[sw] Cleanup example device programs

Signed-off-by: Miguel Young de la Sota <mcyoung@google.com>
diff --git a/sw/device/examples/demos.c b/sw/device/examples/demos.c
new file mode 100644
index 0000000..4af290b
--- /dev/null
+++ b/sw/device/examples/demos.c
@@ -0,0 +1,80 @@
+// Copyright lowRISC contributors.
+// Licensed under the Apache License, Version 2.0, see LICENSE for details.
+// SPDX-License-Identifier: Apache-2.0
+
+#include "sw/device/examples/demos.h"
+
+#include <stdbool.h>
+
+#include "sw/device/lib/arch/device.h"
+#include "sw/device/lib/dif/dif_gpio.h"
+#include "sw/device/lib/log.h"
+#include "sw/device/lib/runtime/hart.h"
+#include "sw/device/lib/spi_device.h"
+#include "sw/device/lib/uart.h"
+
+void demo_gpio_startup(dif_gpio_t *gpio) {
+  LOG_INFO("Watch the LEDs!\n");
+
+  // Give a LED pattern as startup indicator for 5 seconds.
+  dif_gpio_all_write(gpio, 0xff00);
+  for (int i = 0; i < 32; ++i) {
+    usleep(5 * 1000);  // 5 ms
+    dif_gpio_pin_write(gpio, 8 + (i % 8), (i / 8) % 2);
+  }
+  dif_gpio_all_write(gpio, 0x0000);  // All LEDs off.
+}
+
+/**
+ * Mask for "valid" GPIO bits. The first byte represents switch inputs,
+ * while byte 16 represents the FTDI bit.
+ */
+static const uint32_t kGpioMask = 0x100ff;
+
+/**
+ * Mask for the FTDI bit among the GPIO bits.
+ */
+static const uint32_t kFtdiMask = 0x10000;
+
+uint32_t demo_gpio_to_log_echo(dif_gpio_t *gpio, uint32_t prev_gpio_state) {
+  uint32_t gpio_state;
+  dif_gpio_all_read(gpio, &gpio_state);
+  gpio_state &= kGpioMask;
+
+  uint32_t state_delta = prev_gpio_state ^ gpio_state;
+  for (int bit_idx = 0; bit_idx < 8; ++bit_idx) {
+    bool changed = ((state_delta >> bit_idx) & 0x1) != 0;
+    bool is_currently_set = ((gpio_state >> bit_idx) & 0x1) != 0;
+    if (changed) {
+      LOG_INFO("GPIO switch #%d changed to %d\n", bit_idx, is_currently_set);
+    }
+  }
+
+  if ((state_delta & kFtdiMask) != 0) {
+    if ((gpio_state & kFtdiMask) != 0) {
+      LOG_INFO("FTDI control changed. Enable JTAG.\r\n");
+    } else {
+      LOG_INFO("FTDI control changed. Enable JTAG.\r\n");
+    }
+  }
+
+  return gpio_state;
+}
+
+void demo_spi_to_log_echo(void) {
+  uint32_t spi_buf[8];
+  size_t spi_len = spid_read_nb(spi_buf, sizeof(spi_buf));
+  if (spi_len > 0) {
+    uint32_t echo_word = spi_buf[0] ^ 0x01010101;
+    spid_send(&echo_word, sizeof(echo_word));
+    LOG_INFO("SPI: %z\r\n", spi_len, spi_buf);
+  }
+}
+
+void demo_uart_to_uart_and_gpio_echo(dif_gpio_t *gpio) {
+  char rcv_char;
+  while (uart_rcv_char(&rcv_char) != -1) {
+    uart_send_char(rcv_char);
+    dif_gpio_all_write(gpio, rcv_char << 8);
+  }
+}
\ No newline at end of file
diff --git a/sw/device/examples/demos.h b/sw/device/examples/demos.h
new file mode 100644
index 0000000..438aab6
--- /dev/null
+++ b/sw/device/examples/demos.h
@@ -0,0 +1,52 @@
+// Copyright lowRISC contributors.
+// Licensed under the Apache License, Version 2.0, see LICENSE for details.
+// SPDX-License-Identifier: Apache-2.0
+
+#ifndef OPENTITAN_SW_DEVICE_EXAMPLES_DEMOS_H_
+#define OPENTITAN_SW_DEVICE_EXAMPLES_DEMOS_H_
+
+#include <stdint.h>
+
+#include "sw/device/lib/dif/dif_gpio.h"
+
+/**
+ * This header provides a small library of reuseable demos for use with
+ * OpenTitan example code.
+ */
+
+/**
+ * Runs a small demo on the GPIO pins to show that things are working.
+ *
+ * @param gpio the GPIO device to actuate.
+ */
+void demo_gpio_startup(dif_gpio_t *gpio);
+
+/**
+ * Executes a step of a GPIO -> LOG echo demo, by diffing
+ * the previous GPIO state with the current state, and reporting
+ * the difference.
+ *
+ * The new state is returned, so it can be passed in on the next
+ * iteration.
+ *
+ * @param gpio the GPIO device to pull bits from.
+ * @param prev_gpio_state the previous GPIO state to diff against.
+ * @return the new GPIO state.
+ */
+uint32_t demo_gpio_to_log_echo(dif_gpio_t *gpio, uint32_t prev_gpio_state);
+
+/**
+ * Attempts to read at most 32 bytes from SPI, and echo them as printable
+ * characters to LOG.
+ */
+void demo_spi_to_log_echo(void);
+
+/**
+ * Attempts to read characters from UART and immediately echo them back,
+ * as well as to write its bits to GPIO pins 8-15.
+ *
+ * @param gpio the GPIO device to actuate.
+ */
+void demo_uart_to_uart_and_gpio_echo(dif_gpio_t *gpio);
+
+#endif  // OPENTITAN_SW_DEVICE_EXAMPLES_DEMOS_H_
diff --git a/sw/device/examples/hello_usbdev/hello_usbdev.c b/sw/device/examples/hello_usbdev/hello_usbdev.c
index 32fe1b5..90407f6 100644
--- a/sw/device/examples/hello_usbdev/hello_usbdev.c
+++ b/sw/device/examples/hello_usbdev/hello_usbdev.c
@@ -2,12 +2,14 @@
 // Licensed under the Apache License, Version 2.0, see LICENSE for details.
 // SPDX-License-Identifier: Apache-2.0
 
