Important changes to repositories hosted on mbed.com
Mbed hosted mercurial repositories are deprecated and are due to be permanently deleted in July 2026.
To keep a copy of this software download the repository Zip archive or clone locally using Mercurial.
It is also possible to export all your personal repositories from the account settings page.
Fork of SD600A by
SD600A.cpp@28:a905a4738dd4, 2013-07-28 (annotated)
- Committer:
- heroic
- Date:
- Sun Jul 28 08:37:54 2013 +0000
- Revision:
- 28:a905a4738dd4
- Parent:
- 27:6667543f3b28
- Child:
- 29:0200c63e1158
Cleanup.
Who changed what in which revision?
User | Revision | Line number | New contents of line |
---|---|---|---|
heroic | 23:2e60470dd134 | 1 | // Mbed library to control SD600A-based RGB LED Strips |
heroic | 23:2e60470dd134 | 2 | // Partially based on work (c) 2011 Jelmer Tiete |
heroic | 23:2e60470dd134 | 3 | // |
heroic | 23:2e60470dd134 | 4 | // Ported from Arduino by |
heroic | 23:2e60470dd134 | 5 | // Jas Strong <jasmine@electronpusher.org> |
heroic | 23:2e60470dd134 | 6 | /*****************************************************************************/ |
heroic | 23:2e60470dd134 | 7 | |
heroic | 23:2e60470dd134 | 8 | #include "rtos.h" |
heroic | 23:2e60470dd134 | 9 | #include "LedStrip.h" |
heroic | 23:2e60470dd134 | 10 | #include "SD600A.h" |
heroic | 23:2e60470dd134 | 11 | |
heroic | 26:98163818a82e | 12 | SD600ASupervisor::SD600ASupervisor(void) { |
heroic | 26:98163818a82e | 13 | numstrips = 0; |
heroic | 26:98163818a82e | 14 | byte_index = 0; |
heroic | 26:98163818a82e | 15 | bit_index = 0; |
heroic | 26:98163818a82e | 16 | } |
heroic | 26:98163818a82e | 17 | |
heroic | 26:98163818a82e | 18 | void SD600ASupervisor::add(SD600A *strip) { |
heroic | 26:98163818a82e | 19 | strips[numstrips] = strip; |
heroic | 26:98163818a82e | 20 | numstrips++; |
heroic | 26:98163818a82e | 21 | if (numstrips == 1) { // this is the first one, set up the interrupt. |
heroic | 26:98163818a82e | 22 | data_length = strip->data_length; |
heroic | 26:98163818a82e | 23 | NVIC_SetPriority(TIMER3_IRQn, 0); |
heroic | 26:98163818a82e | 24 | idletoggle.attach_us(this, &SD600ASupervisor::spi_isr, INTERRUPT_INTERVAL); |
heroic | 26:98163818a82e | 25 | } |
heroic | 26:98163818a82e | 26 | } |
heroic | 26:98163818a82e | 27 | |
heroic | 23:2e60470dd134 | 28 | /* |
heroic | 25:14e5984d3aae | 29 | * Soft SPI clockout routine. Triggered every few microseconds, |
heroic | 23:2e60470dd134 | 30 | * repeatedly clocks out the strip contents. Since each update function |
heroic | 23:2e60470dd134 | 31 | * actually updates a byte at a time, and the C-m3 peeks the write buffer, |
heroic | 23:2e60470dd134 | 32 | * there is no need to lock the buffer. |
heroic | 23:2e60470dd134 | 33 | */ |
heroic | 23:2e60470dd134 | 34 | |
heroic | 26:98163818a82e | 35 | void SD600ASupervisor::spi_isr(void) { |
heroic | 26:98163818a82e | 36 | int i; |
heroic | 26:98163818a82e | 37 | SD600A *curstrip; |
heroic | 26:98163818a82e | 38 | for (i=0; i<numstrips; i++) { |
heroic | 26:98163818a82e | 39 | curstrip = strips[i]; |
heroic | 26:98163818a82e | 40 | curstrip->clk = 1; |
heroic | 26:98163818a82e | 41 | curstrip->dat = curstrip->pixels[byte_index] & (0x80 >> bit_index); |
heroic | 26:98163818a82e | 42 | } |
heroic | 26:98163818a82e | 43 | for (i=0; i<numstrips; i++) |
heroic | 26:98163818a82e | 44 | strips[i]->clk = 0; |
heroic | 23:2e60470dd134 | 45 | bit_index++; |
heroic | 26:98163818a82e | 46 | for (i=0; i<numstrips; i++) |
heroic | 26:98163818a82e | 47 | strips[i]->clk = 1; |
heroic | 26:98163818a82e | 48 | for (i=0; i<numstrips; i++) { |
heroic | 26:98163818a82e | 49 | curstrip = strips[i]; |
heroic | 26:98163818a82e | 50 | curstrip->dat = curstrip->pixels[byte_index] & (0x80 >> bit_index); |
heroic | 26:98163818a82e | 51 | } |
heroic | 26:98163818a82e | 52 | for (i=0; i<numstrips; i++) |
heroic | 26:98163818a82e | 53 | strips[i]->clk = 0; |
heroic | 23:2e60470dd134 | 54 | bit_index++; |
heroic | 23:2e60470dd134 | 55 | if (bit_index == 8) { |
heroic | 23:2e60470dd134 | 56 | byte_index++; |
heroic | 23:2e60470dd134 | 57 | bit_index = 0; |
heroic | 23:2e60470dd134 | 58 | } |
heroic | 23:2e60470dd134 | 59 | if (byte_index == data_length) |
heroic | 23:2e60470dd134 | 60 | byte_index = 0; |
heroic | 23:2e60470dd134 | 61 | } |
heroic | 23:2e60470dd134 | 62 | |
heroic | 26:98163818a82e | 63 | |
heroic | 26:98163818a82e | 64 | |
heroic | 23:2e60470dd134 | 65 | SD600A::SD600A(PinName dataPin, PinName clockPin, int n) : |
heroic | 23:2e60470dd134 | 66 | dat(dataPin), |
heroic | 23:2e60470dd134 | 67 | clk(clockPin) { |
heroic | 23:2e60470dd134 | 68 | // Allocate 3 bytes per pixel: |
heroic | 23:2e60470dd134 | 69 | numLEDs = n; |
heroic | 28:a905a4738dd4 | 70 | pixels = (uint8_t *)malloc(4+ numLEDs * 3); |
heroic | 28:a905a4738dd4 | 71 | if (pixels) { |
heroic | 23:2e60470dd134 | 72 | memset(pixels, 0, numLEDs * 3); // Init to RGB 'off' state |
heroic | 23:2e60470dd134 | 73 | pixels[numLEDs*3] = 0x7f; |
heroic | 23:2e60470dd134 | 74 | pixels[numLEDs*3+1] = 0xff; |
heroic | 23:2e60470dd134 | 75 | pixels[numLEDs*3+2] = 0xff; |
heroic | 23:2e60470dd134 | 76 | pixels[numLEDs*3+3] = 0x80; |
heroic | 23:2e60470dd134 | 77 | data_length = numLEDs*3 +4; |
heroic | 23:2e60470dd134 | 78 | byte_index = 0; |
heroic | 23:2e60470dd134 | 79 | bit_index = 0; |
heroic | 23:2e60470dd134 | 80 | } else error("SD600A could not allocate memory!\r\n"); |
heroic | 23:2e60470dd134 | 81 | } |
heroic | 23:2e60470dd134 | 82 | |
heroic | 23:2e60470dd134 | 83 | void SD600A::begin(void) { |
heroic | 23:2e60470dd134 | 84 | // Null stub. |
heroic | 23:2e60470dd134 | 85 | } |
heroic | 23:2e60470dd134 | 86 | |
heroic | 23:2e60470dd134 | 87 | uint16_t SD600A::numPixels(void) { |
heroic | 23:2e60470dd134 | 88 | return numLEDs; |
heroic | 23:2e60470dd134 | 89 | } |
heroic | 23:2e60470dd134 | 90 | |
heroic | 23:2e60470dd134 | 91 | void SD600A::blank(void) { |
heroic | 23:2e60470dd134 | 92 | memset(pixels, 0x00, numLEDs * 3); |
heroic | 23:2e60470dd134 | 93 | } |
heroic | 23:2e60470dd134 | 94 | |
heroic | 23:2e60470dd134 | 95 | void SD600A::show(void) { |
heroic | 23:2e60470dd134 | 96 | // Null stub, since shows continuously. |
heroic | 23:2e60470dd134 | 97 | } |
heroic | 23:2e60470dd134 | 98 | |
heroic | 23:2e60470dd134 | 99 | uint32_t SD600A::total_luminance(void) { |
heroic | 23:2e60470dd134 | 100 | uint32_t running_total; |
heroic | 23:2e60470dd134 | 101 | running_total = 0; |
heroic | 23:2e60470dd134 | 102 | for (int i=0; i<numLEDs*3; i++) |
heroic | 23:2e60470dd134 | 103 | running_total += pixels[i]; |
heroic | 23:2e60470dd134 | 104 | return running_total; |
heroic | 23:2e60470dd134 | 105 | } |
heroic | 23:2e60470dd134 | 106 | |
heroic | 23:2e60470dd134 | 107 | // Convert R,G,B to combined 32-bit color |
heroic | 23:2e60470dd134 | 108 | uint32_t SD600A::Color(uint8_t r, uint8_t g, uint8_t b) { |
heroic | 23:2e60470dd134 | 109 | // Take 23 bits of the value and append them end to end |
heroic | 23:2e60470dd134 | 110 | // We cannot drive all ones or it will make the part latch if the previous word ended in one! |
heroic | 23:2e60470dd134 | 111 | return 0xfefefe & ((uint32_t)b << 16) | ((uint32_t)r << 8) | (uint32_t)g; |
heroic | 23:2e60470dd134 | 112 | } |
heroic | 23:2e60470dd134 | 113 | |
heroic | 23:2e60470dd134 | 114 | // store the rgb component in our array |
heroic | 23:2e60470dd134 | 115 | void SD600A::setPixelColor(uint16_t n, uint8_t r, uint8_t g, uint8_t b) { |
heroic | 23:2e60470dd134 | 116 | if (n >= numLEDs) return; // '>=' because arrays are 0-indexed |
heroic | 23:2e60470dd134 | 117 | n=numLEDs-(n+1); |
heroic | 23:2e60470dd134 | 118 | |
heroic | 23:2e60470dd134 | 119 | pixels[n*3 ] = g & 0xfe; |
heroic | 23:2e60470dd134 | 120 | pixels[n*3+1] = b & 0xfe; |
heroic | 23:2e60470dd134 | 121 | pixels[n*3+2] = r & 0xfe; |
heroic | 23:2e60470dd134 | 122 | } |
heroic | 23:2e60470dd134 | 123 | |
heroic | 23:2e60470dd134 | 124 | void SD600A::setPixelR(uint16_t n, uint8_t r) { |
heroic | 23:2e60470dd134 | 125 | if (n >= numLEDs) return; // '>=' because arrays are 0-indexed |
heroic | 23:2e60470dd134 | 126 | n=numLEDs-(n+1); |
heroic | 23:2e60470dd134 | 127 | pixels[n*3+2] = r & 0xfe; |
heroic | 23:2e60470dd134 | 128 | } |
heroic | 23:2e60470dd134 | 129 | |
heroic | 27:6667543f3b28 | 130 | // Set all in one function call, assuming GRB ordering |
heroic | 27:6667543f3b28 | 131 | void SD600A::setPackedPixels(uint8_t * buffer, uint32_t n) { |
heroic | 27:6667543f3b28 | 132 | if (n >= numLEDs) return; |
heroic | 27:6667543f3b28 | 133 | for (int i=0; i<n*3; i++) |
heroic | 27:6667543f3b28 | 134 | buffer[i] &= 0xfe; |
heroic | 27:6667543f3b28 | 135 | memcpy(pixels, buffer, (size_t) (n*3)); |
heroic | 27:6667543f3b28 | 136 | } |
heroic | 27:6667543f3b28 | 137 | |
heroic | 27:6667543f3b28 | 138 | |
heroic | 23:2e60470dd134 | 139 | void SD600A::setPixelG(uint16_t n, uint8_t g) { |
heroic | 23:2e60470dd134 | 140 | if (n >= numLEDs) return; // '>=' because arrays are 0-indexed |
heroic | 23:2e60470dd134 | 141 | n=numLEDs-(n+1); |
heroic | 23:2e60470dd134 | 142 | pixels[n*3+1] = g & 0xfe; |
heroic | 23:2e60470dd134 | 143 | } |
heroic | 23:2e60470dd134 | 144 | |
heroic | 23:2e60470dd134 | 145 | void SD600A::setPixelB(uint16_t n, uint8_t b) { |
heroic | 23:2e60470dd134 | 146 | if (n >= numLEDs) return; // '>=' because arrays are 0-indexed |
heroic | 23:2e60470dd134 | 147 | n=numLEDs-(n+1); |
heroic | 23:2e60470dd134 | 148 | pixels[n*3] = b & 0xfe; |
heroic | 23:2e60470dd134 | 149 | } |
heroic | 23:2e60470dd134 | 150 | |
heroic | 23:2e60470dd134 | 151 | void SD600A::setPixelColor(uint16_t n, uint32_t c) { |
heroic | 23:2e60470dd134 | 152 | if (n >= numLEDs) return; // '>=' because arrays are 0-indexed |
heroic | 23:2e60470dd134 | 153 | n=numLEDs-(n+1); |
heroic | 23:2e60470dd134 | 154 | pixels[n*3 ] = (c >> 16) & 0xfe; |
heroic | 23:2e60470dd134 | 155 | pixels[n*3+1] = (c >> 8) & 0xfe; |
heroic | 23:2e60470dd134 | 156 | pixels[n*3+2] = c & 0xfe; |
heroic | 23:2e60470dd134 | 157 | } |