blob: cc9e0787a1de50e9032a06d4036835bc9d1b35b9 [file] [log] [blame]
Elijah Newrend48be352023-03-21 06:26:07 +00001#include "git-compat-util.h"
Eric Wong412e4ca2021-10-29 00:15:52 +00002#include "config.h"
Jeff King96588462016-02-24 02:40:16 -05003#include "run-command.h"
Elijah Newrend5ebb502023-03-21 06:26:01 +00004#include "wrapper.h"
Elijah Newrend48be352023-03-21 06:26:07 +00005#include "write-or-die.h"
Rene Scharfe7230e6d2006-08-21 20:43:43 +02006
Theodore Ts'o06f59e92007-06-29 13:40:46 -04007/*
8 * Some cases use stdio, but want to flush after the write
9 * to get error handling (and to get better interactive
10 * behaviour - not buffering excessively).
11 *
12 * Of course, if the flush happened within the write itself,
13 * we've already lost the error code, and cannot report it any
14 * more. So we just ignore that case instead (and hope we get
15 * the right error code on the flush).
16 *
17 * If the file handle is stdout, and stdout is a file, then skip the
18 * flush entirely since it's not needed.
19 */
20void maybe_flush_or_die(FILE *f, const char *desc)
21{
22 static int skip_stdout_flush = -1;
23 struct stat st;
24 char *cp;
25
26 if (f == stdout) {
27 if (skip_stdout_flush < 0) {
Junio C Hamanofd017952022-09-15 09:06:57 -070028 /* NEEDSWORK: make this a normal Boolean */
Theodore Ts'o06f59e92007-06-29 13:40:46 -040029 cp = getenv("GIT_FLUSH");
30 if (cp)
31 skip_stdout_flush = (atoi(cp) == 0);
32 else if ((fstat(fileno(stdout), &st) == 0) &&
33 S_ISREG(st.st_mode))
34 skip_stdout_flush = 1;
35 else
36 skip_stdout_flush = 0;
37 }
38 if (skip_stdout_flush && !ferror(f))
39 return;
40 }
41 if (fflush(f)) {
Jeff King756e6762013-02-20 15:01:36 -050042 check_pipe(errno);
Thomas Rastd824cbb2009-06-27 17:58:46 +020043 die_errno("write failure on '%s'", desc);
Theodore Ts'o06f59e92007-06-29 13:40:46 -040044 }
45}
46
Jeff King9540ce52014-09-10 06:03:52 -040047void fprintf_or_die(FILE *f, const char *fmt, ...)
48{
49 va_list ap;
50 int ret;
51
52 va_start(ap, fmt);
53 ret = vfprintf(f, fmt, ap);
54 va_end(ap);
55
56 if (ret < 0) {
57 check_pipe(errno);
58 die_errno("write error");
59 }
60}
61
Neeraj Singh020406e2022-03-10 22:43:21 +000062static int maybe_fsync(int fd)
Linus Torvalds4c81b032008-05-30 08:42:16 -070063{
Eric Wong412e4ca2021-10-29 00:15:52 +000064 if (use_fsync < 0)
65 use_fsync = git_env_bool("GIT_TEST_FSYNC", 1);
66 if (!use_fsync)
Neeraj Singh020406e2022-03-10 22:43:21 +000067 return 0;
Neeraj Singhabf38ab2022-03-10 22:43:20 +000068
69 if (fsync_method == FSYNC_METHOD_WRITEOUT_ONLY &&
70 git_fsync(fd, FSYNC_WRITEOUT_ONLY) >= 0)
Neeraj Singh020406e2022-03-10 22:43:21 +000071 return 0;
Neeraj Singhabf38ab2022-03-10 22:43:20 +000072
Neeraj Singh020406e2022-03-10 22:43:21 +000073 return git_fsync(fd, FSYNC_HARDWARE_FLUSH);
74}
75
76void fsync_or_die(int fd, const char *msg)
77{
78 if (maybe_fsync(fd) < 0)
Neeraj Singhabf38ab2022-03-10 22:43:20 +000079 die_errno("fsync error on '%s'", msg);
Linus Torvalds4c81b032008-05-30 08:42:16 -070080}
81
Neeraj Singh020406e2022-03-10 22:43:21 +000082int fsync_component(enum fsync_component component, int fd)
83{
84 if (fsync_components & component)
85 return maybe_fsync(fd);
86 return 0;
87}
88
89void fsync_component_or_die(enum fsync_component component, int fd, const char *msg)
90{
91 if (fsync_components & component)
92 fsync_or_die(fd, msg);
93}
94
Andy Whitcroft93822c22007-01-08 15:58:23 +000095void write_or_die(int fd, const void *buf, size_t count)
96{
Linus Torvaldsd34cf192007-01-11 20:23:00 -080097 if (write_in_full(fd, buf, count) < 0) {
Jeff King756e6762013-02-20 15:01:36 -050098 check_pipe(errno);
Thomas Rastd824cbb2009-06-27 17:58:46 +020099 die_errno("write error");
Andy Whitcroft93822c22007-01-08 15:58:23 +0000100 }
Andy Whitcrofte0814052007-01-08 15:57:52 +0000101}
Jacob Vosmaer96328392021-09-01 14:54:41 +0200102
103void fwrite_or_die(FILE *f, const void *buf, size_t count)
104{
105 if (fwrite(buf, 1, count, f) != count)
106 die_errno("fwrite error");
107}
108
109void fflush_or_die(FILE *f)
110{
111 if (fflush(f))
112 die_errno("fflush error");
113}