[SR-Dev] git:master: core: parse_cseq wrong use of parse_method fix

Andrei Pelinescu-Onciul andrei at iptel.org
Wed May 20 19:11:03 CEST 2009


Module: sip-router
Branch: master
Commit: f3bda7478c760f01009180e7416cbd2c9d4eebba
URL:    http://git.sip-router.org/cgi-bin/gitweb.cgi/sip-router/?a=commit;h=f3bda7478c760f01009180e7416cbd2c9d4eebba

Author: Andrei Pelinescu-Onciul <andrei at iptel.org>
Committer: Andrei Pelinescu-Onciul <andrei at iptel.org>
Date:   Wed May 20 19:06:43 2009 +0200

core: parse_cseq wrong use of parse_method fix

- parse_method renamed to parse_method_advance(), made static and
  changed to check if the method name ends with a non-token char.
- added parse_method_name() for parsing a single method name with
  known length (no leading or trailing whitespace).
- changed parse_cseq() to use parse_method_name()
  (it's previous use of parse_method() had the side effect of
  setting the method name to "").

Reported-by: Henning Westerholt <henning.westerholt at 1und1.de>

---

 parser/parse_cseq.c    |    4 +-
 parser/parse_methods.c |  176 +++++++++++++++++++++++++++++++++++++++++-------
 parser/parse_methods.h |    2 +-
 3 files changed, 155 insertions(+), 27 deletions(-)

diff --git a/parser/parse_cseq.c b/parser/parse_cseq.c
index 23dbf42..1778b28 100644
--- a/parser/parse_cseq.c
+++ b/parser/parse_cseq.c
@@ -74,8 +74,8 @@ char* parse_cseq(char *buf, char* end, struct cseq_body* cb)
 	cb->method.len=t-cb->method.s;
 
 	/* Cache method id */
-	if (parse_method(&cb->method, &cb->method_id) == 0) {
-		LOG(L_ERR, "ERROR: parse_cseq: Cannot parse method string\n");
+	if (parse_method_name(&cb->method, &cb->method_id)!=0){
+		LOG(L_ERR, "Cannot parse method string\n");
 		goto error;
 	}
 
diff --git a/parser/parse_methods.c b/parser/parse_methods.c
index 3ba42ce..6c145ee 100644
--- a/parser/parse_methods.c
+++ b/parser/parse_methods.c
@@ -43,24 +43,149 @@ static int token_char(char _c)
  		(_c == '*') || (_c == '_') || (_c == '+') || (_c == '`') ||
  		(_c == '\'') || (_c == '~') || (_c == '+') || (_c == '`');
  }
