blob: e363eb906e3c8823050b73e1d6e42aac86bf71fc [file] [log] [blame]
Benny Prijono268ca612006-02-07 12:34:11 +00001/* $Id$ */
2/*
3 * Copyright (C) 2003-2006 Benny Prijono <benny@prijono.org>
4 *
5 * This program is free software; you can redistribute it and/or modify
6 * it under the terms of the GNU General Public License as published by
7 * the Free Software Foundation; either version 2 of the License, or
8 * (at your option) any later version.
9 *
10 * This program is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 * GNU General Public License for more details.
14 *
15 * You should have received a copy of the GNU General Public License
16 * along with this program; if not, write to the Free Software
17 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
18 */
19#include <pjsip-ua/sip_inv.h>
20#include <pjsip/sip_module.h>
21#include <pjsip/sip_endpoint.h>
22#include <pjsip/sip_event.h>
23#include <pjsip/sip_transaction.h>
24#include <pjmedia/sdp.h>
25#include <pjmedia/sdp_neg.h>
Benny Prijono95196582006-02-09 00:13:40 +000026#include <pjmedia/errno.h>
Benny Prijono268ca612006-02-07 12:34:11 +000027#include <pj/string.h>
28#include <pj/pool.h>
29#include <pj/assert.h>
Benny Prijono8ad55352006-02-08 11:16:05 +000030#include <pj/os.h>
Benny Prijonoa66c7152006-02-09 01:26:14 +000031#include <pj/log.h>
32
Benny Prijono268ca612006-02-07 12:34:11 +000033
34#define THIS_FILE "sip_invite_session.c"
35
Benny Prijonoe8b0d3b2006-03-17 17:57:52 +000036static const char *inv_state_names[] =
37{
38 "NULL ",
39 "CALLING ",
40 "INCOMING ",
41 "EARLY ",
42 "CONNECTING",
43 "CONFIRMED ",
44 "DISCONNCTD",
45 "TERMINATED",
46};
47
Benny Prijono268ca612006-02-07 12:34:11 +000048/*
49 * Static prototypes.
50 */
51static pj_status_t mod_inv_load(pjsip_endpoint *endpt);
52static pj_status_t mod_inv_unload(void);
53static pj_bool_t mod_inv_on_rx_request(pjsip_rx_data *rdata);
54static pj_bool_t mod_inv_on_rx_response(pjsip_rx_data *rdata);
55static void mod_inv_on_tsx_state(pjsip_transaction*, pjsip_event*);
56
Benny Prijono8ad55352006-02-08 11:16:05 +000057static void inv_on_state_null( pjsip_inv_session *inv, pjsip_event *e);
58static void inv_on_state_calling( pjsip_inv_session *inv, pjsip_event *e);
59static void inv_on_state_incoming( pjsip_inv_session *inv, pjsip_event *e);
60static void inv_on_state_early( pjsip_inv_session *inv, pjsip_event *e);
61static void inv_on_state_connecting( pjsip_inv_session *inv, pjsip_event *e);
62static void inv_on_state_confirmed( pjsip_inv_session *inv, pjsip_event *e);
63static void inv_on_state_disconnected( pjsip_inv_session *inv, pjsip_event *e);
Benny Prijono268ca612006-02-07 12:34:11 +000064
Benny Prijono8ad55352006-02-08 11:16:05 +000065static void (*inv_state_handler[])( pjsip_inv_session *inv, pjsip_event *e) =
Benny Prijono268ca612006-02-07 12:34:11 +000066{
67 &inv_on_state_null,
68 &inv_on_state_calling,
69 &inv_on_state_incoming,
70 &inv_on_state_early,
71 &inv_on_state_connecting,
72 &inv_on_state_confirmed,
73 &inv_on_state_disconnected,
Benny Prijono268ca612006-02-07 12:34:11 +000074};
75
76static struct mod_inv
77{
78 pjsip_module mod;
79 pjsip_endpoint *endpt;
80 pjsip_inv_callback cb;
Benny Prijono268ca612006-02-07 12:34:11 +000081} mod_inv =
82{
83 {
Benny Prijono2f8992b2006-02-25 21:16:36 +000084 NULL, NULL, /* prev, next. */
85 { "mod-invite", 10 }, /* Name. */
86 -1, /* Id */
87 PJSIP_MOD_PRIORITY_DIALOG_USAGE, /* Priority */
88 &mod_inv_load, /* load() */
89 NULL, /* start() */
90 NULL, /* stop() */
91 &mod_inv_unload, /* unload() */
92 &mod_inv_on_rx_request, /* on_rx_request() */
93 &mod_inv_on_rx_response, /* on_rx_response() */
94 NULL, /* on_tx_request. */
95 NULL, /* on_tx_response() */
96 &mod_inv_on_tsx_state, /* on_tsx_state() */
Benny Prijono268ca612006-02-07 12:34:11 +000097 }
98};
99
100
Benny Prijonoa66c7152006-02-09 01:26:14 +0000101/* Invite session data to be attached to transaction. */
102struct tsx_inv_data
103{
104 pjsip_inv_session *inv;
105 pj_bool_t sdp_done;
106};
107
108
Benny Prijono8ad55352006-02-08 11:16:05 +0000109/*
110 * Module load()
111 */
Benny Prijono268ca612006-02-07 12:34:11 +0000112static pj_status_t mod_inv_load(pjsip_endpoint *endpt)
113{
114 pj_str_t allowed[] = {{"INVITE", 6}, {"ACK",3}, {"BYE",3}, {"CANCEL",6}};
Benny Prijono56315612006-07-18 14:39:40 +0000115 pj_str_t accepted = { "application/sdp", 15 };
Benny Prijono268ca612006-02-07 12:34:11 +0000116
117 /* Register supported methods: INVITE, ACK, BYE, CANCEL */
118 pjsip_endpt_add_capability(endpt, &mod_inv.mod, PJSIP_H_ALLOW, NULL,
119 PJ_ARRAY_SIZE(allowed), allowed);
120
Benny Prijono56315612006-07-18 14:39:40 +0000121 /* Register "application/sdp" in Accept header */
122 pjsip_endpt_add_capability(endpt, &mod_inv.mod, PJSIP_H_ACCEPT, NULL,
123 1, &accepted);
124
Benny Prijono268ca612006-02-07 12:34:11 +0000125 return PJ_SUCCESS;
126}
127
Benny Prijono8ad55352006-02-08 11:16:05 +0000128/*
129 * Module unload()
130 */
Benny Prijono268ca612006-02-07 12:34:11 +0000131static pj_status_t mod_inv_unload(void)
132{
133 /* Should remove capability here */
134 return PJ_SUCCESS;
135}
136
Benny Prijono8ad55352006-02-08 11:16:05 +0000137/*
Benny Prijono38998232006-02-08 22:44:25 +0000138 * Set session state.
139 */
140void inv_set_state(pjsip_inv_session *inv, pjsip_inv_state state,
141 pjsip_event *e)
142{
Benny Prijonod4e0abd2006-03-05 11:53:36 +0000143 pjsip_inv_state prev_state = inv->state;
144
145 /* Set state. */
Benny Prijono38998232006-02-08 22:44:25 +0000146 inv->state = state;
Benny Prijonod4e0abd2006-03-05 11:53:36 +0000147
148 /* If state is DISCONNECTED, cause code MUST have been set. */
149 pj_assert(inv->state != PJSIP_INV_STATE_DISCONNECTED ||
150 inv->cause != 0);
151
152 /* Call on_state_changed() callback. */
153 if (mod_inv.cb.on_state_changed && inv->notify)
Benny Prijono38998232006-02-08 22:44:25 +0000154 (*mod_inv.cb.on_state_changed)(inv, e);
155
Benny Prijonod4e0abd2006-03-05 11:53:36 +0000156 /* Only decrement when previous state is not already DISCONNECTED */
157 if (inv->state == PJSIP_INV_STATE_DISCONNECTED &&
158 prev_state != PJSIP_INV_STATE_DISCONNECTED)
159 {
Benny Prijonoe4f2abb2006-02-10 14:04:05 +0000160 pjsip_dlg_dec_session(inv->dlg, &mod_inv.mod);
Benny Prijonod4e0abd2006-03-05 11:53:36 +0000161 }
Benny Prijono38998232006-02-08 22:44:25 +0000162}
163
164
165/*
Benny Prijono0b6340c2006-06-13 22:21:23 +0000166 * Set cause code.
167 */
168void inv_set_cause(pjsip_inv_session *inv, int cause_code,
169 const pj_str_t *cause_text)
170{
171 if (cause_code > inv->cause) {
172 inv->cause = cause_code;
173 if (cause_text)
174 pj_strdup(inv->pool, &inv->cause_text, cause_text);
175 else if (cause_code/100 == 2)
176 inv->cause_text = pj_str("Normal call clearing");
177 else
178 inv->cause_text = *pjsip_get_status_text(cause_code);
179 }
180}
181
182
183
184/*
Benny Prijono8ad55352006-02-08 11:16:05 +0000185 * Send ACK for 2xx response.
186 */
187static pj_status_t inv_send_ack(pjsip_inv_session *inv, pjsip_rx_data *rdata)
Benny Prijono268ca612006-02-07 12:34:11 +0000188{
189 pjsip_tx_data *tdata;
190 pj_status_t status;
191
Benny Prijonoe4f2abb2006-02-10 14:04:05 +0000192 PJ_LOG(5,(inv->obj_name, "Received %s, sending ACK",
193 pjsip_rx_data_get_info(rdata)));
194
Benny Prijono268ca612006-02-07 12:34:11 +0000195 status = pjsip_dlg_create_request(inv->dlg, &pjsip_ack_method,
196 rdata->msg_info.cseq->cseq, &tdata);
197 if (status != PJ_SUCCESS) {
198 /* Better luck next time */
199 pj_assert(!"Unable to create ACK!");
200 return status;
201 }
202
Benny Prijono64158af2006-04-04 11:06:34 +0000203 status = pjsip_dlg_send_request(inv->dlg, tdata, -1, NULL);
Benny Prijono268ca612006-02-07 12:34:11 +0000204 if (status != PJ_SUCCESS) {
205 /* Better luck next time */
206 pj_assert(!"Unable to send ACK!");
207 return status;
208 }
209
210 return PJ_SUCCESS;
211}
212
Benny Prijono8ad55352006-02-08 11:16:05 +0000213/*
214 * Module on_rx_request()
215 *
216 * This callback is called for these events:
217 * - endpoint receives request which was unhandled by higher priority
218 * modules (e.g. transaction layer, dialog layer).
219 * - dialog distributes incoming request to its usages.
220 */
221static pj_bool_t mod_inv_on_rx_request(pjsip_rx_data *rdata)
222{
223 pjsip_method *method;
Benny Prijono38998232006-02-08 22:44:25 +0000224 pjsip_dialog *dlg;
225 pjsip_inv_session *inv;
Benny Prijono8ad55352006-02-08 11:16:05 +0000226
227 /* Only wants to receive request from a dialog. */
Benny Prijono38998232006-02-08 22:44:25 +0000228 dlg = pjsip_rdata_get_dlg(rdata);
229 if (dlg == NULL)
Benny Prijono8ad55352006-02-08 11:16:05 +0000230 return PJ_FALSE;
231
Benny Prijono38998232006-02-08 22:44:25 +0000232 inv = dlg->mod_data[mod_inv.mod.id];
233
Benny Prijono8ad55352006-02-08 11:16:05 +0000234 /* Report to dialog that we handle INVITE, CANCEL, BYE, ACK.
235 * If we need to send response, it will be sent in the state
236 * handlers.
237 */
238 method = &rdata->msg_info.msg->line.req.method;
239
Benny Prijono70127222006-07-02 14:53:05 +0000240 if (method->id == PJSIP_INVITE_METHOD) {
241 return PJ_TRUE;
242 }
243
244 /* BYE and CANCEL must have existing invite session */
245 if (method->id == PJSIP_BYE_METHOD ||
246 method->id == PJSIP_CANCEL_METHOD)
Benny Prijono8ad55352006-02-08 11:16:05 +0000247 {
Benny Prijono70127222006-07-02 14:53:05 +0000248 if (inv == NULL)
249 return PJ_FALSE;
250
Benny Prijono8ad55352006-02-08 11:16:05 +0000251 return PJ_TRUE;
252 }
253
Benny Prijono38998232006-02-08 22:44:25 +0000254 /* On receipt ACK request, when state is CONNECTING,
255 * move state to CONFIRMED.
256 */
Benny Prijono26ff9062006-02-21 23:47:00 +0000257 if (method->id == PJSIP_ACK_METHOD && inv) {
Benny Prijono38998232006-02-08 22:44:25 +0000258
Benny Prijonof521eb02006-08-06 23:07:25 +0000259 /* Ignore ACK if pending INVITE transaction has not finished. */
260 if (inv->invite_tsx &&
261 inv->invite_tsx->state < PJSIP_TSX_STATE_COMPLETED)
262 {
263 return PJ_TRUE;
264 }
265
Benny Prijono5eff0432006-02-09 14:14:21 +0000266 /* Terminate INVITE transaction, if it's still present. */
267 if (inv->invite_tsx &&
268 inv->invite_tsx->state <= PJSIP_TSX_STATE_COMPLETED)
269 {
Benny Prijonof521eb02006-08-06 23:07:25 +0000270 pj_assert(inv->invite_tsx->status_code >= 200);
Benny Prijono5eff0432006-02-09 14:14:21 +0000271 pjsip_tsx_terminate(inv->invite_tsx,
272 inv->invite_tsx->status_code);
273 inv->invite_tsx = NULL;
274 }
275
Benny Prijono32ac8fe2006-03-02 21:16:55 +0000276 /* On receipt of ACK, only set state to confirmed when state
277 * is CONNECTING (e.g. we don't want to set the state to confirmed
278 * when we receive ACK retransmission after sending non-2xx!)
279 */
280 if (inv->state == PJSIP_INV_STATE_CONNECTING) {
Benny Prijono26ff9062006-02-21 23:47:00 +0000281 pjsip_event event;
282
283 PJSIP_EVENT_INIT_RX_MSG(event, rdata);
284 inv_set_state(inv, PJSIP_INV_STATE_CONFIRMED, &event);
285 }
Benny Prijono38998232006-02-08 22:44:25 +0000286 }
287
Benny Prijono8ad55352006-02-08 11:16:05 +0000288 return PJ_FALSE;
289}
290
291/*
292 * Module on_rx_response().
293 *
294 * This callback is called for these events:
295 * - dialog distributes incoming 2xx response to INVITE (outside
296 * transaction) to its usages.
297 * - endpoint distributes strayed responses.
298 */
Benny Prijono268ca612006-02-07 12:34:11 +0000299static pj_bool_t mod_inv_on_rx_response(pjsip_rx_data *rdata)
300{
301 pjsip_dialog *dlg;
302 pjsip_inv_session *inv;
303 pjsip_msg *msg = rdata->msg_info.msg;
304
305 dlg = pjsip_rdata_get_dlg(rdata);
306
307 /* Ignore responses outside dialog */
308 if (dlg == NULL)
309 return PJ_FALSE;
310
311 /* Ignore responses not belonging to invite session */
312 inv = pjsip_dlg_get_inv_session(dlg);
313 if (inv == NULL)
314 return PJ_FALSE;
315
316 /* This MAY be retransmission of 2xx response to INVITE.
317 * If it is, we need to send ACK.
318 */
319 if (msg->type == PJSIP_RESPONSE_MSG && msg->line.status.code/100==2 &&
Benny Prijono26ff9062006-02-21 23:47:00 +0000320 rdata->msg_info.cseq->method.id == PJSIP_INVITE_METHOD &&
321 inv->invite_tsx == NULL)
322 {
Benny Prijono268ca612006-02-07 12:34:11 +0000323
Benny Prijono8ad55352006-02-08 11:16:05 +0000324 inv_send_ack(inv, rdata);
Benny Prijono268ca612006-02-07 12:34:11 +0000325 return PJ_TRUE;
326
327 }
328
329 /* No other processing needs to be done here. */
330 return PJ_FALSE;
331}
332
Benny Prijono8ad55352006-02-08 11:16:05 +0000333/*
334 * Module on_tsx_state()
335 *
336 * This callback is called by dialog framework for all transactions
337 * inside the dialog for all its dialog usages.
338 */
Benny Prijono268ca612006-02-07 12:34:11 +0000339static void mod_inv_on_tsx_state(pjsip_transaction *tsx, pjsip_event *e)
340{
341 pjsip_dialog *dlg;
342 pjsip_inv_session *inv;
343
344 dlg = pjsip_tsx_get_dlg(tsx);
345 if (dlg == NULL)
346 return;
347
348 inv = pjsip_dlg_get_inv_session(dlg);
349 if (inv == NULL)
350 return;
351
352 /* Call state handler for the invite session. */
353 (*inv_state_handler[inv->state])(inv, e);
354
355 /* Call on_tsx_state */
Benny Prijonod4e0abd2006-03-05 11:53:36 +0000356 if (mod_inv.cb.on_tsx_state_changed && inv->notify)
Benny Prijono268ca612006-02-07 12:34:11 +0000357 (*mod_inv.cb.on_tsx_state_changed)(inv, tsx, e);
358
359 /* Clear invite transaction when tsx is terminated. */
360 if (tsx->state==PJSIP_TSX_STATE_TERMINATED && tsx == inv->invite_tsx)
361 inv->invite_tsx = NULL;
362}
363
Benny Prijono8ad55352006-02-08 11:16:05 +0000364
365/*
366 * Initialize the invite module.
367 */
Benny Prijono268ca612006-02-07 12:34:11 +0000368PJ_DEF(pj_status_t) pjsip_inv_usage_init( pjsip_endpoint *endpt,
Benny Prijono268ca612006-02-07 12:34:11 +0000369 const pjsip_inv_callback *cb)
370{
371 pj_status_t status;
372
373 /* Check arguments. */
Benny Prijonoe8b0d3b2006-03-17 17:57:52 +0000374 PJ_ASSERT_RETURN(endpt && cb, PJ_EINVAL);
Benny Prijono268ca612006-02-07 12:34:11 +0000375
376 /* Some callbacks are mandatory */
377 PJ_ASSERT_RETURN(cb->on_state_changed && cb->on_new_session, PJ_EINVAL);
378
379 /* Check if module already registered. */
380 PJ_ASSERT_RETURN(mod_inv.mod.id == -1, PJ_EINVALIDOP);
381
382 /* Copy param. */
383 pj_memcpy(&mod_inv.cb, cb, sizeof(pjsip_inv_callback));
384
385 mod_inv.endpt = endpt;
Benny Prijono268ca612006-02-07 12:34:11 +0000386
387 /* Register the module. */
388 status = pjsip_endpt_register_module(endpt, &mod_inv.mod);
389
390 return status;
391}
392
Benny Prijono8ad55352006-02-08 11:16:05 +0000393/*
394 * Get the instance of invite module.
395 */
Benny Prijono268ca612006-02-07 12:34:11 +0000396PJ_DEF(pjsip_module*) pjsip_inv_usage_instance(void)
397{
398 return &mod_inv.mod;
399}
400
401
Benny Prijono632ce712006-02-09 14:01:40 +0000402
Benny Prijono8ad55352006-02-08 11:16:05 +0000403/*
404 * Return the invite session for the specified dialog.
405 */
Benny Prijono268ca612006-02-07 12:34:11 +0000406PJ_DEF(pjsip_inv_session*) pjsip_dlg_get_inv_session(pjsip_dialog *dlg)
407{
408 return dlg->mod_data[mod_inv.mod.id];
409}
410
Benny Prijono8ad55352006-02-08 11:16:05 +0000411
Benny Prijono268ca612006-02-07 12:34:11 +0000412/*
Benny Prijonoe8b0d3b2006-03-17 17:57:52 +0000413 * Get INVITE state name.
414 */
415PJ_DEF(const char *) pjsip_inv_state_name(pjsip_inv_state state)
416{
417 PJ_ASSERT_RETURN(state >= PJSIP_INV_STATE_NULL &&
418 state <= PJSIP_INV_STATE_DISCONNECTED,
419 "??");
420
421 return inv_state_names[state];
422}
423
424/*
Benny Prijono8ad55352006-02-08 11:16:05 +0000425 * Create UAC invite session.
Benny Prijono268ca612006-02-07 12:34:11 +0000426 */
427PJ_DEF(pj_status_t) pjsip_inv_create_uac( pjsip_dialog *dlg,
428 const pjmedia_sdp_session *local_sdp,
429 unsigned options,
430 pjsip_inv_session **p_inv)
431{
432 pjsip_inv_session *inv;
433 pj_status_t status;
434
435 /* Verify arguments. */
436 PJ_ASSERT_RETURN(dlg && p_inv, PJ_EINVAL);
437
Benny Prijono8eae8382006-08-10 21:44:26 +0000438 /* Must lock dialog first */
439 pjsip_dlg_inc_lock(dlg);
440
Benny Prijono268ca612006-02-07 12:34:11 +0000441 /* Normalize options */
442 if (options & PJSIP_INV_REQUIRE_100REL)
443 options |= PJSIP_INV_SUPPORT_100REL;
444
445 if (options & PJSIP_INV_REQUIRE_TIMER)
446 options |= PJSIP_INV_SUPPORT_TIMER;
447
448 /* Create the session */
449 inv = pj_pool_zalloc(dlg->pool, sizeof(pjsip_inv_session));
Benny Prijono8eae8382006-08-10 21:44:26 +0000450 pj_assert(inv != NULL);
Benny Prijono268ca612006-02-07 12:34:11 +0000451
452 inv->pool = dlg->pool;
453 inv->role = PJSIP_ROLE_UAC;
454 inv->state = PJSIP_INV_STATE_NULL;
455 inv->dlg = dlg;
456 inv->options = options;
Benny Prijonod4e0abd2006-03-05 11:53:36 +0000457 inv->notify = PJ_TRUE;
458 inv->cause = 0;
Benny Prijono268ca612006-02-07 12:34:11 +0000459
Benny Prijonoe4f2abb2006-02-10 14:04:05 +0000460 /* Object name will use the same dialog pointer. */
Benny Prijonoed811d72006-03-10 12:57:12 +0000461 pj_ansi_snprintf(inv->obj_name, PJ_MAX_OBJ_NAME, "inv%p", dlg);
Benny Prijonoe4f2abb2006-02-10 14:04:05 +0000462
Benny Prijono268ca612006-02-07 12:34:11 +0000463 /* Create negotiator if local_sdp is specified. */
464 if (local_sdp) {
465 status = pjmedia_sdp_neg_create_w_local_offer(dlg->pool, local_sdp,
466 &inv->neg);
Benny Prijono8eae8382006-08-10 21:44:26 +0000467 if (status != PJ_SUCCESS) {
468 pjsip_dlg_dec_lock(dlg);
Benny Prijono268ca612006-02-07 12:34:11 +0000469 return status;
Benny Prijono8eae8382006-08-10 21:44:26 +0000470 }
Benny Prijono268ca612006-02-07 12:34:11 +0000471 }
472
473 /* Register invite as dialog usage. */
474 status = pjsip_dlg_add_usage(dlg, &mod_inv.mod, inv);
Benny Prijono8eae8382006-08-10 21:44:26 +0000475 if (status != PJ_SUCCESS) {
476 pjsip_dlg_dec_lock(dlg);
Benny Prijono268ca612006-02-07 12:34:11 +0000477 return status;
Benny Prijono8eae8382006-08-10 21:44:26 +0000478 }
Benny Prijono268ca612006-02-07 12:34:11 +0000479
480 /* Increment dialog session */
Benny Prijonoe4f2abb2006-02-10 14:04:05 +0000481 pjsip_dlg_inc_session(dlg, &mod_inv.mod);
Benny Prijono268ca612006-02-07 12:34:11 +0000482
483 /* Done */
484 *p_inv = inv;
Benny Prijonoe4f2abb2006-02-10 14:04:05 +0000485
Benny Prijono8eae8382006-08-10 21:44:26 +0000486 pjsip_dlg_dec_lock(dlg);
487
Benny Prijonoe4f2abb2006-02-10 14:04:05 +0000488 PJ_LOG(5,(inv->obj_name, "UAC invite session created for dialog %s",
489 dlg->obj_name));
490
Benny Prijono268ca612006-02-07 12:34:11 +0000491 return PJ_SUCCESS;
492}
493
494/*
495 * Verify incoming INVITE request.
496 */
497PJ_DEF(pj_status_t) pjsip_inv_verify_request(pjsip_rx_data *rdata,
498 unsigned *options,
499 const pjmedia_sdp_session *l_sdp,
500 pjsip_dialog *dlg,
501 pjsip_endpoint *endpt,
502 pjsip_tx_data **p_tdata)
503{
504 pjsip_msg *msg;
505 pjsip_allow_hdr *allow;
506 pjsip_supported_hdr *sup_hdr;
507 pjsip_require_hdr *req_hdr;
508 int code = 200;
509 unsigned rem_option = 0;
510 pj_status_t status = PJ_SUCCESS;
511 pjsip_hdr res_hdr_list;
512
513 /* Init return arguments. */
514 if (p_tdata) *p_tdata = NULL;
515
516 /* Verify arguments. */
517 PJ_ASSERT_RETURN(rdata != NULL && options != NULL, PJ_EINVAL);
518
519 /* Normalize options */
520 if (*options & PJSIP_INV_REQUIRE_100REL)
521 *options |= PJSIP_INV_SUPPORT_100REL;
522
523 if (*options & PJSIP_INV_REQUIRE_TIMER)
524 *options |= PJSIP_INV_SUPPORT_TIMER;
525
526 /* Get the message in rdata */
527 msg = rdata->msg_info.msg;
528
529 /* Must be INVITE request. */
530 PJ_ASSERT_RETURN(msg->type == PJSIP_REQUEST_MSG &&
531 msg->line.req.method.id == PJSIP_INVITE_METHOD,
532 PJ_EINVAL);
533
534 /* If tdata is specified, then either dlg or endpt must be specified */
535 PJ_ASSERT_RETURN((!p_tdata) || (endpt || dlg), PJ_EINVAL);
536
537 /* Get the endpoint */
538 endpt = endpt ? endpt : dlg->endpt;
539
540 /* Init response header list */
541 pj_list_init(&res_hdr_list);
542
Benny Prijono8ad55352006-02-08 11:16:05 +0000543 /* Check the request body, see if it'inv something that we support
Benny Prijono268ca612006-02-07 12:34:11 +0000544 * (i.e. SDP).
545 */
546 if (msg->body) {
547 pjsip_msg_body *body = msg->body;
548 pj_str_t str_application = {"application", 11};
549 pj_str_t str_sdp = { "sdp", 3 };
550 pjmedia_sdp_session *sdp;
551
552 /* Check content type. */
553 if (pj_stricmp(&body->content_type.type, &str_application) != 0 ||
554 pj_stricmp(&body->content_type.subtype, &str_sdp) != 0)
555 {
556 /* Not "application/sdp" */
557 code = PJSIP_SC_UNSUPPORTED_MEDIA_TYPE;
558 status = PJSIP_ERRNO_FROM_SIP_STATUS(code);
559
560 if (p_tdata) {
561 /* Add Accept header to response */
562 pjsip_accept_hdr *acc;
563
564 acc = pjsip_accept_hdr_create(rdata->tp_info.pool);
565 PJ_ASSERT_RETURN(acc, PJ_ENOMEM);
566 acc->values[acc->count++] = pj_str("application/sdp");
567 pj_list_push_back(&res_hdr_list, acc);
568 }
569
570 goto on_return;
571 }
572
573 /* Parse and validate SDP */
574 status = pjmedia_sdp_parse(rdata->tp_info.pool, body->data, body->len,
575 &sdp);
576 if (status == PJ_SUCCESS)
577 status = pjmedia_sdp_validate(sdp);
578
579 if (status != PJ_SUCCESS) {
580 /* Unparseable or invalid SDP */
581 code = PJSIP_SC_BAD_REQUEST;
582
583 if (p_tdata) {
584 /* Add Warning header. */
585 pjsip_warning_hdr *w;
586
587 w = pjsip_warning_hdr_create_from_status(rdata->tp_info.pool,
588 pjsip_endpt_name(endpt),
589 status);
590 PJ_ASSERT_RETURN(w, PJ_ENOMEM);
591
592 pj_list_push_back(&res_hdr_list, w);
593 }
594
595 goto on_return;
596 }
597
598 /* Negotiate with local SDP */
599 if (l_sdp) {
600 pjmedia_sdp_neg *neg;
601
602 /* Local SDP must be valid! */
603 PJ_ASSERT_RETURN((status=pjmedia_sdp_validate(l_sdp))==PJ_SUCCESS,
604 status);
605
606 /* Create SDP negotiator */
607 status = pjmedia_sdp_neg_create_w_remote_offer(
608 rdata->tp_info.pool, l_sdp, sdp, &neg);
609 PJ_ASSERT_RETURN(status == PJ_SUCCESS, status);
610
611 /* Negotiate SDP */
612 status = pjmedia_sdp_neg_negotiate(rdata->tp_info.pool, neg, 0);
613 if (status != PJ_SUCCESS) {
614
615 /* Incompatible media */
616 code = PJSIP_SC_NOT_ACCEPTABLE_HERE;
617 status = PJSIP_ERRNO_FROM_SIP_STATUS(code);
618
619 if (p_tdata) {
620 pjsip_accept_hdr *acc;
621 pjsip_warning_hdr *w;
622
623 /* Add Warning header. */
624 w = pjsip_warning_hdr_create_from_status(
625 rdata->tp_info.pool,
626 pjsip_endpt_name(endpt), status);
627 PJ_ASSERT_RETURN(w, PJ_ENOMEM);
628
629 pj_list_push_back(&res_hdr_list, w);
630
631 /* Add Accept header to response */
632 acc = pjsip_accept_hdr_create(rdata->tp_info.pool);
633 PJ_ASSERT_RETURN(acc, PJ_ENOMEM);
634 acc->values[acc->count++] = pj_str("application/sdp");
635 pj_list_push_back(&res_hdr_list, acc);
636
637 }
638
639 goto on_return;
640 }
641 }
642 }
643
644 /* Check supported methods, see if peer supports UPDATE.
645 * We just assume that peer supports standard INVITE, ACK, CANCEL, and BYE
646 * implicitly by sending this INVITE.
647 */
648 allow = pjsip_msg_find_hdr(msg, PJSIP_H_ALLOW, NULL);
649 if (allow) {
650 unsigned i;
651 const pj_str_t STR_UPDATE = { "UPDATE", 6 };
652
653 for (i=0; i<allow->count; ++i) {
654 if (pj_stricmp(&allow->values[i], &STR_UPDATE)==0)
655 break;
656 }
657
658 if (i != allow->count) {
659 /* UPDATE is present in Allow */
660 rem_option |= PJSIP_INV_SUPPORT_UPDATE;
661 }
662
663 }
664
665 /* Check Supported header */
666 sup_hdr = pjsip_msg_find_hdr(msg, PJSIP_H_SUPPORTED, NULL);
667 if (sup_hdr) {
668 unsigned i;
669 pj_str_t STR_100REL = { "100rel", 6};
670 pj_str_t STR_TIMER = { "timer", 5 };
671
672 for (i=0; i<sup_hdr->count; ++i) {
673 if (pj_stricmp(&sup_hdr->values[i], &STR_100REL)==0)
674 rem_option |= PJSIP_INV_SUPPORT_100REL;
675 else if (pj_stricmp(&sup_hdr->values[i], &STR_TIMER)==0)
676 rem_option |= PJSIP_INV_SUPPORT_TIMER;
677 }
678 }
679
680 /* Check Require header */
681 req_hdr = pjsip_msg_find_hdr(msg, PJSIP_H_REQUIRE, NULL);
682 if (req_hdr) {
683 unsigned i;
684 pj_str_t STR_100REL = { "100rel", 6};
685 pj_str_t STR_TIMER = { "timer", 5 };
686 unsigned unsupp_cnt = 0;
687 pj_str_t unsupp_tags[PJSIP_GENERIC_ARRAY_MAX_COUNT];
688
689 for (i=0; i<req_hdr->count; ++i) {
690 if ((*options & PJSIP_INV_SUPPORT_100REL) &&
691 pj_stricmp(&req_hdr->values[i], &STR_100REL)==0)
692 {
693 rem_option |= PJSIP_INV_REQUIRE_100REL;
694
695 } else if ((*options && PJSIP_INV_SUPPORT_TIMER) &&
696 pj_stricmp(&req_hdr->values[i], &STR_TIMER)==0)
697 {
698 rem_option |= PJSIP_INV_REQUIRE_TIMER;
699
700 } else {
701 /* Unknown/unsupported extension tag! */
702 unsupp_tags[unsupp_cnt++] = req_hdr->values[i];
703 }
704 }
705
706 /* Check if there are required tags that we don't support */
707 if (unsupp_cnt) {
708
709 code = PJSIP_SC_BAD_EXTENSION;
710 status = PJSIP_ERRNO_FROM_SIP_STATUS(code);
711
712 if (p_tdata) {
713 pjsip_unsupported_hdr *unsupp_hdr;
714 const pjsip_hdr *h;
715
716 /* Add Unsupported header. */
717 unsupp_hdr = pjsip_unsupported_hdr_create(rdata->tp_info.pool);
718 PJ_ASSERT_RETURN(unsupp_hdr != NULL, PJ_ENOMEM);
719
720 unsupp_hdr->count = unsupp_cnt;
721 for (i=0; i<unsupp_cnt; ++i)
722 unsupp_hdr->values[i] = unsupp_tags[i];
723
724 pj_list_push_back(&res_hdr_list, unsupp_hdr);
725
726 /* Add Supported header. */
727 h = pjsip_endpt_get_capability(endpt, PJSIP_H_SUPPORTED,
728 NULL);
729 pj_assert(h);
730 if (h) {
731 sup_hdr = pjsip_hdr_clone(rdata->tp_info.pool, h);
732 pj_list_push_back(&res_hdr_list, sup_hdr);
733 }
734 }
735
736 goto on_return;
737 }
738 }
739
740 /* Check if there are local requirements that are not supported
741 * by peer.
742 */
743 if ( ((*options & PJSIP_INV_REQUIRE_100REL)!=0 &&
744 (rem_option & PJSIP_INV_SUPPORT_100REL)==0) ||
745 ((*options & PJSIP_INV_REQUIRE_TIMER)!=0 &&
746 (rem_option & PJSIP_INV_SUPPORT_TIMER)==0))
747 {
748 code = PJSIP_SC_EXTENSION_REQUIRED;
749 status = PJSIP_ERRNO_FROM_SIP_STATUS(code);
750
751 if (p_tdata) {
752 const pjsip_hdr *h;
753
754 /* Add Require header. */
755 req_hdr = pjsip_require_hdr_create(rdata->tp_info.pool);
756 PJ_ASSERT_RETURN(req_hdr != NULL, PJ_ENOMEM);
757
758 if (*options & PJSIP_INV_REQUIRE_100REL)
759 req_hdr->values[req_hdr->count++] = pj_str("100rel");
760
761 if (*options & PJSIP_INV_REQUIRE_TIMER)
762 req_hdr->values[req_hdr->count++] = pj_str("timer");
763
764 pj_list_push_back(&res_hdr_list, req_hdr);
765
766 /* Add Supported header. */
767 h = pjsip_endpt_get_capability(endpt, PJSIP_H_SUPPORTED,
768 NULL);
769 pj_assert(h);
770 if (h) {
771 sup_hdr = pjsip_hdr_clone(rdata->tp_info.pool, h);
772 pj_list_push_back(&res_hdr_list, sup_hdr);
773 }
774
775 }
776
777 goto on_return;
778 }
779
780on_return:
781
782 /* Create response if necessary */
783 if (code != 200 && p_tdata) {
784 pjsip_tx_data *tdata;
785 const pjsip_hdr *h;
786
787 if (dlg) {
788 status = pjsip_dlg_create_response(dlg, rdata, code, NULL,
789 &tdata);
790 } else {
791 status = pjsip_endpt_create_response(endpt, rdata, code, NULL,
792 &tdata);
793 }
794
795 if (status != PJ_SUCCESS)
796 return status;
797
798 /* Add response headers. */
799 h = res_hdr_list.next;
800 while (h != &res_hdr_list) {
801 pjsip_hdr *cloned;
802
803 cloned = pjsip_hdr_clone(tdata->pool, h);
804 PJ_ASSERT_RETURN(cloned, PJ_ENOMEM);
805
806 pjsip_msg_add_hdr(tdata->msg, cloned);
807
808 h = h->next;
809 }
810
811 *p_tdata = tdata;
Benny Prijono70127222006-07-02 14:53:05 +0000812
813 /* Can not return PJ_SUCCESS when response message is produced.
814 * Ref: PROTOS test ~#2490
815 */
816 if (status == PJ_SUCCESS)
817 status = PJSIP_ERRNO_FROM_SIP_STATUS(code);
818
Benny Prijono268ca612006-02-07 12:34:11 +0000819 }
820
821 return status;
822}
823
824/*
Benny Prijono8ad55352006-02-08 11:16:05 +0000825 * Create UAS invite session.
Benny Prijono268ca612006-02-07 12:34:11 +0000826 */
827PJ_DEF(pj_status_t) pjsip_inv_create_uas( pjsip_dialog *dlg,
828 pjsip_rx_data *rdata,
829 const pjmedia_sdp_session *local_sdp,
830 unsigned options,
831 pjsip_inv_session **p_inv)
832{
833 pjsip_inv_session *inv;
Benny Prijonoa66c7152006-02-09 01:26:14 +0000834 struct tsx_inv_data *tsx_inv_data;
Benny Prijono268ca612006-02-07 12:34:11 +0000835 pjsip_msg *msg;
836 pjmedia_sdp_session *rem_sdp = NULL;
837 pj_status_t status;
838
839 /* Verify arguments. */
840 PJ_ASSERT_RETURN(dlg && rdata && p_inv, PJ_EINVAL);
841
842 /* Dialog MUST have been initialised. */
843 PJ_ASSERT_RETURN(pjsip_rdata_get_tsx(rdata) != NULL, PJ_EINVALIDOP);
844
845 msg = rdata->msg_info.msg;
846
847 /* rdata MUST contain INVITE request */
848 PJ_ASSERT_RETURN(msg->type == PJSIP_REQUEST_MSG &&
849 msg->line.req.method.id == PJSIP_INVITE_METHOD,
850 PJ_EINVALIDOP);
851
Benny Prijono8eae8382006-08-10 21:44:26 +0000852 /* Lock dialog */
853 pjsip_dlg_inc_lock(dlg);
854
Benny Prijono268ca612006-02-07 12:34:11 +0000855 /* Normalize options */
856 if (options & PJSIP_INV_REQUIRE_100REL)
857 options |= PJSIP_INV_SUPPORT_100REL;
858
859 if (options & PJSIP_INV_REQUIRE_TIMER)
860 options |= PJSIP_INV_SUPPORT_TIMER;
861
862 /* Create the session */
863 inv = pj_pool_zalloc(dlg->pool, sizeof(pjsip_inv_session));
Benny Prijono8eae8382006-08-10 21:44:26 +0000864 pj_assert(inv != NULL);
Benny Prijono268ca612006-02-07 12:34:11 +0000865
866 inv->pool = dlg->pool;
867 inv->role = PJSIP_ROLE_UAS;
Benny Prijono38998232006-02-08 22:44:25 +0000868 inv->state = PJSIP_INV_STATE_NULL;
Benny Prijono268ca612006-02-07 12:34:11 +0000869 inv->dlg = dlg;
870 inv->options = options;
Benny Prijonod4e0abd2006-03-05 11:53:36 +0000871 inv->notify = PJ_TRUE;
872 inv->cause = 0;
Benny Prijono268ca612006-02-07 12:34:11 +0000873
Benny Prijonoe4f2abb2006-02-10 14:04:05 +0000874 /* Object name will use the same dialog pointer. */
Benny Prijonoed811d72006-03-10 12:57:12 +0000875 pj_ansi_snprintf(inv->obj_name, PJ_MAX_OBJ_NAME, "inv%p", dlg);
Benny Prijonoe4f2abb2006-02-10 14:04:05 +0000876
Benny Prijono268ca612006-02-07 12:34:11 +0000877 /* Parse SDP in message body, if present. */
878 if (msg->body) {
879 pjsip_msg_body *body = msg->body;
880
881 /* Parse and validate SDP */
882 status = pjmedia_sdp_parse(inv->pool, body->data, body->len,
883 &rem_sdp);
884 if (status == PJ_SUCCESS)
885 status = pjmedia_sdp_validate(rem_sdp);
886
Benny Prijono8eae8382006-08-10 21:44:26 +0000887 if (status != PJ_SUCCESS) {
888 pjsip_dlg_dec_lock(dlg);
Benny Prijono268ca612006-02-07 12:34:11 +0000889 return status;
Benny Prijono8eae8382006-08-10 21:44:26 +0000890 }
Benny Prijono268ca612006-02-07 12:34:11 +0000891 }
892
893 /* Create negotiator. */
894 if (rem_sdp) {
895 status = pjmedia_sdp_neg_create_w_remote_offer(inv->pool, local_sdp,
896 rem_sdp, &inv->neg);
897
898 } else if (local_sdp) {
899 status = pjmedia_sdp_neg_create_w_local_offer(inv->pool, local_sdp,
900 &inv->neg);
901 } else {
Benny Prijono95196582006-02-09 00:13:40 +0000902 status = PJ_SUCCESS;
Benny Prijono268ca612006-02-07 12:34:11 +0000903 }
904
Benny Prijono8eae8382006-08-10 21:44:26 +0000905 if (status != PJ_SUCCESS) {
906 pjsip_dlg_dec_lock(dlg);
Benny Prijono268ca612006-02-07 12:34:11 +0000907 return status;
Benny Prijono8eae8382006-08-10 21:44:26 +0000908 }
Benny Prijono268ca612006-02-07 12:34:11 +0000909
910 /* Register invite as dialog usage. */
911 status = pjsip_dlg_add_usage(dlg, &mod_inv.mod, inv);
Benny Prijono8eae8382006-08-10 21:44:26 +0000912 if (status != PJ_SUCCESS) {
913 pjsip_dlg_dec_lock(dlg);
Benny Prijono268ca612006-02-07 12:34:11 +0000914 return status;
Benny Prijono8eae8382006-08-10 21:44:26 +0000915 }
Benny Prijono268ca612006-02-07 12:34:11 +0000916
917 /* Increment session in the dialog. */
Benny Prijonoe4f2abb2006-02-10 14:04:05 +0000918 pjsip_dlg_inc_session(dlg, &mod_inv.mod);
Benny Prijono268ca612006-02-07 12:34:11 +0000919
920 /* Save the invite transaction. */
921 inv->invite_tsx = pjsip_rdata_get_tsx(rdata);
Benny Prijonoa66c7152006-02-09 01:26:14 +0000922
923 /* Attach our data to the transaction. */
924 tsx_inv_data = pj_pool_zalloc(inv->invite_tsx->pool,
925 sizeof(struct tsx_inv_data));
926 tsx_inv_data->inv = inv;
927 inv->invite_tsx->mod_data[mod_inv.mod.id] = tsx_inv_data;
Benny Prijono268ca612006-02-07 12:34:11 +0000928
929 /* Done */
Benny Prijono8eae8382006-08-10 21:44:26 +0000930 pjsip_dlg_dec_lock(dlg);
Benny Prijono268ca612006-02-07 12:34:11 +0000931 *p_inv = inv;
Benny Prijonoe4f2abb2006-02-10 14:04:05 +0000932
933 PJ_LOG(5,(inv->obj_name, "UAS invite session created for dialog %s",
934 dlg->obj_name));
935
Benny Prijono268ca612006-02-07 12:34:11 +0000936 return PJ_SUCCESS;
937}
938
Benny Prijonod4e0abd2006-03-05 11:53:36 +0000939/*
940 * Forcefully terminate the session.
941 */
942PJ_DEF(pj_status_t) pjsip_inv_terminate( pjsip_inv_session *inv,
943 int st_code,
944 pj_bool_t notify)
945{
946 PJ_ASSERT_RETURN(inv, PJ_EINVAL);
947
948 /* Lock dialog. */
949 pjsip_dlg_inc_lock(inv->dlg);
950
951 /* Set callback notify flag. */
952 inv->notify = notify;
953
954 /* If there's pending transaction, terminate the transaction.
955 * This may subsequently set the INVITE session state to
956 * disconnected.
957 */
958 if (inv->invite_tsx &&
959 inv->invite_tsx->state <= PJSIP_TSX_STATE_COMPLETED)
960 {
961 pjsip_tsx_terminate(inv->invite_tsx, st_code);
962
963 }
964
965 /* Set cause. */
Benny Prijono0b6340c2006-06-13 22:21:23 +0000966 inv_set_cause(inv, st_code, NULL);
Benny Prijonod4e0abd2006-03-05 11:53:36 +0000967
968 /* Forcefully terminate the session if state is not DISCONNECTED */
969 if (inv->state != PJSIP_INV_STATE_DISCONNECTED) {
970 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, NULL);
971 }
972
973 /* Done.
974 * The dec_lock() below will actually destroys the dialog if it
975 * has no other session.
976 */
977 pjsip_dlg_dec_lock(inv->dlg);
978
979 return PJ_SUCCESS;
980}
981
982
Benny Prijono268ca612006-02-07 12:34:11 +0000983static void *clone_sdp(pj_pool_t *pool, const void *data, unsigned len)
984{
985 PJ_UNUSED_ARG(len);
986 return pjmedia_sdp_session_clone(pool, data);
987}
988
989static int print_sdp(pjsip_msg_body *body, char *buf, pj_size_t len)
990{
991 return pjmedia_sdp_print(body->data, buf, len);
992}
993
Benny Prijono56315612006-07-18 14:39:40 +0000994
995PJ_DEF(pj_status_t) pjsip_create_sdp_body( pj_pool_t *pool,
996 pjmedia_sdp_session *sdp,
997 pjsip_msg_body **p_body)
998{
999 const pj_str_t STR_APPLICATION = { "application", 11};
1000 const pj_str_t STR_SDP = { "sdp", 3 };
1001 pjsip_msg_body *body;
1002
1003 body = pj_pool_zalloc(pool, sizeof(pjsip_msg_body));
1004 PJ_ASSERT_RETURN(body != NULL, PJ_ENOMEM);
1005
1006 body->content_type.type = STR_APPLICATION;
1007 body->content_type.subtype = STR_SDP;
1008 body->data = sdp;
1009 body->len = 0;
1010 body->clone_data = &clone_sdp;
1011 body->print_body = &print_sdp;
1012
1013 *p_body = body;
1014
1015 return PJ_SUCCESS;
1016}
1017
Benny Prijono268ca612006-02-07 12:34:11 +00001018static pjsip_msg_body *create_sdp_body(pj_pool_t *pool,
1019 const pjmedia_sdp_session *c_sdp)
1020{
1021 pjsip_msg_body *body;
Benny Prijono56315612006-07-18 14:39:40 +00001022 pj_status_t status;
Benny Prijono268ca612006-02-07 12:34:11 +00001023
Benny Prijono56315612006-07-18 14:39:40 +00001024 status = pjsip_create_sdp_body(pool,
1025 pjmedia_sdp_session_clone(pool, c_sdp),
1026 &body);
Benny Prijono268ca612006-02-07 12:34:11 +00001027
Benny Prijono56315612006-07-18 14:39:40 +00001028 if (status != PJ_SUCCESS)
1029 return NULL;
Benny Prijono268ca612006-02-07 12:34:11 +00001030
1031 return body;
1032}
1033
1034/*
1035 * Create initial INVITE request.
1036 */
1037PJ_DEF(pj_status_t) pjsip_inv_invite( pjsip_inv_session *inv,
1038 pjsip_tx_data **p_tdata )
1039{
1040 pjsip_tx_data *tdata;
1041 const pjsip_hdr *hdr;
Benny Prijono26ff9062006-02-21 23:47:00 +00001042 pj_bool_t has_sdp;
Benny Prijono268ca612006-02-07 12:34:11 +00001043 pj_status_t status;
1044
1045 /* Verify arguments. */
1046 PJ_ASSERT_RETURN(inv && p_tdata, PJ_EINVAL);
1047
Benny Prijono26ff9062006-02-21 23:47:00 +00001048 /* State MUST be NULL or CONFIRMED. */
1049 PJ_ASSERT_RETURN(inv->state == PJSIP_INV_STATE_NULL ||
1050 inv->state == PJSIP_INV_STATE_CONFIRMED,
1051 PJ_EINVALIDOP);
Benny Prijono268ca612006-02-07 12:34:11 +00001052
Benny Prijono64f851e2006-02-23 13:49:28 +00001053 /* Lock dialog. */
1054 pjsip_dlg_inc_lock(inv->dlg);
1055
Benny Prijono268ca612006-02-07 12:34:11 +00001056 /* Create the INVITE request. */
1057 status = pjsip_dlg_create_request(inv->dlg, &pjsip_invite_method, -1,
1058 &tdata);
1059 if (status != PJ_SUCCESS)
Benny Prijono64f851e2006-02-23 13:49:28 +00001060 goto on_return;
1061
Benny Prijono268ca612006-02-07 12:34:11 +00001062
Benny Prijono26ff9062006-02-21 23:47:00 +00001063 /* If this is the first INVITE, then copy the headers from inv_hdr.
1064 * These are the headers parsed from the request URI when the
1065 * dialog was created.
1066 */
1067 if (inv->state == PJSIP_INV_STATE_NULL) {
1068 hdr = inv->dlg->inv_hdr.next;
1069
1070 while (hdr != &inv->dlg->inv_hdr) {
1071 pjsip_msg_add_hdr(tdata->msg,
1072 pjsip_hdr_shallow_clone(tdata->pool, hdr));
1073 hdr = hdr->next;
1074 }
1075 }
1076
1077 /* See if we have SDP to send. */
1078 if (inv->neg) {
1079 pjmedia_sdp_neg_state neg_state;
1080
1081 neg_state = pjmedia_sdp_neg_get_state(inv->neg);
1082
1083 has_sdp = (neg_state == PJMEDIA_SDP_NEG_STATE_LOCAL_OFFER ||
1084 (neg_state == PJMEDIA_SDP_NEG_STATE_WAIT_NEGO &&
1085 pjmedia_sdp_neg_has_local_answer(inv->neg)));
1086
1087
1088 } else {
1089 has_sdp = PJ_FALSE;
1090 }
1091
Benny Prijono268ca612006-02-07 12:34:11 +00001092 /* Add SDP, if any. */
Benny Prijono26ff9062006-02-21 23:47:00 +00001093 if (has_sdp) {
Benny Prijono268ca612006-02-07 12:34:11 +00001094 const pjmedia_sdp_session *offer;
1095
1096 status = pjmedia_sdp_neg_get_neg_local(inv->neg, &offer);
1097 if (status != PJ_SUCCESS)
Benny Prijono64f851e2006-02-23 13:49:28 +00001098 goto on_return;
Benny Prijono268ca612006-02-07 12:34:11 +00001099
1100 tdata->msg->body = create_sdp_body(tdata->pool, offer);
1101 }
1102
1103 /* Add Allow header. */
1104 hdr = pjsip_endpt_get_capability(inv->dlg->endpt, PJSIP_H_ALLOW, NULL);
1105 if (hdr) {
1106 pjsip_msg_add_hdr(tdata->msg,
1107 pjsip_hdr_shallow_clone(tdata->pool, hdr));
1108 }
1109
1110 /* Add Supported header */
1111 hdr = pjsip_endpt_get_capability(inv->dlg->endpt, PJSIP_H_SUPPORTED, NULL);
1112 if (hdr) {
1113 pjsip_msg_add_hdr(tdata->msg,
1114 pjsip_hdr_shallow_clone(tdata->pool, hdr));
1115 }
1116
1117 /* Add Require header. */
1118 PJ_TODO(INVITE_ADD_REQUIRE_HEADER);
1119
1120 /* Done. */
1121 *p_tdata = tdata;
1122
Benny Prijono64f851e2006-02-23 13:49:28 +00001123
1124on_return:
1125 pjsip_dlg_dec_lock(inv->dlg);
1126 return status;
Benny Prijono268ca612006-02-07 12:34:11 +00001127}
1128
1129
1130/*
Benny Prijono95196582006-02-09 00:13:40 +00001131 * Negotiate SDP.
1132 */
1133static pj_status_t inv_negotiate_sdp( pjsip_inv_session *inv )
1134{
1135 pj_status_t status;
1136
1137 PJ_ASSERT_RETURN(pjmedia_sdp_neg_get_state(inv->neg) ==
1138 PJMEDIA_SDP_NEG_STATE_WAIT_NEGO,
1139 PJMEDIA_SDPNEG_EINSTATE);
1140
1141 status = pjmedia_sdp_neg_negotiate(inv->pool, inv->neg, 0);
1142
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00001143 PJ_LOG(5,(inv->obj_name, "SDP negotiation done, status=%d", status));
1144
Benny Prijonod4e0abd2006-03-05 11:53:36 +00001145 if (mod_inv.cb.on_media_update && inv->notify)
Benny Prijono95196582006-02-09 00:13:40 +00001146 (*mod_inv.cb.on_media_update)(inv, status);
1147
1148 return status;
1149}
1150
1151/*
Benny Prijonoa66c7152006-02-09 01:26:14 +00001152 * Check in incoming message for SDP offer/answer.
1153 */
Benny Prijono26ff9062006-02-21 23:47:00 +00001154static pj_status_t inv_check_sdp_in_incoming_msg( pjsip_inv_session *inv,
1155 pjsip_transaction *tsx,
1156 pjsip_rx_data *rdata)
Benny Prijonoa66c7152006-02-09 01:26:14 +00001157{
1158 struct tsx_inv_data *tsx_inv_data;
1159 static const pj_str_t str_application = { "application", 11 };
1160 static const pj_str_t str_sdp = { "sdp", 3 };
1161 pj_status_t status;
1162 pjsip_msg *msg;
1163 pjmedia_sdp_session *sdp;
1164
1165 /* Get/attach invite session's transaction data */
1166 tsx_inv_data = tsx->mod_data[mod_inv.mod.id];
1167 if (tsx_inv_data == NULL) {
1168 tsx_inv_data = pj_pool_zalloc(tsx->pool, sizeof(struct tsx_inv_data));
1169 tsx_inv_data->inv = inv;
1170 tsx->mod_data[mod_inv.mod.id] = tsx_inv_data;
1171 }
1172
1173 /* MUST NOT do multiple SDP offer/answer in a single transaction.
1174 */
1175
1176 if (tsx_inv_data->sdp_done)
Benny Prijono26ff9062006-02-21 23:47:00 +00001177 return PJ_SUCCESS;
Benny Prijonoa66c7152006-02-09 01:26:14 +00001178
1179 /* Check if SDP is present in the message. */
1180
1181 msg = rdata->msg_info.msg;
1182 if (msg->body == NULL) {
1183 /* Message doesn't have body. */
Benny Prijono26ff9062006-02-21 23:47:00 +00001184 return PJ_SUCCESS;
Benny Prijonoa66c7152006-02-09 01:26:14 +00001185 }
1186
1187 if (pj_stricmp(&msg->body->content_type.type, &str_application) ||
1188 pj_stricmp(&msg->body->content_type.subtype, &str_sdp))
1189 {
1190 /* Message body is not "application/sdp" */
Benny Prijono26ff9062006-02-21 23:47:00 +00001191 return PJMEDIA_SDP_EINSDP;
Benny Prijonoa66c7152006-02-09 01:26:14 +00001192 }
1193
1194 /* Parse the SDP body. */
1195
1196 status = pjmedia_sdp_parse(rdata->tp_info.pool, msg->body->data,
1197 msg->body->len, &sdp);
1198 if (status != PJ_SUCCESS) {
1199 char errmsg[PJ_ERR_MSG_SIZE];
1200 pj_strerror(status, errmsg, sizeof(errmsg));
1201 PJ_LOG(4,(THIS_FILE, "Error parsing SDP in %s: %s",
1202 pjsip_rx_data_get_info(rdata), errmsg));
Benny Prijono26ff9062006-02-21 23:47:00 +00001203 return PJMEDIA_SDP_EINSDP;
Benny Prijonoa66c7152006-02-09 01:26:14 +00001204 }
1205
1206 /* The SDP can be an offer or answer, depending on negotiator's state */
1207
1208 if (inv->neg == NULL ||
1209 pjmedia_sdp_neg_get_state(inv->neg) == PJMEDIA_SDP_NEG_STATE_DONE)
1210 {
1211
1212 /* This is an offer. */
1213
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00001214 PJ_LOG(5,(inv->obj_name, "Got SDP offer in %s",
1215 pjsip_rx_data_get_info(rdata)));
1216
Benny Prijonoa66c7152006-02-09 01:26:14 +00001217 if (inv->neg == NULL) {
1218 status=pjmedia_sdp_neg_create_w_remote_offer(inv->pool, NULL,
1219 sdp, &inv->neg);
1220 } else {
1221 status=pjmedia_sdp_neg_set_remote_offer(inv->pool, inv->neg, sdp);
1222 }
1223
1224 if (status != PJ_SUCCESS) {
1225 char errmsg[PJ_ERR_MSG_SIZE];
1226 pj_strerror(status, errmsg, sizeof(errmsg));
1227 PJ_LOG(4,(THIS_FILE, "Error processing SDP offer in %s: %s",
1228 pjsip_rx_data_get_info(rdata), errmsg));
Benny Prijono26ff9062006-02-21 23:47:00 +00001229 return PJMEDIA_SDP_EINSDP;
Benny Prijonoa66c7152006-02-09 01:26:14 +00001230 }
1231
1232 /* Inform application about remote offer. */
1233
Benny Prijonod4e0abd2006-03-05 11:53:36 +00001234 if (mod_inv.cb.on_rx_offer && inv->notify) {
Benny Prijono26ff9062006-02-21 23:47:00 +00001235
1236 (*mod_inv.cb.on_rx_offer)(inv, sdp);
1237
1238 }
Benny Prijonoa66c7152006-02-09 01:26:14 +00001239
1240 } else if (pjmedia_sdp_neg_get_state(inv->neg) ==
1241 PJMEDIA_SDP_NEG_STATE_LOCAL_OFFER)
1242 {
1243
1244 /* This is an answer.
1245 * Process and negotiate remote answer.
1246 */
1247
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00001248 PJ_LOG(5,(inv->obj_name, "Got SDP answer in %s",
1249 pjsip_rx_data_get_info(rdata)));
1250
Benny Prijonoa66c7152006-02-09 01:26:14 +00001251 status = pjmedia_sdp_neg_set_remote_answer(inv->pool, inv->neg, sdp);
1252
1253 if (status != PJ_SUCCESS) {
1254 char errmsg[PJ_ERR_MSG_SIZE];
1255 pj_strerror(status, errmsg, sizeof(errmsg));
1256 PJ_LOG(4,(THIS_FILE, "Error processing SDP answer in %s: %s",
1257 pjsip_rx_data_get_info(rdata), errmsg));
Benny Prijono26ff9062006-02-21 23:47:00 +00001258 return PJMEDIA_SDP_EINSDP;
Benny Prijonoa66c7152006-02-09 01:26:14 +00001259 }
1260
1261 /* Negotiate SDP */
1262
1263 inv_negotiate_sdp(inv);
1264
1265 /* Mark this transaction has having SDP offer/answer done. */
1266
1267 tsx_inv_data->sdp_done = 1;
1268
1269 } else {
1270
1271 PJ_LOG(5,(THIS_FILE, "Ignored SDP in %s: negotiator state is %s",
1272 pjsip_rx_data_get_info(rdata),
1273 pjmedia_sdp_neg_state_str(pjmedia_sdp_neg_get_state(inv->neg))));
1274 }
1275
Benny Prijono26ff9062006-02-21 23:47:00 +00001276 return PJ_SUCCESS;
Benny Prijonoa66c7152006-02-09 01:26:14 +00001277}
1278
1279
Benny Prijono26ff9062006-02-21 23:47:00 +00001280/*
1281 * Process INVITE answer, for both initial and subsequent re-INVITE
1282 */
1283static pj_status_t process_answer( pjsip_inv_session *inv,
1284 int st_code,
Benny Prijono64f851e2006-02-23 13:49:28 +00001285 pjsip_tx_data *tdata,
1286 const pjmedia_sdp_session *local_sdp)
Benny Prijono26ff9062006-02-21 23:47:00 +00001287{
1288 pj_status_t status;
Benny Prijonoab7399b2006-02-27 00:40:31 +00001289 const pjmedia_sdp_session *sdp = NULL;
Benny Prijono26ff9062006-02-21 23:47:00 +00001290
Benny Prijono64f851e2006-02-23 13:49:28 +00001291 /* If local_sdp is specified, then we MUST NOT have answered the
1292 * offer before.
Benny Prijono26ff9062006-02-21 23:47:00 +00001293 */
Benny Prijono64f851e2006-02-23 13:49:28 +00001294 if (local_sdp && (st_code/100==1 || st_code/100==2)) {
1295
1296 if (inv->neg == NULL) {
1297 status = pjmedia_sdp_neg_create_w_local_offer(inv->pool, local_sdp,
1298 &inv->neg);
1299 } else if (pjmedia_sdp_neg_get_state(inv->neg)==
1300 PJMEDIA_SDP_NEG_STATE_REMOTE_OFFER)
1301 {
1302 status = pjmedia_sdp_neg_set_local_answer(inv->pool, inv->neg,
1303 local_sdp);
1304 } else {
1305
1306 /* Can not specify local SDP at this state. */
1307 pj_assert(0);
1308 status = PJMEDIA_SDPNEG_EINSTATE;
1309 }
1310
1311 if (status != PJ_SUCCESS)
1312 return status;
1313
1314 }
1315
1316
1317 /* If SDP negotiator is ready, start negotiation. */
1318 if (st_code/100==2 || (st_code/10==18 && st_code!=180)) {
Benny Prijono26ff9062006-02-21 23:47:00 +00001319
1320 pjmedia_sdp_neg_state neg_state;
1321
Benny Prijono64f851e2006-02-23 13:49:28 +00001322 /* Start nego when appropriate. */
Benny Prijono26ff9062006-02-21 23:47:00 +00001323 neg_state = inv->neg ? pjmedia_sdp_neg_get_state(inv->neg) :
1324 PJMEDIA_SDP_NEG_STATE_NULL;
1325
1326 if (neg_state == PJMEDIA_SDP_NEG_STATE_LOCAL_OFFER) {
1327
1328 status = pjmedia_sdp_neg_get_neg_local(inv->neg, &sdp);
1329
1330 } else if (neg_state == PJMEDIA_SDP_NEG_STATE_WAIT_NEGO &&
1331 pjmedia_sdp_neg_has_local_answer(inv->neg) )
1332 {
1333
1334 status = inv_negotiate_sdp(inv);
1335 if (status != PJ_SUCCESS)
1336 return status;
1337
1338 status = pjmedia_sdp_neg_get_active_local(inv->neg, &sdp);
1339 }
Benny Prijono26ff9062006-02-21 23:47:00 +00001340 }
1341
Benny Prijono64f851e2006-02-23 13:49:28 +00001342 /* Include SDP when it's available for 2xx and 18x (but not 180) response.
Benny Prijono26ff9062006-02-21 23:47:00 +00001343 * Subsequent response will include this SDP.
1344 */
1345 if (sdp) {
1346 tdata->msg->body = create_sdp_body(tdata->pool, sdp);
1347 }
1348
Benny Prijono26ff9062006-02-21 23:47:00 +00001349
1350 return PJ_SUCCESS;
1351}
1352
Benny Prijonoa66c7152006-02-09 01:26:14 +00001353
1354/*
Benny Prijono64f851e2006-02-23 13:49:28 +00001355 * Create first response to INVITE
1356 */
1357PJ_DEF(pj_status_t) pjsip_inv_initial_answer( pjsip_inv_session *inv,
1358 pjsip_rx_data *rdata,
1359 int st_code,
1360 const pj_str_t *st_text,
1361 const pjmedia_sdp_session *sdp,
1362 pjsip_tx_data **p_tdata)
1363{
1364 pjsip_tx_data *tdata;
1365 pj_status_t status;
1366
1367 /* Verify arguments. */
1368 PJ_ASSERT_RETURN(inv && p_tdata, PJ_EINVAL);
1369
1370 /* Must have INVITE transaction. */
1371 PJ_ASSERT_RETURN(inv->invite_tsx, PJ_EBUG);
1372
1373 pjsip_dlg_inc_lock(inv->dlg);
1374
1375 /* Create response */
1376 status = pjsip_dlg_create_response(inv->dlg, rdata, st_code, st_text,
1377 &tdata);
1378 if (status != PJ_SUCCESS)
1379 goto on_return;
1380
1381 /* Process SDP in answer */
1382 status = process_answer(inv, st_code, tdata, sdp);
1383 if (status != PJ_SUCCESS) {
1384 pjsip_tx_data_dec_ref(tdata);
1385 goto on_return;
1386 }
1387
1388 *p_tdata = tdata;
1389
1390on_return:
1391 pjsip_dlg_dec_lock(inv->dlg);
1392 return status;
1393}
1394
1395
1396/*
Benny Prijono26ff9062006-02-21 23:47:00 +00001397 * Answer initial INVITE
1398 * Re-INVITE will be answered automatically, and will not use this function.
Benny Prijono268ca612006-02-07 12:34:11 +00001399 */
1400PJ_DEF(pj_status_t) pjsip_inv_answer( pjsip_inv_session *inv,
1401 int st_code,
1402 const pj_str_t *st_text,
1403 const pjmedia_sdp_session *local_sdp,
1404 pjsip_tx_data **p_tdata )
1405{
1406 pjsip_tx_data *last_res;
1407 pj_status_t status;
1408
1409 /* Verify arguments. */
1410 PJ_ASSERT_RETURN(inv && p_tdata, PJ_EINVAL);
1411
1412 /* Must have INVITE transaction. */
1413 PJ_ASSERT_RETURN(inv->invite_tsx, PJ_EBUG);
1414
1415 /* INVITE transaction MUST have transmitted a response (e.g. 100) */
1416 PJ_ASSERT_RETURN(inv->invite_tsx->last_tx, PJ_EINVALIDOP);
1417
Benny Prijono64f851e2006-02-23 13:49:28 +00001418 pjsip_dlg_inc_lock(inv->dlg);
Benny Prijono268ca612006-02-07 12:34:11 +00001419
1420 /* Modify last response. */
Benny Prijono26ff9062006-02-21 23:47:00 +00001421 last_res = inv->invite_tsx->last_tx;
Benny Prijono268ca612006-02-07 12:34:11 +00001422 status = pjsip_dlg_modify_response(inv->dlg, last_res, st_code, st_text);
1423 if (status != PJ_SUCCESS)
Benny Prijono64f851e2006-02-23 13:49:28 +00001424 goto on_return;
Benny Prijono268ca612006-02-07 12:34:11 +00001425
Benny Prijono268ca612006-02-07 12:34:11 +00001426
Benny Prijono26ff9062006-02-21 23:47:00 +00001427 /* Process SDP in answer */
Benny Prijono64f851e2006-02-23 13:49:28 +00001428 status = process_answer(inv, st_code, last_res, local_sdp);
1429 if (status != PJ_SUCCESS) {
1430 pjsip_tx_data_dec_ref(last_res);
1431 goto on_return;
1432 }
Benny Prijono268ca612006-02-07 12:34:11 +00001433
Benny Prijono268ca612006-02-07 12:34:11 +00001434
1435 *p_tdata = last_res;
1436
Benny Prijono64f851e2006-02-23 13:49:28 +00001437on_return:
1438 pjsip_dlg_dec_lock(inv->dlg);
1439 return status;
Benny Prijono268ca612006-02-07 12:34:11 +00001440}
1441
1442
1443/*
Benny Prijono26ff9062006-02-21 23:47:00 +00001444 * Set SDP answer.
1445 */
1446PJ_DEF(pj_status_t) pjsip_inv_set_sdp_answer( pjsip_inv_session *inv,
1447 const pjmedia_sdp_session *sdp )
1448{
1449 pj_status_t status;
1450
1451 PJ_ASSERT_RETURN(inv && sdp, PJ_EINVAL);
1452
1453 pjsip_dlg_inc_lock(inv->dlg);
1454 status = pjmedia_sdp_neg_set_local_answer( inv->pool, inv->neg, sdp);
1455 pjsip_dlg_dec_lock(inv->dlg);
1456
1457 return status;
1458}
1459
1460
1461/*
Benny Prijono268ca612006-02-07 12:34:11 +00001462 * End session.
1463 */
1464PJ_DEF(pj_status_t) pjsip_inv_end_session( pjsip_inv_session *inv,
1465 int st_code,
1466 const pj_str_t *st_text,
1467 pjsip_tx_data **p_tdata )
1468{
1469 pjsip_tx_data *tdata;
1470 pj_status_t status;
1471
1472 /* Verify arguments. */
1473 PJ_ASSERT_RETURN(inv && p_tdata, PJ_EINVAL);
1474
Benny Prijonod4e0abd2006-03-05 11:53:36 +00001475 /* Set cause code. */
Benny Prijono0b6340c2006-06-13 22:21:23 +00001476 inv_set_cause(inv, st_code, st_text);
Benny Prijonod4e0abd2006-03-05 11:53:36 +00001477
Benny Prijono268ca612006-02-07 12:34:11 +00001478 /* Create appropriate message. */
1479 switch (inv->state) {
1480 case PJSIP_INV_STATE_CALLING:
1481 case PJSIP_INV_STATE_EARLY:
1482 case PJSIP_INV_STATE_INCOMING:
1483
1484 if (inv->role == PJSIP_ROLE_UAC) {
1485
1486 /* For UAC when session has not been confirmed, create CANCEL. */
1487
1488 /* MUST have the original UAC INVITE transaction. */
1489 PJ_ASSERT_RETURN(inv->invite_tsx != NULL, PJ_EBUG);
1490
1491 /* But CANCEL should only be called when we have received a
1492 * provisional response. If we haven't received any responses,
1493 * just destroy the transaction.
1494 */
1495 if (inv->invite_tsx->status_code < 100) {
1496
1497 pjsip_tsx_terminate(inv->invite_tsx, 487);
Benny Prijonofccab712006-02-22 22:23:22 +00001498 *p_tdata = NULL;
1499 return PJ_SUCCESS;
Benny Prijono268ca612006-02-07 12:34:11 +00001500 }
1501
1502 /* The CSeq here assumes that the dialog is started with an
1503 * INVITE session. This may not be correct; dialog can be
1504 * started as SUBSCRIBE session.
1505 * So fix this!
1506 */
1507 status = pjsip_endpt_create_cancel(inv->dlg->endpt,
1508 inv->invite_tsx->last_tx,
1509 &tdata);
1510
1511 } else {
1512
1513 /* For UAS, send a final response. */
1514 tdata = inv->invite_tsx->last_tx;
1515 PJ_ASSERT_RETURN(tdata != NULL, PJ_EINVALIDOP);
1516
Benny Prijono26ff9062006-02-21 23:47:00 +00001517 //status = pjsip_dlg_modify_response(inv->dlg, tdata, st_code,
1518 // st_text);
1519 status = pjsip_inv_answer(inv, st_code, st_text, NULL, &tdata);
Benny Prijono268ca612006-02-07 12:34:11 +00001520 }
1521 break;
1522
1523 case PJSIP_INV_STATE_CONNECTING:
1524 case PJSIP_INV_STATE_CONFIRMED:
1525 /* For established dialog, send BYE */
1526 status = pjsip_dlg_create_request(inv->dlg, &pjsip_bye_method, -1,
1527 &tdata);
1528 break;
1529
1530 case PJSIP_INV_STATE_DISCONNECTED:
Benny Prijono268ca612006-02-07 12:34:11 +00001531 /* No need to do anything. */
Benny Prijonod4e0abd2006-03-05 11:53:36 +00001532 return PJSIP_ESESSIONTERMINATED;
Benny Prijono268ca612006-02-07 12:34:11 +00001533
1534 default:
1535 pj_assert("!Invalid operation!");
1536 return PJ_EINVALIDOP;
1537 }
1538
1539 if (status != PJ_SUCCESS)
1540 return status;
1541
1542
1543 /* Done */
1544
1545 *p_tdata = tdata;
1546
1547 return PJ_SUCCESS;
1548}
1549
1550
1551/*
1552 * Create re-INVITE.
1553 */
1554PJ_DEF(pj_status_t) pjsip_inv_reinvite( pjsip_inv_session *inv,
1555 const pj_str_t *new_contact,
1556 const pjmedia_sdp_session *new_offer,
1557 pjsip_tx_data **p_tdata )
1558{
Benny Prijono26ff9062006-02-21 23:47:00 +00001559 pj_status_t status;
1560 pjsip_contact_hdr *contact_hdr = NULL;
Benny Prijono268ca612006-02-07 12:34:11 +00001561
Benny Prijono26ff9062006-02-21 23:47:00 +00001562 /* Check arguments. */
1563 PJ_ASSERT_RETURN(inv && p_tdata, PJ_EINVAL);
1564
1565 /* Must NOT have a pending INVITE transaction */
1566 PJ_ASSERT_RETURN(inv->invite_tsx==NULL, PJ_EINVALIDOP);
1567
1568
1569 pjsip_dlg_inc_lock(inv->dlg);
1570
1571 if (new_contact) {
1572 pj_str_t tmp;
1573 const pj_str_t STR_CONTACT = { "Contact", 7 };
1574
1575 pj_strdup_with_null(inv->dlg->pool, &tmp, new_contact);
1576 contact_hdr = pjsip_parse_hdr(inv->dlg->pool, &STR_CONTACT,
1577 tmp.ptr, tmp.slen, NULL);
1578 if (!contact_hdr) {
1579 status = PJSIP_EINVALIDURI;
1580 goto on_return;
1581 }
1582 }
1583
1584
1585 if (new_offer) {
1586 if (!inv->neg) {
1587 status = pjmedia_sdp_neg_create_w_local_offer(inv->pool, new_offer,
1588 &inv->neg);
1589 if (status != PJ_SUCCESS)
1590 goto on_return;
1591
1592 } else switch (pjmedia_sdp_neg_get_state(inv->neg)) {
1593
1594 case PJMEDIA_SDP_NEG_STATE_NULL:
1595 pj_assert(!"Unexpected SDP neg state NULL");
1596 status = PJ_EBUG;
1597 goto on_return;
1598
1599 case PJMEDIA_SDP_NEG_STATE_LOCAL_OFFER:
1600 PJ_LOG(4,(inv->obj_name,
1601 "pjsip_inv_reinvite: already have an offer, new "
1602 "offer is ignored"));
1603 break;
1604
1605 case PJMEDIA_SDP_NEG_STATE_REMOTE_OFFER:
1606 status = pjmedia_sdp_neg_set_local_answer(inv->pool, inv->neg,
1607 new_offer);
1608 if (status != PJ_SUCCESS)
1609 goto on_return;
1610 break;
1611
1612 case PJMEDIA_SDP_NEG_STATE_WAIT_NEGO:
1613 PJ_LOG(4,(inv->obj_name,
1614 "pjsip_inv_reinvite: SDP in WAIT_NEGO state, new "
1615 "offer is ignored"));
1616 break;
1617
1618 case PJMEDIA_SDP_NEG_STATE_DONE:
1619 status = pjmedia_sdp_neg_modify_local_offer(inv->pool,inv->neg,
1620 new_offer);
1621 if (status != PJ_SUCCESS)
1622 goto on_return;
1623 break;
1624 }
1625 }
1626
1627 if (contact_hdr)
1628 inv->dlg->local.contact = contact_hdr;
1629
1630 status = pjsip_inv_invite(inv, p_tdata);
1631
1632on_return:
1633 pjsip_dlg_dec_lock(inv->dlg);
1634 return status;
Benny Prijono268ca612006-02-07 12:34:11 +00001635}
1636
1637/*
1638 * Create UPDATE.
1639 */
1640PJ_DEF(pj_status_t) pjsip_inv_update ( pjsip_inv_session *inv,
1641 const pj_str_t *new_contact,
1642 const pjmedia_sdp_session *new_offer,
1643 pjsip_tx_data **p_tdata )
1644{
1645 PJ_UNUSED_ARG(inv);
1646 PJ_UNUSED_ARG(new_contact);
1647 PJ_UNUSED_ARG(new_offer);
1648 PJ_UNUSED_ARG(p_tdata);
1649
1650 PJ_TODO(CREATE_UPDATE_REQUEST);
1651 return PJ_ENOTSUP;
1652}
1653
1654/*
1655 * Send a request or response message.
1656 */
1657PJ_DEF(pj_status_t) pjsip_inv_send_msg( pjsip_inv_session *inv,
Benny Prijonoe8b0d3b2006-03-17 17:57:52 +00001658 pjsip_tx_data *tdata)
Benny Prijono268ca612006-02-07 12:34:11 +00001659{
1660 pj_status_t status;
1661
1662 /* Verify arguments. */
1663 PJ_ASSERT_RETURN(inv && tdata, PJ_EINVAL);
1664
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00001665 PJ_LOG(5,(inv->obj_name, "Sending %s",
1666 pjsip_tx_data_get_info(tdata)));
1667
Benny Prijono268ca612006-02-07 12:34:11 +00001668 if (tdata->msg->type == PJSIP_REQUEST_MSG) {
Benny Prijonoa66c7152006-02-09 01:26:14 +00001669 struct tsx_inv_data *tsx_inv_data;
Benny Prijono268ca612006-02-07 12:34:11 +00001670
Benny Prijono64158af2006-04-04 11:06:34 +00001671 pjsip_dlg_inc_lock(inv->dlg);
Benny Prijono268ca612006-02-07 12:34:11 +00001672
Benny Prijono64158af2006-04-04 11:06:34 +00001673 tsx_inv_data = pj_pool_zalloc(inv->pool, sizeof(struct tsx_inv_data));
Benny Prijonoa66c7152006-02-09 01:26:14 +00001674 tsx_inv_data->inv = inv;
1675
Benny Prijono64158af2006-04-04 11:06:34 +00001676 pjsip_dlg_dec_lock(inv->dlg);
1677
1678 status = pjsip_dlg_send_request(inv->dlg, tdata, mod_inv.mod.id,
1679 tsx_inv_data);
1680 if (status != PJ_SUCCESS)
1681 return status;
Benny Prijono268ca612006-02-07 12:34:11 +00001682
1683 } else {
1684 pjsip_cseq_hdr *cseq;
1685
1686 /* Can only do this to send response to original INVITE
1687 * request.
1688 */
Benny Prijono64f851e2006-02-23 13:49:28 +00001689 PJ_ASSERT_RETURN((cseq=pjsip_msg_find_hdr(tdata->msg, PJSIP_H_CSEQ, NULL))!=NULL
1690 && (cseq->cseq == inv->invite_tsx->cseq),
Benny Prijono268ca612006-02-07 12:34:11 +00001691 PJ_EINVALIDOP);
1692
1693 status = pjsip_dlg_send_response(inv->dlg, inv->invite_tsx, tdata);
1694 if (status != PJ_SUCCESS)
1695 return status;
1696 }
1697
1698 /* Done (?) */
1699 return PJ_SUCCESS;
1700}
1701
1702
Benny Prijono8ad55352006-02-08 11:16:05 +00001703/*
1704 * Respond to incoming CANCEL request.
1705 */
1706static void inv_respond_incoming_cancel(pjsip_inv_session *inv,
1707 pjsip_transaction *cancel_tsx,
1708 pjsip_rx_data *rdata)
1709{
1710 pjsip_tx_data *tdata;
1711 pjsip_transaction *invite_tsx;
1712 pj_str_t key;
1713 pj_status_t status;
1714
1715 /* See if we have matching INVITE server transaction: */
1716
1717 pjsip_tsx_create_key(rdata->tp_info.pool, &key, PJSIP_ROLE_UAS,
1718 &pjsip_invite_method, rdata);
1719 invite_tsx = pjsip_tsx_layer_find_tsx(&key, PJ_TRUE);
1720
1721 if (invite_tsx == NULL) {
1722
1723 /* Invite transaction not found!
1724 * Respond CANCEL with 491 (RFC 3261 Section 9.2 page 42)
1725 */
1726 status = pjsip_dlg_create_response( inv->dlg, rdata, 200, NULL,
1727 &tdata);
1728
1729 } else {
1730 /* Always answer CANCEL will 200 (OK) regardless of
1731 * the state of the INVITE transaction.
1732 */
1733 status = pjsip_dlg_create_response( inv->dlg, rdata, 200, NULL,
1734 &tdata);
1735 }
1736
1737 /* See if we have created the response successfully. */
1738 if (status != PJ_SUCCESS) return;
1739
1740 /* Send the CANCEL response */
1741 status = pjsip_dlg_send_response(inv->dlg, cancel_tsx, tdata);
1742 if (status != PJ_SUCCESS) return;
1743
1744
1745 /* See if we need to terminate the UAS INVITE transaction
1746 * with 487 (Request Terminated) response.
1747 */
1748 if (invite_tsx && invite_tsx->status_code < 200) {
1749
1750 pj_assert(invite_tsx->last_tx != NULL);
1751
1752 tdata = invite_tsx->last_tx;
1753
1754 status = pjsip_dlg_modify_response(inv->dlg, tdata, 487, NULL);
1755 if (status == PJ_SUCCESS)
1756 pjsip_dlg_send_response(inv->dlg, invite_tsx, tdata);
1757 }
1758
1759 if (invite_tsx)
1760 pj_mutex_unlock(invite_tsx->mutex);
1761}
1762
1763
1764/*
1765 * Respond to incoming BYE request.
1766 */
1767static void inv_respond_incoming_bye( pjsip_inv_session *inv,
1768 pjsip_transaction *bye_tsx,
1769 pjsip_rx_data *rdata,
1770 pjsip_event *e )
1771{
1772 pj_status_t status;
1773 pjsip_tx_data *tdata;
1774
1775 /* Respond BYE with 200: */
1776
1777 status = pjsip_dlg_create_response(inv->dlg, rdata, 200, NULL, &tdata);
1778 if (status != PJ_SUCCESS) return;
1779
1780 status = pjsip_dlg_send_response(inv->dlg, bye_tsx, tdata);
1781 if (status != PJ_SUCCESS) return;
1782
1783 /* Terminate session: */
1784
Benny Prijonod4e0abd2006-03-05 11:53:36 +00001785 if (inv->state != PJSIP_INV_STATE_DISCONNECTED) {
Benny Prijono0b6340c2006-06-13 22:21:23 +00001786 inv_set_cause(inv, PJSIP_SC_OK, NULL);
Benny Prijono8ad55352006-02-08 11:16:05 +00001787 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
Benny Prijonod4e0abd2006-03-05 11:53:36 +00001788 }
Benny Prijono8ad55352006-02-08 11:16:05 +00001789}
1790
1791/*
Benny Prijono38998232006-02-08 22:44:25 +00001792 * Respond to BYE request.
1793 */
1794static void inv_handle_bye_response( pjsip_inv_session *inv,
1795 pjsip_transaction *tsx,
1796 pjsip_rx_data *rdata,
1797 pjsip_event *e )
1798{
1799 pj_status_t status;
1800
1801 if (e->body.tsx_state.type != PJSIP_EVENT_RX_MSG) {
Benny Prijono0b6340c2006-06-13 22:21:23 +00001802 inv_set_cause(inv, PJSIP_SC_OK, NULL);
Benny Prijono38998232006-02-08 22:44:25 +00001803 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
1804 return;
1805 }
1806
1807 /* Handle 401/407 challenge. */
1808 if (tsx->status_code == 401 || tsx->status_code == 407) {
1809
1810 pjsip_tx_data *tdata;
1811
1812 status = pjsip_auth_clt_reinit_req( &inv->dlg->auth_sess,
1813 rdata,
1814 tsx->last_tx,
1815 &tdata);
1816
1817 if (status != PJ_SUCCESS) {
1818
1819 /* Does not have proper credentials.
1820 * End the session anyway.
1821 */
Benny Prijono0b6340c2006-06-13 22:21:23 +00001822 inv_set_cause(inv, PJSIP_SC_OK, NULL);
Benny Prijono38998232006-02-08 22:44:25 +00001823 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
1824
1825 } else {
1826 /* Re-send BYE. */
Benny Prijonoe8b0d3b2006-03-17 17:57:52 +00001827 status = pjsip_inv_send_msg(inv, tdata);
Benny Prijono38998232006-02-08 22:44:25 +00001828 }
1829
1830 } else {
1831
1832 /* End the session. */
Benny Prijono0b6340c2006-06-13 22:21:23 +00001833 inv_set_cause(inv, PJSIP_SC_OK, NULL);
Benny Prijono38998232006-02-08 22:44:25 +00001834 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
1835 }
1836
1837}
1838
1839/*
Benny Prijono8ad55352006-02-08 11:16:05 +00001840 * State NULL is before anything is sent/received.
1841 */
1842static void inv_on_state_null( pjsip_inv_session *inv, pjsip_event *e)
Benny Prijono268ca612006-02-07 12:34:11 +00001843{
1844 pjsip_transaction *tsx = e->body.tsx_state.tsx;
1845 pjsip_dialog *dlg = pjsip_tsx_get_dlg(tsx);
1846
1847 PJ_ASSERT_ON_FAIL(tsx && dlg, return);
1848
1849 if (tsx->method.id == PJSIP_INVITE_METHOD) {
1850
Benny Prijono64f851e2006-02-23 13:49:28 +00001851 /* Keep the initial INVITE transaction. */
1852 if (inv->invite_tsx == NULL)
1853 inv->invite_tsx = tsx;
Benny Prijono268ca612006-02-07 12:34:11 +00001854
Benny Prijono64f851e2006-02-23 13:49:28 +00001855 if (dlg->role == PJSIP_ROLE_UAC) {
Benny Prijono268ca612006-02-07 12:34:11 +00001856
1857 switch (tsx->state) {
1858 case PJSIP_TSX_STATE_CALLING:
Benny Prijono8ad55352006-02-08 11:16:05 +00001859 inv_set_state(inv, PJSIP_INV_STATE_CALLING, e);
Benny Prijono268ca612006-02-07 12:34:11 +00001860 break;
1861 default:
Benny Prijono64f851e2006-02-23 13:49:28 +00001862 inv_on_state_calling(inv, e);
Benny Prijono268ca612006-02-07 12:34:11 +00001863 break;
1864 }
1865
1866 } else {
1867 switch (tsx->state) {
1868 case PJSIP_TSX_STATE_TRYING:
Benny Prijono8ad55352006-02-08 11:16:05 +00001869 inv_set_state(inv, PJSIP_INV_STATE_INCOMING, e);
Benny Prijono268ca612006-02-07 12:34:11 +00001870 break;
Benny Prijono38998232006-02-08 22:44:25 +00001871 case PJSIP_TSX_STATE_PROCEEDING:
1872 inv_set_state(inv, PJSIP_INV_STATE_INCOMING, e);
1873 if (tsx->status_code > 100)
1874 inv_set_state(inv, PJSIP_INV_STATE_EARLY, e);
1875 break;
Benny Prijono268ca612006-02-07 12:34:11 +00001876 default:
Benny Prijono64f851e2006-02-23 13:49:28 +00001877 inv_on_state_incoming(inv, e);
1878 break;
Benny Prijono268ca612006-02-07 12:34:11 +00001879 }
1880 }
1881
1882 } else {
1883 pj_assert(!"Unexpected transaction type");
1884 }
1885}
1886
Benny Prijono8ad55352006-02-08 11:16:05 +00001887/*
1888 * State CALLING is after sending initial INVITE request but before
1889 * any response (with tag) is received.
1890 */
1891static void inv_on_state_calling( pjsip_inv_session *inv, pjsip_event *e)
Benny Prijono268ca612006-02-07 12:34:11 +00001892{
1893 pjsip_transaction *tsx = e->body.tsx_state.tsx;
1894 pjsip_dialog *dlg = pjsip_tsx_get_dlg(tsx);
Benny Prijonoccf95622006-02-07 18:48:01 +00001895 pj_status_t status;
Benny Prijono268ca612006-02-07 12:34:11 +00001896
1897 PJ_ASSERT_ON_FAIL(tsx && dlg, return);
1898
Benny Prijono8ad55352006-02-08 11:16:05 +00001899 if (tsx == inv->invite_tsx) {
Benny Prijono268ca612006-02-07 12:34:11 +00001900
1901 switch (tsx->state) {
1902
Benny Prijono64f851e2006-02-23 13:49:28 +00001903 case PJSIP_TSX_STATE_CALLING:
1904 inv_set_state(inv, PJSIP_INV_STATE_CALLING, e);
1905 break;
1906
Benny Prijono268ca612006-02-07 12:34:11 +00001907 case PJSIP_TSX_STATE_PROCEEDING:
1908 if (dlg->remote.info->tag.slen) {
Benny Prijonoa66c7152006-02-09 01:26:14 +00001909
Benny Prijono8ad55352006-02-08 11:16:05 +00001910 inv_set_state(inv, PJSIP_INV_STATE_EARLY, e);
Benny Prijonoa66c7152006-02-09 01:26:14 +00001911
1912 inv_check_sdp_in_incoming_msg(inv, tsx,
1913 e->body.tsx_state.src.rdata);
1914
Benny Prijono268ca612006-02-07 12:34:11 +00001915 } else {
1916 /* Ignore 100 (Trying) response, as it doesn't change
1917 * session state. It only ceases retransmissions.
1918 */
1919 }
1920 break;
1921
1922 case PJSIP_TSX_STATE_COMPLETED:
1923 if (tsx->status_code/100 == 2) {
1924
1925 /* This should not happen.
1926 * When transaction receives 2xx, it should be terminated
1927 */
1928 pj_assert(0);
Benny Prijono8ad55352006-02-08 11:16:05 +00001929 inv_set_state(inv, PJSIP_INV_STATE_CONNECTING, e);
Benny Prijonoa66c7152006-02-09 01:26:14 +00001930
1931 inv_check_sdp_in_incoming_msg(inv, tsx,
1932 e->body.tsx_state.src.rdata);
Benny Prijono268ca612006-02-07 12:34:11 +00001933
Benny Prijonoccf95622006-02-07 18:48:01 +00001934 } else if (tsx->status_code==401 || tsx->status_code==407) {
1935
1936 /* Handle authentication failure:
1937 * Resend the request with Authorization header.
1938 */
1939 pjsip_tx_data *tdata;
1940
Benny Prijono8ad55352006-02-08 11:16:05 +00001941 status = pjsip_auth_clt_reinit_req(&inv->dlg->auth_sess,
Benny Prijonoccf95622006-02-07 18:48:01 +00001942 e->body.tsx_state.src.rdata,
1943 tsx->last_tx,
1944 &tdata);
1945
1946 if (status != PJ_SUCCESS) {
1947
1948 /* Does not have proper credentials.
1949 * End the session.
1950 */
Benny Prijono0b6340c2006-06-13 22:21:23 +00001951 inv_set_cause(inv, tsx->status_code, &tsx->status_text);
Benny Prijono8ad55352006-02-08 11:16:05 +00001952 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
Benny Prijonoccf95622006-02-07 18:48:01 +00001953
1954 } else {
1955
1956 /* Restart session. */
Benny Prijono8ad55352006-02-08 11:16:05 +00001957 inv->state = PJSIP_INV_STATE_NULL;
1958 inv->invite_tsx = NULL;
Benny Prijonoccf95622006-02-07 18:48:01 +00001959
1960 /* Send the request. */
Benny Prijonoe8b0d3b2006-03-17 17:57:52 +00001961 status = pjsip_inv_send_msg(inv, tdata);
Benny Prijonoccf95622006-02-07 18:48:01 +00001962 }
1963
Benny Prijono268ca612006-02-07 12:34:11 +00001964 } else {
Benny Prijonoccf95622006-02-07 18:48:01 +00001965
Benny Prijono0b6340c2006-06-13 22:21:23 +00001966 inv_set_cause(inv, tsx->status_code, &tsx->status_text);
Benny Prijono8ad55352006-02-08 11:16:05 +00001967 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
Benny Prijonoccf95622006-02-07 18:48:01 +00001968
Benny Prijono268ca612006-02-07 12:34:11 +00001969 }
1970 break;
1971
1972 case PJSIP_TSX_STATE_TERMINATED:
1973 /* INVITE transaction can be terminated either because UAC
1974 * transaction received 2xx response or because of transport
1975 * error.
1976 */
1977 if (tsx->status_code/100 == 2) {
1978 /* This must be receipt of 2xx response */
1979
1980 /* Set state to CONNECTING */
Benny Prijono8ad55352006-02-08 11:16:05 +00001981 inv_set_state(inv, PJSIP_INV_STATE_CONNECTING, e);
Benny Prijono268ca612006-02-07 12:34:11 +00001982
Benny Prijonoa66c7152006-02-09 01:26:14 +00001983 inv_check_sdp_in_incoming_msg(inv, tsx,
1984 e->body.tsx_state.src.rdata);
1985
Benny Prijono268ca612006-02-07 12:34:11 +00001986 /* Send ACK */
1987 pj_assert(e->body.tsx_state.type == PJSIP_EVENT_RX_MSG);
1988
Benny Prijono8ad55352006-02-08 11:16:05 +00001989 inv_send_ack(inv, e->body.tsx_state.src.rdata);
Benny Prijono5eff0432006-02-09 14:14:21 +00001990 inv_set_state(inv, PJSIP_INV_STATE_CONFIRMED, e);
Benny Prijono268ca612006-02-07 12:34:11 +00001991
Benny Prijonoa66c7152006-02-09 01:26:14 +00001992
Benny Prijono268ca612006-02-07 12:34:11 +00001993 } else {
Benny Prijono0b6340c2006-06-13 22:21:23 +00001994 inv_set_cause(inv, tsx->status_code, &tsx->status_text);
Benny Prijono8ad55352006-02-08 11:16:05 +00001995 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
Benny Prijono268ca612006-02-07 12:34:11 +00001996 }
1997 break;
1998
Benny Prijono34a404e2006-02-09 14:38:30 +00001999 default:
2000 break;
Benny Prijono268ca612006-02-07 12:34:11 +00002001 }
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00002002
2003 } else if (inv->role == PJSIP_ROLE_UAC &&
2004 tsx->role == PJSIP_ROLE_UAC &&
2005 tsx->method.id == PJSIP_CANCEL_METHOD)
2006 {
2007 /*
2008 * Handle case when outgoing CANCEL is answered with 481 (Call/
2009 * Transaction Does Not Exist), 408, or when it's timed out. In these
2010 * cases, disconnect session (i.e. dialog usage only).
2011 */
2012 if (tsx->status_code == PJSIP_SC_CALL_TSX_DOES_NOT_EXIST ||
2013 tsx->status_code == PJSIP_SC_REQUEST_TIMEOUT ||
2014 tsx->status_code == PJSIP_SC_TSX_TIMEOUT ||
Benny Prijono0b6340c2006-06-13 22:21:23 +00002015 tsx->status_code == PJSIP_SC_TSX_TRANSPORT_ERROR)
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00002016 {
Benny Prijono0b6340c2006-06-13 22:21:23 +00002017 inv_set_cause(inv, tsx->status_code, &tsx->status_text);
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00002018 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
2019 }
Benny Prijono268ca612006-02-07 12:34:11 +00002020 }
2021}
2022
Benny Prijono8ad55352006-02-08 11:16:05 +00002023/*
2024 * State INCOMING is after we received the request, but before
2025 * responses with tag are sent.
2026 */
2027static void inv_on_state_incoming( pjsip_inv_session *inv, pjsip_event *e)
Benny Prijono268ca612006-02-07 12:34:11 +00002028{
2029 pjsip_transaction *tsx = e->body.tsx_state.tsx;
2030 pjsip_dialog *dlg = pjsip_tsx_get_dlg(tsx);
2031
2032 PJ_ASSERT_ON_FAIL(tsx && dlg, return);
2033
Benny Prijono8ad55352006-02-08 11:16:05 +00002034 if (tsx == inv->invite_tsx) {
2035
2036 /*
2037 * Handle the INVITE state transition.
2038 */
2039
Benny Prijono268ca612006-02-07 12:34:11 +00002040 switch (tsx->state) {
Benny Prijono8ad55352006-02-08 11:16:05 +00002041
Benny Prijono64f851e2006-02-23 13:49:28 +00002042 case PJSIP_TSX_STATE_TRYING:
2043 inv_set_state(inv, PJSIP_INV_STATE_INCOMING, e);
2044 break;
2045
Benny Prijono268ca612006-02-07 12:34:11 +00002046 case PJSIP_TSX_STATE_PROCEEDING:
Benny Prijono8ad55352006-02-08 11:16:05 +00002047 /*
2048 * Transaction sent provisional response.
2049 */
Benny Prijono268ca612006-02-07 12:34:11 +00002050 if (tsx->status_code > 100)
Benny Prijono8ad55352006-02-08 11:16:05 +00002051 inv_set_state(inv, PJSIP_INV_STATE_EARLY, e);
Benny Prijono268ca612006-02-07 12:34:11 +00002052 break;
Benny Prijono8ad55352006-02-08 11:16:05 +00002053
Benny Prijono268ca612006-02-07 12:34:11 +00002054 case PJSIP_TSX_STATE_COMPLETED:
Benny Prijono8ad55352006-02-08 11:16:05 +00002055 /*
2056 * Transaction sent final response.
2057 */
Benny Prijonod4e0abd2006-03-05 11:53:36 +00002058 if (tsx->status_code/100 == 2) {
Benny Prijono8ad55352006-02-08 11:16:05 +00002059 inv_set_state(inv, PJSIP_INV_STATE_CONNECTING, e);
Benny Prijonod4e0abd2006-03-05 11:53:36 +00002060 } else {
Benny Prijono0b6340c2006-06-13 22:21:23 +00002061 inv_set_cause(inv, tsx->status_code, &tsx->status_text);
Benny Prijono8ad55352006-02-08 11:16:05 +00002062 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
Benny Prijonod4e0abd2006-03-05 11:53:36 +00002063 }
Benny Prijono268ca612006-02-07 12:34:11 +00002064 break;
Benny Prijono8ad55352006-02-08 11:16:05 +00002065
Benny Prijono268ca612006-02-07 12:34:11 +00002066 case PJSIP_TSX_STATE_TERMINATED:
Benny Prijono8ad55352006-02-08 11:16:05 +00002067 /*
2068 * This happens on transport error (e.g. failed to send
2069 * response)
2070 */
Benny Prijono0b6340c2006-06-13 22:21:23 +00002071 inv_set_cause(inv, tsx->status_code, &tsx->status_text);
Benny Prijono8ad55352006-02-08 11:16:05 +00002072 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
Benny Prijono268ca612006-02-07 12:34:11 +00002073 break;
Benny Prijono8ad55352006-02-08 11:16:05 +00002074
Benny Prijono268ca612006-02-07 12:34:11 +00002075 default:
Benny Prijono8ad55352006-02-08 11:16:05 +00002076 pj_assert(!"Unexpected INVITE state");
2077 break;
Benny Prijono268ca612006-02-07 12:34:11 +00002078 }
Benny Prijono8ad55352006-02-08 11:16:05 +00002079
2080 } else if (tsx->method.id == PJSIP_CANCEL_METHOD &&
2081 tsx->role == PJSIP_ROLE_UAS &&
2082 tsx->state < PJSIP_TSX_STATE_COMPLETED &&
2083 e->body.tsx_state.type == PJSIP_EVENT_RX_MSG )
2084 {
2085
2086 /*
2087 * Handle incoming CANCEL request.
2088 */
2089
2090 inv_respond_incoming_cancel(inv, tsx, e->body.tsx_state.src.rdata);
2091
Benny Prijono268ca612006-02-07 12:34:11 +00002092 }
2093}
2094
Benny Prijono8ad55352006-02-08 11:16:05 +00002095/*
2096 * State EARLY is for both UAS and UAC, after response with To tag
2097 * is sent/received.
2098 */
2099static void inv_on_state_early( pjsip_inv_session *inv, pjsip_event *e)
Benny Prijono268ca612006-02-07 12:34:11 +00002100{
2101 pjsip_transaction *tsx = e->body.tsx_state.tsx;
2102 pjsip_dialog *dlg = pjsip_tsx_get_dlg(tsx);
2103
2104 PJ_ASSERT_ON_FAIL(tsx && dlg, return);
2105
Benny Prijono8ad55352006-02-08 11:16:05 +00002106 if (tsx == inv->invite_tsx) {
2107
2108 /*
2109 * Handle the INVITE state progress.
2110 */
Benny Prijono268ca612006-02-07 12:34:11 +00002111
2112 switch (tsx->state) {
2113
2114 case PJSIP_TSX_STATE_PROCEEDING:
2115 /* Send/received another provisional response. */
Benny Prijono8ad55352006-02-08 11:16:05 +00002116 inv_set_state(inv, PJSIP_INV_STATE_EARLY, e);
Benny Prijonoa66c7152006-02-09 01:26:14 +00002117
2118 if (e->body.tsx_state.type == PJSIP_EVENT_RX_MSG) {
2119 inv_check_sdp_in_incoming_msg(inv, tsx,
2120 e->body.tsx_state.src.rdata);
2121 }
Benny Prijono268ca612006-02-07 12:34:11 +00002122 break;
2123
2124 case PJSIP_TSX_STATE_COMPLETED:
Benny Prijonoa66c7152006-02-09 01:26:14 +00002125 if (tsx->status_code/100 == 2) {
Benny Prijono8ad55352006-02-08 11:16:05 +00002126 inv_set_state(inv, PJSIP_INV_STATE_CONNECTING, e);
Benny Prijonoa66c7152006-02-09 01:26:14 +00002127 if (e->body.tsx_state.type == PJSIP_EVENT_RX_MSG) {
2128 inv_check_sdp_in_incoming_msg(inv, tsx,
2129 e->body.tsx_state.src.rdata);
2130 }
2131
Benny Prijonod4e0abd2006-03-05 11:53:36 +00002132 } else {
Benny Prijono0b6340c2006-06-13 22:21:23 +00002133 inv_set_cause(inv, tsx->status_code, &tsx->status_text);
Benny Prijono8ad55352006-02-08 11:16:05 +00002134 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
Benny Prijonod4e0abd2006-03-05 11:53:36 +00002135 }
Benny Prijono268ca612006-02-07 12:34:11 +00002136 break;
2137
Benny Prijonof3195072006-02-14 21:15:30 +00002138 case PJSIP_TSX_STATE_CONFIRMED:
2139 /* For some reason can go here */
2140
Benny Prijono268ca612006-02-07 12:34:11 +00002141 case PJSIP_TSX_STATE_TERMINATED:
2142 /* INVITE transaction can be terminated either because UAC
2143 * transaction received 2xx response or because of transport
2144 * error.
2145 */
2146 if (tsx->status_code/100 == 2) {
2147
2148 /* This must be receipt of 2xx response */
2149
2150 /* Set state to CONNECTING */
Benny Prijono8ad55352006-02-08 11:16:05 +00002151 inv_set_state(inv, PJSIP_INV_STATE_CONNECTING, e);
Benny Prijono268ca612006-02-07 12:34:11 +00002152
Benny Prijonoa66c7152006-02-09 01:26:14 +00002153 if (e->body.tsx_state.type == PJSIP_EVENT_RX_MSG) {
2154 inv_check_sdp_in_incoming_msg(inv, tsx,
2155 e->body.tsx_state.src.rdata);
2156 }
2157
Benny Prijono268ca612006-02-07 12:34:11 +00002158 /* if UAC, send ACK and move state to confirmed. */
2159 if (tsx->role == PJSIP_ROLE_UAC) {
2160 pj_assert(e->body.tsx_state.type == PJSIP_EVENT_RX_MSG);
2161
Benny Prijono8ad55352006-02-08 11:16:05 +00002162 inv_send_ack(inv, e->body.tsx_state.src.rdata);
Benny Prijono8ad55352006-02-08 11:16:05 +00002163 inv_set_state(inv, PJSIP_INV_STATE_CONFIRMED, e);
Benny Prijono268ca612006-02-07 12:34:11 +00002164 }
2165
2166 } else {
Benny Prijono0b6340c2006-06-13 22:21:23 +00002167 inv_set_cause(inv, tsx->status_code, &tsx->status_text);
Benny Prijono8ad55352006-02-08 11:16:05 +00002168 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
Benny Prijono268ca612006-02-07 12:34:11 +00002169 }
2170 break;
2171
2172 default:
Benny Prijono8ad55352006-02-08 11:16:05 +00002173 pj_assert(!"Unexpected INVITE tsx state");
Benny Prijono268ca612006-02-07 12:34:11 +00002174 }
2175
Benny Prijono8ad55352006-02-08 11:16:05 +00002176 } else if (inv->role == PJSIP_ROLE_UAS &&
2177 tsx->role == PJSIP_ROLE_UAS &&
2178 tsx->method.id == PJSIP_CANCEL_METHOD &&
2179 tsx->state < PJSIP_TSX_STATE_COMPLETED &&
2180 e->body.tsx_state.type == PJSIP_EVENT_RX_MSG )
2181 {
Benny Prijono268ca612006-02-07 12:34:11 +00002182
Benny Prijono8ad55352006-02-08 11:16:05 +00002183 /*
2184 * Handle incoming CANCEL request.
Benny Prijonoccf95622006-02-07 18:48:01 +00002185 */
2186
Benny Prijono8ad55352006-02-08 11:16:05 +00002187 inv_respond_incoming_cancel(inv, tsx, e->body.tsx_state.src.rdata);
2188
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00002189 } else if (inv->role == PJSIP_ROLE_UAC &&
2190 tsx->role == PJSIP_ROLE_UAC &&
2191 tsx->method.id == PJSIP_CANCEL_METHOD)
2192 {
2193 /*
2194 * Handle case when outgoing CANCEL is answered with 481 (Call/
2195 * Transaction Does Not Exist), 408, or when it's timed out. In these
2196 * cases, disconnect session (i.e. dialog usage only).
2197 */
2198 if (tsx->status_code == PJSIP_SC_CALL_TSX_DOES_NOT_EXIST ||
2199 tsx->status_code == PJSIP_SC_REQUEST_TIMEOUT ||
2200 tsx->status_code == PJSIP_SC_TSX_TIMEOUT ||
Benny Prijono0b6340c2006-06-13 22:21:23 +00002201 tsx->status_code == PJSIP_SC_TSX_TRANSPORT_ERROR)
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00002202 {
Benny Prijono0b6340c2006-06-13 22:21:23 +00002203 inv_set_cause(inv, tsx->status_code, &tsx->status_text);
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00002204 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
2205 }
Benny Prijono268ca612006-02-07 12:34:11 +00002206 }
2207}
2208
Benny Prijono8ad55352006-02-08 11:16:05 +00002209/*
2210 * State CONNECTING is after 2xx response to INVITE is sent/received.
2211 */
2212static void inv_on_state_connecting( pjsip_inv_session *inv, pjsip_event *e)
Benny Prijono268ca612006-02-07 12:34:11 +00002213{
2214 pjsip_transaction *tsx = e->body.tsx_state.tsx;
2215 pjsip_dialog *dlg = pjsip_tsx_get_dlg(tsx);
2216
2217 PJ_ASSERT_ON_FAIL(tsx && dlg, return);
2218
Benny Prijono8ad55352006-02-08 11:16:05 +00002219 if (tsx == inv->invite_tsx) {
Benny Prijono268ca612006-02-07 12:34:11 +00002220
Benny Prijono8ad55352006-02-08 11:16:05 +00002221 /*
2222 * Handle INVITE state progression.
2223 */
Benny Prijono268ca612006-02-07 12:34:11 +00002224 switch (tsx->state) {
2225
2226 case PJSIP_TSX_STATE_CONFIRMED:
Benny Prijono38998232006-02-08 22:44:25 +00002227 if (tsx->status_code/100 == 2)
2228 inv_set_state(inv, PJSIP_INV_STATE_CONFIRMED, e);
Benny Prijono268ca612006-02-07 12:34:11 +00002229 break;
2230
2231 case PJSIP_TSX_STATE_TERMINATED:
2232 /* INVITE transaction can be terminated either because UAC
2233 * transaction received 2xx response or because of transport
2234 * error.
2235 */
2236 if (tsx->status_code/100 != 2) {
Benny Prijono0b6340c2006-06-13 22:21:23 +00002237 inv_set_cause(inv, tsx->status_code, &tsx->status_text);
Benny Prijono8ad55352006-02-08 11:16:05 +00002238 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
Benny Prijono268ca612006-02-07 12:34:11 +00002239 }
2240 break;
2241
2242 case PJSIP_TSX_STATE_DESTROYED:
2243 /* Do nothing. */
2244 break;
2245
2246 default:
2247 pj_assert(!"Unexpected state");
2248 }
2249
Benny Prijono8ad55352006-02-08 11:16:05 +00002250 } else if (tsx->role == PJSIP_ROLE_UAS &&
2251 tsx->method.id == PJSIP_BYE_METHOD &&
2252 tsx->status_code < 200 &&
2253 e->body.tsx_state.type == PJSIP_EVENT_RX_MSG)
2254 {
2255
2256 /*
2257 * Handle incoming BYE.
2258 */
2259
2260 inv_respond_incoming_bye( inv, tsx, e->body.tsx_state.src.rdata, e );
2261
Benny Prijono38998232006-02-08 22:44:25 +00002262 } else if (tsx->method.id == PJSIP_BYE_METHOD &&
2263 tsx->role == PJSIP_ROLE_UAC &&
Benny Prijonoa66c7152006-02-09 01:26:14 +00002264 (tsx->state == PJSIP_TSX_STATE_COMPLETED ||
2265 tsx->state == PJSIP_TSX_STATE_TERMINATED))
Benny Prijono38998232006-02-08 22:44:25 +00002266 {
2267
2268 /*
2269 * Outgoing BYE
2270 */
2271 inv_handle_bye_response( inv, tsx, e->body.tsx_state.src.rdata, e);
2272
Benny Prijono268ca612006-02-07 12:34:11 +00002273 }
Benny Prijono70127222006-07-02 14:53:05 +00002274 else if (tsx->method.id == PJSIP_CANCEL_METHOD &&
2275 tsx->role == PJSIP_ROLE_UAS &&
2276 tsx->status_code < 200 &&
2277 e->body.tsx_state.type == PJSIP_EVENT_RX_MSG)
2278 {
Benny Prijono38998232006-02-08 22:44:25 +00002279
Benny Prijono70127222006-07-02 14:53:05 +00002280 /*
2281 * Handle strandled incoming CANCEL.
2282 */
2283 pjsip_rx_data *rdata = e->body.tsx_state.src.rdata;
2284 pjsip_tx_data *tdata;
2285 pj_status_t status;
2286
2287 status = pjsip_dlg_create_response(dlg, rdata, 200, NULL, &tdata);
2288 if (status != PJ_SUCCESS) return;
2289
2290 status = pjsip_dlg_send_response(dlg, tsx, tdata);
2291 if (status != PJ_SUCCESS) return;
2292
2293 }
Benny Prijono268ca612006-02-07 12:34:11 +00002294}
2295
Benny Prijono8ad55352006-02-08 11:16:05 +00002296/*
2297 * State CONFIRMED is after ACK is sent/received.
2298 */
2299static void inv_on_state_confirmed( pjsip_inv_session *inv, pjsip_event *e)
Benny Prijono268ca612006-02-07 12:34:11 +00002300{
2301 pjsip_transaction *tsx = e->body.tsx_state.tsx;
2302 pjsip_dialog *dlg = pjsip_tsx_get_dlg(tsx);
2303
2304 PJ_ASSERT_ON_FAIL(tsx && dlg, return);
2305
Benny Prijono268ca612006-02-07 12:34:11 +00002306
Benny Prijono8ad55352006-02-08 11:16:05 +00002307 if (tsx->method.id == PJSIP_BYE_METHOD &&
2308 tsx->role == PJSIP_ROLE_UAC &&
Benny Prijonoa66c7152006-02-09 01:26:14 +00002309 (tsx->state == PJSIP_TSX_STATE_COMPLETED ||
2310 tsx->state == PJSIP_TSX_STATE_TERMINATED))
Benny Prijono8ad55352006-02-08 11:16:05 +00002311 {
Benny Prijono38998232006-02-08 22:44:25 +00002312
Benny Prijono8ad55352006-02-08 11:16:05 +00002313 /*
Benny Prijono38998232006-02-08 22:44:25 +00002314 * Outgoing BYE
Benny Prijono8ad55352006-02-08 11:16:05 +00002315 */
Benny Prijono8ad55352006-02-08 11:16:05 +00002316
Benny Prijonoa66c7152006-02-09 01:26:14 +00002317 inv_handle_bye_response( inv, tsx, e->body.tsx_state.src.rdata, e);
Benny Prijono268ca612006-02-07 12:34:11 +00002318
Benny Prijono8ad55352006-02-08 11:16:05 +00002319 }
2320 else if (tsx->method.id == PJSIP_BYE_METHOD &&
2321 tsx->role == PJSIP_ROLE_UAS &&
2322 tsx->status_code < 200 &&
2323 e->body.tsx_state.type == PJSIP_EVENT_RX_MSG)
2324 {
Benny Prijonoccf95622006-02-07 18:48:01 +00002325
Benny Prijono8ad55352006-02-08 11:16:05 +00002326 /*
2327 * Handle incoming BYE.
2328 */
Benny Prijono268ca612006-02-07 12:34:11 +00002329
Benny Prijono8ad55352006-02-08 11:16:05 +00002330 inv_respond_incoming_bye( inv, tsx, e->body.tsx_state.src.rdata, e );
2331
Benny Prijono268ca612006-02-07 12:34:11 +00002332 }
Benny Prijono70127222006-07-02 14:53:05 +00002333 else if (tsx->method.id == PJSIP_CANCEL_METHOD &&
2334 tsx->role == PJSIP_ROLE_UAS &&
2335 tsx->status_code < 200 &&
2336 e->body.tsx_state.type == PJSIP_EVENT_RX_MSG)
2337 {
2338
2339 /*
2340 * Handle strandled incoming CANCEL.
2341 */
2342 pjsip_rx_data *rdata = e->body.tsx_state.src.rdata;
2343 pjsip_tx_data *tdata;
2344 pj_status_t status;
2345
2346 status = pjsip_dlg_create_response(dlg, rdata, 200, NULL, &tdata);
2347 if (status != PJ_SUCCESS) return;
2348
2349 status = pjsip_dlg_send_response(dlg, tsx, tdata);
2350 if (status != PJ_SUCCESS) return;
2351
2352 }
Benny Prijono26ff9062006-02-21 23:47:00 +00002353 else if (tsx->method.id == PJSIP_INVITE_METHOD &&
2354 tsx->role == PJSIP_ROLE_UAS)
2355 {
2356
2357 /*
2358 * Handle incoming re-INVITE
2359 */
2360 if (tsx->state == PJSIP_TSX_STATE_TRYING) {
2361
2362 pjsip_rx_data *rdata = e->body.tsx_state.src.rdata;
2363 pjsip_tx_data *tdata;
2364 pj_status_t status;
2365
2366 /* Check if we have INVITE pending. */
2367 if (inv->invite_tsx && inv->invite_tsx!=tsx) {
2368
2369 /* Can not receive re-INVITE while another one is pending. */
2370 status = pjsip_dlg_create_response( inv->dlg, rdata, 500, NULL,
2371 &tdata);
2372 if (status != PJ_SUCCESS)
2373 return;
2374
2375 status = pjsip_dlg_send_response( inv->dlg, tsx, tdata);
2376
2377
2378 return;
2379 }
2380
2381 /* Save the invite transaction. */
2382 inv->invite_tsx = tsx;
2383
2384 /* Process SDP in incoming message. */
2385 status = inv_check_sdp_in_incoming_msg(inv, tsx, rdata);
2386
2387 if (status != PJ_SUCCESS) {
2388
2389 /* Not Acceptable */
2390 const pjsip_hdr *accept;
2391
2392 status = pjsip_dlg_create_response(inv->dlg, rdata,
2393 488, NULL, &tdata);
2394 if (status != PJ_SUCCESS)
2395 return;
2396
2397
2398 accept = pjsip_endpt_get_capability(dlg->endpt, PJSIP_H_ACCEPT,
2399 NULL);
2400 if (accept) {
2401 pjsip_msg_add_hdr(tdata->msg,
2402 pjsip_hdr_clone(tdata->pool, accept));
2403 }
2404
2405 status = pjsip_dlg_send_response(dlg, tsx, tdata);
2406
2407 return;
2408 }
2409
2410 /* Create 2xx ANSWER */
2411 status = pjsip_dlg_create_response(dlg, rdata, 200, NULL, &tdata);
2412 if (status != PJ_SUCCESS)
2413 return;
2414
2415 /* Process SDP in the answer */
Benny Prijono64f851e2006-02-23 13:49:28 +00002416 status = process_answer(inv, 200, tdata, NULL);
Benny Prijono26ff9062006-02-21 23:47:00 +00002417 if (status != PJ_SUCCESS)
2418 return;
2419
Benny Prijonoe8b0d3b2006-03-17 17:57:52 +00002420 status = pjsip_inv_send_msg(inv, tdata);
Benny Prijono26ff9062006-02-21 23:47:00 +00002421
2422 }
2423
2424 }
2425 else if (tsx->method.id == PJSIP_INVITE_METHOD &&
2426 tsx->role == PJSIP_ROLE_UAC)
2427 {
2428 /*
2429 * Handle outgoing re-INVITE
2430 */
2431 if (tsx->state == PJSIP_TSX_STATE_TERMINATED &&
2432 tsx->status_code/100 == 2)
2433 {
2434
2435 /* Re-INVITE was accepted. */
2436
2437 /* Process SDP */
2438 inv_check_sdp_in_incoming_msg(inv, tsx,
2439 e->body.tsx_state.src.rdata);
2440
2441 /* Send ACK */
2442 inv_send_ack(inv, e->body.tsx_state.src.rdata);
2443
2444 } else if (tsx->state == PJSIP_TSX_STATE_COMPLETED &&
2445 (tsx->status_code==401 || tsx->status_code==407))
2446 {
2447 pjsip_tx_data *tdata;
2448 pj_status_t status;
2449
2450 /* Handle authentication challenge. */
2451 status = pjsip_auth_clt_reinit_req( &dlg->auth_sess,
2452 e->body.tsx_state.src.rdata,
2453 tsx->last_tx,
2454 &tdata);
2455 if (status != PJ_SUCCESS)
2456 return;
2457
2458 /* Send re-INVITE */
Benny Prijonoe8b0d3b2006-03-17 17:57:52 +00002459 status = pjsip_inv_send_msg( inv, tdata);
Benny Prijono26ff9062006-02-21 23:47:00 +00002460
2461 } else if (tsx->status_code==PJSIP_SC_CALL_TSX_DOES_NOT_EXIST ||
2462 tsx->status_code==PJSIP_SC_REQUEST_TIMEOUT ||
2463 tsx->status_code >= 700)
2464 {
2465 /*
2466 * Handle responses that terminates dialog.
2467 */
Benny Prijono0b6340c2006-06-13 22:21:23 +00002468 inv_set_cause(inv, tsx->status_code, &tsx->status_text);
Benny Prijono26ff9062006-02-21 23:47:00 +00002469 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
2470 }
2471 }
Benny Prijono268ca612006-02-07 12:34:11 +00002472}
2473
Benny Prijono8ad55352006-02-08 11:16:05 +00002474/*
2475 * After session has been terminated, but before dialog is destroyed
2476 * (because dialog has other usages, or because dialog is waiting for
2477 * the last transaction to terminate).
2478 */
2479static void inv_on_state_disconnected( pjsip_inv_session *inv, pjsip_event *e)
Benny Prijono268ca612006-02-07 12:34:11 +00002480{
Benny Prijono8ad55352006-02-08 11:16:05 +00002481 pjsip_transaction *tsx = e->body.tsx_state.tsx;
2482 pjsip_dialog *dlg = pjsip_tsx_get_dlg(tsx);
Benny Prijono268ca612006-02-07 12:34:11 +00002483
Benny Prijono8ad55352006-02-08 11:16:05 +00002484 PJ_ASSERT_ON_FAIL(tsx && dlg, return);
2485
Benny Prijono70127222006-07-02 14:53:05 +00002486 if (tsx->role == PJSIP_ROLE_UAS &&
Benny Prijono8ad55352006-02-08 11:16:05 +00002487 tsx->status_code < 200 &&
2488 e->body.tsx_state.type == PJSIP_EVENT_RX_MSG)
2489 {
Benny Prijono70127222006-07-02 14:53:05 +00002490 pjsip_rx_data *rdata = e->body.tsx_state.src.rdata;
Benny Prijono8ad55352006-02-08 11:16:05 +00002491
2492 /*
Benny Prijono70127222006-07-02 14:53:05 +00002493 * Respond BYE with 200/OK
Benny Prijono8ad55352006-02-08 11:16:05 +00002494 */
Benny Prijono70127222006-07-02 14:53:05 +00002495 if (tsx->method.id == PJSIP_BYE_METHOD) {
2496 inv_respond_incoming_bye( inv, tsx, rdata, e );
2497 } else if (tsx->method.id == PJSIP_CANCEL_METHOD) {
2498 /*
2499 * Respond CANCEL with 200/OK too.
2500 */
2501 pjsip_tx_data *tdata;
2502 pj_status_t status;
Benny Prijono8ad55352006-02-08 11:16:05 +00002503
Benny Prijono70127222006-07-02 14:53:05 +00002504 status = pjsip_dlg_create_response(dlg, rdata, 200, NULL, &tdata);
2505 if (status != PJ_SUCCESS) return;
Benny Prijono8ad55352006-02-08 11:16:05 +00002506
Benny Prijono70127222006-07-02 14:53:05 +00002507 status = pjsip_dlg_send_response(dlg, tsx, tdata);
2508 if (status != PJ_SUCCESS) return;
2509
2510 }
Benny Prijono8ad55352006-02-08 11:16:05 +00002511 }
Benny Prijono268ca612006-02-07 12:34:11 +00002512}
2513