====== DMX led wash ====== ~~META:description abstract=A DMX512 led wash receiver/driver~~ A DMX512 led wash receiver/driver, current source version (single power leds) and open drain version (led strips). Takes 3 DMX channels (RGB). ===== Features ===== * Up to 9A output (open drain version, with heatsink) * Uses 3 DMX channels * 7-30V input * Cheap ATMEGA8L based * Small footprint * Extension header with ADC and INT lines for sound sensor or such (standalone mode) ===== Electronics ===== ==== Maximum ratings ==== === Open drain === AP4800 on 1"2 clad has a junction-to-ambient thermal resistance of 50°C/W, on minimal clad it is 125°C/W, in our case it should be close to 100°C/W. == Average study == Maximum temperature will be 125°C, taking ambient temperature of 25°C we get at most 1W power dissipation per MOS, which means a 7.4A current. == Worse case study == Maximum temperature will be 100°C, taking ambient temperature of 50°C we get at most 0.5W power dissipation per MOS, which means a 5.2A current. == Maximum power study == Taking a current of 9.5A we get a power dissipation of 1.63W. To avoid going over 125°C from a 25°C ambient temperature it is required to use a 35°C/W heatsink (5.7°C/W in the worst case). === Current source === In this case and without heatsink we should avoid dissipation over 0.5W per MOS (by using series resistors). Using heatsink will allow to go higher than that. ==== Schematic ==== {{schematic>schematic.svg}} ===== Source code ===== /** * @defgroup RGB leds DMX512 controller * @type firmware * @file main.c * @author Etienne Meleard * @creation 2014-06-20 * @tabsize 4 * @mcu ATMEGA8L * @fcpu 8MHz */ //============================ WARNING ===============================// //!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!// // // // Fuses must be like SUT1 = 1, SUT0 = 1, CKSEL3 = 1, CKSEL2 = 1, // // CKSEL1 = 1 and CKSEL0 = 0 to ensure using the crystal as clock // // (otherwise it uses the 8MHz internal clock), full swing mode is // // mandatory to acheive 8MHz operation. // // // // CKDIV8 fuse must not be set to avoid clock frequency divided by 8 // // // // Fuse set or = 1 means checked in ponyprog // // // //!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!// //====================================================================// // ================= Fuses ================= // /*FUSES = { // No success writing fuse this way, command line below ... .low = (FUSE_SUT1 & FUSE_SUT0 & FUSE_CKSEL3), .high = HFUSE_DEFAULT, .extended = EFUSE_DEFAULT, };*/ // Run // avrdude -c usbasp -p m8 -U lfuse:w:0xfe:m -U hfuse:w:0xd1:m -B250 // at first (-F may be needed) // ================ Includes =============== // #include #include #include #include // ================ Pinouts ================ // #define LED Bit(PORTD).bit5 #define RX Bit(PIND).bit0 #define RX_COMPLETE SIG_UART_RECV #define RX_INT Bit(PORTD).bit2 #define RX_SIG SIG_INTERRUPT0 #define RX_TIMER TCNT0 #define BREAK_LEN 11 #define MAB_LEN 1 #define DIP_0 Bit(PINC).bit0 #define DIP_1 Bit(PINC).bit1 #define DIP_2 Bit(PINC).bit2 #define DIP_3 Bit(PINC).bit3 #define DIP_A Bit(PORTC).bit4 #define DIP_B Bit(PORTB).bit4 #define DIP_C Bit(PORTB).bit5 #define EXT_ADC 6 #define EXT_INPUT Bit(PIND).bit3 #define EXT_OUTPUT Bit(PORTD).bit3 #define EXT_SIG SIG_INTERRUPT1 #define OUT_RED Bit(PORTB).bit1 #define RED_CH OCR1A #define OUT_GREEN Bit(PORTB).bit2 #define GREEN_CH OCR1B #define OUT_BLUE Bit(PORTB).bit3 #define BLUE_CH OCR2 // ================ Macros ================= // #define BLINK LED=1; _delay_ms(150); LED=0; _delay_ms(150) #define PULSE10 LED=1; _delay_us(10); LED=0; _delay_us(10) #define PULSE5 LED=1; _delay_us(5); LED=0; _delay_us(5) #define PULSE3 LED=1; _delay_us(3); LED=0; _delay_us(3) #define PULSE1 LED=1; _delay_us(1); LED=0; _delay_us(1) #define RX_START UCSRB |= 0x10; receiving = 1 #define RX_STOP UCSRB &= 0xef; receiving = 0 #define ADDR_4 ((uint8_t)((PINC ^ 0x0f) & 0x0f)) #define DIP_SETTIME 10 // us, allows diodes to set (due to diode capacitance) #define RX_ERROR ((UCSRA & 0x10) || (UCSRA & 0x08)) #define MAX_TIME_WITHOUT_START 10000 #define ADDRESS_REFRESH_PERIOD 3000 #define PATTERN_FIXED 0 #define PATTERN_RAINBOW 1 #define PATTERN_BREATH 2 // ================ Globals ================ // volatile uint16_t address = 0x0000; volatile uint8_t receiving = 0; volatile uint16_t received = 0; volatile const uint8_t colors[12][3] = { {0xff, 0x00, 0x00}, // Linear colors {0x7f, 0x3f, 0x00}, {0x7f, 0x7f, 0x00}, {0x3f, 0x7f, 0x00}, {0x00, 0xff, 0x00}, {0x00, 0x7f, 0x3f}, {0x00, 0x7f, 0x7f}, {0x00, 0x3f, 0x7f}, {0x00, 0x00, 0xff}, {0x3f, 0x00, 0x7f}, {0x7f, 0x00, 0x7f}, {0x7f, 0x00, 0x3f} /*{0xff, 0x00, 0x00}, // Not linear {0xff, 0x7f, 0x00}, {0xff, 0xff, 0x00}, {0x7f, 0xff, 0x00}, {0x00, 0xff, 0x00}, {0x00, 0xff, 0x7f}, {0x00, 0xff, 0xff}, {0x00, 0x7f, 0xff}, {0x00, 0x00, 0xff}, {0x7f, 0x00, 0xff}, {0xff, 0x00, 0xff}, {0xff, 0x00, 0x7f}*/ }; volatile const uint16_t speeds[2][8] = { { // Rainbow : 1536 steps for each cycle, measured speed : c = 6.26 T + 17.49 31 /*5s*/, 63 /*10s*/, 186 /*30s*/, 373 /*60s*/, 562 /*90s*/, 752 /*120s*/, 1864 /*300s*/, 3740 /*600s*/ }, { // Breath : 256 steps for each cycle, so speed = 10000 T / 256, min is ??? 19 /*1s*/, 37 /*2s*/, 94 /*5s*/, 187 /*10s*/, 375 /*20s*/, 562 /*30s*/, 1125 /*60s*/, 2250 /*120s*/ } }; // ================ Functions ============== // void init(void) { // Led as output DDRD |= 1 << 5; // USART UCSRA = 0x00; // No doubling of frequency UCSRB = 0x80; // Rx disabled, Rx irq enabled, 8bit mode UCSRC = 0x8e; // Async, no parity, 2 stops, 8 bits UBRRL = 1; // 250kbds // Detection timer (timer 0) TCCR0 = 0x03; // 64 prescaler, increment every 8us // Dip inputs DDRC &= 0xf0; PORTC |= 0x0f; // Pull-up // Dip outputs DDRC |= 0x10; DDRB |= 0x30; DIP_A = 1; DIP_B = 1; DIP_C = 1; // RGB outputs as outputs DDRB |= 0x0e; // R/G timer (timer 1) TCCR1A = 0xa1; // 8 bits, fast, non-inverting PWM on OC1A and OC1B TCCR1B = 0x04; // with 256 prescaler ~ 122Hz RED_CH = 0x00; // Red channel to 0 GREEN_CH = 0x00; // Green channel to 0 // B timer (timer 2) TCCR2 = 0x66; // 8 bits, fast, non-inverting PWM on OC2 BLUE_CH = 0x00; // Blue channel to 0 } void getAddress(void) { uint16_t buffer = 0x0000; DIP_A = 1; DIP_B = 1; DIP_C = 1; DIP_C = 0; _delay_us(DIP_SETTIME); buffer += ADDR_4; DIP_C = 1; buffer <<= 4; DIP_B = 0; _delay_us(DIP_SETTIME); buffer += ADDR_4; DIP_B = 1; buffer <<= 4; DIP_A = 0; _delay_us(DIP_SETTIME); buffer += ADDR_4; DIP_A = 1; address = buffer; } uint8_t getFrameStart(void) { uint32_t t = 1500000; // 1.5s while(t && RX) { _delay_us(1); t--; } if(!t) return 0; RX_TIMER = 0; while(!RX && RX_TIMER < BREAK_LEN) _delay_us(1); if(RX_TIMER < BREAK_LEN) return 0; // Error, break too short while(!RX) _delay_us(1); // Wait for MAB RX_TIMER = 0; while(RX && RX_TIMER < MAB_LEN) _delay_us(1); if(RX_TIMER < MAB_LEN) return 0; // Error, break too short received = 0; RX_START; return 1; } // =============== Interrupts ============== // SIGNAL(RX_COMPLETE) { uint8_t data; data = UDR; if(!receiving) return; if(UCSRA & 0x10) return; if(UCSRA & 0x08)return; received++; if(received == 1) { //if(data != 0x00) RX_STOP; // Should be 0x00, not a setting packet otherwise return; // Ignore first slot } if(received < address + 2) { return; // Address not reached } if(received > address + 4) { // Received enough packets, stop receiving and wait for next frame RX_STOP; return; } // Got channel, set it switch(received - address - 2) { case 0: RED_CH = data; break; case 1: GREEN_CH = data; break; case 2: BLUE_CH = data; break; } } // ================== Main ================= // int main(void) { uint16_t time_without_start = 0; uint16_t address_is_valid = 0; uint8_t auto_mode = 0; uint8_t pattern = 0; uint8_t speed = 0; uint8_t stepped = 0; uint8_t color = 0; uint8_t color_index = 0; uint16_t wait_step = 0; uint16_t step = 0; uint16_t level = 0; cli(); // WatchDog disabled AT ALL TIMES ! init(); sei(); BLINK; BLINK; while(1) { if(!address_is_valid) { getAddress(); auto_mode = address >> 9; pattern = (address >> 7) & 0x03; if(auto_mode) switch(pattern) { case PATTERN_FIXED : color = address & 0x3f; break; case PATTERN_RAINBOW : speed = (address >> 4) & 0x07; stepped = (address >> 3) & 0x01; break; case PATTERN_BREATH : speed = (address >> 4) & 0x07; color = address & 0x0f; //color_index = color; break; } address_is_valid = ADDRESS_REFRESH_PERIOD; } if(auto_mode) { if(!wait_step) { // Update output switch(pattern) { case PATTERN_FIXED : RED_CH = (color << 6) & 0xc0; GREEN_CH = (color << 4) & 0xc0; BLUE_CH = (color << 2) & 0xc0; wait_step = (uint16_t)3500; // 0.35s break; case PATTERN_RAINBOW : if(color) { color_index = color - 1; }else color_index = 11; if(stepped) { // 25% (x2) holding if(step < 32) level = 0; else if(step >= 96) level = 127; else level = 2 * step - 64; }else level = step; RED_CH = ((127 - level) * colors[color_index][0] + level * colors[color][0]) >> 7; GREEN_CH = ((127 - level) * colors[color_index][1] + level * colors[color][1]) >> 7; BLUE_CH = ((127 - level) * colors[color_index][2] + level * colors[color][2]) >> 7; step++; if(step >= 128) { step = 0; color++; if(color >= 12) { color = 0; } } wait_step = speeds[0][speed];// - (stepped ? 25 : 19); // stepped cycle code takes approx. 25 (TODO), fade cycle code takes approx. 19 break; case PATTERN_BREATH : if(step < 256) { level = step; }else{ level = 511 - step; } RED_CH = (level * colors[color_index][0]) >> 8; GREEN_CH = (level * colors[color_index][1]) >> 8; BLUE_CH = (level * colors[color_index][2]) >> 8; step++; if(step >= 512) { step = 0; if(color == 0) { color_index += 2; if(color_index >= 12) color_index = 0; }else color_index = color - 1; } wait_step = speeds[1][speed]; // TODO modulate dep on cycle approx time break; } } if(wait_step) wait_step--; }else{ // DMX if(getFrameStart()) { LED = 0; time_without_start = 0; } if(time_without_start > MAX_TIME_WITHOUT_START) { if(receiving) RX_STOP; BLINK; } time_without_start++; } if(address_is_valid) address_is_valid--; _delay_us(100); } return 0; } # Name: Makefile # Tabsize: 4 DEVICE = atmega8 F_CPU = 8000000 # in Hz FUSE_L = # see below for fuse values for particular devices FUSE_H = AVRDUDE = avrdude -c usbasp -p m8 #$(DEVICE) # edit this line for your programmer #CFLAGS = -Iusbdrv -I. -DDEBUG_LEVEL=0 CFLAGS = -I. -DDEBUG_LEVEL=0 #OBJECTS = usbdrv/usbdrv.o usbdrv/usbdrvasm.o usbdrv/oddebug.o main.o OBJECTS = main.o COMPILE = avr-gcc -Wall -Os -DF_CPU=$(F_CPU) $(CFLAGS) -mmcu=$(DEVICE) # symbolic targets: help: @echo "This Makefile has no default rule. Use one of the following:" @echo "make hex ....... to build main.hex" @echo "make flash ..... to flash the firmware (use this on metaboard)" @echo "make clean ..... to delete objects and hex file" hex: main.hex # rule for uploading firmware: flash: main.hex $(AVRDUDE) -U flash:w:main.hex:i # rule for deleting dependent files (those which can be built by Make): clean: rm -f main.hex main.lst main.obj main.cof main.list main.map main.eep.hex main.elf *.o # rm -f main.hex main.lst main.obj main.cof main.list main.map main.eep.hex main.elf *.o usbdrv/*.o main.s usbdrv/oddebug.s usbdrv/usbdrv.s # Generic rule for compiling C files: .c.o: $(COMPILE) -c $< -o $@ # Generic rule for assembling Assembler source files: .S.o: $(COMPILE) -x assembler-with-cpp -c $< -o $@ # "-x assembler-with-cpp" should not be necessary since this is the default # file type for the .S (with capital S) extension. However, upper case # characters are not always preserved on Windows. To ensure WinAVR # compatibility define the file type manually. # Generic rule for compiling C to assembler, used for debugging only. .c.s: $(COMPILE) -S $< -o $@ # file targets: main.elf: $(OBJECTS) # usbdrv dependency only needed because we copy it $(COMPILE) -o main.elf $(OBJECTS) main.hex: main.elf rm -f main.hex main.eep.hex avr-objcopy -j .text -j .data -O ihex main.elf main.hex avr-size main.hex # debugging targets: disasm: main.elf avr-objdump -d main.elf cpp: $(COMPILE) -E main.c ===== Downloads ===== * {{:projects:dmx_led_wash:main.hex|main.hex}} * {{:projects:dmx_led_wash:cdm_v2.10.00_whql_certified.exe|USB driver for OpenDMX hardware}} * {{:projects:dmx_led_wash:dmx_theatre.zip|Open DMX theater}} ===== How to ===== ===== Manual ===== ==== DMX mode ==== DMX mode is activated by setting the 10th dip switch to 0. In this mode dip switches 1 to 9 set the base channel (1 to 512). The wash uses 3 channels : Red, Green and Blue. ==== Auto mode ==== Auto mode is activated by setting the 10th dip switch to 1. In this mode dip switches 8 and 9 select the pattern : ^ 8-9 dip ^ pattern ^ configuration ^^ ^:::^:::^ dip range ^ role ^ | 00 | fixed color | 1-2 | red level, 0-3 (max) | |:::|:::| 3-4 | green level, 0-3 (max) | |:::|:::| 5-6 | blue level, 0-3 (max) | | 10 | rainbow (12 colors, see breathing) | 4 | step mode in "continous change" (0) and "stay a bit on each color" (1) | |:::|:::| 5-7 | speed in : 2s (000), 10s (100), 30s (010), 60s (110), 90s (001), 120s (101), 300s (011) and 600s (111) | | 01 | color breathing | 1-4 | color, 0 means color rolling, 1-12 select color in : red (1000), orange (0100), yellow (1100), lime (0010), green (1010), green-blue (0110), cyan (1110), light-blue (0001), blue (1001), blue-magenta (0101), magenta (1101) and pink (0011) | |:::|:::| 5-7 | speed in : 1s (000), 2s (100), 5s (010), 10s (110), 20s (001), 30s (101), 60s (011) and 120s (111) | | 11 | Reserved for audio sensing extension ||| {{tag>led microcontroller lighting dmx}}