- 
- 
+
+
+
+/** Parse a string containing a method.
+ * Parse a method pointed by s & assign its enum bit to method. The string
+ * _must_ contain _only_ the method (without trailing or heading whitespace).
+ * @return 0 on success, -1 on error
+ */
+int parse_method_name(str* s, enum request_method* method)
+ {
+	if (unlikely(!s || !method)) {
+		LOG(L_ERR, "Invalid parameter value\n");
+		return -1;
+	}
+	
+	if (unlikely(!s->len || (s->s==0))) {
+		DBG("No input\n");
+		*method = METHOD_OTHER;
+		return 0;
+	}
+	
+	switch ((s->s)[0]) {
+		/* ordered after probability of aparition on a normal proxy */
+		case 'R':
+		case 'r':
+			if (likely((s->len == 8) &&
+					!strncasecmp(s->s + 1, "egister", 7))) {
+				*method = METHOD_REGISTER;
+				return 0;
+			}
+			if (likely((s->len==5) && !strncasecmp(s->s + 1, "efer", 4))) {
+				*method = METHOD_REFER;
+				return 0;
+			}
+			break;
+		case 'A':
+		case 'a':
+			if (likely((s->len==3) && !strncasecmp(s->s + 1, "ck", 2))) {
+				*method = METHOD_ACK;
+				return 0;
+			}
+			break;
+		case 'I':
+		case 'i':
+			if (likely((s->len==6) && !strncasecmp(s->s + 1, "nvite", 5))){
+				*method = METHOD_INVITE;
+				return 0;
+			}
+			if (likely((s->len==4) && !strncasecmp(s->s + 1, "nfo", 3))) {
+				*method = METHOD_INFO;
+				return 0;
+			}
+			break;
+		case 'P':
+		case 'p':
+			if (likely((s->len==5) && !strncasecmp(s->s + 1, "rack", 4))) {
+				*method = METHOD_PRACK;
+				return 0;
+			}
+			if (likely((s->len==7) && !strncasecmp(s->s + 1, "ublish", 6))) {
+				*method = METHOD_PUBLISH;
+				return 0;
+			}
+			break;
+		case 'C':
+		case 'c':
+			if (likely((s->len==6) && !strncasecmp(s->s + 1, "ancel", 5))) {
+				*method = METHOD_CANCEL;
+				return 0;
+			}
+			break;
+		case 'B':
+		case 'b':
+			if (likely((s->len==3) && !strncasecmp(s->s + 1, "ye", 2))) {
+				*method = METHOD_BYE;
+				return 0;
+			}
+			break;
+		case 'M':
+		case 'm':
+			if (likely((s->len==7) && !strncasecmp(s->s + 1, "essage", 6))) {
+				*method = METHOD_MESSAGE;
+				return 0;
+			}
+			break;
+		case 'O':
+		case 'o':
+			if (likely((s->len==7) && !strncasecmp(s->s + 1, "ptions", 6))) {
+				*method = METHOD_OPTIONS;
+				return 0;
+			}
+			break;
+		case 'S':
+		case 's':
+			if (likely((s->len==9) && !strncasecmp(s->s + 1, "ubscribe", 8))) {
+				*method = METHOD_SUBSCRIBE;
+				return 0;
+			}
+			break;
+		case 'N':
+		case 'n':
+			if (likely((s->len==6) && !strncasecmp(s->s + 1, "otify", 5))){
+				*method = METHOD_NOTIFY;
+				return 0;
+			}
+			break;
+		case 'U':
+		case 'u':
+			if (likely((s->len==6) && !strncasecmp(s->s + 1, "pdate", 5))){
+				*method = METHOD_UPDATE;
+				return 0;
+			}
+			break;
+		default:
+			break;
+	}
+/* unknown method */
+	*method = METHOD_OTHER;
+	return 0;
+}
+
+
+
  /*
   * Parse a method pointed by _next, assign its enum bit to _method, and update
   * _next past the method. Returns 1 if parse succeeded and 0 otherwise.
   */
-int parse_method(str* _next, enum request_method* _method) 
+static int parse_method_advance(str* _next, enum request_method* _method)
  {
-	 if (!_next || !_method) {
-		 LOG(L_ERR, "parse_method: Invalid parameter value\n");
+	char* end;
+	
+	 if (unlikely(!_next || !_method)) {
+		 LOG(L_ERR, "Invalid parameter value\n");
 		 return 0;
 	 }
 	 
-	 if (!_next->len || !_next->s) {
-		 DBG("parse_method: No input\n");
+	 if (unlikely(!_next->len || !_next->s)) {
+		 DBG("No input\n");
+ 		*_method = METHOD_OTHER;
 		 return 1;
 	 }
-
+	end=_next->s+_next->len;
+	
 	 switch ((_next->s)[0]) {
 	 case 'A':
 	 case 'a':
@@ -68,7 +193,7 @@ int parse_method(str* _next, enum request_method* _method)
  			*_method = METHOD_ACK;
  			_next->len -= 3;
  			_next->s += 3;
- 			return 1;
+			goto found;
  		} else {
  			goto unknown;
  		}
@@ -79,7 +204,7 @@ int parse_method(str* _next, enum request_method* _method)
  			*_method = METHOD_BYE;
  			_next->len -= 3;
  			_next->s += 3;
- 			return 1;
+			goto found;
  		} else {
  			goto unknown;
  		}
@@ -90,7 +215,7 @@ int parse_method(str* _next, enum request_method* _method)
  			*_method = METHOD_CANCEL;
  			_next->len -= 6;
  			_next->s += 6;
- 			return 1;
+			goto found;
  		} else {
  			goto unknown;
  		}
@@ -103,14 +228,14 @@ int parse_method(str* _next, enum request_method* _method)
  				*_method = METHOD_INFO;
  				_next->len -= 4;
  				_next->s += 4;
- 				return 1;
+				goto found;
  			}
 
  			if ((_next->len > 5) && !strncasecmp(_next->s + 2, "vite", 4)) {
  				*_method = METHOD_INVITE;
  				_next->len -= 6;
  				_next->s += 6;
- 				return 1;
+				goto found;
  			}
  		}
  		goto unknown;
