/* Feature dependency helpers for AArch64. Copyright (C) 2022-2023 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 . */ #ifndef AARCH64_FEATURE_DEPS_H #define AARCH64_FEATURE_DEPS_H 1 namespace { namespace feature_deps { /* Together, these definitions of get_flags take a list of feature names (representing functions that are defined below) and return the set of associated flags. */ constexpr aarch64_feature_flags get_flags () { return 0; } template constexpr aarch64_feature_flags get_flags (T1 i, Ts... args) { return i ().flag | get_flags (args...); } /* Like get_flags, but return the transitive closure of those features and the ones that they rely on. */ constexpr aarch64_feature_flags get_enable () { return 0; } template constexpr aarch64_feature_flags get_enable (T1 i, Ts... args) { return i ().enable | get_enable (args...); } /* Define info such that it has the following static constant variables: - flag: the aarch64_feature_flags bit associated with FEATURE - enable: the transitive closure of the features that FEATURE requires, plus FLAG itself - explicit_on: the transitive closure of the features that an explicit +FEATURE enables, including FLAG itself. This is always a superset of ENABLE Also define a function FEATURE () that returns an info (which is an empty structure, since all members are static). Building up the list feature-by-feature ensures that the definition files are in topological order. */ template struct info; #define HANDLE(IDENT, REQUIRES, EXPLICIT_ON) \ template<> struct info { \ static constexpr auto flag = AARCH64_FL_##IDENT; \ static constexpr auto enable = flag | get_enable REQUIRES; \ static constexpr auto explicit_on = enable | get_enable EXPLICIT_ON; \ }; \ constexpr info IDENT () \ { \ return info (); \ } #define AARCH64_OPT_EXTENSION(A, IDENT, REQUIRES, EXPLICIT_ON, E, F) \ HANDLE (IDENT, REQUIRES, EXPLICIT_ON) #define AARCH64_ARCH(A, B, IDENT, D, REQUIRES) HANDLE (IDENT, REQUIRES, ()) #include "config/aarch64/aarch64-option-extensions.def" #include "config/aarch64/aarch64-arches.def" #undef HANDLE /* Return the set of all features that would need to be disabled if the features in MASK are disabled. Note that the size of the expression varies linearly with the number of features, which means that invoking this function once per feature is quadratic in the number of features. However, collecting the same information at compiler start-up is likely to be quadratic too, so we're better off paying the cost once per compiler build rather than once per compiler run. */ constexpr aarch64_feature_flags get_flags_off (aarch64_feature_flags mask) { return (0 #define AARCH64_OPT_EXTENSION(A, IDENT, C, D, E, F) \ | (feature_deps::IDENT ().enable & mask ? AARCH64_FL_##IDENT : 0) #include "config/aarch64/aarch64-option-extensions.def" ); } /* Define root_off_ variables for each feature, giving the set of features that must be turned off by +noIDENT. This set is not transitively closed; use get_flags_off to complete the closure. */ #define AARCH64_OPT_EXTENSION(A, IDENT, C, D, EXPLICIT_OFF, F) \ constexpr auto root_off_##IDENT \ = AARCH64_FL_##IDENT | get_flags EXPLICIT_OFF; #include "config/aarch64/aarch64-option-extensions.def" /* Define cpu_ variables for each CPU, giving the transitive closure of all the features that the CPU supports. */ #define AARCH64_CORE(A, CORE_IDENT, C, ARCH_IDENT, FEATURES, F, G, H, I) \ constexpr auto cpu_##CORE_IDENT = ARCH_IDENT ().enable | get_enable FEATURES; #include "config/aarch64/aarch64-cores.def" } } #endif