summaryrefslogtreecommitdiff
path: root/gnu/egcs/gcc/cp/errfn.c
diff options
context:
space:
mode:
Diffstat (limited to 'gnu/egcs/gcc/cp/errfn.c')
-rw-r--r--gnu/egcs/gcc/cp/errfn.c352
1 files changed, 352 insertions, 0 deletions
diff --git a/gnu/egcs/gcc/cp/errfn.c b/gnu/egcs/gcc/cp/errfn.c
new file mode 100644
index 00000000000..b5d3eeccd5a
--- /dev/null
+++ b/gnu/egcs/gcc/cp/errfn.c
@@ -0,0 +1,352 @@
+/* Provide a call-back mechanism for handling error output.
+ Copyright (C) 1993, 94-98, 1999 Free Software Foundation, Inc.
+ Contributed by Jason Merrill (jason@cygnus.com)
+
+ This file is part of GNU CC.
+
+GNU CC is free software; you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation; either version 2, or (at your option)
+any later version.
+
+GNU CC is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with GNU CC; see the file COPYING. If not, write to
+the Free Software Foundation, 59 Temple Place - Suite 330,
+Boston, MA 02111-1307, USA. */
+
+#include "config.h"
+#include "system.h"
+#include "tree.h"
+#include "cp-tree.h"
+#include "toplev.h"
+
+/* cp_printer is the type of a function which converts an argument into
+ a string for digestion by printf. The cp_printer function should deal
+ with all memory management; the functions in this file will not free
+ the char*s returned. See error.c for an example use of this code. */
+
+typedef char* cp_printer PROTO((tree, int));
+extern cp_printer * cp_printers[256];
+
+/* Whether or not we should try to be quiet for errors and warnings; this is
+ used to avoid being too talkative about problems with tentative choices
+ when we're computing the conversion costs for a method call. */
+int cp_silent = 0;
+
+typedef void errorfn (); /* deliberately vague */
+
+static void cp_thing PROTO ((errorfn *, int, const char *, va_list));
+
+#define STRDUP(f) (ap = (char *) alloca (strlen (f) +1), strcpy (ap, (f)), ap)
+
+/* This function supports only `%s', `%d', `%%', and the C++ print
+ codes. */
+
+static void
+cp_thing (errfn, atarg1, format, ap)
+ errorfn *errfn;
+ int atarg1;
+ const char *format;
+ va_list ap;
+{
+ static char *buf;
+ static long buflen;
+ int nargs = 0;
+ long len;
+ long offset;
+ const char *f;
+ tree atarg = 0;
+
+ len = strlen (format) + 1;
+ if (len > buflen)
+ {
+ buflen = len;
+ buf = xrealloc (buf, buflen);
+ }
+ offset = 0;
+
+ for (f = format; *f; ++f)
+ {
+ cp_printer * function;
+ int alternate;
+ int maybe_here;
+
+ /* ignore text */
+ if (*f != '%')
+ {
+ buf[offset++] = *f;
+ continue;
+ }
+
+ ++f;
+
+ alternate = 0;
+ maybe_here = 0;
+
+ /* Check for '+' and '#' (in that order). */
+ if (*f == '+')
+ {
+ maybe_here = 1;
+ ++f;
+ }
+ if (*f == '#')
+ {
+ alternate = 1;
+ ++f;
+ }
+
+ /* no field width or precision */
+
+ function = cp_printers[(int)*f];
+
+ if (function || *f == 's')
+ {
+ char *p;
+ int plen;
+
+ if (*f == 's')
+ {
+ p = va_arg (ap, char *);
+ nargs++;
+ }
+ else
+ {
+ tree t = va_arg (ap, tree);
+ nargs++;
+
+ /* This indicates that ATARG comes from a different
+ location than normal. */
+ if (maybe_here && atarg1)
+ atarg = t;
+
+ /* If atarg1 is set and this is the first argument, then
+ set ATARG appropriately. */
+ if (atarg1 && nargs == 1)
+ atarg = t;
+
+ p = (*function) (t, alternate);
+ }
+
+ plen = strlen (p);
+ len += plen;
+ if (len > buflen)
+ {
+ buflen = len;
+ buf = xrealloc (buf, len);
+ }
+ strcpy (buf + offset, p);
+ offset += plen;
+ }
+ else if (*f == '%')
+ {
+ /* A `%%' has occurred in the input string. Replace it with
+ a `%' in the formatted message buf. */
+
+ if (++len > buflen)
+ {
+ buflen = len;
+ buf = xrealloc (buf, len);
+ }
+ buf[offset++] = '%';
+ }
+ else
+ {
+ if (*f != 'd')
+ abort ();
+ len += HOST_BITS_PER_INT / 2;
+ if (len > buflen)
+ {
+ buflen = len;
+ buf = xrealloc (buf, len);
+ }
+ sprintf (buf + offset, "%d", va_arg (ap, int));
+ nargs++;
+ offset += strlen (buf + offset);
+ /* With an ANSI C library one could write
+ out += sprintf (...); */
+ }
+ }
+ buf[offset] = '\0';
+
+ /* If ATARG1 is set, but we haven't extracted any arguments, then
+ extract one tree argument for ATARG. */
+ if (nargs == 0 && atarg1)
+ atarg = va_arg (ap, tree);
+
+ if (atarg)
+ {
+ char *file = cp_file_of (atarg);
+ int line = cp_line_of (atarg);
+ (*errfn) (file, line, "%s", buf);
+ }
+ else
+ (*errfn) ("%s", buf);
+
+}
+
+void
+cp_error VPROTO((const char *format, ...))
+{
+#ifndef ANSI_PROTOTYPES
+ char *format;
+#endif
+ va_list ap;
+
+ VA_START (ap, format);
+
+#ifndef ANSI_PROTOTYPES
+ format = va_arg (ap, char *);
+#endif
+
+ if (! cp_silent)
+ cp_thing ((errorfn *) error, 0, format, ap);
+ va_end (ap);
+}
+
+void
+cp_warning VPROTO((const char *format, ...))
+{
+#ifndef ANSI_PROTOTYPES
+ char *format;
+#endif
+ va_list ap;
+
+ VA_START (ap, format);
+
+#ifndef ANSI_PROTOTYPES
+ format = va_arg (ap, char *);
+#endif
+
+ if (! cp_silent)
+ cp_thing ((errorfn *) warning, 0, format, ap);
+ va_end (ap);
+}
+
+void
+cp_pedwarn VPROTO((const char *format, ...))
+{
+#ifndef ANSI_PROTOTYPES
+ char *format;
+#endif
+ va_list ap;
+
+ VA_START (ap, format);
+
+#ifndef ANSI_PROTOTYPES
+ format = va_arg (ap, char *);
+#endif
+
+ if (! cp_silent)
+ cp_thing ((errorfn *) pedwarn, 0, format, ap);
+ va_end (ap);
+}
+
+void
+cp_compiler_error VPROTO((const char *format, ...))
+{
+#ifndef ANSI_PROTOTYPES
+ char *format;
+#endif
+ va_list ap;
+
+ VA_START (ap, format);
+
+#ifndef ANSI_PROTOTYPES
+ format = va_arg (ap, char *);
+#endif
+
+ if (! cp_silent)
+ cp_thing ((errorfn *) compiler_error, 0, format, ap);
+ va_end (ap);
+}
+
+void
+cp_deprecated (msg)
+ const char *msg;
+{
+ extern int warn_deprecated;
+ if (!warn_deprecated)
+ return;
+ cp_warning ("%s is deprecated.", msg);
+ cp_warning ("Please see the documentation for details.");
+}
+
+void
+cp_sprintf VPROTO((const char *format, ...))
+{
+#ifndef ANSI_PROTOTYPES
+ char *format;
+#endif
+ va_list ap;
+
+ VA_START (ap, format);
+
+#ifndef ANSI_PROTOTYPES
+ format = va_arg (ap, char *);
+#endif
+
+ cp_thing ((errorfn *) sprintf, 0, format, ap);
+ va_end (ap);
+}
+
+void
+cp_error_at VPROTO((const char *format, ...))
+{
+#ifndef ANSI_PROTOTYPES
+ char *format;
+#endif
+ va_list ap;
+
+ VA_START (ap, format);
+
+#ifndef ANSI_PROTOTYPES
+ format = va_arg (ap, char *);
+#endif
+
+ if (! cp_silent)
+ cp_thing ((errorfn *) error_with_file_and_line, 1, format, ap);
+ va_end (ap);
+}
+
+void
+cp_warning_at VPROTO((const char *format, ...))
+{
+#ifndef ANSI_PROTOTYPES
+ char *format;
+#endif
+ va_list ap;
+
+ VA_START (ap, format);
+
+#ifndef ANSI_PROTOTYPES
+ format = va_arg (ap, char *);
+#endif
+
+ if (! cp_silent)
+ cp_thing ((errorfn *) warning_with_file_and_line, 1, format, ap);
+ va_end (ap);
+}
+
+void
+cp_pedwarn_at VPROTO((const char *format, ...))
+{
+#ifndef ANSI_PROTOTYPES
+ char *format;
+#endif
+ va_list ap;
+
+ VA_START (ap, format);
+
+#ifndef ANSI_PROTOTYPES
+ format = va_arg (ap, char *);
+#endif
+
+ if (! cp_silent)
+ cp_thing ((errorfn *) pedwarn_with_file_and_line, 1, format, ap);
+ va_end (ap);
+}