1; RUN: llc -march=bpfel -filetype=asm -o - %s | FileCheck -check-prefixes=CHECK %s
2; RUN: llc -march=bpfeb -filetype=asm -o - %s | FileCheck -check-prefixes=CHECK %s
3;
4; Source:
5;   void foo(const void *);
6;   int test() {
7;     const struct {
8;       unsigned a[4];
9;       char b;
10;     } val = { .a = {2, 3, 4, 5}, .b = 4 };
11;     foo(&val);
12;     return 0;
13;   }
14; Compilation flag:
15;   clang -target bpf -O2 -g -S -emit-llvm t.c
16
17%struct.anon = type { [4 x i32], i8 }
18
19@__const.test.val = private unnamed_addr constant %struct.anon { [4 x i32] [i32 2, i32 3, i32 4, i32 5], i8 4 }, align 4
20
21; Function Attrs: nounwind
22define dso_local i32 @test() local_unnamed_addr #0 !dbg !7 {
23entry:
24  %val = alloca %struct.anon, align 4
25  %0 = bitcast %struct.anon* %val to i8*, !dbg !23
26  call void @llvm.lifetime.start.p0i8(i64 20, i8* nonnull %0) #4, !dbg !23
27  call void @llvm.dbg.declare(metadata %struct.anon* %val, metadata !12, metadata !DIExpression()), !dbg !24
28  call void @llvm.memcpy.p0i8.p0i8.i64(i8* nonnull align 4 dereferenceable(20) %0, i8* nonnull align 4 dereferenceable(20) bitcast (%struct.anon* @__const.test.val to i8*), i64 20, i1 false), !dbg !24
29  call void @foo(i8* nonnull %0) #4, !dbg !25
30  call void @llvm.lifetime.end.p0i8(i64 20, i8* nonnull %0) #4, !dbg !26
31  ret i32 0, !dbg !27
32}
33
34; the init value of local variable "val" is stored in .rodata section
35; CHECK:             .long   42                              # BTF_KIND_DATASEC
36; CHECK-NEXT:        .long   251658240                       # 0xf000000
37; CHECK-NEXT:        .long   0
38
39; CHECK:             .ascii  ".rodata"                       # string offset=42
40
41; Function Attrs: argmemonly nounwind willreturn
42declare void @llvm.lifetime.start.p0i8(i64 immarg, i8* nocapture) #1
43
44; Function Attrs: nounwind readnone speculatable willreturn
45declare void @llvm.dbg.declare(metadata, metadata, metadata) #2
46
47; Function Attrs: argmemonly nounwind willreturn
48declare void @llvm.memcpy.p0i8.p0i8.i64(i8* noalias nocapture writeonly, i8* noalias nocapture readonly, i64, i1 immarg) #1
49
50declare !dbg !28 dso_local void @foo(i8*) local_unnamed_addr #3
51
52; Function Attrs: argmemonly nounwind willreturn
53declare void @llvm.lifetime.end.p0i8(i64 immarg, i8* nocapture) #1
54
55attributes #0 = { nounwind "correctly-rounded-divide-sqrt-fp-math"="false" "disable-tail-calls"="false" "frame-pointer"="all" "less-precise-fpmad"="false" "min-legal-vector-width"="0" "no-infs-fp-math"="false" "no-jump-tables"="false" "no-nans-fp-math"="false" "no-signed-zeros-fp-math"="false" "no-trapping-math"="true" "stack-protector-buffer-size"="8" "unsafe-fp-math"="false" "use-soft-float"="false" }
56attributes #1 = { argmemonly nounwind willreturn }
57attributes #2 = { nounwind readnone speculatable willreturn }
58attributes #3 = { "correctly-rounded-divide-sqrt-fp-math"="false" "disable-tail-calls"="false" "frame-pointer"="all" "less-precise-fpmad"="false" "no-infs-fp-math"="false" "no-nans-fp-math"="false" "no-signed-zeros-fp-math"="false" "no-trapping-math"="true" "stack-protector-buffer-size"="8" "unsafe-fp-math"="false" "use-soft-float"="false" }
59attributes #4 = { nounwind }
60
61!llvm.dbg.cu = !{!0}
62!llvm.module.flags = !{!3, !4, !5}
63!llvm.ident = !{!6}
64
65!0 = distinct !DICompileUnit(language: DW_LANG_C99, file: !1, producer: "clang version 11.0.0 (https://github.com/llvm/llvm-project.git 1e92cffe18a07c12042b57504dfa7fb709b833c8)", isOptimized: true, runtimeVersion: 0, emissionKind: FullDebug, enums: !2, splitDebugInlining: false, nameTableKind: None)
66!1 = !DIFile(filename: "t.c", directory: "/tmp/home/yhs/tmp")
67!2 = !{}
68!3 = !{i32 7, !"Dwarf Version", i32 4}
69!4 = !{i32 2, !"Debug Info Version", i32 3}
70!5 = !{i32 1, !"wchar_size", i32 4}
71!6 = !{!"clang version 11.0.0 (https://github.com/llvm/llvm-project.git 1e92cffe18a07c12042b57504dfa7fb709b833c8)"}
72!7 = distinct !DISubprogram(name: "test", scope: !1, file: !1, line: 2, type: !8, scopeLine: 2, flags: DIFlagAllCallsDescribed, spFlags: DISPFlagDefinition | DISPFlagOptimized, unit: !0, retainedNodes: !11)
73!8 = !DISubroutineType(types: !9)
74!9 = !{!10}
75!10 = !DIBasicType(name: "int", size: 32, encoding: DW_ATE_signed)
76!11 = !{!12}
77!12 = !DILocalVariable(name: "val", scope: !7, file: !1, line: 6, type: !13)
78!13 = !DIDerivedType(tag: DW_TAG_const_type, baseType: !14)
79!14 = distinct !DICompositeType(tag: DW_TAG_structure_type, scope: !7, file: !1, line: 3, size: 160, elements: !15)
80!15 = !{!16, !21}
81!16 = !DIDerivedType(tag: DW_TAG_member, name: "a", scope: !14, file: !1, line: 4, baseType: !17, size: 128)
82!17 = !DICompositeType(tag: DW_TAG_array_type, baseType: !18, size: 128, elements: !19)
83!18 = !DIBasicType(name: "unsigned int", size: 32, encoding: DW_ATE_unsigned)
84!19 = !{!20}
85!20 = !DISubrange(count: 4)
86!21 = !DIDerivedType(tag: DW_TAG_member, name: "b", scope: !14, file: !1, line: 5, baseType: !22, size: 8, offset: 128)
87!22 = !DIBasicType(name: "char", size: 8, encoding: DW_ATE_signed_char)
88!23 = !DILocation(line: 3, column: 3, scope: !7)
89!24 = !DILocation(line: 6, column: 5, scope: !7)
90!25 = !DILocation(line: 7, column: 3, scope: !7)
91!26 = !DILocation(line: 9, column: 1, scope: !7)
92!27 = !DILocation(line: 8, column: 3, scope: !7)
93!28 = !DISubprogram(name: "foo", scope: !1, file: !1, line: 1, type: !29, flags: DIFlagPrototyped, spFlags: DISPFlagOptimized, retainedNodes: !2)
94!29 = !DISubroutineType(types: !30)
95!30 = !{null, !31}
96!31 = !DIDerivedType(tag: DW_TAG_pointer_type, baseType: !32, size: 64)
97!32 = !DIDerivedType(tag: DW_TAG_const_type, baseType: null)
98