Xinqi Bao's Git
1 /* See LICENSE file for copyright and license details. */
13 verr(const char *fmt
, va_list ap
)
15 if (argv0
&& strncmp(fmt
, "usage", sizeof("usage") - 1)) {
16 fprintf(stderr
, "%s: ", argv0
);
19 vfprintf(stderr
, fmt
, ap
);
21 if (fmt
[0] && fmt
[strlen(fmt
) - 1] == ':') {
30 warn(const char *fmt
, ...)
40 die(const char *fmt
, ...)
52 evsnprintf(char *str
, size_t size
, const char *fmt
, va_list ap
)
56 ret
= vsnprintf(str
, size
, fmt
, ap
);
61 } else if ((size_t)ret
>= size
) {
62 warn("vsnprintf: Output truncated");
70 esnprintf(char *str
, size_t size
, const char *fmt
, ...)
76 ret
= evsnprintf(str
, size
, fmt
, ap
);
83 bprintf(const char *fmt
, ...)
89 ret
= evsnprintf(buf
, sizeof(buf
), fmt
, ap
);
92 return (ret
< 0) ? NULL
: buf
;
96 fmt_human(size_t num
, int base
)
101 const char *prefix_1000
[] = { "", "k", "M", "G", "T", "P", "E", "Z", "Y" };
102 const char *prefix_1024
[] = { "", "Ki", "Mi", "Gi", "Ti", "Pi", "Ei",
107 prefix
= prefix_1000
;
108 prefixlen
= LEN(prefix_1000
);
111 prefix
= prefix_1024
;
112 prefixlen
= LEN(prefix_1024
);
115 warn("fmt_human: Invalid base");
120 for (i
= 0; i
< prefixlen
&& scaled
>= base
; i
++) {
124 return bprintf("%.1f%s", scaled
, prefix
[i
]);
128 pscanf(const char *path
, const char *fmt
, ...)
134 if (!(fp
= fopen(path
, "r"))) {
135 warn("fopen '%s':", path
);
139 n
= vfscanf(fp
, fmt
, ap
);
143 return (n
== EOF
) ? -1 : n
;