|
@@ -351,7 +351,8 @@ static void build_epilogue(struct jit_ctx *ctx)
|
|
* >0 - successfully JITed a 16-byte eBPF instruction.
|
|
* >0 - successfully JITed a 16-byte eBPF instruction.
|
|
* <0 - failed to JIT.
|
|
* <0 - failed to JIT.
|
|
*/
|
|
*/
|
|
-static int build_insn(const struct bpf_insn *insn, struct jit_ctx *ctx)
|
|
|
|
|
|
+static int build_insn(const struct bpf_insn *insn, struct jit_ctx *ctx,
|
|
|
|
+ bool extra_pass)
|
|
{
|
|
{
|
|
const u8 code = insn->code;
|
|
const u8 code = insn->code;
|
|
const u8 dst = bpf2a64[insn->dst_reg];
|
|
const u8 dst = bpf2a64[insn->dst_reg];
|
|
@@ -625,12 +626,19 @@ emit_cond_jmp:
|
|
case BPF_JMP | BPF_CALL:
|
|
case BPF_JMP | BPF_CALL:
|
|
{
|
|
{
|
|
const u8 r0 = bpf2a64[BPF_REG_0];
|
|
const u8 r0 = bpf2a64[BPF_REG_0];
|
|
- const u64 func = (u64)__bpf_call_base + imm;
|
|
|
|
|
|
+ bool func_addr_fixed;
|
|
|
|
+ u64 func_addr;
|
|
|
|
+ int ret;
|
|
|
|
|
|
- if (ctx->prog->is_func)
|
|
|
|
- emit_addr_mov_i64(tmp, func, ctx);
|
|
|
|
|
|
+ ret = bpf_jit_get_func_addr(ctx->prog, insn, extra_pass,
|
|
|
|
+ &func_addr, &func_addr_fixed);
|
|
|
|
+ if (ret < 0)
|
|
|
|
+ return ret;
|
|
|
|
+ if (func_addr_fixed)
|
|
|
|
+ /* We can use optimized emission here. */
|
|
|
|
+ emit_a64_mov_i64(tmp, func_addr, ctx);
|
|
else
|
|
else
|
|
- emit_a64_mov_i64(tmp, func, ctx);
|
|
|
|
|
|
+ emit_addr_mov_i64(tmp, func_addr, ctx);
|
|
emit(A64_BLR(tmp), ctx);
|
|
emit(A64_BLR(tmp), ctx);
|
|
emit(A64_MOV(1, r0, A64_R(0)), ctx);
|
|
emit(A64_MOV(1, r0, A64_R(0)), ctx);
|
|
break;
|
|
break;
|
|
@@ -753,7 +761,7 @@ emit_cond_jmp:
|
|
return 0;
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
|
|
-static int build_body(struct jit_ctx *ctx)
|
|
|
|
|
|
+static int build_body(struct jit_ctx *ctx, bool extra_pass)
|
|
{
|
|
{
|
|
const struct bpf_prog *prog = ctx->prog;
|
|
const struct bpf_prog *prog = ctx->prog;
|
|
int i;
|
|
int i;
|
|
@@ -762,7 +770,7 @@ static int build_body(struct jit_ctx *ctx)
|
|
const struct bpf_insn *insn = &prog->insnsi[i];
|
|
const struct bpf_insn *insn = &prog->insnsi[i];
|
|
int ret;
|
|
int ret;
|
|
|
|
|
|
- ret = build_insn(insn, ctx);
|
|
|
|
|
|
+ ret = build_insn(insn, ctx, extra_pass);
|
|
if (ret > 0) {
|
|
if (ret > 0) {
|
|
i++;
|
|
i++;
|
|
if (ctx->image == NULL)
|
|
if (ctx->image == NULL)
|
|
@@ -858,7 +866,7 @@ struct bpf_prog *bpf_int_jit_compile(struct bpf_prog *prog)
|
|
/* 1. Initial fake pass to compute ctx->idx. */
|
|
/* 1. Initial fake pass to compute ctx->idx. */
|
|
|
|
|
|
/* Fake pass to fill in ctx->offset. */
|
|
/* Fake pass to fill in ctx->offset. */
|
|
- if (build_body(&ctx)) {
|
|
|
|
|
|
+ if (build_body(&ctx, extra_pass)) {
|
|
prog = orig_prog;
|
|
prog = orig_prog;
|
|
goto out_off;
|
|
goto out_off;
|
|
}
|
|
}
|
|
@@ -888,7 +896,7 @@ skip_init_ctx:
|
|
|
|
|
|
build_prologue(&ctx, was_classic);
|
|
build_prologue(&ctx, was_classic);
|
|
|
|
|
|
- if (build_body(&ctx)) {
|
|
|
|
|
|
+ if (build_body(&ctx, extra_pass)) {
|
|
bpf_jit_binary_free(header);
|
|
bpf_jit_binary_free(header);
|
|
prog = orig_prog;
|
|
prog = orig_prog;
|
|
goto out_off;
|
|
goto out_off;
|