/* Concrete classes for selftests involving diagnostic paths. Copyright (C) 2019-2025 Free Software Foundation, Inc. Contributed by David Malcolm 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 . */ #ifndef GCC_DIAGNOSTICS_SELFTEST_PATHS_H #define GCC_DIAGNOSTICS_SELFTEST_PATHS_H #include "diagnostics/paths.h" #include "diagnostics/selftest-logical-locations.h" /* The selftest code should entirely disappear in a production configuration, hence we guard all of it with #if CHECKING_P. */ #if CHECKING_P namespace diagnostics { namespace paths { namespace selftest { /* Concrete subclasses of the abstract base classes declared in diagnostic-path.h for use in selftests. This code should have no dependency on "tree". */ /* An implementation of diagnostics::paths::event. */ class test_event : public event { public: using logical_location = logical_locations::key; using thread_id_t = paths::thread_id_t; test_event (location_t loc, logical_location logical_loc, int depth, const char *desc, thread_id_t thread_id = 0); ~test_event (); location_t get_location () const final override { return m_loc; } int get_stack_depth () const final override { return m_depth; } void print_desc (pretty_printer &pp) const final override { pp_string (&pp, m_desc); } logical_location get_logical_location () const final override { return m_logical_loc; } meaning get_meaning () const final override { return meaning (); } bool connect_to_next_event_p () const final override { return m_connected_to_next_event; } thread_id_t get_thread_id () const final override { return m_thread_id; } void connect_to_next_event () { m_connected_to_next_event = true; } private: location_t m_loc; logical_location m_logical_loc; int m_depth; char *m_desc; // has been formatted; doesn't get i18n-ed bool m_connected_to_next_event; thread_id_t m_thread_id; }; /* A simple implementation of diagnostics::paths::thread. */ class test_thread : public thread { public: test_thread (const char *name) : m_name (name) {} label_text get_name (bool) const final override { return label_text::borrow (m_name); } private: const char *m_name; // has been i18n-ed and formatted }; /* A concrete subclass of diagnostics::paths::path for implementing selftests - a vector of test_event instances - adds member functions for adding test event - does no translation of its events - has no dependency on "tree". */ class test_path : public path { public: test_path (logical_locations::selftest::test_manager &logical_loc_mgr, pretty_printer *event_pp); unsigned num_events () const final override; const event & get_event (int idx) const final override; unsigned num_threads () const final override; const thread & get_thread (thread_id_t) const final override; bool same_function_p (int event_idx_a, int event_idx_b) const final override; thread_id_t add_thread (const char *name); event_id_t add_event (location_t loc, const char *funcname, int depth, const char *fmt, ...) ATTRIBUTE_GCC_DIAG(5,6); event_id_t add_thread_event (thread_id_t thread_id, location_t loc, const char *funcname, int depth, const char *fmt, ...) ATTRIBUTE_GCC_DIAG(6,7); void connect_to_next_event (); void add_entry (const char *callee_name, int stack_depth, thread_id_t thread_id = 0); void add_return (const char *caller_name, int stack_depth, thread_id_t thread_id = 0); void add_call (const char *caller_name, int caller_stack_depth, const char *callee_name, thread_id_t thread_id = 0); private: logical_locations::key logical_location_from_funcname (const char *funcname); logical_locations::selftest::test_manager &m_test_logical_loc_mgr; auto_delete_vec m_threads; auto_delete_vec m_events; /* (for use by add_event). */ pretty_printer *m_event_pp; }; } // namespace diagnostics::paths::selftest } // namespace diagnostics::paths } // namespace diagnostics #endif /* #if CHECKING_P */ #endif /* ! GCC_DIAGNOSTICS_SELFTEST_PATHS_H */