/* Support for complaint handling during symbol reading in GDB. Copyright (C) 1990-2021 Free Software Foundation, Inc. This file is part of GDB. This program 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 of the License, or (at your option) any later version. This program 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 this program. If not, see . */ #include "defs.h" #include "complaints.h" #include "command.h" #include "gdbcmd.h" #include "gdbsupport/selftest.h" #include /* Map format strings to counters. */ static std::unordered_map counters; /* How many complaints about a particular thing should be printed before we stop whining about it? Default is no whining at all, since so many systems have ill-constructed symbol files. */ int stop_whining = 0; /* See complaints.h. */ void complaint_internal (const char *fmt, ...) { va_list args; if (++counters[fmt] > stop_whining) return; va_start (args, fmt); if (deprecated_warning_hook) (*deprecated_warning_hook) (fmt, args); else { fputs_filtered (_("During symbol reading: "), gdb_stderr); vfprintf_filtered (gdb_stderr, fmt, args); fputs_filtered ("\n", gdb_stderr); } va_end (args); } /* See complaints.h. */ void clear_complaints () { counters.clear (); } static void complaints_show_value (struct ui_file *file, int from_tty, struct cmd_list_element *cmd, const char *value) { fprintf_filtered (file, _("Max number of complaints about incorrect" " symbols is %s.\n"), value); } #if GDB_SELF_TEST namespace selftests { /* Entry point for complaints unit tests. */ static void test_complaints () { std::unordered_map tmp; scoped_restore reset_counters = make_scoped_restore (&counters, tmp); scoped_restore reset_stop_whining = make_scoped_restore (&stop_whining, 2); #define CHECK_COMPLAINT(STR, CNT) \ do \ { \ std::string output; \ output = execute_fn_to_string ([]() { complaint (STR); }, false); \ std::string expected \ = _("During symbol reading: ") + std::string (STR "\n"); \ SELF_CHECK (output == expected); \ SELF_CHECK (counters[STR] == CNT); \ } while (0) #define CHECK_COMPLAINT_SILENT(STR, CNT) \ do \ { \ std::string output; \ output = execute_fn_to_string ([]() { complaint (STR); }, false); \ SELF_CHECK (output.empty ()); \ SELF_CHECK (counters[STR] == CNT); \ } while (0) CHECK_COMPLAINT ("maintenance complaint 0", 1); CHECK_COMPLAINT ("maintenance complaint 0", 2); CHECK_COMPLAINT_SILENT ("maintenance complaint 0", 3); CHECK_COMPLAINT ("maintenance complaint 1", 1); clear_complaints (); CHECK_COMPLAINT ("maintenance complaint 0", 1); #undef CHECK_COMPLAINT #undef CHECK_COMPLAINT_SILENT } } // namespace selftests #endif /* GDB_SELF_TEST */ void _initialize_complaints (); void _initialize_complaints () { add_setshow_zinteger_cmd ("complaints", class_support, &stop_whining, _("\ Set max number of complaints about incorrect symbols."), _("\ Show max number of complaints about incorrect symbols."), NULL, NULL, complaints_show_value, &setlist, &showlist); #if GDB_SELF_TEST selftests::register_test ("complaints", selftests::test_complaints); #endif /* GDB_SELF_TEST */ }