Choppy/inaudible playback with chunked audio through Web Audio API

后端 未结 1 1015
青春惊慌失措
青春惊慌失措 2020-12-29 12:08

I brought this up in my last post but since it was off topic from the original question I\'m posting it separately. I\'m having trouble with getting my transmitted audio to

相关标签:
1条回答
  • 2020-12-29 12:53

    You really can't just call source.start(audioContext.currentTime) like that.

    setTimeout() has a long and imprecise latency - other main-thread stuff can be going on, so your setTimeout() calls can be delayed by milliseconds, even tens of milliseconds (by garbage collection, JS execution, layout...) Your code is trying to immediately play audio - which needs to be started within about 0.02ms accuracy to not glitch - on a timer that has tens of milliseconds of imprecision.

    The whole point of the web audio system is that the audio scheduler works in a separate high-priority thread, and you can pre-schedule audio (starts, stops, and audioparam changes) at very high accuracy. You should rewrite your system to:

    1) track when the first block was scheduled in audiocontext time - and DON'T schedule the first block immediately, give some latency so your network can hopefully keep up.

    2) schedule each successive block received in the future based on its "next block" timing.

    e.g. (note I haven't tested this code, this is off the top of my head):

    window.AudioContext = window.AudioContext || window.webkitAudioContext;
    var context = new AudioContext();
    var delayTime = 0;
    var init = 0;
    var audioStack = [];
    var nextTime = 0;
    
    client.on('stream', function(stream, meta){
        stream.on('data', function(data) {
            context.decodeAudioData(data, function(buffer) {
                audioStack.push(buffer);
                if ((init!=0) || (audioStack.length > 10)) { // make sure we put at least 10 chunks in the buffer before starting
                    init++;
                    scheduleBuffers();
                }
            }, function(err) {
                console.log("err(decodeAudioData): "+err);
            });
        });
    });
    
    function scheduleBuffers() {
        while ( audioStack.length) {
            var buffer = audioStack.shift();
            var source    = context.createBufferSource();
            source.buffer = buffer;
            source.connect(context.destination);
            if (nextTime == 0)
                nextTime = context.currentTime + 0.05;  /// add 50ms latency to work well across systems - tune this if you like
            source.start(nextTime);
            nextTime+=source.buffer.duration; // Make the next buffer wait the length of the last buffer before being played
        };
    }
    
    0 讨论(0)
提交回复
热议问题