C++程序  |  120行  |  4.13 KB

/*
 * Copyright (C) 2017 The Android Open Source Project
 *
 * 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.
 */

#ifndef ANDROID_RECORD_BUFFER_CONVERTER_H
#define ANDROID_RECORD_BUFFER_CONVERTER_H

#include <stdint.h>
#include <sys/types.h>

#include <media/AudioBufferProvider.h>
#include <system/audio.h>

class AudioResampler;
class PassthruBufferProvider;

namespace android {

/* The RecordBufferConverter is used for format, channel, and sample rate
 * conversion for a RecordTrack.
 *
 * RecordBufferConverter uses the convert() method rather than exposing a
 * buffer provider interface; this is to save a memory copy.
 *
 * There are legacy conversion requirements for this converter, specifically
 * due to mono handling, so be careful about modifying.
 *
 * Original source audioflinger/Threads.{h,cpp}
 */
class RecordBufferConverter
{
public:
    RecordBufferConverter(
            audio_channel_mask_t srcChannelMask, audio_format_t srcFormat,
            uint32_t srcSampleRate,
            audio_channel_mask_t dstChannelMask, audio_format_t dstFormat,
            uint32_t dstSampleRate);

    ~RecordBufferConverter();

    /* Converts input data from an AudioBufferProvider by format, channelMask,
     * and sampleRate to a destination buffer.
     *
     * Parameters
     *      dst:  buffer to place the converted data.
     * provider:  buffer provider to obtain source data.
     *   frames:  number of frames to convert
     *
     * Returns the number of frames converted.
     */
    size_t convert(void *dst, AudioBufferProvider *provider, size_t frames);

    // returns NO_ERROR if constructor was successful
    status_t initCheck() const {
        // mSrcChannelMask set on successful updateParameters
        return mSrcChannelMask != AUDIO_CHANNEL_INVALID ? NO_ERROR : NO_INIT;
    }

    // allows dynamic reconfigure of all parameters
    status_t updateParameters(
            audio_channel_mask_t srcChannelMask, audio_format_t srcFormat,
            uint32_t srcSampleRate,
            audio_channel_mask_t dstChannelMask, audio_format_t dstFormat,
            uint32_t dstSampleRate);

    // called to reset resampler buffers on record track discontinuity
    void reset();

private:
    // format conversion when not using resampler
    void convertNoResampler(void *dst, const void *src, size_t frames);

    // format conversion when using resampler; modifies src in-place
    void convertResampler(void *dst, /*not-a-const*/ void *src, size_t frames);

    // user provided information
    audio_channel_mask_t mSrcChannelMask;
    audio_format_t       mSrcFormat;
    uint32_t             mSrcSampleRate;
    audio_channel_mask_t mDstChannelMask;
    audio_format_t       mDstFormat;
    uint32_t             mDstSampleRate;

    // derived information
    uint32_t             mSrcChannelCount;
    uint32_t             mDstChannelCount;
    size_t               mDstFrameSize;

    // format conversion buffer
    void                *mBuf;
    size_t               mBufFrames;
    size_t               mBufFrameSize;

    // resampler info
    AudioResampler      *mResampler;

    bool                 mIsLegacyDownmix;  // legacy stereo to mono conversion needed
    bool                 mIsLegacyUpmix;    // legacy mono to stereo conversion needed
    bool                 mRequiresFloat;    // data processing requires float (e.g. resampler)
    PassthruBufferProvider *mInputConverterProvider;    // converts input to float
    int8_t               mIdxAry[sizeof(uint32_t) * 8]; // used for channel mask conversion
};

// ----------------------------------------------------------------------------
} // namespace android

#endif // ANDROID_RECORD_BUFFER_CONVERTER_H