Skip to content
Snippets Groups Projects
utils.c 125 KiB
Newer Older
  • Learn to ignore specific revisions
  •     st->pts.val= pkt->dts;
    
    
            frame_size = get_audio_frame_size(st->codec, pkt->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 && (pkt->size || st->pts.num!=st->pts.den>>1 || st->pts.val)) {
    
                frac_add(&st->pts, (int64_t)st->time_base.den * frame_size);
    
            frac_add(&st->pts, (int64_t)st->time_base.den * st->codec->time_base.num);
    
    }
    
    int av_write_frame(AVFormatContext *s, AVPacket *pkt)
    {
    
        int ret = compute_pkt_fields2(s, s->streams[pkt->stream_index], pkt);
    
        if(ret<0 && !(s->oformat->flags & AVFMT_NOTIMESTAMPS))
    
        ret= s->oformat->write_packet(s, pkt);
    
    
        if (ret >= 0)
            s->streams[pkt->stream_index]->nb_frames++;
    
        return ret;
    
    void ff_interleave_add_packet(AVFormatContext *s, AVPacket *pkt,
                                  int (*compare)(AVFormatContext *, AVPacket *, AVPacket *))
    {
        AVPacketList **next_point, *this_pktl;
    
        this_pktl = av_mallocz(sizeof(AVPacketList));
        this_pktl->pkt= *pkt;
    
        pkt->destruct= NULL;             // do not free original but only the copy
        av_dup_packet(&this_pktl->pkt);  // duplicate the packet if it uses non-alloced memory
    
        if(s->streams[pkt->stream_index]->last_in_packet_buffer){
            next_point = &(s->streams[pkt->stream_index]->last_in_packet_buffer->next);
        }else
    
            next_point = &s->packet_buffer;
    
    
        if(*next_point){
            if(compare(s, &s->packet_buffer_end->pkt, pkt)){
                while(!compare(s, &(*next_point)->pkt, pkt)){
    
                    next_point= &(*next_point)->next;
                }
    
            }else{
                next_point = &(s->packet_buffer_end->next);
    
        assert(!*next_point);
    
        s->packet_buffer_end= this_pktl;
    
        s->streams[pkt->stream_index]->last_in_packet_buffer=
        *next_point= this_pktl;
    
    static int ff_interleave_compare_dts(AVFormatContext *s, AVPacket *next, AVPacket *pkt)
    
    {
        AVStream *st = s->streams[ pkt ->stream_index];
        AVStream *st2= s->streams[ next->stream_index];
    
        int comp = av_compare_ts(next->dts, st2->time_base, pkt->dts,
                                 st->time_base);
    
        if (comp == 0)
    
            return pkt->stream_index < next->stream_index;
    
    int av_interleave_packet_per_dts(AVFormatContext *s, AVPacket *out, AVPacket *pkt, int flush){
    
            ff_interleave_add_packet(s, pkt, ff_interleave_compare_dts);
    
            stream_count+= !!s->streams[i]->last_in_packet_buffer;
    
        if(stream_count && (s->nb_streams == stream_count || flush)){
    
    
            s->packet_buffer= pktl->next;
    
            if(!s->packet_buffer)
                s->packet_buffer_end= NULL;
    
    
            if(s->streams[out->stream_index]->last_in_packet_buffer == pktl)
                s->streams[out->stream_index]->last_in_packet_buffer= NULL;
    
     * Interleave an AVPacket correctly so it can be muxed.
    
     * @param out the interleaved packet will be output here
     * @param in the input packet
     * @param flush 1 if no further packets are available as input and all
     *              remaining packets should be output
    
     * @return 1 if a packet was output, 0 if no packet could be output,
    
     *         < 0 if an error occurred
    
    static int interleave_packet(AVFormatContext *s, AVPacket *out, AVPacket *in, int flush){
    
        if(s->oformat->interleave_packet)
            return s->oformat->interleave_packet(s, out, in, flush);
        else
            return av_interleave_packet_per_dts(s, out, in, flush);
    }
    
    
    int av_interleaved_write_frame(AVFormatContext *s, AVPacket *pkt){
        AVStream *st= s->streams[ pkt->stream_index];
    
        //FIXME/XXX/HACK drop zero sized packets
    
        if(st->codec->codec_type == AVMEDIA_TYPE_AUDIO && pkt->size==0)
    
        av_dlog(s, "av_interleaved_write_frame size:%d dts:%"PRId64" pts:%"PRId64"\n",
                pkt->size, pkt->dts, pkt->pts);
    
        if((ret = compute_pkt_fields2(s, st, pkt)) < 0 && !(s->oformat->flags & AVFMT_NOTIMESTAMPS))
            return ret;
    
        if(pkt->dts == AV_NOPTS_VALUE && !(s->oformat->flags & AVFMT_NOTIMESTAMPS))
    
            int ret= interleave_packet(s, &opkt, pkt, 0);
    
            if(ret<=0) //FIXME cleanup needed for ret<0 ?
                return ret;
    
            if (ret >= 0)
                s->streams[opkt.stream_index]->nb_frames++;
    
            ret= interleave_packet(s, &pkt, NULL, 1);
    
            if (ret >= 0)
                s->streams[pkt.stream_index]->nb_frames++;
    
        if(s->oformat->write_trailer)
            ret = s->oformat->write_trailer(s);
    
        for(i=0;i<s->nb_streams;i++) {
    
            av_freep(&s->streams[i]->priv_data);
    
            av_freep(&s->streams[i]->index_entries);
        }
    
        if (s->iformat && s->iformat->priv_class)
            av_opt_free(s->priv_data);
    
    void ff_program_add_stream_index(AVFormatContext *ac, int progid, unsigned int idx)
    
        if (idx >= ac->nb_streams) {
            av_log(ac, AV_LOG_ERROR, "stream index %d is not valid\n", idx);
            return;
        }
    
    
        for(i=0; i<ac->nb_programs; i++){
            if(ac->programs[i]->id != progid)
                continue;
            program = ac->programs[i];
            for(j=0; j<program->nb_stream_indexes; j++)
                if(program->stream_index[j] == idx)
                    return;
    
            tmp = av_realloc(program->stream_index, sizeof(unsigned int)*(program->nb_stream_indexes+1));
            if(!tmp)
                return;
            program->stream_index = tmp;
            program->stream_index[program->nb_stream_indexes++] = idx;
            return;
        }
    }
    
    
    static void print_fps(double d, const char *postfix){
        uint64_t v= lrintf(d*100);
        if     (v% 100      ) av_log(NULL, AV_LOG_INFO, ", %3.2f %s", d, postfix);
        else if(v%(100*1000)) av_log(NULL, AV_LOG_INFO, ", %1.0f %s", d, postfix);
        else                  av_log(NULL, AV_LOG_INFO, ", %1.0fk %s", d/1000, postfix);
    }
    
    
    static void dump_metadata(void *ctx, AVDictionary *m, const char *indent)
    
        if(m && !(m->count == 1 && av_dict_get(m, "language", NULL, 0))){
            AVDictionaryEntry *tag=NULL;
    
    
            av_log(ctx, AV_LOG_INFO, "%sMetadata:\n", indent);
    
            while((tag=av_dict_get(m, "", tag, AV_DICT_IGNORE_SUFFIX))) {
    
                if(strcmp("language", tag->key))
    
    Michael Niedermayer's avatar
    Michael Niedermayer committed
                    av_log(ctx, AV_LOG_INFO, "%s  %-16s: %s\n", indent, tag->key, tag->value);
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
    /* "user interface" functions */
    
    static void dump_stream_format(AVFormatContext *ic, int i, int index, int is_output)
    
        int flags = (is_output ? ic->oformat->flags : ic->iformat->flags);
        AVStream *st = ic->streams[i];
    
        int g = av_gcd(st->time_base.num, st->time_base.den);
    
        AVDictionaryEntry *lang = av_dict_get(st->metadata, "language", NULL, 0);
    
        avcodec_string(buf, sizeof(buf), st->codec, is_output);
        av_log(NULL, AV_LOG_INFO, "    Stream #%d.%d", index, i);
        /* the pid is an important information, so we display it */
        /* XXX: add a generic system */
        if (flags & AVFMT_SHOW_IDS)
            av_log(NULL, AV_LOG_INFO, "[0x%x]", st->id);
    
        if (lang)
            av_log(NULL, AV_LOG_INFO, "(%s)", lang->value);
    
        av_log(NULL, AV_LOG_DEBUG, ", %d, %d/%d", st->codec_info_nb_frames, st->time_base.num/g, st->time_base.den/g);
    
        if (st->sample_aspect_ratio.num && // default
            av_cmp_q(st->sample_aspect_ratio, st->codec->sample_aspect_ratio)) {
            AVRational display_aspect_ratio;
            av_reduce(&display_aspect_ratio.num, &display_aspect_ratio.den,
                      st->codec->width*st->sample_aspect_ratio.num,
                      st->codec->height*st->sample_aspect_ratio.den,
                      1024*1024);
            av_log(NULL, AV_LOG_INFO, ", PAR %d:%d DAR %d:%d",
                     st->sample_aspect_ratio.num, st->sample_aspect_ratio.den,
                     display_aspect_ratio.num, display_aspect_ratio.den);
        }
    
        if(st->codec->codec_type == AVMEDIA_TYPE_VIDEO){
    
    Michael Niedermayer's avatar
    Michael Niedermayer committed
            if(st->avg_frame_rate.den && st->avg_frame_rate.num)
                print_fps(av_q2d(st->avg_frame_rate), "fps");
    
            if(st->r_frame_rate.den && st->r_frame_rate.num)
    
                print_fps(av_q2d(st->r_frame_rate), "tbr");
    
            if(st->time_base.den && st->time_base.num)
    
                print_fps(1/av_q2d(st->time_base), "tbn");
    
            if(st->codec->time_base.den && st->codec->time_base.num)
    
                print_fps(1/av_q2d(st->codec->time_base), "tbc");
    
        if (st->disposition & AV_DISPOSITION_DEFAULT)
            av_log(NULL, AV_LOG_INFO, " (default)");
        if (st->disposition & AV_DISPOSITION_DUB)
            av_log(NULL, AV_LOG_INFO, " (dub)");
        if (st->disposition & AV_DISPOSITION_ORIGINAL)
            av_log(NULL, AV_LOG_INFO, " (original)");
        if (st->disposition & AV_DISPOSITION_COMMENT)
            av_log(NULL, AV_LOG_INFO, " (comment)");
        if (st->disposition & AV_DISPOSITION_LYRICS)
            av_log(NULL, AV_LOG_INFO, " (lyrics)");
        if (st->disposition & AV_DISPOSITION_KARAOKE)
            av_log(NULL, AV_LOG_INFO, " (karaoke)");
        if (st->disposition & AV_DISPOSITION_FORCED)
            av_log(NULL, AV_LOG_INFO, " (forced)");
        if (st->disposition & AV_DISPOSITION_HEARING_IMPAIRED)
            av_log(NULL, AV_LOG_INFO, " (hearing impaired)");
        if (st->disposition & AV_DISPOSITION_VISUAL_IMPAIRED)
            av_log(NULL, AV_LOG_INFO, " (visual impaired)");
    
        if (st->disposition & AV_DISPOSITION_CLEAN_EFFECTS)
            av_log(NULL, AV_LOG_INFO, " (clean effects)");
    
        dump_metadata(NULL, st->metadata, "    ");
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
    
    
    #if FF_API_DUMP_FORMAT
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
    void dump_format(AVFormatContext *ic,
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
                     const char *url,
                     int is_output)
    
    {
        av_dump_format(ic, index, url, is_output);
    }
    #endif
    
    void av_dump_format(AVFormatContext *ic,
                        int index,
                        const char *url,
                        int is_output)
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
    {
    
        uint8_t *printed = av_mallocz(ic->nb_streams);
        if (ic->nb_streams && !printed)
            return;
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
    
    
        av_log(NULL, AV_LOG_INFO, "%s #%d, %s, %s '%s':\n",
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
                is_output ? "Output" : "Input",
    
                index,
                is_output ? ic->oformat->name : ic->iformat->name,
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
                is_output ? "to" : "from", url);
    
        dump_metadata(NULL, ic->metadata, "  ");
    
            av_log(NULL, AV_LOG_INFO, "  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;
    
                av_log(NULL, AV_LOG_INFO, "%02d:%02d:%02d.%02d", hours, mins, secs,
                       (100 * us) / AV_TIME_BASE);
    
                av_log(NULL, AV_LOG_INFO, "N/A");
    
            if (ic->start_time != AV_NOPTS_VALUE) {
                int secs, us;
    
                av_log(NULL, AV_LOG_INFO, ", start: ");
    
                us = abs(ic->start_time % AV_TIME_BASE);
    
                av_log(NULL, AV_LOG_INFO, "%d.%06d",
    
            av_log(NULL, AV_LOG_INFO, ", bitrate: ");
    
                av_log(NULL, AV_LOG_INFO,"%d kb/s", ic->bit_rate / 1000);
    
                av_log(NULL, AV_LOG_INFO, "N/A");
    
            av_log(NULL, AV_LOG_INFO, "\n");
    
        for (i = 0; i < ic->nb_chapters; i++) {
            AVChapter *ch = ic->chapters[i];
            av_log(NULL, AV_LOG_INFO, "    Chapter #%d.%d: ", index, i);
            av_log(NULL, AV_LOG_INFO, "start %f, ", ch->start * av_q2d(ch->time_base));
            av_log(NULL, AV_LOG_INFO, "end %f\n",   ch->end   * av_q2d(ch->time_base));
    
            dump_metadata(NULL, ch->metadata, "    ");
        }
    
                AVDictionaryEntry *name = av_dict_get(ic->programs[j]->metadata,
    
                                                      "name", NULL, 0);
    
                av_log(NULL, AV_LOG_INFO, "  Program %d %s\n", ic->programs[j]->id,
    
                       name ? name->value : "");
    
                dump_metadata(NULL, ic->programs[j]->metadata, "    ");
    
                for(k=0; k<ic->programs[j]->nb_stream_indexes; k++) {
    
                    dump_stream_format(ic, ic->programs[j]->stream_index[k], index, is_output);
    
                    printed[ic->programs[j]->stream_index[k]] = 1;
                }
                total += ic->programs[j]->nb_stream_indexes;
            }
            if (total < ic->nb_streams)
                av_log(NULL, AV_LOG_INFO, "  No Program\n");
        }
    
        for(i=0;i<ic->nb_streams;i++)
            if (!printed[i])
    
                dump_stream_format(ic, i, index, is_output);
    
    int64_t av_gettime(void)
    {
        struct timeval tv;
        gettimeofday(&tv,NULL);
        return (int64_t)tv.tv_sec * 1000000 + tv.tv_usec;
    }
    
    
    uint64_t ff_ntp_time(void)
    {
      return (av_gettime() / 1000) * 1000 + NTP_OFFSET_US;
    }
    
    
    #if FF_API_PARSE_DATE
    #include "libavutil/parseutils.h"
    
    int64_t parse_date(const char *timestr, int duration)
    {
        int64_t timeval;
        av_parse_time(&timeval, timestr, duration);
        return timeval;
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
    }
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
    
    
    #if FF_API_FIND_INFO_TAG
    #include "libavutil/parseutils.h"
    
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
    int find_info_tag(char *arg, int arg_size, const char *tag1, const char *info)
    {
    
        return av_find_info_tag(arg, arg_size, tag1, info);
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
    }
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
    
    
    int av_get_frame_filename(char *buf, int buf_size,
                              const char *path, int number)
    
    {
        const char *p;
    
    
        q = buf;
        p = path;
        percentd_found = 0;
        for(;;) {
            c = *p++;
            if (c == '\0')
                break;
            if (c == '%') {
    
                do {
                    nd = 0;
                    while (isdigit(*p)) {
                        nd = nd * 10 + *p++ - '0';
                    }
                    c = *p++;
                } while (isdigit(c));
    
    
                switch(c) {
                case '%':
                    goto addchar;
                case 'd':
                    if (percentd_found)
                        goto fail;
                    percentd_found = 1;
                    snprintf(buf1, sizeof(buf1), "%0*d", nd, number);
                    len = strlen(buf1);
                    if ((q - buf + len) > buf_size - 1)
                        goto fail;
                    memcpy(q, buf1, len);
                    q += len;
                    break;
                default:
                    goto fail;
                }
            } else {
            addchar:
                if ((q - buf) < buf_size - 1)
                    *q++ = c;
            }
        }
        if (!percentd_found)
            goto fail;
        *q = '\0';
        return 0;
     fail:
        *q = '\0';
        return -1;
    }
    
    
    static void hex_dump_internal(void *avcl, FILE *f, int level, uint8_t *buf, int size)
    
    #define PRINT(...) do { if (!f) av_log(avcl, level, __VA_ARGS__); else fprintf(f, __VA_ARGS__); } while(0)
    
    #undef PRINT
    }
    
    void av_hex_dump(FILE *f, uint8_t *buf, int size)
    {
        hex_dump_internal(NULL, f, 0, buf, size);
    }
    
    void av_hex_dump_log(void *avcl, int level, uint8_t *buf, int size)
    {
        hex_dump_internal(avcl, NULL, level, buf, size);
    
    static void pkt_dump_internal(void *avcl, FILE *f, int level, AVPacket *pkt, int dump_payload, AVRational time_base)
    
    #define PRINT(...) do { if (!f) av_log(avcl, level, __VA_ARGS__); else fprintf(f, __VA_ARGS__); } while(0)
        PRINT("stream #%d:\n", pkt->stream_index);
    
        PRINT("  keyframe=%d\n", ((pkt->flags & AV_PKT_FLAG_KEY) != 0));
    
        PRINT("  duration=%0.3f\n", pkt->duration * av_q2d(time_base));
    
        /* DTS is _always_ valid after av_read_frame() */
    
        if (pkt->dts == AV_NOPTS_VALUE)
    
            PRINT("%0.3f", pkt->dts * av_q2d(time_base));
    
        /* PTS may not be known if B-frames are present. */
    
        if (pkt->pts == AV_NOPTS_VALUE)
    
            PRINT("%0.3f", pkt->pts * av_q2d(time_base));
    
        PRINT("\n");
        PRINT("  size=%d\n", pkt->size);
    #undef PRINT
    
        if (dump_payload)
            av_hex_dump(f, pkt->data, pkt->size);
    }
    
    
    void av_pkt_dump(FILE *f, AVPacket *pkt, int dump_payload)
    {
    
        AVRational tb = { 1, AV_TIME_BASE };
        pkt_dump_internal(NULL, f, 0, pkt, dump_payload, tb);
    }
    
    
    void av_pkt_dump2(FILE *f, AVPacket *pkt, int dump_payload, AVStream *st)
    {
        pkt_dump_internal(NULL, f, 0, pkt, dump_payload, st->time_base);
    
    void av_pkt_dump_log(void *avcl, int level, AVPacket *pkt, int dump_payload)
    {
    
        AVRational tb = { 1, AV_TIME_BASE };
        pkt_dump_internal(avcl, NULL, level, pkt, dump_payload, tb);
    }
    
    
    void av_pkt_dump_log2(void *avcl, int level, AVPacket *pkt, int dump_payload,
                          AVStream *st)
    {
        pkt_dump_internal(avcl, NULL, level, pkt, dump_payload, st->time_base);
    
    void av_url_split(char *proto, int proto_size,
                      char *authorization, int authorization_size,
                      char *hostname, int hostname_size,
                      int *port_ptr,
                      char *path, int path_size,
                      const char *url)
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
    {
    
    
        if (port_ptr)               *port_ptr = -1;
        if (proto_size > 0)         proto[0] = 0;
        if (authorization_size > 0) authorization[0] = 0;
        if (hostname_size > 0)      hostname[0] = 0;
        if (path_size > 0)          path[0] = 0;
    
        /* parse protocol */
        if ((p = strchr(url, ':'))) {
            av_strlcpy(proto, url, FFMIN(proto_size, p + 1 - url));
            p++; /* skip ':' */
            if (*p == '/') p++;
            if (*p == '/') p++;
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
        } else {
    
            /* no protocol means plain filename */
            av_strlcpy(path, url, path_size);
            return;
        }
    
        /* separate path from hostname */
    
    Michael Niedermayer's avatar
    Michael Niedermayer committed
            av_strlcpy(path, ls, path_size);
    
            ls = &p[strlen(p)]; // XXX
    
        /* the rest is hostname, use that to parse auth/port */
        if (ls != p) {
            /* authorization (user[:pass]@hostname) */
            if ((at = strchr(p, '@')) && at < ls) {
                av_strlcpy(authorization, p,
                           FFMIN(authorization_size, at + 1 - p));
                p = at + 1; /* skip '@' */
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
            }
    
            if (*p == '[' && (brk = strchr(p, ']')) && brk < ls) {
                /* [host]:port */
                av_strlcpy(hostname, p + 1,
                           FFMIN(hostname_size, brk - p));
                if (brk[1] == ':' && port_ptr)
                    *port_ptr = atoi(brk + 2);
            } else if ((col = strchr(p, ':')) && col < ls) {
                av_strlcpy(hostname, p,
                           FFMIN(col + 1 - p, hostname_size));
                if (port_ptr) *port_ptr = atoi(col + 1);
            } else
                av_strlcpy(hostname, p,
                           FFMIN(ls + 1 - p, hostname_size));
    
    char *ff_data_to_hex(char *buff, const uint8_t *src, int s, int lowercase)
    
        static const char hex_table_uc[16] = { '0', '1', '2', '3',
    
    Martin Storsjö's avatar
    Martin Storsjö committed
                                               '4', '5', '6', '7',
                                               '8', '9', 'A', 'B',
                                               'C', 'D', 'E', 'F' };
    
        static const char hex_table_lc[16] = { '0', '1', '2', '3',
                                               '4', '5', '6', '7',
                                               '8', '9', 'a', 'b',
                                               'c', 'd', 'e', 'f' };
        const char *hex_table = lowercase ? hex_table_lc : hex_table_uc;
    
            buff[i * 2]     = hex_table[src[i] >> 4];
            buff[i * 2 + 1] = hex_table[src[i] & 0xF];
    
    int ff_hex_to_data(uint8_t *data, const char *p)
    {
        int c, len, v;
    
        len = 0;
        v = 1;
        for (;;) {
            p += strspn(p, SPACE_CHARS);
            if (*p == '\0')
                break;
            c = toupper((unsigned char) *p++);
            if (c >= '0' && c <= '9')
                c = c - '0';
            else if (c >= 'A' && c <= 'F')
                c = c - 'A' + 10;
            else
                break;
            v = (v << 4) | c;
            if (v & 0x100) {
                if (data)
                    data[len] = v;
                len++;
                v = 1;
            }
        }
        return len;
    }
    
    
    void av_set_pts_info(AVStream *s, int pts_wrap_bits,
    
                         unsigned int pts_num, unsigned int pts_den)
    
        AVRational new_tb;
        if(av_reduce(&new_tb.num, &new_tb.den, pts_num, pts_den, INT_MAX)){
            if(new_tb.num != pts_num)
                av_log(NULL, AV_LOG_DEBUG, "st:%d removing common factor %d from timebase\n", s->index, pts_num/new_tb.num);
    
        }else
            av_log(NULL, AV_LOG_WARNING, "st:%d has too large timebase, reducing\n", s->index);
    
        if(new_tb.num <= 0 || new_tb.den <= 0) {
            av_log(NULL, AV_LOG_ERROR, "Ignoring attempt to set invalid timebase for st:%d\n", s->index);
            return;
        }
        s->time_base = new_tb;
        s->pts_wrap_bits = pts_wrap_bits;
    
    
    int ff_url_join(char *str, int size, const char *proto,
                    const char *authorization, const char *hostname,
                    int port, const char *fmt, ...)
    {
    #if CONFIG_NETWORK
        struct addrinfo hints, *ai;
    #endif
    
        str[0] = '\0';
        if (proto)
            av_strlcatf(str, size, "%s://", proto);
    
        if (authorization && authorization[0])
    
            av_strlcatf(str, size, "%s@", authorization);
    #if CONFIG_NETWORK && defined(AF_INET6)
        /* Determine if hostname is a numerical IPv6 address,
         * properly escape it within [] in that case. */
        memset(&hints, 0, sizeof(hints));
        hints.ai_flags = AI_NUMERICHOST;
        if (!getaddrinfo(hostname, NULL, &hints, &ai)) {
            if (ai->ai_family == AF_INET6) {
                av_strlcat(str, "[", size);
                av_strlcat(str, hostname, size);
                av_strlcat(str, "]", size);
            } else {
                av_strlcat(str, hostname, size);
            }
            freeaddrinfo(ai);
        } else
    #endif
            /* Not an IPv6 address, just output the plain string. */
            av_strlcat(str, hostname, size);
    
        if (port >= 0)
            av_strlcatf(str, size, ":%d", port);
        if (fmt) {
            va_list vl;
            int len = strlen(str);
    
            va_start(vl, fmt);
            vsnprintf(str + len, size > len ? size - len : 0, fmt, vl);
            va_end(vl);
        }
        return strlen(str);
    }
    
    
    int ff_write_chained(AVFormatContext *dst, int dst_stream, AVPacket *pkt,
                         AVFormatContext *src)
    {
        AVPacket local_pkt;
    
        local_pkt = *pkt;
        local_pkt.stream_index = dst_stream;
        if (pkt->pts != AV_NOPTS_VALUE)
            local_pkt.pts = av_rescale_q(pkt->pts,
                                         src->streams[pkt->stream_index]->time_base,
                                         dst->streams[dst_stream]->time_base);
        if (pkt->dts != AV_NOPTS_VALUE)
            local_pkt.dts = av_rescale_q(pkt->dts,
                                         src->streams[pkt->stream_index]->time_base,
                                         dst->streams[dst_stream]->time_base);
        return av_write_frame(dst, &local_pkt);
    }
    
    
    void ff_parse_key_value(const char *str, ff_parse_key_val_cb callback_get_buf,
                            void *context)
    {
        const char *ptr = str;
    
        /* Parse key=value pairs. */
        for (;;) {
            const char *key;
            char *dest = NULL, *dest_end;
            int key_len, dest_len = 0;
    
            /* Skip whitespace and potential commas. */
            while (*ptr && (isspace(*ptr) || *ptr == ','))
                ptr++;
            if (!*ptr)
                break;
    
            key = ptr;
    
            if (!(ptr = strchr(key, '=')))
                break;
            ptr++;
            key_len = ptr - key;
    
            callback_get_buf(context, key, key_len, &dest, &dest_len);
            dest_end = dest + dest_len - 1;
    
            if (*ptr == '\"') {
                ptr++;
                while (*ptr && *ptr != '\"') {
                    if (*ptr == '\\') {
                        if (!ptr[1])
                            break;
                        if (dest && dest < dest_end)
                            *dest++ = ptr[1];
                        ptr += 2;
                    } else {
                        if (dest && dest < dest_end)
                            *dest++ = *ptr;
                        ptr++;
                    }
                }
                if (*ptr == '\"')
                    ptr++;
            } else {
                for (; *ptr && !(isspace(*ptr) || *ptr == ','); ptr++)
                    if (dest && dest < dest_end)
                        *dest++ = *ptr;
            }
            if (dest)
                *dest = 0;
        }
    }
    
    
    Peter Ross's avatar
    Peter Ross committed
    int ff_find_stream_index(AVFormatContext *s, int id)
    {
        int i;
        for (i = 0; i < s->nb_streams; i++) {
            if (s->streams[i]->id == id)
                return i;
        }
        return -1;
    }
    
    
    void ff_make_absolute_url(char *buf, int size, const char *base,
                              const char *rel)
    {
        char *sep;
        /* Absolute path, relative to the current server */
        if (base && strstr(base, "://") && rel[0] == '/') {
            if (base != buf)
                av_strlcpy(buf, base, size);
            sep = strstr(buf, "://");
            if (sep) {
                sep += 3;
                sep = strchr(sep, '/');
                if (sep)
                    *sep = '\0';
            }
            av_strlcat(buf, rel, size);
            return;
        }
        /* If rel actually is an absolute url, just copy it */
        if (!base || strstr(rel, "://") || rel[0] == '/') {
            av_strlcpy(buf, rel, size);
            return;
        }
        if (base != buf)
            av_strlcpy(buf, base, size);
        /* Remove the file name from the base url */
        sep = strrchr(buf, '/');
        if (sep)
            sep[1] = '\0';
        else
            buf[0] = '\0';
        while (av_strstart(rel, "../", NULL) && sep) {
            /* Remove the path delimiter at the end */
            sep[0] = '\0';
            sep = strrchr(buf, '/');
            /* If the next directory name to pop off is "..", break here */
            if (!strcmp(sep ? &sep[1] : buf, "..")) {
                /* Readd the slash we just removed */
                av_strlcat(buf, "/", size);
                break;
            }
            /* Cut off the directory name */
            if (sep)
                sep[1] = '\0';
            else
                buf[0] = '\0';
            rel += 3;
        }
        av_strlcat(buf, rel, size);
    }
    
    
    int64_t ff_iso8601_to_unix_time(const char *datestr)
    {
    
    #if HAVE_STRPTIME
    
        struct tm time = {0};
        strptime(datestr, "%Y - %m - %dT%T", &time);
        return mktime(&time);
    
    #else
        av_log(NULL, AV_LOG_WARNING, "strptime() unavailable on this system, cannot convert "
                                     "the date string.\n");
        return 0;
    #endif