From 0f8faebb974621a7647d52beb3796121c07b3607 Mon Sep 17 00:00:00 2001 From: James Youngman Date: Wed, 20 Feb 2008 21:44:53 +0100 Subject: join: avoid new leaks * src/join.c (prevline): Make prevline module-level static, so that the allocated items can be freed at exit. (free_prevline): new atexit function; frees items in prevline[]. Use ARRAY_CARDINALITY, so include "argmatch.h" to get that. (main): Arrange for free_prevline to be called during exit. (get_line): Free prevline[which - 1] also, as that will have been allocated by dup_line. --- src/join.c | 52 ++++++++++++++++++++++++++++++++++------------------ 1 file changed, 34 insertions(+), 18 deletions(-) (limited to 'src') diff --git a/src/join.c b/src/join.c index 71b11da06..30fbd439c 100644 --- a/src/join.c +++ b/src/join.c @@ -31,6 +31,7 @@ #include "stdio--.h" #include "xmemcoll.h" #include "xstrtol.h" +#include "argmatch.h" /* The official name of this program (e.g., no `g' prefix). */ #define PROGRAM_NAME "join" @@ -81,6 +82,9 @@ struct seq /* The name this program was run with. */ char *program_name; +/* The previous line read from each file. */ +static struct line *prevline[2]; + /* True if the LC_COLLATE locale is hard. */ static bool hard_LC_COLLATE; @@ -300,8 +304,6 @@ freeline (struct line *line) static bool get_line (FILE *fp, struct line *line, int which) { - static struct line *prevline[2]; - initbuffer (&line->buf); if (! readlinebuffer (&line->buf, fp)) @@ -322,11 +324,26 @@ get_line (FILE *fp, struct line *line, int which) { checkorder (prevline[which - 1], line, which); freeline (prevline[which - 1]); + free (prevline[which - 1]); } prevline[which - 1] = dup_line (line); return true; } +static void +free_prevline (void) +{ + size_t i; + + for (i = 0; i < ARRAY_CARDINALITY (prevline); i++) + { + if (prevline[i]) + freeline (prevline[i]); + free (prevline[i]); + prevline[i] = NULL; + } +} + static void initseq (struct seq *seq) { @@ -352,8 +369,7 @@ getseq (FILE *fp, struct seq *seq, int whichfile) } /* Read a line from FP and add it to SEQ, as the first item if FIRST is - * true, else as the next. - */ + true, else as the next. */ static bool advance_seq (FILE *fp, struct seq *seq, bool first, int whichfile) { @@ -438,16 +454,16 @@ keycmp (struct line const *line1, struct line const *line2) /* Check that successive input lines PREV and CURRENT from input file - * WHATFILE are presented in order, unless the user may be relying on - * the GNU extension that input lines may be out of order if no input - * lines are unpairable. - * - * If the user specified --nocheck-order, the check is not made. - * If the user specified --check-order, the problem is fatal. - * Otherwise (the default), the message is simply a warning. - * - * A message is printed at most once per input file. - */ + WHATFILE are presented in order, unless the user may be relying on + the GNU extension that input lines may be out of order if no input + lines are unpairable. + + If the user specified --nocheck-order, the check is not made. + If the user specified --check-order, the problem is fatal. + Otherwise (the default), the message is simply a warning. + + A message is printed at most once per input file. */ + static void checkorder (const struct line *prev, const struct line *current, @@ -668,10 +684,9 @@ join (FILE *fp1, FILE *fp2) } /* If the user did not specify --check-order, and the we read the - * tail ends of both inputs to verify that they are in order. We - * skip the rest of the tail once we have issued a warning for that - * file, unless we actually need to print the unpairable lines. - */ + tail ends of both inputs to verify that they are in order. We + skip the rest of the tail once we have issued a warning for that + file, unless we actually need to print the unpairable lines. */ if (check_input_order != CHECK_ORDER_DISABLED && !(issued_disorder_warning[0] && issued_disorder_warning[1])) checktail = true; @@ -925,6 +940,7 @@ main (int argc, char **argv) hard_LC_COLLATE = hard_locale (LC_COLLATE); atexit (close_stdout); + atexit (free_prevline); print_pairables = true; seen_unpairable = false; -- cgit v1.2.3-70-g09d2