2016-10-30 06:14:56 +03:00
|
|
|
/*
|
2020-10-09 09:44:37 +03:00
|
|
|
* PowerPC internal definitions for qemu.
|
2016-10-30 06:14:56 +03:00
|
|
|
*
|
|
|
|
* 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
|
2020-10-19 09:11:26 +03:00
|
|
|
* version 2.1 of the License, or (at your option) any later version.
|
2016-10-30 06:14:56 +03:00
|
|
|
*
|
|
|
|
* 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/>.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef PPC_INTERNAL_H
|
|
|
|
#define PPC_INTERNAL_H
|
|
|
|
|
|
|
|
#define FUNC_MASK(name, ret_type, size, max_val) \
|
|
|
|
static inline ret_type name(uint##size##_t start, \
|
|
|
|
uint##size##_t end) \
|
|
|
|
{ \
|
|
|
|
ret_type ret, max_bit = size - 1; \
|
|
|
|
\
|
|
|
|
if (likely(start == 0)) { \
|
|
|
|
ret = max_val << (max_bit - end); \
|
|
|
|
} else if (likely(end == max_bit)) { \
|
|
|
|
ret = max_val >> start; \
|
|
|
|
} else { \
|
|
|
|
ret = (((uint##size##_t)(-1ULL)) >> (start)) ^ \
|
|
|
|
(((uint##size##_t)(-1ULL) >> (end)) >> 1); \
|
|
|
|
if (unlikely(start > end)) { \
|
|
|
|
return ~ret; \
|
|
|
|
} \
|
|
|
|
} \
|
|
|
|
\
|
|
|
|
return ret; \
|
|
|
|
}
|
|
|
|
|
|
|
|
#if defined(TARGET_PPC64)
|
|
|
|
FUNC_MASK(MASK, target_ulong, 64, UINT64_MAX);
|
|
|
|
#else
|
|
|
|
FUNC_MASK(MASK, target_ulong, 32, UINT32_MAX);
|
|
|
|
#endif
|
|
|
|
FUNC_MASK(mask_u32, uint32_t, 32, UINT32_MAX);
|
|
|
|
FUNC_MASK(mask_u64, uint64_t, 64, UINT64_MAX);
|
|
|
|
|
2016-11-23 14:37:10 +03:00
|
|
|
/*****************************************************************************/
|
|
|
|
/*** Instruction decoding ***/
|
|
|
|
#define EXTRACT_HELPER(name, shift, nb) \
|
|
|
|
static inline uint32_t name(uint32_t opcode) \
|
|
|
|
{ \
|
2019-01-02 12:14:16 +03:00
|
|
|
return extract32(opcode, shift, nb); \
|
2016-11-23 14:37:10 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
#define EXTRACT_SHELPER(name, shift, nb) \
|
|
|
|
static inline int32_t name(uint32_t opcode) \
|
|
|
|
{ \
|
2019-01-02 12:14:16 +03:00
|
|
|
return sextract32(opcode, shift, nb); \
|
2016-11-23 14:37:10 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
#define EXTRACT_HELPER_SPLIT(name, shift1, nb1, shift2, nb2) \
|
|
|
|
static inline uint32_t name(uint32_t opcode) \
|
|
|
|
{ \
|
2019-01-02 12:14:16 +03:00
|
|
|
return extract32(opcode, shift1, nb1) << nb2 | \
|
|
|
|
extract32(opcode, shift2, nb2); \
|
2016-11-23 14:37:10 +03:00
|
|
|
}
|
|
|
|
|
2017-01-13 12:23:39 +03:00
|
|
|
#define EXTRACT_HELPER_SPLIT_3(name, \
|
2016-11-23 14:37:10 +03:00
|
|
|
d0_bits, shift_op_d0, shift_d0, \
|
|
|
|
d1_bits, shift_op_d1, shift_d1, \
|
|
|
|
d2_bits, shift_op_d2, shift_d2) \
|
|
|
|
static inline int16_t name(uint32_t opcode) \
|
|
|
|
{ \
|
|
|
|
return \
|
|
|
|
(((opcode >> (shift_op_d0)) & ((1 << (d0_bits)) - 1)) << (shift_d0)) | \
|
|
|
|
(((opcode >> (shift_op_d1)) & ((1 << (d1_bits)) - 1)) << (shift_d1)) | \
|
|
|
|
(((opcode >> (shift_op_d2)) & ((1 << (d2_bits)) - 1)) << (shift_d2)); \
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* Opcode part 1 */
|
|
|
|
EXTRACT_HELPER(opc1, 26, 6);
|
|
|
|
/* Opcode part 2 */
|
|
|
|
EXTRACT_HELPER(opc2, 1, 5);
|
|
|
|
/* Opcode part 3 */
|
|
|
|
EXTRACT_HELPER(opc3, 6, 5);
|
|
|
|
/* Opcode part 4 */
|
|
|
|
EXTRACT_HELPER(opc4, 16, 5);
|
|
|
|
/* Update Cr0 flags */
|
|
|
|
EXTRACT_HELPER(Rc, 0, 1);
|
|
|
|
/* Update Cr6 flags (Altivec) */
|
|
|
|
EXTRACT_HELPER(Rc21, 10, 1);
|
|
|
|
/* Destination */
|
|
|
|
EXTRACT_HELPER(rD, 21, 5);
|
|
|
|
/* Source */
|
|
|
|
EXTRACT_HELPER(rS, 21, 5);
|
|
|
|
/* First operand */
|
|
|
|
EXTRACT_HELPER(rA, 16, 5);
|
|
|
|
/* Second operand */
|
|
|
|
EXTRACT_HELPER(rB, 11, 5);
|
|
|
|
/* Third operand */
|
|
|
|
EXTRACT_HELPER(rC, 6, 5);
|
|
|
|
/*** Get CRn ***/
|
|
|
|
EXTRACT_HELPER(crfD, 23, 3);
|
|
|
|
EXTRACT_HELPER(BF, 23, 3);
|
|
|
|
EXTRACT_HELPER(crfS, 18, 3);
|
|
|
|
EXTRACT_HELPER(crbD, 21, 5);
|
|
|
|
EXTRACT_HELPER(crbA, 16, 5);
|
|
|
|
EXTRACT_HELPER(crbB, 11, 5);
|
|
|
|
/* SPR / TBL */
|
|
|
|
EXTRACT_HELPER(_SPR, 11, 10);
|
|
|
|
static inline uint32_t SPR(uint32_t opcode)
|
|
|
|
{
|
|
|
|
uint32_t sprn = _SPR(opcode);
|
|
|
|
|
|
|
|
return ((sprn >> 5) & 0x1F) | ((sprn & 0x1F) << 5);
|
|
|
|
}
|
|
|
|
/*** Get constants ***/
|
|
|
|
/* 16 bits signed immediate value */
|
|
|
|
EXTRACT_SHELPER(SIMM, 0, 16);
|
|
|
|
/* 16 bits unsigned immediate value */
|
|
|
|
EXTRACT_HELPER(UIMM, 0, 16);
|
|
|
|
/* 5 bits signed immediate value */
|
2019-01-02 12:14:15 +03:00
|
|
|
EXTRACT_SHELPER(SIMM5, 16, 5);
|
2016-11-23 14:37:10 +03:00
|
|
|
/* 5 bits signed immediate value */
|
|
|
|
EXTRACT_HELPER(UIMM5, 16, 5);
|
|
|
|
/* 4 bits unsigned immediate value */
|
|
|
|
EXTRACT_HELPER(UIMM4, 16, 4);
|
|
|
|
/* Bit count */
|
|
|
|
EXTRACT_HELPER(NB, 11, 5);
|
|
|
|
/* Shift count */
|
|
|
|
EXTRACT_HELPER(SH, 11, 5);
|
2017-02-09 08:33:30 +03:00
|
|
|
/* lwat/stwat/ldat/lwat */
|
|
|
|
EXTRACT_HELPER(FC, 11, 5);
|
2016-11-23 14:37:10 +03:00
|
|
|
/* Vector shift count */
|
|
|
|
EXTRACT_HELPER(VSH, 6, 4);
|
|
|
|
/* Mask start */
|
|
|
|
EXTRACT_HELPER(MB, 6, 5);
|
|
|
|
/* Mask end */
|
|
|
|
EXTRACT_HELPER(ME, 1, 5);
|
|
|
|
/* Trap operand */
|
|
|
|
EXTRACT_HELPER(TO, 21, 5);
|
|
|
|
|
|
|
|
EXTRACT_HELPER(CRM, 12, 8);
|
|
|
|
|
|
|
|
#ifndef CONFIG_USER_ONLY
|
|
|
|
EXTRACT_HELPER(SR, 16, 4);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* mtfsf/mtfsfi */
|
|
|
|
EXTRACT_HELPER(FPBF, 23, 3);
|
|
|
|
EXTRACT_HELPER(FPIMM, 12, 4);
|
|
|
|
EXTRACT_HELPER(FPL, 25, 1);
|
|
|
|
EXTRACT_HELPER(FPFLM, 17, 8);
|
|
|
|
EXTRACT_HELPER(FPW, 16, 1);
|
|
|
|
|
ppc: Add support for 'mffscrn','mffscrni' instructions
ISA 3.0B added a set of Floating-Point Status and Control Register (FPSCR)
instructions: mffsce, mffscdrn, mffscdrni, mffscrn, mffscrni, mffsl.
This patch adds support for 'mffscrn' and 'mffscrni' instructions.
'mffscrn' and 'mffscrni' are similar to 'mffsl', except they do not return
the status bits (FI, FR, FPRF) and they also set the rounding mode in the
FPSCR.
On CPUs without support for 'mffscrn'/'mffscrni' (below ISA 3.0), the
instructions will execute identically to 'mffs'.
Signed-off-by: Paul A. Clarke <pc@us.ibm.com>
Message-Id: <1568817081-1345-1-git-send-email-pc@us.ibm.com>
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
Signed-off-by: David Gibson <david@gibson.dropbear.id.au>
2019-09-18 17:31:21 +03:00
|
|
|
/* mffscrni */
|
|
|
|
EXTRACT_HELPER(RM, 11, 2);
|
|
|
|
|
2016-11-23 14:37:10 +03:00
|
|
|
/* addpcis */
|
2017-01-13 12:23:39 +03:00
|
|
|
EXTRACT_HELPER_SPLIT_3(DX, 10, 6, 6, 5, 16, 1, 1, 0, 0)
|
2016-11-23 14:37:10 +03:00
|
|
|
#if defined(TARGET_PPC64)
|
|
|
|
/* darn */
|
|
|
|
EXTRACT_HELPER(L, 16, 2);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/*** Jump target decoding ***/
|
|
|
|
/* Immediate address */
|
|
|
|
static inline target_ulong LI(uint32_t opcode)
|
|
|
|
{
|
|
|
|
return (opcode >> 0) & 0x03FFFFFC;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline uint32_t BD(uint32_t opcode)
|
|
|
|
{
|
|
|
|
return (opcode >> 0) & 0xFFFC;
|
|
|
|
}
|
|
|
|
|
|
|
|
EXTRACT_HELPER(BO, 21, 5);
|
|
|
|
EXTRACT_HELPER(BI, 16, 5);
|
|
|
|
/* Absolute/relative address */
|
|
|
|
EXTRACT_HELPER(AA, 1, 1);
|
|
|
|
/* Link */
|
|
|
|
EXTRACT_HELPER(LK, 0, 1);
|
|
|
|
|
|
|
|
/* DFP Z22-form */
|
|
|
|
EXTRACT_HELPER(DCM, 10, 6)
|
|
|
|
|
|
|
|
/* DFP Z23-form */
|
|
|
|
EXTRACT_HELPER(RMC, 9, 2)
|
2017-02-04 01:01:14 +03:00
|
|
|
EXTRACT_HELPER(Rrm, 16, 1)
|
2016-11-23 14:37:10 +03:00
|
|
|
|
2016-11-23 14:37:17 +03:00
|
|
|
EXTRACT_HELPER_SPLIT(DQxT, 3, 1, 21, 5);
|
2016-11-23 14:37:10 +03:00
|
|
|
EXTRACT_HELPER_SPLIT(xT, 0, 1, 21, 5);
|
|
|
|
EXTRACT_HELPER_SPLIT(xS, 0, 1, 21, 5);
|
|
|
|
EXTRACT_HELPER_SPLIT(xA, 2, 1, 16, 5);
|
|
|
|
EXTRACT_HELPER_SPLIT(xB, 1, 1, 11, 5);
|
|
|
|
EXTRACT_HELPER_SPLIT(xC, 3, 1, 6, 5);
|
|
|
|
EXTRACT_HELPER(DM, 8, 2);
|
|
|
|
EXTRACT_HELPER(UIM, 16, 2);
|
|
|
|
EXTRACT_HELPER(SHW, 8, 2);
|
|
|
|
EXTRACT_HELPER(SP, 19, 2);
|
|
|
|
EXTRACT_HELPER(IMM8, 11, 8);
|
2017-01-13 12:23:40 +03:00
|
|
|
EXTRACT_HELPER(DCMX, 16, 7);
|
2017-01-13 12:23:39 +03:00
|
|
|
EXTRACT_HELPER_SPLIT_3(DCMX_XV, 5, 16, 0, 1, 2, 5, 1, 6, 6);
|
2016-11-23 14:37:10 +03:00
|
|
|
|
2017-01-06 09:14:49 +03:00
|
|
|
void helper_compute_fprf_float16(CPUPPCState *env, float16 arg);
|
2017-01-06 09:14:50 +03:00
|
|
|
void helper_compute_fprf_float32(CPUPPCState *env, float32 arg);
|
2017-01-09 17:26:13 +03:00
|
|
|
void helper_compute_fprf_float128(CPUPPCState *env, float128 arg);
|
2018-06-26 19:19:09 +03:00
|
|
|
|
|
|
|
/* Raise a data fault alignment exception for the specified virtual address */
|
|
|
|
void ppc_cpu_do_unaligned_access(CPUState *cs, vaddr addr,
|
|
|
|
MMUAccessType access_type,
|
|
|
|
int mmu_idx, uintptr_t retaddr);
|
2021-04-26 21:47:06 +03:00
|
|
|
|
2021-04-29 19:21:24 +03:00
|
|
|
/* translate.c */
|
|
|
|
|
|
|
|
/* #define PPC_DUMP_CPU */
|
|
|
|
|
|
|
|
int ppc_fixup_cpu(PowerPCCPU *cpu);
|
|
|
|
void create_ppc_opcodes(PowerPCCPU *cpu, Error **errp);
|
|
|
|
void destroy_ppc_opcodes(PowerPCCPU *cpu);
|
|
|
|
|
2021-04-26 21:47:06 +03:00
|
|
|
/* gdbstub.c */
|
|
|
|
void ppc_gdb_init(CPUState *cs, PowerPCCPUClass *ppc);
|
|
|
|
gchar *ppc_gdb_arch_name(CPUState *cs);
|
|
|
|
|
2021-05-18 23:11:23 +03:00
|
|
|
/**
|
|
|
|
* prot_for_access_type:
|
|
|
|
* @access_type: Access type
|
|
|
|
*
|
|
|
|
* Return the protection bit required for the given access type.
|
|
|
|
*/
|
|
|
|
static inline int prot_for_access_type(MMUAccessType access_type)
|
|
|
|
{
|
|
|
|
switch (access_type) {
|
|
|
|
case MMU_INST_FETCH:
|
|
|
|
return PAGE_EXEC;
|
|
|
|
case MMU_DATA_LOAD:
|
|
|
|
return PAGE_READ;
|
|
|
|
case MMU_DATA_STORE:
|
|
|
|
return PAGE_WRITE;
|
|
|
|
}
|
|
|
|
g_assert_not_reached();
|
|
|
|
}
|
|
|
|
|
2016-10-30 06:14:56 +03:00
|
|
|
#endif /* PPC_INTERNAL_H */
|