diff options
Diffstat (limited to 'tools/objtool/arch/loongarch/decode.c')
| -rw-r--r-- | tools/objtool/arch/loongarch/decode.c | 33 | 
1 files changed, 30 insertions, 3 deletions
| diff --git a/tools/objtool/arch/loongarch/decode.c b/tools/objtool/arch/loongarch/decode.c index b6fdc68053cc..2e555c4060c5 100644 --- a/tools/objtool/arch/loongarch/decode.c +++ b/tools/objtool/arch/loongarch/decode.c @@ -278,6 +278,25 @@ static bool decode_insn_reg2i16_fomat(union loongarch_instruction inst,  	return true;  } +static bool decode_insn_reg3_fomat(union loongarch_instruction inst, +				   struct instruction *insn) +{ +	switch (inst.reg3_format.opcode) { +	case amswapw_op: +		if (inst.reg3_format.rd == LOONGARCH_GPR_ZERO && +		    inst.reg3_format.rk == LOONGARCH_GPR_RA && +		    inst.reg3_format.rj == LOONGARCH_GPR_ZERO) { +			/* amswap.w $zero, $ra, $zero */ +			insn->type = INSN_BUG; +		} +		break; +	default: +		return false; +	} + +	return true; +} +  int arch_decode_instruction(struct objtool_file *file, const struct section *sec,  			    unsigned long offset, unsigned int maxlen,  			    struct instruction *insn) @@ -309,11 +328,19 @@ int arch_decode_instruction(struct objtool_file *file, const struct section *sec  		return 0;  	if (decode_insn_reg2i16_fomat(inst, insn))  		return 0; +	if (decode_insn_reg3_fomat(inst, insn)) +		return 0; -	if (inst.word == 0) +	if (inst.word == 0) { +		/* andi $zero, $zero, 0x0 */  		insn->type = INSN_NOP; -	else if (inst.reg0i15_format.opcode == break_op) { -		/* break */ +	} else if (inst.reg0i15_format.opcode == break_op && +		   inst.reg0i15_format.immediate == 0x0) { +		/* break 0x0 */ +		insn->type = INSN_TRAP; +	} else if (inst.reg0i15_format.opcode == break_op && +		   inst.reg0i15_format.immediate == 0x1) { +		/* break 0x1 */  		insn->type = INSN_BUG;  	} else if (inst.reg2_format.opcode == ertn_op) {  		/* ertn */ | 
