/* * Project: Arrow * Author: curiousmuch */ #include #include #include "freertos/FreeRTOS.h" #include "freertos/task.h" #include "driver/gpio.h" #include "sdkconfig.h" #include "driver/spi_master.h" #include "esp_err.h" #include "cc1200.h" #include "cc1200_protocol.h" #include "board.h" #include "esp_task_wdt.h" #include "freertos/ringbuf.h" #define CC1200_WRITE_BIT 0 #define CC1200_READ_BIT BIT(1) #define CC1200_BURST_BIT BIT(0) // Public Configurations for CC1120 SPI Driver spi_bus_config_t bus_config = { .miso_io_num = CC1120_MISO, .mosi_io_num = CC1120_MOSI, .sclk_io_num = CC1120_SCLK, .quadwp_io_num = -1, .quadhd_io_num = -1, .max_transfer_sz = 150, .flags = ESP_INTR_FLAG_IRAM }; spi_device_interface_config_t interface_config = { .command_bits = 2, .address_bits = 6, .dummy_bits = 0, .mode = 0, .spics_io_num = CC1120_CS, .clock_speed_hz = (APB_CLK_FREQ/20), .flags = 0, .queue_size = 20 }; spi_device_handle_t spi; // Private CC1120 Driver Functions void cc1200_gpio_init(void) { gpio_config_t reset_pin_config = { .pin_bit_mask = (uint64_t)(BIT64(CC1120_RESET)), .mode = GPIO_MODE_OUTPUT, .pull_up_en = GPIO_PULLUP_DISABLE, .pull_down_en = GPIO_PULLDOWN_DISABLE, .intr_type = GPIO_INTR_DISABLE }; gpio_config_t gpio_pin_config = { .pin_bit_mask = (uint64_t) (BIT64(CC1120_GPIO0)|BIT64(CC1120_GPIO2)|BIT64(CC1120_GPIO3)), .mode = GPIO_MODE_INPUT, .pull_up_en = GPIO_PULLUP_DISABLE, .pull_down_en = GPIO_PULLDOWN_DISABLE, .intr_type = GPIO_INTR_DISABLE }; gpio_config_t debug_pin_config = { .pin_bit_mask = (uint64_t) (BIT64(DEBUG_0)|BIT64(DEBUG_1)), .mode = GPIO_MODE_OUTPUT, .pull_up_en = GPIO_PULLUP_DISABLE, .pull_down_en = GPIO_PULLDOWN_DISABLE, .intr_type = GPIO_INTR_DISABLE }; gpio_config(&reset_pin_config); gpio_config(&gpio_pin_config); gpio_config(&debug_pin_config); gpio_set_level(CC1120_RESET, 1); } void cc1200_spi_init(void) { esp_err_t ret; ret = spi_bus_initialize(VSPI_HOST, &bus_config, 0); // this uses DMA channel 1 ESP_ERROR_CHECK(ret); ret = spi_bus_add_device(VSPI_HOST, &interface_config, &spi); ESP_ERROR_CHECK(ret); } void IRAM_ATTR cc1200_spi_write_byte(uint16_t addr, uint8_t data) { esp_err_t ret; spi_transaction_t tx_trans = { .flags = SPI_TRANS_USE_TXDATA, .cmd = CC1200_WRITE_BIT, .addr = addr, .length = 8, .rxlength = 0, .tx_data[0] = data }; if ((addr & 0xFF00) != 0) // send data with extended address in command field { tx_trans.flags |= (SPI_TRANS_VARIABLE_CMD | SPI_TRANS_VARIABLE_ADDR); spi_transaction_ext_t tx_trans_ext = { .base = tx_trans, .command_bits = 2, .address_bits = 14 }; ret = spi_device_polling_transmit(spi, (spi_transaction_t*)&tx_trans_ext); } else { ret = spi_device_polling_transmit(spi, &tx_trans); } ESP_ERROR_CHECK(ret); } void IRAM_ATTR cc1200_spi_write_bytes(uint16_t addr, uint8_t* data, uint8_t len) { esp_err_t ret; spi_transaction_t tx_trans = { .cmd = (CC1200_WRITE_BIT | CC1200_BURST_BIT), .addr = addr, .length = 8*len, .tx_buffer = data }; if ((addr & 0xFF00) != 0) // send data with extended address in command field { tx_trans.flags |= (SPI_TRANS_VARIABLE_CMD | SPI_TRANS_VARIABLE_ADDR); spi_transaction_ext_t tx_trans_ext = { .base = tx_trans, .command_bits = 2, .address_bits = 14 }; ret = spi_device_polling_transmit(spi, (spi_transaction_t*)&tx_trans_ext); } else { ret = spi_device_polling_transmit(spi, &tx_trans); } ESP_ERROR_CHECK(ret); } void IRAM_ATTR cc1200_spi_read_byte(uint16_t addr, uint8_t* data) { esp_err_t ret; spi_transaction_t rx_trans = { .cmd = CC1200_READ_BIT, .addr = addr, .length = 8, .rxlength = 8, .rx_buffer = data }; if ((addr & 0xFF00) != 0) // read data with extended address in command field { rx_trans.flags |= (SPI_TRANS_VARIABLE_CMD | SPI_TRANS_VARIABLE_ADDR); spi_transaction_ext_t rx_trans_ext = { .base = rx_trans, .command_bits = 2, .address_bits = 14 }; ret = spi_device_polling_transmit(spi, (spi_transaction_t*)&rx_trans_ext); } else { ret = spi_device_polling_transmit(spi, &rx_trans); } ESP_ERROR_CHECK(ret); } void cc1200_spi_read_bytes(uint16_t addr, uint8_t* data, uint8_t len) { esp_err_t ret; spi_transaction_t rx_trans = { .cmd = (CC1200_READ_BIT | CC1200_BURST_BIT), .addr = addr, .length = 8*len, .rxlength = 8*len, .rx_buffer = data }; if ((addr & 0xFF00) != 0) // read data with extended address in command field { rx_trans.flags |= (SPI_TRANS_VARIABLE_CMD | SPI_TRANS_VARIABLE_ADDR); spi_transaction_ext_t rx_trans_ext = { .base = rx_trans, .command_bits = 2, .address_bits = 14 }; ret = spi_device_polling_transmit(spi, (spi_transaction_t*)&rx_trans_ext); } else { ret = spi_device_polling_transmit(spi, &rx_trans); } ESP_ERROR_CHECK(ret); } rf_status_t IRAM_ATTR cc1200_spi_strobe(uint8_t cmd) { esp_err_t ret; uint8_t temp=0; spi_transaction_t rx_trans = { .flags = (SPI_TRANS_USE_TXDATA | SPI_TRANS_VARIABLE_CMD | SPI_TRANS_VARIABLE_ADDR), .length = 8, .rxlength = 8, .rx_buffer = &temp, .tx_data[0] = cmd }; spi_transaction_ext_t rx_trans_ext = { .base = rx_trans, .command_bits = 0, .address_bits = 0 }; ret = spi_device_transmit(spi, (spi_transaction_t*)&rx_trans_ext); ESP_ERROR_CHECK(ret); return (temp & 0xF0); } // Public CC1120 Driver Functions // These function should have there own error codes as they're dependent upon the radio and // not the ESP32 :) rf_status_t cc1200_radio_reset(void) { rf_status_t status; uint8_t retry_count = 0; cc1200_spi_strobe(CC120X_SRES); status = cc1200_spi_strobe(CC120X_SNOP); vTaskDelay(10 / portTICK_PERIOD_MS); while((CC120X_RDYn_BIT & (status & 0x80))) { vTaskDelay(10 / portTICK_PERIOD_MS); if (retry_count > 3) { // place error CC1120 timeout printf("CC1120 Reset Failure\n"); break; } status = cc1200_spi_strobe(CC120X_SNOP); retry_count++; } printf("%x\n", retry_count); return status; } // f_RF = f_VCO / LO Divider #define CC1200_LO_DIVIDER 24 // 136.7 - 160 MHz Band #define CC1200_XOSC 40000000 // 40MHz // f_VCO = FREQ / 2^16 * f_XOSX + FREQOFF / 2^18 * F_XOSC esp_err_t cc1200_radio_frequency(uint32_t freq) { // calculate FREQ0, FREQ, FREQ2 registers volatile double temp_freq; temp_freq = ((double) freq * 65536 * CC1200_LO_DIVIDER) / CC1200_XOSC; freq = (uint32_t)temp_freq; cc1200_spi_write_byte(CC120X_FREQ0, ((uint8_t *)&freq)[0]); cc1200_spi_write_byte(CC120X_FREQ1, ((uint8_t *)&freq)[1]); cc1200_spi_write_byte(CC120X_FREQ2, ((uint8_t *)&freq)[2]); return ESP_OK; } esp_err_t cc1200_radio_sleep(void) { return ESP_OK; } esp_err_t cc1200_radio_power(uint8_t txPower) { return ESP_OK; } #define HDLC_FLAG 0x7E #define HDLC_FLAG_LEN 10 uint8_t packet_len = 0; uint8_t test_vector[] = {0x71, 0x01, 023, 0xAE, 0x75}; volatile uint8_t sample_count = 0; uint8_t toggle; uint8_t toggle2; uint8_t prev_sample_count = 0; uint32_t tx_symbol = 0; uint8_t prev_tx_symbol = 0; #define SAMPLE_FREQUENCY 13200 #define DAC_MAX 64 #define LUT_SIZE 128 DRAM_ATTR int8_t LUT[LUT_SIZE]; int32_t phase_i = 0; volatile uint8_t new_sample = 0; float phase = 0.0f; float delta_phi = 0.0f; float const delta_phi_1 = (float) 1200 / SAMPLE_FREQUENCY * LUT_SIZE; float const delta_phi_2 = (float) 2200 / SAMPLE_FREQUENCY * LUT_SIZE; uint8_t data; //const uint8_t APRS_TEST_PACKET[] = {168,138,166,168, 64, 64,224,174,132,100,158,166,180,255, 3,240, 44, 84, // 104,101, 32,113,117,105, 99,107, 32, 98,114,111,119,110, 32,102,111,120, // 32,106,117,109,112,115, 32,111,118,101,114, 32,116,104,101, 32,108, 97, // 122,121, 32,100,111,103, 33, 32, 32, 49, 32,111,102, 32, 52, 40,110}; const uint8_t APRS_TEST_PACKET[] = { 0x82, 0x98, 0x98, 0x40, 0x40, 0x40, 0xe0, 0x96, 0x84, 0x66, 0xaa, 0x96, 0xac, 0xe0, 0xae, 0x92, 0x88, 0x8a, 0x62, 0x40, 0x62, 0xae, 0x92, 0x88, 0x8a, 0x64, 0x40, 0x65, 0x03, 0xf0, 0x3a, 0x4b, 0x42, 0x33, 0x55, 0x4b, 0x56, 0x2d, 0x32, 0x20, 0x3a, 0x48, 0x69, 0x21, 0x20, 0x54, 0x68, 0x69, 0x73, 0x20, 0x69, 0x73, 0x20, 0x61, 0x20, 0x54, 0x65, 0x73, 0x74, 0x7b, 0x31, 0xad, 0xa1 }; //#define APRS_TEST_PACKET fbuf extern unsigned char *fbuf; extern uint32_t flen; //const uint8_t APRS_TEST_PACKET[] = {0xFF, 0xFF, 0xFF}; // The output needs to be continous phase. typedef struct { uint8_t one_count; uint32_t sample_count; uint32_t byte; uint32_t packet_len; uint8_t prev_bit; uint8_t cur_bit; uint8_t tone; } aprs_flags_t; aprs_flags_t DRAM_ATTR aprs_flags = { .one_count = 0, .sample_count = 0, .byte = 0, .packet_len = sizeof(APRS_TEST_PACKET)/sizeof(uint8_t), .prev_bit = 0, .cur_bit = 0, .tone = 0 }; static void IRAM_ATTR LUT_lookup(void) { if (aprs_flags.tone) delta_phi = delta_phi_1; else delta_phi = delta_phi_2; phase_i = (int32_t)phase; // get integer part of our phase phase += delta_phi; // increment phase if (phase >= (float)LUT_SIZE) // handle wraparound phase -= (float)LUT_SIZE; } static void IRAM_ATTR cc1200_aprs_tx_isr(void* arg) { cc1200_spi_write_byte(CC120X_CFM_TX_DATA_IN, LUT[phase_i]); sample_count++; new_sample = 1; toggle = toggle ^ 1; gpio_set_level(DEBUG_1, toggle); } void cc1200_lut_init(void) { int16_t i=0; for (i=0; i> 1); while(sample_count < 11) // wait for symbol to be sent { if ( new_sample ) { LUT_lookup(); new_sample = 0; } } sample_count = 0; //printf("Symbol: %x\n", aprs_flags.cur_bit); } } aprs_flags.one_count = 0; /* Send Packet */ for (i=0;i> 1); while(sample_count < 11) // wait for symbol to be sent { if ( new_sample ) { LUT_lookup(); new_sample = 0; } } toggle2 = toggle2 ^ 1; gpio_set_level(DEBUG_0, toggle2); sample_count = 0; //printf("Symbol: %x\n", aprs_flags.cur_bit); } } aprs_flags.one_count = 0; /* Send Flag */ for (i = 0; i> 1); while(sample_count < 11) // wait for symbol to be sent { if ( new_sample ) { LUT_lookup(); new_sample = 0; } } sample_count = 0; } } cc1200_spi_strobe(CC120X_SIDLE); vTaskDelay(10000/portTICK_PERIOD_MS); } } extern RingbufHandle_t cfm_buf_handle; static void IRAM_ATTR cc1200_aprs_rx_isr(void* arg) { //uint8_t data = 0; cc1200_spi_read_byte(CC120X_CFM_RX_DATA_OUT, &data); toggle = toggle ^ 1; gpio_set_level(DEBUG_1, toggle); xRingbufferSendFromISR(cfm_buf_handle, &data, sizeof(data), NULL); //ets_write_char_uart(data); //new_sample = 1; } void IRAM_ATTR cc1200_radio_APRSRXPacket(void) { printf("Starting Continuous RX\n"); // start RX transmission //cc1120_spi_write_byte(CC112X_FIFO, 0x12); cc1200_spi_strobe(CC120X_SRX); vTaskDelay(20/portTICK_PERIOD_MS); cc1200_spi_strobe(CC120X_SRX); // acquire SPI bus for fastest possible SPI transactions spi_device_acquire_bus(spi, portMAX_DELAY); // enable interrupt pin for CC1120 for timing packets gpio_install_isr_service(ESP_INTR_FLAG_IRAM); gpio_isr_handler_add(CC1120_GPIO3, cc1200_aprs_rx_isr, NULL); gpio_set_intr_type(CC1120_GPIO3, GPIO_INTR_POSEDGE); } void cc1200_radio_init(const cc1200_reg_settings_t* rf_settings, uint8_t len) { cc1200_gpio_init(); cc1200_spi_init(); cc1200_lut_init(); cc1200_radio_reset(); gpio_set_level(CC1120_RESET, 1); uint8_t data; cc1200_spi_read_byte(CC120X_PARTNUMBER, &data); printf("%x", data); uint8_t i; for (i=0;i