225 lines
6.1 KiB

/*
* This file is part of FFmpeg.
*
* FFmpeg is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 2.1 of the License, or (at your option) any later version.
*
* FFmpeg 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
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with FFmpeg; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
*/
#include "config.h"
#include "libavutil/avassert.h"
#include "libavutil/avstring.h"
#include "libavcodec/avcodec.h"
#include "libavcodec/bytestream.h"
#include "libavformat/avformat.h"
typedef struct IOContext {
int64_t pos;
int64_t filesize;
uint8_t *fuzz;
int fuzz_size;
} IOContext;
int LLVMFuzzerTestOneInput(const uint8_t *data, size_t size);
int64_t interrupt_counter;
static int interrupt_cb(void *ctx)
{
interrupt_counter --;
return interrupt_counter < 0;
}
static void error(const char *err)
{
fprintf(stderr, "%s", err);
exit(1);
}
static int io_read(void *opaque, uint8_t *buf, int buf_size)
{
IOContext *c = opaque;
int size = FFMIN(buf_size, c->fuzz_size);
if (!c->fuzz_size) {
c->filesize = FFMIN(c->pos, c->filesize);
return AVERROR_EOF;
}
if (c->pos > INT64_MAX - size)
return AVERROR(EIO);
memcpy(buf, c->fuzz, size);
c->fuzz += size;
c->fuzz_size -= size;
c->pos += size;
c->filesize = FFMAX(c->filesize, c->pos);
return size;
}
static int64_t io_seek(void *opaque, int64_t offset, int whence)
{
IOContext *c = opaque;
if (whence == SEEK_CUR) {
if (offset > INT64_MAX - c->pos)
return -1;
offset += c->pos;
} else if (whence == SEEK_END) {
if (offset > INT64_MAX - c->filesize)
return -1;
offset += c->filesize;
} else if (whence == AVSEEK_SIZE) {
return c->filesize;
}
if (offset < 0 || offset > c->filesize)
return -1;
if (IO_FLAT) {
c->fuzz += offset - c->pos;
c->fuzz_size -= offset - c->pos;
}
c->pos = offset;
return 0;
}
// Ensure we don't loop forever
const uint32_t maxiteration = 8096;
const int maxblocks= 50000;
static const uint64_t FUZZ_TAG = 0x4741542D5A5A5546ULL;
int LLVMFuzzerTestOneInput(const uint8_t *data, size_t size) {
const uint64_t fuzz_tag = FUZZ_TAG;
uint32_t it = 0;
AVFormatContext *avfmt = avformat_alloc_context();
AVPacket *pkt;
char filename[1025] = {0};
AVIOContext *fuzzed_pb = NULL;
uint8_t *io_buffer;
int io_buffer_size = 32768;
int64_t filesize = size;
IOContext opaque;
static int c;
int seekable = 0;
int ret;
AVInputFormat *fmt = NULL;
#ifdef FFMPEG_DEMUXER
#define DEMUXER_SYMBOL0(DEMUXER) ff_##DEMUXER##_demuxer
#define DEMUXER_SYMBOL(DEMUXER) DEMUXER_SYMBOL0(DEMUXER)
extern AVInputFormat DEMUXER_SYMBOL(FFMPEG_DEMUXER);
fmt = &DEMUXER_SYMBOL(FFMPEG_DEMUXER);
#endif
if (!c) {
av_log_set_level(AV_LOG_PANIC);
c=1;
}
if (!avfmt)
error("Failed avformat_alloc_context()");
if (IO_FLAT) {
seekable = 1;
io_buffer_size = size;
} else if (size > 2048) {
int flags;
char extension[64];
GetByteContext gbc;
memcpy (filename, data + size - 1024, 1024);
bytestream2_init(&gbc, data + size - 2048, 1024);
size -= 2048;
io_buffer_size = bytestream2_get_le32(&gbc) & 0xFFFFFFF;
flags = bytestream2_get_byte(&gbc);
seekable = flags & 1;
filesize = bytestream2_get_le64(&gbc) & 0x7FFFFFFFFFFFFFFF;
if ((flags & 2) && strlen(filename) < sizeof(filename) / 2) {
const AVInputFormat *avif = NULL;
void *avif_iter = NULL;
int avif_count = 0;
while ((avif = av_demuxer_iterate(&avif_iter))) {
if (avif->extensions)
avif_count ++;
}
avif_count = bytestream2_get_le32(&gbc) % avif_count;
avif_iter = NULL;
while ((avif = av_demuxer_iterate(&avif_iter))) {
if (avif->extensions)
if (!avif_count--)
break;
}
av_strlcpy(extension, avif->extensions, sizeof(extension));
if (strchr(extension, ','))
*strchr(extension, ',') = 0;
av_strlcatf(filename, sizeof(filename), ".%s", extension);
}
interrupt_counter = bytestream2_get_le32(&gbc);
avfmt->interrupt_callback.callback = interrupt_cb;
}
// HLS uses a loop with sleep, we thus must breakout or we timeout
if (fmt && !strcmp(fmt->name, "hls"))
interrupt_counter &= 31;
if (!io_buffer_size || size / io_buffer_size > maxblocks)
io_buffer_size = size;
pkt = av_packet_alloc();
if (!pkt)
error("Failed to allocate pkt");
io_buffer = av_malloc(io_buffer_size);
if (!io_buffer)
error("Failed to allocate io_buffer");
opaque.filesize = filesize;
opaque.pos = 0;
opaque.fuzz = data;
opaque.fuzz_size= size;
fuzzed_pb = avio_alloc_context(io_buffer, io_buffer_size, 0, &opaque,
io_read, NULL, seekable ? io_seek : NULL);
if (!fuzzed_pb)
error("avio_alloc_context failed");
avfmt->pb = fuzzed_pb;
ret = avformat_open_input(&avfmt, filename, fmt, NULL);
if (ret < 0) {
goto fail;
}
ret = avformat_find_stream_info(avfmt, NULL);
//TODO, test seeking
for(it = 0; it < maxiteration; it++) {
ret = av_read_frame(avfmt, pkt);
if (ret < 0)
break;
av_packet_unref(pkt);
}
fail:
av_packet_free(&pkt);
av_freep(&fuzzed_pb->buffer);
avio_context_free(&fuzzed_pb);
avformat_close_input(&avfmt);
return 0;
}