-
Notifications
You must be signed in to change notification settings - Fork 24
/
Copy pathLPD6803.cpp
198 lines (168 loc) · 4.74 KB
/
LPD6803.cpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
/*!
* @file LPD6803.cpp
*
* @mainpage LPD6803 RGB Pixels
*
* @section intro_sec Introduction
*
* Example to control LPD6803-based RGB LED Modules in a strand
* Original code by Bliptronics.com Ben Moyes 2009
* Use this as you wish, but please give credit, or at least buy some of my
*LEDs!
*
* Code cleaned up and Object-ified by ladyada, should be a bit easier to use
*
* Library Optimized for fast refresh rates 2011 by [email protected]
*/
#include "LPD6803.h"
#include <TimerOne.h>
// the arrays of ints that hold each LED's 15 bit color values
static uint16_t *pixels;
static uint16_t numLEDs;
static uint8_t dataPin, clockPin;
/*!
* @brief Various modes for the LPD6803
*/
enum lpd6803mode { START, HEADER, DATA, DONE };
static lpd6803mode
SendMode; // Used in interrupt 0=start,1=header,2=data,3=data done
static byte BitCount; // Used in interrupt
static uint16_t LedIndex; // Used in interrupt - Which LED we are sending.
static byte BlankCounter; // Used in interrupt.
static byte lastdata = 0;
static uint16_t swapAsap =
0; // flag to indicate that the colors need an update asap
// Interrupt routine.
// Frequency was set in setup(). Called once for every bit of data sent
// In your code, set global Sendmode to 0 to re-send the data to the pixels
// Otherwise it will just send clocks.
/*!
* @brief Function for writing to the LEDs
*/
void LedOut() {
// PORTB |= _BV(5); // port 13 LED for timing debug
switch (SendMode) {
case DONE: // Done..just send clocks with zero data
if (swapAsap > 0) {
if (!BlankCounter) // AS SOON AS CURRENT pwm IS DONE. BlankCounter
{
BitCount = 0;
LedIndex = swapAsap; // set current led
SendMode = HEADER;
swapAsap = 0;
}
}
break;
case DATA: // Sending Data
if ((1 << (15 - BitCount)) & pixels[LedIndex]) {
if (!lastdata) { // digitalwrites take a long time, avoid if possible
// If not the first bit then output the next bits
// (Starting with MSB bit 15 down.)
digitalWrite(dataPin, 1);
lastdata = 1;
}
} else {
if (lastdata) { // digitalwrites take a long time, avoid if possible
digitalWrite(dataPin, 0);
lastdata = 0;
}
}
BitCount++;
if (BitCount == 16) // Last bit?
{
LedIndex++; // Move to next LED
if (LedIndex < numLEDs) // Still more leds to go or are we done?
{
BitCount = 0; // Start from the fist bit of the next LED
} else {
// no longer sending data, set the data pin low
digitalWrite(dataPin, 0);
lastdata = 0; // this is a lite optimization
SendMode = DONE; // No more LEDs to go, we are done!
}
}
break;
case HEADER: // Header
if (BitCount < 32) {
digitalWrite(dataPin, 0);
lastdata = 0;
BitCount++;
if (BitCount == 32) {
SendMode =
DATA; // If this was the last bit of header then move on to data.
LedIndex = 0;
BitCount = 0;
}
}
break;
case START: // Start
if (!BlankCounter) // AS SOON AS CURRENT pwm IS DONE. BlankCounter
{
BitCount = 0;
LedIndex = 0;
SendMode = HEADER;
}
break;
}
// Clock out data (or clock LEDs)
digitalWrite(clockPin, HIGH);
digitalWrite(clockPin, LOW);
// Keep track of where the LEDs are at in their pwm cycle.
BlankCounter++;
// PORTB &= ~_BV(5); // pin 13 digital output debug
}
//---
LPD6803::LPD6803(uint16_t n, uint8_t dpin, uint8_t cpin) {
dataPin = dpin;
clockPin = cpin;
numLEDs = n;
pixels = (uint16_t *)malloc(numLEDs);
for (uint16_t i = 0; i < numLEDs; i++) {
setPixelColor(i, 0, 0, 0);
}
SendMode = START;
BitCount = LedIndex = BlankCounter = 0;
cpumax = 50;
}
//---
void LPD6803::begin(void) {
pinMode(dataPin, OUTPUT);
pinMode(clockPin, OUTPUT);
setCPUmax(cpumax);
Timer1.attachInterrupt(
LedOut); // attaches callback() as a timer overflow interrupt
}
//---
uint16_t LPD6803::numPixels(void) { return numLEDs; }
//---
void LPD6803::setCPUmax(uint8_t m) {
cpumax = m;
// each clock out takes 20 microseconds max
long time = 100;
time *= 20; // 20 microseconds per
time /= m; // how long between timers
Timer1.initialize(time);
}
//---
void LPD6803::show(void) { SendMode = START; }
//---
void LPD6803::doSwapBuffersAsap(uint16_t idx) { swapAsap = idx; }
//---
void LPD6803::setPixelColor(uint16_t n, uint8_t r, uint8_t g, uint8_t b) {
uint16_t data;
if (n > numLEDs)
return;
data = g & 0x1F;
data <<= 5;
data |= b & 0x1F;
data <<= 5;
data |= r & 0x1F;
data |= 0x8000;
pixels[n] = data;
}
//---
void LPD6803::setPixelColor(uint16_t n, uint16_t c) {
if (n > numLEDs)
return;
pixels[n] = 0x8000 | c;
}