OggVisitor.hxx 2.73 KB
Newer Older
1
/*
Max Kellermann's avatar
Max Kellermann committed
2
 * Copyright 2003-2021 The Music Player Daemon Project
3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39
 * 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_VISITOR_HXX
#define MPD_OGG_VISITOR_HXX

#include "OggSyncState.hxx"
#include "OggStreamState.hxx"

#include <ogg/ogg.h>

class Reader;

/**
 * Abstract class which iterates over Ogg packets in a #Reader.
 * Subclass it and implement the virtual methods.
 */
class OggVisitor {
	OggSyncState sync;
	OggStreamState stream;

	bool has_stream = false;

40 41 42 43 44 45 46 47
	/**
	 * This is true after seeking; its one-time effect is to
	 * ignore the BOS packet, just in case we have been seeking to
	 * the beginning of the file, because that would disrupt
	 * playback.
	 */
	bool post_seek = false;

48 49 50 51 52 53 54 55
public:
	explicit OggVisitor(Reader &reader)
		:sync(reader), stream(0) {}

	long GetSerialNo() const {
		return stream.GetSerialNo();
	}

56 57 58 59
	uint64_t GetStartOffset() const noexcept {
		return sync.GetStartOffset();
	}

60 61 62 63
	void Visit();

	/**
	 * Call this method after seeking the #Reader.
64 65
	 *
	 * @param offset the current #Reader offset
66
	 */
67
	void PostSeek(uint64_t offset);
68

69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84
	/**
	 * Skip packets (#ogg_packet) from the #OggStreamState until a
	 * packet with a valid granulepos is found or until the stream
	 * has run dry.
	 *
	 * Since this will discard pending packets and will disturb
	 * this object, this should only be used while seeking.
	 *
	 * This method must not be called from within one of the
	 * virtual methods.
	 *
	 * @return the granulepos or -1 if no valid granulepos was
	 * found
	 */
	ogg_int64_t ReadGranulepos() noexcept;

85 86 87 88 89 90 91
private:
	void EndStream();
	bool ReadNextPage();
	void HandlePacket(const ogg_packet &packet);
	void HandlePackets();

protected:
92 93 94 95 96
	/**
	 * Called when the "beginning of stream" packet has been seen.
	 *
	 * @param packet the "beginning of stream" packet
	 */
97
	virtual void OnOggBeginning(const ogg_packet &packet) = 0;
98 99 100 101

	/**
	 * Called for each follow-up packet.
	 */
102
	virtual void OnOggPacket(const ogg_packet &packet) = 0;
103 104 105 106

	/**
	 * Called after the "end of stream" packet has been processed.
	 */
107 108 109 110
	virtual void OnOggEnd() = 0;
};

#endif