File: RTCRtpEncodingParameters-codec-opus-stereo.https.html

package info (click to toggle)
thunderbird 1%3A140.4.0esr-1
  • links: PTS, VCS
  • area: main
  • in suites: forky, sid
  • size: 4,609,432 kB
  • sloc: cpp: 7,672,442; javascript: 5,901,613; ansic: 3,898,954; python: 1,413,343; xml: 653,997; asm: 462,286; java: 180,927; sh: 113,489; makefile: 20,460; perl: 14,288; objc: 13,059; yacc: 4,583; pascal: 3,352; lex: 1,720; ruby: 1,222; exp: 762; sql: 715; awk: 580; php: 436; lisp: 430; sed: 70; csh: 10
file content (115 lines) | stat: -rw-r--r-- 4,459 bytes parent folder | download | duplicates (12)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
<!DOCTYPE html>
<meta charset="utf-8">
<title>RTCRtpEncodingParameters codec opus stereo</title>
<script src="/resources/testharness.js"></script>
<script src="/resources/testharnessreport.js"></script>
<script>
'use strict';

const kNumChannelsMono = 1;
const kNumChannelsStereo = 2;
const kAudioProcessingEnabled = true;
const kAudioProcessingDisabled = false;

// In some implementations (e.g. Chrome), whether or not audio processing is
// used may affect the number of channels we get.
// To isolate this factor from the test, we constrain both channels and APM.
function getAudioConstraints(numChannels, audioProcessing) {
  return {
    channelCount: {exact: numChannels},
    autoGainControl: audioProcessing,
    echoCancellation: audioProcessing,
    noiseSuppression: audioProcessing
  };
}

// Polls getSettings() until `channelCount` is known.
// Remote tracks don't populate their `channelCount` until media is received.
async function pollChannelCount(t, track) {
  while (true) {
    if (track.getSettings().channelCount != undefined) {
      break;
    }
    await new Promise(r => t.step_timeout(r, 50));
  }
  return track.getSettings().channelCount;
}

promise_test(async t => {
  const pc1 = new RTCPeerConnection();
  t.add_cleanup(() => pc1.close());
  const pc2 = new RTCPeerConnection();
  t.add_cleanup(() => pc2.close());
  pc1.onicecandidate = e => pc2.addIceCandidate(e.candidate);
  pc2.onicecandidate = e => pc1.addIceCandidate(e.candidate);

  // Open microphone as mono.
  const stream = await navigator.mediaDevices.getUserMedia(
      {audio: getAudioConstraints(kNumChannelsMono, kAudioProcessingEnabled)});
  const [track] = stream.getTracks();
  t.add_cleanup(() => track.stop());
  // Prerequisite of the test.
  assert_equals(track.getSettings().channelCount, 1,
                'Can open track in mono');

  // Negotiate opus.
  const transceiver = pc1.addTransceiver(track);
  transceiver.setCodecPreferences(
      RTCRtpSender.getCapabilities('audio').codecs.filter(
          codec => codec.mimeType == 'audio/opus'));
  await pc1.setLocalDescription();
  await pc2.setRemoteDescription(pc1.localDescription);
  await pc2.setLocalDescription();
  await pc1.setRemoteDescription(pc2.localDescription);

  const [receiver] = pc2.getReceivers();
  const remoteTrack = receiver.track;
  // Attaching the track to an audio element is needed for media to flow,
  // otherwise the `channelCount` is never known.
  const audio = document.createElement('audio');
  audio.srcObject = new MediaStream();
  audio.srcObject.addTrack(remoteTrack);
  // The stereo opus decoder outputs stereo regardless of the signal on the
  // wire.
  assert_equals(await pollChannelCount(t, remoteTrack), 2,
                'Remote track is stereo (despite local track being mono)');
}, 'Sending and receiving a mono track with opus (stereo decoder upmix)');

promise_test(async t => {
  const pc1 = new RTCPeerConnection();
  t.add_cleanup(() => pc1.close());
  const pc2 = new RTCPeerConnection();
  t.add_cleanup(() => pc2.close());
  pc1.onicecandidate = e => pc2.addIceCandidate(e.candidate);
  pc2.onicecandidate = e => pc1.addIceCandidate(e.candidate);

  // Open microphone as stereo.
  const stream = await navigator.mediaDevices.getUserMedia(
    {audio: getAudioConstraints(kNumChannelsStereo, kAudioProcessingDisabled)});
  const [track] = stream.getTracks();
  t.add_cleanup(() => track.stop());
  // Prerequisite of the test.
  assert_equals(track.getSettings().channelCount, 2,
                'Can open track in stereo');

  // Negotiate opus.
  const transceiver = pc1.addTransceiver(track);
  transceiver.setCodecPreferences(
      RTCRtpSender.getCapabilities('audio').codecs.filter(
          codec => codec.mimeType == 'audio/opus'));
  await pc1.setLocalDescription();
  await pc2.setRemoteDescription(pc1.localDescription);
  await pc2.setLocalDescription();
  await pc1.setRemoteDescription(pc2.localDescription);

  const [receiver] = pc2.getReceivers();
  const remoteTrack = receiver.track;
  // Attaching the track to an audio element is needed for media to flow,
  // otherwise the `channelCount` is never known.
  const audio = document.createElement('audio');
  audio.srcObject = new MediaStream();
  audio.srcObject.addTrack(remoteTrack);
  assert_equals(await pollChannelCount(t, remoteTrack), 2,
                'Remote track is also stereo');
}, 'Sending and receiving a stereo track with opus');
</script>