// 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.');