1 /* Producer string parsers for GDB.
3 Copyright (C) 2012-2024 Free Software Foundation, Inc.
5 This file is part of GDB.
7 This program is free software; you can redistribute it and/or modify
8 it under the terms of the GNU General Public License as published by
9 the Free Software Foundation; either version 3 of the License, or
10 (at your option) any later version.
12 This program is distributed in the hope that it will be useful,
13 but WITHOUT ANY WARRANTY; without even the implied warranty of
14 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 GNU General Public License for more details.
17 You should have received a copy of the GNU General Public License
18 along with this program. If not, see <http://www.gnu.org/licenses/>. */
21 #include "gdbsupport/selftest.h"
22 #include "gdbsupport/gdb_regex.h"
27 producer_is_gcc_ge_4 (const char *producer
)
31 if (! producer_is_gcc (producer
, &major
, &minor
))
43 producer_is_gcc (const char *producer
, int *major
, int *minor
)
47 if (producer
!= NULL
&& startswith (producer
, "GNU "))
57 cs
= &producer
[strlen ("GNU ")];
59 /* Bail out for GNU AS. */
60 if (startswith (cs
, "AS "))
63 /* Skip any identifier after "GNU " - such as "C11" "C++" or "Java".
64 A full producer string might look like:
66 "GNU Fortran 4.8.2 20140120 (Red Hat 4.8.2-16) -mtune=generic ..."
67 "GNU C++14 5.0.0 20150123 (experimental)"
69 while (*cs
&& !isspace (*cs
))
71 if (*cs
&& isspace (*cs
))
73 if (sscanf (cs
, "%d.%d", major
, minor
) == 2)
77 /* Not recognized as GCC. */
84 producer_is_gas (const char *producer
, int *major
, int *minor
)
86 if (producer
== nullptr)
88 /* No producer, don't know. */
93 const char prefix
[] = "GNU AS ";
94 if (!startswith (producer
, prefix
))
96 /* Producer is not gas. */
101 const char *cs
= &producer
[strlen (prefix
)];
103 /* Ensure that major/minor are not nullptrs. */
105 if (major
== nullptr)
107 if (minor
== nullptr)
110 int scanned
= sscanf (cs
, "%d.%d", major
, minor
);
113 /* Unable to scan major/minor version. */
120 /* See producer.h. */
123 producer_is_icc_ge_19 (const char *producer
)
127 if (! producer_is_icc (producer
, &major
, &minor
))
133 /* See producer.h. */
136 producer_is_icc (const char *producer
, int *major
, int *minor
)
138 compiled_regex
i_re ("Intel(R)", 0, "producer_is_icc");
139 if (producer
== nullptr || i_re
.exec (producer
, 0, nullptr, 0) != 0)
142 /* Prepare the used fields. */
144 if (major
== nullptr)
146 if (minor
== nullptr)
152 compiled_regex
re ("[0-9]+\\.[0-9]+", REG_EXTENDED
, "producer_is_icc");
153 regmatch_t version
[1];
154 if (re
.exec (producer
, ARRAY_SIZE (version
), version
, 0) == 0
155 && version
[0].rm_so
!= -1)
157 const char *version_str
= producer
+ version
[0].rm_so
;
158 sscanf (version_str
, "%d.%d", major
, minor
);
165 /* See producer.h. */
168 producer_is_llvm (const char *producer
)
170 return ((producer
!= NULL
) && (startswith (producer
, "clang ")
171 || startswith (producer
, " F90 Flang ")));
174 /* See producer.h. */
177 producer_is_clang (const char *producer
, int *major
, int *minor
)
179 if (producer
!= nullptr && startswith (producer
, "clang version "))
182 if (major
== nullptr)
184 if (minor
== nullptr)
187 /* The full producer string will look something like
188 "clang version XX.X.X ..."
189 So we can safely ignore all characters before the first digit. */
190 const char *cs
= producer
+ strlen ("clang version ");
192 if (sscanf (cs
, "%d.%d", major
, minor
) == 2)
198 #if defined GDB_SELF_TEST
199 namespace selftests
{
203 producer_parsing_tests ()
206 /* Check that we don't crash if "Version" is not found in what
207 looks like an ICC producer string. */
208 static const char icc_no_version
[] = "Intel(R) foo bar";
210 int major
= 0, minor
= 0;
211 SELF_CHECK (!producer_is_icc (icc_no_version
, &major
, &minor
));
212 SELF_CHECK (!producer_is_gcc (icc_no_version
, &major
, &minor
));
216 static const char extern_f_14_0
[] = "\
217 Intel(R) Fortran Intel(R) 64 Compiler XE for applications running on \
219 Version 14.0.1.074 Build 20130716";
221 int major
= 0, minor
= 0;
222 SELF_CHECK (producer_is_icc (extern_f_14_0
, &major
, &minor
)
223 && major
== 14 && minor
== 0);
224 SELF_CHECK (!producer_is_gcc (extern_f_14_0
, &major
, &minor
));
228 static const char intern_f_14
[] = "\
229 Intel(R) Fortran Intel(R) 64 Compiler XE for applications running on \
233 int major
= 0, minor
= 0;
234 SELF_CHECK (producer_is_icc (intern_f_14
, &major
, &minor
)
235 && major
== 14 && minor
== 0);
236 SELF_CHECK (!producer_is_gcc (intern_f_14
, &major
, &minor
));
240 static const char intern_c_14
[] = "\
241 Intel(R) C++ Intel(R) 64 Compiler XE for applications running on \
244 int major
= 0, minor
= 0;
245 SELF_CHECK (producer_is_icc (intern_c_14
, &major
, &minor
)
246 && major
== 14 && minor
== 0);
247 SELF_CHECK (!producer_is_gcc (intern_c_14
, &major
, &minor
));
251 static const char intern_c_18
[] = "\
252 Intel(R) C++ Intel(R) 64 Compiler for applications running on \
255 int major
= 0, minor
= 0;
256 SELF_CHECK (producer_is_icc (intern_c_18
, &major
, &minor
)
257 && major
== 18 && minor
== 0);
261 static const char gnu
[] = "GNU C 4.7.2";
262 SELF_CHECK (!producer_is_icc (gnu
, NULL
, NULL
));
264 int major
= 0, minor
= 0;
265 SELF_CHECK (producer_is_gcc (gnu
, &major
, &minor
)
266 && major
== 4 && minor
== 7);
270 static const char gnu_exp
[] = "GNU C++14 5.0.0 20150123 (experimental)";
271 int major
= 0, minor
= 0;
272 SELF_CHECK (!producer_is_icc (gnu_exp
, NULL
, NULL
));
273 SELF_CHECK (producer_is_gcc (gnu_exp
, &major
, &minor
)
274 && major
== 5 && minor
== 0);
278 static const char clang_llvm_exp
[] = "clang version 12.0.0 (CLANG: bld#8)";
279 int major
= 0, minor
= 0;
280 SELF_CHECK (!producer_is_icc (clang_llvm_exp
, NULL
, NULL
));
281 SELF_CHECK (!producer_is_gcc (clang_llvm_exp
, &major
, &minor
));
282 SELF_CHECK (producer_is_llvm (clang_llvm_exp
));
286 static const char flang_llvm_exp
[] = " F90 Flang - 1.5 2017-05-01";
287 int major
= 0, minor
= 0;
288 SELF_CHECK (!producer_is_icc (flang_llvm_exp
, NULL
, NULL
));
289 SELF_CHECK (!producer_is_gcc (flang_llvm_exp
, &major
, &minor
));
290 SELF_CHECK (producer_is_llvm (flang_llvm_exp
));
294 static const char gas_exp
[] = "GNU AS 2.39.0";
295 int major
= 0, minor
= 0;
296 SELF_CHECK (!producer_is_gcc (gas_exp
, &major
, &minor
));
297 SELF_CHECK (producer_is_gas (gas_exp
, &major
, &minor
));
298 SELF_CHECK (major
== 2 && minor
== 39);
300 static const char gas_incomplete_exp
[] = "GNU AS ";
301 SELF_CHECK (!producer_is_gas (gas_incomplete_exp
, &major
, &minor
));
302 SELF_CHECK (!producer_is_gcc (gas_incomplete_exp
, &major
, &minor
));
304 static const char gas_incomplete_exp_2
[] = "GNU AS 2";
305 SELF_CHECK (!producer_is_gas (gas_incomplete_exp_2
, &major
, &minor
));
306 SELF_CHECK (!producer_is_gcc (gas_incomplete_exp_2
, &major
, &minor
));
314 void _initialize_producer ();
316 _initialize_producer ()
318 #if defined GDB_SELF_TEST
319 selftests::register_test
320 ("producer-parser", selftests::producer::producer_parsing_tests
);