mirror of
git://projects.qi-hardware.com/ben-wpan.git
synced 2024-12-23 19:35:31 +02:00
5129029d3b
This patch adds support for the rzusbstick for the atusb firmware. More detailed information about this usb stick: http://www.atmel.com/tools/rzusbstick.aspx Original I have the rzraven kit: http://www.atmel.com/tools/rzraven.aspx Which comes with a special cable and avr dragon programmer. You need some programmer and wires to the programmers pins. To lookup how to connect the programmer to the rzusbstick pinout, see: http://www.atmel.com/Images/doc8117.pdf page 22 (schematics of the rzusbstick). Difference between atusb and rzusbstick(rzusb) is mainly the at86rf231 vs at86rf230 one. The rzusb contains the at86rf230 which is a little bit hard to deal with it (and has a huge errata inside the datasheet). Nevertheless with small schanges the atusb firmware can run now on the rzusb. The rzusb contains also a bigger mcu, so we can maybe cache more pdus for receive handling. To compile the rzusb firmware call: make NAME=rzusb this will generate the rzusb.bin then call the programmer (in my case avrdude): avrdude -P usb -c dragon_jtag -p usb1287 -U flash:w:rzusb.bin NOTE: currently there is no chance (I suppose) to ensure that the atusb receive the correct firmware, so don't try to flash the atusb with the rzusb firmware! Also the vendor and product id is the same. This currently a RFC, it's a quick hack and I think we should update more the documentation to support the rzusb. Signed-off-by: Alexander Aring <alex.aring@gmail.com> Cc: Stefan Schmidt <stefan@osg.samsung.com> Cc: Werner Almesberger <werner@almesberger.net>
280 lines
4.9 KiB
C
280 lines
4.9 KiB
C
/*
|
|
* fw/mac.c - HardMAC functions
|
|
*
|
|
* 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 <stddef.h>
|
|
#include <stdbool.h>
|
|
#include <stdint.h>
|
|
|
|
#include "usb.h"
|
|
|
|
#include "at86rf230.h"
|
|
#include "spi.h"
|
|
#include "board.h"
|
|
#include "mac.h"
|
|
|
|
#define RX_BUFS 3
|
|
|
|
|
|
bool (*mac_irq)(void) = NULL;
|
|
|
|
|
|
static uint8_t rx_buf[RX_BUFS][MAX_PSDU+2]; /* PHDR+payload+LQ */
|
|
static uint8_t tx_buf[MAX_PSDU];
|
|
static uint8_t tx_size = 0;
|
|
static bool txing = 0;
|
|
static bool queued_tx_ack = 0;
|
|
static uint8_t next_seq, this_seq, queued_seq;
|
|
|
|
|
|
/* ----- Receive buffer management ----------------------------------------- */
|
|
|
|
|
|
static uint8_t rx_in = 0, rx_out = 0;
|
|
|
|
|
|
static inline void next_buf(uint8_t *index)
|
|
{
|
|
*index = (*index+1) % RX_BUFS;
|
|
}
|
|
|
|
|
|
/* ----- Register access --------------------------------------------------- */
|
|
|
|
|
|
static 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;
|
|
}
|
|
|
|
|
|
static void reg_write(uint8_t reg, uint8_t value)
|
|
{
|
|
spi_begin();
|
|
spi_send(AT86RF230_REG_WRITE | reg);
|
|
spi_send(value);
|
|
spi_end();
|
|
}
|
|
|
|
|
|
/* ----- Interrupt handling ------------------------------------------------ */
|
|
|
|
|
|
static void rx_done(void *user);
|
|
static void tx_ack_done(void *user);
|
|
|
|
|
|
static void usb_next(void)
|
|
{
|
|
const uint8_t *buf;
|
|
|
|
if (rx_in != rx_out) {
|
|
buf = rx_buf[rx_out];
|
|
led(1);
|
|
usb_send(&eps[1], buf, buf[0]+2, rx_done, NULL);
|
|
}
|
|
|
|
if (queued_tx_ack) {
|
|
usb_send(&eps[1], &queued_seq, 1, tx_ack_done, NULL);
|
|
queued_tx_ack = 0;
|
|
}
|
|
}
|
|
|
|
|
|
static void tx_ack_done(void *user)
|
|
{
|
|
usb_next();
|
|
}
|
|
|
|
static void change_state(uint8_t new)
|
|
{
|
|
while ((reg_read(REG_TRX_STATUS) & TRX_STATUS_MASK) ==
|
|
TRX_STATUS_TRANSITION);
|
|
reg_write(REG_TRX_STATE, new);
|
|
}
|
|
|
|
static void rx_done(void *user)
|
|
{
|
|
led(0);
|
|
next_buf(&rx_out);
|
|
usb_next();
|
|
#ifdef RZUSB
|
|
/* slap at86rf230 - reduce fragmentation issue */
|
|
change_state(TRX_STATUS_RX_AACK_ON);
|
|
#endif
|
|
}
|
|
|
|
|
|
static void receive_frame(void)
|
|
{
|
|
uint8_t size;
|
|
uint8_t *buf;
|
|
|
|
spi_begin();
|
|
#ifdef ATUSB
|
|
if (!(spi_io(AT86RF230_BUF_READ) & RX_CRC_VALID)) {
|
|
spi_end();
|
|
return;
|
|
}
|
|
#endif
|
|
#ifdef RZUSB
|
|
spi_io(AT86RF230_BUF_READ);
|
|
#endif
|
|
|
|
size = spi_recv();
|
|
if (!size || (size & 0x80)) {
|
|
spi_end();
|
|
return;
|
|
}
|
|
|
|
buf = rx_buf[rx_in];
|
|
spi_recv_block(buf+1, size+1);
|
|
spi_end();
|
|
|
|
buf[0] = size;
|
|
next_buf(&rx_in);
|
|
|
|
if (eps[1].state == EP_IDLE)
|
|
usb_next();
|
|
}
|
|
|
|
|
|
static bool handle_irq(void)
|
|
{
|
|
uint8_t irq;
|
|
|
|
irq = reg_read(REG_IRQ_STATUS);
|
|
if (!(irq & IRQ_TRX_END))
|
|
return 1;
|
|
|
|
if (txing) {
|
|
if (eps[1].state == EP_IDLE) {
|
|
usb_send(&eps[1], &this_seq, 1, tx_ack_done, NULL);
|
|
} else {
|
|
queued_tx_ack = 1;
|
|
queued_seq = this_seq;
|
|
}
|
|
txing = 0;
|
|
return 1;
|
|
}
|
|
|
|
/* likely */
|
|
if (eps[1].state == EP_IDLE || rx_in != rx_out)
|
|
receive_frame();
|
|
|
|
return 1;
|
|
}
|
|
|
|
|
|
/* ----- TX/RX ------------------------------------------------------------- */
|
|
|
|
|
|
bool mac_rx(int on)
|
|
{
|
|
if (on) {
|
|
mac_irq = handle_irq;
|
|
reg_read(REG_IRQ_STATUS);
|
|
change_state(TRX_CMD_RX_AACK_ON);
|
|
} else {
|
|
mac_irq = NULL;
|
|
change_state(TRX_CMD_FORCE_TRX_OFF);
|
|
txing = 0;
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
|
|
static void do_tx(void *user)
|
|
{
|
|
uint16_t timeout = 0xffff;
|
|
uint8_t status;
|
|
uint8_t i;
|
|
|
|
/*
|
|
* If we time out here, the host driver will time out waiting for the
|
|
* TRX_END acknowledgement.
|
|
*/
|
|
do {
|
|
if (!--timeout)
|
|
return;
|
|
status = reg_read(REG_TRX_STATUS) & TRX_STATUS_MASK;
|
|
}
|
|
while (status != TRX_STATUS_RX_ON && status != TRX_STATUS_RX_AACK_ON);
|
|
|
|
#ifdef ATUSB
|
|
/*
|
|
* We use TRX_CMD_FORCE_PLL_ON instead of TRX_CMD_PLL_ON because a new
|
|
* reception may have begun while we were still working on the previous
|
|
* one.
|
|
*/
|
|
reg_write(REG_TRX_STATE, TRX_CMD_FORCE_PLL_ON);
|
|
#endif
|
|
#ifdef RZUSB
|
|
/*
|
|
* at86rf230 doesn't support force change, nevetherless this works
|
|
* somehow
|
|
*/
|
|
reg_write(REG_TRX_STATE, TRX_CMD_PLL_ON);
|
|
#endif
|
|
|
|
handle_irq();
|
|
|
|
spi_begin();
|
|
spi_send(AT86RF230_BUF_WRITE);
|
|
spi_send(tx_size+2); /* CRC */
|
|
for (i = 0; i != tx_size; i++)
|
|
spi_send(tx_buf[i]);
|
|
spi_end();
|
|
|
|
slp_tr();
|
|
|
|
txing = 1;
|
|
this_seq = next_seq;
|
|
|
|
/*
|
|
* Wait until we reach BUSY_TX, so that we command the transition to
|
|
* RX_AACK_ON which will be executed upon TX completion.
|
|
*/
|
|
change_state(TRX_CMD_RX_AACK_ON);
|
|
}
|
|
|
|
|
|
bool mac_tx(uint16_t flags, uint8_t seq, uint16_t len)
|
|
{
|
|
if (len > MAX_PSDU)
|
|
return 0;
|
|
tx_size = len;
|
|
next_seq = seq;
|
|
usb_recv(&eps[0], tx_buf, len, do_tx, NULL);
|
|
return 1;
|
|
}
|
|
|
|
|
|
void mac_reset(void)
|
|
{
|
|
mac_irq = NULL;
|
|
txing = 0;
|
|
queued_tx_ack = 0;
|
|
rx_in = rx_out = 0;
|
|
next_seq = this_seq = queued_seq = 0;
|
|
|
|
/* enable CRC and PHY_RSSI (with RX_CRC_VALID) in SPI status return */
|
|
reg_write(REG_TRX_CTRL_1,
|
|
TX_AUTO_CRC_ON | SPI_CMD_MODE_PHY_RSSI << SPI_CMD_MODE_SHIFT);
|
|
}
|