VirtualBox

source: kBuild/trunk/src/gmake/misc.c@ 530

Last change on this file since 530 was 527, checked in by bird, 19 years ago

CONFIG_WITH_OPTIMIZATION_HACKS

  • Property svn:eol-style set to native
File size: 21.0 KB
Line 
1/* Miscellaneous generic support functions for GNU Make.
2Copyright (C) 1988, 1989, 1990, 1991, 1992, 1993, 1994, 1995, 1996, 1997,
31998, 1999, 2000, 2001, 2002, 2003, 2004, 2005, 2006 Free Software
4Foundation, Inc.
5This file is part of GNU Make.
6
7GNU Make is free software; you can redistribute it and/or modify it under the
8terms of the GNU General Public License as published by the Free Software
9Foundation; either version 2, or (at your option) any later version.
10
11GNU Make is distributed in the hope that it will be useful, but WITHOUT ANY
12WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR
13A PARTICULAR PURPOSE. See the GNU General Public License for more details.
14
15You should have received a copy of the GNU General Public License along with
16GNU Make; see the file COPYING. If not, write to the Free Software
17Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. */
18
19#include "make.h"
20#include "dep.h"
21#include "debug.h"
22
23#if defined(__EMX__) && defined(CONFIG_WITH_OPTIMIZATION_HACKS) /* bird: saves 5-10ms on libc */
24# define bcopy(src, dst, size) __builtin_memcpy((dst), (src), (size))
25#endif
26
27/* Variadic functions. We go through contortions to allow proper function
28 prototypes for both ANSI and pre-ANSI C compilers, and also for those
29 which support stdarg.h vs. varargs.h, and finally those which have
30 vfprintf(), etc. and those who have _doprnt... or nothing.
31
32 This fancy stuff all came from GNU fileutils, except for the VA_PRINTF and
33 VA_END macros used here since we have multiple print functions. */
34
35#if USE_VARIADIC
36# if HAVE_STDARG_H
37# include <stdarg.h>
38# define VA_START(args, lastarg) va_start(args, lastarg)
39# else
40# include <varargs.h>
41# define VA_START(args, lastarg) va_start(args)
42# endif
43# if HAVE_VPRINTF
44# define VA_PRINTF(fp, lastarg, args) vfprintf((fp), (lastarg), (args))
45# else
46# define VA_PRINTF(fp, lastarg, args) _doprnt((lastarg), (args), (fp))
47# endif
48# define VA_END(args) va_end(args)
49#else
50/* We can't use any variadic interface! */
51# define va_alist a1, a2, a3, a4, a5, a6, a7, a8
52# define va_dcl char *a1, *a2, *a3, *a4, *a5, *a6, *a7, *a8;
53# define VA_START(args, lastarg)
54# define VA_PRINTF(fp, lastarg, args) fprintf((fp), (lastarg), va_alist)
55# define VA_END(args)
56#endif
57
58
59/* Compare strings *S1 and *S2.
60 Return negative if the first is less, positive if it is greater,
61 zero if they are equal. */
62
63int
64alpha_compare (const void *v1, const void *v2)
65{
66 const char *s1 = *((char **)v1);
67 const char *s2 = *((char **)v2);
68
69 if (*s1 != *s2)
70 return *s1 - *s2;
71 return strcmp (s1, s2);
72}
73
74
75/* Discard each backslash-newline combination from LINE.
76 Backslash-backslash-newline combinations become backslash-newlines.
77 This is done by copying the text at LINE into itself. */
78
79void
80collapse_continuations (char *line)
81{
82 register char *in, *out, *p;
83 register int backslash;
84 register unsigned int bs_write;
85
86 in = strchr (line, '\n');
87 if (in == 0)
88 return;
89
90 out = in;
91 while (out > line && out[-1] == '\\')
92 --out;
93
94 while (*in != '\0')
95 {
96 /* BS_WRITE gets the number of quoted backslashes at
97 the end just before IN, and BACKSLASH gets nonzero
98 if the next character is quoted. */
99 backslash = 0;
100 bs_write = 0;
101 for (p = in - 1; p >= line && *p == '\\'; --p)
102 {
103 if (backslash)
104 ++bs_write;
105 backslash = !backslash;
106
107 /* It should be impossible to go back this far without exiting,
108 but if we do, we can't get the right answer. */
109 if (in == out - 1)
110 abort ();
111 }
112
113 /* Output the appropriate number of backslashes. */
114 while (bs_write-- > 0)
115 *out++ = '\\';
116
117 /* Skip the newline. */
118 ++in;
119
120 /* If the newline is quoted, discard following whitespace
121 and any preceding whitespace; leave just one space. */
122 if (backslash)
123 {
124 in = next_token (in);
125 while (out > line && isblank ((unsigned char)out[-1]))
126 --out;
127 *out++ = ' ';
128 }
129 else
130 /* If the newline isn't quoted, put it in the output. */
131 *out++ = '\n';
132
133 /* Now copy the following line to the output.
134 Stop when we find backslashes followed by a newline. */
135 while (*in != '\0')
136 if (*in == '\\')
137 {
138 p = in + 1;
139 while (*p == '\\')
140 ++p;
141 if (*p == '\n')
142 {
143 in = p;
144 break;
145 }
146 while (in < p)
147 *out++ = *in++;
148 }
149 else
150 *out++ = *in++;
151 }
152
153 *out = '\0';
154}
155
156
157/* Print N spaces (used in debug for target-depth). */
158
159void
160print_spaces (unsigned int n)
161{
162 while (n-- > 0)
163 putchar (' ');
164}
165
166
167
168/* Return a newly-allocated string whose contents
169 concatenate those of s1, s2, s3. */
170
171char *
172concat (const char *s1, const char *s2, const char *s3)
173{
174 unsigned int len1, len2, len3;
175 char *result;
176
177 len1 = *s1 != '\0' ? strlen (s1) : 0;
178 len2 = *s2 != '\0' ? strlen (s2) : 0;
179 len3 = *s3 != '\0' ? strlen (s3) : 0;
180
181 result = (char *) xmalloc (len1 + len2 + len3 + 1);
182
183 if (*s1 != '\0')
184 bcopy (s1, result, len1);
185 if (*s2 != '\0')
186 bcopy (s2, result + len1, len2);
187 if (*s3 != '\0')
188 bcopy (s3, result + len1 + len2, len3);
189 *(result + len1 + len2 + len3) = '\0';
190
191 return result;
192}
193
194
195/* Print a message on stdout. */
196
197void
198#if HAVE_ANSI_COMPILER && USE_VARIADIC && HAVE_STDARG_H
199message (int prefix, const char *fmt, ...)
200#else
201message (prefix, fmt, va_alist)
202 int prefix;
203 const char *fmt;
204 va_dcl
205#endif
206{
207#if USE_VARIADIC
208 va_list args;
209#endif
210
211 log_working_directory (1);
212
213 if (fmt != 0)
214 {
215 if (prefix)
216 {
217 if (makelevel == 0)
218 printf ("%s: ", program);
219 else
220 printf ("%s[%u]: ", program, makelevel);
221 }
222 VA_START (args, fmt);
223 VA_PRINTF (stdout, fmt, args);
224 VA_END (args);
225 putchar ('\n');
226 }
227
228 fflush (stdout);
229}
230
231/* Print an error message. */
232
233void
234#if HAVE_ANSI_COMPILER && USE_VARIADIC && HAVE_STDARG_H
235error (const struct floc *flocp, const char *fmt, ...)
236#else
237error (flocp, fmt, va_alist)
238 const struct floc *flocp;
239 const char *fmt;
240 va_dcl
241#endif
242{
243#if USE_VARIADIC
244 va_list args;
245#endif
246
247 log_working_directory (1);
248
249 if (flocp && flocp->filenm)
250 fprintf (stderr, "%s:%lu: ", flocp->filenm, flocp->lineno);
251 else if (makelevel == 0)
252 fprintf (stderr, "%s: ", program);
253 else
254 fprintf (stderr, "%s[%u]: ", program, makelevel);
255
256 VA_START(args, fmt);
257 VA_PRINTF (stderr, fmt, args);
258 VA_END (args);
259
260 putc ('\n', stderr);
261 fflush (stderr);
262}
263
264/* Print an error message and exit. */
265
266void
267#if HAVE_ANSI_COMPILER && USE_VARIADIC && HAVE_STDARG_H
268fatal (const struct floc *flocp, const char *fmt, ...)
269#else
270fatal (flocp, fmt, va_alist)
271 const struct floc *flocp;
272 const char *fmt;
273 va_dcl
274#endif
275{
276#if USE_VARIADIC
277 va_list args;
278#endif
279
280 log_working_directory (1);
281
282 if (flocp && flocp->filenm)
283 fprintf (stderr, "%s:%lu: *** ", flocp->filenm, flocp->lineno);
284 else if (makelevel == 0)
285 fprintf (stderr, "%s: *** ", program);
286 else
287 fprintf (stderr, "%s[%u]: *** ", program, makelevel);
288
289 VA_START(args, fmt);
290 VA_PRINTF (stderr, fmt, args);
291 VA_END (args);
292
293 fputs (_(". Stop.\n"), stderr);
294
295 die (2);
296}
297
298#ifndef HAVE_STRERROR
299
300#undef strerror
301
302char *
303strerror (int errnum)
304{
305 extern int errno, sys_nerr;
306#ifndef __DECC
307 extern char *sys_errlist[];
308#endif
309 static char buf[] = "Unknown error 12345678901234567890";
310
311 if (errno < sys_nerr)
312 return sys_errlist[errnum];
313
314 sprintf (buf, _("Unknown error %d"), errnum);
315 return buf;
316}
317#endif
318
319/* Print an error message from errno. */
320
321void
322perror_with_name (const char *str, const char *name)
323{
324 error (NILF, _("%s%s: %s"), str, name, strerror (errno));
325}
326
327/* Print an error message from errno and exit. */
328
329void
330pfatal_with_name (const char *name)
331{
332 fatal (NILF, _("%s: %s"), name, strerror (errno));
333
334 /* NOTREACHED */
335}
336
337
338/* Like malloc but get fatal error if memory is exhausted. */
339/* Don't bother if we're using dmalloc; it provides these for us. */
340
341#ifndef HAVE_DMALLOC_H
342
343#undef xmalloc
344#undef xrealloc
345#undef xstrdup
346
347char *
348xmalloc (unsigned int size)
349{
350 /* Make sure we don't allocate 0, for pre-ANSI libraries. */
351 char *result = (char *) malloc (size ? size : 1);
352 if (result == 0)
353 fatal (NILF, _("virtual memory exhausted"));
354 return result;
355}
356
357
358char *
359xrealloc (char *ptr, unsigned int size)
360{
361 char *result;
362
363 /* Some older implementations of realloc() don't conform to ANSI. */
364 if (! size)
365 size = 1;
366 result = ptr ? realloc (ptr, size) : malloc (size);
367 if (result == 0)
368 fatal (NILF, _("virtual memory exhausted"));
369 return result;
370}
371
372
373char *
374xstrdup (const char *ptr)
375{
376 char *result;
377
378#ifdef HAVE_STRDUP
379 result = strdup (ptr);
380#else
381 result = (char *) malloc (strlen (ptr) + 1);
382#endif
383
384 if (result == 0)
385 fatal (NILF, _("virtual memory exhausted"));
386
387#ifdef HAVE_STRDUP
388 return result;
389#else
390 return strcpy(result, ptr);
391#endif
392}
393
394#endif /* HAVE_DMALLOC_H */
395
396char *
397savestring (const char *str, unsigned int length)
398{
399 register char *out = (char *) xmalloc (length + 1);
400 if (length > 0)
401 bcopy (str, out, length);
402 out[length] = '\0';
403 return out;
404}
405
406
407
408#ifndef CONFIG_WITH_OPTIMIZATION_HACKS /* This is really a reimplemntation of
409 memchr, only slower. It's been replaced by a macro in the header file. */
410
411/* Limited INDEX:
412 Search through the string STRING, which ends at LIMIT, for the character C.
413 Returns a pointer to the first occurrence, or nil if none is found.
414 Like INDEX except that the string searched ends where specified
415 instead of at the first null. */
416
417char *
418lindex (const char *s, const char *limit, int c)
419{
420 while (s < limit)
421 if (*s++ == c)
422 return (char *)(s - 1);
423
424 return 0;
425}
426#endif /* CONFIG_WITH_OPTIMIZATION_HACKS */
427
428
429/* Return the address of the first whitespace or null in the string S. */
430
431char *
432end_of_token (const char *s)
433{
434 while (*s != '\0' && !isblank ((unsigned char)*s))
435 ++s;
436 return (char *)s;
437}
438
439#ifdef WINDOWS32
440/*
441 * Same as end_of_token, but take into account a stop character
442 */
443char *
444end_of_token_w32 (char *s, char stopchar)
445{
446 register char *p = s;
447 register int backslash = 0;
448
449 while (*p != '\0' && *p != stopchar
450 && (backslash || !isblank ((unsigned char)*p)))
451 {
452 if (*p++ == '\\')
453 {
454 backslash = !backslash;
455 while (*p == '\\')
456 {
457 backslash = !backslash;
458 ++p;
459 }
460 }
461 else
462 backslash = 0;
463 }
464
465 return p;
466}
467#endif
468
469/* Return the address of the first nonwhitespace or null in the string S. */
470
471char *
472next_token (const char *s)
473{
474 while (isblank ((unsigned char)*s))
475 ++s;
476 return (char *)s;
477}
478
479/* Find the next token in PTR; return the address of it, and store the
480 length of the token into *LENGTHPTR if LENGTHPTR is not nil. */
481
482char *
483find_next_token (char **ptr, unsigned int *lengthptr)
484{
485 char *p = next_token (*ptr);
486 char *end;
487
488 if (*p == '\0')
489 return 0;
490
491 *ptr = end = end_of_token (p);
492 if (lengthptr != 0)
493 *lengthptr = end - p;
494 return p;
495}
496
497
498
499/* Allocate a new `struct dep' with all fields initialized to 0. */
500
501struct dep *
502alloc_dep ()
503{
504 struct dep *d = (struct dep *) xmalloc (sizeof (struct dep));
505 bzero ((char *) d, sizeof (struct dep));
506 return d;
507}
508
509
510/* Free `struct dep' along with `name' and `stem'. */
511
512void
513free_dep (struct dep *d)
514{
515 if (d->name != 0)
516 free (d->name);
517
518 if (d->stem != 0)
519 free (d->stem);
520
521 free ((char *)d);
522}
523
524/* Copy a chain of `struct dep', making a new chain
525 with the same contents as the old one. */
526
527struct dep *
528copy_dep_chain (const struct dep *d)
529{
530 register struct dep *c;
531 struct dep *firstnew = 0;
532 struct dep *lastnew = 0;
533
534 while (d != 0)
535 {
536 c = (struct dep *) xmalloc (sizeof (struct dep));
537 bcopy ((char *) d, (char *) c, sizeof (struct dep));
538
539 if (c->name != 0)
540 c->name = xstrdup (c->name);
541 if (c->stem != 0)
542 c->stem = xstrdup (c->stem);
543
544 c->next = 0;
545 if (firstnew == 0)
546 firstnew = lastnew = c;
547 else
548 lastnew = lastnew->next = c;
549
550 d = d->next;
551 }
552
553 return firstnew;
554}
555
556/* Free a chain of 'struct dep'. */
557
558void
559free_dep_chain (struct dep *d)
560{
561 while (d != 0)
562 {
563 struct dep *df = d;
564 d = d->next;
565 free_dep (df);
566 }
567}
568
569
570/* Free a chain of `struct nameseq'. Each nameseq->name is freed
571 as well. For `struct dep' chains use free_dep_chain. */
572
573void
574free_ns_chain (struct nameseq *n)
575{
576 register struct nameseq *tmp;
577
578 while (n != 0)
579 {
580 if (n->name != 0)
581 free (n->name);
582
583 tmp = n;
584
585 n = n->next;
586
587 free (tmp);
588 }
589
590}
591
592#ifdef iAPX286
593/* The losing compiler on this machine can't handle this macro. */
594
595char *
596dep_name (struct dep *dep)
597{
598 return dep->name == 0 ? dep->file->name : dep->name;
599}
600#endif
601
602
603#ifdef GETLOADAVG_PRIVILEGED
604
605#ifdef POSIX
606
607/* Hopefully if a system says it's POSIX.1 and has the setuid and setgid
608 functions, they work as POSIX.1 says. Some systems (Alpha OSF/1 1.2,
609 for example) which claim to be POSIX.1 also have the BSD setreuid and
610 setregid functions, but they don't work as in BSD and only the POSIX.1
611 way works. */
612
613#undef HAVE_SETREUID
614#undef HAVE_SETREGID
615
616#else /* Not POSIX. */
617
618/* Some POSIX.1 systems have the seteuid and setegid functions. In a
619 POSIX-like system, they are the best thing to use. However, some
620 non-POSIX systems have them too but they do not work in the POSIX style
621 and we must use setreuid and setregid instead. */
622
623#undef HAVE_SETEUID
624#undef HAVE_SETEGID
625
626#endif /* POSIX. */
627
628#ifndef HAVE_UNISTD_H
629extern int getuid (), getgid (), geteuid (), getegid ();
630extern int setuid (), setgid ();
631#ifdef HAVE_SETEUID
632extern int seteuid ();
633#else
634#ifdef HAVE_SETREUID
635extern int setreuid ();
636#endif /* Have setreuid. */
637#endif /* Have seteuid. */
638#ifdef HAVE_SETEGID
639extern int setegid ();
640#else
641#ifdef HAVE_SETREGID
642extern int setregid ();
643#endif /* Have setregid. */
644#endif /* Have setegid. */
645#endif /* No <unistd.h>. */
646
647/* Keep track of the user and group IDs for user- and make- access. */
648static int user_uid = -1, user_gid = -1, make_uid = -1, make_gid = -1;
649#define access_inited (user_uid != -1)
650static enum { make, user } current_access;
651
652
653/* Under -d, write a message describing the current IDs. */
654
655static void
656log_access (const char *flavor)
657{
658 if (! ISDB (DB_JOBS))
659 return;
660
661 /* All the other debugging messages go to stdout,
662 but we write this one to stderr because it might be
663 run in a child fork whose stdout is piped. */
664
665 fprintf (stderr, _("%s: user %lu (real %lu), group %lu (real %lu)\n"),
666 flavor, (unsigned long) geteuid (), (unsigned long) getuid (),
667 (unsigned long) getegid (), (unsigned long) getgid ());
668 fflush (stderr);
669}
670
671
672static void
673init_access (void)
674{
675#ifndef VMS
676 user_uid = getuid ();
677 user_gid = getgid ();
678
679 make_uid = geteuid ();
680 make_gid = getegid ();
681
682 /* Do these ever fail? */
683 if (user_uid == -1 || user_gid == -1 || make_uid == -1 || make_gid == -1)
684 pfatal_with_name ("get{e}[gu]id");
685
686 log_access (_("Initialized access"));
687
688 current_access = make;
689#endif
690}
691
692#endif /* GETLOADAVG_PRIVILEGED */
693
694/* Give the process appropriate permissions for access to
695 user data (i.e., to stat files, or to spawn a child process). */
696void
697user_access (void)
698{
699#ifdef GETLOADAVG_PRIVILEGED
700
701 if (!access_inited)
702 init_access ();
703
704 if (current_access == user)
705 return;
706
707 /* We are in "make access" mode. This means that the effective user and
708 group IDs are those of make (if it was installed setuid or setgid).
709 We now want to set the effective user and group IDs to the real IDs,
710 which are the IDs of the process that exec'd make. */
711
712#ifdef HAVE_SETEUID
713
714 /* Modern systems have the seteuid/setegid calls which set only the
715 effective IDs, which is ideal. */
716
717 if (seteuid (user_uid) < 0)
718 pfatal_with_name ("user_access: seteuid");
719
720#else /* Not HAVE_SETEUID. */
721
722#ifndef HAVE_SETREUID
723
724 /* System V has only the setuid/setgid calls to set user/group IDs.
725 There is an effective ID, which can be set by setuid/setgid.
726 It can be set (unless you are root) only to either what it already is
727 (returned by geteuid/getegid, now in make_uid/make_gid),
728 the real ID (return by getuid/getgid, now in user_uid/user_gid),
729 or the saved set ID (what the effective ID was before this set-ID
730 executable (make) was exec'd). */
731
732 if (setuid (user_uid) < 0)
733 pfatal_with_name ("user_access: setuid");
734
735#else /* HAVE_SETREUID. */
736
737 /* In 4BSD, the setreuid/setregid calls set both the real and effective IDs.
738 They may be set to themselves or each other. So you have two alternatives
739 at any one time. If you use setuid/setgid, the effective will be set to
740 the real, leaving only one alternative. Using setreuid/setregid, however,
741 you can toggle between your two alternatives by swapping the values in a
742 single setreuid or setregid call. */
743
744 if (setreuid (make_uid, user_uid) < 0)
745 pfatal_with_name ("user_access: setreuid");
746
747#endif /* Not HAVE_SETREUID. */
748#endif /* HAVE_SETEUID. */
749
750#ifdef HAVE_SETEGID
751 if (setegid (user_gid) < 0)
752 pfatal_with_name ("user_access: setegid");
753#else
754#ifndef HAVE_SETREGID
755 if (setgid (user_gid) < 0)
756 pfatal_with_name ("user_access: setgid");
757#else
758 if (setregid (make_gid, user_gid) < 0)
759 pfatal_with_name ("user_access: setregid");
760#endif
761#endif
762
763 current_access = user;
764
765 log_access (_("User access"));
766
767#endif /* GETLOADAVG_PRIVILEGED */
768}
769
770/* Give the process appropriate permissions for access to
771 make data (i.e., the load average). */
772void
773make_access (void)
774{
775#ifdef GETLOADAVG_PRIVILEGED
776
777 if (!access_inited)
778 init_access ();
779
780 if (current_access == make)
781 return;
782
783 /* See comments in user_access, above. */
784
785#ifdef HAVE_SETEUID
786 if (seteuid (make_uid) < 0)
787 pfatal_with_name ("make_access: seteuid");
788#else
789#ifndef HAVE_SETREUID
790 if (setuid (make_uid) < 0)
791 pfatal_with_name ("make_access: setuid");
792#else
793 if (setreuid (user_uid, make_uid) < 0)
794 pfatal_with_name ("make_access: setreuid");
795#endif
796#endif
797
798#ifdef HAVE_SETEGID
799 if (setegid (make_gid) < 0)
800 pfatal_with_name ("make_access: setegid");
801#else
802#ifndef HAVE_SETREGID
803 if (setgid (make_gid) < 0)
804 pfatal_with_name ("make_access: setgid");
805#else
806 if (setregid (user_gid, make_gid) < 0)
807 pfatal_with_name ("make_access: setregid");
808#endif
809#endif
810
811 current_access = make;
812
813 log_access (_("Make access"));
814
815#endif /* GETLOADAVG_PRIVILEGED */
816}
817
818/* Give the process appropriate permissions for a child process.
819 This is like user_access, but you can't get back to make_access. */
820void
821child_access (void)
822{
823#ifdef GETLOADAVG_PRIVILEGED
824
825 if (!access_inited)
826 abort ();
827
828 /* Set both the real and effective UID and GID to the user's.
829 They cannot be changed back to make's. */
830
831#ifndef HAVE_SETREUID
832 if (setuid (user_uid) < 0)
833 pfatal_with_name ("child_access: setuid");
834#else
835 if (setreuid (user_uid, user_uid) < 0)
836 pfatal_with_name ("child_access: setreuid");
837#endif
838
839#ifndef HAVE_SETREGID
840 if (setgid (user_gid) < 0)
841 pfatal_with_name ("child_access: setgid");
842#else
843 if (setregid (user_gid, user_gid) < 0)
844 pfatal_with_name ("child_access: setregid");
845#endif
846
847 log_access (_("Child access"));
848
849#endif /* GETLOADAVG_PRIVILEGED */
850}
851
852
853#ifdef NEED_GET_PATH_MAX
854unsigned int
855get_path_max (void)
856{
857 static unsigned int value;
858
859 if (value == 0)
860 {
861 long int x = pathconf ("/", _PC_PATH_MAX);
862 if (x > 0)
863 value = x;
864 else
865 return MAXPATHLEN;
866 }
867
868 return value;
869}
870#endif
871
872
873
874/* This code is stolen from gnulib.
875 If/when we abandon the requirement to work with K&R compilers, we can
876 remove this (and perhaps other parts of GNU make!) and migrate to using
877 gnulib directly.
878
879 This is called only through atexit(), which means die() has already been
880 invoked. So, call exit() here directly. Apparently that works...?
881*/
882
883/* Close standard output, exiting with status 'exit_failure' on failure.
884 If a program writes *anything* to stdout, that program should close
885 stdout and make sure that it succeeds before exiting. Otherwise,
886 suppose that you go to the extreme of checking the return status
887 of every function that does an explicit write to stdout. The last
888 printf can succeed in writing to the internal stream buffer, and yet
889 the fclose(stdout) could still fail (due e.g., to a disk full error)
890 when it tries to write out that buffered data. Thus, you would be
891 left with an incomplete output file and the offending program would
892 exit successfully. Even calling fflush is not always sufficient,
893 since some file systems (NFS and CODA) buffer written/flushed data
894 until an actual close call.
895
896 Besides, it's wasteful to check the return value from every call
897 that writes to stdout -- just let the internal stream state record
898 the failure. That's what the ferror test is checking below.
899
900 It's important to detect such failures and exit nonzero because many
901 tools (most notably `make' and other build-management systems) depend
902 on being able to detect failure in other tools via their exit status. */
903
904void
905close_stdout (void)
906{
907 int prev_fail = ferror (stdout);
908 int fclose_fail = fclose (stdout);
909
910 if (prev_fail || fclose_fail)
911 {
912 if (fclose_fail)
913 error (NILF, _("write error: %s"), strerror (errno));
914 else
915 error (NILF, _("write error"));
916 exit (EXIT_FAILURE);
917 }
918}
Note: See TracBrowser for help on using the repository browser.

© 2025 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette