 eb90ab9437
			
		
	
	
		eb90ab9437
		
	
	
	
	
		
			
			Provide a new function mips_gictimer_get_freq() which returns the frequency at which a GIC timer will count. This will be useful for boards which perform setup based upon this frequency. Signed-off-by: Paul Burton <paul.burton@imgtec.com> Reviewed-by: Leon Alrae <leon.alrae@imgtec.com> Reviewed-by: Philippe Mathieu-Daudé <f4bug@amsat.org> Signed-off-by: Yongbok Kim <yongbok.kim@imgtec.com>
		
			
				
	
	
		
			148 lines
		
	
	
		
			4.4 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			148 lines
		
	
	
		
			4.4 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * This file is subject to the terms and conditions of the GNU General Public
 | |
|  * License.  See the file "COPYING" in the main directory of this archive
 | |
|  * for more details.
 | |
|  *
 | |
|  * Copyright (C) 2016 Imagination Technologies
 | |
|  */
 | |
| 
 | |
| #include "qemu/osdep.h"
 | |
| #include "hw/hw.h"
 | |
| #include "hw/sysbus.h"
 | |
| #include "qemu/timer.h"
 | |
| #include "hw/timer/mips_gictimer.h"
 | |
| 
 | |
| #define TIMER_PERIOD 10 /* 10 ns period for 100 Mhz frequency */
 | |
| 
 | |
| uint32_t mips_gictimer_get_freq(MIPSGICTimerState *gic)
 | |
| {
 | |
|     return NANOSECONDS_PER_SECOND / TIMER_PERIOD;
 | |
| }
 | |
| 
 | |
| static void gic_vptimer_update(MIPSGICTimerState *gictimer,
 | |
|                                    uint32_t vp_index, uint64_t now)
 | |
| {
 | |
|     uint64_t next;
 | |
|     uint32_t wait;
 | |
| 
 | |
|     wait = gictimer->vptimers[vp_index].comparelo - gictimer->sh_counterlo -
 | |
|            (uint32_t)(now / TIMER_PERIOD);
 | |
|     next = now + (uint64_t)wait * TIMER_PERIOD;
 | |
| 
 | |
|     timer_mod(gictimer->vptimers[vp_index].qtimer, next);
 | |
| }
 | |
| 
 | |
| static void gic_vptimer_expire(MIPSGICTimerState *gictimer, uint32_t vp_index,
 | |
|                                uint64_t now)
 | |
| {
 | |
|     if (gictimer->countstop) {
 | |
|         /* timer stopped */
 | |
|         return;
 | |
|     }
 | |
|     gictimer->cb(gictimer->opaque, vp_index);
 | |
|     gic_vptimer_update(gictimer, vp_index, now);
 | |
| }
 | |
| 
 | |
| static void gic_vptimer_cb(void *opaque)
 | |
| {
 | |
|     MIPSGICTimerVPState *vptimer = opaque;
 | |
|     MIPSGICTimerState *gictimer = vptimer->gictimer;
 | |
|     gic_vptimer_expire(gictimer, vptimer->vp_index,
 | |
|                        qemu_clock_get_ns(QEMU_CLOCK_VIRTUAL));
 | |
| }
 | |
| 
 | |
| uint32_t mips_gictimer_get_sh_count(MIPSGICTimerState *gictimer)
 | |
| {
 | |
|     int i;
 | |
|     if (gictimer->countstop) {
 | |
|         return gictimer->sh_counterlo;
 | |
|     } else {
 | |
|         uint64_t now;
 | |
|         now = qemu_clock_get_ns(QEMU_CLOCK_VIRTUAL);
 | |
|         for (i = 0; i < gictimer->num_vps; i++) {
 | |
|             if (timer_pending(gictimer->vptimers[i].qtimer)
 | |
|                 && timer_expired(gictimer->vptimers[i].qtimer, now)) {
 | |
|                 /* The timer has already expired.  */
 | |
|                 gic_vptimer_expire(gictimer, i, now);
 | |
|             }
 | |
|         }
 | |
|         return gictimer->sh_counterlo + (uint32_t)(now / TIMER_PERIOD);
 | |
|     }
 | |
| }
 | |
