Add firmware for the ATUSB IEEE 802.15.4 USB Adapter
[linux-libre-firmware.git] / atusb / board.c
diff --git a/atusb/board.c b/atusb/board.c
new file mode 100644 (file)
index 0000000..c3b8d26
--- /dev/null
@@ -0,0 +1,120 @@
+/*
+ * fw/board.c - Board-specific functions (for boot loader and application)
+ *
+ * Written 2011, 2013 by Werner Almesberger
+ * Copyright 2011, 2013 Werner Almesberger
+ *
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation; either version 2 of the License, or
+ * (at your option) any later version.
+ */
+
+
+#include <stdbool.h>
+#include <stdint.h>
+
+#include <avr/io.h>
+#include <avr/interrupt.h>
+#include <avr/boot.h>
+
+#define F_CPU   8000000UL
+#include <util/delay.h>
+
+#include "usb.h"
+#include "at86rf230.h"
+#include "board.h"
+#include "spi.h"
+
+
+uint8_t board_sernum[42] = { 42, USB_DT_STRING };
+
+/* ----- Register access --------------------------------------------------- */
+
+void change_state(uint8_t new)
+{
+       while ((reg_read(REG_TRX_STATUS) & TRX_STATUS_MASK) ==
+               TRX_STATUS_TRANSITION);
+       reg_write(REG_TRX_STATE, new);
+}
+
+
+uint8_t reg_read(uint8_t reg)
+{
+       uint8_t value;
+
+       spi_begin();
+       spi_send(AT86RF230_REG_READ | reg);
+       value = spi_recv();
+       spi_end();
+
+       return value;
+}
+
+
+uint8_t subreg_read(uint8_t address, uint8_t mask, uint8_t position)
+{
+       /* Read current register value and mask out subregister. */
+       uint8_t register_value = reg_read(address);
+       register_value &= mask;
+       register_value >>= position; /* Align subregister value. */
+
+       return register_value;
+}
+
+
+void reg_write(uint8_t reg, uint8_t value)
+{
+       spi_begin();
+       spi_send(AT86RF230_REG_WRITE | reg);
+       spi_send(value);
+       spi_end();
+}
+
+
+void subreg_write(uint8_t address, uint8_t mask, uint8_t position, uint8_t value)
+{
+       /* Read current register value and mask area outside the subregister. */
+       uint8_t register_value = reg_read(address);
+       register_value &= ~mask;
+
+       /* Start preparing the new subregister value. shift in place and mask. */
+       value <<= position;
+       value &= mask;
+
+       value |= register_value; /* Set the new subregister value. */
+
+       /* Write the modified register value. */
+       reg_write(address, value);
+}
+
+
+void panic(void)
+{
+       cli();
+       while (1) {
+               SET(LED);
+               _delay_ms(100);
+               CLR(LED);
+               _delay_ms(100);
+       }
+}
+
+
+static char hex(uint8_t nibble)
+{
+       return nibble < 10 ? '0'+nibble : 'a'+nibble-10;
+}
+
+
+void get_sernum(void)
+{
+       uint8_t sig;
+       uint8_t i;
+
+       for (i = 0; i != 10; i++) {
+               sig = boot_signature_byte_get(i+0xe);
+               board_sernum[(i << 2)+2] = hex(sig >> 4);
+               board_sernum[(i << 2)+4] = hex(sig & 0xf);
+       }
+}