Skip to content
Snippets Groups Projects
ffserver.c 145 KiB
Newer Older
  • Learn to ignore specific revisions
  •             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) {
    
    Michael Niedermayer's avatar
    Michael Niedermayer committed
    //                audio_enc.quality = atof(arg) * 1000;
    
            } else if (!strcasecmp(cmd, "VideoBitRateRange")) {
                if (stream) {
                    int minrate, maxrate;
    
                    get_arg(arg, sizeof(arg), &p);
    
                    if (sscanf(arg, "%d-%d", &minrate, &maxrate) == 2) {
                        video_enc.rc_min_rate = minrate * 1000;
                        video_enc.rc_max_rate = maxrate * 1000;
                    } else {
                        fprintf(stderr, "%s:%d: Incorrect format for VideoBitRateRange -- should be <min>-<max>: %s\n", 
                                filename, line_num, arg);
                        errors++;
                    }
                }
            } else if (!strcasecmp(cmd, "VideoBitRateTolerance")) {
                if (stream) {
                    get_arg(arg, sizeof(arg), &p);
                    video_enc.bit_rate_tolerance = atoi(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_base= DEFAULT_FRAME_RATE_BASE;
                    video_enc.frame_rate = (int)(strtod(arg, NULL) * video_enc.frame_rate_base);
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
                }
            } 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, "Video4MotionVector")) {
                if (stream) {
                    video_enc.flags |= CODEC_FLAG_HQ;
                    video_enc.flags |= CODEC_FLAG_4MV;
                }
    
            } 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++;
                    }
                }
    
            } else if (!strcasecmp(cmd, "LumaElim")) {
                get_arg(arg, sizeof(arg), &p);
                if (stream) {
                    video_enc.luma_elim_threshold = atoi(arg);
                }
            } else if (!strcasecmp(cmd, "ChromaElim")) {
                get_arg(arg, sizeof(arg), &p);
                if (stream) {
                    video_enc.chroma_elim_threshold = atoi(arg);
                }
            } else if (!strcasecmp(cmd, "LumiMask")) {
                get_arg(arg, sizeof(arg), &p);
                if (stream) {
                    video_enc.lumi_masking = atof(arg);
                }
            } else if (!strcasecmp(cmd, "DarkMask")) {
                get_arg(arg, sizeof(arg), &p);
                if (stream) {
                    video_enc.dark_masking = atof(arg);
                }
    
    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.s_addr = ntohl(((struct in_addr *) he->h_addr_list[0])->s_addr);
    
                    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.s_addr = ntohl(((struct in_addr *) he->h_addr_list[0])->s_addr);
    
                    }
                }
    
                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);
                    }
                }
    
            } else if (!strcasecmp(cmd, "MulticastAddress")) {
                get_arg(arg, sizeof(arg), &p);
                if (stream) {
                    if (!inet_aton(arg, &stream->multicast_ip)) {
                        fprintf(stderr, "%s:%d: Invalid IP address: %s\n", 
                                filename, line_num, arg);
                        errors++;
                    }
                    stream->is_multicast = 1;
    
                    stream->loop = 1; /* default is looping */
    
                }
            } else if (!strcasecmp(cmd, "MulticastPort")) {
                get_arg(arg, sizeof(arg), &p);
                if (stream) {
                    stream->multicast_port = atoi(arg);
                }
    
            } else if (!strcasecmp(cmd, "MulticastTTL")) {
                get_arg(arg, sizeof(arg), &p);
                if (stream) {
                    stream->multicast_ttl = atoi(arg);
                }
            } else if (!strcasecmp(cmd, "NoLoop")) {
                if (stream) {
                    stream->loop = 0;
                }
    
    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);
    
    #ifdef CONFIG_HAVE_DLOPEN
    
    #else
                fprintf(stderr, "%s:%d: Module support not compiled into this version: '%s'\n", 
                        filename, line_num, arg);
                errors++;
    #endif
    
    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_t *buf, int size)
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
    {
        PacketHeader hdr;
        AVCodecContext *enc = &ffenc->enc;
    
        uint8_t *wptr;
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
        mk_header(&hdr, enc, size);
        wptr = http_fifo.wptr;
    
        fifo_write(&http_fifo, (uint8_t *)&hdr, sizeof(hdr), &wptr);
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
        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
    
    
    static void help(void)
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
    {
    
        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"
               );
    }
    
    
    static void licence(void)
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
    {
        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];
    
        my_program_dir = getcwd(0, 0);
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
        for(;;) {
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
            if (c == -1)
                break;
            switch(c) {
            case 'L':
                licence();
                exit(1);
            case '?':
            case 'h':
                help();
                exit(1);
    
            case 'n':
                no_launch = 1;
                break;
            case 'd':
                ffserver_debug = 1;
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
            case 'f':
                config_filename = optarg;
                break;
            default:
                exit(2);
            }
        }
    
    
        putenv("http_proxy");               /* Kill the http_proxy */
    
    
        /* address on which the server will handle HTTP connections */
        my_http_addr.sin_family = AF_INET;
        my_http_addr.sin_port = htons (8080);
        my_http_addr.sin_addr.s_addr = htonl (INADDR_ANY);
    
        /* address on which the server will handle RTSP connections */
        my_rtsp_addr.sin_family = AF_INET;
        my_rtsp_addr.sin_port = htons (5454);
        my_rtsp_addr.sin_addr.s_addr = htonl (INADDR_ANY);
        
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
        nb_max_connections = 5;
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
        first_stream = NULL;
        logfilename[0] = '\0';
    
    
        memset(&sigact, 0, sizeof(sigact));
        sigact.sa_handler = handle_child_exit;
        sigact.sa_flags = SA_NOCLDSTOP | SA_RESTART;
        sigaction(SIGCHLD, &sigact, 0);
    
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
        if (parse_ffconfig(config_filename) < 0) {
            fprintf(stderr, "Incorrect config file - exiting.\n");
            exit(1);
        }
    
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
        build_feed_streams();
    
    
        /* put the process in background and detach it from its TTY */
        if (ffserver_daemon) {
            int pid;
    
            pid = fork();
            if (pid < 0) {
                perror("fork");
                exit(1);
            } else if (pid > 0) {
                /* parent : exit */
                exit(0);
            } else {
                /* child */
                setsid();
                chdir("/");
                close(0);
                open("/dev/null", O_RDWR);
    
                if (strcmp(logfilename, "-") != 0) {
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
        /* signal init */
        signal(SIGPIPE, SIG_IGN);
    
        /* open log file if needed */
        if (logfilename[0] != '\0') {
            if (!strcmp(logfilename, "-"))
                logfile = stdout;
            else
                logfile = fopen(logfilename, "w");
        }
    
    
        if (http_server() < 0) {
            fprintf(stderr, "Could not start server\n");
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
            exit(1);
        }
    
        return 0;
    }