FLANG
Fortran-features.h
1//===-- include/flang/Support/Fortran-features.h ----------------*- C++ -*-===//
2//
3// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4// See https://llvm.org/LICENSE.txt for license information.
5// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6//
7//===----------------------------------------------------------------------===//
8
9#ifndef FORTRAN_SUPPORT_FORTRAN_FEATURES_H_
10#define FORTRAN_SUPPORT_FORTRAN_FEATURES_H_
11
12#include "Fortran.h"
13#include "flang/Common/enum-set.h"
14#include <string_view>
15#include <vector>
16
17namespace Fortran::common {
18
19// Non-conforming extensions & legacies
20ENUM_CLASS(LanguageFeature, BackslashEscapes, OldDebugLines,
21 FixedFormContinuationWithColumn1Ampersand, LogicalAbbreviations,
22 XOROperator, PunctuationInNames, OptionalFreeFormSpace, BOZExtensions,
23 EmptyStatement, AlternativeNE, ExecutionPartNamelist, DECStructures,
24 DoubleComplex, Byte, StarKind, ExponentMatchingKindParam, QuadPrecision,
25 SlashInitialization, TripletInArrayConstructor, MissingColons,
26 SignedComplexLiteral, OldStyleParameter, ComplexConstructor, PercentLOC,
27 SignedMultOperand, FileName, Carriagecontrol, Convert, Dispose,
28 IOListLeadingComma, AbbreviatedEditDescriptor, ProgramParentheses,
29 PercentRefAndVal, OmitFunctionDummies, CrayPointer, Hollerith, ArithmeticIF,
30 Assign, AssignedGOTO, Pause, OpenACC, OpenMP, CUDA, CruftAfterAmpersand,
31 ClassicCComments, AdditionalFormats, BigIntLiterals, RealDoControls,
32 EquivalenceNumericWithCharacter, EquivalenceNonDefaultNumeric,
33 EquivalenceSameNonSequence, AdditionalIntrinsics, AnonymousParents,
34 OldLabelDoEndStatements, LogicalIntegerAssignment, EmptySourceFile,
35 ProgramReturn, ImplicitNoneTypeNever, ImplicitNoneTypeAlways,
36 ImplicitNoneExternal, ForwardRefImplicitNone, OpenAccessAppend,
37 BOZAsDefaultInteger, DistinguishableSpecifics, DefaultSave,
38 PointerInSeqType, NonCharacterFormat, SaveMainProgram,
39 SaveBigMainProgramVariables, DistinctArrayConstructorLengths, PPCVector,
40 RelaxedIntentInChecking, ForwardRefImplicitNoneData,
41 NullActualForAllocatable, ActualIntegerConvertedToSmallerKind,
42 HollerithOrCharacterAsBOZ, BindingAsProcedure, StatementFunctionExtensions,
43 UseGenericIntrinsicWhenSpecificDoesntMatch, DataStmtExtensions,
44 RedundantContiguous, RedundantAttribute, InitBlankCommon,
45 EmptyBindCDerivedType, MiscSourceExtensions, AllocateToOtherLength,
46 LongNames, IntrinsicAsSpecific, BenignNameClash, BenignRedundancy,
47 NullMoldAllocatableComponentValue, NopassScalarBase, MiscUseExtensions,
48 ImpliedDoIndexScope, DistinctCommonSizes, OddIndexVariableRestrictions,
49 IndistinguishableSpecifics, SubroutineAndFunctionSpecifics,
50 EmptySequenceType, NonSequenceCrayPointee, BranchIntoConstruct,
51 BadBranchTarget, HollerithPolymorphic, ListDirectedSize,
52 NonBindCInteroperability, CudaManaged, CudaUnified,
53 PolymorphicActualAllocatableOrPointerToMonomorphicDummy, RelaxedPureDummy,
54 UndefinableAsynchronousOrVolatileActual, AutomaticInMainProgram, PrintCptr,
55 SavedLocalInSpecExpr, PrintNamelist, AssumedRankPassedToNonAssumedRank,
56 IgnoreIrrelevantAttributes, Unsigned, ContiguousOkForSeqAssociation,
57 ForwardRefExplicitTypeDummy, InaccessibleDeferredOverride,
58 CudaWarpMatchFunction, DoConcurrentOffload, TransferBOZ, Coarray,
59 PointerPassObject, MultipleIdenticalDATA,
60 DefaultStructConstructorNullPointer, AssumedRankIoItem,
61 MultipleProgramUnitsOnSameLine, AllocatedForAssociated,
62 OpenMPThreadprivateEquivalence, RelaxedCLoc, CudaPinned,
63 AccDefaultNoneScalars)
64
65// Portability and suspicious usage warnings
66ENUM_CLASS(UsageWarning, Portability, PointerToUndefinable,
67 NonTargetPassedToTarget, PointerToPossibleNoncontiguous,
68 ShortCharacterActual, ShortArrayActual, ImplicitInterfaceActual,
69 PolymorphicTransferArg, PointerComponentTransferArg, TransferSizePresence,
70 F202XAllocatableBreakingChange, OptionalMustBePresent, CommonBlockPadding,
71 LogicalVsCBool, BindCCharLength, ProcDummyArgShapes, ExternalNameConflict,
72 FoldingException, FoldingAvoidsRuntimeCrash, FoldingValueChecks,
73 FoldingFailure, FoldingLimit, Interoperability, CharacterInteroperability,
74 Bounds, Preprocessing, Scanning, OpenAccUsage, ProcPointerCompatibility,
75 VoidMold, KnownBadImplicitInterface, EmptyCase, CaseOverflow, CUDAUsage,
76 IgnoreTKRUsage, ExternalInterfaceMismatch, DefinedOperatorArgs, Final,
77 ZeroDoStep, UnusedForallIndex, OpenMPUsage, DataLength, IgnoredDirective,
78 HomonymousSpecific, HomonymousResult, IgnoredIntrinsicFunctionType,
79 PreviousScalarUse, RedeclaredInaccessibleComponent, ImplicitShared,
80 IndexVarRedefinition, IncompatibleImplicitInterfaces,
81 VectorSubscriptFinalization, UndefinedFunctionResult, UselessIomsg,
82 MismatchingDummyProcedure, SubscriptedEmptyArray, UnsignedLiteralTruncation,
83 CompatibleDeclarationsFromDistinctModules, ConstantIsContiguous,
84 NullActualForDefaultIntentAllocatable, UseAssociationIntoSameNameSubprogram,
85 HostAssociatedIntentOutInSpecExpr, NonVolatilePointerToVolatile,
86 RealConstantWidening, VolatileOrAsynchronousTemporary, UnusedVariable,
87 UsedUndefinedVariable, BadValueInDeadCode, AssumedTypeSizeDummy,
88 MisplacedIgnoreTKR, NamelistParameter, ImpureFinalInPure,
89 IgnoredNoReallocateLHS, CLoc, ExperimentalOption, AccImplicitScalar)
90
93using LanguageFeatureOrWarning = std::variant<LanguageFeature, UsageWarning>;
94using LanguageControlFlag =
95 std::pair<LanguageFeatureOrWarning, /*shouldEnable=*/bool>;
96
97class LanguageFeatureControl {
98public:
99 LanguageFeatureControl();
100 LanguageFeatureControl(const LanguageFeatureControl &) = default;
101
102 void Enable(LanguageFeature f, bool yes = true) { disable_.set(f, !yes); }
103 void EnableWarning(LanguageFeature f, bool yes = true) {
104 warnLanguage_.set(f, yes);
105 }
106 void EnableWarning(UsageWarning w, bool yes = true) {
107 warnUsage_.set(w, yes);
108 }
109 void EnableWarning(LanguageFeatureOrWarning flag, bool yes = true) {
110 if (std::holds_alternative<LanguageFeature>(flag)) {
111 EnableWarning(std::get<LanguageFeature>(flag), yes);
112 } else {
113 EnableWarning(std::get<UsageWarning>(flag), yes);
114 }
115 }
116 void WarnOnAllNonstandard(bool yes = true);
117 bool IsWarnOnAllNonstandard() const { return warnAllLanguage_; }
118 void WarnOnAllUsage(bool yes = true);
119 bool IsWarnOnAllUsage() const { return warnAllUsage_; }
120 void DisableAllNonstandardWarnings() {
121 warnAllLanguage_ = false;
122 warnLanguage_.clear();
123 }
124 void DisableAllUsageWarnings() {
125 warnAllUsage_ = false;
126 warnUsage_.clear();
127 }
128 void DisableAllWarnings() {
129 disableAllWarnings_ = true;
130 DisableAllNonstandardWarnings();
131 DisableAllUsageWarnings();
132 }
133 bool AreWarningsDisabled() const { return disableAllWarnings_; }
134 bool IsEnabled(LanguageFeature f) const { return !disable_.test(f); }
135 bool ShouldWarn(LanguageFeature f) const { return warnLanguage_.test(f); }
136 bool ShouldWarn(UsageWarning w) const { return warnUsage_.test(w); }
137 // Cli options
138 // Find a warning by its Cli spelling, i.e. '[no-]warning-name'.
139 std::optional<LanguageControlFlag> FindWarning(std::string_view input);
140 // Take a string from the Cli and apply it to the LanguageFeatureControl.
141 // Return true if the option was recognized (and hence applied).
142 bool EnableWarning(std::string_view input);
143 // The add and replace functions are not currently used but are provided
144 // to allow a flexible many-to-one mapping from Cli spellings to enum values.
145 // Taking a string by value because the functions own this string after the
146 // call.
147 void AddAlternativeCliSpelling(LanguageFeature f, std::string input) {
148 cliOptions_.insert({input, {f}});
149 }
150 void AddAlternativeCliSpelling(UsageWarning w, std::string input) {
151 cliOptions_.insert({input, {w}});
152 }
153 void AddDeprecatedCliSpelling(LanguageFeature f,
154 const std::string &deprecated, const std::string &canonical) {
155 cliOptions_.insert({deprecated, {f}});
156 deprecatedCliOptions_.insert({deprecated, canonical});
157 }
158 void AddDeprecatedCliSpelling(UsageWarning w, const std::string &deprecated,
159 const std::string &canonical) {
160 cliOptions_.insert({deprecated, {w}});
161 deprecatedCliOptions_.insert({deprecated, canonical});
162 }
163 // Returns the canonical spelling if the input is a deprecated spelling.
164 std::optional<std::string_view> CheckDeprecatedSpelling(
165 std::string_view input) const;
166 void ReplaceCliCanonicalSpelling(LanguageFeature f, std::string input);
167 void ReplaceCliCanonicalSpelling(UsageWarning w, std::string input);
168 std::string_view getDefaultCliSpelling(LanguageFeature f) const {
169 return languageFeatureCliCanonicalSpelling_[EnumToInt(f)];
170 };
171 std::string_view getDefaultCliSpelling(UsageWarning w) const {
172 return usageWarningCliCanonicalSpelling_[EnumToInt(w)];
173 };
174 // Return all spellings of operators names, depending on features enabled
175 std::vector<const char *> GetNames(LogicalOperator) const;
176 std::vector<const char *> GetNames(RelationalOperator) const;
177
178private:
179 // Map from Cli syntax of language features and usage warnings to their enum
180 // values.
181 std::unordered_map<std::string, LanguageFeatureOrWarning> cliOptions_;
182 // Map from deprecated Cli spellings to their canonical replacements.
183 std::unordered_map<std::string, std::string> deprecatedCliOptions_;
184 // These two arrays map the enum values to their cannonical Cli spellings.
185 // Since each of the CanonicalSpelling is a string in the domain of the map
186 // above we just use a view of the string instead of another copy.
187 std::array<std::string, LanguageFeature_enumSize>
188 languageFeatureCliCanonicalSpelling_;
189 std::array<std::string, UsageWarning_enumSize>
190 usageWarningCliCanonicalSpelling_;
191 LanguageFeatures disable_;
192 LanguageFeatures warnLanguage_;
193 bool warnAllLanguage_{false};
194 UsageWarnings warnUsage_;
195 bool warnAllUsage_{false};
196 bool disableAllWarnings_{false};
197};
198} // namespace Fortran::common
199#endif // FORTRAN_SUPPORT_FORTRAN_FEATURES_H_
Definition enum-set.h:28
Definition bit-population-count.h:20