llvm/test/CodeGen/BPF/sockex2.ll
Alexei Starovoitov bc5bb8c0ae convert bpf assembler to look like kernel verifier output
since bpf instruction set was introduced people learned to
read and understand kernel verifier output whereas llvm asm
output stayed obscure and unknown. Convert llvm to emit
assembler text similar to kernel to avoid this discrepancy

Signed-off-by: Alexei Starovoitov <ast@kernel.org>

git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@287300 91177308-0d34-0410-b5e6-96231b3b80d8
2016-11-18 02:32:35 +00:00

325 lines
12 KiB
LLVM

; RUN: llc < %s -march=bpfel -show-mc-encoding | FileCheck %s
%struct.bpf_map_def = type { i32, i32, i32, i32 }
%struct.sk_buff = type opaque
@hash_map = global %struct.bpf_map_def { i32 1, i32 4, i32 8, i32 1024 }, section "maps", align 4
; Function Attrs: nounwind uwtable
define i32 @bpf_prog2(%struct.sk_buff* %skb) #0 section "socket2" {
%key = alloca i32, align 4
%val = alloca i64, align 8
%1 = bitcast %struct.sk_buff* %skb to i8*
%2 = call i64 @llvm.bpf.load.half(i8* %1, i64 12) #2
%3 = icmp eq i64 %2, 34984
br i1 %3, label %4, label %6
; <label>:4 ; preds = %0
%5 = call i64 @llvm.bpf.load.half(i8* %1, i64 16) #2
br label %6
; <label>:6 ; preds = %4, %0
%proto.0.i = phi i64 [ %5, %4 ], [ %2, %0 ]
%nhoff.0.i = phi i64 [ 18, %4 ], [ 14, %0 ]
%7 = icmp eq i64 %proto.0.i, 33024
br i1 %7, label %8, label %12
; <label>:8 ; preds = %6
%9 = add i64 %nhoff.0.i, 2
%10 = call i64 @llvm.bpf.load.half(i8* %1, i64 %9) #2
%11 = add i64 %nhoff.0.i, 4
br label %12
; <label>:12 ; preds = %8, %6
%proto.1.i = phi i64 [ %10, %8 ], [ %proto.0.i, %6 ]
%nhoff.1.i = phi i64 [ %11, %8 ], [ %nhoff.0.i, %6 ]
switch i64 %proto.1.i, label %flow_dissector.exit.thread [
i64 2048, label %13
i64 34525, label %39
]
; <label>:13 ; preds = %12
%14 = add i64 %nhoff.1.i, 6
%15 = call i64 @llvm.bpf.load.half(i8* %1, i64 %14) #2
%16 = and i64 %15, 16383
%17 = icmp eq i64 %16, 0
br i1 %17, label %18, label %.thread.i.i
; <label>:18 ; preds = %13
%19 = add i64 %nhoff.1.i, 9
%20 = call i64 @llvm.bpf.load.byte(i8* %1, i64 %19) #2
%21 = icmp eq i64 %20, 47
br i1 %21, label %28, label %.thread.i.i
.thread.i.i: ; preds = %18, %13
%22 = phi i64 [ %20, %18 ], [ 0, %13 ]
%23 = add i64 %nhoff.1.i, 12
%24 = call i64 @llvm.bpf.load.word(i8* %1, i64 %23) #2
%25 = add i64 %nhoff.1.i, 16
%26 = call i64 @llvm.bpf.load.word(i8* %1, i64 %25) #2
%27 = trunc i64 %26 to i32
br label %28
; <label>:28 ; preds = %.thread.i.i, %18
%29 = phi i32 [ %27, %.thread.i.i ], [ undef, %18 ]
%30 = phi i64 [ %22, %.thread.i.i ], [ 47, %18 ]
%31 = call i64 @llvm.bpf.load.byte(i8* %1, i64 %nhoff.1.i) #2
%32 = icmp eq i64 %31, 69
br i1 %32, label %33, label %35
; <label>:33 ; preds = %28
%34 = add i64 %nhoff.1.i, 20
br label %parse_ip.exit.i
; <label>:35 ; preds = %28
%36 = shl i64 %31, 2
%37 = and i64 %36, 60
%38 = add i64 %37, %nhoff.1.i
br label %parse_ip.exit.i
; <label>:39 ; preds = %12
%40 = add i64 %nhoff.1.i, 6
%41 = call i64 @llvm.bpf.load.byte(i8* %1, i64 %40) #2
%42 = add i64 %nhoff.1.i, 8
%43 = call i64 @llvm.bpf.load.word(i8* %1, i64 %42) #2
%44 = add i64 %nhoff.1.i, 12
%45 = call i64 @llvm.bpf.load.word(i8* %1, i64 %44) #2
%46 = add i64 %nhoff.1.i, 16
%47 = call i64 @llvm.bpf.load.word(i8* %1, i64 %46) #2
%48 = add i64 %nhoff.1.i, 20
%49 = call i64 @llvm.bpf.load.word(i8* %1, i64 %48) #2
%50 = add i64 %nhoff.1.i, 24
%51 = call i64 @llvm.bpf.load.word(i8* %1, i64 %50) #2
%52 = add i64 %nhoff.1.i, 28
%53 = call i64 @llvm.bpf.load.word(i8* %1, i64 %52) #2
%54 = add i64 %nhoff.1.i, 32
%55 = call i64 @llvm.bpf.load.word(i8* %1, i64 %54) #2
%56 = add i64 %nhoff.1.i, 36
%57 = call i64 @llvm.bpf.load.word(i8* %1, i64 %56) #2
%58 = xor i64 %53, %51
%59 = xor i64 %58, %55
%60 = xor i64 %59, %57
%61 = trunc i64 %60 to i32
%62 = add i64 %nhoff.1.i, 40
br label %parse_ip.exit.i
parse_ip.exit.i: ; preds = %39, %35, %33
%63 = phi i32 [ %61, %39 ], [ %29, %33 ], [ %29, %35 ]
%64 = phi i64 [ %41, %39 ], [ %30, %33 ], [ %30, %35 ]
%nhoff.2.i = phi i64 [ %62, %39 ], [ %34, %33 ], [ %38, %35 ]
switch i64 %64, label %187 [
i64 47, label %65
i64 4, label %137
i64 41, label %163
]
; <label>:65 ; preds = %parse_ip.exit.i
%66 = call i64 @llvm.bpf.load.half(i8* %1, i64 %nhoff.2.i) #2
%67 = add i64 %nhoff.2.i, 2
%68 = call i64 @llvm.bpf.load.half(i8* %1, i64 %67) #2
%69 = and i64 %66, 1856
%70 = icmp eq i64 %69, 0
br i1 %70, label %71, label %187
; <label>:71 ; preds = %65
%72 = lshr i64 %66, 5
%73 = and i64 %72, 4
%74 = add i64 %nhoff.2.i, 4
%..i = add i64 %74, %73
%75 = and i64 %66, 32
%76 = icmp eq i64 %75, 0
%77 = add i64 %..i, 4
%nhoff.4.i = select i1 %76, i64 %..i, i64 %77
%78 = and i64 %66, 16
%79 = icmp eq i64 %78, 0
%80 = add i64 %nhoff.4.i, 4
%nhoff.4..i = select i1 %79, i64 %nhoff.4.i, i64 %80
%81 = icmp eq i64 %68, 33024
br i1 %81, label %82, label %86
; <label>:82 ; preds = %71
%83 = add i64 %nhoff.4..i, 2
%84 = call i64 @llvm.bpf.load.half(i8* %1, i64 %83) #2
%85 = add i64 %nhoff.4..i, 4
br label %86
; <label>:86 ; preds = %82, %71
%proto.2.i = phi i64 [ %84, %82 ], [ %68, %71 ]
%nhoff.6.i = phi i64 [ %85, %82 ], [ %nhoff.4..i, %71 ]
switch i64 %proto.2.i, label %flow_dissector.exit.thread [
i64 2048, label %87
i64 34525, label %113
]
; <label>:87 ; preds = %86
%88 = add i64 %nhoff.6.i, 6
%89 = call i64 @llvm.bpf.load.half(i8* %1, i64 %88) #2
%90 = and i64 %89, 16383
%91 = icmp eq i64 %90, 0
br i1 %91, label %92, label %.thread.i4.i
; <label>:92 ; preds = %87
%93 = add i64 %nhoff.6.i, 9
%94 = call i64 @llvm.bpf.load.byte(i8* %1, i64 %93) #2
%95 = icmp eq i64 %94, 47
br i1 %95, label %102, label %.thread.i4.i
.thread.i4.i: ; preds = %92, %87
%96 = phi i64 [ %94, %92 ], [ 0, %87 ]
%97 = add i64 %nhoff.6.i, 12
%98 = call i64 @llvm.bpf.load.word(i8* %1, i64 %97) #2
%99 = add i64 %nhoff.6.i, 16
%100 = call i64 @llvm.bpf.load.word(i8* %1, i64 %99) #2
%101 = trunc i64 %100 to i32
br label %102
; <label>:102 ; preds = %.thread.i4.i, %92
%103 = phi i32 [ %101, %.thread.i4.i ], [ %63, %92 ]
%104 = phi i64 [ %96, %.thread.i4.i ], [ 47, %92 ]
%105 = call i64 @llvm.bpf.load.byte(i8* %1, i64 %nhoff.6.i) #2
%106 = icmp eq i64 %105, 69
br i1 %106, label %107, label %109
; <label>:107 ; preds = %102
%108 = add i64 %nhoff.6.i, 20
br label %187
; <label>:109 ; preds = %102
%110 = shl i64 %105, 2
%111 = and i64 %110, 60
%112 = add i64 %111, %nhoff.6.i
br label %187
; <label>:113 ; preds = %86
%114 = add i64 %nhoff.6.i, 6
%115 = call i64 @llvm.bpf.load.byte(i8* %1, i64 %114) #2
%116 = add i64 %nhoff.6.i, 8
%117 = call i64 @llvm.bpf.load.word(i8* %1, i64 %116) #2
%118 = add i64 %nhoff.6.i, 12
%119 = call i64 @llvm.bpf.load.word(i8* %1, i64 %118) #2
%120 = add i64 %nhoff.6.i, 16
%121 = call i64 @llvm.bpf.load.word(i8* %1, i64 %120) #2
%122 = add i64 %nhoff.6.i, 20
%123 = call i64 @llvm.bpf.load.word(i8* %1, i64 %122) #2
%124 = add i64 %nhoff.6.i, 24
%125 = call i64 @llvm.bpf.load.word(i8* %1, i64 %124) #2
%126 = add i64 %nhoff.6.i, 28
%127 = call i64 @llvm.bpf.load.word(i8* %1, i64 %126) #2
%128 = add i64 %nhoff.6.i, 32
%129 = call i64 @llvm.bpf.load.word(i8* %1, i64 %128) #2
%130 = add i64 %nhoff.6.i, 36
%131 = call i64 @llvm.bpf.load.word(i8* %1, i64 %130) #2
%132 = xor i64 %127, %125
%133 = xor i64 %132, %129
%134 = xor i64 %133, %131
%135 = trunc i64 %134 to i32
%136 = add i64 %nhoff.6.i, 40
br label %187
; <label>:137 ; preds = %parse_ip.exit.i
%138 = add i64 %nhoff.2.i, 6
%139 = call i64 @llvm.bpf.load.half(i8* %1, i64 %138) #2
%140 = and i64 %139, 16383
%141 = icmp eq i64 %140, 0
br i1 %141, label %142, label %.thread.i1.i
; <label>:142 ; preds = %137
%143 = add i64 %nhoff.2.i, 9
%144 = call i64 @llvm.bpf.load.byte(i8* %1, i64 %143) #2
%145 = icmp eq i64 %144, 47
br i1 %145, label %152, label %.thread.i1.i
.thread.i1.i: ; preds = %142, %137
%146 = phi i64 [ %144, %142 ], [ 0, %137 ]
%147 = add i64 %nhoff.2.i, 12
%148 = call i64 @llvm.bpf.load.word(i8* %1, i64 %147) #2
%149 = add i64 %nhoff.2.i, 16
%150 = call i64 @llvm.bpf.load.word(i8* %1, i64 %149) #2
%151 = trunc i64 %150 to i32
br label %152
; <label>:152 ; preds = %.thread.i1.i, %142
%153 = phi i32 [ %151, %.thread.i1.i ], [ %63, %142 ]
%154 = phi i64 [ %146, %.thread.i1.i ], [ 47, %142 ]
%155 = call i64 @llvm.bpf.load.byte(i8* %1, i64 %nhoff.2.i) #2
%156 = icmp eq i64 %155, 69
br i1 %156, label %157, label %159
; <label>:157 ; preds = %152
%158 = add i64 %nhoff.2.i, 20
br label %187
; <label>:159 ; preds = %152
%160 = shl i64 %155, 2
%161 = and i64 %160, 60
%162 = add i64 %161, %nhoff.2.i
br label %187
; <label>:163 ; preds = %parse_ip.exit.i
%164 = add i64 %nhoff.2.i, 6
%165 = call i64 @llvm.bpf.load.byte(i8* %1, i64 %164) #2
%166 = add i64 %nhoff.2.i, 8
%167 = call i64 @llvm.bpf.load.word(i8* %1, i64 %166) #2
%168 = add i64 %nhoff.2.i, 12
%169 = call i64 @llvm.bpf.load.word(i8* %1, i64 %168) #2
%170 = add i64 %nhoff.2.i, 16
%171 = call i64 @llvm.bpf.load.word(i8* %1, i64 %170) #2
%172 = add i64 %nhoff.2.i, 20
%173 = call i64 @llvm.bpf.load.word(i8* %1, i64 %172) #2
%174 = add i64 %nhoff.2.i, 24
%175 = call i64 @llvm.bpf.load.word(i8* %1, i64 %174) #2
%176 = add i64 %nhoff.2.i, 28
%177 = call i64 @llvm.bpf.load.word(i8* %1, i64 %176) #2
%178 = add i64 %nhoff.2.i, 32
%179 = call i64 @llvm.bpf.load.word(i8* %1, i64 %178) #2
%180 = add i64 %nhoff.2.i, 36
%181 = call i64 @llvm.bpf.load.word(i8* %1, i64 %180) #2
%182 = xor i64 %177, %175
%183 = xor i64 %182, %179
%184 = xor i64 %183, %181
%185 = trunc i64 %184 to i32
%186 = add i64 %nhoff.2.i, 40
br label %187
; <label>:187 ; preds = %163, %159, %157, %113, %109, %107, %65, %parse_ip.exit.i
%188 = phi i32 [ %63, %parse_ip.exit.i ], [ %185, %163 ], [ %63, %65 ], [ %135, %113 ], [ %103, %107 ], [ %103, %109 ], [ %153, %157 ], [ %153, %159 ]
%189 = phi i64 [ %64, %parse_ip.exit.i ], [ %165, %163 ], [ 47, %65 ], [ %115, %113 ], [ %104, %107 ], [ %104, %109 ], [ %154, %157 ], [ %154, %159 ]
%nhoff.7.i = phi i64 [ %nhoff.2.i, %parse_ip.exit.i ], [ %186, %163 ], [ %nhoff.2.i, %65 ], [ %136, %113 ], [ %108, %107 ], [ %112, %109 ], [ %158, %157 ], [ %162, %159 ]
%cond.i.i = icmp eq i64 %189, 51
%190 = select i1 %cond.i.i, i64 4, i64 0
%191 = add i64 %190, %nhoff.7.i
%192 = call i64 @llvm.bpf.load.word(i8* %1, i64 %191) #2
store i32 %188, i32* %key, align 4
%193 = bitcast i32* %key to i8*
%194 = call i8* inttoptr (i64 1 to i8* (i8*, i8*)*)(i8* bitcast (%struct.bpf_map_def* @hash_map to i8*), i8* %193) #2
%195 = icmp eq i8* %194, null
br i1 %195, label %199, label %196
; <label>:196 ; preds = %187
%197 = bitcast i8* %194 to i64*
%198 = atomicrmw add i64* %197, i64 1 seq_cst
br label %flow_dissector.exit.thread
; <label>:199 ; preds = %187
store i64 1, i64* %val, align 8
%200 = bitcast i64* %val to i8*
%201 = call i32 inttoptr (i64 2 to i32 (i8*, i8*, i8*, i64)*)(i8* bitcast (%struct.bpf_map_def* @hash_map to i8*), i8* %193, i8* %200, i64 0) #2
br label %flow_dissector.exit.thread
flow_dissector.exit.thread: ; preds = %86, %12, %196, %199
ret i32 0
; CHECK-LABEL: bpf_prog2:
; CHECK: r0 = *(u16 *)skb[12] # encoding: [0x28,0x00,0x00,0x00,0x0c,0x00,0x00,0x00]
; CHECK: r0 = *(u16 *)skb[16] # encoding: [0x28,0x00,0x00,0x00,0x10,0x00,0x00,0x00]
; CHECK: implicit-def: %R1
; CHECK: r1 =
; CHECK: call 1 # encoding: [0x85,0x00,0x00,0x00,0x01,0x00,0x00,0x00]
; CHECK: call 2 # encoding: [0x85,0x00,0x00,0x00,0x02,0x00,0x00,0x00]
}
declare i64 @llvm.bpf.load.half(i8*, i64) #1
declare i64 @llvm.bpf.load.word(i8*, i64) #1
declare i64 @llvm.bpf.load.byte(i8*, i64) #1