123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149 |
- /* Paths through the code associated with a diagnostic.
- Copyright (C) 2019-2020 Free Software Foundation, Inc.
- Contributed by David Malcolm <dmalcolm@redhat.com>
- This file is part of GCC.
- GCC 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 3, or (at your option) any later
- version.
- GCC 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 GCC; see the file COPYING3. If not see
- <http://www.gnu.org/licenses/>. */
- #ifndef GCC_DIAGNOSTIC_PATH_H
- #define GCC_DIAGNOSTIC_PATH_H
- #include "diagnostic.h" /* for ATTRIBUTE_GCC_DIAG. */
- #include "diagnostic-event-id.h"
- /* A diagnostic_path is an optional additional piece of metadata associated
- with a diagnostic (via its rich_location).
- It describes a sequence of events predicted by the compiler that
- lead to the problem occurring, with their locations in the user's source,
- and text descriptions.
- For example, the following error has a 3-event path:
- test.c: In function 'demo':
- test.c:29:5: error: passing NULL as argument 1 to 'PyList_Append' which
- requires a non-NULL parameter
- 29 | PyList_Append(list, item);
- | ^~~~~~~~~~~~~~~~~~~~~~~~~
- 'demo': events 1-3
- |
- | 25 | list = PyList_New(0);
- | | ^~~~~~~~~~~~~
- | | |
- | | (1) when 'PyList_New' fails, returning NULL
- | 26 |
- | 27 | for (i = 0; i < count; i++) {
- | | ~~~
- | | |
- | | (2) when 'i < count'
- | 28 | item = PyLong_FromLong(random());
- | 29 | PyList_Append(list, item);
- | | ~~~~~~~~~~~~~~~~~~~~~~~~~
- | | |
- | | (3) when calling 'PyList_Append', passing NULL from (1) as argument 1
- |
- The diagnostic-printing code has consolidated the path into a single
- run of events, since all the events are near each other and within the same
- function; more complicated examples (such as interprocedural paths)
- might be printed as multiple runs of events. */
- /* Abstract base classes, describing events within a path, and the paths
- themselves. */
- /* One event within a diagnostic_path. */
- class diagnostic_event
- {
- public:
- virtual ~diagnostic_event () {}
- virtual location_t get_location () const = 0;
- virtual tree get_fndecl () const = 0;
- /* Stack depth, so that consumers can visualizes the interprocedural
- calls, returns, and frame nesting. */
- virtual int get_stack_depth () const = 0;
- /* Get a localized (and possibly colorized) description of this event. */
- virtual label_text get_desc (bool can_colorize) const = 0;
- };
- /* Abstract base class for getting at a sequence of events. */
- class diagnostic_path
- {
- public:
- virtual ~diagnostic_path () {}
- virtual unsigned num_events () const = 0;
- virtual const diagnostic_event & get_event (int idx) const = 0;
- bool interprocedural_p () const;
- };
- /* Concrete subclasses. */
- /* A simple implementation of diagnostic_event. */
- class simple_diagnostic_event : public diagnostic_event
- {
- public:
- simple_diagnostic_event (location_t loc, tree fndecl, int depth,
- const char *desc);
- ~simple_diagnostic_event ();
- location_t get_location () const FINAL OVERRIDE { return m_loc; }
- tree get_fndecl () const FINAL OVERRIDE { return m_fndecl; }
- int get_stack_depth () const FINAL OVERRIDE { return m_depth; }
- label_text get_desc (bool) const FINAL OVERRIDE
- {
- return label_text::borrow (m_desc);
- }
- private:
- location_t m_loc;
- tree m_fndecl;
- int m_depth;
- char *m_desc; // has been i18n-ed and formatted
- };
- /* A simple implementation of diagnostic_path, as a vector of
- simple_diagnostic_event instances. */
- class simple_diagnostic_path : public diagnostic_path
- {
- public:
- simple_diagnostic_path (pretty_printer *event_pp)
- : m_event_pp (event_pp) {}
- unsigned num_events () const FINAL OVERRIDE;
- const diagnostic_event & get_event (int idx) const FINAL OVERRIDE;
- diagnostic_event_id_t add_event (location_t loc, tree fndecl, int depth,
- const char *fmt, ...)
- ATTRIBUTE_GCC_DIAG(5,6);
- private:
- auto_delete_vec<simple_diagnostic_event> m_events;
- /* (for use by add_event). */
- pretty_printer *m_event_pp;
- };
- extern void debug (diagnostic_path *path);
- #endif /* ! GCC_DIAGNOSTIC_PATH_H */
|