mirror of https://github.com/xemu-project/xemu.git
target-arm: Add the Hypervisor timer
Signed-off-by: Edgar E. Iglesias <edgar.iglesias@xilinx.com> Reviewed-by: Peter Maydell <peter.maydell@linaro.org> Message-id: 1436791864-4582-6-git-send-email-edgar.iglesias@gmail.com Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
This commit is contained in:
parent
0e3eca4c26
commit
b0e66d95e4
|
@ -224,6 +224,7 @@ int arm_cpu_gdb_write_register(CPUState *cpu, uint8_t *buf, int reg);
|
||||||
/* Callback functions for the generic timer's timers. */
|
/* Callback functions for the generic timer's timers. */
|
||||||
void arm_gt_ptimer_cb(void *opaque);
|
void arm_gt_ptimer_cb(void *opaque);
|
||||||
void arm_gt_vtimer_cb(void *opaque);
|
void arm_gt_vtimer_cb(void *opaque);
|
||||||
|
void arm_gt_htimer_cb(void *opaque);
|
||||||
|
|
||||||
#ifdef TARGET_AARCH64
|
#ifdef TARGET_AARCH64
|
||||||
int aarch64_cpu_gdb_read_register(CPUState *cpu, uint8_t *buf, int reg);
|
int aarch64_cpu_gdb_read_register(CPUState *cpu, uint8_t *buf, int reg);
|
||||||
|
|
|
@ -453,6 +453,8 @@ static void arm_cpu_initfn(Object *obj)
|
||||||
arm_gt_ptimer_cb, cpu);
|
arm_gt_ptimer_cb, cpu);
|
||||||
cpu->gt_timer[GTIMER_VIRT] = timer_new(QEMU_CLOCK_VIRTUAL, GTIMER_SCALE,
|
cpu->gt_timer[GTIMER_VIRT] = timer_new(QEMU_CLOCK_VIRTUAL, GTIMER_SCALE,
|
||||||
arm_gt_vtimer_cb, cpu);
|
arm_gt_vtimer_cb, cpu);
|
||||||
|
cpu->gt_timer[GTIMER_HYP] = timer_new(QEMU_CLOCK_VIRTUAL, GTIMER_SCALE,
|
||||||
|
arm_gt_htimer_cb, cpu);
|
||||||
qdev_init_gpio_out(DEVICE(cpu), cpu->gt_timer_outputs,
|
qdev_init_gpio_out(DEVICE(cpu), cpu->gt_timer_outputs,
|
||||||
ARRAY_SIZE(cpu->gt_timer_outputs));
|
ARRAY_SIZE(cpu->gt_timer_outputs));
|
||||||
#endif
|
#endif
|
||||||
|
|
|
@ -113,7 +113,8 @@ typedef struct ARMGenericTimer {
|
||||||
|
|
||||||
#define GTIMER_PHYS 0
|
#define GTIMER_PHYS 0
|
||||||
#define GTIMER_VIRT 1
|
#define GTIMER_VIRT 1
|
||||||
#define NUM_GTIMERS 2
|
#define GTIMER_HYP 2
|
||||||
|
#define NUM_GTIMERS 3
|
||||||
|
|
||||||
typedef struct {
|
typedef struct {
|
||||||
uint64_t raw_tcr;
|
uint64_t raw_tcr;
|
||||||
|
|
|
@ -1392,6 +1392,34 @@ static void gt_cntvoff_write(CPUARMState *env, const ARMCPRegInfo *ri,
|
||||||
gt_recalc_timer(cpu, GTIMER_VIRT);
|
gt_recalc_timer(cpu, GTIMER_VIRT);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void gt_hyp_timer_reset(CPUARMState *env, const ARMCPRegInfo *ri)
|
||||||
|
{
|
||||||
|
gt_timer_reset(env, ri, GTIMER_HYP);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void gt_hyp_cval_write(CPUARMState *env, const ARMCPRegInfo *ri,
|
||||||
|
uint64_t value)
|
||||||
|
{
|
||||||
|
gt_cval_write(env, ri, GTIMER_HYP, value);
|
||||||
|
}
|
||||||
|
|
||||||
|
static uint64_t gt_hyp_tval_read(CPUARMState *env, const ARMCPRegInfo *ri)
|
||||||
|
{
|
||||||
|
return gt_tval_read(env, ri, GTIMER_HYP);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void gt_hyp_tval_write(CPUARMState *env, const ARMCPRegInfo *ri,
|
||||||
|
uint64_t value)
|
||||||
|
{
|
||||||
|
gt_tval_write(env, ri, GTIMER_HYP, value);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void gt_hyp_ctl_write(CPUARMState *env, const ARMCPRegInfo *ri,
|
||||||
|
uint64_t value)
|
||||||
|
{
|
||||||
|
gt_ctl_write(env, ri, GTIMER_HYP, value);
|
||||||
|
}
|
||||||
|
|
||||||
void arm_gt_ptimer_cb(void *opaque)
|
void arm_gt_ptimer_cb(void *opaque)
|
||||||
{
|
{
|
||||||
ARMCPU *cpu = opaque;
|
ARMCPU *cpu = opaque;
|
||||||
|
@ -1406,6 +1434,13 @@ void arm_gt_vtimer_cb(void *opaque)
|
||||||
gt_recalc_timer(cpu, GTIMER_VIRT);
|
gt_recalc_timer(cpu, GTIMER_VIRT);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void arm_gt_htimer_cb(void *opaque)
|
||||||
|
{
|
||||||
|
ARMCPU *cpu = opaque;
|
||||||
|
|
||||||
|
gt_recalc_timer(cpu, GTIMER_HYP);
|
||||||
|
}
|
||||||
|
|
||||||
static const ARMCPRegInfo generic_timer_cp_reginfo[] = {
|
static const ARMCPRegInfo generic_timer_cp_reginfo[] = {
|
||||||
/* Note that CNTFRQ is purely reads-as-written for the benefit
|
/* Note that CNTFRQ is purely reads-as-written for the benefit
|
||||||
* of software; writing it doesn't actually change the timer frequency.
|
* of software; writing it doesn't actually change the timer frequency.
|
||||||
|
@ -2711,6 +2746,18 @@ static const ARMCPRegInfo el3_no_el2_cp_reginfo[] = {
|
||||||
{ .name = "CNTVOFF", .cp = 15, .opc1 = 4, .crm = 14,
|
{ .name = "CNTVOFF", .cp = 15, .opc1 = 4, .crm = 14,
|
||||||
.access = PL2_RW, .type = ARM_CP_64BIT | ARM_CP_CONST,
|
.access = PL2_RW, .type = ARM_CP_64BIT | ARM_CP_CONST,
|
||||||
.resetvalue = 0 },
|
.resetvalue = 0 },
|
||||||
|
{ .name = "CNTHP_CVAL_EL2", .state = ARM_CP_STATE_AA64,
|
||||||
|
.opc0 = 3, .opc1 = 4, .crn = 14, .crm = 2, .opc2 = 2,
|
||||||
|
.access = PL2_RW, .type = ARM_CP_CONST, .resetvalue = 0 },
|
||||||
|
{ .name = "CNTHP_CVAL", .cp = 15, .opc1 = 6, .crm = 14,
|
||||||
|
.access = PL2_RW, .type = ARM_CP_64BIT | ARM_CP_CONST,
|
||||||
|
.resetvalue = 0 },
|
||||||
|
{ .name = "CNTHP_TVAL_EL2", .state = ARM_CP_STATE_BOTH,
|
||||||
|
.opc0 = 3, .opc1 = 4, .crn = 14, .crm = 2, .opc2 = 0,
|
||||||
|
.access = PL2_RW, .type = ARM_CP_CONST, .resetvalue = 0 },
|
||||||
|
{ .name = "CNTHP_CTL_EL2", .state = ARM_CP_STATE_BOTH,
|
||||||
|
.opc0 = 3, .opc1 = 4, .crn = 14, .crm = 2, .opc2 = 1,
|
||||||
|
.access = PL2_RW, .type = ARM_CP_CONST, .resetvalue = 0 },
|
||||||
REGINFO_SENTINEL
|
REGINFO_SENTINEL
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -2840,6 +2887,27 @@ static const ARMCPRegInfo el2_cp_reginfo[] = {
|
||||||
.access = PL2_RW, .type = ARM_CP_64BIT | ARM_CP_ALIAS | ARM_CP_IO,
|
.access = PL2_RW, .type = ARM_CP_64BIT | ARM_CP_ALIAS | ARM_CP_IO,
|
||||||
.writefn = gt_cntvoff_write,
|
.writefn = gt_cntvoff_write,
|
||||||
.fieldoffset = offsetof(CPUARMState, cp15.cntvoff_el2) },
|
.fieldoffset = offsetof(CPUARMState, cp15.cntvoff_el2) },
|
||||||
|
{ .name = "CNTHP_CVAL_EL2", .state = ARM_CP_STATE_AA64,
|
||||||
|
.opc0 = 3, .opc1 = 4, .crn = 14, .crm = 2, .opc2 = 2,
|
||||||
|
.fieldoffset = offsetof(CPUARMState, cp15.c14_timer[GTIMER_HYP].cval),
|
||||||
|
.type = ARM_CP_IO, .access = PL2_RW,
|
||||||
|
.writefn = gt_hyp_cval_write, .raw_writefn = raw_write },
|
||||||
|
{ .name = "CNTHP_CVAL", .cp = 15, .opc1 = 6, .crm = 14,
|
||||||
|
.fieldoffset = offsetof(CPUARMState, cp15.c14_timer[GTIMER_HYP].cval),
|
||||||
|
.access = PL2_RW, .type = ARM_CP_64BIT | ARM_CP_IO,
|
||||||
|
.writefn = gt_hyp_cval_write, .raw_writefn = raw_write },
|
||||||
|
{ .name = "CNTHP_TVAL_EL2", .state = ARM_CP_STATE_BOTH,
|
||||||
|
.opc0 = 3, .opc1 = 4, .crn = 14, .crm = 2, .opc2 = 0,
|
||||||
|
.type = ARM_CP_IO, .access = PL2_RW,
|
||||||
|
.resetfn = gt_hyp_timer_reset,
|
||||||
|
.readfn = gt_hyp_tval_read, .writefn = gt_hyp_tval_write },
|
||||||
|
{ .name = "CNTHP_CTL_EL2", .state = ARM_CP_STATE_BOTH,
|
||||||
|
.type = ARM_CP_IO,
|
||||||
|
.opc0 = 3, .opc1 = 4, .crn = 14, .crm = 2, .opc2 = 1,
|
||||||
|
.access = PL2_RW,
|
||||||
|
.fieldoffset = offsetof(CPUARMState, cp15.c14_timer[GTIMER_HYP].ctl),
|
||||||
|
.resetvalue = 0,
|
||||||
|
.writefn = gt_hyp_ctl_write, .raw_writefn = raw_write },
|
||||||
#endif
|
#endif
|
||||||
REGINFO_SENTINEL
|
REGINFO_SENTINEL
|
||||||
};
|
};
|
||||||
|
|
Loading…
Reference in New Issue