mirror of
git://projects.qi-hardware.com/ben-wpan.git
synced 2025-01-10 12:30:15 +02:00
2961482cac
- include/atrf.h (atrf_driver_spec), lib/atrf.c (atrf_driver_spec): new function to retrieve the local or remote/final driver spec - lib/atrf.c (struct atrf_dsc, atrf_open, atrf_close): record the local driver spec - lib/driver.h (struct atrf_driver): new driver function "driver_spec" to retrieve the driver spec - lib/atnet.c (struct atnet_dsc, atnet_open, atnet_close): maintain a cache for the driver spec - lib/atnet.c (atnet_driver_spec, atnet_driver): added support for the "driver_spec" function - atrf-proxy/PROTOCOL, atrf-proxy/atrf-proxy.c (cmd_zero): added command SPEC to retrieve the (final) driver spec - atrf-id/atrf-id.c (usage, main): added option -s to retrieve the driver spec. One -s retrieves the local spec, -s -s the remote/final.
213 lines
3.9 KiB
C
213 lines
3.9 KiB
C
/*
|
|
* atrf-id/atrf-id.c - Identify a ben-wpan AT86RF230 board
|
|
*
|
|
* 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 <stdlib.h>
|
|
#include <stdio.h>
|
|
#include <unistd.h>
|
|
|
|
#ifdef HAVE_USB
|
|
#include <usb.h>
|
|
#endif
|
|
|
|
#include "at86rf230.h"
|
|
#include "atusb/ep0.h"
|
|
#include "atrf.h"
|
|
|
|
|
|
#ifdef HAVE_USB
|
|
|
|
#define FROM_DEV ATUSB_FROM_DEV(0)
|
|
|
|
#define BUF_SIZE 256
|
|
|
|
|
|
static int get_id(usb_dev_handle *dev, void *data, int size)
|
|
{
|
|
int res;
|
|
|
|
res = usb_control_msg(dev, FROM_DEV, ATUSB_ID, 0, 0, data, size, 1000);
|
|
if (res < 0)
|
|
fprintf(stderr, "ATUSB_ID: %s\n", usb_strerror());
|
|
return res;
|
|
}
|
|
|
|
|
|
static int get_protocol(usb_dev_handle *dev,
|
|
uint8_t *major, uint8_t *minor, uint8_t *target)
|
|
{
|
|
uint8_t ids[3];
|
|
|
|
if (get_id(dev, ids, 3) < 0)
|
|
return -1;
|
|
if (major)
|
|
*major = ids[0];
|
|
if (minor)
|
|
*minor = ids[1];
|
|
if (target)
|
|
*target = ids[2];
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
static int get_build(usb_dev_handle *dev, char *buf, size_t size)
|
|
{
|
|
int res;
|
|
|
|
res = usb_control_msg(dev, FROM_DEV, ATUSB_BUILD, 0, 0, buf, size,
|
|
1000);
|
|
if (res < 0)
|
|
fprintf(stderr, "ATUSB_BUILD: %s\n", usb_strerror());
|
|
return res;
|
|
}
|
|
|
|
|
|
static void show_usb_info(struct atrf_dsc *dsc)
|
|
{
|
|
usb_dev_handle *dev;
|
|
const struct usb_device *device;
|
|
uint8_t major, minor, target;
|
|
const char *mcu;
|
|
char buf[BUF_SIZE+1]; /* +1 for terminating \0 */
|
|
int len;
|
|
|
|
dev = atrf_usb_handle(dsc);
|
|
if (!dev)
|
|
return;
|
|
device = usb_device(dev);
|
|
|
|
printf("%04x:%04x ",
|
|
device->descriptor.idVendor, device->descriptor.idProduct);
|
|
|
|
if (get_protocol(dev, &major, &minor, &target) < 0)
|
|
exit(1);
|
|
switch (target) {
|
|
case HW_TYPE_100813:
|
|
case HW_TYPE_101216:
|
|
mcu = "C8051F326";
|
|
break;
|
|
case HW_TYPE_110131:
|
|
mcu = "ATmega32U2";
|
|
break;
|
|
default:
|
|
mcu = "???";
|
|
}
|
|
printf("protocol %u.%u hw %u (%s)\n", major, minor, target, mcu);
|
|
|
|
len = get_build(dev, buf, sizeof(buf)-1);
|
|
if (len < 0)
|
|
exit(1);
|
|
buf[len] = 0;
|
|
printf("%10s%s\n", "", buf);
|
|
}
|
|
|
|
|
|
#else /* HAVE_USB */
|
|
|
|
|
|
static void show_usb_info(struct atrf_dsc *dsc)
|
|
{
|
|
}
|
|
|
|
|
|
#endif /* !HAVE_USB */
|
|
|
|
|
|
static void show_info(struct atrf_dsc *dsc)
|
|
{
|
|
uint8_t part, version, man_id_0, man_id_1;
|
|
|
|
show_usb_info(dsc);
|
|
|
|
printf("%10s", "");
|
|
|
|
switch (atrf_identify(dsc)) {
|
|
case atrf_unknown_chip:
|
|
printf("???");
|
|
break;
|
|
case artf_at86rf230:
|
|
printf("AT86RF230");
|
|
break;
|
|
case artf_at86rf231:
|
|
printf("AT86RF231");
|
|
break;
|
|
default:
|
|
abort();
|
|
}
|
|
|
|
part = atrf_reg_read(dsc, REG_PART_NUM);
|
|
version = atrf_reg_read(dsc, REG_VERSION_NUM);
|
|
man_id_0 = atrf_reg_read(dsc, REG_MAN_ID_0);
|
|
man_id_1 = atrf_reg_read(dsc, REG_MAN_ID_1);
|
|
printf(", part 0x%02x version %u manufacturer xxxx%02x%02x",
|
|
part, version, man_id_1, man_id_0);
|
|
|
|
printf(" (%s)\n", man_id_1 == 0 && man_id_0 == 0x1f ? "Atmel" : "???");
|
|
}
|
|
|
|
|
|
static void usage(const char *name)
|
|
{
|
|
fprintf(stderr,
|
|
"usage: %s [-d driver[:arg]] [-s [-s]]\n\n"
|
|
" -d driver[:arg] use the specified driver (default: %s)\n"
|
|
" -s print only the local driver specification\n"
|
|
" -s -s print only the remote driver specification\n"
|
|
, name, atrf_default_driver_name());
|
|
exit(1);
|
|
}
|
|
|
|
|
|
int main(int argc, char *const *argv)
|
|
{
|
|
const char *driver = NULL;
|
|
struct atrf_dsc *dsc;
|
|
int spec_only = 0;
|
|
int c;
|
|
|
|
while ((c = getopt(argc, argv, "d:s")) != EOF)
|
|
switch (c) {
|
|
case 'd':
|
|
driver = optarg;
|
|
break;
|
|
case 's':
|
|
spec_only++;
|
|
break;
|
|
default:
|
|
usage(*argv);
|
|
}
|
|
if (argc != optind)
|
|
usage(*argv);
|
|
|
|
dsc = atrf_open(driver);
|
|
if (!dsc)
|
|
return 1;
|
|
|
|
if (spec_only) {
|
|
const char *spec = atrf_driver_spec(dsc, spec_only > 1);
|
|
|
|
if (spec)
|
|
printf("%s\n", spec);
|
|
else {
|
|
fprintf(stderr, "can't obtain specification\n");
|
|
exit(1);
|
|
}
|
|
} else {
|
|
show_info(dsc);
|
|
}
|
|
|
|
atrf_close(dsc);
|
|
|
|
return 0;
|
|
}
|