[rtems commit] bsp/riscv: Add and use riscv_fdt_get_address()

Sebastian Huber sebh at rtems.org
Wed Jul 25 08:11:10 UTC 2018


Module:    rtems
Branch:    master
Commit:    3a263a9b02227ebb056eef7bcbbc378cacbd476e
Changeset: http://git.rtems.org/rtems/commit/?id=3a263a9b02227ebb056eef7bcbbc378cacbd476e

Author:    Sebastian Huber <sebastian.huber at embedded-brains.de>
Date:      Thu Jul 19 14:38:44 2018 +0200

bsp/riscv: Add and use riscv_fdt_get_address()

Update #3433.

---

 bsps/include/bsp/fatal.h                  |  3 +-
 bsps/riscv/riscv/clock/clockdrv.c         | 46 +++++++++++++++++++++----------
 bsps/riscv/riscv/console/console-config.c | 12 ++------
 bsps/riscv/riscv/include/bsp/riscv.h      |  2 ++
 bsps/riscv/riscv/start/bspstart.c         | 43 +++++++++++++++++++++++++++++
 5 files changed, 81 insertions(+), 25 deletions(-)

diff --git a/bsps/include/bsp/fatal.h b/bsps/include/bsp/fatal.h
index 92ab55f..71edcb0 100644
--- a/bsps/include/bsp/fatal.h
+++ b/bsps/include/bsp/fatal.h
@@ -144,7 +144,8 @@ typedef enum {
   RISCV_FATAL_NO_NS16550_REG_IN_DEVICE_TREE,
   RISCV_FATAL_NO_NS16550_CLOCK_FREQUENCY_IN_DEVICE_TREE,
   RISCV_FATAL_UNEXPECTED_INTERRUPT_EXCEPTION,
-  RISCV_FATAL_CLOCK_IRQ_INSTALL
+  RISCV_FATAL_CLOCK_IRQ_INSTALL,
+  RISCV_FATAL_NO_CLINT_REG_IN_DEVICE_TREE
 } bsp_fatal_code;
 
 RTEMS_NO_RETURN static inline void
diff --git a/bsps/riscv/riscv/clock/clockdrv.c b/bsps/riscv/riscv/clock/clockdrv.c
index 52fb441..1420e31 100644
--- a/bsps/riscv/riscv/clock/clockdrv.c
+++ b/bsps/riscv/riscv/clock/clockdrv.c
@@ -35,30 +35,33 @@
 #include <rtems/timecounter.h>
 #include <rtems/score/riscv-utility.h>
 
-#include <bsp.h>
 #include <bsp/fatal.h>
 #include <bsp/fdt.h>
 #include <bsp/irq.h>
+#include <bsp/riscv.h>
 
 #include <dev/irq/clint.h>
 
 #include <libfdt.h>
 
-#define CLINT ((volatile clint_regs *) 0x02000000)
-
 /* This is defined in dev/clock/clockimpl.h */
 void Clock_isr(void *arg);
 
-static struct timecounter riscv_clock_tc;
+typedef struct {
+  struct timecounter base;
+  volatile clint_regs *clint;
+} riscv_timecounter;
+
+static riscv_timecounter riscv_clock_tc;
 
 static uint32_t riscv_clock_interval;
 
-static void riscv_clock_at_tick(void)
+static void riscv_clock_at_tick(riscv_timecounter *tc)
 {
   volatile clint_regs *clint;
   uint64_t cmp;
 
-  clint = CLINT;
+  clint = tc->clint;
 
   cmp = clint->mtimecmp[0].val_64;
   cmp += riscv_clock_interval;
@@ -88,11 +91,13 @@ static void riscv_clock_handler_install(void)
   }
 }
 
-static uint32_t riscv_clock_get_timecount(struct timecounter *tc)
+static uint32_t riscv_clock_get_timecount(struct timecounter *base)
 {
+  riscv_timecounter *tc;
   volatile clint_regs *clint;
 
-  clint = CLINT;
+  tc = (riscv_timecounter *) base;
+  clint = tc->clint;
   return clint->mtime.val_32[0];
 }
 
@@ -114,25 +119,36 @@ static uint32_t riscv_clock_get_timebase_frequency(const void *fdt)
 static void riscv_clock_initialize(void)
 {
   const char *fdt;
+  riscv_timecounter *tc;
+  int node;
   uint32_t tb_freq;
   uint64_t us_per_tick;
 
   fdt = bsp_fdt_get();
+  tc = &riscv_clock_tc;
+
+  node = fdt_node_offset_by_compatible(fdt, -1, "riscv,clint0");
+  tc->clint = riscv_fdt_get_address(fdt, node);
+
+  if (tc->clint == NULL) {
+    bsp_fatal(RISCV_FATAL_NO_CLINT_REG_IN_DEVICE_TREE);
+  }
+
   tb_freq = riscv_clock_get_timebase_frequency(fdt);
   us_per_tick = rtems_configuration_get_microseconds_per_tick();
   riscv_clock_interval = (uint32_t) ((tb_freq * us_per_tick) / 1000000);
 
-  riscv_clock_at_tick();
+  riscv_clock_at_tick(tc);
 
   /* Enable mtimer interrupts */
   set_csr(mie, MIP_MTIP);
 
   /* Initialize timecounter */
-  riscv_clock_tc.tc_get_timecount = riscv_clock_get_timecount;
-  riscv_clock_tc.tc_counter_mask = 0xffffffff;
-  riscv_clock_tc.tc_frequency = tb_freq;
-  riscv_clock_tc.tc_quality = RTEMS_TIMECOUNTER_QUALITY_CLOCK_DRIVER;
-  rtems_timecounter_install(&riscv_clock_tc);
+  tc->base.tc_get_timecount = riscv_clock_get_timecount;
+  tc->base.tc_counter_mask = 0xffffffff;
+  tc->base.tc_frequency = tb_freq;
+  tc->base.tc_quality = RTEMS_TIMECOUNTER_QUALITY_CLOCK_DRIVER;
+  rtems_timecounter_install(&tc->base);
 }
 
 uint32_t _CPU_Counter_frequency( void )
