#include #include #include #include #include #include #include #include #include #include "driver/uart.h" #include "hardware.h" #include "managed_i2c.h" #include "pax_gfx.h" #include "ice40.h" #include "system_wrapper.h" #include "graphics_wrapper.h" #include "esp32/rom/crc.h" void fpga_install_uart() { fflush(stdout); ESP_ERROR_CHECK(uart_driver_install(0, 2048, 0, 0, NULL, 0)); uart_config_t uart_config = { .baud_rate = 921600, .data_bits = UART_DATA_8_BITS, .parity = UART_PARITY_DISABLE, .stop_bits = UART_STOP_BITS_1, .flow_ctrl = UART_HW_FLOWCTRL_DISABLE, .source_clk = UART_SCLK_APB, }; ESP_ERROR_CHECK(uart_param_config(0, &uart_config)); } void fpga_uninstall_uart() { uart_driver_delete(0); } bool fpga_read_stdin(uint8_t* buffer, uint32_t len, uint32_t timeout) { int read = uart_read_bytes(0, buffer, len, timeout / portTICK_PERIOD_MS); return (read == len); } bool fpga_uart_sync(uint32_t* length, uint32_t* crc) { uint8_t data[256]; uart_read_bytes(0, data, sizeof(data), 10 / portTICK_PERIOD_MS); char command[] = "FPGA"; uart_write_bytes(0, command, 4); uint8_t rx_buffer[4 * 3]; fpga_read_stdin(rx_buffer, sizeof(rx_buffer), 1000); if (memcmp(rx_buffer, "FPGA", 4) != 0) return false; memcpy((uint8_t*) length, &rx_buffer[4 * 1], 4); memcpy((uint8_t*) crc, &rx_buffer[4 * 2], 4); return true; } bool fpga_uart_load(uint8_t* buffer, uint32_t length) { return fpga_read_stdin(buffer, length, 3000); } void fpga_uart_mess(const char *mess) { uart_write_bytes(0, mess, strlen(mess)); } esp_err_t fpga_process_events(xQueueHandle buttonQueue, ICE40* ice40, uint16_t *key_state, uint16_t *idle_count) { rp2040_input_message_t buttonMessage = {0}; while (xQueueReceive(buttonQueue, &buttonMessage, 0) == pdTRUE) { uint8_t pin = buttonMessage.input; bool value = buttonMessage.state; uint16_t key_mask = 0; switch(pin) { case RP2040_INPUT_JOYSTICK_DOWN: key_mask = 1 << 0; break; case RP2040_INPUT_JOYSTICK_UP: key_mask = 1 << 1; break; case RP2040_INPUT_JOYSTICK_LEFT: key_mask = 1 << 2; break; case RP2040_INPUT_JOYSTICK_RIGHT: key_mask = 1 << 3; break; case RP2040_INPUT_JOYSTICK_PRESS: key_mask = 1 << 4; break; case RP2040_INPUT_BUTTON_HOME: key_mask = 1 << 5; break; case RP2040_INPUT_BUTTON_MENU: key_mask = 1 << 6; break; case RP2040_INPUT_BUTTON_SELECT: key_mask = 1 << 7; break; case RP2040_INPUT_BUTTON_START: key_mask = 1 << 8; break; case RP2040_INPUT_BUTTON_ACCEPT: key_mask = 1 << 9; break; case RP2040_INPUT_BUTTON_BACK: key_mask = 1 << 10; default: break; } if (key_mask != 0) { if (value) { *key_state |= key_mask; } else { *key_state &= ~key_mask; } uint8_t spi_message[5] = { 0xf4 }; spi_message[1] = *key_state >> 8; spi_message[2] = *key_state & 0xff; spi_message[3] = key_mask >> 8; spi_message[4] = key_mask & 0xff; esp_err_t res = ice40_send(ice40, spi_message, 5); if (res != ESP_OK) { return res; } } *idle_count = 0; } return ESP_OK; } void fpga_download(xQueueHandle buttonQueue, ICE40* ice40, pax_buf_t* pax_buffer, ILI9341* ili9341) { char message[64]; pax_noclip(pax_buffer); pax_background(pax_buffer, 0x325aa8); pax_draw_text(pax_buffer, 0xFFFFFFFF, NULL, 18, 0, 20*0, "FPGA download mode"); pax_draw_text(pax_buffer, 0xFFFFFFFF, NULL, 18, 0, 20*1, "Preparing..."); ili9341_write(ili9341, pax_buffer->buf); fpga_install_uart(); ice40_disable(ice40); ili9341_init(ili9341); uint8_t counter = 0; uint32_t length = 0; uint32_t crc = 0; while (!fpga_uart_sync(&length, &crc)) { pax_noclip(pax_buffer); pax_background(pax_buffer, 0x325aa8); pax_draw_text(pax_buffer, 0xFFFFFFFF, NULL, 18, 0, 20*0, "FPGA download mode"); snprintf(message, sizeof(message), "Waiting for bitstream%s%s%s", (counter > 0) ? "." : " ", (counter > 1) ? "." : " ", (counter > 2) ? "." : " "); pax_draw_text(pax_buffer, 0xFFFFFFFF, NULL, 18, 0, 20*1, message); ili9341_write(ili9341, pax_buffer->buf); counter++; if (counter > 3) counter = 0; } while (true) { pax_noclip(pax_buffer); pax_background(pax_buffer, 0x325aa8); pax_draw_text(pax_buffer, 0xFFFFFFFF, NULL, 18, 0, 20*0, "FPGA download mode"); pax_draw_text(pax_buffer, 0xFFFFFFFF, NULL, 18, 0, 20*1, "Receiving bitstream..."); ili9341_write(ili9341, pax_buffer->buf); uint8_t* buffer = malloc(length); if (buffer == NULL) { pax_noclip(pax_buffer); pax_background(pax_buffer, 0xa85a32); pax_draw_text(pax_buffer, 0xFFFFFFFF, NULL, 18, 0, 20*0, "FPGA download mode"); pax_draw_text(pax_buffer, 0xFFFFFFFF, NULL, 18, 0, 20*1, "Malloc failed"); ili9341_write(ili9341, pax_buffer->buf); vTaskDelay(1000 / portTICK_PERIOD_MS); fpga_uninstall_uart(); return; } if (!fpga_uart_load(buffer, length)) { free(buffer); pax_noclip(pax_buffer); pax_background(pax_buffer, 0xa85a32); pax_draw_text(pax_buffer, 0xFFFFFFFF, NULL, 18, 0, 20*0, "FPGA download mode"); pax_draw_text(pax_buffer, 0xFFFFFFFF, NULL, 18, 0, 20*1, "Timeout while loading"); ili9341_write(ili9341, pax_buffer->buf); vTaskDelay(1000 / portTICK_PERIOD_MS); fpga_uninstall_uart(); return; } uint32_t checkCrc = crc32_le(0, buffer, length); if (checkCrc != crc) { free(buffer); pax_noclip(pax_buffer); pax_background(pax_buffer, 0xa85a32); pax_draw_text(pax_buffer, 0xFFFFFFFF, NULL, 18, 0, 20*0, "FPGA download mode"); pax_draw_text(pax_buffer, 0xFFFFFFFF, NULL, 18, 0, 20*1, "CRC incorrect"); snprintf(message, sizeof(message), "Provided CRC: %08X", crc); pax_draw_text(pax_buffer, 0xFFFFFFFF, NULL, 18, 0, 20*2, message); snprintf(message, sizeof(message), "Calculated CRC: %08X", checkCrc); pax_draw_text(pax_buffer, 0xFFFFFFFF, NULL, 18, 0, 20*3, message); ili9341_write(ili9341, pax_buffer->buf); vTaskDelay(1000 / portTICK_PERIOD_MS); snprintf(message, sizeof(message), "CRC incorrect %08X %08x\n", crc, checkCrc); fpga_uart_mess(message); fpga_uninstall_uart(); return; } fpga_uart_mess("CRC correct\n"); ili9341_deinit(ili9341); ili9341_select(ili9341, false); vTaskDelay(200 / portTICK_PERIOD_MS); ili9341_select(ili9341, true); esp_err_t res = ice40_load_bitstream(ice40, buffer, length); free(buffer); if (res != ESP_OK) { ice40_disable(ice40); ili9341_init(ili9341); pax_noclip(pax_buffer); pax_background(pax_buffer, 0xa85a32); pax_draw_text(pax_buffer, 0xFFFFFFFF, NULL, 18, 0, 20*0, "FPGA download mode"); snprintf(message, sizeof(message), "Upload failed: %d", res); pax_draw_text(pax_buffer, 0xFFFFFFFF, NULL, 18, 0, 20*1, message); ili9341_write(ili9341, pax_buffer->buf); vTaskDelay(1000 / portTICK_PERIOD_MS); snprintf(message, sizeof(message), "uploading bitstream failed with %d\n", res); fpga_uart_mess(message); fpga_uninstall_uart(); return; } snprintf(message, sizeof(message), "bitstream has uploaded\n"); fpga_uart_mess(message); // Waiting for next download and sending key strokes to FPGA uint16_t key_state = 0; uint16_t idle_count = 0; while (true) { if (idle_count >= 200) { if (fpga_uart_sync(&length, &crc)) { break; } idle_count = 0; } esp_err_t res = fpga_process_events(buttonQueue, ice40, &key_state, &idle_count); if (res != ESP_OK) { ice40_disable(ice40); ili9341_init(ili9341); pax_noclip(pax_buffer); pax_background(pax_buffer, 0xa85a32); pax_draw_text(pax_buffer, 0xFFFFFFFF, NULL, 18, 0, 20*0, "FPGA download mode"); snprintf(message, sizeof(message), "Error: %d", res); pax_draw_text(pax_buffer, 0xFFFFFFFF, NULL, 18, 0, 20*1, message); ili9341_write(ili9341, pax_buffer->buf); vTaskDelay(1000 / portTICK_PERIOD_MS); snprintf(message, sizeof(message), "processing events failed with %d\n", res); fpga_uart_mess(message); fpga_uninstall_uart(); return; } vTaskDelay(10 / portTICK_PERIOD_MS); idle_count++; } ice40_disable(ice40); ili9341_init(ili9341); } }