/* See LICENSE file for copyright and license details. */
#include "libsimple.h"
#ifndef TEST
extern char *argv0;
int libsimple_default_failure_exit = 1;
void (*libsimple_eprintf_preprint)(void) = NULL;
void (*libsimple_eprintf_postprint)(void) = NULL;
void
libsimple_vweprintf(const char *fmt, va_list ap)
{
int saved_errno = errno, r;
const char *end = strchr(fmt, '\0');
const char *prefix1 = argv0;
const char *prefix2 = ": ";
const char *suffix1 = "";
const char *suffix2 = "";
const char *suffix3 = "";
char *message = NULL;
va_list ap1;
va_list ap2;
if (!argv0 || !strncmp(fmt, "usage: ", sizeof("usage: ") - 1))
prefix1 = prefix2 = "";
va_copy(ap1, ap);
va_copy(ap2, ap);
r = vsnprintf(NULL, 0, fmt, ap1);
if (0 <= r && r < 8096) {
message = alloca((size_t)r + 1);
vsprintf(message, fmt, ap2);
}
va_end(ap2);
va_end(ap1);
if (!*fmt) {
suffix1 = strerror(saved_errno);
suffix2 = "\n";
} else if (end[-1] == ':') {
suffix1 = " ";
suffix2 = strerror(saved_errno);
suffix3 = "\n";
} else if (end[-1] != '\n') {
suffix1 = "\n";
}
if (libsimple_eprintf_preprint)
libsimple_eprintf_preprint();
if (message) {
/* This is to avoid mangling when multiple processes are writting. */
fprintf(stderr, "%s%s%s%s%s%s", prefix1, prefix2, message, suffix1, suffix2, suffix3);
} else {
fprintf(stderr, "%s%s", prefix1, prefix2);
vfprintf(stderr, fmt, ap);
fprintf(stderr, "%s%s%s", suffix1, suffix2, suffix3);
}
if (libsimple_eprintf_postprint)
libsimple_eprintf_postprint();
errno = saved_errno;
}
#else
#include "test.h"
#if defined(__GNUC__) && !defined(__clang__)
# pragma GCC diagnostic ignored "-Wformat-zero-length"
#endif
static void
preprint(void)
{
test_fprintf(stderr, "pre\n");
}
static void
postprint(void)
{
test_fprintf(stderr, "post\n");
}
int
main(void)
{
char prefix[100];
int i, j;
#define T(...)\
do {\
stderr_n = 0;\
if (j == 0) {\
libsimple_weprintf(__VA_ARGS__);\
} else if (j == 1) {\
assert_exit(libsimple_enprintf(107, __VA_ARGS__));\
assert(exit_status == 107);\
} else {\
libsimple_default_failure_exit = 108;\
assert_exit(libsimple_eprintf(__VA_ARGS__));\
assert(exit_status == 108);\
}\
} while (0)
sprintf(prefix, "%s: ", argv0);
stderr_ok = 1;
for (i = 0; i < 2; i++, argv0 = NULL, *prefix = '\0') {
for (j = 0; j < 3; j++) {
T("%s%s%s\n", "abc", "123", "xyz");
assert_stderr("%s%s\n", prefix, "abc123xyz");
libsimple_eprintf_preprint = preprint;
T("%s%s%s\n", "abc", "123", "\n");
assert_stderr("pre\n%s%s\n", prefix, "abc123\n");
libsimple_eprintf_preprint = NULL;
libsimple_eprintf_postprint = postprint;
T("%s%s%s", "abc", "123", "xyz");
assert_stderr("%s%s\npost\n", prefix, "abc123xyz");
libsimple_eprintf_postprint = NULL;
T("%s%s%s", "abc", "123", "\n");
assert_stderr("%s%s\n", prefix, "abc123\n");
errno = EDOM;
libsimple_eprintf_preprint = preprint;
libsimple_eprintf_postprint = postprint;
T("%s%s%s:", "abc", "123", "\n");
assert_stderr("pre\n%s%s: %s\npost\n", prefix, "abc123\n", strerror(EDOM));
libsimple_eprintf_preprint = NULL;
libsimple_eprintf_postprint = NULL;
errno = ERANGE;
T("%s%s%s:", "abc", "123", "\n");
assert_stderr("%s%s: %s\n", prefix, "abc123\n", strerror(ERANGE));
errno = EOVERFLOW;
T("");
assert_stderr("%s%s\n", prefix, strerror(EOVERFLOW));
T("%s%s%s\n", "usage: ", "123", "xyz");
assert_stderr("%s%s\n", prefix, "usage: 123xyz");
T("%s%s%s\n", "usage: ", "123", "\n");
assert_stderr("%s%s\n", prefix, "usage: 123\n");
T("%s%s%s", "usage: ", "123", "xyz");
assert_stderr("%s%s\n", prefix, "usage: 123xyz");
T("%s%s%s", "usage: ", "123", "\n");
assert_stderr("%s%s\n", prefix, "usage: 123\n");
errno = EDOM;
T("%s%s%s:", "usage: ", "123", "\n");
assert_stderr("%s%s: %s\n", prefix, "usage: 123\n", strerror(EDOM));
errno = ERANGE;
T("%s%s%s:", "usage: ", "123", "\n");
assert_stderr("%s%s: %s\n", prefix, "usage: 123\n", strerror(ERANGE));
T("usage: %s%s%s\n", "abc", "123", "xyz");
assert_stderr("%s: %s\n", "usage", "abc123xyz");
T("usage: %s%s%s\n", "abc", "123", "\n");
assert_stderr("%s: %s\n", "usage", "abc123\n");
T("usage: %s%s%s", "abc", "123", "xyz");
assert_stderr("%s: %s\n", "usage", "abc123xyz");
T("usage: %s%s%s", "abc", "123", "\n");
assert_stderr("%s: %s\n", "usage", "abc123\n");
errno = EDOM;
T("usage: %s%s%s:", "abc", "123", "\n");
assert_stderr("%s: %s: %s\n", "usage", "abc123\n", strerror(EDOM));
errno = ERANGE;
T("usage: %s%s%s:", "abc", "123", "\n");
assert_stderr("%s: %s: %s\n", "usage", "abc123\n", strerror(ERANGE));
T("usage: ");
assert_stderr("%s", "usage: \n");
}
}
#undef T
return 0;
}
#endif