]> git.ipfire.org Git - thirdparty/git.git/commitdiff
csum-file: introduce discard_hashfile()
authorJunio C Hamano <gitster@pobox.com>
Thu, 25 Jul 2024 23:07:28 +0000 (16:07 -0700)
committerJunio C Hamano <gitster@pobox.com>
Fri, 26 Jul 2024 16:04:02 +0000 (09:04 -0700)
The hashfile API is used to write out a "hashfile", which has a
final checksum (typically SHA-1) at the end.  An in-core hashfile
structure has up to two file descriptors and a few buffers that can
only be freed by calling a helper function that is private to the
csum-file implementation.

The usual flow of a user of the API is to first open a file
descriptor for writing, obtain a hashfile associated with that write
file descriptor by calling either hashfd() or hashfd_check(), call
hashwrite() number of times to write data to the file, and then call
finalize_hashfile(), which appends th checksum to the end of the
file, closes file descriptors and releases associated buffers.

But what if a caller finds some error after calling hashfd() to
start the process and/or hashwrite() to send some data to the file,
and wants to abort the operation?  The underlying file descriptor is
often managed by the tempfile API, so aborting will clean the file
out of the filesystem, but the resources associated with the in-core
hashfile structure is lost.

Introduce discard_hashfile() API function to allow them to release
the resources held by a hashfile structure the callers want to
dispose of, and use that in read-cache.c:do_write_index(), which is
a central place that writes the index file.

Mark t2107 as leak-free, as this leak in "update-index --cacheinfo"
test that deliberately makes it fail is now plugged.

Signed-off-by: Junio C Hamano <gitster@pobox.com>
csum-file.c
csum-file.h
read-cache.c
t/t2107-update-index-basic.sh

index 8abbf013250eef0d5bc33cf2f47ce6797c51e0bb..2131ee6b12e2a64a444786a924f04b941b2b64bb 100644 (file)
@@ -102,6 +102,15 @@ int finalize_hashfile(struct hashfile *f, unsigned char *result,
        return fd;
 }
 
