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

📄 rtsp.c

📁 自己移植的linux下的流媒体播放器原代码,支持mms协议,支持ftp和http协议.
💻 C
📖 第 1 页 / 共 2 页
字号:
/* * This file was ported to MPlayer from xine CVS rtsp.c,v 1.9 2003/04/10 02:30:48 *//* * Copyright (C) 2000-2002 the xine project * * This file is part of xine, a free video player. * * xine is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * xine is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA  02111-1307, USA * * * a minimalistic implementation of rtsp protocol, * *not* RFC 2326 compilant yet. */#include <unistd.h>#include <stdio.h>#include <assert.h>#include "config.h"#ifndef HAVE_WINSOCK2#define closesocket close#include <sys/socket.h>#include <netinet/in.h>#include <netdb.h>#else#include <winsock2.h>#endif#include <string.h>#include <sys/stat.h>#include <fcntl.h>#include <errno.h>#include <stdlib.h>#include <time.h>#include <sys/time.h>#include <sys/types.h>#include <inttypes.h>#include "rtsp.h"/*#define LOG*/#define BUF_SIZE 4096#define HEADER_SIZE 1024#define MAX_FIELDS 256struct rtsp_s {  int           s;  char         *host;  int           port;  char         *path;  char         *param;  char         *mrl;  char         *user_agent;  char         *server;  unsigned int  server_state;  uint32_t      server_caps;    unsigned int  cseq;  char         *session;  char        *answers[MAX_FIELDS];   /* data of last message */  char        *scheduled[MAX_FIELDS]; /* will be sent with next message */};/* * constants */const char rtsp_protocol_version[]="RTSP/1.0";/* server states */#define RTSP_CONNECTED 1#define RTSP_INIT      2#define RTSP_READY     4#define RTSP_PLAYING   8#define RTSP_RECORDING 16/* server capabilities */#define RTSP_OPTIONS       0x001#define RTSP_DESCRIBE      0x002#define RTSP_ANNOUNCE      0x004#define RTSP_SETUP         0x008#define RTSP_GET_PARAMETER 0x010#define RTSP_SET_PARAMETER 0x020#define RTSP_TEARDOWN      0x040#define RTSP_PLAY          0x080#define RTSP_RECORD        0x100/* * network utilities */ static int host_connect_attempt(struct in_addr ia, int port) {  int                s;  struct sockaddr_in sin;  s = socket(PF_INET, SOCK_STREAM, IPPROTO_TCP);    if (s == -1) {    printf ("rtsp: socket(): %s\n", strerror(errno));    return -1;  }  sin.sin_family = AF_INET;  sin.sin_addr   = ia;  sin.sin_port   = htons(port);    if (connect(s, (struct sockaddr *)&sin, sizeof(sin))==-1 #ifndef HAVE_WINSOCK2      && errno != EINPROGRESS) {#else      && WSAGetLastError() == WSAEINPROGRESS) {#endif    printf ("rtsp: connect(): %s\n", strerror(errno));    closesocket(s);    return -1;  }  return s;}static int host_connect(const char *host, int port) {  struct hostent *h;  int             i, s;    h = gethostbyname(host);  if (h == NULL) {    printf ("rtsp: unable to resolve '%s'.\n", host);    return -1;  }  for (i = 0; h->h_addr_list[i]; i++) {    struct in_addr ia;    memcpy (&ia, h->h_addr_list[i], 4);    s = host_connect_attempt(ia, port);    if(s != -1)      return s;  }  printf ("rtsp: unable to connect to '%s'.\n", host);  return -1;}static int write_stream(int s, const char *buf, int len) {  int total, timeout;  total = 0; timeout = 30;  while (total < len){     int n;    n = send (s, &buf[total], len - total, 0);    if (n > 0)      total += n;    else if (n < 0) {#ifndef HAVE_WINSOCK2      if ((timeout>0) && ((errno == EAGAIN) || (errno == EINPROGRESS))) {#else      if ((timeout>0) && ((errno == EAGAIN) || (WSAGetLastError() == WSAEINPROGRESS))) {#endif        sleep (1); timeout--;      } else        return -1;    }  }  return total;}static ssize_t read_stream(int fd, void *buf, size_t count) {    ssize_t ret, total;  total = 0;  while (total < count) {      ret=recv (fd, ((uint8_t*)buf)+total, count-total, 0);    if (ret<0) {      if(errno == EAGAIN) {        fd_set rset;        struct timeval timeout;            FD_ZERO (&rset);        FD_SET  (fd, &rset);                timeout.tv_sec  = 30;        timeout.tv_usec = 0;                if (select (fd+1, &rset, NULL, NULL, &timeout) <= 0) {          return -1;        }        continue;      }            printf ("rtsp: read error.\n");      return ret;    } else      total += ret;        /* end of stream */    if (!ret) break;  }  return total;}/* * debugging utilities */#if 0 static void hexdump (char *buf, int length) {  int i;  printf ("rtsp: ascii>");  for (i = 0; i < length; i++) {    unsigned char c = buf[i];    if ((c >= 32) && (c <= 128))      printf ("%c", c);    else      printf (".");  }  printf ("\n");  printf ("rtsp: hexdump> ");  for (i = 0; i < length; i++) {    unsigned char c = buf[i];    printf ("%02x", c);    if ((i % 16) == 15)      printf ("\nrtsp:         ");    if ((i % 2) == 1)      printf (" ");  }  printf ("\n");}#endif/* * rtsp_get gets a line from stream * and returns a null terminated string. */ static char *rtsp_get(rtsp_t *s) {  int n=0;  char *buffer = malloc(BUF_SIZE);  char *string = NULL;  while (n<BUF_SIZE) {    read_stream(s->s, &(buffer[n]), 1);    if ((buffer[n-1]==0x0d)&&(buffer[n]==0x0a)) break;    n++;  }  if (n>=BUF_SIZE) {    printf("librtsp: buffer overflow in rtsp_get\n");    exit(1);  }  string=malloc(sizeof(char)*n);  memcpy(string,buffer,n-1);  string[n-1]=0;#ifdef LOG  printf("librtsp: << '%s'\n", string);#endif    free(buffer);  return string;}/* * rtsp_put puts a line on stream */ static void rtsp_put(rtsp_t *s, const char *string) {  int len=strlen(string);  char *buf=malloc(sizeof(char)*len+2);#ifdef LOG  printf("librtsp: >> '%s'", string);#endif  memcpy(buf,string,len);  buf[len]=0x0d;  buf[len+1]=0x0a;  write_stream(s->s, buf, len+2);  #ifdef LOG  printf(" done.\n");#endif  free(buf);}/* * extract server status code */static int rtsp_get_code(const char *string) {  char buf[4];  int code=0;   if (!strncmp(string, rtsp_protocol_version, strlen(rtsp_protocol_version)))  {    memcpy(buf, string+strlen(rtsp_protocol_version)+1, 3);    buf[3]=0;    code=atoi(buf);  } else if (!strncmp(string, "SET_PARAMETER",8))  {    return RTSP_STATUS_SET_PARAMETER;  }  if(code != 200) printf("librtsp: server responds: '%s'\n",string);  return code;}/* * send a request */static void rtsp_send_request(rtsp_t *s, const char *type, const char *what) {  char **payload=s->scheduled;  char *buf;    buf = malloc(strlen(type)+strlen(what)+strlen(rtsp_protocol_version)+3);    sprintf(buf,"%s %s %s",type, what, rtsp_protocol_version);  rtsp_put(s,buf);  free(buf);  if (payload)    while (*payload) {      rtsp_put(s,*payload);      payload++;    }  rtsp_put(s,"");  rtsp_unschedule_all(s);}/* * schedule standard fields */static void rtsp_schedule_standard(rtsp_t *s) {  char tmp[16];    snprintf(tmp, 16, "Cseq: %u", s->cseq);  rtsp_schedule_field(s, tmp);    if (s->session) {    char *buf;    buf = malloc(strlen(s->session)+15);    sprintf(buf, "Session: %s", s->session);    rtsp_schedule_field(s, buf);    free(buf);  }}/* * get the answers, if server responses with something != 200, return NULL */ static int rtsp_get_answers(rtsp_t *s) {  char *answer=NULL;  unsigned int answer_seq;  char **answer_ptr=s->answers;  int code;  int ans_count = 0;    answer=rtsp_get(s);  if (!answer)    return 0;  code=rtsp_get_code(answer);  free(answer);  rtsp_free_answers(s);    do { /* while we get answer lines */      answer=rtsp_get(s);    if (!answer)      return 0;        if (!strncmp(answer,"Cseq:",5)) {      sscanf(answer,"Cseq: %u",&answer_seq);      if (s->cseq != answer_seq) {#ifdef LOG        printf("librtsp: warning: Cseq mismatch. got %u, assumed %u", answer_seq, s->cseq);#endif

⌨️ 快捷键说明

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