2011-02-14 17:48:42 +02:00
|
|
|
/*
|
2011-06-09 18:48:44 +03:00
|
|
|
* fw/board.c - Board-specific functions (for boot loader and application)
|
2011-02-14 17:48:42 +02:00
|
|
|
*
|
2013-03-30 00:19:49 +02:00
|
|
|
* Written 2011, 2013 by Werner Almesberger
|
|
|
|
* Copyright 2011, 2013 Werner Almesberger
|
2011-02-14 17:48:42 +02:00
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
2013-03-30 00:19:49 +02:00
|
|
|
#include <stdbool.h>
|
2011-02-14 17:48:42 +02:00
|
|
|
#include <stdint.h>
|
|
|
|
|
|
|
|
#include <avr/io.h>
|
|
|
|
#include <avr/interrupt.h>
|
2011-05-10 23:23:08 +03:00
|
|
|
#include <avr/boot.h>
|
2011-02-14 17:48:42 +02:00
|
|
|
|
|
|
|
#define F_CPU 8000000UL
|
|
|
|
#include <util/delay.h>
|
|
|
|
|
2011-05-10 23:23:08 +03:00
|
|
|
#include "usb.h"
|
2011-02-14 17:48:42 +02:00
|
|
|
#include "at86rf230.h"
|
|
|
|
#include "board.h"
|
|
|
|
#include "spi.h"
|
|
|
|
|
|
|
|
|
2011-05-10 23:23:08 +03:00
|
|
|
uint8_t board_sernum[42] = { 42, USB_DT_STRING };
|
|
|
|
|
2017-09-11 14:58:33 +03:00
|
|
|
/* ----- Register access --------------------------------------------------- */
|
|
|
|
|
|
|
|
void change_state(uint8_t new)
|
2011-02-14 17:48:42 +02:00
|
|
|
{
|
2017-09-11 14:58:33 +03:00
|
|
|
while ((reg_read(REG_TRX_STATUS) & TRX_STATUS_MASK) ==
|
|
|
|
TRX_STATUS_TRANSITION);
|
|
|
|
reg_write(REG_TRX_STATE, new);
|
|
|
|
}
|
2011-06-04 15:40:34 +03:00
|
|
|
|
|
|
|
|
2017-09-11 14:58:33 +03:00
|
|
|
uint8_t reg_read(uint8_t reg)
|
|
|
|
{
|
|
|
|
uint8_t value;
|
2011-06-04 15:40:34 +03:00
|
|
|
|
2017-09-11 14:58:33 +03:00
|
|
|
spi_begin();
|
|
|
|
spi_send(AT86RF230_REG_READ | reg);
|
|
|
|
value = spi_recv();
|
|
|
|
spi_end();
|
2011-06-04 15:40:34 +03:00
|
|
|
|
2017-09-11 14:58:33 +03:00
|
|
|
return value;
|
|
|
|
}
|
2011-02-14 17:48:42 +02:00
|
|
|
|
|
|
|
|
2017-09-11 14:58:33 +03:00
|
|
|
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. */
|
2011-02-14 17:48:42 +02:00
|
|
|
|
2017-09-11 14:58:33 +03:00
|
|
|
return register_value;
|
|
|
|
}
|
2011-02-14 17:48:42 +02:00
|
|
|
|
|
|
|
|
2017-09-11 14:58:33 +03:00
|
|
|
void reg_write(uint8_t reg, uint8_t value)
|
|
|
|
{
|
|
|
|
spi_begin();
|
|
|
|
spi_send(AT86RF230_REG_WRITE | reg);
|
|
|
|
spi_send(value);
|
|
|
|
spi_end();
|
2011-02-14 17:48:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-09-11 14:58:33 +03:00
|
|
|
void subreg_write(uint8_t address, uint8_t mask, uint8_t position, uint8_t value)
|
2011-02-14 17:48:42 +02:00
|
|
|
{
|
2017-09-11 14:58:33 +03:00
|
|
|
/* 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);
|
2011-02-14 17:48:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void panic(void)
|
|
|
|
{
|
|
|
|
cli();
|
|
|
|
while (1) {
|
|
|
|
SET(LED);
|
|
|
|
_delay_ms(100);
|
|
|
|
CLR(LED);
|
|
|
|
_delay_ms(100);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-05-10 23:23:08 +03:00
|
|
|
static char hex(uint8_t nibble)
|
|
|
|
{
|
|
|
|
return nibble < 10 ? '0'+nibble : 'a'+nibble-10;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-03-30 01:03:48 +03:00
|
|
|
void get_sernum(void)
|
2011-05-10 23:23:08 +03:00
|
|
|
{
|
|
|
|
uint8_t sig;
|
2013-03-30 00:19:49 +02:00
|
|
|
uint8_t i;
|
2011-05-10 23:23:08 +03:00
|
|
|
|
|
|
|
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);
|
|
|
|
}
|
|
|
|
}
|