new recursive AMF parser, new and improved AMF class
This commit is contained in:
parent
3a3bd060be
commit
4f0f71716f
3 changed files with 202 additions and 141 deletions
|
@ -19,6 +19,6 @@ clean:
|
|||
run-test: $(OUT)
|
||||
rm -rf ./meh
|
||||
mkfifo ./meh
|
||||
nc -o test -l -p 1935 -e './Connector_RTMP 2>./meh'
|
||||
nc -l -p 1935 -e './Connector_RTMP 2>./meh'
|
||||
run-cat:
|
||||
cat < ./meh
|
||||
|
|
|
@ -4,31 +4,134 @@
|
|||
|
||||
class AMFType {
|
||||
public:
|
||||
std::string Indice(){return myIndice;};
|
||||
unsigned char GetType(){return myType;};
|
||||
double NumValue(){return numval;};
|
||||
std::string StrValue(){return strval;};
|
||||
AMFType(double val){strval = ""; numval = val;};
|
||||
AMFType(std::string val){strval = val; numval = 0;};
|
||||
private:
|
||||
const char * Str(){return strval.c_str();};
|
||||
int hasContent(){
|
||||
if (!contents){return 0;}
|
||||
return contents->size();
|
||||
};
|
||||
void addContent(AMFType c){if (contents != 0){contents->push_back(c);}};
|
||||
AMFType* getContentP(int i){if (contents != 0){return &contents->at(i);}else{return 0;}};
|
||||
AMFType getContent(int i){if (contents != 0){return contents->at(i);}else{return AMFType("error");}};
|
||||
AMFType* getContentP(std::string s){
|
||||
if (contents != 0){
|
||||
for (std::vector<AMFType>::iterator it = contents->begin(); it != contents->end(); it++){
|
||||
if (it->Indice() == s){
|
||||
return &(*it);
|
||||
}
|
||||
}
|
||||
}
|
||||
return this;
|
||||
};
|
||||
AMFType getContent(std::string s){
|
||||
if (contents != 0){
|
||||
for (std::vector<AMFType>::iterator it = contents->begin(); it != contents->end(); it++){
|
||||
if (it->Indice() == s){
|
||||
return *it;
|
||||
}
|
||||
}
|
||||
}
|
||||
return AMFType("error");
|
||||
};
|
||||
AMFType(std::string indice, double val, unsigned char setType = 0x00){//num type initializer
|
||||
myIndice = indice;
|
||||
myType = setType;
|
||||
strval = "";
|
||||
numval = val;
|
||||
contents = 0;
|
||||
};
|
||||
AMFType(std::string indice, std::string val, unsigned char setType = 0x02){//str type initializer
|
||||
myIndice = indice;
|
||||
myType = setType;
|
||||
strval = val;
|
||||
numval = 0;
|
||||
contents = 0;
|
||||
};
|
||||
AMFType(std::string indice, unsigned char setType = 0x03){//object type initializer
|
||||
myIndice = indice;
|
||||
myType = setType;
|
||||
strval = "";
|
||||
numval = 0;
|
||||
contents = new std::vector<AMFType>;
|
||||
};
|
||||
~AMFType(){if (contents != 0){delete contents;contents=0;}};
|
||||
AMFType& operator=(const AMFType &a) {
|
||||
myIndice = a.myIndice;
|
||||
myType = a.myType;
|
||||
strval = a.strval;
|
||||
numval = a.numval;
|
||||
if (contents){
|
||||
if (a.contents != contents){
|
||||
delete contents;
|
||||
if (a.contents){
|
||||
contents = new std::vector<AMFType>;
|
||||
for (std::vector<AMFType>::iterator it = a.contents->begin(); it < a.contents->end(); it++){
|
||||
contents->push_back(*it);
|
||||
}
|
||||
}else{
|
||||
contents = 0;
|
||||
}
|
||||
}
|
||||
}else{
|
||||
if (a.contents){
|
||||
contents = new std::vector<AMFType>;
|
||||
for (std::vector<AMFType>::iterator it = a.contents->begin(); it < a.contents->end(); it++){
|
||||
contents->push_back(*it);
|
||||
}
|
||||
}
|
||||
}
|
||||
return *this;
|
||||
};
|
||||
AMFType(const AMFType &a){
|
||||
myIndice = a.myIndice;
|
||||
myType = a.myType;
|
||||
strval = a.strval;
|
||||
numval = a.numval;
|
||||
if (a.contents){
|
||||
contents = new std::vector<AMFType>;
|
||||
for (std::vector<AMFType>::iterator it = a.contents->begin(); it < a.contents->end(); it++){
|
||||
contents->push_back(*it);
|
||||
}
|
||||
}else{contents = 0;}
|
||||
};
|
||||
void Print(std::string indent = ""){
|
||||
std::cerr << indent;
|
||||
switch (myType){
|
||||
case 0x00: std::cerr << "Number"; break;
|
||||
case 0x01: std::cerr << "Bool"; break;
|
||||
case 0x02://short string
|
||||
case 0x0C: std::cerr << "String"; break;
|
||||
case 0x03: std::cerr << "Object"; break;
|
||||
case 0x05: std::cerr << "Null"; break;
|
||||
case 0x06: std::cerr << "Undefined"; break;
|
||||
case 0x0D: std::cerr << "Unsupported"; break;
|
||||
case 0xFF: std::cerr << "Container"; break;
|
||||
}
|
||||
std::cerr << " " << myIndice << " ";
|
||||
switch (myType){
|
||||
case 0x00: case 0x01: std::cerr << numval; break;
|
||||
case 0x02: case 0x0C: std::cerr << strval; break;
|
||||
}
|
||||
std::cerr << std::endl;
|
||||
if (contents){
|
||||
for (std::vector<AMFType>::iterator it = contents->begin(); it != contents->end(); it++){it->Print(indent+" ");}
|
||||
}
|
||||
};
|
||||
protected:
|
||||
std::string myIndice;
|
||||
unsigned char myType;
|
||||
std::string strval;
|
||||
double numval;
|
||||
std::vector<AMFType> * contents;
|
||||
};//AMFType
|
||||
|
||||
//scans the vector for the indice, returns the next AMFType from it or null
|
||||
AMFType * getAMF(std::vector<AMFType> * vect, std::string indice){
|
||||
std::vector<AMFType>::iterator it;
|
||||
for (it=vect->begin(); it < vect->end(); it++){
|
||||
if ((*it).StrValue() == indice){it++; return &(*it);}
|
||||
}
|
||||
return 0;
|
||||
}//getAMF
|
||||
|
||||
std::vector<AMFType> * parseAMF(unsigned char * data, unsigned int len){
|
||||
std::vector<AMFType> * ret = new std::vector<AMFType>;
|
||||
unsigned int i = 0;
|
||||
AMFType parseOneAMF(unsigned char *& data, unsigned int &len, unsigned int &i, std::string name){
|
||||
std::string tmpstr;
|
||||
unsigned int tmpi = 0;
|
||||
unsigned char tmpdbl[8];
|
||||
while (i < len){
|
||||
switch (data[i]){
|
||||
case 0x00://number
|
||||
tmpdbl[7] = data[i+1];
|
||||
|
@ -39,100 +142,54 @@ std::vector<AMFType> * parseAMF(unsigned char * data, unsigned int len){
|
|||
tmpdbl[2] = data[i+6];
|
||||
tmpdbl[1] = data[i+7];
|
||||
tmpdbl[0] = data[i+8];
|
||||
ret->push_back(*(double*)tmpdbl);
|
||||
i+=9;
|
||||
fprintf(stderr, "AMF: Number %f\n", *(double*)tmpdbl);
|
||||
i += 8;
|
||||
return AMFType(name, *(double*)tmpdbl, 0x00);
|
||||
break;
|
||||
case 0x01://bool
|
||||
if (data[i+1] == 0){
|
||||
ret->push_back((double)0);
|
||||
i+=2;
|
||||
if (data[i-1] == 0){
|
||||
fprintf(stderr, "AMF: Bool false\n");
|
||||
return AMFType(name, (double)0, 0x01);
|
||||
}else{
|
||||
ret->push_back((double)1);
|
||||
fprintf(stderr, "AMF: Bool true\n");
|
||||
return AMFType(name, (double)1, 0x01);
|
||||
}
|
||||
++i;
|
||||
break;
|
||||
case 0x0C://long string
|
||||
tmpi = data[i+1]*256*256*256+data[i+2]*256*256+data[i+3]*256+data[i+4];
|
||||
tmpstr = (char*)(data+i+5);
|
||||
ret->push_back(tmpstr);
|
||||
i += tmpi + 4;
|
||||
i += tmpi + 5;
|
||||
fprintf(stderr, "AMF: String %s\n", tmpstr.c_str());
|
||||
return AMFType(name, tmpstr, 0x0C);
|
||||
break;
|
||||
case 0x02://string
|
||||
tmpi = data[i+1]*256+data[i+2];
|
||||
tmpstr = (char*)(data+i+3);
|
||||
ret->push_back(tmpstr);
|
||||
i += tmpi + 2;
|
||||
i += tmpi + 3;
|
||||
fprintf(stderr, "AMF: String %s\n", tmpstr.c_str());
|
||||
return AMFType(name, tmpstr, 0x02);
|
||||
break;
|
||||
case 0x05://null
|
||||
case 0x06://undefined
|
||||
case 0x0D://unsupported
|
||||
fprintf(stderr, "AMF: Null\n");
|
||||
ret->push_back((double)0);
|
||||
break;
|
||||
case 0x03://object
|
||||
++i;
|
||||
return AMFType(name, (double)0, data[i-1]);
|
||||
break;
|
||||
case 0x03:{//object
|
||||
++i;
|
||||
AMFType ret = AMFType(name, data[i-1]);
|
||||
while (data[i] + data[i+1] != 0){
|
||||
tmpi = data[i]*256+data[i+1];
|
||||
tmpstr = (char*)(data+i+2);
|
||||
ret->push_back(tmpstr);
|
||||
i += tmpi + 2;
|
||||
fprintf(stderr, "AMF: Indice %s\n", tmpstr.c_str());
|
||||
switch (data[i]){
|
||||
case 0x00://number
|
||||
tmpdbl[7] = data[i+1];
|
||||
tmpdbl[6] = data[i+2];
|
||||
tmpdbl[5] = data[i+3];
|
||||
tmpdbl[4] = data[i+4];
|
||||
tmpdbl[3] = data[i+5];
|
||||
tmpdbl[2] = data[i+6];
|
||||
tmpdbl[1] = data[i+7];
|
||||
tmpdbl[0] = data[i+8];
|
||||
ret->push_back(*(double*)tmpdbl);
|
||||
fprintf(stderr, "AMF: Value Number %f\n", *(double*)tmpdbl);
|
||||
i += 8;
|
||||
break;
|
||||
case 0x01://bool
|
||||
if (data[i+1] == 0){
|
||||
ret->push_back((double)0);
|
||||
fprintf(stderr, "AMF: Value Bool false\n");
|
||||
}else{
|
||||
ret->push_back((double)1);
|
||||
fprintf(stderr, "AMF: Value Bool true\n");
|
||||
ret.addContent(parseOneAMF(data, len, i, tmpstr));
|
||||
}
|
||||
++i;
|
||||
break;
|
||||
case 0x0C://long string
|
||||
tmpi = data[i+1]*256*256*256+data[i+2]*256*256+data[i+3]*256+data[i+4];
|
||||
tmpstr = (char*)(data+i+5);
|
||||
ret->push_back(tmpstr);
|
||||
i += tmpi + 4;
|
||||
fprintf(stderr, "AMF: Value String %s\n", tmpstr.c_str());
|
||||
break;
|
||||
case 0x02://string
|
||||
tmpi = data[i+1]*256+data[i+2];
|
||||
tmpstr = (char*)(data+i+3);
|
||||
ret->push_back(tmpstr);
|
||||
i += tmpi + 2;
|
||||
fprintf(stderr, "AMF: Value String %s\n", tmpstr.c_str());
|
||||
break;
|
||||
case 0x05://null
|
||||
case 0x06://undefined
|
||||
case 0x0D://unsupported
|
||||
fprintf(stderr, "AMF: Value Null\n");
|
||||
ret->push_back((double)0);
|
||||
break;
|
||||
default:
|
||||
fprintf(stderr, "Error: Unknown AMF object contents type %hhx - returning.\n", data[i]);
|
||||
break;
|
||||
}
|
||||
++i;
|
||||
}
|
||||
i += 2;
|
||||
break;
|
||||
i += 3;
|
||||
return ret;
|
||||
} break;
|
||||
case 0x07://reference
|
||||
case 0x08://array
|
||||
case 0x0A://strict array
|
||||
|
@ -141,11 +198,15 @@ std::vector<AMFType> * parseAMF(unsigned char * data, unsigned int len){
|
|||
case 0x10://typed object
|
||||
case 0x11://AMF+
|
||||
default:
|
||||
fprintf(stderr, "Error: Unknown AMF type %hhx - returning.\n", data[i]);
|
||||
return ret;
|
||||
fprintf(stderr, "Error: Unimplemented AMF type %hhx - returning.\n", data[i]);
|
||||
return AMFType("error", (unsigned char)0xFF);
|
||||
break;
|
||||
}
|
||||
++i;
|
||||
}
|
||||
}//parseOneAMF
|
||||
|
||||
AMFType parseAMF(unsigned char * data, unsigned int len){
|
||||
AMFType ret("returned", (unsigned char)0xFF);//container type
|
||||
unsigned int i = 0;
|
||||
while (i < len){ret.addContent(parseOneAMF(data, len, i, ""));}
|
||||
return ret;
|
||||
}//parseAMF
|
||||
|
|
|
@ -5,8 +5,8 @@
|
|||
//gets and parses one chunk
|
||||
void parseChunk(){
|
||||
static chunkpack next;
|
||||
static std::vector<AMFType> * amfdata = 0;
|
||||
static AMFType * amfelem = 0;
|
||||
static AMFType amfdata("empty", (unsigned char)0xFF);
|
||||
static AMFType amfelem("empty", (unsigned char)0xFF);
|
||||
static int tmpint;
|
||||
next = getWholeChunk();
|
||||
if (next.cs_id == 2 && next.msg_stream_id == 0){
|
||||
|
@ -67,14 +67,14 @@ void parseChunk(){
|
|||
fprintf(stderr, "Received AFM0 shared object\n");
|
||||
break;
|
||||
case 20:
|
||||
if (amfdata != 0){delete amfdata;}
|
||||
amfdata = parseAMF(next.data, next.real_len);
|
||||
fprintf(stderr, "Received AFM0 command message: %s\n", (*amfdata)[0].StrValue().c_str());
|
||||
if ((*amfdata)[0].StrValue() == "connect"){
|
||||
tmpint = getAMF(amfdata, "videoCodecs")->NumValue();
|
||||
amfdata.Print();
|
||||
fprintf(stderr, "Received AFM0 command message: %s\n", amfdata.getContentP(0)->Str());
|
||||
if (amfdata.getContentP(0)->StrValue() == "connect"){
|
||||
tmpint = amfdata.getContentP(2)->getContentP("videoCodecs")->NumValue();
|
||||
if (tmpint & 0x04){fprintf(stderr, "Sorensen video support detected\n");}
|
||||
if (tmpint & 0x80){fprintf(stderr, "H264 video support detected\n");}
|
||||
tmpint = getAMF(amfdata, "audioCodecs")->NumValue();
|
||||
tmpint = amfdata.getContentP(2)->getContentP("audioCodecs")->NumValue();
|
||||
if (tmpint & 0x04){fprintf(stderr, "MP3 audio support detected\n");}
|
||||
if (tmpint & 0x400){fprintf(stderr, "AAC video support detected\n");}
|
||||
SendCTL(5, snd_window_size);//send window acknowledgement size (msg 5)
|
||||
|
@ -84,7 +84,7 @@ void parseChunk(){
|
|||
}else{
|
||||
//call, close, createStream
|
||||
//TODO: play (&& play2?)
|
||||
fprintf(stderr, "Ignored AFM0 command.\n");
|
||||
//fprintf(stderr, "Ignored AFM0 command.\n");
|
||||
}
|
||||
break;
|
||||
case 22:
|
||||
|
|
Loading…
Add table
Reference in a new issue