-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathdecoder.h
332 lines (297 loc) · 9.84 KB
/
decoder.h
1
2
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
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
#ifndef DECODER_H
#define DECODER_H
#include "audio.h"
#include "playlist.h"
#include "io.h"
#ifdef __cplusplus
extern "C" {
#endif
/** Version of the decoder API.
*
* On every change in the decoder API this number will be changed, so MOC will
* not load plugins compiled with older/newer decoder.h. */
#define DECODER_API_VERSION 7
/** Type of the decoder error. */
enum decoder_error_type
{
ERROR_OK, /*!< There was no error. */
ERROR_STREAM, /*!< Recoverable error in the stream. */
ERROR_FATAL /*!< Fatal error in the stream - further decoding can't be
performed. */
};
/** Decoder error.
*
* Describes decoder error. Fields don't need to be accessed directly, there are
* functions to modify/access decoder_error object. */
struct decoder_error
{
enum decoder_error_type type; /*!< Type of the error. */
char *err; /*!< malloc()ed error string or NULL. */
};
/** @struct decoder
* Functions provided by the decoder plugin.
*
* Describes decoder - contains pointers to decoder's functions. If some field
* is optional, it may have NULL value. */
struct decoder
{
/** API version used by the plugin.
*
* Set it to DECODER_API_VERSION to recognize if MOC and the plugin can
* communicate. This is the first field in the structure, so even after
* changing other fields it will hopefully be always read properly.
*/
int api_version;
/** Initialize the plugin.
*
* This function is called once at MOC startup (once for the client and
* once for the server). Optional. */
void (*init) ();
/** Cleanup the plugin.
*
* This function is called once at exit (once for the client and
* once for the server). Optional. */
void (*destroy) ();
/** Open the resource.
*
* Open the given resource (file).
*
* \param uri URL to the resource that can be used as the file parameter and return pointer
* to io_open().
*
* \return Private decoder data. This pointer will be passed to every
* other function that operates on the stream.
*/
void *(*open)(const char *uri);
/** Open the resource for an already opened stream.
*
* Handle the stream that was already opened, but no data were read.
* You must operate on the stream using io_*() functions. This is used
* for internet streams, so seeking is not possible. This function is
* optional.
*
* \param stream Opened stream from which the decoder must read.
*
* \return Private decoder data. This pointer will be passed to every
* other function that operates on the stream.
*/
void *(*open_stream)(struct io_stream *stream);
/** Check if the decoder is able to decode from this stream.
*
* Used to check if the decoder is able to read from an already opened
* stream. This is used to find the proper decoder for an internet
* stream when searching by the MIME type failed. The decoder must not
* read from this stream (io_read()), but can peek data (io_peek()).
* The decoder is expected to peek a few bytes to recognize it's format.
* Optional.
*
* \param stream Opened stream.
*
* \return 1 if the decoder is able to decode data from this stream.
*/
int (*can_decode)(struct io_stream *stream);
/** Close the resource and cleanup.
*
* Free all decoder's private data and allocated resources.
*
* \param data Decoder's private data.
*/
void (*close)(void *data);
/** Decode a piece of input.
*
* Decode a piece of input and write it to the buffer. The buffer size
* is at least 32KB, but don't make any assumptions that it is always
* true. It is prefered to decode as few bytes as possible without
* loosing much performance to cause very small delays.
*
* \param data Decoder's private data.
* \param buf Buffer to put data in.
* \param buf_len Size of the buffer in bytes.
* \param sound_params Parameters of the decoded sound. This must be
* always filled.
*
* \return Number of bytes written or 0 on EOF.
*/
int (*decode)(void *data, char *buf, int buf_len,
struct sound_params *sound_params);
/** Seek in the stream.
*
* Seek to the given position.
*
* \param data Decoder's private data.
* \param sec Where to seek in seconds.
*
* \return The position that we actually seek to or -1 on error.
* -1 is not a fatal error and further decoding will be performed.
*/
int (*seek)(void *data, int sec);
/** Get tags for a file.
*
* Get requested file's tags. Is some tags are not available, the
* decoder can just not fill the field. The function can even not
* fill any field.
*
* \param file File to get tags for.
* \param tags Pointer to the tags structure where we must put
* the tags. All strings must be malloc()ed.
* \param tags_sel OR'ed list of requested tags (values of
* enum tags_select).
*/
void (*info)(const char *file, struct file_tags *tags,
const int tags_sel);
/** Get the current bitrate.
*
* Get the bitrate of the last decoded piece of sound.
*
* \param data Decoder's private data.
*
* \return Current bitrate in kbps or -1 if not available.
*/
int (*get_bitrate)(void *data);
/** Get duration of the stream.
*
* Get duration of the stream. It is used as a faster alternative
* for getting duration using info() if the file is opened.
*
* \param data Decoder's private data.
*
* \return Duration in seconds or -1 on error. -1 is not a fatal
* error, further decoding fill be performed.
*/
int (*get_duration)(void *data);
/** Get error for the last decode() invokation.
*
* Get the error for the last decode() invokation. If there was no
* error the type of the error must be ERROR_OK. Don't access the
* error object's fields directly, there are proper functions for
* that.
*
* \param data Decoder's private data.
* \param error Pointer to the decoder_error opiect to fill.
*/
void (*get_error)(void *data, struct decoder_error *error);
/** Check if the file extension is for a file that this decoder
* supports.
*
* \param ext Extension (chars after the last dot in the file name).
*
* \return Value other than 0 if the extension if a file with this
* extension is supported.
*/
int (*our_format_ext)(const char *ext);
/** Check if a stream with the given MIME type is supported by this
* decoder. Optional.
*
* \param mime_type MIME type.
*
* \return Value other than 0 if a stream with this mime type is
* supported.
*/
int (*our_format_mime)(const char *mime_type);
/** Get a 3-chars format name for a file.
*
* Get an abbreviated format name (up to 3 chars) for a file.
*
* \param file File for which we want the format name.
* \param buf Buffer where the format name and ending zero-byte
* must be put.
*/
void (*get_name)(const char *file, char buf[4]);
/** Get current tags for the stream.
*
* Fill the tags structure with the current tags for the stream. This
* is intended for internet streams and used when the source of the
* stream doesn't provide tags while broadcasting. This function is
* optional.
*
* \param data Decoder's private data.
*
* \return 1 if the tags were changed from the last call of this
* function or 0 if not.
*/
int (*current_tags)(void *data, struct file_tags *tags);
/** Get the IO stream used by the decoder.
*
* Get the pointer to the io_stream object used by the decoder. This is
* used for fast interrupting especially when the stream reads from
* a network. This function is optional.
*
* \param data Decoder's private data.
*
* \return Pointer to the used IO stream.
*/
struct io_stream *(*get_stream)(void *data);
/** Get the average bitrate.
*
* Get the bitrate of the whole file.
*
* \param data Decoder's private data.
*
* \return Average bitrate in kbps or -1 if not available.
*/
int (*get_avg_bitrate)(void *data);
};
/** Initialize decoder plugin.
*
* Each decoder plugin must export a function name plugin_init of this type.
* The function must return a pointer to the struct decoder variable filled
* with pointers to decoder's functions.
*/
typedef struct decoder *(*plugin_init_func)();
int is_sound_file (const char *name);
struct decoder *get_decoder (const char *file);
struct decoder *get_decoder_by_content (struct io_stream *stream);
void decoder_init (int debug_info);
void decoder_cleanup ();
char *file_type_name (const char *file);
/** @defgroup decoder_error_funcs Decoder error functions
*
* Those functions can be used to modify variables of type struct decoder_error.
*/
/*@{*/
/** Fill decoder_error structure with an error.
*
* Fills decoder error variable with an error. It can be used like printf().
*
* \param error Pointer to the decoder error variable to fill.
* \param type Type of the error.
* \param add_errno If this value is non-zero, a space and a string describing
* system error for errno equal to the value of add_errno is appended to the
* error message.
* \param format Format, like in the printf() function.
*/
#ifdef HAVE__ATTRIBUTE__
void decoder_error (struct decoder_error *error,
const enum decoder_error_type type, const int add_errno,
const char *format, ...) __attribute__((format (printf, 4, 5)));
#else
void decoder_error (struct decoder_error *error,
const enum decoder_error_type type, const int add_errno,
const char *format, ...);
#endif
/** Clear decoder_error structure.
*
* Clear decoder_error structure. Set the system type to ERROR_OK and
* the error message to NULL. Frees all memory used by the error's fields.
*/
void decoder_error_clear (struct decoder_error *error);
/** Copy decoder_error variable.
*
* Copies the decoder_error variable to another decoder_error variable.
*
* \param dst Destination.
* \param src Source.
*/
void decoder_error_copy (struct decoder_error *dst,
const struct decoder_error *src);
/** Initialize decoder_error variable.
*
* Initialize decoder_error variable and set the error to ERROR_OK with no
* message.
*/
void decoder_error_init (struct decoder_error *error);
/*@}*/
#ifdef __cplusplus
}
#endif
#endif