Index: clang/lib/Basic/Targets/ARM.cpp =================================================================== --- clang/lib/Basic/Targets/ARM.cpp +++ clang/lib/Basic/Targets/ARM.cpp @@ -480,10 +480,8 @@ } else if (Feature == "+dotprod") { DotProd = true; } else if (Feature == "+mve") { - DSP = 1; MVE |= MVE_INT; } else if (Feature == "+mve.fp") { - DSP = 1; HasLegalHalfType = true; FPU |= FPARMV8; MVE |= MVE_INT | MVE_FP; Index: clang/test/Driver/arm-mfpu.c =================================================================== --- clang/test/Driver/arm-mfpu.c +++ clang/test/Driver/arm-mfpu.c @@ -414,12 +414,14 @@ // CHECK-MVEFP-FPUNONE-DAG: "-target-feature" "-d32" // CHECK-MVEFP-FPUNONE-DAG: "-target-feature" "-neon" // CHECK-MVEFP-FPUNONE-DAG: "-target-feature" "-crypto" +// CHECK-MVEFP-FPUNONE-DAG: "-target-feature" "+mve" +// CHECK-MVEFP-FPUNONE-DAG: "-target-feature" "+dsp" // CHECK-MVEFP-FPUNONE-DAG: "-target-feature" "-mve.fp" // CHECK-MVEFP-FPUNONE-NOT: "-target-feature" "-fpregs" - // RUN: %clang -target arm-none-none-eabi %s -march=armv8.1-m.main+mve -mfpu=none -### -c 2>&1 \ // RUN: | FileCheck --check-prefix=CHECK-MVEI-FPUNONE %s // CHECK-MVEI-FPUNONE-DAG: "-target-feature" "-mve.fp" // CHECK-MVEI-FPUNONE-DAG: "-target-feature" "+mve" +// CHECK-MVEI-FPUNONE-DAG: "-target-feature" "+dsp" // CHECK-MVEI-FPUNONE-NOT: "-target-feature" "-fpregs" Index: clang/test/Preprocessor/arm-target-features.c =================================================================== --- clang/test/Preprocessor/arm-target-features.c +++ clang/test/Preprocessor/arm-target-features.c @@ -761,9 +761,10 @@ // CHECK-V81M-MVEFP: #define __ARM_FEATURE_SIMD32 1 // CHECK-V81M-MVEFP: #define __ARM_FPV5__ 1 -// nofp discards mve.fp +// nofp discards mve.fp, but not mve/dsp // RUN: %clang -target arm-arm-none-eabi -march=armv8.1-m.main+mve.fp+nofp -x c -E -dM %s -o - | FileCheck -match-full-lines --check-prefix=CHECK-V81M-MVEFP-NOFP %s -// CHECK-V81M-MVEFP-NOFP-NOT: #define __ARM_FEATURE_MVE +// CHECK-V81M-MVEFP-NOFP: #define __ARM_FEATURE_DSP 1 +// CHECK-V81M-MVEFP-NOFP: #define __ARM_FEATURE_MVE 1 // nomve discards mve.fp // RUN: %clang -target arm-arm-none-eabi -march=armv8.1-m.main+mve.fp+nomve -x c -E -dM %s -o - | FileCheck -match-full-lines --check-prefix=CHECK-V81M-MVEFP-NOMVE %s @@ -773,11 +774,16 @@ // RUN: %clang -target arm-arm-none-eabi -march=armv8.1-m.main+mve+fp -x c -E -dM %s -o - | FileCheck -match-full-lines --check-prefix=CHECK-V81M-MVE-FP %s // CHECK-V81M-MVE-FP: #define __ARM_FEATURE_MVE 1 -// nodsp discards both dsp and mve +// nodsp discards both dsp and mve ... // RUN: %clang -target arm-arm-none-eabi -march=armv8.1-m.main+mve+nodsp -x c -E -dM %s -o - | FileCheck -match-full-lines --check-prefix=CHECK-V81M-MVE-NODSP %s // CHECK-V81M-MVE-NODSP-NOT: #define __ARM_FEATURE_MVE // CHECK-V81M-MVE-NODSP-NOT: #define __ARM_FEATURE_DSP +// ... and also mve.fp +// RUN: %clang -target arm-arm-none-eabi -march=armv8.1-m.main+mve.fp+nodsp -x c -E -dM %s -o - | FileCheck -match-full-lines --check-prefix=CHECK-V81M-MVE-NODSP %s +// CHECK-V81M-MVE-NODSP-NOT: #define __ARM_FEATURE_MVE +// CHECK-V81M-MVE-NODSP-NOT: #define __ARM_FEATURE_DSP + // RUN: %clang -target armv8.1a-none-none-eabi -x c -E -dM %s -o - | FileCheck -match-full-lines --check-prefix=CHECK-V81A %s // CHECK-V81A: #define __ARM_ARCH 8 // CHECK-V81A: #define __ARM_ARCH_8_1A__ 1 Index: llvm/lib/Support/ARMTargetParser.cpp =================================================================== --- llvm/lib/Support/ARMTargetParser.cpp +++ llvm/lib/Support/ARMTargetParser.cpp @@ -505,10 +505,13 @@ return false; for (const auto AE : ARCHExtNames) { - if (Negated && (AE.ID & ID) == ID && AE.NegFeature) - Features.push_back(AE.NegFeature); - else if (AE.ID == ID && AE.Feature) - Features.push_back(AE.Feature); + if (Negated) { + if ((AE.ID & ID) == ID && AE.NegFeature) + Features.push_back(AE.NegFeature); + } else { + if ((AE.ID & ID) == AE.ID && AE.Feature) + Features.push_back(AE.Feature); + } } if (CPU == "") Index: llvm/unittests/Support/TargetParserTest.cpp =================================================================== --- llvm/unittests/Support/TargetParserTest.cpp +++ llvm/unittests/Support/TargetParserTest.cpp @@ -637,6 +637,27 @@ } } +static bool +testArchExtDependency(const char *ArchExt, + const std::initializer_list &Expected) { + std::vector Features; + + if (!ARM::appendArchExtFeatures("", ARM::ArchKind::ARMV8_1MMainline, ArchExt, + Features)) + return false; + + return llvm::all_of(Expected, [&](StringRef Ext) { + return llvm::is_contained(Features, Ext); + }); +} + +TEST(TargetParserTest, ARMArchExtDependencies) { + EXPECT_TRUE(testArchExtDependency("mve", {"+mve", "+dsp"})); + EXPECT_TRUE(testArchExtDependency("mve.fp", {"+mve.fp", "+mve", "+dsp"})); + EXPECT_TRUE(testArchExtDependency("nodsp", {"-dsp", "-mve", "-mve.fp"})); + EXPECT_TRUE(testArchExtDependency("nomve", {"-mve", "-mve.fp"})); +} + TEST(TargetParserTest, ARMparseHWDiv) { const char *hwdiv[] = {"thumb", "arm", "arm,thumb", "thumb,arm"};