release v0.19.10

-----BEGIN PGP SIGNATURE-----
 Version: GnuPG v2
 
 iQIcBAABCAAGBQJVhsF2AAoJECNuiljG20US2OUQAIyDcaUCFgUa2CYa7MftCPGc
 dYJFwf+6Y2fJdSXcogYZmXNKNjFbPAZ4qqGdoNfI5a0rLxMarmL4DyjziAWi3ETB
 MOkZK65Y5ySyyw69e+i/XsOwOn8rm6jDuwHKpT7wVJNjvZ8nA8esvu5b1Ief5LBd
 UwEmn5DtKjA7dErHEYd2YvMK0xm+YmoKXuhmJKAn3sQdCEldgH4T5BCdOqmfrHWX
 BYmNxmP3PU9Tqi7XHTSFZJn6vWiXhOoWr4Cb7K54j49sRV2B4QMWX1CLyK4+Jwmk
 NZwD1IoGtoks5twfMTA9F9dBV/CPAWT69E0LIvaFJwCyoPCEnEi6k41bRAWK7P65
 QwKxdtY/GZnVFpiqXba+wkD6VBa5wmkjS10+cIBhz3CVCEE+N5YacubUw5JieYg3
 kOG2htSF1YP/Mo+IAObO9doQWHnDUavVhhpQ8UyqQ1bDMsmaEpfE16TRmlY/l5wx
 Aor2p6D3c0E0IGpEwjOl6T9pDql4dyTdrRLLXJ6oD9iYv2rDdahctBRdyFZ1mRwX
 oNUz8bfGDrshHVvwjQTr2b4O+w+yc+RSuJcyCGGcn5LakBuiM6vYNYuZzq3Yj/RK
 Wk7RErVsbtY4ZRH06Lf5MSM5TflnrfQmzkUB0rZ0XoDyweoHOHPyzKhvBaKhadNh
 UnEx4kCOvWdjFXUVWH3Q
 =MrVf
 -----END PGP SIGNATURE-----

Merge tag 'v0.19.10'
This commit is contained in:
Max Kellermann 2015-06-21 16:01:44 +02:00
commit 90e7ace980
11 changed files with 108 additions and 16 deletions

15
NEWS
View File

@ -30,6 +30,21 @@ ver 0.20 (not yet released)
* database * database
- proxy: add TCP keepalive option - proxy: add TCP keepalive option
ver 0.19.10 (2015/06/21)
* input
- curl: fix deadlock on small responses
- smbclient: fix DFF playback
* decoder
- ffmpeg: improve seeking accuracy
- fix stuck stream tags
* encoder
- opus: fix bogus granulepos
* output
- fix failure to open device right after booting
* neighbor
- nfs: fix deadlock when connecting
* fix "single" mode breakage due to queue edits
ver 0.19.9 (2015/02/06) ver 0.19.9 (2015/02/06)
* decoder * decoder
- dsdiff, dsf: raise ID3 tag limit to 1 MB - dsdiff, dsf: raise ID3 tag limit to 1 MB

View File

@ -612,6 +612,12 @@ Player::ProcessCommand()
queued = true; queued = true;
pc.CommandFinished(); pc.CommandFinished();
pc.Unlock();
if (dc.LockIsIdle())
StartDecoder(*new MusicPipe());
pc.Lock();
break; break;
case PlayerCommand::PAUSE: case PlayerCommand::PAUSE:

View File