+#include <stdalign.h>
 #include <stdbool.h>
 
+#include "sw/device/examples/demos.h"
 #include "sw/device/lib/arch/device.h"
-#include "sw/device/lib/base/stdasm.h"
 #include "sw/device/lib/common.h"
 #include "sw/device/lib/dif/dif_gpio.h"
+#include "sw/device/lib/log.h"
 #include "sw/device/lib/pinmux.h"
 #include "sw/device/lib/runtime/hart.h"
 #include "sw/device/lib/spi_device.h"
@@ -20,154 +22,122 @@
 #define USBDEV_BASE_ADDR 0x40150000
 #include "usbdev_regs.h"  // Generated.
 
-#define SPI_MAX 32
-#define GPIO0_BASE_ADDR 0x40010000u
+/**
+ * Configuration values for USB.
+ */
+static uint8_t config_descriptors[] = {
+    USB_CFG_DSCR_HEAD(USB_CFG_DSCR_LEN +
+                          2 * (USB_INTERFACE_DSCR_LEN + 2 * USB_EP_DSCR_LEN),
+                      2),
+    VEND_INTERFACE_DSCR(0, 2, 0x50, 1),
+    USB_BULK_EP_DSCR(0, 1, 32, 0),
+    USB_BULK_EP_DSCR(1, 1, 32, 4),
+    VEND_INTERFACE_DSCR(1, 2, 0x50, 1),
+    USB_BULK_EP_DSCR(0, 2, 32, 0),
+    USB_BULK_EP_DSCR(1, 2, 32, 4),
+};
+
+/**
+ * USB device context types.
+ */
+static usbdev_ctx_t usbdev;
+static usb_controlep_ctx_t usbdev_control;
+static usb_ss_ctx_t simple_serial0;
+static usb_ss_ctx_t simple_serial1;
+
+/**
+ * Makes |c| into a printable character, replacing it with |replacement|
+ * as necessary.
+ */
+static char make_printable(char c, char replacement) {
+  if (c == 0xa || c == 0xd) {
+    return c;
+  }
+
+  if (c < ' ' || c > '~') {
+    c = replacement;
+  }
+  return c;
+}
+
+static const size_t kExpectedUsbCharsRecved = 6;
+static size_t usb_chars_recved_total;
+
+/**
+ * Callbacks for processing USB reciept. The latter increments the
+ * recieved character by one, to make them distinct.
+ */
+static void usb_receipt_callback_0(uint8_t c) {
+  c = make_printable(c, '?');
+  uart_send_char(c);
+  ++usb_chars_recved_total;
+}
+static void usb_receipt_callback_1(uint8_t c) {
+  c = make_printable(c + 1, '?');
+  uart_send_char(c);
+  ++usb_chars_recved_total;
+}
 
 static dif_gpio_t gpio;
 
