forked from Mirrors/opensbi
		
	lib: utils/timer: Allow separate base addresses for MTIME and MTIMECMP
We extend the ACLINT library to support separate base addresses for MTIME and MTIMECMP registers. Signed-off-by: Anup Patel <anup.patel@wdc.com> Reviewed-by: Atish Patra <atish.patra@wdc.com>
This commit is contained in:
		@@ -10,6 +10,7 @@
 | 
			
		||||
#include <sbi/riscv_asm.h>
 | 
			
		||||
#include <sbi/riscv_atomic.h>
 | 
			
		||||
#include <sbi/riscv_io.h>
 | 
			
		||||
#include <sbi/sbi_bitops.h>
 | 
			
		||||
#include <sbi/sbi_domain.h>
 | 
			
		||||
#include <sbi/sbi_error.h>
 | 
			
		||||
#include <sbi/sbi_hartmask.h>
 | 
			
		||||
@@ -17,9 +18,6 @@
 | 
			
		||||
#include <sbi/sbi_timer.h>
 | 
			
		||||
#include <sbi_utils/timer/aclint_mtimer.h>
 | 
			
		||||
 | 
			
		||||
#define MTIMER_CMP_OFF		0x0000
 | 
			
		||||
#define MTIMER_VAL_OFF		0x7ff8
 | 
			
		||||
 | 
			
		||||
static struct aclint_mtimer_data *mtimer_hartid2data[SBI_HARTMASK_MAX_BITS];
 | 
			
		||||
 | 
			
		||||
#if __riscv_xlen != 32
 | 
			
		||||
@@ -56,7 +54,7 @@ static void mtimer_time_wr32(bool timecmp, u64 value, volatile u64 *addr)
 | 
			
		||||
