//Copyright 2015 <>< Charles Lohr under the ColorChord License. #include "outdrivers.h" #include "notefinder.h" #include #include #include "parameters.h" #include #include #include "color.h" #include #include struct LEDOutDriver { struct libusb_device_handle *devh; int did_init; int zigzag; int total_leds; int array; float outamp; uint8_t * last_leds; volatile int readyFlag; volatile int transferring; int xn; int yn; int rot90; }; static void isocb(struct libusb_transfer *tf) { struct LEDOutDriver *led = (struct LEDOutDriver*)tf->user_data; led->transferring = 0; } static void * LEDOutThread( void * v ) { struct LEDOutDriver * led = (struct LEDOutDriver*)v; while(1) { while(!led->readyFlag || !led->did_init || !led->devh) usleep(100); int transfers = ((led->total_leds + 19) / 20); int totallength = transfers * 64; struct libusb_transfer *tf = libusb_alloc_transfer(transfers); tf->dev_handle = led->devh; tf->flags = LIBUSB_TRANSFER_FREE_TRANSFER; tf->type = LIBUSB_TRANSFER_TYPE_ISOCHRONOUS; tf->num_iso_packets = transfers; libusb_set_iso_packet_lengths(tf, 64); tf->endpoint = 1; tf->flags = 0; tf->buffer = led->last_leds; tf->callback = isocb; tf->length = totallength; tf->user_data = led; if(libusb_submit_transfer(tf)) { led->did_init = 0; printf( "Fault sending LEDs.\n" ); } led->transferring = 1; while(led->transferring) { libusb_handle_events(NULL); } led->readyFlag = 0; } return 0; } static void LEDUpdate(void * id, struct NoteFinder*nf) { int i; struct LEDOutDriver * led = (struct LEDOutDriver*)id; if( !led->did_init ) { led->did_init = 1; if( libusb_init(NULL) < 0 ) { fprintf( stderr, "Error: Could not initialize libUSB\n" ); // exit( -99 ); } led->devh = libusb_open_device_with_vid_pid( NULL, 0x16c0, 0x05dc ); if( !led->devh ) { fprintf( stderr, "Error: Cannot find device.\n" ); // exit( -98 ); } } while( led->readyFlag ) usleep(100); //Advance the LEDs to this position when outputting the values. for( i = 0; i < led->total_leds; i++ ) { int source = i; if( !led->array ) { int sx, sy; if( led->rot90 ) { sy = i % led->yn; sx = i / led->yn; } else { sx = i % led->xn; sy = i / led->xn; } if( led->zigzag ) { if( led->rot90 ) { if( sx & 1 ) { sy = led->yn - sy - 1; } } else { if( sy & 1 ) { sx = led->xn - sx - 1; } } } if( led->rot90 ) { source = sx + sy * led->xn; } else { source = sx + sy * led->yn; } } led->last_leds[i*3+0 + (i / 20 + 1) * 4] = OutLEDs[source*3+1] * led->outamp; led->last_leds[i*3+1 + (i / 20 + 1) * 4] = OutLEDs[source*3+0] * led->outamp; led->last_leds[i*3+2 + (i / 20 + 1) * 4] = OutLEDs[source*3+2] * led->outamp; if(i % 20 == 0) { led->last_leds[(i / 20) * 64 + 0] = i; led->last_leds[(i / 20) * 64 + 1] = i >> 8; led->last_leds[(i / 20) * 64 + 2] = i >> 16; led->last_leds[(i / 20) * 64 + 3] = i >> 24; } } led->readyFlag = 1; } static void LEDParams(void * id ) { struct LEDOutDriver * led = (struct LEDOutDriver*)id; led->total_leds = GetParameterI( "leds", 300 ); led->last_leds = (uint8_t*)malloc( led->total_leds * 3 + 1 + ((led->total_leds + 19) / 20) * 4 + 64); led->outamp = .1; RegisterValue( "ledoutamp", PAFLOAT, &led->outamp, sizeof( led->outamp ) ); led->zigzag = 0; RegisterValue( "zigzag", PAINT, &led->zigzag, sizeof( led->zigzag ) ); led->xn = 16; RegisterValue( "lightx", PAINT, &led->xn, sizeof( led->xn ) ); led->yn = 9; RegisterValue( "lighty", PAINT, &led->yn, sizeof( led->yn ) ); led->rot90 = 0; RegisterValue( "rot90", PAINT, &led->rot90, sizeof( led->rot90 ) ); led->array = 0; RegisterValue( "ledarray", PAINT, &led->array, sizeof( led->array ) ); led->did_init = 0; } static struct DriverInstances * DisplayUSBIsochronous() { struct DriverInstances * ret = malloc( sizeof( struct DriverInstances ) ); memset( ret, 0, sizeof( struct DriverInstances ) ); struct LEDOutDriver * led = ret->id = malloc( sizeof( struct LEDOutDriver ) ); ret->Func = LEDUpdate; ret->Params = LEDParams; OGCreateThread( LEDOutThread, led ); led->readyFlag = 0; LEDParams( led ); return ret; } REGISTER_OUT_DRIVER(DisplayUSBIsochronous);