mirror of
git://projects.qi-hardware.com/ben-wpan.git
synced 2024-12-23 14:45: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>
285 lines
5.3 KiB
C
285 lines
5.3 KiB
C
/*
|
|
* fw/usb/atu2.c - Chip-specific driver for Atmel ATxxxU2 USB chips
|
|
*
|
|
* Written 2008-2011, 2013-2014 by Werner Almesberger
|
|
* Copyright 2008-2011, 2013-2014 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.
|
|
*/
|
|
|
|
/*
|
|
* Known issues:
|
|
* - no suspend/resume
|
|
* - we don't call back after failed transmissions,
|
|
* - we don't reset the EP buffer after failed receptions
|
|
* - enumeration often encounters an error -71 (from which it recovers)
|
|
*/
|
|
|
|
#include <stdbool.h>
|
|
#include <stdint.h>
|
|
|
|
#define F_CPU 8000000UL
|
|
#include <util/delay.h>
|
|
|
|
#include <avr/io.h>
|
|
#include <avr/interrupt.h>
|
|
#include "usb.h"
|
|
#include "board.h"
|
|
|
|
|
|
#ifndef NULL
|
|
#define NULL 0
|
|
#endif
|
|
|
|
#if 1
|
|
#define BUG_ON(cond) do { if (cond) panic(); } while (0)
|
|
#else
|
|
#define BUG_ON(cond)
|
|
#endif
|
|
|
|
|
|
struct ep_descr eps[NUM_EPS];
|
|
|
|
|
|
static uint16_t usb_read_word(void)
|
|
{
|
|
uint8_t low;
|
|
|
|
low = UEDATX;
|
|
return low | UEDATX << 8;
|
|
}
|
|
|
|
|
|
static void enable_addr(void *user)
|
|
{
|
|
while (!(UEINTX & (1 << TXINI)));
|
|
UDADDR |= 1 << ADDEN;
|
|
}
|
|
|
|
|
|
void set_addr(uint8_t addr)
|
|
{
|
|
UDADDR = addr;
|
|
usb_send(&eps[0], NULL, 0, enable_addr, NULL);
|
|
}
|
|
|
|
|
|
void usb_ep_change(struct ep_descr *ep)
|
|
{
|
|
if (ep->state == EP_TX) {
|
|
UENUM = ep-eps;
|
|
UEIENX |= 1 << TXINE;
|
|
}
|
|
}
|
|
|
|
|
|
static bool ep_setup(void)
|
|
{
|
|
struct setup_request setup;
|
|
|
|
BUG_ON(UEBCLX < 8);
|
|
|
|
setup.bmRequestType = UEDATX;
|
|
setup.bRequest = UEDATX;
|
|
setup.wValue = usb_read_word();
|
|
setup.wIndex = usb_read_word();
|
|
setup.wLength = usb_read_word();
|
|
|
|
if (!handle_setup(&setup))
|
|
return 0;
|
|
if (!(setup.bmRequestType & 0x80) && eps[0].state == EP_IDLE)
|
|
usb_send(&eps[0], NULL, 0, NULL, NULL);
|
|
return 1;
|
|
}
|
|
|
|
|
|
static bool ep_rx(struct ep_descr *ep)
|
|
{
|
|
uint8_t size;
|
|
|
|
size = UEBCLX;
|
|
if (size > ep->end-ep->buf)
|
|
return 0;
|
|
while (size--)
|
|
*ep->buf++ = UEDATX;
|
|
if (ep->buf == ep->end) {
|
|
ep->state = EP_IDLE;
|
|
if (ep->callback)
|
|
ep->callback(ep->user);
|
|
// if (ep == &eps[0])
|
|
usb_send(ep, NULL, 0, NULL, NULL);
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
|
|
static void ep_tx(struct ep_descr *ep)
|
|
{
|
|
uint8_t size = ep->end-ep->buf;
|
|
uint8_t left;
|
|
|
|
if (size > ep->size)
|
|
size = ep->size;
|
|
for (left = size; left; left--)
|
|
UEDATX = *ep->buf++;
|
|
if (size == ep->size)
|
|
return;
|
|
ep->state = EP_IDLE;
|
|
}
|
|
|
|
|
|
static void handle_ep(int n)
|
|
{
|
|
struct ep_descr *ep = eps+n;
|
|
uint8_t mask;
|
|
|
|
UENUM = n;
|
|
if (UEINTX & (1 << RXSTPI)) {
|
|
/* @@@ EP_RX. EP_TX: cancel */
|
|
ep->state = EP_IDLE;
|
|
if (!ep_setup())
|
|
goto stall;
|
|
UEINTX = ~(1 << RXSTPI);
|
|
}
|
|
if (UEINTX & (1 << RXOUTI)) {
|
|
/* @@ EP_TX: cancel */
|
|
if (ep->state != EP_RX)
|
|
goto stall;
|
|
if (!ep_rx(ep))
|
|
goto stall;
|
|
/* @@@ gcc 4.5.2 wants this cast */
|
|
UEINTX = (uint8_t) ~(1 << RXOUTI | 1 << FIFOCON);
|
|
}
|
|
if (UEINTX & (1 << STALLEDI)) {
|
|
ep->state = EP_IDLE;
|
|
UEINTX = ~(1 << STALLEDI);
|
|
}
|
|
if (UEINTX & (1 << TXINI)) {
|
|
/* @@ EP_RX: cancel (?) */
|
|
if (ep->state == EP_TX) {
|
|
ep_tx(ep);
|
|
mask = 1 << TXINI;
|
|
if (n)
|
|
mask |= 1 << FIFOCON;
|
|
UEINTX = ~mask;
|
|
if (ep->state == EP_IDLE && ep->callback)
|
|
ep->callback(ep->user);
|
|
} else {
|
|
UEIENX &= ~(1 << TXINE);
|
|
}
|
|
}
|
|
return;
|
|
|
|
stall:
|
|
UEINTX = ~(1 << RXSTPI | 1 << RXOUTI | 1 << STALLEDI);
|
|
ep->state = EP_IDLE;
|
|
UECONX |= 1 << STALLRQ;
|
|
}
|
|
|
|
|
|
static void ep_init(void)
|
|
{
|
|
UENUM = 0;
|
|
UECONX = (1 << RSTDT) | (1 << EPEN); /* enable */
|
|
UECFG0X = 0; /* control, direction is ignored */
|
|
UECFG1X = 3 << EPSIZE0; /* 64 bytes */
|
|
UECFG1X |= 1 << ALLOC;
|
|
|
|
while (!(UESTA0X & (1 << CFGOK)));
|
|
|
|
UEIENX =
|
|
(1 << RXSTPE) | (1 << RXOUTE) | (1 << STALLEDE) | (1 << TXINE);
|
|
|
|
eps[0].state = EP_IDLE;
|
|
eps[0].size = 64;
|
|
|
|
#ifndef BOOT_LOADER
|
|
|
|
UENUM = 1;
|
|
UECONX = (1 << RSTDT) | (1 << EPEN); /* enable */
|
|
UECFG0X = (1 << EPTYPE1) | (1 << EPDIR); /* bulk IN */
|
|
UECFG1X = 3 << EPSIZE0; /* 64 bytes */
|
|
UECFG1X |= 1 << ALLOC;
|
|
|
|
while (!(UESTA0X & (1 << CFGOK)));
|
|
|
|
UEIENX = (1 << STALLEDE) | (1 << TXINE);
|
|
|
|
eps[1].state = EP_IDLE;
|
|
eps[1].size = 64;
|
|
|
|
#endif
|
|
}
|
|
|
|
|
|
ISR(USB_GEN_vect)
|
|
{
|
|
uint8_t flags;
|
|
|
|
flags = UDINT;
|
|
if (flags & (1 << EORSTI)) {
|
|
if (user_reset)
|
|
user_reset();
|
|
ep_init();
|
|
UDINT = ~(1 << EORSTI);
|
|
}
|
|
}
|
|
|
|
|
|
ISR(USB_COM_vect)
|
|
{
|
|
uint8_t flags, i;
|
|
|
|
flags = UEINT;
|
|
for (i = 0; i != NUM_EPS; i++)
|
|
if (flags & (1 << i))
|
|
handle_ep(i);
|
|
}
|
|
|
|
|
|
void usb_reset(void)
|
|
{
|
|
UDCON |= 1 << DETACH; /* detach the pull-up */
|
|
_delay_ms(1);
|
|
}
|
|
|
|
|
|
void usb_init(void)
|
|
{
|
|
USBCON |= 1 << FRZCLK; /* freeze the clock */
|
|
|
|
/* enable the PLL and wait for it to lock */
|
|
#ifdef ATUSB
|
|
PLLCSR &= ~(1 << PLLP2 | 1 << PLLP1 | 1 << PLLP0);
|
|
#endif
|
|
#ifdef RZUSB
|
|
/* TODO sheet page 50 For Atmel AT90USB128x only. Do not use with Atmel AT90USB64x. */
|
|
/* FOR 8 XTAL Mhz only!!! */
|
|
PLLCSR = ((1 << PLLP1) | (1 << PLLP0));
|
|
#endif
|
|
PLLCSR |= 1 << PLLE;
|
|
while (!(PLLCSR & (1 << PLOCK)));
|
|
|
|
#ifdef ATUSB
|
|
USBCON &= ~(1 << USBE); /* reset the controller */
|
|
USBCON |= 1 << USBE;
|
|
#endif
|
|
#ifdef RZUSB
|
|
UHWCON |= (1 << UVREGE);
|
|
|
|
USBCON &= ~((1 << USBE) | (1 << OTGPADE)); /* reset the controller */
|
|
USBCON |= ((1 << USBE) | (1 << OTGPADE));
|
|
#endif
|
|
|
|
USBCON &= ~(1 << FRZCLK); /* thaw the clock */
|
|
|
|
UDCON &= ~(1 << DETACH); /* attach the pull-up */
|
|
UDIEN = 1 << EORSTE; /* enable device interrupts */
|
|
// UDCON |= 1 << RSTCPU; /* reset CPU on bus reset */
|
|
|
|
ep_init();
|
|
}
|