@@ -140,7 +156,7 @@ uint32_t _CPU_Counter_frequency( void )
   return riscv_clock_get_timebase_frequency(bsp_fdt_get());
 }
 
-#define Clock_driver_support_at_tick() riscv_clock_at_tick()
+#define Clock_driver_support_at_tick() riscv_clock_at_tick(&riscv_clock_tc)
 
 #define Clock_driver_support_initialize_hardware() riscv_clock_initialize()
 
diff --git a/bsps/riscv/riscv/console/console-config.c b/bsps/riscv/riscv/console/console-config.c
index 50d292d..270646f 100644
--- a/bsps/riscv/riscv/console/console-config.c
+++ b/bsps/riscv/riscv/console/console-config.c
@@ -17,10 +17,10 @@
 #include <rtems/sysinit.h>
 #include <rtems/termiostypes.h>
 
-#include <bsp.h>
 #include <bsp/fatal.h>
 #include <bsp/fdt.h>
 #include <bsp/irq.h>
+#include <bsp/riscv.h>
 
 #include <dev/serial/htif.h>
 #include <libchip/ns16550.h>
@@ -119,18 +119,12 @@ static void riscv_console_probe(void)
         ctx->set_reg = set_register;
         ctx->initial_baud = BSP_CONSOLE_BAUD;
 
-        val = (fdt32_t *) fdt_getprop(fdt, node, "reg", &len);
+        ctx->port = (uintptr_t) riscv_fdt_get_address(fdt, node);
 
-        if (val == NULL || (len != 8 && len != 16)) {
+        if (ctx->port == 0) {
           bsp_fatal(RISCV_FATAL_NO_NS16550_REG_IN_DEVICE_TREE);
         }
 
-        if (len == 16) {
-          ctx->port = fdt32_to_cpu(val[1]);
-        } else {
-          ctx->port = fdt32_to_cpu(val[0]);
-        }
-
         val = (fdt32_t *) fdt_getprop(fdt, node, "clock-frequency", &len);
 
         if (val == NULL || len != 4) {
diff --git a/bsps/riscv/riscv/include/bsp/riscv.h b/bsps/riscv/riscv/include/bsp/riscv.h
index b2b9d24..b85b250 100644
--- a/bsps/riscv/riscv/include/bsp/riscv.h
+++ b/bsps/riscv/riscv/include/bsp/riscv.h
@@ -32,6 +32,8 @@
 extern "C" {
 #endif
 
+void *riscv_fdt_get_address(const void *fdt, int node);
+
 #if RISCV_ENABLE_HTIF_SUPPORT != 0
 void htif_poweroff(void);
 #endif
diff --git a/bsps/riscv/riscv/start/bspstart.c b/bsps/riscv/riscv/start/bspstart.c
index 217e6f2..6b57977 100644
--- a/bsps/riscv/riscv/start/bspstart.c
+++ b/bsps/riscv/riscv/start/bspstart.c
@@ -25,6 +25,49 @@
 
 #include <bsp/bootcard.h>
 #include <bsp/irq-generic.h>
+#include <bsp/riscv.h>
+
+#include <libfdt.h>
+
+void *riscv_fdt_get_address(const void *fdt, int node)
+{
+  int parent;
+  int ac;
+  int len;
+  const uint32_t *reg;
+  uint64_t addr;
+
+  parent = fdt_parent_offset(fdt, node);
+  if (parent < 0) {
+    return NULL;
+  }
+
+  ac = fdt_address_cells(fdt, parent);
+  if (ac != 1 && ac != 2) {
+    return NULL;
+  }
+
+  reg = fdt_getprop(fdt, node, "reg", &len);
+  if (reg == NULL || len < ac) {
+    return NULL;
+  }
+
+  addr = 0;
+
+  while (ac > 0) {
+    addr = (addr << 32) | fdt32_to_cpu(*reg);
+    ++reg;
+    --ac;
+  }
+
+#if __riscv_xlen < 64
+  if (addr > 0xffffffff) {
+    return NULL;
+  }
+#endif
+
+  return (void *)(uintptr_t) addr;
+}
 
 void bsp_start(void)
 {



More information about the vc mailing list