/* * This file is part of the Sofia-SIP package * * Copyright (C) 2005 Nokia Corporation. * * Contact: Pekka Pessi * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public License * as published by the Free Software Foundation; either version 2.1 of * the License, or (at your option) any later version. * * This library 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 * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA * 02110-1301 USA * */ /**@ingroup msg_headers * @CFILE msg_basic.c * @brief Basic header handling. * * This file contains implementation of basic headers, that is, generic * headers like Subject or Organization containing non-structured text only, * numeric headers like Content-Length or Max-Forwards containing only an * 32-bit unsigned integer, or token list headers like Supported or Allow. * * @author Pekka Pessi * * @date Created: Fri Feb 23 19:51:55 2001 ppessi */ #include "config.h" #include #include #include #include #include #include #include #include #include #include #include #include #define msg_generic_update NULL /* ====================================================================== */ /**@ingroup msg_headers * @defgroup msg_error Erroneous Headers * * The erroneous headers are stored in #msg_error_t structure. * * @note Parser may put other headers (like duplicate Content-Length * headers) into the list of erroneous headers. If the list of erroneous * headers is processed, the header type must be validated first by calling * msg_is_error() (or by other relevant tests). */ /**@ingroup msg_error * @typedef typedef struct msg_error_s msg_error_t; * Type for erroneous headers. */ isize_t msg_error_dup_xtra(msg_header_t const *h, isize_t offset); char *msg_error_dup_one(msg_header_t *dst, msg_header_t const *src, char *b, isize_t xtra); msg_hclass_t msg_error_class[] = MSG_HEADER_CLASS(msg_, error, "", "", er_common, append, msg_error, msg_generic); issize_t msg_error_d(su_home_t *home, msg_header_t *h, char *s, isize_t slen) { return 0; } issize_t msg_error_e(char b[], isize_t bsiz, msg_header_t const *h, int flags) { /* There is no way to encode an erroneous header */ return 0; } isize_t msg_error_dup_xtra(msg_header_t const *h, isize_t offset) { return msg_default_dup_xtra(h, offset); } char *msg_error_dup_one(msg_header_t *dst, msg_header_t const *src, char *b, isize_t xtra) { return msg_default_dup_one(dst, src, b, xtra); } /* ====================================================================== */ /**@ingroup msg_headers * @defgroup msg_unknown Unknown Headers * * The unknown headers are handled with #msg_unknown_t structure. The whole * unknown header including its name is included in the header value string * @a g_value. * * @note It is possible to speed up parsing process by creating a parser * which does understand only a minimum number of headers. If such a parser * is used, some well-known headers are not regocnized or parser, but they * are treated as unknown and put unparsed into the list of unknown headers. */ /**@ingroup msg_unknown * @typedef typedef struct msg_unknown_s msg_unknown_t; * * Type for unknown headers. */ msg_hclass_t msg_unknown_class[] = MSG_HEADER_CLASS(msg_, unknown, "", "", un_common, append, msg_unknown, msg_generic); issize_t msg_unknown_d(su_home_t *home, msg_header_t *h, char *s, isize_t slen) { msg_unknown_t *un = (msg_unknown_t *)h; if (msg_token_d(&s, &un->un_name) < 0 || *s != ':') return -1; *s++ = '\0'; skip_lws(&s); un->un_value = s; return 0; } issize_t msg_unknown_e(char b[], isize_t bsiz, msg_header_t const *h, int flags) { char *b0 = b, *end = b + bsiz; msg_unknown_t *un = (msg_unknown_t *)h; int const compact = MSG_IS_COMPACT(flags); MSG_STRING_E(b, end, un->un_name); MSG_CHAR_E(b, end, ':'); if (!compact) MSG_CHAR_E(b, end, ' '); MSG_STRING_E(b, end, un->un_value); return b - b0; } isize_t msg_unknown_dup_xtra(msg_header_t const *h, isize_t offset) { msg_unknown_t const *un = (msg_unknown_t *)h; return offset + MSG_STRING_SIZE(un->un_name) + MSG_STRING_SIZE(un->un_value); } char *msg_unknown_dup_one(msg_header_t *dst, msg_header_t const *src, char *b, isize_t xtra) { msg_unknown_t *un = (msg_unknown_t *)dst; msg_unknown_t const *o = (msg_unknown_t *)src; char *end = b + xtra; MSG_STRING_DUP(b, un->un_name, o->un_name); MSG_STRING_DUP(b, un->un_value, o->un_value); assert(b <= end); (void)end; return b; } /* ====================================================================== */ /**@ingroup msg_headers * @defgroup msg_payload Message Body * * The payload object contains the message body. The message body has no * structure, but it is stored in the @a pl_data buffer as a byte array. * Multiple payload objects may be linked to a list. */ /**@ingroup msg_payload * @typedef typedef struct msg_payload_s msg_payload_t; * * The structure msg_payload_t contains representation of MIME message payload. * * The msg_payload_t is defined as follows: * @code * typedef struct msg_payload_s { * msg_common_t pl_common[1]; // Common fragment info * msg_header_t *pl_next; // Next payload object * char *pl_data; // Data - may contain zero bytes * usize_t pl_len; // Length of message payload * } msg_payload_t; * @endcode */ msg_hclass_t msg_payload_class[1] = MSG_HEADER_CLASS(msg_, payload, NULL, "", pl_common, append, msg_payload, msg_generic); /** Create a MIME payload */ msg_payload_t *msg_payload_create(su_home_t *home, void const *data, usize_t len) { msg_header_t *h = msg_header_alloc(home, msg_payload_class, len + 1); if (h) { msg_payload_t *pl = (msg_payload_t *)h; char *b = msg_header_data(h->sh_common); if (data) memcpy(b, data, len); else memset(b, 0, len); b[len] = 0; h->sh_data = pl->pl_data = b; h->sh_len = pl->pl_len = len; return pl; } return NULL; } /** Parse payload. */ issize_t msg_payload_d(su_home_t *home, msg_header_t *h, char *s, isize_t slen) { h->sh_payload->pl_len = slen; h->sh_payload->pl_data = s; h->sh_len = slen; h->sh_data = s; return 0; } issize_t msg_payload_e(char b[], isize_t bsiz, msg_header_t const *h, int flags) { size_t len = h->sh_payload->pl_len; if (bsiz > 0) { memcpy(b, h->sh_payload->pl_data, bsiz > len ? len : bsiz); b[bsiz > len ? len : bsiz - 1] = '\0'; } return len; } isize_t msg_payload_dup_xtra(msg_header_t const *h, isize_t offset) { return offset + h->sh_payload->pl_len + 1; } char *msg_payload_dup_one(msg_header_t *dst, msg_header_t const *src, char *b, isize_t xtra) { msg_payload_t *pl = dst->sh_payload; msg_payload_t const *o = src->sh_payload; memcpy(pl->pl_data = b, o->pl_data, pl->pl_len = o->pl_len); pl->pl_common->h_data = pl->pl_data; pl->pl_common->h_len = pl->pl_len; pl->pl_data[pl->pl_len] = 0; /* NUL terminate just in case */ return b + pl->pl_len + 1; } usize_t msg_payload_length(msg_payload_t const *pl) { /* XXX */ return 0; } /* ====================================================================== */ /**@ingroup msg_headers * @defgroup msg_separator Message Separator * * An empty line separates headers from the message body. In order to avoid * modifying messages with integrity protection, the separator line has its * own header structure which is included in the msg_t structure. */ /**@ingroup msg_separator * @typedef typedef struct msg_separator_s msg_separator_t; * * The structure msg_separator_t contains representation of separator line * between message headers and body. * * The msg_separator_t is defined as follows: * @code * typedef struct msg_separator_s { * msg_common_t sep_common[1]; // Common fragment info * msg_header_t *sep_next; // Pointer to next header * char sep_data[4]; // NUL-terminated separator * } msg_separator_t; * @endcode */ msg_hclass_t msg_separator_class[] = MSG_HEADER_CLASS(msg_, separator, NULL, "", sep_common, single, msg_default, msg_generic); /** Calculate length of line ending (0, 1 or 2). @internal */ #define CRLF_TEST(s) ((s[0]) == '\r' ? ((s[1]) == '\n') + 1 : (s[0])=='\n') /** Parse a separator line. */ issize_t msg_separator_d(su_home_t *home, msg_header_t *h, char *s, isize_t slen) { int len = CRLF_TEST(s); if (len == 0 && slen > 0) return -1; memcpy(h->sh_separator->sep_data, s, len); h->sh_separator->sep_data[len] = '\0'; return 0; } /** Encode a separator line. */ issize_t msg_separator_e(char b[], isize_t bsiz, msg_header_t const *h, int flags) { size_t n = strlen(h->sh_separator->sep_data); if (bsiz > n) strcpy(b, h->sh_separator->sep_data); return (issize_t)n; } msg_separator_t *msg_separator_create(su_home_t *home) { msg_separator_t *sep = msg_header_alloc(home, msg_separator_class, 0)->sh_separator; if (sep) strcpy(sep->sep_data, CRLF); return sep; } /* ====================================================================== */