-// called from ctr0 when something bad happens
-// char I=illegal instruction, A=lsu error (address), E=ecall
-void trap_handler(uint32_t mepc, char c) {
-  uart_send_char(c);
-  uart_send_uint(mepc, 32);
-  while (1) {
-    dif_gpio_all_write(&gpio, 0xAA00);  // pattern
-    usleep(200 * 1000);
-    dif_gpio_all_write(&gpio, 0x5500);  // pattern
-    usleep(100 * 1000);
-  }
-}
-
-#define MK_PRINT(c) \
-  (((c != 0xa) && (c != 0xd) && ((c < 32) || (c > 126))) ? '_' : c)
-
-// Build Configuration descriptor array
-static uint8_t cfg_dscr[] = {
-    USB_CFG_DSCR_HEAD(
-        USB_CFG_DSCR_LEN + 2 * (USB_INTERFACE_DSCR_LEN + 2 * USB_EP_DSCR_LEN),
-        2) VEND_INTERFACE_DSCR(0, 2, 0x50, 1) USB_BULK_EP_DSCR(0, 1, 32, 0)
-        USB_BULK_EP_DSCR(1, 1, 32, 4) VEND_INTERFACE_DSCR(1, 2, 0x50, 1)
-            USB_BULK_EP_DSCR(0, 2, 32, 0) USB_BULK_EP_DSCR(1, 2, 32, 4)};
-
-/* context areas */
-static usbdev_ctx_t usbdev_ctx;
-static usb_controlep_ctx_t control_ctx;
-static usb_ss_ctx_t ss_ctx[2];
-
-// We signal PASS! after receiving USB_MAX.
-static volatile unsigned int num_chars_rx = 0;
-#ifndef USB_MAX
-#define USB_MAX 6
-#endif
-
-/* Inbound USB characters get printed to the UART via these callbacks */
-/* Not ideal because the UART is slower */
-static void serial_rx0(uint8_t c) {
-  uart_send_char(MK_PRINT(c));
-  num_chars_rx++;
-}
-/* Add one to rx character so you can tell it is the second instance */
-static void serial_rx1(uint8_t c) {
-  uart_send_char(MK_PRINT(c + 1));
-  num_chars_rx++;
-}
-
 int main(int argc, char **argv) {
   uart_init(kUartBaudrate);
   pinmux_init();
   spid_init();
-  // Enable GPIO: 0-7 and 16 is input, 8-15 is output
+
   dif_gpio_config_t gpio_config = {.base_addr =
-                                       mmio_region_from_addr(GPIO0_BASE_ADDR)};
+                                       mmio_region_from_addr(0x40010000)};
   dif_gpio_init(&gpio_config, &gpio);
-  dif_gpio_output_mode_all_set(&gpio, 0xFF00);
+  // Enable GPIO: 0-7 and 16 is input; 8-15 is output.
+  dif_gpio_output_mode_all_set(&gpio, 0x0ff00);
 
-  // Add DATE and TIME because I keep fooling myself with old versions
-  uart_send_str(
-      "Hello USB! "__DATE__
-      " "__TIME__
-      "\r\n");
-  uart_send_str("Characters from UART go to USB and GPIO\r\n");
-  uart_send_str("Characters from USB go to UART\r\n");
+  LOG_INFO("Hello, USB!\n");
+  LOG_INFO("Built at " __DATE__ " " __TIME__ "\n");
 
