xref: /freebsd/lib/libstdbuf/stdbuf.c (revision b3e7694832e81d7a904a10f525f8797b753bf0d3)
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