build: update gnulib submodule to latest
[coreutils.git] / src / yes.c
blob07885be9b402370d9c670d4e9c97a66ffe4da5f3
1 /* yes - output a string repeatedly until killed
2 Copyright (C) 1991-2015 Free Software Foundation, Inc.
4 This program is free software: you can redistribute it and/or modify
5 it under the terms of the GNU General Public License as published by
6 the Free Software Foundation, either version 3 of the License, or
7 (at your option) any later version.
9 This program is distributed in the hope that it will be useful,
10 but WITHOUT ANY WARRANTY; without even the implied warranty of
11 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 GNU General Public License for more details.
14 You should have received a copy of the GNU General Public License
15 along with this program. If not, see <http://www.gnu.org/licenses/>. */
17 /* David MacKenzie <djm@gnu.ai.mit.edu> */
19 #include <config.h>
20 #include <stdio.h>
21 #include <sys/types.h>
22 #include <getopt.h>
24 #include "system.h"
26 #include "error.h"
27 #include "long-options.h"
29 /* The official name of this program (e.g., no 'g' prefix). */
30 #define PROGRAM_NAME "yes"
32 #define AUTHORS proper_name ("David MacKenzie")
34 void
35 usage (int status)
37 if (status != EXIT_SUCCESS)
38 emit_try_help ();
39 else
41 printf (_("\
42 Usage: %s [STRING]...\n\
43 or: %s OPTION\n\
44 "),
45 program_name, program_name);
47 fputs (_("\
48 Repeatedly output a line with all specified STRING(s), or 'y'.\n\
49 \n\
50 "), stdout);
51 fputs (HELP_OPTION_DESCRIPTION, stdout);
52 fputs (VERSION_OPTION_DESCRIPTION, stdout);
53 emit_ancillary_info (PROGRAM_NAME);
55 exit (status);
58 int
59 main (int argc, char **argv)
61 char buf[BUFSIZ];
62 char *pbuf = buf;
63 int i;
65 initialize_main (&argc, &argv);
66 set_program_name (argv[0]);
67 setlocale (LC_ALL, "");
68 bindtextdomain (PACKAGE, LOCALEDIR);
69 textdomain (PACKAGE);
71 atexit (close_stdout);
73 parse_long_options (argc, argv, PROGRAM_NAME, PACKAGE_NAME, Version,
74 usage, AUTHORS, (char const *) NULL);
75 if (getopt_long (argc, argv, "+", NULL, NULL) != -1)
76 usage (EXIT_FAILURE);
78 if (argc <= optind)
80 optind = argc;
81 argv[argc++] = bad_cast ("y");
84 /* Buffer data locally once, rather than having the
85 large overhead of stdio buffering each item. */
86 for (i = optind; i < argc; i++)
88 size_t len = strlen (argv[i]);
89 if (BUFSIZ < len || BUFSIZ - len <= pbuf - buf)
90 break;
91 memcpy (pbuf, argv[i], len);
92 pbuf += len;
93 *pbuf++ = i == argc - 1 ? '\n' : ' ';
95 if (i == argc)
97 size_t line_len = pbuf - buf;
98 size_t lines = BUFSIZ / line_len;
99 while (--lines)
101 memcpy (pbuf, pbuf - line_len, line_len);
102 pbuf += line_len;
106 /* The normal case is to continuously output the local buffer. */
107 while (i == argc)
109 if (write (STDOUT_FILENO, buf, pbuf - buf) == -1)
111 error (0, errno, _("standard output"));
112 return EXIT_FAILURE;
116 /* If the data doesn't fit in BUFSIZ then output
117 what we've buffered, and iterate over the remaining items. */
118 while (true /* i != argc */)
120 int j;
121 if ((pbuf - buf) && fwrite (buf, pbuf - buf, 1, stdout) != 1)
123 error (0, errno, _("standard output"));
124 return EXIT_FAILURE;
126 for (j = i; j < argc; j++)
127 if (fputs (argv[j], stdout) == EOF
128 || putchar (j == argc - 1 ? '\n' : ' ') == EOF)
130 error (0, errno, _("standard output"));
131 return EXIT_FAILURE;