如何使用PJSIP库解码SIP字节流?

时间:2013-09-18 05:43:06

标签: c++ sip decode pjsip

我正在尝试使用pjsip库来解码以下SIP字节流,但我遇到了分段错误。我的代码出了什么问题?

#include <pjsip.h>

int main()
{

    char __MSG[] = {
        0x49, 0x4e, 0x56, 0x49, 0x54, 0x45, 0x20, 0x73, 0x69, 0x70,
        0x3a, 0x40, 0x31, 0x32, 0x37, 0x2e, 0x30, 0x2e, 0x30, 0x2e,
        0x31, 0x20, 0x53, 0x49, 0x50, 0x2f, 0x32, 0x2e, 0x30, 0x0d,
        0x0a, 0x54, 0x6f, 0x3a, 0x20, 0x3c, 0x73, 0x69, 0x70, 0x3a,
        0x31, 0x30, 0x2e, 0x30, 0x2e, 0x31, 0x2e, 0x34, 0x35, 0x3e,
        0x0d, 0x0a, 0x56, 0x69, 0x61, 0x3a, 0x20, 0x53, 0x49, 0x50,
        0x2f, 0x32, 0x2e, 0x30, 0x2f, 0x55, 0x44, 0x50, 0x20, 0x31,
        0x30, 0x2e, 0x30, 0x2e, 0x31, 0x2e, 0x34, 0x35, 0x0d, 0x0a,
        0x46, 0x72, 0x6f, 0x6d, 0x3a, 0x20, 0x22, 0x74, 0x65, 0x73,
        0x74, 0x74, 0x65, 0x73, 0x74, 0x22, 0x3c, 0x73, 0x69, 0x70,
        0x3a, 0x31, 0x30, 0x2e, 0x30, 0x2e, 0x31, 0x2e, 0x31, 0x39,
        0x39, 0x3e, 0x0d, 0x0a, 0x43, 0x61, 0x6c, 0x6c, 0x2d, 0x49,
        0x44, 0x3a, 0x20, 0x31, 0x34, 0x38, 0x31, 0x30, 0x2e, 0x30,
        0x2e, 0x31, 0x2e, 0x34, 0x35, 0x0d, 0x0a, 0x43, 0x53, 0x65,
        0x71, 0x3a, 0x20, 0x31, 0x20, 0x49, 0x4e, 0x56, 0x49, 0x54,
        0x45, 0x0d, 0x0a, 0x4d, 0x61, 0x78, 0x2d, 0x46, 0x6f, 0x72,
        0x77, 0x61, 0x72, 0x64, 0x73, 0x3a, 0x20, 0x32, 0x30, 0x0d,
        0x0a, 0x43, 0x6f, 0x6e, 0x74, 0x61, 0x63, 0x74, 0x3a, 0x20,
        0x3c, 0x73, 0x69, 0x70, 0x3a, 0x31, 0x32, 0x37, 0x2e, 0x30,
        0x2e, 0x30, 0x2e, 0x31, 0x3e, 0x0d, 0x0a, 0x0d, 0x0a, NULL
    };



    char *testmsg = __MSG;

    pj_size_t msgsize;
    pj_status_t status;


    // INIT
    status = pj_init();
    PJ_ASSERT_RETURN(status == PJ_SUCCESS, 1);

    status = pjlib_util_init();
    PJ_ASSERT_RETURN(status == PJ_SUCCESS, 1);

    // PARSING
    pj_caching_pool cp;
    pj_caching_pool_init(&cp, NULL, 1024*1024);
    pj_pool_t *pool = pj_pool_create(&cp.factory, "parser_pool", 4000, 4000, NULL);
    pjsip_parser_err_report err;

    int len = strlen(testmsg);
    pjsip_msg *msg = pjsip_parse_msg(pool, __MSG, len, &err);

    printf("The end...");

    return 0;
}
  

程序因信号11,分段故障而终止   #0 0x0805befb在pj_scan_peek()(gdb)bt中   pj_scan_peek()中的#0 0x0805befb   int_parse_msg()中的#1 0x080507f6   pjsip_parse_msg()中的#2 0x080523a2   #3 0x0804fa89在main()(gdb)

1 个答案:

答案 0 :(得分:1)

您需要使用SIP端点实例。来自doc

  

SIP端点实例(pjsip_endpoint)可以视为   应用程序中所有SIP对象的主/所有者。它执行   以下角色:

     
      
  • 它管理所有对象的内存池的分配/释放。
  •   
  • 它管理侦听器和传输,以及事务如何使用它们。
  •   
  • 它从传输层接收传入消息并自动将它们分派到正确的事务(或创建一个新事务)。
  •   
  • 它有一个定时器管理实例(定时器堆)。
  •   
  • 它管理模块,这是扩展库的主要方法。
  •   
  • 它为所有对象提供单一轮询功能并分发事件。
  •   
  • 它会自动处理现有模块无法处理的传入请求(例如传入请求不受支持时)   法)。
  •   
  • 依旧......
  •   
     

应用程序应仅为实例化一个SIP端点实例   每个过程。

您需要致电pjsip_endpt_create来创建端点。

查看Stateless SIP Endpoint示例,了解初始化和设置端点时需要进行的库调用的一般结构。

其他注意事项:

  • pjsip_parse_msg函数将pjsip_parser_err_report列表作为最后一个参数。您必须使用pj_list_init(&amp; err)初始化列表。
  • 确保在调用pj_pool_create后检查池是否为NULL。
  • 文档没有说明是否可以将NULL池工厂策略传递给pj_caching_pool_init;可能最好只传递默认值(如下面的示例所示)。

我认为它应该看起来像这样(但没有测试过):

    // INIT
    status = pj_init();
    PJ_ASSERT_RETURN(status == PJ_SUCCESS, 1);

    status = pjlib_util_init();
    PJ_ASSERT_RETURN(status == PJ_SUCCESS, 1);

    // Create pool factory (for memory allocations)
    pj_caching_pool cp;
    pj_caching_pool_init(&cp, &pj_pool_factory_default_policy, 1024*1024);

    // Create global endpoint
    static pjsip_endpoint *sip_endpt;
    status = pjsip_endpt_create(&cp.factory, "uniquesipendpointname", &sip_endpt);
    PJ_ASSERT_RETURN(status == PJ_SUCCESS, 1);

    // PARSING    
    pj_pool_t *pool = pj_pool_create(&cp.factory, "parser_pool", 4000, 4000, NULL);
    pjsip_parser_err_report err;

    int len = strlen(testmsg);
    pj_list_init(&err);
    pjsip_msg *msg = pjsip_parse_msg(pool, __MSG, len, &err);