xref: /btstack/src/classic/hfp_msbc.c (revision 6535961a1f105e850ecdf294c128b9fc982ee05a)
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