summaryrefslogtreecommitdiff
path: root/plat/common/sys
diff options
context:
space:
mode:
Diffstat (limited to 'plat/common/sys')
-rw-r--r--plat/common/sys/clint.c136
-rw-r--r--plat/common/sys/objects.mk10
2 files changed, 146 insertions, 0 deletions
diff --git a/plat/common/sys/clint.c b/plat/common/sys/clint.c
new file mode 100644
index 0000000..b6c6e9b
--- /dev/null
+++ b/plat/common/sys/clint.c
@@ -0,0 +1,136 @@
+/*
+ * 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;
+}
diff --git a/plat/common/sys/objects.mk b/plat/common/sys/objects.mk
new file mode 100644
index 0000000..451adbb
--- /dev/null
+++ b/plat/common/sys/objects.mk
@@ -0,0 +1,10 @@
+#
+# Copyright (c) 2018 Western Digital Corporation or its affiliates.
+#
+# Authors:
+# Anup Patel <anup.patel@wdc.com>
+#
+# SPDX-License-Identifier: BSD-2-Clause
+#
+
+plat-common-objs-$(PLAT_SYS_CLINT) += sys/clint.o