/* * Copyright (C) 2009-2012 by Matthias Ringwald * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * 3. Neither the name of the copyright holders nor the names of * contributors may be used to endorse or promote products derived * from this software without specific prior written permission. * 4. Any redistribution, use, or modification is done solely for * personal benefit and not for any commercial purpose or for * monetary gain. * * THIS SOFTWARE IS PROVIDED BY MATTHIAS RINGWALD AND CONTRIBUTORS * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL MATTHIAS * RINGWALD OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF * THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF * SUCH DAMAGE. * * Please inquire about commercial licensing options at btstack@ringwald.ch * */ /* * bt_control_cc256x.c * * Adapter to use cc256x-based chipsets with BTstack * * Handles init script (a.k.a. Service Patch) * Allows for non-standard UART baud rate * Allows to configure transmit power * Allows to activate eHCILL deep sleep mode * * Issues with mspgcc LTS: * - 20 bit support is not there yet -> .text cannot get bigger than 48 kb * - arrays cannot have more than 32k entries * * workarounds: * - store init script in .fartext and use assembly code to read from there * - split into two arrays * * Issues with AVR * - Harvard architecture doesn't allow to store init script directly -> use avr-libc helpers * * Documentation for TI VS CC256x commands: http://processors.wiki.ti.com/index.php/CC256x_VS_HCI_Commands * */ #include "btstack-config.h" #include "bt_control_cc256x.h" #include /* NULL */ #include #include /* memcpy */ #if defined(__GNUC__) && defined(__MSP430X__) && (__MSP430X__ > 0) #include "hal_compat.h" #endif #ifdef __AVR__ #include #endif #include "bt_control.h" // actual init script provided by seperate .c file extern const uint8_t cc256x_init_script[]; extern const uint32_t cc256x_init_script_size; // static uint32_t init_script_offset = 0; static int16_t init_power_in_dB = 13; // 13 dBm static int init_ehcill_enabled = 0; static int bt_control_cc256x_on(void *config){ init_script_offset = 0; return 0; } // UART Baud Rate control from: http://e2e.ti.com/support/low_power_rf/f/660/p/134850/484763.aspx static int cc256x_baudrate_cmd(void * config, uint32_t baudrate, uint8_t *hci_cmd_buffer){ hci_cmd_buffer[0] = 0x36; hci_cmd_buffer[1] = 0xFF; hci_cmd_buffer[2] = 0x04; hci_cmd_buffer[3] = baudrate & 0xff; hci_cmd_buffer[4] = (baudrate >> 8) & 0xff; hci_cmd_buffer[5] = (baudrate >> 16) & 0xff; hci_cmd_buffer[6] = 0; return 0; } // Output Power control from: http://e2e.ti.com/support/low_power_rf/f/660/p/134853/484767.aspx #define NUM_POWER_LEVELS 16 #define DB_MIN_LEVEL -35 #define DB_PER_LEVEL 5 #define DB_DYNAMIC_RANGE 30 static int get_max_power_for_modulation_type(int type){ // a) limit max output power int power_db; switch (type){ case 0: // GFSK power_db = 12; break; default: // EDRx power_db = 10; break; } if (power_db > init_power_in_dB) { power_db = init_power_in_dB; } return power_db; } static int get_highest_level_for_given_power(int power_db, int recommended_db){ int i = NUM_POWER_LEVELS-1; while (i) { if (power_db < recommended_db) { return i; } power_db -= DB_PER_LEVEL; i--; } return 0; } static void update_set_power_vector(uint8_t *hci_cmd_buffer){ int i; int modulation_type = hci_cmd_buffer[3]; int power_db = get_max_power_for_modulation_type(modulation_type); int dynamic_range = 0; // f) don't touch level 0 for ( i = (NUM_POWER_LEVELS-1) ; i >= 1 ; i--){ #ifdef HAVE_BLE // level 1 is BLE transmit power for GFSK if (i == 1 && modulation_type == 0) { hci_cmd_buffer[4+1] = 2 * get_max_power_for_modulation_type(modulation_type); // as level 0 isn't set, we're done continue; } #endif hci_cmd_buffer[4+i] = 2 * power_db; if (dynamic_range + DB_PER_LEVEL > DB_DYNAMIC_RANGE) continue; // e) power_db -= DB_PER_LEVEL; // d) dynamic_range += DB_PER_LEVEL; if (power_db > DB_MIN_LEVEL) continue; power_db = DB_MIN_LEVEL; // b) } } static void update_set_class2_single_power(uint8_t * hci_cmd_buffer){ int i = 0; for (i=0;i<3;i++){ hci_cmd_buffer[3+i] = get_highest_level_for_given_power(get_max_power_for_modulation_type(i), 4); } } // eHCILL activate from http://e2e.ti.com/support/low_power_rf/f/660/p/134855/484776.aspx static void update_sleep_mode_configurations(uint8_t * hci_cmd_buffer){ if (init_ehcill_enabled) { hci_cmd_buffer[4] = 1; } else { hci_cmd_buffer[4] = 0; } } // @returns 1 if command was injected before this one static int bt_control_cc256x_update_command(uint8_t *hci_cmd_buffer){ uint16_t opcode = hci_cmd_buffer[0] | (hci_cmd_buffer[1] << 8); switch (opcode){ case 0xFD87: update_set_class2_single_power(hci_cmd_buffer); break; case 0xFD82: update_set_power_vector(hci_cmd_buffer); break; case 0xFD0C: update_sleep_mode_configurations(hci_cmd_buffer); break; default: break; } return 0; } static int bt_control_cc256x_next_cmd(void *config, uint8_t *hci_cmd_buffer){ if (init_script_offset >= cc256x_init_script_size) { return 0; } // store current position in case command needs to get expanded uint32_t current_offset = init_script_offset; // extracted init script has 0x01 cmd packet type, but BTstack expects them without init_script_offset++; #if defined(__GNUC__) && defined(__MSP430X__) && (__MSP430X__ > 0) // workaround: use FlashReadBlock with 32-bit integer and assume init script starts at 0x10000 uint32_t init_script_addr = 0x10000; FlashReadBlock(&hci_cmd_buffer[0], init_script_addr + init_script_offset, 3); // cmd header init_script_offset += 3; int payload_len = hci_cmd_buffer[2]; FlashReadBlock(&hci_cmd_buffer[3], init_script_addr + init_script_offset, payload_len); // cmd payload #elif defined (__AVR__) // workaround: use memcpy_P to access init script in lower 64 kB of flash memcpy_P(&hci_cmd_buffer[0], &cc256x_init_script[init_script_offset], 3); init_script_offset += 3; int payload_len = hci_cmd_buffer[2]; memcpy_P(&hci_cmd_buffer[3], &cc256x_init_script[init_script_offset], payload_len); #else // use memcpy with pointer uint8_t * init_script_ptr = (uint8_t*) &cc256x_init_script[0]; memcpy(&hci_cmd_buffer[0], init_script_ptr + init_script_offset, 3); // cmd header init_script_offset += 3; int payload_len = hci_cmd_buffer[2]; memcpy(&hci_cmd_buffer[3], init_script_ptr + init_script_offset, payload_len); // cmd payload #endif init_script_offset += payload_len; // support for cc256x power commands and ehcill int command_injected = bt_control_cc256x_update_command(hci_cmd_buffer); if (command_injected){ // stay at this command init_script_offset = current_offset; } return 1; } // MARK: const structs static const bt_control_t bt_control_cc256x = { bt_control_cc256x_on, // on NULL, // off NULL, // sleep NULL, // wake NULL, // valid NULL, // name cc256x_baudrate_cmd, // baudrate_cmd bt_control_cc256x_next_cmd, // next_cmd NULL, // register_for_power_notifications NULL, // hw_error }; static const hci_uart_config_t hci_uart_config_cc256x = { NULL, 115200, 1000000, 0 }; // MARK: public API void bt_control_cc256x_enable_ehcill(int on){ init_ehcill_enabled = on; } int bt_control_cc256x_ehcill_enabled(void){ return init_ehcill_enabled; } void bt_control_cc256x_set_power(int16_t power_in_dB){ init_power_in_dB = power_in_dB; } bt_control_t *bt_control_cc256x_instance(void){ return (bt_control_t*) &bt_control_cc256x; } hci_uart_config_t *hci_uart_config_cc256x_instance(void){ return (hci_uart_config_t*) &hci_uart_config_cc256x; }