/* * CDDL HEADER START * * The contents of this file are subject to the terms of the * Common Development and Distribution License (the "License"). * You may not use this file except in compliance with the License. * * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE * or http://www.opensolaris.org/os/licensing. * See the License for the specific language governing permissions * and limitations under the License. * * When distributing Covered Code, include this CDDL HEADER in each * file and include the License file at usr/src/OPENSOLARIS.LICENSE. * If applicable, add the following below this CDDL HEADER, with the * fields enclosed by brackets "[]" replaced with your own identifying * information: Portions Copyright [yyyy] [name of copyright owner] * * CDDL HEADER END */ /* * Copyright 2007 Sun Microsystems, Inc. All rights reserved. * Use is subject to license terms. */ #pragma ident "%Z%%M% %I% %E% SMI" #include <stdlib.h> #include <assert.h> #include <ctype.h> #include <pthread.h> #include <strings.h> #include <sip.h> #include "sip_miscdefs.h" /* * Local version of case insensitive strstr(). */ static char * sip_reass_strstr(const char *as1, const char *as2) { const char *s1; const char *s2; const char *tptr; char c; s1 = as1; s2 = as2; if (s2 == NULL || *s2 == '\0') return ((char *)s1); c = *s2; while (*s1) if (tolower(*s1++) == c) { tptr = s1; while ((c = *++s2) == tolower(*s1++) && c) ; if (c == 0) return ((char *)tptr - 1); s1 = tptr; s2 = as2; c = *s2; } return (NULL); } /* * Get the value in the content-length field and add it to the header length * and return the total length. returns -1 if the length cannot be determined * or if the message does not contain the entire message. */ static int sip_get_msglen(char *p, size_t msglen) { int value = 0; int hlen; char *c; char *e; int base = 10; char *edge; int digits = 0; edge = p + msglen; if ((c = sip_reass_strstr(p, "content-length")) == NULL) return (-1); hlen = c - p; if ((hlen + strlen("content-length")) >= msglen) return (-1); c += strlen("content-length"); e = c + 1; while (*e == ' ' || *e == ':') { e++; if (e == edge) return (-1); } while (*e != '\r' && *e != ' ') { if (e == edge) return (-1); if (*e >= '0' && *e <= '9') digits = *e - '0'; else return (-1); value = (value * base) + digits; e++; } while (*e != '\r') { e++; if (e == edge) return (-1); } hlen = e - p + 4; /* 4 for 2 CRLFs ?? */ value += hlen; return (value); } /* * We have determined that msg does not contain a *single* complete message. * Add it to the reassembly list and check if we have a complete message. * a NULL 'msg' means we are just checking if there are more complete * messages in the list that can be passed up. */ char * sip_get_tcp_msg(sip_conn_object_t obj, char *msg, size_t *msglen) { int value; sip_conn_obj_pvt_t *pvt_data; sip_reass_entry_t *reass; void **obj_val; char *msgbuf = NULL; int splitlen; char *splitbuf; if (msg != NULL) { assert(*msglen > 0); msgbuf = (char *)malloc(*msglen + 1); if (msgbuf == NULL) return (NULL); (void) strncpy(msgbuf, msg, *msglen); msgbuf[*msglen] = '\0'; msg = msgbuf; } obj_val = (void *)obj; pvt_data = (sip_conn_obj_pvt_t *)*obj_val; /* * connection object not initialized */ if (pvt_data == NULL) { if (msg == NULL) return (NULL); value = sip_get_msglen(msg, *msglen); if (value == *msglen) { return (msg); } else { if (msgbuf != NULL) free(msgbuf); return (NULL); } } (void) pthread_mutex_lock(&pvt_data->sip_conn_obj_reass_lock); reass = pvt_data->sip_conn_obj_reass; assert(reass != NULL); if (reass->sip_reass_msg == NULL) { assert(reass->sip_reass_msglen == 0); if (msg == NULL) { (void) pthread_mutex_unlock( &pvt_data->sip_conn_obj_reass_lock); return (NULL); } value = sip_get_msglen(msg, *msglen); if (value == *msglen) { (void) pthread_mutex_unlock( &pvt_data->sip_conn_obj_reass_lock); return (msg); } reass->sip_reass_msg = msg; reass->sip_reass_msglen = *msglen; if (value != -1 && value < reass->sip_reass_msglen) goto tryone; (void) pthread_mutex_unlock(&pvt_data->sip_conn_obj_reass_lock); return (NULL); } else if (msg != NULL) { /* * Resize, not optimal */ int newlen = reass->sip_reass_msglen + *msglen; char *newmsg; assert(strlen(reass->sip_reass_msg) == reass->sip_reass_msglen); newmsg = malloc(newlen + 1); if (newmsg == NULL) { (void) pthread_mutex_unlock( &pvt_data->sip_conn_obj_reass_lock); if (msgbuf != NULL) free(msgbuf); return (NULL); } (void) strncpy(newmsg, reass->sip_reass_msg, reass->sip_reass_msglen); newmsg[reass->sip_reass_msglen] = '\0'; (void) strncat(newmsg, msg, *msglen); newmsg[newlen] = '\0'; assert(strlen(newmsg) == newlen); reass->sip_reass_msglen = newlen; free(msg); free(reass->sip_reass_msg); reass->sip_reass_msg = newmsg; } value = sip_get_msglen(reass->sip_reass_msg, reass->sip_reass_msglen); if (value == -1 || value > reass->sip_reass_msglen) { (void) pthread_mutex_unlock(&pvt_data->sip_conn_obj_reass_lock); return (NULL); } tryone: if (value == reass->sip_reass_msglen) { msg = reass->sip_reass_msg; *msglen = reass->sip_reass_msglen; reass->sip_reass_msg = NULL; reass->sip_reass_msglen = 0; (void) pthread_mutex_unlock(&pvt_data->sip_conn_obj_reass_lock); return (msg); } splitlen = reass->sip_reass_msglen - value; msg = (char *)malloc(value + 1); splitbuf = (char *)malloc(splitlen + 1); if (msg == NULL || splitbuf == NULL) { if (msg != NULL) free(msg); if (splitbuf != NULL) free(splitbuf); (void) pthread_mutex_unlock(&pvt_data->sip_conn_obj_reass_lock); return (NULL); } (void) strncpy(msg, reass->sip_reass_msg, value); msg[value] = '\0'; (void) strncpy(splitbuf, reass->sip_reass_msg + value, splitlen); splitbuf[splitlen] = '\0'; free(reass->sip_reass_msg); reass->sip_reass_msg = splitbuf; reass->sip_reass_msglen = splitlen; (void) pthread_mutex_unlock(&pvt_data->sip_conn_obj_reass_lock); *msglen = value; return (msg); }