forked from RoXXoR/ws2812
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmain.c
149 lines (125 loc) · 4.13 KB
/
main.c
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
#include <msp430.h>
#include "ws2812/ws2812.h"
#define NUMBEROFLEDS 15
#define ENCODING 3 // possible values 3 and 4
void configClock(void);
void configSPI(void);
void sendBuffer(uint8_t* buffer, ledcount_t ledCount);
void sendBufferDma(uint8_t* buffer, ledcount_t ledCount);
int main(void) {
// buffer to store encoded transport data
uint8_t frameBuffer[(ENCODING * sizeof(ledcolor_t) * NUMBEROFLEDS)] = { 0, };
// array with 15 rbg colors
ledcolor_t leds[NUMBEROFLEDS] = {
// rainbow colors
{ 0xFF, 0x00, 0x00 },
{ 0xFF, 0x3F, 0x00 },
{ 0xFF, 0x7F, 0x00 },
{ 0xFF, 0xFF, 0x00 },
{ 0x00, 0xFF, 0x00 },
{ 0x00, 0xFF, 0x3F },
{ 0x00, 0xFF, 0x7F },
{ 0x00, 0xFF, 0xFF },
{ 0x00, 0x00, 0xFF },
{ 0x3F, 0x00, 0xFF },
{ 0x7F, 0x00, 0xFF },
{ 0xFF, 0x00, 0xFF },
{ 0xFF, 0x00, 0x7F },
{ 0xFF, 0x00, 0x3F },
{ 0xFF, 0x00, 0x0F },
};
uint8_t update;
ledcolor_t blankLed = {0x00, 0x00, 0x00};
uint8_t colorIdx;
ledcolor_t led;
WDTCTL = WDTPW | WDTHOLD; // Stop watchdog timer
configClock();
configSPI();
while(1) {
// blank all LEDs
fillFrameBufferSingleColor(&blankLed, NUMBEROFLEDS, frameBuffer, ENCODING);
sendBufferDma(frameBuffer, NUMBEROFLEDS);
__delay_cycles(0xFFFFF);
// set one LED after an other (one more with each round) with the colors from the LEDs array
fillFrameBuffer(leds, NUMBEROFLEDS, frameBuffer, ENCODING);
for(update=1; update <= NUMBEROFLEDS; update++) {
sendBuffer(frameBuffer, update);
__delay_cycles(0xFFFFF);
}
__delay_cycles(0xFFFFFF);
led = blankLed;
// set all LEDs with the same color and simulate a sunrise
for(colorIdx=0; colorIdx < 0xFF; colorIdx++) {
led.red = colorIdx + 1;
fillFrameBufferSingleColor(&led, NUMBEROFLEDS, frameBuffer, ENCODING);
sendBufferDma(frameBuffer, NUMBEROFLEDS);
__delay_cycles(0x1FFFF);
}
for(colorIdx=0; colorIdx < 0xD0; colorIdx++) {
led.green = colorIdx;
fillFrameBufferSingleColor(&led, NUMBEROFLEDS, frameBuffer, ENCODING);
sendBufferDma(frameBuffer, NUMBEROFLEDS);
__delay_cycles(0x2FFFF);
}
for(colorIdx=0; colorIdx < 0x50; colorIdx++) {
led.blue = colorIdx;
fillFrameBufferSingleColor(&led, NUMBEROFLEDS, frameBuffer, ENCODING);
sendBuffer(frameBuffer, NUMBEROFLEDS);
__delay_cycles(0x3FFFF);
}
__delay_cycles(0xFFFFF);
}
return 0;
}
// copy bytes from the buffer to SPI transmit register
// should be reworked to use DMA
void sendBuffer(uint8_t* buffer, ledcount_t ledCount) {
uint16_t bufferIdx;
for (bufferIdx=0; bufferIdx < (ENCODING * sizeof(ledcolor_t) * ledCount); bufferIdx++) {
while (!(UCA0IFG & UCTXIFG)); // wait for TX buffer to be ready
UCA0TXBUF_L = buffer[bufferIdx];
}
__delay_cycles(300);
}
void sendBufferDma(uint8_t* buffer, ledcount_t ledCount) {
DMA0SA = (__SFR_FARPTR) buffer; // source address
DMA0DA = (__SFR_FARPTR) &UCA0TXBUF_L; // destination address
// single transfer, source increment, source and destination byte access, interrupt enable
DMACTL0 = DMA0TSEL__UCA0TXIFG; // DMA0 trigger input
DMA0SZ = (ENCODING * sizeof(ledcolor_t) * ledCount);
DMA0CTL = DMADT_0 | DMASRCINCR_3 | DMASRCBYTE | DMADSTBYTE | DMAEN; //| DMAIE;
//start DMA
UCA0IFG ^= UCTXIFG;
UCA0IFG ^= UCTXIFG;
}
// configure MCLK and SMCLK to be sourced by DCO with a frequency of
// 8Mhz (3-bit encoding)
// 6.7MHz (4-bit encoding)
void configClock(void) {
CSCTL0_H = 0xA5;
#if ENCODING == 3
CSCTL1 = DCOFSEL_3; // DCO frequency setting = 8MHz
#else
CSCTL1 = DCOFSEL_2; // DCO frequency setting = 6.7MHz
#endif
CSCTL2 = SELS__DCOCLK + SELM__DCOCLK;
CSCTL3 = DIVS__1 + DIVM__1;
}
void configSPI(void) {
UCA0CTLW0 |= UCSWRST; // **Put state machine in reset**
UCA0CTLW0 |= UCMST + UCSYNC + UCCKPL + UCMSB; // 3-pin, 8-bit SPI master
// Clock polarity high, MSB
UCA0CTLW0 |= UCSSEL__SMCLK; // SMCLK
#if ENCODING == 3
UCA0BR0 = 3; // SPICLK 8MHz/3 = 2.66MHz
UCA0BR1 = 0;
#else
UCA0BR0 = 2; // SPICLK 6.7MHz/2 = 3.35MHz
UCA0BR1 = 0;
#endif
UCA0MCTLW = 0; // No modulation
UCA0CTLW0 &= ~UCSWRST;
// UCB0SIMO = LED data signal on P2.0
P2SEL0 &= ~BIT0;
P2SEL1 |= BIT0;
}