| 
 | |
| void mips_gictimer_store_sh_count(MIPSGICTimerState *gictimer, uint64_t count)
 | |
| {
 | |
|     int i;
 | |
|     uint64_t now;
 | |
| 
 | |
|     if (gictimer->countstop || !gictimer->vptimers[0].qtimer) {
 | |
|         gictimer->sh_counterlo = count;
 | |
|     } else {
 | |
|         /* Store new count register */
 | |
|         now = qemu_clock_get_ns(QEMU_CLOCK_VIRTUAL);
 | |
|         gictimer->sh_counterlo = count - (uint32_t)(now / TIMER_PERIOD);
 | |
|         /* Update timer timer */
 | |
|         for (i = 0; i < gictimer->num_vps; i++) {
 | |
|             gic_vptimer_update(gictimer, i, now);
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| uint32_t mips_gictimer_get_vp_compare(MIPSGICTimerState *gictimer,
 | |
|                                       uint32_t vp_index)
 | |
| {
 | |
|     return gictimer->vptimers[vp_index].comparelo;
 | |
| }
 | |
| 
 | |
| void mips_gictimer_store_vp_compare(MIPSGICTimerState *gictimer,
 | |
|                                     uint32_t vp_index, uint64_t compare)
 | |
| {
 | |
|     gictimer->vptimers[vp_index].comparelo = (uint32_t) compare;
 | |
|     gic_vptimer_update(gictimer, vp_index,
 | |
|                        qemu_clock_get_ns(QEMU_CLOCK_VIRTUAL));
 | |
| }
 | |
| 
 | |
| uint8_t mips_gictimer_get_countstop(MIPSGICTimerState *gictimer)
 | |
| {
 | |
|     return gictimer->countstop;
 | |
| }
 | |
| 
 | |
| void mips_gictimer_start_count(MIPSGICTimerState *gictimer)
 | |
| {
 | |
|     gictimer->countstop = 0;
 | |
|     mips_gictimer_store_sh_count(gictimer, gictimer->sh_counterlo);
 | |
| }
 | |
| 
 | |
| void mips_gictimer_stop_count(MIPSGICTimerState *gictimer)
 | |
| {
 | |
|     int i;
 | |
| 
 | |
|     gictimer->countstop = 1;
 | |
|     /* Store the current value */
 | |
|     gictimer->sh_counterlo +=
 | |
|         (uint32_t)(qemu_clock_get_ns(QEMU_CLOCK_VIRTUAL) / TIMER_PERIOD);
 | |
|     for (i = 0; i < gictimer->num_vps; i++) {
 | |
|         timer_del(gictimer->vptimers[i].qtimer);
 | |
|     }
 | |
| }
 | |
| 
 | |
| MIPSGICTimerState *mips_gictimer_init(void *opaque, uint32_t nvps,
 | |
|                                       MIPSGICTimerCB *cb)
 | |
| {
 | |
|     int i;
 | |
|     MIPSGICTimerState *gictimer = g_new(MIPSGICTimerState, 1);
 | |
|     gictimer->vptimers = g_new(MIPSGICTimerVPState, nvps);
 | |
|     gictimer->countstop = 1;
 | |
|     gictimer->num_vps = nvps;
 | |
|     gictimer->opaque = opaque;
 | |
|     gictimer->cb = cb;
 | |
|     for (i = 0; i < nvps; i++) {
 | |
|         gictimer->vptimers[i].gictimer = gictimer;
 | |
|         gictimer->vptimers[i].vp_index = i;
 | |
|         gictimer->vptimers[i].qtimer = timer_new_ns(QEMU_CLOCK_VIRTUAL,
 | |
|                                             &gic_vptimer_cb,
 | |
|                                             &gictimer->vptimers[i]);
 | |
|     }
 | |
|     return gictimer;
 | |
| }
 |