Nieuwe flv parser, nieuwe unix sockets support...
This commit is contained in:
		
							parent
							
								
									5f3d64ba0b
								
							
						
					
					
						commit
						6949269ccc
					
				
					 2 changed files with 131 additions and 29 deletions
				
			
		| 
						 | 
				
			
			@ -1,13 +1,34 @@
 | 
			
		|||
#include <sys/types.h>
 | 
			
		||||
#include <sys/socket.h>
 | 
			
		||||
#include <sys/un.h>
 | 
			
		||||
#include <arpa/inet.h>
 | 
			
		||||
#include <unistd.h>
 | 
			
		||||
#include <stdio.h>
 | 
			
		||||
#include <errno.h>
 | 
			
		||||
#include <string.h>
 | 
			
		||||
#include <fcntl.h>
 | 
			
		||||
 | 
			
		||||
bool socketError = false;
 | 
			
		||||
 | 
			
		||||
int DDV_OpenUnix(const char adres[], bool nonblock = false){
 | 
			
		||||
  int s = socket(AF_UNIX, SOCK_STREAM, 0);
 | 
			
		||||
  sockaddr_un addr;
 | 
			
		||||
  addr.sun_family = AF_UNIX;
 | 
			
		||||
  strcpy(addr.sun_path, adres);
 | 
			
		||||
  int r = connect(s, (sockaddr*)&adres, sizeof(addr));
 | 
			
		||||
  if (r == 0){
 | 
			
		||||
    if (nonblock){
 | 
			
		||||
      int flags = fcntl(s, F_GETFL, 0);
 | 
			
		||||
      flags |= O_NONBLOCK;
 | 
			
		||||
      fcntl(s, F_SETFL, flags);
 | 
			
		||||
    }
 | 
			
		||||
    return s;
 | 
			
		||||
  }else{
 | 
			
		||||
    close(s);
 | 
			
		||||
    return 0;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
int DDV_Listen(int port){
 | 
			
		||||
  int s = socket(AF_INET, SOCK_STREAM, 0);
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			@ -36,9 +57,8 @@ int DDV_Accept(int sock){
 | 
			
		|||
  return accept(sock, 0, 0);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
bool DDV_write(void * buffer, int width, int count, int sock){
 | 
			
		||||
bool DDV_write(void * buffer, int todo, int sock){
 | 
			
		||||
  int sofar = 0;
 | 
			
		||||
  int todo = width*count;
 | 
			
		||||
  while (sofar != todo){
 | 
			
		||||
    int r = send(sock, (char*)buffer + sofar, todo-sofar, 0);
 | 
			
		||||
    if (r <= 0){
 | 
			
		||||
| 
						 | 
				
			
			@ -51,9 +71,8 @@ bool DDV_write(void * buffer, int width, int count, int sock){
 | 
			
		|||
  return true;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
bool DDV_read(void * buffer, int width, int count, int sock){
 | 
			
		||||
bool DDV_read(void * buffer, int todo, int sock){
 | 
			
		||||
  int sofar = 0;
 | 
			
		||||
  int todo = width*count;
 | 
			
		||||
  while (sofar != todo){
 | 
			
		||||
    int r = recv(sock, (char*)buffer + sofar, todo-sofar, 0);
 | 
			
		||||
    if (r <= 0){
 | 
			
		||||
| 
						 | 
				
			
			@ -65,3 +84,28 @@ bool DDV_read(void * buffer, int width, int count, int sock){
 | 
			
		|||
  }
 | 
			
		||||
  return true;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
bool DDV_read(void * buffer, int width, int count, int sock){return DDV_read(buffer, width*count, sock);}
 | 
			
		||||
bool DDV_write(void * buffer, int width, int count, int sock){return DDV_write(buffer, width*count, sock);}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
int DDV_iwrite(void * buffer, int todo, int sock){
 | 
			
		||||
  int r = send(sock, buffer, todo, 0);
 | 
			
		||||
  if (r < 0){
 | 
			
		||||
    socketError = true;
 | 
			
		||||
    printf("Could not write! %s\n", strerror(errno));
 | 
			
		||||
  }
 | 
			
		||||
  return r;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
int DDV_iread(void * buffer, int todo, int sock){
 | 
			
		||||
  int r = recv(sock, buffer, todo, 0);
 | 
			
		||||
  if (r < 0){
 | 
			
		||||
    socketError = true;
 | 
			
		||||
    printf("Could not write! %s\n", strerror(errno));
 | 
			
		||||
  }
 | 
			
		||||
  return r;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -1,34 +1,92 @@
 | 
			
		|||
SWBaseSocket::SWBaseError SWBerr;
 | 
			
		||||
char * FLVbuffer;
 | 
			
		||||
int FLV_len;
 | 
			
		||||
int FLVbs = 0;
 | 
			
		||||
bool HeaderDone = false;
 | 
			
		||||
static char FLVheader[13];
 | 
			
		||||
 | 
			
		||||
void FLV_Readheader(SWUnixSocket & ss){
 | 
			
		||||
}//FLV_Readheader
 | 
			
		||||
struct FLV_Pack {
 | 
			
		||||
  int len;
 | 
			
		||||
  int buf;
 | 
			
		||||
  bool isKeyframe;
 | 
			
		||||
  char * data;
 | 
			
		||||
};//FLV_Pack
 | 
			
		||||
 | 
			
		||||
void FLV_Dump(){FLV_len = 0;}
 | 
			
		||||
char FLVHeader[13];
 | 
			
		||||
bool All_Hell_Broke_Loose = false;
 | 
			
		||||
 | 
			
		||||
bool FLV_GetPacket(SWUnixSocket & ss){
 | 
			
		||||
  if (!HeaderDone){
 | 
			
		||||
    if (ss.frecv(FLVheader, 13, &SWBerr) == 13){HeaderDone = true;}
 | 
			
		||||
    return false;
 | 
			
		||||
  }
 | 
			
		||||
//checks FLV Header for correctness
 | 
			
		||||
//returns true if everything is alright, false otherwise
 | 
			
		||||
bool FLV_Checkheader(char * header){
 | 
			
		||||
  if (header[0] != 'F') return false;
 | 
			
		||||
  if (header[1] != 'L') return false;
 | 
			
		||||
  if (header[2] != 'V') return false;
 | 
			
		||||
  if (header[8] != 0x09) return false;
 | 
			
		||||
  if (header[9] != 0) return false;
 | 
			
		||||
  if (header[10] != 0) return false;
 | 
			
		||||
  if (header[11] != 0) return false;
 | 
			
		||||
  if (header[12] != 0) return false;
 | 
			
		||||
  return true;
 | 
			
		||||
}//FLV_Checkheader
 | 
			
		||||
 | 
			
		||||
  
 | 
			
		||||
  if (FLVbs < 15){FLVbuffer = (char*)realloc(FLVbuffer, 15); FLVbs = 15;}
 | 
			
		||||
  //if received a whole header, receive a whole packet
 | 
			
		||||
  //if not, retry header next pass
 | 
			
		||||
  if (FLV_len == 0){
 | 
			
		||||
    if (ss.frecv(FLVbuffer, 11, &SWBerr) == 11){
 | 
			
		||||
      FLV_len = FLVbuffer[3] + 15;
 | 
			
		||||
      FLV_len += (FLVbuffer[2] << 8);
 | 
			
		||||
      FLV_len += (FLVbuffer[1] << 16);
 | 
			
		||||
      if (FLVbs < FLV_len){FLVbuffer = (char*)realloc(FLVbuffer, FLV_len);FLVbs = FLV_len;}
 | 
			
		||||
//returns true if header is an FLV header
 | 
			
		||||
bool FLV_Isheader(char * header){
 | 
			
		||||
  if (header[0] != 'F') return false;
 | 
			
		||||
  if (header[1] != 'L') return false;
 | 
			
		||||
  if (header[2] != 'V') return false;
 | 
			
		||||
  return true;
 | 
			
		||||
}//FLV_Isheader
 | 
			
		||||
 | 
			
		||||
bool ReadUntil(char * buffer, unsigned int count, unsigned int & sofar, int sock){
 | 
			
		||||
  if (sofar >= count){return true;}
 | 
			
		||||
  int r = 0;
 | 
			
		||||
  r = DDV_iread(buffer + sofar,count-sofar,sock);
 | 
			
		||||
  if (r < 0){All_Hell_Broke_Loose = true; return false;}
 | 
			
		||||
  sofar += r;
 | 
			
		||||
  if (sofar >= count){return true;}
 | 
			
		||||
  return false;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
//gets a packet, storing in given FLV_Pack pointer.
 | 
			
		||||
//will assign pointer if null
 | 
			
		||||
//resizes FLV_Pack data field bigger if data doesn't fit
 | 
			
		||||
// (does not auto-shrink for speed!)
 | 
			
		||||
bool FLV_GetPacket(FLV_Pack *& p, int sock){
 | 
			
		||||
  int preflags = fcntl(sock, F_GETFL, 0);
 | 
			
		||||
  int postflags = preflags | O_NONBLOCK;
 | 
			
		||||
  fcntl(sock, F_SETFL, postflags);
 | 
			
		||||
  static bool done = true;
 | 
			
		||||
  static unsigned int sofar = 0;
 | 
			
		||||
  if (!p){p = (FLV_Pack*)calloc(1, sizeof(FLV_Pack));}
 | 
			
		||||
  if (p->buf < 15){p->data = (char*)realloc(p->data, 15); p->buf = 15;}
 | 
			
		||||
 | 
			
		||||
  if (done){
 | 
			
		||||
    //read a header
 | 
			
		||||
    if (ReadUntil(p->data, 11, sofar, sock)){
 | 
			
		||||
      //if its a correct FLV header, throw away and read tag header
 | 
			
		||||
      if (FLV_Isheader(p->data)){
 | 
			
		||||
        if (ReadUntil(p->data, 13, sofar, sock)){
 | 
			
		||||
          if (FLV_Checkheader(p->data)){
 | 
			
		||||
            sofar = 0;
 | 
			
		||||
            memcpy(FLVHeader, p->data, 13);
 | 
			
		||||
          }else{All_Hell_Broke_Loose = true;}
 | 
			
		||||
        }
 | 
			
		||||
      }else{
 | 
			
		||||
        //if a tag header, calculate length and read tag body
 | 
			
		||||
        p->len = p->data[3] + 15;
 | 
			
		||||
        p->len += (p->data[2] << 8);
 | 
			
		||||
        p->len += (p->data[1] << 16);
 | 
			
		||||
        if (p->buf < p->len){p->data = (char*)realloc(p->data, p->len);p->buf = p->len;}
 | 
			
		||||
        done = false;
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
  }else{
 | 
			
		||||
    if (ss.frecv(FLVbuffer+11, FLV_len-11, &SWBerr) == FLV_len-11){return true;}
 | 
			
		||||
    //read tag body
 | 
			
		||||
    if (ReadUntil(p->data, p->len, sofar, sock)){
 | 
			
		||||
      //calculate keyframeness, next time read header again, return true
 | 
			
		||||
      p->isKeyframe = false;
 | 
			
		||||
      if ((p->data[0] == 0x09) && (((p->data[11] & 0xf0) >> 4) == 1)){p->isKeyframe = true;}
 | 
			
		||||
      done = true;
 | 
			
		||||
      sofar = 0;
 | 
			
		||||
      fcntl(sock, F_SETFL, preflags);
 | 
			
		||||
      return true;
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
  fcntl(sock, F_SETFL, preflags);
 | 
			
		||||
  return false;
 | 
			
		||||
}//FLV_GetPacket
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue