Asterisk 1.8 sip 协议栈分析 2

来源:互联网 发布:类似“一个”的软件 编辑:程序博客网 时间:2024/05/14 05:43

上一篇文章分析了 sip注册消息的流程,下面分析一下 invite请求的处理流程。

 

从handle_request_invite入口,invite请求此处处理replace请求头,如果为replace则认为是咨询,此时不会创建新的通道,而是找到一个通道植入(masqued),大多数情况下是根据invite创建新的请求,所以此处我们从这里开始,不考虑咨询情况

 

首先检查此请求是否为重复请求,if (!req->ignore) ,接下来调用check_via检查via头域,这个函数涉及到nat穿越问题,此函数解析rport头域参数,如果via头域有rport= ,则设置标记此请求包含rport 域标志,同时检查maddr= 域是否存在,如果此处rport=存在则设置nat mode 为 nat,否则为no nat, 至此check_via结束。

 

返回 invite函数,这里 invite 有两种情况,一个为 call-id已经存在,则asterisk认为此请求是re-invite(!p->owner),否则认为是一个新的invite,关于re-invite有很多故事,涉及到asterisk是b2bua还是proxy的问题,下面先讨论非 re-invite请求。

 

从打印信息看到

ast_verbose("Using INVITE request as basis request - %s/n", p->callid);

 

Using INVITE request as basis request - ZjRiYjZkYzYzZDNjNDRmMjhmMmNlNzdmODE4NTYzZmE.

 

如果开启sip history 可以看到会调用,

append_history(p, "Invite", "New call: %s", p->callid);

 

接下来调用parse_ok_contact ()函数保存 此invite的contact头域,以备将来做响应(200 ok,bye, re-invite)

fullcontact 变量保存 全部cantact头域,用来做bye,re-invite,okcontacturi保存uri of acks, bye, re-invite.

 

 接下来调用 下面代码:

 

