DecoderControl.hxx 6.59 KB
Newer Older
1
/*
2
 * Copyright (C) 2003-2013 The Music Player Daemon Project
3
 * http://www.musicpd.org
Warren Dukes's avatar
Warren Dukes committed
4 5 6 7 8 9 10 11 12 13
 *
 * 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.
14 15 16 17
 *
 * 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.
Warren Dukes's avatar
Warren Dukes committed
18 19
 */

20 21
#ifndef MPD_DECODER_CONTROL_HXX
#define MPD_DECODER_CONTROL_HXX
Warren Dukes's avatar
Warren Dukes committed
22

23
#include "decoder_command.h"
24
#include "audio_format.h"
25 26
#include "thread/Mutex.hxx"
#include "thread/Cond.hxx"
27 28

#include <glib.h>
Warren Dukes's avatar
Warren Dukes committed
29

30 31
#include <assert.h>

32 33 34
enum decoder_state {
	DECODE_STATE_STOP = 0,
	DECODE_STATE_START,
35 36 37 38 39 40 41 42 43
	DECODE_STATE_DECODE,

	/**
	 * The last "START" command failed, because there was an I/O
	 * error or because no decoder was able to decode the file.
	 * This state will only come after START; once the state has
	 * turned to DECODE, by definition no such error can occur.
	 */
	DECODE_STATE_ERROR,
44
};
45

46
struct decoder_control {
47 48 49 50
	/** the handle of the decoder thread, or NULL if the decoder
	    thread isn't running */
	GThread *thread;

51 52 53
	/**
	 * This lock protects #state and #command.
	 */
54
	mutable Mutex mutex;
55 56 57 58 59 60

	/**
	 * Trigger this object after you have modified #command.  This
	 * is also used by the decoder thread to notify the caller
	 * when it has finished a command.
	 */
61
	Cond cond;
62

63 64 65 66
	/**
	 * The trigger of this object's client.  It is signalled
	 * whenever an event occurs.
	 */
67
	Cond client_cond;
68

69 70
	enum decoder_state state;
	enum decoder_command command;
71

72 73 74 75 76 77 78 79
	/**
	 * The error that occurred in the decoder thread.  This
	 * attribute is only valid if #state is #DECODE_STATE_ERROR.
	 * The object must be freed when this object transitions to
	 * any other state (usually #DECODE_STATE_START).
	 */
	GError *error;

80
	bool quit;
Max Kellermann's avatar
Max Kellermann committed
81
	bool seek_error;
82
	bool seekable;
83
	double seek_where;
84 85 86 87 88 89 90

	/** the format of the song file */
	struct audio_format in_audio_format;

	/** the format being sent to the music pipe */
	struct audio_format out_audio_format;

91 92 93 94
	/**
	 * The song currently being decoded.  This attribute is set by
	 * the player thread, when it sends the #DECODE_COMMAND_START
	 * command.
95 96 97
	 *
	 * This is a duplicate, and must be freed when this attribute
	 * is cleared.
98
	 */
99
	struct song *song;
100

101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117
	/**
	 * The initial seek position (in milliseconds), e.g. to the
	 * start of a sub-track described by a CUE file.
	 *
	 * This attribute is set by dc_start().
	 */
	unsigned start_ms;

	/**
	 * The decoder will stop when it reaches this position (in
	 * milliseconds).  0 means don't stop before the end of the
	 * file.
	 *
	 * This attribute is set by dc_start().
	 */
	unsigned end_ms;

118
	float total_time;
119 120 121 122

	/** the #music_chunk allocator */
	struct music_buffer *buffer;

123 124 125 126
	/**
	 * The destination pipe for decoded chunks.  The caller thread
	 * owns this object, and is responsible for freeing it.
	 */
127
	struct music_pipe *pipe;
128

129 130
	float replay_gain_db;
	float replay_gain_prev_db;
131 132 133
	char *mixramp_start;
	char *mixramp_end;
	char *mixramp_prev_end;
134

135 136 137 138 139 140 141
	decoder_control();
	~decoder_control();

	/**
	 * Locks the object.
	 */
	void Lock() const {
142
		mutex.lock();
143 144 145 146 147 148
	}

	/**
	 * Unlocks the object.
	 */
	void Unlock() const {
149
		mutex.unlock();
150 151 152 153 154 155 156 157
	}

