Skip to content

Commit 50fe7eb

Browse files
lukenelsAlexei Starovoitov
authored andcommitted
bpf, x86_32: Fix clobbering of dst for BPF_JSET
The current JIT clobbers the destination register for BPF_JSET BPF_X and BPF_K by using "and" and "or" instructions. This is fine when the destination register is a temporary loaded from a register stored on the stack but not otherwise. This patch fixes the problem (for both BPF_K and BPF_X) by always loading the destination register into temporaries since BPF_JSET should not modify the destination register. This bug may not be currently triggerable as BPF_REG_AX is the only register not stored on the stack and the verifier uses it in a limited way. Fixes: 03f5781 ("bpf, x86_32: add eBPF JIT compiler for ia32") Signed-off-by: Xi Wang <xi.wang@gmail.com> Signed-off-by: Luke Nelson <luke.r.nels@gmail.com> Signed-off-by: Alexei Starovoitov <ast@kernel.org> Acked-by: Wang YanQing <udknight@gmail.com> Link: https://lore.kernel.org/bpf/20200422173630.8351-2-luke.r.nels@gmail.com
1 parent 5fa9a98 commit 50fe7eb

File tree

1 file changed

+18
-4
lines changed

1 file changed

+18
-4
lines changed

arch/x86/net/bpf_jit_comp32.c

Lines changed: 18 additions & 4 deletions
Original file line numberDiff line numberDiff line change
@@ -2015,8 +2015,8 @@ static int do_jit(struct bpf_prog *bpf_prog, int *addrs, u8 *image,
20152015
case BPF_JMP | BPF_JSET | BPF_X:
20162016
case BPF_JMP32 | BPF_JSET | BPF_X: {
20172017
bool is_jmp64 = BPF_CLASS(insn->code) == BPF_JMP;
2018-
u8 dreg_lo = dstk ? IA32_EAX : dst_lo;
2019-
u8 dreg_hi = dstk ? IA32_EDX : dst_hi;
2018+
u8 dreg_lo = IA32_EAX;
2019+
u8 dreg_hi = IA32_EDX;
20202020
u8 sreg_lo = sstk ? IA32_ECX : src_lo;
20212021
u8 sreg_hi = sstk ? IA32_EBX : src_hi;
20222022

@@ -2028,6 +2028,13 @@ static int do_jit(struct bpf_prog *bpf_prog, int *addrs, u8 *image,
20282028
add_2reg(0x40, IA32_EBP,
20292029
IA32_EDX),
20302030
STACK_VAR(dst_hi));
2031+
} else {
2032+
/* mov dreg_lo,dst_lo */
2033+
EMIT2(0x89, add_2reg(0xC0, dreg_lo, dst_lo));
2034+
if (is_jmp64)
2035+
/* mov dreg_hi,dst_hi */
2036+
EMIT2(0x89,
2037+
add_2reg(0xC0, dreg_hi, dst_hi));
20312038
}
20322039

20332040
if (sstk) {
@@ -2052,8 +2059,8 @@ static int do_jit(struct bpf_prog *bpf_prog, int *addrs, u8 *image,
20522059
case BPF_JMP | BPF_JSET | BPF_K:
20532060
case BPF_JMP32 | BPF_JSET | BPF_K: {
20542061
bool is_jmp64 = BPF_CLASS(insn->code) == BPF_JMP;
2055-
u8 dreg_lo = dstk ? IA32_EAX : dst_lo;
2056-
u8 dreg_hi = dstk ? IA32_EDX : dst_hi;
2062+
u8 dreg_lo = IA32_EAX;
2063+
u8 dreg_hi = IA32_EDX;
20572064
u8 sreg_lo = IA32_ECX;
20582065
u8 sreg_hi = IA32_EBX;
20592066
u32 hi;
@@ -2066,6 +2073,13 @@ static int do_jit(struct bpf_prog *bpf_prog, int *addrs, u8 *image,
20662073
add_2reg(0x40, IA32_EBP,
20672074
IA32_EDX),
20682075
STACK_VAR(dst_hi));
2076+
} else {
2077+
/* mov dreg_lo,dst_lo */
2078+
EMIT2(0x89, add_2reg(0xC0, dreg_lo, dst_lo));
2079+
if (is_jmp64)
2080+
/* mov dreg_hi,dst_hi */
2081+
EMIT2(0x89,
2082+
add_2reg(0xC0, dreg_hi, dst_hi));
20692083
}
20702084

20712085
/* mov ecx,imm32 */

0 commit comments

Comments
 (0)