1 /*
2 * Monkey's Audio APE demuxer
3 * Copyright (c) 2007 Benjamin Zores <ben@geexbox.org>
4 * based upon libdemac from Dave Chapman.
5 *
6 * This file is part of FFmpeg.
7 *
8 * FFmpeg is free software; you can redistribute it and/or
9 * modify it under the terms of the GNU Lesser General Public
10 * License as published by the Free Software Foundation; either
11 * version 2.1 of the License, or (at your option) any later version.
12 *
13 * FFmpeg is distributed in the hope that it will be useful,
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
16 * Lesser General Public License for more details.
17 *
18 * You should have received a copy of the GNU Lesser General Public
19 * License along with FFmpeg; if not, write to the Free Software
20 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
21 */
22
23 #include <stdio.h>
24
29
30 /* The earliest and latest file formats supported by this library */
31 #define APE_MIN_VERSION 3800
32 #define APE_MAX_VERSION 3990
33
34 #define MAC_FORMAT_FLAG_8_BIT 1 // is 8-bit [OBSOLETE]
35 #define MAC_FORMAT_FLAG_CRC 2 // uses the new CRC32 error detection [OBSOLETE]
36 #define MAC_FORMAT_FLAG_HAS_PEAK_LEVEL 4 // uint32 nPeakLevel after the header [OBSOLETE]
37 #define MAC_FORMAT_FLAG_24_BIT 8 // is 24-bit [OBSOLETE]
38 #define MAC_FORMAT_FLAG_HAS_SEEK_ELEMENTS 16 // has the number of seek elements after the peak level
39 #define MAC_FORMAT_FLAG_CREATE_WAV_HEADER 32 // create the wave header on decompression (not stored)
40
41 #define APE_EXTRADATA_SIZE 6
42
50
52 /* Derived fields */
58
59 /* Info from Descriptor Block */
70
71 /* Info from Header Block */
80
81 /* Seektable */
85
87 {
90 return 0;
91
94
96 }
97
99 {
100 #ifdef DEBUG
101 int i;
102
113 for (i = 0; i < 16; i++)
116
118
127
131 } else {
133 if (i < ape_ctx->totalframes - 1) {
141 } else {
143 }
144 }
145 }
146
152
157 #endif
158 }
159
161 {
166 int i;
167 int total_blocks, final_size = 0;
168 int64_t
pts, file_size;
169
170 /* Skip any leading junk such as id3v2 tags */
172
174 if (tag !=
MKTAG(
'M',
'A',
'C',
' '))
176
178
183 }
184
195
196 /* Skip any unknown bytes at the end of the descriptor.
197 This is for future compatibility */
200
201 /* Read header data */
210 } else {
213
222
224 avio_skip(pb, 4);
/* Skip the peak level */
226 }
227
232 } else
234
239 else
241
246 else
248
249 /* Skip any stored wav header */
252 }
253
257 }
262 }
265 "Number of seek entries is less than number of frames: %"SIZE_SPECIFIER " vs. %"PRIu32
"\n",
268 }
276
277
281
294 }
297 }
298
307 }
309 /* calculate final packet size from total file size, if available */
311 if (file_size > 0) {
314 final_size -= final_size & 3;
315 }
316 if (file_size <= 0 || final_size <= 0)
319
324 }
326 }
329 if (i < ape->totalframes - 1 && ape->
bittable[i + 1])
333 }
334 }
335
337
341
342 /* now we are ready: build format streams */
344 if (!st)
346
348
355
360
366
367 pts = 0;
372 }
373
374 /* try to read APE tags */
378 }
379
380 return 0;
381 }
382
384 {
385 int ret;
386 int nblocks;
388 uint32_t extra_size = 8;
389
394
397
398 /* Calculate how many blocks there are in this frame */
401 else
403
410 }
411
414
418 if (ret < 0) {
420 return ret;
421 }
422
425
426 /* note: we need to modify the packet size here to handle the last
427 packet */
428 pkt->
size = ret + extra_size;
429
431
432 return 0;
433 }
434
436 {
438
442 return 0;
443 }
444
446 {
450
451 if (index < 0)
452 return -1;
453
455 return -1;
457 return 0;
458 }
459
469 .extensions = "ape,apl,mac",
470 };
static int ape_read_close(AVFormatContext *s)
#define AVERROR_INVALIDDATA
Invalid data found when processing input.
int64_t avio_size(AVIOContext *s)
Get the filesize.
int av_add_index_entry(AVStream *st, int64_t pos, int64_t timestamp, int size, int distance, int flags)
Add an index entry into a sorted list.
int fileversion
codec version, very important in decoding process
#define AV_LOG_WARNING
Something somehow does not look correct.
#define MAC_FORMAT_FLAG_24_BIT
static int read_seek(AVFormatContext *ctx, int stream_index, int64_t timestamp, int flags)
enum AVCodecID codec_id
Specific type of the encoded data (the codec used).
int64_t avio_seek(AVIOContext *s, int64_t offset, int whence)
fseek() equivalent for AVIOContext.
AVIndexEntry * index_entries
Only used if the format does not support seeking natively.
int64_t avio_skip(AVIOContext *s, int64_t offset)
Skip given number of bytes forward.
void * av_mallocz(size_t size)
Allocate a memory block with alignment suitable for all memory accesses (including vectors if availab...
uint64_t_TMPL AV_WL64 unsigned int_TMPL AV_WL32 unsigned int_TMPL AV_WL24 unsigned int_TMPL AV_RL16
#define MAC_FORMAT_FLAG_HAS_PEAK_LEVEL
static int ape_read_seek(AVFormatContext *s, int stream_index, int64_t timestamp, int flags)
static int ape_probe(AVProbeData *p)
AVStream * avformat_new_stream(AVFormatContext *s, const AVCodec *c)
Add a new stream to a media file.
AVStream ** streams
A list of all streams in the file.
#define AVERROR_EOF
End of file.
static av_cold int read_close(AVFormatContext *ctx)
#define AV_LOG_VERBOSE
Detailed information.
static av_always_inline int64_t avio_tell(AVIOContext *s)
ftell() equivalent for AVIOContext.
#define MAC_FORMAT_FLAG_8_BIT
int avio_read(AVIOContext *s, unsigned char *buf, int size)
Read size bytes from AVIOContext into buf.
int av_new_packet(AVPacket *pkt, int size)
Allocate the payload of a packet and initialize its fields with default values.
#define AV_LOG_ERROR
Something went wrong and cannot losslessly be recovered.
int av_index_search_timestamp(AVStream *st, int64_t timestamp, int flags)
Get the index for a specific timestamp.
unsigned int avio_rl32(AVIOContext *s)
#define NULL_IF_CONFIG_SMALL(x)
Return NULL if CONFIG_SMALL is true, otherwise the argument without modification. ...
#define AV_LOG_DEBUG
Stuff which is only useful for libav* developers.
enum AVMediaType codec_type
General type of the encoded data.
#define MAC_FORMAT_FLAG_HAS_SEEK_ELEMENTS
int64_t ff_ape_parse_tag(AVFormatContext *s)
Read and parse an APE tag.
int avio_r8(AVIOContext *s)
unsigned char * buf
Buffer must have AVPROBE_PADDING_SIZE of extra allocated bytes filled with zero.
int seekable
A combination of AVIO_SEEKABLE_ flags or 0 when the stream is not seekable.
static int ape_read_packet(AVFormatContext *s, AVPacket *pkt)
uint32_t finalframeblocks
AVInputFormat ff_ape_demuxer
static int read_header(FFV1Context *f)
#define AVERROR_PATCHWELCOME
Not yet implemented in FFmpeg, patches welcome.
static int read_packet(void *opaque, uint8_t *buf, int buf_size)
#define AVIO_SEEKABLE_NORMAL
Seeking works like for a local file.
AVIOContext * pb
I/O context.
void av_packet_unref(AVPacket *pkt)
Wipe the packet.
static int ape_read_header(AVFormatContext *s)
This structure contains the data a format has to probe a file.
int64_t duration
Decoding: duration of the stream, in stream time base.
uint32_t descriptorlength
int sample_rate
Audio only.
unsigned int avio_rl16(AVIOContext *s)
int64_t start_time
Decoding: pts of the first frame of the stream in presentation order, in stream time base...
int64_t nb_frames
number of frames in this stream if known or 0
int eof_reached
true if eof reached
void * priv_data
Format private data.
#define APE_EXTRADATA_SIZE
int bits_per_coded_sample
The number of bits per sample in the codedwords.
uint8_t * extradata
Extra binary data needed for initializing the decoder, codec-dependent.
AVCodecParameters * codecpar
Codec parameters associated with this stream.
#define av_malloc_array(a, b)
int avio_feof(AVIOContext *s)
feof() equivalent for AVIOContext.
uint32_t codec_tag
Additional information about the codec (corresponds to the AVI FOURCC).
static void ape_dumpinfo(AVFormatContext *s, APEContext *ape_ctx)
#define MKTAG(a, b, c, d)
uint32_t audiodatalength_high
#define MAC_FORMAT_FLAG_CREATE_WAV_HEADER
uint64_t_TMPL AV_WL64 unsigned int_TMPL AV_RL32
This structure stores compressed data.
int64_t pts
Presentation timestamp in AVStream->time_base units; the time at which the decompressed packet will b...