	/**
	 * Signals the object.  This function is only valid in the
	 * player thread.  The object should be locked prior to
	 * calling this function.
	 */
	void Signal() {
158
		cond.signal();
159 160 161 162 163 164 165 166
	}

	/**
	 * Waits for a signal on the #decoder_control object.  This function
	 * is only valid in the decoder thread.  The object must be locked
	 * prior to calling this function.
	 */
	void Wait() {
167
		cond.wait(mutex);
168 169
	}

170 171 172 173 174 175
	/**
	 * Waits for a signal from the decoder thread.  This object
	 * must be locked prior to calling this function.  This method
	 * is only valid in the player thread.
	 */
	void WaitForDecoder() {
176
		client_cond.wait(mutex);
177
	}
178

179 180 181
	bool IsIdle() const {
		return state == DECODE_STATE_STOP ||
			state == DECODE_STATE_ERROR;
182 183
	}

184 185 186 187 188 189 190
	gcc_pure
	bool LockIsIdle() const {
		Lock();
		bool result = IsIdle();
		Unlock();
		return result;
	}
191

192 193 194
	bool IsStarting() const {
		return state == DECODE_STATE_START;
	}
195

196 197 198 199 200 201 202
	gcc_pure
	bool LockIsStarting() const {
		Lock();
		bool result = IsStarting();
		Unlock();
		return result;
	}
203

204 205
	bool HasFailed() const {
		assert(command == DECODE_COMMAND_NONE);
206

207 208
		return state == DECODE_STATE_ERROR;
	}
209

210 211 212 213 214 215 216
	gcc_pure
	bool LockHasFailed() const {
		Lock();
		bool result = HasFailed();
		Unlock();
		return result;
	}
217

218 219 220 221 222 223 224 225 226
	/**
	 * Checks whether an error has occurred, and if so, returns a newly
	 * allocated copy of the #GError object.
	 *
	 * Caller must lock the object.
	 */
	GError *GetError() const {
		assert(command == DECODE_COMMAND_NONE);
		assert(state != DECODE_STATE_ERROR || error != nullptr);
227

228 229 230 231
		return state == DECODE_STATE_ERROR
			? g_error_copy(error)
			: nullptr;
	}
232

233 234 235 236 237 238 239 240 241
	/**
	 * Like dc_get_error(), but locks and unlocks the object.
	 */
	GError *LockGetError() const {
		Lock();
		GError *result = GetError();
		Unlock();
		return result;
	}
242

243 244 245 246 247 248 249 250 251 252 253
	/**
	 * Clear the error condition and free the #GError object (if any).
	 *
	 * Caller must lock the object.
	 */
	void ClearError() {
		if (state == DECODE_STATE_ERROR) {
			g_error_free(error);
			state = DECODE_STATE_STOP;
		}
	}
Max Kellermann's avatar
Max Kellermann committed
254

255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271
	/**
	 * Check if the specified song is currently being decoded.  If the
	 * decoder is not running currently (or being started), then this
	 * function returns false in any case.
	 *
	 * Caller must lock the object.
	 */
	gcc_pure
	bool IsCurrentSong(const struct song *_song) const;

	gcc_pure
	bool LockIsCurrentSong(const struct song *_song) const {
		Lock();
		const bool result = IsCurrentSong(_song);
		Unlock();
		return result;
	}
Max Kellermann's avatar
Max Kellermann committed
272

273 274 275 276 277 278 279 280 281 282 283 284
	/**
	 * Start the decoder.
	 *
	 * @param song the song to be decoded; the given instance will be
	 * owned and freed by the decoder
	 * @param start_ms see #decoder_control
	 * @param end_ms see #decoder_control
	 * @param pipe the pipe which receives the decoded chunks (owned by
	 * the caller)
	 */
	void Start(struct song *song, unsigned start_ms, unsigned end_ms,
		   music_buffer *buffer, music_pipe *pipe);
Max Kellermann's avatar
Max Kellermann committed
285

286
	void Stop();
287

288
	bool Seek(double where);
289

290
	void Quit();
291

292 293 294 295
	void MixRampStart(char *_mixramp_start);
	void MixRampEnd(char *_mixramp_end);
	void MixRampPrevEnd(char *_mixramp_prev_end);
};
296

Warren Dukes's avatar
Warren Dukes committed
297
#endif