mirror of
https://github.com/FFmpeg/FFmpeg.git
synced 2025-01-13 21:28:01 +02:00
Merge commit 'e7a44f87d07655ec0cd31c315936931674434340'
* commit 'e7a44f87d07655ec0cd31c315936931674434340': 4xm: refactor fourxm_read_header Conflicts: libavformat/4xm.c Merged-by: Michael Niedermayer <michaelni@gmx.at>
This commit is contained in:
commit
60657ee37a
@ -411,12 +411,14 @@ static int decode_p_frame(FourXContext *f, AVFrame *frame,
|
|||||||
int x, y;
|
int x, y;
|
||||||
const int width = f->avctx->width;
|
const int width = f->avctx->width;
|
||||||
const int height = f->avctx->height;
|
const int height = f->avctx->height;
|
||||||
uint16_t *src = (uint16_t *)f->last_picture->data[0];
|
|
||||||
uint16_t *dst = (uint16_t *)frame->data[0];
|
uint16_t *dst = (uint16_t *)frame->data[0];
|
||||||
const int stride = frame->linesize[0] >> 1;
|
const int stride = frame->linesize[0] >> 1;
|
||||||
|
uint16_t *src;
|
||||||
unsigned int bitstream_size, bytestream_size, wordstream_size, extra,
|
unsigned int bitstream_size, bytestream_size, wordstream_size, extra,
|
||||||
bytestream_offset, wordstream_offset;
|
bytestream_offset, wordstream_offset;
|
||||||
|
|
||||||
|
src = (uint16_t *)f->last_picture->data[0];
|
||||||
|
|
||||||
if (f->version > 1) {
|
if (f->version > 1) {
|
||||||
extra = 20;
|
extra = 20;
|
||||||
if (length < extra)
|
if (length < extra)
|
||||||
|
@ -72,8 +72,6 @@ typedef struct AudioTrack {
|
|||||||
} AudioTrack;
|
} AudioTrack;
|
||||||
|
|
||||||
typedef struct FourxmDemuxContext {
|
typedef struct FourxmDemuxContext {
|
||||||
int width;
|
|
||||||
int height;
|
|
||||||
int video_stream_index;
|
int video_stream_index;
|
||||||
int track_count;
|
int track_count;
|
||||||
AudioTrack *tracks;
|
AudioTrack *tracks;
|
||||||
@ -91,6 +89,105 @@ static int fourxm_probe(AVProbeData *p)
|
|||||||
return AVPROBE_SCORE_MAX;
|
return AVPROBE_SCORE_MAX;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int parse_vtrk(AVFormatContext *s,
|
||||||
|
FourxmDemuxContext *fourxm, uint8_t *buf, int size)
|
||||||
|
{
|
||||||
|
AVStream *st;
|
||||||
|
/* check that there is enough data */
|
||||||
|
if (size != vtrk_SIZE) {
|
||||||
|
return AVERROR_INVALIDDATA;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* allocate a new AVStream */
|
||||||
|
st = avformat_new_stream(s, NULL);
|
||||||
|
if (!st)
|
||||||
|
return AVERROR(ENOMEM);
|
||||||
|
|
||||||
|
avpriv_set_pts_info(st, 60, 1, fourxm->fps);
|
||||||
|
|
||||||
|
fourxm->video_stream_index = st->index;
|
||||||
|
|
||||||
|
st->codec->codec_type = AVMEDIA_TYPE_VIDEO;
|
||||||
|
st->codec->codec_id = AV_CODEC_ID_4XM;
|
||||||
|
st->codec->extradata_size = 4;
|
||||||
|
st->codec->extradata = av_malloc(4);
|
||||||
|
AV_WL32(st->codec->extradata, AV_RL32(buf + 16));
|
||||||
|
st->codec->width = AV_RL32(buf + 36);
|
||||||
|
st->codec->height = AV_RL32(buf + 40);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static int parse_strk(AVFormatContext *s,
|
||||||
|
FourxmDemuxContext *fourxm, uint8_t *buf, int size)
|
||||||
|
{
|
||||||
|
AVStream *st;
|
||||||
|
int track;
|
||||||
|
/* check that there is enough data */
|
||||||
|
if (size != strk_SIZE)
|
||||||
|
return AVERROR_INVALIDDATA;
|
||||||
|
|
||||||
|
track = AV_RL32(buf + 8);
|
||||||
|
if ((unsigned)track >= UINT_MAX / sizeof(AudioTrack) - 1) {
|
||||||
|
av_log(s, AV_LOG_ERROR, "current_track too large\n");
|
||||||
|
return AVERROR_INVALIDDATA;
|
||||||
|
}
|
||||||
|
if (track + 1 > fourxm->track_count) {
|
||||||
|
if (av_reallocp_array(&fourxm->tracks, track + 1, sizeof(AudioTrack)))
|
||||||
|
return AVERROR(ENOMEM);
|
||||||
|
memset(&fourxm->tracks[fourxm->track_count], 0,
|
||||||
|
sizeof(AudioTrack) * (track + 1 - fourxm->track_count));
|
||||||
|
fourxm->track_count = track + 1;
|
||||||
|
}
|
||||||
|
fourxm->tracks[track].adpcm = AV_RL32(buf + 12);
|
||||||
|
fourxm->tracks[track].channels = AV_RL32(buf + 36);
|
||||||
|
fourxm->tracks[track].sample_rate = AV_RL32(buf + 40);
|
||||||
|
fourxm->tracks[track].bits = AV_RL32(buf + 44);
|
||||||
|
fourxm->tracks[track].audio_pts = 0;
|
||||||
|
|
||||||
|
if (fourxm->tracks[track].channels <= 0 ||
|
||||||
|
fourxm->tracks[track].sample_rate <= 0 ||
|
||||||
|
fourxm->tracks[track].bits < 0) {
|
||||||
|
av_log(s, AV_LOG_ERROR, "audio header invalid\n");
|
||||||
|
return AVERROR_INVALIDDATA;
|
||||||
|
}
|
||||||
|
if (!fourxm->tracks[track].adpcm && fourxm->tracks[track].bits<8) {
|
||||||
|
av_log(s, AV_LOG_ERROR, "bits unspecified for non ADPCM\n");
|
||||||
|
return AVERROR_INVALIDDATA;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* allocate a new AVStream */
|
||||||
|
st = avformat_new_stream(s, NULL);
|
||||||
|
if (!st)
|
||||||
|
return AVERROR(ENOMEM);
|
||||||
|
|
||||||
|
st->id = track;
|
||||||
|
avpriv_set_pts_info(st, 60, 1, fourxm->tracks[track].sample_rate);
|
||||||
|
|
||||||
|
fourxm->tracks[track].stream_index = st->index;
|
||||||
|
|
||||||
|
st->codec->codec_type = AVMEDIA_TYPE_AUDIO;
|
||||||
|
st->codec->codec_tag = 0;
|
||||||
|
st->codec->channels = fourxm->tracks[track].channels;
|
||||||
|
st->codec->sample_rate = fourxm->tracks[track].sample_rate;
|
||||||
|
st->codec->bits_per_coded_sample = fourxm->tracks[track].bits;
|
||||||
|
st->codec->bit_rate = st->codec->channels *
|
||||||
|
st->codec->sample_rate *
|
||||||
|
st->codec->bits_per_coded_sample;
|
||||||
|
st->codec->block_align = st->codec->channels *
|
||||||
|
st->codec->bits_per_coded_sample;
|
||||||
|
|
||||||
|
if (fourxm->tracks[track].adpcm){
|
||||||
|
st->codec->codec_id = AV_CODEC_ID_ADPCM_4XM;
|
||||||
|
} else if (st->codec->bits_per_coded_sample == 8) {
|
||||||
|
st->codec->codec_id = AV_CODEC_ID_PCM_U8;
|
||||||
|
} else
|
||||||
|
st->codec->codec_id = AV_CODEC_ID_PCM_S16LE;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
static int fourxm_read_header(AVFormatContext *s)
|
static int fourxm_read_header(AVFormatContext *s)
|
||||||
{
|
{
|
||||||
AVIOContext *pb = s->pb;
|
AVIOContext *pb = s->pb;
|
||||||
@ -100,7 +197,6 @@ static int fourxm_read_header(AVFormatContext *s)
|
|||||||
FourxmDemuxContext *fourxm = s->priv_data;
|
FourxmDemuxContext *fourxm = s->priv_data;
|
||||||
unsigned char *header;
|
unsigned char *header;
|
||||||
int i, ret;
|
int i, ret;
|
||||||
AVStream *st;
|
|
||||||
|
|
||||||
fourxm->track_count = 0;
|
fourxm->track_count = 0;
|
||||||
fourxm->tracks = NULL;
|
fourxm->tracks = NULL;
|
||||||
@ -140,104 +236,15 @@ static int fourxm_read_header(AVFormatContext *s)
|
|||||||
}
|
}
|
||||||
fourxm->fps = av_int2float(AV_RL32(&header[i + 12]));
|
fourxm->fps = av_int2float(AV_RL32(&header[i + 12]));
|
||||||
} else if (fourcc_tag == vtrk_TAG) {
|
} else if (fourcc_tag == vtrk_TAG) {
|
||||||
/* check that there is enough data */
|
if ((ret = parse_vtrk(s, fourxm, header + i, size)) < 0)
|
||||||
if (size != vtrk_SIZE) {
|
|
||||||
ret = AVERROR_INVALIDDATA;
|
|
||||||
goto fail;
|
goto fail;
|
||||||
}
|
|
||||||
fourxm->width = AV_RL32(&header[i + 36]);
|
|
||||||
fourxm->height = AV_RL32(&header[i + 40]);
|
|
||||||
|
|
||||||
/* allocate a new AVStream */
|
|
||||||
st = avformat_new_stream(s, NULL);
|
|
||||||
if (!st) {
|
|
||||||
ret = AVERROR(ENOMEM);
|
|
||||||
goto fail;
|
|
||||||
}
|
|
||||||
avpriv_set_pts_info(st, 60, 1, fourxm->fps);
|
|
||||||
|
|
||||||
fourxm->video_stream_index = st->index;
|
|
||||||
|
|
||||||
st->codec->codec_type = AVMEDIA_TYPE_VIDEO;
|
|
||||||
st->codec->codec_id = AV_CODEC_ID_4XM;
|
|
||||||
st->codec->extradata_size = 4;
|
|
||||||
st->codec->extradata = av_malloc(4);
|
|
||||||
AV_WL32(st->codec->extradata, AV_RL32(&header[i + 16]));
|
|
||||||
st->codec->width = fourxm->width;
|
|
||||||
st->codec->height = fourxm->height;
|
|
||||||
|
|
||||||
i += 8 + size;
|
i += 8 + size;
|
||||||
} else if (fourcc_tag == strk_TAG) {
|
} else if (fourcc_tag == strk_TAG) {
|
||||||
int current_track;
|
if ((ret = parse_strk(s, fourxm, header + i, size)) < 0)
|
||||||
/* check that there is enough data */
|
|
||||||
if (size != strk_SIZE) {
|
|
||||||
ret= AVERROR_INVALIDDATA;
|
|
||||||
goto fail;
|
goto fail;
|
||||||
}
|
|
||||||
current_track = AV_RL32(&header[i + 8]);
|
|
||||||
if((unsigned)current_track >= UINT_MAX / sizeof(AudioTrack) - 1){
|
|
||||||
av_log(s, AV_LOG_ERROR, "current_track too large\n");
|
|
||||||
ret = AVERROR_INVALIDDATA;
|
|
||||||
goto fail;
|
|
||||||
}
|
|
||||||
if (current_track + 1 > fourxm->track_count) {
|
|
||||||
fourxm->tracks = av_realloc_f(fourxm->tracks,
|
|
||||||
sizeof(AudioTrack),
|
|
||||||
current_track + 1);
|
|
||||||
if (!fourxm->tracks) {
|
|
||||||
ret = AVERROR(ENOMEM);
|
|
||||||
goto fail;
|
|
||||||
}
|
|
||||||
memset(&fourxm->tracks[fourxm->track_count], 0,
|
|
||||||
sizeof(AudioTrack) * (current_track + 1 - fourxm->track_count));
|
|
||||||
fourxm->track_count = current_track + 1;
|
|
||||||
}
|
|
||||||
fourxm->tracks[current_track].adpcm = AV_RL32(&header[i + 12]);
|
|
||||||
fourxm->tracks[current_track].channels = AV_RL32(&header[i + 36]);
|
|
||||||
fourxm->tracks[current_track].sample_rate = AV_RL32(&header[i + 40]);
|
|
||||||
fourxm->tracks[current_track].bits = AV_RL32(&header[i + 44]);
|
|
||||||
fourxm->tracks[current_track].audio_pts = 0;
|
|
||||||
if (fourxm->tracks[current_track].channels <= 0 ||
|
|
||||||
fourxm->tracks[current_track].sample_rate <= 0 ||
|
|
||||||
fourxm->tracks[current_track].bits < 0) {
|
|
||||||
av_log(s, AV_LOG_ERROR, "audio header invalid\n");
|
|
||||||
ret = AVERROR_INVALIDDATA;
|
|
||||||
goto fail;
|
|
||||||
}
|
|
||||||
if(!fourxm->tracks[current_track].adpcm && fourxm->tracks[current_track].bits<8){
|
|
||||||
av_log(s, AV_LOG_ERROR, "bits unspecified for non ADPCM\n");
|
|
||||||
ret = AVERROR_INVALIDDATA;
|
|
||||||
goto fail;
|
|
||||||
}
|
|
||||||
i += 8 + size;
|
i += 8 + size;
|
||||||
|
|
||||||
/* allocate a new AVStream */
|
|
||||||
st = avformat_new_stream(s, NULL);
|
|
||||||
if (!st) {
|
|
||||||
ret = AVERROR(ENOMEM);
|
|
||||||
goto fail;
|
|
||||||
}
|
|
||||||
|
|
||||||
st->id = current_track;
|
|
||||||
avpriv_set_pts_info(st, 60, 1,
|
|
||||||
fourxm->tracks[current_track].sample_rate);
|
|
||||||
|
|
||||||
fourxm->tracks[current_track].stream_index = st->index;
|
|
||||||
|
|
||||||
st->codec->codec_type = AVMEDIA_TYPE_AUDIO;
|
|
||||||
st->codec->codec_tag = 0;
|
|
||||||
st->codec->channels = fourxm->tracks[current_track].channels;
|
|
||||||
st->codec->sample_rate = fourxm->tracks[current_track].sample_rate;
|
|
||||||
st->codec->bits_per_coded_sample = fourxm->tracks[current_track].bits;
|
|
||||||
st->codec->bit_rate = st->codec->channels * st->codec->sample_rate *
|
|
||||||
st->codec->bits_per_coded_sample;
|
|
||||||
st->codec->block_align = st->codec->channels * st->codec->bits_per_coded_sample;
|
|
||||||
if (fourxm->tracks[current_track].adpcm){
|
|
||||||
st->codec->codec_id = AV_CODEC_ID_ADPCM_4XM;
|
|
||||||
} else if (st->codec->bits_per_coded_sample == 8) {
|
|
||||||
st->codec->codec_id = AV_CODEC_ID_PCM_U8;
|
|
||||||
} else
|
|
||||||
st->codec->codec_id = AV_CODEC_ID_PCM_S16LE;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user