diff --git a/.cargo/config b/.cargo/config deleted file mode 100644 index 92823f72145afc5aa99dda78574337db7c8ffafc..0000000000000000000000000000000000000000 --- a/.cargo/config +++ /dev/null @@ -1,7 +0,0 @@ -# Target configuration for the CTAP2 app on the nRF52840 chip -[target.thumbv7em-none-eabi] -rustflags = [ - "-C", "link-arg=-Tnrf52840dk_layout.ld", - "-C", "relocation-model=static", - "-D", "warnings", -] diff --git a/README.md b/README.md index 34d7a9bedb7dde5b5c493fd783e20e4f0d78eec9..48e69a6210df0c213908f53bc27bdb0e62e29615 100644 --- a/README.md +++ b/README.md @@ -19,8 +19,9 @@ successfully tested on the following boards: ## Disclaimer -This project is proof-of-concept and a research platform. It's still under -development and as such comes with a few limitations: +This project is **proof-of-concept and a research platform**. It is **NOT** +meant for a day to day usage. It's still under development and as such comes +with a few limitations: ### FIDO2 @@ -53,25 +54,17 @@ For a more detailed guide, please refer to our ./setup.sh ``` -2. If Tock OS is already installed on your board, move to the next step. - Otherwise, just run one of the following commands, depending on the board - you have: +2. Next step is to install Tock OS as well as the OpenSK application on your + board (**Warning**: it will erase the locally stored credentials). Run: ```shell # Nordic nRF52840-DK board - ./deploy.py os --board=nrf52840_dk + ./deploy.py --board=nrf52840dk --opensk # Nordic nRF52840-Dongle - ./deploy.py os --board=nrf52840_dongle + ./deploy.py --board=nrf52840_dongle --opensk ``` -3. Next step is to install/update the OpenSK application on your board - (**Warning**: it will erase the locally stored credentials). Run: - - ```shell - ./deploy.py app --opensk - ``` - -4. On Linux, you may want to avoid the need for `root` privileges to interact +3. On Linux, you may want to avoid the need for `root` privileges to interact with the key. For that purpose we provide a udev rule file that can be installed with the following command: diff --git a/boards/nrf52840_dongle_dfu/Cargo.toml b/boards/nrf52840_dongle_dfu/Cargo.toml new file mode 100644 index 0000000000000000000000000000000000000000..6944eb33fc308caba08ec539d5d6af5783d4e3be --- /dev/null +++ b/boards/nrf52840_dongle_dfu/Cargo.toml @@ -0,0 +1,30 @@ +[package] +name = "nrf52840_dongle_dfu" +version = "0.1.0" +authors = ["Tock Project Developers <tock-dev@googlegroups.com>"] +build = "build.rs" +edition = "2018" + +[profile.dev] +panic = "abort" +lto = false +opt-level = "z" +debug = true + +[profile.release] +panic = "abort" +lto = true +opt-level = "z" +debug = true + +[[bin]] +path = "../../third_party/tock/boards/nordic/nrf52840_dongle/src/main.rs" +name = "nrf52840_dongle_dfu" + +[dependencies] +components = { path = "../../third_party/tock/boards/components" } +cortexm4 = { path = "../../third_party/tock/arch/cortex-m4" } +capsules = { path = "../../third_party/tock/capsules" } +kernel = { path = "../../third_party/tock/kernel" } +nrf52840 = { path = "../../third_party/tock/chips/nrf52840" } +nrf52dk_base = { path = "../../third_party/tock/boards/nordic/nrf52dk_base" } diff --git a/boards/nrf52840_dongle_dfu/build.rs b/boards/nrf52840_dongle_dfu/build.rs new file mode 100644 index 0000000000000000000000000000000000000000..2631dccb75f21f432f5186b17215367490e2d3f2 --- /dev/null +++ b/boards/nrf52840_dongle_dfu/build.rs @@ -0,0 +1,4 @@ +fn main() { + println!("cargo:rerun-if-changed=layout.ld"); + println!("cargo:rerun-if-changed=../../third_party/tock/boards/kernel_layout.ld"); +} diff --git a/boards/nrf52840_dongle_dfu/layout.ld b/boards/nrf52840_dongle_dfu/layout.ld new file mode 100644 index 0000000000000000000000000000000000000000..834133c00b8d0adb87c9e8eb355ed0b2e91c0582 --- /dev/null +++ b/boards/nrf52840_dongle_dfu/layout.ld @@ -0,0 +1,10 @@ +MEMORY +{ + rom (rx) : ORIGIN = 0x00001000, LENGTH = 188K + prog (rx) : ORIGIN = 0x00030000, LENGTH = 832K + ram (rwx) : ORIGIN = 0x20000000, LENGTH = 256K +} + +MPU_MIN_ALIGN = 8K; + +INCLUDE ../../third_party/tock/boards/kernel_layout.ld diff --git a/boards/nrf52840_mdk_dfu/Cargo.toml b/boards/nrf52840_mdk_dfu/Cargo.toml new file mode 100644 index 0000000000000000000000000000000000000000..1047869ba67c2d46b0af09d0ec7dd214a051fa43 --- /dev/null +++ b/boards/nrf52840_mdk_dfu/Cargo.toml @@ -0,0 +1,26 @@ +[package] +name = "nrf52840_mdk_dfu" +version = "0.1.0" +authors = ["Yihui Xiong <yihui.xiong@hotmail.com>"] +build = "build.rs" +edition = "2018" + +[profile.dev] +panic = "abort" +lto = false +opt-level = "z" +debug = true + +[profile.release] +panic = "abort" +lto = true +opt-level = "z" +debug = true + +[dependencies] +components = { path = "../../third_party/tock/boards/components" } +cortexm4 = { path = "../../third_party/tock/arch/cortex-m4" } +capsules = { path = "../../third_party/tock/capsules" } +kernel = { path = "../../third_party/tock/kernel" } +nrf52840 = { path = "../../third_party/tock/chips/nrf52840" } +nrf52dk_base = { path = "../../third_party/tock/boards/nordic/nrf52dk_base" } diff --git a/boards/nrf52840_mdk_dfu/build.rs b/boards/nrf52840_mdk_dfu/build.rs new file mode 100644 index 0000000000000000000000000000000000000000..2631dccb75f21f432f5186b17215367490e2d3f2 --- /dev/null +++ b/boards/nrf52840_mdk_dfu/build.rs @@ -0,0 +1,4 @@ +fn main() { + println!("cargo:rerun-if-changed=layout.ld"); + println!("cargo:rerun-if-changed=../../third_party/tock/boards/kernel_layout.ld"); +} diff --git a/boards/nrf52840_mdk_dfu/layout.ld b/boards/nrf52840_mdk_dfu/layout.ld new file mode 100644 index 0000000000000000000000000000000000000000..834133c00b8d0adb87c9e8eb355ed0b2e91c0582 --- /dev/null +++ b/boards/nrf52840_mdk_dfu/layout.ld @@ -0,0 +1,10 @@ +MEMORY +{ + rom (rx) : ORIGIN = 0x00001000, LENGTH = 188K + prog (rx) : ORIGIN = 0x00030000, LENGTH = 832K + ram (rwx) : ORIGIN = 0x20000000, LENGTH = 256K +} + +MPU_MIN_ALIGN = 8K; + +INCLUDE ../../third_party/tock/boards/kernel_layout.ld diff --git a/boards/nrf52840_mdk_dfu/src/io.rs b/boards/nrf52840_mdk_dfu/src/io.rs new file mode 100644 index 0000000000000000000000000000000000000000..1f5f5a6f98501aa52cf69a394025471111a6a4f7 --- /dev/null +++ b/boards/nrf52840_mdk_dfu/src/io.rs @@ -0,0 +1,65 @@ +use core::fmt::Write; +use core::panic::PanicInfo; +use cortexm4; +use kernel::debug; +use kernel::debug::IoWrite; +use kernel::hil::led; +use kernel::hil::uart::{self, Configure}; +use nrf52840::gpio::Pin; + +use crate::CHIP; +use crate::PROCESSES; + +struct Writer { + initialized: bool, +} + +static mut WRITER: Writer = Writer { initialized: false }; + +impl Write for Writer { + fn write_str(&mut self, s: &str) -> ::core::fmt::Result { + self.write(s.as_bytes()); + Ok(()) + } +} + +impl IoWrite for Writer { + fn write(&mut self, buf: &[u8]) { + let uart = unsafe { &mut nrf52840::uart::UARTE0 }; + if !self.initialized { + self.initialized = true; + uart.configure(uart::Parameters { + baud_rate: 115200, + stop_bits: uart::StopBits::One, + parity: uart::Parity::None, + hw_flow_control: false, + width: uart::Width::Eight, + }); + } + for &c in buf { + unsafe { + uart.send_byte(c); + } + while !uart.tx_ready() {} + } + } +} + +#[cfg(not(test))] +#[no_mangle] +#[panic_handler] +/// Panic handler +pub unsafe extern "C" fn panic_fmt(pi: &PanicInfo) -> ! { + // The nRF52840 Dongle LEDs (see back of board) + const LED1_PIN: Pin = Pin::P0_23; + let led = &mut led::LedLow::new(&mut nrf52840::gpio::PORT[LED1_PIN]); + let writer = &mut WRITER; + debug::panic( + &mut [led], + writer, + pi, + &cortexm4::support::nop, + &PROCESSES, + &CHIP, + ) +} diff --git a/boards/nrf52840_mdk_dfu/src/main.rs b/boards/nrf52840_mdk_dfu/src/main.rs new file mode 100644 index 0000000000000000000000000000000000000000..0dc58d8a0de981a381bcd89b6f55c3b0ea21458b --- /dev/null +++ b/boards/nrf52840_mdk_dfu/src/main.rs @@ -0,0 +1,123 @@ +//! Tock kernel for the Makerdiary nRF52840 MDK USB dongle. +//! +//! It is based on nRF52840 SoC (Cortex M4 core with a BLE transceiver) with +//! many exported I/O and peripherals. + +#![no_std] +#![no_main] +#![deny(missing_docs)] + +use kernel::component::Component; +#[allow(unused_imports)] +use kernel::{debug, debug_gpio, debug_verbose, static_init}; +use nrf52840::gpio::Pin; +use nrf52dk_base::{SpiPins, UartChannel, UartPins}; + +// The nRF52840 MDK USB Dongle LEDs +const LED1_R_PIN: Pin = Pin::P0_23; +const LED1_G_PIN: Pin = Pin::P0_22; +const LED1_B_PIN: Pin = Pin::P0_24; + +// The nRF52840 Dongle button +const BUTTON_PIN: Pin = Pin::P0_18; +const BUTTON_RST_PIN: Pin = Pin::P0_02; + +const UART_RTS: Pin = Pin::P0_21; +const UART_TXD: Pin = Pin::P0_20; +const UART_CTS: Pin = Pin::P0_03; +const UART_RXD: Pin = Pin::P0_19; + +const SPI_MOSI: Pin = Pin::P0_05; +const SPI_MISO: Pin = Pin::P0_06; +const SPI_CLK: Pin = Pin::P0_07; + +/// UART Writer +pub mod io; + +// State for loading and holding applications. +// How should the kernel respond when a process faults. +const FAULT_RESPONSE: kernel::procs::FaultResponse = kernel::procs::FaultResponse::Panic; + +// Number of concurrent processes this platform supports. +const NUM_PROCS: usize = 8; + +// RAM to be shared by all application processes. +#[link_section = ".app_memory"] +static mut APP_MEMORY: [u8; 0x3C000] = [0; 0x3C000]; + +static mut PROCESSES: [Option<&'static dyn kernel::procs::ProcessType>; NUM_PROCS] = + [None, None, None, None, None, None, None, None]; + +// Static reference to chip for panic dumps +static mut CHIP: Option<&'static nrf52840::chip::Chip> = None; + +/// Dummy buffer that causes the linker to reserve enough space for the stack. +#[no_mangle] +#[link_section = ".stack_buffer"] +pub static mut STACK_MEMORY: [u8; 0x1000] = [0; 0x1000]; + +/// Entry point in the vector table called on hard reset. +#[no_mangle] +pub unsafe fn reset_handler() { + // Loads relocations and clears BSS + nrf52840::init(); + + let board_kernel = static_init!(kernel::Kernel, kernel::Kernel::new(&PROCESSES)); + // GPIOs + let gpio = components::gpio::GpioComponent::new(board_kernel).finalize( + components::gpio_component_helper!( + &nrf52840::gpio::PORT[Pin::P0_04], + &nrf52840::gpio::PORT[Pin::P0_05], + &nrf52840::gpio::PORT[Pin::P0_06], + &nrf52840::gpio::PORT[Pin::P0_07], + &nrf52840::gpio::PORT[Pin::P0_08] + ), + ); + let button = components::button::ButtonComponent::new(board_kernel).finalize( + components::button_component_helper!(( + &nrf52840::gpio::PORT[BUTTON_PIN], + capsules::button::GpioMode::LowWhenPressed, + kernel::hil::gpio::FloatingState::PullUp + )), + ); + + let led = components::led::LedsComponent::new().finalize(components::led_component_helper!( + ( + &nrf52840::gpio::PORT[LED1_R_PIN], + capsules::led::ActivationMode::ActiveLow + ), + ( + &nrf52840::gpio::PORT[LED1_G_PIN], + capsules::led::ActivationMode::ActiveLow + ), + ( + &nrf52840::gpio::PORT[LED1_B_PIN], + capsules::led::ActivationMode::ActiveLow + ) + )); + let chip = static_init!(nrf52840::chip::Chip, nrf52840::chip::new()); + CHIP = Some(chip); + + nrf52dk_base::setup_board( + board_kernel, + BUTTON_RST_PIN, + &nrf52840::gpio::PORT, + gpio, + LED1_R_PIN, + LED1_G_PIN, + LED1_B_PIN, + led, + UartChannel::Pins(UartPins::new(UART_RTS, UART_TXD, UART_CTS, UART_RXD)), + &SpiPins::new(SPI_MOSI, SPI_MISO, SPI_CLK), + &None, + button, + true, + &mut APP_MEMORY, + &mut PROCESSES, + FAULT_RESPONSE, + nrf52840::uicr::Regulator0Output::V3_0, + false, + &Some(&nrf52840::usbd::USBD), + chip, + ); +} diff --git a/deploy.py b/deploy.py index dc69c24ea4921d73e03577649edbcbdf995d677b..1f000484cb05f608b096047b180fda3996ee99bc 100755 --- a/deploy.py +++ b/deploy.py @@ -20,6 +20,7 @@ from __future__ import division from __future__ import print_function import argparse +import collections import copy import os import shutil @@ -32,10 +33,115 @@ from tockloader import tbfh from tockloader import tockloader as loader from tockloader.exceptions import TockLoaderException -# This structure allows us in the future to also support out-of-tree boards. +PROGRAMMERS = frozenset(("jlink", "openocd", "pyocd", "nordicdfu", "none")) + +# This structure allows us to support out-of-tree boards as well as (in the +# future) more achitectures. +OpenSKBoard = collections.namedtuple( + "OpenSKBoard", + [ + # Location of the Tock board (where the Makefile file is + "path", + # Targeted architecture (e.g. thumbv7em-none-eabi) + "arch", + # Size of 1 page of flash memory + "page_size", + # Flash address at which the kernel will be written + "kernel_address", + # Set to None is padding is not required for the board + "padding_address", + # Linker script to produce a working app for this board + "app_ldscript", + # Flash address at which the app should be written + "app_address", + # Target name for flashing the board using pyOCD + "pyocd_target", + # The cfg file in OpenOCD board folder + "openocd_board", + # Options to tell Tockloader how to work with OpenOCD + # Default: [] + "openocd_options", + # Dictionnary specifying custom commands for OpenOCD + # Default is an empty dict + # Valid keys are: program, read, erase + "openocd_commands", + # Interface to use with JLink (e.g. swd, jtag, etc.) + "jlink_if", + # Device name as supported by JLinkExe + "jlink_device", + # Whether Nordic DFU flashing method is supported + "nordic_dfu", + ]) + SUPPORTED_BOARDS = { - "nrf52840_dk": "third_party/tock/boards/nordic/nrf52840dk", - "nrf52840_dongle": "third_party/tock/boards/nordic/nrf52840_dongle" + "nrf52840dk": + OpenSKBoard( + path="third_party/tock/boards/nordic/nrf52840dk", + arch="thumbv7em-none-eabi", + page_size=4096, + kernel_address=0, + padding_address=0x30000, + app_ldscript="nrf52840dk_layout.ld", + app_address=0x40000, + pyocd_target="nrf52840", + openocd_board="nordic_nrf52840_dongle.cfg", + openocd_options=[], + openocd_commands={}, + jlink_if="swd", + jlink_device="nrf52840_xxaa", + nordic_dfu=False, + ), + "nrf52840_dongle": + OpenSKBoard( + path="third_party/tock/boards/nordic/nrf52840_dongle", + arch="thumbv7em-none-eabi", + page_size=4096, + kernel_address=0, + padding_address=0x30000, + app_ldscript="nrf52840dk_layout.ld", + app_address=0x40000, + pyocd_target="nrf52840", + openocd_board="nordic_nrf52840_dongle.cfg", + openocd_options=[], + openocd_commands={}, + jlink_if="swd", + jlink_device="nrf52840_xxaa", + nordic_dfu=False, + ), + "nrf52840_dongle_dfu": + OpenSKBoard( + path="boards/nrf52840_dongle_dfu", + arch="thumbv7em-none-eabi", + page_size=4096, + kernel_address=0x1000, + padding_address=0x30000, + app_ldscript="nrf52840dk_layout.ld", + app_address=0x40000, + pyocd_target="nrf52840", + openocd_board="nordic_nrf52840_dongle.cfg", + openocd_options=[], + openocd_commands={}, + jlink_if="swd", + jlink_device="nrf52840_xxaa", + nordic_dfu=True, + ), + "nrf52840_mdk_dfu": + OpenSKBoard( + path="boards/nrf52840_mdk_dfu", + arch="thumbv7em-none-eabi", + page_size=4096, + kernel_address=0x1000, + padding_address=0x30000, + app_ldscript="nrf52840dk_layout.ld", + app_address=0x40000, + pyocd_target="nrf52840", + openocd_board="nordic_nrf52840_dongle.cfg", + openocd_options=[], + openocd_commands={}, + jlink_if="swd", + jlink_device="nrf52840_xxaa", + nordic_dfu=True, + ), } # The STACK_SIZE value below must match the one used in the linker script @@ -50,9 +156,9 @@ APP_HEAP_SIZE = 90000 def get_supported_boards(): boards = [] - for name, root in SUPPORTED_BOARDS.items(): - if all((os.path.exists(os.path.join(root, "Cargo.toml")), - os.path.exists(os.path.join(root, "Makefile")))): + for name, props in SUPPORTED_BOARDS.items(): + if all((os.path.exists(os.path.join(props.path, "Cargo.toml")), + (props.app_ldscript and os.path.exists(props.app_ldscript)))): boards.append(name) return tuple(set(boards)) @@ -79,9 +185,23 @@ def info(msg): message=msg)) +def assert_mandatory_binary(binary): + if not shutil.which(binary): + fatal(("Couldn't find {} binary. Make sure it is installed and " + "that your PATH is set correctly.").format(binary)) + + +def assert_python_library(module): + try: + __import__(module) + except ModuleNotFoundError: + fatal(("Couldn't load python3 module {name}. " + "Try to run: pip3 install {name}").format(name=module)) + + class RemoveConstAction(argparse.Action): - # pylint: disable=W0622 + # pylint: disable=redefined-builtin def __init__(self, option_strings, dest, @@ -121,28 +241,34 @@ class OpenSKInstaller: self.args = args # Where all the TAB files should go self.tab_folder = os.path.join("target", "tab") - # This is the filename that elf2tab command expects in order - # to create a working TAB file. - self.target_elf_filename = os.path.join(self.tab_folder, "cortex-m4.elf") + board = SUPPORTED_BOARDS[self.args.board] self.tockloader_default_args = argparse.Namespace( - arch="cortex-m4", - board=getattr(self.args, "board", "nrf52840"), + arch=board.arch, + board=self.args.board, debug=False, force=False, - jlink=True, - jlink_device="nrf52840_xxaa", - jlink_if="swd", + jlink=self.args.programmer == "jlink", + jlink_device=board.jlink_device, + jlink_if=board.jlink_if, jlink_speed=1200, + openocd=self.args.programmer == "openocd", + openocd_board=board.openocd_board, jtag=False, no_bootloader_entry=False, - page_size=4096, + page_size=board.page_size, port=None, ) - def checked_command_output(self, cmd): + def checked_command_output(self, cmd, env=None, cwd=None): cmd_output = "" try: - cmd_output = subprocess.check_output(cmd) + cmd_output = subprocess.run( + cmd, + stdout=subprocess.PIPE, + timeout=None, + check=True, + env=env, + cwd=cwd).stdout except subprocess.CalledProcessError as e: fatal("Failed to execute {}: {}".format(cmd[0], str(e))) # Unreachable because fatal() will exit @@ -166,38 +292,104 @@ class OpenSKInstaller: # Need to update self.checked_command_output( ["rustup", "install", target_toolchain_fullstring]) - self.checked_command_output( - ["rustup", "target", "add", "thumbv7em-none-eabi"]) + targets = set([x.arch for _, x in SUPPORTED_BOARDS.items()]) + for arch in targets: + self.checked_command_output(["rustup", "target", "add", arch]) info("Rust toolchain up-to-date") - def build_and_install_tockos(self): + def search_binary(self, name, start_directory="."): + for root, _, files in os.walk(start_directory): + for fname in files: + if fname == name: + return os.path.join(root, fname) + return None + + def build_tockos(self): + info("Building Tock OS for board {}".format(self.args.board)) + props = SUPPORTED_BOARDS[self.args.board] + out_directory = os.path.join(props.path, "target", props.arch, "release") + os.makedirs(out_directory, exist_ok=True) + rust_flags = [ + "-C", + "link-arg=-Tlayout.ld", + "-C", + "linker=rust-lld", + "-C", + "linker-flavor=ld.lld", + "-C", + "relocation-model=dynamic-no-pic", + "-C", + "link-arg=-zmax-page-size=512", + "--remap-path-prefix={}=".format(os.path.realpath(props.path)), + ] + env = os.environ.copy() + env["RUSTFLAGS"] = " ".join(rust_flags) self.checked_command_output( - ["make", "-C", SUPPORTED_BOARDS[self.args.board], "flash"]) - - def build_and_install_example(self): - assert self.args.application + ["cargo", "build", "--release", "--target={}".format(props.arch)], + env=env, + cwd=props.path) + info("Converting Tock OS file into a binary") + kernel_name = os.path.basename(props.path) + shutil.copyfile( + os.path.join(out_directory, kernel_name), + os.path.join(out_directory, "{}.elf".format(kernel_name))) + # Find appropriate llvm-objcopy + llvm_dir = self.checked_command_output(["rustc", "--print=sysroot"], + cwd=props.path).strip() + if not llvm_dir: + fatal("Couldn't determine where rustc is installed. " + "This shouldn't happen.") + if not os.path.isdir(llvm_dir): + fatal("Something went wrong while locating llvm-objcopy.") + objcopy = self.search_binary("llvm-objcopy", start_directory=llvm_dir) + if not objcopy: + fatal("Couldn't locate llvm-objcopy binary in your system.") self.checked_command_output([ - "cargo", "build", "--release", "--target=thumbv7em-none-eabi", - "--features={}".format(",".join(self.args.features)), "--example", - self.args.application + objcopy, "--output-target=binary", + os.path.join(out_directory, "{}.elf".format(kernel_name)), + os.path.join(out_directory, "{}.bin".format(kernel_name)) ]) - self.install_elf_file( - os.path.join("target/thumbv7em-none-eabi/release/examples", - self.args.application)) - def build_and_install_opensk(self): - assert self.args.application + def build_example(self): + info("Building example {}".format(self.args.application)) + self._build_app_or_example(is_example=True) + + def build_opensk(self): info("Building OpenSK application") - self.checked_command_output([ - "cargo", - "build", - "--release", - "--target=thumbv7em-none-eabi", - "--features={}".format(",".join(self.args.features)), - ]) - self.install_elf_file( - os.path.join("target/thumbv7em-none-eabi/release", - self.args.application)) + self._build_app_or_example(is_example=False) + + def _build_app_or_example(self, is_example): + assert self.args.application + # Ideally we would build a TAB file for all boards at once but depending on + # the chip on the board, the link script could be totally different. + # And elf2tab doesn't seem to let us set the boards a TAB file has been + # created for. So at the moment we only build for the selected board. + props = SUPPORTED_BOARDS[self.args.board] + rust_flags = [ + "-C", + "link-arg=-T{}".format(props.app_ldscript), + "-C", + "relocation-model=static", + "-D", + "warnings", + "--remap-path-prefix={}=".format(os.getcwd()), + ] + env = os.environ.copy() + env["RUSTFLAGS"] = " ".join(rust_flags) + + command = [ + "cargo", "build", "--release", "--target={}".format(props.arch), + "--features={}".format(",".join(self.args.features)) + ] + if is_example: + command.extend(["--example", self.args.application]) + self.checked_command_output(command, env=env) + app_path = os.path.join("target", props.arch, "release") + if is_example: + app_path = os.path.join(app_path, "examples") + app_path = os.path.join(app_path, self.args.application) + # Create a TAB file + self.create_tab_file({props.arch: app_path}) def generate_crypto_materials(self, force_regenerate): has_error = subprocess.call([ @@ -208,8 +400,11 @@ class OpenSKInstaller: error(("Something went wrong while trying to generate ECC " "key and/or certificate for OpenSK")) - def install_elf_file(self, elf_path): + def create_tab_file(self, binaries): + assert binaries assert self.args.application + info("Generating Tock TAB file for application/example {}".format( + self.args.application)) package_parameter = "-n" elf2tab_ver = self.checked_command_output(["elf2tab", "--version"]).split( " ", maxsplit=1)[1] @@ -221,17 +416,26 @@ class OpenSKInstaller: os.makedirs(self.tab_folder, exist_ok=True) tab_filename = os.path.join(self.tab_folder, "{}.tab".format(self.args.application)) - shutil.copyfile(elf_path, self.target_elf_filename) - self.checked_command_output([ - "elf2tab", package_parameter, self.args.application, "-o", tab_filename, - self.target_elf_filename, "--stack={}".format(STACK_SIZE), - "--app-heap={}".format(APP_HEAP_SIZE), "--kernel-heap=1024", - "--protected-region-size=64" + elf2tab_args = [ + "elf2tab", package_parameter, self.args.application, "-o", tab_filename + ] + for arch, app_file in binaries.items(): + dest_file = os.path.join(self.tab_folder, "{}.elf".format(arch)) + shutil.copyfile(app_file, dest_file) + elf2tab_args.append(dest_file) + + elf2tab_args.extend([ + "--stack={}".format(STACK_SIZE), "--app-heap={}".format(APP_HEAP_SIZE), + "--kernel-heap=1024", "--protected-region-size=64" ]) - self.install_padding() + self.checked_command_output(elf2tab_args) + + def install_tab_file(self, tab_filename): + assert self.args.application info("Installing Tock application {}".format(self.args.application)) + board_props = SUPPORTED_BOARDS[self.args.board] args = copy.copy(self.tockloader_default_args) - setattr(args, "app_address", 0x40000) + setattr(args, "app_address", board_props.app_address) setattr(args, "erase", self.args.clear_apps) setattr(args, "make", False) setattr(args, "no_replace", False) @@ -244,16 +448,38 @@ class OpenSKInstaller: fatal("Couldn't install Tock application {}: {}".format( self.args.application, str(e))) - def install_padding(self): + def get_padding(self): fake_header = tbfh.TBFHeader("") fake_header.version = 2 fake_header.fields["header_size"] = 0x10 - fake_header.fields["total_size"] = 0x10000 + fake_header.fields["total_size"] = ( + SUPPORTED_BOARDS[self.args.board].app_address - + SUPPORTED_BOARDS[self.args.board].padding_address) fake_header.fields["flags"] = 0 - padding = fake_header.get_binary() + return fake_header.get_binary() + + def install_tock_os(self): + board_props = SUPPORTED_BOARDS[self.args.board] + kernel_file = os.path.join(board_props.path, "target", board_props.arch, + "release", "{}.bin".format(self.args.board)) + info("Flashing file {}.".format(kernel_file)) + with open(kernel_file, "rb") as f: + kernel = f.read() + args = copy.copy(self.tockloader_default_args) + setattr(args, "address", board_props.app_address) + tock = loader.TockLoader(args) + tock.open(args) + try: + tock.flash_binary(kernel, board_props.kernel_address) + except TockLoaderException as e: + fatal("Couldn't install padding: {}".format(str(e))) + + def install_padding(self): + padding = self.get_padding() + board_props = SUPPORTED_BOARDS[self.args.board] info("Flashing padding application") args = copy.copy(self.tockloader_default_args) - setattr(args, "address", 0x30000) + setattr(args, "address", board_props.padding_address) tock = loader.TockLoader(args) tock.open(args) try: @@ -263,7 +489,8 @@ class OpenSKInstaller: def clear_apps(self): args = copy.copy(self.tockloader_default_args) - setattr(args, "app_address", 0x40000) + board_props = SUPPORTED_BOARDS[self.args.board] + setattr(args, "app_address", board_props.app_address) info("Erasing all installed applications") tock = loader.TockLoader(args) tock.open(args) @@ -271,11 +498,13 @@ class OpenSKInstaller: tock.erase_apps(False) except TockLoaderException as e: # Erasing apps is not critical - info(("A non-critical error occured while erasing " + info(("A non-critical error occurred while erasing " "apps: {}".format(str(e)))) - # pylint: disable=W0212 + # pylint: disable=protected-access def verify_flashed_app(self, expected_app): + if self.args.programmer not in ("jlink", "openocd"): + return False args = copy.copy(self.tockloader_default_args) tock = loader.TockLoader(args) app_found = False @@ -284,51 +513,192 @@ class OpenSKInstaller: app_found = expected_app in apps return app_found + def create_hex_file(self, dest_file): + # We produce an intelhex file with everything in it + # pylint: disable=g-import-not-at-top,import-outside-toplevel + import intelhex + board_props = SUPPORTED_BOARDS[self.args.board] + final_hex = intelhex.IntelHex() + + if self.args.tockos: + # Process kernel + kernel_path = os.path.join(board_props.path, "target", board_props.arch, + "release", "{}.bin".format(self.args.board)) + with open(kernel_path, "rb") as kernel: + kern_hex = intelhex.IntelHex() + kern_hex.frombytes(kernel.read(), offset=board_props.kernel_address) + final_hex.merge(kern_hex, overlap="error") + + # Add padding + if board_props.padding_address: + padding_hex = intelhex.IntelHex() + padding_hex.frombytes( + self.get_padding(), offset=board_props.padding_address) + final_hex.merge(padding_hex, overlap="error") + + # Now we can add the application from the TAB file + app_tab_path = "target/tab/{}.tab".format(self.args.application) + assert os.path.exists(app_tab_path) + app_tab = tab.TAB(app_tab_path) + if board_props.arch not in app_tab.get_supported_architectures(): + fatal(("It seems that the TAB file was not produced for the " + "architecture {}".format(board_props.arch))) + app_hex = intelhex.IntelHex() + app_hex.frombytes( + app_tab.extract_app(board_props.arch).get_binary(), + offset=board_props.app_address) + final_hex.merge(app_hex) + info("Generating all-merged HEX file: {}".format(dest_file)) + final_hex.tofile(dest_file, format="hex") + + def check_prerequisites(self): + if self.args.programmer == "jlink": + assert_mandatory_binary("JLinkExe") + + if self.args.programmer == "openocd": + assert_mandatory_binary("openocd") + + if self.args.programmer == "pyocd": + assert_mandatory_binary("pyocd") + assert_python_library("intelhex") + if not SUPPORTED_BOARDS[self.args.board].pyocd_target: + fatal("This board doesn't seem to support flashing through pyocd.") + + if self.args.programmer == "nordicdfu": + assert_mandatory_binary("nrfutil") + assert_python_library("intelhex") + assert_python_library("nordicsemi.lister") + if not SUPPORTED_BOARDS[self.args.board].nordic_dfu: + fatal("This board doesn't support flashing over DFU.") + + if self.args.programmer == "none": + assert_python_library("intelhex") + def run(self): - if self.args.action is None: - # Nothing to do + if self.args.listing == "boards": + print(os.linesep.join(get_supported_boards())) return 0 + if self.args.listing == "programmers": + print(os.linesep.join(PROGRAMMERS)) + return 0 + + if self.args.listing: + # Missing check? + fatal("Listing {} is not implemented.".format(self.args.listing)) + + self.check_prerequisites() self.update_rustc_if_needed() - if self.args.action == "os": - info("Installing Tock on board {}".format(self.args.board)) - self.build_and_install_tockos() - return 0 + if self.args.application is None: + fatal("Please specify an application to be flashed") + + # Compile what needs to be compiled + if self.args.tockos: + self.build_tockos() + + if self.args.application == "ctap2": + self.generate_crypto_materials(self.args.regenerate_keys) + self.build_opensk() + else: + self.build_example() - if self.args.action == "app": - if self.args.application is None: - fatal("Unspecified application") + # Flashing + board_props = SUPPORTED_BOARDS[self.args.board] + if self.args.programmer in ("jlink", "openocd"): + # We rely on Tockloader to do the job if self.args.clear_apps: self.clear_apps() - if self.args.application == "ctap2": - self.generate_crypto_materials(self.args.regenerate_keys) - self.build_and_install_opensk() - else: - self.build_and_install_example() + if self.args.tockos: + # Install Tock OS + self.install_tock_os() + # Install padding and application + self.install_padding() + self.install_tab_file("target/tab/{}.tab".format(self.args.application)) if self.verify_flashed_app(self.args.application): info("You're all set!") return 0 - error(("It seems that something went wrong. " - "App/example not found on your board.")) + error(("It seems that something went wrong. App/example not found " + "on your board. Ensure the connections between the programmer and " + "the board are correct.")) return 1 + + if self.args.programmer in ("pyocd", "nordicdfu", "none"): + dest_file = "target/{}_merged.hex".format(self.args.board) + self.create_hex_file(dest_file) + + if self.args.programmer == "pyocd": + info("Flashing HEX file") + self.checked_command_output([ + "pyocd", "flash", "--target={}".format(board_props.pyocd_target), + "--format=hex", "--erase=auto", dest_file + ]) + if self.args.programmer == "nordicdfu": + info("Creating DFU package") + dfu_pkg_file = "target/{}_dfu.zip".format(self.args.board) + self.checked_command_output([ + "nrfutil", "pkg", "generate", "--hw-version=52", "--sd-req=0", + "--application-version=1", "--application={}".format(dest_file), + dfu_pkg_file + ]) + info( + "Please insert the dongle and switch it to DFU mode by keeping the " + "button pressed while inserting...") + info("Press [ENTER] when ready.") + _ = input() + # Search for the DFU devices + serial_number = [] + # pylint: disable=g-import-not-at-top,import-outside-toplevel + from nordicsemi.lister import device_lister + for device in device_lister.DeviceLister().enumerate(): + if device.vendor_id == "1915" and device.product_id == "521F": + serial_number.append(device.serial_number) + if not serial_number: + fatal("Couldn't find any DFU device on your system.") + if len(serial_number) > 1: + fatal("Multiple DFU devices are detected. Please only connect one.") + # Run the command without capturing stdout so that we show progress + info("Flashing device using DFU...") + return subprocess.run( + [ + "nrfutil", "dfu", "usb-serial", + "--package={}".format(dfu_pkg_file), + "--serial-number={}".format(serial_number[0]) + ], + check=False, + timeout=None, + ).returncode return 0 def main(args): # Make sure the current working directory is the right one before running os.chdir(os.path.realpath(os.path.dirname(__file__))) - # Check for pre-requisite executable files. - if not shutil.which("JLinkExe"): - fatal(("Couldn't find JLinkExe binary. Make sure Segger JLink tools " - "are installed and correctly set up.")) OpenSKInstaller(args).run() if __name__ == "__main__": - shared_parser = argparse.ArgumentParser(add_help=False) - shared_parser.add_argument( + main_parser = argparse.ArgumentParser() + action_group = main_parser.add_mutually_exclusive_group(required=True) + action_group.add_argument( + "--list", + metavar="WHAT", + choices=("boards", "programmers"), + default=None, + dest="listing", + help=("List supported boards or programmers, 1 per line and then exit."), + ) + action_group.add_argument( + "--board", + metavar="BOARD_NAME", + dest="board", + default=None, + choices=get_supported_boards(), + help="Indicates which board Tock OS will be compiled for.", + ) + + main_parser.add_argument( "--dont-clear-apps", action="store_false", default=True, @@ -336,32 +706,26 @@ if __name__ == "__main__": help=("When installing an application, previously installed " "applications won't be erased from the board."), ) + main_parser.add_argument( + "--programmer", + metavar="METHOD", + dest="programmer", + choices=PROGRAMMERS, + default="jlink", + help=("Sets the method to be used to flash Tock OS or the application " + "on the target board."), + ) - main_parser = argparse.ArgumentParser() - commands = main_parser.add_subparsers( - dest="action", - help=("Indicates which part of the firmware should be compiled and " - "flashed to the connected board.")) - - os_commands = commands.add_parser( - "os", - parents=[shared_parser], - help=("Compiles and installs Tock OS. The target board must be " - "specified by setting the --board argument."), + main_parser.add_argument( + "--no-tockos", + action="store_false", + default=True, + dest="tockos", + help=("Only compiles and flash the application/example. " + "Otherwise TockOS will also be bundled and flashed."), ) - os_commands.add_argument( - "--board", - metavar="BOARD_NAME", - dest="board", - choices=get_supported_boards(), - help="Indicates which board Tock OS will be compiled for.", - required=True) - app_commands = commands.add_parser( - "app", - parents=[shared_parser], - help="compiles and installs an application.") - app_commands.add_argument( + main_parser.add_argument( "--panic-console", action="append_const", const="panic_console", @@ -370,7 +734,7 @@ if __name__ == "__main__": "output messages before starting blinking the LEDs on the " "board."), ) - app_commands.add_argument( + main_parser.add_argument( "--debug-allocations", action="append_const", const="debug_allocations", @@ -378,7 +742,7 @@ if __name__ == "__main__": help=("The console will be used to output allocator statistics every " "time an allocation/deallocation happens."), ) - app_commands.add_argument( + main_parser.add_argument( "--no-u2f", action=RemoveConstAction, const="with_ctap1", @@ -386,7 +750,7 @@ if __name__ == "__main__": help=("Compiles the OpenSK application without backward compatible " "support for U2F/CTAP1 protocol."), ) - app_commands.add_argument( + main_parser.add_argument( "--regen-keys", action="store_true", default=False, @@ -396,7 +760,7 @@ if __name__ == "__main__": "This is useful to allow flashing multiple OpenSK authenticators " "in a row without them being considered clones."), ) - app_commands.add_argument( + main_parser.add_argument( "--debug", action="append_const", const="debug_ctap", @@ -405,7 +769,7 @@ if __name__ == "__main__": "(i.e. more debug messages will be sent over the console port " "such as hexdumps of packets)."), ) - app_commands.add_argument( + main_parser.add_argument( "--no-persistent-storage", action="append_const", const="ram_storage", @@ -413,7 +777,8 @@ if __name__ == "__main__": help=("Compiles and installs the OpenSK application without persistent " "storage (i.e. unplugging the key will reset the key)."), ) - apps_group = app_commands.add_mutually_exclusive_group() + + apps_group = main_parser.add_mutually_exclusive_group() apps_group.add_argument( "--opensk", dest="application", @@ -428,6 +793,6 @@ if __name__ == "__main__": help=("Compiles and installs the crypto_bench example that tests " "the performance of the cryptographic algorithms on the board.")) - app_commands.set_defaults(features=["with_ctap1"]) + main_parser.set_defaults(features=["with_ctap1"]) main(main_parser.parse_args()) diff --git a/docs/install.md b/docs/install.md index 11a9147bd1fb0f34a60279846ede1abee0c73f12..5b5763b1b83ee2c58da2c1006a1d581e6c1daca7 100644 --- a/docs/install.md +++ b/docs/install.md @@ -16,8 +16,9 @@ You will need one the following supported boards: scenarios as the JTAG probe is already on the board. * [Nordic nRF52840 Dongle](https://www.nordicsemi.com/Software-and-tools/Development-Kits/nRF52840-Dongle) to have a more practical form factor. +* [Makerdiary nRF52840-MDK USB dongle](https://wiki.makerdiary.com/nrf52840-mdk/). -In the case of the Nordic USB dongle, you will also need the following extra +In the case of the Nordic USB dongle, you may also need the following extra hardware: * a [Segger J-Link](https://www.segger.com/products/debug-probes/j-link/) JTAG @@ -30,13 +31,18 @@ hardware: [Tag-Connect TC2050 retainer clip](http://www.tag-connect.com/TC2050-CLIP) to keep the spring loaded connector pressed to the PCB. -Although [OpenOCD](http://openocd.org/) should be supported we encountered some -issues while trying to flash a firmware with it. Therefore we suggest at the -moment to use a -[Segger J-Link](https://www.segger.com/products/debug-probes/j-link/) probe -instead. +Additionnaly, OpenSK supports other ways to flash your board: -This guide **does not** cover how to setup the JTAG probe on your system. +* [OpenOCD](http://openocd.org/). +* [Segger J-Link](https://www.segger.com/products/debug-probes/j-link/) + (default method). +* [pyOCD](https://pypi.org/project/pyocd/). +* [nrfutil](https://pypi.org/project/nrfutil/) for the USB dongle boards that + supports it, which allows you to directly flash a working board over USB + without additional hardware. + +This guide **does not** cover how to setup the JTAG probe and their related +tools on your system. ### Software @@ -141,17 +147,17 @@ Our build script `build.rs` is responsible for converting `opensk_cert.pem` and 1. Connect a micro USB cable to the JTAG USB port. -1. Run our script for compiling/flashing Tock OS on your device (_output may - differ_): +1. Run our script for compiling/flashing Tock OS and OpenSK on your device + (_output may differ_): ```shell - $ ./deploy.py os --board=nrf52840_dk + $ ./deploy.py --board=nrf52840dk --opensk info: Updating rust toolchain to nightly-2020-02-03 info: syncing channel updates for 'nightly-2020-02-03-x86_64-unknown-linux-gnu' info: checking for self-updates info: component 'rust-std' for target 'thumbv7em-none-eabi' is up to date info: Rust toolchain up-to-date - info: Installing Tock on board nrf52840_dk + info: Building Tock OS for board nrf52840dk Compiling tock-registers v0.5.0 (./third_party/tock/libraries/tock-register-interface) Compiling tock-cells v0.1.0 (./third_party/tock/libraries/tock-cells) Compiling enum_primitive v0.1.0 (./third_party/tock/libraries/enum_primitive) @@ -166,47 +172,17 @@ Our build script `build.rs` is responsible for converting `opensk_cert.pem` and Compiling nrf52840 v0.1.0 (./third_party/tock/chips/nrf52840) Compiling components v0.1.0 (./third_party/tock/boards/components) Compiling nrf52dk_base v0.1.0 (./third_party/tock/boards/nordic/nrf52dk_base) - Finished release [optimized + debuginfo] target(s) in 11.97s - [STATUS ] Flashing binar(y|ies) to board... - [INFO ] Using known arch and jtag-device for known board nrf52dk - [INFO ] Finished in 0.284 seconds - ``` - -1. Run our script for compiling/flashing the OpenSK application on your device - (_output may differ_): - - ```shell - $ ./deploy.py app --opensk - info: Updating rust toolchain to nightly-2020-02-03 - info: syncing channel updates for 'nightly-2020-02-03-x86_64-unknown-linux-gnu' - info: checking for self-updates - info: component 'rust-std' for target 'thumbv7em-none-eabi' is up to date - info: Rust toolchain up-to-date - info: Erasing all installed applications - All apps have been erased. - info: Building OpenSK application - Compiling autocfg v1.0.0 - Compiling pkg-config v0.3.17 - Compiling cc v1.0.50 - Compiling libc v0.2.66 - Compiling bitflags v1.2.1 - Compiling foreign-types-shared v0.1.1 - Compiling openssl v0.10.28 - Compiling cfg-if v0.1.10 - Compiling lazy_static v1.4.0 - Compiling byteorder v1.3.2 - Compiling linked_list_allocator v0.6.6 - Compiling arrayref v0.3.6 - Compiling cbor v0.1.0 (./libraries/cbor) - Compiling subtle v2.2.2 - Compiling foreign-types v0.3.2 - Compiling libtock v0.1.0 (./third_party/libtock-rs) - Compiling crypto v0.1.0 (./libraries/crypto) - Compiling openssl-sys v0.9.54 - Compiling ctap2 v0.1.0 (.) - Finished release [optimized] target(s) in 15.34s - info: Flashing padding application - info: Installing Tock application ctap2 + Finished release [optimized + debuginfo] target(s) in 13.15s + info: Converting Tock OS file into a binary + info: Building OpenSK application + Finished release [optimized] target(s) in 0.02s + info: Generating Tock TAB file for application/example ctap2 + info: Erasing all installed applications + All apps have been erased. + info: Flashing file third_party/tock/boards/nordic/nrf52840dk/target/thumbv7em-none-eabi/release/nrf52840dk.bin. + info: Flashing padding application + info: Installing Tock application ctap2 + info: You're all set! ``` 1. Connect a micro USB cable to the device USB port. @@ -217,6 +193,8 @@ the board in order to see your OpenSK device on your system. #### Nordic nRF52840 Dongle +##### Using external programmer (JLink, OpenOCD, etc.) +  1. The JTAG probe used for programming won't provide power to the board. @@ -232,17 +210,18 @@ the board in order to see your OpenSK device on your system.  -1. Run our script for compiling/flashing Tock OS on your device (_output may - differ_): +1. Depending on the programmer you're using, you may have to adapt the next + command line. Run our script for compiling/flashing Tock OS on your device + (_output may differ_): ```shell - $ ./deploy.py os --board=nrf52840_dongle + $ ./deploy.py os --board=nrf52840_dongle --programmer=jlink info: Updating rust toolchain to nightly-2020-02-03 info: syncing channel updates for 'nightly-2020-02-03-x86_64-unknown-linux-gnu' info: checking for self-updates info: component 'rust-std' for target 'thumbv7em-none-eabi' is up to date info: Rust toolchain up-to-date - info: Installing Tock on board nrf52840_dongle + info: Building Tock OS for board nrf52840_dongle Compiling tock-cells v0.1.0 (./third_party/tock/libraries/tock-cells) Compiling tock-registers v0.5.0 (./third_party/tock/libraries/tock-register-interface) Compiling enum_primitive v0.1.0 (./third_party/tock/libraries/enum_primitive) @@ -258,50 +237,39 @@ the board in order to see your OpenSK device on your system. Compiling components v0.1.0 (./third_party/tock/boards/components) Compiling nrf52dk_base v0.1.0 (./third_party/tock/boards/nordic/nrf52dk_base) Finished release [optimized + debuginfo] target(s) in 11.72s - [STATUS ] Flashing binar(y|ies) to board... - [INFO ] Using known arch and jtag-device for known board nrf52dk - [INFO ] Finished in 0.280 seconds - ``` - -1. Run our script for compiling/flashing the OpenSK application on your device - (_output may differ_): - - ```shell - $ ./deploy.py app --opensk - info: Updating rust toolchain to nightly-2020-02-03 - info: syncing channel updates for 'nightly-2020-02-03-x86_64-unknown-linux-gnu' - info: checking for self-updates - info: component 'rust-std' for target 'thumbv7em-none-eabi' is up to date - info: Rust toolchain up-to-date + info: Converting Tock OS file into a binary + info: Building OpenSK application + Finished release [optimized] target(s) in 0.02s + info: Generating Tock TAB file for application/example ctap2 info: Erasing all installed applications All apps have been erased. - info: Building OpenSK application - Compiling autocfg v1.0.0 - Compiling pkg-config v0.3.17 - Compiling cc v1.0.50 - Compiling libc v0.2.66 - Compiling bitflags v1.2.1 - Compiling foreign-types-shared v0.1.1 - Compiling openssl v0.10.28 - Compiling cfg-if v0.1.10 - Compiling lazy_static v1.4.0 - Compiling byteorder v1.3.2 - Compiling linked_list_allocator v0.6.6 - Compiling arrayref v0.3.6 - Compiling cbor v0.1.0 (./libraries/cbor) - Compiling subtle v2.2.2 - Compiling foreign-types v0.3.2 - Compiling libtock v0.1.0 (./third_party/libtock-rs) - Compiling crypto v0.1.0 (./libraries/crypto) - Compiling openssl-sys v0.9.54 - Compiling ctap2 v0.1.0 (.) - Finished release [optimized] target(s) in 15.34s + info: Flashing file third_party/tock/boards/nordic/nrf52840_dongle/target/thumbv7em-none-eabi/release/nrf52840_dongle.bin. info: Flashing padding application info: Installing Tock application ctap2 + info: You're all set! ``` 1. Remove the programming cable and the USB-A extension cable. +### Advanced installation + +Although flashing using a Segger JLink probe is the officially supported way, +our tool, `deploy.py` also supports other methods: + +* OpenOCD: `./deploy.py --board=nrf52840_dongle --opensk --programmer=openocd` +* pyOCD: `./deploy.py --board=nrf52840_dongle --opensk --programmer=pyocd` +* Nordic DFU: `./deploy.py --board=nrf52840_dongle --opensk + --programmer=nordicdfu` +* Custom: `./deploy.py --board=nrf52840_dongle --opensk --programmer=none`. In + this case, an IntelHex file will be created and how to program a board is + left to the user. + +If your board is already flashed with Tock OS, you may skip installing it: +`./deploy.py --board=nrf52840dk --opensk --no-tockos` + +For more options, we invite you to read the help of our `deploy.py` script by +running `./deploy.py --help`. + ### Installing the udev rule (Linux only) By default on Linux, a USB device will require root privilege in order interact diff --git a/patches/tock/06-add-set_vector_table_offset.patch b/patches/tock/06-add-set_vector_table_offset.patch new file mode 100644 index 0000000000000000000000000000000000000000..e3b3618254c86d81e2075e4904dc8c990f6a31c0 --- /dev/null +++ b/patches/tock/06-add-set_vector_table_offset.patch @@ -0,0 +1,23 @@ +diff --git a/arch/cortex-m/src/scb.rs b/arch/cortex-m/src/scb.rs +index 8107f16..a271db9 100644 +--- a/arch/cortex-m/src/scb.rs ++++ b/arch/cortex-m/src/scb.rs +@@ -9,7 +9,7 @@ use kernel::common::StaticRef; + struct ScbRegisters { + cpuid: VolatileCell<u32>, + icsr: VolatileCell<u32>, +- vtor: VolatileCell<u32>, ++ vtor: VolatileCell<*const ()>, + aircr: VolatileCell<u32>, + scr: VolatileCell<u32>, + ccr: VolatileCell<u32>, +@@ -54,3 +54,8 @@ pub unsafe fn reset() { + let reset = (0x5FA << 16) | (aircr & (0x7 << 8)) | (1 << 2); + SCB.aircr.set(reset); + } ++ ++/// relocate interrupt vector table ++pub unsafe fn set_vector_table_offset(offset: *const ()) { ++ SCB.vtor.set(offset); ++} + diff --git a/patches/tock/07-nrf52-bootloader.patch b/patches/tock/07-nrf52-bootloader.patch new file mode 100644 index 0000000000000000000000000000000000000000..02f57be6552ce9ca2b09c1c98475015db6c08494 --- /dev/null +++ b/patches/tock/07-nrf52-bootloader.patch @@ -0,0 +1,21 @@ +diff --git a/chips/nrf52/src/crt1.rs b/chips/nrf52/src/crt1.rs +index 9703aac..281ceeb 100644 +--- a/chips/nrf52/src/crt1.rs ++++ b/chips/nrf52/src/crt1.rs +@@ -1,4 +1,4 @@ +-use cortexm4::{generic_isr, hard_fault_handler, nvic, svc_handler, systick_handler}; ++use cortexm4::{generic_isr, hard_fault_handler, nvic, scb, svc_handler, systick_handler}; + use tock_rt0; + + /* +@@ -168,5 +168,9 @@ pub unsafe extern "C" fn init() { + tock_rt0::init_data(&mut _etext, &mut _srelocate, &mut _erelocate); + tock_rt0::zero_bss(&mut _szero, &mut _ezero); + ++ // Ensure that we are compatible with a bootloader. ++ // For this we need to offset our vector table ++ scb::set_vector_table_offset(BASE_VECTORS.as_ptr() as *const ()); ++ + nvic::enable_all(); + } +