blob: 7b3b1d2327c418e433ccc23b764a7b75074edd70 [file] [log] [blame]
turaj@webrtc.org7959e162013-09-12 18:30:26 +00001/*
2 * Copyright (c) 2012 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
11/*
12 * This file generates databases with information about all supported audio
13 * codecs.
14 */
15
16// TODO(tlegrand): Change constant input pointers in all functions to constant
17// references, where appropriate.
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020018#include "modules/audio_coding/acm2/acm_codec_database.h"
turaj@webrtc.org7959e162013-09-12 18:30:26 +000019
20#include <assert.h>
21
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020022#include "rtc_base/checks.h"
turaj@webrtc.org7959e162013-09-12 18:30:26 +000023
kwiberg65cb70d2017-03-03 06:16:28 -080024#if ((defined WEBRTC_CODEC_ISAC) && (defined WEBRTC_CODEC_ISACFX))
25#error iSAC and iSACFX codecs cannot be enabled at the same time
26#endif
27
turaj@webrtc.org7959e162013-09-12 18:30:26 +000028namespace webrtc {
29
turaj@webrtc.org6d5d2482013-10-06 04:47:28 +000030namespace acm2 {
31
kwibergec249d42015-09-24 04:32:03 -070032namespace {
33
kwibergec249d42015-09-24 04:32:03 -070034// Checks if the bitrate is valid for iSAC.
35bool IsISACRateValid(int rate) {
36 return (rate == -1) || ((rate <= 56000) && (rate >= 10000));
37}
38
39// Checks if the bitrate is valid for iLBC.
40bool IsILBCRateValid(int rate, int frame_size_samples) {
41 if (((frame_size_samples == 240) || (frame_size_samples == 480)) &&
42 (rate == 13300)) {
43 return true;
44 } else if (((frame_size_samples == 160) || (frame_size_samples == 320)) &&
45 (rate == 15200)) {
46 return true;
47 } else {
48 return false;
49 }
50}
51
52// Checks if the bitrate is valid for Opus.
53bool IsOpusRateValid(int rate) {
54 return (rate >= 6000) && (rate <= 510000);
55}
56
57} // namespace
58
turaj@webrtc.org532f3dc2013-09-19 00:12:23 +000059// Not yet used payload-types.
60// 83, 82, 81, 80, 79, 78, 77, 76, 75, 74, 73, 72, 71, 70, 69, 68,
61// 67, 66, 65
turaj@webrtc.org7959e162013-09-12 18:30:26 +000062
63const CodecInst ACMCodecDB::database_[] = {
64#if (defined(WEBRTC_CODEC_ISAC) || defined(WEBRTC_CODEC_ISACFX))
kwiberg65cb70d2017-03-03 06:16:28 -080065 {103, "ISAC", 16000, 480, 1, 32000},
turaj@webrtc.org7959e162013-09-12 18:30:26 +000066# if (defined(WEBRTC_CODEC_ISAC))
kwiberg65cb70d2017-03-03 06:16:28 -080067 {104, "ISAC", 32000, 960, 1, 56000},
turaj@webrtc.org7959e162013-09-12 18:30:26 +000068# endif
69#endif
turaj@webrtc.org7959e162013-09-12 18:30:26 +000070 // Mono
turaj@webrtc.org532f3dc2013-09-19 00:12:23 +000071 {107, "L16", 8000, 80, 1, 128000},
72 {108, "L16", 16000, 160, 1, 256000},
73 {109, "L16", 32000, 320, 1, 512000},
turaj@webrtc.org7959e162013-09-12 18:30:26 +000074 // Stereo
turaj@webrtc.org532f3dc2013-09-19 00:12:23 +000075 {111, "L16", 8000, 80, 2, 128000},
76 {112, "L16", 16000, 160, 2, 256000},
77 {113, "L16", 32000, 320, 2, 512000},
turaj@webrtc.org7959e162013-09-12 18:30:26 +000078 // G.711, PCM mu-law and A-law.
79 // Mono
80 {0, "PCMU", 8000, 160, 1, 64000},
81 {8, "PCMA", 8000, 160, 1, 64000},
82 // Stereo
83 {110, "PCMU", 8000, 160, 2, 64000},
84 {118, "PCMA", 8000, 160, 2, 64000},
85#ifdef WEBRTC_CODEC_ILBC
86 {102, "ILBC", 8000, 240, 1, 13300},
87#endif
turaj@webrtc.org7959e162013-09-12 18:30:26 +000088#ifdef WEBRTC_CODEC_G722
89 // Mono
90 {9, "G722", 16000, 320, 1, 64000},
91 // Stereo
92 {119, "G722", 16000, 320, 2, 64000},
93#endif
turaj@webrtc.org7959e162013-09-12 18:30:26 +000094#ifdef WEBRTC_CODEC_OPUS
95 // Opus internally supports 48, 24, 16, 12, 8 kHz.
96 // Mono and stereo.
97 {120, "opus", 48000, 960, 2, 64000},
98#endif
turaj@webrtc.org7959e162013-09-12 18:30:26 +000099 // Comfort noise for four different sampling frequencies.
100 {13, "CN", 8000, 240, 1, 0},
101 {98, "CN", 16000, 480, 1, 0},
102 {99, "CN", 32000, 960, 1, 0},
turaj@webrtc.org532f3dc2013-09-19 00:12:23 +0000103#ifdef ENABLE_48000_HZ
turaj@webrtc.org7959e162013-09-12 18:30:26 +0000104 {100, "CN", 48000, 1440, 1, 0},
turaj@webrtc.org532f3dc2013-09-19 00:12:23 +0000105#endif
turaj@webrtc.org7959e162013-09-12 18:30:26 +0000106 {106, "telephone-event", 8000, 240, 1, 0},
solenberg2779bab2016-11-17 04:45:19 -0800107 {114, "telephone-event", 16000, 240, 1, 0},
108 {115, "telephone-event", 32000, 240, 1, 0},
109 {116, "telephone-event", 48000, 240, 1, 0},
turaj@webrtc.org7959e162013-09-12 18:30:26 +0000110#ifdef WEBRTC_CODEC_RED
111 {127, "red", 8000, 0, 1, 0},
112#endif
113 // To prevent compile errors due to trailing commas.
pkasting25702cb2016-01-08 13:50:27 -0800114 {-1, "Null", -1, -1, 0, -1}
turaj@webrtc.org7959e162013-09-12 18:30:26 +0000115};
116
117// Create database with all codec settings at compile time.
118// Each entry needs the following parameters in the given order:
119// Number of allowed packet sizes, a vector with the allowed packet sizes,
120// Basic block samples, max number of channels that are supported.
121const ACMCodecDB::CodecSettings ACMCodecDB::codec_settings_[] = {
122#if (defined(WEBRTC_CODEC_ISAC) || defined(WEBRTC_CODEC_ISACFX))
kwiberg65cb70d2017-03-03 06:16:28 -0800123 {2, {480, 960}, 0, 1},
turaj@webrtc.org7959e162013-09-12 18:30:26 +0000124# if (defined(WEBRTC_CODEC_ISAC))
kwiberg65cb70d2017-03-03 06:16:28 -0800125 {1, {960}, 0, 1},
turaj@webrtc.org7959e162013-09-12 18:30:26 +0000126# endif
127#endif
turaj@webrtc.org7959e162013-09-12 18:30:26 +0000128 // Mono
kwibergec249d42015-09-24 04:32:03 -0700129 {4, {80, 160, 240, 320}, 0, 2},
130 {4, {160, 320, 480, 640}, 0, 2},
131 {2, {320, 640}, 0, 2},
turaj@webrtc.org7959e162013-09-12 18:30:26 +0000132 // Stereo
kwibergec249d42015-09-24 04:32:03 -0700133 {4, {80, 160, 240, 320}, 0, 2},
134 {4, {160, 320, 480, 640}, 0, 2},
turaj@webrtc.org7959e162013-09-12 18:30:26 +0000135 {2, {320, 640}, 0, 2},
turaj@webrtc.org7959e162013-09-12 18:30:26 +0000136 // G.711, PCM mu-law and A-law.
137 // Mono
kwibergec249d42015-09-24 04:32:03 -0700138 {6, {80, 160, 240, 320, 400, 480}, 0, 2},
139 {6, {80, 160, 240, 320, 400, 480}, 0, 2},
turaj@webrtc.org7959e162013-09-12 18:30:26 +0000140 // Stereo
kwibergec249d42015-09-24 04:32:03 -0700141 {6, {80, 160, 240, 320, 400, 480}, 0, 2},
142 {6, {80, 160, 240, 320, 400, 480}, 0, 2},
turaj@webrtc.org7959e162013-09-12 18:30:26 +0000143#ifdef WEBRTC_CODEC_ILBC
kwibergec249d42015-09-24 04:32:03 -0700144 {4, {160, 240, 320, 480}, 0, 1},
turaj@webrtc.org7959e162013-09-12 18:30:26 +0000145#endif
turaj@webrtc.org7959e162013-09-12 18:30:26 +0000146#ifdef WEBRTC_CODEC_G722
147 // Mono
kwibergec249d42015-09-24 04:32:03 -0700148 {6, {160, 320, 480, 640, 800, 960}, 0, 2},
turaj@webrtc.org7959e162013-09-12 18:30:26 +0000149 // Stereo
kwibergec249d42015-09-24 04:32:03 -0700150 {6, {160, 320, 480, 640, 800, 960}, 0, 2},
turaj@webrtc.org7959e162013-09-12 18:30:26 +0000151#endif
turaj@webrtc.org7959e162013-09-12 18:30:26 +0000152#ifdef WEBRTC_CODEC_OPUS
153 // Opus supports frames shorter than 10ms,
154 // but it doesn't help us to use them.
155 // Mono and stereo.
minyue2e03c662017-02-01 17:31:11 -0800156#if WEBRTC_OPUS_SUPPORT_120MS_PTIME
157 {5, {480, 960, 1920, 2880, 5760}, 0, 2},
158#else
kwibergec249d42015-09-24 04:32:03 -0700159 {4, {480, 960, 1920, 2880}, 0, 2},
turaj@webrtc.org7959e162013-09-12 18:30:26 +0000160#endif
minyue2e03c662017-02-01 17:31:11 -0800161#endif
turaj@webrtc.org7959e162013-09-12 18:30:26 +0000162 // Comfort noise for three different sampling frequencies.
kwibergec249d42015-09-24 04:32:03 -0700163 {1, {240}, 240, 1},
164 {1, {480}, 480, 1},
165 {1, {960}, 960, 1},
solenberg2779bab2016-11-17 04:45:19 -0800166// TODO(solenberg): What is this flag? It is never set in the build files.
turaj@webrtc.org532f3dc2013-09-19 00:12:23 +0000167#ifdef ENABLE_48000_HZ
kwibergec249d42015-09-24 04:32:03 -0700168 {1, {1440}, 1440, 1},
turaj@webrtc.org532f3dc2013-09-19 00:12:23 +0000169#endif
kwibergec249d42015-09-24 04:32:03 -0700170 {1, {240}, 240, 1},
solenberg2779bab2016-11-17 04:45:19 -0800171 {1, {240}, 240, 1},
172 {1, {240}, 240, 1},
173 {1, {240}, 240, 1},
turaj@webrtc.org7959e162013-09-12 18:30:26 +0000174#ifdef WEBRTC_CODEC_RED
kwibergec249d42015-09-24 04:32:03 -0700175 {1, {0}, 0, 1},
turaj@webrtc.org7959e162013-09-12 18:30:26 +0000176#endif
177 // To prevent compile errors due to trailing commas.
pkasting25702cb2016-01-08 13:50:27 -0800178 {-1, {-1}, -1, 0}
turaj@webrtc.org7959e162013-09-12 18:30:26 +0000179};
180
181// Create a database of all NetEQ decoders at compile time.
182const NetEqDecoder ACMCodecDB::neteq_decoders_[] = {
183#if (defined(WEBRTC_CODEC_ISAC) || defined(WEBRTC_CODEC_ISACFX))
kwibergee1879c2015-10-29 06:20:28 -0700184 NetEqDecoder::kDecoderISAC,
turaj@webrtc.org7959e162013-09-12 18:30:26 +0000185# if (defined(WEBRTC_CODEC_ISAC))
kwibergee1879c2015-10-29 06:20:28 -0700186 NetEqDecoder::kDecoderISACswb,
turaj@webrtc.org7959e162013-09-12 18:30:26 +0000187# endif
188#endif
turaj@webrtc.org7959e162013-09-12 18:30:26 +0000189 // Mono
kwibergee1879c2015-10-29 06:20:28 -0700190 NetEqDecoder::kDecoderPCM16B, NetEqDecoder::kDecoderPCM16Bwb,
191 NetEqDecoder::kDecoderPCM16Bswb32kHz,
turaj@webrtc.org7959e162013-09-12 18:30:26 +0000192 // Stereo
kwibergee1879c2015-10-29 06:20:28 -0700193 NetEqDecoder::kDecoderPCM16B_2ch, NetEqDecoder::kDecoderPCM16Bwb_2ch,
194 NetEqDecoder::kDecoderPCM16Bswb32kHz_2ch,
turaj@webrtc.org7959e162013-09-12 18:30:26 +0000195 // G.711, PCM mu-las and A-law.
196 // Mono
kwibergee1879c2015-10-29 06:20:28 -0700197 NetEqDecoder::kDecoderPCMu, NetEqDecoder::kDecoderPCMa,
turaj@webrtc.org7959e162013-09-12 18:30:26 +0000198 // Stereo
kwibergee1879c2015-10-29 06:20:28 -0700199 NetEqDecoder::kDecoderPCMu_2ch, NetEqDecoder::kDecoderPCMa_2ch,
turaj@webrtc.org7959e162013-09-12 18:30:26 +0000200#ifdef WEBRTC_CODEC_ILBC
kwibergee1879c2015-10-29 06:20:28 -0700201 NetEqDecoder::kDecoderILBC,
turaj@webrtc.org7959e162013-09-12 18:30:26 +0000202#endif
turaj@webrtc.org7959e162013-09-12 18:30:26 +0000203#ifdef WEBRTC_CODEC_G722
204 // Mono
kwibergee1879c2015-10-29 06:20:28 -0700205 NetEqDecoder::kDecoderG722,
turaj@webrtc.org7959e162013-09-12 18:30:26 +0000206 // Stereo
kwibergee1879c2015-10-29 06:20:28 -0700207 NetEqDecoder::kDecoderG722_2ch,
turaj@webrtc.org7959e162013-09-12 18:30:26 +0000208#endif
turaj@webrtc.org7959e162013-09-12 18:30:26 +0000209#ifdef WEBRTC_CODEC_OPUS
210 // Mono and stereo.
kwibergee1879c2015-10-29 06:20:28 -0700211 NetEqDecoder::kDecoderOpus,
turaj@webrtc.org7959e162013-09-12 18:30:26 +0000212#endif
turaj@webrtc.org7959e162013-09-12 18:30:26 +0000213 // Comfort noise for three different sampling frequencies.
kwibergee1879c2015-10-29 06:20:28 -0700214 NetEqDecoder::kDecoderCNGnb, NetEqDecoder::kDecoderCNGwb,
215 NetEqDecoder::kDecoderCNGswb32kHz,
turaj@webrtc.org532f3dc2013-09-19 00:12:23 +0000216#ifdef ENABLE_48000_HZ
kwibergee1879c2015-10-29 06:20:28 -0700217 NetEqDecoder::kDecoderCNGswb48kHz,
turaj@webrtc.org532f3dc2013-09-19 00:12:23 +0000218#endif
kwibergee1879c2015-10-29 06:20:28 -0700219 NetEqDecoder::kDecoderAVT,
solenberg2779bab2016-11-17 04:45:19 -0800220 NetEqDecoder::kDecoderAVT16kHz,
221 NetEqDecoder::kDecoderAVT32kHz,
222 NetEqDecoder::kDecoderAVT48kHz,
turaj@webrtc.org7959e162013-09-12 18:30:26 +0000223#ifdef WEBRTC_CODEC_RED
kwibergee1879c2015-10-29 06:20:28 -0700224 NetEqDecoder::kDecoderRED,
turaj@webrtc.org7959e162013-09-12 18:30:26 +0000225#endif
226};
227
turaj@webrtc.org7959e162013-09-12 18:30:26 +0000228// Enumerator for error codes when asking for codec database id.
229enum {
230 kInvalidCodec = -10,
231 kInvalidPayloadtype = -30,
232 kInvalidPacketSize = -40,
233 kInvalidRate = -50
234};
235
236// Gets the codec id number from the database. If there is some mismatch in
237// the codec settings, the function will return an error code.
238// NOTE! The first mismatch found will generate the return value.
Henrik Lundin93ef1d82015-04-13 09:31:16 +0200239int ACMCodecDB::CodecNumber(const CodecInst& codec_inst) {
turaj@webrtc.org7959e162013-09-12 18:30:26 +0000240 // Look for a matching codec in the database.
241 int codec_id = CodecId(codec_inst);
242
243 // Checks if we found a matching codec.
244 if (codec_id == -1) {
245 return kInvalidCodec;
246 }
247
248 // Checks the validity of payload type
kwiberg93a2feb2015-11-05 07:39:37 -0800249 if (!RentACodec::IsPayloadTypeValid(codec_inst.pltype)) {
turaj@webrtc.org7959e162013-09-12 18:30:26 +0000250 return kInvalidPayloadtype;
251 }
252
253 // Comfort Noise is special case, packet-size & rate is not checked.
254 if (STR_CASE_CMP(database_[codec_id].plname, "CN") == 0) {
turaj@webrtc.org7959e162013-09-12 18:30:26 +0000255 return codec_id;
256 }
257
258 // RED is special case, packet-size & rate is not checked.
259 if (STR_CASE_CMP(database_[codec_id].plname, "red") == 0) {
turaj@webrtc.org7959e162013-09-12 18:30:26 +0000260 return codec_id;
261 }
262
263 // Checks the validity of packet size.
264 if (codec_settings_[codec_id].num_packet_sizes > 0) {
265 bool packet_size_ok = false;
266 int i;
267 int packet_size_samples;
268 for (i = 0; i < codec_settings_[codec_id].num_packet_sizes; i++) {
269 packet_size_samples =
270 codec_settings_[codec_id].packet_sizes_samples[i];
271 if (codec_inst.pacsize == packet_size_samples) {
272 packet_size_ok = true;
273 break;
274 }
275 }
276
277 if (!packet_size_ok) {
278 return kInvalidPacketSize;
279 }
280 }
281
282 if (codec_inst.pacsize < 1) {
283 return kInvalidPacketSize;
284 }
285
286 // Check the validity of rate. Codecs with multiple rates have their own
287 // function for this.
turaj@webrtc.org7959e162013-09-12 18:30:26 +0000288 if (STR_CASE_CMP("isac", codec_inst.plname) == 0) {
Henrik Lundin93ef1d82015-04-13 09:31:16 +0200289 return IsISACRateValid(codec_inst.rate) ? codec_id : kInvalidRate;
turaj@webrtc.org7959e162013-09-12 18:30:26 +0000290 } else if (STR_CASE_CMP("ilbc", codec_inst.plname) == 0) {
291 return IsILBCRateValid(codec_inst.rate, codec_inst.pacsize)
292 ? codec_id : kInvalidRate;
turaj@webrtc.org7959e162013-09-12 18:30:26 +0000293 } else if (STR_CASE_CMP("opus", codec_inst.plname) == 0) {
294 return IsOpusRateValid(codec_inst.rate)
295 ? codec_id : kInvalidRate;
turaj@webrtc.org7959e162013-09-12 18:30:26 +0000296 }
297
kwiberg4b938e52015-11-03 12:38:27 -0800298 return database_[codec_id].rate == codec_inst.rate ? codec_id : kInvalidRate;
turaj@webrtc.org7959e162013-09-12 18:30:26 +0000299}
300
301// Looks for a matching payload name, frequency, and channels in the
302// codec list. Need to check all three since some codecs have several codec
303// entries with different frequencies and/or channels.
304// Does not check other codec settings, such as payload type and packet size.
305// Returns the id of the codec, or -1 if no match is found.
306int ACMCodecDB::CodecId(const CodecInst& codec_inst) {
307 return (CodecId(codec_inst.plname, codec_inst.plfreq,
308 codec_inst.channels));
309}
310
Peter Kasting69558702016-01-12 16:26:35 -0800311int ACMCodecDB::CodecId(const char* payload_name,
312 int frequency,
313 size_t channels) {
kwibergfce4a942015-10-27 11:40:24 -0700314 for (const CodecInst& ci : RentACodec::Database()) {
turaj@webrtc.org7959e162013-09-12 18:30:26 +0000315 bool name_match = false;
316 bool frequency_match = false;
317 bool channels_match = false;
318
319 // Payload name, sampling frequency and number of channels need to match.
320 // NOTE! If |frequency| is -1, the frequency is not applicable, and is
321 // always treated as true, like for RED.
kwibergfce4a942015-10-27 11:40:24 -0700322 name_match = (STR_CASE_CMP(ci.plname, payload_name) == 0);
323 frequency_match = (frequency == ci.plfreq) || (frequency == -1);
turaj@webrtc.org7959e162013-09-12 18:30:26 +0000324 // The number of channels must match for all codecs but Opus.
325 if (STR_CASE_CMP(payload_name, "opus") != 0) {
kwibergfce4a942015-10-27 11:40:24 -0700326 channels_match = (channels == ci.channels);
turaj@webrtc.org7959e162013-09-12 18:30:26 +0000327 } else {
328 // For opus we just check that number of channels is valid.
329 channels_match = (channels == 1 || channels == 2);
330 }
331
332 if (name_match && frequency_match && channels_match) {
333 // We have found a matching codec in the list.
kwibergfce4a942015-10-27 11:40:24 -0700334 return &ci - RentACodec::Database().data();
turaj@webrtc.org7959e162013-09-12 18:30:26 +0000335 }
336 }
337
338 // We didn't find a matching codec.
339 return -1;
340}
Henrik Lundin93ef1d82015-04-13 09:31:16 +0200341// Gets codec id number from database for the receiver.
342int ACMCodecDB::ReceiverCodecNumber(const CodecInst& codec_inst) {
turaj@webrtc.org7959e162013-09-12 18:30:26 +0000343 // Look for a matching codec in the database.
Henrik Lundin93ef1d82015-04-13 09:31:16 +0200344 return CodecId(codec_inst);
turaj@webrtc.org7959e162013-09-12 18:30:26 +0000345}
346
turaj@webrtc.org6d5d2482013-10-06 04:47:28 +0000347} // namespace acm2
348
turaj@webrtc.org7959e162013-09-12 18:30:26 +0000349} // namespace webrtc