⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 asf.c

📁 mediastreamer2是开源的网络传输媒体流的库
💻 C
📖 第 1 页 / 共 3 页
字号:
/* * ASF compatible demuxer * Copyright (c) 2000, 2001 Fabrice Bellard. * * 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 "avformat.h"#include "riff.h"#include "mpegaudio.h"#include "asf.h"#include "common.h"#include "asfcrypt.h"extern void ff_mms_set_stream_selection(URLContext *h, AVFormatContext *format);#undef NDEBUG#include <assert.h>#define FRAME_HEADER_SIZE 17// Fix Me! FRAME_HEADER_SIZE may be different.static const GUID index_guid = {    0x90, 0x08, 0x00, 0x33, 0xb1, 0xe5, 0xcf, 0x11, 0x89, 0xf4, 0x00, 0xa0, 0xc9, 0x03, 0x49, 0xcb};static const GUID stream_bitrate_guid = { /* (http://get.to/sdp) */    0xce, 0x75, 0xf8, 0x7b, 0x8d, 0x46, 0xd1, 0x11, 0x8d, 0x82, 0x00, 0x60, 0x97, 0xc9, 0xa2, 0xb2};/**********************************//* decoding *///#define DEBUG#ifdef DEBUG#define PRINT_IF_GUID(g,cmp) \if (!memcmp(g, &cmp, sizeof(GUID))) \    printf("(GUID: %s) ", #cmp)static void print_guid(const GUID *g){    int i;    PRINT_IF_GUID(g, asf_header);    else PRINT_IF_GUID(g, file_header);    else PRINT_IF_GUID(g, stream_header);    else PRINT_IF_GUID(g, audio_stream);    else PRINT_IF_GUID(g, audio_conceal_none);    else PRINT_IF_GUID(g, video_stream);    else PRINT_IF_GUID(g, video_conceal_none);    else PRINT_IF_GUID(g, command_stream);    else PRINT_IF_GUID(g, comment_header);    else PRINT_IF_GUID(g, codec_comment_header);    else PRINT_IF_GUID(g, codec_comment1_header);    else PRINT_IF_GUID(g, data_header);    else PRINT_IF_GUID(g, index_guid);    else PRINT_IF_GUID(g, head1_guid);    else PRINT_IF_GUID(g, head2_guid);    else PRINT_IF_GUID(g, my_guid);    else PRINT_IF_GUID(g, ext_stream_header);    else PRINT_IF_GUID(g, extended_content_header);    else PRINT_IF_GUID(g, ext_stream_embed_stream_header);    else PRINT_IF_GUID(g, ext_stream_audio_stream);    else PRINT_IF_GUID(g, metadata_header);    else PRINT_IF_GUID(g, stream_bitrate_guid);    else        printf("(GUID: unknown) ");    for(i=0;i<16;i++)        printf(" 0x%02x,", (*g)[i]);    printf("}\n");}#undef PRINT_IF_GUID#endifstatic void get_guid(ByteIOContext *s, GUID *g){    assert(sizeof(*g) == 16);    get_buffer(s, *g, sizeof(*g));}#if 0static void get_str16(ByteIOContext *pb, char *buf, int buf_size){    int len, c;    char *q;    len = get_le16(pb);    q = buf;    while (len > 0) {        c = get_le16(pb);        if ((q - buf) < buf_size - 1)            *q++ = c;        len--;    }    *q = '\0';}#endifstatic void get_str16_nolen(ByteIOContext *pb, int len, char *buf, int buf_size){    char* q = buf;    len /= 2;    while (len--) {        uint8_t tmp;        PUT_UTF8(get_le16(pb), tmp, if (q - buf < buf_size - 1) *q++ = tmp;)    }    *q = '\0';}static int asf_probe(AVProbeData *pd){    /* check file header */    if (!memcmp(pd->buf, &asf_header, sizeof(GUID)))        return AVPROBE_SCORE_MAX;    else        return 0;}static int get_value(ByteIOContext *pb, int type){    switch(type){        case 2: return get_le32(pb);        case 3: return get_le32(pb);        case 4: return get_le64(pb);        case 5: return get_le16(pb);        default:return INT_MIN;    }}static int asf_read_header(AVFormatContext *s, AVFormatParameters *ap){    ASFContext *asf = s->priv_data;    GUID g;    ByteIOContext *pb = s->pb;    AVStream *st;    ASFStream *asf_st;    int size, i;    int64_t gsize;    AVRational dar[128];    uint32_t bitrate[128];    memset(dar, 0, sizeof(dar));    memset(bitrate, 0, sizeof(bitrate));    get_guid(pb, &g);    if (memcmp(&g, &asf_header, sizeof(GUID)))        goto fail;    get_le64(pb);    get_le32(pb);    get_byte(pb);    get_byte(pb);    memset(&asf->asfid2avid, -1, sizeof(asf->asfid2avid));    for(;;) {        get_guid(pb, &g);        gsize = get_le64(pb);#ifdef DEBUG        printf("%08"PRIx64": ", url_ftell(pb) - 24);        print_guid(&g);        printf("  size=0x%"PRIx64"\n", gsize);#endif        if (!memcmp(&g, &data_header, sizeof(GUID))) {            asf->data_object_offset = url_ftell(pb);            // if not streaming, gsize is not unlimited (how?), and there is enough space in the file..            if (!(asf->hdr.flags & 0x01) && gsize >= 100) {                asf->data_object_size = gsize - 24;            } else {                asf->data_object_size = (uint64_t)-1;            }            break;        }        if (gsize < 24)            goto fail;        if (!memcmp(&g, &file_header, sizeof(GUID))) {            get_guid(pb, &asf->hdr.guid);            asf->hdr.file_size          = get_le64(pb);            asf->hdr.create_time        = get_le64(pb);            asf->nb_packets             = get_le64(pb);            asf->hdr.send_time          = get_le64(pb);            asf->hdr.play_time          = get_le64(pb);            asf->hdr.preroll            = get_le32(pb);            asf->hdr.ignore             = get_le32(pb);            asf->hdr.flags              = get_le32(pb);            asf->hdr.min_pktsize        = get_le32(pb);            asf->hdr.max_pktsize        = get_le32(pb);            asf->hdr.max_bitrate        = get_le32(pb);            asf->packet_size = asf->hdr.max_pktsize;        } else if (!memcmp(&g, &stream_header, sizeof(GUID))) {            int type, type_specific_size, sizeX;            uint64_t total_size;            unsigned int tag1;            int64_t pos1, pos2, start_time;            int test_for_ext_stream_audio, is_dvr_ms_audio=0;            pos1 = url_ftell(pb);            st = av_new_stream(s, 0);            if (!st)                goto fail;            av_set_pts_info(st, 32, 1, 1000); /* 32 bit pts in ms */            asf_st = av_mallocz(sizeof(ASFStream));            if (!asf_st)                goto fail;            st->priv_data = asf_st;            start_time = asf->hdr.preroll;            if(!(asf->hdr.flags & 0x01)) { // if we aren't streaming...                st->duration = asf->hdr.send_time /                    (10000000 / 1000) - start_time;            }            get_guid(pb, &g);            test_for_ext_stream_audio = 0;            if (!memcmp(&g, &audio_stream, sizeof(GUID))) {                type = CODEC_TYPE_AUDIO;            } else if (!memcmp(&g, &video_stream, sizeof(GUID))) {                type = CODEC_TYPE_VIDEO;            } else if (!memcmp(&g, &command_stream, sizeof(GUID))) {                type = CODEC_TYPE_UNKNOWN;            } else if (!memcmp(&g, &ext_stream_embed_stream_header, sizeof(GUID))) {                test_for_ext_stream_audio = 1;                type = CODEC_TYPE_UNKNOWN;            } else {                goto fail;            }            get_guid(pb, &g);            total_size = get_le64(pb);            type_specific_size = get_le32(pb);            get_le32(pb);            st->id = get_le16(pb) & 0x7f; /* stream id */            // mapping of asf ID to AV stream ID;            asf->asfid2avid[st->id] = s->nb_streams - 1;            get_le32(pb);            if (test_for_ext_stream_audio) {                get_guid(pb, &g);                if (!memcmp(&g, &ext_stream_audio_stream, sizeof(GUID))) {                    type = CODEC_TYPE_AUDIO;                    is_dvr_ms_audio=1;                    get_guid(pb, &g);                    get_le32(pb);                    get_le32(pb);                    get_le32(pb);                    get_guid(pb, &g);                    get_le32(pb);                }            }            st->codec->codec_type = type;            if (type == CODEC_TYPE_AUDIO) {                get_wav_header(pb, st->codec, type_specific_size);                if (is_dvr_ms_audio) {                    // codec_id and codec_tag are unreliable in dvr_ms                    // files. Set them later by probing stream.                    st->codec->codec_id = CODEC_ID_NONE;                    st->codec->codec_tag = 0;                }                st->need_parsing = AVSTREAM_PARSE_FULL;                /* We have to init the frame size at some point .... */                pos2 = url_ftell(pb);                if (gsize >= (pos2 + 8 - pos1 + 24)) {                    asf_st->ds_span = get_byte(pb);                    asf_st->ds_packet_size = get_le16(pb);                    asf_st->ds_chunk_size = get_le16(pb);                    get_le16(pb); //ds_data_size                    get_byte(pb); //ds_silence_data                }                //printf("Descrambling: ps:%d cs:%d ds:%d s:%d  sd:%d\n",                //       asf_st->ds_packet_size, asf_st->ds_chunk_size,                //       asf_st->ds_data_size, asf_st->ds_span, asf_st->ds_silence_data);                if (asf_st->ds_span > 1) {                    if (!asf_st->ds_chunk_size                        || (asf_st->ds_packet_size/asf_st->ds_chunk_size <= 1)                        || asf_st->ds_packet_size % asf_st->ds_chunk_size)                        asf_st->ds_span = 0; // disable descrambling                }                switch (st->codec->codec_id) {                case CODEC_ID_MP3:                    st->codec->frame_size = MPA_FRAME_SIZE;                    break;                case CODEC_ID_PCM_S16LE:                case CODEC_ID_PCM_S16BE:                case CODEC_ID_PCM_U16LE:                case CODEC_ID_PCM_U16BE:                case CODEC_ID_PCM_S8:                case CODEC_ID_PCM_U8:                case CODEC_ID_PCM_ALAW:                case CODEC_ID_PCM_MULAW:                    st->codec->frame_size = 1;                    break;                default:                    /* This is probably wrong, but it prevents a crash later */                    st->codec->frame_size = 1;                    break;                }            } else if (type == CODEC_TYPE_VIDEO) {                get_le32(pb);                get_le32(pb);                get_byte(pb);                size = get_le16(pb); /* size */                sizeX= get_le32(pb); /* size */                st->codec->width = get_le32(pb);                st->codec->height = get_le32(pb);                /* not available for asf */                get_le16(pb); /* panes */                st->codec->bits_per_sample = get_le16(pb); /* depth */                tag1 = get_le32(pb);                url_fskip(pb, 20);//                av_log(NULL, AV_LOG_DEBUG, "size:%d tsize:%d sizeX:%d\n", size, total_size, sizeX);                size= sizeX;                if (size > 40) {                    st->codec->extradata_size = size - 40;                    st->codec->extradata = av_mallocz(st->codec->extradata_size + FF_INPUT_BUFFER_PADDING_SIZE);                    get_buffer(pb, st->codec->extradata, st->codec->extradata_size);                }        /* Extract palette from extradata if bpp <= 8 */        /* This code assumes that extradata contains only palette */        /* This is true for all paletted codecs implemented in ffmpeg */        if (st->codec->extradata_size && (st->codec->bits_per_sample <= 8)) {            st->codec->palctrl = av_mallocz(sizeof(AVPaletteControl));#ifdef WORDS_BIGENDIAN            for (i = 0; i < FFMIN(st->codec->extradata_size, AVPALETTE_SIZE)/4; i++)                st->codec->palctrl->palette[i] = bswap_32(((uint32_t*)st->codec->extradata)[i]);#else            memcpy(st->codec->palctrl->palette, st->codec->extradata,                   FFMIN(st->codec->extradata_size, AVPALETTE_SIZE));#endif            st->codec->palctrl->palette_changed = 1;        }                st->codec->codec_tag = tag1;                st->codec->codec_id = codec_get_id(codec_bmp_tags, tag1);                if(tag1 == MKTAG('D', 'V', 'R', ' '))                    st->need_parsing = AVSTREAM_PARSE_FULL;            }            pos2 = url_ftell(pb);            url_fskip(pb, gsize - (pos2 - pos1 + 24));        } else if (!memcmp(&g, &comment_header, sizeof(GUID))) {            int len1, len2, len3, len4, len5;            len1 = get_le16(pb);            len2 = get_le16(pb);            len3 = get_le16(pb);            len4 = get_le16(pb);            len5 = get_le16(pb);            get_str16_nolen(pb, len1, s->title    , sizeof(s->title));            get_str16_nolen(pb, len2, s->author   , sizeof(s->author));            get_str16_nolen(pb, len3, s->copyright, sizeof(s->copyright));            get_str16_nolen(pb, len4, s->comment  , sizeof(s->comment));            url_fskip(pb, len5);        } else if (!memcmp(&g, &stream_bitrate_guid, sizeof(GUID))) {            int stream_count = get_le16(pb);            int j;//            av_log(NULL, AV_LOG_ERROR, "stream bitrate properties\n");//            av_log(NULL, AV_LOG_ERROR, "streams %d\n", streams);

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -