diff options
| author | Russ Allbery <eagle@eyrie.org> | 2014-07-16 13:46:50 -0700 | 
|---|---|---|
| committer | Russ Allbery <eagle@eyrie.org> | 2014-07-16 13:46:50 -0700 | 
| commit | 1796d631f0846ec98cd286bc4284898a7300ee78 (patch) | |
| tree | 6fd42de6dc858ef06c6d270410c32ec61f39e593 /tests/tap/basic.c | |
| parent | f5194217566a6f4cdeffbae551153feb1412210d (diff) | |
| parent | 6409733ee3b7b1910dc1c166a392cc628834146c (diff) | |
Merge tag 'upstream/1.1' into debian
Upstream version 1.1
Conflicts:
	NEWS
	README
	client/keytab.c
	perl/lib/Wallet/ACL.pm
	perl/sql/Wallet-Schema-0.08-PostgreSQL.sql
	perl/t/general/admin.t
	perl/t/verifier/ldap-attr.t
Change-Id: I1a1dc09b97c9258e61f1c8877d0837193c8ae2c6
Diffstat (limited to 'tests/tap/basic.c')
| -rw-r--r-- | tests/tap/basic.c | 586 | 
1 files changed, 437 insertions, 149 deletions
| diff --git a/tests/tap/basic.c b/tests/tap/basic.c index e8196fc..92a749b 100644 --- a/tests/tap/basic.c +++ b/tests/tap/basic.c @@ -12,8 +12,8 @@   * This file is part of C TAP Harness.  The current version plus supporting   * documentation is at <http://www.eyrie.org/~eagle/software/c-tap-harness/>.   * - * Copyright 2009, 2010, 2011, 2012 Russ Allbery <rra@stanford.edu> - * Copyright 2001, 2002, 2004, 2005, 2006, 2007, 2008, 2011, 2012 + * Copyright 2009, 2010, 2011, 2012, 2013, 2014 Russ Allbery <eagle@eyrie.org> + * Copyright 2001, 2002, 2004, 2005, 2006, 2007, 2008, 2011, 2012, 2013, 2014   *     The Board of Trustees of the Leland Stanford Junior University   *   * Permission is hereby granted, free of charge, to any person obtaining a @@ -36,6 +36,7 @@   */  #include <errno.h> +#include <limits.h>  #include <stdarg.h>  #include <stdio.h>  #include <stdlib.h> @@ -58,7 +59,7 @@  /*   * The test count.  Always contains the number that will be used for the next - * test status. + * test status.  This is exported to callers of the library.   */  unsigned long testnum = 1; @@ -66,72 +67,298 @@ unsigned long testnum = 1;   * Status information stored so that we can give a test summary at the end of   * the test case.  We store the planned final test and the count of failures.   * We can get the highest test count from testnum. - * - * We also store the PID of the process that called plan() and only summarize - * results when that process exits, so as to not misreport results in forked - * processes. - * - * If _lazy is true, we're doing lazy planning and will print out the plan - * based on the last test number at the end of testing.   */  static unsigned long _planned = 0;  static unsigned long _failed  = 0; + +/* + * Store the PID of the process that called plan() and only summarize + * results when that process exits, so as to not misreport results in forked + * processes. + */  static pid_t _process = 0; + +/* + * If true, we're doing lazy planning and will print out the plan based on the + * last test number at the end of testing. + */  static int _lazy = 0; +/* + * If true, the test was aborted by calling bail().  Currently, this is only + * used to ensure that we pass a false value to any cleanup functions even if + * all tests to that point have passed. + */ +static int _aborted = 0; + +/* + * Registered cleanup functions.  These are stored as a linked list and run in + * registered order by finish when the test program exits.  Each function is + * passed a boolean value indicating whether all tests were successful. + */ +struct cleanup_func { +    test_cleanup_func func; +    struct cleanup_func *next; +}; +static struct cleanup_func *cleanup_funcs = NULL; + +/* + * Registered diag files.  Any output found in these files will be printed out + * as if it were passed to diag() before any other output we do.  This allows + * background processes to log to a file and have that output interleved with + * the test output. + */ +struct diag_file { +    char *name; +    FILE *file; +    char *buffer; +    size_t bufsize; +    struct diag_file *next; +}; +static struct diag_file *diag_files = NULL; + +/* + * Print a specified prefix and then the test description.  Handles turning + * the argument list into a va_args structure suitable for passing to + * print_desc, which has to be done in a macro.  Assumes that format is the + * argument immediately before the variadic arguments. + */ +#define PRINT_DESC(prefix, format)              \ +    do {                                        \ +        if (format != NULL) {                   \ +            va_list args;                       \ +            if (prefix != NULL)                 \ +                printf("%s", prefix);           \ +            va_start(args, format);             \ +            vprintf(format, args);              \ +            va_end(args);                       \ +        }                                       \ +    } while (0) + + +/* + * Form a new string by concatenating multiple strings.  The arguments must be + * terminated by (const char *) 0. + * + * This function only exists because we can't assume asprintf.  We can't + * simulate asprintf with snprintf because we're only assuming SUSv3, which + * does not require that snprintf with a NULL buffer return the required + * length.  When those constraints are relaxed, this should be ripped out and + * replaced with asprintf or a more trivial replacement with snprintf. + */ +static char * +concat(const char *first, ...) +{ +    va_list args; +    char *result; +    const char *string; +    size_t offset; +    size_t length = 0; + +    /* +     * Find the total memory required.  Ensure we don't overflow length.  See +     * the comment for breallocarray for why we're using UINT_MAX here. +     */ +    va_start(args, first); +    for (string = first; string != NULL; string = va_arg(args, const char *)) { +        if (length >= UINT_MAX - strlen(string)) +            bail("strings too long in concat"); +        length += strlen(string); +    } +    va_end(args); +    length++; + +    /* Create the string. */ +    result = bmalloc(length); +    va_start(args, first); +    offset = 0; +    for (string = first; string != NULL; string = va_arg(args, const char *)) { +        memcpy(result + offset, string, strlen(string)); +        offset += strlen(string); +    } +    va_end(args); +    result[offset] = '\0'; +    return result; +} + + +/* + * Check all registered diag_files for any output.  We only print out the + * output if we see a complete line; otherwise, we wait for the next newline. + */ +static void +check_diag_files(void) +{ +    struct diag_file *file; +    fpos_t where; +    size_t length; +    int incomplete; + +    /* +     * Walk through each file and read each line of output available.  The +     * general scheme here used is as follows: try to read a line of output at +     * a time.  If we get NULL, check for EOF; on EOF, advance to the next +     * file. +     * +     * If we get some data, see if it ends in a newline.  If it doesn't end in +     * a newline, we have one of two cases: our buffer isn't large enough, in +     * which case we resize it and try again, or we have incomplete data in +     * the file, in which case we rewind the file and will try again next +     * time. +     */ +    for (file = diag_files; file != NULL; file = file->next) { +        clearerr(file->file); + +        /* Store the current position in case we have to rewind. */ +        if (fgetpos(file->file, &where) < 0) +            sysbail("cannot get position in %s", file->name); + +        /* Continue until we get EOF or an incomplete line of data. */ +        incomplete = 0; +        while (!feof(file->file) && !incomplete) { +            if (fgets(file->buffer, file->bufsize, file->file) == NULL) { +                if (ferror(file->file)) +                    sysbail("cannot read from %s", file->name); +                continue; +            } + +            /* +             * See if the line ends in a newline.  If not, see which error +             * case we have.  Use UINT_MAX as a substitute for SIZE_MAX (see +             * the comment for breallocarray). +             */ +            length = strlen(file->buffer); +            if (file->buffer[length - 1] != '\n') { +                if (length < file->bufsize - 1) +                    incomplete = 1; +                else { +                    if (file->bufsize >= UINT_MAX - BUFSIZ) +                        sysbail("line too long in %s", file->name); +                    file->bufsize += BUFSIZ; +                    file->buffer = brealloc(file->buffer, file->bufsize); +                } + +                /* +                 * On either incomplete lines or too small of a buffer, rewind +                 * and read the file again (on the next pass, if incomplete). +                 * It's simpler than trying to double-buffer the file. +                 */ +                if (fsetpos(file->file, &where) < 0) +                    sysbail("cannot set position in %s", file->name); +                continue; +            } + +            /* We saw a complete line.  Print it out. */ +            printf("# %s", file->buffer); +        } +    } +} +  /*   * Our exit handler.  Called on completion of the test to report a summary of   * results provided we're still in the original process.  This also handles   * printing out the plan if we used plan_lazy(), although that's suppressed if - * we never ran a test (due to an early bail, for example). + * we never ran a test (due to an early bail, for example), and running any + * registered cleanup functions.   */  static void  finish(void)  { +    int success, primary; +    struct cleanup_func *current;      unsigned long highest = testnum - 1; +    struct diag_file *file, *tmp; + +    /* Check for pending diag_file output. */ +    check_diag_files(); + +    /* Free the diag_files. */ +    file = diag_files; +    while (file != NULL) { +        tmp = file; +        file = file->next; +        fclose(tmp->file); +        free(tmp->name); +        free(tmp->buffer); +        free(tmp); +    } +    diag_files = NULL; + +    /* +     * Determine whether all tests were successful, which is needed before +     * calling cleanup functions since we pass that fact to the functions. +     */ +    if (_planned == 0 && _lazy) +        _planned = highest; +    success = (!_aborted && _planned == highest && _failed == 0); + +    /* +     * If there are any registered cleanup functions, we run those first.  We +     * always run them, even if we didn't run a test.  Don't do anything +     * except free the diag_files and call cleanup functions if we aren't the +     * primary process (the process in which plan or plan_lazy was called), +     * and tell the cleanup functions that fact. +     */ +    primary = (_process == 0 || getpid() == _process); +    while (cleanup_funcs != NULL) { +        cleanup_funcs->func(success, primary); +        current = cleanup_funcs; +        cleanup_funcs = cleanup_funcs->next; +        free(current); +    } +    if (!primary) +        return; -    if (_planned == 0 && !_lazy) +    /* Don't do anything further if we never planned a test. */ +    if (_planned == 0)          return; + +    /* If we're aborting due to bail, don't print summaries. */ +    if (_aborted) +        return; + +    /* Print out the lazy plan if needed. */      fflush(stderr); -    if (_process != 0 && getpid() == _process) { -        if (_lazy && highest > 0) { -            printf("1..%lu\n", highest); -            _planned = highest; -        } -        if (_planned > highest) -            printf("# Looks like you planned %lu test%s but only ran %lu\n", -                   _planned, (_planned > 1 ? "s" : ""), highest); -        else if (_planned < highest) -            printf("# Looks like you planned %lu test%s but ran %lu extra\n", -                   _planned, (_planned > 1 ? "s" : ""), highest - _planned); -        else if (_failed > 0) -            printf("# Looks like you failed %lu test%s of %lu\n", _failed, -                   (_failed > 1 ? "s" : ""), _planned); -        else if (_planned > 1) -            printf("# All %lu tests successful or skipped\n", _planned); -        else -            printf("# %lu test successful or skipped\n", _planned); -    } +    if (_lazy && _planned > 0) +        printf("1..%lu\n", _planned); + +    /* Print out a summary of the results. */ +    if (_planned > highest) +        diag("Looks like you planned %lu test%s but only ran %lu", _planned, +             (_planned > 1 ? "s" : ""), highest); +    else if (_planned < highest) +        diag("Looks like you planned %lu test%s but ran %lu extra", _planned, +             (_planned > 1 ? "s" : ""), highest - _planned); +    else if (_failed > 0) +        diag("Looks like you failed %lu test%s of %lu", _failed, +             (_failed > 1 ? "s" : ""), _planned); +    else if (_planned != 1) +        diag("All %lu tests successful or skipped", _planned); +    else +        diag("%lu test successful or skipped", _planned);  }  /*   * Initialize things.  Turns on line buffering on stdout and then prints out - * the number of tests in the test suite. + * the number of tests in the test suite.  We intentionally don't check for + * pending diag_file output here, since it should really come after the plan.   */  void  plan(unsigned long count)  {      if (setvbuf(stdout, NULL, _IOLBF, BUFSIZ) != 0) -        fprintf(stderr, "# cannot set stdout to line buffered: %s\n", -                strerror(errno)); +        sysdiag("cannot set stdout to line buffered");      fflush(stderr);      printf("1..%lu\n", count);      testnum = 1;      _planned = count;      _process = getpid(); -    atexit(finish); +    if (atexit(finish) != 0) { +        sysdiag("cannot register exit handler"); +        diag("cleanups will not be run"); +    }  } @@ -143,83 +370,66 @@ void  plan_lazy(void)  {      if (setvbuf(stdout, NULL, _IOLBF, BUFSIZ) != 0) -        fprintf(stderr, "# cannot set stdout to line buffered: %s\n", -                strerror(errno)); +        sysdiag("cannot set stdout to line buffered");      testnum = 1;      _process = getpid();      _lazy = 1; -    atexit(finish); +    if (atexit(finish) != 0) +        sysbail("cannot register exit handler to display plan");  }  /*   * Skip the entire test suite and exits.  Should be called instead of plan(), - * not after it, since it prints out a special plan line. + * not after it, since it prints out a special plan line.  Ignore diag_file + * output here, since it's not clear if it's allowed before the plan.   */  void  skip_all(const char *format, ...)  {      fflush(stderr);      printf("1..0 # skip"); -    if (format != NULL) { -        va_list args; - -        putchar(' '); -        va_start(args, format); -        vprintf(format, args); -        va_end(args); -    } +    PRINT_DESC(" ", format);      putchar('\n');      exit(0);  }  /* - * Print the test description. - */ -static void -print_desc(const char *format, va_list args) -{ -    printf(" - "); -    vprintf(format, args); -} - - -/*   * Takes a boolean success value and assumes the test passes if that value   * is true and fails if that value is false.   */ -void +int  ok(int success, const char *format, ...)  {      fflush(stderr); +    check_diag_files();      printf("%sok %lu", success ? "" : "not ", testnum++);      if (!success)          _failed++; -    if (format != NULL) { -        va_list args; - -        va_start(args, format); -        print_desc(format, args); -        va_end(args); -    } +    PRINT_DESC(" - ", format);      putchar('\n'); +    return success;  }  /*   * Same as ok(), but takes the format arguments as a va_list.   */ -void +int  okv(int success, const char *format, va_list args)  {      fflush(stderr); +    check_diag_files();      printf("%sok %lu", success ? "" : "not ", testnum++);      if (!success)          _failed++; -    if (format != NULL) -        print_desc(format, args); +    if (format != NULL) { +        printf(" - "); +        vprintf(format, args); +    }      putchar('\n'); +    return success;  } @@ -230,15 +440,9 @@ void  skip(const char *reason, ...)  {      fflush(stderr); +    check_diag_files();      printf("ok %lu # skip", testnum++); -    if (reason != NULL) { -        va_list args; - -        va_start(args, reason); -        putchar(' '); -        vprintf(reason, args); -        va_end(args); -    } +    PRINT_DESC(" ", reason);      putchar('\n');  } @@ -246,25 +450,21 @@ skip(const char *reason, ...)  /*   * Report the same status on the next count tests.   */ -void -ok_block(unsigned long count, int status, const char *format, ...) +int +ok_block(unsigned long count, int success, const char *format, ...)  {      unsigned long i;      fflush(stderr); +    check_diag_files();      for (i = 0; i < count; i++) { -        printf("%sok %lu", status ? "" : "not ", testnum++); -        if (!status) +        printf("%sok %lu", success ? "" : "not ", testnum++); +        if (!success)              _failed++; -        if (format != NULL) { -            va_list args; - -            va_start(args, format); -            print_desc(format, args); -            va_end(args); -        } +        PRINT_DESC(" - ", format);          putchar('\n');      } +    return success;  } @@ -277,16 +477,10 @@ skip_block(unsigned long count, const char *reason, ...)      unsigned long i;      fflush(stderr); +    check_diag_files();      for (i = 0; i < count; i++) {          printf("ok %lu # skip", testnum++); -        if (reason != NULL) { -            va_list args; - -            va_start(args, reason); -            putchar(' '); -            vprintf(reason, args); -            va_end(args); -        } +        PRINT_DESC(" ", reason);          putchar('\n');      }  } @@ -296,25 +490,25 @@ skip_block(unsigned long count, const char *reason, ...)   * Takes an expected integer and a seen integer and assumes the test passes   * if those two numbers match.   */ -void +int  is_int(long wanted, long seen, const char *format, ...)  { +    int success; +      fflush(stderr); -    if (wanted == seen) +    check_diag_files(); +    success = (wanted == seen); +    if (success)          printf("ok %lu", testnum++);      else { -        printf("# wanted: %ld\n#   seen: %ld\n", wanted, seen); +        diag("wanted: %ld", wanted); +        diag("  seen: %ld", seen);          printf("not ok %lu", testnum++);          _failed++;      } -    if (format != NULL) { -        va_list args; - -        va_start(args, format); -        print_desc(format, args); -        va_end(args); -    } +    PRINT_DESC(" - ", format);      putchar('\n'); +    return success;  } @@ -322,29 +516,29 @@ is_int(long wanted, long seen, const char *format, ...)   * Takes a string and what the string should be, and assumes the test passes   * if those strings match (using strcmp).   */ -void +int  is_string(const char *wanted, const char *seen, const char *format, ...)  { +    int success; +      if (wanted == NULL)          wanted = "(null)";      if (seen == NULL)          seen = "(null)";      fflush(stderr); -    if (strcmp(wanted, seen) == 0) +    check_diag_files(); +    success = (strcmp(wanted, seen) == 0); +    if (success)          printf("ok %lu", testnum++);      else { -        printf("# wanted: %s\n#   seen: %s\n", wanted, seen); +        diag("wanted: %s", wanted); +        diag("  seen: %s", seen);          printf("not ok %lu", testnum++);          _failed++;      } -    if (format != NULL) { -        va_list args; - -        va_start(args, format); -        print_desc(format, args); -        va_end(args); -    } +    PRINT_DESC(" - ", format);      putchar('\n'); +    return success;  } @@ -352,26 +546,25 @@ is_string(const char *wanted, const char *seen, const char *format, ...)   * Takes an expected unsigned long and a seen unsigned long and assumes the   * test passes if the two numbers match.  Otherwise, reports them in hex.   */ -void +int  is_hex(unsigned long wanted, unsigned long seen, const char *format, ...)  { +    int success; +      fflush(stderr); -    if (wanted == seen) +    check_diag_files(); +    success = (wanted == seen); +    if (success)          printf("ok %lu", testnum++);      else { -        printf("# wanted: %lx\n#   seen: %lx\n", (unsigned long) wanted, -               (unsigned long) seen); +        diag("wanted: %lx", (unsigned long) wanted); +        diag("  seen: %lx", (unsigned long) seen);          printf("not ok %lu", testnum++);          _failed++;      } -    if (format != NULL) { -        va_list args; - -        va_start(args, format); -        print_desc(format, args); -        va_end(args); -    } +    PRINT_DESC(" - ", format);      putchar('\n'); +    return success;  } @@ -383,14 +576,16 @@ bail(const char *format, ...)  {      va_list args; +    _aborted = 1;      fflush(stderr); +    check_diag_files();      fflush(stdout);      printf("Bail out! ");      va_start(args, format);      vprintf(format, args);      va_end(args);      printf("\n"); -    exit(1); +    exit(255);  } @@ -403,51 +598,110 @@ sysbail(const char *format, ...)      va_list args;      int oerrno = errno; +    _aborted = 1;      fflush(stderr); +    check_diag_files();      fflush(stdout);      printf("Bail out! ");      va_start(args, format);      vprintf(format, args);      va_end(args);      printf(": %s\n", strerror(oerrno)); -    exit(1); +    exit(255);  }  /* - * Report a diagnostic to stderr. + * Report a diagnostic to stderr.  Always returns 1 to allow embedding in + * compound statements.   */ -void +int  diag(const char *format, ...)  {      va_list args;      fflush(stderr); +    check_diag_files();      fflush(stdout);      printf("# ");      va_start(args, format);      vprintf(format, args);      va_end(args);      printf("\n"); +    return 1;  }  /* - * Report a diagnostic to stderr, appending strerror(errno). + * Report a diagnostic to stderr, appending strerror(errno).  Always returns 1 + * to allow embedding in compound statements.   */ -void +int  sysdiag(const char *format, ...)  {      va_list args;      int oerrno = errno;      fflush(stderr); +    check_diag_files();      fflush(stdout);      printf("# ");      va_start(args, format);      vprintf(format, args);      va_end(args);      printf(": %s\n", strerror(oerrno)); +    return 1; +} + + +/* + * Register a new file for diag_file processing. + */ +void +diag_file_add(const char *name) +{ +    struct diag_file *file, *prev; + +    file = bcalloc(1, sizeof(struct diag_file)); +    file->name = bstrdup(name); +    file->file = fopen(file->name, "r"); +    if (file->file == NULL) +        sysbail("cannot open %s", name); +    file->buffer = bmalloc(BUFSIZ); +    file->bufsize = BUFSIZ; +    if (diag_files == NULL) +        diag_files = file; +    else { +        for (prev = diag_files; prev->next != NULL; prev = prev->next) +            ; +        prev->next = file; +    } +} + + +/* + * Remove a file from diag_file processing.  If the file is not found, do + * nothing, since there are some situations where it can be removed twice + * (such as if it's removed from a cleanup function, since cleanup functions + * are called after freeing all the diag_files). + */ +void +diag_file_remove(const char *name) +{ +    struct diag_file *file; +    struct diag_file **prev = &diag_files; + +    for (file = diag_files; file != NULL; file = file->next) { +        if (strcmp(file->name, name) == 0) { +            *prev = file->next; +            fclose(file->file); +            free(file->name); +            free(file->buffer); +            free(file); +            return; +        } +        prev = &file->next; +    }  } @@ -495,6 +749,32 @@ brealloc(void *p, size_t size)  /* + * The same as brealloc, but determine the size by multiplying an element + * count by a size, similar to calloc.  The multiplication is checked for + * integer overflow. + * + * We should technically use SIZE_MAX here for the overflow check, but + * SIZE_MAX is C99 and we're only assuming C89 + SUSv3, which does not + * guarantee that it exists.  They do guarantee that UINT_MAX exists, and we + * can assume that UINT_MAX <= SIZE_MAX. + * + * (In theory, C89 and C99 permit size_t to be smaller than unsigned int, but + * I disbelieve in the existence of such systems and they will have to cope + * without overflow checks.) + */ +void * +breallocarray(void *p, size_t n, size_t size) +{ +    if (n > 0 && UINT_MAX / n <= size) +        bail("reallocarray too large"); +    p = realloc(p, n * size); +    if (p == NULL) +        sysbail("failed to realloc %lu bytes", (unsigned long) (n * size)); +    return p; +} + + +/*   * Copy a string, reporting a fatal error with bail on failure.   */  char * @@ -542,17 +822,12 @@ bstrndup(const char *s, size_t n)   * then SOURCE for the file and return the full path to the file.  Returns   * NULL if the file doesn't exist.  A non-NULL return should be freed with   * test_file_path_free(). - * - * This function uses sprintf because it attempts to be independent of all - * other portability layers.  The use immediately after a memory allocation - * should be safe without using snprintf or strlcpy/strlcat.   */  char *  test_file_path(const char *file)  {      char *base;      char *path = NULL; -    size_t length;      const char *envs[] = { "BUILD", "SOURCE", NULL };      int i; @@ -560,9 +835,7 @@ test_file_path(const char *file)          base = getenv(envs[i]);          if (base == NULL)              continue; -        length = strlen(base) + 1 + strlen(file) + 1; -        path = bmalloc(length); -        sprintf(path, "%s/%s", base, file); +        path = concat(base, "/", file, (const char *) 0);          if (access(path, R_OK) == 0)              break;          free(path); @@ -580,8 +853,7 @@ test_file_path(const char *file)  void  test_file_path_free(char *path)  { -    if (path != NULL) -        free(path); +    free(path);  } @@ -600,14 +872,11 @@ test_tmpdir(void)  {      const char *build;      char *path = NULL; -    size_t length;      build = getenv("BUILD");      if (build == NULL)          build = "."; -    length = strlen(build) + strlen("/tmp") + 1; -    path = bmalloc(length); -    sprintf(path, "%s/tmp", build); +    path = concat(build, "/tmp", (const char *) 0);      if (access(path, X_OK) < 0)          if (mkdir(path, 0777) < 0)              sysbail("error creating temporary directory %s", path); @@ -623,7 +892,26 @@ test_tmpdir(void)  void  test_tmpdir_free(char *path)  { -    rmdir(path);      if (path != NULL) -        free(path); +        rmdir(path); +    free(path); +} + + +/* + * Register a cleanup function that is called when testing ends.  All such + * registered functions will be run by finish. + */ +void +test_cleanup_register(test_cleanup_func func) +{ +    struct cleanup_func *cleanup, **last; + +    cleanup = bmalloc(sizeof(struct cleanup_func)); +    cleanup->func = func; +    cleanup->next = NULL; +    last = &cleanup_funcs; +    while (*last != NULL) +        last = &(*last)->next; +    *last = cleanup;  } | 
