diff options
| -rw-r--r-- | arch/arm64/net/bpf_jit_comp.c | 26 | 
1 files changed, 17 insertions, 9 deletions
| diff --git a/arch/arm64/net/bpf_jit_comp.c b/arch/arm64/net/bpf_jit_comp.c index a6fdaea07c63..89198017e8e6 100644 --- a/arch/arm64/net/bpf_jit_comp.c +++ b/arch/arm64/net/bpf_jit_comp.c @@ -351,7 +351,8 @@ static void build_epilogue(struct jit_ctx *ctx)   * >0 - successfully JITed a 16-byte eBPF instruction.   * <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 dst = bpf2a64[insn->dst_reg]; @@ -625,12 +626,19 @@ emit_cond_jmp:  	case BPF_JMP | BPF_CALL:  	{  		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 -			emit_a64_mov_i64(tmp, func, ctx); +			emit_addr_mov_i64(tmp, func_addr, ctx);  		emit(A64_BLR(tmp), ctx);  		emit(A64_MOV(1, r0, A64_R(0)), ctx);  		break; @@ -753,7 +761,7 @@ emit_cond_jmp:  	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;  	int i; @@ -762,7 +770,7 @@ static int build_body(struct jit_ctx *ctx)  		const struct bpf_insn *insn = &prog->insnsi[i];  		int ret; -		ret = build_insn(insn, ctx); +		ret = build_insn(insn, ctx, extra_pass);  		if (ret > 0) {  			i++;  			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. */  	/* Fake pass to fill in ctx->offset. */ -	if (build_body(&ctx)) { +	if (build_body(&ctx, extra_pass)) {  		prog = orig_prog;  		goto out_off;  	} @@ -888,7 +896,7 @@ skip_init_ctx:  	build_prologue(&ctx, was_classic); -	if (build_body(&ctx)) { +	if (build_body(&ctx, extra_pass)) {  		bpf_jit_binary_free(header);  		prog = orig_prog;  		goto out_off; | 
