xref: /btstack/src/classic/hfp_msbc.c (revision e501bae08349e058caa4648e0af3dd01cbd89d20)
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 
38*e501bae0SMatthias 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){
75469fc11aSMatthias 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);
885b4ff0f7SMilanka Ringwald     memcpy(msbc_buffer + msbc_buffer_offset, btstack_sbc_encoder_sbc_buffer(), MSBC_FRAME_SIZE);
895b4ff0f7SMilanka Ringwald     msbc_buffer_offset += MSBC_FRAME_SIZE;
90469fc11aSMatthias Ringwald 
91469fc11aSMatthias Ringwald     // Final padding to use 60 bytes for 120 audio samples
92469fc11aSMatthias Ringwald     msbc_buffer[msbc_buffer_offset++] = 0;
935b4ff0f7SMilanka Ringwald }
945b4ff0f7SMilanka Ringwald 
955b4ff0f7SMilanka Ringwald void hfp_msbc_read_from_stream(uint8_t * buf, int size){
965b4ff0f7SMilanka Ringwald     int bytes_to_copy = size;
975b4ff0f7SMilanka Ringwald     if (size > msbc_buffer_offset){
985b4ff0f7SMilanka Ringwald         bytes_to_copy = msbc_buffer_offset;
995b4ff0f7SMilanka Ringwald         log_error("sbc frame storage is smaller then the output buffer");
1005b4ff0f7SMilanka Ringwald         return;
1015b4ff0f7SMilanka Ringwald     }
1025b4ff0f7SMilanka Ringwald 
1035b4ff0f7SMilanka Ringwald     memcpy(buf, msbc_buffer, bytes_to_copy);
1045b4ff0f7SMilanka Ringwald     memmove(msbc_buffer, msbc_buffer + bytes_to_copy, sizeof(msbc_buffer) - bytes_to_copy);
1055b4ff0f7SMilanka Ringwald     msbc_buffer_offset -= bytes_to_copy;
1065b4ff0f7SMilanka Ringwald }
1075b4ff0f7SMilanka Ringwald 
1085b4ff0f7SMilanka Ringwald int hfp_msbc_num_bytes_in_stream(void){
1095b4ff0f7SMilanka Ringwald     return msbc_buffer_offset;
1105b4ff0f7SMilanka Ringwald }
1115b4ff0f7SMilanka Ringwald 
1125b4ff0f7SMilanka Ringwald int hfp_msbc_num_audio_samples_per_frame(void){
113747ec646SMilanka Ringwald     return btstack_sbc_encoder_num_audio_frames();
1145b4ff0f7SMilanka Ringwald }
1155b4ff0f7SMilanka Ringwald 
1165b4ff0f7SMilanka Ringwald 
117