blob: e08e10d288dd61696db5095f201234b5230f69a3 [file] [log] [blame]
Benny Prijono268ca612006-02-07 12:34:11 +00001/* $Id$ */
2/*
Benny Prijonoa771a512007-02-19 01:13:53 +00003 * Copyright (C) 2003-2007 Benny Prijono <benny@prijono.org>
Benny Prijono268ca612006-02-07 12:34:11 +00004 *
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{
Benny Prijono4be63b52006-11-25 14:50:25 +000038 "NULL",
39 "CALLING",
40 "INCOMING",
41 "EARLY",
Benny Prijonoe8b0d3b2006-03-17 17:57:52 +000042 "CONNECTING",
Benny Prijonoc5055702007-01-13 23:20:18 +000043 "CONFIRMED",
Benny Prijonoe8b0d3b2006-03-17 17:57:52 +000044 "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) {
Benny Prijonoba5926a2007-05-02 11:29:37 +0000172 inv->cause = (pjsip_status_code) cause_code;
Benny Prijono0b6340c2006-06-13 22:21:23 +0000173 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
Benny Prijono46249942007-02-19 22:23:14 +0000359 /* Clear invite transaction when tsx is confirmed.
360 * Previously we set invite_tsx to NULL only when transaction has
361 * terminated, but this didn't work when ACK has the same Via branch
362 * value as the INVITE (see http://www.pjsip.org/trac/ticket/113)
363 */
364 if (tsx->state>=PJSIP_TSX_STATE_CONFIRMED && tsx == inv->invite_tsx)
365 inv->invite_tsx = NULL;
Benny Prijono268ca612006-02-07 12:34:11 +0000366}
367
Benny Prijono8ad55352006-02-08 11:16:05 +0000368
369/*
370 * Initialize the invite module.
371 */
Benny Prijono268ca612006-02-07 12:34:11 +0000372PJ_DEF(pj_status_t) pjsip_inv_usage_init( pjsip_endpoint *endpt,
Benny Prijono268ca612006-02-07 12:34:11 +0000373 const pjsip_inv_callback *cb)
374{
375 pj_status_t status;
376
377 /* Check arguments. */
Benny Prijonoe8b0d3b2006-03-17 17:57:52 +0000378 PJ_ASSERT_RETURN(endpt && cb, PJ_EINVAL);
Benny Prijono268ca612006-02-07 12:34:11 +0000379
380 /* Some callbacks are mandatory */
381 PJ_ASSERT_RETURN(cb->on_state_changed && cb->on_new_session, PJ_EINVAL);
382
383 /* Check if module already registered. */
384 PJ_ASSERT_RETURN(mod_inv.mod.id == -1, PJ_EINVALIDOP);
385
386 /* Copy param. */
387 pj_memcpy(&mod_inv.cb, cb, sizeof(pjsip_inv_callback));
388
389 mod_inv.endpt = endpt;
Benny Prijono268ca612006-02-07 12:34:11 +0000390
391 /* Register the module. */
392 status = pjsip_endpt_register_module(endpt, &mod_inv.mod);
Benny Prijono053f5222006-11-11 16:16:04 +0000393 if (status != PJ_SUCCESS)
394 return status;
Benny Prijono268ca612006-02-07 12:34:11 +0000395
Benny Prijono053f5222006-11-11 16:16:04 +0000396 return PJ_SUCCESS;
Benny Prijono268ca612006-02-07 12:34:11 +0000397}
398
Benny Prijono8ad55352006-02-08 11:16:05 +0000399/*
400 * Get the instance of invite module.
401 */
Benny Prijono268ca612006-02-07 12:34:11 +0000402PJ_DEF(pjsip_module*) pjsip_inv_usage_instance(void)
403{
404 return &mod_inv.mod;
405}
406
407
Benny Prijono632ce712006-02-09 14:01:40 +0000408
Benny Prijono8ad55352006-02-08 11:16:05 +0000409/*
410 * Return the invite session for the specified dialog.
411 */
Benny Prijono268ca612006-02-07 12:34:11 +0000412PJ_DEF(pjsip_inv_session*) pjsip_dlg_get_inv_session(pjsip_dialog *dlg)
413{
414 return dlg->mod_data[mod_inv.mod.id];
415}
416
Benny Prijono8ad55352006-02-08 11:16:05 +0000417
Benny Prijono268ca612006-02-07 12:34:11 +0000418/*
Benny Prijonoe8b0d3b2006-03-17 17:57:52 +0000419 * Get INVITE state name.
420 */
421PJ_DEF(const char *) pjsip_inv_state_name(pjsip_inv_state state)
422{
423 PJ_ASSERT_RETURN(state >= PJSIP_INV_STATE_NULL &&
424 state <= PJSIP_INV_STATE_DISCONNECTED,
425 "??");
426
427 return inv_state_names[state];
428}
429
430/*
Benny Prijono8ad55352006-02-08 11:16:05 +0000431 * Create UAC invite session.
Benny Prijono268ca612006-02-07 12:34:11 +0000432 */
433PJ_DEF(pj_status_t) pjsip_inv_create_uac( pjsip_dialog *dlg,
434 const pjmedia_sdp_session *local_sdp,
435 unsigned options,
436 pjsip_inv_session **p_inv)
437{
438 pjsip_inv_session *inv;
439 pj_status_t status;
440
441 /* Verify arguments. */
442 PJ_ASSERT_RETURN(dlg && p_inv, PJ_EINVAL);
443
Benny Prijono8eae8382006-08-10 21:44:26 +0000444 /* Must lock dialog first */
445 pjsip_dlg_inc_lock(dlg);
446
Benny Prijono268ca612006-02-07 12:34:11 +0000447 /* Normalize options */
448 if (options & PJSIP_INV_REQUIRE_100REL)
449 options |= PJSIP_INV_SUPPORT_100REL;
450
451 if (options & PJSIP_INV_REQUIRE_TIMER)
452 options |= PJSIP_INV_SUPPORT_TIMER;
453
454 /* Create the session */
455 inv = pj_pool_zalloc(dlg->pool, sizeof(pjsip_inv_session));
Benny Prijono8eae8382006-08-10 21:44:26 +0000456 pj_assert(inv != NULL);
Benny Prijono268ca612006-02-07 12:34:11 +0000457
458 inv->pool = dlg->pool;
459 inv->role = PJSIP_ROLE_UAC;
460 inv->state = PJSIP_INV_STATE_NULL;
461 inv->dlg = dlg;
462 inv->options = options;
Benny Prijonod4e0abd2006-03-05 11:53:36 +0000463 inv->notify = PJ_TRUE;
Benny Prijonoba5926a2007-05-02 11:29:37 +0000464 inv->cause = (pjsip_status_code) 0;
Benny Prijono268ca612006-02-07 12:34:11 +0000465
Benny Prijonoe4f2abb2006-02-10 14:04:05 +0000466 /* Object name will use the same dialog pointer. */
Benny Prijonoed811d72006-03-10 12:57:12 +0000467 pj_ansi_snprintf(inv->obj_name, PJ_MAX_OBJ_NAME, "inv%p", dlg);
Benny Prijonoe4f2abb2006-02-10 14:04:05 +0000468
Benny Prijono268ca612006-02-07 12:34:11 +0000469 /* Create negotiator if local_sdp is specified. */
470 if (local_sdp) {
471 status = pjmedia_sdp_neg_create_w_local_offer(dlg->pool, local_sdp,
472 &inv->neg);
Benny Prijono8eae8382006-08-10 21:44:26 +0000473 if (status != PJ_SUCCESS) {
474 pjsip_dlg_dec_lock(dlg);
Benny Prijono268ca612006-02-07 12:34:11 +0000475 return status;
Benny Prijono8eae8382006-08-10 21:44:26 +0000476 }
Benny Prijono268ca612006-02-07 12:34:11 +0000477 }
478
479 /* Register invite as dialog usage. */
480 status = pjsip_dlg_add_usage(dlg, &mod_inv.mod, inv);
Benny Prijono8eae8382006-08-10 21:44:26 +0000481 if (status != PJ_SUCCESS) {
482 pjsip_dlg_dec_lock(dlg);
Benny Prijono268ca612006-02-07 12:34:11 +0000483 return status;
Benny Prijono8eae8382006-08-10 21:44:26 +0000484 }
Benny Prijono268ca612006-02-07 12:34:11 +0000485
486 /* Increment dialog session */
Benny Prijonoe4f2abb2006-02-10 14:04:05 +0000487 pjsip_dlg_inc_session(dlg, &mod_inv.mod);
Benny Prijono268ca612006-02-07 12:34:11 +0000488
489 /* Done */
490 *p_inv = inv;
Benny Prijonoe4f2abb2006-02-10 14:04:05 +0000491
Benny Prijono8eae8382006-08-10 21:44:26 +0000492 pjsip_dlg_dec_lock(dlg);
493
Benny Prijonoe4f2abb2006-02-10 14:04:05 +0000494 PJ_LOG(5,(inv->obj_name, "UAC invite session created for dialog %s",
495 dlg->obj_name));
496
Benny Prijono268ca612006-02-07 12:34:11 +0000497 return PJ_SUCCESS;
498}
499
500/*
501 * Verify incoming INVITE request.
502 */
503PJ_DEF(pj_status_t) pjsip_inv_verify_request(pjsip_rx_data *rdata,
504 unsigned *options,
505 const pjmedia_sdp_session *l_sdp,
506 pjsip_dialog *dlg,
507 pjsip_endpoint *endpt,
508 pjsip_tx_data **p_tdata)
509{
510 pjsip_msg *msg;
511 pjsip_allow_hdr *allow;
512 pjsip_supported_hdr *sup_hdr;
513 pjsip_require_hdr *req_hdr;
514 int code = 200;
515 unsigned rem_option = 0;
516 pj_status_t status = PJ_SUCCESS;
517 pjsip_hdr res_hdr_list;
518
519 /* Init return arguments. */
520 if (p_tdata) *p_tdata = NULL;
521
522 /* Verify arguments. */
523 PJ_ASSERT_RETURN(rdata != NULL && options != NULL, PJ_EINVAL);
524
525 /* Normalize options */
526 if (*options & PJSIP_INV_REQUIRE_100REL)
527 *options |= PJSIP_INV_SUPPORT_100REL;
528
529 if (*options & PJSIP_INV_REQUIRE_TIMER)
530 *options |= PJSIP_INV_SUPPORT_TIMER;
531
532 /* Get the message in rdata */
533 msg = rdata->msg_info.msg;
534
535 /* Must be INVITE request. */
536 PJ_ASSERT_RETURN(msg->type == PJSIP_REQUEST_MSG &&
537 msg->line.req.method.id == PJSIP_INVITE_METHOD,
538 PJ_EINVAL);
539
540 /* If tdata is specified, then either dlg or endpt must be specified */
541 PJ_ASSERT_RETURN((!p_tdata) || (endpt || dlg), PJ_EINVAL);
542
543 /* Get the endpoint */
544 endpt = endpt ? endpt : dlg->endpt;
545
546 /* Init response header list */
547 pj_list_init(&res_hdr_list);
548
Benny Prijono8ad55352006-02-08 11:16:05 +0000549 /* Check the request body, see if it'inv something that we support
Benny Prijono268ca612006-02-07 12:34:11 +0000550 * (i.e. SDP).
551 */
552 if (msg->body) {
553 pjsip_msg_body *body = msg->body;
554 pj_str_t str_application = {"application", 11};
555 pj_str_t str_sdp = { "sdp", 3 };
556 pjmedia_sdp_session *sdp;
557
558 /* Check content type. */
559 if (pj_stricmp(&body->content_type.type, &str_application) != 0 ||
560 pj_stricmp(&body->content_type.subtype, &str_sdp) != 0)
561 {
562 /* Not "application/sdp" */
563 code = PJSIP_SC_UNSUPPORTED_MEDIA_TYPE;
564 status = PJSIP_ERRNO_FROM_SIP_STATUS(code);
565
566 if (p_tdata) {
567 /* Add Accept header to response */
568 pjsip_accept_hdr *acc;
569
570 acc = pjsip_accept_hdr_create(rdata->tp_info.pool);
571 PJ_ASSERT_RETURN(acc, PJ_ENOMEM);
572 acc->values[acc->count++] = pj_str("application/sdp");
573 pj_list_push_back(&res_hdr_list, acc);
574 }
575
576 goto on_return;
577 }
578
579 /* Parse and validate SDP */
580 status = pjmedia_sdp_parse(rdata->tp_info.pool, body->data, body->len,
581 &sdp);
582 if (status == PJ_SUCCESS)
583 status = pjmedia_sdp_validate(sdp);
584
585 if (status != PJ_SUCCESS) {
586 /* Unparseable or invalid SDP */
587 code = PJSIP_SC_BAD_REQUEST;
588
589 if (p_tdata) {
590 /* Add Warning header. */
591 pjsip_warning_hdr *w;
592
593 w = pjsip_warning_hdr_create_from_status(rdata->tp_info.pool,
594 pjsip_endpt_name(endpt),
595 status);
596 PJ_ASSERT_RETURN(w, PJ_ENOMEM);
597
598 pj_list_push_back(&res_hdr_list, w);
599 }
600
601 goto on_return;
602 }
603
604 /* Negotiate with local SDP */
605 if (l_sdp) {
606 pjmedia_sdp_neg *neg;
607
608 /* Local SDP must be valid! */
609 PJ_ASSERT_RETURN((status=pjmedia_sdp_validate(l_sdp))==PJ_SUCCESS,
610 status);
611
612 /* Create SDP negotiator */
613 status = pjmedia_sdp_neg_create_w_remote_offer(
614 rdata->tp_info.pool, l_sdp, sdp, &neg);
615 PJ_ASSERT_RETURN(status == PJ_SUCCESS, status);
616
617 /* Negotiate SDP */
618 status = pjmedia_sdp_neg_negotiate(rdata->tp_info.pool, neg, 0);
619 if (status != PJ_SUCCESS) {
620
621 /* Incompatible media */
622 code = PJSIP_SC_NOT_ACCEPTABLE_HERE;
Benny Prijono268ca612006-02-07 12:34:11 +0000623
624 if (p_tdata) {
625 pjsip_accept_hdr *acc;
626 pjsip_warning_hdr *w;
627
628 /* Add Warning header. */
629 w = pjsip_warning_hdr_create_from_status(
630 rdata->tp_info.pool,
631 pjsip_endpt_name(endpt), status);
632 PJ_ASSERT_RETURN(w, PJ_ENOMEM);
633
634 pj_list_push_back(&res_hdr_list, w);
635
636 /* Add Accept header to response */
637 acc = pjsip_accept_hdr_create(rdata->tp_info.pool);
638 PJ_ASSERT_RETURN(acc, PJ_ENOMEM);
639 acc->values[acc->count++] = pj_str("application/sdp");
640 pj_list_push_back(&res_hdr_list, acc);
641
642 }
643
644 goto on_return;
645 }
646 }
647 }
648
649 /* Check supported methods, see if peer supports UPDATE.
650 * We just assume that peer supports standard INVITE, ACK, CANCEL, and BYE
651 * implicitly by sending this INVITE.
652 */
653 allow = pjsip_msg_find_hdr(msg, PJSIP_H_ALLOW, NULL);
654 if (allow) {
655 unsigned i;
656 const pj_str_t STR_UPDATE = { "UPDATE", 6 };
657
658 for (i=0; i<allow->count; ++i) {
659 if (pj_stricmp(&allow->values[i], &STR_UPDATE)==0)
660 break;
661 }
662
663 if (i != allow->count) {
664 /* UPDATE is present in Allow */
665 rem_option |= PJSIP_INV_SUPPORT_UPDATE;
666 }
667
668 }
669
670 /* Check Supported header */
671 sup_hdr = pjsip_msg_find_hdr(msg, PJSIP_H_SUPPORTED, NULL);
672 if (sup_hdr) {
673 unsigned i;
674 pj_str_t STR_100REL = { "100rel", 6};
675 pj_str_t STR_TIMER = { "timer", 5 };
676
677 for (i=0; i<sup_hdr->count; ++i) {
678 if (pj_stricmp(&sup_hdr->values[i], &STR_100REL)==0)
679 rem_option |= PJSIP_INV_SUPPORT_100REL;
680 else if (pj_stricmp(&sup_hdr->values[i], &STR_TIMER)==0)
681 rem_option |= PJSIP_INV_SUPPORT_TIMER;
682 }
683 }
684
685 /* Check Require header */
686 req_hdr = pjsip_msg_find_hdr(msg, PJSIP_H_REQUIRE, NULL);
687 if (req_hdr) {
688 unsigned i;
Benny Prijono053f5222006-11-11 16:16:04 +0000689 const pj_str_t STR_100REL = { "100rel", 6};
690 const pj_str_t STR_TIMER = { "timer", 5 };
691 const pj_str_t STR_REPLACES = { "replaces", 8 };
Benny Prijono268ca612006-02-07 12:34:11 +0000692 unsigned unsupp_cnt = 0;
693 pj_str_t unsupp_tags[PJSIP_GENERIC_ARRAY_MAX_COUNT];
694
695 for (i=0; i<req_hdr->count; ++i) {
696 if ((*options & PJSIP_INV_SUPPORT_100REL) &&
697 pj_stricmp(&req_hdr->values[i], &STR_100REL)==0)
698 {
699 rem_option |= PJSIP_INV_REQUIRE_100REL;
700
701 } else if ((*options && PJSIP_INV_SUPPORT_TIMER) &&
702 pj_stricmp(&req_hdr->values[i], &STR_TIMER)==0)
703 {
704 rem_option |= PJSIP_INV_REQUIRE_TIMER;
705
Benny Prijono053f5222006-11-11 16:16:04 +0000706 } else if (pj_stricmp(&req_hdr->values[i], &STR_REPLACES)==0) {
707 pj_bool_t supp;
708
709 supp = pjsip_endpt_has_capability(endpt, PJSIP_H_SUPPORTED,
710 NULL, &STR_REPLACES);
711 if (!supp)
712 unsupp_tags[unsupp_cnt++] = req_hdr->values[i];
713
Benny Prijono268ca612006-02-07 12:34:11 +0000714 } else {
715 /* Unknown/unsupported extension tag! */
716 unsupp_tags[unsupp_cnt++] = req_hdr->values[i];
717 }
718 }
719
720 /* Check if there are required tags that we don't support */
721 if (unsupp_cnt) {
722
723 code = PJSIP_SC_BAD_EXTENSION;
724 status = PJSIP_ERRNO_FROM_SIP_STATUS(code);
725
726 if (p_tdata) {
727 pjsip_unsupported_hdr *unsupp_hdr;
728 const pjsip_hdr *h;
729
730 /* Add Unsupported header. */
731 unsupp_hdr = pjsip_unsupported_hdr_create(rdata->tp_info.pool);
732 PJ_ASSERT_RETURN(unsupp_hdr != NULL, PJ_ENOMEM);
733
734 unsupp_hdr->count = unsupp_cnt;
735 for (i=0; i<unsupp_cnt; ++i)
736 unsupp_hdr->values[i] = unsupp_tags[i];
737
738 pj_list_push_back(&res_hdr_list, unsupp_hdr);
739
740 /* Add Supported header. */
741 h = pjsip_endpt_get_capability(endpt, PJSIP_H_SUPPORTED,
742 NULL);
743 pj_assert(h);
744 if (h) {
745 sup_hdr = pjsip_hdr_clone(rdata->tp_info.pool, h);
746 pj_list_push_back(&res_hdr_list, sup_hdr);
747 }
748 }
749
750 goto on_return;
751 }
752 }
753
754 /* Check if there are local requirements that are not supported
755 * by peer.
756 */
757 if ( ((*options & PJSIP_INV_REQUIRE_100REL)!=0 &&
758 (rem_option & PJSIP_INV_SUPPORT_100REL)==0) ||
759 ((*options & PJSIP_INV_REQUIRE_TIMER)!=0 &&
760 (rem_option & PJSIP_INV_SUPPORT_TIMER)==0))
761 {
762 code = PJSIP_SC_EXTENSION_REQUIRED;
763 status = PJSIP_ERRNO_FROM_SIP_STATUS(code);
764
765 if (p_tdata) {
766 const pjsip_hdr *h;
767
768 /* Add Require header. */
769 req_hdr = pjsip_require_hdr_create(rdata->tp_info.pool);
770 PJ_ASSERT_RETURN(req_hdr != NULL, PJ_ENOMEM);
771
772 if (*options & PJSIP_INV_REQUIRE_100REL)
773 req_hdr->values[req_hdr->count++] = pj_str("100rel");
774
775 if (*options & PJSIP_INV_REQUIRE_TIMER)
776 req_hdr->values[req_hdr->count++] = pj_str("timer");
777
778 pj_list_push_back(&res_hdr_list, req_hdr);
779
780 /* Add Supported header. */
781 h = pjsip_endpt_get_capability(endpt, PJSIP_H_SUPPORTED,
782 NULL);
783 pj_assert(h);
784 if (h) {
785 sup_hdr = pjsip_hdr_clone(rdata->tp_info.pool, h);
786 pj_list_push_back(&res_hdr_list, sup_hdr);
787 }
788
789 }
790
791 goto on_return;
792 }
793
794on_return:
795
796 /* Create response if necessary */
797 if (code != 200 && p_tdata) {
798 pjsip_tx_data *tdata;
799 const pjsip_hdr *h;
800
801 if (dlg) {
802 status = pjsip_dlg_create_response(dlg, rdata, code, NULL,
803 &tdata);
804 } else {
805 status = pjsip_endpt_create_response(endpt, rdata, code, NULL,
806 &tdata);
807 }
808
809 if (status != PJ_SUCCESS)
810 return status;
811
812 /* Add response headers. */
813 h = res_hdr_list.next;
814 while (h != &res_hdr_list) {
815 pjsip_hdr *cloned;
816
817 cloned = pjsip_hdr_clone(tdata->pool, h);
818 PJ_ASSERT_RETURN(cloned, PJ_ENOMEM);
819
820 pjsip_msg_add_hdr(tdata->msg, cloned);
821
822 h = h->next;
823 }
824
825 *p_tdata = tdata;
Benny Prijono70127222006-07-02 14:53:05 +0000826
827 /* Can not return PJ_SUCCESS when response message is produced.
828 * Ref: PROTOS test ~#2490
829 */
830 if (status == PJ_SUCCESS)
831 status = PJSIP_ERRNO_FROM_SIP_STATUS(code);
832
Benny Prijono268ca612006-02-07 12:34:11 +0000833 }
834
835 return status;
836}
837
838/*
Benny Prijono8ad55352006-02-08 11:16:05 +0000839 * Create UAS invite session.
Benny Prijono268ca612006-02-07 12:34:11 +0000840 */
841PJ_DEF(pj_status_t) pjsip_inv_create_uas( pjsip_dialog *dlg,
842 pjsip_rx_data *rdata,
843 const pjmedia_sdp_session *local_sdp,
844 unsigned options,
845 pjsip_inv_session **p_inv)
846{
847 pjsip_inv_session *inv;
Benny Prijonoa66c7152006-02-09 01:26:14 +0000848 struct tsx_inv_data *tsx_inv_data;
Benny Prijono268ca612006-02-07 12:34:11 +0000849 pjsip_msg *msg;
850 pjmedia_sdp_session *rem_sdp = NULL;
851 pj_status_t status;
852
853 /* Verify arguments. */
854 PJ_ASSERT_RETURN(dlg && rdata && p_inv, PJ_EINVAL);
855
856 /* Dialog MUST have been initialised. */
857 PJ_ASSERT_RETURN(pjsip_rdata_get_tsx(rdata) != NULL, PJ_EINVALIDOP);
858
859 msg = rdata->msg_info.msg;
860
861 /* rdata MUST contain INVITE request */
862 PJ_ASSERT_RETURN(msg->type == PJSIP_REQUEST_MSG &&
863 msg->line.req.method.id == PJSIP_INVITE_METHOD,
864 PJ_EINVALIDOP);
865
Benny Prijono8eae8382006-08-10 21:44:26 +0000866 /* Lock dialog */
867 pjsip_dlg_inc_lock(dlg);
868
Benny Prijono268ca612006-02-07 12:34:11 +0000869 /* Normalize options */
870 if (options & PJSIP_INV_REQUIRE_100REL)
871 options |= PJSIP_INV_SUPPORT_100REL;
872
873 if (options & PJSIP_INV_REQUIRE_TIMER)
874 options |= PJSIP_INV_SUPPORT_TIMER;
875
876 /* Create the session */
877 inv = pj_pool_zalloc(dlg->pool, sizeof(pjsip_inv_session));
Benny Prijono8eae8382006-08-10 21:44:26 +0000878 pj_assert(inv != NULL);
Benny Prijono268ca612006-02-07 12:34:11 +0000879
880 inv->pool = dlg->pool;
881 inv->role = PJSIP_ROLE_UAS;
Benny Prijono38998232006-02-08 22:44:25 +0000882 inv->state = PJSIP_INV_STATE_NULL;
Benny Prijono268ca612006-02-07 12:34:11 +0000883 inv->dlg = dlg;
884 inv->options = options;
Benny Prijonod4e0abd2006-03-05 11:53:36 +0000885 inv->notify = PJ_TRUE;
Benny Prijonoba5926a2007-05-02 11:29:37 +0000886 inv->cause = (pjsip_status_code) 0;
Benny Prijono268ca612006-02-07 12:34:11 +0000887
Benny Prijonoe4f2abb2006-02-10 14:04:05 +0000888 /* Object name will use the same dialog pointer. */
Benny Prijonoed811d72006-03-10 12:57:12 +0000889 pj_ansi_snprintf(inv->obj_name, PJ_MAX_OBJ_NAME, "inv%p", dlg);
Benny Prijonoe4f2abb2006-02-10 14:04:05 +0000890
Benny Prijono268ca612006-02-07 12:34:11 +0000891 /* Parse SDP in message body, if present. */
892 if (msg->body) {
893 pjsip_msg_body *body = msg->body;
894
895 /* Parse and validate SDP */
896 status = pjmedia_sdp_parse(inv->pool, body->data, body->len,
897 &rem_sdp);
898 if (status == PJ_SUCCESS)
899 status = pjmedia_sdp_validate(rem_sdp);
900
Benny Prijono8eae8382006-08-10 21:44:26 +0000901 if (status != PJ_SUCCESS) {
902 pjsip_dlg_dec_lock(dlg);
Benny Prijono268ca612006-02-07 12:34:11 +0000903 return status;
Benny Prijono8eae8382006-08-10 21:44:26 +0000904 }
Benny Prijono268ca612006-02-07 12:34:11 +0000905 }
906
907 /* Create negotiator. */
908 if (rem_sdp) {
909 status = pjmedia_sdp_neg_create_w_remote_offer(inv->pool, local_sdp,
910 rem_sdp, &inv->neg);
911
912 } else if (local_sdp) {
913 status = pjmedia_sdp_neg_create_w_local_offer(inv->pool, local_sdp,
914 &inv->neg);
915 } else {
Benny Prijono95196582006-02-09 00:13:40 +0000916 status = PJ_SUCCESS;
Benny Prijono268ca612006-02-07 12:34:11 +0000917 }
918
Benny Prijono8eae8382006-08-10 21:44:26 +0000919 if (status != PJ_SUCCESS) {
920 pjsip_dlg_dec_lock(dlg);
Benny Prijono268ca612006-02-07 12:34:11 +0000921 return status;
Benny Prijono8eae8382006-08-10 21:44:26 +0000922 }
Benny Prijono268ca612006-02-07 12:34:11 +0000923
924 /* Register invite as dialog usage. */
925 status = pjsip_dlg_add_usage(dlg, &mod_inv.mod, inv);
Benny Prijono8eae8382006-08-10 21:44:26 +0000926 if (status != PJ_SUCCESS) {
927 pjsip_dlg_dec_lock(dlg);
Benny Prijono268ca612006-02-07 12:34:11 +0000928 return status;
Benny Prijono8eae8382006-08-10 21:44:26 +0000929 }
Benny Prijono268ca612006-02-07 12:34:11 +0000930
931 /* Increment session in the dialog. */
Benny Prijonoe4f2abb2006-02-10 14:04:05 +0000932 pjsip_dlg_inc_session(dlg, &mod_inv.mod);
Benny Prijono268ca612006-02-07 12:34:11 +0000933
934 /* Save the invite transaction. */
935 inv->invite_tsx = pjsip_rdata_get_tsx(rdata);
Benny Prijonoa66c7152006-02-09 01:26:14 +0000936
937 /* Attach our data to the transaction. */
938 tsx_inv_data = pj_pool_zalloc(inv->invite_tsx->pool,
939 sizeof(struct tsx_inv_data));
940 tsx_inv_data->inv = inv;
941 inv->invite_tsx->mod_data[mod_inv.mod.id] = tsx_inv_data;
Benny Prijono268ca612006-02-07 12:34:11 +0000942
943 /* Done */
Benny Prijono8eae8382006-08-10 21:44:26 +0000944 pjsip_dlg_dec_lock(dlg);
Benny Prijono268ca612006-02-07 12:34:11 +0000945 *p_inv = inv;
Benny Prijonoe4f2abb2006-02-10 14:04:05 +0000946
947 PJ_LOG(5,(inv->obj_name, "UAS invite session created for dialog %s",
948 dlg->obj_name));
949
Benny Prijono268ca612006-02-07 12:34:11 +0000950 return PJ_SUCCESS;
951}
952
Benny Prijonod4e0abd2006-03-05 11:53:36 +0000953/*
954 * Forcefully terminate the session.
955 */
956PJ_DEF(pj_status_t) pjsip_inv_terminate( pjsip_inv_session *inv,
957 int st_code,
958 pj_bool_t notify)
959{
960 PJ_ASSERT_RETURN(inv, PJ_EINVAL);
961
962 /* Lock dialog. */
963 pjsip_dlg_inc_lock(inv->dlg);
964
965 /* Set callback notify flag. */
966 inv->notify = notify;
967
968 /* If there's pending transaction, terminate the transaction.
969 * This may subsequently set the INVITE session state to
970 * disconnected.
971 */
972 if (inv->invite_tsx &&
973 inv->invite_tsx->state <= PJSIP_TSX_STATE_COMPLETED)
974 {
975 pjsip_tsx_terminate(inv->invite_tsx, st_code);
976
977 }
978
979 /* Set cause. */
Benny Prijono0b6340c2006-06-13 22:21:23 +0000980 inv_set_cause(inv, st_code, NULL);
Benny Prijonod4e0abd2006-03-05 11:53:36 +0000981
982 /* Forcefully terminate the session if state is not DISCONNECTED */
983 if (inv->state != PJSIP_INV_STATE_DISCONNECTED) {
984 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, NULL);
985 }
986
987 /* Done.
988 * The dec_lock() below will actually destroys the dialog if it
989 * has no other session.
990 */
991 pjsip_dlg_dec_lock(inv->dlg);
992
993 return PJ_SUCCESS;
994}
995
996
Benny Prijono268ca612006-02-07 12:34:11 +0000997static void *clone_sdp(pj_pool_t *pool, const void *data, unsigned len)
998{
999 PJ_UNUSED_ARG(len);
1000 return pjmedia_sdp_session_clone(pool, data);
1001}
1002
1003static int print_sdp(pjsip_msg_body *body, char *buf, pj_size_t len)
1004{
1005 return pjmedia_sdp_print(body->data, buf, len);
1006}
1007
Benny Prijono56315612006-07-18 14:39:40 +00001008
1009PJ_DEF(pj_status_t) pjsip_create_sdp_body( pj_pool_t *pool,
1010 pjmedia_sdp_session *sdp,
1011 pjsip_msg_body **p_body)
1012{
1013 const pj_str_t STR_APPLICATION = { "application", 11};
1014 const pj_str_t STR_SDP = { "sdp", 3 };
1015 pjsip_msg_body *body;
1016
1017 body = pj_pool_zalloc(pool, sizeof(pjsip_msg_body));
1018 PJ_ASSERT_RETURN(body != NULL, PJ_ENOMEM);
1019
1020 body->content_type.type = STR_APPLICATION;
1021 body->content_type.subtype = STR_SDP;
1022 body->data = sdp;
1023 body->len = 0;
1024 body->clone_data = &clone_sdp;
1025 body->print_body = &print_sdp;
1026
1027 *p_body = body;
1028
1029 return PJ_SUCCESS;
1030}
1031
Benny Prijono268ca612006-02-07 12:34:11 +00001032static pjsip_msg_body *create_sdp_body(pj_pool_t *pool,
1033 const pjmedia_sdp_session *c_sdp)
1034{
1035 pjsip_msg_body *body;
Benny Prijono56315612006-07-18 14:39:40 +00001036 pj_status_t status;
Benny Prijono268ca612006-02-07 12:34:11 +00001037
Benny Prijono56315612006-07-18 14:39:40 +00001038 status = pjsip_create_sdp_body(pool,
1039 pjmedia_sdp_session_clone(pool, c_sdp),
1040 &body);
Benny Prijono268ca612006-02-07 12:34:11 +00001041
Benny Prijono56315612006-07-18 14:39:40 +00001042 if (status != PJ_SUCCESS)
1043 return NULL;
Benny Prijono268ca612006-02-07 12:34:11 +00001044
1045 return body;
1046}
1047
1048/*
1049 * Create initial INVITE request.
1050 */
1051PJ_DEF(pj_status_t) pjsip_inv_invite( pjsip_inv_session *inv,
1052 pjsip_tx_data **p_tdata )
1053{
1054 pjsip_tx_data *tdata;
1055 const pjsip_hdr *hdr;
Benny Prijono26ff9062006-02-21 23:47:00 +00001056 pj_bool_t has_sdp;
Benny Prijono268ca612006-02-07 12:34:11 +00001057 pj_status_t status;
1058
1059 /* Verify arguments. */
1060 PJ_ASSERT_RETURN(inv && p_tdata, PJ_EINVAL);
1061
Benny Prijono26ff9062006-02-21 23:47:00 +00001062 /* State MUST be NULL or CONFIRMED. */
1063 PJ_ASSERT_RETURN(inv->state == PJSIP_INV_STATE_NULL ||
1064 inv->state == PJSIP_INV_STATE_CONFIRMED,
1065 PJ_EINVALIDOP);
Benny Prijono268ca612006-02-07 12:34:11 +00001066
Benny Prijono64f851e2006-02-23 13:49:28 +00001067 /* Lock dialog. */
1068 pjsip_dlg_inc_lock(inv->dlg);
1069
Benny Prijono268ca612006-02-07 12:34:11 +00001070 /* Create the INVITE request. */
1071 status = pjsip_dlg_create_request(inv->dlg, &pjsip_invite_method, -1,
1072 &tdata);
1073 if (status != PJ_SUCCESS)
Benny Prijono64f851e2006-02-23 13:49:28 +00001074 goto on_return;
1075
Benny Prijono268ca612006-02-07 12:34:11 +00001076
Benny Prijono26ff9062006-02-21 23:47:00 +00001077 /* If this is the first INVITE, then copy the headers from inv_hdr.
1078 * These are the headers parsed from the request URI when the
1079 * dialog was created.
1080 */
1081 if (inv->state == PJSIP_INV_STATE_NULL) {
1082 hdr = inv->dlg->inv_hdr.next;
1083
1084 while (hdr != &inv->dlg->inv_hdr) {
1085 pjsip_msg_add_hdr(tdata->msg,
1086 pjsip_hdr_shallow_clone(tdata->pool, hdr));
1087 hdr = hdr->next;
1088 }
1089 }
1090
1091 /* See if we have SDP to send. */
1092 if (inv->neg) {
1093 pjmedia_sdp_neg_state neg_state;
1094
1095 neg_state = pjmedia_sdp_neg_get_state(inv->neg);
1096
1097 has_sdp = (neg_state == PJMEDIA_SDP_NEG_STATE_LOCAL_OFFER ||
1098 (neg_state == PJMEDIA_SDP_NEG_STATE_WAIT_NEGO &&
1099 pjmedia_sdp_neg_has_local_answer(inv->neg)));
1100
1101
1102 } else {
1103 has_sdp = PJ_FALSE;
1104 }
1105
Benny Prijono268ca612006-02-07 12:34:11 +00001106 /* Add SDP, if any. */
Benny Prijono26ff9062006-02-21 23:47:00 +00001107 if (has_sdp) {
Benny Prijono268ca612006-02-07 12:34:11 +00001108 const pjmedia_sdp_session *offer;
1109
1110 status = pjmedia_sdp_neg_get_neg_local(inv->neg, &offer);
1111 if (status != PJ_SUCCESS)
Benny Prijono64f851e2006-02-23 13:49:28 +00001112 goto on_return;
Benny Prijono268ca612006-02-07 12:34:11 +00001113
1114 tdata->msg->body = create_sdp_body(tdata->pool, offer);
1115 }
1116
1117 /* Add Allow header. */
1118 hdr = pjsip_endpt_get_capability(inv->dlg->endpt, PJSIP_H_ALLOW, NULL);
1119 if (hdr) {
1120 pjsip_msg_add_hdr(tdata->msg,
1121 pjsip_hdr_shallow_clone(tdata->pool, hdr));
1122 }
1123
1124 /* Add Supported header */
1125 hdr = pjsip_endpt_get_capability(inv->dlg->endpt, PJSIP_H_SUPPORTED, NULL);
1126 if (hdr) {
1127 pjsip_msg_add_hdr(tdata->msg,
1128 pjsip_hdr_shallow_clone(tdata->pool, hdr));
1129 }
1130
1131 /* Add Require header. */
1132 PJ_TODO(INVITE_ADD_REQUIRE_HEADER);
1133
1134 /* Done. */
1135 *p_tdata = tdata;
1136
Benny Prijono64f851e2006-02-23 13:49:28 +00001137
1138on_return:
1139 pjsip_dlg_dec_lock(inv->dlg);
1140 return status;
Benny Prijono268ca612006-02-07 12:34:11 +00001141}
1142
1143
1144/*
Benny Prijono95196582006-02-09 00:13:40 +00001145 * Negotiate SDP.
1146 */
1147static pj_status_t inv_negotiate_sdp( pjsip_inv_session *inv )
1148{
1149 pj_status_t status;
1150
1151 PJ_ASSERT_RETURN(pjmedia_sdp_neg_get_state(inv->neg) ==
1152 PJMEDIA_SDP_NEG_STATE_WAIT_NEGO,
1153 PJMEDIA_SDPNEG_EINSTATE);
1154
1155 status = pjmedia_sdp_neg_negotiate(inv->pool, inv->neg, 0);
1156
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00001157 PJ_LOG(5,(inv->obj_name, "SDP negotiation done, status=%d", status));
1158
Benny Prijonod4e0abd2006-03-05 11:53:36 +00001159 if (mod_inv.cb.on_media_update && inv->notify)
Benny Prijono95196582006-02-09 00:13:40 +00001160 (*mod_inv.cb.on_media_update)(inv, status);
1161
1162 return status;
1163}
1164
1165/*
Benny Prijonoa66c7152006-02-09 01:26:14 +00001166 * Check in incoming message for SDP offer/answer.
1167 */
Benny Prijono26ff9062006-02-21 23:47:00 +00001168static pj_status_t inv_check_sdp_in_incoming_msg( pjsip_inv_session *inv,
1169 pjsip_transaction *tsx,
1170 pjsip_rx_data *rdata)
Benny Prijonoa66c7152006-02-09 01:26:14 +00001171{
1172 struct tsx_inv_data *tsx_inv_data;
1173 static const pj_str_t str_application = { "application", 11 };
1174 static const pj_str_t str_sdp = { "sdp", 3 };
1175 pj_status_t status;
1176 pjsip_msg *msg;
1177 pjmedia_sdp_session *sdp;
1178
1179 /* Get/attach invite session's transaction data */
1180 tsx_inv_data = tsx->mod_data[mod_inv.mod.id];
1181 if (tsx_inv_data == NULL) {
1182 tsx_inv_data = pj_pool_zalloc(tsx->pool, sizeof(struct tsx_inv_data));
1183 tsx_inv_data->inv = inv;
1184 tsx->mod_data[mod_inv.mod.id] = tsx_inv_data;
1185 }
1186
1187 /* MUST NOT do multiple SDP offer/answer in a single transaction.
1188 */
1189
1190 if (tsx_inv_data->sdp_done)
Benny Prijono26ff9062006-02-21 23:47:00 +00001191 return PJ_SUCCESS;
Benny Prijonoa66c7152006-02-09 01:26:14 +00001192
1193 /* Check if SDP is present in the message. */
1194
1195 msg = rdata->msg_info.msg;
1196 if (msg->body == NULL) {
1197 /* Message doesn't have body. */
Benny Prijono26ff9062006-02-21 23:47:00 +00001198 return PJ_SUCCESS;
Benny Prijonoa66c7152006-02-09 01:26:14 +00001199 }
1200
1201 if (pj_stricmp(&msg->body->content_type.type, &str_application) ||
1202 pj_stricmp(&msg->body->content_type.subtype, &str_sdp))
1203 {
1204 /* Message body is not "application/sdp" */
Benny Prijono26ff9062006-02-21 23:47:00 +00001205 return PJMEDIA_SDP_EINSDP;
Benny Prijonoa66c7152006-02-09 01:26:14 +00001206 }
1207
1208 /* Parse the SDP body. */
1209
1210 status = pjmedia_sdp_parse(rdata->tp_info.pool, msg->body->data,
1211 msg->body->len, &sdp);
1212 if (status != PJ_SUCCESS) {
1213 char errmsg[PJ_ERR_MSG_SIZE];
1214 pj_strerror(status, errmsg, sizeof(errmsg));
1215 PJ_LOG(4,(THIS_FILE, "Error parsing SDP in %s: %s",
1216 pjsip_rx_data_get_info(rdata), errmsg));
Benny Prijono26ff9062006-02-21 23:47:00 +00001217 return PJMEDIA_SDP_EINSDP;
Benny Prijonoa66c7152006-02-09 01:26:14 +00001218 }
1219
1220 /* The SDP can be an offer or answer, depending on negotiator's state */
1221
1222 if (inv->neg == NULL ||
1223 pjmedia_sdp_neg_get_state(inv->neg) == PJMEDIA_SDP_NEG_STATE_DONE)
1224 {
1225
1226 /* This is an offer. */
1227
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00001228 PJ_LOG(5,(inv->obj_name, "Got SDP offer in %s",
1229 pjsip_rx_data_get_info(rdata)));
1230
Benny Prijonoa66c7152006-02-09 01:26:14 +00001231 if (inv->neg == NULL) {
1232 status=pjmedia_sdp_neg_create_w_remote_offer(inv->pool, NULL,
1233 sdp, &inv->neg);
1234 } else {
1235 status=pjmedia_sdp_neg_set_remote_offer(inv->pool, inv->neg, sdp);
1236 }
1237
1238 if (status != PJ_SUCCESS) {
1239 char errmsg[PJ_ERR_MSG_SIZE];
1240 pj_strerror(status, errmsg, sizeof(errmsg));
1241 PJ_LOG(4,(THIS_FILE, "Error processing SDP offer in %s: %s",
1242 pjsip_rx_data_get_info(rdata), errmsg));
Benny Prijono26ff9062006-02-21 23:47:00 +00001243 return PJMEDIA_SDP_EINSDP;
Benny Prijonoa66c7152006-02-09 01:26:14 +00001244 }
1245
1246 /* Inform application about remote offer. */
1247
Benny Prijonod4e0abd2006-03-05 11:53:36 +00001248 if (mod_inv.cb.on_rx_offer && inv->notify) {
Benny Prijono26ff9062006-02-21 23:47:00 +00001249
1250 (*mod_inv.cb.on_rx_offer)(inv, sdp);
1251
1252 }
Benny Prijonoa66c7152006-02-09 01:26:14 +00001253
1254 } else if (pjmedia_sdp_neg_get_state(inv->neg) ==
1255 PJMEDIA_SDP_NEG_STATE_LOCAL_OFFER)
1256 {
1257
1258 /* This is an answer.
1259 * Process and negotiate remote answer.
1260 */
1261
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00001262 PJ_LOG(5,(inv->obj_name, "Got SDP answer in %s",
1263 pjsip_rx_data_get_info(rdata)));
1264
Benny Prijonoa66c7152006-02-09 01:26:14 +00001265 status = pjmedia_sdp_neg_set_remote_answer(inv->pool, inv->neg, sdp);
1266
1267 if (status != PJ_SUCCESS) {
1268 char errmsg[PJ_ERR_MSG_SIZE];
1269 pj_strerror(status, errmsg, sizeof(errmsg));
1270 PJ_LOG(4,(THIS_FILE, "Error processing SDP answer in %s: %s",
1271 pjsip_rx_data_get_info(rdata), errmsg));
Benny Prijono26ff9062006-02-21 23:47:00 +00001272 return PJMEDIA_SDP_EINSDP;
Benny Prijonoa66c7152006-02-09 01:26:14 +00001273 }
1274
1275 /* Negotiate SDP */
1276
1277 inv_negotiate_sdp(inv);
1278
1279 /* Mark this transaction has having SDP offer/answer done. */
1280
1281 tsx_inv_data->sdp_done = 1;
1282
1283 } else {
1284
1285 PJ_LOG(5,(THIS_FILE, "Ignored SDP in %s: negotiator state is %s",
1286 pjsip_rx_data_get_info(rdata),
1287 pjmedia_sdp_neg_state_str(pjmedia_sdp_neg_get_state(inv->neg))));
1288 }
1289
Benny Prijono26ff9062006-02-21 23:47:00 +00001290 return PJ_SUCCESS;
Benny Prijonoa66c7152006-02-09 01:26:14 +00001291}
1292
1293
Benny Prijono26ff9062006-02-21 23:47:00 +00001294/*
1295 * Process INVITE answer, for both initial and subsequent re-INVITE
1296 */
1297static pj_status_t process_answer( pjsip_inv_session *inv,
1298 int st_code,
Benny Prijono64f851e2006-02-23 13:49:28 +00001299 pjsip_tx_data *tdata,
1300 const pjmedia_sdp_session *local_sdp)
Benny Prijono26ff9062006-02-21 23:47:00 +00001301{
1302 pj_status_t status;
Benny Prijonoab7399b2006-02-27 00:40:31 +00001303 const pjmedia_sdp_session *sdp = NULL;
Benny Prijono26ff9062006-02-21 23:47:00 +00001304
Benny Prijono64f851e2006-02-23 13:49:28 +00001305 /* If local_sdp is specified, then we MUST NOT have answered the
1306 * offer before.
Benny Prijono26ff9062006-02-21 23:47:00 +00001307 */
Benny Prijono64f851e2006-02-23 13:49:28 +00001308 if (local_sdp && (st_code/100==1 || st_code/100==2)) {
1309
1310 if (inv->neg == NULL) {
1311 status = pjmedia_sdp_neg_create_w_local_offer(inv->pool, local_sdp,
1312 &inv->neg);
1313 } else if (pjmedia_sdp_neg_get_state(inv->neg)==
1314 PJMEDIA_SDP_NEG_STATE_REMOTE_OFFER)
1315 {
1316 status = pjmedia_sdp_neg_set_local_answer(inv->pool, inv->neg,
1317 local_sdp);
1318 } else {
1319
1320 /* Can not specify local SDP at this state. */
1321 pj_assert(0);
1322 status = PJMEDIA_SDPNEG_EINSTATE;
1323 }
1324
1325 if (status != PJ_SUCCESS)
1326 return status;
1327
1328 }
1329
1330
1331 /* If SDP negotiator is ready, start negotiation. */
1332 if (st_code/100==2 || (st_code/10==18 && st_code!=180)) {
Benny Prijono26ff9062006-02-21 23:47:00 +00001333
1334 pjmedia_sdp_neg_state neg_state;
1335
Benny Prijono64f851e2006-02-23 13:49:28 +00001336 /* Start nego when appropriate. */
Benny Prijono26ff9062006-02-21 23:47:00 +00001337 neg_state = inv->neg ? pjmedia_sdp_neg_get_state(inv->neg) :
1338 PJMEDIA_SDP_NEG_STATE_NULL;
1339
1340 if (neg_state == PJMEDIA_SDP_NEG_STATE_LOCAL_OFFER) {
1341
1342 status = pjmedia_sdp_neg_get_neg_local(inv->neg, &sdp);
1343
1344 } else if (neg_state == PJMEDIA_SDP_NEG_STATE_WAIT_NEGO &&
1345 pjmedia_sdp_neg_has_local_answer(inv->neg) )
1346 {
1347
1348 status = inv_negotiate_sdp(inv);
1349 if (status != PJ_SUCCESS)
1350 return status;
1351
1352 status = pjmedia_sdp_neg_get_active_local(inv->neg, &sdp);
1353 }
Benny Prijono26ff9062006-02-21 23:47:00 +00001354 }
1355
Benny Prijono64f851e2006-02-23 13:49:28 +00001356 /* Include SDP when it's available for 2xx and 18x (but not 180) response.
Benny Prijono26ff9062006-02-21 23:47:00 +00001357 * Subsequent response will include this SDP.
1358 */
1359 if (sdp) {
1360 tdata->msg->body = create_sdp_body(tdata->pool, sdp);
1361 }
1362
Benny Prijono26ff9062006-02-21 23:47:00 +00001363
1364 return PJ_SUCCESS;
1365}
1366
Benny Prijonoa66c7152006-02-09 01:26:14 +00001367
1368/*
Benny Prijono64f851e2006-02-23 13:49:28 +00001369 * Create first response to INVITE
1370 */
1371PJ_DEF(pj_status_t) pjsip_inv_initial_answer( pjsip_inv_session *inv,
1372 pjsip_rx_data *rdata,
1373 int st_code,
1374 const pj_str_t *st_text,
1375 const pjmedia_sdp_session *sdp,
1376 pjsip_tx_data **p_tdata)
1377{
1378 pjsip_tx_data *tdata;
1379 pj_status_t status;
1380
1381 /* Verify arguments. */
1382 PJ_ASSERT_RETURN(inv && p_tdata, PJ_EINVAL);
1383
1384 /* Must have INVITE transaction. */
1385 PJ_ASSERT_RETURN(inv->invite_tsx, PJ_EBUG);
1386
1387 pjsip_dlg_inc_lock(inv->dlg);
1388
1389 /* Create response */
1390 status = pjsip_dlg_create_response(inv->dlg, rdata, st_code, st_text,
1391 &tdata);
1392 if (status != PJ_SUCCESS)
1393 goto on_return;
1394
1395 /* Process SDP in answer */
1396 status = process_answer(inv, st_code, tdata, sdp);
1397 if (status != PJ_SUCCESS) {
1398 pjsip_tx_data_dec_ref(tdata);
1399 goto on_return;
1400 }
1401
1402 *p_tdata = tdata;
1403
1404on_return:
1405 pjsip_dlg_dec_lock(inv->dlg);
1406 return status;
1407}
1408
1409
1410/*
Benny Prijono26ff9062006-02-21 23:47:00 +00001411 * Answer initial INVITE
1412 * Re-INVITE will be answered automatically, and will not use this function.
Benny Prijono268ca612006-02-07 12:34:11 +00001413 */
1414PJ_DEF(pj_status_t) pjsip_inv_answer( pjsip_inv_session *inv,
1415 int st_code,
1416 const pj_str_t *st_text,
1417 const pjmedia_sdp_session *local_sdp,
1418 pjsip_tx_data **p_tdata )
1419{
1420 pjsip_tx_data *last_res;
1421 pj_status_t status;
1422
1423 /* Verify arguments. */
1424 PJ_ASSERT_RETURN(inv && p_tdata, PJ_EINVAL);
1425
1426 /* Must have INVITE transaction. */
1427 PJ_ASSERT_RETURN(inv->invite_tsx, PJ_EBUG);
1428
1429 /* INVITE transaction MUST have transmitted a response (e.g. 100) */
1430 PJ_ASSERT_RETURN(inv->invite_tsx->last_tx, PJ_EINVALIDOP);
1431
Benny Prijono64f851e2006-02-23 13:49:28 +00001432 pjsip_dlg_inc_lock(inv->dlg);
Benny Prijono268ca612006-02-07 12:34:11 +00001433
1434 /* Modify last response. */
Benny Prijono26ff9062006-02-21 23:47:00 +00001435 last_res = inv->invite_tsx->last_tx;
Benny Prijono268ca612006-02-07 12:34:11 +00001436 status = pjsip_dlg_modify_response(inv->dlg, last_res, st_code, st_text);
1437 if (status != PJ_SUCCESS)
Benny Prijono64f851e2006-02-23 13:49:28 +00001438 goto on_return;
Benny Prijono268ca612006-02-07 12:34:11 +00001439
Benny Prijono268ca612006-02-07 12:34:11 +00001440
Benny Prijono26ff9062006-02-21 23:47:00 +00001441 /* Process SDP in answer */
Benny Prijono64f851e2006-02-23 13:49:28 +00001442 status = process_answer(inv, st_code, last_res, local_sdp);
1443 if (status != PJ_SUCCESS) {
1444 pjsip_tx_data_dec_ref(last_res);
1445 goto on_return;
1446 }
Benny Prijono268ca612006-02-07 12:34:11 +00001447
Benny Prijono268ca612006-02-07 12:34:11 +00001448
1449 *p_tdata = last_res;
1450
Benny Prijono64f851e2006-02-23 13:49:28 +00001451on_return:
1452 pjsip_dlg_dec_lock(inv->dlg);
1453 return status;
Benny Prijono268ca612006-02-07 12:34:11 +00001454}
1455
1456
1457/*
Benny Prijono26ff9062006-02-21 23:47:00 +00001458 * Set SDP answer.
1459 */
1460PJ_DEF(pj_status_t) pjsip_inv_set_sdp_answer( pjsip_inv_session *inv,
1461 const pjmedia_sdp_session *sdp )
1462{
1463 pj_status_t status;
1464
1465 PJ_ASSERT_RETURN(inv && sdp, PJ_EINVAL);
1466
1467 pjsip_dlg_inc_lock(inv->dlg);
1468 status = pjmedia_sdp_neg_set_local_answer( inv->pool, inv->neg, sdp);
1469 pjsip_dlg_dec_lock(inv->dlg);
1470
1471 return status;
1472}
1473
1474
1475/*
Benny Prijono268ca612006-02-07 12:34:11 +00001476 * End session.
1477 */
1478PJ_DEF(pj_status_t) pjsip_inv_end_session( pjsip_inv_session *inv,
1479 int st_code,
1480 const pj_str_t *st_text,
1481 pjsip_tx_data **p_tdata )
1482{
1483 pjsip_tx_data *tdata;
1484 pj_status_t status;
1485
1486 /* Verify arguments. */
1487 PJ_ASSERT_RETURN(inv && p_tdata, PJ_EINVAL);
1488
Benny Prijonod4e0abd2006-03-05 11:53:36 +00001489 /* Set cause code. */
Benny Prijono0b6340c2006-06-13 22:21:23 +00001490 inv_set_cause(inv, st_code, st_text);
Benny Prijonod4e0abd2006-03-05 11:53:36 +00001491
Benny Prijono268ca612006-02-07 12:34:11 +00001492 /* Create appropriate message. */
1493 switch (inv->state) {
1494 case PJSIP_INV_STATE_CALLING:
1495 case PJSIP_INV_STATE_EARLY:
1496 case PJSIP_INV_STATE_INCOMING:
1497
1498 if (inv->role == PJSIP_ROLE_UAC) {
1499
1500 /* For UAC when session has not been confirmed, create CANCEL. */
1501
1502 /* MUST have the original UAC INVITE transaction. */
1503 PJ_ASSERT_RETURN(inv->invite_tsx != NULL, PJ_EBUG);
1504
1505 /* But CANCEL should only be called when we have received a
1506 * provisional response. If we haven't received any responses,
1507 * just destroy the transaction.
1508 */
1509 if (inv->invite_tsx->status_code < 100) {
1510
1511 pjsip_tsx_terminate(inv->invite_tsx, 487);
Benny Prijonofccab712006-02-22 22:23:22 +00001512 *p_tdata = NULL;
1513 return PJ_SUCCESS;
Benny Prijono268ca612006-02-07 12:34:11 +00001514 }
1515
1516 /* The CSeq here assumes that the dialog is started with an
1517 * INVITE session. This may not be correct; dialog can be
1518 * started as SUBSCRIBE session.
1519 * So fix this!
1520 */
1521 status = pjsip_endpt_create_cancel(inv->dlg->endpt,
1522 inv->invite_tsx->last_tx,
1523 &tdata);
1524
1525 } else {
1526
1527 /* For UAS, send a final response. */
1528 tdata = inv->invite_tsx->last_tx;
1529 PJ_ASSERT_RETURN(tdata != NULL, PJ_EINVALIDOP);
1530
Benny Prijono26ff9062006-02-21 23:47:00 +00001531 //status = pjsip_dlg_modify_response(inv->dlg, tdata, st_code,
1532 // st_text);
1533 status = pjsip_inv_answer(inv, st_code, st_text, NULL, &tdata);
Benny Prijono268ca612006-02-07 12:34:11 +00001534 }
1535 break;
1536
1537 case PJSIP_INV_STATE_CONNECTING:
1538 case PJSIP_INV_STATE_CONFIRMED:
1539 /* For established dialog, send BYE */
1540 status = pjsip_dlg_create_request(inv->dlg, &pjsip_bye_method, -1,
1541 &tdata);
1542 break;
1543
1544 case PJSIP_INV_STATE_DISCONNECTED:
Benny Prijono268ca612006-02-07 12:34:11 +00001545 /* No need to do anything. */
Benny Prijonod4e0abd2006-03-05 11:53:36 +00001546 return PJSIP_ESESSIONTERMINATED;
Benny Prijono268ca612006-02-07 12:34:11 +00001547
1548 default:
1549 pj_assert("!Invalid operation!");
1550 return PJ_EINVALIDOP;
1551 }
1552
1553 if (status != PJ_SUCCESS)
1554 return status;
1555
1556
1557 /* Done */
1558
1559 *p_tdata = tdata;
1560
1561 return PJ_SUCCESS;
1562}
1563
1564
1565/*
1566 * Create re-INVITE.
1567 */
1568PJ_DEF(pj_status_t) pjsip_inv_reinvite( pjsip_inv_session *inv,
1569 const pj_str_t *new_contact,
1570 const pjmedia_sdp_session *new_offer,
1571 pjsip_tx_data **p_tdata )
1572{
Benny Prijono26ff9062006-02-21 23:47:00 +00001573 pj_status_t status;
1574 pjsip_contact_hdr *contact_hdr = NULL;
Benny Prijono268ca612006-02-07 12:34:11 +00001575
Benny Prijono26ff9062006-02-21 23:47:00 +00001576 /* Check arguments. */
1577 PJ_ASSERT_RETURN(inv && p_tdata, PJ_EINVAL);
1578
1579 /* Must NOT have a pending INVITE transaction */
1580 PJ_ASSERT_RETURN(inv->invite_tsx==NULL, PJ_EINVALIDOP);
1581
1582
1583 pjsip_dlg_inc_lock(inv->dlg);
1584
1585 if (new_contact) {
1586 pj_str_t tmp;
1587 const pj_str_t STR_CONTACT = { "Contact", 7 };
1588
1589 pj_strdup_with_null(inv->dlg->pool, &tmp, new_contact);
1590 contact_hdr = pjsip_parse_hdr(inv->dlg->pool, &STR_CONTACT,
1591 tmp.ptr, tmp.slen, NULL);
1592 if (!contact_hdr) {
1593 status = PJSIP_EINVALIDURI;
1594 goto on_return;
1595 }
1596 }
1597
1598
1599 if (new_offer) {
1600 if (!inv->neg) {
1601 status = pjmedia_sdp_neg_create_w_local_offer(inv->pool, new_offer,
1602 &inv->neg);
1603 if (status != PJ_SUCCESS)
1604 goto on_return;
1605
1606 } else switch (pjmedia_sdp_neg_get_state(inv->neg)) {
1607
1608 case PJMEDIA_SDP_NEG_STATE_NULL:
1609 pj_assert(!"Unexpected SDP neg state NULL");
1610 status = PJ_EBUG;
1611 goto on_return;
1612
1613 case PJMEDIA_SDP_NEG_STATE_LOCAL_OFFER:
1614 PJ_LOG(4,(inv->obj_name,
1615 "pjsip_inv_reinvite: already have an offer, new "
1616 "offer is ignored"));
1617 break;
1618
1619 case PJMEDIA_SDP_NEG_STATE_REMOTE_OFFER:
1620 status = pjmedia_sdp_neg_set_local_answer(inv->pool, inv->neg,
1621 new_offer);
1622 if (status != PJ_SUCCESS)
1623 goto on_return;
1624 break;
1625
1626 case PJMEDIA_SDP_NEG_STATE_WAIT_NEGO:
1627 PJ_LOG(4,(inv->obj_name,
1628 "pjsip_inv_reinvite: SDP in WAIT_NEGO state, new "
1629 "offer is ignored"));
1630 break;
1631
1632 case PJMEDIA_SDP_NEG_STATE_DONE:
1633 status = pjmedia_sdp_neg_modify_local_offer(inv->pool,inv->neg,
1634 new_offer);
1635 if (status != PJ_SUCCESS)
1636 goto on_return;
1637 break;
1638 }
1639 }
1640
1641 if (contact_hdr)
1642 inv->dlg->local.contact = contact_hdr;
1643
1644 status = pjsip_inv_invite(inv, p_tdata);
1645
1646on_return:
1647 pjsip_dlg_dec_lock(inv->dlg);
1648 return status;
Benny Prijono268ca612006-02-07 12:34:11 +00001649}
1650
1651/*
1652 * Create UPDATE.
1653 */
1654PJ_DEF(pj_status_t) pjsip_inv_update ( pjsip_inv_session *inv,
1655 const pj_str_t *new_contact,
1656 const pjmedia_sdp_session *new_offer,
1657 pjsip_tx_data **p_tdata )
1658{
1659 PJ_UNUSED_ARG(inv);
1660 PJ_UNUSED_ARG(new_contact);
1661 PJ_UNUSED_ARG(new_offer);
1662 PJ_UNUSED_ARG(p_tdata);
1663
1664 PJ_TODO(CREATE_UPDATE_REQUEST);
1665 return PJ_ENOTSUP;
1666}
1667
1668/*
1669 * Send a request or response message.
1670 */
1671PJ_DEF(pj_status_t) pjsip_inv_send_msg( pjsip_inv_session *inv,
Benny Prijonoe8b0d3b2006-03-17 17:57:52 +00001672 pjsip_tx_data *tdata)
Benny Prijono268ca612006-02-07 12:34:11 +00001673{
1674 pj_status_t status;
1675
1676 /* Verify arguments. */
1677 PJ_ASSERT_RETURN(inv && tdata, PJ_EINVAL);
1678
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00001679 PJ_LOG(5,(inv->obj_name, "Sending %s",
1680 pjsip_tx_data_get_info(tdata)));
1681
Benny Prijono268ca612006-02-07 12:34:11 +00001682 if (tdata->msg->type == PJSIP_REQUEST_MSG) {
Benny Prijonoa66c7152006-02-09 01:26:14 +00001683 struct tsx_inv_data *tsx_inv_data;
Benny Prijono268ca612006-02-07 12:34:11 +00001684
Benny Prijono64158af2006-04-04 11:06:34 +00001685 pjsip_dlg_inc_lock(inv->dlg);
Benny Prijono268ca612006-02-07 12:34:11 +00001686
Benny Prijono64158af2006-04-04 11:06:34 +00001687 tsx_inv_data = pj_pool_zalloc(inv->pool, sizeof(struct tsx_inv_data));
Benny Prijonoa66c7152006-02-09 01:26:14 +00001688 tsx_inv_data->inv = inv;
1689
Benny Prijono64158af2006-04-04 11:06:34 +00001690 pjsip_dlg_dec_lock(inv->dlg);
1691
1692 status = pjsip_dlg_send_request(inv->dlg, tdata, mod_inv.mod.id,
1693 tsx_inv_data);
1694 if (status != PJ_SUCCESS)
1695 return status;
Benny Prijono268ca612006-02-07 12:34:11 +00001696
1697 } else {
1698 pjsip_cseq_hdr *cseq;
1699
1700 /* Can only do this to send response to original INVITE
1701 * request.
1702 */
Benny Prijono64f851e2006-02-23 13:49:28 +00001703 PJ_ASSERT_RETURN((cseq=pjsip_msg_find_hdr(tdata->msg, PJSIP_H_CSEQ, NULL))!=NULL
1704 && (cseq->cseq == inv->invite_tsx->cseq),
Benny Prijono268ca612006-02-07 12:34:11 +00001705 PJ_EINVALIDOP);
1706
1707 status = pjsip_dlg_send_response(inv->dlg, inv->invite_tsx, tdata);
1708 if (status != PJ_SUCCESS)
1709 return status;
1710 }
1711
1712 /* Done (?) */
1713 return PJ_SUCCESS;
1714}
1715
1716
Benny Prijono8ad55352006-02-08 11:16:05 +00001717/*
1718 * Respond to incoming CANCEL request.
1719 */
1720static void inv_respond_incoming_cancel(pjsip_inv_session *inv,
1721 pjsip_transaction *cancel_tsx,
1722 pjsip_rx_data *rdata)
1723{
1724 pjsip_tx_data *tdata;
1725 pjsip_transaction *invite_tsx;
1726 pj_str_t key;
1727 pj_status_t status;
1728
1729 /* See if we have matching INVITE server transaction: */
1730
1731 pjsip_tsx_create_key(rdata->tp_info.pool, &key, PJSIP_ROLE_UAS,
1732 &pjsip_invite_method, rdata);
1733 invite_tsx = pjsip_tsx_layer_find_tsx(&key, PJ_TRUE);
1734
1735 if (invite_tsx == NULL) {
1736
1737 /* Invite transaction not found!
1738 * Respond CANCEL with 491 (RFC 3261 Section 9.2 page 42)
1739 */
1740 status = pjsip_dlg_create_response( inv->dlg, rdata, 200, NULL,
1741 &tdata);
1742
1743 } else {
1744 /* Always answer CANCEL will 200 (OK) regardless of
1745 * the state of the INVITE transaction.
1746 */
1747 status = pjsip_dlg_create_response( inv->dlg, rdata, 200, NULL,
1748 &tdata);
1749 }
1750
1751 /* See if we have created the response successfully. */
1752 if (status != PJ_SUCCESS) return;
1753
1754 /* Send the CANCEL response */
1755 status = pjsip_dlg_send_response(inv->dlg, cancel_tsx, tdata);
1756 if (status != PJ_SUCCESS) return;
1757
1758
1759 /* See if we need to terminate the UAS INVITE transaction
1760 * with 487 (Request Terminated) response.
1761 */
1762 if (invite_tsx && invite_tsx->status_code < 200) {
1763
1764 pj_assert(invite_tsx->last_tx != NULL);
1765
1766 tdata = invite_tsx->last_tx;
1767
1768 status = pjsip_dlg_modify_response(inv->dlg, tdata, 487, NULL);
1769 if (status == PJ_SUCCESS)
1770 pjsip_dlg_send_response(inv->dlg, invite_tsx, tdata);
1771 }
1772
1773 if (invite_tsx)
1774 pj_mutex_unlock(invite_tsx->mutex);
1775}
1776
1777
1778/*
1779 * Respond to incoming BYE request.
1780 */
1781static void inv_respond_incoming_bye( pjsip_inv_session *inv,
1782 pjsip_transaction *bye_tsx,
1783 pjsip_rx_data *rdata,
1784 pjsip_event *e )
1785{
1786 pj_status_t status;
1787 pjsip_tx_data *tdata;
1788
1789 /* Respond BYE with 200: */
1790
1791 status = pjsip_dlg_create_response(inv->dlg, rdata, 200, NULL, &tdata);
1792 if (status != PJ_SUCCESS) return;
1793
1794 status = pjsip_dlg_send_response(inv->dlg, bye_tsx, tdata);
1795 if (status != PJ_SUCCESS) return;
1796
1797 /* Terminate session: */
1798
Benny Prijonod4e0abd2006-03-05 11:53:36 +00001799 if (inv->state != PJSIP_INV_STATE_DISCONNECTED) {
Benny Prijono0b6340c2006-06-13 22:21:23 +00001800 inv_set_cause(inv, PJSIP_SC_OK, NULL);
Benny Prijono8ad55352006-02-08 11:16:05 +00001801 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
Benny Prijonod4e0abd2006-03-05 11:53:36 +00001802 }
Benny Prijono8ad55352006-02-08 11:16:05 +00001803}
1804
1805/*
Benny Prijono38998232006-02-08 22:44:25 +00001806 * Respond to BYE request.
1807 */
1808static void inv_handle_bye_response( pjsip_inv_session *inv,
1809 pjsip_transaction *tsx,
1810 pjsip_rx_data *rdata,
1811 pjsip_event *e )
1812{
1813 pj_status_t status;
1814
1815 if (e->body.tsx_state.type != PJSIP_EVENT_RX_MSG) {
Benny Prijono0b6340c2006-06-13 22:21:23 +00001816 inv_set_cause(inv, PJSIP_SC_OK, NULL);
Benny Prijono38998232006-02-08 22:44:25 +00001817 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
1818 return;
1819 }
1820
1821 /* Handle 401/407 challenge. */
1822 if (tsx->status_code == 401 || tsx->status_code == 407) {
1823
1824 pjsip_tx_data *tdata;
1825
1826 status = pjsip_auth_clt_reinit_req( &inv->dlg->auth_sess,
1827 rdata,
1828 tsx->last_tx,
1829 &tdata);
1830
1831 if (status != PJ_SUCCESS) {
1832
1833 /* Does not have proper credentials.
1834 * End the session anyway.
1835 */
Benny Prijono0b6340c2006-06-13 22:21:23 +00001836 inv_set_cause(inv, PJSIP_SC_OK, NULL);
Benny Prijono38998232006-02-08 22:44:25 +00001837 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
1838
1839 } else {
1840 /* Re-send BYE. */
Benny Prijonoe8b0d3b2006-03-17 17:57:52 +00001841 status = pjsip_inv_send_msg(inv, tdata);
Benny Prijono38998232006-02-08 22:44:25 +00001842 }
1843
1844 } else {
1845
1846 /* End the session. */
Benny Prijono0b6340c2006-06-13 22:21:23 +00001847 inv_set_cause(inv, PJSIP_SC_OK, NULL);
Benny Prijono38998232006-02-08 22:44:25 +00001848 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
1849 }
1850
1851}
1852
1853/*
Benny Prijono8ad55352006-02-08 11:16:05 +00001854 * State NULL is before anything is sent/received.
1855 */
1856static void inv_on_state_null( pjsip_inv_session *inv, pjsip_event *e)
Benny Prijono268ca612006-02-07 12:34:11 +00001857{
1858 pjsip_transaction *tsx = e->body.tsx_state.tsx;
1859 pjsip_dialog *dlg = pjsip_tsx_get_dlg(tsx);
1860
1861 PJ_ASSERT_ON_FAIL(tsx && dlg, return);
1862
1863 if (tsx->method.id == PJSIP_INVITE_METHOD) {
1864
Benny Prijono64f851e2006-02-23 13:49:28 +00001865 /* Keep the initial INVITE transaction. */
1866 if (inv->invite_tsx == NULL)
1867 inv->invite_tsx = tsx;
Benny Prijono268ca612006-02-07 12:34:11 +00001868
Benny Prijono64f851e2006-02-23 13:49:28 +00001869 if (dlg->role == PJSIP_ROLE_UAC) {
Benny Prijono268ca612006-02-07 12:34:11 +00001870
1871 switch (tsx->state) {
1872 case PJSIP_TSX_STATE_CALLING:
Benny Prijono8ad55352006-02-08 11:16:05 +00001873 inv_set_state(inv, PJSIP_INV_STATE_CALLING, e);
Benny Prijono268ca612006-02-07 12:34:11 +00001874 break;
1875 default:
Benny Prijono64f851e2006-02-23 13:49:28 +00001876 inv_on_state_calling(inv, e);
Benny Prijono268ca612006-02-07 12:34:11 +00001877 break;
1878 }
1879
1880 } else {
1881 switch (tsx->state) {
1882 case PJSIP_TSX_STATE_TRYING:
Benny Prijono8ad55352006-02-08 11:16:05 +00001883 inv_set_state(inv, PJSIP_INV_STATE_INCOMING, e);
Benny Prijono268ca612006-02-07 12:34:11 +00001884 break;
Benny Prijono38998232006-02-08 22:44:25 +00001885 case PJSIP_TSX_STATE_PROCEEDING:
1886 inv_set_state(inv, PJSIP_INV_STATE_INCOMING, e);
1887 if (tsx->status_code > 100)
1888 inv_set_state(inv, PJSIP_INV_STATE_EARLY, e);
1889 break;
Benny Prijono268ca612006-02-07 12:34:11 +00001890 default:
Benny Prijono64f851e2006-02-23 13:49:28 +00001891 inv_on_state_incoming(inv, e);
1892 break;
Benny Prijono268ca612006-02-07 12:34:11 +00001893 }
1894 }
1895
1896 } else {
1897 pj_assert(!"Unexpected transaction type");
1898 }
1899}
1900
Benny Prijono8ad55352006-02-08 11:16:05 +00001901/*
1902 * State CALLING is after sending initial INVITE request but before
1903 * any response (with tag) is received.
1904 */
1905static void inv_on_state_calling( pjsip_inv_session *inv, pjsip_event *e)
Benny Prijono268ca612006-02-07 12:34:11 +00001906{
1907 pjsip_transaction *tsx = e->body.tsx_state.tsx;
1908 pjsip_dialog *dlg = pjsip_tsx_get_dlg(tsx);
Benny Prijonoccf95622006-02-07 18:48:01 +00001909 pj_status_t status;
Benny Prijono268ca612006-02-07 12:34:11 +00001910
1911 PJ_ASSERT_ON_FAIL(tsx && dlg, return);
1912
Benny Prijono8ad55352006-02-08 11:16:05 +00001913 if (tsx == inv->invite_tsx) {
Benny Prijono268ca612006-02-07 12:34:11 +00001914
1915 switch (tsx->state) {
1916
Benny Prijono64f851e2006-02-23 13:49:28 +00001917 case PJSIP_TSX_STATE_CALLING:
1918 inv_set_state(inv, PJSIP_INV_STATE_CALLING, e);
1919 break;
1920
Benny Prijono268ca612006-02-07 12:34:11 +00001921 case PJSIP_TSX_STATE_PROCEEDING:
1922 if (dlg->remote.info->tag.slen) {
Benny Prijonoa66c7152006-02-09 01:26:14 +00001923
Benny Prijono8ad55352006-02-08 11:16:05 +00001924 inv_set_state(inv, PJSIP_INV_STATE_EARLY, e);
Benny Prijonoa66c7152006-02-09 01:26:14 +00001925
1926 inv_check_sdp_in_incoming_msg(inv, tsx,
1927 e->body.tsx_state.src.rdata);
1928
Benny Prijono268ca612006-02-07 12:34:11 +00001929 } else {
1930 /* Ignore 100 (Trying) response, as it doesn't change
1931 * session state. It only ceases retransmissions.
1932 */
1933 }
1934 break;
1935
1936 case PJSIP_TSX_STATE_COMPLETED:
1937 if (tsx->status_code/100 == 2) {
1938
1939 /* This should not happen.
1940 * When transaction receives 2xx, it should be terminated
1941 */
1942 pj_assert(0);
Benny Prijono8ad55352006-02-08 11:16:05 +00001943 inv_set_state(inv, PJSIP_INV_STATE_CONNECTING, e);
Benny Prijonoa66c7152006-02-09 01:26:14 +00001944
1945 inv_check_sdp_in_incoming_msg(inv, tsx,
1946 e->body.tsx_state.src.rdata);
Benny Prijono268ca612006-02-07 12:34:11 +00001947
Benny Prijonoccf95622006-02-07 18:48:01 +00001948 } else if (tsx->status_code==401 || tsx->status_code==407) {
1949
1950 /* Handle authentication failure:
1951 * Resend the request with Authorization header.
1952 */
1953 pjsip_tx_data *tdata;
1954
Benny Prijono8ad55352006-02-08 11:16:05 +00001955 status = pjsip_auth_clt_reinit_req(&inv->dlg->auth_sess,
Benny Prijonoccf95622006-02-07 18:48:01 +00001956 e->body.tsx_state.src.rdata,
1957 tsx->last_tx,
1958 &tdata);
1959
1960 if (status != PJ_SUCCESS) {
1961
1962 /* Does not have proper credentials.
1963 * End the session.
1964 */
Benny Prijono0b6340c2006-06-13 22:21:23 +00001965 inv_set_cause(inv, tsx->status_code, &tsx->status_text);
Benny Prijono8ad55352006-02-08 11:16:05 +00001966 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
Benny Prijonoccf95622006-02-07 18:48:01 +00001967
1968 } else {
1969
1970 /* Restart session. */
Benny Prijono8ad55352006-02-08 11:16:05 +00001971 inv->state = PJSIP_INV_STATE_NULL;
1972 inv->invite_tsx = NULL;
Benny Prijonoccf95622006-02-07 18:48:01 +00001973
1974 /* Send the request. */
Benny Prijonoe8b0d3b2006-03-17 17:57:52 +00001975 status = pjsip_inv_send_msg(inv, tdata);
Benny Prijonoccf95622006-02-07 18:48:01 +00001976 }
1977
Benny Prijono268ca612006-02-07 12:34:11 +00001978 } else {
Benny Prijonoccf95622006-02-07 18:48:01 +00001979
Benny Prijono0b6340c2006-06-13 22:21:23 +00001980 inv_set_cause(inv, tsx->status_code, &tsx->status_text);
Benny Prijono8ad55352006-02-08 11:16:05 +00001981 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
Benny Prijonoccf95622006-02-07 18:48:01 +00001982
Benny Prijono268ca612006-02-07 12:34:11 +00001983 }
1984 break;
1985
1986 case PJSIP_TSX_STATE_TERMINATED:
1987 /* INVITE transaction can be terminated either because UAC
1988 * transaction received 2xx response or because of transport
1989 * error.
1990 */
1991 if (tsx->status_code/100 == 2) {
1992 /* This must be receipt of 2xx response */
1993
1994 /* Set state to CONNECTING */
Benny Prijono8ad55352006-02-08 11:16:05 +00001995 inv_set_state(inv, PJSIP_INV_STATE_CONNECTING, e);
Benny Prijono268ca612006-02-07 12:34:11 +00001996
Benny Prijonoa66c7152006-02-09 01:26:14 +00001997 inv_check_sdp_in_incoming_msg(inv, tsx,
1998 e->body.tsx_state.src.rdata);
1999
Benny Prijono268ca612006-02-07 12:34:11 +00002000 /* Send ACK */
2001 pj_assert(e->body.tsx_state.type == PJSIP_EVENT_RX_MSG);
2002
Benny Prijono8ad55352006-02-08 11:16:05 +00002003 inv_send_ack(inv, e->body.tsx_state.src.rdata);
Benny Prijono5eff0432006-02-09 14:14:21 +00002004 inv_set_state(inv, PJSIP_INV_STATE_CONFIRMED, e);
Benny Prijono268ca612006-02-07 12:34:11 +00002005
Benny Prijonoa66c7152006-02-09 01:26:14 +00002006
Benny Prijono268ca612006-02-07 12:34:11 +00002007 } else {
Benny Prijono0b6340c2006-06-13 22:21:23 +00002008 inv_set_cause(inv, tsx->status_code, &tsx->status_text);
Benny Prijono8ad55352006-02-08 11:16:05 +00002009 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
Benny Prijono268ca612006-02-07 12:34:11 +00002010 }
2011 break;
2012
Benny Prijono34a404e2006-02-09 14:38:30 +00002013 default:
2014 break;
Benny Prijono268ca612006-02-07 12:34:11 +00002015 }
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00002016
2017 } else if (inv->role == PJSIP_ROLE_UAC &&
2018 tsx->role == PJSIP_ROLE_UAC &&
2019 tsx->method.id == PJSIP_CANCEL_METHOD)
2020 {
2021 /*
2022 * Handle case when outgoing CANCEL is answered with 481 (Call/
2023 * Transaction Does Not Exist), 408, or when it's timed out. In these
2024 * cases, disconnect session (i.e. dialog usage only).
2025 */
2026 if (tsx->status_code == PJSIP_SC_CALL_TSX_DOES_NOT_EXIST ||
2027 tsx->status_code == PJSIP_SC_REQUEST_TIMEOUT ||
2028 tsx->status_code == PJSIP_SC_TSX_TIMEOUT ||
Benny Prijono0b6340c2006-06-13 22:21:23 +00002029 tsx->status_code == PJSIP_SC_TSX_TRANSPORT_ERROR)
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00002030 {
Benny Prijono0b6340c2006-06-13 22:21:23 +00002031 inv_set_cause(inv, tsx->status_code, &tsx->status_text);
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00002032 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
2033 }
Benny Prijono268ca612006-02-07 12:34:11 +00002034 }
2035}
2036
Benny Prijono8ad55352006-02-08 11:16:05 +00002037/*
2038 * State INCOMING is after we received the request, but before
2039 * responses with tag are sent.
2040 */
2041static void inv_on_state_incoming( pjsip_inv_session *inv, pjsip_event *e)
Benny Prijono268ca612006-02-07 12:34:11 +00002042{
2043 pjsip_transaction *tsx = e->body.tsx_state.tsx;
2044 pjsip_dialog *dlg = pjsip_tsx_get_dlg(tsx);
2045
2046 PJ_ASSERT_ON_FAIL(tsx && dlg, return);
2047
Benny Prijono8ad55352006-02-08 11:16:05 +00002048 if (tsx == inv->invite_tsx) {
2049
2050 /*
2051 * Handle the INVITE state transition.
2052 */
2053
Benny Prijono268ca612006-02-07 12:34:11 +00002054 switch (tsx->state) {
Benny Prijono8ad55352006-02-08 11:16:05 +00002055
Benny Prijono64f851e2006-02-23 13:49:28 +00002056 case PJSIP_TSX_STATE_TRYING:
2057 inv_set_state(inv, PJSIP_INV_STATE_INCOMING, e);
2058 break;
2059
Benny Prijono268ca612006-02-07 12:34:11 +00002060 case PJSIP_TSX_STATE_PROCEEDING:
Benny Prijono8ad55352006-02-08 11:16:05 +00002061 /*
2062 * Transaction sent provisional response.
2063 */
Benny Prijono268ca612006-02-07 12:34:11 +00002064 if (tsx->status_code > 100)
Benny Prijono8ad55352006-02-08 11:16:05 +00002065 inv_set_state(inv, PJSIP_INV_STATE_EARLY, e);
Benny Prijono268ca612006-02-07 12:34:11 +00002066 break;
Benny Prijono8ad55352006-02-08 11:16:05 +00002067
Benny Prijono268ca612006-02-07 12:34:11 +00002068 case PJSIP_TSX_STATE_COMPLETED:
Benny Prijono8ad55352006-02-08 11:16:05 +00002069 /*
2070 * Transaction sent final response.
2071 */
Benny Prijonod4e0abd2006-03-05 11:53:36 +00002072 if (tsx->status_code/100 == 2) {
Benny Prijono8ad55352006-02-08 11:16:05 +00002073 inv_set_state(inv, PJSIP_INV_STATE_CONNECTING, e);
Benny Prijonod4e0abd2006-03-05 11:53:36 +00002074 } else {
Benny Prijono0b6340c2006-06-13 22:21:23 +00002075 inv_set_cause(inv, tsx->status_code, &tsx->status_text);
Benny Prijono8ad55352006-02-08 11:16:05 +00002076 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
Benny Prijonod4e0abd2006-03-05 11:53:36 +00002077 }
Benny Prijono268ca612006-02-07 12:34:11 +00002078 break;
Benny Prijono8ad55352006-02-08 11:16:05 +00002079
Benny Prijono268ca612006-02-07 12:34:11 +00002080 case PJSIP_TSX_STATE_TERMINATED:
Benny Prijono8ad55352006-02-08 11:16:05 +00002081 /*
2082 * This happens on transport error (e.g. failed to send
2083 * response)
2084 */
Benny Prijono0b6340c2006-06-13 22:21:23 +00002085 inv_set_cause(inv, tsx->status_code, &tsx->status_text);
Benny Prijono8ad55352006-02-08 11:16:05 +00002086 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
Benny Prijono268ca612006-02-07 12:34:11 +00002087 break;
Benny Prijono8ad55352006-02-08 11:16:05 +00002088
Benny Prijono268ca612006-02-07 12:34:11 +00002089 default:
Benny Prijono8ad55352006-02-08 11:16:05 +00002090 pj_assert(!"Unexpected INVITE state");
2091 break;
Benny Prijono268ca612006-02-07 12:34:11 +00002092 }
Benny Prijono8ad55352006-02-08 11:16:05 +00002093
2094 } else if (tsx->method.id == PJSIP_CANCEL_METHOD &&
2095 tsx->role == PJSIP_ROLE_UAS &&
2096 tsx->state < PJSIP_TSX_STATE_COMPLETED &&
2097 e->body.tsx_state.type == PJSIP_EVENT_RX_MSG )
2098 {
2099
2100 /*
2101 * Handle incoming CANCEL request.
2102 */
2103
2104 inv_respond_incoming_cancel(inv, tsx, e->body.tsx_state.src.rdata);
2105
Benny Prijono268ca612006-02-07 12:34:11 +00002106 }
2107}
2108
Benny Prijono8ad55352006-02-08 11:16:05 +00002109/*
2110 * State EARLY is for both UAS and UAC, after response with To tag
2111 * is sent/received.
2112 */
2113static void inv_on_state_early( pjsip_inv_session *inv, pjsip_event *e)
Benny Prijono268ca612006-02-07 12:34:11 +00002114{
2115 pjsip_transaction *tsx = e->body.tsx_state.tsx;
2116 pjsip_dialog *dlg = pjsip_tsx_get_dlg(tsx);
2117
2118 PJ_ASSERT_ON_FAIL(tsx && dlg, return);
2119
Benny Prijono8ad55352006-02-08 11:16:05 +00002120 if (tsx == inv->invite_tsx) {
2121
2122 /*
2123 * Handle the INVITE state progress.
2124 */
Benny Prijono268ca612006-02-07 12:34:11 +00002125
2126 switch (tsx->state) {
2127
2128 case PJSIP_TSX_STATE_PROCEEDING:
2129 /* Send/received another provisional response. */
Benny Prijono8ad55352006-02-08 11:16:05 +00002130 inv_set_state(inv, PJSIP_INV_STATE_EARLY, e);
Benny Prijonoa66c7152006-02-09 01:26:14 +00002131
2132 if (e->body.tsx_state.type == PJSIP_EVENT_RX_MSG) {
2133 inv_check_sdp_in_incoming_msg(inv, tsx,
2134 e->body.tsx_state.src.rdata);
2135 }
Benny Prijono268ca612006-02-07 12:34:11 +00002136 break;
2137
2138 case PJSIP_TSX_STATE_COMPLETED:
Benny Prijonoa66c7152006-02-09 01:26:14 +00002139 if (tsx->status_code/100 == 2) {
Benny Prijono8ad55352006-02-08 11:16:05 +00002140 inv_set_state(inv, PJSIP_INV_STATE_CONNECTING, e);
Benny Prijonoa66c7152006-02-09 01:26:14 +00002141 if (e->body.tsx_state.type == PJSIP_EVENT_RX_MSG) {
2142 inv_check_sdp_in_incoming_msg(inv, tsx,
2143 e->body.tsx_state.src.rdata);
2144 }
2145
Benny Prijonod4e0abd2006-03-05 11:53:36 +00002146 } else {
Benny Prijono0b6340c2006-06-13 22:21:23 +00002147 inv_set_cause(inv, tsx->status_code, &tsx->status_text);
Benny Prijono8ad55352006-02-08 11:16:05 +00002148 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
Benny Prijonod4e0abd2006-03-05 11:53:36 +00002149 }
Benny Prijono268ca612006-02-07 12:34:11 +00002150 break;
2151
Benny Prijonof3195072006-02-14 21:15:30 +00002152 case PJSIP_TSX_STATE_CONFIRMED:
2153 /* For some reason can go here */
2154
Benny Prijono268ca612006-02-07 12:34:11 +00002155 case PJSIP_TSX_STATE_TERMINATED:
2156 /* INVITE transaction can be terminated either because UAC
2157 * transaction received 2xx response or because of transport
2158 * error.
2159 */
2160 if (tsx->status_code/100 == 2) {
2161
2162 /* This must be receipt of 2xx response */
2163
2164 /* Set state to CONNECTING */
Benny Prijono8ad55352006-02-08 11:16:05 +00002165 inv_set_state(inv, PJSIP_INV_STATE_CONNECTING, e);
Benny Prijono268ca612006-02-07 12:34:11 +00002166
Benny Prijonoa66c7152006-02-09 01:26:14 +00002167 if (e->body.tsx_state.type == PJSIP_EVENT_RX_MSG) {
2168 inv_check_sdp_in_incoming_msg(inv, tsx,
2169 e->body.tsx_state.src.rdata);
2170 }
2171
Benny Prijono268ca612006-02-07 12:34:11 +00002172 /* if UAC, send ACK and move state to confirmed. */
2173 if (tsx->role == PJSIP_ROLE_UAC) {
2174 pj_assert(e->body.tsx_state.type == PJSIP_EVENT_RX_MSG);
2175
Benny Prijono8ad55352006-02-08 11:16:05 +00002176 inv_send_ack(inv, e->body.tsx_state.src.rdata);
Benny Prijono8ad55352006-02-08 11:16:05 +00002177 inv_set_state(inv, PJSIP_INV_STATE_CONFIRMED, e);
Benny Prijono268ca612006-02-07 12:34:11 +00002178 }
2179
2180 } else {
Benny Prijono0b6340c2006-06-13 22:21:23 +00002181 inv_set_cause(inv, tsx->status_code, &tsx->status_text);
Benny Prijono8ad55352006-02-08 11:16:05 +00002182 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
Benny Prijono268ca612006-02-07 12:34:11 +00002183 }
2184 break;
2185
2186 default:
Benny Prijono8ad55352006-02-08 11:16:05 +00002187 pj_assert(!"Unexpected INVITE tsx state");
Benny Prijono268ca612006-02-07 12:34:11 +00002188 }
2189
Benny Prijono8ad55352006-02-08 11:16:05 +00002190 } else if (inv->role == PJSIP_ROLE_UAS &&
2191 tsx->role == PJSIP_ROLE_UAS &&
2192 tsx->method.id == PJSIP_CANCEL_METHOD &&
2193 tsx->state < PJSIP_TSX_STATE_COMPLETED &&
2194 e->body.tsx_state.type == PJSIP_EVENT_RX_MSG )
2195 {
Benny Prijono268ca612006-02-07 12:34:11 +00002196
Benny Prijono8ad55352006-02-08 11:16:05 +00002197 /*
2198 * Handle incoming CANCEL request.
Benny Prijonoccf95622006-02-07 18:48:01 +00002199 */
2200
Benny Prijono8ad55352006-02-08 11:16:05 +00002201 inv_respond_incoming_cancel(inv, tsx, e->body.tsx_state.src.rdata);
2202
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00002203 } else if (inv->role == PJSIP_ROLE_UAC &&
2204 tsx->role == PJSIP_ROLE_UAC &&
2205 tsx->method.id == PJSIP_CANCEL_METHOD)
2206 {
2207 /*
2208 * Handle case when outgoing CANCEL is answered with 481 (Call/
2209 * Transaction Does Not Exist), 408, or when it's timed out. In these
2210 * cases, disconnect session (i.e. dialog usage only).
2211 */
2212 if (tsx->status_code == PJSIP_SC_CALL_TSX_DOES_NOT_EXIST ||
2213 tsx->status_code == PJSIP_SC_REQUEST_TIMEOUT ||
2214 tsx->status_code == PJSIP_SC_TSX_TIMEOUT ||
Benny Prijono0b6340c2006-06-13 22:21:23 +00002215 tsx->status_code == PJSIP_SC_TSX_TRANSPORT_ERROR)
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00002216 {
Benny Prijono0b6340c2006-06-13 22:21:23 +00002217 inv_set_cause(inv, tsx->status_code, &tsx->status_text);
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00002218 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
2219 }
Benny Prijono268ca612006-02-07 12:34:11 +00002220 }
2221}
2222
Benny Prijono8ad55352006-02-08 11:16:05 +00002223/*
2224 * State CONNECTING is after 2xx response to INVITE is sent/received.
2225 */
2226static void inv_on_state_connecting( pjsip_inv_session *inv, pjsip_event *e)
Benny Prijono268ca612006-02-07 12:34:11 +00002227{
2228 pjsip_transaction *tsx = e->body.tsx_state.tsx;
2229 pjsip_dialog *dlg = pjsip_tsx_get_dlg(tsx);
2230
2231 PJ_ASSERT_ON_FAIL(tsx && dlg, return);
2232
Benny Prijono8ad55352006-02-08 11:16:05 +00002233 if (tsx == inv->invite_tsx) {
Benny Prijono268ca612006-02-07 12:34:11 +00002234
Benny Prijono8ad55352006-02-08 11:16:05 +00002235 /*
2236 * Handle INVITE state progression.
2237 */
Benny Prijono268ca612006-02-07 12:34:11 +00002238 switch (tsx->state) {
2239
2240 case PJSIP_TSX_STATE_CONFIRMED:
Benny Prijono38998232006-02-08 22:44:25 +00002241 if (tsx->status_code/100 == 2)
2242 inv_set_state(inv, PJSIP_INV_STATE_CONFIRMED, e);
Benny Prijono268ca612006-02-07 12:34:11 +00002243 break;
2244
2245 case PJSIP_TSX_STATE_TERMINATED:
2246 /* INVITE transaction can be terminated either because UAC
2247 * transaction received 2xx response or because of transport
2248 * error.
2249 */
2250 if (tsx->status_code/100 != 2) {
Benny Prijono0b6340c2006-06-13 22:21:23 +00002251 inv_set_cause(inv, tsx->status_code, &tsx->status_text);
Benny Prijono8ad55352006-02-08 11:16:05 +00002252 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
Benny Prijono268ca612006-02-07 12:34:11 +00002253 }
2254 break;
2255
2256 case PJSIP_TSX_STATE_DESTROYED:
2257 /* Do nothing. */
2258 break;
2259
2260 default:
2261 pj_assert(!"Unexpected state");
2262 }
2263
Benny Prijono8ad55352006-02-08 11:16:05 +00002264 } else if (tsx->role == PJSIP_ROLE_UAS &&
2265 tsx->method.id == PJSIP_BYE_METHOD &&
2266 tsx->status_code < 200 &&
2267 e->body.tsx_state.type == PJSIP_EVENT_RX_MSG)
2268 {
2269
2270 /*
2271 * Handle incoming BYE.
2272 */
2273
2274 inv_respond_incoming_bye( inv, tsx, e->body.tsx_state.src.rdata, e );
2275
Benny Prijono38998232006-02-08 22:44:25 +00002276 } else if (tsx->method.id == PJSIP_BYE_METHOD &&
2277 tsx->role == PJSIP_ROLE_UAC &&
Benny Prijonoa66c7152006-02-09 01:26:14 +00002278 (tsx->state == PJSIP_TSX_STATE_COMPLETED ||
2279 tsx->state == PJSIP_TSX_STATE_TERMINATED))
Benny Prijono38998232006-02-08 22:44:25 +00002280 {
2281
2282 /*
2283 * Outgoing BYE
2284 */
2285 inv_handle_bye_response( inv, tsx, e->body.tsx_state.src.rdata, e);
2286
Benny Prijono268ca612006-02-07 12:34:11 +00002287 }
Benny Prijono70127222006-07-02 14:53:05 +00002288 else if (tsx->method.id == PJSIP_CANCEL_METHOD &&
2289 tsx->role == PJSIP_ROLE_UAS &&
2290 tsx->status_code < 200 &&
2291 e->body.tsx_state.type == PJSIP_EVENT_RX_MSG)
2292 {
Benny Prijono38998232006-02-08 22:44:25 +00002293
Benny Prijono70127222006-07-02 14:53:05 +00002294 /*
2295 * Handle strandled incoming CANCEL.
2296 */
2297 pjsip_rx_data *rdata = e->body.tsx_state.src.rdata;
2298 pjsip_tx_data *tdata;
2299 pj_status_t status;
2300
2301 status = pjsip_dlg_create_response(dlg, rdata, 200, NULL, &tdata);
2302 if (status != PJ_SUCCESS) return;
2303
2304 status = pjsip_dlg_send_response(dlg, tsx, tdata);
2305 if (status != PJ_SUCCESS) return;
2306
2307 }
Benny Prijono268ca612006-02-07 12:34:11 +00002308}
2309
Benny Prijono8ad55352006-02-08 11:16:05 +00002310/*
2311 * State CONFIRMED is after ACK is sent/received.
2312 */
2313static void inv_on_state_confirmed( pjsip_inv_session *inv, pjsip_event *e)
Benny Prijono268ca612006-02-07 12:34:11 +00002314{
2315 pjsip_transaction *tsx = e->body.tsx_state.tsx;
2316 pjsip_dialog *dlg = pjsip_tsx_get_dlg(tsx);
2317
2318 PJ_ASSERT_ON_FAIL(tsx && dlg, return);
2319
Benny Prijono268ca612006-02-07 12:34:11 +00002320
Benny Prijono8ad55352006-02-08 11:16:05 +00002321 if (tsx->method.id == PJSIP_BYE_METHOD &&
2322 tsx->role == PJSIP_ROLE_UAC &&
Benny Prijonoa66c7152006-02-09 01:26:14 +00002323 (tsx->state == PJSIP_TSX_STATE_COMPLETED ||
2324 tsx->state == PJSIP_TSX_STATE_TERMINATED))
Benny Prijono8ad55352006-02-08 11:16:05 +00002325 {
Benny Prijono38998232006-02-08 22:44:25 +00002326
Benny Prijono8ad55352006-02-08 11:16:05 +00002327 /*
Benny Prijono38998232006-02-08 22:44:25 +00002328 * Outgoing BYE
Benny Prijono8ad55352006-02-08 11:16:05 +00002329 */
Benny Prijono8ad55352006-02-08 11:16:05 +00002330
Benny Prijonoa66c7152006-02-09 01:26:14 +00002331 inv_handle_bye_response( inv, tsx, e->body.tsx_state.src.rdata, e);
Benny Prijono268ca612006-02-07 12:34:11 +00002332
Benny Prijono8ad55352006-02-08 11:16:05 +00002333 }
2334 else if (tsx->method.id == PJSIP_BYE_METHOD &&
2335 tsx->role == PJSIP_ROLE_UAS &&
2336 tsx->status_code < 200 &&
2337 e->body.tsx_state.type == PJSIP_EVENT_RX_MSG)
2338 {
Benny Prijonoccf95622006-02-07 18:48:01 +00002339
Benny Prijono8ad55352006-02-08 11:16:05 +00002340 /*
2341 * Handle incoming BYE.
2342 */
Benny Prijono268ca612006-02-07 12:34:11 +00002343
Benny Prijono8ad55352006-02-08 11:16:05 +00002344 inv_respond_incoming_bye( inv, tsx, e->body.tsx_state.src.rdata, e );
2345
Benny Prijono268ca612006-02-07 12:34:11 +00002346 }
Benny Prijono70127222006-07-02 14:53:05 +00002347 else if (tsx->method.id == PJSIP_CANCEL_METHOD &&
2348 tsx->role == PJSIP_ROLE_UAS &&
2349 tsx->status_code < 200 &&
2350 e->body.tsx_state.type == PJSIP_EVENT_RX_MSG)
2351 {
2352
2353 /*
2354 * Handle strandled incoming CANCEL.
2355 */
2356 pjsip_rx_data *rdata = e->body.tsx_state.src.rdata;
2357 pjsip_tx_data *tdata;
2358 pj_status_t status;
2359
2360 status = pjsip_dlg_create_response(dlg, rdata, 200, NULL, &tdata);
2361 if (status != PJ_SUCCESS) return;
2362
2363 status = pjsip_dlg_send_response(dlg, tsx, tdata);
2364 if (status != PJ_SUCCESS) return;
2365
2366 }
Benny Prijono26ff9062006-02-21 23:47:00 +00002367 else if (tsx->method.id == PJSIP_INVITE_METHOD &&
2368 tsx->role == PJSIP_ROLE_UAS)
2369 {
2370
2371 /*
2372 * Handle incoming re-INVITE
2373 */
2374 if (tsx->state == PJSIP_TSX_STATE_TRYING) {
2375
2376 pjsip_rx_data *rdata = e->body.tsx_state.src.rdata;
2377 pjsip_tx_data *tdata;
2378 pj_status_t status;
2379
2380 /* Check if we have INVITE pending. */
2381 if (inv->invite_tsx && inv->invite_tsx!=tsx) {
Benny Prijono46249942007-02-19 22:23:14 +00002382 pj_str_t reason;
2383
2384 reason = pj_str("Another INVITE transaction in progress");
Benny Prijono26ff9062006-02-21 23:47:00 +00002385
2386 /* Can not receive re-INVITE while another one is pending. */
Benny Prijono46249942007-02-19 22:23:14 +00002387 status = pjsip_dlg_create_response( inv->dlg, rdata, 500,
2388 &reason, &tdata);
Benny Prijono26ff9062006-02-21 23:47:00 +00002389 if (status != PJ_SUCCESS)
2390 return;
2391
2392 status = pjsip_dlg_send_response( inv->dlg, tsx, tdata);
2393
2394
2395 return;
2396 }
2397
2398 /* Save the invite transaction. */
2399 inv->invite_tsx = tsx;
2400
2401 /* Process SDP in incoming message. */
2402 status = inv_check_sdp_in_incoming_msg(inv, tsx, rdata);
2403
2404 if (status != PJ_SUCCESS) {
2405
2406 /* Not Acceptable */
2407 const pjsip_hdr *accept;
2408
2409 status = pjsip_dlg_create_response(inv->dlg, rdata,
2410 488, NULL, &tdata);
2411 if (status != PJ_SUCCESS)
2412 return;
2413
2414
2415 accept = pjsip_endpt_get_capability(dlg->endpt, PJSIP_H_ACCEPT,
2416 NULL);
2417 if (accept) {
2418 pjsip_msg_add_hdr(tdata->msg,
2419 pjsip_hdr_clone(tdata->pool, accept));
2420 }
2421
2422 status = pjsip_dlg_send_response(dlg, tsx, tdata);
2423
2424 return;
2425 }
2426
2427 /* Create 2xx ANSWER */
2428 status = pjsip_dlg_create_response(dlg, rdata, 200, NULL, &tdata);
2429 if (status != PJ_SUCCESS)
2430 return;
2431
2432 /* Process SDP in the answer */
Benny Prijono64f851e2006-02-23 13:49:28 +00002433 status = process_answer(inv, 200, tdata, NULL);
Benny Prijono26ff9062006-02-21 23:47:00 +00002434
Benny Prijono1d9b9a42006-09-25 13:40:12 +00002435 if (status != PJ_SUCCESS) {
2436 /*
2437 * SDP negotiation has failed.
2438 */
2439 pj_status_t rc;
2440 pj_str_t reason;
2441
2442 /* Delete the 2xx answer */
2443 pjsip_tx_data_dec_ref(tdata);
2444
2445 /* Create 500 response */
2446 reason = pj_str("SDP negotiation failed");
2447 rc = pjsip_dlg_create_response(dlg, rdata, 500, &reason,
2448 &tdata);
2449 if (rc == PJ_SUCCESS) {
2450 pjsip_warning_hdr *w;
2451 const pj_str_t *endpt_name;
2452
2453 endpt_name = pjsip_endpt_name(dlg->endpt);
2454 w = pjsip_warning_hdr_create_from_status(tdata->pool,
2455 endpt_name,
2456 status);
2457 if (w)
2458 pjsip_msg_add_hdr(tdata->msg, (pjsip_hdr*)w);
2459
2460 pjsip_inv_send_msg(inv, tdata);
2461 }
2462 return;
2463 }
2464
2465 /* Send 2xx regardless of the status of negotiation */
Benny Prijonoe8b0d3b2006-03-17 17:57:52 +00002466 status = pjsip_inv_send_msg(inv, tdata);
Benny Prijono26ff9062006-02-21 23:47:00 +00002467
2468 }
2469
2470 }
2471 else if (tsx->method.id == PJSIP_INVITE_METHOD &&
2472 tsx->role == PJSIP_ROLE_UAC)
2473 {
2474 /*
2475 * Handle outgoing re-INVITE
2476 */
2477 if (tsx->state == PJSIP_TSX_STATE_TERMINATED &&
2478 tsx->status_code/100 == 2)
2479 {
2480
2481 /* Re-INVITE was accepted. */
2482
2483 /* Process SDP */
2484 inv_check_sdp_in_incoming_msg(inv, tsx,
2485 e->body.tsx_state.src.rdata);
2486
2487 /* Send ACK */
2488 inv_send_ack(inv, e->body.tsx_state.src.rdata);
2489
2490 } else if (tsx->state == PJSIP_TSX_STATE_COMPLETED &&
2491 (tsx->status_code==401 || tsx->status_code==407))
2492 {
2493 pjsip_tx_data *tdata;
2494 pj_status_t status;
2495
2496 /* Handle authentication challenge. */
2497 status = pjsip_auth_clt_reinit_req( &dlg->auth_sess,
2498 e->body.tsx_state.src.rdata,
2499 tsx->last_tx,
2500 &tdata);
2501 if (status != PJ_SUCCESS)
2502 return;
2503
2504 /* Send re-INVITE */
Benny Prijonoe8b0d3b2006-03-17 17:57:52 +00002505 status = pjsip_inv_send_msg( inv, tdata);
Benny Prijono26ff9062006-02-21 23:47:00 +00002506
2507 } else if (tsx->status_code==PJSIP_SC_CALL_TSX_DOES_NOT_EXIST ||
2508 tsx->status_code==PJSIP_SC_REQUEST_TIMEOUT ||
2509 tsx->status_code >= 700)
2510 {
2511 /*
2512 * Handle responses that terminates dialog.
2513 */
Benny Prijono0b6340c2006-06-13 22:21:23 +00002514 inv_set_cause(inv, tsx->status_code, &tsx->status_text);
Benny Prijono26ff9062006-02-21 23:47:00 +00002515 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
2516 }
2517 }
Benny Prijono268ca612006-02-07 12:34:11 +00002518}
2519
Benny Prijono8ad55352006-02-08 11:16:05 +00002520/*
2521 * After session has been terminated, but before dialog is destroyed
2522 * (because dialog has other usages, or because dialog is waiting for
2523 * the last transaction to terminate).
2524 */
2525static void inv_on_state_disconnected( pjsip_inv_session *inv, pjsip_event *e)
Benny Prijono268ca612006-02-07 12:34:11 +00002526{
Benny Prijono8ad55352006-02-08 11:16:05 +00002527 pjsip_transaction *tsx = e->body.tsx_state.tsx;
2528 pjsip_dialog *dlg = pjsip_tsx_get_dlg(tsx);
Benny Prijono268ca612006-02-07 12:34:11 +00002529
Benny Prijono8ad55352006-02-08 11:16:05 +00002530 PJ_ASSERT_ON_FAIL(tsx && dlg, return);
2531
Benny Prijono70127222006-07-02 14:53:05 +00002532 if (tsx->role == PJSIP_ROLE_UAS &&
Benny Prijono8ad55352006-02-08 11:16:05 +00002533 tsx->status_code < 200 &&
2534 e->body.tsx_state.type == PJSIP_EVENT_RX_MSG)
2535 {
Benny Prijono70127222006-07-02 14:53:05 +00002536 pjsip_rx_data *rdata = e->body.tsx_state.src.rdata;
Benny Prijono8ad55352006-02-08 11:16:05 +00002537
2538 /*
Benny Prijono70127222006-07-02 14:53:05 +00002539 * Respond BYE with 200/OK
Benny Prijono8ad55352006-02-08 11:16:05 +00002540 */
Benny Prijono70127222006-07-02 14:53:05 +00002541 if (tsx->method.id == PJSIP_BYE_METHOD) {
2542 inv_respond_incoming_bye( inv, tsx, rdata, e );
2543 } else if (tsx->method.id == PJSIP_CANCEL_METHOD) {
2544 /*
2545 * Respond CANCEL with 200/OK too.
2546 */
2547 pjsip_tx_data *tdata;
2548 pj_status_t status;
Benny Prijono8ad55352006-02-08 11:16:05 +00002549
Benny Prijono70127222006-07-02 14:53:05 +00002550 status = pjsip_dlg_create_response(dlg, rdata, 200, NULL, &tdata);
2551 if (status != PJ_SUCCESS) return;
Benny Prijono8ad55352006-02-08 11:16:05 +00002552
Benny Prijono70127222006-07-02 14:53:05 +00002553 status = pjsip_dlg_send_response(dlg, tsx, tdata);
2554 if (status != PJ_SUCCESS) return;
2555
2556 }
Benny Prijono8ad55352006-02-08 11:16:05 +00002557 }
Benny Prijono268ca612006-02-07 12:34:11 +00002558}
2559