diff --git a/lib/Transforms/IPO/FunctionAttrs.cpp b/lib/Transforms/IPO/FunctionAttrs.cpp index d40ee30f453..72c850fca99 100644 --- a/lib/Transforms/IPO/FunctionAttrs.cpp +++ b/lib/Transforms/IPO/FunctionAttrs.cpp @@ -281,6 +281,13 @@ static bool addReadAttrs(const SCCNodeSet &SCCNodes, AARGetterT &&AARGetter) { F->removeFnAttr(Attribute::ReadNone); F->removeFnAttr(Attribute::WriteOnly); + if (!WritesMemory && !ReadsMemory) { + // Clear out any "access range attributes" if readnone was deduced. + F->removeFnAttr(Attribute::ArgMemOnly); + F->removeFnAttr(Attribute::InaccessibleMemOnly); + F->removeFnAttr(Attribute::InaccessibleMemOrArgMemOnly); + } + // Add in the new attribute. if (WritesMemory && !ReadsMemory) F->addFnAttr(Attribute::WriteOnly); diff --git a/test/Transforms/FunctionAttrs/incompatible_fn_attrs.ll b/test/Transforms/FunctionAttrs/incompatible_fn_attrs.ll new file mode 100644 index 00000000000..79af817ff03 --- /dev/null +++ b/test/Transforms/FunctionAttrs/incompatible_fn_attrs.ll @@ -0,0 +1,32 @@ +; RUN: opt -S -o - -functionattrs %s | FileCheck %s +; RUN: opt -S -o - -passes=function-attrs %s | FileCheck %s + +; Verify we remove argmemonly/inaccessiblememonly/inaccessiblemem_or_argmemonly +; function attributes when we derive readnone. + +; Function Attrs: argmemonly +define i32* @given_argmem_infer_readnone(i32* %p) #0 { +; CHECK: define i32* @given_argmem_infer_readnone(i32* readnone returned %p) #0 { +entry: + ret i32* %p +} + +; Function Attrs: inaccessiblememonly +define i32* @given_inaccessible_infer_readnone(i32* %p) #1 { +; CHECK: define i32* @given_inaccessible_infer_readnone(i32* readnone returned %p) #0 { +entry: + ret i32* %p +} + +; Function Attrs: inaccessiblemem_or_argmemonly +define i32* @given_inaccessible_or_argmem_infer_readnone(i32* %p) #2 { +; CHECK: define i32* @given_inaccessible_or_argmem_infer_readnone(i32* readnone returned %p) #0 { +entry: + ret i32* %p +} + +attributes #0 = { argmemonly } +attributes #1 = { inaccessiblememonly } +attributes #2 = { inaccessiblemem_or_argmemonly } +; CHECK: attributes #0 = { norecurse nounwind readnone } +; CHECK-NOT: attributes