15b4ff0f7SMilanka Ringwald /* 25b4ff0f7SMilanka Ringwald * Copyright (C) 2014 BlueKitchen GmbH 35b4ff0f7SMilanka Ringwald * 45b4ff0f7SMilanka Ringwald * Redistribution and use in source and binary forms, with or without 55b4ff0f7SMilanka Ringwald * modification, are permitted provided that the following conditions 65b4ff0f7SMilanka Ringwald * are met: 75b4ff0f7SMilanka Ringwald * 85b4ff0f7SMilanka Ringwald * 1. Redistributions of source code must retain the above copyright 95b4ff0f7SMilanka Ringwald * notice, this list of conditions and the following disclaimer. 105b4ff0f7SMilanka Ringwald * 2. Redistributions in binary form must reproduce the above copyright 115b4ff0f7SMilanka Ringwald * notice, this list of conditions and the following disclaimer in the 125b4ff0f7SMilanka Ringwald * documentation and/or other materials provided with the distribution. 135b4ff0f7SMilanka Ringwald * 3. Neither the name of the copyright holders nor the names of 145b4ff0f7SMilanka Ringwald * contributors may be used to endorse or promote products derived 155b4ff0f7SMilanka Ringwald * from this software without specific prior written permission. 165b4ff0f7SMilanka Ringwald * 4. Any redistribution, use, or modification is done solely for 175b4ff0f7SMilanka Ringwald * personal benefit and not for any commercial purpose or for 185b4ff0f7SMilanka Ringwald * monetary gain. 195b4ff0f7SMilanka Ringwald * 205b4ff0f7SMilanka Ringwald * THIS SOFTWARE IS PROVIDED BY BLUEKITCHEN GMBH AND CONTRIBUTORS 215b4ff0f7SMilanka Ringwald * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT 225b4ff0f7SMilanka Ringwald * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS 235b4ff0f7SMilanka Ringwald * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL MATTHIAS 245b4ff0f7SMilanka Ringwald * RINGWALD OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, 255b4ff0f7SMilanka Ringwald * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, 265b4ff0f7SMilanka Ringwald * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS 275b4ff0f7SMilanka Ringwald * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED 285b4ff0f7SMilanka Ringwald * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, 295b4ff0f7SMilanka Ringwald * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF 305b4ff0f7SMilanka Ringwald * THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 315b4ff0f7SMilanka Ringwald * SUCH DAMAGE. 325b4ff0f7SMilanka Ringwald * 335b4ff0f7SMilanka Ringwald * Please inquire about commercial licensing options at 345b4ff0f7SMilanka Ringwald * [email protected] 355b4ff0f7SMilanka Ringwald * 365b4ff0f7SMilanka Ringwald */ 375b4ff0f7SMilanka Ringwald 385b4ff0f7SMilanka Ringwald // ***************************************************************************** 395b4ff0f7SMilanka Ringwald // 405b4ff0f7SMilanka Ringwald // HFP mSBC encoder wrapper 415b4ff0f7SMilanka Ringwald // 425b4ff0f7SMilanka Ringwald // ***************************************************************************** 435b4ff0f7SMilanka Ringwald 445b4ff0f7SMilanka Ringwald #include "btstack_config.h" 455b4ff0f7SMilanka Ringwald 465b4ff0f7SMilanka Ringwald #include <string.h> 475b4ff0f7SMilanka Ringwald 485b4ff0f7SMilanka Ringwald #include "btstack_debug.h" 490c87db9eSMilanka Ringwald #include "btstack_sbc.h" 505b4ff0f7SMilanka Ringwald #include "hfp_msbc.h" 515b4ff0f7SMilanka Ringwald 525b4ff0f7SMilanka Ringwald #define MSBC_FRAME_SIZE 57 53469fc11aSMatthias Ringwald #define MSBC_HEADER_H2_SIZE 2 54469fc11aSMatthias Ringwald #define MSBC_PADDING_SIZE 1 55469fc11aSMatthias Ringwald #define MSBC_EXTRA_SIZE (MSBC_HEADER_H2_SIZE + MSBC_PADDING_SIZE) 56469fc11aSMatthias Ringwald 57469fc11aSMatthias Ringwald static const uint8_t msbc_header_h2_byte_0 = 1; 58469fc11aSMatthias Ringwald static const uint8_t msbc_header_h2_byte_1_table[] = { 0x08, 0x38, 0xc8, 0xf8 }; 595b4ff0f7SMilanka Ringwald 605b4ff0f7SMilanka Ringwald static btstack_sbc_encoder_state_t state; 61469fc11aSMatthias Ringwald static int msbc_sequence_number; 625b4ff0f7SMilanka Ringwald 63469fc11aSMatthias Ringwald static uint8_t msbc_buffer[2*(MSBC_FRAME_SIZE + MSBC_EXTRA_SIZE)]; 645b4ff0f7SMilanka Ringwald static int msbc_buffer_offset = 0; 655b4ff0f7SMilanka Ringwald 665b4ff0f7SMilanka Ringwald void hfp_msbc_init(void){ 675b4ff0f7SMilanka Ringwald btstack_sbc_encoder_init(&state, SBC_MODE_mSBC, 16, 8, 0, 16000, 26); 685b4ff0f7SMilanka Ringwald msbc_buffer_offset = 0; 69469fc11aSMatthias Ringwald msbc_sequence_number = 0; 705b4ff0f7SMilanka Ringwald } 715b4ff0f7SMilanka Ringwald 725b4ff0f7SMilanka Ringwald int hfp_msbc_can_encode_audio_frame_now(void){ 73469fc11aSMatthias Ringwald return sizeof(msbc_buffer) - msbc_buffer_offset >= MSBC_FRAME_SIZE + MSBC_EXTRA_SIZE; 745b4ff0f7SMilanka Ringwald } 755b4ff0f7SMilanka Ringwald 765b4ff0f7SMilanka Ringwald void hfp_msbc_encode_audio_frame(int16_t * pcm_samples){ 775b4ff0f7SMilanka Ringwald if (!hfp_msbc_can_encode_audio_frame_now()) return; 785b4ff0f7SMilanka Ringwald 79469fc11aSMatthias Ringwald // Synchronization Header H2 80469fc11aSMatthias Ringwald msbc_buffer[msbc_buffer_offset++] = msbc_header_h2_byte_0; 81469fc11aSMatthias Ringwald msbc_buffer[msbc_buffer_offset++] = msbc_header_h2_byte_1_table[msbc_sequence_number]; 82469fc11aSMatthias Ringwald msbc_sequence_number = (msbc_sequence_number + 1) & 3; 83469fc11aSMatthias Ringwald 84469fc11aSMatthias Ringwald // SBC Frame 855b4ff0f7SMilanka Ringwald btstack_sbc_encoder_process_data(pcm_samples); 865b4ff0f7SMilanka Ringwald memcpy(msbc_buffer + msbc_buffer_offset, btstack_sbc_encoder_sbc_buffer(), MSBC_FRAME_SIZE); 875b4ff0f7SMilanka Ringwald msbc_buffer_offset += MSBC_FRAME_SIZE; 88469fc11aSMatthias Ringwald 89469fc11aSMatthias Ringwald // Final padding to use 60 bytes for 120 audio samples 90469fc11aSMatthias Ringwald msbc_buffer[msbc_buffer_offset++] = 0; 915b4ff0f7SMilanka Ringwald } 925b4ff0f7SMilanka Ringwald 935b4ff0f7SMilanka Ringwald void hfp_msbc_read_from_stream(uint8_t * buf, int size){ 945b4ff0f7SMilanka Ringwald int bytes_to_copy = size; 955b4ff0f7SMilanka Ringwald if (size > msbc_buffer_offset){ 965b4ff0f7SMilanka Ringwald bytes_to_copy = msbc_buffer_offset; 975b4ff0f7SMilanka Ringwald log_error("sbc frame storage is smaller then the output buffer"); 985b4ff0f7SMilanka Ringwald return; 995b4ff0f7SMilanka Ringwald } 1005b4ff0f7SMilanka Ringwald 1015b4ff0f7SMilanka Ringwald memcpy(buf, msbc_buffer, bytes_to_copy); 1025b4ff0f7SMilanka Ringwald memmove(msbc_buffer, msbc_buffer + bytes_to_copy, sizeof(msbc_buffer) - bytes_to_copy); 1035b4ff0f7SMilanka Ringwald msbc_buffer_offset -= bytes_to_copy; 1045b4ff0f7SMilanka Ringwald } 1055b4ff0f7SMilanka Ringwald 1065b4ff0f7SMilanka Ringwald int hfp_msbc_num_bytes_in_stream(void){ 1075b4ff0f7SMilanka Ringwald return msbc_buffer_offset; 1085b4ff0f7SMilanka Ringwald } 1095b4ff0f7SMilanka Ringwald 1105b4ff0f7SMilanka Ringwald int hfp_msbc_num_audio_samples_per_frame(void){ 111*747ec646SMilanka Ringwald return btstack_sbc_encoder_num_audio_frames(); 1125b4ff0f7SMilanka Ringwald } 1135b4ff0f7SMilanka Ringwald 1145b4ff0f7SMilanka Ringwald 115