[llvm] 8700f2b - [Verifier] Verify scoped noalias metadata

Nikita Popov via llvm-commits llvm-commits at lists.llvm.org
Mon Sep 20 09:30:14 PDT 2021


Author: Nikita Popov
Date: 2021-09-20T18:27:28+02:00
New Revision: 8700f2bd36bb9b7d7075ed4dac0aef92b9489237

URL: https://github.com/llvm/llvm-project/commit/8700f2bd36bb9b7d7075ed4dac0aef92b9489237
DIFF: https://github.com/llvm/llvm-project/commit/8700f2bd36bb9b7d7075ed4dac0aef92b9489237.diff

LOG: [Verifier] Verify scoped noalias metadata

Verify that !noalias, !alias.scope and llvm.experimental.noalias.scope
arguments have the format specified in
https://llvm.org/docs/LangRef.html#noalias-and-alias-scope-metadata.
I've fixed up a lot of broken metadata used by tests in advance.
Especially using a scope instead of the expected scope list is a
commonly made mistake.

Differential Revision: https://reviews.llvm.org/D110026

Added: 
    llvm/test/Verifier/alias-scope-metadata.ll

Modified: 
    llvm/lib/IR/Verifier.cpp

Removed: 
    


################################################################################
diff  --git a/llvm/lib/IR/Verifier.cpp b/llvm/lib/IR/Verifier.cpp
index 354f09c028d7..4c235d7c7c5d 100644
--- a/llvm/lib/IR/Verifier.cpp
+++ b/llvm/lib/IR/Verifier.cpp
@@ -461,6 +461,8 @@ class Verifier : public InstVisitor<Verifier>, VerifierSupport {
   void visitDereferenceableMetadata(Instruction &I, MDNode *MD);
   void visitProfMetadata(Instruction &I, MDNode *MD);
   void visitAnnotationMetadata(MDNode *Annotation);
+  void visitAliasScopeMetadata(const MDNode *MD);
+  void visitAliasScopeListMetadata(const MDNode *MD);
 
   template <class Ty> bool isValidMetadataArray(const MDTuple &N);
 #define HANDLE_SPECIALIZED_MDNODE_LEAF(CLASS) void visit##CLASS(const CLASS &N);
@@ -4343,6 +4345,38 @@ void Verifier::visitAnnotationMetadata(MDNode *Annotation) {
     Assert(isa<MDString>(Op.get()), "operands must be strings");
 }
 
+void Verifier::visitAliasScopeMetadata(const MDNode *MD) {
+  unsigned NumOps = MD->getNumOperands();
+  Assert(NumOps >= 2 && NumOps <= 3, "scope must have two or three operands",
+         MD);
+  Assert(MD->getOperand(0) == MD || isa<MDString>(MD->getOperand(0)),
+         "first scope operand must be self-referential or string", MD);
+  if (NumOps == 3)
+    Assert(isa<MDString>(MD->getOperand(2)),
+           "third scope operand must be string (if used)", MD);
+
+  MDNode *Domain = dyn_cast<MDNode>(MD->getOperand(1));
+  Assert(Domain != nullptr, "second scope operand must be MDNode", MD);
+
+  unsigned NumDomainOps = Domain->getNumOperands();
+  Assert(NumDomainOps >= 1 && NumDomainOps <= 2,
+         "domain must have one or two operands", Domain);
+  Assert(Domain->getOperand(0) == Domain ||
+             isa<MDString>(Domain->getOperand(0)),
+         "first domain operand must be self-referential or string", Domain);
+  if (NumDomainOps == 2)
+    Assert(isa<MDString>(Domain->getOperand(1)),
+           "second domain operand must be string (if used)", Domain);
+}
+
+void Verifier::visitAliasScopeListMetadata(const MDNode *MD) {
+  for (const MDOperand &Op : MD->operands()) {
+    const MDNode *OpMD = dyn_cast<MDNode>(Op);
+    Assert(OpMD != nullptr, "scope list must consist of MDNodes", MD);
+    visitAliasScopeMetadata(OpMD);
+  }
+}
+
 /// verifyInstruction - Verify that an instruction is well formed.
 ///
 void Verifier::visitInstruction(Instruction &I) {
@@ -4502,6 +4536,11 @@ void Verifier::visitInstruction(Instruction &I) {
   if (MDNode *TBAA = I.getMetadata(LLVMContext::MD_tbaa))
     TBAAVerifyHelper.visitTBAAMetadata(I, TBAA);
 
+  if (MDNode *MD = I.getMetadata(LLVMContext::MD_noalias))
+    visitAliasScopeListMetadata(MD);
+  if (MDNode *MD = I.getMetadata(LLVMContext::MD_alias_scope))
+    visitAliasScopeListMetadata(MD);
+
   if (MDNode *AlignMD = I.getMetadata(LLVMContext::MD_align)) {
     Assert(I.getType()->isPointerTy(), "align applies only to pointer types",
            &I);
@@ -5719,6 +5758,7 @@ void Verifier::verifyNoAliasScopeDecl() {
            II);
     Assert(ScopeListMD->getNumOperands() == 1,
            "!id.scope.list must point to a list with a single scope", II);
+    visitAliasScopeListMetadata(ScopeListMD);
   }
 
   // Only check the domination rule when requested. Once all passes have been

diff  --git a/llvm/test/Verifier/alias-scope-metadata.ll b/llvm/test/Verifier/alias-scope-metadata.ll
new file mode 100644
index 000000000000..072fcd11ce36
--- /dev/null
+++ b/llvm/test/Verifier/alias-scope-metadata.ll
@@ -0,0 +1,61 @@
+; RUN: not llvm-as %s -o /dev/null 2>&1 | FileCheck %s
+
+define void @test(i8* %p) {
+  load i8, i8* %p, !noalias !0
+  load i8, i8* %p, !noalias !1
+  load i8, i8* %p, !noalias !3
+  load i8, i8* %p, !noalias !5
+  load i8, i8* %p, !noalias !7
+  load i8, i8* %p, !noalias !9
+  load i8, i8* %p, !noalias !11
+  load i8, i8* %p, !noalias !14
+  load i8, i8* %p, !alias.scope !17
+  call void @llvm.experimental.noalias.scope.decl(metadata !20)
+  ret void
+}
+
+declare void @llvm.experimental.noalias.scope.decl(metadata)
+
+; CHECK: scope list must consist of MDNodes
+!0 = !{!"str"}
+
+; CHECK: scope must have two or three operands
+!1 = !{!2}
+!2 = !{!2}
+
+; CHECK: scope must have two or three operands
+!3 = !{!4}
+!4 = !{!4, !5, !6, !7}
+
+; CHECK: first scope operand must be self-referential or string
+!5 = !{!6}
+!6 = !{!7, !8}
+
+; CHECK: third scope operand must be string (if used)
+!7 = !{!8}
+!8 = !{!8, !9, !10}
+
+; CHECK: second scope operand must be MDNode
+!9 = !{!10}
+!10 = !{!10, !"str"}
+
+; CHECK: domain must have one or two operands
+!11 = !{!12}
+!12 = !{!12, !13}
+!13 = !{}
+
+; CHECK: domain must have one or two operands
+!14 = !{!15}
+!15 = !{!15, !16}
+!16 = !{!17, !18, !19}
+
+; CHECK: first domain operand must be self-referential or string
+!17 = !{!18}
+!18 = !{!18, !19}
+!19 = !{!20}
+
+; CHECK: second domain operand must be string (if used)
+!20 = !{!21}
+!21 = !{!21, !22}
+!22 = !{!22, !23}
+!23 = !{}


        


More information about the llvm-commits mailing list