2003-03-19 09:13:01 +00:00
|
|
|
/* vi: set sw=4 ts=4: */
|
|
|
|
/*
|
|
|
|
* *printf implementations for busybox
|
|
|
|
*
|
|
|
|
* Copyright (C) 2003 Manuel Novoa III <mjn3@codepoet.org>
|
|
|
|
*
|
2006-03-23 15:30:26 +00:00
|
|
|
* Licensed under GPLv2 or later, see file LICENSE in this tarball for details.
|
2003-03-19 09:13:01 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
/* 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.
|
|
|
|
*
|
2004-04-14 17:51:38 +00:00
|
|
|
* In some cases, it would be desirable to have a group of *printf()
|
2003-03-19 09:13:01 +00:00
|
|
|
* 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().
|
2006-03-23 15:30:26 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
/* Mar 22, 2006 Rich Felker III
|
2003-03-19 09:13:01 +00:00
|
|
|
*
|
2006-03-23 15:30:26 +00:00
|
|
|
* 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.
|
2003-03-19 09:13:01 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdarg.h>
|
2006-03-23 15:30:26 +00:00
|
|
|
#include <unistd.h>
|
|
|
|
#include <errno.h>
|
2003-03-19 09:13:01 +00:00
|
|
|
#include "libbb.h"
|
|
|
|
|
2006-03-29 16:52:56 +00:00
|
|
|
int bb_vfprintf(FILE * __restrict stream,
|
2003-03-19 09:13:01 +00:00
|
|
|
const char * __restrict format,
|
|
|
|
va_list arg)
|
|
|
|
{
|
|
|
|
int rv;
|
|
|
|
|
|
|
|
if ((rv = vfprintf(stream, format, arg)) < 0) {
|
2006-03-23 15:30:26 +00:00
|
|
|
/* 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;
|
2003-03-19 09:13:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
2006-03-06 20:47:33 +00:00
|
|
|
int bb_vprintf(const char * __restrict format, va_list arg)
|
2003-03-19 09:13:01 +00:00
|
|
|
{
|
|
|
|
return bb_vfprintf(stdout, format, arg);
|
|
|
|
}
|
|
|
|
|
2006-03-29 16:52:56 +00:00
|
|
|
int bb_fprintf(FILE * __restrict stream,
|
2003-03-19 09:13:01 +00:00
|
|
|
const char * __restrict format, ...)
|
|
|
|
{
|
|
|
|
va_list arg;
|
|
|
|
int rv;
|
|
|
|
|
|
|
|
va_start(arg, format);
|
|
|
|
rv = bb_vfprintf(stream, format, arg);
|
|
|
|
va_end(arg);
|
|
|
|
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
2006-03-06 20:47:33 +00:00
|
|
|
int bb_printf(const char * __restrict format, ...)
|
2003-03-19 09:13:01 +00:00
|
|
|
{
|
|
|
|
va_list arg;
|
|
|
|
int rv;
|
|
|
|
|
|
|
|
va_start(arg, format);
|
|
|
|
rv = bb_vfprintf(stdout, format, arg);
|
|
|
|
va_end(arg);
|
|
|
|
|
|
|
|
return rv;
|
|
|
|
}
|