pimoroni-pico/drivers/esp32spi/spi_drv.cpp

314 lines
7.5 KiB
C++

#include "spi_drv.hpp"
namespace pimoroni {
void SpiDrv::init() {
spi_init(spi, 8000000);
gpio_set_function(miso, GPIO_FUNC_SPI);
gpio_set_function(sck, GPIO_FUNC_SPI);
gpio_set_function(mosi, GPIO_FUNC_SPI);
// Chip select is active-low, so we'll initialise it to a driven-high state
gpio_init(cs);
gpio_set_dir(cs, GPIO_OUT);
gpio_put(cs, true);
gpio_init(gpio0);
gpio_set_dir(gpio0, GPIO_OUT);
gpio_init(resetn);
gpio_set_dir(resetn, GPIO_OUT);
}
void SpiDrv::reset() {
gpio_put(gpio0, true);
gpio_put(cs, true);
gpio_put(resetn, false);
sleep_ms(10);
gpio_put(resetn, true);
sleep_ms(750);
}
bool SpiDrv::available() {
return gpio_get(gpio0);
}
void SpiDrv::esp_select() {
gpio_put(cs, false);
}
void SpiDrv::esp_deselect() {
gpio_put(cs, true);
}
bool SpiDrv::get_esp_ready() {
return !gpio_get(ack);
}
bool SpiDrv::get_esp_ack() {
return gpio_get(ack);
}
void SpiDrv::wait_for_esp_ack() {
while(!get_esp_ack()) {
tight_loop_contents();
}
}
void SpiDrv::wait_for_esp_ready() {
while(!get_esp_ready()) {
tight_loop_contents();
}
}
void SpiDrv::wait_for_esp_select() {
wait_for_esp_ready();
esp_select();
wait_for_esp_ack();
}
int SpiDrv::wait_for_byte(uint8_t wait_byte) {
int timeout = BYTE_TIMEOUT;
uint8_t byte_read = 0;
do{
byte_read = read_byte(); //get data byte
if (byte_read == ERR_CMD) {
printf("Err cmd received\n");
return -1;
}
} while((timeout-- > 0) && (byte_read != wait_byte));
return (byte_read == wait_byte);
}
bool SpiDrv::read_and_check_byte(uint8_t check_byte, uint8_t *byte_out) {
get_param(byte_out);
return (*byte_out == check_byte);
}
uint8_t SpiDrv::read_byte() {
uint8_t byte_read = 0;
get_param(&byte_read);
return byte_read;
}
bool SpiDrv::wait_response_params(uint8_t cmd, uint8_t num_param, tParam *params_out) {
uint8_t data = 0;
int i = 0;
IF_CHECK_START_CMD() {
CHECK_DATA(cmd | REPLY_FLAG, data){};
uint8_t num_param_read = read_byte();
if(num_param_read != 0) {
for(i = 0; i < num_param_read; ++i) {
params_out[i].param_len = read_param_len8();
spi_read_blocking(spi, DUMMY_DATA, params_out[i].param, params_out[i].param_len);
}
}
else {
printf("Error num_param == 0\n");
return false;
}
if(num_param != num_param_read) {
printf("Mismatch num_param\n");
return false;
}
read_and_check_byte(END_CMD, &data);
}
return true;
}
bool SpiDrv::wait_response_cmd(uint8_t cmd, uint8_t num_param, uint8_t *param_out, uint8_t *param_len_out) {
uint8_t data = 0;
int ii = 0;
IF_CHECK_START_CMD() {
CHECK_DATA(cmd | REPLY_FLAG, data){};
CHECK_DATA(num_param, data) {
read_param_len8(param_len_out);
for(ii = 0; ii < (*param_len_out); ++ii) {
get_param(&param_out[ii]);
}
}
read_and_check_byte(END_CMD, &data);
}
return true;
}
bool SpiDrv::wait_response_data8(uint8_t cmd, uint8_t *param_out, uint8_t *param_len_out) {
uint8_t data = 0;
IF_CHECK_START_CMD() {
CHECK_DATA(cmd | REPLY_FLAG, data){};
uint8_t num_param_read = read_byte();
if(num_param_read != 0) {
read_param_len8(param_len_out);
spi_read_blocking(spi, DUMMY_DATA, param_out, *param_len_out);
}
read_and_check_byte(END_CMD, &data);
}
return true;
}
bool SpiDrv::wait_response_data16(uint8_t cmd, uint8_t* param_out, uint16_t *param_len_out) {
uint8_t data = 0;
IF_CHECK_START_CMD() {
CHECK_DATA(cmd | REPLY_FLAG, data){};
uint8_t num_param_read = read_byte();
if(num_param_read != 0) {
read_param_len16(param_len_out);
spi_read_blocking(spi, DUMMY_DATA, param_out, *param_len_out);
}
read_and_check_byte(END_CMD, &data);
}
return false;
}
bool SpiDrv::wait_response(uint8_t cmd, uint8_t *num_param_out, uint8_t **params_out, uint8_t max_num_params) {
uint8_t data = 0;
int i = 0;
uint8_t* index[WL_SSID_MAX_LENGTH];
for(i = 0 ; i < WL_NETWORKS_LIST_MAXNUM; i++)
index[i] = (uint8_t*)params_out + (WL_SSID_MAX_LENGTH * i);
IF_CHECK_START_CMD() {
CHECK_DATA(cmd | REPLY_FLAG, data){};
uint8_t num_param_read = read_byte();
if(num_param_read > max_num_params) {
num_param_read = max_num_params;
}
*num_param_out = num_param_read;
if(num_param_read != 0) {
for(i = 0; i < num_param_read; ++i) {
uint8_t param_len = read_param_len8();
spi_read_blocking(spi, DUMMY_DATA, index[i], param_len);
index[i][param_len] = 0;
}
}
else {
printf("Error numParams == 0\n");
read_and_check_byte(END_CMD, &data);
return false;
}
read_and_check_byte(END_CMD, &data);
}
return true;
}
void SpiDrv::send_param(const uint8_t *param, uint8_t param_len, lastParam last_param) {
send_param_len8(param_len);
spi_write_blocking(spi, param, param_len);
if(last_param) {
uint8_t buf = END_CMD;
spi_write_blocking(spi, &buf, 1);
}
}
void SpiDrv::send_param_len8(uint8_t param_len) {
spi_write_blocking(spi, &param_len, 1);
}
void SpiDrv::send_param_len16(uint16_t param_len) {
uint8_t buf[2];
buf[0] = (uint8_t)((param_len & 0xff00) >> 8);
buf[1] = (uint8_t)(param_len & 0xff);
spi_write_blocking(spi, buf, 2);
}
uint8_t SpiDrv::read_param_len8(uint8_t *param_len_out) {
uint8_t param_len;
get_param(&param_len);
if(param_len_out != nullptr) {
*param_len_out = param_len;
}
return param_len;
}
uint16_t SpiDrv::read_param_len16(uint16_t *param_len_out) {
uint8_t buf[2];
spi_read_blocking(spi, DUMMY_DATA, buf, 2);
uint16_t param_len = (buf[0] << 8) | (buf[1] & 0xff);
if(param_len_out != nullptr) {
*param_len_out = param_len;
}
return param_len;
}
void SpiDrv::send_buffer(const uint8_t* param, uint16_t param_len, lastParam last_param) {
send_param_len16(param_len);
spi_write_blocking(spi, param, param_len);
if(last_param) {
uint8_t buf = END_CMD;
spi_write_blocking(spi, &buf, 1);
}
}
void SpiDrv::send_param(uint16_t param, lastParam last_param) {
send_param_len8(2);
uint8_t buf[2];
buf[0] = (uint8_t)((param & 0xff00) >> 8);
buf[1] = (uint8_t)(param & 0xff);
spi_write_blocking(spi, buf, 2);
if(last_param) {
uint8_t buf = END_CMD;
spi_write_blocking(spi, &buf, 1);
}
}
void SpiDrv::send_byte_param(uint8_t param, lastParam last_param) {
send_param_len8(1);
spi_write_blocking(spi, &param, 1);
if(last_param) {
uint8_t buf = END_CMD;
spi_write_blocking(spi, &buf, 1);
}
}
void SpiDrv::send_cmd(uint8_t cmd, uint8_t num_param) {
uint8_t buf[3];
buf[0] = START_CMD;
buf[1] = cmd & ~(REPLY_FLAG);
buf[2] = num_param;
spi_write_blocking(spi, buf, 3);
if(num_param == 0) {
uint8_t buf = END_CMD;
spi_write_blocking(spi, &buf, 1);
}
}
void SpiDrv::pad_to_multiple_of_4(int command_size) {
while(command_size % 4) {
read_byte();
command_size++;
}
}
void SpiDrv::get_param(uint8_t* param_out) {
spi_read_blocking(spi, DUMMY_DATA, param_out, 1);
}
}