summaryrefslogtreecommitdiffstats
path: root/ble/src/main.rs
blob: 0fed859c7d276738d9c7e78bfab84986506e7db0 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
//! BLE interface.

#![no_std]
#![no_main]

mod logger;
mod macros;
mod rtc;

use clint::HandlerArray;
use core::mem;
use cortex_m::asm::wfi;
use cortex_m_rt::{entry, exception, ExceptionFrame};
use log::{info, trace, LevelFilter};

#[allow(unused_imports)]
extern crate panic_semihosting;

use nrf52840_mdk_bsp::{
    hal::{
        target::{interrupt, Interrupt, UARTE0},
        Clocks, Rtc,
    },
    Board,
};

// TODO:
//   * set up serial reader for trinket
//   * set up i²c interface for keyboard reports
//   * and, finally, bluetooth

// Interrupt handler table.
static HANDLERS: HandlerArray = HandlerArray::new();

#[entry]
fn main() -> ! {
    let nrf52 = Board::take().unwrap();

    let uart_wrapped = logger::WriteWrapper::new(nrf52.cdc);
    let logger = logger::SerialLogger::new(uart_wrapped);
    let logger_ref: &'static logger::SerialLogger<UARTE0> = unsafe { mem::transmute(&logger) };
    log::set_logger(logger_ref).expect("setting logger");
    log::set_max_level(LevelFilter::Trace);

    nrf52.RTC0.intenset.write(|w| w.tick().set());

    let mut nvic = nrf52.NVIC;
    let mut rtc_handler = rtc::setup(Rtc::new(nrf52.RTC0), Clocks::new(nrf52.CLOCK));

    HANDLERS.with_overrides(|hs| {
        hs.register(0, &mut rtc_handler);
        nvic.enable(Interrupt::RTC0);

        info!("Bootstrap complete.");

        let mut last_tick = rtc::millis();
        last_tick -= last_tick % 1024;
        loop {
            let tick = rtc::millis();
            if tick >= last_tick + 1024 {
                last_tick = tick;
                last_tick -= last_tick % 1024;
                trace!(".");
            }
            wfi();
        }
    });
    unreachable!();
}

#[exception]
fn HardFault(ef: &ExceptionFrame) -> ! {
    log::logger().flush();
    logln_now!("!!! Hard Fault - ef: {:?} !!!", ef);
    logln_now!("flushing log");
    loop {
        log::logger().flush();
        wfi()
    }
}

#[interrupt]
fn RTC0() {
    HANDLERS.call(0);
}