-  // Give a LED pattern as startup indicator for 5 seconds
-  dif_gpio_all_write(&gpio, 0xAA00);  // pattern
-  usleep(1000);
-  dif_gpio_all_write(&gpio, 0x5500);  // pattern
-  // usbdev_init here so dpi code will not start until simulation
-  // got through all the printing (which takes a while if --trace)
-  usbdev_init(&usbdev_ctx);
-  usb_controlep_init(&control_ctx, &usbdev_ctx, 0, cfg_dscr, sizeof(cfg_dscr));
-  usb_simpleserial_init(&ss_ctx[0], &usbdev_ctx, 1, serial_rx0);
-  usb_simpleserial_init(&ss_ctx[1], &usbdev_ctx, 2, serial_rx1);
-  bool pass_signaled = false;
+  demo_gpio_startup(&gpio);
 
-  uint32_t gpio_in;
-  uint32_t gpio_in_prev = 0;
-  uint32_t gpio_in_changes;
-  uint8_t spi_buf[SPI_MAX];
-  uint32_t spi_in;
+  // Call |usbdev_init| here so that DPI will not start until the
+  // simulation has finished all of the printing, which takes a while
+  // if |--trace| was passed in.
+  usbdev_init(&usbdev);
+  usb_controlep_init(&usbdev_control, &usbdev, 0, config_descriptors,
+                     sizeof(config_descriptors));
+  usb_simpleserial_init(&simple_serial0, &usbdev, 1, usb_receipt_callback_0);
+  usb_simpleserial_init(&simple_serial1, &usbdev, 2, usb_receipt_callback_1);
 
   spid_send("SPI!", 4);
 