@@ -121,7 +246,7 @@ int parse_method(str* _next, enum request_method* _method)
  			*_method = METHOD_MESSAGE;
  			_next->len -= 7;
  			_next->s += 7;
- 			return 1;
+			goto found;
  		} else {
  			goto unknown;
  		}
@@ -132,7 +257,7 @@ int parse_method(str* _next, enum request_method* _method)
  			*_method = METHOD_NOTIFY;
  			_next->len -= 6;
  			_next->s += 6;
- 			return 1;
+			goto found;
  		} else {
  			goto unknown;
  		}
@@ -143,7 +268,7 @@ int parse_method(str* _next, enum request_method* _method)
  			*_method = METHOD_OPTIONS;
  			_next->len -= 7;
  			_next->s += 7;
- 			return 1;
+			goto found;
  		} else {
  			goto unknown;
  		}
@@ -154,13 +279,13 @@ int parse_method(str* _next, enum request_method* _method)
  			*_method = METHOD_PRACK;
  			_next->len -= 5;
  			_next->s += 5;
- 			return 1;
+			goto found;
  		}
  		if ((_next->len > 6) && !strncasecmp(_next->s + 1, "ublish", 6)) {
  			*_method = METHOD_PUBLISH;
  			_next->len -= 7;
  			_next->s += 7;
- 			return 1;
+			goto found;
  		}
  		goto unknown;
 
@@ -172,14 +297,14 @@ int parse_method(str* _next, enum request_method* _method)
  				*_method = METHOD_REFER;
  				_next->len -= 5;
  				_next->s += 5;
- 				return 1;
+				goto found;
  			}
 
  			if ((_next->len > 7) && !strncasecmp(_next->s + 2, "gister", 6)) {
  				*_method = METHOD_REGISTER;
  				_next->len -= 8;
  				_next->s += 8;
- 				return 1;
+				goto found;
  			}
  		}
  		goto unknown;
@@ -190,7 +315,7 @@ int parse_method(str* _next, enum request_method* _method)
  			*_method = METHOD_SUBSCRIBE;
  			_next->len -= 9;
  			_next->s += 9;
- 			return 1;
+			goto found;
  		} else {
  			goto unknown;
  		}
@@ -201,7 +326,7 @@ int parse_method(str* _next, enum request_method* _method)
  			*_method = METHOD_UPDATE;
  			_next->len -= 6;
  			_next->s += 6;
- 			return 1;
+			goto found;
  		} else {
  			goto unknown;
  		}
@@ -209,7 +334,7 @@ int parse_method(str* _next, enum request_method* _method)
  	default:
  		goto unknown;
  	}
- 
+
  unknown:
  	if (token_char(*(_next->s))) {
  		do { 
@@ -221,6 +346,9 @@ int parse_method(str* _next, enum request_method* _method)
  	} else {
  		return 0;
  	}
+found:
+	/* check if the method really ends here (if not return 0) */
+	return (_next->s>=end) || (!token_char(*(_next->s)));
  }
  
  
@@ -252,7 +380,7 @@ int parse_method(str* _next, enum request_method* _method)
  	}
 
  	while (1) {
- 		if (parse_method(&next, &method)) {
+ 		if (parse_method_advance(&next, &method)) {
  			*_methods |= method;
  		} else {
  			LOG(L_ERR, "ERROR: parse_methods: Invalid method\n");
diff --git a/parser/parse_methods.h b/parser/parse_methods.h
index 867484f..15ca2b9 100644
--- a/parser/parse_methods.h
+++ b/parser/parse_methods.h
@@ -39,7 +39,7 @@
  */
 int parse_methods(str* _body, unsigned int* _methods);
 
-int parse_method(str* _next, enum request_method* _method);
+int parse_method_name(str* s, enum request_method* _method);
 
 
 #endif /* PARSE_METHODS_H */




More information about the sr-dev mailing list