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…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user