⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 msg_basic.c

📁 Sofia SIP is an open-source SIP User-Agent library, compliant with the IETF RFC3261 specification.
💻 C
字号:
/* * This file is part of the Sofia-SIP package * * Copyright (C) 2005 Nokia Corporation. * * Contact: Pekka Pessi <pekka.pessi@nokia.com> * * 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 <Pekka.Pessi@nokia.com> * * @date Created: Fri Feb 23 19:51:55 2001 ppessi */#include "config.h"#include <sofia-sip/su_alloc.h>#include <sofia-sip/msg.h>#include <sofia-sip/bnf.h>#include <sofia-sip/msg_parser.h>#include <sofia-sip/msg_header.h>#include <stddef.h>#include <stdlib.h>#include <string.h>#include <assert.h>#include <stdio.h>#include <stdarg.h>#include <limits.h>#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;}/* ====================================================================== */

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -