X-Git-Url: http://www.chiark.greenend.org.uk/ucgi/~mdw/git/disorder/blobdiff_plain/2d2effe25f5635bec27e00f556fd8eec123cfc9a..e84fb5f0c934ff2b8054a582c2ce09bcb05cba40:/server/speaker-network.c diff --git a/server/speaker-network.c b/server/speaker-network.c index 89412f5..c8edbfe 100644 --- a/server/speaker-network.c +++ b/server/speaker-network.c @@ -306,14 +306,16 @@ static size_t network_play(size_t frames) { static int bfd_slot; /** @brief Set up poll array for network play */ -static void network_beforepoll(void) { +static void network_beforepoll(int *timeoutp) { struct timeval now; uint64_t target_us; uint64_t target_rtp_time; + const int64_t samples_per_second = config->sample_format.rate + * config->sample_format.channels; const int64_t samples_ahead = ((uint64_t)RTP_AHEAD_MS - * config->sample_format.rate - * config->sample_format.channels + * samples_per_second / 1000); + int64_t lead, ahead_ms; /* If we're starting then initialize the base time */ if(!rtp_time) @@ -326,8 +328,17 @@ static void network_beforepoll(void) { target_rtp_time = (target_us * config->sample_format.rate * config->sample_format.channels) / 1000000; - if((int64_t)(rtp_time - target_rtp_time) < samples_ahead) + lead = rtp_time - target_rtp_time; + if(lead < samples_ahead) + /* We've not reached the desired lead, write as fast as we can */ bfd_slot = addfd(bfd, POLLOUT); + else { + /* We've reached the desired lead, we can afford to wait a bit even if the + * IP stack thinks it can accept more. */ + ahead_ms = 1000 * (lead - samples_ahead) / samples_per_second; + if(ahead_ms < *timeoutp) + *timeoutp = ahead_ms; + } } /** @brief Process poll() results for network play */