if (!p->lastinvite && !req->ignore && !p->owner) {        // 全新invite
  /* This is a new invite */
  /* Handle authentication if this is our first invite */
  int cc_recall_core_id = -1;
  set_pvt_allowed_methods(p, req);
  res = check_user_full(p, req, SIP_INVITE, e, XMIT_RELIABLE, addr, &authpeer);
  if (res == AUTH_CHALLENGE_SENT) {
   p->invitestate = INV_COMPLETED;  /* Needs to restart in another INVITE transaction */
   res = 0;
   goto request_invite_cleanup;
  }

 

对于第一次请求会做验证,调用check_user_full,下面分析一下此函数。

 

此函数用 请求的 from 头域 的usr name 和 peer的 ip/port做匹配,check_user_full 调用get_calleridname 从from头域分理出

 

callid_name,最终调用 check_peer_ok ,check_peer_ok内部查找 peer name 是否在 peers 链表中存在。这里先尝试用

from 头域中的 user name查找,找不到则用 ip/port查找。

这里找到后 输出如下。

if (debug)
  ast_verbose("Found peer '%s' for '%s' from %s/n",
   peer->name, of, ast_sockaddr_stringify(&p->recv));

 

 

Found peer '1501159973' for '1501159973' from 10.10.10.84:59584

然后 把peer 相关属性拷贝到为此peer创建的channle中,如 acc,language,amaflags。callgroup,fullcontact等,

 

设置定时器管理 事务。。。

 

接下来调用 dialog_initialize_rtp函数初始化此peer的rtp信息。

 

是否peer有RTP,有则设置编码。设置RTP 引擎,这里需要说明的是 asterisk1.8开始RTP协议栈改动很大,默认使用Asteirsk提供的rtp协议栈,开发者可以自己嵌入其他rtp协议栈。

设置完协议栈后设置此peer 建立rtp会话的默认编码规则。

Found peer '1501159973' for '1501159973' from 10.10.10.84:59584
[Dec 21 15:30:13] DEBUG[28437]: rtp_engine.c:344 ast_rtp_instance_new: Using engine 'asterisk' for RTP instance '0xc1c0078'
[Dec 21 15:30:13] DEBUG[28437]: res_rtp_asterisk.c:472 ast_rtp_new: Allocated port 18084 for RTP instance '0xc1c0078'
[Dec 21 15:30:13] DEBUG[28437]: rtp_engine.c:353 ast_rtp_instance_new: RTP instance '0xc1c0078' is setup and ready to go
[Dec 21 15:30:13] DEBUG[28437]: res_rtp_asterisk.c:2370 ast_rtp_prop_set: Setup RTCP on RTP instance '0xc1c0078'

 

此处路线: check_peer_ok->dialog_initialize_rtp->ast_rtp_instance_new->ast_rtp_instance_set_timeout

ast_rtp_instance_set_hold_timeout

ast_rtp_instance_set_prop

 

ast_rtp_instance_set_qos

 

do_setnat

 

上面为一些列调用过程,初始化此peer的RTP信息,包括 qos,nat mode,rtcp,dtmf几项任务。check_peer_ok函数做了很多工作哇。。。。

 

 

至此 验证通过,RTP信息也初始化完毕,返回 handl_request_invite函数 ,开始处理 SDP啦。。。。

 

 

/* We have a successful authentication, process the SDP portion if there is one */
  if (find_sdp(req)) {

if (process_sdp(p, req, SDP_T38_INITIATE)) {
    /* Asterisk does not yet support any Content-Encoding methods.  Always
     * attempt to process the sdp, but return a 415 if a Content-Encoding header
     * was present after processing fails. */
    if (!ast_strlen_zero(get_header(req, "Content-Encoding"))) {
     transmit_response_reliable(p, "415 Unsupported Media type", req);
    } else {
     /* Unacceptable codecs */
     transmit_response_reliable(p, "488 Not acceptable here", req);
    }

。。。。

 

可以看到,开始处理SDP...

首先当然是从invite请求中找sdp 了,,调用find_sdp (),

 

/*!
  /brief 返回sip invite请求包中是否存在SDP信息,

 */
static int find_sdp(struct sip_request *req)

 

此处当然是找到了。。

调用 process_sdp 处理SDP,

/*! /brief Process SIP SDP offer, select formats and activate RTP channels
 If offer is rejected, we will not change any properties of the call
  Return 0 on success, a negative value on errors.
 Must be called after find_sdp().
*/
static int process_sdp(struct sip_pvt *p, struct sip_request *req, int t38action)

 

 

解析SDP包头。。。

 

[Dec 21 15:30:13] DEBUG[28437]: chan_sip.c:8197 process_sdp: Processing session-level SDP v=0... UNSUPPORTED.
[Dec 21 15:30:13] DEBUG[28437]: chan_sip.c:8197 process_sdp: Processing session-level SDP o=- 0 2 IN IP4 10.10.10.84... UNSUPPORTED.
[Dec 21 15:30:13] DEBUG[28437]: chan_sip.c:8197 process_sdp: Processing session-level SDP s=CounterPath X-Lite 3.0... UNSUPPORTED.
[Dec 21 15:30:13] DEBUG[28437]: netsock2.c:125 ast_sockaddr_split_hostport: Splitting '10.10.10.84' gives...
[Dec 21 15:30:13] DEBUG[28437]: netsock2.c:155 ast_sockaddr_split_hostport: ...host '10.10.10.84' and port '(null)'.
[Dec 21 15:30:13] DEBUG[28437]: chan_sip.c:8197 process_sdp: Processing session-level SDP c=IN IP4 10.10.10.84... OK.
[Dec 21 15:30:13] DEBUG[28437]: chan_sip.c:8197 process_sdp: Processing session-level SDP t=0 0... UNSUPPORTED.

 

首先扫描 m=头域,media stream

 

这里包含一些SDP 信息,解释如下

SDP Data from Example

SDP Parameter

Parameter Name

v=0

Version number

o=Tesla 2890844526 2890844526 IN IP4 lab.high-voltage.org

Origin containing name

s=Phone Call

Subject

c=IN IP4 100.101.102.103

Connection

t=0 0

Time

m=audio 49170 RTP/AVP 0

Media

a=rtpmap:0 PCMU/8000

Attributes

  • Connection IP address (100.101.102.103);

  • Media format (audio);

  • Port number (49170);

  • Media transport protocol (RTP);

  • Media encoding (PCM μ Law);

  • Sampling rate (8,000 Hz).

 

每个SDP头域包含 m,o,c,等对不同SDP头域的处理,

每种类型调用 process_sdp_类型 函数处理,

如 连接地址 处理函数process_sdp_c,这里最重要的是 process_sdp_a_媒体类型,此函数处理SDP包的属性,如音频,则为

process_sdp_a_audio,关于SDP处理的核心位置都在这个函数中。。。process_sdp根据SDP包属性(媒体编码类型)匹配支持的编码类型。

 

[Dec 21 15:30:13] DEBUG[28437]: rtp_engine.c:535 ast_rtp_codecs_payloads_set_m_type: Setting payload 98 based on m type on 0xb7b30490
Found RTP audio format 8
[Dec 21 15:30:13] DEBUG[28437]: rtp_engine.c:535 ast_rtp_codecs_payloads_set_m_type: Setting payload 8 based on m type on 0xb7b30490
Found RTP audio format 101
[Dec 21 15:30:13] DEBUG[28437]: rtp_engine.c:535 ast_rtp_codecs_payloads_set_m_type: Setting payload 101 based on m type on 0xb7b30490
[Dec 21 15:30:13] DEBUG[28437]: chan_sip.c:8384 process_sdp: Processing media-level (audio) SDP a=alt:1 1 : Bwd30+5D C9DdG2tq 10.10.10.84 50946... UNSUPPORTED.
[Dec 21 15:30:13] DEBUG[28437]: chan_sip.c:8384 process_sdp: Processing media-level (audio) SDP a=fmtp:101 0-15... UNSUPPORTED.
Found audio description format BV32 for ID 107
[Dec 21 15:30:13] DEBUG[28437]: chan_sip.c:8384 process_sdp: Processing media-level (audio) SDP a=rtpmap:107 BV32/16000... OK.
Found audio description format BV32-FEC for ID 119
[Dec 21 15:30:13] DEBUG[28437]: chan_sip.c:8384 process_sdp: Processing media-level (audio) SDP a=rtpmap:119 BV32-FEC/16000... OK.
Found audio description format SPEEX for ID 100
[Dec 21 15:30:13] DEBUG[28437]: chan_sip.c:8384 process_sdp: Processing media-level (audio) SDP a=rtpmap:100 SPEEX/16000... OK.
Found audio description format SPEEX-FEC for ID 106
[Dec 21 15:30:13] DEBUG[28437]: chan_sip.c:8384 process_sdp: Processing media-level (audio) SDP a=rtpmap:106 SPEEX-FEC/16000... OK.
Found audio description format SPEEX-FEC for ID 105
[Dec 21 15:30:13] DEBUG[28437]: chan_sip.c:8384 process_sdp: Processing media-level (audio) SDP a=rtpmap:105 SPEEX-FEC/8000... OK.
Found audio description format iLBC for ID 98
[Dec 21 15:30:13] DEBUG[28437]: chan_sip.c:8384 process_sdp: Processing media-level (audio) SDP a=rtpmap:98 iLBC/8000... OK.
Found audio description format telephone-event for ID 101
[Dec 21 15:30:13] DEBUG[28437]: chan_sip.c:8384 process_sdp: Processing media-level (audio) SDP a=rtpmap:101 telephone-event/8000... OK.
[Dec 21 15:30:13] DEBUG[28437]: chan_sip.c:8384 process_sdp: Processing media-level (audio) SDP a=sendrecv... OK.

 

 

 

至此,关于此peer的 SDP信息及 RTP信息都已初始化完毕。

 

回到 handle_request_invite()

/* Check number of concurrent calls -vs- incoming limit HERE */
  ast_debug(1, "Checking SIP call limits for device %s/n", p->username);

 

开始 检查 peer的 call-limit

如果此peer达到 上限,则返回 480 Temporarily Unavailable (Call limit)响应。。。所以当调试时返回此响应我们应该猜测到 此设备已经达到并发上限。。。。

 

接下来 调用get_destination(),我们得给此请求 送到哪????

此函数用 invite请求的 to 头域 作为请求地址,

根据请求的peer name@ip 查找请求的peerr是否在我这check_sip_domain()。。。找不到则到dialplan中查找。。。。。

 

 

/* If we don't have a peer (i.e. we're a guest call),
   * overwrite the original context */
  if (!ast_test_flag(&p->flags[1], SIP_PAGE2_HAVEPEERCONTEXT) && !ast_strlen_zero(domain_context)) {
   ast_string_field_set(p, context, domain_context);
  }

 

这里是当 我们在sip config 里设置润许guest invite时设置 context为默认。。

 

当在dialplan中找到对应分机时我们得给此SIP 请求 创建 channle啦。。

 

调用 sip_new()...这里,sip_new函数是真正创建sip通道的地方,此函数 在呼入请求,及外呼请求时调用,分别为函数 sip_request_call及handle_request_invite函数。。。

 

sip_new函用 sip_pvt结构创建sip structuer, 设置此通道的编码类型,dtfm, caller id等信息。。。

调用ast_channel_alloc 宏(channel.c)创建sip channle,创建细节在__ast_channel_alloc_ap 函数中,channle的分配用

ao2_alloc函数,同时指定了析构函数释放通道,申请channel内存后设置 管道句柄初始状态,创建此channle调度器上下文,

1.8新增了 caller party information 统计信息,所以此处先初始化这些结构,接下来申请channle无名管道,

if (pipe(tmp->alertpipe)) {
   ast_log(LOG_WARNING, "Channel allocation failed: Can't create alert pipe! Try increasing max file descriptors with ulimit -n/n");
   return ast_channel_unref(tmp);
  } else {
   flags = fcntl(tmp->alertpipe[0], F_GETFL);
   if (fcntl(tmp->alertpipe[0], F_SETFL, flags | O_NONBLOCK) < 0) {
    ast_log(LOG_WARNING, "Channel allocation failed: Unable to set alertpipe nonblocking! (%d: %s)/n", errno, strerror(errno));
    return ast_channel_unref(tmp);
   }
   flags = fcntl(tmp->alertpipe[1], F_GETFL);
   if (fcntl(tmp->alertpipe[1], F_SETFL, flags | O_NONBLOCK) < 0) {
    ast_log(LOG_WARNING, "Channel allocation failed: Unable to set alertpipe nonblocking! (%d: %s)/n", errno, strerror(errno));
    return ast_channel_unref(tmp);
   }
  }

 

把创建的管道加入fd列表监听。。。。这里channle fd数量是有限制的,默认一个channle最大10个。

 

 

/* Always watch the alertpipe */
 ast_channel_set_fd(tmp, AST_ALERT_FD, tmp->alertpipe[0]);
 /* And timing pipe */
 ast_channel_set_fd(tmp, AST_TIMING_FD, tmp->timingfd);

 

 

接下来初始化uniqueid,linkeid,这里uniqueid 最大150个字符,包括系统名字(最大127)+unix时间戳+递增序列。。。

 

初始化channle的amaflags,accountcode,context以便于计费使用(cdr)。

 

 

接下来 开始 分配此 channle cdr 结构并初始化。。

 

 

tmp->cdr = ast_cdr_alloc();// 分配
 ast_cdr_init(tmp->cdr, tmp);//初始化
 ast_cdr_start(tmp->cdr); //设置起始 时间,,,cdr->start..

 

 ast_cel_report_event(tmp, AST_CEL_CHANNEL_START, NULL, NULL, NULL);// cdel引擎启动。。发channle start事件。。

1.8多了个CEL ,也是在这里 初始化。。把channle放到 channles 容内部器。。

 

channle建立完毕,,,发送ami事件 Newchannel。。完毕后返回 创建channle 到sip_new

 

sip为一种通道类型,实际创建为channle.c中的ast_channle 结构,此结构为众多通道的接口层,。。

这里 设置  SIPURI,SIPDOMAIN,parkinglot,accountcode,language,等全局数据,初始化 fd 事件,值得注意的是  asterisk 1.8中 支持了 epoll 异步Io,在一些情况下对系统的并发应该提高很多。

 

接下来 SIP_new 对于有rtp的peer,则初始化 jt引擎(rtp 抖动),

 

然后,此函数调用 Ast_pbx_start进入Asterisk内核。。。。ast_pbx_start 启动新的线程处理此channle..

返回后,调用build_route(),记录 record_Route头域,此处作用是记录路由路径作为未来的请求。

 

ast_debug(2, "%s: New call is still down.... Trying... /n", c->name);

 

至此,channle 已经建立完成,发个临时响应吧。。。transmit_provisional_response(p, "100 Trying", req, 0);

 

[Dec 21 15:30:13] DEBUG[28437]: chan_sip.c:21609 handle_request_invite: SIP/1501159973-0000000b: New call is still down.... Trying..

 

 

 

<--- Transmitting (no NAT) to 10.10.10.84:59584 --->
SIP/2.0 100 Trying
Via: SIP/2.0/UDP 10.10.10.84:59584;branch=z9hG4bK-d87543-7c6375234a299e1c-1--d87543-;received=10.10.10.84;rport=59584
From: "1501159973"<sip:1501159973@10.10.10.182>;tag=e648a101
To: "6969 (Softphone)"<sip:6969@10.10.10.182>
Call-ID: ZjRiYjZkYzYzZDNjNDRmMjhmMmNlNzdmODE4NTYzZmE.
CSeq: 2 INVITE
Server: Asterisk PBX 1.8.2-rc1
Allow: INVITE, ACK, CANCEL, OPTIONS, BYE, REFER, SUBSCRIBE, NOTIFY, INFO, PUBLISH
Supported: replaces, timer
Contact: <sip:6969@10.10.10.182:5060>
Content-Length: 0

 

这里,我们可以看到,via头域多了一个received及rport有值了,解决nat穿透。。。

 

同时记住 100 trying没有 sdp信息。。 调用 顺序,sip_xmit<-send_response<--transmit_response<---transmit_provisional_response<--Handle_request_response.

 

接下来 执行  dialplan.......至此 一次呼入系统的请求 基本完成。

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

原创粉丝点击