decoder/OggUtil,Opus: move code to new class OggSyncState
This commit is contained in:
parent
8f7adf79a3
commit
bf4311cd9b
@ -19,16 +19,15 @@
|
|||||||
|
|
||||||
#include "config.h"
|
#include "config.h"
|
||||||
#include "OggFind.hxx"
|
#include "OggFind.hxx"
|
||||||
#include "OggUtil.hxx"
|
#include "OggSyncState.hxx"
|
||||||
|
|
||||||
bool
|
bool
|
||||||
OggFindEOS(ogg_sync_state &oy, ogg_stream_state &os, ogg_packet &packet,
|
OggFindEOS(OggSyncState &oy, ogg_stream_state &os, ogg_packet &packet)
|
||||||
decoder *decoder, input_stream *is)
|
|
||||||
{
|
{
|
||||||
while (true) {
|
while (true) {
|
||||||
int r = ogg_stream_packetout(&os, &packet);
|
int r = ogg_stream_packetout(&os, &packet);
|
||||||
if (r == 0) {
|
if (r == 0) {
|
||||||
if (!OggExpectPageIn(oy, os, decoder, is))
|
if (!oy.ExpectPageIn(os))
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
continue;
|
continue;
|
||||||
|
@ -24,8 +24,7 @@
|
|||||||
|
|
||||||
#include <ogg/ogg.h>
|
#include <ogg/ogg.h>
|
||||||
|
|
||||||
struct decoder;
|
class OggSyncState;
|
||||||
struct input_stream;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Skip all pages/packets until an end-of-stream (EOS) packet for the
|
* Skip all pages/packets until an end-of-stream (EOS) packet for the
|
||||||
@ -34,7 +33,6 @@ struct input_stream;
|
|||||||
* @return true if the EOS packet was found
|
* @return true if the EOS packet was found
|
||||||
*/
|
*/
|
||||||
bool
|
bool
|
||||||
OggFindEOS(ogg_sync_state &oy, ogg_stream_state &os, ogg_packet &packet,
|
OggFindEOS(OggSyncState &oy, ogg_stream_state &os, ogg_packet &packet);
|
||||||
decoder *decoder, input_stream *is);
|
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
78
src/decoder/OggSyncState.hxx
Normal file
78
src/decoder/OggSyncState.hxx
Normal file
@ -0,0 +1,78 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (C) 2003-2013 The Music Player Daemon Project
|
||||||
|
* http://www.musicpd.org
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
* the Free Software Foundation; either version 2 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License along
|
||||||
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
||||||
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef MPD_OGG_SYNC_STATE_HXX
|
||||||
|
#define MPD_OGG_SYNC_STATE_HXX
|
||||||
|
|
||||||
|
#include "check.h"
|
||||||
|
#include "OggUtil.hxx"
|
||||||
|
|
||||||
|
#include <ogg/ogg.h>
|
||||||
|
|
||||||
|
#include <stddef.h>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Wrapper for an ogg_sync_state.
|
||||||
|
*/
|
||||||
|
class OggSyncState {
|
||||||
|
ogg_sync_state oy;
|
||||||
|
|
||||||
|
input_stream &is;
|
||||||
|
struct decoder *const decoder;
|
||||||
|
|
||||||
|
public:
|
||||||
|
OggSyncState(input_stream &_is, struct decoder *const _decoder=nullptr)
|
||||||
|
:is(_is), decoder(_decoder) {
|
||||||
|
ogg_sync_init(&oy);
|
||||||
|
}
|
||||||
|
|
||||||
|
~OggSyncState() {
|
||||||
|
ogg_sync_clear(&oy);
|
||||||
|
}
|
||||||
|
|
||||||
|
void Reset() {
|
||||||
|
ogg_sync_reset(&oy);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool Feed(size_t size) {
|
||||||
|
return OggFeed(oy, decoder, &is, size);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool ExpectPage(ogg_page &page) {
|
||||||
|
return OggExpectPage(oy, page, decoder, &is);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool ExpectFirstPage(ogg_stream_state &os) {
|
||||||
|
return OggExpectFirstPage(oy, os, decoder, &is);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool ExpectPageIn(ogg_stream_state &os) {
|
||||||
|
return OggExpectPageIn(oy, os, decoder, &is);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool ExpectPageSeek(ogg_page &page) {
|
||||||
|
return OggExpectPageSeek(oy, page, decoder, &is);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool ExpectPageSeekIn(ogg_stream_state &os) {
|
||||||
|
return OggExpectPageSeekIn(oy, os, decoder, &is);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif
|
@ -23,6 +23,7 @@
|
|||||||
#include "OpusTags.hxx"
|
#include "OpusTags.hxx"
|
||||||
#include "OggUtil.hxx"
|
#include "OggUtil.hxx"
|
||||||
#include "OggFind.hxx"
|
#include "OggFind.hxx"
|
||||||
|
#include "OggSyncState.hxx"
|
||||||
#include "decoder_api.h"
|
#include "decoder_api.h"
|
||||||
#include "OggCodec.hxx"
|
#include "OggCodec.hxx"
|
||||||
#include "audio_check.h"
|
#include "audio_check.h"
|
||||||
@ -85,8 +86,8 @@ public:
|
|||||||
:decoder(_decoder), input_stream(_input_stream) {}
|
:decoder(_decoder), input_stream(_input_stream) {}
|
||||||
~MPDOpusDecoder();
|
~MPDOpusDecoder();
|
||||||
|
|
||||||
bool ReadFirstPage(ogg_sync_state &oy);
|
bool ReadFirstPage(OggSyncState &oy);
|
||||||
bool ReadNextPage(ogg_sync_state &oy);
|
bool ReadNextPage(OggSyncState &oy);
|
||||||
|
|
||||||
enum decoder_command HandlePackets();
|
enum decoder_command HandlePackets();
|
||||||
enum decoder_command HandlePacket(const ogg_packet &packet);
|
enum decoder_command HandlePacket(const ogg_packet &packet);
|
||||||
@ -107,11 +108,11 @@ MPDOpusDecoder::~MPDOpusDecoder()
|
|||||||
}
|
}
|
||||||
|
|
||||||
inline bool
|
inline bool
|
||||||
MPDOpusDecoder::ReadFirstPage(ogg_sync_state &oy)
|
MPDOpusDecoder::ReadFirstPage(OggSyncState &oy)
|
||||||
{
|
{
|
||||||
assert(!os_initialized);
|
assert(!os_initialized);
|
||||||
|
|
||||||
if (!OggExpectFirstPage(oy, os, decoder, input_stream))
|
if (!oy.ExpectFirstPage(os))
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
os_initialized = true;
|
os_initialized = true;
|
||||||
@ -119,12 +120,12 @@ MPDOpusDecoder::ReadFirstPage(ogg_sync_state &oy)
|
|||||||
}
|
}
|
||||||
|
|
||||||
inline bool
|
inline bool
|
||||||
MPDOpusDecoder::ReadNextPage(ogg_sync_state &oy)
|
MPDOpusDecoder::ReadNextPage(OggSyncState &oy)
|
||||||
{
|
{
|
||||||
assert(os_initialized);
|
assert(os_initialized);
|
||||||
|
|
||||||
ogg_page page;
|
ogg_page page;
|
||||||
if (!OggExpectPage(oy, page, decoder, input_stream))
|
if (!oy.ExpectPage(page))
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
const auto page_serialno = ogg_page_serialno(&page);
|
const auto page_serialno = ogg_page_serialno(&page);
|
||||||
@ -269,14 +270,10 @@ mpd_opus_stream_decode(struct decoder *decoder,
|
|||||||
input_stream_lock_seek(input_stream, 0, SEEK_SET, nullptr);
|
input_stream_lock_seek(input_stream, 0, SEEK_SET, nullptr);
|
||||||
|
|
||||||
MPDOpusDecoder d(decoder, input_stream);
|
MPDOpusDecoder d(decoder, input_stream);
|
||||||
|
OggSyncState oy(*input_stream, decoder);
|
||||||
|
|
||||||
ogg_sync_state oy;
|
if (!d.ReadFirstPage(oy))
|
||||||
ogg_sync_init(&oy);
|
|
||||||
|
|
||||||
if (!d.ReadFirstPage(oy)) {
|
|
||||||
ogg_sync_clear(&oy);
|
|
||||||
return;
|
return;
|
||||||
}
|
|
||||||
|
|
||||||
while (true) {
|
while (true) {
|
||||||
enum decoder_command cmd = d.HandlePackets();
|
enum decoder_command cmd = d.HandlePackets();
|
||||||
@ -287,39 +284,34 @@ mpd_opus_stream_decode(struct decoder *decoder,
|
|||||||
break;
|
break;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ogg_sync_clear(&oy);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static bool
|
static bool
|
||||||
SeekFindEOS(ogg_sync_state &oy, ogg_stream_state &os, ogg_packet &packet,
|
SeekFindEOS(OggSyncState &oy, ogg_stream_state &os, ogg_packet &packet,
|
||||||
decoder *decoder, input_stream *is)
|
input_stream *is)
|
||||||
{
|
{
|
||||||
if (is->size > 0 && is->size - is->offset < 65536)
|
if (is->size > 0 && is->size - is->offset < 65536)
|
||||||
return OggFindEOS(oy, os, packet, decoder, is);
|
return OggFindEOS(oy, os, packet);
|
||||||
|
|
||||||
if (!input_stream_cheap_seeking(is))
|
if (!input_stream_cheap_seeking(is))
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
ogg_sync_reset(&oy);
|
oy.Reset();
|
||||||
|
|
||||||
return input_stream_lock_seek(is, -65536, SEEK_END, nullptr) &&
|
return input_stream_lock_seek(is, -65536, SEEK_END, nullptr) &&
|
||||||
OggExpectPageSeekIn(oy, os, decoder, is) &&
|
oy.ExpectPageSeekIn(os) &&
|
||||||
OggFindEOS(oy, os, packet, decoder, is);
|
OggFindEOS(oy, os, packet);
|
||||||
}
|
}
|
||||||
|
|
||||||
static bool
|
static bool
|
||||||
mpd_opus_scan_stream(struct input_stream *is,
|
mpd_opus_scan_stream(struct input_stream *is,
|
||||||
const struct tag_handler *handler, void *handler_ctx)
|
const struct tag_handler *handler, void *handler_ctx)
|
||||||
{
|
{
|
||||||
ogg_sync_state oy;
|
OggSyncState oy(*is);
|
||||||
ogg_sync_init(&oy);
|
|
||||||
|
|
||||||
ogg_stream_state os;
|
ogg_stream_state os;
|
||||||
if (!OggExpectFirstPage(oy, os, nullptr, is)) {
|
if (!oy.ExpectFirstPage(os))
|
||||||
ogg_sync_clear(&oy);
|
|
||||||
return false;
|
return false;
|
||||||
}
|
|
||||||
|
|
||||||
/* read at most two more pages */
|
/* read at most two more pages */
|
||||||
unsigned remaining_pages = 2;
|
unsigned remaining_pages = 2;
|
||||||
@ -338,7 +330,7 @@ mpd_opus_scan_stream(struct input_stream *is,
|
|||||||
if (remaining_pages-- == 0)
|
if (remaining_pages-- == 0)
|
||||||
break;
|
break;
|
||||||
|
|
||||||
if (!OggExpectPageIn(oy, os, nullptr, is)) {
|
if (!oy.ExpectPageIn(os)) {
|
||||||
result = false;
|
result = false;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
@ -369,12 +361,11 @@ mpd_opus_scan_stream(struct input_stream *is,
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (packet.e_o_s || SeekFindEOS(oy, os, packet, nullptr, is))
|
if (packet.e_o_s || SeekFindEOS(oy, os, packet, is))
|
||||||
tag_handler_invoke_duration(handler, handler_ctx,
|
tag_handler_invoke_duration(handler, handler_ctx,
|
||||||
packet.granulepos / opus_sample_rate);
|
packet.granulepos / opus_sample_rate);
|
||||||
|
|
||||||
ogg_stream_clear(&os);
|
ogg_stream_clear(&os);
|
||||||
ogg_sync_clear(&oy);
|
|
||||||
|
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user