mirror of https://gitlab.com/qemu-project/qemu
You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
83 lines
2.6 KiB
C
83 lines
2.6 KiB
C
/* SPDX-License-Identifier: GPL-2.0-or-later */
|
|
|
|
#include "qemu/osdep.h"
|
|
#include "qemu.h"
|
|
#include "loader.h"
|
|
#include "elf.h"
|
|
#include "target_elf.h"
|
|
|
|
|
|
const char *get_elf_cpu_model(uint32_t eflags)
|
|
{
|
|
return "qemu";
|
|
}
|
|
|
|
#define GET_FEATURE(_feat, _hwcap) \
|
|
do { if (s390_has_feat(_feat)) { hwcap |= _hwcap; } } while (0)
|
|
|
|
abi_ulong get_elf_hwcap(CPUState *cs)
|
|
{
|
|
/*
|
|
* Let's assume we always have esan3 and zarch.
|
|
* 31-bit processes can use 64-bit registers (high gprs).
|
|
*/
|
|
uint32_t hwcap = HWCAP_S390_ESAN3 | HWCAP_S390_ZARCH | HWCAP_S390_HIGH_GPRS;
|
|
|
|
GET_FEATURE(S390_FEAT_STFLE, HWCAP_S390_STFLE);
|
|
GET_FEATURE(S390_FEAT_MSA, HWCAP_S390_MSA);
|
|
GET_FEATURE(S390_FEAT_LONG_DISPLACEMENT, HWCAP_S390_LDISP);
|
|
GET_FEATURE(S390_FEAT_EXTENDED_IMMEDIATE, HWCAP_S390_EIMM);
|
|
if (s390_has_feat(S390_FEAT_EXTENDED_TRANSLATION_3) &&
|
|
s390_has_feat(S390_FEAT_ETF3_ENH)) {
|
|
hwcap |= HWCAP_S390_ETF3EH;
|
|
}
|
|
GET_FEATURE(S390_FEAT_VECTOR, HWCAP_S390_VXRS);
|
|
GET_FEATURE(S390_FEAT_VECTOR_ENH, HWCAP_S390_VXRS_EXT);
|
|
GET_FEATURE(S390_FEAT_VECTOR_ENH2, HWCAP_S390_VXRS_EXT2);
|
|
|
|
return hwcap;
|
|
}
|
|
|
|
const char *elf_hwcap_str(uint32_t bit)
|
|
{
|
|
static const char *hwcap_str[] = {
|
|
[HWCAP_S390_NR_ESAN3] = "esan3",
|
|
[HWCAP_S390_NR_ZARCH] = "zarch",
|
|
[HWCAP_S390_NR_STFLE] = "stfle",
|
|
[HWCAP_S390_NR_MSA] = "msa",
|
|
[HWCAP_S390_NR_LDISP] = "ldisp",
|
|
[HWCAP_S390_NR_EIMM] = "eimm",
|
|
[HWCAP_S390_NR_DFP] = "dfp",
|
|
[HWCAP_S390_NR_HPAGE] = "edat",
|
|
[HWCAP_S390_NR_ETF3EH] = "etf3eh",
|
|
[HWCAP_S390_NR_HIGH_GPRS] = "highgprs",
|
|
[HWCAP_S390_NR_TE] = "te",
|
|
[HWCAP_S390_NR_VXRS] = "vx",
|
|
[HWCAP_S390_NR_VXRS_BCD] = "vxd",
|
|
[HWCAP_S390_NR_VXRS_EXT] = "vxe",
|
|
[HWCAP_S390_NR_GS] = "gs",
|
|
[HWCAP_S390_NR_VXRS_EXT2] = "vxe2",
|
|
[HWCAP_S390_NR_VXRS_PDE] = "vxp",
|
|
[HWCAP_S390_NR_SORT] = "sort",
|
|
[HWCAP_S390_NR_DFLT] = "dflt",
|
|
[HWCAP_S390_NR_NNPA] = "nnpa",
|
|
[HWCAP_S390_NR_PCI_MIO] = "pcimio",
|
|
[HWCAP_S390_NR_SIE] = "sie",
|
|
};
|
|
|
|
return bit < ARRAY_SIZE(hwcap_str) ? hwcap_str[bit] : NULL;
|
|
}
|
|
|
|
void elf_core_copy_regs(target_elf_gregset_t *r, const CPUS390XState *env)
|
|
{
|
|
r->pt.psw.mask = tswapal(env->psw.mask);
|
|
r->pt.psw.addr = tswapal(env->psw.addr);
|
|
for (int i = 0; i < 16; i++) {
|
|
r->pt.gprs[i] = tswapal(env->regs[i]);
|
|
}
|
|
for (int i = 0; i < 16; i++) {
|
|
r->pt.acrs[i] = tswap32(env->aregs[i]);
|
|
}
|
|
r->pt.orig_gpr2 = 0;
|
|
}
|