static u64 mtimer_value(void)
 | 
			
		||||
{
 | 
			
		||||
	struct aclint_mtimer_data *mt = mtimer_hartid2data[current_hartid()];
 | 
			
		||||
	u64 *time_val = ((void *)mt->addr) + MTIMER_VAL_OFF;
 | 
			
		||||
	u64 *time_val = (void *)mt->mtime_addr;
 | 
			
		||||
 | 
			
		||||
	/* Read MTIMER Time Value */
 | 
			
		||||
	return mt->time_rd(time_val) + mt->time_delta;
 | 
			
		||||
@@ -66,7 +64,7 @@ static void mtimer_event_stop(void)
 | 
			
		||||
{
 | 
			
		||||
	u32 target_hart = current_hartid();
 | 
			
		||||
	struct aclint_mtimer_data *mt = mtimer_hartid2data[target_hart];
 | 
			
		||||
	u64 *time_cmp = (void *)mt->addr + MTIMER_CMP_OFF;
 | 
			
		||||
	u64 *time_cmp = (void *)mt->mtimecmp_addr;
 | 
			
		||||
 | 
			
		||||
	/* Clear MTIMER Time Compare */
 | 
			
		||||
	mt->time_wr(true, -1ULL, &time_cmp[target_hart - mt->first_hartid]);
 | 
			
		||||
@@ -76,7 +74,7 @@ static void mtimer_event_start(u64 next_event)
 | 
			
		||||
{
 | 
			
		||||
	u32 target_hart = current_hartid();
 | 
			
		||||
	struct aclint_mtimer_data *mt = mtimer_hartid2data[target_hart];
 | 
			
		||||
	u64 *time_cmp = (void *)mt->addr + MTIMER_CMP_OFF;
 | 
			
		||||
	u64 *time_cmp = (void *)mt->mtimecmp_addr;
 | 
			
		||||
 | 
			
		||||
	/* Program MTIMER Time Compare */
 | 
			
		||||
	mt->time_wr(true, next_event - mt->time_delta,
 | 
			
		||||
@@ -111,8 +109,8 @@ int aclint_mtimer_warm_init(void)
 | 
			
		||||
	 */
 | 
			
		||||
	if (mt->time_delta_reference) {
 | 
			
		||||
		reference = mt->time_delta_reference;
 | 
			
		||||
		mt_time_val = (void *)mt->addr + MTIMER_VAL_OFF;
 | 
			
		||||
		ref_time_val = (void *)reference->addr + MTIMER_VAL_OFF;
 | 
			
		||||
		mt_time_val = (void *)mt->mtime_addr;
 | 
			
		||||
		ref_time_val = (void *)reference->mtime_addr;
 | 
			
		||||
		if (!atomic_raw_xchg_ulong(&mt->time_delta_computed, 1)) {
 | 
			
		||||
			v1 = mt->time_rd(mt_time_val);
 | 
			
		||||
			mv = reference->time_rd(ref_time_val);
 | 
			
		||||
@@ -122,24 +120,54 @@ int aclint_mtimer_warm_init(void)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	/* Clear Time Compare */
 | 
			
		||||
	mt_time_cmp = (void *)mt->addr + MTIMER_CMP_OFF;
 | 
			
		||||
	mt_time_cmp = (void *)mt->mtimecmp_addr;
 | 
			
		||||
	mt->time_wr(true, -1ULL,
 | 
			
		||||
		    &mt_time_cmp[target_hart - mt->first_hartid]);
 | 
			
		||||
 | 
			
		||||
	return 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static int aclint_mtimer_add_regions(unsigned long addr, unsigned long size)
 | 
			
		||||
{
 | 
			
		||||
#define MTIMER_ADD_REGION_ALIGN		0x1000
 | 
			
		||||
	int rc;
 | 
			
		||||
	unsigned long pos, end, rsize;
 | 
			
		||||
	struct sbi_domain_memregion reg;
 | 
			
		||||
 | 
			
		||||
	pos = addr;
 | 
			
		||||
	end = addr + size;
 | 
			
		||||
	while (pos < end) {
 | 
			
		||||
		rsize = pos & (MTIMER_ADD_REGION_ALIGN - 1);
 | 
			
		||||
		if (rsize)
 | 
			
		||||
			rsize = 1UL << __ffs(pos);
 | 
			
		||||
		else
 | 
			
		||||
			rsize = ((end - pos) < MTIMER_ADD_REGION_ALIGN) ?
 | 
			
		||||
				(end - pos) : MTIMER_ADD_REGION_ALIGN;
 | 
			
		||||
 | 
			
		||||
		sbi_domain_memregion_init(pos, rsize,
 | 
			
		||||
					  SBI_DOMAIN_MEMREGION_MMIO, ®);
 | 
			
		||||
		rc = sbi_domain_root_add_memregion(®);
 | 
			
		||||
		if (rc)
 | 
			
		||||
			return rc;
 | 
			
		||||
		pos += rsize;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	return 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
int aclint_mtimer_cold_init(struct aclint_mtimer_data *mt,
 | 
			
		||||
			    struct aclint_mtimer_data *reference)
 | 
			
		||||
{
 | 
			
		||||
	u32 i;
 | 
			
		||||
	int rc;
 | 
			
		||||
	unsigned long pos, region_size;
 | 
			
		||||
	struct sbi_domain_memregion reg;
 | 
			
		||||
 | 
			
		||||
	/* Sanity checks */
 | 
			
		||||
	if (!mt || (mt->addr & (ACLINT_MTIMER_ALIGN - 1)) ||
 | 
			
		||||
	    (mt->size < ACLINT_MTIMER_SIZE) ||
 | 
			
		||||
	if (!mt || !mt->mtime_size ||
 | 
			
		||||
	    (mt->hart_count && !mt->mtimecmp_size) ||
 | 
			
		||||
	    (mt->mtime_addr & (ACLINT_MTIMER_ALIGN - 1)) ||
 | 
			
		||||
	    (mt->mtime_size & (ACLINT_MTIMER_ALIGN - 1)) ||
 | 
			
		||||
	    (mt->mtimecmp_addr & (ACLINT_MTIMER_ALIGN - 1)) ||
 | 
			
		||||
	    (mt->mtimecmp_size & (ACLINT_MTIMER_ALIGN - 1)) ||
 | 
			
		||||
	    (mt->first_hartid >= SBI_HARTMASK_MAX_BITS) ||
 | 
			
		||||
	    (mt->hart_count > ACLINT_MTIMER_MAX_HARTS))
 | 
			
		||||
		return SBI_EINVAL;
 | 
			
		||||
@@ -164,12 +192,24 @@ int aclint_mtimer_cold_init(struct aclint_mtimer_data *mt,
 | 
			
		||||
		mtimer_hartid2data[mt->first_hartid + i] = mt;
 | 
			
		||||
 | 
			
		||||
	/* Add MTIMER regions to the root domain */
 | 
			
		||||
	for (pos = 0; pos < mt->size; pos += ACLINT_MTIMER_ALIGN) {
 | 
			
		||||
		region_size = ((mt->size - pos) < ACLINT_MTIMER_ALIGN) ?
 | 
			
		||||
			      (mt->size - pos) : ACLINT_MTIMER_ALIGN;
 | 
			
		||||
		sbi_domain_memregion_init(mt->addr + pos, region_size,
 | 
			
		||||
					  SBI_DOMAIN_MEMREGION_MMIO, ®);
 | 
			
		||||
		rc = sbi_domain_root_add_memregion(®);
 | 
			
		||||
	if (mt->mtime_addr == (mt->mtimecmp_addr + mt->mtimecmp_size)) {
 | 
			
		||||
		rc = aclint_mtimer_add_regions(mt->mtimecmp_addr,
 | 
			
		||||
					mt->mtime_size + mt->mtimecmp_size);
 | 
			
		||||
		if (rc)
 | 
			
		||||
			return rc;
 | 
			
		||||
	} else if (mt->mtimecmp_addr == (mt->mtime_addr + mt->mtime_size)) {
 | 
			
		||||
		rc = aclint_mtimer_add_regions(mt->mtime_addr,
 | 
			
		||||
					mt->mtime_size + mt->mtimecmp_size);
 | 
			
		||||
		if (rc)
 | 
			
		||||
			return rc;
 | 
			
		||||
	} else {
 | 
			
		||||
		rc = aclint_mtimer_add_regions(mt->mtime_addr,
 | 
			
		||||
						mt->mtime_size);
 | 
			
		||||
		if (rc)
 | 
			
		||||
			return rc;
 | 
			
		||||
 | 
			
		||||
		rc = aclint_mtimer_add_regions(mt->mtimecmp_addr,
 | 
			
		||||
						mt->mtimecmp_size);
 | 
			
		||||
		if (rc)
 | 
			
		||||
			return rc;
 | 
			
		||||
	}
 | 
			
		||||
 
 | 
			
		||||
@@ -22,7 +22,7 @@ static int timer_mtimer_cold_init(void *fdt, int nodeoff,
 | 
			
		||||
				  const struct fdt_match *match)
 | 
			
		||||
{
 | 
			
		||||
	int rc;
 | 
			
		||||
	unsigned long offset;
 | 
			
		||||
	unsigned long offset, addr, size;
 | 
			
		||||
	struct aclint_mtimer_data *mt, *mtmaster = NULL;
 | 
			
		||||
 | 
			
		||||
	if (MTIMER_MAX_NR <= mtimer_count)
 | 
			
		||||
@@ -31,19 +31,23 @@ static int timer_mtimer_cold_init(void *fdt, int nodeoff,
 | 
			
		||||
	if (0 < mtimer_count)
 | 
			
		||||
		mtmaster = &mtimer[0];
 | 
			
		||||
 | 
			
		||||
	rc = fdt_parse_aclint_node(fdt, nodeoff, true, &mt->addr, &mt->size,
 | 
			
		||||
	rc = fdt_parse_aclint_node(fdt, nodeoff, true, &addr, &size,
 | 
			
		||||
				   &mt->first_hartid, &mt->hart_count);
 | 
			
		||||
	if (rc)
 | 
			
		||||
		return rc;
 | 
			
		||||
	mt->has_64bit_mmio = true;
 | 
			
		||||
 | 
			
		||||
	mt->mtimecmp_addr = addr + ACLINT_DEFAULT_MTIMECMP_OFFSET;
 | 
			
		||||
	mt->mtimecmp_size = ACLINT_DEFAULT_MTIMECMP_SIZE;
 | 
			
		||||
	mt->mtime_addr = addr + ACLINT_DEFAULT_MTIME_OFFSET;
 | 
			
		||||
	mt->mtime_size = size - mt->mtimecmp_size;
 | 
			
		||||
 | 
			
		||||
	if (match->data) {
 | 
			
		||||
		/* Adjust MTIMER address and size for CLINT device */
 | 
			
		||||
		offset = *((unsigned long *)match->data);
 | 
			
		||||
		mt->addr += offset;
 | 
			
		||||
		if ((mt->size - offset) < ACLINT_MTIMER_SIZE)
 | 
			
		||||
			return SBI_EINVAL;
 | 
			
		||||
		mt->size -= offset;
 | 
			
		||||
		mt->mtime_addr += offset;
 | 
			
		||||
		mt->mtimecmp_addr += offset;
 | 
			
		||||
		mt->mtime_size -= offset;
 | 
			
		||||
		/* Parse additional CLINT properties */
 | 
			
		||||
		if (fdt_getprop(fdt, nodeoff, "clint,has-no-64bit-mmio", &rc))
 | 
			
		||||
			mt->has_64bit_mmio = false;
 | 
			
		||||
 
 | 
			
		||||
		Reference in New Issue
	
	Block a user