From 7d1d5e77d8bd338360ce00e60c35a658ea9fdc67 Mon Sep 17 00:00:00 2001
From: kenrabold <kenrabold@hotmail.com>
Date: Mon, 16 Apr 2018 10:03:33 -0700
Subject: [PATCH] cpu/fe310: add RISC-V cpu FE310

New CPU FE310 from SiFive based on RISC-V architecture

build: add makefile for RISC-V builds

Makefile for builds using RISC-V tools
---
 cpu/fe310/Makefile                     |    7 +
 cpu/fe310/Makefile.dep                 |    3 +
 cpu/fe310/Makefile.features            |    2 +
 cpu/fe310/Makefile.include             |   10 +
 cpu/fe310/context_frame.c              |   83 ++
 cpu/fe310/cpu.c                        |  342 ++++++
 cpu/fe310/include/context_frame.h      |  129 +++
 cpu/fe310/include/cpu.h                |   57 +
 cpu/fe310/include/cpu_conf.h           |   43 +
 cpu/fe310/include/cpucycle.h           |   42 +
 cpu/fe310/include/periph_cpu.h         |   51 +
 cpu/fe310/include/vendor/LICENSE       |  206 ++++
 cpu/fe310/include/vendor/aon.h         |   88 ++
 cpu/fe310/include/vendor/clint.h       |   14 +
 cpu/fe310/include/vendor/encoding.h    | 1313 ++++++++++++++++++++++++
 cpu/fe310/include/vendor/gpio.h        |   24 +
 cpu/fe310/include/vendor/otp.h         |   23 +
 cpu/fe310/include/vendor/platform.h    |  129 +++
 cpu/fe310/include/vendor/plic.h        |   30 +
 cpu/fe310/include/vendor/plic_driver.h |   34 +
 cpu/fe310/include/vendor/prci.h        |   56 +
 cpu/fe310/include/vendor/prci_driver.h |   79 ++
 cpu/fe310/include/vendor/pwm.h         |   37 +
 cpu/fe310/include/vendor/spi.h         |   80 ++
 cpu/fe310/include/vendor/uart.h        |   31 +
 cpu/fe310/intr.S                       |  151 +++
 cpu/fe310/ldscripts/fe310.ld           |  182 ++++
 cpu/fe310/nano/Makefile                |    3 +
 cpu/fe310/nano/nanostubs.c             |  150 +++
 cpu/fe310/periph/Makefile              |    1 +
 cpu/fe310/periph/cpucycle.c            |   34 +
 cpu/fe310/periph/cpuid.c               |   38 +
 cpu/fe310/periph/gpio.c                |  214 ++++
 cpu/fe310/periph/pm.c                  |   33 +
 cpu/fe310/periph/rtc.c                 |  106 ++
 cpu/fe310/periph/rtt.c                 |  206 ++++
 cpu/fe310/periph/timer.c               |  171 +++
 cpu/fe310/periph/uart.c                |  170 +++
 cpu/fe310/start.S                      |   59 ++
 cpu/fe310/vendor/LICENSE               |  206 ++++
 cpu/fe310/vendor/Makefile              |    3 +
 cpu/fe310/vendor/plic_driver.c         |  153 +++
 cpu/fe310/vendor/prci_driver.c         |  268 +++++
 makefiles/arch/riscv.inc.mk            |   19 +
 44 files changed, 5080 insertions(+)
 create mode 100644 cpu/fe310/Makefile
 create mode 100644 cpu/fe310/Makefile.dep
 create mode 100644 cpu/fe310/Makefile.features
 create mode 100644 cpu/fe310/Makefile.include
 create mode 100644 cpu/fe310/context_frame.c
 create mode 100644 cpu/fe310/cpu.c
 create mode 100644 cpu/fe310/include/context_frame.h
 create mode 100644 cpu/fe310/include/cpu.h
 create mode 100644 cpu/fe310/include/cpu_conf.h
 create mode 100644 cpu/fe310/include/cpucycle.h
 create mode 100644 cpu/fe310/include/periph_cpu.h
 create mode 100644 cpu/fe310/include/vendor/LICENSE
 create mode 100644 cpu/fe310/include/vendor/aon.h
 create mode 100644 cpu/fe310/include/vendor/clint.h
 create mode 100644 cpu/fe310/include/vendor/encoding.h
 create mode 100644 cpu/fe310/include/vendor/gpio.h
 create mode 100644 cpu/fe310/include/vendor/otp.h
 create mode 100644 cpu/fe310/include/vendor/platform.h
 create mode 100644 cpu/fe310/include/vendor/plic.h
 create mode 100644 cpu/fe310/include/vendor/plic_driver.h
 create mode 100644 cpu/fe310/include/vendor/prci.h
 create mode 100644 cpu/fe310/include/vendor/prci_driver.h
 create mode 100644 cpu/fe310/include/vendor/pwm.h
 create mode 100644 cpu/fe310/include/vendor/spi.h
 create mode 100644 cpu/fe310/include/vendor/uart.h
 create mode 100644 cpu/fe310/intr.S
 create mode 100644 cpu/fe310/ldscripts/fe310.ld
 create mode 100644 cpu/fe310/nano/Makefile
 create mode 100644 cpu/fe310/nano/nanostubs.c
 create mode 100644 cpu/fe310/periph/Makefile
 create mode 100644 cpu/fe310/periph/cpucycle.c
 create mode 100644 cpu/fe310/periph/cpuid.c
 create mode 100644 cpu/fe310/periph/gpio.c
 create mode 100644 cpu/fe310/periph/pm.c
 create mode 100644 cpu/fe310/periph/rtc.c
 create mode 100644 cpu/fe310/periph/rtt.c
 create mode 100644 cpu/fe310/periph/timer.c
 create mode 100644 cpu/fe310/periph/uart.c
 create mode 100644 cpu/fe310/start.S
 create mode 100644 cpu/fe310/vendor/LICENSE
 create mode 100644 cpu/fe310/vendor/Makefile
 create mode 100644 cpu/fe310/vendor/plic_driver.c
 create mode 100644 cpu/fe310/vendor/prci_driver.c
 create mode 100644 makefiles/arch/riscv.inc.mk

