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