Merge branch 'ds/write-index-with-hashfile-api'

Use the hashfile API in the codepath that writes the index file to
reduce code duplication.

* ds/write-index-with-hashfile-api:
  read-cache: delete unused hashing methods
  read-cache: use hashfile instead of git_hash_ctx
  csum-file.h: increase hashfile buffer size
  hashfile: use write_in_full()
This commit is contained in:
Junio C Hamano 2021-06-14 13:33:26 +09:00
commit 0dd2fd18f8
4 changed files with 135 additions and 168 deletions

View File

@ -58,9 +58,11 @@ void add_chunk(struct chunkfile *cf,
int write_chunkfile(struct chunkfile *cf, void *data) int write_chunkfile(struct chunkfile *cf, void *data)
{ {
int i; int i, result = 0;
uint64_t cur_offset = hashfile_total(cf->f); uint64_t cur_offset = hashfile_total(cf->f);
trace2_region_enter("chunkfile", "write", the_repository);
/* Add the table of contents to the current offset */ /* Add the table of contents to the current offset */
cur_offset += (cf->chunks_nr + 1) * CHUNK_TOC_ENTRY_SIZE; cur_offset += (cf->chunks_nr + 1) * CHUNK_TOC_ENTRY_SIZE;
@ -77,10 +79,10 @@ int write_chunkfile(struct chunkfile *cf, void *data)
for (i = 0; i < cf->chunks_nr; i++) { for (i = 0; i < cf->chunks_nr; i++) {
off_t start_offset = hashfile_total(cf->f); off_t start_offset = hashfile_total(cf->f);
int result = cf->chunks[i].write_fn(cf->f, data); result = cf->chunks[i].write_fn(cf->f, data);
if (result) if (result)
return result; goto cleanup;
if (hashfile_total(cf->f) - start_offset != cf->chunks[i].size) if (hashfile_total(cf->f) - start_offset != cf->chunks[i].size)
BUG("expected to write %"PRId64" bytes to chunk %"PRIx32", but wrote %"PRId64" instead", BUG("expected to write %"PRId64" bytes to chunk %"PRIx32", but wrote %"PRId64" instead",
@ -88,7 +90,9 @@ int write_chunkfile(struct chunkfile *cf, void *data)
hashfile_total(cf->f) - start_offset); hashfile_total(cf->f) - start_offset);
} }
return 0; cleanup:
trace2_region_leave("chunkfile", "write", the_repository);
return result;
} }
int read_table_of_contents(struct chunkfile *cf, int read_table_of_contents(struct chunkfile *cf,

View File

@ -11,35 +11,33 @@
#include "progress.h" #include "progress.h"
#include "csum-file.h" #include "csum-file.h"
static void verify_buffer_or_die(struct hashfile *f,
const void *buf,
unsigned int count)
{
ssize_t ret = read_in_full(f->check_fd, f->check_buffer, count);
if (ret < 0)
die_errno("%s: sha1 file read error", f->name);
if (ret != count)
die("%s: sha1 file truncated", f->name);
if (memcmp(buf, f->check_buffer, count))
die("sha1 file '%s' validation error", f->name);
}
static void flush(struct hashfile *f, const void *buf, unsigned int count) static void flush(struct hashfile *f, const void *buf, unsigned int count)
{ {
if (0 <= f->check_fd && count) { if (0 <= f->check_fd && count)
unsigned char check_buffer[8192]; verify_buffer_or_die(f, buf, count);
ssize_t ret = read_in_full(f->check_fd, check_buffer, count);
if (ret < 0) if (write_in_full(f->fd, buf, count) < 0) {
die_errno("%s: sha1 file read error", f->name); if (errno == ENOSPC)
if (ret != count)
die("%s: sha1 file truncated", f->name);
if (memcmp(buf, check_buffer, count))
die("sha1 file '%s' validation error", f->name);
}
for (;;) {
int ret = xwrite(f->fd, buf, count);
if (ret > 0) {
f->total += ret;
display_throughput(f->tp, f->total);
buf = (char *) buf + ret;
count -= ret;
if (count)
continue;
return;
}
if (!ret)
die("sha1 file '%s' write error. Out of diskspace", f->name); die("sha1 file '%s' write error. Out of diskspace", f->name);
die_errno("sha1 file '%s' write error", f->name); die_errno("sha1 file '%s' write error", f->name);
} }
f->total += count;
display_throughput(f->tp, f->total);
} }
void hashflush(struct hashfile *f) void hashflush(struct hashfile *f)
@ -53,6 +51,13 @@ void hashflush(struct hashfile *f)
} }
} }
static void free_hashfile(struct hashfile *f)
{
free(f->buffer);
free(f->check_buffer);
free(f);
}
int finalize_hashfile(struct hashfile *f, unsigned char *result, unsigned int flags) int finalize_hashfile(struct hashfile *f, unsigned char *result, unsigned int flags)
{ {
int fd; int fd;
@ -82,20 +87,20 @@ int finalize_hashfile(struct hashfile *f, unsigned char *result, unsigned int fl
if (close(f->check_fd)) if (close(f->check_fd))
die_errno("%s: sha1 file error on close", f->name); die_errno("%s: sha1 file error on close", f->name);
} }
free(f); free_hashfile(f);
return fd; return fd;
} }
void hashwrite(struct hashfile *f, const void *buf, unsigned int count) void hashwrite(struct hashfile *f, const void *buf, unsigned int count)
{ {
while (count) { while (count) {
unsigned left = sizeof(f->buffer) - f->offset; unsigned left = f->buffer_len - f->offset;
unsigned nr = count > left ? left : count; unsigned nr = count > left ? left : count;
if (f->do_crc) if (f->do_crc)
f->crc32 = crc32(f->crc32, buf, nr); f->crc32 = crc32(f->crc32, buf, nr);
if (nr == sizeof(f->buffer)) { if (nr == f->buffer_len) {
/* /*
* Flush a full batch worth of data directly * Flush a full batch worth of data directly
* from the input, skipping the memcpy() to * from the input, skipping the memcpy() to
@ -121,11 +126,6 @@ void hashwrite(struct hashfile *f, const void *buf, unsigned int count)
} }
} }
struct hashfile *hashfd(int fd, const char *name)
{
return hashfd_throughput(fd, name, NULL);
}
struct hashfile *hashfd_check(const char *name) struct hashfile *hashfd_check(const char *name)
{ {
int sink, check; int sink, check;
@ -139,10 +139,14 @@ struct hashfile *hashfd_check(const char *name)
die_errno("unable to open '%s'", name); die_errno("unable to open '%s'", name);
f = hashfd(sink, name); f = hashfd(sink, name);
f->check_fd = check; f->check_fd = check;
f->check_buffer = xmalloc(f->buffer_len);
return f; return f;
} }
struct hashfile *hashfd_throughput(int fd, const char *name, struct progress *tp) static struct hashfile *hashfd_internal(int fd, const char *name,
struct progress *tp,
size_t buffer_len)
{ {
struct hashfile *f = xmalloc(sizeof(*f)); struct hashfile *f = xmalloc(sizeof(*f));
f->fd = fd; f->fd = fd;
@ -153,9 +157,35 @@ struct hashfile *hashfd_throughput(int fd, const char *name, struct progress *tp
f->name = name; f->name = name;
f->do_crc = 0; f->do_crc = 0;
the_hash_algo->init_fn(&f->ctx); the_hash_algo->init_fn(&f->ctx);
f->buffer_len = buffer_len;
f->buffer = xmalloc(buffer_len);
f->check_buffer = NULL;
return f; return f;
} }
struct hashfile *hashfd(int fd, const char *name)
{
/*
* Since we are not going to use a progress meter to
* measure the rate of data passing through this hashfile,
* use a larger buffer size to reduce fsync() calls.
*/
return hashfd_internal(fd, name, NULL, 128 * 1024);
}
struct hashfile *hashfd_throughput(int fd, const char *name, struct progress *tp)
{
/*
* Since we are expecting to report progress of the
* write into this hashfile, use a smaller buffer
* size so the progress indicators arrive at a more
* frequent rate.
*/
return hashfd_internal(fd, name, tp, 8 * 1024);
}
void hashfile_checkpoint(struct hashfile *f, struct hashfile_checkpoint *checkpoint) void hashfile_checkpoint(struct hashfile *f, struct hashfile_checkpoint *checkpoint)
{ {
hashflush(f); hashflush(f);

View File

@ -16,7 +16,9 @@ struct hashfile {
const char *name; const char *name;
int do_crc; int do_crc;
uint32_t crc32; uint32_t crc32;
unsigned char buffer[8192]; size_t buffer_len;
unsigned char *buffer;
unsigned char *check_buffer;
}; };
/* Checkpoint */ /* Checkpoint */

View File

@ -26,6 +26,7 @@
#include "thread-utils.h" #include "thread-utils.h"
#include "progress.h" #include "progress.h"
#include "sparse-index.h" #include "sparse-index.h"
#include "csum-file.h"
/* Mask for the name length in ce_flags in the on-disk index */ /* Mask for the name length in ce_flags in the on-disk index */
@ -2521,80 +2522,23 @@ int repo_index_has_changes(struct repository *repo,
} }
} }
#define WRITE_BUFFER_SIZE (128 * 1024) static int write_index_ext_header(struct hashfile *f,
static unsigned char write_buffer[WRITE_BUFFER_SIZE]; git_hash_ctx *eoie_f,
static unsigned long write_buffer_len; unsigned int ext,
unsigned int sz)
static int ce_write_flush(git_hash_ctx *context, int fd)
{ {
unsigned int buffered = write_buffer_len; hashwrite_be32(f, ext);
if (buffered) { hashwrite_be32(f, sz);
the_hash_algo->update_fn(context, write_buffer, buffered);
if (write_in_full(fd, write_buffer, buffered) < 0) if (eoie_f) {
return -1; ext = htonl(ext);
write_buffer_len = 0; sz = htonl(sz);
the_hash_algo->update_fn(eoie_f, &ext, sizeof(ext));
the_hash_algo->update_fn(eoie_f, &sz, sizeof(sz));
} }
return 0; return 0;
} }
static int ce_write(git_hash_ctx *context, int fd, void *data, unsigned int len)
{
while (len) {
unsigned int buffered = write_buffer_len;
unsigned int partial = WRITE_BUFFER_SIZE - buffered;
if (partial > len)
partial = len;
memcpy(write_buffer + buffered, data, partial);
buffered += partial;
if (buffered == WRITE_BUFFER_SIZE) {
write_buffer_len = buffered;
if (ce_write_flush(context, fd))
return -1;
buffered = 0;
}
write_buffer_len = buffered;
len -= partial;
data = (char *) data + partial;
}
return 0;
}
static int write_index_ext_header(git_hash_ctx *context, git_hash_ctx *eoie_context,
int fd, unsigned int ext, unsigned int sz)
{
ext = htonl(ext);
sz = htonl(sz);
if (eoie_context) {
the_hash_algo->update_fn(eoie_context, &ext, 4);
the_hash_algo->update_fn(eoie_context, &sz, 4);
}
return ((ce_write(context, fd, &ext, 4) < 0) ||
(ce_write(context, fd, &sz, 4) < 0)) ? -1 : 0;
}
static int ce_flush(git_hash_ctx *context, int fd, unsigned char *hash)
{
unsigned int left = write_buffer_len;
if (left) {
write_buffer_len = 0;
the_hash_algo->update_fn(context, write_buffer, left);
}
/* Flush first if not enough space for hash signature */
if (left + the_hash_algo->rawsz > WRITE_BUFFER_SIZE) {
if (write_in_full(fd, write_buffer, left) < 0)
return -1;
left = 0;
}
/* Append the hash signature at the end */
the_hash_algo->final_fn(write_buffer + left, context);
hashcpy(hash, write_buffer + left);
left += the_hash_algo->rawsz;
return (write_in_full(fd, write_buffer, left) < 0) ? -1 : 0;
}
static void ce_smudge_racily_clean_entry(struct index_state *istate, static void ce_smudge_racily_clean_entry(struct index_state *istate,
struct cache_entry *ce) struct cache_entry *ce)
{ {
@ -2673,11 +2617,10 @@ static void copy_cache_entry_to_ondisk(struct ondisk_cache_entry *ondisk,
} }
} }
static int ce_write_entry(git_hash_ctx *c, int fd, struct cache_entry *ce, static int ce_write_entry(struct hashfile *f, struct cache_entry *ce,
struct strbuf *previous_name, struct ondisk_cache_entry *ondisk) struct strbuf *previous_name, struct ondisk_cache_entry *ondisk)
{ {
int size; int size;
int result;
unsigned int saved_namelen; unsigned int saved_namelen;
int stripped_name = 0; int stripped_name = 0;
static unsigned char padding[8] = { 0x00 }; static unsigned char padding[8] = { 0x00 };
@ -2693,11 +2636,9 @@ static int ce_write_entry(git_hash_ctx *c, int fd, struct cache_entry *ce,
if (!previous_name) { if (!previous_name) {
int len = ce_namelen(ce); int len = ce_namelen(ce);
copy_cache_entry_to_ondisk(ondisk, ce); copy_cache_entry_to_ondisk(ondisk, ce);
result = ce_write(c, fd, ondisk, size); hashwrite(f, ondisk, size);
if (!result) hashwrite(f, ce->name, len);
result = ce_write(c, fd, ce->name, len); hashwrite(f, padding, align_padding_size(size, len));
if (!result)
result = ce_write(c, fd, padding, align_padding_size(size, len));
} else { } else {
int common, to_remove, prefix_size; int common, to_remove, prefix_size;
unsigned char to_remove_vi[16]; unsigned char to_remove_vi[16];
@ -2711,13 +2652,10 @@ static int ce_write_entry(git_hash_ctx *c, int fd, struct cache_entry *ce,
prefix_size = encode_varint(to_remove, to_remove_vi); prefix_size = encode_varint(to_remove, to_remove_vi);
copy_cache_entry_to_ondisk(ondisk, ce); copy_cache_entry_to_ondisk(ondisk, ce);
result = ce_write(c, fd, ondisk, size); hashwrite(f, ondisk, size);
if (!result) hashwrite(f, to_remove_vi, prefix_size);
result = ce_write(c, fd, to_remove_vi, prefix_size); hashwrite(f, ce->name + common, ce_namelen(ce) - common);
if (!result) hashwrite(f, padding, 1);
result = ce_write(c, fd, ce->name + common, ce_namelen(ce) - common);
if (!result)
result = ce_write(c, fd, padding, 1);
strbuf_splice(previous_name, common, to_remove, strbuf_splice(previous_name, common, to_remove,
ce->name + common, ce_namelen(ce) - common); ce->name + common, ce_namelen(ce) - common);
@ -2727,7 +2665,7 @@ static int ce_write_entry(git_hash_ctx *c, int fd, struct cache_entry *ce,
ce->ce_flags &= ~CE_STRIP_NAME; ce->ce_flags &= ~CE_STRIP_NAME;
} }
return result; return 0;
} }
/* /*
@ -2839,8 +2777,8 @@ static int do_write_index(struct index_state *istate, struct tempfile *tempfile,
int strip_extensions) int strip_extensions)
{ {
uint64_t start = getnanotime(); uint64_t start = getnanotime();
int newfd = tempfile->fd; struct hashfile *f;
git_hash_ctx c, eoie_c; git_hash_ctx *eoie_c = NULL;
struct cache_header hdr; struct cache_header hdr;
int i, err = 0, removed, extended, hdr_version; int i, err = 0, removed, extended, hdr_version;
struct cache_entry **cache = istate->cache; struct cache_entry **cache = istate->cache;
@ -2854,6 +2792,8 @@ static int do_write_index(struct index_state *istate, struct tempfile *tempfile,
struct index_entry_offset_table *ieot = NULL; struct index_entry_offset_table *ieot = NULL;
int nr, nr_threads; int nr, nr_threads;
f = hashfd(tempfile->fd, tempfile->filename.buf);
for (i = removed = extended = 0; i < entries; i++) { for (i = removed = extended = 0; i < entries; i++) {
if (cache[i]->ce_flags & CE_REMOVE) if (cache[i]->ce_flags & CE_REMOVE)
removed++; removed++;
@ -2882,9 +2822,7 @@ static int do_write_index(struct index_state *istate, struct tempfile *tempfile,
hdr.hdr_version = htonl(hdr_version); hdr.hdr_version = htonl(hdr_version);
hdr.hdr_entries = htonl(entries - removed); hdr.hdr_entries = htonl(entries - removed);
the_hash_algo->init_fn(&c); hashwrite(f, &hdr, sizeof(hdr));
if (ce_write(&c, newfd, &hdr, sizeof(hdr)) < 0)
return -1;
if (!HAVE_THREADS || git_config_get_index_threads(&nr_threads)) if (!HAVE_THREADS || git_config_get_index_threads(&nr_threads))
nr_threads = 1; nr_threads = 1;
@ -2919,12 +2857,8 @@ static int do_write_index(struct index_state *istate, struct tempfile *tempfile,
} }
} }
offset = lseek(newfd, 0, SEEK_CUR); offset = hashfile_total(f);
if (offset < 0) {
free(ieot);
return -1;
}
offset += write_buffer_len;
nr = 0; nr = 0;
previous_name = (hdr_version == 4) ? &previous_name_buf : NULL; previous_name = (hdr_version == 4) ? &previous_name_buf : NULL;
@ -2959,14 +2893,10 @@ static int do_write_index(struct index_state *istate, struct tempfile *tempfile,
if (previous_name) if (previous_name)
previous_name->buf[0] = 0; previous_name->buf[0] = 0;
nr = 0; nr = 0;
offset = lseek(newfd, 0, SEEK_CUR);
if (offset < 0) { offset = hashfile_total(f);
free(ieot);
return -1;
}
offset += write_buffer_len;
} }
if (ce_write_entry(&c, newfd, ce, previous_name, (struct ondisk_cache_entry *)&ondisk) < 0) if (ce_write_entry(f, ce, previous_name, (struct ondisk_cache_entry *)&ondisk) < 0)
err = -1; err = -1;
if (err) if (err)
@ -2985,14 +2915,16 @@ static int do_write_index(struct index_state *istate, struct tempfile *tempfile,
return err; return err;
} }
/* Write extension data here */ offset = hashfile_total(f);
offset = lseek(newfd, 0, SEEK_CUR);
if (offset < 0) { /*
free(ieot); * The extension headers must be hashed on their own for the
return -1; * EOIE extension. Create a hashfile here to compute that hash.
*/
if (offset && record_eoie()) {
CALLOC_ARRAY(eoie_c, 1);
the_hash_algo->init_fn(eoie_c);
} }
offset += write_buffer_len;
the_hash_algo->init_fn(&eoie_c);
/* /*
* Lets write out CACHE_EXT_INDEXENTRYOFFSETTABLE first so that we * Lets write out CACHE_EXT_INDEXENTRYOFFSETTABLE first so that we
@ -3005,8 +2937,8 @@ static int do_write_index(struct index_state *istate, struct tempfile *tempfile,
struct strbuf sb = STRBUF_INIT; struct strbuf sb = STRBUF_INIT;
write_ieot_extension(&sb, ieot); write_ieot_extension(&sb, ieot);
err = write_index_ext_header(&c, &eoie_c, newfd, CACHE_EXT_INDEXENTRYOFFSETTABLE, sb.len) < 0 err = write_index_ext_header(f, eoie_c, CACHE_EXT_INDEXENTRYOFFSETTABLE, sb.len) < 0;
|| ce_write(&c, newfd, sb.buf, sb.len) < 0; hashwrite(f, sb.buf, sb.len);
strbuf_release(&sb); strbuf_release(&sb);
free(ieot); free(ieot);
if (err) if (err)
@ -3018,9 +2950,9 @@ static int do_write_index(struct index_state *istate, struct tempfile *tempfile,
struct strbuf sb = STRBUF_INIT; struct strbuf sb = STRBUF_INIT;
err = write_link_extension(&sb, istate) < 0 || err = write_link_extension(&sb, istate) < 0 ||
write_index_ext_header(&c, &eoie_c, newfd, CACHE_EXT_LINK, write_index_ext_header(f, eoie_c, CACHE_EXT_LINK,
sb.len) < 0 || sb.len) < 0;
ce_write(&c, newfd, sb.buf, sb.len) < 0; hashwrite(f, sb.buf, sb.len);
strbuf_release(&sb); strbuf_release(&sb);
if (err) if (err)
return -1; return -1;
@ -3029,8 +2961,8 @@ static int do_write_index(struct index_state *istate, struct tempfile *tempfile,
struct strbuf sb = STRBUF_INIT; struct strbuf sb = STRBUF_INIT;
cache_tree_write(&sb, istate->cache_tree); cache_tree_write(&sb, istate->cache_tree);
err = write_index_ext_header(&c, &eoie_c, newfd, CACHE_EXT_TREE, sb.len) < 0 err = write_index_ext_header(f, eoie_c, CACHE_EXT_TREE, sb.len) < 0;
|| ce_write(&c, newfd, sb.buf, sb.len) < 0; hashwrite(f, sb.buf, sb.len);
strbuf_release(&sb); strbuf_release(&sb);
if (err) if (err)
return -1; return -1;
@ -3039,9 +2971,9 @@ static int do_write_index(struct index_state *istate, struct tempfile *tempfile,
struct strbuf sb = STRBUF_INIT; struct strbuf sb = STRBUF_INIT;
resolve_undo_write(&sb, istate->resolve_undo); resolve_undo_write(&sb, istate->resolve_undo);
err = write_index_ext_header(&c, &eoie_c, newfd, CACHE_EXT_RESOLVE_UNDO, err = write_index_ext_header(f, eoie_c, CACHE_EXT_RESOLVE_UNDO,
sb.len) < 0 sb.len) < 0;
|| ce_write(&c, newfd, sb.buf, sb.len) < 0; hashwrite(f, sb.buf, sb.len);
strbuf_release(&sb); strbuf_release(&sb);
if (err) if (err)
return -1; return -1;
@ -3050,9 +2982,9 @@ static int do_write_index(struct index_state *istate, struct tempfile *tempfile,
struct strbuf sb = STRBUF_INIT; struct strbuf sb = STRBUF_INIT;
write_untracked_extension(&sb, istate->untracked); write_untracked_extension(&sb, istate->untracked);
err = write_index_ext_header(&c, &eoie_c, newfd, CACHE_EXT_UNTRACKED, err = write_index_ext_header(f, eoie_c, CACHE_EXT_UNTRACKED,
sb.len) < 0 || sb.len) < 0;
ce_write(&c, newfd, sb.buf, sb.len) < 0; hashwrite(f, sb.buf, sb.len);
strbuf_release(&sb); strbuf_release(&sb);
if (err) if (err)
return -1; return -1;
@ -3061,14 +2993,14 @@ static int do_write_index(struct index_state *istate, struct tempfile *tempfile,
struct strbuf sb = STRBUF_INIT; struct strbuf sb = STRBUF_INIT;
write_fsmonitor_extension(&sb, istate); write_fsmonitor_extension(&sb, istate);
err = write_index_ext_header(&c, &eoie_c, newfd, CACHE_EXT_FSMONITOR, sb.len) < 0 err = write_index_ext_header(f, eoie_c, CACHE_EXT_FSMONITOR, sb.len) < 0;
|| ce_write(&c, newfd, sb.buf, sb.len) < 0; hashwrite(f, sb.buf, sb.len);
strbuf_release(&sb); strbuf_release(&sb);
if (err) if (err)
return -1; return -1;
} }
if (istate->sparse_index) { if (istate->sparse_index) {
if (write_index_ext_header(&c, &eoie_c, newfd, CACHE_EXT_SPARSE_DIRECTORIES, 0) < 0) if (write_index_ext_header(f, eoie_c, CACHE_EXT_SPARSE_DIRECTORIES, 0) < 0)
return -1; return -1;
} }
@ -3078,19 +3010,18 @@ static int do_write_index(struct index_state *istate, struct tempfile *tempfile,
* read. Write it out regardless of the strip_extensions parameter as we need it * read. Write it out regardless of the strip_extensions parameter as we need it
* when loading the shared index. * when loading the shared index.
*/ */
if (offset && record_eoie()) { if (eoie_c) {
struct strbuf sb = STRBUF_INIT; struct strbuf sb = STRBUF_INIT;
write_eoie_extension(&sb, &eoie_c, offset); write_eoie_extension(&sb, eoie_c, offset);
err = write_index_ext_header(&c, NULL, newfd, CACHE_EXT_ENDOFINDEXENTRIES, sb.len) < 0 err = write_index_ext_header(f, NULL, CACHE_EXT_ENDOFINDEXENTRIES, sb.len) < 0;
|| ce_write(&c, newfd, sb.buf, sb.len) < 0; hashwrite(f, sb.buf, sb.len);
strbuf_release(&sb); strbuf_release(&sb);
if (err) if (err)
return -1; return -1;
} }
if (ce_flush(&c, newfd, istate->oid.hash)) finalize_hashfile(f, istate->oid.hash, CSUM_HASH_IN_STREAM);
return -1;
if (close_tempfile_gently(tempfile)) { if (close_tempfile_gently(tempfile)) {
error(_("could not close '%s'"), get_tempfile_path(tempfile)); error(_("could not close '%s'"), get_tempfile_path(tempfile));
return -1; return -1;