1faa6c1f6SMatthias Ringwald /*
2faa6c1f6SMatthias Ringwald * Copyright (C) 2009-2012 by Matthias Ringwald
3faa6c1f6SMatthias Ringwald *
4faa6c1f6SMatthias Ringwald * Redistribution and use in source and binary forms, with or without
5faa6c1f6SMatthias Ringwald * modification, are permitted provided that the following conditions
6faa6c1f6SMatthias Ringwald * are met:
7faa6c1f6SMatthias Ringwald *
8faa6c1f6SMatthias Ringwald * 1. Redistributions of source code must retain the above copyright
9faa6c1f6SMatthias Ringwald * notice, this list of conditions and the following disclaimer.
10faa6c1f6SMatthias Ringwald * 2. Redistributions in binary form must reproduce the above copyright
11faa6c1f6SMatthias Ringwald * notice, this list of conditions and the following disclaimer in the
12faa6c1f6SMatthias Ringwald * documentation and/or other materials provided with the distribution.
13faa6c1f6SMatthias Ringwald * 3. Neither the name of the copyright holders nor the names of
14faa6c1f6SMatthias Ringwald * contributors may be used to endorse or promote products derived
15faa6c1f6SMatthias Ringwald * from this software without specific prior written permission.
16faa6c1f6SMatthias Ringwald * 4. Any redistribution, use, or modification is done solely for
17faa6c1f6SMatthias Ringwald * personal benefit and not for any commercial purpose or for
18faa6c1f6SMatthias Ringwald * monetary gain.
19faa6c1f6SMatthias Ringwald *
20faa6c1f6SMatthias Ringwald * THIS SOFTWARE IS PROVIDED BY MATTHIAS RINGWALD AND CONTRIBUTORS
21faa6c1f6SMatthias Ringwald * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
22faa6c1f6SMatthias Ringwald * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
23faa6c1f6SMatthias Ringwald * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL MATTHIAS
24faa6c1f6SMatthias Ringwald * RINGWALD OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
25faa6c1f6SMatthias Ringwald * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
26faa6c1f6SMatthias Ringwald * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS
27faa6c1f6SMatthias Ringwald * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
28faa6c1f6SMatthias Ringwald * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
29faa6c1f6SMatthias Ringwald * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF
30faa6c1f6SMatthias Ringwald * THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
31faa6c1f6SMatthias Ringwald * SUCH DAMAGE.
32faa6c1f6SMatthias Ringwald *
3351acb414SMatthias Ringwald * Please inquire about commercial licensing options at [email protected]
34faa6c1f6SMatthias Ringwald *
35faa6c1f6SMatthias Ringwald */
36faa6c1f6SMatthias Ringwald
37e501bae0SMatthias Ringwald #define BTSTACK_FILE__ "btstack_chipset_cc256x.c"
38ab2c6ae4SMatthias Ringwald
39faa6c1f6SMatthias Ringwald /*
40fb55bd0aSMatthias Ringwald * btstack_chipset_cc256x.c
41faa6c1f6SMatthias Ringwald *
42faa6c1f6SMatthias Ringwald * Adapter to use cc256x-based chipsets with BTstack
43faa6c1f6SMatthias Ringwald *
44faa6c1f6SMatthias Ringwald * Handles init script (a.k.a. Service Patch)
45faa6c1f6SMatthias Ringwald * Allows for non-standard UART baud rate
46faa6c1f6SMatthias Ringwald * Allows to configure transmit power
47faa6c1f6SMatthias Ringwald * Allows to activate eHCILL deep sleep mode
48faa6c1f6SMatthias Ringwald *
49faa6c1f6SMatthias Ringwald * Issues with mspgcc LTS:
50faa6c1f6SMatthias Ringwald * - 20 bit support is not there yet -> .text cannot get bigger than 48 kb
51faa6c1f6SMatthias Ringwald * - arrays cannot have more than 32k entries
52faa6c1f6SMatthias Ringwald *
53faa6c1f6SMatthias Ringwald * workarounds:
54faa6c1f6SMatthias Ringwald * - store init script in .fartext and use assembly code to read from there
55faa6c1f6SMatthias Ringwald * - split into two arrays
56faa6c1f6SMatthias Ringwald *
57faa6c1f6SMatthias Ringwald * Issues with AVR
58faa6c1f6SMatthias Ringwald * - Harvard architecture doesn't allow to store init script directly -> use avr-libc helpers
59faa6c1f6SMatthias Ringwald *
60faa6c1f6SMatthias Ringwald * Documentation for TI VS CC256x commands: http://processors.wiki.ti.com/index.php/CC256x_VS_HCI_Commands
61faa6c1f6SMatthias Ringwald *
62faa6c1f6SMatthias Ringwald */
63faa6c1f6SMatthias Ringwald
64faa6c1f6SMatthias Ringwald #include "btstack_config.h"
65fb55bd0aSMatthias Ringwald #include "btstack_chipset_cc256x.h"
66e2ee49eaSMatthias Ringwald #include "btstack_debug.h"
677a4d61a3SMatthias Ringwald #include "hci.h"
68faa6c1f6SMatthias Ringwald
69faa6c1f6SMatthias Ringwald #include <stddef.h> /* NULL */
70faa6c1f6SMatthias Ringwald #include <stdio.h>
71faa6c1f6SMatthias Ringwald #include <string.h> /* memcpy */
72faa6c1f6SMatthias Ringwald
737a4d61a3SMatthias Ringwald // assert outgoing and incoming hci packet buffers can hold max hci command resp. event packet
747a4d61a3SMatthias Ringwald #if HCI_OUTGOING_PACKET_BUFFER_SIZE < (HCI_CMD_HEADER_SIZE + 255)
757a4d61a3SMatthias Ringwald #error "HCI_OUTGOING_PACKET_BUFFER_SIZE to small. Outgoing HCI packet buffer to small for largest HCI Command packet. Please set HCI_ACL_PAYLOAD_SIZE to 258 or higher."
767a4d61a3SMatthias Ringwald #endif
776ffe3f3eSMatthias Ringwald #if HCI_INCOMING_PACKET_BUFFER_SIZE < (HCI_EVENT_HEADER_SIZE + 255)
787a4d61a3SMatthias Ringwald #error "HCI_INCOMING_PACKET_BUFFER_SIZE to small. Incoming HCI packet buffer to small for largest HCI Event packet. Please set HCI_ACL_PAYLOAD_SIZE to 257 or higher."
797a4d61a3SMatthias Ringwald #endif
807a4d61a3SMatthias Ringwald
8149ed206dSMatthias Ringwald #if defined(ENABLE_SCO_OVER_HCI) && defined(ENABLE_SCO_OVER_PCM)
8249ed206dSMatthias Ringwald #error "SCO can either be routed over HCI or PCM, please define only one of: ENABLE_SCO_OVER_HCI or ENABLE_SCO_OVER_PCM"
8349ed206dSMatthias Ringwald #endif
8449ed206dSMatthias Ringwald
85faa6c1f6SMatthias Ringwald #if defined(__GNUC__) && defined(__MSP430X__) && (__MSP430X__ > 0)
86faa6c1f6SMatthias Ringwald #include "hal_compat.h"
87faa6c1f6SMatthias Ringwald #endif
88faa6c1f6SMatthias Ringwald
89faa6c1f6SMatthias Ringwald #ifdef __AVR__
90faa6c1f6SMatthias Ringwald #include <avr/pgmspace.h>
91faa6c1f6SMatthias Ringwald #endif
92faa6c1f6SMatthias Ringwald
93faa6c1f6SMatthias Ringwald #include "btstack_control.h"
94faa6c1f6SMatthias Ringwald
95faa6c1f6SMatthias Ringwald
96e2ee49eaSMatthias Ringwald // default init script provided by separate .c file
97faa6c1f6SMatthias Ringwald extern const uint8_t cc256x_init_script[];
98faa6c1f6SMatthias Ringwald extern const uint32_t cc256x_init_script_size;
99faa6c1f6SMatthias Ringwald
100e2ee49eaSMatthias Ringwald // custom init script set by btstack_chipset_cc256x_set_init_script
101e2ee49eaSMatthias Ringwald // used to select init scripts before each power up
102e2ee49eaSMatthias Ringwald static const uint8_t * custom_init_script;
103e2ee49eaSMatthias Ringwald static uint32_t custom_init_script_size;
104e2ee49eaSMatthias Ringwald
105e2ee49eaSMatthias Ringwald // init script to use: either cc256x_init_script or custom_init_script
106e2ee49eaSMatthias Ringwald static const uint8_t * init_script;
107e2ee49eaSMatthias Ringwald static uint32_t init_script_size;
108e2ee49eaSMatthias Ringwald
109e2ee49eaSMatthias Ringwald // power in db - set by btstack_chipset_cc256x_set_power
110faa6c1f6SMatthias Ringwald static int16_t init_power_in_dB = 13; // 13 dBm
111faa6c1f6SMatthias Ringwald
11263efc2edSMatthias Ringwald // explicit power vectors of 16 uint8_t bytes
11363efc2edSMatthias Ringwald static const uint8_t * init_power_vectors[3];
11463efc2edSMatthias Ringwald
115e2ee49eaSMatthias Ringwald // upload position
116e2ee49eaSMatthias Ringwald static uint32_t init_script_offset = 0;
117e2ee49eaSMatthias Ringwald
118faa6c1f6SMatthias Ringwald // support for SCO over HCI
11971c4de7aSMatthias Ringwald #ifdef ENABLE_SCO_OVER_HCI
120faa6c1f6SMatthias Ringwald static int init_send_route_sco_over_hci = 0;
121638e16f3SMatthias Ringwald static const uint8_t hci_route_sco_over_hci[] = {
1225415113fSMatthias Ringwald // Follow recommendation from https://e2e.ti.com/support/wireless_connectivity/bluetooth_cc256x/f/660/t/397004
1235415113fSMatthias Ringwald // route SCO over HCI (connection type=1, tx buffer size = 120, tx buffer max latency= 720, accept packets with CRC Error
1245415113fSMatthias Ringwald 0x10, 0xfe, 0x05, 0x01, 0x78, 0xd0, 0x02, 0x01,
12549ed206dSMatthias Ringwald };
126638e16f3SMatthias Ringwald #endif
12749ed206dSMatthias Ringwald #ifdef ENABLE_SCO_OVER_PCM
12849ed206dSMatthias Ringwald static int init_send_sco_i2s_config_cvsd = 0;
12949ed206dSMatthias Ringwald static const uint8_t hci_write_codec_config_cvsd[] = {
13049ed206dSMatthias Ringwald 0x06, 0xFD, // HCI opcode = HCI_VS_Write_CODEC_Config
13149ed206dSMatthias Ringwald 0x22, // HCI param length
13249ed206dSMatthias Ringwald 0x00, 0x01, // PCM clock rate 256, - clock rate 256000 Hz
13349ed206dSMatthias Ringwald 0x00, // PCM clock direction = master
13449ed206dSMatthias Ringwald 0x40, 0x1F, 0x00, 0x00, // PCM frame sync = 8kHz
13549ed206dSMatthias Ringwald 0x10, 0x00, // PCM frame sync duty cycle = 16 clk
13649ed206dSMatthias Ringwald 0x01, // PCM frame edge = rising edge
13749ed206dSMatthias Ringwald 0x00, // PCM frame polarity = active high
13849ed206dSMatthias Ringwald 0x00, // Reserved
13949ed206dSMatthias Ringwald 0x10, 0x00, // PCM channel 1 out size = 16
14049ed206dSMatthias Ringwald 0x01, 0x00, // PCM channel 1 out offset = 1
14149ed206dSMatthias Ringwald 0x01, // PCM channel 1 out edge = rising
14249ed206dSMatthias Ringwald 0x10, 0x00, // PCM channel 1 in size = 16
14349ed206dSMatthias Ringwald 0x01, 0x00, // PCM channel 1 in offset = 1
14449ed206dSMatthias Ringwald 0x00, // PCM channel 1 in edge = falling
14549ed206dSMatthias Ringwald 0x00, // Reserved
14649ed206dSMatthias Ringwald 0x10, 0x00, // PCM channel 2 out size = 16
14749ed206dSMatthias Ringwald 0x11, 0x00, // PCM channel 2 out offset = 17
14849ed206dSMatthias Ringwald 0x01, // PCM channel 2 out edge = rising
14949ed206dSMatthias Ringwald 0x10, 0x00, // PCM channel 2 in size = 16
15049ed206dSMatthias Ringwald 0x11, 0x00, // PCM channel 2 in offset = 17
15149ed206dSMatthias Ringwald 0x00, // PCM channel 2 in edge = falling
15249ed206dSMatthias Ringwald 0x00, // Reserved
153faa6c1f6SMatthias Ringwald };
154faa6c1f6SMatthias Ringwald #endif
155faa6c1f6SMatthias Ringwald
chipset_init(const void * config)156fb55bd0aSMatthias Ringwald static void chipset_init(const void * config){
157*ee45d18fSMatthias Ringwald UNUSED(config);
158faa6c1f6SMatthias Ringwald init_script_offset = 0;
15948900dc5SMatthias Ringwald #if defined(__GNUC__) && defined(__MSP430X__) && (__MSP430X__ > 0)
16048900dc5SMatthias Ringwald // On MSP430, custom init script is not supported
16148900dc5SMatthias Ringwald init_script_size = cc256x_init_script_size;
16248900dc5SMatthias Ringwald #else
163e2ee49eaSMatthias Ringwald if (custom_init_script){
164e2ee49eaSMatthias Ringwald log_info("cc256x: using custom init script");
165e2ee49eaSMatthias Ringwald init_script = custom_init_script;
166e2ee49eaSMatthias Ringwald init_script_size = custom_init_script_size;
167e2ee49eaSMatthias Ringwald } else {
168e2ee49eaSMatthias Ringwald log_info("cc256x: using default init script");
169e2ee49eaSMatthias Ringwald init_script = cc256x_init_script;
170e2ee49eaSMatthias Ringwald init_script_size = cc256x_init_script_size;
171e2ee49eaSMatthias Ringwald }
17248900dc5SMatthias Ringwald #endif
17371c4de7aSMatthias Ringwald #ifdef ENABLE_SCO_OVER_HCI
174faa6c1f6SMatthias Ringwald init_send_route_sco_over_hci = 1;
175faa6c1f6SMatthias Ringwald #endif
17649ed206dSMatthias Ringwald #ifdef ENABLE_SCO_OVER_PCM
17749ed206dSMatthias Ringwald init_send_sco_i2s_config_cvsd = 1;
17849ed206dSMatthias Ringwald #endif
179faa6c1f6SMatthias Ringwald }
180faa6c1f6SMatthias Ringwald
chipset_set_baudrate_command(uint32_t baudrate,uint8_t * hci_cmd_buffer)181faa6c1f6SMatthias Ringwald static void chipset_set_baudrate_command(uint32_t baudrate, uint8_t *hci_cmd_buffer){
182faa6c1f6SMatthias Ringwald hci_cmd_buffer[0] = 0x36;
183faa6c1f6SMatthias Ringwald hci_cmd_buffer[1] = 0xFF;
184faa6c1f6SMatthias Ringwald hci_cmd_buffer[2] = 0x04;
185faa6c1f6SMatthias Ringwald hci_cmd_buffer[3] = baudrate & 0xff;
186faa6c1f6SMatthias Ringwald hci_cmd_buffer[4] = (baudrate >> 8) & 0xff;
187faa6c1f6SMatthias Ringwald hci_cmd_buffer[5] = (baudrate >> 16) & 0xff;
188faa6c1f6SMatthias Ringwald hci_cmd_buffer[6] = 0;
189faa6c1f6SMatthias Ringwald }
190faa6c1f6SMatthias Ringwald
chipset_set_bd_addr_command(bd_addr_t addr,uint8_t * hci_cmd_buffer)191faa6c1f6SMatthias Ringwald static void chipset_set_bd_addr_command(bd_addr_t addr, uint8_t *hci_cmd_buffer){
192e63eef5dSMatthias Ringwald hci_cmd_buffer[0] = 0x06;
193e63eef5dSMatthias Ringwald hci_cmd_buffer[1] = 0xFC;
194e63eef5dSMatthias Ringwald hci_cmd_buffer[2] = 0x06;
195e63eef5dSMatthias Ringwald reverse_bd_addr(addr, &hci_cmd_buffer[3]);
196faa6c1f6SMatthias Ringwald }
197faa6c1f6SMatthias Ringwald
198faa6c1f6SMatthias Ringwald // Output Power control from: http://e2e.ti.com/support/low_power_rf/f/660/p/134853/484767.aspx
199faa6c1f6SMatthias Ringwald #define NUM_POWER_LEVELS 16
200faa6c1f6SMatthias Ringwald #define DB_MIN_LEVEL -35
201faa6c1f6SMatthias Ringwald #define DB_PER_LEVEL 5
202faa6c1f6SMatthias Ringwald #define DB_DYNAMIC_RANGE 30
203faa6c1f6SMatthias Ringwald
get_max_power_for_modulation_type(int type)204faa6c1f6SMatthias Ringwald static int get_max_power_for_modulation_type(int type){
205faa6c1f6SMatthias Ringwald // a) limit max output power
206faa6c1f6SMatthias Ringwald int power_db;
207faa6c1f6SMatthias Ringwald switch (type){
208faa6c1f6SMatthias Ringwald case 0: // GFSK
209faa6c1f6SMatthias Ringwald power_db = 12;
210faa6c1f6SMatthias Ringwald break;
211faa6c1f6SMatthias Ringwald default: // EDRx
212faa6c1f6SMatthias Ringwald power_db = 10;
213faa6c1f6SMatthias Ringwald break;
214faa6c1f6SMatthias Ringwald }
215faa6c1f6SMatthias Ringwald if (power_db > init_power_in_dB) {
216faa6c1f6SMatthias Ringwald power_db = init_power_in_dB;
217faa6c1f6SMatthias Ringwald }
218faa6c1f6SMatthias Ringwald return power_db;
219faa6c1f6SMatthias Ringwald }
220faa6c1f6SMatthias Ringwald
get_highest_level_for_given_power(int power_db,int recommended_db)221faa6c1f6SMatthias Ringwald static int get_highest_level_for_given_power(int power_db, int recommended_db){
222faa6c1f6SMatthias Ringwald int i = NUM_POWER_LEVELS-1;
223faa6c1f6SMatthias Ringwald while (i) {
224faa6c1f6SMatthias Ringwald if (power_db <= recommended_db) {
225faa6c1f6SMatthias Ringwald return i;
226faa6c1f6SMatthias Ringwald }
227faa6c1f6SMatthias Ringwald power_db -= DB_PER_LEVEL;
228faa6c1f6SMatthias Ringwald i--;
229faa6c1f6SMatthias Ringwald }
230faa6c1f6SMatthias Ringwald return 0;
231faa6c1f6SMatthias Ringwald }
232faa6c1f6SMatthias Ringwald
update_set_power_vector(uint8_t * hci_cmd_buffer)233faa6c1f6SMatthias Ringwald static void update_set_power_vector(uint8_t *hci_cmd_buffer){
23463efc2edSMatthias Ringwald uint8_t modulation_type = hci_cmd_buffer[3];
23563efc2edSMatthias Ringwald btstack_assert(modulation_type <= 2);
23663efc2edSMatthias Ringwald
23763efc2edSMatthias Ringwald // explicit power vector provided by user
23863efc2edSMatthias Ringwald if (init_power_vectors[modulation_type] != NULL){
23963efc2edSMatthias Ringwald (void)memcpy(&hci_cmd_buffer[4], init_power_vectors[modulation_type], 16);
24063efc2edSMatthias Ringwald return;
24163efc2edSMatthias Ringwald }
24263efc2edSMatthias Ringwald
24363efc2edSMatthias Ringwald unsigned int i;
244faa6c1f6SMatthias Ringwald int power_db = get_max_power_for_modulation_type(modulation_type);
245faa6c1f6SMatthias Ringwald int dynamic_range = 0;
246faa6c1f6SMatthias Ringwald
247faa6c1f6SMatthias Ringwald // f) don't touch level 0
248faa6c1f6SMatthias Ringwald for ( i = (NUM_POWER_LEVELS-1) ; i >= 1 ; i--){
249faa6c1f6SMatthias Ringwald
250faa6c1f6SMatthias Ringwald #ifdef ENABLE_BLE
251faa6c1f6SMatthias Ringwald // level 1 is BLE transmit power for GFSK
252faa6c1f6SMatthias Ringwald if (i == 1 && modulation_type == 0) {
253faa6c1f6SMatthias Ringwald hci_cmd_buffer[4+1] = 2 * get_max_power_for_modulation_type(modulation_type);
254faa6c1f6SMatthias Ringwald // as level 0 isn't set, we're done
255faa6c1f6SMatthias Ringwald continue;
256faa6c1f6SMatthias Ringwald }
257faa6c1f6SMatthias Ringwald #endif
258faa6c1f6SMatthias Ringwald hci_cmd_buffer[4+i] = 2 * power_db;
259faa6c1f6SMatthias Ringwald
260faa6c1f6SMatthias Ringwald if (dynamic_range + DB_PER_LEVEL > DB_DYNAMIC_RANGE) continue; // e)
261faa6c1f6SMatthias Ringwald
262faa6c1f6SMatthias Ringwald power_db -= DB_PER_LEVEL; // d)
263faa6c1f6SMatthias Ringwald dynamic_range += DB_PER_LEVEL;
264faa6c1f6SMatthias Ringwald
265faa6c1f6SMatthias Ringwald if (power_db > DB_MIN_LEVEL) continue;
266faa6c1f6SMatthias Ringwald
267faa6c1f6SMatthias Ringwald power_db = DB_MIN_LEVEL; // b)
268faa6c1f6SMatthias Ringwald }
269faa6c1f6SMatthias Ringwald }
270faa6c1f6SMatthias Ringwald
271faa6c1f6SMatthias Ringwald // max permitted power for class 2 devices: 4 dBm
update_set_class2_single_power(uint8_t * hci_cmd_buffer)272faa6c1f6SMatthias Ringwald static void update_set_class2_single_power(uint8_t * hci_cmd_buffer){
273faa6c1f6SMatthias Ringwald const int max_power_class_2 = 4;
274faa6c1f6SMatthias Ringwald int i = 0;
275faa6c1f6SMatthias Ringwald for (i=0;i<3;i++){
276faa6c1f6SMatthias Ringwald hci_cmd_buffer[3+i] = get_highest_level_for_given_power(get_max_power_for_modulation_type(i), max_power_class_2);
277faa6c1f6SMatthias Ringwald }
278faa6c1f6SMatthias Ringwald }
279faa6c1f6SMatthias Ringwald
280faa6c1f6SMatthias Ringwald // eHCILL activate from http://e2e.ti.com/support/low_power_rf/f/660/p/134855/484776.aspx
update_sleep_mode_configurations(uint8_t * hci_cmd_buffer)281faa6c1f6SMatthias Ringwald static void update_sleep_mode_configurations(uint8_t * hci_cmd_buffer){
282f6a20ec9SMatthias Ringwald #ifdef ENABLE_EHCILL
283faa6c1f6SMatthias Ringwald hci_cmd_buffer[4] = 1;
284f6a20ec9SMatthias Ringwald #else
285faa6c1f6SMatthias Ringwald hci_cmd_buffer[4] = 0;
286f6a20ec9SMatthias Ringwald #endif
287faa6c1f6SMatthias Ringwald }
288faa6c1f6SMatthias Ringwald
update_init_script_command(uint8_t * hci_cmd_buffer)289faa6c1f6SMatthias Ringwald static void update_init_script_command(uint8_t *hci_cmd_buffer){
290faa6c1f6SMatthias Ringwald
291faa6c1f6SMatthias Ringwald uint16_t opcode = hci_cmd_buffer[0] | (hci_cmd_buffer[1] << 8);
292faa6c1f6SMatthias Ringwald
293faa6c1f6SMatthias Ringwald switch (opcode){
294faa6c1f6SMatthias Ringwald case 0xFD87:
295faa6c1f6SMatthias Ringwald update_set_class2_single_power(hci_cmd_buffer);
296faa6c1f6SMatthias Ringwald break;
297faa6c1f6SMatthias Ringwald case 0xFD82:
298faa6c1f6SMatthias Ringwald update_set_power_vector(hci_cmd_buffer);
299faa6c1f6SMatthias Ringwald break;
300faa6c1f6SMatthias Ringwald case 0xFD0C:
301faa6c1f6SMatthias Ringwald update_sleep_mode_configurations(hci_cmd_buffer);
302faa6c1f6SMatthias Ringwald break;
303faa6c1f6SMatthias Ringwald default:
304faa6c1f6SMatthias Ringwald break;
305faa6c1f6SMatthias Ringwald }
306faa6c1f6SMatthias Ringwald }
307faa6c1f6SMatthias Ringwald
chipset_next_command(uint8_t * hci_cmd_buffer)308faa6c1f6SMatthias Ringwald static btstack_chipset_result_t chipset_next_command(uint8_t * hci_cmd_buffer){
309e2ee49eaSMatthias Ringwald if (init_script_offset >= init_script_size) {
310faa6c1f6SMatthias Ringwald
31171c4de7aSMatthias Ringwald #ifdef ENABLE_SCO_OVER_HCI
312faa6c1f6SMatthias Ringwald // append send route SCO over HCI if requested
313faa6c1f6SMatthias Ringwald if (init_send_route_sco_over_hci){
314faa6c1f6SMatthias Ringwald init_send_route_sco_over_hci = 0;
315faa6c1f6SMatthias Ringwald memcpy(hci_cmd_buffer, hci_route_sco_over_hci, sizeof(hci_route_sco_over_hci));
316faa6c1f6SMatthias Ringwald return BTSTACK_CHIPSET_VALID_COMMAND;
317faa6c1f6SMatthias Ringwald }
318faa6c1f6SMatthias Ringwald #endif
31949ed206dSMatthias Ringwald #ifdef ENABLE_SCO_OVER_PCM
32049ed206dSMatthias Ringwald // append sco i2s cvsd config
32149ed206dSMatthias Ringwald if (init_send_sco_i2s_config_cvsd){
32249ed206dSMatthias Ringwald init_send_sco_i2s_config_cvsd = 0;
32349ed206dSMatthias Ringwald memcpy(hci_cmd_buffer, hci_write_codec_config_cvsd, sizeof(hci_write_codec_config_cvsd));
32449ed206dSMatthias Ringwald return BTSTACK_CHIPSET_VALID_COMMAND;
32549ed206dSMatthias Ringwald }
32649ed206dSMatthias Ringwald #endif
327faa6c1f6SMatthias Ringwald return BTSTACK_CHIPSET_DONE;
328faa6c1f6SMatthias Ringwald }
329faa6c1f6SMatthias Ringwald
330faa6c1f6SMatthias Ringwald // extracted init script has 0x01 cmd packet type, but BTstack expects them without
331faa6c1f6SMatthias Ringwald init_script_offset++;
332faa6c1f6SMatthias Ringwald
333faa6c1f6SMatthias Ringwald #if defined(__GNUC__) && defined(__MSP430X__) && (__MSP430X__ > 0)
334faa6c1f6SMatthias Ringwald
335faa6c1f6SMatthias Ringwald // workaround: use FlashReadBlock with 32-bit integer and assume init script starts at 0x10000
336faa6c1f6SMatthias Ringwald uint32_t init_script_addr = 0x10000;
337faa6c1f6SMatthias Ringwald FlashReadBlock(&hci_cmd_buffer[0], init_script_addr + init_script_offset, 3); // cmd header
338faa6c1f6SMatthias Ringwald init_script_offset += 3;
339faa6c1f6SMatthias Ringwald int payload_len = hci_cmd_buffer[2];
340faa6c1f6SMatthias Ringwald FlashReadBlock(&hci_cmd_buffer[3], init_script_addr + init_script_offset, payload_len); // cmd payload
341faa6c1f6SMatthias Ringwald
342faa6c1f6SMatthias Ringwald #elif defined (__AVR__)
343faa6c1f6SMatthias Ringwald
344faa6c1f6SMatthias Ringwald // workaround: use memcpy_P to access init script in lower 64 kB of flash
345e2ee49eaSMatthias Ringwald memcpy_P(&hci_cmd_buffer[0], &init_script[init_script_offset], 3);
346faa6c1f6SMatthias Ringwald init_script_offset += 3;
347faa6c1f6SMatthias Ringwald int payload_len = hci_cmd_buffer[2];
348e2ee49eaSMatthias Ringwald memcpy_P(&hci_cmd_buffer[3], &init_script[init_script_offset], payload_len);
349faa6c1f6SMatthias Ringwald
350faa6c1f6SMatthias Ringwald #else
351faa6c1f6SMatthias Ringwald
352faa6c1f6SMatthias Ringwald // use memcpy with pointer
353e2ee49eaSMatthias Ringwald uint8_t * init_script_ptr = (uint8_t*) &init_script[0];
354faa6c1f6SMatthias Ringwald memcpy(&hci_cmd_buffer[0], init_script_ptr + init_script_offset, 3); // cmd header
355faa6c1f6SMatthias Ringwald init_script_offset += 3;
356faa6c1f6SMatthias Ringwald int payload_len = hci_cmd_buffer[2];
357faa6c1f6SMatthias Ringwald memcpy(&hci_cmd_buffer[3], init_script_ptr + init_script_offset, payload_len); // cmd payload
358faa6c1f6SMatthias Ringwald
359faa6c1f6SMatthias Ringwald #endif
360faa6c1f6SMatthias Ringwald
361faa6c1f6SMatthias Ringwald init_script_offset += payload_len;
362faa6c1f6SMatthias Ringwald
363faa6c1f6SMatthias Ringwald // control power commands and ehcill
364faa6c1f6SMatthias Ringwald update_init_script_command(hci_cmd_buffer);
365faa6c1f6SMatthias Ringwald
366faa6c1f6SMatthias Ringwald return BTSTACK_CHIPSET_VALID_COMMAND;
367faa6c1f6SMatthias Ringwald }
368faa6c1f6SMatthias Ringwald
369faa6c1f6SMatthias Ringwald
370faa6c1f6SMatthias Ringwald // MARK: public API
btstack_chipset_cc256x_set_power(int16_t power_in_dB)371faa6c1f6SMatthias Ringwald void btstack_chipset_cc256x_set_power(int16_t power_in_dB){
372faa6c1f6SMatthias Ringwald init_power_in_dB = power_in_dB;
373faa6c1f6SMatthias Ringwald }
374faa6c1f6SMatthias Ringwald
btstack_chipset_cc256x_set_power_vector(uint8_t modulation_type,const uint8_t * power_vector)37563efc2edSMatthias Ringwald void btstack_chipset_cc256x_set_power_vector(uint8_t modulation_type, const uint8_t * power_vector){
37663efc2edSMatthias Ringwald btstack_assert(modulation_type <= 2);
3774bf37171SMatthias Ringwald init_power_vectors[modulation_type] = power_vector;
37863efc2edSMatthias Ringwald }
37963efc2edSMatthias Ringwald
btstack_chipset_cc256x_set_init_script(uint8_t * data,uint32_t size)380e2ee49eaSMatthias Ringwald void btstack_chipset_cc256x_set_init_script(uint8_t * data, uint32_t size){
381e2ee49eaSMatthias Ringwald custom_init_script = data;
382e2ee49eaSMatthias Ringwald custom_init_script_size = size;
383e2ee49eaSMatthias Ringwald }
384e2ee49eaSMatthias Ringwald
385faa6c1f6SMatthias Ringwald static const btstack_chipset_t btstack_chipset_cc256x = {
386faa6c1f6SMatthias Ringwald "CC256x",
387faa6c1f6SMatthias Ringwald chipset_init,
388faa6c1f6SMatthias Ringwald chipset_next_command,
389faa6c1f6SMatthias Ringwald chipset_set_baudrate_command,
390e63eef5dSMatthias Ringwald chipset_set_bd_addr_command,
391faa6c1f6SMatthias Ringwald };
392faa6c1f6SMatthias Ringwald
btstack_chipset_cc256x_instance(void)393faa6c1f6SMatthias Ringwald const btstack_chipset_t * btstack_chipset_cc256x_instance(void){
394faa6c1f6SMatthias Ringwald return &btstack_chipset_cc256x;
395faa6c1f6SMatthias Ringwald }
396faa6c1f6SMatthias Ringwald
397