Skip to content
Snippets Groups Projects
ffserver.c 130 KiB
Newer Older
  • Learn to ignore specific revisions
  • Fabrice Bellard's avatar
    Fabrice Bellard committed
        for(i=0;i<feed->nb_streams;i++) {
            st = feed->streams[i];
            av1 = &st->codec;
    
            if (av1->codec_id == av->codec_id &&
                av1->codec_type == av->codec_type &&
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
                av1->bit_rate == av->bit_rate) {
    
                switch(av->codec_type) {
                case CODEC_TYPE_AUDIO:
                    if (av1->channels == av->channels &&
                        av1->sample_rate == av->sample_rate)
                        goto found;
                    break;
                case CODEC_TYPE_VIDEO:
                    if (av1->width == av->width &&
                        av1->height == av->height &&
                        av1->frame_rate == av->frame_rate &&
                        av1->gop_size == av->gop_size)
                        goto found;
                    break;
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
        if (!fst)
            return -1;
        return feed->nb_streams - 1;
     found:
        return i;
    }
    
    
    void remove_stream(FFStream *stream)
    {
        FFStream **ps;
        ps = &first_stream;
        while (*ps != NULL) {
            if (*ps == stream) {
                *ps = (*ps)->next;
            } else {
                ps = &(*ps)->next;
            }
        }
    }
    
    /* compute the needed AVStream for each file */
    void build_file_streams(void)
    {
        FFStream *stream, *stream_next;
        AVFormatContext *infile;
        int i;
    
        /* gather all streams */
        for(stream = first_stream; stream != NULL; stream = stream_next) {
            stream_next = stream->next;
            if (stream->stream_type == STREAM_TYPE_LIVE &&
                !stream->feed) {
                /* the stream comes from a file */
                /* try to open the file */
                /* open stream */
                if (av_open_input_file(&infile, stream->feed_filename, 
                                       NULL, 0, NULL) < 0) {
                    http_log("%s not found", stream->feed_filename);
                    /* remove stream (no need to spend more time on it) */
                fail:
                    remove_stream(stream);
                } else {
                    /* find all the AVStreams inside and reference them in
                       'stream' */
                    if (av_find_stream_info(infile) < 0) {
                        http_log("Could not find codec parameters from '%s'", 
                                 stream->feed_filename);
                        av_close_input_file(infile);
                        goto fail;
                    }
                    for(i=0;i<infile->nb_streams;i++) {
                        add_av_stream1(stream, &infile->streams[i]->codec);
                    }
                    av_close_input_file(infile);
                }
            }
        }
    }
    
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
    /* compute the needed AVStream for each feed */
    void build_feed_streams(void)
    {
        FFStream *stream, *feed;
        int i;
    
        /* gather all streams */
        for(stream = first_stream; stream != NULL; stream = stream->next) {
            feed = stream->feed;
            if (feed) {
                if (!stream->is_feed) {
    
                    /* we handle a stream coming from a feed */
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
                    for(i=0;i<stream->nb_streams;i++) {
                        stream->feed_streams[i] = add_av_stream(feed, stream->streams[i]);
                    }
    
                }
            }
        }
    
        /* gather all streams */
        for(stream = first_stream; stream != NULL; stream = stream->next) {
            feed = stream->feed;
            if (feed) {
                if (stream->is_feed) {
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
                    for(i=0;i<stream->nb_streams;i++) {
                        stream->feed_streams[i] = i;
                    }
                }
            }
        }
    
        /* create feed files if needed */
        for(feed = first_feed; feed != NULL; feed = feed->next_feed) {
            int fd;
    
    
            if (url_exist(feed->feed_filename)) {
                /* See if it matches */
                AVFormatContext *s;
                int matches = 0;
    
                if (av_open_input_file(&s, feed->feed_filename, NULL, FFM_PACKET_SIZE, NULL) >= 0) {
                    /* Now see if it matches */
                    if (s->nb_streams == feed->nb_streams) {
                        matches = 1;
                        for(i=0;i<s->nb_streams;i++) {
                            AVStream *sf, *ss;
                            sf = feed->streams[i];
                            ss = s->streams[i];
    
                            if (sf->index != ss->index ||
                                sf->id != ss->id) {
                                printf("Index & Id do not match for stream %d\n", i);
                                matches = 0;
                            } else {
                                AVCodecContext *ccf, *ccs;
    
                                ccf = &sf->codec;
                                ccs = &ss->codec;
    #define CHECK_CODEC(x)  (ccf->x != ccs->x)
    
                                if (CHECK_CODEC(codec) || CHECK_CODEC(codec_type)) {
                                    printf("Codecs do not match for stream %d\n", i);
                                    matches = 0;
                                } else if (CHECK_CODEC(bit_rate) || CHECK_CODEC(flags)) {
                                    printf("Codec bitrates do not match for stream %d\n", i);
                                    matches = 0;
                                } else if (ccf->codec_type == CODEC_TYPE_VIDEO) {
                                    if (CHECK_CODEC(frame_rate) ||
                                        CHECK_CODEC(width) ||
                                        CHECK_CODEC(height)) {
                                        printf("Codec width, height and framerate do not match for stream %d\n", i);
                                        matches = 0;
                                    }
                                } else if (ccf->codec_type == CODEC_TYPE_AUDIO) {
                                    if (CHECK_CODEC(sample_rate) ||
                                        CHECK_CODEC(channels) ||
                                        CHECK_CODEC(frame_size)) {
                                        printf("Codec sample_rate, channels, frame_size do not match for stream %d\n", i);
                                        matches = 0;
                                    }
                                } else {
                                    printf("Unknown codec type\n");
                                    matches = 0;
                                }
                            }
                            if (!matches) {
                                break;
                            }
                        }
                    } else {
                        printf("Deleting feed file '%s' as stream counts differ (%d != %d)\n",
                            feed->feed_filename, s->nb_streams, feed->nb_streams);
                    }
    
                    av_close_input_file(s);
                } else {
                    printf("Deleting feed file '%s' as it appears to be corrupt\n",
                            feed->feed_filename);
                }
                if (!matches)
                    unlink(feed->feed_filename);
            }
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
            if (!url_exist(feed->feed_filename)) {
                AVFormatContext s1, *s = &s1;
    
                /* only write the header of the ffm file */
                if (url_fopen(&s->pb, feed->feed_filename, URL_WRONLY) < 0) {
                    fprintf(stderr, "Could not open output feed file '%s'\n",
                            feed->feed_filename);
                    exit(1);
                }
    
                s->oformat = feed->fmt;
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
                s->nb_streams = feed->nb_streams;
                for(i=0;i<s->nb_streams;i++) {
                    AVStream *st;
                    st = feed->streams[i];
                    s->streams[i] = st;
                }
    
                av_write_header(s);
                /* XXX: need better api */
                av_freep(&s->priv_data);
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
                url_fclose(&s->pb);
            }
            /* get feed size and write index */
            fd = open(feed->feed_filename, O_RDONLY);
            if (fd < 0) {
                fprintf(stderr, "Could not open output feed file '%s'\n",
                        feed->feed_filename);
                exit(1);
            }
    
            feed->feed_write_index = ffm_read_write_index(fd);
            feed->feed_size = lseek(fd, 0, SEEK_END);
            /* ensure that we do not wrap before the end of file */
            if (feed->feed_max_size < feed->feed_size)
                feed->feed_max_size = feed->feed_size;
    
            close(fd);
        }
    }
    
    static void get_arg(char *buf, int buf_size, const char **pp)
    {
        const char *p;
        char *q;
        int quote;
    
        p = *pp;
        while (isspace(*p)) p++;
        q = buf;
        quote = 0;
        if (*p == '\"' || *p == '\'')
            quote = *p++;
        for(;;) {
            if (quote) {
                if (*p == quote)
                    break;
            } else {
                if (isspace(*p))
                    break;
            }
            if (*p == '\0')
                break;
            if ((q - buf) < buf_size - 1)
                *q++ = *p;
            p++;
        }
        *q = '\0';
        if (quote && *p == quote)
            p++;
        *pp = p;
    }
    
    /* add a codec and set the default parameters */
    void add_codec(FFStream *stream, AVCodecContext *av)
    {
        AVStream *st;
    
        /* compute default parameters */
        switch(av->codec_type) {
        case CODEC_TYPE_AUDIO:
            if (av->bit_rate == 0)
                av->bit_rate = 64000;
            if (av->sample_rate == 0)
                av->sample_rate = 22050;
            if (av->channels == 0)
                av->channels = 1;
            break;
        case CODEC_TYPE_VIDEO:
            if (av->bit_rate == 0)
                av->bit_rate = 64000;
            if (av->frame_rate == 0)
                av->frame_rate = 5 * FRAME_RATE_BASE;
            if (av->width == 0 || av->height == 0) {
                av->width = 160;
                av->height = 128;
            }
    
            /* Bitrate tolerance is less for streaming */
    
            if (av->bit_rate_tolerance == 0)
                av->bit_rate_tolerance = av->bit_rate / 4;
            if (av->qmin == 0)
                av->qmin = 3;
            if (av->qmax == 0)
                av->qmax = 31;
            if (av->max_qdiff == 0)
                av->max_qdiff = 3;
    
            av->qcompress = 0.5;
            av->qblur = 0.5;
    
            if (!av->rc_eq)
                av->rc_eq = "tex^qComp";
            if (!av->i_quant_factor)
                av->i_quant_factor = 0.8;
            if (!av->b_quant_factor)
                av->b_quant_factor = 1.25;
            if (!av->b_quant_offset)
                av->b_quant_offset = 1.25;
                
    
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
            break;
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
        }
    
        st = av_mallocz(sizeof(AVStream));
        if (!st)
            return;
        stream->streams[stream->nb_streams++] = st;
        memcpy(&st->codec, av, sizeof(AVCodecContext));
    }
    
    
    int opt_audio_codec(const char *arg)
    {
        AVCodec *p;
    
        p = first_avcodec;
        while (p) {
            if (!strcmp(p->name, arg) && p->type == CODEC_TYPE_AUDIO)
                break;
            p = p->next;
        }
        if (p == NULL) {
            return CODEC_ID_NONE;
        }
    
        return p->id;
    }
    
    int opt_video_codec(const char *arg)
    {
        AVCodec *p;
    
        p = first_avcodec;
        while (p) {
            if (!strcmp(p->name, arg) && p->type == CODEC_TYPE_VIDEO)
                break;
            p = p->next;
        }
        if (p == NULL) {
            return CODEC_ID_NONE;
        }
    
        return p->id;
    }
    
    
    /* simplistic plugin support */
    
    void load_module(const char *filename)
    {
        void *dll;
        void (*init_func)(void);
        dll = dlopen(filename, RTLD_NOW);
        if (!dll) {
            fprintf(stderr, "Could not load module '%s' - %s\n",
                    filename, dlerror());
            return;
        }
        
        init_func = dlsym(dll, "ffserver_module_init");
        if (!init_func) {
            fprintf(stderr, 
                    "%s: init function 'ffserver_module_init()' not found\n",
                    filename);
            dlclose(dll);
        }
    
        init_func();
    }
    
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
    int parse_ffconfig(const char *filename)
    {
        FILE *f;
        char line[1024];
        char cmd[64];
        char arg[1024];
        const char *p;
        int val, errors, line_num;
    
        FFStream **last_stream, *stream, *redirect;
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
        FFStream **last_feed, *feed;
        AVCodecContext audio_enc, video_enc;
        int audio_id, video_id;
    
        f = fopen(filename, "r");
        if (!f) {
            perror(filename);
            return -1;
        }
        
        errors = 0;
        line_num = 0;
        first_stream = NULL;
        last_stream = &first_stream;
        first_feed = NULL;
        last_feed = &first_feed;
        stream = NULL;
        feed = NULL;
    
        redirect = NULL;
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
        audio_id = CODEC_ID_NONE;
        video_id = CODEC_ID_NONE;
        for(;;) {
            if (fgets(line, sizeof(line), f) == NULL)
                break;
            line_num++;
            p = line;
            while (isspace(*p)) 
                p++;
            if (*p == '\0' || *p == '#')
                continue;
    
            get_arg(cmd, sizeof(cmd), &p);
            
            if (!strcasecmp(cmd, "Port")) {
                get_arg(arg, sizeof(arg), &p);
    
                my_http_addr.sin_port = htons (atoi(arg));
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
            } else if (!strcasecmp(cmd, "BindAddress")) {
                get_arg(arg, sizeof(arg), &p);
    
                if (!inet_aton(arg, &my_http_addr.sin_addr)) {
                    fprintf(stderr, "%s:%d: Invalid IP address: %s\n", 
                            filename, line_num, arg);
                    errors++;
                }
            } else if (!strcasecmp(cmd, "NoDaemon")) {
                ffserver_daemon = 0;
            } else if (!strcasecmp(cmd, "RTSPPort")) {
                get_arg(arg, sizeof(arg), &p);
                my_rtsp_addr.sin_port = htons (atoi(arg));
            } else if (!strcasecmp(cmd, "RTSPBindAddress")) {
                get_arg(arg, sizeof(arg), &p);
                if (!inet_aton(arg, &my_rtsp_addr.sin_addr)) {
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
                    fprintf(stderr, "%s:%d: Invalid IP address: %s\n", 
                            filename, line_num, arg);
                    errors++;
                }
            } else if (!strcasecmp(cmd, "MaxClients")) {
                get_arg(arg, sizeof(arg), &p);
                val = atoi(arg);
                if (val < 1 || val > HTTP_MAX_CONNECTIONS) {
                    fprintf(stderr, "%s:%d: Invalid MaxClients: %s\n", 
                            filename, line_num, arg);
                    errors++;
                } else {
                    nb_max_connections = val;
                }
    
            } else if (!strcasecmp(cmd, "MaxBandwidth")) {
                get_arg(arg, sizeof(arg), &p);
                val = atoi(arg);
                if (val < 10 || val > 100000) {
                    fprintf(stderr, "%s:%d: Invalid MaxBandwidth: %s\n", 
                            filename, line_num, arg);
                    errors++;
                } else {
                    nb_max_bandwidth = val;
                }
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
            } else if (!strcasecmp(cmd, "CustomLog")) {
                get_arg(logfilename, sizeof(logfilename), &p);
            } else if (!strcasecmp(cmd, "<Feed")) {
                /*********************************************/
                /* Feed related options */
                char *q;
                if (stream || feed) {
                    fprintf(stderr, "%s:%d: Already in a tag\n",
                            filename, line_num);
                } else {
                    feed = av_mallocz(sizeof(FFStream));
                    /* add in stream list */
                    *last_stream = feed;
                    last_stream = &feed->next;
                    /* add in feed list */
                    *last_feed = feed;
                    last_feed = &feed->next_feed;
                    
                    get_arg(feed->filename, sizeof(feed->filename), &p);
                    q = strrchr(feed->filename, '>');
                    if (*q)
                        *q = '\0';
                    feed->fmt = guess_format("ffm", NULL, NULL);
                    /* defaut feed file */
                    snprintf(feed->feed_filename, sizeof(feed->feed_filename),
                             "/tmp/%s.ffm", feed->filename);
                    feed->feed_max_size = 5 * 1024 * 1024;
                    feed->is_feed = 1;
                    feed->feed = feed; /* self feeding :-) */
                }
    
            } else if (!strcasecmp(cmd, "Launch")) {
                if (feed) {
                    int i;
    
                    feed->child_argv = (char **) av_mallocz(64 * sizeof(char *));
    
                    feed->child_argv[0] = av_malloc(7);
                    strcpy(feed->child_argv[0], "ffmpeg");
    
                    for (i = 1; i < 62; i++) {
                        char argbuf[256];
    
                        get_arg(argbuf, sizeof(argbuf), &p);
                        if (!argbuf[0])
                            break;
    
                        feed->child_argv[i] = av_malloc(strlen(argbuf + 1));
                        strcpy(feed->child_argv[i], argbuf);
                    }
    
                    feed->child_argv[i] = av_malloc(30 + strlen(feed->filename));
    
                    snprintf(feed->child_argv[i], 256, "http://127.0.0.1:%d/%s", 
    
                        ntohs(my_http_addr.sin_port), feed->filename);
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
            } else if (!strcasecmp(cmd, "File")) {
                if (feed) {
                    get_arg(feed->feed_filename, sizeof(feed->feed_filename), &p);
                } else if (stream) {
                    get_arg(stream->feed_filename, sizeof(stream->feed_filename), &p);
                }
            } else if (!strcasecmp(cmd, "FileMaxSize")) {
                if (feed) {
                    const char *p1;
                    double fsize;
    
                    get_arg(arg, sizeof(arg), &p);
                    p1 = arg;
                    fsize = strtod(p1, (char **)&p1);
                    switch(toupper(*p1)) {
                    case 'K':
                        fsize *= 1024;
                        break;
                    case 'M':
                        fsize *= 1024 * 1024;
                        break;
                    case 'G':
                        fsize *= 1024 * 1024 * 1024;
                        break;
                    }
                    feed->feed_max_size = (INT64)fsize;
                }
            } else if (!strcasecmp(cmd, "</Feed>")) {
                if (!feed) {
                    fprintf(stderr, "%s:%d: No corresponding <Feed> for </Feed>\n",
                            filename, line_num);
                    errors++;
    
                } else {
                    /* Make sure that we start out clean */
    
                    if (unlink(feed->feed_filename) < 0 
                        && errno != ENOENT) {
                        fprintf(stderr, "%s:%d: Unable to clean old feed file '%s': %s\n",
                            filename, line_num, feed->feed_filename, strerror(errno));
                        errors++;
                    }
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
                }
                feed = NULL;
            } else if (!strcasecmp(cmd, "<Stream")) {
                /*********************************************/
                /* Stream related options */
                char *q;
                if (stream || feed) {
                    fprintf(stderr, "%s:%d: Already in a tag\n",
                            filename, line_num);
                } else {
                    stream = av_mallocz(sizeof(FFStream));
                    *last_stream = stream;
                    last_stream = &stream->next;
    
                    get_arg(stream->filename, sizeof(stream->filename), &p);
                    q = strrchr(stream->filename, '>');
                    if (*q)
                        *q = '\0';
    
                    stream->fmt = guess_stream_format(NULL, stream->filename, NULL);
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
                    memset(&audio_enc, 0, sizeof(AVCodecContext));
                    memset(&video_enc, 0, sizeof(AVCodecContext));
                    audio_id = CODEC_ID_NONE;
                    video_id = CODEC_ID_NONE;
                    if (stream->fmt) {
                        audio_id = stream->fmt->audio_codec;
                        video_id = stream->fmt->video_codec;
                    }
                }
            } else if (!strcasecmp(cmd, "Feed")) {
                get_arg(arg, sizeof(arg), &p);
                if (stream) {
                    FFStream *sfeed;
                    
                    sfeed = first_feed;
                    while (sfeed != NULL) {
                        if (!strcmp(sfeed->filename, arg))
                            break;
                        sfeed = sfeed->next_feed;
                    }
                    if (!sfeed) {
                        fprintf(stderr, "%s:%d: feed '%s' not defined\n",
                                filename, line_num, arg);
                    } else {
                        stream->feed = sfeed;
                    }
                }
            } else if (!strcasecmp(cmd, "Format")) {
                get_arg(arg, sizeof(arg), &p);
                if (!strcmp(arg, "status")) {
                    stream->stream_type = STREAM_TYPE_STATUS;
                    stream->fmt = NULL;
                } else {
                    stream->stream_type = STREAM_TYPE_LIVE;
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
                    /* jpeg cannot be used here, so use single frame jpeg */
                    if (!strcmp(arg, "jpeg"))
                        strcpy(arg, "singlejpeg");
    
                    stream->fmt = guess_stream_format(arg, NULL, NULL);
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
                    if (!stream->fmt) {
                        fprintf(stderr, "%s:%d: Unknown Format: %s\n", 
                                filename, line_num, arg);
                        errors++;
                    }
                }
                if (stream->fmt) {
                    audio_id = stream->fmt->audio_codec;
                    video_id = stream->fmt->video_codec;
                }
    
            } else if (!strcasecmp(cmd, "FaviconURL")) {
                if (stream && stream->stream_type == STREAM_TYPE_STATUS) {
                    get_arg(stream->feed_filename, sizeof(stream->feed_filename), &p);
                } else {
                    fprintf(stderr, "%s:%d: FaviconURL only permitted for status streams\n", 
                                filename, line_num);
                    errors++;
                }
    
            } else if (!strcasecmp(cmd, "Author")) {
                if (stream) {
                    get_arg(stream->author, sizeof(stream->author), &p);
                }
            } else if (!strcasecmp(cmd, "Comment")) {
                if (stream) {
                    get_arg(stream->comment, sizeof(stream->comment), &p);
                }
            } else if (!strcasecmp(cmd, "Copyright")) {
                if (stream) {
                    get_arg(stream->copyright, sizeof(stream->copyright), &p);
                }
            } else if (!strcasecmp(cmd, "Title")) {
                if (stream) {
                    get_arg(stream->title, sizeof(stream->title), &p);
                }
    
            } else if (!strcasecmp(cmd, "Preroll")) {
                get_arg(arg, sizeof(arg), &p);
                if (stream) {
    
                    stream->prebuffer = atof(arg) * 1000;
    
            } else if (!strcasecmp(cmd, "StartSendOnKey")) {
                if (stream) {
                    stream->send_on_key = 1;
                }
    
            } else if (!strcasecmp(cmd, "AudioCodec")) {
                get_arg(arg, sizeof(arg), &p);
                audio_id = opt_audio_codec(arg);
                if (audio_id == CODEC_ID_NONE) {
                    fprintf(stderr, "%s:%d: Unknown AudioCodec: %s\n", 
                            filename, line_num, arg);
                    errors++;
                }
            } else if (!strcasecmp(cmd, "VideoCodec")) {
                get_arg(arg, sizeof(arg), &p);
                video_id = opt_video_codec(arg);
                if (video_id == CODEC_ID_NONE) {
                    fprintf(stderr, "%s:%d: Unknown VideoCodec: %s\n", 
                            filename, line_num, arg);
                    errors++;
                }
    
            } else if (!strcasecmp(cmd, "MaxTime")) {
                get_arg(arg, sizeof(arg), &p);
                if (stream) {
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
            } else if (!strcasecmp(cmd, "AudioBitRate")) {
                get_arg(arg, sizeof(arg), &p);
                if (stream) {
                    audio_enc.bit_rate = atoi(arg) * 1000;
                }
            } else if (!strcasecmp(cmd, "AudioChannels")) {
                get_arg(arg, sizeof(arg), &p);
                if (stream) {
                    audio_enc.channels = atoi(arg);
                }
            } else if (!strcasecmp(cmd, "AudioSampleRate")) {
                get_arg(arg, sizeof(arg), &p);
                if (stream) {
                    audio_enc.sample_rate = atoi(arg);
                }
    
    	} else if (!strcasecmp(cmd, "AudioQuality")) {
    	    get_arg(arg, sizeof(arg), &p);
                if (stream) {
                    audio_enc.quality = atof(arg) * 1000;
                }
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
            } else if (!strcasecmp(cmd, "VideoBitRate")) {
                get_arg(arg, sizeof(arg), &p);
                if (stream) {
                    video_enc.bit_rate = atoi(arg) * 1000;
                }
            } else if (!strcasecmp(cmd, "VideoSize")) {
                get_arg(arg, sizeof(arg), &p);
                if (stream) {
                    parse_image_size(&video_enc.width, &video_enc.height, arg);
                    if ((video_enc.width % 16) != 0 ||
                        (video_enc.height % 16) != 0) {
                        fprintf(stderr, "%s:%d: Image size must be a multiple of 16\n",
                                filename, line_num);
                        errors++;
                    }
                }
            } else if (!strcasecmp(cmd, "VideoFrameRate")) {
                get_arg(arg, sizeof(arg), &p);
                if (stream) {
                    video_enc.frame_rate = (int)(strtod(arg, NULL) * FRAME_RATE_BASE);
                }
            } else if (!strcasecmp(cmd, "VideoGopSize")) {
                get_arg(arg, sizeof(arg), &p);
                if (stream) {
                    video_enc.gop_size = atoi(arg);
                }
            } else if (!strcasecmp(cmd, "VideoIntraOnly")) {
                if (stream) {
                    video_enc.gop_size = 1;
                }
    
            } else if (!strcasecmp(cmd, "VideoHighQuality")) {
                if (stream) {
                    video_enc.flags |= CODEC_FLAG_HQ;
                }
    
            } else if (!strcasecmp(cmd, "VideoQDiff")) {
    
                if (stream) {
                    video_enc.max_qdiff = atoi(arg);
                    if (video_enc.max_qdiff < 1 || video_enc.max_qdiff > 31) {
                        fprintf(stderr, "%s:%d: VideoQDiff out of range\n",
                                filename, line_num);
                        errors++;
                    }
                }
            } else if (!strcasecmp(cmd, "VideoQMax")) {
    
                if (stream) {
                    video_enc.qmax = atoi(arg);
                    if (video_enc.qmax < 1 || video_enc.qmax > 31) {
                        fprintf(stderr, "%s:%d: VideoQMax out of range\n",
                                filename, line_num);
                        errors++;
                    }
                }
            } else if (!strcasecmp(cmd, "VideoQMin")) {
    
                if (stream) {
                    video_enc.qmin = atoi(arg);
                    if (video_enc.qmin < 1 || video_enc.qmin > 31) {
                        fprintf(stderr, "%s:%d: VideoQMin out of range\n",
                                filename, line_num);
                        errors++;
                    }
                }
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
            } else if (!strcasecmp(cmd, "NoVideo")) {
                video_id = CODEC_ID_NONE;
            } else if (!strcasecmp(cmd, "NoAudio")) {
                audio_id = CODEC_ID_NONE;
    
            } else if (!strcasecmp(cmd, "ACL")) {
                IPAddressACL acl;
                struct hostent *he;
    
                get_arg(arg, sizeof(arg), &p);
                if (strcasecmp(arg, "allow") == 0) {
                    acl.action = IP_ALLOW;
                } else if (strcasecmp(arg, "deny") == 0) {
                    acl.action = IP_DENY;
                } else {
                    fprintf(stderr, "%s:%d: ACL action '%s' is not ALLOW or DENY\n",
                            filename, line_num, arg);
                    errors++;
                }
    
                get_arg(arg, sizeof(arg), &p);
    
                he = gethostbyname(arg);
                if (!he) {
                    fprintf(stderr, "%s:%d: ACL refers to invalid host or ip address '%s'\n",
                            filename, line_num, arg);
                    errors++;
                } else {
                    /* Only take the first */
                    acl.first = *(struct in_addr *) he->h_addr_list[0];
                    acl.last = acl.first;
                }
    
                get_arg(arg, sizeof(arg), &p);
    
                if (arg[0]) {
                    he = gethostbyname(arg);
                    if (!he) {
                        fprintf(stderr, "%s:%d: ACL refers to invalid host or ip address '%s'\n",
                                filename, line_num, arg);
                        errors++;
                    } else {
                        /* Only take the first */
                        acl.last = *(struct in_addr *) he->h_addr_list[0];
                    }
                }
    
                if (!errors) {
                    IPAddressACL *nacl = (IPAddressACL *) av_mallocz(sizeof(*nacl));
                    IPAddressACL **naclp = 0;
    
                    *nacl = acl;
                    nacl->next = 0;
    
                    if (stream) {
                        naclp = &stream->acl;
                    } else if (feed) {
                        naclp = &feed->acl;
                    } else {
                        fprintf(stderr, "%s:%d: ACL found not in <stream> or <feed>\n",
                                filename, line_num);
                        errors++;
                    }
    
                    if (naclp) {
                        while (*naclp)
                            naclp = &(*naclp)->next;
    
                        *naclp = nacl;
                    }
                }
    
            } else if (!strcasecmp(cmd, "RTSPOption")) {
                get_arg(arg, sizeof(arg), &p);
                if (stream) {
                    av_freep(&stream->rtsp_option);
                    /* XXX: av_strdup ? */
                    stream->rtsp_option = av_malloc(strlen(arg) + 1);
                    if (stream->rtsp_option) {
                        strcpy(stream->rtsp_option, arg);
                    }
                }
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
            } else if (!strcasecmp(cmd, "</Stream>")) {
                if (!stream) {
                    fprintf(stderr, "%s:%d: No corresponding <Stream> for </Stream>\n",
                            filename, line_num);
                    errors++;
                }
                if (stream->feed && stream->fmt && strcmp(stream->fmt->name, "ffm") != 0) {
                    if (audio_id != CODEC_ID_NONE) {
                        audio_enc.codec_type = CODEC_TYPE_AUDIO;
                        audio_enc.codec_id = audio_id;
                        add_codec(stream, &audio_enc);
                    }
                    if (video_id != CODEC_ID_NONE) {
                        video_enc.codec_type = CODEC_TYPE_VIDEO;
                        video_enc.codec_id = video_id;
                        add_codec(stream, &video_enc);
                    }
                }
                stream = NULL;
    
            } else if (!strcasecmp(cmd, "<Redirect")) {
                /*********************************************/
                char *q;
                if (stream || feed || redirect) {
                    fprintf(stderr, "%s:%d: Already in a tag\n",
                            filename, line_num);
                    errors++;
                } else {
                    redirect = av_mallocz(sizeof(FFStream));
                    *last_stream = redirect;
                    last_stream = &redirect->next;
    
                    get_arg(redirect->filename, sizeof(redirect->filename), &p);
                    q = strrchr(redirect->filename, '>');
                    if (*q)
                        *q = '\0';
                    redirect->stream_type = STREAM_TYPE_REDIRECT;
                }
            } else if (!strcasecmp(cmd, "URL")) {
                if (redirect) {
                    get_arg(redirect->feed_filename, sizeof(redirect->feed_filename), &p);
                }
            } else if (!strcasecmp(cmd, "</Redirect>")) {
                if (!redirect) {
                    fprintf(stderr, "%s:%d: No corresponding <Redirect> for </Redirect>\n",
                            filename, line_num);
                    errors++;
                }
                if (!redirect->feed_filename[0]) {
                    fprintf(stderr, "%s:%d: No URL found for <Redirect>\n",
                            filename, line_num);
                    errors++;
                }
                redirect = NULL;
    
            } else if (!strcasecmp(cmd, "LoadModule")) {
                get_arg(arg, sizeof(arg), &p);
                load_module(arg);
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
            } else {
                fprintf(stderr, "%s:%d: Incorrect keyword: '%s'\n", 
                        filename, line_num, cmd);
                errors++;
            }
        }
    
        fclose(f);
        if (errors)
            return -1;
        else
            return 0;
    }
    
    
    #if 0
    static void write_packet(FFCodec *ffenc,
                             UINT8 *buf, int size)
    {
        PacketHeader hdr;
        AVCodecContext *enc = &ffenc->enc;
        UINT8 *wptr;
        mk_header(&hdr, enc, size);
        wptr = http_fifo.wptr;
        fifo_write(&http_fifo, (UINT8 *)&hdr, sizeof(hdr), &wptr);
        fifo_write(&http_fifo, buf, size, &wptr);
        /* atomic modification of wptr */
        http_fifo.wptr = wptr;
        ffenc->data_count += size;
        ffenc->avg_frame_size = ffenc->avg_frame_size * AVG_COEF + size * (1.0 - AVG_COEF);
    }
    #endif
    
    void help(void)
    {
    
        printf("ffserver version " FFMPEG_VERSION ", Copyright (c) 2000, 2001, 2002 Fabrice Bellard\n"
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
               "usage: ffserver [-L] [-h] [-f configfile]\n"
               "Hyper fast multi format Audio/Video streaming server\n"
               "\n"
               "-L            : print the LICENCE\n"
               "-h            : this help\n"
               "-f configfile : use configfile instead of /etc/ffserver.conf\n"
               );
    }
    
    void licence(void)
    {
        printf(
        "ffserver version " FFMPEG_VERSION "\n"
    
        "Copyright (c) 2000, 2001, 2002 Fabrice Bellard\n"
        "This library is free software; you can redistribute it and/or\n"
        "modify it under the terms of the GNU Lesser General Public\n"
        "License as published by the Free Software Foundation; either\n"
        "version 2 of the License, or (at your option) any later version.\n"
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
        "\n"
    
        "This library is distributed in the hope that it will be useful,\n"
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
        "but WITHOUT ANY WARRANTY; without even the implied warranty of\n"
    
        "MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n"
        "Lesser General Public License for more details.\n"
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
        "\n"
    
        "You should have received a copy of the GNU Lesser General Public\n"
        "License along with this library; if not, write to the Free Software\n"
        "Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA\n"
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
        );
    }
    
    
    static void handle_child_exit(int sig)
    {
        pid_t pid;
        int status;
    
        while ((pid = waitpid(-1, &status, WNOHANG)) > 0) {
            FFStream *feed;
    
            for (feed = first_feed; feed; feed = feed->next) {
                if (feed->pid == pid) {
                    int uptime = time(0) - feed->pid_start;
    
                    feed->pid = 0;
                    fprintf(stderr, "%s: Pid %d exited with status %d after %d seconds\n", feed->filename, pid, status, uptime);
    
                    if (uptime < 30) {
                        /* Turn off any more restarts */
                        feed->child_argv = 0;
                    }    
                }
            }
        }
    
        need_to_start_children = 1;
    }
    
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
    int main(int argc, char **argv)
    {
        const char *config_filename;
        int c;
    
        struct sigaction sigact;
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
    
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
    
        config_filename = "/etc/ffserver.conf";
    
    
        my_program_name = argv[0];