decoder/API: move functions into class DecoderBridge
This commit is contained in:
parent
697c3f8cb9
commit
8c342a764b
@ -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
|
||||||
|
@ -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;
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user