forked from raspberrypi/pico-examples
-
Notifications
You must be signed in to change notification settings - Fork 0
/
max7219_32x8_spi.c
139 lines (112 loc) · 4.14 KB
/
max7219_32x8_spi.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
/**
* Copyright (c) 2022 Raspberry Pi (Trading) Ltd.
*
* SPDX-License-Identifier: BSD-3-Clause
*/
#include <stdio.h>
#include <string.h>
#include "pico/stdlib.h"
#include "pico/binary_info.h"
#include "hardware/spi.h"
/* Example code to talk to a Max7219 driving an 32x8 pixel LED display via SPI
NOTE: The device is driven at 5v, but SPI communications are at 3v3
Connections on Raspberry Pi Pico board and a generic Max7219 board, other
boards may vary.
* GPIO 17 (pin 22) Chip select -> CS on Max7219 board
* GPIO 18 (pin 24) SCK/spi0_sclk -> CLK on Max7219 board
* GPIO 19 (pin 25) MOSI/spi0_tx -> DIN on Max7219 board
* 5v (pin 40) -> VCC on Max7219 board
* GND (pin 38) -> GND on Max7219 board
Note: SPI devices can have a number of different naming schemes for pins. See
the Wikipedia page at https://en.wikipedia.org/wiki/Serial_Peripheral_Interface
for variations.
*/
// This defines how many Max7219 modules we have cascaded together, in this case, we have 4 x 8x8 matrices giving a total of 32x8
#define NUM_MODULES 4
const uint8_t CMD_NOOP = 0;
const uint8_t CMD_DIGIT0 = 1; // Goes up to 8, for each line
const uint8_t CMD_DECODEMODE = 9;
const uint8_t CMD_BRIGHTNESS = 10;
const uint8_t CMD_SCANLIMIT = 11;
const uint8_t CMD_SHUTDOWN = 12;
const uint8_t CMD_DISPLAYTEST = 15;
#ifdef PICO_DEFAULT_SPI_CSN_PIN
static inline void cs_select() {
asm volatile("nop \n nop \n nop");
gpio_put(PICO_DEFAULT_SPI_CSN_PIN, 0); // Active low
asm volatile("nop \n nop \n nop");
}
static inline void cs_deselect() {
asm volatile("nop \n nop \n nop");
gpio_put(PICO_DEFAULT_SPI_CSN_PIN, 1);
asm volatile("nop \n nop \n nop");
}
#endif
#if defined(spi_default) && defined(PICO_DEFAULT_SPI_CSN_PIN)
static void write_register(uint8_t reg, uint8_t data) {
uint8_t buf[2];
buf[0] = reg;
buf[1] = data;
cs_select();
spi_write_blocking(spi_default, buf, 2);
cs_deselect();
sleep_ms(1);
}
static void write_register_all(uint8_t reg, uint8_t data) {
uint8_t buf[2];
buf[0] = reg;
buf[1] = data;
cs_select();
for (int i = 0; i< NUM_MODULES;i++) {
spi_write_blocking(spi_default, buf, 2);
}
cs_deselect();
}
#endif
int main() {
stdio_init_all();
#if !defined(spi_default) || !defined(PICO_DEFAULT_SPI_SCK_PIN) || !defined(PICO_DEFAULT_SPI_TX_PIN) || !defined(PICO_DEFAULT_SPI_CSN_PIN)
#warning spi/max7219_32x8_spi example requires a board with SPI pins
puts("Default SPI pins were not defined");
#else
printf("Hello, max7219! Drawing things on a LED matrix since 2022...\n");
// This example will use SPI0 at 10MHz.
spi_init(spi_default, 10 * 1000 * 1000);
gpio_set_function(PICO_DEFAULT_SPI_SCK_PIN, GPIO_FUNC_SPI);
gpio_set_function(PICO_DEFAULT_SPI_TX_PIN, GPIO_FUNC_SPI);
// Make the SPI pins available to picotool
bi_decl(bi_2pins_with_func(PICO_DEFAULT_SPI_TX_PIN, PICO_DEFAULT_SPI_SCK_PIN, GPIO_FUNC_SPI));
// Chip select is active-low, so we'll initialise it to a driven-high state
gpio_init(PICO_DEFAULT_SPI_CSN_PIN);
gpio_set_dir(PICO_DEFAULT_SPI_CSN_PIN, GPIO_OUT);
gpio_put(PICO_DEFAULT_SPI_CSN_PIN, 1);
// Make the CS pin available to picotool
bi_decl(bi_1pin_with_name(PICO_DEFAULT_SPI_CSN_PIN, "SPI CS"));
// Send init sequence to device
write_register_all(CMD_SHUTDOWN, 0);
write_register_all(CMD_DISPLAYTEST, 0);
write_register_all(CMD_SCANLIMIT, 7); // Use all lines
write_register_all(CMD_DECODEMODE, 0); // No BCD decode, just use bit pattern.
write_register_all(CMD_SHUTDOWN, 1);
write_register_all(CMD_BRIGHTNESS, 8);
for (int i=0; i<256; i++) {
for (int j=0; j<8; j++) {
write_register_all(CMD_DIGIT0+j, i);
}
sleep_ms(20);
}
int bright = 1;
while (true) {
for (int i=0; i<8; i++) {
if (bright & 1) {
write_register_all(CMD_DIGIT0 + i, 170 >> (i%2));
} else {
write_register_all(CMD_DIGIT0 + i, (170 >> 1) << (i%2));
}
write_register_all(CMD_BRIGHTNESS, bright % 16);
}
sleep_ms(250);
bright++;
}
#endif
}