/* * 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_parser * @file msg_generic.c * @brief Functions for generic headers * * @author Pekka Pessi * * @date Created: Thu Jan 23 20:08:00 2003 ppessi * */ #include "config.h" #include #include "sofia-sip/msg.h" #include "sofia-sip/bnf.h" #include "sofia-sip/msg_parser.h" #include "sofia-sip/msg_header.h" #include #include #include #include #include #include #include /** * Parse a generic header. * * The function msg_generic_d() parses a generic header structure. * * @param[in] home memory home * @param[in,out] h header structure * @param[in] s string to be parsed * @param[in] slen length of the string * * @retval 0 when successful, * @retval -1 upon an error. */ issize_t msg_generic_d(su_home_t *home, msg_header_t *h, char *s, isize_t slen) { h->sh_generic->g_string = s; return 0; } /** * Encode a generic header. * * The function @c msg_generic_e encodes a generic header. * */ issize_t msg_generic_e(char b[], isize_t bsiz, msg_header_t const *h, int flags) { msg_generic_t const *g = h->sh_generic; size_t n = strlen(g->g_string); if (bsiz > n) strcpy(b, g->g_string); return (issize_t)n; } /** Calculate the size of strings associated with a @c msg_generic_t object. */ isize_t msg_generic_dup_xtra(msg_header_t const *h, isize_t offset) { msg_generic_t const *g = h->sh_generic; return offset + MSG_STRING_SIZE(g->g_string); } /** Duplicate one @c msg_generic_t object. */ char *msg_generic_dup_one(msg_header_t *dst, msg_header_t const *src, char *b, isize_t xtra) { char *end = b + xtra; MSG_STRING_DUP(b, dst->sh_generic->g_string, src->sh_generic->g_string); assert(b <= end); (void)end; return b; } issize_t msg_numeric_d(su_home_t *home, msg_header_t *h, char *s, isize_t slen) { msg_numeric_t *x = (msg_numeric_t *)h; uint32_t value = 0; issize_t retval = msg_uint32_d(&s, &value); assert(x->x_common->h_class->hc_size >= sizeof *x); x->x_value = value; if (*s) return -1; return retval; } issize_t msg_numeric_e(char b[], isize_t bsiz, msg_header_t const *h, int flags) { msg_numeric_t *x = (msg_numeric_t *)h; assert(x->x_common->h_class->hc_size >= sizeof *x); if (x->x_value > 0xffffffffU) return -1; return snprintf(b, bsiz, "%lu", x->x_value); } /* ====================================================================== */ /* Comma-separated list */ /** @typedef struct msg_list_s msg_list_t; * * Type for token list headers. * */ issize_t msg_list_d(su_home_t *home, msg_header_t *h, char *s, isize_t slen) { return msg_commalist_d(home, &s, &h->sh_list->k_items, NULL); } issize_t msg_list_e(char b[], isize_t bsiz, msg_header_t const *h, int flags) { int compact = MSG_IS_COMPACT(flags); char *b0 = b, *end = b + bsiz; MSG_COMMALIST_E(b, end, h->sh_list->k_items, compact); MSG_TERM_E(b, end); return b - b0; } /**@internal * Extra size of a msg_auth_t object. * * This function calculates extra size required by a msg_auth_t object. * * @param a pointer to a msg_auth_t object * * @return * Size of strings related to msg_auth_t object. */ isize_t msg_list_dup_xtra(msg_header_t const *h, isize_t offset) { MSG_PARAMS_SIZE(offset, h->sh_list->k_items); return offset; } char *msg_list_dup_one(msg_header_t *dst, msg_header_t const *src, char *b, isize_t xtra) { char *end = b + xtra; msg_param_t const ** items = (msg_param_t const **)&dst->sh_list->k_items; b = msg_params_dup(items, src->sh_list->k_items, b, xtra); assert(b <= end); (void)end; return b; } /** Append a list of constant items to a list. * * @retval 0 when successful * @retval -1 upon an error */ int msg_list_append_items(su_home_t *home, msg_list_t *k, msg_param_t const items[]) { size_t i; if (k == NULL) return -1; if (items == NULL) return 0; for (i = 0; items[i]; i++) { if (msg_header_add_param(home, (msg_common_t *)k, items[i]) < 0) return -1; } return 0; } /** Replace a list of constant items. * * @retval 0 when successful * @retval -1 upon an error */ int msg_list_replace_items(su_home_t *home, msg_list_t *k, msg_param_t const items[]) { size_t i; if (k == NULL) return -1; if (items == NULL) return 0; for (i = 0; items[i]; i++) { if (msg_header_replace_item(home, (msg_common_t *)k, items[i]) < 0) return -1; } return 0; }