-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathhardware.c
295 lines (241 loc) · 8.23 KB
/
hardware.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
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
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
#include "hardware.h"
#include <driver/gpio.h>
#include <driver/spi_master.h>
#include <driver/i2c.h>
#include <esp_log.h>
#include "managed_i2c.h"
#include "rp2040.h"
#include "pax_gfx.h"
static const char* TAG = "hardware";
static BNO055 dev_bno055 = {0};
static ILI9341 dev_ili9341 = {0};
static ICE40 dev_ice40 = {0};
static RP2040 dev_rp2040 = {0};
static BME680 dev_bme680 = {0};
static uint8_t rp2040_fw_version = 0;
static bool bsp_ready = false;
static bool rp2040_ready = false;
static bool ice40_ready = false;
static bool bno055_ready = false;
static bool bme680_ready = false;
static xSemaphoreHandle i2c_semaphore = NULL;
static pax_buf_t pax_buffer;
esp_err_t ice40_get_done_wrapper(bool* done) {
uint16_t buttons;
esp_err_t res = rp2040_read_buttons(&dev_rp2040, &buttons);
if (res != ESP_OK) return res;
*done = !((buttons >> 5) & 0x01);
return ESP_OK;
}
esp_err_t ice40_set_reset_wrapper(bool reset) {
esp_err_t res = rp2040_set_fpga(&dev_rp2040, reset);
vTaskDelay(100 / portTICK_PERIOD_MS);
return res;
}
void ili9341_set_lcd_mode(bool mode) {
ESP_LOGI(TAG, "LCD mode switch to %s", mode ? "FPGA" : "ESP32");
esp_err_t res = gpio_set_level(GPIO_LCD_MODE, mode);
if (res != ESP_OK) {
ESP_LOGE(TAG, "Setting LCD mode failed");
}
}
static esp_err_t _bus_init() {
esp_err_t res;
// I2C bus
i2c_config_t i2c_config = {
.mode = I2C_MODE_MASTER,
.sda_io_num = GPIO_I2C_SDA,
.scl_io_num = GPIO_I2C_SCL,
.master.clk_speed = I2C_SPEED,
.sda_pullup_en = false,
.scl_pullup_en = false,
.clk_flags = 0
};
res = i2c_param_config(I2C_BUS, &i2c_config);
if (res != ESP_OK) {
ESP_LOGE(TAG, "Configuring I2C bus parameters failed");
return res;
}
res = i2c_set_timeout(I2C_BUS, I2C_TIMEOUT * 80);
if (res != ESP_OK) {
ESP_LOGE(TAG, "Configuring I2C bus timeout failed");
return res;
}
res = i2c_driver_install(I2C_BUS, i2c_config.mode, 0, 0, 0);
if (res != ESP_OK) {
ESP_LOGE(TAG, "Initializing system I2C bus failed");
return res;
}
i2c_semaphore = xSemaphoreCreateBinary();
xSemaphoreGive( i2c_semaphore );
// SPI bus
spi_bus_config_t busConfiguration = {0};
busConfiguration.mosi_io_num = GPIO_SPI_MOSI;
busConfiguration.miso_io_num = GPIO_SPI_MISO;
busConfiguration.sclk_io_num = GPIO_SPI_CLK;
busConfiguration.quadwp_io_num = -1;
busConfiguration.quadhd_io_num = -1;
busConfiguration.max_transfer_sz = SPI_MAX_TRANSFER_SIZE;
res = spi_bus_initialize(SPI_BUS, &busConfiguration, SPI_DMA_CHANNEL);
if (res != ESP_OK) {
ESP_LOGE(TAG, "Initializing SPI bus failed");
return res;
}
return ESP_OK;
}
esp_err_t bsp_init() {
if (bsp_ready) return ESP_OK;
esp_err_t res;
// Interrupts
res = gpio_install_isr_service(0);
if (res != ESP_OK) {
ESP_LOGE(TAG, "Installing ISR service failed");
return res;
}
// Communication busses
res = _bus_init();
if (res != ESP_OK) return res;
// LCD display
dev_ili9341.spi_bus = SPI_BUS;
dev_ili9341.pin_cs = GPIO_SPI_CS_LCD;
dev_ili9341.pin_dcx = GPIO_SPI_DC_LCD;
dev_ili9341.pin_reset = GPIO_LCD_RESET;
dev_ili9341.rotation = 1;
dev_ili9341.color_mode = true; // Blue and red channels are swapped
dev_ili9341.spi_speed = 40000000; // 40MHz
dev_ili9341.spi_max_transfer_size = SPI_MAX_TRANSFER_SIZE;
dev_ili9341.callback = ili9341_set_lcd_mode; // Callback for changing LCD mode between ESP32 and FPGA
res = gpio_set_direction(GPIO_LCD_MODE, GPIO_MODE_OUTPUT);
if (res != ESP_OK) {
ESP_LOGE(TAG, "Initializing LCD mode GPIO failed");
return res;
}
res = ili9341_init(&dev_ili9341);
if (res != ESP_OK) {
ESP_LOGE(TAG, "Initializing LCD failed");
return res;
}
pax_buf_init(&pax_buffer, NULL, ILI9341_WIDTH, ILI9341_HEIGHT, PAX_BUF_16_565RGB);
pax_buf_reversed(&pax_buffer, true);
bsp_ready = true;
return ESP_OK;
}
esp_err_t bsp_rp2040_init() {
if (!bsp_ready) return ESP_FAIL;
if (rp2040_ready) return ESP_OK;
// RP2040 co-processor
dev_rp2040.i2c_bus = I2C_BUS;
dev_rp2040.i2c_address = RP2040_ADDR;
dev_rp2040.pin_interrupt = GPIO_INT_RP2040;
dev_rp2040.queue = xQueueCreate(8, sizeof(rp2040_input_message_t));
dev_rp2040.i2c_semaphore = i2c_semaphore;
esp_err_t res = rp2040_init(&dev_rp2040);
if (res != ESP_OK) {
ESP_LOGE(TAG, "Initializing RP2040 failed");
return res;
}
if (rp2040_get_firmware_version(&dev_rp2040, &rp2040_fw_version) != ESP_OK) {
ESP_LOGE(TAG, "Failed to read RP2040 firmware version");
return ESP_FAIL;
}
rp2040_ready = true;
return ESP_OK;
}
esp_err_t bsp_ice40_init() {
if (!bsp_ready) return ESP_FAIL;
if (!rp2040_ready) return ESP_FAIL;
if (rp2040_fw_version == 0xFF) return ESP_FAIL; // The ICE40 FPGA can only be controlled when the RP2040 is not in bootloader mode
if (ice40_ready) return ESP_OK;
dev_ice40.spi_bus = SPI_BUS;
dev_ice40.pin_cs = GPIO_SPI_CS_FPGA;
dev_ice40.pin_done = -1;
dev_ice40.pin_reset = -1;
dev_ice40.pin_int = GPIO_INT_FPGA;
dev_ice40.spi_speed_full_duplex = 26700000;
dev_ice40.spi_speed_half_duplex = 40000000;
dev_ice40.spi_speed_turbo = 80000000;
dev_ice40.spi_input_delay_ns = 10;
dev_ice40.spi_max_transfer_size = SPI_MAX_TRANSFER_SIZE;
dev_ice40.get_done = ice40_get_done_wrapper;
dev_ice40.set_reset = ice40_set_reset_wrapper;
esp_err_t res = ice40_init(&dev_ice40);
if (res != ESP_OK) {
ESP_LOGE(TAG, "Initializing ICE40 failed");
return res;
}
bool done;
res = ice40_get_done(&dev_ice40, &done);
if (res != ESP_OK) {
ESP_LOGE(TAG, "Failed to read ICE40 done state");
return res;
}
if (done) {
ESP_LOGE(TAG, "ICE40 indicates done in disabled state");
return ESP_FAIL;
}
ice40_ready = true;
return ESP_OK;
}
esp_err_t bsp_bno055_init() {
if (!bsp_ready) return ESP_FAIL;
if (bno055_ready) return ESP_OK;
esp_err_t res = bno055_init(&dev_bno055, I2C_BUS, BNO055_ADDR, GPIO_INT_BNO055, true);
if (res != ESP_OK) {
ESP_LOGE(TAG, "Initializing BNO055 failed");
return res;
}
res = bno055_set_power_mode(&dev_bno055, BNO055_POWER_MODE_SUSPEND);
if (res != ESP_OK) {
ESP_LOGE(TAG, "Failed to switch BNO055 power mode to suspended state");
return res;
}
bno055_ready = true;
return ESP_OK;
}
esp_err_t bsp_bme680_init() {
if (!bsp_ready) return ESP_FAIL;
if (bme680_ready) return ESP_OK;
dev_bme680.i2c_bus = I2C_BUS;
dev_bme680.i2c_address = BME680_ADDR;
esp_err_t res = bme680_init(&dev_bme680);
if (res != ESP_OK) {
ESP_LOGE(TAG, "Initializing BME680 failed");
return res;
}
bme680_ready = true;
return ESP_OK;
}
ILI9341* get_ili9341() {
if (!bsp_ready) return NULL;
return &dev_ili9341;
}
esp_err_t display_flush() {
if (!bsp_ready) return ESP_FAIL;
if (!pax_is_dirty(&pax_buffer)) return ESP_OK;
//ESP_LOGI(TAG, "Flush %u to %u\n", pax_buffer.dirty_y0, pax_buffer.dirty_y1);
uint8_t* buffer = (uint8_t*)(pax_buffer.buf);
esp_err_t res = ili9341_write_partial_direct(&dev_ili9341, &buffer[pax_buffer.dirty_y0 * ILI9341_WIDTH * 2], 0, pax_buffer.dirty_y0, ILI9341_WIDTH, pax_buffer.dirty_y1 - pax_buffer.dirty_y0 + 1);
if (res != ESP_OK) return res;
pax_mark_clean(&pax_buffer);
return res;
}
pax_buf_t* get_pax_buffer() {
if (!bsp_ready) return NULL;
return &pax_buffer;
}
RP2040* get_rp2040() {
if (!rp2040_ready) return NULL;
return &dev_rp2040;
}
ICE40* get_ice40() {
if (!ice40_ready) return NULL;
return &dev_ice40;
}
BNO055* get_bno055() {
if (!bno055_ready) return NULL;
return &dev_bno055;
}
BME680* get_bme680() {
if (!bme680_ready) return NULL;
return &dev_bme680;
}