diff --git a/cpu/fe310/Makefile b/cpu/fe310/Makefile
new file mode 100644
index 0000000000..6fdbf475a4
--- /dev/null
+++ b/cpu/fe310/Makefile
@@ -0,0 +1,7 @@
+# define the module that is built
+MODULE = cpu
+
+# add a list of subdirectories, that should also be built
+DIRS = periph nano vendor
+
+include $(RIOTBASE)/Makefile.base
diff --git a/cpu/fe310/Makefile.dep b/cpu/fe310/Makefile.dep
new file mode 100644
index 0000000000..2728a327db
--- /dev/null
+++ b/cpu/fe310/Makefile.dep
@@ -0,0 +1,3 @@
+ifneq (,$(filter periph_rtc,$(USEMODULE)))
+  USEMODULE += periph_rtt
+endif
diff --git a/cpu/fe310/Makefile.features b/cpu/fe310/Makefile.features
new file mode 100644
index 0000000000..fee8720798
--- /dev/null
+++ b/cpu/fe310/Makefile.features
@@ -0,0 +1,2 @@
+FEATURES_PROVIDED += periph_cpuid
+FEATURES_PROVIDED += periph_pm
diff --git a/cpu/fe310/Makefile.include b/cpu/fe310/Makefile.include
new file mode 100644
index 0000000000..e3afe4befc
--- /dev/null
+++ b/cpu/fe310/Makefile.include
@@ -0,0 +1,10 @@
+
+USEMODULE += newlib_nano
+
+USEMODULE += newlib_syscalls_fe310
+USEMODULE += sifive_drivers_fe310
+
+USEMODULE += periph
+USEMODULE += periph_pm
+
+include $(RIOTMAKE)/arch/riscv.inc.mk
diff --git a/cpu/fe310/context_frame.c b/cpu/fe310/context_frame.c
new file mode 100644
index 0000000000..e1b2a81bd9
--- /dev/null
+++ b/cpu/fe310/context_frame.c
@@ -0,0 +1,83 @@
+
+/*
+ * Copyright (C) 2017 JP Bonn
+ *
+ * This file is subject to the terms and conditions of the GNU Lesser
+ * General Public License v2.1. See the file LICENSE in the top level
+ * directory for more details.
+ */
+
+/**
+ * @ingroup     cpu_fe310
+ * @ingroup     cpu
+ * @{
+ *
+ * @file
+ * @brief       compile-time check of context_switch_frame offsets
+ *
+ * @author      JP Bonn
+ * @}
+ */
+
+#include "context_frame.h"
+#include "thread.h"
+
+
+#define CHECK_OFFSET(member)     \
+    _Static_assert(offsetof(struct context_switch_frame, member) == member ## _OFFSET, \
+                   "context_switch_frame offset mismatch for offset member");
+
+static void check_context_switch_frame_alignment(void) __attribute__ ((unused));
+
+/**
+ * @brief   Check size and offsets of context_switch_frame
+ *
+ * This does nothing at runtime.  It is optimized out since it's only
+ * doing compile-time checks.
+ */
+static void check_context_switch_frame_alignment(void)
+{
+    _Static_assert(sizeof(struct context_switch_frame) % 16 == 0,
+                   "Stack pointer should be 16 byte aligned");
+    _Static_assert(sizeof(struct context_switch_frame) == CONTEXT_FRAME_SIZE,
+                   "context_switch_frame size mismatch");
+    CHECK_OFFSET(filler0);
+    CHECK_OFFSET(filler1);
+    CHECK_OFFSET(pc);
+    CHECK_OFFSET(s0);
+    CHECK_OFFSET(s1);
+    CHECK_OFFSET(s2);
+    CHECK_OFFSET(s3);
+    CHECK_OFFSET(s4);
+    CHECK_OFFSET(s5);
+    CHECK_OFFSET(s6);
+    CHECK_OFFSET(s7);
+    CHECK_OFFSET(s8);
+    CHECK_OFFSET(s9);
+    CHECK_OFFSET(s10);
+    CHECK_OFFSET(s11);
+    CHECK_OFFSET(ra);
+    CHECK_OFFSET(tp);
+    CHECK_OFFSET(t0);
+    CHECK_OFFSET(t1);
+    CHECK_OFFSET(t2);
+    CHECK_OFFSET(t3);
+    CHECK_OFFSET(t4);
+    CHECK_OFFSET(t5);
+    CHECK_OFFSET(t6);
+    CHECK_OFFSET(a0);
+    CHECK_OFFSET(a1);
+    CHECK_OFFSET(a2);
+    CHECK_OFFSET(a3);
+    CHECK_OFFSET(a4);
+    CHECK_OFFSET(a5);
+    CHECK_OFFSET(a6);
+    CHECK_OFFSET(a7);
+
+    /*
+     * also check the SP offset in the _frame structure
+     */
+    _Static_assert( offsetof(struct _thread, sp) == SP_OFFSET_IN_THREAD,
+                    "Offset of SP in _thread mismatch");
+
+}
diff --git a/cpu/fe310/cpu.c b/cpu/fe310/cpu.c
new file mode 100644
index 0000000000..d145738c5b
--- /dev/null
+++ b/cpu/fe310/cpu.c
@@ -0,0 +1,342 @@
+/*
+ * Copyright (C) 2017 Ken Rabold, JP Bonn
+ *
+ * This file is subject to the terms and conditions of the GNU Lesser
+ * General Public License v2.1. See the file LICENSE in the top level
+ * directory for more details.
+ */
+
+/**
+ * @ingroup     cpu_fe310
+ * @{
+ *
+ * @file        cpu.c
+ * @brief       Implementation of the CPU initialization for SiFive FE310
+ *
+ * @author      Ken Rabold
+ * @}
+ */
+
+#include <stdio.h>
+#include <errno.h>
+
+#include "thread.h"
+#include "irq.h"
+#include "sched.h"
+#include "thread.h"
+#include "irq.h"
+#include "cpu.h"
+#include "context_frame.h"
+#include "periph_cpu.h"
+#include "panic.h"
+#include "vendor/encoding.h"
+#include "vendor/platform.h"
+#include "vendor/plic_driver.h"
+
+/* Default state of mstatus register */
+#define MSTATUS_DEFAULT     (MSTATUS_MPP | MSTATUS_MPIE)
+
+volatile int __in_isr = 0;
+
+void trap_entry(void);
+void thread_start(void);
+
+/* PLIC external ISR function list */
+static external_isr_ptr_t _ext_isrs[PLIC_NUM_INTERRUPTS];
+
+/* NULL interrupt handler */
+void null_isr(int num)
+{
+    (void) num;
+}
+
+/**
+ * @brief Initialize the CPU, set IRQ priorities, clocks
+ */
+void cpu_init(void)
+{
+    volatile uint64_t *mtimecmp =
+        (uint64_t *) (CLINT_CTRL_ADDR + CLINT_MTIMECMP);
+
+    /* Setup trap handler function */
+    write_csr(mtvec, &trap_entry);
+
+    /* Enable FPU if present */
+    if (read_csr(misa) & (1 << ('F' - 'A'))) {
+        write_csr(mstatus, MSTATUS_FS); /* allow FPU instructions without trapping */
+        write_csr(fcsr, 0);             /* initialize rounding mode, undefined at reset */
+    }
+
+    /* Clear all interrupt enables */
+    write_csr(mie, 0);
+
+    /* Initial PLIC external interrupt controller */
+    PLIC_init(PLIC_CTRL_ADDR, PLIC_NUM_INTERRUPTS, PLIC_NUM_PRIORITIES);
+
+    /* Initialize ISR function list */
+    for (int i = 0; i < PLIC_NUM_INTERRUPTS; i++) {
+        _ext_isrs[i] = null_isr;
+    }
+
+    /* Set mtimecmp to largest value to avoid spurious timer interrupts */
+    *mtimecmp = 0xFFFFFFFFFFFFFFFF;
+
+    /* Enable SW, timer and external interrupts */
+    set_csr(mie, MIP_MSIP);
+    set_csr(mie, MIP_MTIP);
+    set_csr(mie, MIP_MEIP);
+
+    /*  Set default state of mstatus */
+    set_csr(mstatus, MSTATUS_DEFAULT);
+}
+
+/**
+ * @brief Enable all maskable interrupts
+ */
+unsigned int irq_enable(void)
+{
+    /* Enable all interrupts */
+    set_csr(mstatus, MSTATUS_MIE);
+    return read_csr(mstatus);
+}
+
+/**
+ * @brief Disable all maskable interrupts
+ */
+unsigned int irq_disable(void)
+{
+    unsigned int state = read_csr(mstatus);
+
+    /* Disable all interrupts */
+    clear_csr(mstatus, MSTATUS_MIE);
+    return state;
+}
+
+/**
+ * @brief Restore the state of the IRQ flags
+ */
+void irq_restore(unsigned int state)
+{
+    /* Restore all interrupts to given state */
+    write_csr(mstatus, state);
+}
+
+/**
+ * @brief See if the current context is inside an ISR
+ */
+int irq_is_in(void)
+{
+    return __in_isr;
+}
+
+/**
+ * @brief   Set External ISR callback
+ */
+void set_external_isr_cb(int intNum, external_isr_ptr_t cbFunc)
+{
+    if ((intNum > 0) && (intNum < PLIC_NUM_INTERRUPTS)) {
+        _ext_isrs[intNum] = cbFunc;
+    }
+}
+
+/**
+ * @brief External interrupt handler
+ */
+void external_isr(void)
+{
+    plic_source intNum = PLIC_claim_interrupt();
+
+    if ((intNum > 0) && (intNum < PLIC_NUM_INTERRUPTS)) {
+        _ext_isrs[intNum]((uint32_t) intNum);
+    }
+
+    PLIC_complete_interrupt(intNum);
+}
+
+/**
+ * @brief Global trap and interrupt handler
+ */
+void handle_trap(unsigned int mcause)
+{
+    /*  Tell RIOT to set sched_context_switch_request instead of
+     *  calling thread_yield(). */
+    __in_isr = 1;
+
+    /* Check for INT or TRAP */
+    if ((mcause & MCAUSE_INT) == MCAUSE_INT) {
+        /* Cause is an interrupt - determine type */
+        switch (mcause & MCAUSE_CAUSE) {
+#ifdef MODULE_PERIPH_TIMER
+            case IRQ_M_TIMER:
+                /* Handle timer interrupt */
+                timer_isr();
+                break;
+#endif
+            case IRQ_M_EXT:
+                /* Handle external interrupt */
+                external_isr();
+                break;
+
+            default:
+                /* Unknown interrupt */
+                core_panic(PANIC_GENERAL_ERROR, "Unhandled interrupt");
+                break;
+        }
+    }
+
+    /* ISR done - no more changes to thread states */
+    __in_isr = 0;
+}
+
+/**
+ * @brief   Noticeable marker marking the beginning of a stack segment
+ *
+ * This marker is used e.g. by *thread_start_threading* to identify the
+ * stacks beginning.
+ */
+#define STACK_MARKER                (0x77777777)
+
+/**
+ * @brief Initialize a thread's stack
+ *
+ * RIOT saves the tasks registers on the stack, not in the task control
+ * block.  thread_stack_init() is responsible for allocating space for
+ * the registers on the stack and adjusting the stack pointer to account for
+ * the saved registers.
+ *
+ * The stack_start parameter is the bottom of the stack (low address).  The
+ * return value is the top of stack: stack_start + stack_size - space reserved
+ * for thread context save - space reserved to align stack.
+ *
+ * thread_stack_init is called for each thread.
+ *
+ * RISCV ABI is here: https://github.com/riscv/riscv-elf-psabi-doc
+ * From ABI:
+ * The stack grows downwards and the stack pointer shall be aligned to a
+ * 128-bit boundary upon procedure entry, except for the RV32E ABI, where it
+ * need only be aligned to 32 bits. In the standard ABI, the stack pointer
+ * must remain aligned throughout procedure execution. Non-standard ABI code
+ * must realign the stack pointer prior to invoking standard ABI procedures.
+ * The operating system must realign the stack pointer prior to invoking a
+ * signal handler; hence, POSIX signal handlers need not realign the stack
+ * pointer. In systems that service interrupts using the interruptee's stack,
+ * the interrupt service routine must realign the stack pointer if linked
+ * with any code that uses a non-standard stack-alignment discipline, but
+ * need not realign the stack pointer if all code adheres to the standard ABI.
+ *
+ * @param[in] task_func     pointer to the thread's code
+ * @param[in] arg           argument to task_func
+ * @param[in] stack_start   pointer to the start address of the thread
+ * @param[in] stack_size    the maximum size of the stack
+ *
+ * @return                  pointer to the new top of the stack (128bit aligned)
+ *
+ */
+char *thread_stack_init(thread_task_func_t task_func,
+                             void *arg,
+                             void *stack_start,
+                             int stack_size)
+{
+    struct context_switch_frame *sf;
+    uint32_t *reg;
+    uint32_t *stk_top;
+
+    /* calculate the top of the stack */
+    stk_top = (uint32_t *)((uintptr_t)stack_start + stack_size);
+
+    /* Put a marker at the top of the stack.  This is used by
+     * thread_stack_print to determine where to stop dumping the
+     * stack.
+     */
+    stk_top--;
+    *stk_top = STACK_MARKER;
+
+    /* per ABI align stack pointer to 16 byte boundary. */
+    stk_top = (uint32_t *)(((uint32_t)stk_top) & ~((uint32_t)0xf));
+
+    /* reserve space for the stack frame. */
+    stk_top = (uint32_t *)((uint8_t *) stk_top - sizeof(*sf));
+
+    /* populate the stack frame with default values for starting the thread. */
+    sf = (struct context_switch_frame *) stk_top;
+
+    /* a7 is register with highest memory address in frame */
+    reg = &sf->a7;
+    while (reg != &sf->pc) {
+        *reg-- = 0;
+    }
+    sf->pc = (uint32_t) task_func;
+    sf->a0 = (uint32_t) arg;
+
+    /* if the thread exits go to sched_task_exit() */
+    sf->ra = (uint32_t) sched_task_exit;
+
+    return (char *) stk_top;
+}
+
+void thread_print_stack(void)
+{
+    int count = 0;
+    uint32_t *sp = (uint32_t *) sched_active_thread->sp;
+
+    printf("printing the current stack of thread %" PRIkernel_pid "\n",
+           thread_getpid());
+
+#ifdef DEVELHELP
+    printf("thread name: %s\n", sched_active_thread->name);
+    printf("stack start: 0x%08x\n", (unsigned int)(sched_active_thread->stack_start));
+    printf("stack end  : 0x%08x\n", (unsigned int)(sched_active_thread->stack_start + sched_active_thread->stack_size));
+#endif
+
+    printf("  address:      data:\n");
+
+    do {
+        printf("  0x%08x:   0x%08x\n", (unsigned int) sp, (unsigned int) *sp);
+        sp++;
+        count++;
+    } while (*sp != STACK_MARKER);
+
+    printf("current stack size: %i words\n", count);
+}
+
+int thread_isr_stack_usage(void)
+{
+    return 0;
+}
+
+void *thread_isr_stack_pointer(void)
+{
+    return NULL;
+}
+
+void *thread_isr_stack_start(void)
+{
+    return NULL;
+}
+
+/**
+ * @brief Start or resume threading by loading a threads initial information
+ * from the stack.
+ *
+ * This is called is two situations: 1) after the initial main and idle threads
+ * have been created and 2) when a thread exits.
+ *
+ * sched_active_thread is not valid when cpu_switch_context_exit() is
+ * called.  sched_run() must be called to determine the next valid thread.
+ * This is exploited in the context switch code.
+ */
+void cpu_switch_context_exit(void)
+{
+    /* enable interrupts */
+    irq_enable();
+
+    /* start the thread */
+    thread_yield();
+    UNREACHABLE();
+}
+
+void thread_yield_higher(void)
+{
+    /* Use SW intr to schedule context switch */
+    CLINT_REG(CLINT_MSIP) = 1;
+}
diff --git a/cpu/fe310/include/context_frame.h b/cpu/fe310/include/context_frame.h
new file mode 100644
index 0000000000..e8fc29210d
--- /dev/null
+++ b/cpu/fe310/include/context_frame.h
@@ -0,0 +1,129 @@
+
+/*
+ * Copyright (C) 2017 JP Bonn
+ *
+ * This file is subject to the terms and conditions of the GNU Lesser
+ * General Public License v2.1. See the file LICENSE in the top level
+ * directory for more details.
+ */
+
+/**
+ * @defgroup    cpu_fe310
+ * @ingroup     cpu
+ * @brief       Freedom E cpu
+ * @{
+ *
+ * @file
+ * @brief       Thread context frame stored on stack.
+ *
+ * @author      JP Bonn
+ */
+
+#ifndef CONTEXT_FRAME_H
+#define CONTEXT_FRAME_H
+
+#if !defined(__ASSEMBLER__)
+#include <stdint.h>
+#include <assert.h>
+#endif /* __ASSEMBLER__ */
+
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+/**
+ * @brief   context_switch_frame stores the registers and PC for a context switch.
+ *
+ * This also defines context_switch_frame offsets for assembly language.  The
+ * structure is sized to maintain 16 byte stack alignment per the ABI.
+ * https://github.com/riscv/riscv-elf-psabi-doc
+ *
+ */
+#if !defined(__ASSEMBLER__)
+
+/* N.B.: update the definitions below if this changes */
+struct context_switch_frame {
+    uint32_t filler0;   /* filler to maintain 16 byte alignment */
+    uint32_t filler1;   /* filler to maintain 16 byte alignment */
+    uint32_t pc;
+    /* Callee saved registers */
+    uint32_t s0;
+    uint32_t s1;
+    uint32_t s2;
+    uint32_t s3;
+    uint32_t s4;
+    uint32_t s5;
+    uint32_t s6;
+    uint32_t s7;
+    uint32_t s8;
+    uint32_t s9;
+    uint32_t s10;
+    uint32_t s11;
+    /* Caller saved register */
+    uint32_t ra;
+    uint32_t tp;
+    uint32_t t0;
+    uint32_t t1;
+    uint32_t t2;
+    uint32_t t3;
+    uint32_t t4;
+    uint32_t t5;
+    uint32_t t6;
+    uint32_t a0;
+    uint32_t a1;
+    uint32_t a2;
+    uint32_t a3;
+    uint32_t a4;
+    uint32_t a5;
+    uint32_t a6;
+    uint32_t a7;
+};
+#endif /* __ASSEMBLER__ */
+
+
+/* These values are checked for correctness in context_frame.c */
+#define filler0_OFFSET 0
+#define filler1_OFFSET 4
+#define pc_OFFSET     8
+#define s0_OFFSET     12
+#define s1_OFFSET     16
+#define s2_OFFSET     20
+#define s3_OFFSET     24
+#define s4_OFFSET     28
+#define s5_OFFSET     32
+#define s6_OFFSET     36
+#define s7_OFFSET     40
+#define s8_OFFSET     44
+#define s9_OFFSET     48
+#define s10_OFFSET    52
+#define s11_OFFSET    56
+#define ra_OFFSET     60
+#define tp_OFFSET     64
+#define t0_OFFSET     68
+#define t1_OFFSET     72
+#define t2_OFFSET     76
+#define t3_OFFSET     80
+#define t4_OFFSET     84
+#define t5_OFFSET     88
+#define t6_OFFSET     92
+#define a0_OFFSET     96
+#define a1_OFFSET     100
+#define a2_OFFSET     104
+#define a3_OFFSET     108
+#define a4_OFFSET     112
+#define a5_OFFSET     116
+#define a6_OFFSET     120
+#define a7_OFFSET     124
+
+#define CONTEXT_FRAME_SIZE (a7_OFFSET + 4)
+
+#define SP_OFFSET_IN_THREAD 0
+
+
+#ifdef __cplusplus
+}
+#endif
+
+#endif /* CONTEXT_FRAME_H */
+/** @} */
diff --git a/cpu/fe310/include/cpu.h b/cpu/fe310/include/cpu.h
new file mode 100644
index 0000000000..0da3944f65
--- /dev/null
+++ b/cpu/fe310/include/cpu.h
@@ -0,0 +1,57 @@
+/*
+ * Copyright (C) 2017 Ken Rabold
+ *
+ * This file is subject to the terms and conditions of the GNU Lesser
+ * General Public License v2.1. See the file LICENSE in the top level
+ * directory for more details.
+ */
+
+/**
+ * @defgroup    cpu_fe310
+ * @ingroup     cpu
+ * @brief       Common implementations and headers for RISC-V
+ * @{
+ *
+ * @file
+ * @brief       Basic definitions for the RISC-V common module
+ *
+ * When ever you want to do something hardware related, that is accessing MCUs
+ * registers, just include this file. It will then make sure that the MCU
+ * specific headers are included.
+ *
+ * @author      Ken Rabold
+ */
+
+#ifndef CPU_H
+#define CPU_H
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+/**
+ * @brief   Initialization of the CPU
+ */
+void cpu_init(void);
+
+/**
+ * @brief   Print the last instruction's address
+ *
+ * @todo:   Not supported
+ */
+static inline void cpu_print_last_instruction(void)
+{
+    /* This function must exist else RIOT won't compile */
+}
+
+/**
+ * @brief   Initialization of the Newlib-nano stub
+ */
+void nanostubs_init(void);
+
+#ifdef __cplusplus
+}
+#endif
+
+#endif /* CPU_H */
+/** @} */
diff --git a/cpu/fe310/include/cpu_conf.h b/cpu/fe310/include/cpu_conf.h
new file mode 100644
index 0000000000..0bf6bfbc06
--- /dev/null
+++ b/cpu/fe310/include/cpu_conf.h
@@ -0,0 +1,43 @@
+/*
+ * Copyright (C) 2017 Ken Rabold
+ *
+ * This file is subject to the terms and conditions of the GNU Lesser General
+ * Public License v2.1. See the file LICENSE in the top level directory for more
+ * details.
+ */
+
+/**
+ * @defgroup        cpu_fe310
+ * @ingroup         cpu
+ * @brief           CPU specific implementations for the SiFive FE310 cpu
+ * @{
+ *
+ * @file
+ * @brief           CPU specific configuration options
+ *
+ * @author          Ken Rabold
+ */
+
+#ifndef CPU_CONF_H
+#define CPU_CONF_H
+
+#ifndef THREAD_EXTRA_STACKSIZE_PRINTF
+#define THREAD_EXTRA_STACKSIZE_PRINTF   (256)
+#endif
+#ifndef THREAD_STACKSIZE_DEFAULT
+#define THREAD_STACKSIZE_DEFAULT        (1024)
+#endif
+#ifndef THREAD_STACKSIZE_IDLE
+#define THREAD_STACKSIZE_IDLE           (256)
+#endif
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+#ifdef __cplusplus
+}
+#endif
+
+#endif /* CPU_CONF_H */
+/** @} */
diff --git a/cpu/fe310/include/cpucycle.h b/cpu/fe310/include/cpucycle.h
new file mode 100644
index 0000000000..fb37d2a4b7
--- /dev/null
+++ b/cpu/fe310/include/cpucycle.h
@@ -0,0 +1,42 @@
+/*
+ * Copyright (C) 2017 JP Bonn
+ *
+ * This file is subject to the terms and conditions of the GNU Lesser General
+ * Public License v2.1. See the file LICENSE in the top level directory for more
+ * details.
+ */
+
+/**
+ * @ingroup         cpu_fe310
+ * @{
+ *
+ * @file
+ * @brief           Functions to read CPU cycle counter
+ *
+ * @author          JP Bonn
+ */
+
+#ifndef CPUCYCLE_H
+#define CPUCYCLE_H
+
+#include <stdint.h>
+
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+/**
+ * @brief   Returns a count of the number of clock cycles executed by the
+ *          processor core on which the hart is running from an arbitrary
+ *          start time in the past.
+ */
+uint64_t get_cycle_count(void);
+
+
+#ifdef __cplusplus
+}
+#endif
+
+#endif /* CPUCYCLE_H */
+/** @} */
diff --git a/cpu/fe310/include/periph_cpu.h b/cpu/fe310/include/periph_cpu.h
new file mode 100644
index 0000000000..02ce572e76
--- /dev/null
+++ b/cpu/fe310/include/periph_cpu.h
@@ -0,0 +1,51 @@
+/*
+ * Copyright (C) 2017 Ken Rabold
+ *
+ * This file is subject to the terms and conditions of the GNU Lesser General
+ * Public License v2.1. See the file LICENSE in the top level directory for more
+ * details.
+ */
+
+/**
+ * @ingroup         cpu_fe310
+ * @{
+ *
+ * @file
+ * @brief           CPU specific definitions for internal peripheral handling
+ *
+ * @author          Ken Rabold
+ */
+
+#ifndef PERIPH_CPU_H
+#define PERIPH_CPU_H
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+/**
+ * @brief   Length of the CPU_ID in octets
+ */
+#define CPUID_LEN           (12U)
+
+/**
+ * @brief   Timer ISR
+ */
+void timer_isr(void);
+
+/**
+ * @brief   External ISR callback
+ */
+typedef void (*external_isr_ptr_t)(int intNum);
+
+/**
+ * @brief   Set External ISR callback
+ */
+void set_external_isr_cb(int intNum, external_isr_ptr_t cbFunc);
+
+#ifdef __cplusplus
+}
+#endif
+
+#endif /* PERIPH_CPU_H */
+/** @} */
diff --git a/cpu/fe310/include/vendor/LICENSE b/cpu/fe310/include/vendor/LICENSE
new file mode 100644
index 0000000000..0b0b6c8fc6
--- /dev/null
+++ b/cpu/fe310/include/vendor/LICENSE
@@ -0,0 +1,206 @@
+
+This software, except as otherwise noted in subrepositories, 
+is licensed under the Apache 2 license, quoted below.
+
+
+                                 Apache License
+                           Version 2.0, January 2004
+                        http://www.apache.org/licenses/
+
+   TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION
+
+   1. Definitions.
+
+      "License" shall mean the terms and conditions for use, reproduction,
+      and distribution as defined by Sections 1 through 9 of this document.
+
+      "Licensor" shall mean the copyright owner or entity authorized by
+      the copyright owner that is granting the License.
+
+      "Legal Entity" shall mean the union of the acting entity and all
+      other entities that control, are controlled by, or are under common
+      control with that entity. For the purposes of this definition,
+      "control" means (i) the power, direct or indirect, to cause the
+      direction or management of such entity, whether by contract or
+      otherwise, or (ii) ownership of fifty percent (50%) or more of the
+      outstanding shares, or (iii) beneficial ownership of such entity.
+
+      "You" (or "Your") shall mean an individual or Legal Entity
+      exercising permissions granted by this License.
+
+      "Source" form shall mean the preferred form for making modifications,
+      including but not limited to software source code, documentation
+      source, and configuration files.
+
+      "Object" form shall mean any form resulting from mechanical
+      transformation or translation of a Source form, including but
+      not limited to compiled object code, generated documentation,
+      and conversions to other media types.
+
+      "Work" shall mean the work of authorship, whether in Source or
+      Object form, made available under the License, as indicated by a
+      copyright notice that is included in or attached to the work
+      (an example is provided in the Appendix below).
+
+      "Derivative Works" shall mean any work, whether in Source or Object
+      form, that is based on (or derived from) the Work and for which the
+      editorial revisions, annotations, elaborations, or other modifications
+      represent, as a whole, an original work of authorship. For the purposes
+      of this License, Derivative Works shall not include works that remain
+      separable from, or merely link (or bind by name) to the interfaces of,
+      the Work and Derivative Works thereof.
+
+      "Contribution" shall mean any work of authorship, including
+      the original version of the Work and any modifications or additions
+      to that Work or Derivative Works thereof, that is intentionally
+      submitted to Licensor for inclusion in the Work by the copyright owner
+      or by an individual or Legal Entity authorized to submit on behalf of
+      the copyright owner. For the purposes of this definition, "submitted"
+      means any form of electronic, verbal, or written communication sent
+      to the Licensor or its representatives, including but not limited to
+      communication on electronic mailing lists, source code control systems,
+      and issue tracking systems that are managed by, or on behalf of, the
+      Licensor for the purpose of discussing and improving the Work, but
+      excluding communication that is conspicuously marked or otherwise
+      designated in writing by the copyright owner as "Not a Contribution."
+
+      "Contributor" shall mean Licensor and any individual or Legal Entity
+      on behalf of whom a Contribution has been received by Licensor and
+      subsequently incorporated within the Work.
+
+   2. Grant of Copyright License. Subject to the terms and conditions of
+      this License, each Contributor hereby grants to You a perpetual,
+      worldwide, non-exclusive, no-charge, royalty-free, irrevocable
+      copyright license to reproduce, prepare Derivative Works of,
+      publicly display, publicly perform, sublicense, and distribute the
+      Work and such Derivative Works in Source or Object form.
+
+   3. Grant of Patent License. Subject to the terms and conditions of
+      this License, each Contributor hereby grants to You a perpetual,
+      worldwide, non-exclusive, no-charge, royalty-free, irrevocable
+      (except as stated in this section) patent license to make, have made,
+      use, offer to sell, sell, import, and otherwise transfer the Work,
+      where such license applies only to those patent claims licensable
+      by such Contributor that are necessarily infringed by their
+      Contribution(s) alone or by combination of their Contribution(s)
+      with the Work to which such Contribution(s) was submitted. If You
+      institute patent litigation against any entity (including a
+      cross-claim or counterclaim in a lawsuit) alleging that the Work
+      or a Contribution incorporated within the Work constitutes direct
+      or contributory patent infringement, then any patent licenses
+      granted to You under this License for that Work shall terminate
+      as of the date such litigation is filed.
+
+   4. Redistribution. You may reproduce and distribute copies of the
+      Work or Derivative Works thereof in any medium, with or without
+      modifications, and in Source or Object form, provided that You
+      meet the following conditions:
+
+      (a) You must give any other recipients of the Work or
+          Derivative Works a copy of this License; and
+
+      (b) You must cause any modified files to carry prominent notices
+          stating that You changed the files; and
+
+      (c) You must retain, in the Source form of any Derivative Works
+          that You distribute, all copyright, patent, trademark, and
+          attribution notices from the Source form of the Work,
+          excluding those notices that do not pertain to any part of
+          the Derivative Works; and
+
+      (d) If the Work includes a "NOTICE" text file as part of its
+          distribution, then any Derivative Works that You distribute must
+          include a readable copy of the attribution notices contained
+          within such NOTICE file, excluding those notices that do not
+          pertain to any part of the Derivative Works, in at least one
+          of the following places: within a NOTICE text file distributed
+          as part of the Derivative Works; within the Source form or
+          documentation, if provided along with the Derivative Works; or,
+          within a display generated by the Derivative Works, if and
+          wherever such third-party notices normally appear. The contents
+          of the NOTICE file are for informational purposes only and
+          do not modify the License. You may add Your own attribution
+          notices within Derivative Works that You distribute, alongside
+          or as an addendum to the NOTICE text from the Work, provided
+          that such additional attribution notices cannot be construed
+          as modifying the License.
+
+      You may add Your own copyright statement to Your modifications and
+      may provide additional or different license terms and conditions
+      for use, reproduction, or distribution of Your modifications, or
+      for any such Derivative Works as a whole, provided Your use,
+      reproduction, and distribution of the Work otherwise complies with
+      the conditions stated in this License.
+
+   5. Submission of Contributions. Unless You explicitly state otherwise,
+      any Contribution intentionally submitted for inclusion in the Work
+      by You to the Licensor shall be under the terms and conditions of
+      this License, without any additional terms or conditions.
+      Notwithstanding the above, nothing herein shall supersede or modify
+      the terms of any separate license agreement you may have executed
+      with Licensor regarding such Contributions.
+
+   6. Trademarks. This License does not grant permission to use the trade
+      names, trademarks, service marks, or product names of the Licensor,
+      except as required for reasonable and customary use in describing the
+      origin of the Work and reproducing the content of the NOTICE file.
+
+   7. Disclaimer of Warranty. Unless required by applicable law or
+      agreed to in writing, Licensor provides the Work (and each
+      Contributor provides its Contributions) on an "AS IS" BASIS,
+      WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
+      implied, including, without limitation, any warranties or conditions
+      of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A
+      PARTICULAR PURPOSE. You are solely responsible for determining the
+      appropriateness of using or redistributing the Work and assume any
+      risks associated with Your exercise of permissions under this License.
+
+   8. Limitation of Liability. In no event and under no legal theory,
+      whether in tort (including negligence), contract, or otherwise,
+      unless required by applicable law (such as deliberate and grossly
+      negligent acts) or agreed to in writing, shall any Contributor be
+      liable to You for damages, including any direct, indirect, special,
+      incidental, or consequential damages of any character arising as a
+      result of this License or out of the use or inability to use the
+      Work (including but not limited to damages for loss of goodwill,
+      work stoppage, computer failure or malfunction, or any and all
+      other commercial damages or losses), even if such Contributor
+      has been advised of the possibility of such damages.
+
+   9. Accepting Warranty or Additional Liability. While redistributing
+      the Work or Derivative Works thereof, You may choose to offer,
+      and charge a fee for, acceptance of support, warranty, indemnity,
+      or other liability obligations and/or rights consistent with this
+      License. However, in accepting such obligations, You may act only
+      on Your own behalf and on Your sole responsibility, not on behalf
+      of any other Contributor, and only if You agree to indemnify,
+      defend, and hold each Contributor harmless for any liability
+      incurred by, or claims asserted against, such Contributor by reason
+      of your accepting any such warranty or additional liability.
+
+   END OF TERMS AND CONDITIONS
+
+   APPENDIX: How to apply the Apache License to your work.
+
+      To apply the Apache License to your work, attach the following
+      boilerplate notice, with the fields enclosed by brackets "[]"
+      replaced with your own identifying information. (Don't include
+      the brackets!)  The text should be enclosed in the appropriate
+      comment syntax for the file format. We also recommend that a
+      file or class name and description of purpose be included on the
+      same "printed page" as the copyright notice for easier
+      identification within third-party archives.
+
+   Copyright 2016 SiFive, Inc.
+
+   Licensed under the Apache License, Version 2.0 (the "License");
+   you may not use this file except in compliance with the License.
+   You may obtain a copy of the License at
+
+       http://www.apache.org/licenses/LICENSE-2.0
+
+   Unless required by applicable law or agreed to in writing, software
+   distributed under the License is distributed on an "AS IS" BASIS,
+   WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+   See the License for the specific language governing permissions and
+   limitations under the License.
diff --git a/cpu/fe310/include/vendor/aon.h b/cpu/fe310/include/vendor/aon.h
new file mode 100644
index 0000000000..63f1db3cd6
--- /dev/null
+++ b/cpu/fe310/include/vendor/aon.h
@@ -0,0 +1,88 @@
+// See LICENSE for license details.
+
+#ifndef _SIFIVE_AON_H
+#define _SIFIVE_AON_H
+
+/* Register offsets */
+
+#define AON_WDOGCFG     0x000
+#define AON_WDOGCOUNT   0x008
+#define AON_WDOGS       0x010
+#define AON_WDOGFEED    0x018
+#define AON_WDOGKEY     0x01C
+#define AON_WDOGCMP     0x020
+
+#define AON_RTCCFG      0x040
+#define AON_RTCLO       0x048
+#define AON_RTCHI       0x04C
+#define AON_RTCS        0x050
+#define AON_RTCCMP      0x060
+
+#define AON_BACKUP0     0x080
+#define AON_BACKUP1     0x084
+#define AON_BACKUP2     0x088
+#define AON_BACKUP3     0x08C
+#define AON_BACKUP4     0x090
+#define AON_BACKUP5     0x094
+#define AON_BACKUP6     0x098
+#define AON_BACKUP7     0x09C
+#define AON_BACKUP8     0x0A0
+#define AON_BACKUP9     0x0A4
+#define AON_BACKUP10    0x0A8
+#define AON_BACKUP11    0x0AC
+#define AON_BACKUP12    0x0B0
+#define AON_BACKUP13    0x0B4
+#define AON_BACKUP14    0x0B8
+#define AON_BACKUP15    0x0BC
+
+#define AON_PMUWAKEUPI0 0x100
+#define AON_PMUWAKEUPI1 0x104
+#define AON_PMUWAKEUPI2 0x108
+#define AON_PMUWAKEUPI3 0x10C
+#define AON_PMUWAKEUPI4 0x110
+#define AON_PMUWAKEUPI5 0x114
+#define AON_PMUWAKEUPI6 0x118
+#define AON_PMUWAKEUPI7 0x11C
+#define AON_PMUSLEEPI0  0x120
+#define AON_PMUSLEEPI1  0x124
+#define AON_PMUSLEEPI2  0x128
+#define AON_PMUSLEEPI3  0x12C
+#define AON_PMUSLEEPI4  0x130
+#define AON_PMUSLEEPI5  0x134
+#define AON_PMUSLEEPI6  0x138
+#define AON_PMUSLEEPI7  0x13C
+#define AON_PMUIE       0x140
+#define AON_PMUCAUSE    0x144
+#define AON_PMUSLEEP    0x148
+#define AON_PMUKEY      0x14C
+
+#define AON_LFROSC      0x070
+/* Constants */
+
+#define AON_WDOGKEY_VALUE  0x51F15E
+#define AON_WDOGFEED_VALUE 0xD09F00D
+
+#define AON_WDOGCFG_SCALE       0x0000000F
+#define AON_WDOGCFG_RSTEN       0x00000100
+#define AON_WDOGCFG_ZEROCMP     0x00000200
+#define AON_WDOGCFG_ENALWAYS    0x00001000
+#define AON_WDOGCFG_ENCOREAWAKE 0x00002000
+#define AON_WDOGCFG_CMPIP       0x10000000
+
+#define AON_RTCCFG_SCALE     0x0000000F
+#define AON_RTCCFG_ENALWAYS  0x00001000
+#define AON_RTCCFG_CMPIP     0x10000000
+
+#define AON_WAKEUPCAUSE_RESET   0x00
+#define AON_WAKEUPCAUSE_RTC     0x01
+#define AON_WAKEUPCAUSE_DWAKEUP 0x02
+#define AON_WAKEUPCAUSE_AWAKEUP 0x03
+
+#define AON_RESETCAUSE_POWERON  0x0000
+#define AON_RESETCAUSE_EXTERNAL 0x0100
+#define AON_RESETCAUSE_WATCHDOG 0x0200
+
+#define AON_PMUCAUSE_WAKEUPCAUSE 0x00FF
+#define AON_PMUCAUSE_RESETCAUSE  0xFF00
+
+#endif /* _SIFIVE_AON_H */
diff --git a/cpu/fe310/include/vendor/clint.h b/cpu/fe310/include/vendor/clint.h
new file mode 100644
index 0000000000..cd3e0c7a34
--- /dev/null
+++ b/cpu/fe310/include/vendor/clint.h
@@ -0,0 +1,14 @@
+// See LICENSE for license details
+
+#ifndef _SIFIVE_CLINT_H
+#define _SIFIVE_CLINT_H
+
+
+#define CLINT_MSIP 0x0000
+#define CLINT_MSIP_size   0x4
+#define CLINT_MTIMECMP 0x4000
+#define CLINT_MTIMECMP_size 0x8
+#define CLINT_MTIME 0xBFF8
+#define CLINT_MTIME_size 0x8
+
+#endif /* _SIFIVE_CLINT_H */ 
diff --git a/cpu/fe310/include/vendor/encoding.h b/cpu/fe310/include/vendor/encoding.h
new file mode 100644
index 0000000000..2573eb73e3
--- /dev/null
+++ b/cpu/fe310/include/vendor/encoding.h
@@ -0,0 +1,1313 @@
+// See LICENSE for license details.
+
+#ifndef RISCV_CSR_ENCODING_H
+#define RISCV_CSR_ENCODING_H
+
+#define MSTATUS_UIE         0x00000001
+#define MSTATUS_SIE         0x00000002
+#define MSTATUS_HIE         0x00000004
+#define MSTATUS_MIE         0x00000008
+#define MSTATUS_UPIE        0x00000010
+#define MSTATUS_SPIE        0x00000020
+#define MSTATUS_HPIE        0x00000040
+#define MSTATUS_MPIE        0x00000080
+#define MSTATUS_SPP         0x00000100
+#define MSTATUS_HPP         0x00000600
+#define MSTATUS_MPP         0x00001800
+#define MSTATUS_FS          0x00006000
+#define MSTATUS_XS          0x00018000
+#define MSTATUS_MPRV        0x00020000
+#define MSTATUS_PUM         0x00040000
+#define MSTATUS_MXR         0x00080000
+#define MSTATUS_VM          0x1F000000
+#define MSTATUS32_SD        0x80000000
+#define MSTATUS64_SD        0x8000000000000000
+
+#define SSTATUS_UIE         0x00000001
+#define SSTATUS_SIE         0x00000002
+#define SSTATUS_UPIE        0x00000010
+#define SSTATUS_SPIE        0x00000020
+#define SSTATUS_SPP         0x00000100
+#define SSTATUS_FS          0x00006000
+#define SSTATUS_XS          0x00018000
+#define SSTATUS_PUM         0x00040000
+#define SSTATUS32_SD        0x80000000
+#define SSTATUS64_SD        0x8000000000000000
+
+#define DCSR_XDEBUGVER      (3U<<30)
+#define DCSR_NDRESET        (1<<29)
+#define DCSR_FULLRESET      (1<<28)
+#define DCSR_EBREAKM        (1<<15)
+#define DCSR_EBREAKH        (1<<14)
+#define DCSR_EBREAKS        (1<<13)
+#define DCSR_EBREAKU        (1<<12)
+#define DCSR_STOPCYCLE      (1<<10)
+#define DCSR_STOPTIME       (1<<9)
+#define DCSR_CAUSE          (7<<6)
+#define DCSR_DEBUGINT       (1<<5)
+#define DCSR_HALT           (1<<3)
+#define DCSR_STEP           (1<<2)
+#define DCSR_PRV            (3<<0)
+
+#define DCSR_CAUSE_NONE     0
+#define DCSR_CAUSE_SWBP     1
+#define DCSR_CAUSE_HWBP     2
+#define DCSR_CAUSE_DEBUGINT 3
+#define DCSR_CAUSE_STEP     4
+#define DCSR_CAUSE_HALT     5
+
+#define MCONTROL_TYPE(xlen)    (0xfULL<<((xlen)-4))
+#define MCONTROL_DMODE(xlen)   (1ULL<<((xlen)-5))
+#define MCONTROL_MASKMAX(xlen) (0x3fULL<<((xlen)-11))
+
+#define MCONTROL_SELECT     (1<<19)
+#define MCONTROL_TIMING     (1<<18)
+#define MCONTROL_ACTION     (0x3f<<12)
+#define MCONTROL_CHAIN      (1<<11)
+#define MCONTROL_MATCH      (0xf<<7)
+#define MCONTROL_M          (1<<6)
+#define MCONTROL_H          (1<<5)
+#define MCONTROL_S          (1<<4)
+#define MCONTROL_U          (1<<3)
+#define MCONTROL_EXECUTE    (1<<2)
+#define MCONTROL_STORE      (1<<1)
+#define MCONTROL_LOAD       (1<<0)
+
+#define MCONTROL_TYPE_NONE      0
+#define MCONTROL_TYPE_MATCH     2
+
+#define MCONTROL_ACTION_DEBUG_EXCEPTION   0
+#define MCONTROL_ACTION_DEBUG_MODE        1
+#define MCONTROL_ACTION_TRACE_START       2
+#define MCONTROL_ACTION_TRACE_STOP        3
+#define MCONTROL_ACTION_TRACE_EMIT        4
+
+#define MCONTROL_MATCH_EQUAL     0
+#define MCONTROL_MATCH_NAPOT     1
+#define MCONTROL_MATCH_GE        2
+#define MCONTROL_MATCH_LT        3
+#define MCONTROL_MATCH_MASK_LOW  4
+#define MCONTROL_MATCH_MASK_HIGH 5
+
+#define MIP_SSIP            (1 << IRQ_S_SOFT)
+#define MIP_HSIP            (1 << IRQ_H_SOFT)
+#define MIP_MSIP            (1 << IRQ_M_SOFT)
+#define MIP_STIP            (1 << IRQ_S_TIMER)
+#define MIP_HTIP            (1 << IRQ_H_TIMER)
+#define MIP_MTIP            (1 << IRQ_M_TIMER)
+#define MIP_SEIP            (1 << IRQ_S_EXT)
+#define MIP_HEIP            (1 << IRQ_H_EXT)
+#define MIP_MEIP            (1 << IRQ_M_EXT)
+
+#define SIP_SSIP MIP_SSIP
+#define SIP_STIP MIP_STIP
+
+#define PRV_U 0
+#define PRV_S 1
+#define PRV_H 2
+#define PRV_M 3
+
+#define VM_MBARE 0
+#define VM_MBB   1
+#define VM_MBBID 2
+#define VM_SV32  8
+#define VM_SV39  9
+#define VM_SV48  10
+
+#define IRQ_S_SOFT   1
+#define IRQ_H_SOFT   2
+#define IRQ_M_SOFT   3
+#define IRQ_S_TIMER  5
+#define IRQ_H_TIMER  6
+#define IRQ_M_TIMER  7
+#define IRQ_S_EXT    9
+#define IRQ_H_EXT    10
+#define IRQ_M_EXT    11
+#define IRQ_COP      12
+#define IRQ_HOST     13
+
+#define DEFAULT_RSTVEC     0x00001000
+#define DEFAULT_NMIVEC     0x00001004
+#define DEFAULT_MTVEC      0x00001010
+#define CONFIG_STRING_ADDR 0x0000100C
+#define EXT_IO_BASE        0x40000000
+#define DRAM_BASE          0x80000000
+
+// page table entry (PTE) fields
+#define PTE_V     0x001 // Valid
+#define PTE_R     0x002 // Read
+#define PTE_W     0x004 // Write
+#define PTE_X     0x008 // Execute
+#define PTE_U     0x010 // User
+#define PTE_G     0x020 // Global
+#define PTE_A     0x040 // Accessed
+#define PTE_D     0x080 // Dirty
+#define PTE_SOFT  0x300 // Reserved for Software
+
+#define PTE_PPN_SHIFT 10
+
+#define PTE_TABLE(PTE) (((PTE) & (PTE_V | PTE_R | PTE_W | PTE_X)) == PTE_V)
+
+#ifdef __riscv
+
+#ifdef __riscv64
+# define MSTATUS_SD MSTATUS64_SD
+# define SSTATUS_SD SSTATUS64_SD
+# define RISCV_PGLEVEL_BITS 9
+#else
+# define MSTATUS_SD MSTATUS32_SD
+# define SSTATUS_SD SSTATUS32_SD
+# define RISCV_PGLEVEL_BITS 10
+#endif
+#define RISCV_PGSHIFT 12
+#define RISCV_PGSIZE (1 << RISCV_PGSHIFT)
+
+#ifndef __ASSEMBLER__
+
+#ifdef __GNUC__
+
+#define read_csr(reg) ({ unsigned long __tmp; \
+  __asm__ volatile ("csrr %0, " #reg : "=r"(__tmp)); \
+  __tmp; })
+
+#define write_csr(reg, val) ({ \
+  if (__builtin_constant_p(val) && (unsigned long)(val) < 32) \
+    __asm__ volatile ("csrw " #reg ", %0" :: "i"(val)); \
+  else \
+    __asm__ volatile ("csrw " #reg ", %0" :: "r"(val)); })
+
+#define swap_csr(reg, val) ({ unsigned long __tmp; \
+  if (__builtin_constant_p(val) && (unsigned long)(val) < 32) \
+    __asm__ volatile ("csrrw %0, " #reg ", %1" : "=r"(__tmp) : "i"(val)); \
+  else \
+    __asm__ volatile ("csrrw %0, " #reg ", %1" : "=r"(__tmp) : "r"(val)); \
+  __tmp; })
+
+#define set_csr(reg, bit) ({ unsigned long __tmp; \
+  if (__builtin_constant_p(bit) && (unsigned long)(bit) < 32) \
+    __asm__ volatile ("csrrs %0, " #reg ", %1" : "=r"(__tmp) : "i"(bit)); \
+  else \
+    __asm__ volatile ("csrrs %0, " #reg ", %1" : "=r"(__tmp) : "r"(bit)); \
+  __tmp; })
+
+#define clear_csr(reg, bit) ({ unsigned long __tmp; \
+  if (__builtin_constant_p(bit) && (unsigned long)(bit) < 32) \
+    __asm__ volatile ("csrrc %0, " #reg ", %1" : "=r"(__tmp) : "i"(bit)); \
+  else \
+    __asm__ volatile ("csrrc %0, " #reg ", %1" : "=r"(__tmp) : "r"(bit)); \
+  __tmp; })
+
+#define rdtime() read_csr(time)
+#define rdcycle() read_csr(cycle)
+#define rdinstret() read_csr(instret)
+
+#endif
+
+#endif
+
+#endif
+
+#endif
+/* Automatically generated by parse-opcodes */
+#ifndef RISCV_ENCODING_H
+#define RISCV_ENCODING_H
+#define MATCH_BEQ 0x63
+#define MASK_BEQ  0x707f
+#define MATCH_BNE 0x1063
+#define MASK_BNE  0x707f
+#define MATCH_BLT 0x4063
+#define MASK_BLT  0x707f
+#define MATCH_BGE 0x5063
+#define MASK_BGE  0x707f
+#define MATCH_BLTU 0x6063
+#define MASK_BLTU  0x707f
+#define MATCH_BGEU 0x7063
+#define MASK_BGEU  0x707f
+#define MATCH_JALR 0x67
+#define MASK_JALR  0x707f
+#define MATCH_JAL 0x6f
+#define MASK_JAL  0x7f
+#define MATCH_LUI 0x37
+#define MASK_LUI  0x7f
+#define MATCH_AUIPC 0x17
+#define MASK_AUIPC  0x7f
+#define MATCH_ADDI 0x13
+#define MASK_ADDI  0x707f
+#define MATCH_SLLI 0x1013
+#define MASK_SLLI  0xfc00707f
+#define MATCH_SLTI 0x2013
+#define MASK_SLTI  0x707f
+#define MATCH_SLTIU 0x3013
+#define MASK_SLTIU  0x707f
+#define MATCH_XORI 0x4013
+#define MASK_XORI  0x707f
+#define MATCH_SRLI 0x5013
+#define MASK_SRLI  0xfc00707f
+#define MATCH_SRAI 0x40005013
+#define MASK_SRAI  0xfc00707f
+#define MATCH_ORI 0x6013
+#define MASK_ORI  0x707f
+#define MATCH_ANDI 0x7013
+#define MASK_ANDI  0x707f
+#define MATCH_ADD 0x33
+#define MASK_ADD  0xfe00707f
+#define MATCH_SUB 0x40000033
+#define MASK_SUB  0xfe00707f
+#define MATCH_SLL 0x1033
+#define MASK_SLL  0xfe00707f
+#define MATCH_SLT 0x2033
+#define MASK_SLT  0xfe00707f
+#define MATCH_SLTU 0x3033
+#define MASK_SLTU  0xfe00707f
+#define MATCH_XOR 0x4033
+#define MASK_XOR  0xfe00707f
+#define MATCH_SRL 0x5033
+#define MASK_SRL  0xfe00707f
+#define MATCH_SRA 0x40005033
+#define MASK_SRA  0xfe00707f
+#define MATCH_OR 0x6033
+#define MASK_OR  0xfe00707f
+#define MATCH_AND 0x7033
+#define MASK_AND  0xfe00707f
+#define MATCH_ADDIW 0x1b
+#define MASK_ADDIW  0x707f
+#define MATCH_SLLIW 0x101b
+#define MASK_SLLIW  0xfe00707f
+#define MATCH_SRLIW 0x501b
+#define MASK_SRLIW  0xfe00707f
+#define MATCH_SRAIW 0x4000501b
+#define MASK_SRAIW  0xfe00707f
+#define MATCH_ADDW 0x3b
+#define MASK_ADDW  0xfe00707f
+#define MATCH_SUBW 0x4000003b
+#define MASK_SUBW  0xfe00707f
+#define MATCH_SLLW 0x103b
+#define MASK_SLLW  0xfe00707f
+#define MATCH_SRLW 0x503b
+#define MASK_SRLW  0xfe00707f
+#define MATCH_SRAW 0x4000503b
+#define MASK_SRAW  0xfe00707f
+#define MATCH_LB 0x3
+#define MASK_LB  0x707f
+#define MATCH_LH 0x1003
+#define MASK_LH  0x707f
+#define MATCH_LW 0x2003
+#define MASK_LW  0x707f
+#define MATCH_LD 0x3003
+#define MASK_LD  0x707f
+#define MATCH_LBU 0x4003
+#define MASK_LBU  0x707f
+#define MATCH_LHU 0x5003
+#define MASK_LHU  0x707f
+#define MATCH_LWU 0x6003
+#define MASK_LWU  0x707f
+#define MATCH_SB 0x23
+#define MASK_SB  0x707f
+#define MATCH_SH 0x1023
+#define MASK_SH  0x707f
+#define MATCH_SW 0x2023
+#define MASK_SW  0x707f
+#define MATCH_SD 0x3023
+#define MASK_SD  0x707f
+#define MATCH_FENCE 0xf
+#define MASK_FENCE  0x707f
+#define MATCH_FENCE_I 0x100f
+#define MASK_FENCE_I  0x707f
+#define MATCH_MUL 0x2000033
+#define MASK_MUL  0xfe00707f
+#define MATCH_MULH 0x2001033
+#define MASK_MULH  0xfe00707f
+#define MATCH_MULHSU 0x2002033
+#define MASK_MULHSU  0xfe00707f
+#define MATCH_MULHU 0x2003033
+#define MASK_MULHU  0xfe00707f
+#define MATCH_DIV 0x2004033
+#define MASK_DIV  0xfe00707f
+#define MATCH_DIVU 0x2005033
+#define MASK_DIVU  0xfe00707f
+#define MATCH_REM 0x2006033
+#define MASK_REM  0xfe00707f
+#define MATCH_REMU 0x2007033
+#define MASK_REMU  0xfe00707f
+#define MATCH_MULW 0x200003b
+#define MASK_MULW  0xfe00707f
+#define MATCH_DIVW 0x200403b
+#define MASK_DIVW  0xfe00707f
+#define MATCH_DIVUW 0x200503b
+#define MASK_DIVUW  0xfe00707f
+#define MATCH_REMW 0x200603b
+#define MASK_REMW  0xfe00707f
+#define MATCH_REMUW 0x200703b
+#define MASK_REMUW  0xfe00707f
+#define MATCH_AMOADD_W 0x202f
+#define MASK_AMOADD_W  0xf800707f
+#define MATCH_AMOXOR_W 0x2000202f
+#define MASK_AMOXOR_W  0xf800707f
+#define MATCH_AMOOR_W 0x4000202f
+#define MASK_AMOOR_W  0xf800707f
+#define MATCH_AMOAND_W 0x6000202f
+#define MASK_AMOAND_W  0xf800707f
+#define MATCH_AMOMIN_W 0x8000202f
+#define MASK_AMOMIN_W  0xf800707f
+#define MATCH_AMOMAX_W 0xa000202f
+#define MASK_AMOMAX_W  0xf800707f
+#define MATCH_AMOMINU_W 0xc000202f
+#define MASK_AMOMINU_W  0xf800707f
+#define MATCH_AMOMAXU_W 0xe000202f
+#define MASK_AMOMAXU_W  0xf800707f
+#define MATCH_AMOSWAP_W 0x800202f
+#define MASK_AMOSWAP_W  0xf800707f
+#define MATCH_LR_W 0x1000202f
+#define MASK_LR_W  0xf9f0707f
+#define MATCH_SC_W 0x1800202f
+#define MASK_SC_W  0xf800707f
+#define MATCH_AMOADD_D 0x302f
+#define MASK_AMOADD_D  0xf800707f
+#define MATCH_AMOXOR_D 0x2000302f
+#define MASK_AMOXOR_D  0xf800707f
+#define MATCH_AMOOR_D 0x4000302f
+#define MASK_AMOOR_D  0xf800707f
+#define MATCH_AMOAND_D 0x6000302f
+#define MASK_AMOAND_D  0xf800707f
+#define MATCH_AMOMIN_D 0x8000302f
+#define MASK_AMOMIN_D  0xf800707f
+#define MATCH_AMOMAX_D 0xa000302f
+#define MASK_AMOMAX_D  0xf800707f
+#define MATCH_AMOMINU_D 0xc000302f
+#define MASK_AMOMINU_D  0xf800707f
+#define MATCH_AMOMAXU_D 0xe000302f
+#define MASK_AMOMAXU_D  0xf800707f
+#define MATCH_AMOSWAP_D 0x800302f
+#define MASK_AMOSWAP_D  0xf800707f
+#define MATCH_LR_D 0x1000302f
+#define MASK_LR_D  0xf9f0707f
+#define MATCH_SC_D 0x1800302f
+#define MASK_SC_D  0xf800707f
+#define MATCH_ECALL 0x73
+#define MASK_ECALL  0xffffffff
+#define MATCH_EBREAK 0x100073
+#define MASK_EBREAK  0xffffffff
+#define MATCH_URET 0x200073
+#define MASK_URET  0xffffffff
+#define MATCH_SRET 0x10200073
+#define MASK_SRET  0xffffffff
+#define MATCH_HRET 0x20200073
+#define MASK_HRET  0xffffffff
+#define MATCH_MRET 0x30200073
+#define MASK_MRET  0xffffffff
+#define MATCH_DRET 0x7b200073
+#define MASK_DRET  0xffffffff
+#define MATCH_SFENCE_VM 0x10400073
+#define MASK_SFENCE_VM  0xfff07fff
+#define MATCH_WFI 0x10500073
+#define MASK_WFI  0xffffffff
+#define MATCH_CSRRW 0x1073
+#define MASK_CSRRW  0x707f
+#define MATCH_CSRRS 0x2073
+#define MASK_CSRRS  0x707f
+#define MATCH_CSRRC 0x3073
+#define MASK_CSRRC  0x707f
+#define MATCH_CSRRWI 0x5073
+#define MASK_CSRRWI  0x707f
+#define MATCH_CSRRSI 0x6073
+#define MASK_CSRRSI  0x707f
+#define MATCH_CSRRCI 0x7073
+#define MASK_CSRRCI  0x707f
+#define MATCH_FADD_S 0x53
+#define MASK_FADD_S  0xfe00007f
+#define MATCH_FSUB_S 0x8000053
+#define MASK_FSUB_S  0xfe00007f
+#define MATCH_FMUL_S 0x10000053
+#define MASK_FMUL_S  0xfe00007f
+#define MATCH_FDIV_S 0x18000053
+#define MASK_FDIV_S  0xfe00007f
+#define MATCH_FSGNJ_S 0x20000053
+#define MASK_FSGNJ_S  0xfe00707f
+#define MATCH_FSGNJN_S 0x20001053
+#define MASK_FSGNJN_S  0xfe00707f
+#define MATCH_FSGNJX_S 0x20002053
+#define MASK_FSGNJX_S  0xfe00707f
+#define MATCH_FMIN_S 0x28000053
+#define MASK_FMIN_S  0xfe00707f
+#define MATCH_FMAX_S 0x28001053
+#define MASK_FMAX_S  0xfe00707f
+#define MATCH_FSQRT_S 0x58000053
+#define MASK_FSQRT_S  0xfff0007f
+#define MATCH_FADD_D 0x2000053
+#define MASK_FADD_D  0xfe00007f
+#define MATCH_FSUB_D 0xa000053
+#define MASK_FSUB_D  0xfe00007f
+#define MATCH_FMUL_D 0x12000053
+#define MASK_FMUL_D  0xfe00007f
+#define MATCH_FDIV_D 0x1a000053
+#define MASK_FDIV_D  0xfe00007f
+#define MATCH_FSGNJ_D 0x22000053
+#define MASK_FSGNJ_D  0xfe00707f
+#define MATCH_FSGNJN_D 0x22001053
+#define MASK_FSGNJN_D  0xfe00707f
+#define MATCH_FSGNJX_D 0x22002053
+#define MASK_FSGNJX_D  0xfe00707f
+#define MATCH_FMIN_D 0x2a000053
+#define MASK_FMIN_D  0xfe00707f
+#define MATCH_FMAX_D 0x2a001053
+#define MASK_FMAX_D  0xfe00707f
+#define MATCH_FCVT_S_D 0x40100053
+#define MASK_FCVT_S_D  0xfff0007f
+#define MATCH_FCVT_D_S 0x42000053
+#define MASK_FCVT_D_S  0xfff0007f
+#define MATCH_FSQRT_D 0x5a000053
+#define MASK_FSQRT_D  0xfff0007f
+#define MATCH_FLE_S 0xa0000053
+#define MASK_FLE_S  0xfe00707f
+#define MATCH_FLT_S 0xa0001053
+#define MASK_FLT_S  0xfe00707f
+#define MATCH_FEQ_S 0xa0002053
+#define MASK_FEQ_S  0xfe00707f
+#define MATCH_FLE_D 0xa2000053
+#define MASK_FLE_D  0xfe00707f
+#define MATCH_FLT_D 0xa2001053
+#define MASK_FLT_D  0xfe00707f
+#define MATCH_FEQ_D 0xa2002053
+#define MASK_FEQ_D  0xfe00707f
+#define MATCH_FCVT_W_S 0xc0000053
+#define MASK_FCVT_W_S  0xfff0007f
+#define MATCH_FCVT_WU_S 0xc0100053
+#define MASK_FCVT_WU_S  0xfff0007f
+#define MATCH_FCVT_L_S 0xc0200053
+#define MASK_FCVT_L_S  0xfff0007f
+#define MATCH_FCVT_LU_S 0xc0300053
+#define MASK_FCVT_LU_S  0xfff0007f
+#define MATCH_FMV_X_S 0xe0000053
+#define MASK_FMV_X_S  0xfff0707f
+#define MATCH_FCLASS_S 0xe0001053
+#define MASK_FCLASS_S  0xfff0707f
+#define MATCH_FCVT_W_D 0xc2000053
+#define MASK_FCVT_W_D  0xfff0007f
+#define MATCH_FCVT_WU_D 0xc2100053
+#define MASK_FCVT_WU_D  0xfff0007f
+#define MATCH_FCVT_L_D 0xc2200053
+#define MASK_FCVT_L_D  0xfff0007f
+#define MATCH_FCVT_LU_D 0xc2300053
+#define MASK_FCVT_LU_D  0xfff0007f
+#define MATCH_FMV_X_D 0xe2000053
+#define MASK_FMV_X_D  0xfff0707f
+#define MATCH_FCLASS_D 0xe2001053
+#define MASK_FCLASS_D  0xfff0707f
+#define MATCH_FCVT_S_W 0xd0000053
+#define MASK_FCVT_S_W  0xfff0007f
+#define MATCH_FCVT_S_WU 0xd0100053
+#define MASK_FCVT_S_WU  0xfff0007f
+#define MATCH_FCVT_S_L 0xd0200053
+#define MASK_FCVT_S_L  0xfff0007f
+#define MATCH_FCVT_S_LU 0xd0300053
+#define MASK_FCVT_S_LU  0xfff0007f
+#define MATCH_FMV_S_X 0xf0000053
+#define MASK_FMV_S_X  0xfff0707f
+#define MATCH_FCVT_D_W 0xd2000053
+#define MASK_FCVT_D_W  0xfff0007f
+#define MATCH_FCVT_D_WU 0xd2100053
+#define MASK_FCVT_D_WU  0xfff0007f
+#define MATCH_FCVT_D_L 0xd2200053
+#define MASK_FCVT_D_L  0xfff0007f
+#define MATCH_FCVT_D_LU 0xd2300053
+#define MASK_FCVT_D_LU  0xfff0007f
+#define MATCH_FMV_D_X 0xf2000053
+#define MASK_FMV_D_X  0xfff0707f
+#define MATCH_FLW 0x2007
+#define MASK_FLW  0x707f
+#define MATCH_FLD 0x3007
+#define MASK_FLD  0x707f
+#define MATCH_FSW 0x2027
+#define MASK_FSW  0x707f
+#define MATCH_FSD 0x3027
+#define MASK_FSD  0x707f
+#define MATCH_FMADD_S 0x43
+#define MASK_FMADD_S  0x600007f
+#define MATCH_FMSUB_S 0x47
+#define MASK_FMSUB_S  0x600007f
+#define MATCH_FNMSUB_S 0x4b
+#define MASK_FNMSUB_S  0x600007f
+#define MATCH_FNMADD_S 0x4f
+#define MASK_FNMADD_S  0x600007f
+#define MATCH_FMADD_D 0x2000043
+#define MASK_FMADD_D  0x600007f
+#define MATCH_FMSUB_D 0x2000047
+#define MASK_FMSUB_D  0x600007f
+#define MATCH_FNMSUB_D 0x200004b
+#define MASK_FNMSUB_D  0x600007f
+#define MATCH_FNMADD_D 0x200004f
+#define MASK_FNMADD_D  0x600007f
+#define MATCH_C_NOP 0x1
+#define MASK_C_NOP  0xffff
+#define MATCH_C_ADDI16SP 0x6101
+#define MASK_C_ADDI16SP  0xef83
+#define MATCH_C_JR 0x8002
+#define MASK_C_JR  0xf07f
+#define MATCH_C_JALR 0x9002
+#define MASK_C_JALR  0xf07f
+#define MATCH_C_EBREAK 0x9002
+#define MASK_C_EBREAK  0xffff
+#define MATCH_C_LD 0x6000
+#define MASK_C_LD  0xe003
+#define MATCH_C_SD 0xe000
+#define MASK_C_SD  0xe003
+#define MATCH_C_ADDIW 0x2001
+#define MASK_C_ADDIW  0xe003
+#define MATCH_C_LDSP 0x6002
+#define MASK_C_LDSP  0xe003
+#define MATCH_C_SDSP 0xe002
+#define MASK_C_SDSP  0xe003
+#define MATCH_C_ADDI4SPN 0x0
+#define MASK_C_ADDI4SPN  0xe003
+#define MATCH_C_FLD 0x2000
+#define MASK_C_FLD  0xe003
+#define MATCH_C_LW 0x4000
+#define MASK_C_LW  0xe003
+#define MATCH_C_FLW 0x6000
+#define MASK_C_FLW  0xe003
+#define MATCH_C_FSD 0xa000
+#define MASK_C_FSD  0xe003
+#define MATCH_C_SW 0xc000
+#define MASK_C_SW  0xe003
+#define MATCH_C_FSW 0xe000
+#define MASK_C_FSW  0xe003
+#define MATCH_C_ADDI 0x1
+#define MASK_C_ADDI  0xe003
+#define MATCH_C_JAL 0x2001
+#define MASK_C_JAL  0xe003
+#define MATCH_C_LI 0x4001
+#define MASK_C_LI  0xe003
+#define MATCH_C_LUI 0x6001
+#define MASK_C_LUI  0xe003
+#define MATCH_C_SRLI 0x8001
+#define MASK_C_SRLI  0xec03
+#define MATCH_C_SRAI 0x8401
+#define MASK_C_SRAI  0xec03
+#define MATCH_C_ANDI 0x8801
+#define MASK_C_ANDI  0xec03
+#define MATCH_C_SUB 0x8c01
+#define MASK_C_SUB  0xfc63
+#define MATCH_C_XOR 0x8c21
+#define MASK_C_XOR  0xfc63
+#define MATCH_C_OR 0x8c41
+#define MASK_C_OR  0xfc63
+#define MATCH_C_AND 0x8c61
+#define MASK_C_AND  0xfc63
+#define MATCH_C_SUBW 0x9c01
+#define MASK_C_SUBW  0xfc63
+#define MATCH_C_ADDW 0x9c21
+#define MASK_C_ADDW  0xfc63
+#define MATCH_C_J 0xa001
+#define MASK_C_J  0xe003
+#define MATCH_C_BEQZ 0xc001
+#define MASK_C_BEQZ  0xe003
+#define MATCH_C_BNEZ 0xe001
+#define MASK_C_BNEZ  0xe003
+#define MATCH_C_SLLI 0x2
+#define MASK_C_SLLI  0xe003
+#define MATCH_C_FLDSP 0x2002
+#define MASK_C_FLDSP  0xe003
+#define MATCH_C_LWSP 0x4002
+#define MASK_C_LWSP  0xe003
+#define MATCH_C_FLWSP 0x6002
+#define MASK_C_FLWSP  0xe003
+#define MATCH_C_MV 0x8002
+#define MASK_C_MV  0xf003
+#define MATCH_C_ADD 0x9002
+#define MASK_C_ADD  0xf003
+#define MATCH_C_FSDSP 0xa002
+#define MASK_C_FSDSP  0xe003
+#define MATCH_C_SWSP 0xc002
+#define MASK_C_SWSP  0xe003
+#define MATCH_C_FSWSP 0xe002
+#define MASK_C_FSWSP  0xe003
+#define MATCH_CUSTOM0 0xb
+#define MASK_CUSTOM0  0x707f
+#define MATCH_CUSTOM0_RS1 0x200b
+#define MASK_CUSTOM0_RS1  0x707f
+#define MATCH_CUSTOM0_RS1_RS2 0x300b
+#define MASK_CUSTOM0_RS1_RS2  0x707f
+#define MATCH_CUSTOM0_RD 0x400b
+#define MASK_CUSTOM0_RD  0x707f
+#define MATCH_CUSTOM0_RD_RS1 0x600b
+#define MASK_CUSTOM0_RD_RS1  0x707f
+#define MATCH_CUSTOM0_RD_RS1_RS2 0x700b
+#define MASK_CUSTOM0_RD_RS1_RS2  0x707f
+#define MATCH_CUSTOM1 0x2b
+#define MASK_CUSTOM1  0x707f
+#define MATCH_CUSTOM1_RS1 0x202b
+#define MASK_CUSTOM1_RS1  0x707f
+#define MATCH_CUSTOM1_RS1_RS2 0x302b
+#define MASK_CUSTOM1_RS1_RS2  0x707f
+#define MATCH_CUSTOM1_RD 0x402b
+#define MASK_CUSTOM1_RD  0x707f
+#define MATCH_CUSTOM1_RD_RS1 0x602b
+#define MASK_CUSTOM1_RD_RS1  0x707f
+#define MATCH_CUSTOM1_RD_RS1_RS2 0x702b
+#define MASK_CUSTOM1_RD_RS1_RS2  0x707f
+#define MATCH_CUSTOM2 0x5b
+#define MASK_CUSTOM2  0x707f
+#define MATCH_CUSTOM2_RS1 0x205b
+#define MASK_CUSTOM2_RS1  0x707f
+#define MATCH_CUSTOM2_RS1_RS2 0x305b
+#define MASK_CUSTOM2_RS1_RS2  0x707f
+#define MATCH_CUSTOM2_RD 0x405b
+#define MASK_CUSTOM2_RD  0x707f
+#define MATCH_CUSTOM2_RD_RS1 0x605b
+#define MASK_CUSTOM2_RD_RS1  0x707f
+#define MATCH_CUSTOM2_RD_RS1_RS2 0x705b
+#define MASK_CUSTOM2_RD_RS1_RS2  0x707f
+#define MATCH_CUSTOM3 0x7b
+#define MASK_CUSTOM3  0x707f
+#define MATCH_CUSTOM3_RS1 0x207b
+#define MASK_CUSTOM3_RS1  0x707f
+#define MATCH_CUSTOM3_RS1_RS2 0x307b
+#define MASK_CUSTOM3_RS1_RS2  0x707f
+#define MATCH_CUSTOM3_RD 0x407b
+#define MASK_CUSTOM3_RD  0x707f
+#define MATCH_CUSTOM3_RD_RS1 0x607b
+#define MASK_CUSTOM3_RD_RS1  0x707f
+#define MATCH_CUSTOM3_RD_RS1_RS2 0x707b
+#define MASK_CUSTOM3_RD_RS1_RS2  0x707f
+#define CSR_FFLAGS 0x1
+#define CSR_FRM 0x2
+#define CSR_FCSR 0x3
+#define CSR_CYCLE 0xc00
+#define CSR_TIME 0xc01
+#define CSR_INSTRET 0xc02
+#define CSR_HPMCOUNTER3 0xc03
+#define CSR_HPMCOUNTER4 0xc04
+#define CSR_HPMCOUNTER5 0xc05
+#define CSR_HPMCOUNTER6 0xc06
+#define CSR_HPMCOUNTER7 0xc07
+#define CSR_HPMCOUNTER8 0xc08
+#define CSR_HPMCOUNTER9 0xc09
+#define CSR_HPMCOUNTER10 0xc0a
+#define CSR_HPMCOUNTER11 0xc0b
+#define CSR_HPMCOUNTER12 0xc0c
+#define CSR_HPMCOUNTER13 0xc0d
+#define CSR_HPMCOUNTER14 0xc0e
+#define CSR_HPMCOUNTER15 0xc0f
+#define CSR_HPMCOUNTER16 0xc10
+#define CSR_HPMCOUNTER17 0xc11
+#define CSR_HPMCOUNTER18 0xc12
+#define CSR_HPMCOUNTER19 0xc13
+#define CSR_HPMCOUNTER20 0xc14
+#define CSR_HPMCOUNTER21 0xc15
+#define CSR_HPMCOUNTER22 0xc16
+#define CSR_HPMCOUNTER23 0xc17
+#define CSR_HPMCOUNTER24 0xc18
+#define CSR_HPMCOUNTER25 0xc19
+#define CSR_HPMCOUNTER26 0xc1a
+#define CSR_HPMCOUNTER27 0xc1b
+#define CSR_HPMCOUNTER28 0xc1c
+#define CSR_HPMCOUNTER29 0xc1d
+#define CSR_HPMCOUNTER30 0xc1e
+#define CSR_HPMCOUNTER31 0xc1f
+#define CSR_SSTATUS 0x100
+#define CSR_SIE 0x104
+#define CSR_STVEC 0x105
+#define CSR_SSCRATCH 0x140
+#define CSR_SEPC 0x141
+#define CSR_SCAUSE 0x142
+#define CSR_SBADADDR 0x143
+#define CSR_SIP 0x144
+#define CSR_SPTBR 0x180
+#define CSR_MSTATUS 0x300
+#define CSR_MISA 0x301
+#define CSR_MEDELEG 0x302
+#define CSR_MIDELEG 0x303
+#define CSR_MIE 0x304
+#define CSR_MTVEC 0x305
+#define CSR_MSCRATCH 0x340
+#define CSR_MEPC 0x341
+#define CSR_MCAUSE 0x342
+#define CSR_MBADADDR 0x343
+#define CSR_MIP 0x344
+#define CSR_TSELECT 0x7a0
+#define CSR_TDATA1 0x7a1
+#define CSR_TDATA2 0x7a2
+#define CSR_TDATA3 0x7a3
+#define CSR_DCSR 0x7b0
+#define CSR_DPC 0x7b1
+#define CSR_DSCRATCH 0x7b2
+#define CSR_MCYCLE 0xb00
+#define CSR_MINSTRET 0xb02
+#define CSR_MHPMCOUNTER3 0xb03
+#define CSR_MHPMCOUNTER4 0xb04
+#define CSR_MHPMCOUNTER5 0xb05
+#define CSR_MHPMCOUNTER6 0xb06
+#define CSR_MHPMCOUNTER7 0xb07
+#define CSR_MHPMCOUNTER8 0xb08
+#define CSR_MHPMCOUNTER9 0xb09
+#define CSR_MHPMCOUNTER10 0xb0a
+#define CSR_MHPMCOUNTER11 0xb0b
+#define CSR_MHPMCOUNTER12 0xb0c
+#define CSR_MHPMCOUNTER13 0xb0d
+#define CSR_MHPMCOUNTER14 0xb0e
+#define CSR_MHPMCOUNTER15 0xb0f
+#define CSR_MHPMCOUNTER16 0xb10
+#define CSR_MHPMCOUNTER17 0xb11
+#define CSR_MHPMCOUNTER18 0xb12
+#define CSR_MHPMCOUNTER19 0xb13
+#define CSR_MHPMCOUNTER20 0xb14
+#define CSR_MHPMCOUNTER21 0xb15
+#define CSR_MHPMCOUNTER22 0xb16
+#define CSR_MHPMCOUNTER23 0xb17
+#define CSR_MHPMCOUNTER24 0xb18
+#define CSR_MHPMCOUNTER25 0xb19
+#define CSR_MHPMCOUNTER26 0xb1a
+#define CSR_MHPMCOUNTER27 0xb1b
+#define CSR_MHPMCOUNTER28 0xb1c
+#define CSR_MHPMCOUNTER29 0xb1d
+#define CSR_MHPMCOUNTER30 0xb1e
+#define CSR_MHPMCOUNTER31 0xb1f
+#define CSR_MUCOUNTEREN 0x320
+#define CSR_MSCOUNTEREN 0x321
+#define CSR_MHPMEVENT3 0x323
+#define CSR_MHPMEVENT4 0x324
+#define CSR_MHPMEVENT5 0x325
+#define CSR_MHPMEVENT6 0x326
+#define CSR_MHPMEVENT7 0x327
+#define CSR_MHPMEVENT8 0x328
+#define CSR_MHPMEVENT9 0x329
+#define CSR_MHPMEVENT10 0x32a
+#define CSR_MHPMEVENT11 0x32b
+#define CSR_MHPMEVENT12 0x32c
+#define CSR_MHPMEVENT13 0x32d
+#define CSR_MHPMEVENT14 0x32e
+#define CSR_MHPMEVENT15 0x32f
+#define CSR_MHPMEVENT16 0x330
+#define CSR_MHPMEVENT17 0x331
+#define CSR_MHPMEVENT18 0x332
+#define CSR_MHPMEVENT19 0x333
+#define CSR_MHPMEVENT20 0x334
+#define CSR_MHPMEVENT21 0x335
+#define CSR_MHPMEVENT22 0x336
+#define CSR_MHPMEVENT23 0x337
+#define CSR_MHPMEVENT24 0x338
+#define CSR_MHPMEVENT25 0x339
+#define CSR_MHPMEVENT26 0x33a
+#define CSR_MHPMEVENT27 0x33b
+#define CSR_MHPMEVENT28 0x33c
+#define CSR_MHPMEVENT29 0x33d
+#define CSR_MHPMEVENT30 0x33e
+#define CSR_MHPMEVENT31 0x33f
+#define CSR_MVENDORID 0xf11
+#define CSR_MARCHID 0xf12
+#define CSR_MIMPID 0xf13
+#define CSR_MHARTID 0xf14
+#define CSR_CYCLEH 0xc80
+#define CSR_TIMEH 0xc81
+#define CSR_INSTRETH 0xc82
+#define CSR_HPMCOUNTER3H 0xc83
+#define CSR_HPMCOUNTER4H 0xc84
+#define CSR_HPMCOUNTER5H 0xc85
+#define CSR_HPMCOUNTER6H 0xc86
+#define CSR_HPMCOUNTER7H 0xc87
+#define CSR_HPMCOUNTER8H 0xc88
+#define CSR_HPMCOUNTER9H 0xc89
+#define CSR_HPMCOUNTER10H 0xc8a
+#define CSR_HPMCOUNTER11H 0xc8b
+#define CSR_HPMCOUNTER12H 0xc8c
+#define CSR_HPMCOUNTER13H 0xc8d
+#define CSR_HPMCOUNTER14H 0xc8e
+#define CSR_HPMCOUNTER15H 0xc8f
+#define CSR_HPMCOUNTER16H 0xc90
+#define CSR_HPMCOUNTER17H 0xc91
+#define CSR_HPMCOUNTER18H 0xc92
+#define CSR_HPMCOUNTER19H 0xc93
+#define CSR_HPMCOUNTER20H 0xc94
+#define CSR_HPMCOUNTER21H 0xc95
+#define CSR_HPMCOUNTER22H 0xc96
+#define CSR_HPMCOUNTER23H 0xc97
+#define CSR_HPMCOUNTER24H 0xc98
+#define CSR_HPMCOUNTER25H 0xc99
+#define CSR_HPMCOUNTER26H 0xc9a
+#define CSR_HPMCOUNTER27H 0xc9b
+#define CSR_HPMCOUNTER28H 0xc9c
+#define CSR_HPMCOUNTER29H 0xc9d
+#define CSR_HPMCOUNTER30H 0xc9e
+#define CSR_HPMCOUNTER31H 0xc9f
+#define CSR_MCYCLEH 0xb80
+#define CSR_MINSTRETH 0xb82
+#define CSR_MHPMCOUNTER3H 0xb83
+#define CSR_MHPMCOUNTER4H 0xb84
+#define CSR_MHPMCOUNTER5H 0xb85
+#define CSR_MHPMCOUNTER6H 0xb86
+#define CSR_MHPMCOUNTER7H 0xb87
+#define CSR_MHPMCOUNTER8H 0xb88
+#define CSR_MHPMCOUNTER9H 0xb89
+#define CSR_MHPMCOUNTER10H 0xb8a
+#define CSR_MHPMCOUNTER11H 0xb8b
+#define CSR_MHPMCOUNTER12H 0xb8c
+#define CSR_MHPMCOUNTER13H 0xb8d
+#define CSR_MHPMCOUNTER14H 0xb8e
+#define CSR_MHPMCOUNTER15H 0xb8f
+#define CSR_MHPMCOUNTER16H 0xb90
+#define CSR_MHPMCOUNTER17H 0xb91
+#define CSR_MHPMCOUNTER18H 0xb92
+#define CSR_MHPMCOUNTER19H 0xb93
+#define CSR_MHPMCOUNTER20H 0xb94
+#define CSR_MHPMCOUNTER21H 0xb95
+#define CSR_MHPMCOUNTER22H 0xb96
+#define CSR_MHPMCOUNTER23H 0xb97
+#define CSR_MHPMCOUNTER24H 0xb98
+#define CSR_MHPMCOUNTER25H 0xb99
+#define CSR_MHPMCOUNTER26H 0xb9a
+#define CSR_MHPMCOUNTER27H 0xb9b
+#define CSR_MHPMCOUNTER28H 0xb9c
+#define CSR_MHPMCOUNTER29H 0xb9d
+#define CSR_MHPMCOUNTER30H 0xb9e
+#define CSR_MHPMCOUNTER31H 0xb9f
+#define CAUSE_MISALIGNED_FETCH 0x0
+#define CAUSE_FAULT_FETCH 0x1
+#define CAUSE_ILLEGAL_INSTRUCTION 0x2
+#define CAUSE_BREAKPOINT 0x3
+#define CAUSE_MISALIGNED_LOAD 0x4
+#define CAUSE_FAULT_LOAD 0x5
+#define CAUSE_MISALIGNED_STORE 0x6
+#define CAUSE_FAULT_STORE 0x7
+#define CAUSE_USER_ECALL 0x8
+#define CAUSE_SUPERVISOR_ECALL 0x9
+#define CAUSE_HYPERVISOR_ECALL 0xa
+#define CAUSE_MACHINE_ECALL 0xb
+#endif
+#ifdef DECLARE_INSN
+DECLARE_INSN(beq, MATCH_BEQ, MASK_BEQ)
+DECLARE_INSN(bne, MATCH_BNE, MASK_BNE)
+DECLARE_INSN(blt, MATCH_BLT, MASK_BLT)
+DECLARE_INSN(bge, MATCH_BGE, MASK_BGE)
+DECLARE_INSN(bltu, MATCH_BLTU, MASK_BLTU)
+DECLARE_INSN(bgeu, MATCH_BGEU, MASK_BGEU)
+DECLARE_INSN(jalr, MATCH_JALR, MASK_JALR)
+DECLARE_INSN(jal, MATCH_JAL, MASK_JAL)
+DECLARE_INSN(lui, MATCH_LUI, MASK_LUI)
+DECLARE_INSN(auipc, MATCH_AUIPC, MASK_AUIPC)
+DECLARE_INSN(addi, MATCH_ADDI, MASK_ADDI)
+DECLARE_INSN(slli, MATCH_SLLI, MASK_SLLI)
+DECLARE_INSN(slti, MATCH_SLTI, MASK_SLTI)
+DECLARE_INSN(sltiu, MATCH_SLTIU, MASK_SLTIU)
+DECLARE_INSN(xori, MATCH_XORI, MASK_XORI)
+DECLARE_INSN(srli, MATCH_SRLI, MASK_SRLI)
+DECLARE_INSN(srai, MATCH_SRAI, MASK_SRAI)
+DECLARE_INSN(ori, MATCH_ORI, MASK_ORI)
+DECLARE_INSN(andi, MATCH_ANDI, MASK_ANDI)
+DECLARE_INSN(add, MATCH_ADD, MASK_ADD)
+DECLARE_INSN(sub, MATCH_SUB, MASK_SUB)
+DECLARE_INSN(sll, MATCH_SLL, MASK_SLL)
+DECLARE_INSN(slt, MATCH_SLT, MASK_SLT)
+DECLARE_INSN(sltu, MATCH_SLTU, MASK_SLTU)
+DECLARE_INSN(xor, MATCH_XOR, MASK_XOR)
+DECLARE_INSN(srl, MATCH_SRL, MASK_SRL)
+DECLARE_INSN(sra, MATCH_SRA, MASK_SRA)
+DECLARE_INSN(or, MATCH_OR, MASK_OR)
+DECLARE_INSN(and, MATCH_AND, MASK_AND)
+DECLARE_INSN(addiw, MATCH_ADDIW, MASK_ADDIW)
+DECLARE_INSN(slliw, MATCH_SLLIW, MASK_SLLIW)
+DECLARE_INSN(srliw, MATCH_SRLIW, MASK_SRLIW)
+DECLARE_INSN(sraiw, MATCH_SRAIW, MASK_SRAIW)
+DECLARE_INSN(addw, MATCH_ADDW, MASK_ADDW)
+DECLARE_INSN(subw, MATCH_SUBW, MASK_SUBW)
+DECLARE_INSN(sllw, MATCH_SLLW, MASK_SLLW)
+DECLARE_INSN(srlw, MATCH_SRLW, MASK_SRLW)
+DECLARE_INSN(sraw, MATCH_SRAW, MASK_SRAW)
+DECLARE_INSN(lb, MATCH_LB, MASK_LB)
+DECLARE_INSN(lh, MATCH_LH, MASK_LH)
+DECLARE_INSN(lw, MATCH_LW, MASK_LW)
+DECLARE_INSN(ld, MATCH_LD, MASK_LD)
+DECLARE_INSN(lbu, MATCH_LBU, MASK_LBU)
+DECLARE_INSN(lhu, MATCH_LHU, MASK_LHU)
+DECLARE_INSN(lwu, MATCH_LWU, MASK_LWU)
+DECLARE_INSN(sb, MATCH_SB, MASK_SB)
+DECLARE_INSN(sh, MATCH_SH, MASK_SH)
+DECLARE_INSN(sw, MATCH_SW, MASK_SW)
+DECLARE_INSN(sd, MATCH_SD, MASK_SD)
+DECLARE_INSN(fence, MATCH_FENCE, MASK_FENCE)
+DECLARE_INSN(fence_i, MATCH_FENCE_I, MASK_FENCE_I)
+DECLARE_INSN(mul, MATCH_MUL, MASK_MUL)
+DECLARE_INSN(mulh, MATCH_MULH, MASK_MULH)
+DECLARE_INSN(mulhsu, MATCH_MULHSU, MASK_MULHSU)
+DECLARE_INSN(mulhu, MATCH_MULHU, MASK_MULHU)
+DECLARE_INSN(div, MATCH_DIV, MASK_DIV)
+DECLARE_INSN(divu, MATCH_DIVU, MASK_DIVU)
+DECLARE_INSN(rem, MATCH_REM, MASK_REM)
+DECLARE_INSN(remu, MATCH_REMU, MASK_REMU)
+DECLARE_INSN(mulw, MATCH_MULW, MASK_MULW)
+DECLARE_INSN(divw, MATCH_DIVW, MASK_DIVW)
+DECLARE_INSN(divuw, MATCH_DIVUW, MASK_DIVUW)
+DECLARE_INSN(remw, MATCH_REMW, MASK_REMW)
+DECLARE_INSN(remuw, MATCH_REMUW, MASK_REMUW)
+DECLARE_INSN(amoadd_w, MATCH_AMOADD_W, MASK_AMOADD_W)
+DECLARE_INSN(amoxor_w, MATCH_AMOXOR_W, MASK_AMOXOR_W)
+DECLARE_INSN(amoor_w, MATCH_AMOOR_W, MASK_AMOOR_W)
+DECLARE_INSN(amoand_w, MATCH_AMOAND_W, MASK_AMOAND_W)
+DECLARE_INSN(amomin_w, MATCH_AMOMIN_W, MASK_AMOMIN_W)
+DECLARE_INSN(amomax_w, MATCH_AMOMAX_W, MASK_AMOMAX_W)
+DECLARE_INSN(amominu_w, MATCH_AMOMINU_W, MASK_AMOMINU_W)
+DECLARE_INSN(amomaxu_w, MATCH_AMOMAXU_W, MASK_AMOMAXU_W)
+DECLARE_INSN(amoswap_w, MATCH_AMOSWAP_W, MASK_AMOSWAP_W)
+DECLARE_INSN(lr_w, MATCH_LR_W, MASK_LR_W)
+DECLARE_INSN(sc_w, MATCH_SC_W, MASK_SC_W)
+DECLARE_INSN(amoadd_d, MATCH_AMOADD_D, MASK_AMOADD_D)
+DECLARE_INSN(amoxor_d, MATCH_AMOXOR_D, MASK_AMOXOR_D)
+DECLARE_INSN(amoor_d, MATCH_AMOOR_D, MASK_AMOOR_D)
+DECLARE_INSN(amoand_d, MATCH_AMOAND_D, MASK_AMOAND_D)
+DECLARE_INSN(amomin_d, MATCH_AMOMIN_D, MASK_AMOMIN_D)
+DECLARE_INSN(amomax_d, MATCH_AMOMAX_D, MASK_AMOMAX_D)
+DECLARE_INSN(amominu_d, MATCH_AMOMINU_D, MASK_AMOMINU_D)
+DECLARE_INSN(amomaxu_d, MATCH_AMOMAXU_D, MASK_AMOMAXU_D)
+DECLARE_INSN(amoswap_d, MATCH_AMOSWAP_D, MASK_AMOSWAP_D)
+DECLARE_INSN(lr_d, MATCH_LR_D, MASK_LR_D)
+DECLARE_INSN(sc_d, MATCH_SC_D, MASK_SC_D)
+DECLARE_INSN(ecall, MATCH_ECALL, MASK_ECALL)
+DECLARE_INSN(ebreak, MATCH_EBREAK, MASK_EBREAK)
+DECLARE_INSN(uret, MATCH_URET, MASK_URET)
+DECLARE_INSN(sret, MATCH_SRET, MASK_SRET)
+DECLARE_INSN(hret, MATCH_HRET, MASK_HRET)
+DECLARE_INSN(mret, MATCH_MRET, MASK_MRET)
+DECLARE_INSN(dret, MATCH_DRET, MASK_DRET)
+DECLARE_INSN(sfence_vm, MATCH_SFENCE_VM, MASK_SFENCE_VM)
+DECLARE_INSN(wfi, MATCH_WFI, MASK_WFI)
+DECLARE_INSN(csrrw, MATCH_CSRRW, MASK_CSRRW)
+DECLARE_INSN(csrrs, MATCH_CSRRS, MASK_CSRRS)
+DECLARE_INSN(csrrc, MATCH_CSRRC, MASK_CSRRC)
+DECLARE_INSN(csrrwi, MATCH_CSRRWI, MASK_CSRRWI)
+DECLARE_INSN(csrrsi, MATCH_CSRRSI, MASK_CSRRSI)
+DECLARE_INSN(csrrci, MATCH_CSRRCI, MASK_CSRRCI)
+DECLARE_INSN(fadd_s, MATCH_FADD_S, MASK_FADD_S)
+DECLARE_INSN(fsub_s, MATCH_FSUB_S, MASK_FSUB_S)
+DECLARE_INSN(fmul_s, MATCH_FMUL_S, MASK_FMUL_S)
+DECLARE_INSN(fdiv_s, MATCH_FDIV_S, MASK_FDIV_S)
+DECLARE_INSN(fsgnj_s, MATCH_FSGNJ_S, MASK_FSGNJ_S)
+DECLARE_INSN(fsgnjn_s, MATCH_FSGNJN_S, MASK_FSGNJN_S)
+DECLARE_INSN(fsgnjx_s, MATCH_FSGNJX_S, MASK_FSGNJX_S)
+DECLARE_INSN(fmin_s, MATCH_FMIN_S, MASK_FMIN_S)
+DECLARE_INSN(fmax_s, MATCH_FMAX_S, MASK_FMAX_S)
+DECLARE_INSN(fsqrt_s, MATCH_FSQRT_S, MASK_FSQRT_S)
+DECLARE_INSN(fadd_d, MATCH_FADD_D, MASK_FADD_D)
+DECLARE_INSN(fsub_d, MATCH_FSUB_D, MASK_FSUB_D)
+DECLARE_INSN(fmul_d, MATCH_FMUL_D, MASK_FMUL_D)
+DECLARE_INSN(fdiv_d, MATCH_FDIV_D, MASK_FDIV_D)
+DECLARE_INSN(fsgnj_d, MATCH_FSGNJ_D, MASK_FSGNJ_D)
+DECLARE_INSN(fsgnjn_d, MATCH_FSGNJN_D, MASK_FSGNJN_D)
+DECLARE_INSN(fsgnjx_d, MATCH_FSGNJX_D, MASK_FSGNJX_D)
+DECLARE_INSN(fmin_d, MATCH_FMIN_D, MASK_FMIN_D)
+DECLARE_INSN(fmax_d, MATCH_FMAX_D, MASK_FMAX_D)
+DECLARE_INSN(fcvt_s_d, MATCH_FCVT_S_D, MASK_FCVT_S_D)
+DECLARE_INSN(fcvt_d_s, MATCH_FCVT_D_S, MASK_FCVT_D_S)
+DECLARE_INSN(fsqrt_d, MATCH_FSQRT_D, MASK_FSQRT_D)
+DECLARE_INSN(fle_s, MATCH_FLE_S, MASK_FLE_S)
+DECLARE_INSN(flt_s, MATCH_FLT_S, MASK_FLT_S)
+DECLARE_INSN(feq_s, MATCH_FEQ_S, MASK_FEQ_S)
+DECLARE_INSN(fle_d, MATCH_FLE_D, MASK_FLE_D)
+DECLARE_INSN(flt_d, MATCH_FLT_D, MASK_FLT_D)
+DECLARE_INSN(feq_d, MATCH_FEQ_D, MASK_FEQ_D)
+DECLARE_INSN(fcvt_w_s, MATCH_FCVT_W_S, MASK_FCVT_W_S)
+DECLARE_INSN(fcvt_wu_s, MATCH_FCVT_WU_S, MASK_FCVT_WU_S)
+DECLARE_INSN(fcvt_l_s, MATCH_FCVT_L_S, MASK_FCVT_L_S)
+DECLARE_INSN(fcvt_lu_s, MATCH_FCVT_LU_S, MASK_FCVT_LU_S)
+DECLARE_INSN(fmv_x_s, MATCH_FMV_X_S, MASK_FMV_X_S)
+DECLARE_INSN(fclass_s, MATCH_FCLASS_S, MASK_FCLASS_S)
+DECLARE_INSN(fcvt_w_d, MATCH_FCVT_W_D, MASK_FCVT_W_D)
+DECLARE_INSN(fcvt_wu_d, MATCH_FCVT_WU_D, MASK_FCVT_WU_D)
+DECLARE_INSN(fcvt_l_d, MATCH_FCVT_L_D, MASK_FCVT_L_D)
+DECLARE_INSN(fcvt_lu_d, MATCH_FCVT_LU_D, MASK_FCVT_LU_D)
+DECLARE_INSN(fmv_x_d, MATCH_FMV_X_D, MASK_FMV_X_D)
+DECLARE_INSN(fclass_d, MATCH_FCLASS_D, MASK_FCLASS_D)
+DECLARE_INSN(fcvt_s_w, MATCH_FCVT_S_W, MASK_FCVT_S_W)
+DECLARE_INSN(fcvt_s_wu, MATCH_FCVT_S_WU, MASK_FCVT_S_WU)
+DECLARE_INSN(fcvt_s_l, MATCH_FCVT_S_L, MASK_FCVT_S_L)
+DECLARE_INSN(fcvt_s_lu, MATCH_FCVT_S_LU, MASK_FCVT_S_LU)
+DECLARE_INSN(fmv_s_x, MATCH_FMV_S_X, MASK_FMV_S_X)
+DECLARE_INSN(fcvt_d_w, MATCH_FCVT_D_W, MASK_FCVT_D_W)
+DECLARE_INSN(fcvt_d_wu, MATCH_FCVT_D_WU, MASK_FCVT_D_WU)
+DECLARE_INSN(fcvt_d_l, MATCH_FCVT_D_L, MASK_FCVT_D_L)
+DECLARE_INSN(fcvt_d_lu, MATCH_FCVT_D_LU, MASK_FCVT_D_LU)
+DECLARE_INSN(fmv_d_x, MATCH_FMV_D_X, MASK_FMV_D_X)
+DECLARE_INSN(flw, MATCH_FLW, MASK_FLW)
+DECLARE_INSN(fld, MATCH_FLD, MASK_FLD)
+DECLARE_INSN(fsw, MATCH_FSW, MASK_FSW)
+DECLARE_INSN(fsd, MATCH_FSD, MASK_FSD)
+DECLARE_INSN(fmadd_s, MATCH_FMADD_S, MASK_FMADD_S)
+DECLARE_INSN(fmsub_s, MATCH_FMSUB_S, MASK_FMSUB_S)
+DECLARE_INSN(fnmsub_s, MATCH_FNMSUB_S, MASK_FNMSUB_S)
+DECLARE_INSN(fnmadd_s, MATCH_FNMADD_S, MASK_FNMADD_S)
+DECLARE_INSN(fmadd_d, MATCH_FMADD_D, MASK_FMADD_D)
+DECLARE_INSN(fmsub_d, MATCH_FMSUB_D, MASK_FMSUB_D)
+DECLARE_INSN(fnmsub_d, MATCH_FNMSUB_D, MASK_FNMSUB_D)
+DECLARE_INSN(fnmadd_d, MATCH_FNMADD_D, MASK_FNMADD_D)
+DECLARE_INSN(c_nop, MATCH_C_NOP, MASK_C_NOP)
+DECLARE_INSN(c_addi16sp, MATCH_C_ADDI16SP, MASK_C_ADDI16SP)
+DECLARE_INSN(c_jr, MATCH_C_JR, MASK_C_JR)
+DECLARE_INSN(c_jalr, MATCH_C_JALR, MASK_C_JALR)
+DECLARE_INSN(c_ebreak, MATCH_C_EBREAK, MASK_C_EBREAK)
+DECLARE_INSN(c_ld, MATCH_C_LD, MASK_C_LD)
+DECLARE_INSN(c_sd, MATCH_C_SD, MASK_C_SD)
+DECLARE_INSN(c_addiw, MATCH_C_ADDIW, MASK_C_ADDIW)
+DECLARE_INSN(c_ldsp, MATCH_C_LDSP, MASK_C_LDSP)
+DECLARE_INSN(c_sdsp, MATCH_C_SDSP, MASK_C_SDSP)
+DECLARE_INSN(c_addi4spn, MATCH_C_ADDI4SPN, MASK_C_ADDI4SPN)
+DECLARE_INSN(c_fld, MATCH_C_FLD, MASK_C_FLD)
+DECLARE_INSN(c_lw, MATCH_C_LW, MASK_C_LW)
+DECLARE_INSN(c_flw, MATCH_C_FLW, MASK_C_FLW)
+DECLARE_INSN(c_fsd, MATCH_C_FSD, MASK_C_FSD)
+DECLARE_INSN(c_sw, MATCH_C_SW, MASK_C_SW)
+DECLARE_INSN(c_fsw, MATCH_C_FSW, MASK_C_FSW)
+DECLARE_INSN(c_addi, MATCH_C_ADDI, MASK_C_ADDI)
+DECLARE_INSN(c_jal, MATCH_C_JAL, MASK_C_JAL)
+DECLARE_INSN(c_li, MATCH_C_LI, MASK_C_LI)
+DECLARE_INSN(c_lui, MATCH_C_LUI, MASK_C_LUI)
+DECLARE_INSN(c_srli, MATCH_C_SRLI, MASK_C_SRLI)
+DECLARE_INSN(c_srai, MATCH_C_SRAI, MASK_C_SRAI)
+DECLARE_INSN(c_andi, MATCH_C_ANDI, MASK_C_ANDI)
+DECLARE_INSN(c_sub, MATCH_C_SUB, MASK_C_SUB)
+DECLARE_INSN(c_xor, MATCH_C_XOR, MASK_C_XOR)
+DECLARE_INSN(c_or, MATCH_C_OR, MASK_C_OR)
+DECLARE_INSN(c_and, MATCH_C_AND, MASK_C_AND)
+DECLARE_INSN(c_subw, MATCH_C_SUBW, MASK_C_SUBW)
+DECLARE_INSN(c_addw, MATCH_C_ADDW, MASK_C_ADDW)
+DECLARE_INSN(c_j, MATCH_C_J, MASK_C_J)
+DECLARE_INSN(c_beqz, MATCH_C_BEQZ, MASK_C_BEQZ)
+DECLARE_INSN(c_bnez, MATCH_C_BNEZ, MASK_C_BNEZ)
+DECLARE_INSN(c_slli, MATCH_C_SLLI, MASK_C_SLLI)
+DECLARE_INSN(c_fldsp, MATCH_C_FLDSP, MASK_C_FLDSP)
+DECLARE_INSN(c_lwsp, MATCH_C_LWSP, MASK_C_LWSP)
+DECLARE_INSN(c_flwsp, MATCH_C_FLWSP, MASK_C_FLWSP)
+DECLARE_INSN(c_mv, MATCH_C_MV, MASK_C_MV)
+DECLARE_INSN(c_add, MATCH_C_ADD, MASK_C_ADD)
+DECLARE_INSN(c_fsdsp, MATCH_C_FSDSP, MASK_C_FSDSP)
+DECLARE_INSN(c_swsp, MATCH_C_SWSP, MASK_C_SWSP)
+DECLARE_INSN(c_fswsp, MATCH_C_FSWSP, MASK_C_FSWSP)
+DECLARE_INSN(custom0, MATCH_CUSTOM0, MASK_CUSTOM0)
+DECLARE_INSN(custom0_rs1, MATCH_CUSTOM0_RS1, MASK_CUSTOM0_RS1)
+DECLARE_INSN(custom0_rs1_rs2, MATCH_CUSTOM0_RS1_RS2, MASK_CUSTOM0_RS1_RS2)
+DECLARE_INSN(custom0_rd, MATCH_CUSTOM0_RD, MASK_CUSTOM0_RD)
+DECLARE_INSN(custom0_rd_rs1, MATCH_CUSTOM0_RD_RS1, MASK_CUSTOM0_RD_RS1)
+DECLARE_INSN(custom0_rd_rs1_rs2, MATCH_CUSTOM0_RD_RS1_RS2, MASK_CUSTOM0_RD_RS1_RS2)
+DECLARE_INSN(custom1, MATCH_CUSTOM1, MASK_CUSTOM1)
+DECLARE_INSN(custom1_rs1, MATCH_CUSTOM1_RS1, MASK_CUSTOM1_RS1)
+DECLARE_INSN(custom1_rs1_rs2, MATCH_CUSTOM1_RS1_RS2, MASK_CUSTOM1_RS1_RS2)
+DECLARE_INSN(custom1_rd, MATCH_CUSTOM1_RD, MASK_CUSTOM1_RD)
+DECLARE_INSN(custom1_rd_rs1, MATCH_CUSTOM1_RD_RS1, MASK_CUSTOM1_RD_RS1)
+DECLARE_INSN(custom1_rd_rs1_rs2, MATCH_CUSTOM1_RD_RS1_RS2, MASK_CUSTOM1_RD_RS1_RS2)
+DECLARE_INSN(custom2, MATCH_CUSTOM2, MASK_CUSTOM2)
+DECLARE_INSN(custom2_rs1, MATCH_CUSTOM2_RS1, MASK_CUSTOM2_RS1)
+DECLARE_INSN(custom2_rs1_rs2, MATCH_CUSTOM2_RS1_RS2, MASK_CUSTOM2_RS1_RS2)
+DECLARE_INSN(custom2_rd, MATCH_CUSTOM2_RD, MASK_CUSTOM2_RD)
+DECLARE_INSN(custom2_rd_rs1, MATCH_CUSTOM2_RD_RS1, MASK_CUSTOM2_RD_RS1)
+DECLARE_INSN(custom2_rd_rs1_rs2, MATCH_CUSTOM2_RD_RS1_RS2, MASK_CUSTOM2_RD_RS1_RS2)
+DECLARE_INSN(custom3, MATCH_CUSTOM3, MASK_CUSTOM3)
+DECLARE_INSN(custom3_rs1, MATCH_CUSTOM3_RS1, MASK_CUSTOM3_RS1)
+DECLARE_INSN(custom3_rs1_rs2, MATCH_CUSTOM3_RS1_RS2, MASK_CUSTOM3_RS1_RS2)
+DECLARE_INSN(custom3_rd, MATCH_CUSTOM3_RD, MASK_CUSTOM3_RD)
+DECLARE_INSN(custom3_rd_rs1, MATCH_CUSTOM3_RD_RS1, MASK_CUSTOM3_RD_RS1)
+DECLARE_INSN(custom3_rd_rs1_rs2, MATCH_CUSTOM3_RD_RS1_RS2, MASK_CUSTOM3_RD_RS1_RS2)
+#endif
+#ifdef DECLARE_CSR
+DECLARE_CSR(fflags, CSR_FFLAGS)
+DECLARE_CSR(frm, CSR_FRM)
+DECLARE_CSR(fcsr, CSR_FCSR)
+DECLARE_CSR(cycle, CSR_CYCLE)
+DECLARE_CSR(time, CSR_TIME)
+DECLARE_CSR(instret, CSR_INSTRET)
+DECLARE_CSR(hpmcounter3, CSR_HPMCOUNTER3)
+DECLARE_CSR(hpmcounter4, CSR_HPMCOUNTER4)
+DECLARE_CSR(hpmcounter5, CSR_HPMCOUNTER5)
+DECLARE_CSR(hpmcounter6, CSR_HPMCOUNTER6)
+DECLARE_CSR(hpmcounter7, CSR_HPMCOUNTER7)
+DECLARE_CSR(hpmcounter8, CSR_HPMCOUNTER8)
+DECLARE_CSR(hpmcounter9, CSR_HPMCOUNTER9)
+DECLARE_CSR(hpmcounter10, CSR_HPMCOUNTER10)
+DECLARE_CSR(hpmcounter11, CSR_HPMCOUNTER11)
+DECLARE_CSR(hpmcounter12, CSR_HPMCOUNTER12)
+DECLARE_CSR(hpmcounter13, CSR_HPMCOUNTER13)
+DECLARE_CSR(hpmcounter14, CSR_HPMCOUNTER14)
+DECLARE_CSR(hpmcounter15, CSR_HPMCOUNTER15)
+DECLARE_CSR(hpmcounter16, CSR_HPMCOUNTER16)
+DECLARE_CSR(hpmcounter17, CSR_HPMCOUNTER17)
+DECLARE_CSR(hpmcounter18, CSR_HPMCOUNTER18)
+DECLARE_CSR(hpmcounter19, CSR_HPMCOUNTER19)
+DECLARE_CSR(hpmcounter20, CSR_HPMCOUNTER20)
+DECLARE_CSR(hpmcounter21, CSR_HPMCOUNTER21)
+DECLARE_CSR(hpmcounter22, CSR_HPMCOUNTER22)
+DECLARE_CSR(hpmcounter23, CSR_HPMCOUNTER23)
+DECLARE_CSR(hpmcounter24, CSR_HPMCOUNTER24)
+DECLARE_CSR(hpmcounter25, CSR_HPMCOUNTER25)
+DECLARE_CSR(hpmcounter26, CSR_HPMCOUNTER26)
+DECLARE_CSR(hpmcounter27, CSR_HPMCOUNTER27)
+DECLARE_CSR(hpmcounter28, CSR_HPMCOUNTER28)
+DECLARE_CSR(hpmcounter29, CSR_HPMCOUNTER29)
+DECLARE_CSR(hpmcounter30, CSR_HPMCOUNTER30)
+DECLARE_CSR(hpmcounter31, CSR_HPMCOUNTER31)
+DECLARE_CSR(sstatus, CSR_SSTATUS)
+DECLARE_CSR(sie, CSR_SIE)
+DECLARE_CSR(stvec, CSR_STVEC)
+DECLARE_CSR(sscratch, CSR_SSCRATCH)
+DECLARE_CSR(sepc, CSR_SEPC)
+DECLARE_CSR(scause, CSR_SCAUSE)
+DECLARE_CSR(sbadaddr, CSR_SBADADDR)
+DECLARE_CSR(sip, CSR_SIP)
+DECLARE_CSR(sptbr, CSR_SPTBR)
+DECLARE_CSR(mstatus, CSR_MSTATUS)
+DECLARE_CSR(misa, CSR_MISA)
+DECLARE_CSR(medeleg, CSR_MEDELEG)
+DECLARE_CSR(mideleg, CSR_MIDELEG)
+DECLARE_CSR(mie, CSR_MIE)
+DECLARE_CSR(mtvec, CSR_MTVEC)
+DECLARE_CSR(mscratch, CSR_MSCRATCH)
+DECLARE_CSR(mepc, CSR_MEPC)
+DECLARE_CSR(mcause, CSR_MCAUSE)
+DECLARE_CSR(mbadaddr, CSR_MBADADDR)
+DECLARE_CSR(mip, CSR_MIP)
+DECLARE_CSR(tselect, CSR_TSELECT)
+DECLARE_CSR(tdata1, CSR_TDATA1)
+DECLARE_CSR(tdata2, CSR_TDATA2)
+DECLARE_CSR(tdata3, CSR_TDATA3)
+DECLARE_CSR(dcsr, CSR_DCSR)
+DECLARE_CSR(dpc, CSR_DPC)
+DECLARE_CSR(dscratch, CSR_DSCRATCH)
+DECLARE_CSR(mcycle, CSR_MCYCLE)
+DECLARE_CSR(minstret, CSR_MINSTRET)
+DECLARE_CSR(mhpmcounter3, CSR_MHPMCOUNTER3)
+DECLARE_CSR(mhpmcounter4, CSR_MHPMCOUNTER4)
+DECLARE_CSR(mhpmcounter5, CSR_MHPMCOUNTER5)
+DECLARE_CSR(mhpmcounter6, CSR_MHPMCOUNTER6)
+DECLARE_CSR(mhpmcounter7, CSR_MHPMCOUNTER7)
+DECLARE_CSR(mhpmcounter8, CSR_MHPMCOUNTER8)
+DECLARE_CSR(mhpmcounter9, CSR_MHPMCOUNTER9)
+DECLARE_CSR(mhpmcounter10, CSR_MHPMCOUNTER10)
+DECLARE_CSR(mhpmcounter11, CSR_MHPMCOUNTER11)
+DECLARE_CSR(mhpmcounter12, CSR_MHPMCOUNTER12)
+DECLARE_CSR(mhpmcounter13, CSR_MHPMCOUNTER13)
+DECLARE_CSR(mhpmcounter14, CSR_MHPMCOUNTER14)
+DECLARE_CSR(mhpmcounter15, CSR_MHPMCOUNTER15)
+DECLARE_CSR(mhpmcounter16, CSR_MHPMCOUNTER16)
+DECLARE_CSR(mhpmcounter17, CSR_MHPMCOUNTER17)
+DECLARE_CSR(mhpmcounter18, CSR_MHPMCOUNTER18)
+DECLARE_CSR(mhpmcounter19, CSR_MHPMCOUNTER19)
+DECLARE_CSR(mhpmcounter20, CSR_MHPMCOUNTER20)
+DECLARE_CSR(mhpmcounter21, CSR_MHPMCOUNTER21)
+DECLARE_CSR(mhpmcounter22, CSR_MHPMCOUNTER22)
+DECLARE_CSR(mhpmcounter23, CSR_MHPMCOUNTER23)
+DECLARE_CSR(mhpmcounter24, CSR_MHPMCOUNTER24)
+DECLARE_CSR(mhpmcounter25, CSR_MHPMCOUNTER25)
+DECLARE_CSR(mhpmcounter26, CSR_MHPMCOUNTER26)
+DECLARE_CSR(mhpmcounter27, CSR_MHPMCOUNTER27)
+DECLARE_CSR(mhpmcounter28, CSR_MHPMCOUNTER28)
+DECLARE_CSR(mhpmcounter29, CSR_MHPMCOUNTER29)
+DECLARE_CSR(mhpmcounter30, CSR_MHPMCOUNTER30)
+DECLARE_CSR(mhpmcounter31, CSR_MHPMCOUNTER31)
+DECLARE_CSR(mucounteren, CSR_MUCOUNTEREN)
+DECLARE_CSR(mscounteren, CSR_MSCOUNTEREN)
+DECLARE_CSR(mhpmevent3, CSR_MHPMEVENT3)
+DECLARE_CSR(mhpmevent4, CSR_MHPMEVENT4)
+DECLARE_CSR(mhpmevent5, CSR_MHPMEVENT5)
+DECLARE_CSR(mhpmevent6, CSR_MHPMEVENT6)
+DECLARE_CSR(mhpmevent7, CSR_MHPMEVENT7)
+DECLARE_CSR(mhpmevent8, CSR_MHPMEVENT8)
+DECLARE_CSR(mhpmevent9, CSR_MHPMEVENT9)
+DECLARE_CSR(mhpmevent10, CSR_MHPMEVENT10)
+DECLARE_CSR(mhpmevent11, CSR_MHPMEVENT11)
+DECLARE_CSR(mhpmevent12, CSR_MHPMEVENT12)
+DECLARE_CSR(mhpmevent13, CSR_MHPMEVENT13)
+DECLARE_CSR(mhpmevent14, CSR_MHPMEVENT14)
+DECLARE_CSR(mhpmevent15, CSR_MHPMEVENT15)
+DECLARE_CSR(mhpmevent16, CSR_MHPMEVENT16)
+DECLARE_CSR(mhpmevent17, CSR_MHPMEVENT17)
+DECLARE_CSR(mhpmevent18, CSR_MHPMEVENT18)
+DECLARE_CSR(mhpmevent19, CSR_MHPMEVENT19)
+DECLARE_CSR(mhpmevent20, CSR_MHPMEVENT20)
+DECLARE_CSR(mhpmevent21, CSR_MHPMEVENT21)
+DECLARE_CSR(mhpmevent22, CSR_MHPMEVENT22)
+DECLARE_CSR(mhpmevent23, CSR_MHPMEVENT23)
+DECLARE_CSR(mhpmevent24, CSR_MHPMEVENT24)
+DECLARE_CSR(mhpmevent25, CSR_MHPMEVENT25)
+DECLARE_CSR(mhpmevent26, CSR_MHPMEVENT26)
+DECLARE_CSR(mhpmevent27, CSR_MHPMEVENT27)
+DECLARE_CSR(mhpmevent28, CSR_MHPMEVENT28)
+DECLARE_CSR(mhpmevent29, CSR_MHPMEVENT29)
+DECLARE_CSR(mhpmevent30, CSR_MHPMEVENT30)
+DECLARE_CSR(mhpmevent31, CSR_MHPMEVENT31)
+DECLARE_CSR(mvendorid, CSR_MVENDORID)
+DECLARE_CSR(marchid, CSR_MARCHID)
+DECLARE_CSR(mimpid, CSR_MIMPID)
+DECLARE_CSR(mhartid, CSR_MHARTID)
+DECLARE_CSR(cycleh, CSR_CYCLEH)
+DECLARE_CSR(timeh, CSR_TIMEH)
+DECLARE_CSR(instreth, CSR_INSTRETH)
+DECLARE_CSR(hpmcounter3h, CSR_HPMCOUNTER3H)
+DECLARE_CSR(hpmcounter4h, CSR_HPMCOUNTER4H)
+DECLARE_CSR(hpmcounter5h, CSR_HPMCOUNTER5H)
+DECLARE_CSR(hpmcounter6h, CSR_HPMCOUNTER6H)
+DECLARE_CSR(hpmcounter7h, CSR_HPMCOUNTER7H)
+DECLARE_CSR(hpmcounter8h, CSR_HPMCOUNTER8H)
+DECLARE_CSR(hpmcounter9h, CSR_HPMCOUNTER9H)
+DECLARE_CSR(hpmcounter10h, CSR_HPMCOUNTER10H)
+DECLARE_CSR(hpmcounter11h, CSR_HPMCOUNTER11H)
+DECLARE_CSR(hpmcounter12h, CSR_HPMCOUNTER12H)
+DECLARE_CSR(hpmcounter13h, CSR_HPMCOUNTER13H)
+DECLARE_CSR(hpmcounter14h, CSR_HPMCOUNTER14H)
+DECLARE_CSR(hpmcounter15h, CSR_HPMCOUNTER15H)
+DECLARE_CSR(hpmcounter16h, CSR_HPMCOUNTER16H)
+DECLARE_CSR(hpmcounter17h, CSR_HPMCOUNTER17H)
+DECLARE_CSR(hpmcounter18h, CSR_HPMCOUNTER18H)
+DECLARE_CSR(hpmcounter19h, CSR_HPMCOUNTER19H)
+DECLARE_CSR(hpmcounter20h, CSR_HPMCOUNTER20H)
+DECLARE_CSR(hpmcounter21h, CSR_HPMCOUNTER21H)
+DECLARE_CSR(hpmcounter22h, CSR_HPMCOUNTER22H)
+DECLARE_CSR(hpmcounter23h, CSR_HPMCOUNTER23H)
+DECLARE_CSR(hpmcounter24h, CSR_HPMCOUNTER24H)
+DECLARE_CSR(hpmcounter25h, CSR_HPMCOUNTER25H)
+DECLARE_CSR(hpmcounter26h, CSR_HPMCOUNTER26H)
+DECLARE_CSR(hpmcounter27h, CSR_HPMCOUNTER27H)
+DECLARE_CSR(hpmcounter28h, CSR_HPMCOUNTER28H)
+DECLARE_CSR(hpmcounter29h, CSR_HPMCOUNTER29H)
+DECLARE_CSR(hpmcounter30h, CSR_HPMCOUNTER30H)
+DECLARE_CSR(hpmcounter31h, CSR_HPMCOUNTER31H)
+DECLARE_CSR(mcycleh, CSR_MCYCLEH)
+DECLARE_CSR(minstreth, CSR_MINSTRETH)
+DECLARE_CSR(mhpmcounter3h, CSR_MHPMCOUNTER3H)
+DECLARE_CSR(mhpmcounter4h, CSR_MHPMCOUNTER4H)
+DECLARE_CSR(mhpmcounter5h, CSR_MHPMCOUNTER5H)
+DECLARE_CSR(mhpmcounter6h, CSR_MHPMCOUNTER6H)
+DECLARE_CSR(mhpmcounter7h, CSR_MHPMCOUNTER7H)
+DECLARE_CSR(mhpmcounter8h, CSR_MHPMCOUNTER8H)
+DECLARE_CSR(mhpmcounter9h, CSR_MHPMCOUNTER9H)
+DECLARE_CSR(mhpmcounter10h, CSR_MHPMCOUNTER10H)
+DECLARE_CSR(mhpmcounter11h, CSR_MHPMCOUNTER11H)
+DECLARE_CSR(mhpmcounter12h, CSR_MHPMCOUNTER12H)
+DECLARE_CSR(mhpmcounter13h, CSR_MHPMCOUNTER13H)
+DECLARE_CSR(mhpmcounter14h, CSR_MHPMCOUNTER14H)
+DECLARE_CSR(mhpmcounter15h, CSR_MHPMCOUNTER15H)
+DECLARE_CSR(mhpmcounter16h, CSR_MHPMCOUNTER16H)
+DECLARE_CSR(mhpmcounter17h, CSR_MHPMCOUNTER17H)
+DECLARE_CSR(mhpmcounter18h, CSR_MHPMCOUNTER18H)
+DECLARE_CSR(mhpmcounter19h, CSR_MHPMCOUNTER19H)
+DECLARE_CSR(mhpmcounter20h, CSR_MHPMCOUNTER20H)
+DECLARE_CSR(mhpmcounter21h, CSR_MHPMCOUNTER21H)
+DECLARE_CSR(mhpmcounter22h, CSR_MHPMCOUNTER22H)
+DECLARE_CSR(mhpmcounter23h, CSR_MHPMCOUNTER23H)
+DECLARE_CSR(mhpmcounter24h, CSR_MHPMCOUNTER24H)
+DECLARE_CSR(mhpmcounter25h, CSR_MHPMCOUNTER25H)
+DECLARE_CSR(mhpmcounter26h, CSR_MHPMCOUNTER26H)
+DECLARE_CSR(mhpmcounter27h, CSR_MHPMCOUNTER27H)
+DECLARE_CSR(mhpmcounter28h, CSR_MHPMCOUNTER28H)
+DECLARE_CSR(mhpmcounter29h, CSR_MHPMCOUNTER29H)
+DECLARE_CSR(mhpmcounter30h, CSR_MHPMCOUNTER30H)
+DECLARE_CSR(mhpmcounter31h, CSR_MHPMCOUNTER31H)
+#endif
+#ifdef DECLARE_CAUSE
+DECLARE_CAUSE("misaligned fetch", CAUSE_MISALIGNED_FETCH)
+DECLARE_CAUSE("fault fetch", CAUSE_FAULT_FETCH)
+DECLARE_CAUSE("illegal instruction", CAUSE_ILLEGAL_INSTRUCTION)
+DECLARE_CAUSE("breakpoint", CAUSE_BREAKPOINT)
+DECLARE_CAUSE("misaligned load", CAUSE_MISALIGNED_LOAD)
+DECLARE_CAUSE("fault load", CAUSE_FAULT_LOAD)
+DECLARE_CAUSE("misaligned store", CAUSE_MISALIGNED_STORE)
+DECLARE_CAUSE("fault store", CAUSE_FAULT_STORE)
+DECLARE_CAUSE("user_ecall", CAUSE_USER_ECALL)
+DECLARE_CAUSE("supervisor_ecall", CAUSE_SUPERVISOR_ECALL)
+DECLARE_CAUSE("hypervisor_ecall", CAUSE_HYPERVISOR_ECALL)
+DECLARE_CAUSE("machine_ecall", CAUSE_MACHINE_ECALL)
+#endif
diff --git a/cpu/fe310/include/vendor/gpio.h b/cpu/fe310/include/vendor/gpio.h
new file mode 100644
index 0000000000..f7f0acb40f
--- /dev/null
+++ b/cpu/fe310/include/vendor/gpio.h
@@ -0,0 +1,24 @@
+// See LICENSE for license details.
+
+#ifndef _SIFIVE_GPIO_H
+#define _SIFIVE_GPIO_H
+
+#define GPIO_INPUT_VAL  (0x00)
+#define GPIO_INPUT_EN   (0x04)
+#define GPIO_OUTPUT_EN  (0x08)
+#define GPIO_OUTPUT_VAL (0x0C)
+#define GPIO_PULLUP_EN  (0x10)
+#define GPIO_DRIVE      (0x14)
+#define GPIO_RISE_IE    (0x18)
+#define GPIO_RISE_IP    (0x1C)
+#define GPIO_FALL_IE    (0x20)
+#define GPIO_FALL_IP    (0x24)
+#define GPIO_HIGH_IE    (0x28)
+#define GPIO_HIGH_IP    (0x2C)
+#define GPIO_LOW_IE     (0x30)
+#define GPIO_LOW_IP     (0x34)
+#define GPIO_IOF_EN     (0x38)
+#define GPIO_IOF_SEL    (0x3C)
+#define GPIO_OUTPUT_XOR    (0x40)
+
+#endif /* _SIFIVE_GPIO_H */
diff --git a/cpu/fe310/include/vendor/otp.h b/cpu/fe310/include/vendor/otp.h
new file mode 100644
index 0000000000..93833e2b21
--- /dev/null
+++ b/cpu/fe310/include/vendor/otp.h
@@ -0,0 +1,23 @@
+// See LICENSE for license details.
+
+#ifndef _SIFIVE_OTP_H
+#define _SIFIVE_OTP_H
+
+/* Register offsets */
+
+#define OTP_LOCK         0x00
+#define OTP_CK           0x04
+#define OTP_OE           0x08
+#define OTP_SEL          0x0C
+#define OTP_WE           0x10
+#define OTP_MR           0x14
+#define OTP_MRR          0x18
+#define OTP_MPP          0x1C
+#define OTP_VRREN        0x20
+#define OTP_VPPEN        0x24
+#define OTP_A            0x28
+#define OTP_D            0x2C
+#define OTP_Q            0x30
+#define OTP_READ_TIMINGS 0x34
+
+#endif
diff --git a/cpu/fe310/include/vendor/platform.h b/cpu/fe310/include/vendor/platform.h
new file mode 100644
index 0000000000..dfb585e484
--- /dev/null
+++ b/cpu/fe310/include/vendor/platform.h
@@ -0,0 +1,129 @@
+// See LICENSE for license details.
+
+#ifndef _SIFIVE_PLATFORM_H
+#define _SIFIVE_PLATFORM_H
+
+// Some things missing from the official encoding.h
+#define MCAUSE_INT         0x80000000
+#define MCAUSE_CAUSE       0x7FFFFFFF
+
+
+
+#include "vendor/aon.h"
+#include "vendor/clint.h"
+#include "vendor/gpio.h"
+#include "vendor/otp.h"
+#include "vendor/plic.h"
+#include "vendor/prci.h"
+#include "vendor/pwm.h"
+#include "vendor/spi.h"
+#include "vendor/uart.h"
+
+
+/****************************************************************************
+ * Platform definitions
+ *****************************************************************************/
+
+// Memory map
+#define MASKROM_MEM_ADDR 		(0x00001000)
+#define TRAPVEC_TABLE_CTRL_ADDR	(0x00001010)
+#define OTP_MEM_ADDR 			(0x00020000)
+#define CLINT_CTRL_ADDR 		(0x02000000)
+#define PLIC_CTRL_ADDR 			(0x0C000000)
+#define AON_CTRL_ADDR 			(0x10000000)
+#define PRCI_CTRL_ADDR 			(0x10008000)
+#define OTP_CTRL_ADDR 			(0x10010000)
+#define GPIO_CTRL_ADDR 			(0x10012000)
+#define UART0_CTRL_ADDR 		(0x10013000)
+#define SPI0_CTRL_ADDR 			(0x10014000)
+#define PWM0_CTRL_ADDR 			(0x10015000)
+#define UART1_CTRL_ADDR 		(0x10023000)
+#define SPI1_CTRL_ADDR 			(0x10024000)
+#define PWM1_CTRL_ADDR 			(0x10025000)
+#define SPI2_CTRL_ADDR 			(0x10034000)
+#define PWM2_CTRL_ADDR 			(0x10035000)
+#define SPI0_MEM_ADDR 			(0x20000000)
+#define MEM_CTRL_ADDR 			(0x80000000)
+
+// IOF masks
+#define IOF0_SPI1_MASK          (0x000007FC)
+#define SPI11_NUM_SS     		(4)
+#define IOF_SPI1_SS0          	(2u)
+#define IOF_SPI1_SS1          	(8u)
+#define IOF_SPI1_SS2          	(9u)
+#define IOF_SPI1_SS3          	(10u)
+#define IOF_SPI1_MOSI         	(3u)
+#define IOF_SPI1_MISO         	(4u)
+#define IOF_SPI1_SCK          	(5u)
+#define IOF_SPI1_DQ0          	(3u)
+#define IOF_SPI1_DQ1          	(4u)
+#define IOF_SPI1_DQ2          	(6u)
+#define IOF_SPI1_DQ3          	(7u)
+
+#define IOF0_SPI2_MASK          (0xFC000000)
+#define SPI2_NUM_SS       		(1)
+#define IOF_SPI2_SS0          	(26u)
+#define IOF_SPI2_MOSI         	(27u)
+#define IOF_SPI2_MISO         	(28u)
+#define IOF_SPI2_SCK          	(29u)
+#define IOF_SPI2_DQ0          	(27u)
+#define IOF_SPI2_DQ1          	(28u)
+#define IOF_SPI2_DQ2          	(30u)
+#define IOF_SPI2_DQ3          	(31u)
+
+//#define IOF0_I2C_MASK          (0x00003000)
+
+#define IOF0_UART0_MASK         (0x00030000)
+#define IOF_UART0_RX   			(16u)
+#define IOF_UART0_TX   			(17u)
+
+#define IOF0_UART1_MASK         (0x03000000)
+#define IOF_UART1_RX 			(24u)
+#define IOF_UART1_TX 			(25u)
+
+#define IOF1_PWM0_MASK          (0x0000000F)
+#define IOF1_PWM1_MASK          (0x00780000)
+#define IOF1_PWM2_MASK          (0x00003C00)
+
+// Interrupt numbers
+#define INT_RESERVED 			0
+#define INT_WDOGCMP 			1
+#define INT_RTCCMP 				2
+#define INT_UART0_BASE 			3
+#define INT_UART1_BASE 			4
+#define INT_SPI0_BASE 			5
+#define INT_SPI1_BASE 			6
+#define INT_SPI2_BASE 			7
+#define INT_GPIO_BASE 			8
+#define INT_PWM0_BASE 			40
+#define INT_PWM1_BASE 			44
+#define INT_PWM2_BASE 			48
+
+// Helper functions
+#define _REG32(p, i) 			(*(volatile uint32_t *) ((p) + (i)))
+#define _REG32P(p, i) 			((volatile uint32_t *) ((p) + (i)))
+#define AON_REG(offset) 		_REG32(AON_CTRL_ADDR, offset)
+#define CLINT_REG(offset) 		_REG32(CLINT_CTRL_ADDR, offset)
+#define GPIO_REG(offset) 		_REG32(GPIO_CTRL_ADDR, offset)
+#define OTP_REG(offset)  		_REG32(OTP_CTRL_ADDR, offset)
+#define PLIC_REG(offset) 		_REG32(PLIC_CTRL_ADDR, offset)
+#define PRCI_REG(offset) 		_REG32(PRCI_CTRL_ADDR, offset)
+#define PWM0_REG(offset) 		_REG32(PWM0_CTRL_ADDR, offset)
+#define PWM1_REG(offset) 		_REG32(PWM1_CTRL_ADDR, offset)
+#define PWM2_REG(offset) 		_REG32(PWM2_CTRL_ADDR, offset)
+#define SPI0_REG(offset) 		_REG32(SPI0_CTRL_ADDR, offset)
+#define SPI1_REG(offset) 		_REG32(SPI1_CTRL_ADDR, offset)
+#define SPI2_REG(offset) 		_REG32(SPI2_CTRL_ADDR, offset)
+#define UART0_REG(offset) 		_REG32(UART0_CTRL_ADDR, offset)
+#define UART1_REG(offset) 		_REG32(UART1_CTRL_ADDR, offset)
+
+// Misc
+
+#define NUM_GPIO 			32
+
+#define PLIC_NUM_INTERRUPTS 52
+#define PLIC_NUM_PRIORITIES 7
+
+#define RTC_FREQ 			32768
+
+#endif /* _SIFIVE_PLATFORM_H */
diff --git a/cpu/fe310/include/vendor/plic.h b/cpu/fe310/include/vendor/plic.h
new file mode 100644
index 0000000000..f59faa47af
--- /dev/null
+++ b/cpu/fe310/include/vendor/plic.h
@@ -0,0 +1,30 @@
+// See LICENSE for license details.
+
+#ifndef PLIC_H
+#define PLIC_H
+
+
+// 32 bits per source
+#define PLIC_PRIORITY_OFFSET            (0x0000)
+#define PLIC_PRIORITY_SHIFT_PER_SOURCE  2
+// 1 bit per source (1 address)
+#define PLIC_PENDING_OFFSET             (0x1000)
+#define PLIC_PENDING_SHIFT_PER_SOURCE   0
+
+//0x80 per target
+#define PLIC_ENABLE_OFFSET              (0x2000)
+#define PLIC_ENABLE_SHIFT_PER_TARGET    7
+
+
+#define PLIC_THRESHOLD_OFFSET           (0x200000)
+#define PLIC_CLAIM_OFFSET               (0x200004)
+#define PLIC_THRESHOLD_SHIFT_PER_TARGET 12
+#define PLIC_CLAIM_SHIFT_PER_TARGET     12
+
+#define PLIC_MAX_SOURCE                 1023
+#define PLIC_SOURCE_MASK                0x3FF
+
+#define PLIC_MAX_TARGET                 15871
+#define PLIC_TARGET_MASK                0x3FFF
+
+#endif /* PLIC_H */
diff --git a/cpu/fe310/include/vendor/plic_driver.h b/cpu/fe310/include/vendor/plic_driver.h
new file mode 100644
index 0000000000..a22272a75a
--- /dev/null
+++ b/cpu/fe310/include/vendor/plic_driver.h
@@ -0,0 +1,34 @@
+// See LICENSE file for licence details
+
+#ifndef PLIC_DRIVER_H
+#define PLIC_DRIVER_H
+
+
+typedef uint32_t plic_source;
+typedef uint32_t plic_priority;
+typedef uint32_t plic_threshold;
+
+
+void PLIC_init (
+                uintptr_t base_addr,
+                uint32_t num_sources,
+                uint32_t num_priorities
+                );
+
+void PLIC_set_threshold (plic_threshold threshold);
+  
+void PLIC_enable_interrupt (plic_source source);
+
+void PLIC_disable_interrupt (plic_source source);
+  
+void PLIC_set_priority (
+			plic_source source,
+			plic_priority priority);
+
+plic_source PLIC_claim_interrupt(void);
+
+void PLIC_complete_interrupt(plic_source source);
+
+
+
+#endif
diff --git a/cpu/fe310/include/vendor/prci.h b/cpu/fe310/include/vendor/prci.h
new file mode 100644
index 0000000000..1a3de58d29
--- /dev/null
+++ b/cpu/fe310/include/vendor/prci.h
@@ -0,0 +1,56 @@
+// See LICENSE for license details.
+
+#ifndef _SIFIVE_PRCI_H
+#define _SIFIVE_PRCI_H
+
+/* Register offsets */
+
+#define PRCI_HFROSCCFG   (0x0000)
+#define PRCI_HFXOSCCFG   (0x0004)
+#define PRCI_PLLCFG      (0x0008)
+#define PRCI_PLLDIV      (0x000C)
+#define PRCI_PROCMONCFG  (0x00F0)
+
+/* Fields */
+#define ROSC_DIV(x)    (((x) & 0x2F) << 0 ) 
+#define ROSC_TRIM(x)   (((x) & 0x1F) << 16)
+#define ROSC_EN(x)     (((x) & 0x1 ) << 30) 
+#define ROSC_RDY(x)    (((x) & 0x1 ) << 31)
+
+#define XOSC_EN(x)     (((x) & 0x1) << 30)
+#define XOSC_RDY(x)    (((x) & 0x1) << 31)
+
+#define PLL_R(x)       (((x) & 0x7)  << 0)
+// single reserved bit for F LSB.
+#define PLL_F(x)       (((x) & 0x3F) << 4)
+#define PLL_Q(x)       (((x) & 0x3)  << 10)
+#define PLL_SEL(x)     (((x) & 0x1)  << 16)
+#define PLL_REFSEL(x)  (((x) & 0x1)  << 17)
+#define PLL_BYPASS(x)  (((x) & 0x1)  << 18)
+#define PLL_LOCK(x)    (((x) & 0x1)  << 31)
+
+#define PLL_R_default 0x1
+#define PLL_F_default 0x1F
+#define PLL_Q_default 0x3
+
+#define PLL_REFSEL_HFROSC 0x0
+#define PLL_REFSEL_HFXOSC 0x1
+
+#define PLL_SEL_HFROSC 0x0
+#define PLL_SEL_PLL    0x1
+
+#define PLL_FINAL_DIV(x)      (((x) & 0x3F) << 0)
+#define PLL_FINAL_DIV_BY_1(x) (((x) & 0x1 ) << 8)
+
+#define PROCMON_DIV(x)   (((x) & 0x1F) << 0)
+#define PROCMON_TRIM(x)  (((x) & 0x1F) << 8)
+#define PROCMON_EN(x)    (((x) & 0x1)  << 16)
+#define PROCMON_SEL(x)   (((x) & 0x3)  << 24)
+#define PROCMON_NT_EN(x) (((x) & 0x1)  << 28)
+
+#define PROCMON_SEL_HFCLK     0
+#define PROCMON_SEL_HFXOSCIN  1
+#define PROCMON_SEL_PLLOUTDIV 2
+#define PROCMON_SEL_PROCMON   3
+
+#endif // _SIFIVE_PRCI_H
diff --git a/cpu/fe310/include/vendor/prci_driver.h b/cpu/fe310/include/vendor/prci_driver.h
new file mode 100644
index 0000000000..91fba4b54b
--- /dev/null
+++ b/cpu/fe310/include/vendor/prci_driver.h
@@ -0,0 +1,79 @@
+// See LICENSE file for license details
+
+#ifndef _PRCI_DRIVER_H_
+#define _PRCI_DRIVER_H_
+
+
+typedef enum prci_freq_target {
+  
+  PRCI_FREQ_OVERSHOOT,
+  PRCI_FREQ_CLOSEST,
+  PRCI_FREQ_UNDERSHOOT
+
+} PRCI_freq_target;
+
+/* Measure and return the approximate frequency of the 
+ * CPU, as given by measuring the mcycle counter against 
+ * the mtime ticks.
+ */
+uint32_t PRCI_measure_mcycle_freq(uint32_t mtime_ticks, uint32_t mtime_freq);
+
+/* Safely switch over to the HFROSC using the given div
+ * and trim settings.
+ */
+void PRCI_use_hfrosc(int div, int trim);
+
+/* Safely switch over to the 16MHz HFXOSC,
+ * applying the finaldiv clock divider (1 is the lowest
+ * legal value).
+ */
+void PRCI_use_hfxosc(uint32_t finaldiv);
+
+/* Safely switch over to the PLL using the given
+ * settings.
+ * 
+ * Note that not all combinations of the inputs are actually
+ * legal, and this function does not check for their
+ * legality ("safely" means that this function won't turn off
+ * or glitch the clock the CPU is actually running off, but
+ * doesn't protect against you making it too fast or slow.)
+ */
+
+void PRCI_use_pll(int refsel, int bypass,
+			 int r, int f, int q, int finaldiv,
+			 int hfroscdiv, int hfrosctrim);
+
+/* Use the default clocks configured at reset.
+ * This is ~16Mhz HFROSC and turns off the LFROSC
+ * (on the current FE310 Dev Platforms, an external LFROSC is 
+ * used as it is more power efficient).
+ */
+void PRCI_use_default_clocks(void);
+
+/* Use the external bypass crystal clock
+ */
+void PRCI_use_bypass_clock(void);
+
+/* This routine will adjust the HFROSC trim
+ * while using HFROSC as the clock source, 
+ * measure the resulting frequency, then
+ * use it as the PLL clock source, 
+ * in an attempt to get over, under, or close to the 
+ * requested frequency. It returns the actual measured 
+ * frequency. 
+ *
+ * Note that the requested frequency must be within the 
+ * range supported by the PLL so not all values are 
+ * achievable with this function, and not all 
+ * are guaranteed to actually work. The PLL
+ * is rated higher than the hardware.
+ * 
+ * There is no check on the desired f_cpu frequency, it
+ * is up to the user to specify something reasonable.
+ */
+
+uint32_t PRCI_set_hfrosctrim_for_f_cpu(uint32_t f_cpu, PRCI_freq_target target);
+
+
+#endif
+  
diff --git a/cpu/fe310/include/vendor/pwm.h b/cpu/fe310/include/vendor/pwm.h
new file mode 100644
index 0000000000..067889abba
--- /dev/null
+++ b/cpu/fe310/include/vendor/pwm.h
@@ -0,0 +1,37 @@
+// See LICENSE for license details.
+
+#ifndef _SIFIVE_PWM_H
+#define _SIFIVE_PWM_H
+
+/* Register offsets */
+
+#define PWM_CFG   0x00
+#define PWM_COUNT 0x08
+#define PWM_S     0x10
+#define PWM_CMP0  0x20
+#define PWM_CMP1  0x24
+#define PWM_CMP2  0x28
+#define PWM_CMP3  0x2C
+
+/* Constants */
+
+#define PWM_CFG_SCALE       0x0000000F
+#define PWM_CFG_STICKY      0x00000100
+#define PWM_CFG_ZEROCMP     0x00000200
+#define PWM_CFG_DEGLITCH    0x00000400
+#define PWM_CFG_ENALWAYS    0x00001000
+#define PWM_CFG_ONESHOT     0x00002000
+#define PWM_CFG_CMP0CENTER  0x00010000
+#define PWM_CFG_CMP1CENTER  0x00020000
+#define PWM_CFG_CMP2CENTER  0x00040000
+#define PWM_CFG_CMP3CENTER  0x00080000
+#define PWM_CFG_CMP0GANG    0x01000000
+#define PWM_CFG_CMP1GANG    0x02000000
+#define PWM_CFG_CMP2GANG    0x04000000
+#define PWM_CFG_CMP3GANG    0x08000000
+#define PWM_CFG_CMP0IP      0x10000000
+#define PWM_CFG_CMP1IP      0x20000000
+#define PWM_CFG_CMP2IP      0x40000000
+#define PWM_CFG_CMP3IP      0x80000000
+
+#endif /* _SIFIVE_PWM_H */
diff --git a/cpu/fe310/include/vendor/spi.h b/cpu/fe310/include/vendor/spi.h
new file mode 100644
index 0000000000..916d86be26
--- /dev/null
+++ b/cpu/fe310/include/vendor/spi.h
@@ -0,0 +1,80 @@
+// See LICENSE for license details.
+
+#ifndef _SIFIVE_SPI_H
+#define _SIFIVE_SPI_H
+
+/* Register offsets */
+
+#define SPI_REG_SCKDIV          0x00
+#define SPI_REG_SCKMODE         0x04
+#define SPI_REG_CSID            0x10
+#define SPI_REG_CSDEF           0x14
+#define SPI_REG_CSMODE          0x18
+
+#define SPI_REG_DCSSCK          0x28
+#define SPI_REG_DSCKCS          0x2a
+#define SPI_REG_DINTERCS        0x2c
+#define SPI_REG_DINTERXFR       0x2e
+
+#define SPI_REG_FMT             0x40
+#define SPI_REG_TXFIFO          0x48
+#define SPI_REG_RXFIFO          0x4c
+#define SPI_REG_TXCTRL          0x50
+#define SPI_REG_RXCTRL          0x54
+
+#define SPI_REG_FCTRL           0x60
+#define SPI_REG_FFMT            0x64
+
+#define SPI_REG_IE              0x70
+#define SPI_REG_IP              0x74
+
+/* Fields */
+
+#define SPI_SCK_POL             0x1
+#define SPI_SCK_PHA             0x2
+
+#define SPI_FMT_PROTO(x)        ((x) & 0x3)
+#define SPI_FMT_ENDIAN(x)       (((x) & 0x1) << 2)
+#define SPI_FMT_DIR(x)          (((x) & 0x1) << 3)
+#define SPI_FMT_LEN(x)          (((x) & 0xf) << 16)
+
+/* TXCTRL register */
+#define SPI_TXWM(x)             ((x) & 0xffff)
+/* RXCTRL register */
+#define SPI_RXWM(x)             ((x) & 0xffff)
+
+#define SPI_IP_TXWM             0x1
+#define SPI_IP_RXWM             0x2
+
+#define SPI_FCTRL_EN            0x1
+
+#define SPI_INSN_CMD_EN         0x1
+#define SPI_INSN_ADDR_LEN(x)    (((x) & 0x7) << 1)
+#define SPI_INSN_PAD_CNT(x)     (((x) & 0xf) << 4)
+#define SPI_INSN_CMD_PROTO(x)   (((x) & 0x3) << 8)
+#define SPI_INSN_ADDR_PROTO(x)  (((x) & 0x3) << 10)
+#define SPI_INSN_DATA_PROTO(x)  (((x) & 0x3) << 12)
+#define SPI_INSN_CMD_CODE(x)    (((x) & 0xff) << 16)
+#define SPI_INSN_PAD_CODE(x)    (((x) & 0xff) << 24)
+
+#define SPI_TXFIFO_FULL  (1 << 31)   
+#define SPI_RXFIFO_EMPTY (1 << 31)   
+
+/* Values */
+
+#define SPI_CSMODE_AUTO         0
+#define SPI_CSMODE_HOLD         2
+#define SPI_CSMODE_OFF          3
+
+#define SPI_DIR_RX              0
+#define SPI_DIR_TX              1
+
+#define SPI_PROTO_S             0
+#define SPI_PROTO_D             1
+#define SPI_PROTO_Q             2
+
+#define SPI_ENDIAN_MSB          0
+#define SPI_ENDIAN_LSB          1
+
+
+#endif /* _SIFIVE_SPI_H */
diff --git a/cpu/fe310/include/vendor/uart.h b/cpu/fe310/include/vendor/uart.h
new file mode 100644
index 0000000000..2d90587071
--- /dev/null
+++ b/cpu/fe310/include/vendor/uart.h
@@ -0,0 +1,31 @@
+// See LICENSE for license details.
+
+#ifndef _SIFIVE_UART_H
+#define _SIFIVE_UART_H
+
+/* Register offsets */
+#define UART_REG_TXFIFO         0x00
+#define UART_REG_RXFIFO         0x04
+#define UART_REG_TXCTRL         0x08
+#define UART_REG_RXCTRL         0x0c
+#define UART_REG_IE             0x10
+#define UART_REG_IP             0x14
+#define UART_REG_DIV            0x18
+
+/* TXFIFO register */
+#define UART_TXFIFO_FULL        (1 << 31)
+#define UART_RXFIFO_EMPTY       (1 << 31)
+
+/* TXCTRL register */
+#define UART_TXEN               0x1
+#define UART_TXWM(x)            (((x) & 0xffff) << 16)
+
+/* RXCTRL register */
+#define UART_RXEN               0x1
+#define UART_RXWM(x)            (((x) & 0xffff) << 16)
+
+/* IP register */
+#define UART_IP_TXWM            0x1
+#define UART_IP_RXWM            0x2
+
+#endif /* _SIFIVE_UART_H */
diff --git a/cpu/fe310/intr.S b/cpu/fe310/intr.S
new file mode 100644
index 0000000000..1755545c1b
--- /dev/null
+++ b/cpu/fe310/intr.S
@@ -0,0 +1,151 @@
+/*
+ * Copyright (C) 2017 JP Bonn
+ *
+ * This file is subject to the terms and conditions of the GNU Lesser
+ * General Public License v2.1. See the file LICENSE in the top level
+ * directory for more details.
+ */
+
+#include "vendor/encoding.h"
+#include "context_frame.h"
+
+/* from platform.h TODO:fix this hard code.... */
+CLINT_CTRL_ADDR = 0x02000000
+
+  .section      .text.entry
+  .align 2
+  .global trap_entry
+
+trap_entry:
+    /*
+     * Save all regs on the currently active stack.
+     * This coule be the active thread's stack,
+     * or if no thread is active, it is saved on ISR stack
+     * (if initial startup) or on the deactivated threads
+     * stack (in the case of thread exit).  In the latter
+     * two cases the stack is just abandoned.
+     */
+    addi sp, sp, -CONTEXT_FRAME_SIZE
+
+    sw s0, s0_OFFSET(sp)
+    sw s1, s1_OFFSET(sp)
+    sw s2, s2_OFFSET(sp)
+    sw s3, s3_OFFSET(sp)
+    sw s4, s4_OFFSET(sp)
+    sw s5, s5_OFFSET(sp)
+    sw s6, s6_OFFSET(sp)
+    sw s7, s7_OFFSET(sp)
+    sw s8, s8_OFFSET(sp)
+    sw s9, s9_OFFSET(sp)
+    sw s10, s10_OFFSET(sp)
+    sw s11, s11_OFFSET(sp)
+    sw ra, ra_OFFSET(sp)
+    sw tp, tp_OFFSET(sp)
+    sw t0, t0_OFFSET(sp)
+    sw t1, t1_OFFSET(sp)
+    sw t2, t2_OFFSET(sp)
+    sw t3, t3_OFFSET(sp)
+    sw t4, t4_OFFSET(sp)
+    sw t5, t5_OFFSET(sp)
+    sw t6, t6_OFFSET(sp)
+    sw a0, a0_OFFSET(sp)
+    sw a1, a1_OFFSET(sp)
+    sw a2, a2_OFFSET(sp)
+    sw a3, a3_OFFSET(sp)
+    sw a4, a4_OFFSET(sp)
+    sw a5, a5_OFFSET(sp)
+    sw a6, a6_OFFSET(sp)
+    sw a7, a7_OFFSET(sp)
+
+
+    /* Get the interrupt cause */
+    csrr a0, mcause
+
+    /* Save active thread stack pointer in a callee save register */
+    mv s1, sp
+
+    /* Switch to ISR stack.  Interrupts are not nested so use fixed
+     *  starting address and just abandon stack when finished. */
+    la  sp, _sp
+    addi sp, sp, -4
+
+    /*  Is it a software interrupt? */
+    li t0, 0x80000003
+    beq a0, t0, context_switch
+
+    /*  Call handle_trap with MCAUSE register value as arg */
+    jal handle_trap
+
+    /*  See if a context switch was requested by the ISR */
+    lw a0, sched_context_switch_request
+    bnez a0, context_switch
+
+    /*  Restore active thread stack pointer */
+    mv sp, s1
+
+    /* Restore remaining registers */
+trap_exit:
+    lw s0, s0_OFFSET(sp)
+    lw s1, s1_OFFSET(sp)
+    lw s2, s2_OFFSET(sp)
+    lw s3, s3_OFFSET(sp)
+    lw s4, s4_OFFSET(sp)
+    lw s5, s5_OFFSET(sp)
+    lw s6, s6_OFFSET(sp)
+    lw s7, s7_OFFSET(sp)
+    lw s8, s8_OFFSET(sp)
+    lw s9, s9_OFFSET(sp)
+    lw s10, s10_OFFSET(sp)
+    lw s11, s11_OFFSET(sp)
+    lw ra, ra_OFFSET(sp)
+    lw tp, tp_OFFSET(sp)
+    lw t0, t0_OFFSET(sp)
+    lw t1, t1_OFFSET(sp)
+    lw t2, t2_OFFSET(sp)
+    lw t3, t3_OFFSET(sp)
+    lw t4, t4_OFFSET(sp)
+    lw t5, t5_OFFSET(sp)
+    lw t6, t6_OFFSET(sp)
+    lw a0, a0_OFFSET(sp)
+    lw a1, a1_OFFSET(sp)
+    lw a2, a2_OFFSET(sp)
+    lw a3, a3_OFFSET(sp)
+    lw a4, a4_OFFSET(sp)
+    lw a5, a5_OFFSET(sp)
+    lw a6, a6_OFFSET(sp)
+    lw a7, a7_OFFSET(sp)
+
+    addi sp, sp, CONTEXT_FRAME_SIZE
+    mret
+
+
+ context_switch:
+    /*  clear the software interrupt */
+    li t0, CLINT_CTRL_ADDR
+    sw zero, (t0)
+
+    /*  save the active thread's PC prior to interrupt on the stack */
+    csrr a0, mepc
+    sw a0, pc_OFFSET(s1)
+
+    /*  get the active thread - it may be 0 if none currently active */
+    lw t0, sched_active_thread
+    /*  was there a previously running thread? */
+    beqz t0, no_sp_save
+    /*  if so, save the thread's SP in the _thread structure */
+    sw s1,SP_OFFSET_IN_THREAD(t0)
+
+no_sp_save:
+    /*  all current thread state is saved - schedule a new thread */
+    call sched_run
+    lw tp, sched_active_thread
+
+    /*  set the threads SP from the newly scheduled thread
+     *  and abandon ISR stack. */
+    lw sp, SP_OFFSET_IN_THREAD(tp)
+
+    /*  restore the PC */
+    lw a0, pc_OFFSET(sp)
+    csrw mepc, a0
+
+    j trap_exit
diff --git a/cpu/fe310/ldscripts/fe310.ld b/cpu/fe310/ldscripts/fe310.ld
new file mode 100644
index 0000000000..85181789e6
--- /dev/null
+++ b/cpu/fe310/ldscripts/fe310.ld
@@ -0,0 +1,182 @@
+/*
+ * Copyright (C) 2017   Ken Rabold
+ *
+ * This file is subject to the terms and conditions of the GNU Lesser
+ * General Public License v2.1. See the file LICENSE in the top level
+ * directory for more details.
+ */
+
+/**
+ * @addtogroup      cpu_fe310
+ * @{
+ *
+ * @file
+ * @brief           Memory definitions for the SiFive FE310
+ *
+ * @author          Ken Rabold
+ *
+ * @}
+ */
+
+OUTPUT_ARCH( "riscv" )
+
+ENTRY( _start )
+
+MEMORY
+{
+  flash (rxai!w) : ORIGIN = 0x20400000, LENGTH = 512M
+  ram (wxa!ri) : ORIGIN = 0x80000000, LENGTH = 16K
+}
+
+PHDRS
+{
+  flash PT_LOAD;
+  ram_init PT_LOAD;
+  ram PT_NULL;
+}
+
+SECTIONS
+{
+  __stack_size     = DEFINED(__stack_size) ? __stack_size : 1K;
+
+  .init           :
+  {
+    KEEP (*(SORT_NONE(.init)))
+  } >flash AT>flash :flash
+
+  .text           :
+  {
+    *(.text.unlikely .text.unlikely.*)
+    *(.text.startup .text.startup.*)
+    *(.text .text.*)
+    *(.gnu.linkonce.t.*)
+  } >flash AT>flash :flash
+
+  .fini           :
+  {
+    KEEP (*(SORT_NONE(.fini)))
+  } >flash AT>flash :flash
+
+  PROVIDE (__etext = .);
+  PROVIDE (_etext = .);
+  PROVIDE (etext = .);
+
+  .rodata         :
+  {
+    *(.rdata)
+    *(.rodata .rodata.*)
+    *(.gnu.linkonce.r.*)
+  } >flash AT>flash :flash
+
+  . = ALIGN(4);
+
+  .preinit_array  :
+  {
+    PROVIDE_HIDDEN (__preinit_array_start = .);
+    KEEP (*(.preinit_array))
+    PROVIDE_HIDDEN (__preinit_array_end = .);
+  } >flash AT>flash :flash
+
+  .init_array     :
+  {
+    PROVIDE_HIDDEN (__init_array_start = .);
+    KEEP (*(SORT_BY_INIT_PRIORITY(.init_array.*) SORT_BY_INIT_PRIORITY(.ctors.*)))
+    KEEP (*(.init_array EXCLUDE_FILE (*crtbegin.o *crtbegin?.o *crtend.o *crtend?.o ) .ctors))
+    PROVIDE_HIDDEN (__init_array_end = .);
+  } >flash AT>flash :flash
+
+  .fini_array     :
+  {
+    PROVIDE_HIDDEN (__fini_array_start = .);
+    KEEP (*(SORT_BY_INIT_PRIORITY(.fini_array.*) SORT_BY_INIT_PRIORITY(.dtors.*)))
+    KEEP (*(.fini_array EXCLUDE_FILE (*crtbegin.o *crtbegin?.o *crtend.o *crtend?.o ) .dtors))
+    PROVIDE_HIDDEN (__fini_array_end = .);
+  } >flash AT>flash :flash
+
+  .ctors          :
+  {
+    /* gcc uses crtbegin.o to find the start of
+       the constructors, so we make sure it is
+       first.  Because this is a wildcard, it
+       doesn't matter if the user does not
+       actually link against crtbegin.o; the
+       linker won't look for a file to match a
+       wildcard.  The wildcard also means that it
+       doesn't matter which directory crtbegin.o
+       is in.  */
+    KEEP (*crtbegin.o(.ctors))
+    KEEP (*crtbegin?.o(.ctors))
+    /* We don't want to include the .ctor section from
+       the crtend.o file until after the sorted ctors.
+       The .ctor section from the crtend file contains the
+       end of ctors marker and it must be last */
+    KEEP (*(EXCLUDE_FILE (*crtend.o *crtend?.o ) .ctors))
+    KEEP (*(SORT(.ctors.*)))
+    KEEP (*(.ctors))
+  } >flash AT>flash :flash
+
+  .dtors          :
+  {
+    KEEP (*crtbegin.o(.dtors))
+    KEEP (*crtbegin?.o(.dtors))
+    KEEP (*(EXCLUDE_FILE (*crtend.o *crtend?.o ) .dtors))
+    KEEP (*(SORT(.dtors.*)))
+    KEEP (*(.dtors))
+  } >flash AT>flash :flash
+
+  .lalign         :
+  {
+    . = ALIGN(4);
+    PROVIDE( _data_lma = . );
+  } >flash AT>flash :flash
+
+  .dalign         :
+  {
+    . = ALIGN(4);
+    PROVIDE( _data = . );
+  } >ram AT>flash :ram_init
+
+  .data          :
+  {
+    *(.data .data.*)
+    *(.gnu.linkonce.d.*)
+    . = ALIGN(8);
+    PROVIDE( __global_pointer$ = . + 0x800 );
+    *(.sdata .sdata.*)
+    *(.gnu.linkonce.s.*)
+    . = ALIGN(8);
+    *(.srodata.cst16)
+    *(.srodata.cst8)
+    *(.srodata.cst4)
+    *(.srodata.cst2)
+    *(.srodata .srodata.*)
+  } >ram AT>flash :ram_init
+
+  . = ALIGN(4);
+  PROVIDE( _edata = . );
+  PROVIDE( edata = . );
+
+  PROVIDE( _fbss = . );
+  PROVIDE( __bss_start = . );
+  .bss            :
+  {
+    *(.sbss*)
+    *(.gnu.linkonce.sb.*)
+    *(.bss .bss.*)
+    *(.gnu.linkonce.b.*)
+    *(COMMON)
+    . = ALIGN(4);
+  } >ram AT>ram :ram
+
+  . = ALIGN(8);
+  PROVIDE( _end = . );
+  PROVIDE( end = . );
+  PROVIDE( _heap_start = . );
+
+  .stack ORIGIN(ram) + LENGTH(ram) - __stack_size :
+  {
+    PROVIDE( _heap_end = . );
+    . = __stack_size;
+    PROVIDE( _sp = . );
+  } >ram AT>ram :ram
+}
diff --git a/cpu/fe310/nano/Makefile b/cpu/fe310/nano/Makefile
new file mode 100644
index 0000000000..7e05da389e
--- /dev/null
+++ b/cpu/fe310/nano/Makefile
@@ -0,0 +1,3 @@
+MODULE = newlib_syscalls_fe310
+
+include $(RIOTBASE)/Makefile.base
diff --git a/cpu/fe310/nano/nanostubs.c b/cpu/fe310/nano/nanostubs.c
new file mode 100644
index 0000000000..f68ddffce0
--- /dev/null
+++ b/cpu/fe310/nano/nanostubs.c
@@ -0,0 +1,150 @@
+/*
+ * Copyright (C) 2017 Ken Rabold
+ *
+ * This file is subject to the terms and conditions of the GNU Lesser
+ * General Public License v2.1. See the file LICENSE in the top level
+ * directory for more details.
+ */
+
+/**
+ * @ingroup     cpu_fe310
+ * @{
+ *
+ * @file        nanostubs.c
+ * @brief       Implementation of the Newlib-nano for SiFive FE310
+ *
+ * @author      Ken Rabold
+ * @}
+ */
+
+#include <reent.h>
+#include <errno.h>
+#include <stdio.h>
+#include <sys/stat.h>
+#include <sys/unistd.h>
+
+#include "sched.h"
+#include "thread.h"
+#include "irq.h"
+#include "cpu.h"
+#include "uart_stdio.h"
+
+extern char _heap_start;            /* Heap markers from fe310.ld file */
+extern char _heap_end;
+char *heap_top = &_heap_start + 4;
+
+/**
+ * @brief Initialize the Newlib-nano functions (also forces inclusion of stubs for linking)
+ */
+void nanostubs_init(void)
+{
+#if defined(MODULE_UART_STDIO)
+    /* STDIO redirected to UART, no line buffering */
+    uart_stdio_init();
+    setvbuf(stdout, NULL, _IONBF, 0);
+#endif
+}
+
+void _init(void)
+{
+}
+
+void _fini(void)
+{
+}
+
+void *_sbrk(ptrdiff_t incr)
+{
+    unsigned int state = irq_disable();
+    void *res = heap_top;
+
+    /* Allocate memory from heap */
+    if ((heap_top + incr > &_heap_end) || (heap_top + incr < &_heap_start)) {
+        errno = ENOMEM;
+        res = (void *) -1;
+    }
+    else {
+        heap_top += incr;
+    }
+
+    irq_restore(state);
+    return res;
+}
+
+int _open(const char *name, int flags, int mode)
+{
+    (void) name;
+    (void) flags;
+    (void) mode;
+    errno = ENODEV;
+    return -1;
+}
+
+_ssize_t _read(int fd, void *buffer, size_t count)
+{
+#if defined(MODULE_UART_STDIO)
+    if (fd == STDIN_FILENO) {
+        return uart_stdio_read(buffer, count);
+    }
+    errno = EBADF;
+    return -1;
+#else
+    (void) fd;
+    (void) buffer;
+    (void) count;
+    errno = ENODEV;
+    return -1;
+#endif
+}
+
+_ssize_t _write(int fd, const void *data, size_t count)
+{
+#if defined(MODULE_UART_STDIO)
+    if (fd == STDOUT_FILENO || fd == STDERR_FILENO) {
+        return uart_stdio_write(data, count);
+    }
+    errno = EBADF;
+    return -1;
+#else
+    (void) fd;
+    (void) data;
+    (void) count;
+    errno = ENODEV;
+    return -1;
+#endif
+}
+
+int _close(int fd)
+{
+    (void) fd;
+    errno = ENODEV;
+    return -1;
+}
+
+int _isatty(int fd)
+{
+    errno = 0;
+
+    if (fd == STDIN_FILENO || fd == STDOUT_FILENO || fd == STDERR_FILENO) {
+        return 1;
+    }
+
+    return 0;
+}
+
+int _fstat(int fd, struct stat *st)
+{
+    (void) fd;
+    (void) st;
+    errno = ENODEV;
+    return -1;
+}
+
+_off_t _lseek(int fd, _off_t pos, int dir)
+{
+    (void) fd;
+    (void) pos;
+    (void) dir;
+    errno = ENODEV;
+    return -1;
+}
diff --git a/cpu/fe310/periph/Makefile b/cpu/fe310/periph/Makefile
new file mode 100644
index 0000000000..a36df249ac
--- /dev/null
+++ b/cpu/fe310/periph/Makefile
@@ -0,0 +1 @@
+include $(RIOTMAKE)/periph.mk
diff --git a/cpu/fe310/periph/cpucycle.c b/cpu/fe310/periph/cpucycle.c
new file mode 100644
index 0000000000..956f7d5b49
--- /dev/null
+++ b/cpu/fe310/periph/cpucycle.c
@@ -0,0 +1,34 @@
+/*
+ * Copyright 2017  JP Bonn
+ *
+ * This file is subject to the terms and conditions of the GNU Lesser
+ * General Public License v2.1. See the file LICENSE in the top level
+ * directory for more details.
+ */
+
+/**
+ * @ingroup     cpu_fe310
+ * @{
+ *
+ * @file        cpu_cycle.c
+ * @brief       Read CPU cycle counter
+ *
+ * @author		JP Bonn
+ * @}
+ */
+#include "cpucycle.h"
+#include "vendor/encoding.h"
+
+
+uint64_t get_cycle_count(void)
+{
+    uint32_t lo, hi, hi2;
+
+    __asm__ __volatile__ ("1:\n\t"             \
+                          "csrr %0, mcycleh\n\t"       \
+                          "csrr %1, mcycle\n\t"        \
+                          "csrr %2, mcycleh\n\t"        \
+                          "bne  %0, %2, 1b\n\t"         \
+                          : "=r" (hi), "=r" (lo), "=r" (hi2));
+    return lo | ((uint64_t) hi << 32);
+}
diff --git a/cpu/fe310/periph/cpuid.c b/cpu/fe310/periph/cpuid.c
new file mode 100644
index 0000000000..ce9ba9b1ef
--- /dev/null
+++ b/cpu/fe310/periph/cpuid.c
@@ -0,0 +1,38 @@
+/*
+ * Copyright 2017 Ken Rabold
+ *
+ * This file is subject to the terms and conditions of the GNU Lesser
+ * General Public License v2.1. See the file LICENSE in the top level
+ * directory for more details.
+ */
+
+/**
+ * @ingroup     cpu_fe310
+ * @{
+ *
+ * @file        cpuid.c
+ * @brief       Low-level CPUID driver implementation
+ *
+ * @author      Ken Rabold
+ * @}
+ */
+
+#include <stdint.h>
+#include <string.h>
+
+#include "periph/cpuid.h"
+#include "cpu.h"
+#include "sifive/encoding.h"
+#include "sifive/platform.h"
+
+void cpuid_get(void *id)
+{
+    uint32_t result[5];
+
+    /* Read RISC-V CSRs for CPU ID */
+    result[0] = read_csr(mvendorid);
+    result[1] = read_csr(marchid);
+    result[2] = read_csr(mimpid);
+
+    memcpy(id, &result[0], CPUID_LEN);
+}
diff --git a/cpu/fe310/periph/gpio.c b/cpu/fe310/periph/gpio.c
new file mode 100644
index 0000000000..85e1934995
--- /dev/null
+++ b/cpu/fe310/periph/gpio.c
@@ -0,0 +1,214 @@
+/*
+ * Copyright 2017 Ken Rabold
+ *
+ * This file is subject to the terms and conditions of the GNU Lesser
+ * General Public License v2.1. See the file LICENSE in the top level
+ * directory for more details.
+ */
+
+/**
+ * @ingroup     cpu_fe310
+ * @{
+ *
+ * @file        gpio.c
+ * @brief       Low-level GPIO implementation
+ *
+ * @author      Ken Rabold
+ * @}
+ */
+
+#include <stdlib.h>
+#include <unistd.h>
+
+#include "irq.h"
+#include "cpu.h"
+#include "periph_cpu.h"
+#include "periph_conf.h"
+#include "periph/gpio.h"
+#include "vendor/encoding.h"
+#include "vendor/platform.h"
+#include "vendor/plic_driver.h"
+
+/* Num of GPIOs supported */
+#define GPIO_NUMOF (32)
+
+static gpio_flank_t isr_flank[GPIO_NUMOF];
+static gpio_isr_ctx_t isr_ctx[GPIO_NUMOF];
+
+void gpio_isr(int num)
+{
+    uint32_t pin = num - INT_GPIO_BASE;
+
+    /* Invoke callback function */
+    if (isr_ctx[pin].cb) {
+        isr_ctx[pin].cb(isr_ctx[pin].arg);
+    }
+
+    /* Clear interupt */
+    switch (isr_flank[pin]) {
+        case GPIO_FALLING:
+            GPIO_REG(GPIO_FALL_IP) |= (1 << pin);
+            break;
+
+        case GPIO_RISING:
+            GPIO_REG(GPIO_RISE_IP) |= (1 << pin);
+            break;
+
+        case GPIO_BOTH:
+            GPIO_REG(GPIO_FALL_IP) |= (1 << pin);
+            GPIO_REG(GPIO_RISE_IP) |= (1 << pin);
+            break;
+    }
+}
+
+int gpio_init(gpio_t pin, gpio_mode_t mode)
+{
+    /* Check for valid pin */
+    if (pin >= GPIO_NUMOF) {
+        return -1;
+    }
+
+    /*  Configure the mode */
+    switch (mode) {
+        case GPIO_IN:
+            GPIO_REG(GPIO_INPUT_EN) |= (1 << pin);
+            GPIO_REG(GPIO_OUTPUT_EN) &= ~(1 << pin);
+            GPIO_REG(GPIO_PULLUP_EN) &= ~(1 << pin);
+            break;
+
+        case GPIO_IN_PU:
+            GPIO_REG(GPIO_INPUT_EN) |= (1 << pin);
+            GPIO_REG(GPIO_OUTPUT_EN) &= ~(1 << pin);
+            GPIO_REG(GPIO_PULLUP_EN) |= (1 << pin);
+            break;
+
+        case GPIO_OUT:
+            GPIO_REG(GPIO_INPUT_EN) &= ~(1 << pin);
+            GPIO_REG(GPIO_OUTPUT_EN) |= (1 << pin);
+            GPIO_REG(GPIO_PULLUP_EN) &= ~(1 << pin);
+            break;
+
+        default:
+            return -1;
+    }
+
+    /* Configure the pin muxing for the GPIO */
+    GPIO_REG(GPIO_IOF_EN) &= ~(1 << pin);
+    GPIO_REG(GPIO_IOF_SEL) &= ~(1 << pin);
+
+    return 0;
+}
+
+int gpio_init_int(gpio_t pin, gpio_mode_t mode, gpio_flank_t flank,
+                  gpio_cb_t cb, void *arg)
+{
+    /* Configure pin */
+    if (gpio_init(pin, mode) != 0) {
+        return -1;
+    }
+
+    /* Disable ext interrupts when setting up */
+    clear_csr(mie, MIP_MEIP);
+
+    /* Configure GPIO ISR with PLIC */
+    set_external_isr_cb(INT_GPIO_BASE + pin, gpio_isr);
+    PLIC_enable_interrupt(INT_GPIO_BASE + pin);
+    PLIC_set_priority(INT_GPIO_BASE + pin, GPIO_INTR_PRIORITY);
+
+    /*  Configure the active flank(s) */
+    gpio_irq_enable(pin);
+
+    /* Save callback */
+    isr_ctx[pin].cb = cb;
+    isr_ctx[pin].arg = arg;
+    isr_flank[pin] = flank;
+
+    /* Re-eanble ext interrupts */
+    set_csr(mie, MIP_MEIP);
+
+    return 0;
+}
+
+void gpio_irq_enable(gpio_t pin)
+{
+    /* Check for valid pin */
+    if (pin >= GPIO_NUMOF) {
+        return;
+    }
+
+    /* Enable interupt for pin */
+    switch (isr_flank[pin]) {
+        case GPIO_FALLING:
+            GPIO_REG(GPIO_FALL_IE) |= (1 << pin);
+            break;
+
+        case GPIO_RISING:
+            GPIO_REG(GPIO_RISE_IE) |= (1 << pin);
+            break;
+
+        case GPIO_BOTH:
+            GPIO_REG(GPIO_FALL_IE) |= (1 << pin);
+            GPIO_REG(GPIO_RISE_IE) |= (1 << pin);
+            break;
+
+        default:
+            break;
+    }
+}
+
+void gpio_irq_disable(gpio_t pin)
+{
+    /* Check for valid pin */
+    if (pin >= GPIO_NUMOF) {
+        return;
+    }
+
+    /* Disable interupt for pin */
+    switch (isr_flank[pin]) {
+        case GPIO_FALLING:
+            GPIO_REG(GPIO_FALL_IE) &= ~(1 << pin);
+            break;
+
+        case GPIO_RISING:
+            GPIO_REG(GPIO_RISE_IE) &= ~(1 << pin);
+            break;
+
+        case GPIO_BOTH:
+            GPIO_REG(GPIO_FALL_IE) &= ~(1 << pin);
+            GPIO_REG(GPIO_RISE_IE) &= ~(1 << pin);
+            break;
+
+        default:
+            break;
+    }
+}
+
+int gpio_read(gpio_t pin)
+{
+    return (GPIO_REG(GPIO_INPUT_VAL) & (1 << pin)) ? 1 : 0;
+}
+
+void gpio_set(gpio_t pin)
+{
+    GPIO_REG(GPIO_OUTPUT_VAL) |= (1 << pin);
+}
+
+void gpio_clear(gpio_t pin)
+{
+    GPIO_REG(GPIO_OUTPUT_VAL) &= ~(1 << pin);
+}
+
+void gpio_toggle(gpio_t pin)
+{
+    GPIO_REG(GPIO_OUTPUT_VAL) ^= (1 << pin);
+}
+
+void gpio_write(gpio_t pin, int value)
+{
+    if (value) {
+        GPIO_REG(GPIO_OUTPUT_VAL) |= (1 << pin);
+    }
+    else {
+        GPIO_REG(GPIO_OUTPUT_VAL) &= ~(1 << pin);
+    }
+}
diff --git a/cpu/fe310/periph/pm.c b/cpu/fe310/periph/pm.c
new file mode 100644
index 0000000000..a5bb503431
--- /dev/null
+++ b/cpu/fe310/periph/pm.c
@@ -0,0 +1,33 @@
+/*
+ * Copyright 2017 Ken Rabold
+ *
+ * This file is subject to the terms and conditions of the GNU Lesser
+ * General Public License v2.1. See the file LICENSE in the top level
+ * directory for more details.
+ */
+
+/**
+ * @ingroup     cpu_fe310
+ * @{
+ *
+ * @file        pm.c
+ * @brief       Implementation of the CPU power management for SiFive FE310
+ *
+ * @author      Ken Rabold
+ * @}
+ */
+
+#include "periph/pm.h"
+
+void pm_set_lowest(void)
+{
+    /* __asm__("wfi"); */
+}
+
+void pm_off(void)
+{
+}
+
+void pm_reboot(void)
+{
+}
diff --git a/cpu/fe310/periph/rtc.c b/cpu/fe310/periph/rtc.c
new file mode 100644
index 0000000000..91621c4878
--- /dev/null
+++ b/cpu/fe310/periph/rtc.c
@@ -0,0 +1,106 @@
+/*
+ * Copyright 2017 Ken Rabold
+ *
+ * This file is subject to the terms and conditions of the GNU Lesser
+ * General Public License v2.1. See the file LICENSE in the top level
+ * directory for more details.
+ */
+
+/**
+ * @ingroup     cpu_fe310
+ * @{
+ *
+ * @file        rtc.c
+ * @brief       RTC interface wrapper for use with RTT modules
+ *
+ * @author      Ken Rabold
+ * @}
+ */
+
+#include <stdlib.h>
+#include <unistd.h>
+#include <time.h>
+
+#include "cpu.h"
+#include "periph_cpu.h"
+#include "periph_conf.h"
+#include "periph/rtt.h"
+#include "periph/rtc.h"
+
+#define ENABLE_DEBUG (0)
+#include "debug.h"
+
+
+typedef struct {
+    rtc_alarm_cb_t cb; /**< callback called from RTC interrupt */
+}rtc_state_t;
+
+static rtc_state_t rtc_callback;
+
+static void rtc_cb(void *arg);
+
+void rtc_init(void)
+{
+    rtt_init();
+}
+
+int rtc_set_time(struct tm *time)
+{
+    time_t t = mktime(time);
+
+    rtt_set_counter((uint32_t)t);
+
+    return 0;
+}
+
+int rtc_get_time(struct tm *time)
+{
+    time_t t = (time_t)rtt_get_counter();
+
+    gmtime_r(&t, time);
+
+    return 0;
+}
+
+int rtc_set_alarm(struct tm *time, rtc_alarm_cb_t cb, void *arg)
+{
+    time_t t = mktime(time);
+
+    rtc_callback.cb = cb;
+
+    rtt_set_alarm((uint32_t)t, rtc_cb, arg);
+
+    return 0;
+}
+
+int rtc_get_alarm(struct tm *time)
+{
+    time_t t = (time_t)rtt_get_alarm();
+
+    gmtime_r(&t, time);
+
+    return 0;
+}
+
+void rtc_clear_alarm(void)
+{
+    rtt_clear_alarm();
+    rtc_callback.cb = NULL;
+}
+
+void rtc_poweron(void)
+{
+    rtt_poweron();
+}
+
+void rtc_poweroff(void)
+{
+    rtt_poweroff();
+}
+
+static void rtc_cb(void *arg)
+{
+    if (rtc_callback.cb != NULL) {
+        rtc_callback.cb(arg);
+    }
+}
diff --git a/cpu/fe310/periph/rtt.c b/cpu/fe310/periph/rtt.c
new file mode 100644
index 0000000000..9ed6e578e8
--- /dev/null
+++ b/cpu/fe310/periph/rtt.c
@@ -0,0 +1,206 @@
+/*
+ * Copyright 2017 Ken Rabold
+ *
+ * This file is subject to the terms and conditions of the GNU Lesser
+ * General Public License v2.1. See the file LICENSE in the top level
+ * directory for more details.
+ */
+
+/**
+ * @ingroup     cpu_fe310
+ * @{
+ *
+ * @file        rtt.c
+ * @brief       Low-level RTT interface implementation for SiFive FE310
+ *              The FE320's RTC module is what RIOT calls a Real-Time
+ *              Timer (RTT), a simple counter which counts seconds; RIOT Real-
+ *              Time Clocks (RTC) counts seconds, minutes, hours etc. There is
+ *              an RTT->RTC wrapper layer in a separate file to allow using the
+ *              RTT as a system real time clock.
+ *
+ * @author      Ken Rabold
+ * @}
+ */
+
+#include <stdlib.h>
+#include <unistd.h>
+
+#include "cpu.h"
+#include "periph_cpu.h"
+#include "periph_conf.h"
+#include "periph/rtt.h"
+#include "vendor/encoding.h"
+#include "vendor/platform.h"
+#include "vendor/plic_driver.h"
+
+#define ENABLE_DEBUG (0)
+#include "debug.h"
+
+/* Convert RTT freq to pre-scaler value */
+#if (RTT_FREQUENCY == 32768)
+#define RTT_SCALE       (0)
+#elif (RTT_FREQUENCY == 16384)
+#define RTT_SCALE       (1)
+#elif (RTT_FREQUENCY == 8192)
+#define RTT_SCALE       (2)
+#elif (RTT_FREQUENCY == 4096)
+#define RTT_SCALE       (3)
+#elif (RTT_FREQUENCY == 2048)
+#define RTT_SCALE       (4)
+#elif (RTT_FREQUENCY == 1024)
+#define RTT_SCALE       (5)
+#elif (RTT_FREQUENCY == 512)
+#define RTT_SCALE       (6)
+#elif (RTT_FREQUENCY == 256)
+#define RTT_SCALE       (7)
+#elif (RTT_FREQUENCY == 128)
+#define RTT_SCALE       (8)
+#elif (RTT_FREQUENCY == 64)
+#define RTT_SCALE       (9)
+#elif (RTT_FREQUENCY == 32)
+#define RTT_SCALE       (10)
+#elif (RTT_FREQUENCY == 16)
+#define RTT_SCALE       (11)
+#elif (RTT_FREQUENCY == 8)
+#define RTT_SCALE       (12)
+#elif (RTT_FREQUENCY == 4)
+#define RTT_SCALE       (13)
+#elif (RTT_FREQUENCY == 2)
+#define RTT_SCALE       (14)
+#elif (RTT_FREQUENCY == 1)
+#define RTT_SCALE       (15)
+#else
+#error "Invalid RTT_FREQUENCY: Must be power of 2"
+#endif
+
+typedef struct {
+    uint32_t alarm_val;     /**< cached alarm val */
+    rtt_cb_t alarm_cb;      /**< callback called from RTC alarm */
+    void *alarm_arg;        /**< argument passed to the callback */
+    rtt_cb_t overflow_cb;   /**< callback called when RTC overflows */
+    void *overflow_arg;     /**< argument passed to the callback */
+}rtt_state_t;
+
+static rtt_state_t rtt_callback;
+
+void rtt_isr(int num)
+{
+    (void) num;
+
+    /* Clear intr */
+    AON_REG(AON_RTCCMP) = RTT_MAX_VALUE;
+
+    /* Invoke callback function for alarm or overflow */
+    if (rtt_callback.alarm_cb) {
+        rtt_callback.alarm_cb(rtt_callback.alarm_arg);
+    }
+    else if (rtt_callback.overflow_cb) {
+        rtt_callback.overflow_cb(rtt_callback.overflow_arg);
+    }
+}
+
+void rtt_init(void)
+{
+    /* Disable ext interrupts when setting up */
+    clear_csr(mie, MIP_MEIP);
+
+    /* Configure RTC ISR with PLIC */
+    set_external_isr_cb(INT_RTCCMP, rtt_isr);
+    PLIC_enable_interrupt(INT_RTCCMP);
+    PLIC_set_priority(INT_RTCCMP, RTT_INTR_PRIORITY);
+
+    /*  Configure RTC scaler, etc... */
+    AON_REG(AON_RTCCFG) = RTT_SCALE;
+    AON_REG(AON_RTCLO) = 0;
+    AON_REG(AON_RTCHI) = 0;
+    AON_REG(AON_RTCCMP) = RTT_MAX_VALUE;
+
+    /* Configure state struct */
+    rtt_callback.alarm_val = RTT_MAX_VALUE;
+    rtt_callback.alarm_cb = NULL;
+    rtt_callback.alarm_arg = NULL;
+    rtt_callback.overflow_cb = NULL;
+    rtt_callback.overflow_arg = NULL;
+
+    /* Enable the RTC counter */
+    rtt_poweron();
+
+    /* Re-eanble ext interrupts */
+    set_csr(mie, MIP_MEIP);
+}
+
+void rtt_set_overflow_cb(rtt_cb_t cb, void *arg)
+{
+    /*
+     * No separate overflow intr
+     * If no alarm cb is set, this cb will fire on RTC intr
+     */
+    rtt_callback.overflow_cb = cb;
+    rtt_callback.overflow_arg = arg;
+}
+
+void rtt_clear_overflow_cb(void)
+{
+    /* No separate overflow intr */
+    rtt_callback.overflow_cb = NULL;
+    rtt_callback.overflow_arg = NULL;
+}
+
+uint32_t rtt_get_counter(void)
+{
+    /* Read scaled counter reg value */
+    uint32_t t = AON_REG(AON_RTCS);
+
+    return t;
+}
+
+void rtt_set_counter(uint32_t counter)
+{
+    /*
+     * Can't write to scaled RTC reg
+     * Must program HI/LO regs
+     * Write scaled counter reg value
+     */
+    AON_REG(AON_RTCLO) = counter << RTT_SCALE;
+    AON_REG(AON_RTCHI) = counter >> (32 - RTT_SCALE);
+}
+
+void rtt_set_alarm(uint32_t alarm, rtt_cb_t cb, void *arg)
+{
+    /* Set cmp reg to given value */
+    AON_REG(AON_RTCCMP) = alarm;
+    rtt_callback.alarm_val = alarm;
+    rtt_callback.alarm_cb = cb;
+    rtt_callback.alarm_arg = arg;
+}
+
+uint32_t rtt_get_alarm(void)
+{
+    /*
+     * Read back cached alarm value
+     * When alarm fires, cmp reg is set to highest val
+     * to clear intr, so used cached value for this function
+     */
+    return rtt_callback.alarm_val;
+}
+
+void rtt_clear_alarm(void)
+{
+    /* Set cmp reg to highest value */
+    AON_REG(AON_RTCCMP) = RTT_MAX_VALUE;
+    rtt_callback.alarm_val = RTT_MAX_VALUE;
+    rtt_callback.alarm_cb = NULL;
+    rtt_callback.alarm_arg = NULL;
+}
+
+void rtt_poweron(void)
+{
+    /* Enable the RTC counter */
+    AON_REG(AON_RTCCFG) |= AON_RTCCFG_ENALWAYS;
+}
+
+void rtt_poweroff(void)
+{
+    /* Disable the RTC counter */
+    AON_REG(AON_RTCCFG) &= ~AON_RTCCFG_ENALWAYS;
+}
diff --git a/cpu/fe310/periph/timer.c b/cpu/fe310/periph/timer.c
new file mode 100644
index 0000000000..e7d111a7f0
--- /dev/null
+++ b/cpu/fe310/periph/timer.c
@@ -0,0 +1,171 @@
+/*
+ * Copyright 2017 Ken Rabold
+ *
+ * This file is subject to the terms and conditions of the GNU Lesser
+ * General Public License v2.1. See the file LICENSE in the top level
+ * directory for more details.
+ */
+
+/**
+ * @ingroup     cpu_fe310
+ * @{
+ *
+ * @file        timer.c
+ * @brief       Low-level timer implementation
+ *
+ * @author      Ken Rabold
+ * @}
+ */
+
+#include <stdlib.h>
+#include <unistd.h>
+
+#include "cpu.h"
+#include "periph_cpu.h"
+#include "periph_conf.h"
+#include "periph/timer.h"
+#include "vendor/encoding.h"
+#include "vendor/platform.h"
+
+/**
+ * @brief   Save reference to the timer callback
+ */
+static timer_cb_t isr_cb;
+
+/**
+ * @brief   Save argument for the callback
+ */
+static void *isr_arg;
+
+int timer_init(tim_t dev, unsigned long freq, timer_cb_t cb, void *arg)
+{
+    /* Using RISC-V built in timer (64bit value) */
+    if (dev != 0) {
+        return -1;
+    }
+
+    /* Built in timer for FE310 is 32KHz */
+    if (freq != RTC_FREQ) {
+        return -1;
+    }
+
+    /* Save timer callback and arg */
+    isr_cb = cb;
+    isr_arg = arg;
+
+    /* No other configuration */
+    return 0;
+}
+
+int timer_set(tim_t dev, int channel, unsigned int timeout)
+{
+    volatile uint64_t *mtime = (uint64_t *) (CLINT_CTRL_ADDR + CLINT_MTIME);
+    volatile uint64_t *mtimecmp =
+        (uint64_t *) (CLINT_CTRL_ADDR + CLINT_MTIMECMP);
+
+    /* Compute delta for timer */
+    uint64_t now = *mtime;
+    uint64_t then = now + (uint64_t) timeout;
+
+    if (dev != 0 || channel != 0) {
+        return -1;
+    }
+
+    /* Avoid spurious timer intr */
+    clear_csr(mie, MIP_MTIP);
+
+    /* New intr time */
+    *mtimecmp = then;
+
+    /* Re-enalble timer intr */
+    set_csr(mie, MIP_MTIP);
+    return 0;
+}
+
+int timer_set_absolute(tim_t dev, int channel, unsigned int value)
+{
+
+    volatile uint64_t *mtime = (uint64_t *) (CLINT_CTRL_ADDR + CLINT_MTIME);
+    volatile uint64_t *mtimecmp =
+        (uint64_t *) (CLINT_CTRL_ADDR + CLINT_MTIMECMP);
+
+    /* Compute absolute for timer */
+    uint64_t now = *mtime;
+    uint64_t then = (now & 0xFFFFFFFF00000000) + (uint64_t) value;
+
+    if (dev != 0 || channel != 0) {
+        return -1;
+    }
+
+    /* Avoid spurious timer intr */
+    clear_csr(mie, MIP_MTIP);
+
+    /* New intr time (handle 32bit rollover) */
+    *mtimecmp = (then > now) ? then : then + 0x100000000;
+
+    /* Re-enable timer intr */
+    set_csr(mie, MIP_MTIP);
+    return 0;
+}
+
+int timer_clear(tim_t dev, int channel)
+{
+    if (dev != 0 || channel != 0) {
+        return -1;
+    }
+
+    return 0;
+}
+
+unsigned int timer_read(tim_t dev)
+{
+    uint32_t lo = *(volatile uint32_t *) (CLINT_CTRL_ADDR + CLINT_MTIME);
+
+    if (dev != 0) {
+        return 0;
+    }
+
+    /* Read current timer value */
+    return (unsigned int) lo;
+}
+
+void timer_start(tim_t dev)
+{
+    if (dev != 0) {
+        return;
+    }
+
+    /* Timer is continuous running
+     * Enable the timer interrupt */
+    set_csr(mie, MIP_MTIP);
+}
+
+void timer_stop(tim_t dev)
+{
+    if (dev != 0) {
+        return;
+    }
+
+    /* Disable the timer interrupt */
+    clear_csr(mie, MIP_MTIP);
+}
+
+void timer_isr(void)
+{
+    volatile uint64_t *mtimecmp =
+        (uint64_t *) (CLINT_CTRL_ADDR + CLINT_MTIMECMP);
+
+    /* Clear intr */
+    clear_csr(mie, MIP_MTIP);
+
+    /* Set mtimecmp to largest value to clear the interrupt */
+    *mtimecmp = 0xFFFFFFFFFFFFFFFF;
+
+    /* Call timer callback function */
+    if (isr_cb) {
+        isr_cb(isr_arg, 0);
+    }
+
+    /* Reset interrupt */
+    set_csr(mie, MIP_MTIP);
+}
diff --git a/cpu/fe310/periph/uart.c b/cpu/fe310/periph/uart.c
new file mode 100644
index 0000000000..675d34f4e8
--- /dev/null
+++ b/cpu/fe310/periph/uart.c
@@ -0,0 +1,170 @@
+/*
+ * Copyright 2017 Ken Rabold
+ *
+ * This file is subject to the terms and conditions of the GNU Lesser
+ * General Public License v2.1. See the file LICENSE in the top level
+ * directory for more details.
+ */
+
+/**
+ * @ingroup     cpu_fe310
+ * @{
+ *
+ * @file        timer.c
+ * @brief       Low-level timer implementation
+ *
+ * @author      Ken Rabold
+ * @}
+ */
+
+#include <stdlib.h>
+#include <unistd.h>
+
+#include "irq.h"
+#include "cpu.h"
+#include "periph_cpu.h"
+#include "periph_conf.h"
+#include "periph/uart.h"
+#include "vendor/encoding.h"
+#include "vendor/platform.h"
+#include "vendor/plic_driver.h"
+#include "vendor/prci_driver.h"
+
+/**
+ * @brief   Allocate memory to store the callback functions
+ */
+static uart_isr_ctx_t isr_ctx[UART_NUMOF];
+
+void uart_isr(int num)
+{
+    uint32_t data;
+
+    /* Invoke callback function */
+    if (num == INT_UART0_BASE) {
+        data = UART0_REG(UART_REG_RXFIFO);
+
+        /* Intr cleared automatically when data is read */
+        while ((data & UART_RXFIFO_EMPTY) != (uint32_t)UART_RXFIFO_EMPTY) {
+            isr_ctx[0].rx_cb(isr_ctx[0].arg, (uint8_t) data);
+            data = UART0_REG(UART_REG_RXFIFO);
+        }
+    }
+
+    if (num == INT_UART1_BASE) {
+        data = UART1_REG(UART_REG_RXFIFO);
+
+        /* Intr cleared automatically when data is read */
+        while ((data & UART_RXFIFO_EMPTY) != (uint32_t)UART_RXFIFO_EMPTY) {
+            isr_ctx[1].rx_cb(isr_ctx[1].arg, (uint8_t) data);
+            data = UART1_REG(UART_REG_RXFIFO);
+        }
+    }
+}
+
+int uart_init(uart_t dev, uint32_t baudrate, uart_rx_cb_t rx_cb, void *arg)
+{
+    uint32_t uartDiv;
+
+    /* Check for valid UART dev */
+    if (dev >= UART_NUMOF) {
+        return UART_NODEV;
+    }
+
+    /* Save interrupt callback context */
+    isr_ctx[dev].rx_cb = rx_cb;
+    isr_ctx[dev].arg = arg;
+
+    /* Power on the device */
+    uart_poweron(dev);
+
+    /* Calculate buadrate divisor given current CPU clk rate
+     * Ignore the first run (icache needs to be warm) */
+    uartDiv = PRCI_measure_mcycle_freq(1000, RTC_FREQ);
+    /* cppcheck-suppress redundantAssignment */
+    uartDiv = PRCI_measure_mcycle_freq(1000, RTC_FREQ);
+    uartDiv = uartDiv / baudrate;
+
+    /* Enable UART 8-N-1 at given baudrate */
+    if (dev == 0) {
+        /* Config UART */
+        UART0_REG(UART_REG_DIV) = uartDiv;
+        UART0_REG(UART_REG_TXCTRL) = UART_TXEN;
+
+        /* Enable RX intr if there is a callback */
+        if (rx_cb != NULL) {
+            /* Disable ext interrupts when setting up */
+            clear_csr(mie, MIP_MEIP);
+
+            /* Configure UART ISR with PLIC */
+            set_external_isr_cb(INT_UART0_BASE, uart_isr);
+            PLIC_enable_interrupt(INT_UART0_BASE);
+            PLIC_set_priority(INT_UART0_BASE, UART0_RX_INTR_PRIORITY);
+            UART0_REG(UART_REG_IE) = UART_IP_RXWM;
+            UART0_REG(UART_REG_RXCTRL) = UART_RXEN;
+
+            /* Re-eanble ext interrupts */
+            set_csr(mie, MIP_MEIP);
+        }
+    }
+
+    if (dev == 1) {
+        /* Config UART */
+        UART1_REG(UART_REG_DIV) = uartDiv;
+        UART1_REG(UART_REG_TXCTRL) = UART_TXEN;
+
+        /* Enable RX intr if there is a callback */
+        if (rx_cb != NULL) {
+            /* Disable ext interrupts when setting up */
+            clear_csr(mie, MIP_MEIP);
+
+            /* Configure UART ISR with PLIC */
+            set_external_isr_cb(INT_UART1_BASE, uart_isr);
+            PLIC_enable_interrupt(INT_UART1_BASE);
+            PLIC_set_priority(INT_UART1_BASE, UART1_RX_INTR_PRIORITY);
+            UART1_REG(UART_REG_IE) = UART_IP_RXWM;
+            UART1_REG(UART_REG_RXCTRL) = UART_RXEN;
+
+            /* Re-eanble ext interrupts */
+            set_csr(mie, MIP_MEIP);
+        }
+    }
+
+    return UART_OK;
+}
+
+void uart_write(uart_t dev, const uint8_t *data, size_t len)
+{
+    if (dev == 0) {
+        for (size_t i = 0; i < len; i++) {
+            /* Wait for FIFO to empty */
+            while ((UART0_REG(UART_REG_TXFIFO) & UART_TXFIFO_FULL)
+                   == (uint32_t)UART_TXFIFO_FULL)
+                ;
+
+            /* Write a byte */
+            UART0_REG(UART_REG_TXFIFO) = data[i];
+        }
+    }
+
+    if (dev == 1) {
+        for (size_t i = 0; i < len; i++) {
+            /* Wait for FIFO to empty */
+            while ((UART1_REG(UART_REG_TXFIFO) & UART_TXFIFO_FULL)
+                   == (uint32_t)UART_TXFIFO_FULL)
+                ;
+
+            /* Write a byte */
+            UART1_REG(UART_REG_TXFIFO) = data[i];
+        }
+    }
+}
+
+void uart_poweron(uart_t dev)
+{
+    (void) dev;
+}
+
+void uart_poweroff(uart_t dev)
+{
+    (void) dev;
+}
diff --git a/cpu/fe310/start.S b/cpu/fe310/start.S
new file mode 100644
index 0000000000..d3518e5344
--- /dev/null
+++ b/cpu/fe310/start.S
@@ -0,0 +1,59 @@
+/*
+ * Copyright (C) 2017 JP Bonn, Ken Rabold
+ *
+ * This file is subject to the terms and conditions of the GNU Lesser
+ * General Public License v2.1. See the file LICENSE in the top level
+ * directory for more details.
+ */
+    .section .init
+    .globl _start
+    .type _start,@function
+
+_start:
+    .cfi_startproc
+    .cfi_undefined ra
+.option push
+.option norelax
+    la gp, __global_pointer$
+.option pop
+    la sp, _sp
+
+
+    /* Load data section */
+    la a0, _data_lma
+    la a1, _data
+    la a2, _edata
+    bgeu a1, a2, 2f
+1:
+    lw t0, (a0)
+    sw t0, (a1)
+    addi a0, a0, 4
+    addi a1, a1, 4
+    bltu a1, a2, 1b
+2:
+
+    /* Clear bss section */
+    la a0, __bss_start
+    la a1, _end
+    bgeu a0, a1, 2f
+1:
+    sw zero, (a0)
+    addi a0, a0, 4
+    bltu a0, a1, 1b
+2:
+
+    /* Call global constructors */
+    la a0, __libc_fini_array
+    call atexit
+    call __libc_init_array
+
+
+    /* Initialize board and start kernel */
+    call board_init
+    call kernel_init
+
+    /* Loop forever (should never get here) */
+1:
+    j 1b
+
+    .cfi_endproc
diff --git a/cpu/fe310/vendor/LICENSE b/cpu/fe310/vendor/LICENSE
new file mode 100644
index 0000000000..0b0b6c8fc6
--- /dev/null
+++ b/cpu/fe310/vendor/LICENSE
@@ -0,0 +1,206 @@
+
+This software, except as otherwise noted in subrepositories, 
+is licensed under the Apache 2 license, quoted below.
+
+
+                                 Apache License
+                           Version 2.0, January 2004
+                        http://www.apache.org/licenses/
+
+   TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION
+
+   1. Definitions.
+
+      "License" shall mean the terms and conditions for use, reproduction,
+      and distribution as defined by Sections 1 through 9 of this document.
+
+      "Licensor" shall mean the copyright owner or entity authorized by
+      the copyright owner that is granting the License.
+
+      "Legal Entity" shall mean the union of the acting entity and all
+      other entities that control, are controlled by, or are under common
+      control with that entity. For the purposes of this definition,
+      "control" means (i) the power, direct or indirect, to cause the
+      direction or management of such entity, whether by contract or
+      otherwise, or (ii) ownership of fifty percent (50%) or more of the
+      outstanding shares, or (iii) beneficial ownership of such entity.
+
+      "You" (or "Your") shall mean an individual or Legal Entity
+      exercising permissions granted by this License.
+
+      "Source" form shall mean the preferred form for making modifications,
+      including but not limited to software source code, documentation
+      source, and configuration files.
+
+      "Object" form shall mean any form resulting from mechanical
+      transformation or translation of a Source form, including but
+      not limited to compiled object code, generated documentation,
+      and conversions to other media types.
+
+      "Work" shall mean the work of authorship, whether in Source or
+      Object form, made available under the License, as indicated by a
+      copyright notice that is included in or attached to the work
+      (an example is provided in the Appendix below).
+
+      "Derivative Works" shall mean any work, whether in Source or Object
+      form, that is based on (or derived from) the Work and for which the
+      editorial revisions, annotations, elaborations, or other modifications
+      represent, as a whole, an original work of authorship. For the purposes
+      of this License, Derivative Works shall not include works that remain
+      separable from, or merely link (or bind by name) to the interfaces of,
+      the Work and Derivative Works thereof.
+
+      "Contribution" shall mean any work of authorship, including
+      the original version of the Work and any modifications or additions
+      to that Work or Derivative Works thereof, that is intentionally
+      submitted to Licensor for inclusion in the Work by the copyright owner
+      or by an individual or Legal Entity authorized to submit on behalf of
+      the copyright owner. For the purposes of this definition, "submitted"
+      means any form of electronic, verbal, or written communication sent
+      to the Licensor or its representatives, including but not limited to
+      communication on electronic mailing lists, source code control systems,
+      and issue tracking systems that are managed by, or on behalf of, the
+      Licensor for the purpose of discussing and improving the Work, but
+      excluding communication that is conspicuously marked or otherwise
+      designated in writing by the copyright owner as "Not a Contribution."
+
+      "Contributor" shall mean Licensor and any individual or Legal Entity
+      on behalf of whom a Contribution has been received by Licensor and
+      subsequently incorporated within the Work.
+
+   2. Grant of Copyright License. Subject to the terms and conditions of
+      this License, each Contributor hereby grants to You a perpetual,
+      worldwide, non-exclusive, no-charge, royalty-free, irrevocable
+      copyright license to reproduce, prepare Derivative Works of,
+      publicly display, publicly perform, sublicense, and distribute the
+      Work and such Derivative Works in Source or Object form.
+
+   3. Grant of Patent License. Subject to the terms and conditions of
+      this License, each Contributor hereby grants to You a perpetual,
+      worldwide, non-exclusive, no-charge, royalty-free, irrevocable
+      (except as stated in this section) patent license to make, have made,
+      use, offer to sell, sell, import, and otherwise transfer the Work,
+      where such license applies only to those patent claims licensable
+      by such Contributor that are necessarily infringed by their
+      Contribution(s) alone or by combination of their Contribution(s)
+      with the Work to which such Contribution(s) was submitted. If You
+      institute patent litigation against any entity (including a
+      cross-claim or counterclaim in a lawsuit) alleging that the Work
+      or a Contribution incorporated within the Work constitutes direct
+      or contributory patent infringement, then any patent licenses
+      granted to You under this License for that Work shall terminate
+      as of the date such litigation is filed.
+
+   4. Redistribution. You may reproduce and distribute copies of the
+      Work or Derivative Works thereof in any medium, with or without
+      modifications, and in Source or Object form, provided that You
+      meet the following conditions:
+
+      (a) You must give any other recipients of the Work or
+          Derivative Works a copy of this License; and
+
+      (b) You must cause any modified files to carry prominent notices
+          stating that You changed the files; and
+
+      (c) You must retain, in the Source form of any Derivative Works
+          that You distribute, all copyright, patent, trademark, and
+          attribution notices from the Source form of the Work,
+          excluding those notices that do not pertain to any part of
+          the Derivative Works; and
+
+      (d) If the Work includes a "NOTICE" text file as part of its
+          distribution, then any Derivative Works that You distribute must
+          include a readable copy of the attribution notices contained
+          within such NOTICE file, excluding those notices that do not
+          pertain to any part of the Derivative Works, in at least one
+          of the following places: within a NOTICE text file distributed
+          as part of the Derivative Works; within the Source form or
+          documentation, if provided along with the Derivative Works; or,
+          within a display generated by the Derivative Works, if and
+          wherever such third-party notices normally appear. The contents
+          of the NOTICE file are for informational purposes only and
+          do not modify the License. You may add Your own attribution
+          notices within Derivative Works that You distribute, alongside
+          or as an addendum to the NOTICE text from the Work, provided
+          that such additional attribution notices cannot be construed
+          as modifying the License.
+
+      You may add Your own copyright statement to Your modifications and
+      may provide additional or different license terms and conditions
+      for use, reproduction, or distribution of Your modifications, or
+      for any such Derivative Works as a whole, provided Your use,
+      reproduction, and distribution of the Work otherwise complies with
+      the conditions stated in this License.
+
+   5. Submission of Contributions. Unless You explicitly state otherwise,
+      any Contribution intentionally submitted for inclusion in the Work
+      by You to the Licensor shall be under the terms and conditions of
+      this License, without any additional terms or conditions.
+      Notwithstanding the above, nothing herein shall supersede or modify
+      the terms of any separate license agreement you may have executed
+      with Licensor regarding such Contributions.
+
+   6. Trademarks. This License does not grant permission to use the trade
+      names, trademarks, service marks, or product names of the Licensor,
+      except as required for reasonable and customary use in describing the
+      origin of the Work and reproducing the content of the NOTICE file.
+
+   7. Disclaimer of Warranty. Unless required by applicable law or
+      agreed to in writing, Licensor provides the Work (and each
+      Contributor provides its Contributions) on an "AS IS" BASIS,
+      WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
+      implied, including, without limitation, any warranties or conditions
+      of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A
+      PARTICULAR PURPOSE. You are solely responsible for determining the
+      appropriateness of using or redistributing the Work and assume any
+      risks associated with Your exercise of permissions under this License.
+
+   8. Limitation of Liability. In no event and under no legal theory,
+      whether in tort (including negligence), contract, or otherwise,
+      unless required by applicable law (such as deliberate and grossly
+      negligent acts) or agreed to in writing, shall any Contributor be
+      liable to You for damages, including any direct, indirect, special,
+      incidental, or consequential damages of any character arising as a
+      result of this License or out of the use or inability to use the
+      Work (including but not limited to damages for loss of goodwill,
+      work stoppage, computer failure or malfunction, or any and all
+      other commercial damages or losses), even if such Contributor
+      has been advised of the possibility of such damages.
+
+   9. Accepting Warranty or Additional Liability. While redistributing
+      the Work or Derivative Works thereof, You may choose to offer,
+      and charge a fee for, acceptance of support, warranty, indemnity,
+      or other liability obligations and/or rights consistent with this
+      License. However, in accepting such obligations, You may act only
+      on Your own behalf and on Your sole responsibility, not on behalf
+      of any other Contributor, and only if You agree to indemnify,
+      defend, and hold each Contributor harmless for any liability
+      incurred by, or claims asserted against, such Contributor by reason
+      of your accepting any such warranty or additional liability.
+
+   END OF TERMS AND CONDITIONS
+
+   APPENDIX: How to apply the Apache License to your work.
+
+      To apply the Apache License to your work, attach the following
+      boilerplate notice, with the fields enclosed by brackets "[]"
+      replaced with your own identifying information. (Don't include
+      the brackets!)  The text should be enclosed in the appropriate
+      comment syntax for the file format. We also recommend that a
+      file or class name and description of purpose be included on the
+      same "printed page" as the copyright notice for easier
+      identification within third-party archives.
+
+   Copyright 2016 SiFive, Inc.
+
+   Licensed under the Apache License, Version 2.0 (the "License");
+   you may not use this file except in compliance with the License.
+   You may obtain a copy of the License at
+
+       http://www.apache.org/licenses/LICENSE-2.0
+
+   Unless required by applicable law or agreed to in writing, software
+   distributed under the License is distributed on an "AS IS" BASIS,
+   WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+   See the License for the specific language governing permissions and
+   limitations under the License.
diff --git a/cpu/fe310/vendor/Makefile b/cpu/fe310/vendor/Makefile
new file mode 100644
index 0000000000..35f667540f
--- /dev/null
+++ b/cpu/fe310/vendor/Makefile
@@ -0,0 +1,3 @@
+MODULE = sifive_drivers_fe310
+
+include $(RIOTBASE)/Makefile.base
diff --git a/cpu/fe310/vendor/plic_driver.c b/cpu/fe310/vendor/plic_driver.c
new file mode 100644
index 0000000000..3009f0623e
--- /dev/null
+++ b/cpu/fe310/vendor/plic_driver.c
@@ -0,0 +1,153 @@
+// See LICENSE for license details.
+
+#include <stdlib.h>
+#include <unistd.h>
+
+#include "vendor/encoding.h"
+#include "vendor/platform.h"
+#include "vendor/plic_driver.h"
+
+
+typedef struct __plic_instance_t
+{
+  uintptr_t base_addr;
+  uint32_t num_sources;
+  uint32_t num_priorities;
+
+} plic_instance_t;
+
+
+//	PLIC instance
+static plic_instance_t 	_plic;
+
+
+
+// Note that there are no assertions or bounds checking on these
+// parameter values.
+
+void volatile_memzero(uint8_t * base, unsigned int size)
+{
+  volatile uint8_t * ptr;
+  for (ptr = base; ptr < (base + size); ptr++){
+    *ptr = 0;
+  }
+}
+
+void PLIC_init (
+                uintptr_t base_addr,
+                uint32_t num_sources,
+                uint32_t num_priorities
+                )
+{
+  plic_instance_t* this_plic = &_plic;
+  
+  this_plic->base_addr = base_addr;
+  this_plic->num_sources = num_sources;
+  this_plic->num_priorities = num_priorities;
+  
+  // Disable all interrupts (don't assume that these registers are reset).
+  unsigned long hart_id = read_csr(mhartid);
+  volatile_memzero((uint8_t*) (this_plic->base_addr +
+                               PLIC_ENABLE_OFFSET +
+                               (hart_id << PLIC_ENABLE_SHIFT_PER_TARGET)),
+                   (num_sources + 8) / 8);
+  
+  // Set all priorities to 0 (equal priority -- don't assume that these are reset).
+  volatile_memzero ((uint8_t *)(this_plic->base_addr +
+                                PLIC_PRIORITY_OFFSET),
+                    (num_sources + 1) << PLIC_PRIORITY_SHIFT_PER_SOURCE);
+
+  // Set the threshold to 0.
+  volatile plic_threshold* threshold = (plic_threshold*)
+    (this_plic->base_addr +
+     PLIC_THRESHOLD_OFFSET +
+     (hart_id << PLIC_THRESHOLD_SHIFT_PER_TARGET));
+
+  *threshold = 0;
+  
+}
+
+void PLIC_set_threshold (plic_threshold threshold)
+{
+  plic_instance_t* this_plic = &_plic;
+
+  unsigned long hart_id = read_csr(mhartid);  
+  volatile plic_threshold* threshold_ptr = (plic_threshold*) (this_plic->base_addr +
+                                                              PLIC_THRESHOLD_OFFSET +
+                                                              (hart_id << PLIC_THRESHOLD_SHIFT_PER_TARGET));
+
+  *threshold_ptr = threshold;
+
+}
+  
+
+void PLIC_enable_interrupt (plic_source source)
+{
+  plic_instance_t* this_plic = &_plic;
+
+  unsigned long hart_id = read_csr(mhartid);
+  volatile uint8_t * current_ptr = (volatile uint8_t *)(this_plic->base_addr +
+                                                        PLIC_ENABLE_OFFSET +
+                                                        (hart_id << PLIC_ENABLE_SHIFT_PER_TARGET) +
+                                                        (source >> 3));
+  uint8_t current = *current_ptr;
+  current = current | ( 1 << (source & 0x7));
+  *current_ptr = current;
+
+}
+
+void PLIC_disable_interrupt (plic_source source)
+{
+  plic_instance_t* this_plic = &_plic;
+  
+  unsigned long hart_id = read_csr(mhartid);
+  volatile uint8_t * current_ptr = (volatile uint8_t *) (this_plic->base_addr +
+                                                         PLIC_ENABLE_OFFSET +
+                                                         (hart_id << PLIC_ENABLE_SHIFT_PER_TARGET) +
+                                                         (source >> 3));
+  uint8_t current = *current_ptr;
+  current = current & ~(( 1 << (source & 0x7)));
+  *current_ptr = current;
+  
+}
+
+void PLIC_set_priority (plic_source source, plic_priority priority)
+{
+  plic_instance_t* this_plic = &_plic;
+
+  if (this_plic->num_priorities > 0) {
+    volatile plic_priority * priority_ptr = (volatile plic_priority *)
+      (this_plic->base_addr +
+       PLIC_PRIORITY_OFFSET +
+       (source << PLIC_PRIORITY_SHIFT_PER_SOURCE));
+    *priority_ptr = priority;
+  }
+}
+
+plic_source PLIC_claim_interrupt(void)
+{
+  plic_instance_t* this_plic = &_plic;
+  
+  unsigned long hart_id = read_csr(mhartid);
+
+  volatile plic_source * claim_addr = (volatile plic_source * )
+    (this_plic->base_addr +
+     PLIC_CLAIM_OFFSET +
+     (hart_id << PLIC_CLAIM_SHIFT_PER_TARGET));
+
+  return  *claim_addr;
+  
+}
+
+void PLIC_complete_interrupt(plic_source source)
+{
+  plic_instance_t* this_plic = &_plic;
+  
+  unsigned long hart_id = read_csr(mhartid);
+  volatile plic_source * claim_addr = (volatile plic_source *) (this_plic->base_addr +
+                                                                PLIC_CLAIM_OFFSET +
+                                                                (hart_id << PLIC_CLAIM_SHIFT_PER_TARGET));
+  *claim_addr = source;
+  
+}
+
diff --git a/cpu/fe310/vendor/prci_driver.c b/cpu/fe310/vendor/prci_driver.c
new file mode 100644
index 0000000000..ecdbd597fc
--- /dev/null
+++ b/cpu/fe310/vendor/prci_driver.c
@@ -0,0 +1,268 @@
+// See LICENSE file for license details
+
+#include <stdlib.h>
+#include <unistd.h>
+
+#include "vendor/encoding.h"
+#include "vendor/platform.h"
+#include "vendor/prci_driver.h"
+
+
+#define rdmcycle(x)  {				       \
+    uint32_t lo, hi, hi2;			       \
+    __asm__ __volatile__ ("1:\n\t"		       \
+			  "csrr %0, mcycleh\n\t"       \
+			  "csrr %1, mcycle\n\t"	       \
+			  "csrr %2, mcycleh\n\t"		\
+			  "bne  %0, %2, 1b\n\t"			\
+			  : "=r" (hi), "=r" (lo), "=r" (hi2)) ;	\
+    *(x) = lo | ((uint64_t) hi << 32); 				\
+  }
+
+uint32_t PRCI_measure_mcycle_freq(uint32_t mtime_ticks, uint32_t mtime_freq)
+{
+
+  uint32_t start_mtime = CLINT_REG(CLINT_MTIME);
+  uint32_t end_mtime = start_mtime + mtime_ticks + 1;
+
+  // Make sure we won't get rollover.
+  while (end_mtime < start_mtime){
+    start_mtime = CLINT_REG(CLINT_MTIME);
+    end_mtime = start_mtime + mtime_ticks + 1;
+  }
+
+  // Don't start measuring until mtime edge.
+  uint32_t tmp = start_mtime;
+  do {
+    start_mtime = CLINT_REG(CLINT_MTIME);
+  } while (start_mtime == tmp);
+  
+  uint64_t start_mcycle;
+  rdmcycle(&start_mcycle);
+  
+  while (CLINT_REG(CLINT_MTIME) < end_mtime) ;
+  
+  uint64_t end_mcycle;
+  rdmcycle(&end_mcycle);
+  uint32_t difference = (uint32_t) (end_mcycle - start_mcycle);
+
+  uint64_t freq = ((uint64_t) difference * mtime_freq) / mtime_ticks;
+  return (uint32_t) freq & 0xFFFFFFFF;
+  
+}
+ 
+
+void PRCI_use_hfrosc(int div, int trim)
+{
+  // Make sure the HFROSC is running at its default setting
+  // It is OK to change this even if we are running off of it.
+  
+  PRCI_REG(PRCI_HFROSCCFG) = (ROSC_DIV(div) | ROSC_TRIM(trim) | ROSC_EN(1));
+
+  while ((PRCI_REG(PRCI_HFROSCCFG) & ROSC_RDY(1)) == 0);
+  
+  PRCI_REG(PRCI_PLLCFG) &= ~PLL_SEL(1);
+}
+
+void PRCI_use_pll(int refsel, int bypass,
+			 int r, int f, int q, int finaldiv,
+			 int hfroscdiv, int hfrosctrim)
+{
+  // Ensure that we aren't running off the PLL before we mess with it.
+  if (PRCI_REG(PRCI_PLLCFG) & PLL_SEL(1)) {
+    // Make sure the HFROSC is running at its default setting
+    PRCI_use_hfrosc(4, 16);
+  }
+  
+  // Set PLL Source to be HFXOSC if desired.
+  uint32_t config_value = 0;
+
+  config_value |= PLL_REFSEL(refsel);
+  
+  if (bypass) {
+    // Bypass
+    config_value |= PLL_BYPASS(1);
+
+    PRCI_REG(PRCI_PLLCFG) = config_value;
+
+    // If we don't have an HFXTAL, this doesn't really matter.
+    // Set our Final output divide to divide-by-1:
+    PRCI_REG(PRCI_PLLDIV) = (PLL_FINAL_DIV_BY_1(1) | PLL_FINAL_DIV(0));
+  } else {
+  
+    // To overclock, use the hfrosc
+    if (hfrosctrim >= 0 && hfroscdiv >= 0) {
+      PRCI_use_hfrosc(hfroscdiv, hfrosctrim);
+    }
+    
+    // Set DIV Settings for PLL
+    
+    // (Legal values of f_REF are 6-48MHz)
+
+    // Set DIVR to divide-by-2 to get 8MHz frequency
+    // (legal values of f_R are 6-12 MHz)
+
+    config_value |= PLL_BYPASS(1);
+    config_value |= PLL_R(r);
+
+    // Set DIVF to get 512Mhz frequncy
+    // There is an implied multiply-by-2, 16Mhz.
+    // So need to write 32-1
+    // (legal values of f_F are 384-768 MHz)
+    config_value |= PLL_F(f);
+
+    // Set DIVQ to divide-by-2 to get 256 MHz frequency
+    // (legal values of f_Q are 50-400Mhz)
+    config_value |= PLL_Q(q);
+
+    // Set our Final output divide to divide-by-1:
+    if (finaldiv == 1){
+      PRCI_REG(PRCI_PLLDIV) = (PLL_FINAL_DIV_BY_1(1) | PLL_FINAL_DIV(0));
+    } else {
+      PRCI_REG(PRCI_PLLDIV) = (PLL_FINAL_DIV(finaldiv-1));
+    }
+
+    PRCI_REG(PRCI_PLLCFG) = config_value;
+
+    // Un-Bypass the PLL.
+    PRCI_REG(PRCI_PLLCFG) &= ~PLL_BYPASS(1);
+
+    // Wait for PLL Lock
+    // Note that the Lock signal can be glitchy.
+    // Need to wait 100 us
+    // RTC is running at 32kHz.
+    // So wait 4 ticks of RTC.
+    uint32_t now = CLINT_REG(CLINT_MTIME);
+    while (CLINT_REG(CLINT_MTIME) - now < 4) ;
+    
+    // Now it is safe to check for PLL Lock
+    while ((PRCI_REG(PRCI_PLLCFG) & PLL_LOCK(1)) == 0);
+
+  }
+
+  // Switch over to PLL Clock source
+  PRCI_REG(PRCI_PLLCFG) |= PLL_SEL(1);
+
+  // If we're running off HFXOSC, turn off the HFROSC to
+  // save power.
+  if (refsel) {
+    PRCI_REG(PRCI_HFROSCCFG) &= ~ROSC_EN(1);
+  }
+  
+}
+
+void PRCI_use_default_clocks(void)
+{
+  // Turn off the LFROSC
+  AON_REG(AON_LFROSC) &= ~ROSC_EN(1);
+
+  // Use HFROSC
+  PRCI_use_hfrosc(4, 16);
+}
+
+void PRCI_use_hfxosc(uint32_t finaldiv)
+{
+  
+  PRCI_use_pll(1, // Use HFXTAL
+	       1, // Bypass = 1
+	       0, // PLL settings don't matter
+	       0, // PLL settings don't matter
+	       0, // PLL settings don't matter
+	       finaldiv,
+	       -1,
+	       -1);
+}
+
+void PRCI_use_bypass_clock(void)
+{
+    // Make sure the HFROSC is on before the next line:
+    PRCI_REG(PRCI_HFROSCCFG) |= ROSC_EN(1);
+
+    // Run off 16 MHz crystal
+    PRCI_REG(PRCI_PLLCFG) = (PLL_REFSEL(1) | PLL_BYPASS(1));
+    PRCI_REG(PRCI_PLLCFG) |= (PLL_SEL(1));
+
+    // Turn off HFROSC to save power
+    PRCI_REG(PRCI_HFROSCCFG) &= ~(ROSC_EN(1));
+}
+
+
+// This is a generic function, which
+// doesn't span the entire range of HFROSC settings.
+// It only adjusts the trim, which can span a hundred MHz or so.
+// This function does not check the legality of the PLL settings
+// at all, and it is quite possible to configure invalid PLL settings
+// this way.
+// It returns the actual measured CPU frequency.
+
+
+uint32_t PRCI_set_hfrosctrim_for_f_cpu(uint32_t f_cpu, PRCI_freq_target target )
+{
+
+  uint32_t hfrosctrim = 0;
+  uint32_t hfroscdiv = 4;
+  uint32_t prev_trim = 0;
+
+  // In this function we use PLL settings which
+  // will give us a 32x multiplier from the output
+  // of the HFROSC source to the output of the
+  // PLL. We first measure our HFROSC to get the
+  // right trim, then finally use it as the PLL source.
+  // We should really check here that the f_cpu
+  // requested is something in the limit of the PLL. For
+  // now that is up to the user.
+
+  // This will undershoot for frequencies not divisible by 16.
+  uint32_t desired_hfrosc_freq = (f_cpu/ 16);
+
+  PRCI_use_hfrosc(hfroscdiv, hfrosctrim);
+  
+  // Ignore the first run (for icache reasons)
+  uint32_t cpu_freq = PRCI_measure_mcycle_freq(3000, RTC_FREQ);
+
+  cpu_freq = PRCI_measure_mcycle_freq(3000, RTC_FREQ);
+  uint32_t prev_freq = cpu_freq;
+  
+  while ((cpu_freq < desired_hfrosc_freq) && (hfrosctrim < 0x1F)){
+    prev_trim = hfrosctrim;
+    prev_freq = cpu_freq;
+    hfrosctrim ++;
+    PRCI_use_hfrosc(hfroscdiv, hfrosctrim);
+    cpu_freq = PRCI_measure_mcycle_freq(3000, RTC_FREQ);
+  } 
+
+  // We couldn't go low enough
+  if (prev_freq > desired_hfrosc_freq){
+    PRCI_use_pll(0, 0, 1, 31, 1, 1, hfroscdiv, prev_trim);
+    cpu_freq = PRCI_measure_mcycle_freq(1000, RTC_FREQ);
+    return cpu_freq;
+  }
+  
+  // We couldn't go high enough
+  if (cpu_freq < desired_hfrosc_freq){
+    PRCI_use_pll(0, 0, 1, 31, 1, 1, hfroscdiv, prev_trim);
+    cpu_freq = PRCI_measure_mcycle_freq(1000, RTC_FREQ);
+    return cpu_freq;
+  }
+
+  // Check for over/undershoot
+  switch(target) {
+  case(PRCI_FREQ_CLOSEST):
+    if ((desired_hfrosc_freq - prev_freq) < (cpu_freq - desired_hfrosc_freq)) {
+      PRCI_use_pll(0, 0, 1, 31, 1, 1, hfroscdiv, prev_trim);
+    } else {
+      PRCI_use_pll(0, 0, 1, 31, 1, 1, hfroscdiv, hfrosctrim);
+    }
+    break;
+  case(PRCI_FREQ_UNDERSHOOT):
+    PRCI_use_pll(0, 0, 1, 31, 1, 1, hfroscdiv, prev_trim);
+    break;
+  default:
+    PRCI_use_pll(0, 0, 1, 31, 1, 1, hfroscdiv, hfrosctrim);
+  }
+
+  cpu_freq =  PRCI_measure_mcycle_freq(1000, RTC_FREQ);
+  return cpu_freq;
+
+}
+
diff --git a/makefiles/arch/riscv.inc.mk b/makefiles/arch/riscv.inc.mk
new file mode 100644
index 0000000000..549e3d87f3
--- /dev/null
+++ b/makefiles/arch/riscv.inc.mk
@@ -0,0 +1,19 @@
+# Target architecture for the build.
+export TARGET_ARCH ?= riscv-none-embed
+
+# define build specific options
+CFLAGS_CPU   = -march=rv32imac -mabi=ilp32 -mcmodel=medlow -msmall-data-limit=8
+CFLAGS_LINK  = -nostartfiles -ffunction-sections -fdata-sections
+CFLAGS_DBG  ?= -g3 -Og
+#CFLAGS_OPT  ?= -Os
+
+export LINKFLAGS += -L$(RIOTCPU)/$(CPU)/ldscripts
+export LINKER_SCRIPT ?= $(CPU_MODEL).ld
+export LINKFLAGS += -T$(LINKER_SCRIPT)
+
+# export compiler flags
+export CFLAGS += $(CFLAGS_CPU) $(CFLAGS_DBG) $(CFLAGS_OPT) $(CFLAGS_LINK)
+# export assmebly flags
+export ASFLAGS += $(CFLAGS_CPU) $(CFLAGS_DBG)
+# export linker flags
+export LINKFLAGS += $(CFLAGS_CPU) $(CFLAGS_LINK) $(CFLAGS_DBG) $(CFLAGS_OPT) -Wl,--gc-sections -static -lgcc
-- 
GitLab