/* Support for suggestions about missing #include directives. Copyright (C) 2017-2018 Free Software Foundation, Inc. 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 . */ #include "config.h" #define INCLUDE_UNIQUE_PTR #include "system.h" #include "coretypes.h" #include "c-family/c-common.h" #include "c-family/name-hint.h" #include "c-family/known-headers.h" #include "gcc-rich-location.h" /* An enum for distinguishing between the C and C++ stdlibs. */ enum stdlib { STDLIB_C, STDLIB_CPLUSPLUS, NUM_STDLIBS }; /* A struct for associating names in a standard library with the header that should be included to locate them, for each of the C and C++ stdlibs (or NULL, for names that aren't in a header for a particular stdlib). */ struct stdlib_hint { const char *name; const char *header[NUM_STDLIBS]; }; /* Given non-NULL NAME, return the header name defining it within either the standard library (with '<' and '>'), or NULL. Only handles a subset of the most common names within the stdlibs. */ static const char * get_stdlib_header_for_name (const char *name, enum stdlib lib) { gcc_assert (name); gcc_assert (lib < NUM_STDLIBS); static const stdlib_hint hints[] = { /* and . */ {"assert", {"", ""} }, /* and . */ {"errno", {"", ""} }, /* and . */ {"CHAR_BIT", {"", ""} }, {"CHAR_MAX", {"", ""} }, {"CHAR_MIN", {"", ""} }, {"INT_MAX", {"", ""} }, {"INT_MIN", {"", ""} }, {"LLONG_MAX", {"", ""} }, {"LLONG_MIN", {"", ""} }, {"LONG_MAX", {"", ""} }, {"LONG_MIN", {"", ""} }, {"MB_LEN_MAX", {"", ""} }, {"SCHAR_MAX", {"", ""} }, {"SCHAR_MIN", {"", ""} }, {"SHRT_MAX", {"", ""} }, {"SHRT_MIN", {"", ""} }, {"UCHAR_MAX", {"", ""} }, {"UINT_MAX", {"", ""} }, {"ULLONG_MAX", {"", ""} }, {"ULONG_MAX", {"", ""} }, {"USHRT_MAX", {"", ""} }, /* and . */ {"va_list", {"", ""} }, /* and . */ {"NULL", {"", ""} }, {"nullptr_t", {NULL, ""} }, {"offsetof", {"", ""} }, {"ptrdiff_t", {"", ""} }, {"size_t", {"", ""} }, {"wchar_t", {"", NULL /* a keyword in C++ */} }, /* and . */ {"BUFSIZ", {"", ""} }, {"EOF", {"", ""} }, {"FILE", {"", ""} }, {"FILENAME_MAX", {"", ""} }, {"fopen", {"", ""} }, {"fpos_t", {"", ""} }, {"getchar", {"", ""} }, {"printf", {"", ""} }, {"snprintf", {"", ""} }, {"sprintf", {"", ""} }, {"stderr", {"", ""} }, {"stdin", {"", ""} }, {"stdout", {"", ""} }, /* and . */ {"free", {"", ""} }, {"malloc", {"", ""} }, {"realloc", {"", ""} }, /* and . */ {"memchr", {"", ""} }, {"memcmp", {"", ""} }, {"memcpy", {"", ""} }, {"memmove", {"", ""} }, {"memset", {"", ""} }, {"strcat", {"", ""} }, {"strchr", {"", ""} }, {"strcmp", {"", ""} }, {"strcpy", {"", ""} }, {"strlen", {"", ""} }, {"strncat", {"", ""} }, {"strncmp", {"", ""} }, {"strncpy", {"", ""} }, {"strrchr", {"", ""} }, {"strspn", {"", ""} }, {"strstr", {"", ""} }, /* . */ {"PTRDIFF_MAX", {"", ""} }, {"PTRDIFF_MIN", {"", ""} }, {"SIG_ATOMIC_MAX", {"", ""} }, {"SIG_ATOMIC_MIN", {"", ""} }, {"SIZE_MAX", {"", ""} }, {"WINT_MAX", {"", ""} }, {"WINT_MIN", {"", ""} }, /* . */ {"WCHAR_MAX", {"", ""} }, {"WCHAR_MIN", {"", ""} } }; const size_t num_hints = sizeof (hints) / sizeof (hints[0]); for (size_t i = 0; i < num_hints; i++) if (strcmp (name, hints[i].name) == 0) return hints[i].header[lib]; return NULL; } /* Given non-NULL NAME, return the header name defining it within the C standard library (with '<' and '>'), or NULL. */ const char * get_c_stdlib_header_for_name (const char *name) { return get_stdlib_header_for_name (name, STDLIB_C); } /* Given non-NULL NAME, return the header name defining it within the C++ standard library (with '<' and '>'), or NULL. */ const char * get_cp_stdlib_header_for_name (const char *name) { return get_stdlib_header_for_name (name, STDLIB_CPLUSPLUS); } /* Implementation of class suggest_missing_header. */ /* suggest_missing_header's ctor. */ suggest_missing_header::suggest_missing_header (location_t loc, const char *name, const char *header_hint) : deferred_diagnostic (loc), m_name_str (name), m_header_hint (header_hint) { gcc_assert (name); gcc_assert (header_hint); } /* suggest_missing_header's dtor. */ suggest_missing_header::~suggest_missing_header () { if (is_suppressed_p ()) return; gcc_rich_location richloc (get_location ()); maybe_add_include_fixit (&richloc, m_header_hint); inform (&richloc, "%qs is defined in header %qs;" " did you forget to %<#include %s%>?", m_name_str, m_header_hint, m_header_hint); }