1 /* Functions for recorded errors, warnings, and verbose messages.
2    Copyright (C) 1998-2022 Free Software Foundation, Inc.
3    This file is part of the GNU C Library.
4 
5    This program is free software; you can redistribute it and/or modify
6    it under the terms of the GNU General Public License as published
7    by the Free Software Foundation; version 2 of the License, or
8    (at your option) any later version.
9 
10    This program is distributed in the hope that it will be useful,
11    but WITHOUT ANY WARRANTY; without even the implied warranty of
12    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
13    GNU General Public License for more details.
14 
15    You should have received a copy of the GNU General Public License
16    along with this program; if not, see <https://www.gnu.org/licenses/>.  */
17 
18 #include <stdio.h>
19 #include <stdlib.h>
20 #include <stdarg.h>
21 #include <stdbool.h>
22 #include <string.h>
23 #include <error.h>
24 #include <errno.h>
25 #include <locale.h>
26 
27 #include "record-status.h"
28 
29 /* Warnings recorded by record_warnings.  */
30 int recorded_warning_count;
31 
32 /* Errors recorded by record_errors.  */
33 int recorded_error_count;
34 
35 /* If not zero suppress warnings and information messages.  */
36 int be_quiet;
37 
38 /* If not zero give a lot more messages.  */
39 int verbose;
40 
41 /* Warnings which can be disabled:  */
42 /* By default we check the character map for ASCII compatibility.  */
43 bool warn_ascii = true;
44 /* By default we check that the international currency symbol matches a
45    known country code.  */
46 bool warn_int_curr_symbol = true;
47 
48 /* Alter the current locale to match the locale configured by the
49    user, and return the previous saved state.  */
50 struct locale_state
push_locale(void)51 push_locale (void)
52 {
53   int saved_errno;
54   const char *orig;
55   char *copy = NULL;
56 
57   saved_errno = errno;
58 
59   orig = setlocale (LC_CTYPE, NULL);
60   if (orig == NULL)
61     error (0, 0, "failed to read locale!");
62 
63   if (setlocale (LC_CTYPE, "") == NULL)
64     error (0, 0, "failed to set locale!");
65 
66   errno = saved_errno;
67 
68   if (orig != NULL)
69     copy = strdup (orig);
70 
71   /* We will return either a valid locale or NULL if we failed
72      to save the locale.  */
73   return (struct locale_state) { .cur_locale = copy };
74 }
75 
76 /* Use the saved state to restore the locale.  */
77 void
pop_locale(struct locale_state ls)78 pop_locale (struct locale_state ls)
79 {
80   const char *set = NULL;
81   /* We might have failed to save the locale, so only attempt to
82      restore a validly saved non-NULL locale.  */
83   if (ls.cur_locale != NULL)
84     {
85       set = setlocale (LC_CTYPE, ls.cur_locale);
86       if (set == NULL)
87 	error (0, 0, "failed to restore %s locale!", ls.cur_locale);
88 
89       free (ls.cur_locale);
90     }
91 }
92 
93 /* Wrapper to print verbose informative messages.
94    Verbose messages are only printed if --verbose
95    is in effect and --quiet is not.  */
96 void
97 __attribute__ ((__format__ (__printf__, 2, 3), nonnull (1, 2), unused))
record_verbose(FILE * stream,const char * format,...)98 record_verbose (FILE *stream, const char *format, ...)
99 {
100   char *str;
101   va_list arg;
102 
103   if (!verbose)
104     return;
105 
106   if (!be_quiet)
107     {
108       struct locale_state ls;
109       int ret;
110 
111       va_start (arg, format);
112       ls = push_locale ();
113 
114       ret = vasprintf (&str, format, arg);
115       if (ret == -1)
116 	abort ();
117 
118       pop_locale (ls);
119       va_end (arg);
120 
121       fprintf (stream, "[verbose] %s\n", str);
122 
123       free (str);
124     }
125 }
126 
127 /* Wrapper to print warning messages.  We keep track of how
128    many were called because this effects our exit code.
129    Nothing is printed if --quiet is in effect, but warnings
130    are always counted.  */
131 void
132 __attribute__ ((__format__ (__printf__, 1, 2), nonnull (1), unused))
record_warning(const char * format,...)133 record_warning (const char *format, ...)
134 {
135   char *str;
136   va_list arg;
137 
138   recorded_warning_count++;
139 
140   if (!be_quiet)
141     {
142       struct locale_state ls;
143       int ret;
144 
145       va_start (arg, format);
146       ls = push_locale ();
147 
148       ret = vasprintf (&str, format, arg);
149       if (ret == -1)
150 	abort ();
151 
152       pop_locale (ls);
153       va_end (arg);
154 
155       fprintf (stderr, "[warning] %s\n", str);
156 
157       free (str);
158     }
159 }
160 
161 /* Wrapper to print error messages.  We keep track of how
162    many were called because this effects our exit code.
163    Nothing is printed if --quiet is in effect, but errors
164    are always counted, and fatal errors always exit the
165    program.  */
166 void
167 __attribute__ ((__format__ (__printf__, 3, 4), nonnull (3), unused))
record_error(int status,int errnum,const char * format,...)168 record_error (int status, int errnum, const char *format, ...)
169 {
170   char *str;
171   va_list arg;
172 
173   recorded_error_count++;
174 
175   /* The existing behaviour is that even if you use --quiet, a fatal
176      error is always printed and terminates the process.  */
177   if (!be_quiet || status != 0)
178     {
179       struct locale_state ls;
180       int ret;
181 
182       va_start (arg, format);
183       ls = push_locale ();
184 
185       ret = vasprintf (&str, format, arg);
186       if (ret == -1)
187         abort ();
188 
189       pop_locale (ls);
190       va_end (arg);
191 
192       error (status, errnum, "[error] %s", str);
193 
194       free (str);
195     }
196 }
197 /* ... likewise for error_at_line.  */
198 void
199 __attribute__ ((__format__ (__printf__, 5, 6), nonnull (3, 5), unused))
record_error_at_line(int status,int errnum,const char * filename,unsigned int linenum,const char * format,...)200 record_error_at_line (int status, int errnum, const char *filename,
201 		      unsigned int linenum, const char *format, ...)
202 {
203   char *str;
204   va_list arg;
205 
206   recorded_error_count++;
207 
208   /* The existing behaviour is that even if you use --quiet, a fatal
209      error is always printed and terminates the process.  */
210   if (!be_quiet || status != 0)
211     {
212       struct locale_state ls;
213       int ret;
214 
215       va_start (arg, format);
216       ls = push_locale ();
217 
218       ret = vasprintf (&str, format, arg);
219       if (ret == -1)
220         abort ();
221 
222       pop_locale (ls);
223       va_end (arg);
224 
225       error_at_line (status, errnum, filename, linenum, "[error] %s", str);
226 
227       free (str);
228     }
229 }
230