From: Junio C Hamano <gitster@pobox.com>
To: git@vger.kernel.org
Subject: [PATCH] diff: --dirstat leakfix
Date: Thu, 17 Aug 2023 14:15:10 -0700 [thread overview]
Message-ID: <xmqqbkf5bcbl.fsf@gitster.g> (raw)
The dirstat_dir structure holds a list of files that had "damages"
and is used to summarize the change by directory. It, specifically
its .files member, was allocated, used, and then left behind,
leaking.
What is tricky is that dir.files[] array is allocated and walked by
not pointing into the array with an incrementing offset, but by
incrementing the beginning of the array, so we need to remember the
original address of dir.files[] array before letting gather_dirstat()
walk over it, and free the original address once we are done.
Signed-off-by: Junio C Hamano <gitster@pobox.com>
---
diff.c | 14 ++++++++++++--
t/t4047-diff-dirstat.sh | 2 ++
2 files changed, 14 insertions(+), 2 deletions(-)
diff --git c/diff.c w/diff.c
index 648f6717a5..03d0cfc700 100644
--- c/diff.c
+++ w/diff.c
@@ -2977,6 +2977,7 @@ static void show_dirstat(struct diff_options *options)
unsigned long changed;
struct dirstat_dir dir;
struct diff_queue_struct *q = &diff_queued_diff;
+ struct dirstat_file *to_free;
dir.files = NULL;
dir.alloc = 0;
@@ -3060,13 +3061,17 @@ static void show_dirstat(struct diff_options *options)
dir.nr++;
}
+ to_free = dir.files;
+
/* This can happen even with many files, if everything was renames */
if (!changed)
- return;
+ goto free_return;
/* Show all directories with more than x% of the changes */
QSORT(dir.files, dir.nr, dirstat_compare);
gather_dirstat(options, &dir, changed, "", 0);
+free_return:
+ free(to_free);
}
static void show_dirstat_by_line(struct diffstat_t *data, struct diff_options *options)
@@ -3074,6 +3079,7 @@ static void show_dirstat_by_line(struct diffstat_t *data, struct diff_options *o
int i;
unsigned long changed;
struct dirstat_dir dir;
+ struct dirstat_file *to_free;
if (data->nr == 0)
return;
@@ -3104,13 +3110,17 @@ static void show_dirstat_by_line(struct diffstat_t *data, struct diff_options *o
dir.nr++;
}
+ to_free = dir.files;
+
/* This can happen even with many files, if everything was renames */
if (!changed)
- return;
+ goto free_return;
/* Show all directories with more than x% of the changes */
QSORT(dir.files, dir.nr, dirstat_compare);
gather_dirstat(options, &dir, changed, "", 0);
+free_return:
+ free(to_free);
}
static void free_diffstat_file(struct diffstat_file *f)
diff --git c/t/t4047-diff-dirstat.sh w/t/t4047-diff-dirstat.sh
index 7fec2cb9cd..70224c3da1 100755
--- c/t/t4047-diff-dirstat.sh
+++ w/t/t4047-diff-dirstat.sh
@@ -1,6 +1,8 @@
#!/bin/sh
test_description='diff --dirstat tests'
+
+TEST_PASSES_SANITIZE_LEAK=true
. ./test-lib.sh
# set up two commits where the second commit has these files
reply other threads:[~2023-08-17 21:16 UTC|newest]
Thread overview: [no followups] expand[flat|nested] mbox.gz Atom feed
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to=xmqqbkf5bcbl.fsf@gitster.g \
--to=gitster@pobox.com \
--cc=git@vger.kernel.org \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line
before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for NNTP newsgroup(s).