From 72033f38227c21e25863e78d294900adfc269a45 Mon Sep 17 00:00:00 2001 From: Chris Spencer <spencercw@gmail.com> Date: Sun, 27 Feb 2022 17:09:58 +0000 Subject: [PATCH] [client] audio: reduce hardcoded minimum latency The current minimum target latency is partially based upon the default qemu behaviour whereby audio packets are delivered in a sawtooth pattern, with packet timestamps drifting between 5ms above and below the measured clock. This 5ms error is baked into the minimum target latency to avoid underrunning. This sawtooth pattern can be reduced by specifying a lower timer period in the qemu configuration, so remove it from the hardcoded minimum latency and add it to the default configurable buffer latency instead. This allows users that have configured their VM appropriately to reduce the overall latency. --- client/src/audio.c | 20 ++++---------------- client/src/config.c | 2 +- 2 files changed, 5 insertions(+), 17 deletions(-) diff --git a/client/src/audio.c b/client/src/audio.c index 9416ab48..f2e45d24 100644 --- a/client/src/audio.c +++ b/client/src/audio.c @@ -516,24 +516,12 @@ void audio_playbackData(uint8_t * data, size_t size) spiceData->devNextPosition = deviceTick.nextPosition; } - /* Determine the target latency. This is made up of three components: - * 1. Half the Spice period. This is necessary due to the way qemu handles - * audio. Data is not sent as soon as it is produced by the virtual sound - * card; instead, qemu polls for new data every ~10ms. This results in a - * sawtooth pattern in the packet timing as it drifts in and out of phase - * with the virtual device. LG measures the average progression of the - * Spice clock, so sees the packet timing error drift by half a period - * above and below the measured clock. We need to account for this in the - * target latency to avoid underrunning. - * 2. The maximum audio device period, plus a little extra to absorb timing - * jitter. - * 3. A configurable additional buffer period. The default value is set high - * enough to absorb typical timing jitter from Spice, which can be quite - * significant. Users may reduce this if they care more about latency than - * audio quality. */ + /* Determine the target latency. This is made up of the maximum audio device + * period (plus a little extra to absorb timing jitter) and a configurable + * additional buffer period. The default is set high enough to absorb typical + * timing jitter from qemu. */ int configLatencyMs = max(g_params.audioBufferLatency, 0); double targetLatencyFrames = - spiceData->periodFrames / 2.0 + audio.playback.deviceMaxPeriodFrames * 1.1 + configLatencyMs * audio.playback.sampleRate / 1000.0; diff --git a/client/src/config.c b/client/src/config.c index 7772f3ec..9504fda1 100644 --- a/client/src/config.c +++ b/client/src/config.c @@ -478,7 +478,7 @@ static struct Option options[] = .name = "bufferLatency", .description = "Additional buffer latency in milliseconds", .type = OPTION_TYPE_INT, - .value.x_int = 8 + .value.x_int = 13 }, {0} };