target-tricore: Add instructions of SRC opcode format
Add instructions of SRC opcode format. Add micro-op generator functions for add, conditional add/sub and shi/shai. Signed-off-by: Bastian Koppelmann <kbastian@mail.uni-paderborn.de> Reviewed-by: Richard Henderson <rth@twiddle.net> Message-id: 1409572800-4116-7-git-send-email-kbastian@mail.uni-paderborn.de Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
This commit is contained in:
parent
7c87d074d2
commit
0707ec1bea
@ -0,0 +1,16 @@
|
||||
/*
|
||||
* Copyright (c) 2012-2014 Bastian Koppelmann C-Lab/University Paderborn
|
||||
*
|
||||
* This library is free software; you can redistribute it and/or
|
||||
* modify it under the terms of the GNU Lesser General Public
|
||||
* License as published by the Free Software Foundation; either
|
||||
* version 2 of the License, or (at your option) any later version.
|
||||
*
|
||||
* This library is distributed in the hope that it will be useful,
|
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||
* Lesser General Public License for more details.
|
||||
*
|
||||
* You should have received a copy of the GNU Lesser General Public
|
||||
* License along with this library; if not, see <http://www.gnu.org/licenses/>.
|
||||
*/
|
@ -27,6 +27,7 @@
|
||||
#include "exec/helper-gen.h"
|
||||
|
||||
#include "tricore-opcodes.h"
|
||||
|
||||
/*
|
||||
* TCG registers
|
||||
*/
|
||||
@ -102,8 +103,258 @@ void tricore_cpu_dump_state(CPUState *cs, FILE *f,
|
||||
|
||||
}
|
||||
|
||||
/*
|
||||
* Functions to generate micro-ops
|
||||
*/
|
||||
|
||||
/* Functions for arithmetic instructions */
|
||||
|
||||
static inline void gen_add_d(TCGv ret, TCGv r1, TCGv r2)
|
||||
{
|
||||
TCGv t0 = tcg_temp_new_i32();
|
||||
TCGv result = tcg_temp_new_i32();
|
||||
/* Addition and set V/SV bits */
|
||||
tcg_gen_add_tl(result, r1, r2);
|
||||
/* calc V bit */
|
||||
tcg_gen_xor_tl(cpu_PSW_V, result, r1);
|
||||
tcg_gen_xor_tl(t0, r1, r2);
|
||||
tcg_gen_andc_tl(cpu_PSW_V, cpu_PSW_V, t0);
|
||||
/* Calc SV bit */
|
||||
tcg_gen_or_tl(cpu_PSW_SV, cpu_PSW_SV, cpu_PSW_V);
|
||||
/* Calc AV/SAV bits */
|
||||
tcg_gen_add_tl(cpu_PSW_AV, result, result);
|
||||
tcg_gen_xor_tl(cpu_PSW_AV, result, cpu_PSW_AV);
|
||||
/* calc SAV */
|
||||
tcg_gen_or_tl(cpu_PSW_SAV, cpu_PSW_SAV, cpu_PSW_AV);
|
||||
/* write back result */
|
||||
tcg_gen_mov_tl(ret, result);
|
||||
|
||||
tcg_temp_free(result);
|
||||
tcg_temp_free(t0);
|
||||
}
|
||||
|
||||
static inline void gen_addi_d(TCGv ret, TCGv r1, target_ulong r2)
|
||||
{
|
||||
TCGv temp = tcg_const_i32(r2);
|
||||
gen_add_d(ret, r1, temp);
|
||||
tcg_temp_free(temp);
|
||||
}
|
||||
|
||||
static inline void gen_cond_add(TCGCond cond, TCGv r1, TCGv r2, TCGv r3,
|
||||
TCGv r4)
|
||||
{
|
||||
TCGv temp = tcg_temp_new();
|
||||
TCGv temp2 = tcg_temp_new();
|
||||
TCGv result = tcg_temp_new();
|
||||
TCGv mask = tcg_temp_new();
|
||||
TCGv t0 = tcg_const_i32(0);
|
||||
|
||||
/* create mask for sticky bits */
|
||||
tcg_gen_setcond_tl(cond, mask, r4, t0);
|
||||
tcg_gen_shli_tl(mask, mask, 31);
|
||||
|
||||
tcg_gen_add_tl(result, r1, r2);
|
||||
/* Calc PSW_V */
|
||||
tcg_gen_xor_tl(temp, result, r1);
|
||||
tcg_gen_xor_tl(temp2, r1, r2);
|
||||
tcg_gen_andc_tl(temp, temp, temp2);
|
||||
tcg_gen_movcond_tl(cond, cpu_PSW_V, r4, t0, temp, cpu_PSW_V);
|
||||
/* Set PSW_SV */
|
||||
tcg_gen_and_tl(temp, temp, mask);
|
||||
tcg_gen_or_tl(cpu_PSW_SV, temp, cpu_PSW_SV);
|
||||
/* calc AV bit */
|
||||
tcg_gen_add_tl(temp, result, result);
|
||||
tcg_gen_xor_tl(temp, temp, result);
|
||||
tcg_gen_movcond_tl(cond, cpu_PSW_AV, r4, t0, temp, cpu_PSW_AV);
|
||||
/* calc SAV bit */
|
||||
tcg_gen_and_tl(temp, temp, mask);
|
||||
tcg_gen_or_tl(cpu_PSW_SAV, temp, cpu_PSW_SAV);
|
||||
/* write back result */
|
||||
tcg_gen_movcond_tl(cond, r3, r4, t0, result, r3);
|
||||
|
||||
tcg_temp_free(t0);
|
||||
tcg_temp_free(temp);
|
||||
tcg_temp_free(temp2);
|
||||
tcg_temp_free(result);
|
||||
tcg_temp_free(mask);
|
||||
}
|
||||
|
||||
static inline void gen_condi_add(TCGCond cond, TCGv r1, int32_t r2,
|
||||
TCGv r3, TCGv r4)
|
||||
{
|
||||
TCGv temp = tcg_const_i32(r2);
|
||||
gen_cond_add(cond, r1, temp, r3, r4);
|
||||
tcg_temp_free(temp);
|
||||
}
|
||||
|
||||
static void gen_shi(TCGv ret, TCGv r1, int32_t shift_count)
|
||||
{
|
||||
if (shift_count == -32) {
|
||||
tcg_gen_movi_tl(ret, 0);
|
||||
} else if (shift_count >= 0) {
|
||||
tcg_gen_shli_tl(ret, r1, shift_count);
|
||||
} else {
|
||||
tcg_gen_shri_tl(ret, r1, -shift_count);
|
||||
}
|
||||
}
|
||||
|
||||
static void gen_shaci(TCGv ret, TCGv r1, int32_t shift_count)
|
||||
{
|
||||
uint32_t msk, msk_start;
|
||||
TCGv temp = tcg_temp_new();
|
||||
TCGv temp2 = tcg_temp_new();
|
||||
TCGv t_0 = tcg_const_i32(0);
|
||||
|
||||
if (shift_count == 0) {
|
||||
/* Clear PSW.C and PSW.V */
|
||||
tcg_gen_movi_tl(cpu_PSW_C, 0);
|
||||
tcg_gen_mov_tl(cpu_PSW_V, cpu_PSW_C);
|
||||
tcg_gen_mov_tl(ret, r1);
|
||||
} else if (shift_count == -32) {
|
||||
/* set PSW.C */
|
||||
tcg_gen_mov_tl(cpu_PSW_C, r1);
|
||||
/* fill ret completly with sign bit */
|
||||
tcg_gen_sari_tl(ret, r1, 31);
|
||||
/* clear PSW.V */
|
||||
tcg_gen_movi_tl(cpu_PSW_V, 0);
|
||||
} else if (shift_count > 0) {
|
||||
TCGv t_max = tcg_const_i32(0x7FFFFFFF >> shift_count);
|
||||
TCGv t_min = tcg_const_i32(((int32_t) -0x80000000) >> shift_count);
|
||||
|
||||
/* calc carry */
|
||||
msk_start = 32 - shift_count;
|
||||
msk = ((1 << shift_count) - 1) << msk_start;
|
||||
tcg_gen_andi_tl(cpu_PSW_C, r1, msk);
|
||||
/* calc v/sv bits */
|
||||
tcg_gen_setcond_tl(TCG_COND_GT, temp, r1, t_max);
|
||||
tcg_gen_setcond_tl(TCG_COND_LT, temp2, r1, t_min);
|
||||
tcg_gen_or_tl(cpu_PSW_V, temp, temp2);
|
||||
tcg_gen_shli_tl(cpu_PSW_V, cpu_PSW_V, 31);
|
||||
/* calc sv */
|
||||
tcg_gen_or_tl(cpu_PSW_SV, cpu_PSW_V, cpu_PSW_SV);
|
||||
/* do shift */
|
||||
tcg_gen_shli_tl(ret, r1, shift_count);
|
||||
|
||||
tcg_temp_free(t_max);
|
||||
tcg_temp_free(t_min);
|
||||
} else {
|
||||
/* clear PSW.V */
|
||||
tcg_gen_movi_tl(cpu_PSW_V, 0);
|
||||
/* calc carry */
|
||||
msk = (1 << -shift_count) - 1;
|
||||
tcg_gen_andi_tl(cpu_PSW_C, r1, msk);
|
||||
/* do shift */
|
||||
tcg_gen_sari_tl(ret, r1, -shift_count);
|
||||
}
|
||||
/* calc av overflow bit */
|
||||
tcg_gen_add_tl(cpu_PSW_AV, ret, ret);
|
||||
tcg_gen_xor_tl(cpu_PSW_AV, ret, cpu_PSW_AV);
|
||||
/* calc sav overflow bit */
|
||||
tcg_gen_or_tl(cpu_PSW_SAV, cpu_PSW_SAV, cpu_PSW_AV);
|
||||
|
||||
tcg_temp_free(temp);
|
||||
tcg_temp_free(temp2);
|
||||
tcg_temp_free(t_0);
|
||||
}
|
||||
|
||||
/*
|
||||
* Functions for decoding instructions
|
||||
*/
|
||||
|
||||
static void decode_src_opc(DisasContext *ctx, int op1)
|
||||
{
|
||||
int r1;
|
||||
int32_t const4;
|
||||
TCGv temp, temp2;
|
||||
|
||||
r1 = MASK_OP_SRC_S1D(ctx->opcode);
|
||||
const4 = MASK_OP_SRC_CONST4_SEXT(ctx->opcode);
|
||||
|
||||
switch (op1) {
|
||||
case OPC1_16_SRC_ADD:
|
||||
gen_addi_d(cpu_gpr_d[r1], cpu_gpr_d[r1], const4);
|
||||
break;
|
||||
case OPC1_16_SRC_ADD_A15:
|
||||
gen_addi_d(cpu_gpr_d[r1], cpu_gpr_d[15], const4);
|
||||
break;
|
||||
case OPC1_16_SRC_ADD_15A:
|
||||
gen_addi_d(cpu_gpr_d[15], cpu_gpr_d[r1], const4);
|
||||
break;
|
||||
case OPC1_16_SRC_ADD_A:
|
||||
tcg_gen_addi_tl(cpu_gpr_a[r1], cpu_gpr_a[r1], const4);
|
||||
break;
|
||||
case OPC1_16_SRC_CADD:
|
||||
gen_condi_add(TCG_COND_NE, cpu_gpr_d[r1], const4, cpu_gpr_d[r1],
|
||||
cpu_gpr_d[15]);
|
||||
break;
|
||||
case OPC1_16_SRC_CADDN:
|
||||
gen_condi_add(TCG_COND_EQ, cpu_gpr_d[r1], const4, cpu_gpr_d[r1],
|
||||
cpu_gpr_d[15]);
|
||||
break;
|
||||
case OPC1_16_SRC_CMOV:
|
||||
temp = tcg_const_tl(0);
|
||||
temp2 = tcg_const_tl(const4);
|
||||
tcg_gen_movcond_tl(TCG_COND_NE, cpu_gpr_d[r1], cpu_gpr_d[15], temp,
|
||||
temp2, cpu_gpr_d[r1]);
|
||||
tcg_temp_free(temp);
|
||||
tcg_temp_free(temp2);
|
||||
break;
|
||||
case OPC1_16_SRC_CMOVN:
|
||||
temp = tcg_const_tl(0);
|
||||
temp2 = tcg_const_tl(const4);
|
||||
tcg_gen_movcond_tl(TCG_COND_EQ, cpu_gpr_d[r1], cpu_gpr_d[15], temp,
|
||||
temp2, cpu_gpr_d[r1]);
|
||||
tcg_temp_free(temp);
|
||||
tcg_temp_free(temp2);
|
||||
break;
|
||||
case OPC1_16_SRC_EQ:
|
||||
tcg_gen_setcondi_tl(TCG_COND_EQ, cpu_gpr_d[15], cpu_gpr_d[r1],
|
||||
const4);
|
||||
break;
|
||||
case OPC1_16_SRC_LT:
|
||||
tcg_gen_setcondi_tl(TCG_COND_LT, cpu_gpr_d[15], cpu_gpr_d[r1],
|
||||
const4);
|
||||
break;
|
||||
case OPC1_16_SRC_MOV:
|
||||
tcg_gen_movi_tl(cpu_gpr_d[r1], const4);
|
||||
break;
|
||||
case OPC1_16_SRC_MOV_A:
|
||||
const4 = MASK_OP_SRC_CONST4(ctx->opcode);
|
||||
tcg_gen_movi_tl(cpu_gpr_a[r1], const4);
|
||||
break;
|
||||
case OPC1_16_SRC_SH:
|
||||
gen_shi(cpu_gpr_d[r1], cpu_gpr_d[r1], const4);
|
||||
break;
|
||||
case OPC1_16_SRC_SHA:
|
||||
gen_shaci(cpu_gpr_d[r1], cpu_gpr_d[r1], const4);
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
static void decode_16Bit_opc(CPUTriCoreState *env, DisasContext *ctx)
|
||||
{
|
||||
int op1;
|
||||
|
||||
op1 = MASK_OP_MAJOR(ctx->opcode);
|
||||
|
||||
switch (op1) {
|
||||
case OPC1_16_SRC_ADD:
|
||||
case OPC1_16_SRC_ADD_A15:
|
||||
case OPC1_16_SRC_ADD_15A:
|
||||
case OPC1_16_SRC_ADD_A:
|
||||
case OPC1_16_SRC_CADD:
|
||||
case OPC1_16_SRC_CADDN:
|
||||
case OPC1_16_SRC_CMOV:
|
||||
case OPC1_16_SRC_CMOVN:
|
||||
case OPC1_16_SRC_EQ:
|
||||
case OPC1_16_SRC_LT:
|
||||
case OPC1_16_SRC_MOV:
|
||||
case OPC1_16_SRC_MOV_A:
|
||||
case OPC1_16_SRC_SH:
|
||||
case OPC1_16_SRC_SHA:
|
||||
decode_src_opc(ctx, op1);
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
static void decode_32Bit_opc(CPUTriCoreState *env, DisasContext *ctx)
|
||||
|
Loading…
x
Reference in New Issue
Block a user