@ -433,8 +433,11 @@ update_stream_tag(Decoder &decoder, InputStream *is)
/* no stream tag present - submit the song tag /* no stream tag present - submit the song tag
instead */ instead */
decoder.song_tag = nullptr; } else
} /* discard the song tag; we don't need it */
delete decoder.song_tag;
decoder.song_tag = nullptr;
delete decoder.stream_tag; delete decoder.stream_tag;
decoder.stream_tag = tag; decoder.stream_tag = tag;
@ -566,7 +569,7 @@ decoder_tag(Decoder &decoder, InputStream *is,
/* save the tag */ /* save the tag */
delete decoder.decoder_tag; delete decoder.decoder_tag;
decoder.decoder_tag = new Tag(tag); decoder.decoder_tag = new Tag(std::move(tag));
/* check for a new stream tag */ /* check for a new stream tag */

View File

@ -378,7 +378,11 @@ decoder_run_song(DecoderControl &dc,
const DetachedSong &song, const char *uri, Path path_fs) const DetachedSong &song, const char *uri, Path path_fs)
{ {
Decoder decoder(dc, dc.start_time.IsPositive(), Decoder decoder(dc, dc.start_time.IsPositive(),
new Tag(song.GetTag())); /* pass the song tag only if it's
authoritative, i.e. if it's a local file -
tags on "stream" songs are just remembered
from the last time we played it*/
song.IsFile() ? new Tag(song.GetTag()) : nullptr);
dc.state = DecoderState::START; dc.state = DecoderState::START;

View File

@ -162,9 +162,41 @@ copy_interleave_frame(const AVCodecContext &codec_context,
return { output_buffer, (size_t)data_size }; return { output_buffer, (size_t)data_size };
} }
/**
* Convert AVPacket::pts to a stream-relative time stamp (still in
* AVStream::time_base units). Returns a negative value on error.
*/
gcc_pure
static int64_t
StreamRelativePts(const AVPacket &packet, const AVStream &stream)
{
auto pts = packet.pts;
if (pts < 0 || pts == int64_t(AV_NOPTS_VALUE))
return -1;
auto start = start_time_fallback(stream);
return pts - start;
}
/**
* Convert a non-negative stream-relative time stamp in
* AVStream::time_base units to a PCM frame number.
*/
gcc_pure
static uint64_t
PtsToPcmFrame(uint64_t pts, const AVStream &stream,
const AVCodecContext &codec_context)
{
return av_rescale_q(pts, stream.time_base, codec_context.time_base);
}
/** /**
* Decode an #AVPacket and send the resulting PCM data to the decoder * Decode an #AVPacket and send the resulting PCM data to the decoder
* API. * API.
*
* @param min_frame skip all data before this PCM frame number; this
* is used after seeking to skip data in an AVPacket until the exact
* desired time stamp has been reached
*/ */
static DecoderCommand static DecoderCommand
ffmpeg_send_packet(Decoder &decoder, InputStream &is, ffmpeg_send_packet(Decoder &decoder, InputStream &is,
@ -172,13 +204,21 @@ ffmpeg_send_packet(Decoder &decoder, InputStream &is,
AVCodecContext &codec_context, AVCodecContext &codec_context,
const AVStream &stream, const AVStream &stream,
AVFrame &frame, AVFrame &frame,
uint64_t min_frame, size_t pcm_frame_size,
FfmpegBuffer &buffer) FfmpegBuffer &buffer)
{ {
if (packet.pts >= 0 && packet.pts != (int64_t)AV_NOPTS_VALUE) { size_t skip_bytes = 0;
auto start = start_time_fallback(stream);
if (packet.pts >= start) const auto pts = StreamRelativePts(packet, stream);
if (pts >= 0) {
if (min_frame > 0) {
auto cur_frame = PtsToPcmFrame(pts, stream,
codec_context);
if (cur_frame < min_frame)
skip_bytes = pcm_frame_size * (min_frame - cur_frame);
} else
decoder_timestamp(decoder, decoder_timestamp(decoder,
FfmpegTimeToDouble(packet.pts - start, FfmpegTimeToDouble(pts,
stream.time_base)); stream.time_base));
} }
@ -212,6 +252,18 @@ ffmpeg_send_packet(Decoder &decoder, InputStream &is,
return DecoderCommand::STOP; return DecoderCommand::STOP;
} }
if (skip_bytes > 0) {
if (skip_bytes >= output_buffer.size) {
skip_bytes -= output_buffer.size;
continue;
}
output_buffer.data =
(const uint8_t *)output_buffer.data + skip_bytes;
output_buffer.size -= skip_bytes;
skip_bytes = 0;
}
cmd = decoder_data(decoder, is, cmd = decoder_data(decoder, is,
output_buffer.data, output_buffer.size, output_buffer.data, output_buffer.size,
codec_context.bit_rate / 1000); codec_context.bit_rate / 1000);
@ -489,6 +541,8 @@ FfmpegDecode(Decoder &decoder, InputStream &input,
FfmpegBuffer interleaved_buffer; FfmpegBuffer interleaved_buffer;
uint64_t min_frame = 0;
DecoderCommand cmd; DecoderCommand cmd;
do { do {
AVPacket packet; AVPacket packet;
@ -500,13 +554,15 @@ FfmpegDecode(Decoder &decoder, InputStream &input,
FfmpegCheckTag(decoder, input, format_context, audio_stream); FfmpegCheckTag(decoder, input, format_context, audio_stream);
#endif #endif
if (packet.stream_index == audio_stream) if (packet.stream_index == audio_stream) {
cmd = ffmpeg_send_packet(decoder, input, cmd = ffmpeg_send_packet(decoder, input,
packet, codec_context, packet, codec_context,
av_stream, av_stream,
*frame, *frame,
min_frame, audio_format.GetFrameSize(),
interleaved_buffer); interleaved_buffer);
else min_frame = 0;
} else
cmd = decoder_get_command(decoder); cmd = decoder_get_command(decoder);
av_free_packet(&packet); av_free_packet(&packet);
@ -517,11 +573,15 @@ FfmpegDecode(Decoder &decoder, InputStream &input,
av_stream.time_base) + av_stream.time_base) +
start_time_fallback(av_stream); start_time_fallback(av_stream);
/* AVSEEK_FLAG_BACKWARD asks FFmpeg to seek to
the packet boundary before the seek time
stamp, not after */
if (av_seek_frame(&format_context, audio_stream, where, if (av_seek_frame(&format_context, audio_stream, where,
AVSEEK_FLAG_ANY) < 0) AVSEEK_FLAG_ANY|AVSEEK_FLAG_BACKWARD) < 0)
decoder_seek_error(decoder); decoder_seek_error(decoder);
else { else {
avcodec_flush_buffers(&codec_context); avcodec_flush_buffers(&codec_context);
min_frame = decoder_seek_where_frame(decoder);
decoder_command_finished(decoder); decoder_command_finished(decoder);
} }
} }

View File

@ -66,7 +66,7 @@ struct opus_encoder {
ogg_int64_t granulepos; ogg_int64_t granulepos;
opus_encoder():encoder(opus_encoder_plugin) {} opus_encoder():encoder(opus_encoder_plugin), granulepos(0) {}
}; };
static constexpr Domain opus_encoder_domain("opus_encoder"); static constexpr Domain opus_encoder_domain("opus_encoder");

View File

@ -453,6 +453,8 @@ CurlInputStream::RequestDone(CURLcode result, long status)
SeekDone(); SeekDone();
else if (!IsReady()) else if (!IsReady())
SetReady(); SetReady();
else
cond.broadcast();
} }
static void static void

View File

@ -132,6 +132,7 @@ SmbclientInputStream::Read(void *ptr, size_t read_size, Error &error)
nbytes = 0; nbytes = 0;
} }
offset += nbytes;
return nbytes; return nbytes;
} }

View File

@ -186,7 +186,8 @@ AudioOutput::LockUpdate(const AudioFormat audio_format,
const ScopeLock protect(mutex); const ScopeLock protect(mutex);
if (enabled && really_enabled) { if (enabled && really_enabled) {
if (fail_timer.Check(REOPEN_AFTER * 1000)) { if (!fail_timer.IsDefined() ||
fail_timer.Check(REOPEN_AFTER * 1000)) {
return Open(audio_format, mp); return Open(audio_format, mp);
} }
} else if (IsOpen()) } else if (IsOpen())

View File

@ -177,6 +177,8 @@ private:
mutex.unlock(); mutex.unlock();
DeferredMonitor::Schedule(); DeferredMonitor::Schedule();
mutex.lock(); mutex.lock();
if (state == State::INITIAL)
cond.wait(mutex);
break; break;
case State::CONNECTING: case State::CONNECTING:
@ -188,8 +190,6 @@ private:
error.Set(last_error); error.Set(last_error);
return false; return false;
} }
cond.wait(mutex);
} }
} }

View File

@ -63,7 +63,7 @@ protected:
} }
public: public:
bool IsDefined() const { constexpr bool IsDefined() const {
return last != 0; return last != 0;
} }