|  | ; RUN: llvm-as %s -o %t.bc | 
|  | ; RUN: llvm-bcanalyzer -dump %t.bc | FileCheck %s | 
|  | ; RUN: llvm-dis %t.bc | 
|  |  | 
|  | ; There's a curious case where blockaddress constants may refer to functions | 
|  | ; outside of the function they're used in. There's a special bitcode function | 
|  | ; code, FUNC_CODE_BLOCKADDR_USERS, used to signify that this is the case. | 
|  |  | 
|  | ; The intent of this test is two-fold: | 
|  | ; 1. Ensure we produce BLOCKADDR_USERS bitcode function code on the first fn, | 
|  | ;    @repro, since @fun and @fun2 both refer to @repro via blockaddress | 
|  | ;    constants. | 
|  | ; 2. Ensure we can round-trip serializing+desearlizing such case. | 
|  |  | 
|  | ; CHECK: <FUNCTION_BLOCK | 
|  | ; CHECK: <BLOCKADDR_USERS op0=2 op1=1 | 
|  | ; CHECK: <FUNCTION_BLOCK | 
|  | ; CHECK: <FUNCTION_BLOCK | 
|  |  | 
|  | %struct.ptrs = type { i8*, i8* } | 
|  |  | 
|  | define void @repro() { | 
|  | br label %label | 
|  |  | 
|  | label: | 
|  | call void @fun() | 
|  | ret void | 
|  | } | 
|  |  | 
|  | define void @fun() noinline { | 
|  | call void @f(%struct.ptrs { i8* blockaddress(@repro, %label), i8* blockaddress(@repro, %label) }) | 
|  | ret void | 
|  | } | 
|  |  | 
|  | define void @fun2() noinline { | 
|  | call void @f(%struct.ptrs { i8* blockaddress(@repro, %label), i8* blockaddress(@repro, %label) }) | 
|  | ret void | 
|  | } | 
|  |  | 
|  | declare void @f(%struct.ptrs) |