| // META: global=window |
| // META: script=/webcodecs/utils.js |
| |
| function make_silent_audio_data(timestamp, channels, sampleRate, frames) { |
| let data = new Float32Array(frames*channels); |
| |
| return new AudioData({ |
| timestamp: timestamp, |
| data: data, |
| numberOfChannels: channels, |
| numberOfFrames: frames, |
| sampleRate: sampleRate, |
| format: "f32-planar", |
| }); |
| } |
| |
| // The Opus DTX flag (discontinuous transmission) reduces the encoding bitrate |
| // for silence. This test ensures the DTX flag is working properly by encoding |
| // almost 10s of silence and comparing the bitrate with and without the flag. |
| promise_test(async t => { |
| let sample_rate = 48000; |
| let total_duration_s = 10; |
| let data_count = 100; |
| let normal_outputs = []; |
| let dtx_outputs = []; |
| |
| let normal_encoder = new AudioEncoder({ |
| error: e => { |
| assert_unreached('error: ' + e); |
| }, |
| output: chunk => { |
| normal_outputs.push(chunk); |
| } |
| }); |
| |
| let dtx_encoder = new AudioEncoder({ |
| error: e => { |
| assert_unreached('error: ' + e); |
| }, |
| output: chunk => { |
| dtx_outputs.push(chunk); |
| } |
| }); |
| |
| let config = { |
| codec: 'opus', |
| sampleRate: sample_rate, |
| numberOfChannels: 2, |
| bitrate: 256000, // 256kbit |
| }; |
| |
| let normal_config = {...config, opus: {usedtx: false}}; |
| let dtx_config = {...config, opus: {usedtx: true}}; |
| |
| let normal_config_support = await AudioEncoder.isConfigSupported(normal_config); |
| assert_implements_optional(normal_config_support.supported, "Opus not supported"); |
| |
| let dtx_config_support = await AudioEncoder.isConfigSupported(dtx_config); |
| assert_implements_optional(dtx_config_support.supported, "Opus DTX not supported"); |
| |
| // Configure one encoder with and one without the DTX flag |
| normal_encoder.configure(normal_config); |
| dtx_encoder.configure(dtx_config); |
| |
| let timestamp_us = 0; |
| let data_duration_s = total_duration_s / data_count; |
| let data_length = data_duration_s * config.sampleRate; |
| for (let i = 0; i < data_count; i++) { |
| let data; |
| |
| if (i == 0 || i == (data_count - 1)) { |
| // Send real data for the first and last 100ms. |
| data = make_audio_data( |
| timestamp_us, config.numberOfChannels, config.sampleRate, |
| data_length); |
| |
| } else { |
| // Send silence for the rest of the 10s. |
| data = make_silent_audio_data( |
| timestamp_us, config.numberOfChannels, config.sampleRate, |
| data_length); |
| } |
| |
| normal_encoder.encode(data); |
| dtx_encoder.encode(data); |
| data.close(); |
| |
| timestamp_us += data_duration_s * 1_000_000; |
| } |
| |
| await Promise.all([normal_encoder.flush(), dtx_encoder.flush()]) |
| |
| normal_encoder.close(); |
| dtx_encoder.close(); |
| |
| // We expect a significant reduction in the number of packets, over ~10s of silence. |
| assert_less_than(dtx_outputs.length, (normal_outputs.length / 2)); |
| }, 'Test the Opus DTX flag works.'); |
| |
| |
| // The Opus bitrateMode enum chooses whether we use a constant or variable bitrate. |
| // This test ensures that VBR/CBR is respected properly by encoding almost 10s of |
| // silence and comparing the size of the encoded variable or constant bitrates. |
| promise_test(async t => { |
| let sample_rate = 48000; |
| let total_duration_s = 10; |
| let data_count = 100; |
| let vbr_outputs = []; |
| let cbr_outputs = []; |
| |
| let cbr_encoder = new AudioEncoder({ |
| error: e => { |
| assert_unreached('error: ' + e); |
| }, |
| output: chunk => { |
| cbr_outputs.push(chunk); |
| } |
| }); |
| |
| let vbr_encoder = new AudioEncoder({ |
| error: e => { |
| assert_unreached('error: ' + e); |
| }, |
| output: chunk => { |
| vbr_outputs.push(chunk); |
| } |
| }); |
| |
| let config = { |
| codec: 'opus', |
| sampleRate: sample_rate, |
| numberOfChannels: 2, |
| bitrate: 256000, // 256kbit |
| }; |
| |
| let cbr_config = { ...config, bitrateMode: "constant" }; |
| let vbr_config = { ...config, bitrateMode: "variable" }; |
| |
| let cbr_config_support = await AudioEncoder.isConfigSupported(cbr_config); |
| assert_implements_optional(cbr_config_support.supported, "Opus CBR not supported"); |
| |
| let vbr_config_support = await AudioEncoder.isConfigSupported(vbr_config); |
| assert_implements_optional(vbr_config_support.supported, "Opus VBR not supported"); |
| |
| // Configure one encoder with VBR and one CBR. |
| cbr_encoder.configure(cbr_config); |
| vbr_encoder.configure(vbr_config); |
| |
| let timestamp_us = 0; |
| let data_duration_s = total_duration_s / data_count; |
| let data_length = data_duration_s * config.sampleRate; |
| for (let i = 0; i < data_count; i++) { |
| let data; |
| |
| if (i == 0 || i == (data_count - 1)) { |
| // Send real data for the first and last 100ms. |
| data = make_audio_data( |
| timestamp_us, config.numberOfChannels, config.sampleRate, |
| data_length); |
| |
| } else { |
| // Send silence for the rest of the 10s. |
| data = make_silent_audio_data( |
| timestamp_us, config.numberOfChannels, config.sampleRate, |
| data_length); |
| } |
| |
| vbr_encoder.encode(data); |
| cbr_encoder.encode(data); |
| data.close(); |
| |
| timestamp_us += data_duration_s * 1_000_000; |
| } |
| |
| await Promise.all([cbr_encoder.flush(), vbr_encoder.flush()]) |
| |
| cbr_encoder.close(); |
| vbr_encoder.close(); |
| |
| let vbr_total_bytes = 0; |
| vbr_outputs.forEach(chunk => vbr_total_bytes += chunk.byteLength) |
| |
| let cbr_total_bytes = 0; |
| cbr_outputs.forEach(chunk => cbr_total_bytes += chunk.byteLength) |
| |
| // We expect a significant reduction in the size of the packets, over ~10s of silence. |
| assert_less_than(vbr_total_bytes, (cbr_total_bytes / 2)); |
| }, 'Test the Opus bitrateMode flag works.'); |
| |
| |
| // The AAC bitrateMode enum chooses whether we use a constant or variable bitrate. |
| // This test exercises the VBR/CBR paths. Some platforms don't support VBR for AAC, |
| // and still emit a constant bitrate. |
| promise_test(async t => { |
| let sample_rate = 48000; |
| let total_duration_s = 10; |
| let data_count = 100; |
| let vbr_outputs = []; |
| let cbr_outputs = []; |
| |
| let cbr_encoder = new AudioEncoder({ |
| error: e => { |
| assert_unreached('error: ' + e); |
| }, |
| output: chunk => { |
| cbr_outputs.push(chunk); |
| } |
| }); |
| |
| let vbr_encoder = new AudioEncoder({ |
| error: e => { |
| assert_unreached('error: ' + e); |
| }, |
| output: chunk => { |
| vbr_outputs.push(chunk); |
| } |
| }); |
| |
| let config = { |
| codec: 'mp4a.40.2', |
| sampleRate: sample_rate, |
| numberOfChannels: 2, |
| bitrate: 192000, // 256kbit |
| }; |
| |
| let cbr_config = { ...config, bitrateMode: "constant" }; |
| let vbr_config = { ...config, bitrateMode: "variable" }; |
| |
| let cbr_config_support = await AudioEncoder.isConfigSupported(cbr_config); |
| assert_implements_optional(cbr_config_support.supported, "AAC CBR not supported"); |
| |
| let vbr_config_support = await AudioEncoder.isConfigSupported(vbr_config); |
| assert_implements_optional(vbr_config_support.supported, "AAC VBR not supported"); |
| |
| // Configure one encoder with VBR and one CBR. |
| cbr_encoder.configure(cbr_config); |
| vbr_encoder.configure(vbr_config); |
| |
| let timestamp_us = 0; |
| let data_duration_s = total_duration_s / data_count; |
| let data_length = data_duration_s * config.sampleRate; |
| for (let i = 0; i < data_count; i++) { |
| let data; |
| |
| if (i == 0 || i == (data_count - 1)) { |
| // Send real data for the first and last 100ms. |
| data = make_audio_data( |
| timestamp_us, config.numberOfChannels, config.sampleRate, |
| data_length); |
| |
| } else { |
| // Send silence for the rest of the 10s. |
| data = make_silent_audio_data( |
| timestamp_us, config.numberOfChannels, config.sampleRate, |
| data_length); |
| } |
| |
| vbr_encoder.encode(data); |
| cbr_encoder.encode(data); |
| data.close(); |
| |
| timestamp_us += data_duration_s * 1_000_000; |
| } |
| |
| await Promise.all([cbr_encoder.flush(), vbr_encoder.flush()]) |
| |
| cbr_encoder.close(); |
| vbr_encoder.close(); |
| |
| let vbr_total_bytes = 0; |
| vbr_outputs.forEach(chunk => vbr_total_bytes += chunk.byteLength) |
| |
| let cbr_total_bytes = 0; |
| cbr_outputs.forEach(chunk => cbr_total_bytes += chunk.byteLength) |
| |
| // We'd like to confirm that the encoded size using VBR is less than CBR, but |
| // platforms without VBR support will silently revert to CBR (which is |
| // technically a subset of VBR). |
| assert_less_than_equal(vbr_total_bytes, cbr_total_bytes); |
| }, 'Test the AAC bitrateMode flag works.'); |