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 */ 37ab2c6ae4SMatthias Ringwald 38e501bae0SMatthias Ringwald #define BTSTACK_FILE__ "hfp_msbc.c" 395b4ff0f7SMilanka Ringwald 405b4ff0f7SMilanka Ringwald // ***************************************************************************** 415b4ff0f7SMilanka Ringwald // 425b4ff0f7SMilanka Ringwald // HFP mSBC encoder wrapper 435b4ff0f7SMilanka Ringwald // 445b4ff0f7SMilanka Ringwald // ***************************************************************************** 455b4ff0f7SMilanka Ringwald 465b4ff0f7SMilanka Ringwald #include "btstack_config.h" 475b4ff0f7SMilanka Ringwald 485b4ff0f7SMilanka Ringwald #include <string.h> 495b4ff0f7SMilanka Ringwald 505b4ff0f7SMilanka Ringwald #include "btstack_debug.h" 510c87db9eSMilanka Ringwald #include "btstack_sbc.h" 525b4ff0f7SMilanka Ringwald #include "hfp_msbc.h" 535b4ff0f7SMilanka Ringwald 545b4ff0f7SMilanka Ringwald #define MSBC_FRAME_SIZE 57 55469fc11aSMatthias Ringwald #define MSBC_HEADER_H2_SIZE 2 56469fc11aSMatthias Ringwald #define MSBC_PADDING_SIZE 1 57469fc11aSMatthias Ringwald #define MSBC_EXTRA_SIZE (MSBC_HEADER_H2_SIZE + MSBC_PADDING_SIZE) 58469fc11aSMatthias Ringwald 59469fc11aSMatthias Ringwald static const uint8_t msbc_header_h2_byte_0 = 1; 60469fc11aSMatthias Ringwald static const uint8_t msbc_header_h2_byte_1_table[] = { 0x08, 0x38, 0xc8, 0xf8 }; 615b4ff0f7SMilanka Ringwald 625b4ff0f7SMilanka Ringwald static btstack_sbc_encoder_state_t state; 63469fc11aSMatthias Ringwald static int msbc_sequence_number; 645b4ff0f7SMilanka Ringwald 65469fc11aSMatthias Ringwald static uint8_t msbc_buffer[2*(MSBC_FRAME_SIZE + MSBC_EXTRA_SIZE)]; 665b4ff0f7SMilanka Ringwald static int msbc_buffer_offset = 0; 675b4ff0f7SMilanka Ringwald 685b4ff0f7SMilanka Ringwald void hfp_msbc_init(void){ 6991a08b11SMatthias Ringwald btstack_sbc_encoder_init(&state, SBC_MODE_mSBC, 16, 8, 0, 16000, 26, 0); 705b4ff0f7SMilanka Ringwald msbc_buffer_offset = 0; 71469fc11aSMatthias Ringwald msbc_sequence_number = 0; 725b4ff0f7SMilanka Ringwald } 735b4ff0f7SMilanka Ringwald 745b4ff0f7SMilanka Ringwald int hfp_msbc_can_encode_audio_frame_now(void){ 75505f1c30SMatthias Ringwald return (sizeof(msbc_buffer) - msbc_buffer_offset) >= (MSBC_FRAME_SIZE + MSBC_EXTRA_SIZE); 765b4ff0f7SMilanka Ringwald } 775b4ff0f7SMilanka Ringwald 785b4ff0f7SMilanka Ringwald void hfp_msbc_encode_audio_frame(int16_t * pcm_samples){ 795b4ff0f7SMilanka Ringwald if (!hfp_msbc_can_encode_audio_frame_now()) return; 805b4ff0f7SMilanka Ringwald 81469fc11aSMatthias Ringwald // Synchronization Header H2 82469fc11aSMatthias Ringwald msbc_buffer[msbc_buffer_offset++] = msbc_header_h2_byte_0; 83469fc11aSMatthias Ringwald msbc_buffer[msbc_buffer_offset++] = msbc_header_h2_byte_1_table[msbc_sequence_number]; 84469fc11aSMatthias Ringwald msbc_sequence_number = (msbc_sequence_number + 1) & 3; 85469fc11aSMatthias Ringwald 86469fc11aSMatthias Ringwald // SBC Frame 875b4ff0f7SMilanka Ringwald btstack_sbc_encoder_process_data(pcm_samples); 88*6535961aSMatthias Ringwald (void)memcpy(msbc_buffer + msbc_buffer_offset, 89*6535961aSMatthias Ringwald btstack_sbc_encoder_sbc_buffer(), MSBC_FRAME_SIZE); 905b4ff0f7SMilanka Ringwald msbc_buffer_offset += MSBC_FRAME_SIZE; 91469fc11aSMatthias Ringwald 92469fc11aSMatthias Ringwald // Final padding to use 60 bytes for 120 audio samples 93469fc11aSMatthias Ringwald msbc_buffer[msbc_buffer_offset++] = 0; 945b4ff0f7SMilanka Ringwald } 955b4ff0f7SMilanka Ringwald 965b4ff0f7SMilanka Ringwald void hfp_msbc_read_from_stream(uint8_t * buf, int size){ 975b4ff0f7SMilanka Ringwald int bytes_to_copy = size; 985b4ff0f7SMilanka Ringwald if (size > msbc_buffer_offset){ 995b4ff0f7SMilanka Ringwald bytes_to_copy = msbc_buffer_offset; 1005b4ff0f7SMilanka Ringwald log_error("sbc frame storage is smaller then the output buffer"); 1015b4ff0f7SMilanka Ringwald return; 1025b4ff0f7SMilanka Ringwald } 1035b4ff0f7SMilanka Ringwald 104*6535961aSMatthias Ringwald (void)memcpy(buf, msbc_buffer, bytes_to_copy); 1055b4ff0f7SMilanka Ringwald memmove(msbc_buffer, msbc_buffer + bytes_to_copy, sizeof(msbc_buffer) - bytes_to_copy); 1065b4ff0f7SMilanka Ringwald msbc_buffer_offset -= bytes_to_copy; 1075b4ff0f7SMilanka Ringwald } 1085b4ff0f7SMilanka Ringwald 1095b4ff0f7SMilanka Ringwald int hfp_msbc_num_bytes_in_stream(void){ 1105b4ff0f7SMilanka Ringwald return msbc_buffer_offset; 1115b4ff0f7SMilanka Ringwald } 1125b4ff0f7SMilanka Ringwald 1135b4ff0f7SMilanka Ringwald int hfp_msbc_num_audio_samples_per_frame(void){ 114747ec646SMilanka Ringwald return btstack_sbc_encoder_num_audio_frames(); 1155b4ff0f7SMilanka Ringwald } 1165b4ff0f7SMilanka Ringwald 1175b4ff0f7SMilanka Ringwald 118