1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
|
/* vi: set sw=4 ts=4: */
/*
* *printf implementations for busybox
*
* Copyright (C) 2003 Manuel Novoa III <mjn3@codepoet.org>
*
* Licensed under GPLv2 or later, see file LICENSE in this tarball for details.
*/
/* Mar 12, 2003 Manuel Novoa III
*
* While fwrite(), fputc(), fputs(), etc. all set the stream error flag
* on failure, the *printf functions are unique in that they can fail
* for reasons not related to the actual output itself. Among the possible
* reasons for failure which don't set the streams error indicator,
* SUSv3 lists EILSEQ, EINVAL, and ENOMEM.
*
* In some cases, it would be desirable to have a group of *printf()
* functions available that _always_ set the stream error indicator on
* failure. That would allow us to defer error checking until applet
* exit. Unfortunately, there is no standard way of setting a streams
* error indicator... even though we can clear it with clearerr().
*/
/* Mar 22, 2006 Rich Felker III
*
* Actually there is a portable way to set the error indicator. See below.
* It is not thread-safe as written due to a race condition with file
* descriptors but since BB is not threaded that does not matter. It can be
* made thread-safe at the expense of slightly more code, if this is ever
* needed in the future.
*/
#include <stdio.h>
#include <stdarg.h>
#include <unistd.h>
#include <errno.h>
#include "libbb.h"
#ifdef L_bb_vfprintf
extern int bb_vfprintf(FILE * __restrict stream,
const char * __restrict format,
va_list arg)
{
int rv;
if ((rv = vfprintf(stream, format, arg)) < 0) {
/* The following sequence portably sets the error flag for
* stream on any remotely POSIX-compliant implementation. */
int errno_save = errno;
int fd = fileno(stream);
int tmp = dup(fd);
fflush(stream);
close(fd);
/* Force an attempted write to nonexistant fd => EBADF */
fputc(0, stream);
fflush(stream);
/* Restore the stream's original fd */
dup2(tmp, fd);
close(tmp);
errno = errno_save;
}
return rv;
}
#endif
#ifdef L_bb_vprintf
int bb_vprintf(const char * __restrict format, va_list arg)
{
return bb_vfprintf(stdout, format, arg);
}
#endif
#ifdef L_bb_fprintf
extern int bb_fprintf(FILE * __restrict stream,
const char * __restrict format, ...)
{
va_list arg;
int rv;
va_start(arg, format);
rv = bb_vfprintf(stream, format, arg);
va_end(arg);
return rv;
}
#endif
#ifdef L_bb_printf
int bb_printf(const char * __restrict format, ...)
{
va_list arg;
int rv;
va_start(arg, format);
rv = bb_vfprintf(stdout, format, arg);
va_end(arg);
return rv;
}
#endif
|