1
0
mirror of git://projects.qi-hardware.com/ben-wpan.git synced 2024-08-24 07:52:06 +03:00
ben-wpan/tools/lib/atusb.c
Werner Almesberger a75edf0cc0 tools/lib/: copied USB device opening code over from f32xbase
We shared it across projects so far, but since the code will be further
customized, it's better to keep it local (at least for a while).

- usbopen.h, usbopen.c: moved over f32xbase/include/usb.h and
  f32xbase/lib/usb.c (for further customization)
- Makefile (CFLAGS, OBJS_host): use usbopen.o instead of usb.o from
  f32xbase
- atusb.c: use usbopen.h instead of usb.h from f32xbase (also removes
  include path ambiguity)
2011-04-10 07:43:42 -03:00

292 lines
5.2 KiB
C

/*
* lib/atusb.c - ATUSB access functions library (USB version)
*
* Written 2010-2011 by Werner Almesberger
* Copyright 2010-2011 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 <stdio.h>
#include <usb.h>
#include "atusb/ep0.h"
#include "atusb/usb-ids.h"
#include "usbopen.h"
#include "driver.h"
#define FROM_DEV ATUSB_FROM_DEV(0)
#define TO_DEV ATUSB_TO_DEV(0)
/* ----- error handling ---------------------------------------------------- */
static int error;
static int atusb_error(void *dsc)
{
return error;
}
static int atusb_clear_error(void *dsc)
{
int ret;
ret = error;
error = 0;
return ret;
}
/* ----- open/close -------------------------------------------------------- */
static void *atusb_open(const char *arg)
{
usb_dev_handle *dev;
dev = open_usb(USB_VENDOR, USB_PRODUCT);
if (dev) {
error = 0;
} else {
fprintf(stderr, ":-(\n");
error = 1;
}
return dev;
}
static void atusb_close(void *dsc)
{
/* to do */
}
/* ----- device mode ------------------------------------------------------- */
static void atusb_reset(void *dsc)
{
usb_dev_handle *dev = dsc;
int res;
if (error)
return;
res =
usb_control_msg(dev, TO_DEV, ATUSB_RESET, 0, 0, NULL, 0, 1000);
if (res < 0) {
fprintf(stderr, "ATUSB_RESET: %d\n", res);
error = 1;
}
}
static void atusb_reset_rf(void *dsc)
{
usb_dev_handle *dev = dsc;
int res;
if (error)
return;
res =
usb_control_msg(dev, TO_DEV, ATUSB_RF_RESET, 0, 0, NULL, 0, 1000);
if (res < 0) {
fprintf(stderr, "ATUSB_RF_RESET: %d\n", res);
error = 1;
}
}
static void atusb_test_mode(void *dsc)
{
usb_dev_handle *dev = dsc;
int res;
if (error)
return;
res =
usb_control_msg(dev, TO_DEV, ATUSB_TEST, 0, 0, NULL, 0, 1000);
if (res < 0) {
fprintf(stderr, "ATUSB_TEST: %d\n", res);
error = 1;
}
}
/* ----- register access --------------------------------------------------- */
static void atusb_reg_write(void *dsc, uint8_t reg, uint8_t value)
{
usb_dev_handle *dev = dsc;
int res;
if (error)
return;
res = usb_control_msg(dev, TO_DEV, ATUSB_REG_WRITE, value, reg,
NULL, 0, 1000);
if (res < 0) {
fprintf(stderr, "ATUSB_REG_WRITE: %d\n", res);
error = 1;
}
}
static uint8_t atusb_reg_read(void *dsc, uint8_t reg)
{
usb_dev_handle *dev = dsc;
uint8_t value = 0;
int res;
if (error)
return 0;
res = usb_control_msg(dev, FROM_DEV, ATUSB_REG_READ, 0, reg,
(void *) &value, 1, 1000);
if (res < 0) {
fprintf(stderr, "ATUSB_REG_READ: %d\n", res);
error = 1;
}
return value;
}
/* ----- frame buffer access ----------------------------------------------- */
static void atusb_buf_write(void *dsc, const void *buf, int size)
{
usb_dev_handle *dev = dsc;
int res;
if (error)
return;
res = usb_control_msg(dev, TO_DEV, ATUSB_BUF_WRITE, 0, 0,
(void *) buf, size, 1000);
if (res < 0) {
fprintf(stderr, "ATUSB_BUF_WRITE: %d\n", res);
error = 1;
}
}
static int atusb_buf_read(void *dsc, void *buf, int size)
{
usb_dev_handle *dev = dsc;
int res;
if (error)
return -1;
res = usb_control_msg(dev, FROM_DEV, ATUSB_BUF_READ, 0, 0,
buf, size, 1000);
if (res < 0) {
fprintf(stderr, "ATUSB_BUF_READ: %d\n", res);
error = 1;
}
return res;
}
/* ----- RF interrupt ------------------------------------------------------ */
static int atusb_interrupt(void *dsc)
{
usb_dev_handle *dev = dsc;
uint8_t buf;
int res;
if (error)
return -1;
res = usb_control_msg(dev, FROM_DEV, ATUSB_POLL_INT, 0, 0,
(void *) &buf, 1, 1000);
if (res < 0) {
fprintf(stderr, "ATUSB_POLL_INT: %d\n", res);
error = 1;
}
return buf;
}
/* ----- CLKM handling ----------------------------------------------------- */
/*
* ATmega32U2-based boards don't allow disabling CLKM, so we keep it at 8 MHz.
* We could accommodate a choice between 8 MHz and 16 MHz, but that's for
* later.
*/
static int atusb_set_clkm(void *dsc, int mhz)
{
usb_dev_handle *dev = dsc;
uint8_t ids[3];
int res;
if (error)
return 0;
res = usb_control_msg(dev, FROM_DEV, ATUSB_ID, 0, 0,
(void *) ids, 3, 1000);
if (res < 0) {
fprintf(stderr, "ATUSB_ID: %s\n", usb_strerror());
error = 1;
return 0;
}
switch (ids[2]) {
case HW_TYPE_100813:
case HW_TYPE_101216:
break;
case HW_TYPE_110131:
if (mhz == 0 || mhz == 8)
return 1;
fprintf(stderr, "this board only supports CLKM = 8 MHz\n");
return 0;
default:
fprintf(stderr,
"atusb_set_clkm: unknown hardware type 0x%02x\n", ids[2]);
return 0;
}
return atrf_set_clkm_generic(atusb_reg_write, dsc, mhz);
}
/* ----- driver interface -------------------------------------------------- */
struct atrf_driver atusb_driver = {
.name = "usb",
.open = atusb_open,
.close = atusb_close,
.error = atusb_error,
.clear_error = atusb_clear_error,
.reset = atusb_reset,
.reset_rf = atusb_reset_rf,
.test_mode = atusb_test_mode,
.slp_tr = NULL, /* @@@ not yet */
.set_clkm = atusb_set_clkm,
.reg_write = atusb_reg_write,
.reg_read = atusb_reg_read,
.buf_write = atusb_buf_write,
.buf_read = atusb_buf_read,
.interrupt = atusb_interrupt,
};