mirror of
https://github.com/RPCS3/llvm-mirror.git
synced 2024-11-23 19:23:23 +01:00
6a8d2b6215
This is a fix for PR28697. An MDNode can indirectly refer to a GlobalValue, through a ConstantAsMetadata. When the GlobalValue is deleted, the MDNode operand is reset to `nullptr`. If the node is uniqued, this can lead to a hard-to-detect cache invalidation in a Metadata map that's shared across an LLVMContext. Consider: 1. A map from Metadata* to `T` called RemappedMDs. 2. A node that references a global variable, `!{i1* @GV}`. 3. Insert `!{i1* @GV} -> SomeT` in the map. 4. Delete `@GV`, leaving behind `!{null} -> SomeT`. Looking up the generic and uninteresting `!{null}` gives you `SomeT`, which is likely related to `@GV`. Worse, `SomeT`'s lifetime may be tied to the deleted `@GV`. This occurs in practice in the shared ValueMap used since r266579 in the IRMover. Other code that handles more than one Module (with different lifetimes) in the same LLVMContext could hit it too. The fix here is a partial revert of r225223: in the rare case that an MDNode operand is a ConstantAsMetadata (i.e., wrapping a node from the Value hierarchy), drop uniquing if it gets replaced with `nullptr`. This changes step #4 above to leave behind `distinct !{null} -> SomeT`, which can't be confused with the generic `!{null}`. In theory, this can cause some churn in the LLVMContext's MDNode uniquing map when Values are being deleted. However: - The number of GlobalValues referenced from uniqued MDNodes is expected to be quite small. E.g., the debug info metadata schema only references GlobalValues from distinct nodes. - Other Constants have the lifetime of the LLVMContext, whose teardown is careful to drop references before deleting the constants. As a result, I don't expect a compile time regression from this change. llvm-svn: 277625
33 lines
1.0 KiB
LLVM
33 lines
1.0 KiB
LLVM
; RUN: opt -S -globalopt < %s | FileCheck %s
|
|
|
|
; PR6112 - When globalopt does RAUW(@G, %G), the metadata reference should drop
|
|
; to null. Function local metadata that references @G from a different function
|
|
; to that containing %G should likewise drop to null.
|
|
@G = internal global i8** null
|
|
|
|
define i32 @main(i32 %argc, i8** %argv) norecurse {
|
|
; CHECK-LABEL: @main(
|
|
; CHECK: %G = alloca
|
|
store i8** %argv, i8*** @G
|
|
ret i32 0
|
|
}
|
|
|
|
define void @foo(i32 %x) {
|
|
; Note: these arguments look like MDNodes, but they're really syntactic sugar
|
|
; for 'MetadataAsValue::get(ValueAsMetadata::get(Value*))'. When @G drops to
|
|
; null, the ValueAsMetadata instance gets replaced by metadata !{}, or
|
|
; MDNode::get({}).
|
|
call void @llvm.foo(metadata i8*** @G, metadata i32 %x)
|
|
; CHECK: call void @llvm.foo(metadata ![[EMPTY:[0-9]+]], metadata i32 %x)
|
|
ret void
|
|
}
|
|
|
|
declare void @llvm.foo(metadata, metadata) nounwind readnone
|
|
|
|
!named = !{!0}
|
|
; CHECK: !named = !{![[NULL:[0-9]+]]}
|
|
|
|
!0 = !{i8*** @G}
|
|
; CHECK-DAG: ![[NULL]] = distinct !{null}
|
|
; CHECK-DAG: ![[EMPTY]] = !{}
|