+void discard_hashfile(struct hashfile *f)
+{
+       if (0 <= f->check_fd)
+               close(f->check_fd);
+       if (0 <= f->fd)
+               close(f->fd);
+       free_hashfile(f);
+}
+
 void hashwrite(struct hashfile *f, const void *buf, unsigned int count)
 {
        while (count) {
index 566e05cbd25a04be9977356e3cfb5a77637bf4c8..36c7c5585f5672d4af089aee760c0240653b0942 100644 (file)
@@ -47,6 +47,7 @@ struct hashfile *hashfd(int fd, const char *name);
 struct hashfile *hashfd_check(const char *name);
 struct hashfile *hashfd_throughput(int fd, const char *name, struct progress *tp);
 int finalize_hashfile(struct hashfile *, unsigned char *, enum fsync_component, unsigned int);
+void discard_hashfile(struct hashfile *);
 void hashwrite(struct hashfile *, const void *, unsigned int);
 void hashflush(struct hashfile *f);
 void crc32_begin(struct hashfile *);
index 48bf24f87c00c15e5a4b249b26eabc624546f901..1f67bb755b1cd11ad35fc59c1a339e3a0d60b420 100644 (file)
@@ -2963,7 +2963,7 @@ static int do_write_index(struct index_state *istate, struct tempfile *tempfile,
 
        if (err) {
                free(ieot);
-               return err;
+               goto cleanup;
        }
 
        offset = hashfile_total(f);
@@ -2992,8 +2992,14 @@ static int do_write_index(struct index_state *istate, struct tempfile *tempfile,
                hashwrite(f, sb.buf, sb.len);
                strbuf_release(&sb);
                free(ieot);
-               if (err)
-                       return -1;
+               /*
+                * NEEDSWORK: write_index_ext_header() never returns a failure,
+                * and this part may want to be simplified.
+                */
+               if (err) {
+                       err = -1;
+                       goto cleanup;
+               }
        }
 
        if (write_extensions & WRITE_SPLIT_INDEX_EXTENSION &&
@@ -3008,8 +3014,14 @@ static int do_write_index(struct index_state *istate, struct tempfile *tempfile,
                                               sb.len) < 0;
                hashwrite(f, sb.buf, sb.len);
                strbuf_release(&sb);
-               if (err)
-                       return -1;
+               /*
+                * NEEDSWORK: write_link_extension() never returns a failure,
+                * and this part may want to be simplified.
+                */
+               if (err) {
+                       err = -1;
+                       goto cleanup;
+               }
        }
        if (write_extensions & WRITE_CACHE_TREE_EXTENSION &&
            !drop_cache_tree && istate->cache_tree) {
@@ -3019,8 +3031,14 @@ static int do_write_index(struct index_state *istate, struct tempfile *tempfile,
                err = write_index_ext_header(f, eoie_c, CACHE_EXT_TREE, sb.len) < 0;
                hashwrite(f, sb.buf, sb.len);
                strbuf_release(&sb);
-               if (err)
-                       return -1;
+               /*
+                * NEEDSWORK: write_index_ext_header() never returns a failure,
+                * and this part may want to be simplified.
+                */
+               if (err) {
+                       err = -1;
+                       goto cleanup;
+               }
        }
        if (write_extensions & WRITE_RESOLVE_UNDO_EXTENSION &&
            istate->resolve_undo) {
@@ -3031,8 +3049,14 @@ static int do_write_index(struct index_state *istate, struct tempfile *tempfile,
                                             sb.len) < 0;
                hashwrite(f, sb.buf, sb.len);
                strbuf_release(&sb);
-               if (err)
-                       return -1;
+               /*
+                * NEEDSWORK: write_index_ext_header() never returns a failure,
+                * and this part may want to be simplified.
+                */
+               if (err) {
+                       err = -1;
+                       goto cleanup;
+               }
        }
        if (write_extensions & WRITE_UNTRACKED_CACHE_EXTENSION &&
            istate->untracked) {
@@ -3043,8 +3067,14 @@ static int do_write_index(struct index_state *istate, struct tempfile *tempfile,
                                             sb.len) < 0;
                hashwrite(f, sb.buf, sb.len);
                strbuf_release(&sb);
-               if (err)
-                       return -1;
+               /*
+                * NEEDSWORK: write_index_ext_header() never returns a failure,
+                * and this part may want to be simplified.
+                */
+               if (err) {
+                       err = -1;
+                       goto cleanup;
+               }
        }
        if (write_extensions & WRITE_FSMONITOR_EXTENSION &&
            istate->fsmonitor_last_update) {
@@ -3054,12 +3084,25 @@ static int do_write_index(struct index_state *istate, struct tempfile *tempfile,
                err = write_index_ext_header(f, eoie_c, CACHE_EXT_FSMONITOR, sb.len) < 0;
                hashwrite(f, sb.buf, sb.len);
                strbuf_release(&sb);
-               if (err)
-                       return -1;
+               /*
+                * NEEDSWORK: write_index_ext_header() never returns a failure,
+                * and this part may want to be simplified.
+                */
+               if (err) {
+                       err = -1;
+                       goto cleanup;
+               }
        }
        if (istate->sparse_index) {
-               if (write_index_ext_header(f, eoie_c, CACHE_EXT_SPARSE_DIRECTORIES, 0) < 0)
-                       return -1;
+               err = write_index_ext_header(f, eoie_c, CACHE_EXT_SPARSE_DIRECTORIES, 0);
+               /*
+                * NEEDSWORK: write_index_ext_header() never returns a failure,
+                * and this part may want to be simplified.
+                */
+               if (err) {
+                       err = -1;
+                       goto cleanup;
+               }
        }
 
        /*
@@ -3075,8 +3118,14 @@ static int do_write_index(struct index_state *istate, struct tempfile *tempfile,
                err = write_index_ext_header(f, NULL, CACHE_EXT_ENDOFINDEXENTRIES, sb.len) < 0;
                hashwrite(f, sb.buf, sb.len);
                strbuf_release(&sb);
-               if (err)
-                       return -1;
+               /*
+                * NEEDSWORK: write_index_ext_header() never returns a failure,
+                * and this part may want to be simplified.
+                */
+               if (err) {
+                       err = -1;
+                       goto cleanup;
+               }
        }
 
        csum_fsync_flag = 0;
@@ -3085,13 +3134,16 @@ static int do_write_index(struct index_state *istate, struct tempfile *tempfile,
 
        finalize_hashfile(f, istate->oid.hash, FSYNC_COMPONENT_INDEX,
                          CSUM_HASH_IN_STREAM | csum_fsync_flag);
+       f = NULL;
 
        if (close_tempfile_gently(tempfile)) {
-               error(_("could not close '%s'"), get_tempfile_path(tempfile));
-               return -1;
+               err = error(_("could not close '%s'"), get_tempfile_path(tempfile));
+               goto cleanup;
+       }
+       if (stat(get_tempfile_path(tempfile), &st)) {
+               err = error_errno(_("could not stat '%s'"), get_tempfile_path(tempfile));
+               goto cleanup;
        }
-       if (stat(get_tempfile_path(tempfile), &st))
-               return -1;
        istate->timestamp.sec = (unsigned int)st.st_mtime;
        istate->timestamp.nsec = ST_MTIME_NSEC(st);
        trace_performance_since(start, "write index, changed mask = %x", istate->cache_changed);
@@ -3106,6 +3158,11 @@ static int do_write_index(struct index_state *istate, struct tempfile *tempfile,
                           istate->cache_nr);
 
        return 0;
+
+cleanup:
+       if (f)
+               discard_hashfile(f);
+       return err;
 }
 
 void set_alternate_index_output(const char *name)
index cc72ead79f397873b5005196a99baad6b0ba2450..f0eab13f96a6b946f982db2543112cc7467d857d 100755 (executable)
@@ -5,6 +5,7 @@ test_description='basic update-index tests
 Tests for command-line parsing and basic operation.
 '
 
+TEST_PASSES_SANITIZE_LEAK=true
 . ./test-lib.sh
 
 test_expect_success 'update-index --nonsense fails' '