decoder/API: move functions into class DecoderBridge

This commit is contained in:
Max Kellermann 2016-11-21 21:44:57 +01:00
parent 697c3f8cb9
commit 8c342a764b
2 changed files with 63 additions and 62 deletions

View File

@ -138,6 +138,29 @@ public:
DecoderCommand SubmitTag(InputStream *is, Tag &&tag) override ; DecoderCommand SubmitTag(InputStream *is, Tag &&tag) override ;
void SubmitReplayGain(const ReplayGainInfo *replay_gain_info) override; void SubmitReplayGain(const ReplayGainInfo *replay_gain_info) override;
void SubmitMixRamp(MixRampInfo &&mix_ramp) override; void SubmitMixRamp(MixRampInfo &&mix_ramp) override;
private:
/**
* Checks if we need an "initial seek". If so, then the
* initial seek is prepared, and the function returns true.
*/
bool PrepareInitialSeek();
/**
* Returns the current decoder command. May return a
* "virtual" synthesized command, e.g. to seek to the
* beginning of the CUE track.
*/
DecoderCommand GetVirtualCommand();
DecoderCommand LockGetVirtualCommand();
/**
* Sends a #Tag as-is to the #MusicPipe. Flushes the current
* chunk (DecoderBridge::chunk) if there is one.
*/
DecoderCommand DoSendTag(const Tag &tag);
bool UpdateStreamTag(InputStream *is);
}; };
#endif #endif

View File

