aboutsummaryrefslogtreecommitdiff
path: root/lib/utils/timer/aclint_mtimer.c
diff options
context:
space:
mode:
Diffstat (limited to 'lib/utils/timer/aclint_mtimer.c')
-rw-r--r--lib/utils/timer/aclint_mtimer.c78
1 files changed, 59 insertions, 19 deletions
diff --git a/lib/utils/timer/aclint_mtimer.c b/lib/utils/timer/aclint_mtimer.c
index 41b0290..0e4e846 100644
--- a/lib/utils/timer/aclint_mtimer.c
+++ b/lib/utils/timer/aclint_mtimer.c
@@ -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, &reg);
+ rc = sbi_domain_root_add_memregion(&reg);
+ 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, &reg);
- rc = sbi_domain_root_add_memregion(&reg);
+ 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;
}