mirror of
https://github.com/riscv-software-src/opensbi.git
synced 2025-08-24 15:31:22 +01:00
137 lines
2.8 KiB
C
137 lines
2.8 KiB
C
/*
|
|
* Copyright (c) 2018 Western Digital Corporation or its affiliates.
|
|
*
|
|
* Authors:
|
|
* Anup Patel <anup.patel@wdc.com>
|
|
*
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
|
*/
|
|
|
|
#include <sbi/riscv_io.h>
|
|
#include <sbi/riscv_atomic.h>
|
|
#include <plat/sys/clint.h>
|
|
|
|
static u32 clint_ipi_hart_count;
|
|
static volatile void *clint_ipi_base;
|
|
static volatile u32 *clint_ipi;
|
|
|
|
void clint_ipi_inject(u32 target_hart, u32 source_hart)
|
|
{
|
|
if ((clint_ipi_hart_count <= target_hart) ||
|
|
(clint_ipi_hart_count <= source_hart))
|
|
return;
|
|
|
|
/* Set CLINT IPI */
|
|
__io_bw();
|
|
atomic_raw_xchg_uint(&clint_ipi[target_hart], 1);
|
|
}
|
|
|
|
void clint_ipi_sync(u32 target_hart, u32 source_hart)
|
|
{
|
|
u32 target_ipi, incoming_ipi;
|
|
|
|
if ((clint_ipi_hart_count <= target_hart) ||
|
|
(clint_ipi_hart_count <= source_hart))
|
|
return;
|
|
|
|
/* Wait until target HART has handled IPI */
|
|
incoming_ipi = 0;
|
|
while (1) {
|
|
target_ipi = readl(&clint_ipi[target_hart]);
|
|
if (!target_ipi)
|
|
break;
|
|
|
|
__io_bw();
|
|
incoming_ipi |=
|
|
atomic_raw_xchg_uint(&clint_ipi[source_hart], 0);
|
|
}
|
|
|
|
if (incoming_ipi) {
|
|
__io_bw();
|
|
atomic_raw_xchg_uint(&clint_ipi[source_hart], incoming_ipi);
|
|
}
|
|
}
|
|
|
|
void clint_ipi_clear(u32 target_hart)
|
|
{
|
|
if (clint_ipi_hart_count <= target_hart)
|
|
return;
|
|
|
|
/* Clear CLINT IPI */
|
|
__io_bw();
|
|
atomic_raw_xchg_uint(&clint_ipi[target_hart], 0);
|
|
}
|
|
|
|
int clint_warm_ipi_init(u32 target_hart)
|
|
{
|
|
if (clint_ipi_hart_count <= target_hart ||
|
|
!clint_ipi_base)
|
|
return -1;
|
|
|
|
/* Clear CLINT IPI */
|
|
clint_ipi_clear(target_hart);
|
|
|
|
return 0;
|
|
}
|
|
|
|
int clint_cold_ipi_init(unsigned long base, u32 hart_count)
|
|
{
|
|
/* Figure-out CLINT IPI register address */
|
|
clint_ipi_hart_count = hart_count;
|
|
clint_ipi_base = (void *)base;
|
|
clint_ipi = (u32 *)clint_ipi_base;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static u32 clint_time_hart_count;
|
|
static volatile void *clint_time_base;
|
|
static volatile u64 *clint_time_val;
|
|
static volatile u64 *clint_time_cmp;
|
|
|
|
u64 clint_timer_value(void)
|
|
{
|
|
return readq_relaxed(clint_time_val);
|
|
}
|
|
|
|
void clint_timer_event_stop(u32 target_hart)
|
|
{
|
|
if (clint_time_hart_count <= target_hart)
|
|
return;
|
|
|
|
/* Clear CLINT Time Compare */
|
|
writeq_relaxed(-1ULL, &clint_time_cmp[target_hart]);
|
|
}
|
|
|
|
void clint_timer_event_start(u32 target_hart, u64 next_event)
|
|
{
|
|
if (clint_time_hart_count <= target_hart)
|
|
return;
|
|
|
|
/* Program CLINT Time Compare */
|
|
writeq_relaxed(next_event, &clint_time_cmp[target_hart]);
|
|
}
|
|
|
|
int clint_warm_timer_init(u32 target_hart)
|
|
{
|
|
if (clint_time_hart_count <= target_hart ||
|
|
!clint_time_base)
|
|
return -1;
|
|
|
|
/* Clear CLINT Time Compare */
|
|
writeq_relaxed(-1ULL, &clint_time_cmp[target_hart]);
|
|
|
|
return 0;
|
|
}
|
|
|
|
int clint_cold_timer_init(unsigned long base, u32 hart_count)
|
|
{
|
|
/* Figure-out CLINT Time register address */
|
|
clint_time_hart_count = hart_count;
|
|
clint_time_base = (void *)base;
|
|
clint_time_val = (u64 *)(clint_time_base + 0xbff8);
|
|
clint_time_cmp = (u64 *)(clint_time_base + 0x4000);
|
|
|
|
return 0;
|
|
}
|