|  | /* Producer string parsers for GDB. | 
|  |  | 
|  | Copyright (C) 2012-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 <http://www.gnu.org/licenses/>.  */ | 
|  |  | 
|  | #include "defs.h" | 
|  | #include "producer.h" | 
|  | #include "gdbsupport/selftest.h" | 
|  | #include "gdb_regex.h" | 
|  |  | 
|  | /* See producer.h.  */ | 
|  |  | 
|  | int | 
|  | producer_is_gcc_ge_4 (const char *producer) | 
|  | { | 
|  | int major, minor; | 
|  |  | 
|  | if (! producer_is_gcc (producer, &major, &minor)) | 
|  | return -1; | 
|  | if (major < 4) | 
|  | return -1; | 
|  | if (major > 4) | 
|  | return INT_MAX; | 
|  | return minor; | 
|  | } | 
|  |  | 
|  | /* See producer.h.  */ | 
|  |  | 
|  | int | 
|  | producer_is_gcc (const char *producer, int *major, int *minor) | 
|  | { | 
|  | const char *cs; | 
|  |  | 
|  | if (producer != NULL && startswith (producer, "GNU ")) | 
|  | { | 
|  | int maj, min; | 
|  |  | 
|  | if (major == NULL) | 
|  | major = &maj; | 
|  | if (minor == NULL) | 
|  | minor = &min; | 
|  |  | 
|  | /* Skip any identifier after "GNU " - such as "C11" "C++" or "Java". | 
|  | A full producer string might look like: | 
|  | "GNU C 4.7.2" | 
|  | "GNU Fortran 4.8.2 20140120 (Red Hat 4.8.2-16) -mtune=generic ..." | 
|  | "GNU C++14 5.0.0 20150123 (experimental)" | 
|  | */ | 
|  | cs = &producer[strlen ("GNU ")]; | 
|  | while (*cs && !isspace (*cs)) | 
|  | cs++; | 
|  | if (*cs && isspace (*cs)) | 
|  | cs++; | 
|  | if (sscanf (cs, "%d.%d", major, minor) == 2) | 
|  | return 1; | 
|  | } | 
|  |  | 
|  | /* Not recognized as GCC.  */ | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | /* See producer.h.  */ | 
|  |  | 
|  | bool | 
|  | producer_is_icc_ge_19 (const char *producer) | 
|  | { | 
|  | int major, minor; | 
|  |  | 
|  | if (! producer_is_icc (producer, &major, &minor)) | 
|  | return false; | 
|  |  | 
|  | return major >= 19; | 
|  | } | 
|  |  | 
|  | /* See producer.h.  */ | 
|  |  | 
|  | bool | 
|  | producer_is_icc (const char *producer, int *major, int *minor) | 
|  | { | 
|  | compiled_regex i_re ("Intel(R)", 0, "producer_is_icc"); | 
|  | if (producer == nullptr || i_re.exec (producer, 0, nullptr, 0) != 0) | 
|  | return false; | 
|  |  | 
|  | /* Prepare the used fields.  */ | 
|  | int maj, min; | 
|  | if (major == nullptr) | 
|  | major = &maj; | 
|  | if (minor == nullptr) | 
|  | minor = &min; | 
|  |  | 
|  | *minor = 0; | 
|  | *major = 0; | 
|  |  | 
|  | compiled_regex re ("[0-9]+\\.[0-9]+", REG_EXTENDED, "producer_is_icc"); | 
|  | regmatch_t version[1]; | 
|  | if (re.exec (producer, ARRAY_SIZE (version), version, 0) == 0 | 
|  | && version[0].rm_so != -1) | 
|  | { | 
|  | const char *version_str = producer + version[0].rm_so; | 
|  | sscanf (version_str, "%d.%d", major, minor); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | return false; | 
|  | } | 
|  |  | 
|  | /* See producer.h.  */ | 
|  |  | 
|  | bool | 
|  | producer_is_llvm (const char *producer) | 
|  | { | 
|  | return ((producer != NULL) && (startswith (producer, "clang ") | 
|  | || startswith (producer, " F90 Flang "))); | 
|  | } | 
|  |  | 
|  | #if defined GDB_SELF_TEST | 
|  | namespace selftests { | 
|  | namespace producer { | 
|  |  | 
|  | static void | 
|  | producer_parsing_tests () | 
|  | { | 
|  | { | 
|  | /* Check that we don't crash if "Version" is not found in what | 
|  | looks like an ICC producer string.  */ | 
|  | static const char icc_no_version[] = "Intel(R) foo bar"; | 
|  |  | 
|  | int major = 0, minor = 0; | 
|  | SELF_CHECK (!producer_is_icc (icc_no_version, &major, &minor)); | 
|  | SELF_CHECK (!producer_is_gcc (icc_no_version, &major, &minor)); | 
|  | } | 
|  |  | 
|  | { | 
|  | static const char extern_f_14_0[] = "\ | 
|  | Intel(R) Fortran Intel(R) 64 Compiler XE for applications running on \ | 
|  | Intel(R) 64, \ | 
|  | Version 14.0.1.074 Build 20130716"; | 
|  |  | 
|  | int major = 0, minor = 0; | 
|  | SELF_CHECK (producer_is_icc (extern_f_14_0, &major, &minor) | 
|  | && major == 14 && minor == 0); | 
|  | SELF_CHECK (!producer_is_gcc (extern_f_14_0, &major, &minor)); | 
|  | } | 
|  |  | 
|  | { | 
|  | static const char intern_f_14[] = "\ | 
|  | Intel(R) Fortran Intel(R) 64 Compiler XE for applications running on \ | 
|  | Intel(R) 64, \ | 
|  | Version 14.0"; | 
|  |  | 
|  | int major = 0, minor = 0; | 
|  | SELF_CHECK (producer_is_icc (intern_f_14, &major, &minor) | 
|  | && major == 14 && minor == 0); | 
|  | SELF_CHECK (!producer_is_gcc (intern_f_14, &major, &minor)); | 
|  | } | 
|  |  | 
|  | { | 
|  | static const char intern_c_14[] = "\ | 
|  | Intel(R) C++ Intel(R) 64 Compiler XE for applications running on \ | 
|  | Intel(R) 64, \ | 
|  | Version 14.0"; | 
|  | int major = 0, minor = 0; | 
|  | SELF_CHECK (producer_is_icc (intern_c_14, &major, &minor) | 
|  | && major == 14 && minor == 0); | 
|  | SELF_CHECK (!producer_is_gcc (intern_c_14, &major, &minor)); | 
|  | } | 
|  |  | 
|  | { | 
|  | static const char intern_c_18[] = "\ | 
|  | Intel(R) C++ Intel(R) 64 Compiler for applications running on \ | 
|  | Intel(R) 64, \ | 
|  | Version 18.0 Beta"; | 
|  | int major = 0, minor = 0; | 
|  | SELF_CHECK (producer_is_icc (intern_c_18, &major, &minor) | 
|  | && major == 18 && minor == 0); | 
|  | } | 
|  |  | 
|  | { | 
|  | static const char gnu[] = "GNU C 4.7.2"; | 
|  | SELF_CHECK (!producer_is_icc (gnu, NULL, NULL)); | 
|  |  | 
|  | int major = 0, minor = 0; | 
|  | SELF_CHECK (producer_is_gcc (gnu, &major, &minor) | 
|  | && major == 4 && minor == 7); | 
|  | } | 
|  |  | 
|  | { | 
|  | static const char gnu_exp[] = "GNU C++14 5.0.0 20150123 (experimental)"; | 
|  | int major = 0, minor = 0; | 
|  | SELF_CHECK (!producer_is_icc (gnu_exp, NULL, NULL)); | 
|  | SELF_CHECK (producer_is_gcc (gnu_exp, &major, &minor) | 
|  | && major == 5 && minor == 0); | 
|  | } | 
|  |  | 
|  | { | 
|  | static const char clang_llvm_exp[] = "clang version 12.0.0 (CLANG: bld#8)"; | 
|  | int major = 0, minor = 0; | 
|  | SELF_CHECK (!producer_is_icc (clang_llvm_exp, NULL, NULL)); | 
|  | SELF_CHECK (!producer_is_gcc (clang_llvm_exp, &major, &minor)); | 
|  | SELF_CHECK (producer_is_llvm (clang_llvm_exp)); | 
|  | } | 
|  |  | 
|  | { | 
|  | static const char flang_llvm_exp[] = " F90 Flang - 1.5 2017-05-01"; | 
|  | int major = 0, minor = 0; | 
|  | SELF_CHECK (!producer_is_icc (flang_llvm_exp, NULL, NULL)); | 
|  | SELF_CHECK (!producer_is_gcc (flang_llvm_exp, &major, &minor)); | 
|  | SELF_CHECK (producer_is_llvm (flang_llvm_exp)); | 
|  | } | 
|  | } | 
|  | } | 
|  | } | 
|  | #endif | 
|  |  | 
|  | void _initialize_producer (); | 
|  | void | 
|  | _initialize_producer () | 
|  | { | 
|  | #if defined GDB_SELF_TEST | 
|  | selftests::register_test | 
|  | ("producer-parser", selftests::producer::producer_parsing_tests); | 
|  | #endif | 
|  | } |