16486b015SJeremie Le Hen /*-
2*4d846d26SWarner Losh * SPDX-License-Identifier: BSD-2-Clause
35e53a4f9SPedro F. Giffuni *
46486b015SJeremie Le Hen * Copyright (c) 2012 Jeremie Le Hen <jlh@FreeBSD.org>
56486b015SJeremie Le Hen * All rights reserved.
66486b015SJeremie Le Hen *
76486b015SJeremie Le Hen * Redistribution and use in source and binary forms, with or without
86486b015SJeremie Le Hen * modification, are permitted provided that the following conditions
96486b015SJeremie Le Hen * are met:
106486b015SJeremie Le Hen * 1. Redistributions of source code must retain the above copyright
116486b015SJeremie Le Hen * notice, this list of conditions and the following disclaimer.
126486b015SJeremie Le Hen * 2. Redistributions in binary form must reproduce the above copyright
136486b015SJeremie Le Hen * notice, this list of conditions and the following disclaimer in the
146486b015SJeremie Le Hen * documentation and/or other materials provided with the distribution.
156486b015SJeremie Le Hen *
166486b015SJeremie Le Hen * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
176486b015SJeremie Le Hen * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
186486b015SJeremie Le Hen * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
196486b015SJeremie Le Hen * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
206486b015SJeremie Le Hen * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
216486b015SJeremie Le Hen * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
226486b015SJeremie Le Hen * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
236486b015SJeremie Le Hen * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
246486b015SJeremie Le Hen * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
256486b015SJeremie Le Hen * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
266486b015SJeremie Le Hen * SUCH DAMAGE.
276486b015SJeremie Le Hen */
286486b015SJeremie Le Hen
296486b015SJeremie Le Hen #include <err.h>
306486b015SJeremie Le Hen #include <errno.h>
316486b015SJeremie Le Hen #include <limits.h>
326486b015SJeremie Le Hen #include <stdio.h>
336486b015SJeremie Le Hen #include <stdlib.h>
346486b015SJeremie Le Hen #include <string.h>
356486b015SJeremie Le Hen
366486b015SJeremie Le Hen static const char *
stream_name(FILE * s)376486b015SJeremie Le Hen stream_name(FILE *s)
386486b015SJeremie Le Hen {
396486b015SJeremie Le Hen
406486b015SJeremie Le Hen if (s == stdin)
416486b015SJeremie Le Hen return "stdin";
426486b015SJeremie Le Hen if (s == stdout)
436486b015SJeremie Le Hen return "stdout";
446486b015SJeremie Le Hen if (s == stderr)
456486b015SJeremie Le Hen return "stderr";
466486b015SJeremie Le Hen /* This should not happen. */
476486b015SJeremie Le Hen abort();
486486b015SJeremie Le Hen }
496486b015SJeremie Le Hen
506486b015SJeremie Le Hen static void
change_buf(FILE * s,const char * bufmode)516486b015SJeremie Le Hen change_buf(FILE *s, const char *bufmode)
526486b015SJeremie Le Hen {
536486b015SJeremie Le Hen char *unit;
546486b015SJeremie Le Hen size_t bufsize;
556486b015SJeremie Le Hen int mode;
566486b015SJeremie Le Hen
576486b015SJeremie Le Hen bufsize = 0;
586486b015SJeremie Le Hen if (bufmode[0] == '0' && bufmode[1] == '\0')
596486b015SJeremie Le Hen mode = _IONBF;
606486b015SJeremie Le Hen else if (bufmode[0] == 'L' && bufmode[1] == '\0')
616486b015SJeremie Le Hen mode = _IOLBF;
626486b015SJeremie Le Hen else if (bufmode[0] == 'B' && bufmode[1] == '\0') {
636486b015SJeremie Le Hen mode = _IOFBF;
646486b015SJeremie Le Hen bufsize = 0;
656486b015SJeremie Le Hen } else {
666486b015SJeremie Le Hen /*
676486b015SJeremie Le Hen * This library being preloaded, depending on libutil
686486b015SJeremie Le Hen * would lead to excessive namespace pollution.
696486b015SJeremie Le Hen * Thus we do not use expand_number().
706486b015SJeremie Le Hen */
716486b015SJeremie Le Hen errno = 0;
726486b015SJeremie Le Hen bufsize = strtol(bufmode, &unit, 0);
736486b015SJeremie Le Hen if (errno == EINVAL || errno == ERANGE || unit == bufmode)
746486b015SJeremie Le Hen warn("Wrong buffer mode '%s' for %s", bufmode,
756486b015SJeremie Le Hen stream_name(s));
766486b015SJeremie Le Hen switch (*unit) {
776486b015SJeremie Le Hen case 'G':
786486b015SJeremie Le Hen bufsize *= 1024 * 1024 * 1024;
796486b015SJeremie Le Hen break;
806486b015SJeremie Le Hen case 'M':
816486b015SJeremie Le Hen bufsize *= 1024 * 1024;
826486b015SJeremie Le Hen break;
836486b015SJeremie Le Hen case 'k':
846486b015SJeremie Le Hen bufsize *= 1024;
856486b015SJeremie Le Hen break;
866486b015SJeremie Le Hen case '\0':
876486b015SJeremie Le Hen break;
886486b015SJeremie Le Hen default:
896486b015SJeremie Le Hen warnx("Unknown suffix '%c' for %s", *unit,
906486b015SJeremie Le Hen stream_name(s));
916486b015SJeremie Le Hen return;
926486b015SJeremie Le Hen }
936486b015SJeremie Le Hen mode = _IOFBF;
946486b015SJeremie Le Hen }
956486b015SJeremie Le Hen if (setvbuf(s, NULL, mode, bufsize) != 0)
966486b015SJeremie Le Hen warn("Cannot set buffer mode '%s' for %s", bufmode,
976486b015SJeremie Le Hen stream_name(s));
986486b015SJeremie Le Hen }
996486b015SJeremie Le Hen
1006486b015SJeremie Le Hen __attribute__ ((constructor)) static void
stdbuf(void)1016486b015SJeremie Le Hen stdbuf(void)
1026486b015SJeremie Le Hen {
1036486b015SJeremie Le Hen char *i_mode, *o_mode, *e_mode;
1046486b015SJeremie Le Hen
1056486b015SJeremie Le Hen i_mode = getenv("_STDBUF_I");
1066486b015SJeremie Le Hen o_mode = getenv("_STDBUF_O");
1076486b015SJeremie Le Hen e_mode = getenv("_STDBUF_E");
1086486b015SJeremie Le Hen
1096486b015SJeremie Le Hen if (e_mode != NULL)
1106486b015SJeremie Le Hen change_buf(stderr, e_mode);
1116486b015SJeremie Le Hen if (i_mode != NULL)
1126486b015SJeremie Le Hen change_buf(stdin, i_mode);
1136486b015SJeremie Le Hen if (o_mode != NULL)
1146486b015SJeremie Le Hen change_buf(stdout, o_mode);
1156486b015SJeremie Le Hen }
116