-  while (1) {
-    usbdev_poll(&usbdev_ctx);
+  uint32_t gpio_state = 0;
+  bool pass_signaled = false;
+  while (true) {
+    usbdev_poll(&usbdev);
 
-    // report changed switches over UART
-    dif_gpio_all_read(&gpio, &gpio_in);
-    gpio_in &= 0x100FF;  // 0-7 is switch input, 16 is FTDI
-    gpio_in_changes = (gpio_in & ~gpio_in_prev) | (~gpio_in & gpio_in_prev);
-    for (int b = 0; b < 8; b++) {
-      if (gpio_in_changes & (1 << b)) {
-        int val_now = (gpio_in >> b) & 0x01;
-        uart_send_str("GPIO: Switch ");
-        uart_send_char(b + 48);
-        uart_send_str(" changed to ");
-        uart_send_char(val_now + 48);
-        uart_send_str("\r\n");
-      }
-    }
-    if (gpio_in_changes & 0x10000) {
-      uart_send_str("FTDI control changed. Enable ");
-      uart_send_str((gpio_in & 0x10000) ? "JTAG\r\n" : "SPI\r\n");
-    }
-    gpio_in_prev = gpio_in;
+    gpio_state = demo_gpio_to_log_echo(&gpio, gpio_state);
+    demo_spi_to_log_echo();
 
-    // SPI character echo
-    spi_in = spid_read_nb(spi_buf, SPI_MAX);
-    if (spi_in) {
-      uint32_t d = (*(uint32_t *)spi_buf) ^ 0x01010101;
-      spid_send(&d, 4);
-      uart_send_str("SPI: ");
-      for (int i = 0; i < spi_in; i++) {
-        uart_send_char(MK_PRINT(spi_buf[i]));
-      }
-      uart_send_str("\r\n");
-    }
-    // UART echo
     char rcv_char;
     while (uart_rcv_char(&rcv_char) != -1) {
       uart_send_char(rcv_char);
       dif_gpio_all_write(&gpio, rcv_char << 8);
+
       if (rcv_char == '/') {
-        uart_send_char('I');
-        uart_send_uint(REG32(USBDEV_INTR_STATE()), 16);
-        uart_send_char('-');
-        uart_send_uint(REG32(USBDEV_USBSTAT()), 32);
-        uart_send_char(' ');
+        uint32_t usb_irq_state = REG32(USBDEV_INTR_STATE());
+        uint32_t usb_stat = REG32(USBDEV_USBSTAT());
+        LOG_INFO("I%4x-%8x\r\n", usb_irq_state, usb_stat);
       } else {
-        usb_simpleserial_send_byte(&ss_ctx[0], rcv_char);
-        usb_simpleserial_send_byte(&ss_ctx[1], rcv_char + 1);
+        usb_simpleserial_send_byte(&simple_serial0, rcv_char);
+        usb_simpleserial_send_byte(&simple_serial1, rcv_char + 1);
       }
     }
 
-    // Signal that the simulation passed
-    if ((num_chars_rx >= USB_MAX) && (pass_signaled == false)) {
-      uart_send_str("\r\nPASS!\r\n");
+    // Signal that the simulation succeeded.
+    if (usb_chars_recved_total >= kExpectedUsbCharsRecved && !pass_signaled) {
+      uart_send_str("\r\n");
+      uart_send_str("PASS!\r\n");
       pass_signaled = true;
     }
   }
-  uart_send_str("\r\nUSB received 0x");
-  uart_send_uint(num_chars_rx, 32);
-  uart_send_str(" characters.\r\n");
+
+  uart_send_str("\r\n");
+  LOG_INFO("USB recieved %d characters.\r\n", usb_chars_recved_total);
 }
diff --git a/sw/device/examples/hello_usbdev/meson.build b/sw/device/examples/hello_usbdev/meson.build
index 9c23020..d325666 100644
--- a/sw/device/examples/hello_usbdev/meson.build
+++ b/sw/device/examples/hello_usbdev/meson.build
@@ -8,6 +8,7 @@
     sources: ['hello_usbdev.c'],
     name_suffix: 'elf',
     dependencies: [
+    sw_examples_demos,
     sw_lib_runtime_hart,
     sw_lib_pinmux,
     sw_lib_dif_gpio,
diff --git a/sw/device/examples/hello_world/hello_world.c b/sw/device/examples/hello_world/hello_world.c
index 2ac76c1..9ecd9e9 100644
--- a/sw/device/examples/hello_world/hello_world.c
+++ b/sw/device/examples/hello_world/hello_world.c
@@ -2,110 +2,47 @@
 // Licensed under the Apache License, Version 2.0, see LICENSE for details.
 // SPDX-License-Identifier: Apache-2.0
 
+#include "sw/device/examples/demos.h"
 #include "sw/device/lib/arch/device.h"
 #include "sw/device/lib/base/print.h"
-#include "sw/device/lib/base/stdasm.h"
-#include "sw/device/lib/common.h"
 #include "sw/device/lib/dif/dif_gpio.h"
+#include "sw/device/lib/log.h"
 #include "sw/device/lib/pinmux.h"
 #include "sw/device/lib/runtime/hart.h"
 #include "sw/device/lib/spi_device.h"
 #include "sw/device/lib/uart.h"
 
-#define SPI_MAX 32
-#define GPIO0_BASE_ADDR 0x40010000u
-
 static dif_gpio_t gpio;
 
-// called from ctr0 when something bad happens
-// char I=illegal instruction, A=lsu error (address), E=ecall
-void trap_handler(uint32_t mepc, char c) {
-  uart_send_char(c);
-  uart_send_uint(mepc, 32);
-  while (1) {
-    dif_gpio_all_write(&gpio, 0xAA00);  // pattern
-    usleep(200 * 1000);
-    dif_gpio_all_write(&gpio, 0x5500);  // pattern
-    usleep(100 * 1000);
-  }
-}
-
-#define MK_PRINT(c) (((c < 32) || (c > 126)) ? '_' : c)
-
 int main(int argc, char **argv) {
   uart_init(kUartBaudrate);
   base_set_stdout(uart_stdout);
   
   pinmux_init();
   spid_init();
-  // Enable GPIO: 0-7 and 16 is input, 8-15 is output
+
   dif_gpio_config_t gpio_config = {.base_addr =
-                                       mmio_region_from_addr(GPIO0_BASE_ADDR)};
+                                       mmio_region_from_addr(0x40010000)};
   dif_gpio_init(&gpio_config, &gpio);
-  dif_gpio_output_mode_all_set(&gpio, 0xFF00);
-  // Add DATE and TIME because I keep fooling myself with old versions
-  base_printf("Hello World!\r\n");
-  base_printf("Built at: " __DATE__ ", " __TIME__ "\r\n");
-  base_printf("Watch the LEDs!\r\n");
+  // Enable GPIO: 0-7 and 16 is input; 8-15 is output.
+  dif_gpio_output_mode_all_set(&gpio, 0x0ff00);
 
-  // Give a LED pattern as startup indicator for 5 seconds
-  dif_gpio_all_write(&gpio, 0xFF00);  // all LEDs on
-  for (int i = 0; i < 32; i++) {
-    usleep(100 * 1000);  // 100 ms
+  LOG_INFO("Hello, World!\n");
+  LOG_INFO("Built at " __DATE__ " " __TIME__ "\n");
 
-    dif_gpio_pin_write(&gpio, 8 + (i % 8), (i / 8) % 2);
-  }
+  demo_gpio_startup(&gpio);
 
-  // Now have UART <-> Buttons/LEDs demo
-  // all LEDs off
-  dif_gpio_all_write(&gpio, 0x0000);
-  base_printf("Try out the switches on the board\r\n");
-  base_printf("or type anything into the console window.\r\n");
-  base_printf("The LEDs show the ASCII code of the last character.\r\n");
-
-  uint32_t gpio_in;
-  uint32_t gpio_in_prev = 0;
-  uint32_t gpio_in_changes;
-  uint8_t spi_buf[SPI_MAX];
-  uint32_t spi_in;
+  LOG_INFO("Try out the switches on the board,\n");
+  LOG_INFO("or type anything into the UART console window.\n");
+  LOG_INFO("The LEDs show the ASCII code of the last character.\n");
 
   spid_send("SPI!", 4);
 
-  while (1) {
+  uint32_t gpio_state = 0;
+  while (true) {
     usleep(10 * 1000);  // 10 ms
-
-    // report changed switches over UART
-    dif_gpio_all_read(&gpio, &gpio_in);
-    gpio_in &= 0x100FF;  // 0-7 is switch input, 16 is FTDI
-    gpio_in_changes = (gpio_in & ~gpio_in_prev) | (~gpio_in & gpio_in_prev);
-    for (int b = 0; b < 8; b++) {
-      if (gpio_in_changes & (1 << b)) {
-        int val_now = (gpio_in >> b) & 0x01;
-        base_printf("GPIO: Switch %d changed to %d\r\n", b, val_now);
-      }
-    }
-    if (gpio_in_changes & 0x10000) {
-      const char *port_type = (gpio_in & 0x10000) ? "JTAG" : "SPI";
-      base_printf("FTDI control changed. Enable %s.\r\n", port_type);
-    }
-    gpio_in_prev = gpio_in;
-
-    // SPI character echo
-    spi_in = spid_read_nb(spi_buf, SPI_MAX);
-    if (spi_in) {
-      uint32_t d = (*(uint32_t *)spi_buf) ^ 0x01010101;
-      spid_send(&d, 4);
-      base_printf("SPI: ");
-      for (int i = 0; i < spi_in; i++) {
-        base_printf("%c", MK_PRINT(spi_buf[i]));
-      }
-      base_printf("\r\n");
-    }
-    // UART echo
-    char rcv_char;
-    while (uart_rcv_char(&rcv_char) != -1) {
-      uart_send_char(rcv_char);
-      dif_gpio_all_write(&gpio, rcv_char << 8);
-    }
+    gpio_state = demo_gpio_to_log_echo(&gpio, gpio_state);
+    demo_spi_to_log_echo();
+    demo_uart_to_uart_and_gpio_echo(&gpio);
   }
 }
diff --git a/sw/device/examples/hello_world/meson.build b/sw/device/examples/hello_world/meson.build
index b98ded1..e9a5d08 100644
--- a/sw/device/examples/hello_world/meson.build
+++ b/sw/device/examples/hello_world/meson.build
@@ -8,6 +8,7 @@
     sources: ['hello_world.c'],
     name_suffix: 'elf',
     dependencies: [
+      sw_examples_demos, 
       sw_lib_runtime_hart,
       sw_lib_base_print,
       sw_lib_pinmux,
diff --git a/sw/device/examples/meson.build b/sw/device/examples/meson.build
index fe95ea4..3225a3d 100644
--- a/sw/device/examples/meson.build
+++ b/sw/device/examples/meson.build
@@ -2,5 +2,18 @@
 # Licensed under the Apache License, Version 2.0, see LICENSE for details.
 # SPDX-License-Identifier: Apache-2.0
 
+sw_examples_demos = declare_dependency(
+  link_with: static_library(
+    'sw_examples_demos',
+    sources: ['demos.c'],
+    dependencies: [
+      sw_lib_runtime_hart,
+      sw_lib_dif_gpio,
+      sw_lib_spi_device,
+      sw_lib_uart,
+    ],
+  )
+)
+
 subdir('hello_usbdev')
 subdir('hello_world')
diff --git a/sw/device/lib/usb_controlep.h b/sw/device/lib/usb_controlep.h
index 288e789..4f40bf3 100644
--- a/sw/device/lib/usb_controlep.h
+++ b/sw/device/lib/usb_controlep.h
@@ -46,38 +46,38 @@
 /* Use them to initialize a uint8_t array for cfg_dscr              */
 
 #define USB_CFG_DSCR_LEN 9
-#define USB_CFG_DSCR_HEAD(total_len, nint)                                    \
-  /* This is the actual configuration descriptor                 */           \
-  USB_CFG_DSCR_LEN,     /* bLength                                   */       \
-      2,                /* bDescriptorType                           */       \
-      (total_len)&0xff, /* wTotalLength[0]                           */       \
-      (total_len) >> 8, /* wTotalLength[1]                           */       \
-      (nint),           /* bNumInterfaces                            */       \
-      1,                /* bConfigurationValu                        */       \
-      0,                /* iConfiguration                            */       \
-      0xC0,             /* bmAttributes: must-be-one, self-powered   */       \
-      50, /* bMaxPower                                 */ /* MUST be followed \
-                                                             by (nint)        \
-                                                             Interface +      \
-                                                             Endpoint         \
-                                                             Descriptors */
+#define USB_CFG_DSCR_HEAD(total_len, nint)                                   \
+  /* This is the actual configuration descriptor                 */          \
+  USB_CFG_DSCR_LEN,     /* bLength                                   */      \
+      2,                /* bDescriptorType                           */      \
+      (total_len)&0xff, /* wTotalLength[0]                           */      \
+      (total_len) >> 8, /* wTotalLength[1]                           */      \
+      (nint),           /* bNumInterfaces                            */      \
+      1,                /* bConfigurationValu                        */      \
+      0,                /* iConfiguration                            */      \
+      0xC0,             /* bmAttributes: must-be-one, self-powered   */      \
+      50 /* bMaxPower                                 */ /* MUST be followed \
+                                                            by (nint)        \
+                                                            Interface +      \
+                                                            Endpoint         \
+                                                            Descriptors */
 
 // KEEP BLANK LINE ABOVE, it is in the macro!
 
 #define USB_INTERFACE_DSCR_LEN 9
-#define VEND_INTERFACE_DSCR(inum, nep, subclass, protocol)                \
-  /* interface descriptor, USB spec 9.6.5, page 267-269, Table 9-12 */    \
-  USB_INTERFACE_DSCR_LEN, /* bLength                             */       \
-      4,                  /* bDescriptorType                     */       \
-      (inum),             /* bInterfaceNumber                    */       \
-      0,                  /* bAlternateSetting                   */       \
-      (nep),              /* bNumEndpoints                       */       \
-      0xff,               /* bInterfaceClass (Vendor Specific)   */       \
-      (subclass),         /* bInterfaceSubClass                  */       \
-      (protocol),         /* bInterfaceProtocol                  */       \
-      0, /* iInterface                          */ /* MUST be followed by \
-                                                      (nep) Endpoint      \
-                                                      Descriptors */
+#define VEND_INTERFACE_DSCR(inum, nep, subclass, protocol)               \
+  /* interface descriptor, USB spec 9.6.5, page 267-269, Table 9-12 */   \
+  USB_INTERFACE_DSCR_LEN, /* bLength                             */      \
+      4,                  /* bDescriptorType                     */      \
+      (inum),             /* bInterfaceNumber                    */      \
+      0,                  /* bAlternateSetting                   */      \
+      (nep),              /* bNumEndpoints                       */      \
+      0xff,               /* bInterfaceClass (Vendor Specific)   */      \
+      (subclass),         /* bInterfaceSubClass                  */      \
+      (protocol),         /* bInterfaceProtocol                  */      \
+      0 /* iInterface                          */ /* MUST be followed by \
+                                                     (nep) Endpoint      \
+                                                     Descriptors */
 
 // KEEP BLANK LINE ABOVE, it is in the macro!
 
@@ -90,7 +90,7 @@
       0x02,                        /* bmAttributes (0x02=bulk, data)       */ \
       (maxsize)&0xff,              /* wMaxPacketSize[0]                    */ \
       (maxsize) >> 8,              /* wMaxPacketSize[1]                    */ \
-      (interval),                  /* bInterval                            */
+      (interval)                   /* bInterval                            */
 
 // KEEP BLANK LINE ABOVE, it is in the macro!