| /****************************************************************************** |
| * |
| * Copyright (C) 2003-2012 Broadcom Corporation |
| * |
| * Licensed under the Apache License, Version 2.0 (the "License"); |
| * you may not use this file except in compliance with the License. |
| * You may obtain a copy of the License at: |
| * |
| * http://www.apache.org/licenses/LICENSE-2.0 |
| * |
| * Unless required by applicable law or agreed to in writing, software |
| * distributed under the License is distributed on an "AS IS" BASIS, |
| * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| * See the License for the specific language governing permissions and |
| * limitations under the License. |
| * |
| ******************************************************************************/ |
| |
| /****************************************************************************** |
| * |
| * This file contains the down sampling utility to convert PCM samples in |
| * 16k/32k/48k/44.1k/22050/11025 sampling rate into 8K/16bits samples |
| * required for SCO channel format. One API function isprovided and only |
| * possible to be used when transmitting SCO data is sent via HCI |
| * interface. |
| * |
| ******************************************************************************/ |
| #include <string.h> |
| |
| #include "bta_api.h" |
| #include "bta_sys.h" |
| |
| #if (BTM_SCO_HCI_INCLUDED == TRUE) |
| |
| #ifndef BTA_DM_SCO_DEBUG |
| #define BTA_DM_SCO_DEBUG false |
| #endif |
| /***************************************************************************** |
| * Constants |
| ****************************************************************************/ |
| |
| #define BTA_DM_PCM_OVERLAP_SIZE 48 |
| |
| #define BTA_DM_PCM_SMPL_RATE_44100 44100 |
| #define BTA_DM_PCM_SMPL_RATE_22050 22050 |
| #define BTA_DM_PCM_SMPL_RATE_11025 11025 |
| |
| /***************************************************************************** |
| * Data types for PCM Resampling utility |
| ****************************************************************************/ |
| |
| typedef int32_t (*PCONVERT_TO_BT_FILTERED)(uint8_t* pSrc, void* pDst, |
| uint32_t dwSrcSamples, |
| uint32_t dwSrcSps, |
| int32_t* pLastCurPos, |
| uint8_t* pOverlapArea); |
| typedef int32_t (*PCONVERT_TO_BT_NOFILTER)(void* pSrc, void* pDst, |
| uint32_t dwSrcSamples, |
| uint32_t dwSrcSps); |
| typedef struct { |
| uint8_t overlap_area[BTA_DM_PCM_OVERLAP_SIZE * 4]; |
| uint32_t cur_pos; /* current position */ |
| uint32_t src_sps; /* samples per second (source audio data) */ |
| PCONVERT_TO_BT_FILTERED filter; /* the action function to do the |
| conversion 44100, 22050, 11025*/ |
| PCONVERT_TO_BT_NOFILTER nofilter; /* the action function to do |
| the conversion 48000, 32000, 16000*/ |
| uint32_t bits; /* number of bits per pcm sample */ |
| uint32_t n_channels; /* number of channels (i.e. mono(1), stereo(2)...) */ |
| uint32_t sample_size; |
| uint32_t can_be_filtered; |
| uint32_t divisor; |
| } tBTA_DM_PCM_RESAMPLE_CB; |
| |
| tBTA_DM_PCM_RESAMPLE_CB bta_dm_pcm_cb; |
| |
| /***************************************************************************** |
| * Macro Definition |
| ****************************************************************************/ |
| |
| #define CHECK_SATURATION16(x) \ |
| do { \ |
| if ((x) > 32767) \ |
| (x) = 32767; \ |
| else if ((x) < -32768) \ |
| (x) = -32768; \ |
| } while (0) |
| |
| //////////////////////////////////////////////////////////////////////////////// |
| // |
| #define CONVERT_44100_TO_BLUETOOTH(pStart, pEnd) \ |
| do { \ |
| int32_t out1, out2, out3, out4, out5; \ |
| SRC_TYPE* pS = (SRC_TYPE*)(pStart); \ |
| SRC_TYPE* pSEnd = (SRC_TYPE*)(pEnd); \ |
| \ |
| while (pS < pSEnd) { \ |
| CurrentPos -= 8000; \ |
| \ |
| if (CurrentPos >= 0) { \ |
| pS += SRC_CHANNELS; \ |
| continue; \ |
| } \ |
| CurrentPos += dwSrcSps; \ |
| \ |
| out1 = (SRC_SAMPLE(0) * 1587) + \ |
| ((SRC_SAMPLE(1) + SRC_SAMPLE(-1)) * 1522) + \ |
| ((SRC_SAMPLE(2) + SRC_SAMPLE(-2)) * 1337) + \ |
| ((SRC_SAMPLE(3) + SRC_SAMPLE(-3)) * 1058); \ |
| \ |
| out1 = out1 / 30000; \ |
| \ |
| out2 = ((SRC_SAMPLE(4) + SRC_SAMPLE(-4)) * 725) + \ |
| ((SRC_SAMPLE(5) + SRC_SAMPLE(-5)) * 384) + \ |
| ((SRC_SAMPLE(6) + SRC_SAMPLE(-6)) * 79); \ |
| \ |
| out2 = out2 / 30000; \ |
| \ |
| out3 = ((SRC_SAMPLE(7) + SRC_SAMPLE(-7)) * 156) + \ |
| ((SRC_SAMPLE(8) + SRC_SAMPLE(-8)) * 298) + \ |
| ((SRC_SAMPLE(9) + SRC_SAMPLE(-9)) * 345); \ |
| \ |
| out3 = out3 / 30000; \ |
| \ |
| out4 = ((SRC_SAMPLE(10) + SRC_SAMPLE(-10)) * 306) + \ |
| ((SRC_SAMPLE(11) + SRC_SAMPLE(-11)) * 207) + \ |
| ((SRC_SAMPLE(12) + SRC_SAMPLE(-12)) * 78); \ |
| \ |
| out4 = out4 / 30000; \ |
| \ |
| out5 = out1 + out2 - out3 - out4; \ |
| \ |
| CHECK_SATURATION16(out5); \ |
| *psBtOut++ = (int16_t)out5; \ |
| \ |
| pS += SRC_CHANNELS; \ |
| } \ |
| } while (0) |
| |
| //////////////////////////////////////////////////////////////////////////////// |
| // |
| #define CONVERT_22050_TO_BLUETOOTH(pStart, pEnd) \ |
| do { \ |
| int32_t out1, out2, out3, out4, out5; \ |
| SRC_TYPE* pS = (SRC_TYPE*)(pStart); \ |
| SRC_TYPE* pSEnd = (SRC_TYPE*)(pEnd); \ |
| \ |
| while (pS < pSEnd) { \ |
| CurrentPos -= 8000; \ |
| \ |
| if (CurrentPos >= 0) { \ |
| pS += SRC_CHANNELS; \ |
| continue; \ |
| } \ |
| CurrentPos += dwSrcSps; \ |
| \ |
| out1 = (SRC_SAMPLE(0) * 2993) + \ |
| ((SRC_SAMPLE(1) + SRC_SAMPLE(-1)) * 2568) + \ |
| ((SRC_SAMPLE(2) + SRC_SAMPLE(-2)) * 1509) + \ |
| ((SRC_SAMPLE(3) + SRC_SAMPLE(-3)) * 331); \ |
| \ |
| out1 = out1 / 30000; \ |
| \ |
| out2 = ((SRC_SAMPLE(4) + SRC_SAMPLE(-4)) * 454) + \ |
| ((SRC_SAMPLE(5) + SRC_SAMPLE(-5)) * 620) + \ |
| ((SRC_SAMPLE(6) + SRC_SAMPLE(-6)) * 305); \ |
| \ |
| out2 = out2 / 30000; \ |
| \ |
| out3 = ((SRC_SAMPLE(7) + SRC_SAMPLE(-7)) * 127) + \ |
| ((SRC_SAMPLE(8) + SRC_SAMPLE(-8)) * 350) + \ |
| ((SRC_SAMPLE(9) + SRC_SAMPLE(-9)) * 265) + \ |
| ((SRC_SAMPLE(10) + SRC_SAMPLE(-10)) * 6); \ |
| \ |
| out3 = out3 / 30000; \ |
| \ |
| out4 = ((SRC_SAMPLE(11) + SRC_SAMPLE(-11)) * 201); \ |
| \ |
| out4 = out4 / 30000; \ |
| \ |
| out5 = out1 - out2 + out3 - out4; \ |
| \ |
| CHECK_SATURATION16(out5); \ |
| *psBtOut++ = (int16_t)out5; \ |
| \ |
| pS += SRC_CHANNELS; \ |
| } \ |
| } while (0) |
| |
| //////////////////////////////////////////////////////////////////////////////// |
| // |
| #define CONVERT_11025_TO_BLUETOOTH(pStart, pEnd) \ |
| do { \ |
| int32_t out1; \ |
| SRC_TYPE* pS = (SRC_TYPE*)(pStart); \ |
| SRC_TYPE* pSEnd = (SRC_TYPE*)(pEnd); \ |
| \ |
| while (pS < pSEnd) { \ |
| CurrentPos -= 8000; \ |
| \ |
| if (CurrentPos >= 0) { \ |
| pS += SRC_CHANNELS; \ |
| continue; \ |
| } \ |
| CurrentPos += dwSrcSps; \ |
| \ |
| out1 = (SRC_SAMPLE(0) * 6349) + \ |
| ((SRC_SAMPLE(1) + SRC_SAMPLE(-1)) * 2874) - \ |
| ((SRC_SAMPLE(2) + SRC_SAMPLE(-2)) * 1148) - \ |
| ((SRC_SAMPLE(3) + SRC_SAMPLE(-3)) * 287) + \ |
| ((SRC_SAMPLE(4) + SRC_SAMPLE(-4)) * 675) - \ |
| ((SRC_SAMPLE(5) + SRC_SAMPLE(-5)) * 258) - \ |
| ((SRC_SAMPLE(6) + SRC_SAMPLE(-6)) * 206) + \ |
| ((SRC_SAMPLE(7) + SRC_SAMPLE(-7)) * 266); \ |
| \ |
| out1 = out1 / 30000; \ |
| \ |
| CHECK_SATURATION16(out1); \ |
| *psBtOut++ = (int16_t)out1; \ |
| \ |
| pS += SRC_CHANNELS; \ |
| } \ |
| } while (0) |
| |
| //////////////////////////////////////////////////////////////////////////////// |
| // |
| #undef SRC_CHANNELS |
| #undef SRC_SAMPLE |
| #undef SRC_TYPE |
| |
| #define SRC_TYPE uint8_t |
| #define SRC_CHANNELS 1 |
| #define SRC_SAMPLE(x) ((pS[x] - 0x80) << 8) |
| |
| /***************************************************************************** |
| * Local Function |
| ****************************************************************************/ |
| int32_t Convert_8M_ToBT_Filtered(uint8_t* pSrc, void* pDst, |
| uint32_t dwSrcSamples, uint32_t dwSrcSps, |
| int32_t* pLastCurPos, uint8_t* pOverlapArea) { |
| int32_t CurrentPos = *pLastCurPos; |
| SRC_TYPE *pIn, *pInEnd; |
| SRC_TYPE *pOv, *pOvEnd; |
| int16_t* psBtOut = (int16_t*)pDst; |
| #if (BTA_DM_SCO_DEBUG == TRUE) |
| APPL_TRACE_DEBUG("Convert_8M_ToBT_Filtered, CurrentPos %d\n", CurrentPos); |
| #endif |
| memcpy(pOverlapArea + (BTA_DM_PCM_OVERLAP_SIZE * 2), pSrc, |
| BTA_DM_PCM_OVERLAP_SIZE * 2); |
| |
| pOv = (SRC_TYPE*)(pOverlapArea + BTA_DM_PCM_OVERLAP_SIZE); |
| pOvEnd = (SRC_TYPE*)(pOverlapArea + (BTA_DM_PCM_OVERLAP_SIZE * 3)); |
| |
| pIn = (SRC_TYPE*)(pSrc + BTA_DM_PCM_OVERLAP_SIZE); |
| pInEnd = (SRC_TYPE*)(pSrc + (dwSrcSamples * SRC_CHANNELS * sizeof(SRC_TYPE)) - |
| BTA_DM_PCM_OVERLAP_SIZE); |
| |
| if (dwSrcSps == BTA_DM_PCM_SMPL_RATE_44100) { |
| CONVERT_44100_TO_BLUETOOTH(pOv, pOvEnd); |
| CONVERT_44100_TO_BLUETOOTH(pIn, pInEnd); |
| } else if (dwSrcSps == BTA_DM_PCM_SMPL_RATE_22050) { |
| CONVERT_22050_TO_BLUETOOTH(pOv, pOvEnd); |
| CONVERT_22050_TO_BLUETOOTH(pIn, pInEnd); |
| } else if (dwSrcSps == BTA_DM_PCM_SMPL_RATE_11025) { |
| CONVERT_11025_TO_BLUETOOTH(pOv, pOvEnd); |
| CONVERT_11025_TO_BLUETOOTH(pIn, pInEnd); |
| } |
| |
| memcpy(pOverlapArea, pSrc + (dwSrcSamples * SRC_CHANNELS * sizeof(SRC_TYPE)) - |
| (BTA_DM_PCM_OVERLAP_SIZE * 2), |
| BTA_DM_PCM_OVERLAP_SIZE * 2); |
| |
| *pLastCurPos = CurrentPos; |
| |
| return (psBtOut - (int16_t*)pDst); |
| } |
| |
| int32_t Convert_8M_ToBT_NoFilter(void* pSrc, void* pDst, uint32_t dwSrcSamples, |
| uint32_t dwSrcSps) { |
| int32_t CurrentPos; |
| uint8_t* pbSrc = (uint8_t*)pSrc; |
| int16_t* psDst = (int16_t*)pDst; |
| int16_t sWorker; |
| |
| // start at dwSpsSrc / 2, decrement by 8000 |
| // |
| CurrentPos = (dwSrcSps >> 1); |
| |
| while (dwSrcSamples--) { |
| CurrentPos -= 8000; |
| |
| if (CurrentPos >= 0) |
| pbSrc++; |
| else { |
| sWorker = *pbSrc++; |
| sWorker -= 0x80; |
| sWorker <<= 8; |
| |
| *psDst++ = sWorker; |
| |
| CurrentPos += dwSrcSps; |
| } |
| } |
| |
| return (psDst - (int16_t*)pDst); |
| } |
| |
| //////////////////////////////////////////////////////////////////////////////// |
| // |
| #undef SRC_CHANNELS |
| #undef SRC_SAMPLE |
| #undef SRC_TYPE |
| |
| #define SRC_TYPE int16_t |
| #define SRC_CHANNELS 1 |
| #define SRC_SAMPLE(x) pS[x] |
| |
| int32_t Convert_16M_ToBT_Filtered(uint8_t* pSrc, void* pDst, |
| uint32_t dwSrcSamples, uint32_t dwSrcSps, |
| int32_t* pLastCurPos, uint8_t* pOverlapArea) { |
| int32_t CurrentPos = *pLastCurPos; |
| SRC_TYPE *pIn, *pInEnd; |
| SRC_TYPE *pOv, *pOvEnd; |
| int16_t* psBtOut = (int16_t*)pDst; |
| |
| memcpy(pOverlapArea + (BTA_DM_PCM_OVERLAP_SIZE * 2), pSrc, |
| BTA_DM_PCM_OVERLAP_SIZE * 2); |
| |
| pOv = (SRC_TYPE*)(pOverlapArea + BTA_DM_PCM_OVERLAP_SIZE); |
| pOvEnd = (SRC_TYPE*)(pOverlapArea + (BTA_DM_PCM_OVERLAP_SIZE * 3)); |
| |
| pIn = (SRC_TYPE*)(pSrc + BTA_DM_PCM_OVERLAP_SIZE); |
| pInEnd = (SRC_TYPE*)(pSrc + (dwSrcSamples * SRC_CHANNELS * sizeof(SRC_TYPE)) - |
| BTA_DM_PCM_OVERLAP_SIZE); |
| |
| if (dwSrcSps == BTA_DM_PCM_SMPL_RATE_44100) { |
| CONVERT_44100_TO_BLUETOOTH(pOv, pOvEnd); |
| CONVERT_44100_TO_BLUETOOTH(pIn, pInEnd); |
| } else if (dwSrcSps == BTA_DM_PCM_SMPL_RATE_22050) { |
| CONVERT_22050_TO_BLUETOOTH(pOv, pOvEnd); |
| CONVERT_22050_TO_BLUETOOTH(pIn, pInEnd); |
| } else if (dwSrcSps == BTA_DM_PCM_SMPL_RATE_11025) { |
| CONVERT_11025_TO_BLUETOOTH(pOv, pOvEnd); |
| CONVERT_11025_TO_BLUETOOTH(pIn, pInEnd); |
| } |
| |
| memcpy(pOverlapArea, pSrc + (dwSrcSamples * SRC_CHANNELS * sizeof(SRC_TYPE)) - |
| (BTA_DM_PCM_OVERLAP_SIZE * 2), |
| BTA_DM_PCM_OVERLAP_SIZE * 2); |
| |
| *pLastCurPos = CurrentPos; |
| |
| return (psBtOut - (int16_t*)pDst); |
| } |
| |
| int32_t Convert_16M_ToBT_NoFilter(void* pSrc, void* pDst, uint32_t dwSrcSamples, |
| uint32_t dwSrcSps) { |
| int32_t CurrentPos; |
| int16_t* psSrc = (int16_t*)pSrc; |
| int16_t* psDst = (int16_t*)pDst; |
| |
| // start at dwSpsSrc / 2, decrement by 8000 |
| // |
| CurrentPos = (dwSrcSps >> 1); |
| |
| while (dwSrcSamples--) { |
| CurrentPos -= 8000; |
| |
| if (CurrentPos >= 0) |
| psSrc++; |
| else { |
| *psDst++ = *psSrc++; |
| |
| CurrentPos += dwSrcSps; |
| } |
| } |
| |
| return (psDst - (int16_t*)pDst); |
| } |
| |
| //////////////////////////////////////////////////////////////////////////////// |
| // |
| #undef SRC_CHANNELS |
| #undef SRC_SAMPLE |
| #undef SRC_TYPE |
| |
| #define SRC_TYPE uint8_t |
| #define SRC_CHANNELS 2 |
| #define SRC_SAMPLE(x) \ |
| ((((pS[x * 2] - 0x80) << 8) + ((pS[(x * 2) + 1] - 0x80) << 8)) >> 1) |
| |
| int32_t Convert_8S_ToBT_Filtered(uint8_t* pSrc, void* pDst, |
| uint32_t dwSrcSamples, uint32_t dwSrcSps, |
| int32_t* pLastCurPos, uint8_t* pOverlapArea) { |
| int32_t CurrentPos = *pLastCurPos; |
| SRC_TYPE *pIn, *pInEnd; |
| SRC_TYPE *pOv, *pOvEnd; |
| int16_t* psBtOut = (int16_t*)pDst; |
| |
| #if (BTA_DM_SCO_DEBUG == TRUE) |
| APPL_TRACE_DEBUG( |
| "Convert_8S_ToBT_Filtered CurrentPos %d, SRC_TYPE %d, SRC_CHANNELS %d, \ |
| dwSrcSamples %d, dwSrcSps %d", |
| CurrentPos, sizeof(SRC_TYPE), SRC_CHANNELS, dwSrcSamples, dwSrcSps); |
| #endif |
| memcpy(pOverlapArea + (BTA_DM_PCM_OVERLAP_SIZE * 2), pSrc, |
| BTA_DM_PCM_OVERLAP_SIZE * 2); |
| |
| pOv = (SRC_TYPE*)(pOverlapArea + BTA_DM_PCM_OVERLAP_SIZE); |
| pOvEnd = (SRC_TYPE*)(pOverlapArea + (BTA_DM_PCM_OVERLAP_SIZE * 3)); |
| |
| pIn = (SRC_TYPE*)(pSrc + BTA_DM_PCM_OVERLAP_SIZE); |
| pInEnd = (SRC_TYPE*)(pSrc + (dwSrcSamples * SRC_CHANNELS * sizeof(SRC_TYPE)) - |
| BTA_DM_PCM_OVERLAP_SIZE); |
| |
| if (dwSrcSps == BTA_DM_PCM_SMPL_RATE_44100) { |
| CONVERT_44100_TO_BLUETOOTH(pOv, pOvEnd); |
| CONVERT_44100_TO_BLUETOOTH(pIn, pInEnd); |
| } else if (dwSrcSps == BTA_DM_PCM_SMPL_RATE_22050) { |
| CONVERT_22050_TO_BLUETOOTH(pOv, pOvEnd); |
| CONVERT_22050_TO_BLUETOOTH(pIn, pInEnd); |
| } else if (dwSrcSps == BTA_DM_PCM_SMPL_RATE_11025) { |
| CONVERT_11025_TO_BLUETOOTH(pOv, pOvEnd); |
| CONVERT_11025_TO_BLUETOOTH(pIn, pInEnd); |
| } |
| |
| memcpy(pOverlapArea, pSrc + (dwSrcSamples * SRC_CHANNELS * sizeof(SRC_TYPE)) - |
| (BTA_DM_PCM_OVERLAP_SIZE * 2), |
| BTA_DM_PCM_OVERLAP_SIZE * 2); |
| |
| *pLastCurPos = CurrentPos; |
| |
| return (psBtOut - (int16_t*)pDst); |
| } |
| |
| int32_t Convert_8S_ToBT_NoFilter(void* pSrc, void* pDst, uint32_t dwSrcSamples, |
| uint32_t dwSrcSps) { |
| int32_t CurrentPos; |
| uint8_t* pbSrc = (uint8_t*)pSrc; |
| int16_t* psDst = (int16_t*)pDst; |
| int16_t sWorker, sWorker2; |
| |
| // start at dwSpsSrc / 2, decrement by 8000 |
| // |
| CurrentPos = (dwSrcSps >> 1); |
| |
| while (dwSrcSamples--) { |
| CurrentPos -= 8000; |
| |
| if (CurrentPos >= 0) |
| pbSrc += 2; |
| else { |
| sWorker = *(unsigned char*)pbSrc; |
| sWorker -= 0x80; |
| sWorker <<= 8; |
| pbSrc++; |
| |
| sWorker2 = *(unsigned char*)pbSrc; |
| sWorker2 -= 0x80; |
| sWorker2 <<= 8; |
| pbSrc++; |
| |
| sWorker += sWorker2; |
| sWorker >>= 1; |
| |
| *psDst++ = sWorker; |
| |
| CurrentPos += dwSrcSps; |
| } |
| } |
| |
| return (psDst - (int16_t*)pDst); |
| } |
| |
| //////////////////////////////////////////////////////////////////////////////// |
| // |
| #undef SRC_CHANNELS |
| #undef SRC_SAMPLE |
| #undef SRC_TYPE |
| |
| #define SRC_TYPE int16_t |
| #define SRC_CHANNELS 2 |
| #define SRC_SAMPLE(x) ((pS[x * 2] + pS[(x * 2) + 1]) >> 1) |
| |
| int32_t Convert_16S_ToBT_Filtered(uint8_t* pSrc, void* pDst, |
| uint32_t dwSrcSamples, uint32_t dwSrcSps, |
| int32_t* pLastCurPos, uint8_t* pOverlapArea) { |
| int32_t CurrentPos = *pLastCurPos; |
| SRC_TYPE *pIn, *pInEnd; |
| SRC_TYPE *pOv, *pOvEnd; |
| int16_t* psBtOut = (int16_t*)pDst; |
| |
| memcpy(pOverlapArea + (BTA_DM_PCM_OVERLAP_SIZE * 2), pSrc, |
| BTA_DM_PCM_OVERLAP_SIZE * 2); |
| |
| pOv = (SRC_TYPE*)(pOverlapArea + BTA_DM_PCM_OVERLAP_SIZE); |
| pOvEnd = (SRC_TYPE*)(pOverlapArea + (BTA_DM_PCM_OVERLAP_SIZE * 3)); |
| |
| pIn = (SRC_TYPE*)(pSrc + BTA_DM_PCM_OVERLAP_SIZE); |
| pInEnd = (SRC_TYPE*)(pSrc + (dwSrcSamples * SRC_CHANNELS * sizeof(SRC_TYPE)) - |
| BTA_DM_PCM_OVERLAP_SIZE); |
| |
| if (dwSrcSps == BTA_DM_PCM_SMPL_RATE_44100) { |
| CONVERT_44100_TO_BLUETOOTH(pOv, pOvEnd); |
| CONVERT_44100_TO_BLUETOOTH(pIn, pInEnd); |
| } else if (dwSrcSps == BTA_DM_PCM_SMPL_RATE_22050) { |
| CONVERT_22050_TO_BLUETOOTH(pOv, pOvEnd); |
| CONVERT_22050_TO_BLUETOOTH(pIn, pInEnd); |
| } else if (dwSrcSps == BTA_DM_PCM_SMPL_RATE_11025) { |
| CONVERT_11025_TO_BLUETOOTH(pOv, pOvEnd); |
| CONVERT_11025_TO_BLUETOOTH(pIn, pInEnd); |
| } |
| |
| memcpy(pOverlapArea, pSrc + (dwSrcSamples * SRC_CHANNELS * sizeof(SRC_TYPE)) - |
| (BTA_DM_PCM_OVERLAP_SIZE * 2), |
| BTA_DM_PCM_OVERLAP_SIZE * 2); |
| |
| *pLastCurPos = CurrentPos; |
| |
| return (psBtOut - (int16_t*)pDst); |
| } |
| |
| int32_t Convert_16S_ToBT_NoFilter(void* pSrc, void* pDst, uint32_t dwSrcSamples, |
| uint32_t dwSrcSps) { |
| int32_t CurrentPos; |
| int16_t* psSrc = (int16_t*)pSrc; |
| int16_t* psDst = (int16_t*)pDst; |
| int16_t sWorker; |
| |
| // start at dwSpsSrc / 2, decrement by 8000 |
| // |
| CurrentPos = (dwSrcSps >> 1); |
| |
| while (dwSrcSamples--) { |
| CurrentPos -= 8000; |
| |
| if (CurrentPos >= 0) |
| psSrc += 2; |
| else { |
| /* CR 82894, to avoid overflow, divide before add */ |
| sWorker = ((*psSrc) >> 1); |
| psSrc++; |
| sWorker += ((*psSrc) >> 1); |
| psSrc++; |
| |
| *psDst++ = sWorker; |
| |
| CurrentPos += dwSrcSps; |
| } |
| } |
| |
| return (psDst - (int16_t*)pDst); |
| } |
| |
| /******************************************************************************* |
| * |
| * Function BTA_DmPcmInitSamples |
| * |
| * Description initialize the down sample converter. |
| * |
| * src_sps: original samples per second (source audio data) |
| * (ex. 44100, 48000) |
| * bits: number of bits per pcm sample (16) |
| * n_channels: number of channels (i.e. mono(1), stereo(2)...) |
| * |
| * Returns none |
| * |
| ******************************************************************************/ |
| void BTA_DmPcmInitSamples(uint32_t src_sps, uint32_t bits, |
| uint32_t n_channels) { |
| tBTA_DM_PCM_RESAMPLE_CB* p_cb = &bta_dm_pcm_cb; |
| |
| p_cb->cur_pos = src_sps / 2; |
| p_cb->src_sps = src_sps; |
| p_cb->bits = bits; |
| p_cb->n_channels = n_channels; |
| p_cb->sample_size = 2; |
| p_cb->divisor = 2; |
| |
| memset(p_cb->overlap_area, 0, sizeof(p_cb->overlap_area)); |
| |
| if ((src_sps == BTA_DM_PCM_SMPL_RATE_44100) || |
| (src_sps == BTA_DM_PCM_SMPL_RATE_22050) || |
| (src_sps == BTA_DM_PCM_SMPL_RATE_11025)) |
| p_cb->can_be_filtered = 1; |
| else |
| p_cb->can_be_filtered = 0; |
| |
| #if (BTA_DM_SCO_DEBUG == TRUE) |
| APPL_TRACE_DEBUG("bta_dm_pcm_init_samples: n_channels = %d bits = %d", |
| n_channels, bits); |
| #endif |
| if (n_channels == 1) { |
| /* mono */ |
| if (bits == 8) { |
| p_cb->filter = (PCONVERT_TO_BT_FILTERED)Convert_8M_ToBT_Filtered; |
| p_cb->nofilter = (PCONVERT_TO_BT_NOFILTER)Convert_8M_ToBT_NoFilter; |
| p_cb->divisor = 1; |
| } else { |
| p_cb->filter = (PCONVERT_TO_BT_FILTERED)Convert_16M_ToBT_Filtered; |
| p_cb->nofilter = (PCONVERT_TO_BT_NOFILTER)Convert_16M_ToBT_NoFilter; |
| } |
| } else { |
| /* stereo */ |
| if (bits == 8) { |
| p_cb->filter = (PCONVERT_TO_BT_FILTERED)Convert_8S_ToBT_Filtered; |
| p_cb->nofilter = (PCONVERT_TO_BT_NOFILTER)Convert_8S_ToBT_NoFilter; |
| } else { |
| p_cb->filter = (PCONVERT_TO_BT_FILTERED)Convert_16S_ToBT_Filtered; |
| p_cb->nofilter = (PCONVERT_TO_BT_NOFILTER)Convert_16S_ToBT_NoFilter; |
| p_cb->divisor = 4; |
| } |
| } |
| |
| #if (BTA_DM_SCO_DEBUG == TRUE) |
| APPL_TRACE_DEBUG("bta_pcm_init_dwn_sample: cur_pos %d, src_sps %d", |
| p_cb->cur_pos, p_cb->src_sps); |
| APPL_TRACE_DEBUG( |
| "bta_pcm_init_dwn_sample: bits %d, n_channels %d, sample_size %d, ", |
| p_cb->bits, p_cb->n_channels, p_cb->sample_size); |
| APPL_TRACE_DEBUG( |
| "bta_pcm_init_dwn_sample: can_be_filtered %d, n_channels: %d, \ |
| divisor %d", |
| p_cb->can_be_filtered, p_cb->n_channels, p_cb->divisor); |
| #endif |
| } |
| |
| /******************************************************************************* |
| * Function BTA_DmPcmResample |
| * |
| * Description Down sampling utility to convert higher sampling rate into |
| * 8K/16bits PCM samples. |
| * |
| * Parameters p_src: pointer to the buffer where the original sampling PCM |
| * are stored. |
| * in_bytes: Length of the input PCM sample buffer in byte. |
| * p_dst: pointer to the buffer which is to be used to |
| * store the converted PCM samples. |
| * |
| * |
| * Returns int32_t: number of samples converted. |
| * |
| ******************************************************************************/ |
| int32_t BTA_DmPcmResample(void* p_src, uint32_t in_bytes, void* p_dst) { |
| uint32_t out_sample; |
| |
| #if (BTA_DM_SCO_DEBUG == TRUE) |
| APPL_TRACE_DEBUG("bta_pcm_resample : insamples %d", |
| (in_bytes / bta_dm_pcm_cb.divisor)); |
| #endif |
| if (bta_dm_pcm_cb.can_be_filtered) { |
| out_sample = (*bta_dm_pcm_cb.filter)( |
| p_src, p_dst, (in_bytes / bta_dm_pcm_cb.divisor), bta_dm_pcm_cb.src_sps, |
| (int32_t*)&bta_dm_pcm_cb.cur_pos, bta_dm_pcm_cb.overlap_area); |
| } else { |
| out_sample = (*bta_dm_pcm_cb.nofilter)(p_src, p_dst, |
| (in_bytes / bta_dm_pcm_cb.divisor), |
| bta_dm_pcm_cb.src_sps); |
| } |
| |
| #if (BTA_DM_SCO_DEBUG == TRUE) |
| APPL_TRACE_DEBUG("bta_pcm_resample : outsamples %d", out_sample); |
| #endif |
| |
| return (out_sample * bta_dm_pcm_cb.sample_size); |
| } |
| #endif |