2021-12-29 23:32:55 +00:00
|
|
|
#include "hardware.h"
|
|
|
|
#include <driver/spi_master.h>
|
|
|
|
#include <esp_log.h>
|
|
|
|
#include <driver/gpio.h>
|
|
|
|
#include "managed_i2c.h"
|
2022-01-04 23:58:13 +00:00
|
|
|
#include "sdcard.h"
|
2022-01-24 21:32:33 +00:00
|
|
|
#include "rp2040.h"
|
2021-12-29 23:32:55 +00:00
|
|
|
|
|
|
|
static const char *TAG = "hardware";
|
|
|
|
|
2022-01-11 23:53:38 +00:00
|
|
|
static PCA9555 dev_pca9555 = {0};
|
2022-01-24 21:32:33 +00:00
|
|
|
static BNO055 dev_bno055 = {0};
|
2022-01-11 23:53:38 +00:00
|
|
|
static ILI9341 dev_ili9341 = {0};
|
2022-01-24 21:32:33 +00:00
|
|
|
static ICE40 dev_ice40 = {0};
|
|
|
|
static RP2040 dev_rp2040 = {0};
|
2022-01-01 06:13:50 +00:00
|
|
|
|
2022-01-24 21:32:33 +00:00
|
|
|
esp_err_t ice40_get_done_wrapper(bool* done) {
|
|
|
|
return pca9555_get_gpio_value(&dev_pca9555, PCA9555_PIN_FPGA_CDONE, done);
|
|
|
|
}
|
|
|
|
|
|
|
|
esp_err_t ice40_set_reset_wrapper(bool reset) {
|
|
|
|
return pca9555_set_gpio_value(&dev_pca9555, PCA9555_PIN_FPGA_RESET, reset);
|
|
|
|
}
|
2021-12-29 23:32:55 +00:00
|
|
|
|
2022-01-24 21:32:33 +00:00
|
|
|
void ili9341_set_lcd_mode(bool mode) {
|
2022-01-24 22:41:03 +00:00
|
|
|
ESP_LOGI(TAG, "LCD mode switch to %s", mode ? "FPGA" : "ESP32");
|
2022-01-24 21:32:33 +00:00
|
|
|
rp2040_set_lcd_mode(&dev_rp2040, (lcd_mode_t) mode);
|
2022-01-24 22:41:03 +00:00
|
|
|
vTaskDelay(100 / portTICK_PERIOD_MS);
|
2022-01-24 21:32:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static esp_err_t _bus_init() {
|
2021-12-29 23:32:55 +00:00
|
|
|
esp_err_t res;
|
2022-01-24 21:32:33 +00:00
|
|
|
|
2022-01-01 06:13:50 +00:00
|
|
|
// System I2C bus
|
|
|
|
res = i2c_init(I2C_BUS_SYS, GPIO_I2C_SYS_SDA, GPIO_I2C_SYS_SCL, I2C_SPEED_SYS, false, false);
|
|
|
|
if (res != ESP_OK) {
|
|
|
|
ESP_LOGE(TAG, "Initializing system I2C bus failed");
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
// User I2C bus
|
|
|
|
res = i2c_init(I2C_BUS_EXT, GPIO_I2C_EXT_SDA, GPIO_I2C_EXT_SCL, I2C_SPEED_EXT, false, false);
|
|
|
|
if (res != ESP_OK) {
|
|
|
|
ESP_LOGE(TAG, "Initializing user I2C bus failed");
|
|
|
|
return res;
|
|
|
|
}
|
2021-12-29 23:32:55 +00:00
|
|
|
|
|
|
|
// 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;
|
|
|
|
}
|
2022-01-24 21:32:33 +00:00
|
|
|
|
|
|
|
return ESP_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Board init
|
|
|
|
|
|
|
|
esp_err_t board_init() {
|
|
|
|
esp_err_t res;
|
|
|
|
|
|
|
|
// Interrupts
|
|
|
|
res = gpio_install_isr_service(0);
|
|
|
|
if (res != ESP_OK) {
|
|
|
|
ESP_LOGE(TAG, "Initializing ISR service failed");
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Communication busses
|
|
|
|
res = _bus_init();
|
|
|
|
if (res != ESP_OK) return res;
|
2022-01-01 04:51:12 +00:00
|
|
|
|
2022-01-24 21:32:33 +00:00
|
|
|
// RP2040 co-processor
|
|
|
|
dev_rp2040.i2c_bus = I2C_BUS_SYS;
|
|
|
|
dev_rp2040.i2c_address = RP2040_ADDR;
|
|
|
|
dev_rp2040.pin_interrupt = GPIO_INT_RP2040;
|
|
|
|
|
|
|
|
res = rp2040_init(&dev_rp2040);
|
|
|
|
if (res != ESP_OK) {
|
|
|
|
ESP_LOGE(TAG, "Initializing RP2040 failed");
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
2022-01-01 06:13:50 +00:00
|
|
|
// PCA9555 IO expander on system I2C bus
|
|
|
|
res = pca9555_init(&dev_pca9555, I2C_BUS_SYS, PCA9555_ADDR, GPIO_INT_PCA9555);
|
|
|
|
if (res != ESP_OK) {
|
|
|
|
ESP_LOGE(TAG, "Initializing PCA9555 failed");
|
|
|
|
return res;
|
|
|
|
}
|
2022-01-24 21:32:33 +00:00
|
|
|
|
2022-01-01 06:13:50 +00:00
|
|
|
res = pca9555_set_gpio_direction(&dev_pca9555, PCA9555_PIN_FPGA_RESET, true);
|
|
|
|
if (res != ESP_OK) {
|
|
|
|
ESP_LOGE(TAG, "Setting the FPGA reset pin on the PCA9555 to output failed");
|
|
|
|
return res;
|
|
|
|
}
|
2022-01-24 21:32:33 +00:00
|
|
|
|
2022-01-01 06:13:50 +00:00
|
|
|
res = pca9555_set_gpio_value(&dev_pca9555, PCA9555_PIN_FPGA_RESET, false);
|
|
|
|
if (res != ESP_OK) {
|
|
|
|
ESP_LOGE(TAG, "Setting the FPGA reset pin on the PCA9555 to low failed");
|
|
|
|
return res;
|
|
|
|
}
|
2022-01-24 21:32:33 +00:00
|
|
|
|
2022-01-11 23:53:38 +00:00
|
|
|
pca9555_set_gpio_polarity(&dev_pca9555, PCA9555_PIN_BTN_START, true);
|
|
|
|
pca9555_set_gpio_polarity(&dev_pca9555, PCA9555_PIN_BTN_SELECT, true);
|
|
|
|
pca9555_set_gpio_polarity(&dev_pca9555, PCA9555_PIN_BTN_MENU, true);
|
|
|
|
pca9555_set_gpio_polarity(&dev_pca9555, PCA9555_PIN_BTN_HOME, true);
|
|
|
|
pca9555_set_gpio_polarity(&dev_pca9555, PCA9555_PIN_BTN_JOY_LEFT, true);
|
|
|
|
pca9555_set_gpio_polarity(&dev_pca9555, PCA9555_PIN_BTN_JOY_PRESS, true);
|
|
|
|
pca9555_set_gpio_polarity(&dev_pca9555, PCA9555_PIN_BTN_JOY_DOWN, true);
|
|
|
|
pca9555_set_gpio_polarity(&dev_pca9555, PCA9555_PIN_BTN_JOY_UP, true);
|
|
|
|
pca9555_set_gpio_polarity(&dev_pca9555, PCA9555_PIN_BTN_JOY_RIGHT, true);
|
|
|
|
pca9555_set_gpio_polarity(&dev_pca9555, PCA9555_PIN_BTN_BACK, true);
|
|
|
|
pca9555_set_gpio_polarity(&dev_pca9555, PCA9555_PIN_BTN_ACCEPT, true);
|
2022-01-24 21:32:33 +00:00
|
|
|
dev_pca9555.pin_state = 0;
|
|
|
|
|
2022-01-01 06:13:50 +00:00
|
|
|
// FPGA
|
|
|
|
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;
|
2022-01-04 23:58:13 +00:00
|
|
|
dev_ice40.spi_speed = 23000000; // 23MHz
|
2022-01-01 06:13:50 +00:00
|
|
|
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;
|
|
|
|
|
|
|
|
res = ice40_init(&dev_ice40);
|
|
|
|
if (res != ESP_OK) {
|
|
|
|
ESP_LOGE(TAG, "Initializing FPGA failed");
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
2022-01-01 04:51:12 +00:00
|
|
|
// 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 = -1;
|
|
|
|
dev_ili9341.rotation = 1;
|
|
|
|
dev_ili9341.color_mode = true; // Blue and red channels are swapped
|
|
|
|
dev_ili9341.spi_speed = 60000000; // 60MHz
|
|
|
|
dev_ili9341.spi_max_transfer_size = SPI_MAX_TRANSFER_SIZE;
|
2022-01-24 21:32:33 +00:00
|
|
|
dev_ili9341.callback = ili9341_set_lcd_mode; // Callback for changing LCD mode between ESP32 and FPGA
|
2022-01-23 00:27:22 +00:00
|
|
|
|
2022-01-01 04:51:12 +00:00
|
|
|
res = ili9341_init(&dev_ili9341);
|
|
|
|
if (res != ESP_OK) {
|
|
|
|
ESP_LOGE(TAG, "Initializing LCD failed");
|
|
|
|
return res;
|
|
|
|
}
|
2022-01-24 21:32:33 +00:00
|
|
|
|
2022-01-01 02:06:55 +00:00
|
|
|
// BNO055 sensor on system I2C bus
|
2022-01-24 21:32:33 +00:00
|
|
|
|
2022-01-01 04:51:12 +00:00
|
|
|
res = bno055_init(&dev_bno055, I2C_BUS_SYS, BNO055_ADDR, GPIO_INT_BNO055, true);
|
2022-01-01 02:06:55 +00:00
|
|
|
if (res != ESP_OK) {
|
|
|
|
ESP_LOGE(TAG, "Initializing BNO055 failed");
|
|
|
|
return res;
|
|
|
|
}
|
2021-12-29 23:32:55 +00:00
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
PCA9555* get_pca9555() {
|
2022-01-01 04:51:12 +00:00
|
|
|
return &dev_pca9555;
|
2021-12-29 23:32:55 +00:00
|
|
|
}
|
2022-01-01 02:06:55 +00:00
|
|
|
|
|
|
|
BNO055* get_bno055() {
|
2022-01-01 04:51:12 +00:00
|
|
|
return &dev_bno055;
|
|
|
|
}
|
|
|
|
|
|
|
|
ILI9341* get_ili9341() {
|
|
|
|
return &dev_ili9341;
|
2022-01-01 02:06:55 +00:00
|
|
|
}
|
2022-01-01 06:13:50 +00:00
|
|
|
|
|
|
|
ICE40* get_ice40() {
|
|
|
|
return &dev_ice40;
|
|
|
|
}
|
2022-01-24 21:32:33 +00:00
|
|
|
|
|
|
|
RP2040* get_rp2040() {
|
|
|
|
return &dev_rp2040;
|
|
|
|
}
|