Erste CVS-Version
This commit is contained in:
23
commandline/Makefile
Normal file
23
commandline/Makefile
Normal file
@ -0,0 +1,23 @@
|
||||
# $Id: Makefile,v 1.1 2006/09/26 18:18:27 rschaten Exp $
|
||||
|
||||
CC = gcc
|
||||
LIBUSB_CONFIG = libusb-config
|
||||
# Make sure that libusb-config is in the search path or specify a full path. On
|
||||
# Windows, there is no libusb-config and you must configure the options below
|
||||
# manually. See examples.
|
||||
|
||||
CFLAGS = `$(LIBUSB_CONFIG) --cflags` -O -Wall -I../common
|
||||
|
||||
LIBS = `$(LIBUSB_CONFIG) --libs`
|
||||
|
||||
all: usb-led-fader
|
||||
|
||||
.c.o:
|
||||
$(CC) $(CFLAGS) -c $<
|
||||
|
||||
usb-led-fader: usb-led-fader.o
|
||||
$(CC) -o usb-led-fader usb-led-fader.o $(LIBS)
|
||||
|
||||
clean:
|
||||
rm -f *.o
|
||||
rm -f usb-led-fader
|
426
commandline/usb-led-fader.c
Normal file
426
commandline/usb-led-fader.c
Normal file
@ -0,0 +1,426 @@
|
||||
/**
|
||||
* \file usb-led-fader.c
|
||||
* \brief Commandline-tool for the USB-LED-Fader.
|
||||
* \author Ronald Schaten
|
||||
* \version $Id: usb-led-fader.c,v 1.1 2006/09/26 18:18:27 rschaten Exp $
|
||||
*
|
||||
* License: See documentation.
|
||||
*/
|
||||
|
||||
#include <stdio.h>
|
||||
#include <stdlib.h>
|
||||
#include <string.h>
|
||||
#include <usb.h> /* this is libusb, see http://libusb.sourceforge.net/ */
|
||||
|
||||
#include "usbledfader.h"
|
||||
#include "channels.h"
|
||||
|
||||
#define USBDEV_SHARED_VENDOR 0x16C0 /**< VOTI */
|
||||
#define USBDEV_SHARED_PRODUCT 0x05DC /**< Obdev's free shared PID. Use obdev's generic shared VID/PID pair and follow the rules outlined in firmware/usbdrv/USBID-License.txt. */
|
||||
|
||||
/* These are error codes for the communication via USB. */
|
||||
#define USB_ERROR_NOTFOUND 1 /**< Error code if the device isn't found. */
|
||||
#define USB_ERROR_ACCESS 2 /**< Error code if the device isn't accessible. */
|
||||
#define USB_ERROR_IO 3 /**< Error code if errors in the communication with the device occur. */
|
||||
|
||||
/**
|
||||
* Displays usage-informations. This function is called if the parameters
|
||||
* cannot be parsed.
|
||||
* \param name The name of this application.
|
||||
*/
|
||||
void usage(char *name)
|
||||
{
|
||||
fprintf(stderr, "usage:\n");
|
||||
fprintf(stderr, " %s status\n", name);
|
||||
fprintf(stderr, " %s set ledId waveId waveformId periodDuration repetitionCount\n", name);
|
||||
fprintf(stderr, " %s clear ledId\n", name);
|
||||
fprintf(stderr, " %s reset\n", name);
|
||||
fprintf(stderr, " %s show waveformId\n", name);
|
||||
fprintf(stderr, " %s test\n\n", name);
|
||||
fprintf(stderr, "parameters:\n");
|
||||
fprintf(stderr, " ledId: ID of the LED (0-%d).\n", CHANNELS - 1);
|
||||
fprintf(stderr, " waveId: ID of the wave (0-1: constant waves, 2: override).\n");
|
||||
fprintf(stderr, " waveformId: ID of the waveform (0-31: brightness, 32-37: patterns).\n");
|
||||
fprintf(stderr, " periodDuration: Time in sec/10 for one repetition of the waveform.\n");
|
||||
fprintf(stderr, " A value of 0 can be used to reset the wave.\n");
|
||||
fprintf(stderr, " repetitionCount: Number of repetitions before switching to the next wave.\n");
|
||||
fprintf(stderr, " A value of 0 can be used to repeat this forever.\n");
|
||||
}
|
||||
|
||||
/**
|
||||
* Reads and converts a string from USB. The conversion to ASCII is 'lossy' (unknown characters become '?').
|
||||
* \param dev Handle of the USB-Device.
|
||||
* \param index Index of the required data.
|
||||
* \param langid Index of the expected language.
|
||||
* \param buf Buffer to contain the return-string.
|
||||
* \param buflen Length of buf.
|
||||
* \return Length of the string.
|
||||
*/
|
||||
int usbGetStringAscii(usb_dev_handle * dev, int index, int langid, char *buf, int buflen) {
|
||||
char buffer[256];
|
||||
int rval, i;
|
||||
|
||||
if ((rval = usb_control_msg(dev, USB_ENDPOINT_IN, USB_REQ_GET_DESCRIPTOR, (USB_DT_STRING << 8) + index, langid, buffer, sizeof(buffer), 1000)) < 0) {
|
||||
return rval;
|
||||
}
|
||||
if (buffer[1] != USB_DT_STRING) {
|
||||
return 0;
|
||||
}
|
||||
if ((unsigned char) buffer[0] < rval) {
|
||||
rval = (unsigned char) buffer[0];
|
||||
}
|
||||
rval /= 2;
|
||||
/* lossy conversion to ISO Latin1 */
|
||||
for (i = 1; i < rval; i++) {
|
||||
if (i > buflen) {
|
||||
/* destination buffer overflow */
|
||||
break;
|
||||
}
|
||||
buf[i - 1] = buffer[2 * i];
|
||||
if (buffer[2 * i + 1] != 0) {
|
||||
/* outside of ISO Latin1 range */
|
||||
buf[i - 1] = '?';
|
||||
}
|
||||
}
|
||||
buf[i - 1] = 0;
|
||||
return i - 1;
|
||||
}
|
||||
|
||||
/**
|
||||
* Connect to the USB-device. Loops through all connected USB-Devices and
|
||||
* searches our counterpart.
|
||||
* \param device Handle to address the device.
|
||||
* \param vendor USBDEV_SHARED_VENDOR as defined.
|
||||
* \param vendorName In our case "www.schatenseite.de".
|
||||
* \param product USBDEV_SHARED_PRODUCT as defined.
|
||||
* \param productName In our case "USB-LED-Fader".
|
||||
* \return Error code.
|
||||
*/
|
||||
int usbOpenDevice(usb_dev_handle ** device, int vendor, char *vendorName, int product, char *productName) {
|
||||
struct usb_bus *bus;
|
||||
struct usb_device *dev;
|
||||
usb_dev_handle *handle = NULL;
|
||||
int errorCode = USB_ERROR_NOTFOUND;
|
||||
static int didUsbInit = 0;
|
||||
|
||||
if (!didUsbInit) {
|
||||
didUsbInit = 1;
|
||||
usb_init();
|
||||
}
|
||||
usb_find_busses();
|
||||
usb_find_devices();
|
||||
for (bus = usb_get_busses(); bus; bus = bus->next) {
|
||||
for (dev = bus->devices; dev; dev = dev->next) {
|
||||
if (dev->descriptor.idVendor == vendor && dev->descriptor.idProduct == product) {
|
||||
char string[256];
|
||||
int len;
|
||||
handle = usb_open(dev); /* we need to open the device in order to query strings */
|
||||
if (!handle) {
|
||||
errorCode = USB_ERROR_ACCESS;
|
||||
fprintf(stderr, "Warning: cannot open USB device: %s\n", usb_strerror());
|
||||
continue;
|
||||
}
|
||||
if (vendorName == NULL && productName == NULL) { /* name does not matter */
|
||||
break;
|
||||
}
|
||||
/* now check whether the names match: */
|
||||
len = usbGetStringAscii(handle, dev->descriptor.iManufacturer, 0x0409, string, sizeof(string));
|
||||
if (len < 0) {
|
||||
errorCode = USB_ERROR_IO;
|
||||
fprintf(stderr, "Warning: cannot query manufacturer for device: %s\n", usb_strerror());
|
||||
} else {
|
||||
errorCode = USB_ERROR_NOTFOUND;
|
||||
/* fprintf(stderr, "seen device from vendor ->%s<-\n", string); */
|
||||
if (strcmp(string, vendorName) == 0) {
|
||||
len = usbGetStringAscii(handle, dev->descriptor.iProduct, 0x0409, string, sizeof(string));
|
||||
if (len < 0) {
|
||||
errorCode = USB_ERROR_IO;
|
||||
fprintf(stderr, "Warning: cannot query product for device: %s\n", usb_strerror());
|
||||
} else {
|
||||
errorCode = USB_ERROR_NOTFOUND;
|
||||
/* fprintf(stderr, "seen product ->%s<-\n", string); */
|
||||
if (strcmp(string, productName) == 0) {
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
usb_close(handle);
|
||||
handle = NULL;
|
||||
}
|
||||
}
|
||||
if (handle) {
|
||||
break;
|
||||
}
|
||||
}
|
||||
if (handle != NULL) {
|
||||
errorCode = 0;
|
||||
*device = handle;
|
||||
}
|
||||
return errorCode;
|
||||
}
|
||||
|
||||
/**
|
||||
* Test connection to the device. The test consists of writing 1000 random
|
||||
* numbers to the device and checking the echo. This should discover systematic
|
||||
* bit errors (e.g. in bit stuffing).
|
||||
* \param handle Handle to talk to the device.
|
||||
* \param argc Number of arguments.
|
||||
* \param argv Arguments.
|
||||
*/
|
||||
void dev_test(usb_dev_handle *handle, int argc, char** argv) {
|
||||
unsigned char buffer[8];
|
||||
int nBytes;
|
||||
int i, v, r;
|
||||
if (argc != 2) {
|
||||
usage(argv[0]);
|
||||
exit(1);
|
||||
}
|
||||
for (i = 0; i < 1000; i++) {
|
||||
v = rand() & 0xffff;
|
||||
nBytes = usb_control_msg(handle, USB_TYPE_VENDOR | USB_RECIP_DEVICE | USB_ENDPOINT_IN, CMD_ECHO, v, 0, (char *) buffer, sizeof(buffer), 5000);
|
||||
if (nBytes < 2) {
|
||||
if (nBytes < 0) {
|
||||
fprintf(stderr, "USB error: %s\n", usb_strerror());
|
||||
}
|
||||
fprintf(stderr, "only %d bytes received in iteration %d\n", nBytes, i);
|
||||
exit(1);
|
||||
}
|
||||
r = buffer[0] | (buffer[1] << 8);
|
||||
if (r != v) {
|
||||
fprintf(stderr, "data error: received 0x%x instead of 0x%x in iteration %d\n", r, v, i);
|
||||
exit(1);
|
||||
}
|
||||
}
|
||||
printf("test succeeded\n");
|
||||
}
|
||||
|
||||
/**
|
||||
* Set waves. It is possible to set any number of waves at once.
|
||||
* \param handle Handle to talk to the device.
|
||||
* \param argc Number of arguments.
|
||||
* \param argv Arguments.
|
||||
*/
|
||||
void dev_set(usb_dev_handle *handle, int argc, char** argv) {
|
||||
unsigned char buffer[8];
|
||||
int nBytes;
|
||||
int parameter;
|
||||
if ((argc < 7) || ((argc - 2) % 5 != 0)) {
|
||||
usage(argv[0]);
|
||||
exit(1);
|
||||
}
|
||||
for (parameter = 2; (parameter + 4) < argc; parameter += 5) {
|
||||
int ledId = atoi(argv[parameter + 0]);
|
||||
if ((ledId < 0) || (ledId > (CHANNELS - 1))) {
|
||||
fprintf(stderr, "invalid ledId: %d\n", ledId);
|
||||
exit(1);
|
||||
}
|
||||
int waveId = atoi(argv[parameter + 1]);
|
||||
if ((waveId < 0) || (waveId > 2)) {
|
||||
fprintf(stderr, "invalid waveId: %d\n", waveId);
|
||||
exit(1);
|
||||
}
|
||||
int waveformId = atoi(argv[parameter + 2]);
|
||||
if ((waveformId < 0) || (waveformId > 38)) {
|
||||
fprintf(stderr, "invalid waveformId: %d\n", waveformId);
|
||||
exit(1);
|
||||
}
|
||||
int periodDuration = atoi(argv[parameter + 3]);
|
||||
if ((periodDuration < 0) || (periodDuration > 255)) {
|
||||
fprintf(stderr, "invalid periodDuration: %d\n", periodDuration);
|
||||
exit(1);
|
||||
}
|
||||
int repetitionCount = atoi(argv[parameter + 4]);
|
||||
if ((repetitionCount < 0) || (repetitionCount > 255)) {
|
||||
fprintf(stderr, "invalid repetitionCount: %d\n", repetitionCount);
|
||||
exit(1);
|
||||
}
|
||||
|
||||
buffer[0] = CMD_SET;
|
||||
buffer[1] = ledId;
|
||||
buffer[2] = waveId;
|
||||
buffer[3] = waveformId;
|
||||
buffer[4] = periodDuration;
|
||||
buffer[5] = repetitionCount;
|
||||
|
||||
nBytes = usb_control_msg(handle, USB_TYPE_VENDOR | USB_RECIP_DEVICE | USB_ENDPOINT_OUT, CMD_SET, ledId, 0, (char *) buffer, sizeof(buffer), 5000);
|
||||
|
||||
if (nBytes < 0) {
|
||||
fprintf(stderr, "USB error: %s\n", usb_strerror());
|
||||
exit(1);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Clear all waves on one LED.
|
||||
* \param handle Handle to talk to the device.
|
||||
* \param argc Number of arguments.
|
||||
* \param argv Arguments.
|
||||
*/
|
||||
void dev_clear(usb_dev_handle *handle, int argc, char** argv) {
|
||||
unsigned char buffer[8];
|
||||
int nBytes;
|
||||
if (argc != 3) {
|
||||
usage(argv[0]);
|
||||
exit(1);
|
||||
}
|
||||
int ledId = atoi(argv[2]);
|
||||
if ((ledId < 0) || (ledId > (CHANNELS - 1))) {
|
||||
fprintf(stderr, "invalid LED: %d\n", ledId);
|
||||
exit(1);
|
||||
}
|
||||
nBytes = usb_control_msg(handle, USB_TYPE_VENDOR | USB_RECIP_DEVICE | USB_ENDPOINT_OUT, CMD_CLEAR, ledId, 0, (char *) buffer, sizeof(buffer), 5000);
|
||||
if (nBytes < 0) {
|
||||
fprintf(stderr, "USB error: %s\n", usb_strerror());
|
||||
exit(1);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Get the status of the device. Status information is printed in detail.
|
||||
* \param handle Handle to talk to the device.
|
||||
* \param argc Number of arguments.
|
||||
* \param argv Arguments.
|
||||
*/
|
||||
void dev_status(usb_dev_handle *handle, int argc, char** argv) {
|
||||
int nBytes;
|
||||
int i, j;
|
||||
static fade_GlobalData fade_globalData; /* contains the state of all four LEDs. */
|
||||
if (argc != 2) {
|
||||
usage(argv[0]);
|
||||
exit(1);
|
||||
}
|
||||
nBytes = usb_control_msg(handle, USB_TYPE_VENDOR | USB_RECIP_DEVICE | USB_ENDPOINT_IN, CMD_GET, 0, 0, (char *) &fade_globalData, sizeof(fade_globalData), 5000);
|
||||
if (nBytes < 0) {
|
||||
fprintf(stderr, "USB error: %s\n", usb_strerror());
|
||||
exit(1);
|
||||
}
|
||||
if (nBytes != sizeof(fade_globalData)) {
|
||||
fprintf(stderr, "USB oddity: %d bytes received, %d bytes expected.\n", nBytes, sizeof(fade_globalData));
|
||||
exit(1);
|
||||
}
|
||||
for (i = 0; i < CHANNELS; i++) {
|
||||
printf("LED %d %10s %10s %10s %10s %10s\n", i, "curid", "curvalue", "curpos", "currep", "nextupd");
|
||||
printf(" %10d %10d %10d %10d %10d\n",
|
||||
fade_globalData.led[i].waveCurrentId,
|
||||
fade_globalData.led[i].waveCurrentValue,
|
||||
fade_globalData.led[i].waveCurrentPosition,
|
||||
fade_globalData.led[i].waveCurrentRepetition,
|
||||
fade_globalData.led[i].waveNextUpdate);
|
||||
printf("%10s %10s %10s %10s %10s %10s\n", "wave", "waveform", "length", "repeat", "duration", "updtime");
|
||||
for (j = 0; j < 3; j++) {
|
||||
printf("%10d %10d %10d %10d %10d %10d\n",
|
||||
j,
|
||||
fade_globalData.led[i].wave[j].waveformId,
|
||||
fade_globalData.led[i].wave[j].waveformLength,
|
||||
fade_globalData.led[i].wave[j].waveformRepetition,
|
||||
fade_globalData.led[i].wave[j].waveformDuration,
|
||||
fade_globalData.led[i].wave[j].waveformUpdateTime);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Reset the device.
|
||||
* \param handle Handle to talk to the device.
|
||||
* \param argc Number of arguments.
|
||||
* \param argv Arguments.
|
||||
*/
|
||||
void dev_reset(usb_dev_handle *handle, int argc, char** argv) {
|
||||
unsigned char buffer[8];
|
||||
int nBytes;
|
||||
if (argc != 2) {
|
||||
usage(argv[0]);
|
||||
exit(1);
|
||||
}
|
||||
nBytes = usb_control_msg(handle, USB_TYPE_VENDOR | USB_RECIP_DEVICE | USB_ENDPOINT_OUT, CMD_RESET, 0, 0, (char *) buffer, sizeof(buffer), 5000);
|
||||
if (nBytes < 0) {
|
||||
fprintf(stderr, "USB error: %s\n", usb_strerror());
|
||||
exit(1);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Show a waveform. This will not send a command to the device, the waveform is
|
||||
* only printed on the screen.
|
||||
* \param handle Handle to talk to the device (not needed).
|
||||
* \param argc Number of arguments.
|
||||
* \param argv Arguments.
|
||||
*/
|
||||
int dev_show(int argc, char **argv) {
|
||||
if (argc != 3) {
|
||||
usage(argv[0]);
|
||||
exit(1);
|
||||
}
|
||||
int waveformId = atoi(argv[2]);
|
||||
if ((waveformId < 0) || (waveformId > 38)) {
|
||||
fprintf(stderr, "invalid waveformId: %d\n", waveformId);
|
||||
exit(1);
|
||||
}
|
||||
int i, j;
|
||||
int length = fade_calculateWaveform(waveformId, 0);
|
||||
printf("wave %2d - length %2d\n", waveformId, length);
|
||||
for (i = 31; i > 0; i--) {
|
||||
printf("%2d: ", i);
|
||||
for (j = 1; j <= length; j++) {
|
||||
if (fade_calculateWaveform(waveformId, j) >= i) {
|
||||
printf("*");
|
||||
} else {
|
||||
printf(" ");
|
||||
}
|
||||
}
|
||||
printf("\n");
|
||||
}
|
||||
printf(" ");
|
||||
for (j = 1; j <= length; j++) {
|
||||
printf("=");
|
||||
}
|
||||
printf("\n");
|
||||
exit(0);
|
||||
}
|
||||
|
||||
/**
|
||||
* Main function. Initializes the USB-device, parses commandline-parameters and
|
||||
* calls the functions that communicate with the device.
|
||||
* \param argc Number of arguments.
|
||||
* \param argv Arguments.
|
||||
* \return Error code.
|
||||
*/
|
||||
int main(int argc, char **argv)
|
||||
{
|
||||
usb_dev_handle *handle = NULL;
|
||||
|
||||
if (argc < 2) {
|
||||
usage(argv[0]);
|
||||
exit(1);
|
||||
}
|
||||
usb_init();
|
||||
if (usbOpenDevice (&handle, USBDEV_SHARED_VENDOR, "www.schatenseite.de", USBDEV_SHARED_PRODUCT, "USB-LED-Fader") != 0) {
|
||||
fprintf(stderr, "Could not find USB device \"USB-LED-Fader\" with vid=0x%x pid=0x%x\n", USBDEV_SHARED_VENDOR, USBDEV_SHARED_PRODUCT);
|
||||
exit(1);
|
||||
}
|
||||
/* We have searched all devices on all busses for our USB device above. Now
|
||||
* try to open it and perform the vendor specific control operations for the
|
||||
* function requested by the user.
|
||||
*/
|
||||
if (strcmp(argv[1], "test") == 0) {
|
||||
dev_test(handle, argc, argv);
|
||||
} else if (strcmp(argv[1], "set") == 0) {
|
||||
dev_set(handle, argc, argv);
|
||||
} else if (strcmp(argv[1], "clear") == 0) {
|
||||
dev_clear(handle, argc, argv);
|
||||
} else if (strcmp(argv[1], "status") == 0) {
|
||||
dev_status(handle, argc, argv);
|
||||
} else if (strcmp(argv[1], "reset") == 0) {
|
||||
dev_reset(handle, argc, argv);
|
||||
} else if (strcmp(argv[1], "show") == 0) {
|
||||
dev_reset(handle, argc, argv);
|
||||
} else {
|
||||
usage(argv[0]);
|
||||
exit(1);
|
||||
}
|
||||
usb_close(handle);
|
||||
return 0;
|
||||
}
|
||||
|
Reference in New Issue
Block a user