powerpc/bpf/32: introduce a second source register for ALU operations
At the time being, all ALU operation are performed with same L-source and destination, requiring the L-source to be moved into destination via a separate register move, like: 70: 7f c6 f3 78 mr r6,r30 74: 7f a5 eb 78 mr r5,r29 78: 30 c6 ff f4 addic r6,r6,-12 7c: 7c a5 01 d4 addme r5,r5 Introduce a second source register to all ALU operations. For the time being that second source register is made equal to the destination register. That change will allow, via following patch, to optimise the generated code as: 70: 30 de ff f4 addic r6,r30,-12 74: 7c bd 01 d4 addme r5,r29 Signed-off-by: Christophe Leroy <christophe.leroy@csgroup.eu> Signed-off-by: Michael Ellerman <mpe@ellerman.id.au> Link: https://lore.kernel.org/r/d5aaaba50d9d6b4a0e9f0cd4a5e34101aca1e247.1675245773.git.christophe.leroy@csgroup.eu
This commit is contained in:
parent
8616045fe7
commit
c88da29b4d
@ -294,6 +294,8 @@ int bpf_jit_build_body(struct bpf_prog *fp, u32 *image, struct codegen_context *
|
||||
u32 dst_reg_h = dst_reg - 1;
|
||||
u32 src_reg = bpf_to_ppc(insn[i].src_reg);
|
||||
u32 src_reg_h = src_reg - 1;
|
||||
u32 src2_reg = dst_reg;
|
||||
u32 src2_reg_h = dst_reg_h;
|
||||
u32 ax_reg = bpf_to_ppc(BPF_REG_AX);
|
||||
u32 tmp_reg = bpf_to_ppc(TMP_REG);
|
||||
u32 size = BPF_SIZE(code);
|
||||
@ -338,108 +340,111 @@ int bpf_jit_build_body(struct bpf_prog *fp, u32 *image, struct codegen_context *
|
||||
* Arithmetic operations: ADD/SUB/MUL/DIV/MOD/NEG
|
||||
*/
|
||||
case BPF_ALU | BPF_ADD | BPF_X: /* (u32) dst += (u32) src */
|
||||
EMIT(PPC_RAW_ADD(dst_reg, dst_reg, src_reg));
|
||||
EMIT(PPC_RAW_ADD(dst_reg, src2_reg, src_reg));
|
||||
break;
|
||||
case BPF_ALU64 | BPF_ADD | BPF_X: /* dst += src */
|
||||
EMIT(PPC_RAW_ADDC(dst_reg, dst_reg, src_reg));
|
||||
EMIT(PPC_RAW_ADDE(dst_reg_h, dst_reg_h, src_reg_h));
|
||||
EMIT(PPC_RAW_ADDC(dst_reg, src2_reg, src_reg));
|
||||
EMIT(PPC_RAW_ADDE(dst_reg_h, src2_reg_h, src_reg_h));
|
||||
break;
|
||||
case BPF_ALU | BPF_SUB | BPF_X: /* (u32) dst -= (u32) src */
|
||||
EMIT(PPC_RAW_SUB(dst_reg, dst_reg, src_reg));
|
||||
EMIT(PPC_RAW_SUB(dst_reg, src2_reg, src_reg));
|
||||
break;
|
||||
case BPF_ALU64 | BPF_SUB | BPF_X: /* dst -= src */
|
||||
EMIT(PPC_RAW_SUBFC(dst_reg, src_reg, dst_reg));
|
||||
EMIT(PPC_RAW_SUBFE(dst_reg_h, src_reg_h, dst_reg_h));
|
||||
EMIT(PPC_RAW_SUBFC(dst_reg, src_reg, src2_reg));
|
||||
EMIT(PPC_RAW_SUBFE(dst_reg_h, src_reg_h, src2_reg_h));
|
||||
break;
|
||||
case BPF_ALU | BPF_SUB | BPF_K: /* (u32) dst -= (u32) imm */
|
||||
imm = -imm;
|
||||
fallthrough;
|
||||
case BPF_ALU | BPF_ADD | BPF_K: /* (u32) dst += (u32) imm */
|
||||
if (IMM_HA(imm) & 0xffff)
|
||||
EMIT(PPC_RAW_ADDIS(dst_reg, dst_reg, IMM_HA(imm)));
|
||||
if (!imm) {
|
||||
EMIT(PPC_RAW_MR(dst_reg, src2_reg));
|
||||
} else if (IMM_HA(imm) & 0xffff) {
|
||||
EMIT(PPC_RAW_ADDIS(dst_reg, src2_reg, IMM_HA(imm)));
|
||||
src2_reg = dst_reg;
|
||||
}
|
||||
if (IMM_L(imm))
|
||||
EMIT(PPC_RAW_ADDI(dst_reg, dst_reg, IMM_L(imm)));
|
||||
EMIT(PPC_RAW_ADDI(dst_reg, src2_reg, IMM_L(imm)));
|
||||
break;
|
||||
case BPF_ALU64 | BPF_SUB | BPF_K: /* dst -= imm */
|
||||
imm = -imm;
|
||||
fallthrough;
|
||||
case BPF_ALU64 | BPF_ADD | BPF_K: /* dst += imm */
|
||||
if (!imm)
|
||||
if (!imm) {
|
||||
EMIT(PPC_RAW_MR(dst_reg, src2_reg));
|
||||
EMIT(PPC_RAW_MR(dst_reg_h, src2_reg_h));
|
||||
break;
|
||||
|
||||
}
|
||||
if (imm >= -32768 && imm < 32768) {
|
||||
EMIT(PPC_RAW_ADDIC(dst_reg, dst_reg, imm));
|
||||
EMIT(PPC_RAW_ADDIC(dst_reg, src2_reg, imm));
|
||||
} else {
|
||||
PPC_LI32(_R0, imm);
|
||||
EMIT(PPC_RAW_ADDC(dst_reg, dst_reg, _R0));
|
||||
EMIT(PPC_RAW_ADDC(dst_reg, src2_reg, _R0));
|
||||
}
|
||||
if (imm >= 0 || (BPF_OP(code) == BPF_SUB && imm == 0x80000000))
|
||||
EMIT(PPC_RAW_ADDZE(dst_reg_h, dst_reg_h));
|
||||
EMIT(PPC_RAW_ADDZE(dst_reg_h, src2_reg_h));
|
||||
else
|
||||
EMIT(PPC_RAW_ADDME(dst_reg_h, dst_reg_h));
|
||||
EMIT(PPC_RAW_ADDME(dst_reg_h, src2_reg_h));
|
||||
break;
|
||||
case BPF_ALU64 | BPF_MUL | BPF_X: /* dst *= src */
|
||||
bpf_set_seen_register(ctx, tmp_reg);
|
||||
EMIT(PPC_RAW_MULW(_R0, dst_reg, src_reg_h));
|
||||
EMIT(PPC_RAW_MULW(dst_reg_h, dst_reg_h, src_reg));
|
||||
EMIT(PPC_RAW_MULHWU(tmp_reg, dst_reg, src_reg));
|
||||
EMIT(PPC_RAW_MULW(dst_reg, dst_reg, src_reg));
|
||||
EMIT(PPC_RAW_MULW(_R0, src2_reg, src_reg_h));
|
||||
EMIT(PPC_RAW_MULW(dst_reg_h, src2_reg_h, src_reg));
|
||||
EMIT(PPC_RAW_MULHWU(tmp_reg, src2_reg, src_reg));
|
||||
EMIT(PPC_RAW_MULW(dst_reg, src2_reg, src_reg));
|
||||
EMIT(PPC_RAW_ADD(dst_reg_h, dst_reg_h, _R0));
|
||||
EMIT(PPC_RAW_ADD(dst_reg_h, dst_reg_h, tmp_reg));
|
||||
break;
|
||||
case BPF_ALU | BPF_MUL | BPF_X: /* (u32) dst *= (u32) src */
|
||||
EMIT(PPC_RAW_MULW(dst_reg, dst_reg, src_reg));
|
||||
EMIT(PPC_RAW_MULW(dst_reg, src2_reg, src_reg));
|
||||
break;
|
||||
case BPF_ALU | BPF_MUL | BPF_K: /* (u32) dst *= (u32) imm */
|
||||
if (imm == 1)
|
||||
break;
|
||||
if (imm == -1) {
|
||||
EMIT(PPC_RAW_SUBFIC(dst_reg, dst_reg, 0));
|
||||
if (imm == 1) {
|
||||
EMIT(PPC_RAW_MR(dst_reg, src2_reg));
|
||||
} else if (imm == -1) {
|
||||
EMIT(PPC_RAW_SUBFIC(dst_reg, src2_reg, 0));
|
||||
} else if (is_power_of_2((u32)imm)) {
|
||||
EMIT(PPC_RAW_SLWI(dst_reg, dst_reg, ilog2(imm)));
|
||||
EMIT(PPC_RAW_SLWI(dst_reg, src2_reg, ilog2(imm)));
|
||||
} else if (imm >= -32768 && imm < 32768) {
|
||||
EMIT(PPC_RAW_MULI(dst_reg, dst_reg, imm));
|
||||
EMIT(PPC_RAW_MULI(dst_reg, src2_reg, imm));
|
||||
} else {
|
||||
PPC_LI32(_R0, imm);
|
||||
EMIT(PPC_RAW_MULW(dst_reg, dst_reg, _R0));
|
||||
EMIT(PPC_RAW_MULW(dst_reg, src2_reg, _R0));
|
||||
}
|
||||
break;
|
||||
case BPF_ALU64 | BPF_MUL | BPF_K: /* dst *= imm */
|
||||
if (!imm) {
|
||||
PPC_LI32(dst_reg, 0);
|
||||
PPC_LI32(dst_reg_h, 0);
|
||||
break;
|
||||
}
|
||||
if (imm == 1)
|
||||
break;
|
||||
if (imm == -1) {
|
||||
EMIT(PPC_RAW_SUBFIC(dst_reg, dst_reg, 0));
|
||||
EMIT(PPC_RAW_SUBFZE(dst_reg_h, dst_reg_h));
|
||||
break;
|
||||
}
|
||||
if (imm > 0 && is_power_of_2(imm)) {
|
||||
} else if (imm == 1) {
|
||||
EMIT(PPC_RAW_MR(dst_reg, src2_reg));
|
||||
EMIT(PPC_RAW_MR(dst_reg_h, src2_reg_h));
|
||||
} else if (imm == -1) {
|
||||
EMIT(PPC_RAW_SUBFIC(dst_reg, src2_reg, 0));
|
||||
EMIT(PPC_RAW_SUBFZE(dst_reg_h, src2_reg_h));
|
||||
} else if (imm > 0 && is_power_of_2(imm)) {
|
||||
imm = ilog2(imm);
|
||||
EMIT(PPC_RAW_RLWINM(dst_reg_h, dst_reg_h, imm, 0, 31 - imm));
|
||||
EMIT(PPC_RAW_RLWINM(dst_reg_h, src2_reg_h, imm, 0, 31 - imm));
|
||||
EMIT(PPC_RAW_RLWIMI(dst_reg_h, dst_reg, imm, 32 - imm, 31));
|
||||
EMIT(PPC_RAW_SLWI(dst_reg, dst_reg, imm));
|
||||
break;
|
||||
EMIT(PPC_RAW_SLWI(dst_reg, src2_reg, imm));
|
||||
} else {
|
||||
bpf_set_seen_register(ctx, tmp_reg);
|
||||
PPC_LI32(tmp_reg, imm);
|
||||
EMIT(PPC_RAW_MULW(dst_reg_h, src2_reg_h, tmp_reg));
|
||||
if (imm < 0)
|
||||
EMIT(PPC_RAW_SUB(dst_reg_h, dst_reg_h, src2_reg));
|
||||
EMIT(PPC_RAW_MULHWU(_R0, src2_reg, tmp_reg));
|
||||
EMIT(PPC_RAW_MULW(dst_reg, src2_reg, tmp_reg));
|
||||
EMIT(PPC_RAW_ADD(dst_reg_h, dst_reg_h, _R0));
|
||||
}
|
||||
bpf_set_seen_register(ctx, tmp_reg);
|
||||
PPC_LI32(tmp_reg, imm);
|
||||
EMIT(PPC_RAW_MULW(dst_reg_h, dst_reg_h, tmp_reg));
|
||||
if (imm < 0)
|
||||
EMIT(PPC_RAW_SUB(dst_reg_h, dst_reg_h, dst_reg));
|
||||
EMIT(PPC_RAW_MULHWU(_R0, dst_reg, tmp_reg));
|
||||
EMIT(PPC_RAW_MULW(dst_reg, dst_reg, tmp_reg));
|
||||
EMIT(PPC_RAW_ADD(dst_reg_h, dst_reg_h, _R0));
|
||||
break;
|
||||
case BPF_ALU | BPF_DIV | BPF_X: /* (u32) dst /= (u32) src */
|
||||
EMIT(PPC_RAW_DIVWU(dst_reg, dst_reg, src_reg));
|
||||
EMIT(PPC_RAW_DIVWU(dst_reg, src2_reg, src_reg));
|
||||
break;
|
||||
case BPF_ALU | BPF_MOD | BPF_X: /* (u32) dst %= (u32) src */
|
||||
EMIT(PPC_RAW_DIVWU(_R0, dst_reg, src_reg));
|
||||
EMIT(PPC_RAW_DIVWU(_R0, src2_reg, src_reg));
|
||||
EMIT(PPC_RAW_MULW(_R0, src_reg, _R0));
|
||||
EMIT(PPC_RAW_SUB(dst_reg, dst_reg, _R0));
|
||||
EMIT(PPC_RAW_SUB(dst_reg, src2_reg, _R0));
|
||||
break;
|
||||
case BPF_ALU64 | BPF_DIV | BPF_X: /* dst /= src */
|
||||
return -EOPNOTSUPP;
|
||||
@ -448,14 +453,13 @@ int bpf_jit_build_body(struct bpf_prog *fp, u32 *image, struct codegen_context *
|
||||
case BPF_ALU | BPF_DIV | BPF_K: /* (u32) dst /= (u32) imm */
|
||||
if (!imm)
|
||||
return -EINVAL;
|
||||
if (imm == 1)
|
||||
break;
|
||||
|
||||
if (is_power_of_2((u32)imm)) {
|
||||
EMIT(PPC_RAW_SRWI(dst_reg, dst_reg, ilog2(imm)));
|
||||
if (imm == 1) {
|
||||
EMIT(PPC_RAW_MR(dst_reg, src2_reg));
|
||||
} else if (is_power_of_2((u32)imm)) {
|
||||
EMIT(PPC_RAW_SRWI(dst_reg, src2_reg, ilog2(imm)));
|
||||
} else {
|
||||
PPC_LI32(_R0, imm);
|
||||
EMIT(PPC_RAW_DIVWU(dst_reg, dst_reg, _R0));
|
||||
EMIT(PPC_RAW_DIVWU(dst_reg, src2_reg, _R0));
|
||||
}
|
||||
break;
|
||||
case BPF_ALU | BPF_MOD | BPF_K: /* (u32) dst %= (u32) imm */
|
||||
@ -465,16 +469,15 @@ int bpf_jit_build_body(struct bpf_prog *fp, u32 *image, struct codegen_context *
|
||||
if (!is_power_of_2((u32)imm)) {
|
||||
bpf_set_seen_register(ctx, tmp_reg);
|
||||
PPC_LI32(tmp_reg, imm);
|
||||
EMIT(PPC_RAW_DIVWU(_R0, dst_reg, tmp_reg));
|
||||
EMIT(PPC_RAW_DIVWU(_R0, src2_reg, tmp_reg));
|
||||
EMIT(PPC_RAW_MULW(_R0, tmp_reg, _R0));
|
||||
EMIT(PPC_RAW_SUB(dst_reg, dst_reg, _R0));
|
||||
break;
|
||||
}
|
||||
if (imm == 1)
|
||||
EMIT(PPC_RAW_SUB(dst_reg, src2_reg, _R0));
|
||||
} else if (imm == 1) {
|
||||
EMIT(PPC_RAW_LI(dst_reg, 0));
|
||||
else
|
||||
EMIT(PPC_RAW_RLWINM(dst_reg, dst_reg, 0, 32 - ilog2((u32)imm), 31));
|
||||
|
||||
} else {
|
||||
imm = ilog2((u32)imm);
|
||||
EMIT(PPC_RAW_RLWINM(dst_reg, src2_reg, 0, 32 - imm, 31));
|
||||
}
|
||||
break;
|
||||
case BPF_ALU64 | BPF_MOD | BPF_K: /* dst %= imm */
|
||||
if (!imm)
|
||||
@ -486,7 +489,7 @@ int bpf_jit_build_body(struct bpf_prog *fp, u32 *image, struct codegen_context *
|
||||
if (imm == 1)
|
||||
EMIT(PPC_RAW_LI(dst_reg, 0));
|
||||
else
|
||||
EMIT(PPC_RAW_RLWINM(dst_reg, dst_reg, 0, 32 - ilog2(imm), 31));
|
||||
EMIT(PPC_RAW_RLWINM(dst_reg, src2_reg, 0, 32 - ilog2(imm), 31));
|
||||
EMIT(PPC_RAW_LI(dst_reg_h, 0));
|
||||
break;
|
||||
case BPF_ALU64 | BPF_DIV | BPF_K: /* dst /= imm */
|
||||
@ -496,34 +499,38 @@ int bpf_jit_build_body(struct bpf_prog *fp, u32 *image, struct codegen_context *
|
||||
return -EOPNOTSUPP;
|
||||
|
||||
if (imm < 0) {
|
||||
EMIT(PPC_RAW_SUBFIC(dst_reg, dst_reg, 0));
|
||||
EMIT(PPC_RAW_SUBFZE(dst_reg_h, dst_reg_h));
|
||||
EMIT(PPC_RAW_SUBFIC(dst_reg, src2_reg, 0));
|
||||
EMIT(PPC_RAW_SUBFZE(dst_reg_h, src2_reg_h));
|
||||
imm = -imm;
|
||||
src2_reg = dst_reg;
|
||||
}
|
||||
if (imm == 1) {
|
||||
EMIT(PPC_RAW_MR(dst_reg, src2_reg));
|
||||
EMIT(PPC_RAW_MR(dst_reg_h, src2_reg_h));
|
||||
} else {
|
||||
imm = ilog2(imm);
|
||||
EMIT(PPC_RAW_RLWINM(dst_reg, src2_reg, 32 - imm, imm, 31));
|
||||
EMIT(PPC_RAW_RLWIMI(dst_reg, src2_reg_h, 32 - imm, 0, imm - 1));
|
||||
EMIT(PPC_RAW_SRAWI(dst_reg_h, src2_reg_h, imm));
|
||||
}
|
||||
if (imm == 1)
|
||||
break;
|
||||
imm = ilog2(imm);
|
||||
EMIT(PPC_RAW_RLWINM(dst_reg, dst_reg, 32 - imm, imm, 31));
|
||||
EMIT(PPC_RAW_RLWIMI(dst_reg, dst_reg_h, 32 - imm, 0, imm - 1));
|
||||
EMIT(PPC_RAW_SRAWI(dst_reg_h, dst_reg_h, imm));
|
||||
break;
|
||||
case BPF_ALU | BPF_NEG: /* (u32) dst = -dst */
|
||||
EMIT(PPC_RAW_NEG(dst_reg, dst_reg));
|
||||
EMIT(PPC_RAW_NEG(dst_reg, src2_reg));
|
||||
break;
|
||||
case BPF_ALU64 | BPF_NEG: /* dst = -dst */
|
||||
EMIT(PPC_RAW_SUBFIC(dst_reg, dst_reg, 0));
|
||||
EMIT(PPC_RAW_SUBFZE(dst_reg_h, dst_reg_h));
|
||||
EMIT(PPC_RAW_SUBFIC(dst_reg, src2_reg, 0));
|
||||
EMIT(PPC_RAW_SUBFZE(dst_reg_h, src2_reg_h));
|
||||
break;
|
||||
|
||||
/*
|
||||
* Logical operations: AND/OR/XOR/[A]LSH/[A]RSH
|
||||
*/
|
||||
case BPF_ALU64 | BPF_AND | BPF_X: /* dst = dst & src */
|
||||
EMIT(PPC_RAW_AND(dst_reg, dst_reg, src_reg));
|
||||
EMIT(PPC_RAW_AND(dst_reg_h, dst_reg_h, src_reg_h));
|
||||
EMIT(PPC_RAW_AND(dst_reg, src2_reg, src_reg));
|
||||
EMIT(PPC_RAW_AND(dst_reg_h, src2_reg_h, src_reg_h));
|
||||
break;
|
||||
case BPF_ALU | BPF_AND | BPF_X: /* (u32) dst = dst & src */
|
||||
EMIT(PPC_RAW_AND(dst_reg, dst_reg, src_reg));
|
||||
EMIT(PPC_RAW_AND(dst_reg, src2_reg, src_reg));
|
||||
break;
|
||||
case BPF_ALU64 | BPF_AND | BPF_K: /* dst = dst & imm */
|
||||
if (imm >= 0)
|
||||
@ -531,23 +538,23 @@ int bpf_jit_build_body(struct bpf_prog *fp, u32 *image, struct codegen_context *
|
||||
fallthrough;
|
||||
case BPF_ALU | BPF_AND | BPF_K: /* (u32) dst = dst & imm */
|
||||
if (!IMM_H(imm)) {
|
||||
EMIT(PPC_RAW_ANDI(dst_reg, dst_reg, IMM_L(imm)));
|
||||
EMIT(PPC_RAW_ANDI(dst_reg, src2_reg, IMM_L(imm)));
|
||||
} else if (!IMM_L(imm)) {
|
||||
EMIT(PPC_RAW_ANDIS(dst_reg, dst_reg, IMM_H(imm)));
|
||||
EMIT(PPC_RAW_ANDIS(dst_reg, src2_reg, IMM_H(imm)));
|
||||
} else if (imm == (((1 << fls(imm)) - 1) ^ ((1 << (ffs(i) - 1)) - 1))) {
|
||||
EMIT(PPC_RAW_RLWINM(dst_reg, dst_reg, 0,
|
||||
EMIT(PPC_RAW_RLWINM(dst_reg, src2_reg, 0,
|
||||
32 - fls(imm), 32 - ffs(imm)));
|
||||
} else {
|
||||
PPC_LI32(_R0, imm);
|
||||
EMIT(PPC_RAW_AND(dst_reg, dst_reg, _R0));
|
||||
EMIT(PPC_RAW_AND(dst_reg, src2_reg, _R0));
|
||||
}
|
||||
break;
|
||||
case BPF_ALU64 | BPF_OR | BPF_X: /* dst = dst | src */
|
||||
EMIT(PPC_RAW_OR(dst_reg, dst_reg, src_reg));
|
||||
EMIT(PPC_RAW_OR(dst_reg_h, dst_reg_h, src_reg_h));
|
||||
EMIT(PPC_RAW_OR(dst_reg, src2_reg, src_reg));
|
||||
EMIT(PPC_RAW_OR(dst_reg_h, src2_reg_h, src_reg_h));
|
||||
break;
|
||||
case BPF_ALU | BPF_OR | BPF_X: /* dst = (u32) dst | (u32) src */
|
||||
EMIT(PPC_RAW_OR(dst_reg, dst_reg, src_reg));
|
||||
EMIT(PPC_RAW_OR(dst_reg, src2_reg, src_reg));
|
||||
break;
|
||||
case BPF_ALU64 | BPF_OR | BPF_K:/* dst = dst | imm */
|
||||
/* Sign-extended */
|
||||
@ -555,145 +562,154 @@ int bpf_jit_build_body(struct bpf_prog *fp, u32 *image, struct codegen_context *
|
||||
EMIT(PPC_RAW_LI(dst_reg_h, -1));
|
||||
fallthrough;
|
||||
case BPF_ALU | BPF_OR | BPF_K:/* dst = (u32) dst | (u32) imm */
|
||||
if (IMM_L(imm))
|
||||
EMIT(PPC_RAW_ORI(dst_reg, dst_reg, IMM_L(imm)));
|
||||
if (IMM_L(imm)) {
|
||||
EMIT(PPC_RAW_ORI(dst_reg, src2_reg, IMM_L(imm)));
|
||||
src2_reg = dst_reg;
|
||||
}
|
||||
if (IMM_H(imm))
|
||||
EMIT(PPC_RAW_ORIS(dst_reg, dst_reg, IMM_H(imm)));
|
||||
EMIT(PPC_RAW_ORIS(dst_reg, src2_reg, IMM_H(imm)));
|
||||
break;
|
||||
case BPF_ALU64 | BPF_XOR | BPF_X: /* dst ^= src */
|
||||
if (dst_reg == src_reg) {
|
||||
EMIT(PPC_RAW_LI(dst_reg, 0));
|
||||
EMIT(PPC_RAW_LI(dst_reg_h, 0));
|
||||
} else {
|
||||
EMIT(PPC_RAW_XOR(dst_reg, dst_reg, src_reg));
|
||||
EMIT(PPC_RAW_XOR(dst_reg_h, dst_reg_h, src_reg_h));
|
||||
EMIT(PPC_RAW_XOR(dst_reg, src2_reg, src_reg));
|
||||
EMIT(PPC_RAW_XOR(dst_reg_h, src2_reg_h, src_reg_h));
|
||||
}
|
||||
break;
|
||||
case BPF_ALU | BPF_XOR | BPF_X: /* (u32) dst ^= src */
|
||||
if (dst_reg == src_reg)
|
||||
EMIT(PPC_RAW_LI(dst_reg, 0));
|
||||
else
|
||||
EMIT(PPC_RAW_XOR(dst_reg, dst_reg, src_reg));
|
||||
EMIT(PPC_RAW_XOR(dst_reg, src2_reg, src_reg));
|
||||
break;
|
||||
case BPF_ALU64 | BPF_XOR | BPF_K: /* dst ^= imm */
|
||||
if (imm < 0)
|
||||
EMIT(PPC_RAW_NOR(dst_reg_h, dst_reg_h, dst_reg_h));
|
||||
EMIT(PPC_RAW_NOR(dst_reg_h, src2_reg_h, src2_reg_h));
|
||||
fallthrough;
|
||||
case BPF_ALU | BPF_XOR | BPF_K: /* (u32) dst ^= (u32) imm */
|
||||
if (IMM_L(imm))
|
||||
EMIT(PPC_RAW_XORI(dst_reg, dst_reg, IMM_L(imm)));
|
||||
if (IMM_L(imm)) {
|
||||
EMIT(PPC_RAW_XORI(dst_reg, src2_reg, IMM_L(imm)));
|
||||
src2_reg = dst_reg;
|
||||
}
|
||||
if (IMM_H(imm))
|
||||
EMIT(PPC_RAW_XORIS(dst_reg, dst_reg, IMM_H(imm)));
|
||||
EMIT(PPC_RAW_XORIS(dst_reg, src2_reg, IMM_H(imm)));
|
||||
break;
|
||||
case BPF_ALU | BPF_LSH | BPF_X: /* (u32) dst <<= (u32) src */
|
||||
EMIT(PPC_RAW_SLW(dst_reg, dst_reg, src_reg));
|
||||
EMIT(PPC_RAW_SLW(dst_reg, src2_reg, src_reg));
|
||||
break;
|
||||
case BPF_ALU64 | BPF_LSH | BPF_X: /* dst <<= src; */
|
||||
bpf_set_seen_register(ctx, tmp_reg);
|
||||
EMIT(PPC_RAW_SUBFIC(_R0, src_reg, 32));
|
||||
EMIT(PPC_RAW_SLW(dst_reg_h, dst_reg_h, src_reg));
|
||||
EMIT(PPC_RAW_SLW(dst_reg_h, src2_reg_h, src_reg));
|
||||
EMIT(PPC_RAW_ADDI(tmp_reg, src_reg, 32));
|
||||
EMIT(PPC_RAW_SRW(_R0, dst_reg, _R0));
|
||||
EMIT(PPC_RAW_SLW(tmp_reg, dst_reg, tmp_reg));
|
||||
EMIT(PPC_RAW_SRW(_R0, src2_reg, _R0));
|
||||
EMIT(PPC_RAW_SLW(tmp_reg, src2_reg, tmp_reg));
|
||||
EMIT(PPC_RAW_OR(dst_reg_h, dst_reg_h, _R0));
|
||||
EMIT(PPC_RAW_SLW(dst_reg, dst_reg, src_reg));
|
||||
EMIT(PPC_RAW_SLW(dst_reg, src2_reg, src_reg));
|
||||
EMIT(PPC_RAW_OR(dst_reg_h, dst_reg_h, tmp_reg));
|
||||
break;
|
||||
case BPF_ALU | BPF_LSH | BPF_K: /* (u32) dst <<= (u32) imm */
|
||||
if (!imm)
|
||||
break;
|
||||
EMIT(PPC_RAW_SLWI(dst_reg, dst_reg, imm));
|
||||
if (imm)
|
||||
EMIT(PPC_RAW_SLWI(dst_reg, src2_reg, imm));
|
||||
else
|
||||
EMIT(PPC_RAW_MR(dst_reg, src2_reg));
|
||||
break;
|
||||
case BPF_ALU64 | BPF_LSH | BPF_K: /* dst <<= imm */
|
||||
if (imm < 0)
|
||||
return -EINVAL;
|
||||
if (!imm)
|
||||
break;
|
||||
if (imm < 32) {
|
||||
EMIT(PPC_RAW_RLWINM(dst_reg_h, dst_reg_h, imm, 0, 31 - imm));
|
||||
EMIT(PPC_RAW_RLWIMI(dst_reg_h, dst_reg, imm, 32 - imm, 31));
|
||||
EMIT(PPC_RAW_RLWINM(dst_reg, dst_reg, imm, 0, 31 - imm));
|
||||
break;
|
||||
}
|
||||
if (imm < 64)
|
||||
EMIT(PPC_RAW_RLWINM(dst_reg_h, dst_reg, imm, 0, 31 - imm));
|
||||
else
|
||||
if (!imm) {
|
||||
EMIT(PPC_RAW_MR(dst_reg, src2_reg));
|
||||
} else if (imm < 32) {
|
||||
EMIT(PPC_RAW_RLWINM(dst_reg_h, src2_reg_h, imm, 0, 31 - imm));
|
||||
EMIT(PPC_RAW_RLWIMI(dst_reg_h, src2_reg, imm, 32 - imm, 31));
|
||||
EMIT(PPC_RAW_RLWINM(dst_reg, src2_reg, imm, 0, 31 - imm));
|
||||
} else if (imm < 64) {
|
||||
EMIT(PPC_RAW_RLWINM(dst_reg_h, src2_reg, imm, 0, 31 - imm));
|
||||
EMIT(PPC_RAW_LI(dst_reg, 0));
|
||||
} else {
|
||||
EMIT(PPC_RAW_LI(dst_reg_h, 0));
|
||||
EMIT(PPC_RAW_LI(dst_reg, 0));
|
||||
EMIT(PPC_RAW_LI(dst_reg, 0));
|
||||
}
|
||||
break;
|
||||
case BPF_ALU | BPF_RSH | BPF_X: /* (u32) dst >>= (u32) src */
|
||||
EMIT(PPC_RAW_SRW(dst_reg, dst_reg, src_reg));
|
||||
EMIT(PPC_RAW_SRW(dst_reg, src2_reg, src_reg));
|
||||
break;
|
||||
case BPF_ALU64 | BPF_RSH | BPF_X: /* dst >>= src */
|
||||
bpf_set_seen_register(ctx, tmp_reg);
|
||||
EMIT(PPC_RAW_SUBFIC(_R0, src_reg, 32));
|
||||
EMIT(PPC_RAW_SRW(dst_reg, dst_reg, src_reg));
|
||||
EMIT(PPC_RAW_SRW(dst_reg, src2_reg, src_reg));
|
||||
EMIT(PPC_RAW_ADDI(tmp_reg, src_reg, 32));
|
||||
EMIT(PPC_RAW_SLW(_R0, dst_reg_h, _R0));
|
||||
EMIT(PPC_RAW_SLW(_R0, src2_reg_h, _R0));
|
||||
EMIT(PPC_RAW_SRW(tmp_reg, dst_reg_h, tmp_reg));
|
||||
EMIT(PPC_RAW_OR(dst_reg, dst_reg, _R0));
|
||||
EMIT(PPC_RAW_SRW(dst_reg_h, dst_reg_h, src_reg));
|
||||
EMIT(PPC_RAW_SRW(dst_reg_h, src2_reg_h, src_reg));
|
||||
EMIT(PPC_RAW_OR(dst_reg, dst_reg, tmp_reg));
|
||||
break;
|
||||
case BPF_ALU | BPF_RSH | BPF_K: /* (u32) dst >>= (u32) imm */
|
||||
if (!imm)
|
||||
break;
|
||||
EMIT(PPC_RAW_SRWI(dst_reg, dst_reg, imm));
|
||||
if (imm)
|
||||
EMIT(PPC_RAW_SRWI(dst_reg, src2_reg, imm));
|
||||
else
|
||||
EMIT(PPC_RAW_MR(dst_reg, src2_reg));
|
||||
break;
|
||||
case BPF_ALU64 | BPF_RSH | BPF_K: /* dst >>= imm */
|
||||
if (imm < 0)
|
||||
return -EINVAL;
|
||||
if (!imm)
|
||||
break;
|
||||
if (imm < 32) {
|
||||
EMIT(PPC_RAW_RLWINM(dst_reg, dst_reg, 32 - imm, imm, 31));
|
||||
EMIT(PPC_RAW_RLWIMI(dst_reg, dst_reg_h, 32 - imm, 0, imm - 1));
|
||||
EMIT(PPC_RAW_RLWINM(dst_reg_h, dst_reg_h, 32 - imm, imm, 31));
|
||||
break;
|
||||
}
|
||||
if (imm < 64)
|
||||
EMIT(PPC_RAW_RLWINM(dst_reg, dst_reg_h, 64 - imm, imm - 32, 31));
|
||||
else
|
||||
if (!imm) {
|
||||
EMIT(PPC_RAW_MR(dst_reg, src2_reg));
|
||||
EMIT(PPC_RAW_MR(dst_reg_h, src2_reg_h));
|
||||
} else if (imm < 32) {
|
||||
EMIT(PPC_RAW_RLWINM(dst_reg, src2_reg, 32 - imm, imm, 31));
|
||||
EMIT(PPC_RAW_RLWIMI(dst_reg, src2_reg_h, 32 - imm, 0, imm - 1));
|
||||
EMIT(PPC_RAW_RLWINM(dst_reg_h, src2_reg_h, 32 - imm, imm, 31));
|
||||
} else if (imm < 64) {
|
||||
EMIT(PPC_RAW_RLWINM(dst_reg, src2_reg_h, 64 - imm, imm - 32, 31));
|
||||
EMIT(PPC_RAW_LI(dst_reg_h, 0));
|
||||
} else {
|
||||
EMIT(PPC_RAW_LI(dst_reg, 0));
|
||||
EMIT(PPC_RAW_LI(dst_reg_h, 0));
|
||||
EMIT(PPC_RAW_LI(dst_reg_h, 0));
|
||||
}
|
||||
break;
|
||||
case BPF_ALU | BPF_ARSH | BPF_X: /* (s32) dst >>= src */
|
||||
EMIT(PPC_RAW_SRAW(dst_reg, dst_reg, src_reg));
|
||||
EMIT(PPC_RAW_SRAW(dst_reg, src2_reg, src_reg));
|
||||
break;
|
||||
case BPF_ALU64 | BPF_ARSH | BPF_X: /* (s64) dst >>= src */
|
||||
bpf_set_seen_register(ctx, tmp_reg);
|
||||
EMIT(PPC_RAW_SUBFIC(_R0, src_reg, 32));
|
||||
EMIT(PPC_RAW_SRW(dst_reg, dst_reg, src_reg));
|
||||
EMIT(PPC_RAW_SLW(_R0, dst_reg_h, _R0));
|
||||
EMIT(PPC_RAW_SRW(dst_reg, src2_reg, src_reg));
|
||||
EMIT(PPC_RAW_SLW(_R0, src2_reg_h, _R0));
|
||||
EMIT(PPC_RAW_ADDI(tmp_reg, src_reg, 32));
|
||||
EMIT(PPC_RAW_OR(dst_reg, dst_reg, _R0));
|
||||
EMIT(PPC_RAW_RLWINM(_R0, tmp_reg, 0, 26, 26));
|
||||
EMIT(PPC_RAW_SRAW(tmp_reg, dst_reg_h, tmp_reg));
|
||||
EMIT(PPC_RAW_SRAW(dst_reg_h, dst_reg_h, src_reg));
|
||||
EMIT(PPC_RAW_SRAW(tmp_reg, src2_reg_h, tmp_reg));
|
||||
EMIT(PPC_RAW_SRAW(dst_reg_h, src2_reg_h, src_reg));
|
||||
EMIT(PPC_RAW_SLW(tmp_reg, tmp_reg, _R0));
|
||||
EMIT(PPC_RAW_OR(dst_reg, dst_reg, tmp_reg));
|
||||
break;
|
||||
case BPF_ALU | BPF_ARSH | BPF_K: /* (s32) dst >>= imm */
|
||||
if (!imm)
|
||||
break;
|
||||
EMIT(PPC_RAW_SRAWI(dst_reg, dst_reg, imm));
|
||||
if (imm)
|
||||
EMIT(PPC_RAW_SRAWI(dst_reg, src2_reg, imm));
|
||||
else
|
||||
EMIT(PPC_RAW_MR(dst_reg, src2_reg));
|
||||
break;
|
||||
case BPF_ALU64 | BPF_ARSH | BPF_K: /* (s64) dst >>= imm */
|
||||
if (imm < 0)
|
||||
return -EINVAL;
|
||||
if (!imm)
|
||||
break;
|
||||
if (imm < 32) {
|
||||
EMIT(PPC_RAW_RLWINM(dst_reg, dst_reg, 32 - imm, imm, 31));
|
||||
EMIT(PPC_RAW_RLWIMI(dst_reg, dst_reg_h, 32 - imm, 0, imm - 1));
|
||||
EMIT(PPC_RAW_SRAWI(dst_reg_h, dst_reg_h, imm));
|
||||
break;
|
||||
if (!imm) {
|
||||
EMIT(PPC_RAW_MR(dst_reg, src2_reg));
|
||||
EMIT(PPC_RAW_MR(dst_reg_h, src2_reg_h));
|
||||
} else if (imm < 32) {
|
||||
EMIT(PPC_RAW_RLWINM(dst_reg, src2_reg, 32 - imm, imm, 31));
|
||||
EMIT(PPC_RAW_RLWIMI(dst_reg, src2_reg_h, 32 - imm, 0, imm - 1));
|
||||
EMIT(PPC_RAW_SRAWI(dst_reg_h, src2_reg_h, imm));
|
||||
} else if (imm < 64) {
|
||||
EMIT(PPC_RAW_SRAWI(dst_reg, src2_reg_h, imm - 32));
|
||||
EMIT(PPC_RAW_SRAWI(dst_reg_h, src2_reg_h, 31));
|
||||
} else {
|
||||
EMIT(PPC_RAW_SRAWI(dst_reg, src2_reg_h, 31));
|
||||
EMIT(PPC_RAW_SRAWI(dst_reg_h, src2_reg_h, 31));
|
||||
}
|
||||
if (imm < 64)
|
||||
EMIT(PPC_RAW_SRAWI(dst_reg, dst_reg_h, imm - 32));
|
||||
else
|
||||
EMIT(PPC_RAW_SRAWI(dst_reg, dst_reg_h, 31));
|
||||
EMIT(PPC_RAW_SRAWI(dst_reg_h, dst_reg_h, 31));
|
||||
break;
|
||||
|
||||
/*
|
||||
@ -727,7 +743,7 @@ int bpf_jit_build_body(struct bpf_prog *fp, u32 *image, struct codegen_context *
|
||||
switch (imm) {
|
||||
case 16:
|
||||
/* Copy 16 bits to upper part */
|
||||
EMIT(PPC_RAW_RLWIMI(dst_reg, dst_reg, 16, 0, 15));
|
||||
EMIT(PPC_RAW_RLWIMI(dst_reg, src2_reg, 16, 0, 15));
|
||||
/* Rotate 8 bits right & mask */
|
||||
EMIT(PPC_RAW_RLWINM(dst_reg, dst_reg, 24, 16, 31));
|
||||
break;
|
||||
@ -737,23 +753,23 @@ int bpf_jit_build_body(struct bpf_prog *fp, u32 *image, struct codegen_context *
|
||||
* 2 bytes are already in their final position
|
||||
* -- byte 2 and 4 (of bytes 1, 2, 3 and 4)
|
||||
*/
|
||||
EMIT(PPC_RAW_RLWINM(_R0, dst_reg, 8, 0, 31));
|
||||
EMIT(PPC_RAW_RLWINM(_R0, src2_reg, 8, 0, 31));
|
||||
/* Rotate 24 bits and insert byte 1 */
|
||||
EMIT(PPC_RAW_RLWIMI(_R0, dst_reg, 24, 0, 7));
|
||||
EMIT(PPC_RAW_RLWIMI(_R0, src2_reg, 24, 0, 7));
|
||||
/* Rotate 24 bits and insert byte 3 */
|
||||
EMIT(PPC_RAW_RLWIMI(_R0, dst_reg, 24, 16, 23));
|
||||
EMIT(PPC_RAW_RLWIMI(_R0, src2_reg, 24, 16, 23));
|
||||
EMIT(PPC_RAW_MR(dst_reg, _R0));
|
||||
break;
|
||||
case 64:
|
||||
bpf_set_seen_register(ctx, tmp_reg);
|
||||
EMIT(PPC_RAW_RLWINM(tmp_reg, dst_reg, 8, 0, 31));
|
||||
EMIT(PPC_RAW_RLWINM(_R0, dst_reg_h, 8, 0, 31));
|
||||
EMIT(PPC_RAW_RLWINM(tmp_reg, src2_reg, 8, 0, 31));
|
||||
EMIT(PPC_RAW_RLWINM(_R0, src2_reg_h, 8, 0, 31));
|
||||
/* Rotate 24 bits and insert byte 1 */
|
||||
EMIT(PPC_RAW_RLWIMI(tmp_reg, dst_reg, 24, 0, 7));
|
||||
EMIT(PPC_RAW_RLWIMI(_R0, dst_reg_h, 24, 0, 7));
|
||||
EMIT(PPC_RAW_RLWIMI(tmp_reg, src2_reg, 24, 0, 7));
|
||||
EMIT(PPC_RAW_RLWIMI(_R0, src2_reg_h, 24, 0, 7));
|
||||
/* Rotate 24 bits and insert byte 3 */
|
||||
EMIT(PPC_RAW_RLWIMI(tmp_reg, dst_reg, 24, 16, 23));
|
||||
EMIT(PPC_RAW_RLWIMI(_R0, dst_reg_h, 24, 16, 23));
|
||||
EMIT(PPC_RAW_RLWIMI(tmp_reg, src2_reg, 24, 16, 23));
|
||||
EMIT(PPC_RAW_RLWIMI(_R0, src2_reg_h, 24, 16, 23));
|
||||
EMIT(PPC_RAW_MR(dst_reg, _R0));
|
||||
EMIT(PPC_RAW_MR(dst_reg_h, tmp_reg));
|
||||
break;
|
||||
@ -763,7 +779,7 @@ int bpf_jit_build_body(struct bpf_prog *fp, u32 *image, struct codegen_context *
|
||||
switch (imm) {
|
||||
case 16:
|
||||
/* zero-extend 16 bits into 32 bits */
|
||||
EMIT(PPC_RAW_RLWINM(dst_reg, dst_reg, 0, 16, 31));
|
||||
EMIT(PPC_RAW_RLWINM(dst_reg, src2_reg, 0, 16, 31));
|
||||
break;
|
||||
case 32:
|
||||
case 64:
|
||||
|
Loading…
x
Reference in New Issue
Block a user