blob: d72fced624affc24fd41414d5d64f29dc221fd43 [file] [log] [blame]
kwiberge985b902017-07-31 11:34:57 -07001/*
2 * Copyright (c) 2017 The WebRTC project authors. All Rights Reserved.
3 *
4 * Use of this source code is governed by a BSD-style license
5 * that can be found in the LICENSE file in the root of the source
6 * tree. An additional intellectual property rights grant can be found
7 * in the file PATENTS. All contributing project authors may
8 * be found in the AUTHORS file in the root of the source tree.
9 */
10
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020011#include "api/audio_codecs/g711/audio_encoder_g711.h"
kwiberge985b902017-07-31 11:34:57 -070012
13#include <memory>
14#include <vector>
15
Mirko Bonadei71207422017-09-15 13:58:09 +020016#include "common_types.h" // NOLINT(build/include)
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020017#include "modules/audio_coding/codecs/g711/audio_encoder_pcm.h"
18#include "rtc_base/ptr_util.h"
19#include "rtc_base/safe_conversions.h"
20#include "rtc_base/safe_minmax.h"
21#include "rtc_base/string_to_number.h"
kwiberge985b902017-07-31 11:34:57 -070022
23namespace webrtc {
24
25rtc::Optional<AudioEncoderG711::Config> AudioEncoderG711::SdpToConfig(
26 const SdpAudioFormat& format) {
27 const bool is_pcmu = STR_CASE_CMP(format.name.c_str(), "PCMU") == 0;
28 const bool is_pcma = STR_CASE_CMP(format.name.c_str(), "PCMA") == 0;
29 if (format.clockrate_hz == 8000 && format.num_channels >= 1 &&
30 (is_pcmu || is_pcma)) {
31 Config config;
32 config.type = is_pcmu ? Config::Type::kPcmU : Config::Type::kPcmA;
33 config.num_channels = rtc::dchecked_cast<int>(format.num_channels);
34 config.frame_size_ms = 20;
35 auto ptime_iter = format.parameters.find("ptime");
36 if (ptime_iter != format.parameters.end()) {
37 const auto ptime = rtc::StringToNumber<int>(ptime_iter->second);
38 if (ptime && *ptime > 0) {
39 config.frame_size_ms = rtc::SafeClamp(10 * (*ptime / 10), 10, 60);
40 }
41 }
42 RTC_DCHECK(config.IsOk());
43 return rtc::Optional<Config>(config);
44 } else {
45 return rtc::Optional<Config>();
46 }
47}
48
49void AudioEncoderG711::AppendSupportedEncoders(
50 std::vector<AudioCodecSpec>* specs) {
51 for (const char* type : {"PCMU", "PCMA"}) {
52 specs->push_back({{type, 8000, 1}, {8000, 1, 64000}});
53 }
54}
55
56AudioCodecInfo AudioEncoderG711::QueryAudioEncoder(const Config& config) {
57 RTC_DCHECK(config.IsOk());
58 return {8000, rtc::dchecked_cast<size_t>(config.num_channels),
59 64000 * config.num_channels};
60}
61
62std::unique_ptr<AudioEncoder> AudioEncoderG711::MakeAudioEncoder(
63 const Config& config,
64 int payload_type) {
65 RTC_DCHECK(config.IsOk());
66 switch (config.type) {
67 case Config::Type::kPcmU: {
68 AudioEncoderPcmU::Config impl_config;
69 impl_config.num_channels = config.num_channels;
70 impl_config.frame_size_ms = config.frame_size_ms;
71 impl_config.payload_type = payload_type;
72 return rtc::MakeUnique<AudioEncoderPcmU>(impl_config);
73 }
74 case Config::Type::kPcmA: {
75 AudioEncoderPcmA::Config impl_config;
76 impl_config.num_channels = config.num_channels;
77 impl_config.frame_size_ms = config.frame_size_ms;
78 impl_config.payload_type = payload_type;
79 return rtc::MakeUnique<AudioEncoderPcmA>(impl_config);
80 }
81 default: { return nullptr; }
82 }
83}
84
85} // namespace webrtc