xref: /openbmc/linux/arch/mips/net/bpf_jit_comp64.c (revision 7364d60c)
1fbc802deSJohan Almbladh // SPDX-License-Identifier: GPL-2.0-only
2fbc802deSJohan Almbladh /*
3fbc802deSJohan Almbladh  * Just-In-Time compiler for eBPF bytecode on MIPS.
4fbc802deSJohan Almbladh  * Implementation of JIT functions for 64-bit CPUs.
5fbc802deSJohan Almbladh  *
6fbc802deSJohan Almbladh  * Copyright (c) 2021 Anyfi Networks AB.
7fbc802deSJohan Almbladh  * Author: Johan Almbladh <johan.almbladh@gmail.com>
8fbc802deSJohan Almbladh  *
9fbc802deSJohan Almbladh  * Based on code and ideas from
10fbc802deSJohan Almbladh  * Copyright (c) 2017 Cavium, Inc.
11fbc802deSJohan Almbladh  * Copyright (c) 2017 Shubham Bansal <illusionist.neo@gmail.com>
12fbc802deSJohan Almbladh  * Copyright (c) 2011 Mircea Gherzan <mgherzan@gmail.com>
13fbc802deSJohan Almbladh  */
14fbc802deSJohan Almbladh 
15fbc802deSJohan Almbladh #include <linux/errno.h>
16fbc802deSJohan Almbladh #include <linux/filter.h>
17fbc802deSJohan Almbladh #include <linux/bpf.h>
18fbc802deSJohan Almbladh #include <asm/cpu-features.h>
19fbc802deSJohan Almbladh #include <asm/isa-rev.h>
20fbc802deSJohan Almbladh #include <asm/uasm.h>
21fbc802deSJohan Almbladh 
22fbc802deSJohan Almbladh #include "bpf_jit_comp.h"
23fbc802deSJohan Almbladh 
24fbc802deSJohan Almbladh /* MIPS t0-t3 are not available in the n64 ABI */
25fbc802deSJohan Almbladh #undef MIPS_R_T0
26fbc802deSJohan Almbladh #undef MIPS_R_T1
27fbc802deSJohan Almbladh #undef MIPS_R_T2
28fbc802deSJohan Almbladh #undef MIPS_R_T3
29fbc802deSJohan Almbladh 
30fbc802deSJohan Almbladh /* Stack is 16-byte aligned in n64 ABI */
31fbc802deSJohan Almbladh #define MIPS_STACK_ALIGNMENT 16
32fbc802deSJohan Almbladh 
33fbc802deSJohan Almbladh /* Extra 64-bit eBPF registers used by JIT */
34fbc802deSJohan Almbladh #define JIT_REG_TC (MAX_BPF_JIT_REG + 0)
35fbc802deSJohan Almbladh #define JIT_REG_ZX (MAX_BPF_JIT_REG + 1)
36fbc802deSJohan Almbladh 
37fbc802deSJohan Almbladh /* Number of prologue bytes to skip when doing a tail call */
38fbc802deSJohan Almbladh #define JIT_TCALL_SKIP 4
39fbc802deSJohan Almbladh 
40fbc802deSJohan Almbladh /* Callee-saved CPU registers that the JIT must preserve */
41fbc802deSJohan Almbladh #define JIT_CALLEE_REGS   \
42fbc802deSJohan Almbladh 	(BIT(MIPS_R_S0) | \
43fbc802deSJohan Almbladh 	 BIT(MIPS_R_S1) | \
44fbc802deSJohan Almbladh 	 BIT(MIPS_R_S2) | \
45fbc802deSJohan Almbladh 	 BIT(MIPS_R_S3) | \
46fbc802deSJohan Almbladh 	 BIT(MIPS_R_S4) | \
47fbc802deSJohan Almbladh 	 BIT(MIPS_R_S5) | \
48fbc802deSJohan Almbladh 	 BIT(MIPS_R_S6) | \
49fbc802deSJohan Almbladh 	 BIT(MIPS_R_S7) | \
50fbc802deSJohan Almbladh 	 BIT(MIPS_R_GP) | \
51fbc802deSJohan Almbladh 	 BIT(MIPS_R_FP) | \
52fbc802deSJohan Almbladh 	 BIT(MIPS_R_RA))
53fbc802deSJohan Almbladh 
54fbc802deSJohan Almbladh /* Caller-saved CPU registers available for JIT use */
55fbc802deSJohan Almbladh #define JIT_CALLER_REGS	  \
56fbc802deSJohan Almbladh 	(BIT(MIPS_R_A5) | \
57fbc802deSJohan Almbladh 	 BIT(MIPS_R_A6) | \
58fbc802deSJohan Almbladh 	 BIT(MIPS_R_A7))
59fbc802deSJohan Almbladh /*
60fbc802deSJohan Almbladh  * Mapping of 64-bit eBPF registers to 64-bit native MIPS registers.
61fbc802deSJohan Almbladh  * MIPS registers t4 - t7 may be used by the JIT as temporary registers.
62fbc802deSJohan Almbladh  * MIPS registers t8 - t9 are reserved for single-register common functions.
63fbc802deSJohan Almbladh  */
64fbc802deSJohan Almbladh static const u8 bpf2mips64[] = {
65fbc802deSJohan Almbladh 	/* Return value from in-kernel function, and exit value from eBPF */
66fbc802deSJohan Almbladh 	[BPF_REG_0] = MIPS_R_V0,
67fbc802deSJohan Almbladh 	/* Arguments from eBPF program to in-kernel function */
68fbc802deSJohan Almbladh 	[BPF_REG_1] = MIPS_R_A0,
69fbc802deSJohan Almbladh 	[BPF_REG_2] = MIPS_R_A1,
70fbc802deSJohan Almbladh 	[BPF_REG_3] = MIPS_R_A2,
71fbc802deSJohan Almbladh 	[BPF_REG_4] = MIPS_R_A3,
72fbc802deSJohan Almbladh 	[BPF_REG_5] = MIPS_R_A4,
73fbc802deSJohan Almbladh 	/* Callee-saved registers that in-kernel function will preserve */
74fbc802deSJohan Almbladh 	[BPF_REG_6] = MIPS_R_S0,
75fbc802deSJohan Almbladh 	[BPF_REG_7] = MIPS_R_S1,
76fbc802deSJohan Almbladh 	[BPF_REG_8] = MIPS_R_S2,
77fbc802deSJohan Almbladh 	[BPF_REG_9] = MIPS_R_S3,
78fbc802deSJohan Almbladh 	/* Read-only frame pointer to access the eBPF stack */
79fbc802deSJohan Almbladh 	[BPF_REG_FP] = MIPS_R_FP,
80fbc802deSJohan Almbladh 	/* Temporary register for blinding constants */
81fbc802deSJohan Almbladh 	[BPF_REG_AX] = MIPS_R_AT,
82fbc802deSJohan Almbladh 	/* Tail call count register, caller-saved */
83fbc802deSJohan Almbladh 	[JIT_REG_TC] = MIPS_R_A5,
84fbc802deSJohan Almbladh 	/* Constant for register zero-extension */
85fbc802deSJohan Almbladh 	[JIT_REG_ZX] = MIPS_R_V1,
86fbc802deSJohan Almbladh };
87fbc802deSJohan Almbladh 
88fbc802deSJohan Almbladh /*
89fbc802deSJohan Almbladh  * MIPS 32-bit operations on 64-bit registers generate a sign-extended
90fbc802deSJohan Almbladh  * result. However, the eBPF ISA mandates zero-extension, so we rely on the
91fbc802deSJohan Almbladh  * verifier to add that for us (emit_zext_ver). In addition, ALU arithmetic
92fbc802deSJohan Almbladh  * operations, right shift and byte swap require properly sign-extended
93fbc802deSJohan Almbladh  * operands or the result is unpredictable. We emit explicit sign-extensions
94fbc802deSJohan Almbladh  * in those cases.
95fbc802deSJohan Almbladh  */
96fbc802deSJohan Almbladh 
97fbc802deSJohan Almbladh /* Sign extension */
emit_sext(struct jit_context * ctx,u8 dst,u8 src)98fbc802deSJohan Almbladh static void emit_sext(struct jit_context *ctx, u8 dst, u8 src)
99fbc802deSJohan Almbladh {
100fbc802deSJohan Almbladh 	emit(ctx, sll, dst, src, 0);
101fbc802deSJohan Almbladh 	clobber_reg(ctx, dst);
102fbc802deSJohan Almbladh }
103fbc802deSJohan Almbladh 
104fbc802deSJohan Almbladh /* Zero extension */
emit_zext(struct jit_context * ctx,u8 dst)105fbc802deSJohan Almbladh static void emit_zext(struct jit_context *ctx, u8 dst)
106fbc802deSJohan Almbladh {
107fbc802deSJohan Almbladh 	if (cpu_has_mips64r2 || cpu_has_mips64r6) {
108fbc802deSJohan Almbladh 		emit(ctx, dinsu, dst, MIPS_R_ZERO, 32, 32);
109fbc802deSJohan Almbladh 	} else {
110fbc802deSJohan Almbladh 		emit(ctx, and, dst, dst, bpf2mips64[JIT_REG_ZX]);
111fbc802deSJohan Almbladh 		access_reg(ctx, JIT_REG_ZX); /* We need the ZX register */
112fbc802deSJohan Almbladh 	}
113fbc802deSJohan Almbladh 	clobber_reg(ctx, dst);
114fbc802deSJohan Almbladh }
115fbc802deSJohan Almbladh 
116fbc802deSJohan Almbladh /* Zero extension, if verifier does not do it for us  */
emit_zext_ver(struct jit_context * ctx,u8 dst)117fbc802deSJohan Almbladh static void emit_zext_ver(struct jit_context *ctx, u8 dst)
118fbc802deSJohan Almbladh {
119fbc802deSJohan Almbladh 	if (!ctx->program->aux->verifier_zext)
120fbc802deSJohan Almbladh 		emit_zext(ctx, dst);
121fbc802deSJohan Almbladh }
122fbc802deSJohan Almbladh 
123fbc802deSJohan Almbladh /* dst = imm (64-bit) */
emit_mov_i64(struct jit_context * ctx,u8 dst,u64 imm64)124fbc802deSJohan Almbladh static void emit_mov_i64(struct jit_context *ctx, u8 dst, u64 imm64)
125fbc802deSJohan Almbladh {
126fbc802deSJohan Almbladh 	if (imm64 >= 0xffffffffffff8000ULL || imm64 < 0x8000ULL) {
127fbc802deSJohan Almbladh 		emit(ctx, daddiu, dst, MIPS_R_ZERO, (s16)imm64);
128fbc802deSJohan Almbladh 	} else if (imm64 >= 0xffffffff80000000ULL ||
129fbc802deSJohan Almbladh 		   (imm64 < 0x80000000 && imm64 > 0xffff)) {
130fbc802deSJohan Almbladh 		emit(ctx, lui, dst, (s16)(imm64 >> 16));
131fbc802deSJohan Almbladh 		emit(ctx, ori, dst, dst, (u16)imm64 & 0xffff);
132fbc802deSJohan Almbladh 	} else {
133fbc802deSJohan Almbladh 		u8 acc = MIPS_R_ZERO;
134bbf731b3SJohan Almbladh 		int shift = 0;
135fbc802deSJohan Almbladh 		int k;
136fbc802deSJohan Almbladh 
137fbc802deSJohan Almbladh 		for (k = 0; k < 4; k++) {
138fbc802deSJohan Almbladh 			u16 half = imm64 >> (48 - 16 * k);
139fbc802deSJohan Almbladh 
140fbc802deSJohan Almbladh 			if (acc == dst)
141bbf731b3SJohan Almbladh 				shift += 16;
142fbc802deSJohan Almbladh 
143fbc802deSJohan Almbladh 			if (half) {
144bbf731b3SJohan Almbladh 				if (shift)
145bbf731b3SJohan Almbladh 					emit(ctx, dsll_safe, dst, dst, shift);
146fbc802deSJohan Almbladh 				emit(ctx, ori, dst, acc, half);
147fbc802deSJohan Almbladh 				acc = dst;
148bbf731b3SJohan Almbladh 				shift = 0;
149fbc802deSJohan Almbladh 			}
150fbc802deSJohan Almbladh 		}
151bbf731b3SJohan Almbladh 		if (shift)
152bbf731b3SJohan Almbladh 			emit(ctx, dsll_safe, dst, dst, shift);
153fbc802deSJohan Almbladh 	}
154fbc802deSJohan Almbladh 	clobber_reg(ctx, dst);
155fbc802deSJohan Almbladh }
156fbc802deSJohan Almbladh 
157fbc802deSJohan Almbladh /* ALU immediate operation (64-bit) */
emit_alu_i64(struct jit_context * ctx,u8 dst,s32 imm,u8 op)158fbc802deSJohan Almbladh static void emit_alu_i64(struct jit_context *ctx, u8 dst, s32 imm, u8 op)
159fbc802deSJohan Almbladh {
160fbc802deSJohan Almbladh 	switch (BPF_OP(op)) {
161fbc802deSJohan Almbladh 	/* dst = dst | imm */
162fbc802deSJohan Almbladh 	case BPF_OR:
163fbc802deSJohan Almbladh 		emit(ctx, ori, dst, dst, (u16)imm);
164fbc802deSJohan Almbladh 		break;
165fbc802deSJohan Almbladh 	/* dst = dst ^ imm */
166fbc802deSJohan Almbladh 	case BPF_XOR:
167fbc802deSJohan Almbladh 		emit(ctx, xori, dst, dst, (u16)imm);
168fbc802deSJohan Almbladh 		break;
169fbc802deSJohan Almbladh 	/* dst = -dst */
170fbc802deSJohan Almbladh 	case BPF_NEG:
171fbc802deSJohan Almbladh 		emit(ctx, dsubu, dst, MIPS_R_ZERO, dst);
172fbc802deSJohan Almbladh 		break;
173fbc802deSJohan Almbladh 	/* dst = dst << imm */
174fbc802deSJohan Almbladh 	case BPF_LSH:
175fbc802deSJohan Almbladh 		emit(ctx, dsll_safe, dst, dst, imm);
176fbc802deSJohan Almbladh 		break;
177fbc802deSJohan Almbladh 	/* dst = dst >> imm */
178fbc802deSJohan Almbladh 	case BPF_RSH:
179fbc802deSJohan Almbladh 		emit(ctx, dsrl_safe, dst, dst, imm);
180fbc802deSJohan Almbladh 		break;
181fbc802deSJohan Almbladh 	/* dst = dst >> imm (arithmetic) */
182fbc802deSJohan Almbladh 	case BPF_ARSH:
183fbc802deSJohan Almbladh 		emit(ctx, dsra_safe, dst, dst, imm);
184fbc802deSJohan Almbladh 		break;
185fbc802deSJohan Almbladh 	/* dst = dst + imm */
186fbc802deSJohan Almbladh 	case BPF_ADD:
187fbc802deSJohan Almbladh 		emit(ctx, daddiu, dst, dst, imm);
188fbc802deSJohan Almbladh 		break;
189fbc802deSJohan Almbladh 	/* dst = dst - imm */
190fbc802deSJohan Almbladh 	case BPF_SUB:
191fbc802deSJohan Almbladh 		emit(ctx, daddiu, dst, dst, -imm);
192fbc802deSJohan Almbladh 		break;
193fbc802deSJohan Almbladh 	default:
194fbc802deSJohan Almbladh 		/* Width-generic operations */
195fbc802deSJohan Almbladh 		emit_alu_i(ctx, dst, imm, op);
196fbc802deSJohan Almbladh 	}
197fbc802deSJohan Almbladh 	clobber_reg(ctx, dst);
198fbc802deSJohan Almbladh }
199fbc802deSJohan Almbladh 
200fbc802deSJohan Almbladh /* ALU register operation (64-bit) */
emit_alu_r64(struct jit_context * ctx,u8 dst,u8 src,u8 op)201fbc802deSJohan Almbladh static void emit_alu_r64(struct jit_context *ctx, u8 dst, u8 src, u8 op)
202fbc802deSJohan Almbladh {
203fbc802deSJohan Almbladh 	switch (BPF_OP(op)) {
204fbc802deSJohan Almbladh 	/* dst = dst << src */
205fbc802deSJohan Almbladh 	case BPF_LSH:
206fbc802deSJohan Almbladh 		emit(ctx, dsllv, dst, dst, src);
207fbc802deSJohan Almbladh 		break;
208fbc802deSJohan Almbladh 	/* dst = dst >> src */
209fbc802deSJohan Almbladh 	case BPF_RSH:
210fbc802deSJohan Almbladh 		emit(ctx, dsrlv, dst, dst, src);
211fbc802deSJohan Almbladh 		break;
212fbc802deSJohan Almbladh 	/* dst = dst >> src (arithmetic) */
213fbc802deSJohan Almbladh 	case BPF_ARSH:
214fbc802deSJohan Almbladh 		emit(ctx, dsrav, dst, dst, src);
215fbc802deSJohan Almbladh 		break;
216fbc802deSJohan Almbladh 	/* dst = dst + src */
217fbc802deSJohan Almbladh 	case BPF_ADD:
218fbc802deSJohan Almbladh 		emit(ctx, daddu, dst, dst, src);
219fbc802deSJohan Almbladh 		break;
220fbc802deSJohan Almbladh 	/* dst = dst - src */
221fbc802deSJohan Almbladh 	case BPF_SUB:
222fbc802deSJohan Almbladh 		emit(ctx, dsubu, dst, dst, src);
223fbc802deSJohan Almbladh 		break;
224fbc802deSJohan Almbladh 	/* dst = dst * src */
225fbc802deSJohan Almbladh 	case BPF_MUL:
226fbc802deSJohan Almbladh 		if (cpu_has_mips64r6) {
227fbc802deSJohan Almbladh 			emit(ctx, dmulu, dst, dst, src);
228fbc802deSJohan Almbladh 		} else {
229fbc802deSJohan Almbladh 			emit(ctx, dmultu, dst, src);
230fbc802deSJohan Almbladh 			emit(ctx, mflo, dst);
231*7364d60cSJiaxun Yang 			/* Ensure multiplication is completed */
232*7364d60cSJiaxun Yang 			if (IS_ENABLED(CONFIG_CPU_R4000_WORKAROUNDS))
233*7364d60cSJiaxun Yang 				emit(ctx, mfhi, MIPS_R_ZERO);
234fbc802deSJohan Almbladh 		}
235fbc802deSJohan Almbladh 		break;
236fbc802deSJohan Almbladh 	/* dst = dst / src */
237fbc802deSJohan Almbladh 	case BPF_DIV:
238fbc802deSJohan Almbladh 		if (cpu_has_mips64r6) {
239fbc802deSJohan Almbladh 			emit(ctx, ddivu_r6, dst, dst, src);
240fbc802deSJohan Almbladh 		} else {
241fbc802deSJohan Almbladh 			emit(ctx, ddivu, dst, src);
242fbc802deSJohan Almbladh 			emit(ctx, mflo, dst);
243fbc802deSJohan Almbladh 		}
244fbc802deSJohan Almbladh 		break;
245fbc802deSJohan Almbladh 	/* dst = dst % src */
246fbc802deSJohan Almbladh 	case BPF_MOD:
247fbc802deSJohan Almbladh 		if (cpu_has_mips64r6) {
248fbc802deSJohan Almbladh 			emit(ctx, dmodu, dst, dst, src);
249fbc802deSJohan Almbladh 		} else {
250fbc802deSJohan Almbladh 			emit(ctx, ddivu, dst, src);
251fbc802deSJohan Almbladh 			emit(ctx, mfhi, dst);
252fbc802deSJohan Almbladh 		}
253fbc802deSJohan Almbladh 		break;
254fbc802deSJohan Almbladh 	default:
255fbc802deSJohan Almbladh 		/* Width-generic operations */
256fbc802deSJohan Almbladh 		emit_alu_r(ctx, dst, src, op);
257fbc802deSJohan Almbladh 	}
258fbc802deSJohan Almbladh 	clobber_reg(ctx, dst);
259fbc802deSJohan Almbladh }
260fbc802deSJohan Almbladh 
261fbc802deSJohan Almbladh /* Swap sub words in a register double word */
emit_swap_r64(struct jit_context * ctx,u8 dst,u8 mask,u32 bits)262fbc802deSJohan Almbladh static void emit_swap_r64(struct jit_context *ctx, u8 dst, u8 mask, u32 bits)
263fbc802deSJohan Almbladh {
264fbc802deSJohan Almbladh 	u8 tmp = MIPS_R_T9;
265fbc802deSJohan Almbladh 
266fbc802deSJohan Almbladh 	emit(ctx, and, tmp, dst, mask);  /* tmp = dst & mask  */
267fbc802deSJohan Almbladh 	emit(ctx, dsll, tmp, tmp, bits); /* tmp = tmp << bits */
268fbc802deSJohan Almbladh 	emit(ctx, dsrl, dst, dst, bits); /* dst = dst >> bits */
269fbc802deSJohan Almbladh 	emit(ctx, and, dst, dst, mask);  /* dst = dst & mask  */
270fbc802deSJohan Almbladh 	emit(ctx, or, dst, dst, tmp);    /* dst = dst | tmp   */
271fbc802deSJohan Almbladh }
272fbc802deSJohan Almbladh 
273fbc802deSJohan Almbladh /* Swap bytes and truncate a register double word, word or half word */
emit_bswap_r64(struct jit_context * ctx,u8 dst,u32 width)274fbc802deSJohan Almbladh static void emit_bswap_r64(struct jit_context *ctx, u8 dst, u32 width)
275fbc802deSJohan Almbladh {
276fbc802deSJohan Almbladh 	switch (width) {
277fbc802deSJohan Almbladh 	/* Swap bytes in a double word */
278fbc802deSJohan Almbladh 	case 64:
279fbc802deSJohan Almbladh 		if (cpu_has_mips64r2 || cpu_has_mips64r6) {
280fbc802deSJohan Almbladh 			emit(ctx, dsbh, dst, dst);
281fbc802deSJohan Almbladh 			emit(ctx, dshd, dst, dst);
282fbc802deSJohan Almbladh 		} else {
283fbc802deSJohan Almbladh 			u8 t1 = MIPS_R_T6;
284fbc802deSJohan Almbladh 			u8 t2 = MIPS_R_T7;
285fbc802deSJohan Almbladh 
286fbc802deSJohan Almbladh 			emit(ctx, dsll32, t2, dst, 0);  /* t2 = dst << 32    */
287fbc802deSJohan Almbladh 			emit(ctx, dsrl32, dst, dst, 0); /* dst = dst >> 32   */
288fbc802deSJohan Almbladh 			emit(ctx, or, dst, dst, t2);    /* dst = dst | t2    */
289fbc802deSJohan Almbladh 
290fbc802deSJohan Almbladh 			emit(ctx, ori, t2, MIPS_R_ZERO, 0xffff);
291fbc802deSJohan Almbladh 			emit(ctx, dsll32, t1, t2, 0);   /* t1 = t2 << 32     */
292fbc802deSJohan Almbladh 			emit(ctx, or, t1, t1, t2);      /* t1 = t1 | t2      */
293fbc802deSJohan Almbladh 			emit_swap_r64(ctx, dst, t1, 16);/* dst = swap16(dst) */
294fbc802deSJohan Almbladh 
295fbc802deSJohan Almbladh 			emit(ctx, lui, t2, 0xff);       /* t2 = 0x00ff0000   */
296fbc802deSJohan Almbladh 			emit(ctx, ori, t2, t2, 0xff);   /* t2 = t2 | 0x00ff  */
297fbc802deSJohan Almbladh 			emit(ctx, dsll32, t1, t2, 0);   /* t1 = t2 << 32     */
298fbc802deSJohan Almbladh 			emit(ctx, or, t1, t1, t2);      /* t1 = t1 | t2      */
299fbc802deSJohan Almbladh 			emit_swap_r64(ctx, dst, t1, 8); /* dst = swap8(dst)  */
300fbc802deSJohan Almbladh 		}
301fbc802deSJohan Almbladh 		break;
302fbc802deSJohan Almbladh 	/* Swap bytes in a half word */
303fbc802deSJohan Almbladh 	/* Swap bytes in a word */
304fbc802deSJohan Almbladh 	case 32:
305fbc802deSJohan Almbladh 	case 16:
306fbc802deSJohan Almbladh 		emit_sext(ctx, dst, dst);
307fbc802deSJohan Almbladh 		emit_bswap_r(ctx, dst, width);
308fbc802deSJohan Almbladh 		if (cpu_has_mips64r2 || cpu_has_mips64r6)
309fbc802deSJohan Almbladh 			emit_zext(ctx, dst);
310fbc802deSJohan Almbladh 		break;
311fbc802deSJohan Almbladh 	}
312fbc802deSJohan Almbladh 	clobber_reg(ctx, dst);
313fbc802deSJohan Almbladh }
314fbc802deSJohan Almbladh 
315fbc802deSJohan Almbladh /* Truncate a register double word, word or half word */
emit_trunc_r64(struct jit_context * ctx,u8 dst,u32 width)316fbc802deSJohan Almbladh static void emit_trunc_r64(struct jit_context *ctx, u8 dst, u32 width)
317fbc802deSJohan Almbladh {
318fbc802deSJohan Almbladh 	switch (width) {
319fbc802deSJohan Almbladh 	case 64:
320fbc802deSJohan Almbladh 		break;
321fbc802deSJohan Almbladh 	/* Zero-extend a word */
322fbc802deSJohan Almbladh 	case 32:
323fbc802deSJohan Almbladh 		emit_zext(ctx, dst);
324fbc802deSJohan Almbladh 		break;
325fbc802deSJohan Almbladh 	/* Zero-extend a half word */
326fbc802deSJohan Almbladh 	case 16:
327fbc802deSJohan Almbladh 		emit(ctx, andi, dst, dst, 0xffff);
328fbc802deSJohan Almbladh 		break;
329fbc802deSJohan Almbladh 	}
330fbc802deSJohan Almbladh 	clobber_reg(ctx, dst);
331fbc802deSJohan Almbladh }
332fbc802deSJohan Almbladh 
333fbc802deSJohan Almbladh /* Load operation: dst = *(size*)(src + off) */
emit_ldx(struct jit_context * ctx,u8 dst,u8 src,s16 off,u8 size)334fbc802deSJohan Almbladh static void emit_ldx(struct jit_context *ctx, u8 dst, u8 src, s16 off, u8 size)
335fbc802deSJohan Almbladh {
336fbc802deSJohan Almbladh 	switch (size) {
337fbc802deSJohan Almbladh 	/* Load a byte */
338fbc802deSJohan Almbladh 	case BPF_B:
339fbc802deSJohan Almbladh 		emit(ctx, lbu, dst, off, src);
340fbc802deSJohan Almbladh 		break;
341fbc802deSJohan Almbladh 	/* Load a half word */
342fbc802deSJohan Almbladh 	case BPF_H:
343fbc802deSJohan Almbladh 		emit(ctx, lhu, dst, off, src);
344fbc802deSJohan Almbladh 		break;
345fbc802deSJohan Almbladh 	/* Load a word */
346fbc802deSJohan Almbladh 	case BPF_W:
347fbc802deSJohan Almbladh 		emit(ctx, lwu, dst, off, src);
348fbc802deSJohan Almbladh 		break;
349fbc802deSJohan Almbladh 	/* Load a double word */
350fbc802deSJohan Almbladh 	case BPF_DW:
351fbc802deSJohan Almbladh 		emit(ctx, ld, dst, off, src);
352fbc802deSJohan Almbladh 		break;
353fbc802deSJohan Almbladh 	}
354fbc802deSJohan Almbladh 	clobber_reg(ctx, dst);
355fbc802deSJohan Almbladh }
356fbc802deSJohan Almbladh 
357fbc802deSJohan Almbladh /* Store operation: *(size *)(dst + off) = src */
emit_stx(struct jit_context * ctx,u8 dst,u8 src,s16 off,u8 size)358fbc802deSJohan Almbladh static void emit_stx(struct jit_context *ctx, u8 dst, u8 src, s16 off, u8 size)
359fbc802deSJohan Almbladh {
360fbc802deSJohan Almbladh 	switch (size) {
361fbc802deSJohan Almbladh 	/* Store a byte */
362fbc802deSJohan Almbladh 	case BPF_B:
363fbc802deSJohan Almbladh 		emit(ctx, sb, src, off, dst);
364fbc802deSJohan Almbladh 		break;
365fbc802deSJohan Almbladh 	/* Store a half word */
366fbc802deSJohan Almbladh 	case BPF_H:
367fbc802deSJohan Almbladh 		emit(ctx, sh, src, off, dst);
368fbc802deSJohan Almbladh 		break;
369fbc802deSJohan Almbladh 	/* Store a word */
370fbc802deSJohan Almbladh 	case BPF_W:
371fbc802deSJohan Almbladh 		emit(ctx, sw, src, off, dst);
372fbc802deSJohan Almbladh 		break;
373fbc802deSJohan Almbladh 	/* Store a double word */
374fbc802deSJohan Almbladh 	case BPF_DW:
375fbc802deSJohan Almbladh 		emit(ctx, sd, src, off, dst);
376fbc802deSJohan Almbladh 		break;
377fbc802deSJohan Almbladh 	}
378fbc802deSJohan Almbladh }
379fbc802deSJohan Almbladh 
380fbc802deSJohan Almbladh /* Atomic read-modify-write */
emit_atomic_r64(struct jit_context * ctx,u8 dst,u8 src,s16 off,u8 code)381fbc802deSJohan Almbladh static void emit_atomic_r64(struct jit_context *ctx,
382fbc802deSJohan Almbladh 			    u8 dst, u8 src, s16 off, u8 code)
383fbc802deSJohan Almbladh {
384fbc802deSJohan Almbladh 	u8 t1 = MIPS_R_T6;
385fbc802deSJohan Almbladh 	u8 t2 = MIPS_R_T7;
386fbc802deSJohan Almbladh 
38772570224SJohan Almbladh 	LLSC_sync(ctx);
388fbc802deSJohan Almbladh 	emit(ctx, lld, t1, off, dst);
389fbc802deSJohan Almbladh 	switch (code) {
390fbc802deSJohan Almbladh 	case BPF_ADD:
391fbc802deSJohan Almbladh 	case BPF_ADD | BPF_FETCH:
392fbc802deSJohan Almbladh 		emit(ctx, daddu, t2, t1, src);
393fbc802deSJohan Almbladh 		break;
394fbc802deSJohan Almbladh 	case BPF_AND:
395fbc802deSJohan Almbladh 	case BPF_AND | BPF_FETCH:
396fbc802deSJohan Almbladh 		emit(ctx, and, t2, t1, src);
397fbc802deSJohan Almbladh 		break;
398fbc802deSJohan Almbladh 	case BPF_OR:
399fbc802deSJohan Almbladh 	case BPF_OR | BPF_FETCH:
400fbc802deSJohan Almbladh 		emit(ctx, or, t2, t1, src);
401fbc802deSJohan Almbladh 		break;
402fbc802deSJohan Almbladh 	case BPF_XOR:
403fbc802deSJohan Almbladh 	case BPF_XOR | BPF_FETCH:
404fbc802deSJohan Almbladh 		emit(ctx, xor, t2, t1, src);
405fbc802deSJohan Almbladh 		break;
406fbc802deSJohan Almbladh 	case BPF_XCHG:
407fbc802deSJohan Almbladh 		emit(ctx, move, t2, src);
408fbc802deSJohan Almbladh 		break;
409fbc802deSJohan Almbladh 	}
410fbc802deSJohan Almbladh 	emit(ctx, scd, t2, off, dst);
41172570224SJohan Almbladh 	emit(ctx, LLSC_beqz, t2, -16 - LLSC_offset);
412fbc802deSJohan Almbladh 	emit(ctx, nop); /* Delay slot */
413fbc802deSJohan Almbladh 
414fbc802deSJohan Almbladh 	if (code & BPF_FETCH) {
415fbc802deSJohan Almbladh 		emit(ctx, move, src, t1);
416fbc802deSJohan Almbladh 		clobber_reg(ctx, src);
417fbc802deSJohan Almbladh 	}
418fbc802deSJohan Almbladh }
419fbc802deSJohan Almbladh 
420fbc802deSJohan Almbladh /* Atomic compare-and-exchange */
emit_cmpxchg_r64(struct jit_context * ctx,u8 dst,u8 src,s16 off)421fbc802deSJohan Almbladh static void emit_cmpxchg_r64(struct jit_context *ctx, u8 dst, u8 src, s16 off)
422fbc802deSJohan Almbladh {
423fbc802deSJohan Almbladh 	u8 r0 = bpf2mips64[BPF_REG_0];
424fbc802deSJohan Almbladh 	u8 t1 = MIPS_R_T6;
425fbc802deSJohan Almbladh 	u8 t2 = MIPS_R_T7;
426fbc802deSJohan Almbladh 
42772570224SJohan Almbladh 	LLSC_sync(ctx);
428fbc802deSJohan Almbladh 	emit(ctx, lld, t1, off, dst);
429fbc802deSJohan Almbladh 	emit(ctx, bne, t1, r0, 12);
430fbc802deSJohan Almbladh 	emit(ctx, move, t2, src);      /* Delay slot */
431fbc802deSJohan Almbladh 	emit(ctx, scd, t2, off, dst);
43272570224SJohan Almbladh 	emit(ctx, LLSC_beqz, t2, -20 - LLSC_offset);
433fbc802deSJohan Almbladh 	emit(ctx, move, r0, t1);       /* Delay slot */
434fbc802deSJohan Almbladh 
435fbc802deSJohan Almbladh 	clobber_reg(ctx, r0);
436fbc802deSJohan Almbladh }
437fbc802deSJohan Almbladh 
438fbc802deSJohan Almbladh /* Function call */
emit_call(struct jit_context * ctx,const struct bpf_insn * insn)439fbc802deSJohan Almbladh static int emit_call(struct jit_context *ctx, const struct bpf_insn *insn)
440fbc802deSJohan Almbladh {
441fbc802deSJohan Almbladh 	u8 zx = bpf2mips64[JIT_REG_ZX];
442fbc802deSJohan Almbladh 	u8 tmp = MIPS_R_T6;
443fbc802deSJohan Almbladh 	bool fixed;
444fbc802deSJohan Almbladh 	u64 addr;
445fbc802deSJohan Almbladh 
446fbc802deSJohan Almbladh 	/* Decode the call address */
447fbc802deSJohan Almbladh 	if (bpf_jit_get_func_addr(ctx->program, insn, false,
448fbc802deSJohan Almbladh 				  &addr, &fixed) < 0)
449fbc802deSJohan Almbladh 		return -1;
450fbc802deSJohan Almbladh 	if (!fixed)
451fbc802deSJohan Almbladh 		return -1;
452fbc802deSJohan Almbladh 
453fbc802deSJohan Almbladh 	/* Push caller-saved registers on stack */
454fbc802deSJohan Almbladh 	push_regs(ctx, ctx->clobbered & JIT_CALLER_REGS, 0, 0);
455fbc802deSJohan Almbladh 
456fbc802deSJohan Almbladh 	/* Emit function call */
45772570224SJohan Almbladh 	emit_mov_i64(ctx, tmp, addr & JALR_MASK);
458fbc802deSJohan Almbladh 	emit(ctx, jalr, MIPS_R_RA, tmp);
459fbc802deSJohan Almbladh 	emit(ctx, nop); /* Delay slot */
460fbc802deSJohan Almbladh 
461fbc802deSJohan Almbladh 	/* Restore caller-saved registers */
462fbc802deSJohan Almbladh 	pop_regs(ctx, ctx->clobbered & JIT_CALLER_REGS, 0, 0);
463fbc802deSJohan Almbladh 
464fbc802deSJohan Almbladh 	/* Re-initialize the JIT zero-extension register if accessed */
465fbc802deSJohan Almbladh 	if (ctx->accessed & BIT(JIT_REG_ZX)) {
466fbc802deSJohan Almbladh 		emit(ctx, daddiu, zx, MIPS_R_ZERO, -1);
467fbc802deSJohan Almbladh 		emit(ctx, dsrl32, zx, zx, 0);
468fbc802deSJohan Almbladh 	}
469fbc802deSJohan Almbladh 
470fbc802deSJohan Almbladh 	clobber_reg(ctx, MIPS_R_RA);
471fbc802deSJohan Almbladh 	clobber_reg(ctx, MIPS_R_V0);
472fbc802deSJohan Almbladh 	clobber_reg(ctx, MIPS_R_V1);
473fbc802deSJohan Almbladh 	return 0;
474fbc802deSJohan Almbladh }
475fbc802deSJohan Almbladh 
476fbc802deSJohan Almbladh /* Function tail call */
emit_tail_call(struct jit_context * ctx)477fbc802deSJohan Almbladh static int emit_tail_call(struct jit_context *ctx)
478fbc802deSJohan Almbladh {
479fbc802deSJohan Almbladh 	u8 ary = bpf2mips64[BPF_REG_2];
480fbc802deSJohan Almbladh 	u8 ind = bpf2mips64[BPF_REG_3];
481fbc802deSJohan Almbladh 	u8 tcc = bpf2mips64[JIT_REG_TC];
482fbc802deSJohan Almbladh 	u8 tmp = MIPS_R_T6;
483fbc802deSJohan Almbladh 	int off;
484fbc802deSJohan Almbladh 
485fbc802deSJohan Almbladh 	/*
486fbc802deSJohan Almbladh 	 * Tail call:
487fbc802deSJohan Almbladh 	 * eBPF R1 - function argument (context ptr), passed in a0-a1
488fbc802deSJohan Almbladh 	 * eBPF R2 - ptr to object with array of function entry points
489fbc802deSJohan Almbladh 	 * eBPF R3 - array index of function to be called
490fbc802deSJohan Almbladh 	 */
491fbc802deSJohan Almbladh 
492fbc802deSJohan Almbladh 	/* if (ind >= ary->map.max_entries) goto out */
493fbc802deSJohan Almbladh 	off = offsetof(struct bpf_array, map.max_entries);
494fbc802deSJohan Almbladh 	if (off > 0x7fff)
495fbc802deSJohan Almbladh 		return -1;
496fbc802deSJohan Almbladh 	emit(ctx, lwu, tmp, off, ary);            /* tmp = ary->map.max_entrs*/
497fbc802deSJohan Almbladh 	emit(ctx, sltu, tmp, ind, tmp);           /* tmp = ind < t1          */
498fbc802deSJohan Almbladh 	emit(ctx, beqz, tmp, get_offset(ctx, 1)); /* PC += off(1) if tmp == 0*/
499fbc802deSJohan Almbladh 
500fbc802deSJohan Almbladh 	/* if (--TCC < 0) goto out */
501fbc802deSJohan Almbladh 	emit(ctx, daddiu, tcc, tcc, -1);          /* tcc-- (delay slot)      */
502fbc802deSJohan Almbladh 	emit(ctx, bltz, tcc, get_offset(ctx, 1)); /* PC += off(1) if tcc < 0 */
503fbc802deSJohan Almbladh 						  /* (next insn delay slot)  */
504fbc802deSJohan Almbladh 	/* prog = ary->ptrs[ind] */
505fbc802deSJohan Almbladh 	off = offsetof(struct bpf_array, ptrs);
506fbc802deSJohan Almbladh 	if (off > 0x7fff)
507fbc802deSJohan Almbladh 		return -1;
508fbc802deSJohan Almbladh 	emit(ctx, dsll, tmp, ind, 3);             /* tmp = ind << 3          */
509fbc802deSJohan Almbladh 	emit(ctx, daddu, tmp, tmp, ary);          /* tmp += ary              */
510fbc802deSJohan Almbladh 	emit(ctx, ld, tmp, off, tmp);             /* tmp = *(tmp + off)      */
511fbc802deSJohan Almbladh 
512fbc802deSJohan Almbladh 	/* if (prog == 0) goto out */
513fbc802deSJohan Almbladh 	emit(ctx, beqz, tmp, get_offset(ctx, 1)); /* PC += off(1) if tmp == 0*/
514fbc802deSJohan Almbladh 	emit(ctx, nop);                           /* Delay slot              */
515fbc802deSJohan Almbladh 
516fbc802deSJohan Almbladh 	/* func = prog->bpf_func + 8 (prologue skip offset) */
517fbc802deSJohan Almbladh 	off = offsetof(struct bpf_prog, bpf_func);
518fbc802deSJohan Almbladh 	if (off > 0x7fff)
519fbc802deSJohan Almbladh 		return -1;
520fbc802deSJohan Almbladh 	emit(ctx, ld, tmp, off, tmp);                /* tmp = *(tmp + off)   */
521fbc802deSJohan Almbladh 	emit(ctx, daddiu, tmp, tmp, JIT_TCALL_SKIP); /* tmp += skip (4)      */
522fbc802deSJohan Almbladh 
523fbc802deSJohan Almbladh 	/* goto func */
524fbc802deSJohan Almbladh 	build_epilogue(ctx, tmp);
525fbc802deSJohan Almbladh 	access_reg(ctx, JIT_REG_TC);
526fbc802deSJohan Almbladh 	return 0;
527fbc802deSJohan Almbladh }
528fbc802deSJohan Almbladh 
529fbc802deSJohan Almbladh /*
530fbc802deSJohan Almbladh  * Stack frame layout for a JITed program (stack grows down).
531fbc802deSJohan Almbladh  *
532fbc802deSJohan Almbladh  * Higher address  : Previous stack frame      :
533fbc802deSJohan Almbladh  *                 +===========================+  <--- MIPS sp before call
534fbc802deSJohan Almbladh  *                 | Callee-saved registers,   |
535fbc802deSJohan Almbladh  *                 | including RA and FP       |
536fbc802deSJohan Almbladh  *                 +---------------------------+  <--- eBPF FP (MIPS fp)
537fbc802deSJohan Almbladh  *                 | Local eBPF variables      |
538fbc802deSJohan Almbladh  *                 | allocated by program      |
539fbc802deSJohan Almbladh  *                 +---------------------------+
540fbc802deSJohan Almbladh  *                 | Reserved for caller-saved |
541fbc802deSJohan Almbladh  *                 | registers                 |
542fbc802deSJohan Almbladh  * Lower address   +===========================+  <--- MIPS sp
543fbc802deSJohan Almbladh  */
544fbc802deSJohan Almbladh 
545fbc802deSJohan Almbladh /* Build program prologue to set up the stack and registers */
build_prologue(struct jit_context * ctx)546fbc802deSJohan Almbladh void build_prologue(struct jit_context *ctx)
547fbc802deSJohan Almbladh {
548fbc802deSJohan Almbladh 	u8 fp = bpf2mips64[BPF_REG_FP];
549fbc802deSJohan Almbladh 	u8 tc = bpf2mips64[JIT_REG_TC];
550fbc802deSJohan Almbladh 	u8 zx = bpf2mips64[JIT_REG_ZX];
551fbc802deSJohan Almbladh 	int stack, saved, locals, reserved;
552fbc802deSJohan Almbladh 
553fbc802deSJohan Almbladh 	/*
554bbcf0f55STiezhu Yang 	 * In the unlikely event that the TCC limit is raised to more
555bbcf0f55STiezhu Yang 	 * than 16 bits, it is clamped to the maximum value allowed for
556bbcf0f55STiezhu Yang 	 * the generated code (0xffff). It is better fail to compile
557bbcf0f55STiezhu Yang 	 * instead of degrading gracefully.
558bbcf0f55STiezhu Yang 	 */
559bbcf0f55STiezhu Yang 	BUILD_BUG_ON(MAX_TAIL_CALL_CNT > 0xffff);
560bbcf0f55STiezhu Yang 
561bbcf0f55STiezhu Yang 	/*
562fbc802deSJohan Almbladh 	 * The first instruction initializes the tail call count register.
563fbc802deSJohan Almbladh 	 * On a tail call, the calling function jumps into the prologue
564fbc802deSJohan Almbladh 	 * after this instruction.
565fbc802deSJohan Almbladh 	 */
566bbcf0f55STiezhu Yang 	emit(ctx, ori, tc, MIPS_R_ZERO, MAX_TAIL_CALL_CNT);
567fbc802deSJohan Almbladh 
568fbc802deSJohan Almbladh 	/* === Entry-point for tail calls === */
569fbc802deSJohan Almbladh 
570fbc802deSJohan Almbladh 	/*
571fbc802deSJohan Almbladh 	 * If the eBPF frame pointer and tail call count registers were
572fbc802deSJohan Almbladh 	 * accessed they must be preserved. Mark them as clobbered here
573fbc802deSJohan Almbladh 	 * to save and restore them on the stack as needed.
574fbc802deSJohan Almbladh 	 */
575fbc802deSJohan Almbladh 	if (ctx->accessed & BIT(BPF_REG_FP))
576fbc802deSJohan Almbladh 		clobber_reg(ctx, fp);
577fbc802deSJohan Almbladh 	if (ctx->accessed & BIT(JIT_REG_TC))
578fbc802deSJohan Almbladh 		clobber_reg(ctx, tc);
579fbc802deSJohan Almbladh 	if (ctx->accessed & BIT(JIT_REG_ZX))
580fbc802deSJohan Almbladh 		clobber_reg(ctx, zx);
581fbc802deSJohan Almbladh 
582fbc802deSJohan Almbladh 	/* Compute the stack space needed for callee-saved registers */
583fbc802deSJohan Almbladh 	saved = hweight32(ctx->clobbered & JIT_CALLEE_REGS) * sizeof(u64);
584fbc802deSJohan Almbladh 	saved = ALIGN(saved, MIPS_STACK_ALIGNMENT);
585fbc802deSJohan Almbladh 
586fbc802deSJohan Almbladh 	/* Stack space used by eBPF program local data */
587fbc802deSJohan Almbladh 	locals = ALIGN(ctx->program->aux->stack_depth, MIPS_STACK_ALIGNMENT);
588fbc802deSJohan Almbladh 
589fbc802deSJohan Almbladh 	/*
590fbc802deSJohan Almbladh 	 * If we are emitting function calls, reserve extra stack space for
591fbc802deSJohan Almbladh 	 * caller-saved registers needed by the JIT. The required space is
592fbc802deSJohan Almbladh 	 * computed automatically during resource usage discovery (pass 1).
593fbc802deSJohan Almbladh 	 */
594fbc802deSJohan Almbladh 	reserved = ctx->stack_used;
595fbc802deSJohan Almbladh 
596fbc802deSJohan Almbladh 	/* Allocate the stack frame */
597fbc802deSJohan Almbladh 	stack = ALIGN(saved + locals + reserved, MIPS_STACK_ALIGNMENT);
598fbc802deSJohan Almbladh 	if (stack)
599fbc802deSJohan Almbladh 		emit(ctx, daddiu, MIPS_R_SP, MIPS_R_SP, -stack);
600fbc802deSJohan Almbladh 
601fbc802deSJohan Almbladh 	/* Store callee-saved registers on stack */
602fbc802deSJohan Almbladh 	push_regs(ctx, ctx->clobbered & JIT_CALLEE_REGS, 0, stack - saved);
603fbc802deSJohan Almbladh 
604fbc802deSJohan Almbladh 	/* Initialize the eBPF frame pointer if accessed */
605fbc802deSJohan Almbladh 	if (ctx->accessed & BIT(BPF_REG_FP))
606fbc802deSJohan Almbladh 		emit(ctx, daddiu, fp, MIPS_R_SP, stack - saved);
607fbc802deSJohan Almbladh 
608fbc802deSJohan Almbladh 	/* Initialize the ePF JIT zero-extension register if accessed */
609fbc802deSJohan Almbladh 	if (ctx->accessed & BIT(JIT_REG_ZX)) {
610fbc802deSJohan Almbladh 		emit(ctx, daddiu, zx, MIPS_R_ZERO, -1);
611fbc802deSJohan Almbladh 		emit(ctx, dsrl32, zx, zx, 0);
612fbc802deSJohan Almbladh 	}
613fbc802deSJohan Almbladh 
614fbc802deSJohan Almbladh 	ctx->saved_size = saved;
615fbc802deSJohan Almbladh 	ctx->stack_size = stack;
616fbc802deSJohan Almbladh }
617fbc802deSJohan Almbladh 
618fbc802deSJohan Almbladh /* Build the program epilogue to restore the stack and registers */
build_epilogue(struct jit_context * ctx,int dest_reg)619fbc802deSJohan Almbladh void build_epilogue(struct jit_context *ctx, int dest_reg)
620fbc802deSJohan Almbladh {
621fbc802deSJohan Almbladh 	/* Restore callee-saved registers from stack */
622fbc802deSJohan Almbladh 	pop_regs(ctx, ctx->clobbered & JIT_CALLEE_REGS, 0,
623fbc802deSJohan Almbladh 		 ctx->stack_size - ctx->saved_size);
624fbc802deSJohan Almbladh 
625fbc802deSJohan Almbladh 	/* Release the stack frame */
626fbc802deSJohan Almbladh 	if (ctx->stack_size)
627fbc802deSJohan Almbladh 		emit(ctx, daddiu, MIPS_R_SP, MIPS_R_SP, ctx->stack_size);
628fbc802deSJohan Almbladh 
629fbc802deSJohan Almbladh 	/* Jump to return address and sign-extend the 32-bit return value */
630fbc802deSJohan Almbladh 	emit(ctx, jr, dest_reg);
631fbc802deSJohan Almbladh 	emit(ctx, sll, MIPS_R_V0, MIPS_R_V0, 0); /* Delay slot */
632fbc802deSJohan Almbladh }
633fbc802deSJohan Almbladh 
634fbc802deSJohan Almbladh /* Build one eBPF instruction */
build_insn(const struct bpf_insn * insn,struct jit_context * ctx)635fbc802deSJohan Almbladh int build_insn(const struct bpf_insn *insn, struct jit_context *ctx)
636fbc802deSJohan Almbladh {
637fbc802deSJohan Almbladh 	u8 dst = bpf2mips64[insn->dst_reg];
638fbc802deSJohan Almbladh 	u8 src = bpf2mips64[insn->src_reg];
639fbc802deSJohan Almbladh 	u8 res = bpf2mips64[BPF_REG_0];
640fbc802deSJohan Almbladh 	u8 code = insn->code;
641fbc802deSJohan Almbladh 	s16 off = insn->off;
642fbc802deSJohan Almbladh 	s32 imm = insn->imm;
643fbc802deSJohan Almbladh 	s32 val, rel;
644fbc802deSJohan Almbladh 	u8 alu, jmp;
645fbc802deSJohan Almbladh 
646fbc802deSJohan Almbladh 	switch (code) {
647fbc802deSJohan Almbladh 	/* ALU operations */
648fbc802deSJohan Almbladh 	/* dst = imm */
649fbc802deSJohan Almbladh 	case BPF_ALU | BPF_MOV | BPF_K:
650fbc802deSJohan Almbladh 		emit_mov_i(ctx, dst, imm);
651fbc802deSJohan Almbladh 		emit_zext_ver(ctx, dst);
652fbc802deSJohan Almbladh 		break;
653fbc802deSJohan Almbladh 	/* dst = src */
654fbc802deSJohan Almbladh 	case BPF_ALU | BPF_MOV | BPF_X:
655fbc802deSJohan Almbladh 		if (imm == 1) {
656fbc802deSJohan Almbladh 			/* Special mov32 for zext */
657fbc802deSJohan Almbladh 			emit_zext(ctx, dst);
658fbc802deSJohan Almbladh 		} else {
659fbc802deSJohan Almbladh 			emit_mov_r(ctx, dst, src);
660fbc802deSJohan Almbladh 			emit_zext_ver(ctx, dst);
661fbc802deSJohan Almbladh 		}
662fbc802deSJohan Almbladh 		break;
663fbc802deSJohan Almbladh 	/* dst = -dst */
664fbc802deSJohan Almbladh 	case BPF_ALU | BPF_NEG:
665fbc802deSJohan Almbladh 		emit_sext(ctx, dst, dst);
666fbc802deSJohan Almbladh 		emit_alu_i(ctx, dst, 0, BPF_NEG);
667fbc802deSJohan Almbladh 		emit_zext_ver(ctx, dst);
668fbc802deSJohan Almbladh 		break;
669fbc802deSJohan Almbladh 	/* dst = dst & imm */
670fbc802deSJohan Almbladh 	/* dst = dst | imm */
671fbc802deSJohan Almbladh 	/* dst = dst ^ imm */
672fbc802deSJohan Almbladh 	/* dst = dst << imm */
673fbc802deSJohan Almbladh 	case BPF_ALU | BPF_OR | BPF_K:
674fbc802deSJohan Almbladh 	case BPF_ALU | BPF_AND | BPF_K:
675fbc802deSJohan Almbladh 	case BPF_ALU | BPF_XOR | BPF_K:
676fbc802deSJohan Almbladh 	case BPF_ALU | BPF_LSH | BPF_K:
677fbc802deSJohan Almbladh 		if (!valid_alu_i(BPF_OP(code), imm)) {
678fbc802deSJohan Almbladh 			emit_mov_i(ctx, MIPS_R_T4, imm);
679fbc802deSJohan Almbladh 			emit_alu_r(ctx, dst, MIPS_R_T4, BPF_OP(code));
680fbc802deSJohan Almbladh 		} else if (rewrite_alu_i(BPF_OP(code), imm, &alu, &val)) {
681fbc802deSJohan Almbladh 			emit_alu_i(ctx, dst, val, alu);
682fbc802deSJohan Almbladh 		}
683fbc802deSJohan Almbladh 		emit_zext_ver(ctx, dst);
684fbc802deSJohan Almbladh 		break;
685fbc802deSJohan Almbladh 	/* dst = dst >> imm */
686fbc802deSJohan Almbladh 	/* dst = dst >> imm (arithmetic) */
687fbc802deSJohan Almbladh 	/* dst = dst + imm */
688fbc802deSJohan Almbladh 	/* dst = dst - imm */
689fbc802deSJohan Almbladh 	/* dst = dst * imm */
690fbc802deSJohan Almbladh 	/* dst = dst / imm */
691fbc802deSJohan Almbladh 	/* dst = dst % imm */
692fbc802deSJohan Almbladh 	case BPF_ALU | BPF_RSH | BPF_K:
693fbc802deSJohan Almbladh 	case BPF_ALU | BPF_ARSH | BPF_K:
694fbc802deSJohan Almbladh 	case BPF_ALU | BPF_ADD | BPF_K:
695fbc802deSJohan Almbladh 	case BPF_ALU | BPF_SUB | BPF_K:
696fbc802deSJohan Almbladh 	case BPF_ALU | BPF_MUL | BPF_K:
697fbc802deSJohan Almbladh 	case BPF_ALU | BPF_DIV | BPF_K:
698fbc802deSJohan Almbladh 	case BPF_ALU | BPF_MOD | BPF_K:
699fbc802deSJohan Almbladh 		if (!valid_alu_i(BPF_OP(code), imm)) {
700fbc802deSJohan Almbladh 			emit_sext(ctx, dst, dst);
701fbc802deSJohan Almbladh 			emit_mov_i(ctx, MIPS_R_T4, imm);
702fbc802deSJohan Almbladh 			emit_alu_r(ctx, dst, MIPS_R_T4, BPF_OP(code));
703fbc802deSJohan Almbladh 		} else if (rewrite_alu_i(BPF_OP(code), imm, &alu, &val)) {
704fbc802deSJohan Almbladh 			emit_sext(ctx, dst, dst);
705fbc802deSJohan Almbladh 			emit_alu_i(ctx, dst, val, alu);
706fbc802deSJohan Almbladh 		}
707fbc802deSJohan Almbladh 		emit_zext_ver(ctx, dst);
708fbc802deSJohan Almbladh 		break;
709fbc802deSJohan Almbladh 	/* dst = dst & src */
710fbc802deSJohan Almbladh 	/* dst = dst | src */
711fbc802deSJohan Almbladh 	/* dst = dst ^ src */
712fbc802deSJohan Almbladh 	/* dst = dst << src */
713fbc802deSJohan Almbladh 	case BPF_ALU | BPF_AND | BPF_X:
714fbc802deSJohan Almbladh 	case BPF_ALU | BPF_OR | BPF_X:
715fbc802deSJohan Almbladh 	case BPF_ALU | BPF_XOR | BPF_X:
716fbc802deSJohan Almbladh 	case BPF_ALU | BPF_LSH | BPF_X:
717fbc802deSJohan Almbladh 		emit_alu_r(ctx, dst, src, BPF_OP(code));
718fbc802deSJohan Almbladh 		emit_zext_ver(ctx, dst);
719fbc802deSJohan Almbladh 		break;
720fbc802deSJohan Almbladh 	/* dst = dst >> src */
721fbc802deSJohan Almbladh 	/* dst = dst >> src (arithmetic) */
722fbc802deSJohan Almbladh 	/* dst = dst + src */
723fbc802deSJohan Almbladh 	/* dst = dst - src */
724fbc802deSJohan Almbladh 	/* dst = dst * src */
725fbc802deSJohan Almbladh 	/* dst = dst / src */
726fbc802deSJohan Almbladh 	/* dst = dst % src */
727fbc802deSJohan Almbladh 	case BPF_ALU | BPF_RSH | BPF_X:
728fbc802deSJohan Almbladh 	case BPF_ALU | BPF_ARSH | BPF_X:
729fbc802deSJohan Almbladh 	case BPF_ALU | BPF_ADD | BPF_X:
730fbc802deSJohan Almbladh 	case BPF_ALU | BPF_SUB | BPF_X:
731fbc802deSJohan Almbladh 	case BPF_ALU | BPF_MUL | BPF_X:
732fbc802deSJohan Almbladh 	case BPF_ALU | BPF_DIV | BPF_X:
733fbc802deSJohan Almbladh 	case BPF_ALU | BPF_MOD | BPF_X:
734fbc802deSJohan Almbladh 		emit_sext(ctx, dst, dst);
735fbc802deSJohan Almbladh 		emit_sext(ctx, MIPS_R_T4, src);
736fbc802deSJohan Almbladh 		emit_alu_r(ctx, dst, MIPS_R_T4, BPF_OP(code));
737fbc802deSJohan Almbladh 		emit_zext_ver(ctx, dst);
738fbc802deSJohan Almbladh 		break;
739fbc802deSJohan Almbladh 	/* dst = imm (64-bit) */
740fbc802deSJohan Almbladh 	case BPF_ALU64 | BPF_MOV | BPF_K:
741fbc802deSJohan Almbladh 		emit_mov_i(ctx, dst, imm);
742fbc802deSJohan Almbladh 		break;
743fbc802deSJohan Almbladh 	/* dst = src (64-bit) */
744fbc802deSJohan Almbladh 	case BPF_ALU64 | BPF_MOV | BPF_X:
745fbc802deSJohan Almbladh 		emit_mov_r(ctx, dst, src);
746fbc802deSJohan Almbladh 		break;
747fbc802deSJohan Almbladh 	/* dst = -dst (64-bit) */
748fbc802deSJohan Almbladh 	case BPF_ALU64 | BPF_NEG:
749fbc802deSJohan Almbladh 		emit_alu_i64(ctx, dst, 0, BPF_NEG);
750fbc802deSJohan Almbladh 		break;
751fbc802deSJohan Almbladh 	/* dst = dst & imm (64-bit) */
752fbc802deSJohan Almbladh 	/* dst = dst | imm (64-bit) */
753fbc802deSJohan Almbladh 	/* dst = dst ^ imm (64-bit) */
754fbc802deSJohan Almbladh 	/* dst = dst << imm (64-bit) */
755fbc802deSJohan Almbladh 	/* dst = dst >> imm (64-bit) */
756fbc802deSJohan Almbladh 	/* dst = dst >> imm ((64-bit, arithmetic) */
757fbc802deSJohan Almbladh 	/* dst = dst + imm (64-bit) */
758fbc802deSJohan Almbladh 	/* dst = dst - imm (64-bit) */
759fbc802deSJohan Almbladh 	/* dst = dst * imm (64-bit) */
760fbc802deSJohan Almbladh 	/* dst = dst / imm (64-bit) */
761fbc802deSJohan Almbladh 	/* dst = dst % imm (64-bit) */
762fbc802deSJohan Almbladh 	case BPF_ALU64 | BPF_AND | BPF_K:
763fbc802deSJohan Almbladh 	case BPF_ALU64 | BPF_OR | BPF_K:
764fbc802deSJohan Almbladh 	case BPF_ALU64 | BPF_XOR | BPF_K:
765fbc802deSJohan Almbladh 	case BPF_ALU64 | BPF_LSH | BPF_K:
766fbc802deSJohan Almbladh 	case BPF_ALU64 | BPF_RSH | BPF_K:
767fbc802deSJohan Almbladh 	case BPF_ALU64 | BPF_ARSH | BPF_K:
768fbc802deSJohan Almbladh 	case BPF_ALU64 | BPF_ADD | BPF_K:
769fbc802deSJohan Almbladh 	case BPF_ALU64 | BPF_SUB | BPF_K:
770fbc802deSJohan Almbladh 	case BPF_ALU64 | BPF_MUL | BPF_K:
771fbc802deSJohan Almbladh 	case BPF_ALU64 | BPF_DIV | BPF_K:
772fbc802deSJohan Almbladh 	case BPF_ALU64 | BPF_MOD | BPF_K:
773fbc802deSJohan Almbladh 		if (!valid_alu_i(BPF_OP(code), imm)) {
774fbc802deSJohan Almbladh 			emit_mov_i(ctx, MIPS_R_T4, imm);
775fbc802deSJohan Almbladh 			emit_alu_r64(ctx, dst, MIPS_R_T4, BPF_OP(code));
776fbc802deSJohan Almbladh 		} else if (rewrite_alu_i(BPF_OP(code), imm, &alu, &val)) {
777fbc802deSJohan Almbladh 			emit_alu_i64(ctx, dst, val, alu);
778fbc802deSJohan Almbladh 		}
779fbc802deSJohan Almbladh 		break;
780fbc802deSJohan Almbladh 	/* dst = dst & src (64-bit) */
781fbc802deSJohan Almbladh 	/* dst = dst | src (64-bit) */
782fbc802deSJohan Almbladh 	/* dst = dst ^ src (64-bit) */
783fbc802deSJohan Almbladh 	/* dst = dst << src (64-bit) */
784fbc802deSJohan Almbladh 	/* dst = dst >> src (64-bit) */
785fbc802deSJohan Almbladh 	/* dst = dst >> src (64-bit, arithmetic) */
786fbc802deSJohan Almbladh 	/* dst = dst + src (64-bit) */
787fbc802deSJohan Almbladh 	/* dst = dst - src (64-bit) */
788fbc802deSJohan Almbladh 	/* dst = dst * src (64-bit) */
789fbc802deSJohan Almbladh 	/* dst = dst / src (64-bit) */
790fbc802deSJohan Almbladh 	/* dst = dst % src (64-bit) */
791fbc802deSJohan Almbladh 	case BPF_ALU64 | BPF_AND | BPF_X:
792fbc802deSJohan Almbladh 	case BPF_ALU64 | BPF_OR | BPF_X:
793fbc802deSJohan Almbladh 	case BPF_ALU64 | BPF_XOR | BPF_X:
794fbc802deSJohan Almbladh 	case BPF_ALU64 | BPF_LSH | BPF_X:
795fbc802deSJohan Almbladh 	case BPF_ALU64 | BPF_RSH | BPF_X:
796fbc802deSJohan Almbladh 	case BPF_ALU64 | BPF_ARSH | BPF_X:
797fbc802deSJohan Almbladh 	case BPF_ALU64 | BPF_ADD | BPF_X:
798fbc802deSJohan Almbladh 	case BPF_ALU64 | BPF_SUB | BPF_X:
799fbc802deSJohan Almbladh 	case BPF_ALU64 | BPF_MUL | BPF_X:
800fbc802deSJohan Almbladh 	case BPF_ALU64 | BPF_DIV | BPF_X:
801fbc802deSJohan Almbladh 	case BPF_ALU64 | BPF_MOD | BPF_X:
802fbc802deSJohan Almbladh 		emit_alu_r64(ctx, dst, src, BPF_OP(code));
803fbc802deSJohan Almbladh 		break;
804fbc802deSJohan Almbladh 	/* dst = htole(dst) */
805fbc802deSJohan Almbladh 	/* dst = htobe(dst) */
806fbc802deSJohan Almbladh 	case BPF_ALU | BPF_END | BPF_FROM_LE:
807fbc802deSJohan Almbladh 	case BPF_ALU | BPF_END | BPF_FROM_BE:
808fbc802deSJohan Almbladh 		if (BPF_SRC(code) ==
809fbc802deSJohan Almbladh #ifdef __BIG_ENDIAN
810fbc802deSJohan Almbladh 		    BPF_FROM_LE
811fbc802deSJohan Almbladh #else
812fbc802deSJohan Almbladh 		    BPF_FROM_BE
813fbc802deSJohan Almbladh #endif
814fbc802deSJohan Almbladh 		    )
815fbc802deSJohan Almbladh 			emit_bswap_r64(ctx, dst, imm);
816fbc802deSJohan Almbladh 		else
817fbc802deSJohan Almbladh 			emit_trunc_r64(ctx, dst, imm);
818fbc802deSJohan Almbladh 		break;
819fbc802deSJohan Almbladh 	/* dst = imm64 */
820fbc802deSJohan Almbladh 	case BPF_LD | BPF_IMM | BPF_DW:
821fbc802deSJohan Almbladh 		emit_mov_i64(ctx, dst, (u32)imm | ((u64)insn[1].imm << 32));
822fbc802deSJohan Almbladh 		return 1;
823fbc802deSJohan Almbladh 	/* LDX: dst = *(size *)(src + off) */
824fbc802deSJohan Almbladh 	case BPF_LDX | BPF_MEM | BPF_W:
825fbc802deSJohan Almbladh 	case BPF_LDX | BPF_MEM | BPF_H:
826fbc802deSJohan Almbladh 	case BPF_LDX | BPF_MEM | BPF_B:
827fbc802deSJohan Almbladh 	case BPF_LDX | BPF_MEM | BPF_DW:
828fbc802deSJohan Almbladh 		emit_ldx(ctx, dst, src, off, BPF_SIZE(code));
829fbc802deSJohan Almbladh 		break;
830fbc802deSJohan Almbladh 	/* ST: *(size *)(dst + off) = imm */
831fbc802deSJohan Almbladh 	case BPF_ST | BPF_MEM | BPF_W:
832fbc802deSJohan Almbladh 	case BPF_ST | BPF_MEM | BPF_H:
833fbc802deSJohan Almbladh 	case BPF_ST | BPF_MEM | BPF_B:
834fbc802deSJohan Almbladh 	case BPF_ST | BPF_MEM | BPF_DW:
835fbc802deSJohan Almbladh 		emit_mov_i(ctx, MIPS_R_T4, imm);
836fbc802deSJohan Almbladh 		emit_stx(ctx, dst, MIPS_R_T4, off, BPF_SIZE(code));
837fbc802deSJohan Almbladh 		break;
838fbc802deSJohan Almbladh 	/* STX: *(size *)(dst + off) = src */
839fbc802deSJohan Almbladh 	case BPF_STX | BPF_MEM | BPF_W:
840fbc802deSJohan Almbladh 	case BPF_STX | BPF_MEM | BPF_H:
841fbc802deSJohan Almbladh 	case BPF_STX | BPF_MEM | BPF_B:
842fbc802deSJohan Almbladh 	case BPF_STX | BPF_MEM | BPF_DW:
843fbc802deSJohan Almbladh 		emit_stx(ctx, dst, src, off, BPF_SIZE(code));
844fbc802deSJohan Almbladh 		break;
845fbc802deSJohan Almbladh 	/* Speculation barrier */
846fbc802deSJohan Almbladh 	case BPF_ST | BPF_NOSPEC:
847fbc802deSJohan Almbladh 		break;
848fbc802deSJohan Almbladh 	/* Atomics */
849fbc802deSJohan Almbladh 	case BPF_STX | BPF_ATOMIC | BPF_W:
850fbc802deSJohan Almbladh 	case BPF_STX | BPF_ATOMIC | BPF_DW:
851fbc802deSJohan Almbladh 		switch (imm) {
852fbc802deSJohan Almbladh 		case BPF_ADD:
853fbc802deSJohan Almbladh 		case BPF_ADD | BPF_FETCH:
854fbc802deSJohan Almbladh 		case BPF_AND:
855fbc802deSJohan Almbladh 		case BPF_AND | BPF_FETCH:
856fbc802deSJohan Almbladh 		case BPF_OR:
857fbc802deSJohan Almbladh 		case BPF_OR | BPF_FETCH:
858fbc802deSJohan Almbladh 		case BPF_XOR:
859fbc802deSJohan Almbladh 		case BPF_XOR | BPF_FETCH:
860fbc802deSJohan Almbladh 		case BPF_XCHG:
861fbc802deSJohan Almbladh 			if (BPF_SIZE(code) == BPF_DW) {
862fbc802deSJohan Almbladh 				emit_atomic_r64(ctx, dst, src, off, imm);
863fbc802deSJohan Almbladh 			} else if (imm & BPF_FETCH) {
864fbc802deSJohan Almbladh 				u8 tmp = dst;
865fbc802deSJohan Almbladh 
866fbc802deSJohan Almbladh 				if (src == dst) { /* Don't overwrite dst */
867fbc802deSJohan Almbladh 					emit_mov_r(ctx, MIPS_R_T4, dst);
868fbc802deSJohan Almbladh 					tmp = MIPS_R_T4;
869fbc802deSJohan Almbladh 				}
870fbc802deSJohan Almbladh 				emit_sext(ctx, src, src);
871fbc802deSJohan Almbladh 				emit_atomic_r(ctx, tmp, src, off, imm);
872fbc802deSJohan Almbladh 				emit_zext_ver(ctx, src);
873fbc802deSJohan Almbladh 			} else { /* 32-bit, no fetch */
874fbc802deSJohan Almbladh 				emit_sext(ctx, MIPS_R_T4, src);
875fbc802deSJohan Almbladh 				emit_atomic_r(ctx, dst, MIPS_R_T4, off, imm);
876fbc802deSJohan Almbladh 			}
877fbc802deSJohan Almbladh 			break;
878fbc802deSJohan Almbladh 		case BPF_CMPXCHG:
879fbc802deSJohan Almbladh 			if (BPF_SIZE(code) == BPF_DW) {
880fbc802deSJohan Almbladh 				emit_cmpxchg_r64(ctx, dst, src, off);
881fbc802deSJohan Almbladh 			} else {
882fbc802deSJohan Almbladh 				u8 tmp = res;
883fbc802deSJohan Almbladh 
884fbc802deSJohan Almbladh 				if (res == dst)   /* Don't overwrite dst */
885fbc802deSJohan Almbladh 					tmp = MIPS_R_T4;
886fbc802deSJohan Almbladh 				emit_sext(ctx, tmp, res);
887fbc802deSJohan Almbladh 				emit_sext(ctx, MIPS_R_T5, src);
888fbc802deSJohan Almbladh 				emit_cmpxchg_r(ctx, dst, MIPS_R_T5, tmp, off);
889fbc802deSJohan Almbladh 				if (res == dst)   /* Restore result */
890fbc802deSJohan Almbladh 					emit_mov_r(ctx, res, MIPS_R_T4);
891fbc802deSJohan Almbladh 				/* Result zext inserted by verifier */
892fbc802deSJohan Almbladh 			}
893fbc802deSJohan Almbladh 			break;
894fbc802deSJohan Almbladh 		default:
895fbc802deSJohan Almbladh 			goto notyet;
896fbc802deSJohan Almbladh 		}
897fbc802deSJohan Almbladh 		break;
898fbc802deSJohan Almbladh 	/* PC += off if dst == src */
899fbc802deSJohan Almbladh 	/* PC += off if dst != src */
900fbc802deSJohan Almbladh 	/* PC += off if dst & src */
901fbc802deSJohan Almbladh 	/* PC += off if dst > src */
902fbc802deSJohan Almbladh 	/* PC += off if dst >= src */
903fbc802deSJohan Almbladh 	/* PC += off if dst < src */
904fbc802deSJohan Almbladh 	/* PC += off if dst <= src */
905fbc802deSJohan Almbladh 	/* PC += off if dst > src (signed) */
906fbc802deSJohan Almbladh 	/* PC += off if dst >= src (signed) */
907fbc802deSJohan Almbladh 	/* PC += off if dst < src (signed) */
908fbc802deSJohan Almbladh 	/* PC += off if dst <= src (signed) */
909fbc802deSJohan Almbladh 	case BPF_JMP32 | BPF_JEQ | BPF_X:
910fbc802deSJohan Almbladh 	case BPF_JMP32 | BPF_JNE | BPF_X:
911fbc802deSJohan Almbladh 	case BPF_JMP32 | BPF_JSET | BPF_X:
912fbc802deSJohan Almbladh 	case BPF_JMP32 | BPF_JGT | BPF_X:
913fbc802deSJohan Almbladh 	case BPF_JMP32 | BPF_JGE | BPF_X:
914fbc802deSJohan Almbladh 	case BPF_JMP32 | BPF_JLT | BPF_X:
915fbc802deSJohan Almbladh 	case BPF_JMP32 | BPF_JLE | BPF_X:
916fbc802deSJohan Almbladh 	case BPF_JMP32 | BPF_JSGT | BPF_X:
917fbc802deSJohan Almbladh 	case BPF_JMP32 | BPF_JSGE | BPF_X:
918fbc802deSJohan Almbladh 	case BPF_JMP32 | BPF_JSLT | BPF_X:
919fbc802deSJohan Almbladh 	case BPF_JMP32 | BPF_JSLE | BPF_X:
920fbc802deSJohan Almbladh 		if (off == 0)
921fbc802deSJohan Almbladh 			break;
922fbc802deSJohan Almbladh 		setup_jmp_r(ctx, dst == src, BPF_OP(code), off, &jmp, &rel);
923fbc802deSJohan Almbladh 		emit_sext(ctx, MIPS_R_T4, dst); /* Sign-extended dst */
924fbc802deSJohan Almbladh 		emit_sext(ctx, MIPS_R_T5, src); /* Sign-extended src */
925fbc802deSJohan Almbladh 		emit_jmp_r(ctx, MIPS_R_T4, MIPS_R_T5, rel, jmp);
926fbc802deSJohan Almbladh 		if (finish_jmp(ctx, jmp, off) < 0)
927fbc802deSJohan Almbladh 			goto toofar;
928fbc802deSJohan Almbladh 		break;
929fbc802deSJohan Almbladh 	/* PC += off if dst == imm */
930fbc802deSJohan Almbladh 	/* PC += off if dst != imm */
931fbc802deSJohan Almbladh 	/* PC += off if dst & imm */
932fbc802deSJohan Almbladh 	/* PC += off if dst > imm */
933fbc802deSJohan Almbladh 	/* PC += off if dst >= imm */
934fbc802deSJohan Almbladh 	/* PC += off if dst < imm */
935fbc802deSJohan Almbladh 	/* PC += off if dst <= imm */
936fbc802deSJohan Almbladh 	/* PC += off if dst > imm (signed) */
937fbc802deSJohan Almbladh 	/* PC += off if dst >= imm (signed) */
938fbc802deSJohan Almbladh 	/* PC += off if dst < imm (signed) */
939fbc802deSJohan Almbladh 	/* PC += off if dst <= imm (signed) */
940fbc802deSJohan Almbladh 	case BPF_JMP32 | BPF_JEQ | BPF_K:
941fbc802deSJohan Almbladh 	case BPF_JMP32 | BPF_JNE | BPF_K:
942fbc802deSJohan Almbladh 	case BPF_JMP32 | BPF_JSET | BPF_K:
943fbc802deSJohan Almbladh 	case BPF_JMP32 | BPF_JGT | BPF_K:
944fbc802deSJohan Almbladh 	case BPF_JMP32 | BPF_JGE | BPF_K:
945fbc802deSJohan Almbladh 	case BPF_JMP32 | BPF_JLT | BPF_K:
946fbc802deSJohan Almbladh 	case BPF_JMP32 | BPF_JLE | BPF_K:
947fbc802deSJohan Almbladh 	case BPF_JMP32 | BPF_JSGT | BPF_K:
948fbc802deSJohan Almbladh 	case BPF_JMP32 | BPF_JSGE | BPF_K:
949fbc802deSJohan Almbladh 	case BPF_JMP32 | BPF_JSLT | BPF_K:
950fbc802deSJohan Almbladh 	case BPF_JMP32 | BPF_JSLE | BPF_K:
951fbc802deSJohan Almbladh 		if (off == 0)
952fbc802deSJohan Almbladh 			break;
953fbc802deSJohan Almbladh 		setup_jmp_i(ctx, imm, 32, BPF_OP(code), off, &jmp, &rel);
954fbc802deSJohan Almbladh 		emit_sext(ctx, MIPS_R_T4, dst); /* Sign-extended dst */
955fbc802deSJohan Almbladh 		if (valid_jmp_i(jmp, imm)) {
956fbc802deSJohan Almbladh 			emit_jmp_i(ctx, MIPS_R_T4, imm, rel, jmp);
957fbc802deSJohan Almbladh 		} else {
958fbc802deSJohan Almbladh 			/* Move large immediate to register, sign-extended */
959fbc802deSJohan Almbladh 			emit_mov_i(ctx, MIPS_R_T5, imm);
960fbc802deSJohan Almbladh 			emit_jmp_r(ctx, MIPS_R_T4, MIPS_R_T5, rel, jmp);
961fbc802deSJohan Almbladh 		}
962fbc802deSJohan Almbladh 		if (finish_jmp(ctx, jmp, off) < 0)
963fbc802deSJohan Almbladh 			goto toofar;
964fbc802deSJohan Almbladh 		break;
965fbc802deSJohan Almbladh 	/* PC += off if dst == src */
966fbc802deSJohan Almbladh 	/* PC += off if dst != src */
967fbc802deSJohan Almbladh 	/* PC += off if dst & src */
968fbc802deSJohan Almbladh 	/* PC += off if dst > src */
969fbc802deSJohan Almbladh 	/* PC += off if dst >= src */
970fbc802deSJohan Almbladh 	/* PC += off if dst < src */
971fbc802deSJohan Almbladh 	/* PC += off if dst <= src */
972fbc802deSJohan Almbladh 	/* PC += off if dst > src (signed) */
973fbc802deSJohan Almbladh 	/* PC += off if dst >= src (signed) */
974fbc802deSJohan Almbladh 	/* PC += off if dst < src (signed) */
975fbc802deSJohan Almbladh 	/* PC += off if dst <= src (signed) */
976fbc802deSJohan Almbladh 	case BPF_JMP | BPF_JEQ | BPF_X:
977fbc802deSJohan Almbladh 	case BPF_JMP | BPF_JNE | BPF_X:
978fbc802deSJohan Almbladh 	case BPF_JMP | BPF_JSET | BPF_X:
979fbc802deSJohan Almbladh 	case BPF_JMP | BPF_JGT | BPF_X:
980fbc802deSJohan Almbladh 	case BPF_JMP | BPF_JGE | BPF_X:
981fbc802deSJohan Almbladh 	case BPF_JMP | BPF_JLT | BPF_X:
982fbc802deSJohan Almbladh 	case BPF_JMP | BPF_JLE | BPF_X:
983fbc802deSJohan Almbladh 	case BPF_JMP | BPF_JSGT | BPF_X:
984fbc802deSJohan Almbladh 	case BPF_JMP | BPF_JSGE | BPF_X:
985fbc802deSJohan Almbladh 	case BPF_JMP | BPF_JSLT | BPF_X:
986fbc802deSJohan Almbladh 	case BPF_JMP | BPF_JSLE | BPF_X:
987fbc802deSJohan Almbladh 		if (off == 0)
988fbc802deSJohan Almbladh 			break;
989fbc802deSJohan Almbladh 		setup_jmp_r(ctx, dst == src, BPF_OP(code), off, &jmp, &rel);
990fbc802deSJohan Almbladh 		emit_jmp_r(ctx, dst, src, rel, jmp);
991fbc802deSJohan Almbladh 		if (finish_jmp(ctx, jmp, off) < 0)
992fbc802deSJohan Almbladh 			goto toofar;
993fbc802deSJohan Almbladh 		break;
994fbc802deSJohan Almbladh 	/* PC += off if dst == imm */
995fbc802deSJohan Almbladh 	/* PC += off if dst != imm */
996fbc802deSJohan Almbladh 	/* PC += off if dst & imm */
997fbc802deSJohan Almbladh 	/* PC += off if dst > imm */
998fbc802deSJohan Almbladh 	/* PC += off if dst >= imm */
999fbc802deSJohan Almbladh 	/* PC += off if dst < imm */
1000fbc802deSJohan Almbladh 	/* PC += off if dst <= imm */
1001fbc802deSJohan Almbladh 	/* PC += off if dst > imm (signed) */
1002fbc802deSJohan Almbladh 	/* PC += off if dst >= imm (signed) */
1003fbc802deSJohan Almbladh 	/* PC += off if dst < imm (signed) */
1004fbc802deSJohan Almbladh 	/* PC += off if dst <= imm (signed) */
1005fbc802deSJohan Almbladh 	case BPF_JMP | BPF_JEQ | BPF_K:
1006fbc802deSJohan Almbladh 	case BPF_JMP | BPF_JNE | BPF_K:
1007fbc802deSJohan Almbladh 	case BPF_JMP | BPF_JSET | BPF_K:
1008fbc802deSJohan Almbladh 	case BPF_JMP | BPF_JGT | BPF_K:
1009fbc802deSJohan Almbladh 	case BPF_JMP | BPF_JGE | BPF_K:
1010fbc802deSJohan Almbladh 	case BPF_JMP | BPF_JLT | BPF_K:
1011fbc802deSJohan Almbladh 	case BPF_JMP | BPF_JLE | BPF_K:
1012fbc802deSJohan Almbladh 	case BPF_JMP | BPF_JSGT | BPF_K:
1013fbc802deSJohan Almbladh 	case BPF_JMP | BPF_JSGE | BPF_K:
1014fbc802deSJohan Almbladh 	case BPF_JMP | BPF_JSLT | BPF_K:
1015fbc802deSJohan Almbladh 	case BPF_JMP | BPF_JSLE | BPF_K:
1016fbc802deSJohan Almbladh 		if (off == 0)
1017fbc802deSJohan Almbladh 			break;
1018fbc802deSJohan Almbladh 		setup_jmp_i(ctx, imm, 64, BPF_OP(code), off, &jmp, &rel);
1019fbc802deSJohan Almbladh 		if (valid_jmp_i(jmp, imm)) {
1020fbc802deSJohan Almbladh 			emit_jmp_i(ctx, dst, imm, rel, jmp);
1021fbc802deSJohan Almbladh 		} else {
1022fbc802deSJohan Almbladh 			/* Move large immediate to register */
1023fbc802deSJohan Almbladh 			emit_mov_i(ctx, MIPS_R_T4, imm);
1024fbc802deSJohan Almbladh 			emit_jmp_r(ctx, dst, MIPS_R_T4, rel, jmp);
1025fbc802deSJohan Almbladh 		}
1026fbc802deSJohan Almbladh 		if (finish_jmp(ctx, jmp, off) < 0)
1027fbc802deSJohan Almbladh 			goto toofar;
1028fbc802deSJohan Almbladh 		break;
1029fbc802deSJohan Almbladh 	/* PC += off */
1030fbc802deSJohan Almbladh 	case BPF_JMP | BPF_JA:
1031fbc802deSJohan Almbladh 		if (off == 0)
1032fbc802deSJohan Almbladh 			break;
1033fbc802deSJohan Almbladh 		if (emit_ja(ctx, off) < 0)
1034fbc802deSJohan Almbladh 			goto toofar;
1035fbc802deSJohan Almbladh 		break;
1036fbc802deSJohan Almbladh 	/* Tail call */
1037fbc802deSJohan Almbladh 	case BPF_JMP | BPF_TAIL_CALL:
1038fbc802deSJohan Almbladh 		if (emit_tail_call(ctx) < 0)
1039fbc802deSJohan Almbladh 			goto invalid;
1040fbc802deSJohan Almbladh 		break;
1041fbc802deSJohan Almbladh 	/* Function call */
1042fbc802deSJohan Almbladh 	case BPF_JMP | BPF_CALL:
1043fbc802deSJohan Almbladh 		if (emit_call(ctx, insn) < 0)
1044fbc802deSJohan Almbladh 			goto invalid;
1045fbc802deSJohan Almbladh 		break;
1046fbc802deSJohan Almbladh 	/* Function return */
1047fbc802deSJohan Almbladh 	case BPF_JMP | BPF_EXIT:
1048fbc802deSJohan Almbladh 		/*
1049fbc802deSJohan Almbladh 		 * Optimization: when last instruction is EXIT
1050fbc802deSJohan Almbladh 		 * simply continue to epilogue.
1051fbc802deSJohan Almbladh 		 */
1052fbc802deSJohan Almbladh 		if (ctx->bpf_index == ctx->program->len - 1)
1053fbc802deSJohan Almbladh 			break;
1054fbc802deSJohan Almbladh 		if (emit_exit(ctx) < 0)
1055fbc802deSJohan Almbladh 			goto toofar;
1056fbc802deSJohan Almbladh 		break;
1057fbc802deSJohan Almbladh 
1058fbc802deSJohan Almbladh 	default:
1059fbc802deSJohan Almbladh invalid:
1060fbc802deSJohan Almbladh 		pr_err_once("unknown opcode %02x\n", code);
1061fbc802deSJohan Almbladh 		return -EINVAL;
1062fbc802deSJohan Almbladh notyet:
1063fbc802deSJohan Almbladh 		pr_info_once("*** NOT YET: opcode %02x ***\n", code);
1064fbc802deSJohan Almbladh 		return -EFAULT;
1065fbc802deSJohan Almbladh toofar:
1066fbc802deSJohan Almbladh 		pr_info_once("*** TOO FAR: jump at %u opcode %02x ***\n",
1067fbc802deSJohan Almbladh 			     ctx->bpf_index, code);
1068fbc802deSJohan Almbladh 		return -E2BIG;
1069fbc802deSJohan Almbladh 	}
1070fbc802deSJohan Almbladh 	return 0;
1071fbc802deSJohan Almbladh }
1072