@ -83,15 +83,9 @@ DecoderBridge::Ready(const AudioFormat audio_format,
dc.client_cond.signal(); dc.client_cond.signal();
} }
/** bool
* Checks if we need an "initial seek". If so, then the initial seek DecoderBridge::PrepareInitialSeek()
* is prepared, and the function returns true.
*/
gcc_pure
static bool
decoder_prepare_initial_seek(DecoderBridge &bridge)
{ {
const DecoderControl &dc = bridge.dc;
assert(dc.pipe != nullptr); assert(dc.pipe != nullptr);
if (dc.state != DecoderState::DECODE) if (dc.state != DecoderState::DECODE)
@ -100,69 +94,61 @@ decoder_prepare_initial_seek(DecoderBridge &bridge)
virtual "SEEK" command */ virtual "SEEK" command */
return false; return false;
if (bridge.initial_seek_running) if (initial_seek_running)
/* initial seek has already begun - override any other /* initial seek has already begun - override any other
command */ command */
return true; return true;
if (bridge.initial_seek_pending) { if (initial_seek_pending) {
if (!dc.seekable) { if (!dc.seekable) {
/* seeking is not possible */ /* seeking is not possible */
bridge.initial_seek_pending = false; initial_seek_pending = false;
return false; return false;
} }
if (dc.command == DecoderCommand::NONE) { if (dc.command == DecoderCommand::NONE) {
/* begin initial seek */ /* begin initial seek */
bridge.initial_seek_pending = false; initial_seek_pending = false;
bridge.initial_seek_running = true; initial_seek_running = true;
return true; return true;
} }
/* skip initial seek when there's another command /* skip initial seek when there's another command
(e.g. STOP) */ (e.g. STOP) */
bridge.initial_seek_pending = false; initial_seek_pending = false;
} }
return false; return false;
} }
/** DecoderCommand
* Returns the current decoder command. May return a "virtual" DecoderBridge::GetVirtualCommand()
* synthesized command, e.g. to seek to the beginning of the CUE
* track.
*/
gcc_pure
static DecoderCommand
decoder_get_virtual_command(DecoderBridge &bridge)
{ {
if (bridge.error) if (error)
/* an error has occurred: stop the decoder plugin */ /* an error has occurred: stop the decoder plugin */
return DecoderCommand::STOP; return DecoderCommand::STOP;
const DecoderControl &dc = bridge.dc;
assert(dc.pipe != nullptr); assert(dc.pipe != nullptr);
if (decoder_prepare_initial_seek(bridge)) if (PrepareInitialSeek())
return DecoderCommand::SEEK; return DecoderCommand::SEEK;
return dc.command; return dc.command;
} }
gcc_pure DecoderCommand
static DecoderCommand DecoderBridge::LockGetVirtualCommand()
decoder_lock_get_virtual_command(DecoderBridge &bridge)
{ {
const ScopeLock protect(bridge.dc.mutex); const ScopeLock protect(dc.mutex);
return decoder_get_virtual_command(bridge); return GetVirtualCommand();
} }
DecoderCommand DecoderCommand
DecoderBridge::GetCommand() DecoderBridge::GetCommand()
{ {
return decoder_lock_get_virtual_command(*this); return LockGetVirtualCommand();
} }
void void
@ -382,43 +368,35 @@ DecoderBridge::SubmitTimestamp(double t)
timestamp = t; timestamp = t;
} }
/** DecoderCommand
* Sends a #tag as-is to the music pipe. Flushes the current chunk DecoderBridge::DoSendTag(const Tag &tag)
* (DecoderBridge::chunk) if there is one.
*/
static DecoderCommand
do_send_tag(DecoderClient &client, const Tag &tag)
{ {
auto &bridge = (DecoderBridge &)client; if (current_chunk != nullptr) {
if (bridge.current_chunk != nullptr) {
/* there is a partial chunk - flush it, we want the /* there is a partial chunk - flush it, we want the
tag in a new chunk */ tag in a new chunk */
bridge.FlushChunk(); FlushChunk();
} }
assert(bridge.current_chunk == nullptr); assert(current_chunk == nullptr);
auto *chunk = bridge.GetChunk(); auto *chunk = GetChunk();
if (chunk == nullptr) { if (chunk == nullptr) {
assert(bridge.dc.command != DecoderCommand::NONE); assert(dc.command != DecoderCommand::NONE);
return bridge.dc.command; return dc.command;
} }
chunk->tag = new Tag(tag); chunk->tag = new Tag(tag);
return DecoderCommand::NONE; return DecoderCommand::NONE;
} }
static bool bool
update_stream_tag(DecoderClient &client, InputStream *is) DecoderBridge::UpdateStreamTag(InputStream *is)
{ {
auto &bridge = (DecoderBridge &)client;
auto *tag = is != nullptr auto *tag = is != nullptr
? is->LockReadTag() ? is->LockReadTag()
: nullptr; : nullptr;
if (tag == nullptr) { if (tag == nullptr) {
tag = bridge.song_tag; tag = song_tag;
if (tag == nullptr) if (tag == nullptr)
return false; return false;
@ -426,12 +404,12 @@ update_stream_tag(DecoderClient &client, InputStream *is)
instead */ instead */
} else } else
/* discard the song tag; we don't need it */ /* discard the song tag; we don't need it */
delete bridge.song_tag; delete song_tag;
bridge.song_tag = nullptr; song_tag = nullptr;
delete bridge.stream_tag; delete stream_tag;
bridge.stream_tag = tag; stream_tag = tag;
return true; return true;
} }
@ -444,7 +422,7 @@ DecoderBridge::SubmitData(InputStream *is,
assert(dc.pipe != nullptr); assert(dc.pipe != nullptr);
assert(length % dc.in_audio_format.GetFrameSize() == 0); assert(length % dc.in_audio_format.GetFrameSize() == 0);
DecoderCommand cmd = decoder_lock_get_virtual_command(*this); DecoderCommand cmd = LockGetVirtualCommand();
if (cmd == DecoderCommand::STOP || cmd == DecoderCommand::SEEK || if (cmd == DecoderCommand::STOP || cmd == DecoderCommand::SEEK ||
length == 0) length == 0)
@ -455,16 +433,16 @@ DecoderBridge::SubmitData(InputStream *is,
/* send stream tags */ /* send stream tags */
if (update_stream_tag(*this, is)) { if (UpdateStreamTag(is)) {
if (decoder_tag != nullptr) { if (decoder_tag != nullptr) {
/* merge with tag from decoder plugin */ /* merge with tag from decoder plugin */
Tag *tag = Tag::Merge(*decoder_tag, Tag *tag = Tag::Merge(*decoder_tag,
*stream_tag); *stream_tag);
cmd = do_send_tag(*this, *tag); cmd = DoSendTag(*tag);
delete tag; delete tag;
} else } else
/* send only the stream tag */ /* send only the stream tag */
cmd = do_send_tag(*this, *stream_tag); cmd = DoSendTag(*stream_tag);
if (cmd != DecoderCommand::NONE) if (cmd != DecoderCommand::NONE)
return cmd; return cmd;
@ -553,11 +531,11 @@ DecoderBridge::SubmitTag(InputStream *is, Tag &&tag)
/* check for a new stream tag */ /* check for a new stream tag */
update_stream_tag(*this, is); UpdateStreamTag(is);
/* check if we're seeking */ /* check if we're seeking */
if (decoder_prepare_initial_seek(*this)) if (PrepareInitialSeek())
/* during initial seek, no music chunk must be created /* during initial seek, no music chunk must be created
until seeking is finished; skip the rest of the until seeking is finished; skip the rest of the
function here */ function here */
@ -570,11 +548,11 @@ DecoderBridge::SubmitTag(InputStream *is, Tag &&tag)
Tag *merged; Tag *merged;
merged = Tag::Merge(*stream_tag, *decoder_tag); merged = Tag::Merge(*stream_tag, *decoder_tag);
cmd = do_send_tag(*this, *merged); cmd = DoSendTag(*merged);
delete merged; delete merged;
} else } else
/* send only the decoder tag */ /* send only the decoder tag */
cmd = do_send_tag(*this, *decoder_tag); cmd = DoSendTag(*decoder_tag);
return cmd; return cmd;
} }