Skip to content
Snippets Groups Projects
utils.c 65 KiB
Newer Older
  • Learn to ignore specific revisions
  •                                  int stream_index, int64_t timestamp)
    {
        int index;
        AVStream *st;
        AVIndexEntry *ie;
    
        if (!s->index_built) {
            if (is_raw_stream(s)) {
                av_build_index_raw(s);
            } else {
                return -1;
            }
            s->index_built = 1;
        }
    
        if (stream_index < 0)
            stream_index = 0;
        st = s->streams[stream_index];
    
        index = av_index_search_timestamp(st, timestamp);
    
        if (index < 0)
            return -1;
    
        /* now we have found the index, we can seek */
        ie = &st->index_entries[index];
        av_read_frame_flush(s);
        url_fseek(&s->pb, ie->pos, SEEK_SET);
        st->cur_dts = ie->timestamp;
        return 0;
    }
    
    /**
     * Seek to the key frame just before the frame at timestamp
     * 'timestamp' in 'stream_index'. If stream_index is (-1), a default
     * stream is selected 
     */
    int av_seek_frame(AVFormatContext *s, int stream_index, int64_t timestamp)
    {
        int ret;
        
        av_read_frame_flush(s);
    
        /* first, we try the format specific seek */
        if (s->iformat->read_seek)
            ret = s->iformat->read_seek(s, stream_index, timestamp);
        else
            ret = -1;
        if (ret >= 0) {
            return 0;
        }
        
        return av_seek_frame_generic(s, stream_index, timestamp);
    }
    
    
    /*******************************************************/
    
    
    /* return TRUE if the stream has accurate timings for at least one component */
    static int av_has_timings(AVFormatContext *ic)
    {
        int i;
        AVStream *st;
    
        for(i = 0;i < ic->nb_streams; i++) {
            st = ic->streams[i];
            if (st->start_time != AV_NOPTS_VALUE &&
                st->duration != AV_NOPTS_VALUE)
                return 1;
        }
        return 0;
    }
    
    /* estimate the stream timings from the one of each components. Also
       compute the global bitrate if possible */
    static void av_update_stream_timings(AVFormatContext *ic)
    {
        int64_t start_time, end_time, end_time1;
        int i;
        AVStream *st;
    
        start_time = MAXINT64;
        end_time = MININT64;
        for(i = 0;i < ic->nb_streams; i++) {
            st = ic->streams[i];
            if (st->start_time != AV_NOPTS_VALUE) {
                if (st->start_time < start_time)
                    start_time = st->start_time;
                if (st->duration != AV_NOPTS_VALUE) {
                    end_time1 = st->start_time + st->duration;
                    if (end_time1 > end_time)
                        end_time = end_time1;
                }
            }
        }
        if (start_time != MAXINT64) {
            ic->start_time = start_time;
            if (end_time != MAXINT64) {
                ic->duration = end_time - start_time;
                if (ic->file_size > 0) {
                    /* compute the bit rate */
                    ic->bit_rate = (double)ic->file_size * 8.0 * AV_TIME_BASE / 
                        (double)ic->duration;
                }
            }
        }
    
    }
    
    static void fill_all_stream_timings(AVFormatContext *ic)
    {
        int i;
        AVStream *st;
    
        av_update_stream_timings(ic);
        for(i = 0;i < ic->nb_streams; i++) {
            st = ic->streams[i];
            if (st->start_time == AV_NOPTS_VALUE) {
                st->start_time = ic->start_time;
                st->duration = ic->duration;
            }
        }
    }
    
    static void av_estimate_timings_from_bit_rate(AVFormatContext *ic)
    {
        int64_t filesize, duration;
        int bit_rate, i;
        AVStream *st;
    
        /* if bit_rate is already set, we believe it */
        if (ic->bit_rate == 0) {
            bit_rate = 0;
            for(i=0;i<ic->nb_streams;i++) {
                st = ic->streams[i];
                bit_rate += st->codec.bit_rate;
            }
            ic->bit_rate = bit_rate;
        }
    
        /* if duration is already set, we believe it */
        if (ic->duration == AV_NOPTS_VALUE && 
            ic->bit_rate != 0 && 
            ic->file_size != 0)  {
            filesize = ic->file_size;
            if (filesize > 0) {
                duration = (int64_t)((8 * AV_TIME_BASE * (double)filesize) / (double)ic->bit_rate);
                for(i = 0; i < ic->nb_streams; i++) {
                    st = ic->streams[i];
                    if (st->start_time == AV_NOPTS_VALUE ||
                        st->duration == AV_NOPTS_VALUE) {
                        st->start_time = 0;
                        st->duration = duration;
                    }
                }
            }
        }
    }
    
    #define DURATION_MAX_READ_SIZE 250000
    
    /* only usable for MPEG-PS streams */
    static void av_estimate_timings_from_pts(AVFormatContext *ic)
    {
        AVPacket pkt1, *pkt = &pkt1;
        AVStream *st;
        int read_size, i, ret;
        int64_t start_time, end_time, end_time1;
        int64_t filesize, offset, duration;
        
    
        /* free previous packet */
        if (ic->cur_st && ic->cur_st->parser)
            av_free_packet(&ic->cur_pkt); 
        ic->cur_st = NULL;
    
        /* flush packet queue */
        flush_packet_queue(ic);
        
    
    
        /* we read the first packets to get the first PTS (not fully
           accurate, but it is enough now) */
        url_fseek(&ic->pb, 0, SEEK_SET);
        read_size = 0;
        for(;;) {
            if (read_size >= DURATION_MAX_READ_SIZE)
                break;
            /* if all info is available, we can stop */
            for(i = 0;i < ic->nb_streams; i++) {
                st = ic->streams[i];
                if (st->start_time == AV_NOPTS_VALUE)
                    break;
            }
            if (i == ic->nb_streams)
                break;
    
            ret = av_read_packet(ic, pkt);
            if (ret != 0)
                break;
            read_size += pkt->size;
            st = ic->streams[pkt->stream_index];
            if (pkt->pts != AV_NOPTS_VALUE) {
                if (st->start_time == AV_NOPTS_VALUE)
                    st->start_time = (int64_t)((double)pkt->pts * ic->pts_num * (double)AV_TIME_BASE / ic->pts_den);
    
    
        /* we compute the minimum start_time and use it as default */
        start_time = MAXINT64;
        for(i = 0; i < ic->nb_streams; i++) {
            st = ic->streams[i];
            if (st->start_time != AV_NOPTS_VALUE &&
                st->start_time < start_time)
                start_time = st->start_time;
        }
        if (start_time != MAXINT64)
            ic->start_time = start_time;
        
        /* estimate the end time (duration) */
        /* XXX: may need to support wrapping */
        filesize = ic->file_size;
        offset = filesize - DURATION_MAX_READ_SIZE;
        if (offset < 0)
            offset = 0;
    
        url_fseek(&ic->pb, offset, SEEK_SET);
        read_size = 0;
        for(;;) {
            if (read_size >= DURATION_MAX_READ_SIZE)
                break;
            /* if all info is available, we can stop */
            for(i = 0;i < ic->nb_streams; i++) {
                st = ic->streams[i];
                if (st->duration == AV_NOPTS_VALUE)
                    break;
            }
            if (i == ic->nb_streams)
                break;
            
            ret = av_read_packet(ic, pkt);
            if (ret != 0)
                break;
            read_size += pkt->size;
            st = ic->streams[pkt->stream_index];
            if (pkt->pts != AV_NOPTS_VALUE) {
                end_time = (int64_t)((double)pkt->pts * ic->pts_num * (double)AV_TIME_BASE / ic->pts_den);
                duration = end_time - st->start_time;
                if (duration > 0) {
                    if (st->duration == AV_NOPTS_VALUE ||
                        st->duration < duration)
                        st->duration = duration;
                }
            }
            av_free_packet(pkt);
        }
        
        /* estimate total duration */
        end_time = MININT64;
        for(i = 0;i < ic->nb_streams; i++) {
            st = ic->streams[i];
            if (st->duration != AV_NOPTS_VALUE) {
                end_time1 = st->start_time + st->duration;
                if (end_time1 > end_time)
                    end_time = end_time1;
            }
        }
        
        /* update start_time (new stream may have been created, so we do
           it at the end */
        if (ic->start_time != AV_NOPTS_VALUE) {
            for(i = 0; i < ic->nb_streams; i++) {
                st = ic->streams[i];
                if (st->start_time == AV_NOPTS_VALUE)
                    st->start_time = ic->start_time;
            }
        }
    
        if (end_time != MININT64) {
            /* put dummy values for duration if needed */
            for(i = 0;i < ic->nb_streams; i++) {
                st = ic->streams[i];
                if (st->duration == AV_NOPTS_VALUE && 
                    st->start_time != AV_NOPTS_VALUE)
                    st->duration = end_time - st->start_time;
            }
            ic->duration = end_time - ic->start_time;
        }
    
        url_fseek(&ic->pb, 0, SEEK_SET);
    }
    
    static void av_estimate_timings(AVFormatContext *ic)
    {
        URLContext *h;
        int64_t file_size;
    
        /* get the file size, if possible */
        if (ic->iformat->flags & AVFMT_NOFILE) {
            file_size = 0;
        } else {
            h = url_fileno(&ic->pb);
            file_size = url_filesize(h);
            if (file_size < 0)
                file_size = 0;
        }
        ic->file_size = file_size;
    
        if (ic->iformat == &mpegps_demux) {
            /* get accurate estimate from the PTSes */
            av_estimate_timings_from_pts(ic);
        } else if (av_has_timings(ic)) {
            /* at least one components has timings - we use them for all
               the components */
            fill_all_stream_timings(ic);
        } else {
            /* less precise: use bit rate info */
            av_estimate_timings_from_bit_rate(ic);
        }
        av_update_stream_timings(ic);
    
    #if 0
        {
            int i;
            AVStream *st;
            for(i = 0;i < ic->nb_streams; i++) {
                st = ic->streams[i];
            printf("%d: start_time: %0.3f duration: %0.3f\n", 
                   i, (double)st->start_time / AV_TIME_BASE, 
                   (double)st->duration / AV_TIME_BASE);
            }
            printf("stream: start_time: %0.3f duration: %0.3f bitrate=%d kb/s\n", 
                   (double)ic->start_time / AV_TIME_BASE, 
                   (double)ic->duration / AV_TIME_BASE,
                   ic->bit_rate / 1000);
        }
    #endif
    }
    
    
    static int has_codec_parameters(AVCodecContext *enc)
    {
        int val;
        switch(enc->codec_type) {
        case CODEC_TYPE_AUDIO:
            val = enc->sample_rate;
            break;
        case CODEC_TYPE_VIDEO:
            val = enc->width;
            break;
        default:
            val = 1;
            break;
        }
        return (val != 0);
    }
    
    
    static int try_decode_frame(AVStream *st, const uint8_t *data, int size)
    {
        int16_t *samples;
        AVCodec *codec;
        int got_picture, ret;
        AVFrame picture;
        
        codec = avcodec_find_decoder(st->codec.codec_id);
        if (!codec)
            return -1;
        ret = avcodec_open(&st->codec, codec);
        if (ret < 0)
            return ret;
        switch(st->codec.codec_type) {
        case CODEC_TYPE_VIDEO:
            ret = avcodec_decode_video(&st->codec, &picture, 
                                       &got_picture, (uint8_t *)data, size);
            break;
        case CODEC_TYPE_AUDIO:
            samples = av_malloc(AVCODEC_MAX_AUDIO_FRAME_SIZE);
            if (!samples)
                goto fail;
            ret = avcodec_decode_audio(&st->codec, samples, 
                                       &got_picture, (uint8_t *)data, size);
            av_free(samples);
            break;
        default:
            break;
        }
     fail:
        avcodec_close(&st->codec);
        return ret;
    }
    
    /* absolute maximum size we read until we abort */
    #define MAX_READ_SIZE        5000000
    
    /* maximum duration until we stop analysing the stream */
    #define MAX_STREAM_DURATION  ((int)(AV_TIME_BASE * 1.0))
    
    
    /**
     * Read the beginning of a media file to get stream information. This
     * is useful for file formats with no headers such as MPEG. This
     * function also compute the real frame rate in case of mpeg2 repeat
     * frame mode.
     *
     * @param ic media file handle
     * @return >=0 if OK. AVERROR_xxx if error.  
     */
    int av_find_stream_info(AVFormatContext *ic)
    {
    
        int i, count, ret, read_size;
    
        AVPacketList *pktl=NULL, **ppktl;
    
        count = 0;
        read_size = 0;
        ppktl = &ic->packet_buffer;
        for(;;) {
            /* check if one codec still needs to be handled */
            for(i=0;i<ic->nb_streams;i++) {
                st = ic->streams[i];
    
                if (!has_codec_parameters(&st->codec))
    
                    break;
            }
            if (i == ic->nb_streams) {
                /* NOTE: if the format has no header, then we need to read
                   some packets to get most of the streams, so we cannot
                   stop here */
    
                if (!(ic->ctx_flags & AVFMTCTX_NOHEADER)) {
    
                    /* if we found the info for all the codecs, we can stop */
                    ret = count;
                    break;
                }
            } else {
                /* we did not get all the codec info, but we read too much data */
    
                if (read_size >= MAX_READ_SIZE) {
    
            /* NOTE: a new stream can be added there if no header in file
               (AVFMTCTX_NOHEADER) */
            ret = av_read_frame_internal(ic, &pkt1);
            if (ret < 0) {
                /* EOF or error */
                ret = -1; /* we could not have all the codec parameters before EOF */
                if ((ic->ctx_flags & AVFMTCTX_NOHEADER) &&
                    i == ic->nb_streams)
                    ret = 0;
                break;
            }
    
    
            pktl = av_mallocz(sizeof(AVPacketList));
            if (!pktl) {
                ret = AVERROR_NOMEM;
                break;
            }
    
            /* add the packet in the buffered packet list */
            *ppktl = pktl;
            ppktl = &pktl->next;
    
            pkt = &pktl->pkt;
    
            *pkt = pkt1;
            
            /* duplicate the packet */
            if (av_dup_packet(pkt) < 0) {
                    ret = AVERROR_NOMEM;
                    break;
    
            read_size += pkt->size;
    
            st->codec_info_duration += pkt->duration;
            if (pkt->duration != 0)
                st->codec_info_nb_frames++;
    
            /* if still no information, we try to open the codec and to
               decompress the frame. We try to avoid that in most cases as
               it takes longer and uses more memory. For MPEG4, we need to
               decompress for Quicktime. */
            if (!has_codec_parameters(&st->codec) &&
                (st->codec.codec_id == CODEC_ID_FLV1 ||
                 st->codec.codec_id == CODEC_ID_H264 ||
                 st->codec.codec_id == CODEC_ID_H263 ||
    
                 st->codec.codec_id == CODEC_ID_VORBIS ||
    
                 (st->codec.codec_id == CODEC_ID_MPEG4 && !st->need_parsing)))
                try_decode_frame(st, pkt->data, pkt->size);
            
            if (st->codec_info_duration >= MAX_STREAM_DURATION) {
                break;
    
            }
            count++;
        }
    
        /* set real frame rate info */
        for(i=0;i<ic->nb_streams;i++) {
            st = ic->streams[i];
            if (st->codec.codec_type == CODEC_TYPE_VIDEO) {
    
                /* compute the real frame rate for telecine */
                if ((st->codec.codec_id == CODEC_ID_MPEG1VIDEO ||
                     st->codec.codec_id == CODEC_ID_MPEG2VIDEO) &&
                    st->codec.sub_id == 2) {
                    if (st->codec_info_nb_frames >= 20) {
                        float coded_frame_rate, est_frame_rate;
                        est_frame_rate = ((double)st->codec_info_nb_frames * AV_TIME_BASE) / 
                            (double)st->codec_info_duration ;
                        coded_frame_rate = (double)st->codec.frame_rate /
                            (double)st->codec.frame_rate_base;
    #if 0
                        printf("telecine: coded_frame_rate=%0.3f est_frame_rate=%0.3f\n", 
                               coded_frame_rate, est_frame_rate);
    #endif
                        /* if we detect that it could be a telecine, we
                           signal it. It would be better to do it at a
                           higher level as it can change in a film */
                        if (coded_frame_rate >= 24.97 && 
                            (est_frame_rate >= 23.5 && est_frame_rate < 24.5)) {
                            st->r_frame_rate = 24024;
                            st->r_frame_rate_base = 1001;
                        }
                    }
                }
                /* if no real frame rate, use the codec one */
    
                if (!st->r_frame_rate){
                    st->r_frame_rate      = st->codec.frame_rate;
                    st->r_frame_rate_base = st->codec.frame_rate_base;
                }
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
        }
    
    /*******************************************************/
    
    /**
     * start playing a network based stream (e.g. RTSP stream) at the
     * current position 
     */
    int av_read_play(AVFormatContext *s)
    {
        if (!s->iformat->read_play)
            return AVERROR_NOTSUPP;
        return s->iformat->read_play(s);
    }
    
    /**
     * pause a network based stream (e.g. RTSP stream). Use av_read_play()
     * to resume it.
     */
    int av_read_pause(AVFormatContext *s)
    {
        if (!s->iformat->read_pause)
            return AVERROR_NOTSUPP;
        return s->iformat->read_pause(s);
    }
    
    
    /**
     * Close a media file (but not its codecs)
     *
     * @param s media file handle
     */
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
    void av_close_input_file(AVFormatContext *s)
    {
    
        int i, must_open_file;
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
    
    
        /* free previous packet */
        if (s->cur_st && s->cur_st->parser)
            av_free_packet(&s->cur_pkt); 
    
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
        for(i=0;i<s->nb_streams;i++) {
    
            /* free all data in a stream component */
            st = s->streams[i];
    
            if (st->parser) {
                av_parser_close(st->parser);
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
            }
    
            av_free(st->index_entries);
            av_free(st);
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
        }
    
        must_open_file = 1;
    
        if (s->iformat->flags & AVFMT_NOFILE) {
    
            must_open_file = 0;
        }
        if (must_open_file) {
    
            url_fclose(&s->pb);
        }
    
        av_free(s);
    
    /**
     * Add a new stream to a media file. Can only be called in the
    
     * read_header function. If the flag AVFMTCTX_NOHEADER is in the
     * format context, then new streams can be added in read_packet too.
    
     * @param id file format dependent stream id 
    
     */
    AVStream *av_new_stream(AVFormatContext *s, int id)
    {
        AVStream *st;
    
        if (s->nb_streams >= MAX_STREAMS)
            return NULL;
    
        st = av_mallocz(sizeof(AVStream));
        if (!st)
            return NULL;
    
    Michael Niedermayer's avatar
    Michael Niedermayer committed
        avcodec_get_context_defaults(&st->codec);
    
        if (s->iformat) {
            /* no default bitrate if decoding */
            st->codec.bit_rate = 0;
        }
    
        st->start_time = AV_NOPTS_VALUE;
        st->duration = AV_NOPTS_VALUE;
    
        s->streams[s->nb_streams++] = st;
        return st;
    }
    
    /************************************************************/
    /* output media file */
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
    
    
    int av_set_parameters(AVFormatContext *s, AVFormatParameters *ap)
    {
        int ret;
    
        
        if (s->oformat->priv_data_size > 0) {
            s->priv_data = av_mallocz(s->oformat->priv_data_size);
            if (!s->priv_data)
                return AVERROR_NOMEM;
        } else
            s->priv_data = NULL;
    	
    
        if (s->oformat->set_parameters) {
            ret = s->oformat->set_parameters(s, ap);
            if (ret < 0)
                return ret;
        }
        return 0;
    }
    
    
    /**
     * allocate the stream private data and write the stream header to an
     * output media file
     *
     * @param s media file handle
     * @return 0 if OK. AVERROR_xxx if error.  
     */
    int av_write_header(AVFormatContext *s)
    {
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
        /* default pts settings is MPEG like */
        av_set_pts_info(s, 33, 1, 90000);
    
        ret = s->oformat->write_header(s);
        if (ret < 0)
            return ret;
    
        /* init PTS generation */
        for(i=0;i<s->nb_streams;i++) {
            st = s->streams[i];
    
            switch (st->codec.codec_type) {
            case CODEC_TYPE_AUDIO:
                av_frac_init(&st->pts, 0, 0, 
    
                             (int64_t)s->pts_num * st->codec.sample_rate);
    
                break;
            case CODEC_TYPE_VIDEO:
                av_frac_init(&st->pts, 0, 0, 
    
                             (int64_t)s->pts_num * st->codec.frame_rate);
    
     * Write a packet to an output media file. The packet shall contain
     * one audio or video frame.
    
     * @param stream_index stream index
     * @param buf buffer containing the frame data
     * @param size size of buffer
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
     * @return < 0 if error, = 0 if OK, 1 if end of stream wanted.
    
    int av_write_frame(AVFormatContext *s, int stream_index, const uint8_t *buf, 
                       int size)
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
    {
    
        int64_t pts_mask;
    
    
        st = s->streams[stream_index];
        pts_mask = (1LL << s->pts_wrap_bits) - 1;
    
    
        /* HACK/FIXME we skip all zero size audio packets so a encoder can pass pts by outputing zero size packets */
        if(st->codec.codec_type==CODEC_TYPE_AUDIO && size==0)
            ret = 0;
        else
            ret = s->oformat->write_packet(s, stream_index, buf, size, 
                                           st->pts.val & pts_mask);
        
    
        if (ret < 0)
            return ret;
    
        /* update pts */
        switch (st->codec.codec_type) {
        case CODEC_TYPE_AUDIO:
    
            frame_size = get_audio_frame_size(&st->codec, size);
    
            /* HACK/FIXME, we skip the initial 0-size packets as they are most likely equal to the encoder delay,
    
               but it would be better if we had the real timestamps from the encoder */
            if (frame_size >= 0 && (size || st->pts.num!=st->pts.den>>1 || st->pts.val)) {
    
                av_frac_add(&st->pts, 
                            (int64_t)s->pts_den * frame_size);
    
            break;
        case CODEC_TYPE_VIDEO:
            av_frac_add(&st->pts, 
    
                        (int64_t)s->pts_den * st->codec.frame_rate_base);
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
        return ret;
    
    }
    
    /**
     * write the stream trailer to an output media file and and free the
     * file private data.
     *
     * @param s media file handle
     * @return 0 if OK. AVERROR_xxx if error.  */
    int av_write_trailer(AVFormatContext *s)
    {
        int ret;
        ret = s->oformat->write_trailer(s);
        av_freep(&s->priv_data);
        return ret;
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
    }
    
    /* "user interface" functions */
    
    void dump_format(AVFormatContext *ic,
                     int index, 
                     const char *url,
                     int is_output)
    {
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
        char buf[256];
    
    
    Michael Niedermayer's avatar
    Michael Niedermayer committed
        av_log(NULL, AV_LOG_DEBUG, "%s #%d, %s, %s '%s':\n", 
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
                is_output ? "Output" : "Input",
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
                is_output ? "to" : "from", url);
    
    Michael Niedermayer's avatar
    Michael Niedermayer committed
            av_log(NULL, AV_LOG_DEBUG, "  Duration: ");
    
            if (ic->duration != AV_NOPTS_VALUE) {
                int hours, mins, secs, us;
                secs = ic->duration / AV_TIME_BASE;
                us = ic->duration % AV_TIME_BASE;
                mins = secs / 60;
                secs %= 60;
                hours = mins / 60;
                mins %= 60;
    
    Michael Niedermayer's avatar
    Michael Niedermayer committed
                av_log(NULL, AV_LOG_DEBUG, "%02d:%02d:%02d.%01d", hours, mins, secs, 
    
    Michael Niedermayer's avatar
    Michael Niedermayer committed
                av_log(NULL, AV_LOG_DEBUG, "N/A");
    
    Michael Niedermayer's avatar
    Michael Niedermayer committed
            av_log(NULL, AV_LOG_DEBUG, ", bitrate: ");
    
    Michael Niedermayer's avatar
    Michael Niedermayer committed
                av_log(NULL, AV_LOG_DEBUG,"%d kb/s", ic->bit_rate / 1000);
    
    Michael Niedermayer's avatar
    Michael Niedermayer committed
                av_log(NULL, AV_LOG_DEBUG, "N/A");
    
    Michael Niedermayer's avatar
    Michael Niedermayer committed
            av_log(NULL, AV_LOG_DEBUG, "\n");
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
        for(i=0;i<ic->nb_streams;i++) {
            AVStream *st = ic->streams[i];
            avcodec_string(buf, sizeof(buf), &st->codec, is_output);
    
    Michael Niedermayer's avatar
    Michael Niedermayer committed
            av_log(NULL, AV_LOG_DEBUG, "  Stream #%d.%d", index, i);
    
            /* the pid is an important information, so we display it */
            /* XXX: add a generic system */
            if (is_output)
                flags = ic->oformat->flags;
            else
                flags = ic->iformat->flags;
            if (flags & AVFMT_SHOW_IDS) {
    
    Michael Niedermayer's avatar
    Michael Niedermayer committed
                av_log(NULL, AV_LOG_DEBUG, "[0x%x]", st->id);
    
    Michael Niedermayer's avatar
    Michael Niedermayer committed
            av_log(NULL, AV_LOG_DEBUG, ": %s\n", buf);
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
        }
    }
    
    typedef struct {
    
        const char *abv;
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
        int width, height;
    
        int frame_rate, frame_rate_base;
    } AbvEntry;
    
    static AbvEntry frame_abvs[] = {
    
        { "ntsc",      720, 480, 30000, 1001 },
        { "pal",       720, 576,    25,    1 },
        { "qntsc",     352, 240, 30000, 1001 }, /* VCD compliant ntsc */
        { "qpal",      352, 288,    25,    1 }, /* VCD compliant pal */
    
        { "sntsc",     640, 480, 30000, 1001 }, /* square pixel ntsc */
        { "spal",      768, 576,    25,    1 }, /* square pixel pal */
    
        { "film",      352, 240,    24,    1 },
        { "ntsc-film", 352, 240, 24000, 1001 },
        { "sqcif",     128,  96,     0,    0 },
        { "qcif",      176, 144,     0,    0 },
        { "cif",       352, 288,     0,    0 },
        { "4cif",      704, 576,     0,    0 },
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
    };
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
    int parse_image_size(int *width_ptr, int *height_ptr, const char *str)
    {
        int i;
    
        int n = sizeof(frame_abvs) / sizeof(AbvEntry);
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
        const char *p;
        int frame_width = 0, frame_height = 0;
    
        for(i=0;i<n;i++) {
    
            if (!strcmp(frame_abvs[i].abv, str)) {
                frame_width = frame_abvs[i].width;
                frame_height = frame_abvs[i].height;
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
                break;
            }
        }
        if (i == n) {
            p = str;
            frame_width = strtol(p, (char **)&p, 10);
            if (*p)
                p++;
            frame_height = strtol(p, (char **)&p, 10);
        }
        if (frame_width <= 0 || frame_height <= 0)
            return -1;
        *width_ptr = frame_width;
        *height_ptr = frame_height;
        return 0;
    }
    
    
    int parse_frame_rate(int *frame_rate, int *frame_rate_base, const char *arg)
    {
        int i;
        char* cp;
       
        /* First, we check our abbreviation table */
        for (i = 0; i < sizeof(frame_abvs)/sizeof(*frame_abvs); ++i)
             if (!strcmp(frame_abvs[i].abv, arg)) {
    	     *frame_rate = frame_abvs[i].frame_rate;
    	     *frame_rate_base = frame_abvs[i].frame_rate_base;
    	     return 0;
    	 }
    
        /* Then, we try to parse it as fraction */
        cp = strchr(arg, '/');
        if (cp) {
            char* cpp;
    	*frame_rate = strtol(arg, &cpp, 10);
    	if (cpp != arg || cpp == cp) 
    	    *frame_rate_base = strtol(cp+1, &cpp, 10);
    	else
    	   *frame_rate = 0;
        } 
        else {
            /* Finally we give up and parse it as double */
    
            *frame_rate_base = DEFAULT_FRAME_RATE_BASE; //FIXME use av_d2q()
    
            *frame_rate = (int)(strtod(arg, 0) * (*frame_rate_base) + 0.5);
        }
        if (!*frame_rate || !*frame_rate_base)
            return -1;
        else
            return 0;
    }
    
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
    /* Syntax:
     * - If not a duration:
     *  [{YYYY-MM-DD|YYYYMMDD}]{T| }{HH[:MM[:SS[.m...]]][Z]|HH[MM[SS[.m...]]][Z]}
    
     * Time is localtime unless Z is suffixed to the end. In this case GMT
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
     * Return the date in micro seconds since 1970 
     * - If duration:
     *  HH[:MM[:SS[.m...]]]
     *  S+[.m...]
     */
    
    int64_t parse_date(const char *datestr, int duration)
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
    {
        const char *p;
    
        int64_t t;
    
        int i;
        static const char *date_fmt[] = {
            "%Y-%m-%d",
            "%Y%m%d",
        };
        static const char *time_fmt[] = {
            "%H:%M:%S",
            "%H%M%S",
        };
        const char *q;
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
        int is_utc, len;
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
        len = strlen(datestr);
        if (len > 0)
            lastch = datestr[len - 1];
        else
            lastch = '\0';
    
        is_utc = (lastch == 'z' || lastch == 'Z');
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
    
        p = datestr;
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
        q = NULL;
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
        if (!duration) {
    
            for (i = 0; i < sizeof(date_fmt) / sizeof(date_fmt[0]); i++) {
    
                q = small_strptime(p, date_fmt[i], &dt);
    
                if (q) {
                    break;
                }
            }
    
            if (!q) {
                if (is_utc) {
                    dt = *gmtime(&now);
                } else {
                    dt = *localtime(&now);
                }
                dt.tm_hour = dt.tm_min = dt.tm_sec = 0;
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
            } else {
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
            }
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
            for (i = 0; i < sizeof(time_fmt) / sizeof(time_fmt[0]); i++) {
    
                q = small_strptime(p, time_fmt[i], &dt);
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
                if (q) {
                    break;
                }
            }
        } else {
    
            q = small_strptime(p, time_fmt[0], &dt);
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
            if (!q) {
                dt.tm_sec = strtol(p, (char **)&q, 10);
                dt.tm_min = 0;
                dt.tm_hour = 0;
    
            }
        }
    
        /* Now we have all the fields that we can get */
        if (!q) {
            if (duration)
                return 0;
            else
    
                return now * int64_t_C(1000000);
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
        }
    
            t = dt.tm_hour * 3600 + dt.tm_min * 60 + dt.tm_sec;
    
            dt.tm_isdst = -1;       /* unknown */
            if (is_utc) {
                t = mktimegm(&dt);
            } else {
                t = mktime(&dt);
            }
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
        }
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
            int val, n;
    
            q++;
            for (val = 0, n = 100000; n >= 1; n /= 10, q++) {
                if (!isdigit(*q)) 
                    break;
                val += n * (*q - '0');
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
            }
            t += val;
        }
        return t;
    }