美文网首页
webrtc之音频设备采集及封装

webrtc之音频设备采集及封装

作者: hijiang | 来源:发表于2019-07-19 11:32 被阅读0次

音频设备同样抽象出音频设备对象:

class AudioDevice {
public:
    AudioDevice(const std::string& id, const std::string& name) :device_id(id), device_name(name) {
    }
    virtual ~AudioDevice() {
    }

public:
    std::string device_id;
    std::string device_name;
};

设备集合:

class AudioDeviceCollection
{
public:
    AudioDeviceCollection();
    virtual ~AudioDeviceCollection();

private:
    void addAudioDevice(std::shared_ptr<AudioDevice> device);

    std::vector<std::shared_ptr<AudioDevice>> audio_devices_;
public:
    int getCount();
    std::shared_ptr<AudioDevice> getDevice(int index);
    int getDevice(int index, std::string &deviceName, std::string &deviceId);
    int searchDevice(const std::string &deviceId);

    friend class XXX::AudioDeviceManager;
};

音频设备枚举,因为需要使用到webrtc内部的音频设备管理器,而其使用只能在webrtc的几个固定线程中,否则抛异常,故首先需要创建adm对象,并且需要做一个统一的webrtc线程管理对象,需要使用的线程都丢到那里面:

AudioDeviceManager::AudioDeviceManager():/*audio_sink_(nullptr), */recording_devices_enumerated_(false), playout_devices_enumerated_(false)
{
    audio_recording_collection_ = std::make_shared<XXX_PUBLIC::AudioDeviceCollection>();
    audio_playout_collection_ = std::make_shared<XXX_PUBLIC::AudioDeviceCollection>();

    XXXThreadManager::getInstance()->worker_thread_->Invoke<void>(RTC_FROM_HERE, [&]() {
        adm_ = webrtc::AudioDeviceModule::Create(webrtc::AudioDeviceModule::kPlatformDefaultAudio);
        adm_->Init();
    });
    EnableBuiltInAEC(false);
}

void AudioDeviceManager::init()
{
    XXXThreadManager::getInstance()->worker_thread_->Invoke<void>(RTC_FROM_HERE, [&]() {
        if (adm_) {
            adm_->Init();
        }
    });
}

枚举音频采集设备:

std::shared_ptr<XXX_PUBLIC::AudioDeviceCollection> AudioDeviceManager::enumerateRecordingDevices()
{
    if (recording_devices_enumerated_) {
        return audio_recording_collection_;
    }

    XXXThreadManager::getInstance()->worker_thread_->Invoke<void>(RTC_FROM_HERE, [&]() {
        int num_devices = adm_->RecordingDevices();
        for (int i = 0; i < num_devices; i++)
        {
            char name[webrtc::kAdmMaxDeviceNameSize];
            char id[webrtc::kAdmMaxGuidSize];
            if (adm_->RecordingDeviceName(i, name, id) != -1)
            {
                std::shared_ptr<XXX_PUBLIC::AudioDevice> device = std::make_shared<XXX_PUBLIC::AudioDevice>(id, name);
                audio_recording_collection_->addAudioDevice(device);
            }
        }
    });
    recording_devices_enumerated_ = true;

    return audio_recording_collection_;
}

枚举音频播放设备:

std::shared_ptr<XXX_PUBLIC::AudioDeviceCollection> AudioDeviceManager::enumeratePlayoutDevices()
{
    if (playout_devices_enumerated_) {
        return audio_playout_collection_;
    }

    XXXThreadManager::getInstance()->worker_thread_->Invoke<void>(RTC_FROM_HERE, [&]() {
        int num_devices = adm_->PlayoutDevices();
        for (int i = 0; i < num_devices; i++)
        {
            char name[webrtc::kAdmMaxDeviceNameSize];
            char id[webrtc::kAdmMaxGuidSize];
            if (adm_->PlayoutDeviceName(i, name, id) != -1)
            {
                std::shared_ptr<xxx_PUBLIC::AudioDevice> device = std::make_shared<XXX_PUBLIC::AudioDevice>(id, name);
                audio_playout_collection_->addAudioDevice(device);
            }
        }
    });
    playout_devices_enumerated_ = true;
    return audio_playout_collection_;
}

设置音频数据回调:

int AudioDeviceManager::RegisterAudioSink(webrtc::AudioTransport *audio_sink)
{
    int iRet = XXXThreadManager::getInstance()->worker_thread_->Invoke<int>(RTC_FROM_HERE, [=]() {
        return adm_->RegisterAudioCallback(audio_sink);
    });
    return iRet;
}

管理对象总体接口(当然我们还需要在外部封装一层,屏蔽webrtc的头文件):

class AudioDeviceManager : public webrtc::AudioTransport
{
    public:
        static AudioDeviceManager *getInstance();
        virtual ~AudioDeviceManager();
    private:
        AudioDeviceManager();
    public:
        std::shared_ptr<XXX_PUBLIC::AudioDeviceCollection> enumerateRecordingDevices();
        std::shared_ptr<XXX_PUBLIC::AudioDeviceCollection> enumeratePlayoutDevices();

        int setRecordingDevice(const std::string& deviceId);
        int setRecordingDevice(std::shared_ptr<XXX_PUBLIC::AudioDevice> audio_device);
        int setRecordingDeviceVolume(uint32_t volume);
        int getRecordingDeviceVolume();
        int setRecordingDeviceMute(bool mute);
        int getRecordingDeviceMute(bool* mute);
        int EnableBuiltInAEC(bool enable);
        int RegisterAudioSink(webrtc::AudioTransport *audio_sink);
        int initRecording();
        int startRecording();

        int setPlayoutDevice(const std::string& deviceId);
        int setPlayoutDevice(std::shared_ptr<XXX_PUBLIC::AudioDevice> audio_device);
        int setPlayuotDeviceVolume(uint32_t volume);
        int setPlayoutDeviceMute(bool mute);
        int getPlayoutDeviceMute(bool* mute);
        int getPlayoutDeviceVolume();
        int startPlayout();

        void init();

    public:
        rtc::scoped_refptr<webrtc::AudioDeviceModule> adm_;
        rtc::scoped_refptr<webrtc::AudioDeviceModule> getADM();
    private:
        static AudioDeviceManager *instance_;
        std::shared_ptr<XXX_PUBLIC::AudioDeviceCollection> audio_recording_collection_;
        std::shared_ptr<XXX_PUBLIC::AudioDeviceCollection> audio_playout_collection_;

        std::shared_ptr<XXX_PUBLIC::AudioDevice> curr_recording_device_;
        std::shared_ptr<XXX_PUBLIC::AudioDevice> curr_playout_device_;

        bool playout_devices_enumerated_;
        bool recording_devices_enumerated_;
    private:
        int32_t RecordedDataIsAvailable(const void* audioSamples,
            const size_t nSamples,
            const size_t nBytesPerSample,
            const size_t nChannels,
            const uint32_t samplesPerSec,
            const uint32_t totalDelayMS,
            const int32_t clockDrift,
            const uint32_t currentMicLevel,
            const bool keyPressed,
            uint32_t& newMicLevel) {
            return 0;
        }

        // Implementation has to setup safe values for all specified out parameters.
        int32_t NeedMorePlayData(const size_t nSamples,
            const size_t nBytesPerSample,
            const size_t nChannels,
            const uint32_t samplesPerSec,
            void* audioSamples,
            size_t& nSamplesOut,  // NOLINT
            int64_t* elapsed_time_ms,
            int64_t* ntp_time_ms) {
            return 0;
        };

        // Method to pull mixed render audio data from all active VoE channels.
        // The data will not be passed as reference for audio processing internally.
        void PullRenderData(int bits_per_sample,
            int sample_rate,
            size_t number_of_channels,
            size_t number_of_frames,
            void* audio_data,
            int64_t* elapsed_time_ms,
            int64_t* ntp_time_ms) {};
};

相关文章

  • webrtc之音频设备采集及封装

    音频设备同样抽象出音频设备对象: 设备集合: 音频设备枚举,因为需要使用到webrtc内部的音频设备管理器,而其使...

  • webrtc之视频设备枚举及封装

    有一小段时间没研究webrtc,写点笔记。webrtc的视频设备处理,基本都在VideoCatpureFactor...

  • iOS下WebRTC视频编码

    前言 在 iOS下WebRTC视频采集 一文中,向大家介绍了 WebRTC 是如何在 iOS下进行视频采集的。本文...

  • webrtc之封装视频源

    webrtc视频设备采集,需要先创建VideoCapturer 然后设置采集的限制: 然后使用PeerConnec...

  • webrtc本地采集

    本文分享webrtc的ios端上自定义采集视频本质问题:从摄像机里面取出数据给webrtc模块使用。建立 RTCC...

  • webrtc源码分析之视频采集之二

    在webrtc源码分析之视频采集之一中,主要分析了视频采集与分发的流程以及涉及到的主要类,接下来分析一下这些主要类...

  • webrtc之Android视频采集编码

    创建videoCapturer 首先在Call.java中执行函数 private VideoCapturer c...

  • iOS 音频-AVAudioSession

    AVAudioSession 概述 最近在做 webrtc 采集与播放音频,使用AVAudioSession进行播...

  • iOS下WebRTC视频解码

    前言 今天介绍一下 iOS下WebRTC 是如何进行视频解码的。关于iOS下WebRTC视频采集与编码可以看下面的...

  • WebRTC简介

    什么是WebRTC WebRTC是一个由Google发起的实时通讯解决方案,其中包含视频音频采集,编解码,数据传输...

网友评论

      本文标题:webrtc之音频设备采集及封装

      本文链接:https://www.haomeiwen.com/subject/sijjlctx.html