1*faa6c1f6SMatthias Ringwald /* 2*faa6c1f6SMatthias Ringwald * Copyright (C) 2009-2012 by Matthias Ringwald 3*faa6c1f6SMatthias Ringwald * 4*faa6c1f6SMatthias Ringwald * Redistribution and use in source and binary forms, with or without 5*faa6c1f6SMatthias Ringwald * modification, are permitted provided that the following conditions 6*faa6c1f6SMatthias Ringwald * are met: 7*faa6c1f6SMatthias Ringwald * 8*faa6c1f6SMatthias Ringwald * 1. Redistributions of source code must retain the above copyright 9*faa6c1f6SMatthias Ringwald * notice, this list of conditions and the following disclaimer. 10*faa6c1f6SMatthias Ringwald * 2. Redistributions in binary form must reproduce the above copyright 11*faa6c1f6SMatthias Ringwald * notice, this list of conditions and the following disclaimer in the 12*faa6c1f6SMatthias Ringwald * documentation and/or other materials provided with the distribution. 13*faa6c1f6SMatthias Ringwald * 3. Neither the name of the copyright holders nor the names of 14*faa6c1f6SMatthias Ringwald * contributors may be used to endorse or promote products derived 15*faa6c1f6SMatthias Ringwald * from this software without specific prior written permission. 16*faa6c1f6SMatthias Ringwald * 4. Any redistribution, use, or modification is done solely for 17*faa6c1f6SMatthias Ringwald * personal benefit and not for any commercial purpose or for 18*faa6c1f6SMatthias Ringwald * monetary gain. 19*faa6c1f6SMatthias Ringwald * 20*faa6c1f6SMatthias Ringwald * THIS SOFTWARE IS PROVIDED BY MATTHIAS RINGWALD AND CONTRIBUTORS 21*faa6c1f6SMatthias Ringwald * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT 22*faa6c1f6SMatthias Ringwald * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS 23*faa6c1f6SMatthias Ringwald * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL MATTHIAS 24*faa6c1f6SMatthias Ringwald * RINGWALD OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, 25*faa6c1f6SMatthias Ringwald * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, 26*faa6c1f6SMatthias Ringwald * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS 27*faa6c1f6SMatthias Ringwald * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED 28*faa6c1f6SMatthias Ringwald * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, 29*faa6c1f6SMatthias Ringwald * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF 30*faa6c1f6SMatthias Ringwald * THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 31*faa6c1f6SMatthias Ringwald * SUCH DAMAGE. 32*faa6c1f6SMatthias Ringwald * 33*faa6c1f6SMatthias Ringwald * Please inquire about commercial licensing options at [email protected] 34*faa6c1f6SMatthias Ringwald * 35*faa6c1f6SMatthias Ringwald */ 36*faa6c1f6SMatthias Ringwald 37*faa6c1f6SMatthias Ringwald /* 38*faa6c1f6SMatthias Ringwald * bt_control_cc256x.c 39*faa6c1f6SMatthias Ringwald * 40*faa6c1f6SMatthias Ringwald * Adapter to use cc256x-based chipsets with BTstack 41*faa6c1f6SMatthias Ringwald * 42*faa6c1f6SMatthias Ringwald * Handles init script (a.k.a. Service Patch) 43*faa6c1f6SMatthias Ringwald * Allows for non-standard UART baud rate 44*faa6c1f6SMatthias Ringwald * Allows to configure transmit power 45*faa6c1f6SMatthias Ringwald * Allows to activate eHCILL deep sleep mode 46*faa6c1f6SMatthias Ringwald * 47*faa6c1f6SMatthias Ringwald * Issues with mspgcc LTS: 48*faa6c1f6SMatthias Ringwald * - 20 bit support is not there yet -> .text cannot get bigger than 48 kb 49*faa6c1f6SMatthias Ringwald * - arrays cannot have more than 32k entries 50*faa6c1f6SMatthias Ringwald * 51*faa6c1f6SMatthias Ringwald * workarounds: 52*faa6c1f6SMatthias Ringwald * - store init script in .fartext and use assembly code to read from there 53*faa6c1f6SMatthias Ringwald * - split into two arrays 54*faa6c1f6SMatthias Ringwald * 55*faa6c1f6SMatthias Ringwald * Issues with AVR 56*faa6c1f6SMatthias Ringwald * - Harvard architecture doesn't allow to store init script directly -> use avr-libc helpers 57*faa6c1f6SMatthias Ringwald * 58*faa6c1f6SMatthias Ringwald * Documentation for TI VS CC256x commands: http://processors.wiki.ti.com/index.php/CC256x_VS_HCI_Commands 59*faa6c1f6SMatthias Ringwald * 60*faa6c1f6SMatthias Ringwald */ 61*faa6c1f6SMatthias Ringwald 62*faa6c1f6SMatthias Ringwald #include "btstack_config.h" 63*faa6c1f6SMatthias Ringwald #include "bt_control_cc256x.h" 64*faa6c1f6SMatthias Ringwald 65*faa6c1f6SMatthias Ringwald #include <stddef.h> /* NULL */ 66*faa6c1f6SMatthias Ringwald #include <stdio.h> 67*faa6c1f6SMatthias Ringwald #include <string.h> /* memcpy */ 68*faa6c1f6SMatthias Ringwald 69*faa6c1f6SMatthias Ringwald #if defined(__GNUC__) && defined(__MSP430X__) && (__MSP430X__ > 0) 70*faa6c1f6SMatthias Ringwald #include "hal_compat.h" 71*faa6c1f6SMatthias Ringwald #endif 72*faa6c1f6SMatthias Ringwald 73*faa6c1f6SMatthias Ringwald #ifdef __AVR__ 74*faa6c1f6SMatthias Ringwald #include <avr/pgmspace.h> 75*faa6c1f6SMatthias Ringwald #endif 76*faa6c1f6SMatthias Ringwald 77*faa6c1f6SMatthias Ringwald #include "btstack_control.h" 78*faa6c1f6SMatthias Ringwald 79*faa6c1f6SMatthias Ringwald 80*faa6c1f6SMatthias Ringwald // actual init script provided by seperate .c file 81*faa6c1f6SMatthias Ringwald extern const uint8_t cc256x_init_script[]; 82*faa6c1f6SMatthias Ringwald extern const uint32_t cc256x_init_script_size; 83*faa6c1f6SMatthias Ringwald 84*faa6c1f6SMatthias Ringwald // init script 85*faa6c1f6SMatthias Ringwald static uint32_t init_script_offset = 0; 86*faa6c1f6SMatthias Ringwald static int16_t init_power_in_dB = 13; // 13 dBm 87*faa6c1f6SMatthias Ringwald static int init_ehcill_enabled = 0; 88*faa6c1f6SMatthias Ringwald 89*faa6c1f6SMatthias Ringwald // support for SCO over HCI 90*faa6c1f6SMatthias Ringwald #ifdef HAVE_SCO_OVER_HCI 91*faa6c1f6SMatthias Ringwald static int init_send_route_sco_over_hci = 0; 92*faa6c1f6SMatthias Ringwald // route SCO over HCI (connection type=1, tx buffer size = 0x00 (don't change), tx buffer max latency=0x0000(don't chnage)), accept packets - 0) 93*faa6c1f6SMatthias Ringwald static const uint8_t hci_route_sco_over_hci[] = { 94*faa6c1f6SMatthias Ringwald 0x10, 0xfe, 0x05, 0x01, 0x00, 0x00, 0x00, 0x00 95*faa6c1f6SMatthias Ringwald }; 96*faa6c1f6SMatthias Ringwald #endif 97*faa6c1f6SMatthias Ringwald 98*faa6c1f6SMatthias Ringwald static void chipset_init(void * config){ 99*faa6c1f6SMatthias Ringwald init_script_offset = 0; 100*faa6c1f6SMatthias Ringwald #ifdef HAVE_SCO_OVER_HCI 101*faa6c1f6SMatthias Ringwald init_send_route_sco_over_hci = 1; 102*faa6c1f6SMatthias Ringwald #endif 103*faa6c1f6SMatthias Ringwald } 104*faa6c1f6SMatthias Ringwald 105*faa6c1f6SMatthias Ringwald static void chipset_set_baudrate_command(uint32_t baudrate, uint8_t *hci_cmd_buffer){ 106*faa6c1f6SMatthias Ringwald hci_cmd_buffer[0] = 0x36; 107*faa6c1f6SMatthias Ringwald hci_cmd_buffer[1] = 0xFF; 108*faa6c1f6SMatthias Ringwald hci_cmd_buffer[2] = 0x04; 109*faa6c1f6SMatthias Ringwald hci_cmd_buffer[3] = baudrate & 0xff; 110*faa6c1f6SMatthias Ringwald hci_cmd_buffer[4] = (baudrate >> 8) & 0xff; 111*faa6c1f6SMatthias Ringwald hci_cmd_buffer[5] = (baudrate >> 16) & 0xff; 112*faa6c1f6SMatthias Ringwald hci_cmd_buffer[6] = 0; 113*faa6c1f6SMatthias Ringwald } 114*faa6c1f6SMatthias Ringwald 115*faa6c1f6SMatthias Ringwald #if 0 116*faa6c1f6SMatthias Ringwald static void chipset_set_bd_addr_command(bd_addr_t addr, uint8_t *hci_cmd_buffer){ 117*faa6c1f6SMatthias Ringwald } 118*faa6c1f6SMatthias Ringwald #endif 119*faa6c1f6SMatthias Ringwald 120*faa6c1f6SMatthias Ringwald // Output Power control from: http://e2e.ti.com/support/low_power_rf/f/660/p/134853/484767.aspx 121*faa6c1f6SMatthias Ringwald #define NUM_POWER_LEVELS 16 122*faa6c1f6SMatthias Ringwald #define DB_MIN_LEVEL -35 123*faa6c1f6SMatthias Ringwald #define DB_PER_LEVEL 5 124*faa6c1f6SMatthias Ringwald #define DB_DYNAMIC_RANGE 30 125*faa6c1f6SMatthias Ringwald 126*faa6c1f6SMatthias Ringwald static int get_max_power_for_modulation_type(int type){ 127*faa6c1f6SMatthias Ringwald // a) limit max output power 128*faa6c1f6SMatthias Ringwald int power_db; 129*faa6c1f6SMatthias Ringwald switch (type){ 130*faa6c1f6SMatthias Ringwald case 0: // GFSK 131*faa6c1f6SMatthias Ringwald power_db = 12; 132*faa6c1f6SMatthias Ringwald break; 133*faa6c1f6SMatthias Ringwald default: // EDRx 134*faa6c1f6SMatthias Ringwald power_db = 10; 135*faa6c1f6SMatthias Ringwald break; 136*faa6c1f6SMatthias Ringwald } 137*faa6c1f6SMatthias Ringwald if (power_db > init_power_in_dB) { 138*faa6c1f6SMatthias Ringwald power_db = init_power_in_dB; 139*faa6c1f6SMatthias Ringwald } 140*faa6c1f6SMatthias Ringwald return power_db; 141*faa6c1f6SMatthias Ringwald } 142*faa6c1f6SMatthias Ringwald 143*faa6c1f6SMatthias Ringwald static int get_highest_level_for_given_power(int power_db, int recommended_db){ 144*faa6c1f6SMatthias Ringwald int i = NUM_POWER_LEVELS-1; 145*faa6c1f6SMatthias Ringwald while (i) { 146*faa6c1f6SMatthias Ringwald if (power_db <= recommended_db) { 147*faa6c1f6SMatthias Ringwald return i; 148*faa6c1f6SMatthias Ringwald } 149*faa6c1f6SMatthias Ringwald power_db -= DB_PER_LEVEL; 150*faa6c1f6SMatthias Ringwald i--; 151*faa6c1f6SMatthias Ringwald } 152*faa6c1f6SMatthias Ringwald return 0; 153*faa6c1f6SMatthias Ringwald } 154*faa6c1f6SMatthias Ringwald 155*faa6c1f6SMatthias Ringwald static void update_set_power_vector(uint8_t *hci_cmd_buffer){ 156*faa6c1f6SMatthias Ringwald int i; 157*faa6c1f6SMatthias Ringwald int modulation_type = hci_cmd_buffer[3]; 158*faa6c1f6SMatthias Ringwald int power_db = get_max_power_for_modulation_type(modulation_type); 159*faa6c1f6SMatthias Ringwald int dynamic_range = 0; 160*faa6c1f6SMatthias Ringwald 161*faa6c1f6SMatthias Ringwald // f) don't touch level 0 162*faa6c1f6SMatthias Ringwald for ( i = (NUM_POWER_LEVELS-1) ; i >= 1 ; i--){ 163*faa6c1f6SMatthias Ringwald 164*faa6c1f6SMatthias Ringwald #ifdef ENABLE_BLE 165*faa6c1f6SMatthias Ringwald // level 1 is BLE transmit power for GFSK 166*faa6c1f6SMatthias Ringwald if (i == 1 && modulation_type == 0) { 167*faa6c1f6SMatthias Ringwald hci_cmd_buffer[4+1] = 2 * get_max_power_for_modulation_type(modulation_type); 168*faa6c1f6SMatthias Ringwald // as level 0 isn't set, we're done 169*faa6c1f6SMatthias Ringwald continue; 170*faa6c1f6SMatthias Ringwald } 171*faa6c1f6SMatthias Ringwald #endif 172*faa6c1f6SMatthias Ringwald hci_cmd_buffer[4+i] = 2 * power_db; 173*faa6c1f6SMatthias Ringwald 174*faa6c1f6SMatthias Ringwald if (dynamic_range + DB_PER_LEVEL > DB_DYNAMIC_RANGE) continue; // e) 175*faa6c1f6SMatthias Ringwald 176*faa6c1f6SMatthias Ringwald power_db -= DB_PER_LEVEL; // d) 177*faa6c1f6SMatthias Ringwald dynamic_range += DB_PER_LEVEL; 178*faa6c1f6SMatthias Ringwald 179*faa6c1f6SMatthias Ringwald if (power_db > DB_MIN_LEVEL) continue; 180*faa6c1f6SMatthias Ringwald 181*faa6c1f6SMatthias Ringwald power_db = DB_MIN_LEVEL; // b) 182*faa6c1f6SMatthias Ringwald } 183*faa6c1f6SMatthias Ringwald } 184*faa6c1f6SMatthias Ringwald 185*faa6c1f6SMatthias Ringwald // max permitted power for class 2 devices: 4 dBm 186*faa6c1f6SMatthias Ringwald static void update_set_class2_single_power(uint8_t * hci_cmd_buffer){ 187*faa6c1f6SMatthias Ringwald const int max_power_class_2 = 4; 188*faa6c1f6SMatthias Ringwald int i = 0; 189*faa6c1f6SMatthias Ringwald for (i=0;i<3;i++){ 190*faa6c1f6SMatthias Ringwald hci_cmd_buffer[3+i] = get_highest_level_for_given_power(get_max_power_for_modulation_type(i), max_power_class_2); 191*faa6c1f6SMatthias Ringwald } 192*faa6c1f6SMatthias Ringwald } 193*faa6c1f6SMatthias Ringwald 194*faa6c1f6SMatthias Ringwald // eHCILL activate from http://e2e.ti.com/support/low_power_rf/f/660/p/134855/484776.aspx 195*faa6c1f6SMatthias Ringwald static void update_sleep_mode_configurations(uint8_t * hci_cmd_buffer){ 196*faa6c1f6SMatthias Ringwald if (init_ehcill_enabled) { 197*faa6c1f6SMatthias Ringwald hci_cmd_buffer[4] = 1; 198*faa6c1f6SMatthias Ringwald } else { 199*faa6c1f6SMatthias Ringwald hci_cmd_buffer[4] = 0; 200*faa6c1f6SMatthias Ringwald } 201*faa6c1f6SMatthias Ringwald } 202*faa6c1f6SMatthias Ringwald 203*faa6c1f6SMatthias Ringwald static void update_init_script_command(uint8_t *hci_cmd_buffer){ 204*faa6c1f6SMatthias Ringwald 205*faa6c1f6SMatthias Ringwald uint16_t opcode = hci_cmd_buffer[0] | (hci_cmd_buffer[1] << 8); 206*faa6c1f6SMatthias Ringwald 207*faa6c1f6SMatthias Ringwald switch (opcode){ 208*faa6c1f6SMatthias Ringwald case 0xFD87: 209*faa6c1f6SMatthias Ringwald update_set_class2_single_power(hci_cmd_buffer); 210*faa6c1f6SMatthias Ringwald break; 211*faa6c1f6SMatthias Ringwald case 0xFD82: 212*faa6c1f6SMatthias Ringwald update_set_power_vector(hci_cmd_buffer); 213*faa6c1f6SMatthias Ringwald break; 214*faa6c1f6SMatthias Ringwald case 0xFD0C: 215*faa6c1f6SMatthias Ringwald update_sleep_mode_configurations(hci_cmd_buffer); 216*faa6c1f6SMatthias Ringwald break; 217*faa6c1f6SMatthias Ringwald default: 218*faa6c1f6SMatthias Ringwald break; 219*faa6c1f6SMatthias Ringwald } 220*faa6c1f6SMatthias Ringwald } 221*faa6c1f6SMatthias Ringwald 222*faa6c1f6SMatthias Ringwald static btstack_chipset_result_t chipset_next_command(uint8_t * hci_cmd_buffer){ 223*faa6c1f6SMatthias Ringwald if (init_script_offset >= cc256x_init_script_size) { 224*faa6c1f6SMatthias Ringwald 225*faa6c1f6SMatthias Ringwald #ifdef HAVE_SCO_OVER_HCI 226*faa6c1f6SMatthias Ringwald // append send route SCO over HCI if requested 227*faa6c1f6SMatthias Ringwald if (init_send_route_sco_over_hci){ 228*faa6c1f6SMatthias Ringwald init_send_route_sco_over_hci = 0; 229*faa6c1f6SMatthias Ringwald memcpy(hci_cmd_buffer, hci_route_sco_over_hci, sizeof(hci_route_sco_over_hci)); 230*faa6c1f6SMatthias Ringwald return BTSTACK_CHIPSET_VALID_COMMAND; 231*faa6c1f6SMatthias Ringwald } 232*faa6c1f6SMatthias Ringwald #endif 233*faa6c1f6SMatthias Ringwald 234*faa6c1f6SMatthias Ringwald return BTSTACK_CHIPSET_DONE; 235*faa6c1f6SMatthias Ringwald } 236*faa6c1f6SMatthias Ringwald 237*faa6c1f6SMatthias Ringwald // extracted init script has 0x01 cmd packet type, but BTstack expects them without 238*faa6c1f6SMatthias Ringwald init_script_offset++; 239*faa6c1f6SMatthias Ringwald 240*faa6c1f6SMatthias Ringwald #if defined(__GNUC__) && defined(__MSP430X__) && (__MSP430X__ > 0) 241*faa6c1f6SMatthias Ringwald 242*faa6c1f6SMatthias Ringwald // workaround: use FlashReadBlock with 32-bit integer and assume init script starts at 0x10000 243*faa6c1f6SMatthias Ringwald uint32_t init_script_addr = 0x10000; 244*faa6c1f6SMatthias Ringwald FlashReadBlock(&hci_cmd_buffer[0], init_script_addr + init_script_offset, 3); // cmd header 245*faa6c1f6SMatthias Ringwald init_script_offset += 3; 246*faa6c1f6SMatthias Ringwald int payload_len = hci_cmd_buffer[2]; 247*faa6c1f6SMatthias Ringwald FlashReadBlock(&hci_cmd_buffer[3], init_script_addr + init_script_offset, payload_len); // cmd payload 248*faa6c1f6SMatthias Ringwald 249*faa6c1f6SMatthias Ringwald #elif defined (__AVR__) 250*faa6c1f6SMatthias Ringwald 251*faa6c1f6SMatthias Ringwald // workaround: use memcpy_P to access init script in lower 64 kB of flash 252*faa6c1f6SMatthias Ringwald memcpy_P(&hci_cmd_buffer[0], &cc256x_init_script[init_script_offset], 3); 253*faa6c1f6SMatthias Ringwald init_script_offset += 3; 254*faa6c1f6SMatthias Ringwald int payload_len = hci_cmd_buffer[2]; 255*faa6c1f6SMatthias Ringwald memcpy_P(&hci_cmd_buffer[3], &cc256x_init_script[init_script_offset], payload_len); 256*faa6c1f6SMatthias Ringwald 257*faa6c1f6SMatthias Ringwald #else 258*faa6c1f6SMatthias Ringwald 259*faa6c1f6SMatthias Ringwald // use memcpy with pointer 260*faa6c1f6SMatthias Ringwald uint8_t * init_script_ptr = (uint8_t*) &cc256x_init_script[0]; 261*faa6c1f6SMatthias Ringwald memcpy(&hci_cmd_buffer[0], init_script_ptr + init_script_offset, 3); // cmd header 262*faa6c1f6SMatthias Ringwald init_script_offset += 3; 263*faa6c1f6SMatthias Ringwald int payload_len = hci_cmd_buffer[2]; 264*faa6c1f6SMatthias Ringwald memcpy(&hci_cmd_buffer[3], init_script_ptr + init_script_offset, payload_len); // cmd payload 265*faa6c1f6SMatthias Ringwald 266*faa6c1f6SMatthias Ringwald #endif 267*faa6c1f6SMatthias Ringwald 268*faa6c1f6SMatthias Ringwald init_script_offset += payload_len; 269*faa6c1f6SMatthias Ringwald 270*faa6c1f6SMatthias Ringwald // control power commands and ehcill 271*faa6c1f6SMatthias Ringwald update_init_script_command(hci_cmd_buffer); 272*faa6c1f6SMatthias Ringwald 273*faa6c1f6SMatthias Ringwald return BTSTACK_CHIPSET_VALID_COMMAND; 274*faa6c1f6SMatthias Ringwald } 275*faa6c1f6SMatthias Ringwald 276*faa6c1f6SMatthias Ringwald 277*faa6c1f6SMatthias Ringwald // MARK: public API 278*faa6c1f6SMatthias Ringwald void btstack_chipset_cc256x_enable_ehcill(int on){ 279*faa6c1f6SMatthias Ringwald init_ehcill_enabled = on; 280*faa6c1f6SMatthias Ringwald } 281*faa6c1f6SMatthias Ringwald 282*faa6c1f6SMatthias Ringwald int btstack_chipset_cc256x_ehcill_enabled(void){ 283*faa6c1f6SMatthias Ringwald return init_ehcill_enabled; 284*faa6c1f6SMatthias Ringwald } 285*faa6c1f6SMatthias Ringwald 286*faa6c1f6SMatthias Ringwald void btstack_chipset_cc256x_set_power(int16_t power_in_dB){ 287*faa6c1f6SMatthias Ringwald init_power_in_dB = power_in_dB; 288*faa6c1f6SMatthias Ringwald } 289*faa6c1f6SMatthias Ringwald 290*faa6c1f6SMatthias Ringwald static const btstack_chipset_t btstack_chipset_cc256x = { 291*faa6c1f6SMatthias Ringwald "CC256x", 292*faa6c1f6SMatthias Ringwald chipset_init, 293*faa6c1f6SMatthias Ringwald chipset_next_command, 294*faa6c1f6SMatthias Ringwald chipset_set_baudrate_command, 295*faa6c1f6SMatthias Ringwald NULL, // set bd addr command not available or impemented 296*faa6c1f6SMatthias Ringwald }; 297*faa6c1f6SMatthias Ringwald 298*faa6c1f6SMatthias Ringwald const btstack_chipset_t * btstack_chipset_cc256x_instance(void){ 299*faa6c1f6SMatthias Ringwald return &btstack_chipset_cc256x; 300*faa6c1f6SMatthias Ringwald } 301*faa6c1f6SMatthias Ringwald 302