blob: 13feb457a697bf16aad607e313adbe44a389cc2e [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 Prijonoa1e69682007-05-11 15:14:34 +0000232 inv = (pjsip_inv_session*) dlg->mod_data[mod_inv.mod.id];
Benny Prijono38998232006-02-08 22:44:25 +0000233
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 Prijono853ab812007-06-12 15:40:38 +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{
Benny Prijonoa1e69682007-05-11 15:14:34 +0000414 return (pjsip_inv_session*) dlg->mod_data[mod_inv.mod.id];
Benny Prijono268ca612006-02-07 12:34:11 +0000415}
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 */
Benny Prijonoa1e69682007-05-11 15:14:34 +0000455 inv = PJ_POOL_ZALLOC_T(dlg->pool, 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 */
Benny Prijonoa1e69682007-05-11 15:14:34 +0000580 status = pjmedia_sdp_parse(rdata->tp_info.pool,
581 (char*)body->data, body->len, &sdp);
Benny Prijono268ca612006-02-07 12:34:11 +0000582 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 */
Benny Prijonoa1e69682007-05-11 15:14:34 +0000653 allow = (pjsip_allow_hdr*) pjsip_msg_find_hdr(msg, PJSIP_H_ALLOW, NULL);
Benny Prijono268ca612006-02-07 12:34:11 +0000654 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 */
Benny Prijonoa1e69682007-05-11 15:14:34 +0000671 sup_hdr = (pjsip_supported_hdr*)
672 pjsip_msg_find_hdr(msg, PJSIP_H_SUPPORTED, NULL);
Benny Prijono268ca612006-02-07 12:34:11 +0000673 if (sup_hdr) {
674 unsigned i;
675 pj_str_t STR_100REL = { "100rel", 6};
676 pj_str_t STR_TIMER = { "timer", 5 };
677
678 for (i=0; i<sup_hdr->count; ++i) {
679 if (pj_stricmp(&sup_hdr->values[i], &STR_100REL)==0)
680 rem_option |= PJSIP_INV_SUPPORT_100REL;
681 else if (pj_stricmp(&sup_hdr->values[i], &STR_TIMER)==0)
682 rem_option |= PJSIP_INV_SUPPORT_TIMER;
683 }
684 }
685
686 /* Check Require header */
Benny Prijonoa1e69682007-05-11 15:14:34 +0000687 req_hdr = (pjsip_require_hdr*)
688 pjsip_msg_find_hdr(msg, PJSIP_H_REQUIRE, NULL);
Benny Prijono268ca612006-02-07 12:34:11 +0000689 if (req_hdr) {
690 unsigned i;
Benny Prijono053f5222006-11-11 16:16:04 +0000691 const pj_str_t STR_100REL = { "100rel", 6};
692 const pj_str_t STR_TIMER = { "timer", 5 };
693 const pj_str_t STR_REPLACES = { "replaces", 8 };
Benny Prijono268ca612006-02-07 12:34:11 +0000694 unsigned unsupp_cnt = 0;
695 pj_str_t unsupp_tags[PJSIP_GENERIC_ARRAY_MAX_COUNT];
696
697 for (i=0; i<req_hdr->count; ++i) {
698 if ((*options & PJSIP_INV_SUPPORT_100REL) &&
699 pj_stricmp(&req_hdr->values[i], &STR_100REL)==0)
700 {
701 rem_option |= PJSIP_INV_REQUIRE_100REL;
702
703 } else if ((*options && PJSIP_INV_SUPPORT_TIMER) &&
704 pj_stricmp(&req_hdr->values[i], &STR_TIMER)==0)
705 {
706 rem_option |= PJSIP_INV_REQUIRE_TIMER;
707
Benny Prijono053f5222006-11-11 16:16:04 +0000708 } else if (pj_stricmp(&req_hdr->values[i], &STR_REPLACES)==0) {
709 pj_bool_t supp;
710
711 supp = pjsip_endpt_has_capability(endpt, PJSIP_H_SUPPORTED,
712 NULL, &STR_REPLACES);
713 if (!supp)
714 unsupp_tags[unsupp_cnt++] = req_hdr->values[i];
715
Benny Prijono268ca612006-02-07 12:34:11 +0000716 } else {
717 /* Unknown/unsupported extension tag! */
718 unsupp_tags[unsupp_cnt++] = req_hdr->values[i];
719 }
720 }
721
722 /* Check if there are required tags that we don't support */
723 if (unsupp_cnt) {
724
725 code = PJSIP_SC_BAD_EXTENSION;
726 status = PJSIP_ERRNO_FROM_SIP_STATUS(code);
727
728 if (p_tdata) {
729 pjsip_unsupported_hdr *unsupp_hdr;
730 const pjsip_hdr *h;
731
732 /* Add Unsupported header. */
733 unsupp_hdr = pjsip_unsupported_hdr_create(rdata->tp_info.pool);
734 PJ_ASSERT_RETURN(unsupp_hdr != NULL, PJ_ENOMEM);
735
736 unsupp_hdr->count = unsupp_cnt;
737 for (i=0; i<unsupp_cnt; ++i)
738 unsupp_hdr->values[i] = unsupp_tags[i];
739
740 pj_list_push_back(&res_hdr_list, unsupp_hdr);
741
742 /* Add Supported header. */
743 h = pjsip_endpt_get_capability(endpt, PJSIP_H_SUPPORTED,
744 NULL);
745 pj_assert(h);
746 if (h) {
Benny Prijonoa1e69682007-05-11 15:14:34 +0000747 sup_hdr = (pjsip_supported_hdr*)
748 pjsip_hdr_clone(rdata->tp_info.pool, h);
Benny Prijono268ca612006-02-07 12:34:11 +0000749 pj_list_push_back(&res_hdr_list, sup_hdr);
750 }
751 }
752
753 goto on_return;
754 }
755 }
756
757 /* Check if there are local requirements that are not supported
758 * by peer.
759 */
760 if ( ((*options & PJSIP_INV_REQUIRE_100REL)!=0 &&
761 (rem_option & PJSIP_INV_SUPPORT_100REL)==0) ||
762 ((*options & PJSIP_INV_REQUIRE_TIMER)!=0 &&
763 (rem_option & PJSIP_INV_SUPPORT_TIMER)==0))
764 {
765 code = PJSIP_SC_EXTENSION_REQUIRED;
766 status = PJSIP_ERRNO_FROM_SIP_STATUS(code);
767
768 if (p_tdata) {
769 const pjsip_hdr *h;
770
771 /* Add Require header. */
772 req_hdr = pjsip_require_hdr_create(rdata->tp_info.pool);
773 PJ_ASSERT_RETURN(req_hdr != NULL, PJ_ENOMEM);
774
775 if (*options & PJSIP_INV_REQUIRE_100REL)
776 req_hdr->values[req_hdr->count++] = pj_str("100rel");
777
778 if (*options & PJSIP_INV_REQUIRE_TIMER)
779 req_hdr->values[req_hdr->count++] = pj_str("timer");
780
781 pj_list_push_back(&res_hdr_list, req_hdr);
782
783 /* Add Supported header. */
784 h = pjsip_endpt_get_capability(endpt, PJSIP_H_SUPPORTED,
785 NULL);
786 pj_assert(h);
787 if (h) {
Benny Prijonoa1e69682007-05-11 15:14:34 +0000788 sup_hdr = (pjsip_supported_hdr*)
789 pjsip_hdr_clone(rdata->tp_info.pool, h);
Benny Prijono268ca612006-02-07 12:34:11 +0000790 pj_list_push_back(&res_hdr_list, sup_hdr);
791 }
792
793 }
794
795 goto on_return;
796 }
797
798on_return:
799
800 /* Create response if necessary */
801 if (code != 200 && p_tdata) {
802 pjsip_tx_data *tdata;
803 const pjsip_hdr *h;
804
805 if (dlg) {
806 status = pjsip_dlg_create_response(dlg, rdata, code, NULL,
807 &tdata);
808 } else {
809 status = pjsip_endpt_create_response(endpt, rdata, code, NULL,
810 &tdata);
811 }
812
813 if (status != PJ_SUCCESS)
814 return status;
815
816 /* Add response headers. */
817 h = res_hdr_list.next;
818 while (h != &res_hdr_list) {
819 pjsip_hdr *cloned;
820
Benny Prijonoa1e69682007-05-11 15:14:34 +0000821 cloned = (pjsip_hdr*) pjsip_hdr_clone(tdata->pool, h);
Benny Prijono268ca612006-02-07 12:34:11 +0000822 PJ_ASSERT_RETURN(cloned, PJ_ENOMEM);
823
824 pjsip_msg_add_hdr(tdata->msg, cloned);
825
826 h = h->next;
827 }
828
829 *p_tdata = tdata;
Benny Prijono70127222006-07-02 14:53:05 +0000830
831 /* Can not return PJ_SUCCESS when response message is produced.
832 * Ref: PROTOS test ~#2490
833 */
834 if (status == PJ_SUCCESS)
835 status = PJSIP_ERRNO_FROM_SIP_STATUS(code);
836
Benny Prijono268ca612006-02-07 12:34:11 +0000837 }
838
839 return status;
840}
841
842/*
Benny Prijono8ad55352006-02-08 11:16:05 +0000843 * Create UAS invite session.
Benny Prijono268ca612006-02-07 12:34:11 +0000844 */
845PJ_DEF(pj_status_t) pjsip_inv_create_uas( pjsip_dialog *dlg,
846 pjsip_rx_data *rdata,
847 const pjmedia_sdp_session *local_sdp,
848 unsigned options,
849 pjsip_inv_session **p_inv)
850{
851 pjsip_inv_session *inv;
Benny Prijonoa66c7152006-02-09 01:26:14 +0000852 struct tsx_inv_data *tsx_inv_data;
Benny Prijono268ca612006-02-07 12:34:11 +0000853 pjsip_msg *msg;
854 pjmedia_sdp_session *rem_sdp = NULL;
855 pj_status_t status;
856
857 /* Verify arguments. */
858 PJ_ASSERT_RETURN(dlg && rdata && p_inv, PJ_EINVAL);
859
860 /* Dialog MUST have been initialised. */
861 PJ_ASSERT_RETURN(pjsip_rdata_get_tsx(rdata) != NULL, PJ_EINVALIDOP);
862
863 msg = rdata->msg_info.msg;
864
865 /* rdata MUST contain INVITE request */
866 PJ_ASSERT_RETURN(msg->type == PJSIP_REQUEST_MSG &&
867 msg->line.req.method.id == PJSIP_INVITE_METHOD,
868 PJ_EINVALIDOP);
869
Benny Prijono8eae8382006-08-10 21:44:26 +0000870 /* Lock dialog */
871 pjsip_dlg_inc_lock(dlg);
872
Benny Prijono268ca612006-02-07 12:34:11 +0000873 /* Normalize options */
874 if (options & PJSIP_INV_REQUIRE_100REL)
875 options |= PJSIP_INV_SUPPORT_100REL;
876
877 if (options & PJSIP_INV_REQUIRE_TIMER)
878 options |= PJSIP_INV_SUPPORT_TIMER;
879
880 /* Create the session */
Benny Prijonoa1e69682007-05-11 15:14:34 +0000881 inv = PJ_POOL_ZALLOC_T(dlg->pool, pjsip_inv_session);
Benny Prijono8eae8382006-08-10 21:44:26 +0000882 pj_assert(inv != NULL);
Benny Prijono268ca612006-02-07 12:34:11 +0000883
884 inv->pool = dlg->pool;
885 inv->role = PJSIP_ROLE_UAS;
Benny Prijono38998232006-02-08 22:44:25 +0000886 inv->state = PJSIP_INV_STATE_NULL;
Benny Prijono268ca612006-02-07 12:34:11 +0000887 inv->dlg = dlg;
888 inv->options = options;
Benny Prijonod4e0abd2006-03-05 11:53:36 +0000889 inv->notify = PJ_TRUE;
Benny Prijonoba5926a2007-05-02 11:29:37 +0000890 inv->cause = (pjsip_status_code) 0;
Benny Prijono268ca612006-02-07 12:34:11 +0000891
Benny Prijonoe4f2abb2006-02-10 14:04:05 +0000892 /* Object name will use the same dialog pointer. */
Benny Prijonoed811d72006-03-10 12:57:12 +0000893 pj_ansi_snprintf(inv->obj_name, PJ_MAX_OBJ_NAME, "inv%p", dlg);
Benny Prijonoe4f2abb2006-02-10 14:04:05 +0000894
Benny Prijono268ca612006-02-07 12:34:11 +0000895 /* Parse SDP in message body, if present. */
896 if (msg->body) {
897 pjsip_msg_body *body = msg->body;
898
899 /* Parse and validate SDP */
Benny Prijonoa1e69682007-05-11 15:14:34 +0000900 status = pjmedia_sdp_parse(inv->pool, (char*)body->data, body->len,
Benny Prijono268ca612006-02-07 12:34:11 +0000901 &rem_sdp);
902 if (status == PJ_SUCCESS)
903 status = pjmedia_sdp_validate(rem_sdp);
904
Benny Prijono8eae8382006-08-10 21:44:26 +0000905 if (status != PJ_SUCCESS) {
906 pjsip_dlg_dec_lock(dlg);
Benny Prijono268ca612006-02-07 12:34:11 +0000907 return status;
Benny Prijono8eae8382006-08-10 21:44:26 +0000908 }
Benny Prijono268ca612006-02-07 12:34:11 +0000909 }
910
911 /* Create negotiator. */
912 if (rem_sdp) {
913 status = pjmedia_sdp_neg_create_w_remote_offer(inv->pool, local_sdp,
914 rem_sdp, &inv->neg);
915
916 } else if (local_sdp) {
917 status = pjmedia_sdp_neg_create_w_local_offer(inv->pool, local_sdp,
918 &inv->neg);
919 } else {
Benny Prijono95196582006-02-09 00:13:40 +0000920 status = PJ_SUCCESS;
Benny Prijono268ca612006-02-07 12:34:11 +0000921 }
922
Benny Prijono8eae8382006-08-10 21:44:26 +0000923 if (status != PJ_SUCCESS) {
924 pjsip_dlg_dec_lock(dlg);
Benny Prijono268ca612006-02-07 12:34:11 +0000925 return status;
Benny Prijono8eae8382006-08-10 21:44:26 +0000926 }
Benny Prijono268ca612006-02-07 12:34:11 +0000927
928 /* Register invite as dialog usage. */
929 status = pjsip_dlg_add_usage(dlg, &mod_inv.mod, inv);
Benny Prijono8eae8382006-08-10 21:44:26 +0000930 if (status != PJ_SUCCESS) {
931 pjsip_dlg_dec_lock(dlg);
Benny Prijono268ca612006-02-07 12:34:11 +0000932 return status;
Benny Prijono8eae8382006-08-10 21:44:26 +0000933 }
Benny Prijono268ca612006-02-07 12:34:11 +0000934
935 /* Increment session in the dialog. */
Benny Prijonoe4f2abb2006-02-10 14:04:05 +0000936 pjsip_dlg_inc_session(dlg, &mod_inv.mod);
Benny Prijono268ca612006-02-07 12:34:11 +0000937
938 /* Save the invite transaction. */
939 inv->invite_tsx = pjsip_rdata_get_tsx(rdata);
Benny Prijonoa66c7152006-02-09 01:26:14 +0000940
941 /* Attach our data to the transaction. */
Benny Prijonoa1e69682007-05-11 15:14:34 +0000942 tsx_inv_data = PJ_POOL_ZALLOC_T(inv->invite_tsx->pool, struct tsx_inv_data);
Benny Prijonoa66c7152006-02-09 01:26:14 +0000943 tsx_inv_data->inv = inv;
944 inv->invite_tsx->mod_data[mod_inv.mod.id] = tsx_inv_data;
Benny Prijono268ca612006-02-07 12:34:11 +0000945
946 /* Done */
Benny Prijono8eae8382006-08-10 21:44:26 +0000947 pjsip_dlg_dec_lock(dlg);
Benny Prijono268ca612006-02-07 12:34:11 +0000948 *p_inv = inv;
Benny Prijonoe4f2abb2006-02-10 14:04:05 +0000949
950 PJ_LOG(5,(inv->obj_name, "UAS invite session created for dialog %s",
951 dlg->obj_name));
952
Benny Prijono268ca612006-02-07 12:34:11 +0000953 return PJ_SUCCESS;
954}
955
Benny Prijonod4e0abd2006-03-05 11:53:36 +0000956/*
957 * Forcefully terminate the session.
958 */
959PJ_DEF(pj_status_t) pjsip_inv_terminate( pjsip_inv_session *inv,
960 int st_code,
961 pj_bool_t notify)
962{
963 PJ_ASSERT_RETURN(inv, PJ_EINVAL);
964
965 /* Lock dialog. */
966 pjsip_dlg_inc_lock(inv->dlg);
967
968 /* Set callback notify flag. */
969 inv->notify = notify;
970
971 /* If there's pending transaction, terminate the transaction.
972 * This may subsequently set the INVITE session state to
973 * disconnected.
974 */
975 if (inv->invite_tsx &&
976 inv->invite_tsx->state <= PJSIP_TSX_STATE_COMPLETED)
977 {
978 pjsip_tsx_terminate(inv->invite_tsx, st_code);
979
980 }
981
982 /* Set cause. */
Benny Prijono0b6340c2006-06-13 22:21:23 +0000983 inv_set_cause(inv, st_code, NULL);
Benny Prijonod4e0abd2006-03-05 11:53:36 +0000984
985 /* Forcefully terminate the session if state is not DISCONNECTED */
986 if (inv->state != PJSIP_INV_STATE_DISCONNECTED) {
987 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, NULL);
988 }
989
990 /* Done.
991 * The dec_lock() below will actually destroys the dialog if it
992 * has no other session.
993 */
994 pjsip_dlg_dec_lock(inv->dlg);
995
996 return PJ_SUCCESS;
997}
998
999
Benny Prijono268ca612006-02-07 12:34:11 +00001000static void *clone_sdp(pj_pool_t *pool, const void *data, unsigned len)
1001{
1002 PJ_UNUSED_ARG(len);
Benny Prijonoa1e69682007-05-11 15:14:34 +00001003 return pjmedia_sdp_session_clone(pool, (const pjmedia_sdp_session*)data);
Benny Prijono268ca612006-02-07 12:34:11 +00001004}
1005
1006static int print_sdp(pjsip_msg_body *body, char *buf, pj_size_t len)
1007{
Benny Prijonoa1e69682007-05-11 15:14:34 +00001008 return pjmedia_sdp_print((const pjmedia_sdp_session*)body->data, buf, len);
Benny Prijono268ca612006-02-07 12:34:11 +00001009}
1010
Benny Prijono56315612006-07-18 14:39:40 +00001011
1012PJ_DEF(pj_status_t) pjsip_create_sdp_body( pj_pool_t *pool,
1013 pjmedia_sdp_session *sdp,
1014 pjsip_msg_body **p_body)
1015{
1016 const pj_str_t STR_APPLICATION = { "application", 11};
1017 const pj_str_t STR_SDP = { "sdp", 3 };
1018 pjsip_msg_body *body;
1019
Benny Prijonoa1e69682007-05-11 15:14:34 +00001020 body = PJ_POOL_ZALLOC_T(pool, pjsip_msg_body);
Benny Prijono56315612006-07-18 14:39:40 +00001021 PJ_ASSERT_RETURN(body != NULL, PJ_ENOMEM);
1022
1023 body->content_type.type = STR_APPLICATION;
1024 body->content_type.subtype = STR_SDP;
1025 body->data = sdp;
1026 body->len = 0;
1027 body->clone_data = &clone_sdp;
1028 body->print_body = &print_sdp;
1029
1030 *p_body = body;
1031
1032 return PJ_SUCCESS;
1033}
1034
Benny Prijono268ca612006-02-07 12:34:11 +00001035static pjsip_msg_body *create_sdp_body(pj_pool_t *pool,
1036 const pjmedia_sdp_session *c_sdp)
1037{
1038 pjsip_msg_body *body;
Benny Prijono56315612006-07-18 14:39:40 +00001039 pj_status_t status;
Benny Prijono268ca612006-02-07 12:34:11 +00001040
Benny Prijono56315612006-07-18 14:39:40 +00001041 status = pjsip_create_sdp_body(pool,
1042 pjmedia_sdp_session_clone(pool, c_sdp),
1043 &body);
Benny Prijono268ca612006-02-07 12:34:11 +00001044
Benny Prijono56315612006-07-18 14:39:40 +00001045 if (status != PJ_SUCCESS)
1046 return NULL;
Benny Prijono268ca612006-02-07 12:34:11 +00001047
1048 return body;
1049}
1050
1051/*
1052 * Create initial INVITE request.
1053 */
1054PJ_DEF(pj_status_t) pjsip_inv_invite( pjsip_inv_session *inv,
1055 pjsip_tx_data **p_tdata )
1056{
1057 pjsip_tx_data *tdata;
1058 const pjsip_hdr *hdr;
Benny Prijono26ff9062006-02-21 23:47:00 +00001059 pj_bool_t has_sdp;
Benny Prijono268ca612006-02-07 12:34:11 +00001060 pj_status_t status;
1061
1062 /* Verify arguments. */
1063 PJ_ASSERT_RETURN(inv && p_tdata, PJ_EINVAL);
1064
Benny Prijono26ff9062006-02-21 23:47:00 +00001065 /* State MUST be NULL or CONFIRMED. */
1066 PJ_ASSERT_RETURN(inv->state == PJSIP_INV_STATE_NULL ||
1067 inv->state == PJSIP_INV_STATE_CONFIRMED,
1068 PJ_EINVALIDOP);
Benny Prijono268ca612006-02-07 12:34:11 +00001069
Benny Prijono64f851e2006-02-23 13:49:28 +00001070 /* Lock dialog. */
1071 pjsip_dlg_inc_lock(inv->dlg);
1072
Benny Prijono268ca612006-02-07 12:34:11 +00001073 /* Create the INVITE request. */
1074 status = pjsip_dlg_create_request(inv->dlg, &pjsip_invite_method, -1,
1075 &tdata);
1076 if (status != PJ_SUCCESS)
Benny Prijono64f851e2006-02-23 13:49:28 +00001077 goto on_return;
1078
Benny Prijono268ca612006-02-07 12:34:11 +00001079
Benny Prijono26ff9062006-02-21 23:47:00 +00001080 /* If this is the first INVITE, then copy the headers from inv_hdr.
1081 * These are the headers parsed from the request URI when the
1082 * dialog was created.
1083 */
1084 if (inv->state == PJSIP_INV_STATE_NULL) {
1085 hdr = inv->dlg->inv_hdr.next;
1086
1087 while (hdr != &inv->dlg->inv_hdr) {
Benny Prijonoa1e69682007-05-11 15:14:34 +00001088 pjsip_msg_add_hdr(tdata->msg, (pjsip_hdr*)
Benny Prijono26ff9062006-02-21 23:47:00 +00001089 pjsip_hdr_shallow_clone(tdata->pool, hdr));
1090 hdr = hdr->next;
1091 }
1092 }
1093
1094 /* See if we have SDP to send. */
1095 if (inv->neg) {
1096 pjmedia_sdp_neg_state neg_state;
1097
1098 neg_state = pjmedia_sdp_neg_get_state(inv->neg);
1099
1100 has_sdp = (neg_state == PJMEDIA_SDP_NEG_STATE_LOCAL_OFFER ||
1101 (neg_state == PJMEDIA_SDP_NEG_STATE_WAIT_NEGO &&
1102 pjmedia_sdp_neg_has_local_answer(inv->neg)));
1103
1104
1105 } else {
1106 has_sdp = PJ_FALSE;
1107 }
1108
Benny Prijono268ca612006-02-07 12:34:11 +00001109 /* Add SDP, if any. */
Benny Prijono26ff9062006-02-21 23:47:00 +00001110 if (has_sdp) {
Benny Prijono268ca612006-02-07 12:34:11 +00001111 const pjmedia_sdp_session *offer;
1112
1113 status = pjmedia_sdp_neg_get_neg_local(inv->neg, &offer);
1114 if (status != PJ_SUCCESS)
Benny Prijono64f851e2006-02-23 13:49:28 +00001115 goto on_return;
Benny Prijono268ca612006-02-07 12:34:11 +00001116
1117 tdata->msg->body = create_sdp_body(tdata->pool, offer);
1118 }
1119
1120 /* Add Allow header. */
1121 hdr = pjsip_endpt_get_capability(inv->dlg->endpt, PJSIP_H_ALLOW, NULL);
1122 if (hdr) {
Benny Prijonoa1e69682007-05-11 15:14:34 +00001123 pjsip_msg_add_hdr(tdata->msg, (pjsip_hdr*)
Benny Prijono268ca612006-02-07 12:34:11 +00001124 pjsip_hdr_shallow_clone(tdata->pool, hdr));
1125 }
1126
1127 /* Add Supported header */
1128 hdr = pjsip_endpt_get_capability(inv->dlg->endpt, PJSIP_H_SUPPORTED, NULL);
1129 if (hdr) {
Benny Prijonoa1e69682007-05-11 15:14:34 +00001130 pjsip_msg_add_hdr(tdata->msg, (pjsip_hdr*)
Benny Prijono268ca612006-02-07 12:34:11 +00001131 pjsip_hdr_shallow_clone(tdata->pool, hdr));
1132 }
1133
1134 /* Add Require header. */
1135 PJ_TODO(INVITE_ADD_REQUIRE_HEADER);
1136
1137 /* Done. */
1138 *p_tdata = tdata;
1139
Benny Prijono64f851e2006-02-23 13:49:28 +00001140
1141on_return:
1142 pjsip_dlg_dec_lock(inv->dlg);
1143 return status;
Benny Prijono268ca612006-02-07 12:34:11 +00001144}
1145
1146
1147/*
Benny Prijono95196582006-02-09 00:13:40 +00001148 * Negotiate SDP.
1149 */
1150static pj_status_t inv_negotiate_sdp( pjsip_inv_session *inv )
1151{
1152 pj_status_t status;
1153
1154 PJ_ASSERT_RETURN(pjmedia_sdp_neg_get_state(inv->neg) ==
1155 PJMEDIA_SDP_NEG_STATE_WAIT_NEGO,
1156 PJMEDIA_SDPNEG_EINSTATE);
1157
1158 status = pjmedia_sdp_neg_negotiate(inv->pool, inv->neg, 0);
1159
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00001160 PJ_LOG(5,(inv->obj_name, "SDP negotiation done, status=%d", status));
1161
Benny Prijonod4e0abd2006-03-05 11:53:36 +00001162 if (mod_inv.cb.on_media_update && inv->notify)
Benny Prijono95196582006-02-09 00:13:40 +00001163 (*mod_inv.cb.on_media_update)(inv, status);
1164
1165 return status;
1166}
1167
1168/*
Benny Prijonoa66c7152006-02-09 01:26:14 +00001169 * Check in incoming message for SDP offer/answer.
1170 */
Benny Prijono26ff9062006-02-21 23:47:00 +00001171static pj_status_t inv_check_sdp_in_incoming_msg( pjsip_inv_session *inv,
1172 pjsip_transaction *tsx,
1173 pjsip_rx_data *rdata)
Benny Prijonoa66c7152006-02-09 01:26:14 +00001174{
1175 struct tsx_inv_data *tsx_inv_data;
1176 static const pj_str_t str_application = { "application", 11 };
1177 static const pj_str_t str_sdp = { "sdp", 3 };
1178 pj_status_t status;
1179 pjsip_msg *msg;
1180 pjmedia_sdp_session *sdp;
1181
1182 /* Get/attach invite session's transaction data */
Benny Prijonoa1e69682007-05-11 15:14:34 +00001183 tsx_inv_data = (struct tsx_inv_data*) tsx->mod_data[mod_inv.mod.id];
Benny Prijonoa66c7152006-02-09 01:26:14 +00001184 if (tsx_inv_data == NULL) {
Benny Prijonoa1e69682007-05-11 15:14:34 +00001185 tsx_inv_data = PJ_POOL_ZALLOC_T(tsx->pool, struct tsx_inv_data);
Benny Prijonoa66c7152006-02-09 01:26:14 +00001186 tsx_inv_data->inv = inv;
1187 tsx->mod_data[mod_inv.mod.id] = tsx_inv_data;
1188 }
1189
1190 /* MUST NOT do multiple SDP offer/answer in a single transaction.
1191 */
1192
1193 if (tsx_inv_data->sdp_done)
Benny Prijono26ff9062006-02-21 23:47:00 +00001194 return PJ_SUCCESS;
Benny Prijonoa66c7152006-02-09 01:26:14 +00001195
1196 /* Check if SDP is present in the message. */
1197
1198 msg = rdata->msg_info.msg;
1199 if (msg->body == NULL) {
1200 /* Message doesn't have body. */
Benny Prijono26ff9062006-02-21 23:47:00 +00001201 return PJ_SUCCESS;
Benny Prijonoa66c7152006-02-09 01:26:14 +00001202 }
1203
1204 if (pj_stricmp(&msg->body->content_type.type, &str_application) ||
1205 pj_stricmp(&msg->body->content_type.subtype, &str_sdp))
1206 {
1207 /* Message body is not "application/sdp" */
Benny Prijono26ff9062006-02-21 23:47:00 +00001208 return PJMEDIA_SDP_EINSDP;
Benny Prijonoa66c7152006-02-09 01:26:14 +00001209 }
1210
1211 /* Parse the SDP body. */
1212
Benny Prijonoa1e69682007-05-11 15:14:34 +00001213 status = pjmedia_sdp_parse(rdata->tp_info.pool,
1214 (char*)msg->body->data,
Benny Prijonoa66c7152006-02-09 01:26:14 +00001215 msg->body->len, &sdp);
1216 if (status != PJ_SUCCESS) {
1217 char errmsg[PJ_ERR_MSG_SIZE];
1218 pj_strerror(status, errmsg, sizeof(errmsg));
1219 PJ_LOG(4,(THIS_FILE, "Error parsing SDP in %s: %s",
1220 pjsip_rx_data_get_info(rdata), errmsg));
Benny Prijono26ff9062006-02-21 23:47:00 +00001221 return PJMEDIA_SDP_EINSDP;
Benny Prijonoa66c7152006-02-09 01:26:14 +00001222 }
1223
1224 /* The SDP can be an offer or answer, depending on negotiator's state */
1225
1226 if (inv->neg == NULL ||
1227 pjmedia_sdp_neg_get_state(inv->neg) == PJMEDIA_SDP_NEG_STATE_DONE)
1228 {
1229
1230 /* This is an offer. */
1231
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00001232 PJ_LOG(5,(inv->obj_name, "Got SDP offer in %s",
1233 pjsip_rx_data_get_info(rdata)));
1234
Benny Prijonoa66c7152006-02-09 01:26:14 +00001235 if (inv->neg == NULL) {
1236 status=pjmedia_sdp_neg_create_w_remote_offer(inv->pool, NULL,
1237 sdp, &inv->neg);
1238 } else {
1239 status=pjmedia_sdp_neg_set_remote_offer(inv->pool, inv->neg, sdp);
1240 }
1241
1242 if (status != PJ_SUCCESS) {
1243 char errmsg[PJ_ERR_MSG_SIZE];
1244 pj_strerror(status, errmsg, sizeof(errmsg));
1245 PJ_LOG(4,(THIS_FILE, "Error processing SDP offer in %s: %s",
1246 pjsip_rx_data_get_info(rdata), errmsg));
Benny Prijono26ff9062006-02-21 23:47:00 +00001247 return PJMEDIA_SDP_EINSDP;
Benny Prijonoa66c7152006-02-09 01:26:14 +00001248 }
1249
1250 /* Inform application about remote offer. */
1251
Benny Prijonod4e0abd2006-03-05 11:53:36 +00001252 if (mod_inv.cb.on_rx_offer && inv->notify) {
Benny Prijono26ff9062006-02-21 23:47:00 +00001253
1254 (*mod_inv.cb.on_rx_offer)(inv, sdp);
1255
1256 }
Benny Prijonoa66c7152006-02-09 01:26:14 +00001257
1258 } else if (pjmedia_sdp_neg_get_state(inv->neg) ==
1259 PJMEDIA_SDP_NEG_STATE_LOCAL_OFFER)
1260 {
1261
1262 /* This is an answer.
1263 * Process and negotiate remote answer.
1264 */
1265
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00001266 PJ_LOG(5,(inv->obj_name, "Got SDP answer in %s",
1267 pjsip_rx_data_get_info(rdata)));
1268
Benny Prijonoa66c7152006-02-09 01:26:14 +00001269 status = pjmedia_sdp_neg_set_remote_answer(inv->pool, inv->neg, sdp);
1270
1271 if (status != PJ_SUCCESS) {
1272 char errmsg[PJ_ERR_MSG_SIZE];
1273 pj_strerror(status, errmsg, sizeof(errmsg));
1274 PJ_LOG(4,(THIS_FILE, "Error processing SDP answer in %s: %s",
1275 pjsip_rx_data_get_info(rdata), errmsg));
Benny Prijono26ff9062006-02-21 23:47:00 +00001276 return PJMEDIA_SDP_EINSDP;
Benny Prijonoa66c7152006-02-09 01:26:14 +00001277 }
1278
1279 /* Negotiate SDP */
1280
1281 inv_negotiate_sdp(inv);
1282
1283 /* Mark this transaction has having SDP offer/answer done. */
1284
1285 tsx_inv_data->sdp_done = 1;
1286
1287 } else {
1288
1289 PJ_LOG(5,(THIS_FILE, "Ignored SDP in %s: negotiator state is %s",
1290 pjsip_rx_data_get_info(rdata),
1291 pjmedia_sdp_neg_state_str(pjmedia_sdp_neg_get_state(inv->neg))));
1292 }
1293
Benny Prijono26ff9062006-02-21 23:47:00 +00001294 return PJ_SUCCESS;
Benny Prijonoa66c7152006-02-09 01:26:14 +00001295}
1296
1297
Benny Prijono26ff9062006-02-21 23:47:00 +00001298/*
1299 * Process INVITE answer, for both initial and subsequent re-INVITE
1300 */
1301static pj_status_t process_answer( pjsip_inv_session *inv,
1302 int st_code,
Benny Prijono64f851e2006-02-23 13:49:28 +00001303 pjsip_tx_data *tdata,
1304 const pjmedia_sdp_session *local_sdp)
Benny Prijono26ff9062006-02-21 23:47:00 +00001305{
1306 pj_status_t status;
Benny Prijonoab7399b2006-02-27 00:40:31 +00001307 const pjmedia_sdp_session *sdp = NULL;
Benny Prijono26ff9062006-02-21 23:47:00 +00001308
Benny Prijono64f851e2006-02-23 13:49:28 +00001309 /* If local_sdp is specified, then we MUST NOT have answered the
1310 * offer before.
Benny Prijono26ff9062006-02-21 23:47:00 +00001311 */
Benny Prijono64f851e2006-02-23 13:49:28 +00001312 if (local_sdp && (st_code/100==1 || st_code/100==2)) {
1313
1314 if (inv->neg == NULL) {
1315 status = pjmedia_sdp_neg_create_w_local_offer(inv->pool, local_sdp,
1316 &inv->neg);
1317 } else if (pjmedia_sdp_neg_get_state(inv->neg)==
1318 PJMEDIA_SDP_NEG_STATE_REMOTE_OFFER)
1319 {
1320 status = pjmedia_sdp_neg_set_local_answer(inv->pool, inv->neg,
1321 local_sdp);
1322 } else {
1323
1324 /* Can not specify local SDP at this state. */
1325 pj_assert(0);
1326 status = PJMEDIA_SDPNEG_EINSTATE;
1327 }
1328
1329 if (status != PJ_SUCCESS)
1330 return status;
1331
1332 }
1333
1334
1335 /* If SDP negotiator is ready, start negotiation. */
1336 if (st_code/100==2 || (st_code/10==18 && st_code!=180)) {
Benny Prijono26ff9062006-02-21 23:47:00 +00001337
1338 pjmedia_sdp_neg_state neg_state;
1339
Benny Prijono64f851e2006-02-23 13:49:28 +00001340 /* Start nego when appropriate. */
Benny Prijono26ff9062006-02-21 23:47:00 +00001341 neg_state = inv->neg ? pjmedia_sdp_neg_get_state(inv->neg) :
1342 PJMEDIA_SDP_NEG_STATE_NULL;
1343
1344 if (neg_state == PJMEDIA_SDP_NEG_STATE_LOCAL_OFFER) {
1345
1346 status = pjmedia_sdp_neg_get_neg_local(inv->neg, &sdp);
1347
1348 } else if (neg_state == PJMEDIA_SDP_NEG_STATE_WAIT_NEGO &&
1349 pjmedia_sdp_neg_has_local_answer(inv->neg) )
1350 {
1351
1352 status = inv_negotiate_sdp(inv);
1353 if (status != PJ_SUCCESS)
1354 return status;
1355
1356 status = pjmedia_sdp_neg_get_active_local(inv->neg, &sdp);
1357 }
Benny Prijono26ff9062006-02-21 23:47:00 +00001358 }
1359
Benny Prijono64f851e2006-02-23 13:49:28 +00001360 /* Include SDP when it's available for 2xx and 18x (but not 180) response.
Benny Prijono26ff9062006-02-21 23:47:00 +00001361 * Subsequent response will include this SDP.
1362 */
1363 if (sdp) {
1364 tdata->msg->body = create_sdp_body(tdata->pool, sdp);
1365 }
1366
Benny Prijono26ff9062006-02-21 23:47:00 +00001367
1368 return PJ_SUCCESS;
1369}
1370
Benny Prijonoa66c7152006-02-09 01:26:14 +00001371
1372/*
Benny Prijono64f851e2006-02-23 13:49:28 +00001373 * Create first response to INVITE
1374 */
1375PJ_DEF(pj_status_t) pjsip_inv_initial_answer( pjsip_inv_session *inv,
1376 pjsip_rx_data *rdata,
1377 int st_code,
1378 const pj_str_t *st_text,
1379 const pjmedia_sdp_session *sdp,
1380 pjsip_tx_data **p_tdata)
1381{
1382 pjsip_tx_data *tdata;
1383 pj_status_t status;
1384
1385 /* Verify arguments. */
1386 PJ_ASSERT_RETURN(inv && p_tdata, PJ_EINVAL);
1387
1388 /* Must have INVITE transaction. */
1389 PJ_ASSERT_RETURN(inv->invite_tsx, PJ_EBUG);
1390
1391 pjsip_dlg_inc_lock(inv->dlg);
1392
1393 /* Create response */
1394 status = pjsip_dlg_create_response(inv->dlg, rdata, st_code, st_text,
1395 &tdata);
1396 if (status != PJ_SUCCESS)
1397 goto on_return;
1398
1399 /* Process SDP in answer */
1400 status = process_answer(inv, st_code, tdata, sdp);
1401 if (status != PJ_SUCCESS) {
1402 pjsip_tx_data_dec_ref(tdata);
1403 goto on_return;
1404 }
1405
1406 *p_tdata = tdata;
1407
1408on_return:
1409 pjsip_dlg_dec_lock(inv->dlg);
1410 return status;
1411}
1412
1413
1414/*
Benny Prijono26ff9062006-02-21 23:47:00 +00001415 * Answer initial INVITE
1416 * Re-INVITE will be answered automatically, and will not use this function.
Benny Prijono268ca612006-02-07 12:34:11 +00001417 */
1418PJ_DEF(pj_status_t) pjsip_inv_answer( pjsip_inv_session *inv,
1419 int st_code,
1420 const pj_str_t *st_text,
1421 const pjmedia_sdp_session *local_sdp,
1422 pjsip_tx_data **p_tdata )
1423{
1424 pjsip_tx_data *last_res;
1425 pj_status_t status;
1426
1427 /* Verify arguments. */
1428 PJ_ASSERT_RETURN(inv && p_tdata, PJ_EINVAL);
1429
1430 /* Must have INVITE transaction. */
1431 PJ_ASSERT_RETURN(inv->invite_tsx, PJ_EBUG);
1432
1433 /* INVITE transaction MUST have transmitted a response (e.g. 100) */
1434 PJ_ASSERT_RETURN(inv->invite_tsx->last_tx, PJ_EINVALIDOP);
1435
Benny Prijono64f851e2006-02-23 13:49:28 +00001436 pjsip_dlg_inc_lock(inv->dlg);
Benny Prijono268ca612006-02-07 12:34:11 +00001437
1438 /* Modify last response. */
Benny Prijono26ff9062006-02-21 23:47:00 +00001439 last_res = inv->invite_tsx->last_tx;
Benny Prijono268ca612006-02-07 12:34:11 +00001440 status = pjsip_dlg_modify_response(inv->dlg, last_res, st_code, st_text);
1441 if (status != PJ_SUCCESS)
Benny Prijono64f851e2006-02-23 13:49:28 +00001442 goto on_return;
Benny Prijono268ca612006-02-07 12:34:11 +00001443
Benny Prijono268ca612006-02-07 12:34:11 +00001444
Benny Prijono26ff9062006-02-21 23:47:00 +00001445 /* Process SDP in answer */
Benny Prijono64f851e2006-02-23 13:49:28 +00001446 status = process_answer(inv, st_code, last_res, local_sdp);
1447 if (status != PJ_SUCCESS) {
1448 pjsip_tx_data_dec_ref(last_res);
1449 goto on_return;
1450 }
Benny Prijono268ca612006-02-07 12:34:11 +00001451
Benny Prijono268ca612006-02-07 12:34:11 +00001452
1453 *p_tdata = last_res;
1454
Benny Prijono64f851e2006-02-23 13:49:28 +00001455on_return:
1456 pjsip_dlg_dec_lock(inv->dlg);
1457 return status;
Benny Prijono268ca612006-02-07 12:34:11 +00001458}
1459
1460
1461/*
Benny Prijono26ff9062006-02-21 23:47:00 +00001462 * Set SDP answer.
1463 */
1464PJ_DEF(pj_status_t) pjsip_inv_set_sdp_answer( pjsip_inv_session *inv,
1465 const pjmedia_sdp_session *sdp )
1466{
1467 pj_status_t status;
1468
1469 PJ_ASSERT_RETURN(inv && sdp, PJ_EINVAL);
1470
1471 pjsip_dlg_inc_lock(inv->dlg);
1472 status = pjmedia_sdp_neg_set_local_answer( inv->pool, inv->neg, sdp);
1473 pjsip_dlg_dec_lock(inv->dlg);
1474
1475 return status;
1476}
1477
1478
1479/*
Benny Prijono268ca612006-02-07 12:34:11 +00001480 * End session.
1481 */
1482PJ_DEF(pj_status_t) pjsip_inv_end_session( pjsip_inv_session *inv,
1483 int st_code,
1484 const pj_str_t *st_text,
1485 pjsip_tx_data **p_tdata )
1486{
1487 pjsip_tx_data *tdata;
1488 pj_status_t status;
1489
1490 /* Verify arguments. */
1491 PJ_ASSERT_RETURN(inv && p_tdata, PJ_EINVAL);
1492
Benny Prijonod4e0abd2006-03-05 11:53:36 +00001493 /* Set cause code. */
Benny Prijono0b6340c2006-06-13 22:21:23 +00001494 inv_set_cause(inv, st_code, st_text);
Benny Prijonod4e0abd2006-03-05 11:53:36 +00001495
Benny Prijono268ca612006-02-07 12:34:11 +00001496 /* Create appropriate message. */
1497 switch (inv->state) {
1498 case PJSIP_INV_STATE_CALLING:
1499 case PJSIP_INV_STATE_EARLY:
1500 case PJSIP_INV_STATE_INCOMING:
1501
1502 if (inv->role == PJSIP_ROLE_UAC) {
1503
1504 /* For UAC when session has not been confirmed, create CANCEL. */
1505
1506 /* MUST have the original UAC INVITE transaction. */
1507 PJ_ASSERT_RETURN(inv->invite_tsx != NULL, PJ_EBUG);
1508
1509 /* But CANCEL should only be called when we have received a
1510 * provisional response. If we haven't received any responses,
1511 * just destroy the transaction.
1512 */
1513 if (inv->invite_tsx->status_code < 100) {
1514
Benny Prijono1dc8be02007-05-30 04:26:40 +00001515 pjsip_tsx_stop_retransmit(inv->invite_tsx);
1516 inv->cancelling = PJ_TRUE;
1517 inv->pending_cancel = PJ_TRUE;
Benny Prijonofccab712006-02-22 22:23:22 +00001518 *p_tdata = NULL;
Benny Prijono1dc8be02007-05-30 04:26:40 +00001519 PJ_LOG(4, (inv->obj_name, "Stopping retransmission, "
1520 "delaying CANCEL"));
Benny Prijonofccab712006-02-22 22:23:22 +00001521 return PJ_SUCCESS;
Benny Prijono268ca612006-02-07 12:34:11 +00001522 }
1523
1524 /* The CSeq here assumes that the dialog is started with an
1525 * INVITE session. This may not be correct; dialog can be
1526 * started as SUBSCRIBE session.
1527 * So fix this!
1528 */
1529 status = pjsip_endpt_create_cancel(inv->dlg->endpt,
1530 inv->invite_tsx->last_tx,
1531 &tdata);
1532
1533 } else {
1534
1535 /* For UAS, send a final response. */
1536 tdata = inv->invite_tsx->last_tx;
1537 PJ_ASSERT_RETURN(tdata != NULL, PJ_EINVALIDOP);
1538
Benny Prijono26ff9062006-02-21 23:47:00 +00001539 //status = pjsip_dlg_modify_response(inv->dlg, tdata, st_code,
1540 // st_text);
1541 status = pjsip_inv_answer(inv, st_code, st_text, NULL, &tdata);
Benny Prijono268ca612006-02-07 12:34:11 +00001542 }
1543 break;
1544
1545 case PJSIP_INV_STATE_CONNECTING:
1546 case PJSIP_INV_STATE_CONFIRMED:
1547 /* For established dialog, send BYE */
1548 status = pjsip_dlg_create_request(inv->dlg, &pjsip_bye_method, -1,
1549 &tdata);
1550 break;
1551
1552 case PJSIP_INV_STATE_DISCONNECTED:
Benny Prijono268ca612006-02-07 12:34:11 +00001553 /* No need to do anything. */
Benny Prijonod4e0abd2006-03-05 11:53:36 +00001554 return PJSIP_ESESSIONTERMINATED;
Benny Prijono268ca612006-02-07 12:34:11 +00001555
1556 default:
1557 pj_assert("!Invalid operation!");
1558 return PJ_EINVALIDOP;
1559 }
1560
1561 if (status != PJ_SUCCESS)
1562 return status;
1563
1564
1565 /* Done */
1566
Benny Prijono0606e702007-05-22 12:21:40 +00001567 inv->cancelling = PJ_TRUE;
Benny Prijono268ca612006-02-07 12:34:11 +00001568 *p_tdata = tdata;
1569
1570 return PJ_SUCCESS;
1571}
1572
1573
1574/*
1575 * Create re-INVITE.
1576 */
1577PJ_DEF(pj_status_t) pjsip_inv_reinvite( pjsip_inv_session *inv,
1578 const pj_str_t *new_contact,
1579 const pjmedia_sdp_session *new_offer,
1580 pjsip_tx_data **p_tdata )
1581{
Benny Prijono26ff9062006-02-21 23:47:00 +00001582 pj_status_t status;
1583 pjsip_contact_hdr *contact_hdr = NULL;
Benny Prijono268ca612006-02-07 12:34:11 +00001584
Benny Prijono26ff9062006-02-21 23:47:00 +00001585 /* Check arguments. */
1586 PJ_ASSERT_RETURN(inv && p_tdata, PJ_EINVAL);
1587
1588 /* Must NOT have a pending INVITE transaction */
Benny Prijono2b787822007-06-12 15:29:52 +00001589 if (inv->invite_tsx!=NULL)
1590 return PJ_EINVALIDOP;
Benny Prijono26ff9062006-02-21 23:47:00 +00001591
1592
1593 pjsip_dlg_inc_lock(inv->dlg);
1594
1595 if (new_contact) {
1596 pj_str_t tmp;
1597 const pj_str_t STR_CONTACT = { "Contact", 7 };
1598
1599 pj_strdup_with_null(inv->dlg->pool, &tmp, new_contact);
Benny Prijonoa1e69682007-05-11 15:14:34 +00001600 contact_hdr = (pjsip_contact_hdr*)
1601 pjsip_parse_hdr(inv->dlg->pool, &STR_CONTACT,
Benny Prijono26ff9062006-02-21 23:47:00 +00001602 tmp.ptr, tmp.slen, NULL);
1603 if (!contact_hdr) {
1604 status = PJSIP_EINVALIDURI;
1605 goto on_return;
1606 }
1607 }
1608
1609
1610 if (new_offer) {
1611 if (!inv->neg) {
1612 status = pjmedia_sdp_neg_create_w_local_offer(inv->pool, new_offer,
1613 &inv->neg);
1614 if (status != PJ_SUCCESS)
1615 goto on_return;
1616
1617 } else switch (pjmedia_sdp_neg_get_state(inv->neg)) {
1618
1619 case PJMEDIA_SDP_NEG_STATE_NULL:
1620 pj_assert(!"Unexpected SDP neg state NULL");
1621 status = PJ_EBUG;
1622 goto on_return;
1623
1624 case PJMEDIA_SDP_NEG_STATE_LOCAL_OFFER:
1625 PJ_LOG(4,(inv->obj_name,
1626 "pjsip_inv_reinvite: already have an offer, new "
1627 "offer is ignored"));
1628 break;
1629
1630 case PJMEDIA_SDP_NEG_STATE_REMOTE_OFFER:
1631 status = pjmedia_sdp_neg_set_local_answer(inv->pool, inv->neg,
1632 new_offer);
1633 if (status != PJ_SUCCESS)
1634 goto on_return;
1635 break;
1636
1637 case PJMEDIA_SDP_NEG_STATE_WAIT_NEGO:
1638 PJ_LOG(4,(inv->obj_name,
1639 "pjsip_inv_reinvite: SDP in WAIT_NEGO state, new "
1640 "offer is ignored"));
1641 break;
1642
1643 case PJMEDIA_SDP_NEG_STATE_DONE:
1644 status = pjmedia_sdp_neg_modify_local_offer(inv->pool,inv->neg,
1645 new_offer);
1646 if (status != PJ_SUCCESS)
1647 goto on_return;
1648 break;
1649 }
1650 }
1651
1652 if (contact_hdr)
1653 inv->dlg->local.contact = contact_hdr;
1654
1655 status = pjsip_inv_invite(inv, p_tdata);
1656
1657on_return:
1658 pjsip_dlg_dec_lock(inv->dlg);
1659 return status;
Benny Prijono268ca612006-02-07 12:34:11 +00001660}
1661
1662/*
1663 * Create UPDATE.
1664 */
1665PJ_DEF(pj_status_t) pjsip_inv_update ( pjsip_inv_session *inv,
1666 const pj_str_t *new_contact,
1667 const pjmedia_sdp_session *new_offer,
1668 pjsip_tx_data **p_tdata )
1669{
1670 PJ_UNUSED_ARG(inv);
1671 PJ_UNUSED_ARG(new_contact);
1672 PJ_UNUSED_ARG(new_offer);
1673 PJ_UNUSED_ARG(p_tdata);
1674
1675 PJ_TODO(CREATE_UPDATE_REQUEST);
1676 return PJ_ENOTSUP;
1677}
1678
1679/*
1680 * Send a request or response message.
1681 */
1682PJ_DEF(pj_status_t) pjsip_inv_send_msg( pjsip_inv_session *inv,
Benny Prijonoe8b0d3b2006-03-17 17:57:52 +00001683 pjsip_tx_data *tdata)
Benny Prijono268ca612006-02-07 12:34:11 +00001684{
1685 pj_status_t status;
1686
1687 /* Verify arguments. */
1688 PJ_ASSERT_RETURN(inv && tdata, PJ_EINVAL);
1689
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00001690 PJ_LOG(5,(inv->obj_name, "Sending %s",
1691 pjsip_tx_data_get_info(tdata)));
1692
Benny Prijono268ca612006-02-07 12:34:11 +00001693 if (tdata->msg->type == PJSIP_REQUEST_MSG) {
Benny Prijonoa66c7152006-02-09 01:26:14 +00001694 struct tsx_inv_data *tsx_inv_data;
Benny Prijono268ca612006-02-07 12:34:11 +00001695
Benny Prijono64158af2006-04-04 11:06:34 +00001696 pjsip_dlg_inc_lock(inv->dlg);
Benny Prijono268ca612006-02-07 12:34:11 +00001697
Benny Prijonoa1e69682007-05-11 15:14:34 +00001698 tsx_inv_data = PJ_POOL_ZALLOC_T(inv->pool, struct tsx_inv_data);
Benny Prijonoa66c7152006-02-09 01:26:14 +00001699 tsx_inv_data->inv = inv;
1700
Benny Prijono64158af2006-04-04 11:06:34 +00001701 pjsip_dlg_dec_lock(inv->dlg);
1702
1703 status = pjsip_dlg_send_request(inv->dlg, tdata, mod_inv.mod.id,
1704 tsx_inv_data);
1705 if (status != PJ_SUCCESS)
1706 return status;
Benny Prijono268ca612006-02-07 12:34:11 +00001707
1708 } else {
1709 pjsip_cseq_hdr *cseq;
1710
1711 /* Can only do this to send response to original INVITE
1712 * request.
1713 */
Benny Prijonoa1e69682007-05-11 15:14:34 +00001714 PJ_ASSERT_RETURN((cseq=(pjsip_cseq_hdr*)pjsip_msg_find_hdr(tdata->msg, PJSIP_H_CSEQ, NULL))!=NULL
Benny Prijono64f851e2006-02-23 13:49:28 +00001715 && (cseq->cseq == inv->invite_tsx->cseq),
Benny Prijono268ca612006-02-07 12:34:11 +00001716 PJ_EINVALIDOP);
1717
1718 status = pjsip_dlg_send_response(inv->dlg, inv->invite_tsx, tdata);
1719 if (status != PJ_SUCCESS)
1720 return status;
1721 }
1722
1723 /* Done (?) */
1724 return PJ_SUCCESS;
1725}
1726
1727
Benny Prijono8ad55352006-02-08 11:16:05 +00001728/*
1729 * Respond to incoming CANCEL request.
1730 */
1731static void inv_respond_incoming_cancel(pjsip_inv_session *inv,
1732 pjsip_transaction *cancel_tsx,
1733 pjsip_rx_data *rdata)
1734{
1735 pjsip_tx_data *tdata;
1736 pjsip_transaction *invite_tsx;
1737 pj_str_t key;
1738 pj_status_t status;
1739
1740 /* See if we have matching INVITE server transaction: */
1741
1742 pjsip_tsx_create_key(rdata->tp_info.pool, &key, PJSIP_ROLE_UAS,
1743 &pjsip_invite_method, rdata);
1744 invite_tsx = pjsip_tsx_layer_find_tsx(&key, PJ_TRUE);
1745
1746 if (invite_tsx == NULL) {
1747
1748 /* Invite transaction not found!
1749 * Respond CANCEL with 491 (RFC 3261 Section 9.2 page 42)
1750 */
1751 status = pjsip_dlg_create_response( inv->dlg, rdata, 200, NULL,
1752 &tdata);
1753
1754 } else {
1755 /* Always answer CANCEL will 200 (OK) regardless of
1756 * the state of the INVITE transaction.
1757 */
1758 status = pjsip_dlg_create_response( inv->dlg, rdata, 200, NULL,
1759 &tdata);
1760 }
1761
1762 /* See if we have created the response successfully. */
1763 if (status != PJ_SUCCESS) return;
1764
1765 /* Send the CANCEL response */
1766 status = pjsip_dlg_send_response(inv->dlg, cancel_tsx, tdata);
1767 if (status != PJ_SUCCESS) return;
1768
1769
1770 /* See if we need to terminate the UAS INVITE transaction
1771 * with 487 (Request Terminated) response.
1772 */
1773 if (invite_tsx && invite_tsx->status_code < 200) {
1774
1775 pj_assert(invite_tsx->last_tx != NULL);
1776
1777 tdata = invite_tsx->last_tx;
1778
1779 status = pjsip_dlg_modify_response(inv->dlg, tdata, 487, NULL);
1780 if (status == PJ_SUCCESS)
1781 pjsip_dlg_send_response(inv->dlg, invite_tsx, tdata);
1782 }
1783
1784 if (invite_tsx)
1785 pj_mutex_unlock(invite_tsx->mutex);
1786}
1787
1788
1789/*
1790 * Respond to incoming BYE request.
1791 */
1792static void inv_respond_incoming_bye( pjsip_inv_session *inv,
1793 pjsip_transaction *bye_tsx,
1794 pjsip_rx_data *rdata,
1795 pjsip_event *e )
1796{
1797 pj_status_t status;
1798 pjsip_tx_data *tdata;
1799
1800 /* Respond BYE with 200: */
1801
1802 status = pjsip_dlg_create_response(inv->dlg, rdata, 200, NULL, &tdata);
1803 if (status != PJ_SUCCESS) return;
1804
1805 status = pjsip_dlg_send_response(inv->dlg, bye_tsx, tdata);
1806 if (status != PJ_SUCCESS) return;
1807
1808 /* Terminate session: */
1809
Benny Prijonod4e0abd2006-03-05 11:53:36 +00001810 if (inv->state != PJSIP_INV_STATE_DISCONNECTED) {
Benny Prijono0b6340c2006-06-13 22:21:23 +00001811 inv_set_cause(inv, PJSIP_SC_OK, NULL);
Benny Prijono8ad55352006-02-08 11:16:05 +00001812 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
Benny Prijonod4e0abd2006-03-05 11:53:36 +00001813 }
Benny Prijono8ad55352006-02-08 11:16:05 +00001814}
1815
1816/*
Benny Prijono38998232006-02-08 22:44:25 +00001817 * Respond to BYE request.
1818 */
1819static void inv_handle_bye_response( pjsip_inv_session *inv,
1820 pjsip_transaction *tsx,
1821 pjsip_rx_data *rdata,
1822 pjsip_event *e )
1823{
1824 pj_status_t status;
1825
1826 if (e->body.tsx_state.type != PJSIP_EVENT_RX_MSG) {
Benny Prijono0b6340c2006-06-13 22:21:23 +00001827 inv_set_cause(inv, PJSIP_SC_OK, NULL);
Benny Prijono38998232006-02-08 22:44:25 +00001828 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
1829 return;
1830 }
1831
1832 /* Handle 401/407 challenge. */
1833 if (tsx->status_code == 401 || tsx->status_code == 407) {
1834
1835 pjsip_tx_data *tdata;
1836
1837 status = pjsip_auth_clt_reinit_req( &inv->dlg->auth_sess,
1838 rdata,
1839 tsx->last_tx,
1840 &tdata);
1841
1842 if (status != PJ_SUCCESS) {
1843
1844 /* Does not have proper credentials.
1845 * End the session anyway.
1846 */
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 } else {
1851 /* Re-send BYE. */
Benny Prijonoe8b0d3b2006-03-17 17:57:52 +00001852 status = pjsip_inv_send_msg(inv, tdata);
Benny Prijono38998232006-02-08 22:44:25 +00001853 }
1854
1855 } else {
1856
1857 /* End the session. */
Benny Prijono0b6340c2006-06-13 22:21:23 +00001858 inv_set_cause(inv, PJSIP_SC_OK, NULL);
Benny Prijono38998232006-02-08 22:44:25 +00001859 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
1860 }
1861
1862}
1863
1864/*
Benny Prijono8ad55352006-02-08 11:16:05 +00001865 * State NULL is before anything is sent/received.
1866 */
1867static void inv_on_state_null( pjsip_inv_session *inv, pjsip_event *e)
Benny Prijono268ca612006-02-07 12:34:11 +00001868{
1869 pjsip_transaction *tsx = e->body.tsx_state.tsx;
1870 pjsip_dialog *dlg = pjsip_tsx_get_dlg(tsx);
1871
1872 PJ_ASSERT_ON_FAIL(tsx && dlg, return);
1873
1874 if (tsx->method.id == PJSIP_INVITE_METHOD) {
1875
Benny Prijono64f851e2006-02-23 13:49:28 +00001876 /* Keep the initial INVITE transaction. */
1877 if (inv->invite_tsx == NULL)
1878 inv->invite_tsx = tsx;
Benny Prijono268ca612006-02-07 12:34:11 +00001879
Benny Prijono64f851e2006-02-23 13:49:28 +00001880 if (dlg->role == PJSIP_ROLE_UAC) {
Benny Prijono268ca612006-02-07 12:34:11 +00001881
1882 switch (tsx->state) {
1883 case PJSIP_TSX_STATE_CALLING:
Benny Prijono8ad55352006-02-08 11:16:05 +00001884 inv_set_state(inv, PJSIP_INV_STATE_CALLING, e);
Benny Prijono268ca612006-02-07 12:34:11 +00001885 break;
1886 default:
Benny Prijono64f851e2006-02-23 13:49:28 +00001887 inv_on_state_calling(inv, e);
Benny Prijono268ca612006-02-07 12:34:11 +00001888 break;
1889 }
1890
1891 } else {
1892 switch (tsx->state) {
1893 case PJSIP_TSX_STATE_TRYING:
Benny Prijono8ad55352006-02-08 11:16:05 +00001894 inv_set_state(inv, PJSIP_INV_STATE_INCOMING, e);
Benny Prijono268ca612006-02-07 12:34:11 +00001895 break;
Benny Prijono38998232006-02-08 22:44:25 +00001896 case PJSIP_TSX_STATE_PROCEEDING:
1897 inv_set_state(inv, PJSIP_INV_STATE_INCOMING, e);
1898 if (tsx->status_code > 100)
1899 inv_set_state(inv, PJSIP_INV_STATE_EARLY, e);
1900 break;
Benny Prijono268ca612006-02-07 12:34:11 +00001901 default:
Benny Prijono64f851e2006-02-23 13:49:28 +00001902 inv_on_state_incoming(inv, e);
1903 break;
Benny Prijono268ca612006-02-07 12:34:11 +00001904 }
1905 }
1906
1907 } else {
1908 pj_assert(!"Unexpected transaction type");
1909 }
1910}
1911
Benny Prijono8ad55352006-02-08 11:16:05 +00001912/*
1913 * State CALLING is after sending initial INVITE request but before
1914 * any response (with tag) is received.
1915 */
1916static void inv_on_state_calling( pjsip_inv_session *inv, pjsip_event *e)
Benny Prijono268ca612006-02-07 12:34:11 +00001917{
1918 pjsip_transaction *tsx = e->body.tsx_state.tsx;
1919 pjsip_dialog *dlg = pjsip_tsx_get_dlg(tsx);
Benny Prijonoccf95622006-02-07 18:48:01 +00001920 pj_status_t status;
Benny Prijono268ca612006-02-07 12:34:11 +00001921
1922 PJ_ASSERT_ON_FAIL(tsx && dlg, return);
1923
Benny Prijono8ad55352006-02-08 11:16:05 +00001924 if (tsx == inv->invite_tsx) {
Benny Prijono268ca612006-02-07 12:34:11 +00001925
1926 switch (tsx->state) {
1927
Benny Prijono64f851e2006-02-23 13:49:28 +00001928 case PJSIP_TSX_STATE_CALLING:
1929 inv_set_state(inv, PJSIP_INV_STATE_CALLING, e);
1930 break;
1931
Benny Prijono268ca612006-02-07 12:34:11 +00001932 case PJSIP_TSX_STATE_PROCEEDING:
Benny Prijono1dc8be02007-05-30 04:26:40 +00001933 if (inv->pending_cancel) {
1934 pjsip_tx_data *cancel;
1935
1936 inv->pending_cancel = PJ_FALSE;
1937
1938 status = pjsip_inv_end_session(inv, 487, NULL, &cancel);
1939 if (status == PJ_SUCCESS && cancel)
1940 status = pjsip_inv_send_msg(inv, cancel);
1941 }
1942
Benny Prijono268ca612006-02-07 12:34:11 +00001943 if (dlg->remote.info->tag.slen) {
Benny Prijonoa66c7152006-02-09 01:26:14 +00001944
Benny Prijono8ad55352006-02-08 11:16:05 +00001945 inv_set_state(inv, PJSIP_INV_STATE_EARLY, e);
Benny Prijonoa66c7152006-02-09 01:26:14 +00001946
1947 inv_check_sdp_in_incoming_msg(inv, tsx,
1948 e->body.tsx_state.src.rdata);
1949
Benny Prijono268ca612006-02-07 12:34:11 +00001950 } else {
1951 /* Ignore 100 (Trying) response, as it doesn't change
1952 * session state. It only ceases retransmissions.
1953 */
1954 }
1955 break;
1956
1957 case PJSIP_TSX_STATE_COMPLETED:
1958 if (tsx->status_code/100 == 2) {
1959
1960 /* This should not happen.
1961 * When transaction receives 2xx, it should be terminated
1962 */
1963 pj_assert(0);
Benny Prijono8ad55352006-02-08 11:16:05 +00001964 inv_set_state(inv, PJSIP_INV_STATE_CONNECTING, e);
Benny Prijonoa66c7152006-02-09 01:26:14 +00001965
1966 inv_check_sdp_in_incoming_msg(inv, tsx,
1967 e->body.tsx_state.src.rdata);
Benny Prijono268ca612006-02-07 12:34:11 +00001968
Benny Prijono0606e702007-05-22 12:21:40 +00001969 } else if ((tsx->status_code==401 || tsx->status_code==407) &&
1970 !inv->cancelling)
1971 {
Benny Prijonoccf95622006-02-07 18:48:01 +00001972
1973 /* Handle authentication failure:
1974 * Resend the request with Authorization header.
1975 */
1976 pjsip_tx_data *tdata;
1977
Benny Prijono8ad55352006-02-08 11:16:05 +00001978 status = pjsip_auth_clt_reinit_req(&inv->dlg->auth_sess,
Benny Prijonoccf95622006-02-07 18:48:01 +00001979 e->body.tsx_state.src.rdata,
1980 tsx->last_tx,
1981 &tdata);
1982
1983 if (status != PJ_SUCCESS) {
1984
1985 /* Does not have proper credentials.
1986 * End the session.
1987 */
Benny Prijono0b6340c2006-06-13 22:21:23 +00001988 inv_set_cause(inv, tsx->status_code, &tsx->status_text);
Benny Prijono8ad55352006-02-08 11:16:05 +00001989 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
Benny Prijonoccf95622006-02-07 18:48:01 +00001990
1991 } else {
1992
1993 /* Restart session. */
Benny Prijono8ad55352006-02-08 11:16:05 +00001994 inv->state = PJSIP_INV_STATE_NULL;
1995 inv->invite_tsx = NULL;
Benny Prijonoccf95622006-02-07 18:48:01 +00001996
1997 /* Send the request. */
Benny Prijonoe8b0d3b2006-03-17 17:57:52 +00001998 status = pjsip_inv_send_msg(inv, tdata);
Benny Prijonoccf95622006-02-07 18:48:01 +00001999 }
2000
Benny Prijono268ca612006-02-07 12:34:11 +00002001 } else {
Benny Prijonoccf95622006-02-07 18:48:01 +00002002
Benny Prijono0b6340c2006-06-13 22:21:23 +00002003 inv_set_cause(inv, tsx->status_code, &tsx->status_text);
Benny Prijono8ad55352006-02-08 11:16:05 +00002004 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
Benny Prijonoccf95622006-02-07 18:48:01 +00002005
Benny Prijono268ca612006-02-07 12:34:11 +00002006 }
2007 break;
2008
2009 case PJSIP_TSX_STATE_TERMINATED:
2010 /* INVITE transaction can be terminated either because UAC
2011 * transaction received 2xx response or because of transport
2012 * error.
2013 */
2014 if (tsx->status_code/100 == 2) {
2015 /* This must be receipt of 2xx response */
2016
2017 /* Set state to CONNECTING */
Benny Prijono8ad55352006-02-08 11:16:05 +00002018 inv_set_state(inv, PJSIP_INV_STATE_CONNECTING, e);
Benny Prijono268ca612006-02-07 12:34:11 +00002019
Benny Prijonoa66c7152006-02-09 01:26:14 +00002020 inv_check_sdp_in_incoming_msg(inv, tsx,
2021 e->body.tsx_state.src.rdata);
2022
Benny Prijono268ca612006-02-07 12:34:11 +00002023 /* Send ACK */
2024 pj_assert(e->body.tsx_state.type == PJSIP_EVENT_RX_MSG);
2025
Benny Prijono8ad55352006-02-08 11:16:05 +00002026 inv_send_ack(inv, e->body.tsx_state.src.rdata);
Benny Prijono5eff0432006-02-09 14:14:21 +00002027 inv_set_state(inv, PJSIP_INV_STATE_CONFIRMED, e);
Benny Prijono268ca612006-02-07 12:34:11 +00002028
Benny Prijonoa66c7152006-02-09 01:26:14 +00002029
Benny Prijono268ca612006-02-07 12:34:11 +00002030 } else {
Benny Prijono0b6340c2006-06-13 22:21:23 +00002031 inv_set_cause(inv, tsx->status_code, &tsx->status_text);
Benny Prijono8ad55352006-02-08 11:16:05 +00002032 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
Benny Prijono268ca612006-02-07 12:34:11 +00002033 }
2034 break;
2035
Benny Prijono34a404e2006-02-09 14:38:30 +00002036 default:
2037 break;
Benny Prijono268ca612006-02-07 12:34:11 +00002038 }
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00002039
2040 } else if (inv->role == PJSIP_ROLE_UAC &&
2041 tsx->role == PJSIP_ROLE_UAC &&
2042 tsx->method.id == PJSIP_CANCEL_METHOD)
2043 {
2044 /*
2045 * Handle case when outgoing CANCEL is answered with 481 (Call/
2046 * Transaction Does Not Exist), 408, or when it's timed out. In these
2047 * cases, disconnect session (i.e. dialog usage only).
2048 */
2049 if (tsx->status_code == PJSIP_SC_CALL_TSX_DOES_NOT_EXIST ||
2050 tsx->status_code == PJSIP_SC_REQUEST_TIMEOUT ||
2051 tsx->status_code == PJSIP_SC_TSX_TIMEOUT ||
Benny Prijono0b6340c2006-06-13 22:21:23 +00002052 tsx->status_code == PJSIP_SC_TSX_TRANSPORT_ERROR)
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00002053 {
Benny Prijono0b6340c2006-06-13 22:21:23 +00002054 inv_set_cause(inv, tsx->status_code, &tsx->status_text);
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00002055 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
2056 }
Benny Prijono268ca612006-02-07 12:34:11 +00002057 }
2058}
2059
Benny Prijono8ad55352006-02-08 11:16:05 +00002060/*
2061 * State INCOMING is after we received the request, but before
2062 * responses with tag are sent.
2063 */
2064static void inv_on_state_incoming( pjsip_inv_session *inv, pjsip_event *e)
Benny Prijono268ca612006-02-07 12:34:11 +00002065{
2066 pjsip_transaction *tsx = e->body.tsx_state.tsx;
2067 pjsip_dialog *dlg = pjsip_tsx_get_dlg(tsx);
2068
2069 PJ_ASSERT_ON_FAIL(tsx && dlg, return);
2070
Benny Prijono8ad55352006-02-08 11:16:05 +00002071 if (tsx == inv->invite_tsx) {
2072
2073 /*
2074 * Handle the INVITE state transition.
2075 */
2076
Benny Prijono268ca612006-02-07 12:34:11 +00002077 switch (tsx->state) {
Benny Prijono8ad55352006-02-08 11:16:05 +00002078
Benny Prijono64f851e2006-02-23 13:49:28 +00002079 case PJSIP_TSX_STATE_TRYING:
2080 inv_set_state(inv, PJSIP_INV_STATE_INCOMING, e);
2081 break;
2082
Benny Prijono268ca612006-02-07 12:34:11 +00002083 case PJSIP_TSX_STATE_PROCEEDING:
Benny Prijono8ad55352006-02-08 11:16:05 +00002084 /*
2085 * Transaction sent provisional response.
2086 */
Benny Prijono268ca612006-02-07 12:34:11 +00002087 if (tsx->status_code > 100)
Benny Prijono8ad55352006-02-08 11:16:05 +00002088 inv_set_state(inv, PJSIP_INV_STATE_EARLY, e);
Benny Prijono268ca612006-02-07 12:34:11 +00002089 break;
Benny Prijono8ad55352006-02-08 11:16:05 +00002090
Benny Prijono268ca612006-02-07 12:34:11 +00002091 case PJSIP_TSX_STATE_COMPLETED:
Benny Prijono8ad55352006-02-08 11:16:05 +00002092 /*
2093 * Transaction sent final response.
2094 */
Benny Prijonod4e0abd2006-03-05 11:53:36 +00002095 if (tsx->status_code/100 == 2) {
Benny Prijono8ad55352006-02-08 11:16:05 +00002096 inv_set_state(inv, PJSIP_INV_STATE_CONNECTING, e);
Benny Prijonod4e0abd2006-03-05 11:53:36 +00002097 } else {
Benny Prijono0b6340c2006-06-13 22:21:23 +00002098 inv_set_cause(inv, tsx->status_code, &tsx->status_text);
Benny Prijono8ad55352006-02-08 11:16:05 +00002099 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
Benny Prijonod4e0abd2006-03-05 11:53:36 +00002100 }
Benny Prijono268ca612006-02-07 12:34:11 +00002101 break;
Benny Prijono8ad55352006-02-08 11:16:05 +00002102
Benny Prijono268ca612006-02-07 12:34:11 +00002103 case PJSIP_TSX_STATE_TERMINATED:
Benny Prijono8ad55352006-02-08 11:16:05 +00002104 /*
2105 * This happens on transport error (e.g. failed to send
2106 * response)
2107 */
Benny Prijono0b6340c2006-06-13 22:21:23 +00002108 inv_set_cause(inv, tsx->status_code, &tsx->status_text);
Benny Prijono8ad55352006-02-08 11:16:05 +00002109 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
Benny Prijono268ca612006-02-07 12:34:11 +00002110 break;
Benny Prijono8ad55352006-02-08 11:16:05 +00002111
Benny Prijono268ca612006-02-07 12:34:11 +00002112 default:
Benny Prijono8ad55352006-02-08 11:16:05 +00002113 pj_assert(!"Unexpected INVITE state");
2114 break;
Benny Prijono268ca612006-02-07 12:34:11 +00002115 }
Benny Prijono8ad55352006-02-08 11:16:05 +00002116
2117 } else if (tsx->method.id == PJSIP_CANCEL_METHOD &&
2118 tsx->role == PJSIP_ROLE_UAS &&
2119 tsx->state < PJSIP_TSX_STATE_COMPLETED &&
2120 e->body.tsx_state.type == PJSIP_EVENT_RX_MSG )
2121 {
2122
2123 /*
2124 * Handle incoming CANCEL request.
2125 */
2126
2127 inv_respond_incoming_cancel(inv, tsx, e->body.tsx_state.src.rdata);
2128
Benny Prijono268ca612006-02-07 12:34:11 +00002129 }
2130}
2131
Benny Prijono8ad55352006-02-08 11:16:05 +00002132/*
2133 * State EARLY is for both UAS and UAC, after response with To tag
2134 * is sent/received.
2135 */
2136static void inv_on_state_early( pjsip_inv_session *inv, pjsip_event *e)
Benny Prijono268ca612006-02-07 12:34:11 +00002137{
2138 pjsip_transaction *tsx = e->body.tsx_state.tsx;
2139 pjsip_dialog *dlg = pjsip_tsx_get_dlg(tsx);
2140
2141 PJ_ASSERT_ON_FAIL(tsx && dlg, return);
2142
Benny Prijono8ad55352006-02-08 11:16:05 +00002143 if (tsx == inv->invite_tsx) {
2144
2145 /*
2146 * Handle the INVITE state progress.
2147 */
Benny Prijono268ca612006-02-07 12:34:11 +00002148
2149 switch (tsx->state) {
2150
2151 case PJSIP_TSX_STATE_PROCEEDING:
2152 /* Send/received another provisional response. */
Benny Prijono8ad55352006-02-08 11:16:05 +00002153 inv_set_state(inv, PJSIP_INV_STATE_EARLY, e);
Benny Prijonoa66c7152006-02-09 01:26:14 +00002154
2155 if (e->body.tsx_state.type == PJSIP_EVENT_RX_MSG) {
2156 inv_check_sdp_in_incoming_msg(inv, tsx,
2157 e->body.tsx_state.src.rdata);
2158 }
Benny Prijono268ca612006-02-07 12:34:11 +00002159 break;
2160
2161 case PJSIP_TSX_STATE_COMPLETED:
Benny Prijonoa66c7152006-02-09 01:26:14 +00002162 if (tsx->status_code/100 == 2) {
Benny Prijono8ad55352006-02-08 11:16:05 +00002163 inv_set_state(inv, PJSIP_INV_STATE_CONNECTING, e);
Benny Prijonoa66c7152006-02-09 01:26:14 +00002164 if (e->body.tsx_state.type == PJSIP_EVENT_RX_MSG) {
2165 inv_check_sdp_in_incoming_msg(inv, tsx,
2166 e->body.tsx_state.src.rdata);
2167 }
2168
Benny Prijonod4e0abd2006-03-05 11:53:36 +00002169 } else {
Benny Prijono0b6340c2006-06-13 22:21:23 +00002170 inv_set_cause(inv, tsx->status_code, &tsx->status_text);
Benny Prijono8ad55352006-02-08 11:16:05 +00002171 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
Benny Prijonod4e0abd2006-03-05 11:53:36 +00002172 }
Benny Prijono268ca612006-02-07 12:34:11 +00002173 break;
2174
Benny Prijonof3195072006-02-14 21:15:30 +00002175 case PJSIP_TSX_STATE_CONFIRMED:
2176 /* For some reason can go here */
2177
Benny Prijono268ca612006-02-07 12:34:11 +00002178 case PJSIP_TSX_STATE_TERMINATED:
2179 /* INVITE transaction can be terminated either because UAC
2180 * transaction received 2xx response or because of transport
2181 * error.
2182 */
2183 if (tsx->status_code/100 == 2) {
2184
2185 /* This must be receipt of 2xx response */
2186
2187 /* Set state to CONNECTING */
Benny Prijono8ad55352006-02-08 11:16:05 +00002188 inv_set_state(inv, PJSIP_INV_STATE_CONNECTING, e);
Benny Prijono268ca612006-02-07 12:34:11 +00002189
Benny Prijonoa66c7152006-02-09 01:26:14 +00002190 if (e->body.tsx_state.type == PJSIP_EVENT_RX_MSG) {
2191 inv_check_sdp_in_incoming_msg(inv, tsx,
2192 e->body.tsx_state.src.rdata);
2193 }
2194
Benny Prijono268ca612006-02-07 12:34:11 +00002195 /* if UAC, send ACK and move state to confirmed. */
2196 if (tsx->role == PJSIP_ROLE_UAC) {
2197 pj_assert(e->body.tsx_state.type == PJSIP_EVENT_RX_MSG);
2198
Benny Prijono8ad55352006-02-08 11:16:05 +00002199 inv_send_ack(inv, e->body.tsx_state.src.rdata);
Benny Prijono8ad55352006-02-08 11:16:05 +00002200 inv_set_state(inv, PJSIP_INV_STATE_CONFIRMED, e);
Benny Prijono268ca612006-02-07 12:34:11 +00002201 }
2202
2203 } else {
Benny Prijono0b6340c2006-06-13 22:21:23 +00002204 inv_set_cause(inv, tsx->status_code, &tsx->status_text);
Benny Prijono8ad55352006-02-08 11:16:05 +00002205 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
Benny Prijono268ca612006-02-07 12:34:11 +00002206 }
2207 break;
2208
2209 default:
Benny Prijono8ad55352006-02-08 11:16:05 +00002210 pj_assert(!"Unexpected INVITE tsx state");
Benny Prijono268ca612006-02-07 12:34:11 +00002211 }
2212
Benny Prijono8ad55352006-02-08 11:16:05 +00002213 } else if (inv->role == PJSIP_ROLE_UAS &&
2214 tsx->role == PJSIP_ROLE_UAS &&
2215 tsx->method.id == PJSIP_CANCEL_METHOD &&
2216 tsx->state < PJSIP_TSX_STATE_COMPLETED &&
2217 e->body.tsx_state.type == PJSIP_EVENT_RX_MSG )
2218 {
Benny Prijono268ca612006-02-07 12:34:11 +00002219
Benny Prijono8ad55352006-02-08 11:16:05 +00002220 /*
2221 * Handle incoming CANCEL request.
Benny Prijonoccf95622006-02-07 18:48:01 +00002222 */
2223
Benny Prijono8ad55352006-02-08 11:16:05 +00002224 inv_respond_incoming_cancel(inv, tsx, e->body.tsx_state.src.rdata);
2225
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00002226 } else if (inv->role == PJSIP_ROLE_UAC &&
2227 tsx->role == PJSIP_ROLE_UAC &&
2228 tsx->method.id == PJSIP_CANCEL_METHOD)
2229 {
2230 /*
2231 * Handle case when outgoing CANCEL is answered with 481 (Call/
2232 * Transaction Does Not Exist), 408, or when it's timed out. In these
2233 * cases, disconnect session (i.e. dialog usage only).
2234 */
2235 if (tsx->status_code == PJSIP_SC_CALL_TSX_DOES_NOT_EXIST ||
2236 tsx->status_code == PJSIP_SC_REQUEST_TIMEOUT ||
2237 tsx->status_code == PJSIP_SC_TSX_TIMEOUT ||
Benny Prijono0b6340c2006-06-13 22:21:23 +00002238 tsx->status_code == PJSIP_SC_TSX_TRANSPORT_ERROR)
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00002239 {
Benny Prijono0b6340c2006-06-13 22:21:23 +00002240 inv_set_cause(inv, tsx->status_code, &tsx->status_text);
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00002241 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
2242 }
Benny Prijono268ca612006-02-07 12:34:11 +00002243 }
2244}
2245
Benny Prijono8ad55352006-02-08 11:16:05 +00002246/*
2247 * State CONNECTING is after 2xx response to INVITE is sent/received.
2248 */
2249static void inv_on_state_connecting( pjsip_inv_session *inv, pjsip_event *e)
Benny Prijono268ca612006-02-07 12:34:11 +00002250{
2251 pjsip_transaction *tsx = e->body.tsx_state.tsx;
2252 pjsip_dialog *dlg = pjsip_tsx_get_dlg(tsx);
2253
2254 PJ_ASSERT_ON_FAIL(tsx && dlg, return);
2255
Benny Prijono8ad55352006-02-08 11:16:05 +00002256 if (tsx == inv->invite_tsx) {
Benny Prijono268ca612006-02-07 12:34:11 +00002257
Benny Prijono8ad55352006-02-08 11:16:05 +00002258 /*
2259 * Handle INVITE state progression.
2260 */
Benny Prijono268ca612006-02-07 12:34:11 +00002261 switch (tsx->state) {
2262
2263 case PJSIP_TSX_STATE_CONFIRMED:
Benny Prijono38998232006-02-08 22:44:25 +00002264 if (tsx->status_code/100 == 2)
2265 inv_set_state(inv, PJSIP_INV_STATE_CONFIRMED, e);
Benny Prijono268ca612006-02-07 12:34:11 +00002266 break;
2267
2268 case PJSIP_TSX_STATE_TERMINATED:
2269 /* INVITE transaction can be terminated either because UAC
2270 * transaction received 2xx response or because of transport
2271 * error.
2272 */
2273 if (tsx->status_code/100 != 2) {
Benny Prijono0b6340c2006-06-13 22:21:23 +00002274 inv_set_cause(inv, tsx->status_code, &tsx->status_text);
Benny Prijono8ad55352006-02-08 11:16:05 +00002275 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
Benny Prijono268ca612006-02-07 12:34:11 +00002276 }
2277 break;
2278
2279 case PJSIP_TSX_STATE_DESTROYED:
2280 /* Do nothing. */
2281 break;
2282
2283 default:
2284 pj_assert(!"Unexpected state");
2285 }
2286
Benny Prijono8ad55352006-02-08 11:16:05 +00002287 } else if (tsx->role == PJSIP_ROLE_UAS &&
2288 tsx->method.id == PJSIP_BYE_METHOD &&
2289 tsx->status_code < 200 &&
2290 e->body.tsx_state.type == PJSIP_EVENT_RX_MSG)
2291 {
2292
2293 /*
2294 * Handle incoming BYE.
2295 */
2296
2297 inv_respond_incoming_bye( inv, tsx, e->body.tsx_state.src.rdata, e );
2298
Benny Prijono38998232006-02-08 22:44:25 +00002299 } else if (tsx->method.id == PJSIP_BYE_METHOD &&
2300 tsx->role == PJSIP_ROLE_UAC &&
Benny Prijonoa66c7152006-02-09 01:26:14 +00002301 (tsx->state == PJSIP_TSX_STATE_COMPLETED ||
2302 tsx->state == PJSIP_TSX_STATE_TERMINATED))
Benny Prijono38998232006-02-08 22:44:25 +00002303 {
2304
2305 /*
2306 * Outgoing BYE
2307 */
2308 inv_handle_bye_response( inv, tsx, e->body.tsx_state.src.rdata, e);
2309
Benny Prijono268ca612006-02-07 12:34:11 +00002310 }
Benny Prijono70127222006-07-02 14:53:05 +00002311 else if (tsx->method.id == PJSIP_CANCEL_METHOD &&
2312 tsx->role == PJSIP_ROLE_UAS &&
2313 tsx->status_code < 200 &&
2314 e->body.tsx_state.type == PJSIP_EVENT_RX_MSG)
2315 {
Benny Prijono38998232006-02-08 22:44:25 +00002316
Benny Prijono70127222006-07-02 14:53:05 +00002317 /*
2318 * Handle strandled incoming CANCEL.
2319 */
2320 pjsip_rx_data *rdata = e->body.tsx_state.src.rdata;
2321 pjsip_tx_data *tdata;
2322 pj_status_t status;
2323
2324 status = pjsip_dlg_create_response(dlg, rdata, 200, NULL, &tdata);
2325 if (status != PJ_SUCCESS) return;
2326
2327 status = pjsip_dlg_send_response(dlg, tsx, tdata);
2328 if (status != PJ_SUCCESS) return;
2329
2330 }
Benny Prijono268ca612006-02-07 12:34:11 +00002331}
2332
Benny Prijono8ad55352006-02-08 11:16:05 +00002333/*
2334 * State CONFIRMED is after ACK is sent/received.
2335 */
2336static void inv_on_state_confirmed( pjsip_inv_session *inv, pjsip_event *e)
Benny Prijono268ca612006-02-07 12:34:11 +00002337{
2338 pjsip_transaction *tsx = e->body.tsx_state.tsx;
2339 pjsip_dialog *dlg = pjsip_tsx_get_dlg(tsx);
2340
2341 PJ_ASSERT_ON_FAIL(tsx && dlg, return);
2342
Benny Prijono268ca612006-02-07 12:34:11 +00002343
Benny Prijono8ad55352006-02-08 11:16:05 +00002344 if (tsx->method.id == PJSIP_BYE_METHOD &&
2345 tsx->role == PJSIP_ROLE_UAC &&
Benny Prijonoa66c7152006-02-09 01:26:14 +00002346 (tsx->state == PJSIP_TSX_STATE_COMPLETED ||
2347 tsx->state == PJSIP_TSX_STATE_TERMINATED))
Benny Prijono8ad55352006-02-08 11:16:05 +00002348 {
Benny Prijono38998232006-02-08 22:44:25 +00002349
Benny Prijono8ad55352006-02-08 11:16:05 +00002350 /*
Benny Prijono38998232006-02-08 22:44:25 +00002351 * Outgoing BYE
Benny Prijono8ad55352006-02-08 11:16:05 +00002352 */
Benny Prijono8ad55352006-02-08 11:16:05 +00002353
Benny Prijonoa66c7152006-02-09 01:26:14 +00002354 inv_handle_bye_response( inv, tsx, e->body.tsx_state.src.rdata, e);
Benny Prijono268ca612006-02-07 12:34:11 +00002355
Benny Prijono8ad55352006-02-08 11:16:05 +00002356 }
2357 else if (tsx->method.id == PJSIP_BYE_METHOD &&
2358 tsx->role == PJSIP_ROLE_UAS &&
2359 tsx->status_code < 200 &&
2360 e->body.tsx_state.type == PJSIP_EVENT_RX_MSG)
2361 {
Benny Prijonoccf95622006-02-07 18:48:01 +00002362
Benny Prijono8ad55352006-02-08 11:16:05 +00002363 /*
2364 * Handle incoming BYE.
2365 */
Benny Prijono268ca612006-02-07 12:34:11 +00002366
Benny Prijono8ad55352006-02-08 11:16:05 +00002367 inv_respond_incoming_bye( inv, tsx, e->body.tsx_state.src.rdata, e );
2368
Benny Prijono268ca612006-02-07 12:34:11 +00002369 }
Benny Prijono70127222006-07-02 14:53:05 +00002370 else if (tsx->method.id == PJSIP_CANCEL_METHOD &&
2371 tsx->role == PJSIP_ROLE_UAS &&
2372 tsx->status_code < 200 &&
2373 e->body.tsx_state.type == PJSIP_EVENT_RX_MSG)
2374 {
2375
2376 /*
2377 * Handle strandled incoming CANCEL.
2378 */
2379 pjsip_rx_data *rdata = e->body.tsx_state.src.rdata;
2380 pjsip_tx_data *tdata;
2381 pj_status_t status;
2382
2383 status = pjsip_dlg_create_response(dlg, rdata, 200, NULL, &tdata);
2384 if (status != PJ_SUCCESS) return;
2385
2386 status = pjsip_dlg_send_response(dlg, tsx, tdata);
2387 if (status != PJ_SUCCESS) return;
2388
2389 }
Benny Prijono26ff9062006-02-21 23:47:00 +00002390 else if (tsx->method.id == PJSIP_INVITE_METHOD &&
2391 tsx->role == PJSIP_ROLE_UAS)
2392 {
2393
2394 /*
2395 * Handle incoming re-INVITE
2396 */
2397 if (tsx->state == PJSIP_TSX_STATE_TRYING) {
2398
2399 pjsip_rx_data *rdata = e->body.tsx_state.src.rdata;
2400 pjsip_tx_data *tdata;
2401 pj_status_t status;
2402
2403 /* Check if we have INVITE pending. */
2404 if (inv->invite_tsx && inv->invite_tsx!=tsx) {
Benny Prijono46249942007-02-19 22:23:14 +00002405 pj_str_t reason;
2406
2407 reason = pj_str("Another INVITE transaction in progress");
Benny Prijono26ff9062006-02-21 23:47:00 +00002408
2409 /* Can not receive re-INVITE while another one is pending. */
Benny Prijono46249942007-02-19 22:23:14 +00002410 status = pjsip_dlg_create_response( inv->dlg, rdata, 500,
2411 &reason, &tdata);
Benny Prijono26ff9062006-02-21 23:47:00 +00002412 if (status != PJ_SUCCESS)
2413 return;
2414
2415 status = pjsip_dlg_send_response( inv->dlg, tsx, tdata);
2416
2417
2418 return;
2419 }
2420
2421 /* Save the invite transaction. */
2422 inv->invite_tsx = tsx;
2423
2424 /* Process SDP in incoming message. */
2425 status = inv_check_sdp_in_incoming_msg(inv, tsx, rdata);
2426
2427 if (status != PJ_SUCCESS) {
2428
2429 /* Not Acceptable */
2430 const pjsip_hdr *accept;
2431
2432 status = pjsip_dlg_create_response(inv->dlg, rdata,
2433 488, NULL, &tdata);
2434 if (status != PJ_SUCCESS)
2435 return;
2436
2437
2438 accept = pjsip_endpt_get_capability(dlg->endpt, PJSIP_H_ACCEPT,
2439 NULL);
2440 if (accept) {
Benny Prijonoa1e69682007-05-11 15:14:34 +00002441 pjsip_msg_add_hdr(tdata->msg, (pjsip_hdr*)
Benny Prijono26ff9062006-02-21 23:47:00 +00002442 pjsip_hdr_clone(tdata->pool, accept));
2443 }
2444
2445 status = pjsip_dlg_send_response(dlg, tsx, tdata);
2446
2447 return;
2448 }
2449
2450 /* Create 2xx ANSWER */
2451 status = pjsip_dlg_create_response(dlg, rdata, 200, NULL, &tdata);
2452 if (status != PJ_SUCCESS)
2453 return;
2454
2455 /* Process SDP in the answer */
Benny Prijono64f851e2006-02-23 13:49:28 +00002456 status = process_answer(inv, 200, tdata, NULL);
Benny Prijono26ff9062006-02-21 23:47:00 +00002457
Benny Prijono1d9b9a42006-09-25 13:40:12 +00002458 if (status != PJ_SUCCESS) {
2459 /*
2460 * SDP negotiation has failed.
2461 */
2462 pj_status_t rc;
2463 pj_str_t reason;
2464
2465 /* Delete the 2xx answer */
2466 pjsip_tx_data_dec_ref(tdata);
2467
2468 /* Create 500 response */
2469 reason = pj_str("SDP negotiation failed");
2470 rc = pjsip_dlg_create_response(dlg, rdata, 500, &reason,
2471 &tdata);
2472 if (rc == PJ_SUCCESS) {
2473 pjsip_warning_hdr *w;
2474 const pj_str_t *endpt_name;
2475
2476 endpt_name = pjsip_endpt_name(dlg->endpt);
2477 w = pjsip_warning_hdr_create_from_status(tdata->pool,
2478 endpt_name,
2479 status);
2480 if (w)
2481 pjsip_msg_add_hdr(tdata->msg, (pjsip_hdr*)w);
2482
2483 pjsip_inv_send_msg(inv, tdata);
2484 }
2485 return;
2486 }
2487
2488 /* Send 2xx regardless of the status of negotiation */
Benny Prijonoe8b0d3b2006-03-17 17:57:52 +00002489 status = pjsip_inv_send_msg(inv, tdata);
Benny Prijono26ff9062006-02-21 23:47:00 +00002490
2491 }
2492
2493 }
2494 else if (tsx->method.id == PJSIP_INVITE_METHOD &&
2495 tsx->role == PJSIP_ROLE_UAC)
2496 {
2497 /*
2498 * Handle outgoing re-INVITE
2499 */
2500 if (tsx->state == PJSIP_TSX_STATE_TERMINATED &&
2501 tsx->status_code/100 == 2)
2502 {
2503
2504 /* Re-INVITE was accepted. */
2505
2506 /* Process SDP */
2507 inv_check_sdp_in_incoming_msg(inv, tsx,
2508 e->body.tsx_state.src.rdata);
2509
2510 /* Send ACK */
Benny Prijono853ab812007-06-12 15:40:38 +00002511 inv_send_ack(inv, e->body.tsx_state.src.rdata);
Benny Prijono26ff9062006-02-21 23:47:00 +00002512
2513 } else if (tsx->state == PJSIP_TSX_STATE_COMPLETED &&
2514 (tsx->status_code==401 || tsx->status_code==407))
2515 {
2516 pjsip_tx_data *tdata;
2517 pj_status_t status;
2518
2519 /* Handle authentication challenge. */
2520 status = pjsip_auth_clt_reinit_req( &dlg->auth_sess,
2521 e->body.tsx_state.src.rdata,
2522 tsx->last_tx,
2523 &tdata);
2524 if (status != PJ_SUCCESS)
2525 return;
2526
2527 /* Send re-INVITE */
Benny Prijonoe8b0d3b2006-03-17 17:57:52 +00002528 status = pjsip_inv_send_msg( inv, tdata);
Benny Prijono26ff9062006-02-21 23:47:00 +00002529
2530 } else if (tsx->status_code==PJSIP_SC_CALL_TSX_DOES_NOT_EXIST ||
2531 tsx->status_code==PJSIP_SC_REQUEST_TIMEOUT ||
2532 tsx->status_code >= 700)
2533 {
2534 /*
2535 * Handle responses that terminates dialog.
2536 */
Benny Prijono0b6340c2006-06-13 22:21:23 +00002537 inv_set_cause(inv, tsx->status_code, &tsx->status_text);
Benny Prijono26ff9062006-02-21 23:47:00 +00002538 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
2539 }
2540 }
Benny Prijono268ca612006-02-07 12:34:11 +00002541}
2542
Benny Prijono8ad55352006-02-08 11:16:05 +00002543/*
2544 * After session has been terminated, but before dialog is destroyed
2545 * (because dialog has other usages, or because dialog is waiting for
2546 * the last transaction to terminate).
2547 */
2548static void inv_on_state_disconnected( pjsip_inv_session *inv, pjsip_event *e)
Benny Prijono268ca612006-02-07 12:34:11 +00002549{
Benny Prijono8ad55352006-02-08 11:16:05 +00002550 pjsip_transaction *tsx = e->body.tsx_state.tsx;
2551 pjsip_dialog *dlg = pjsip_tsx_get_dlg(tsx);
Benny Prijono268ca612006-02-07 12:34:11 +00002552
Benny Prijono8ad55352006-02-08 11:16:05 +00002553 PJ_ASSERT_ON_FAIL(tsx && dlg, return);
2554
Benny Prijono70127222006-07-02 14:53:05 +00002555 if (tsx->role == PJSIP_ROLE_UAS &&
Benny Prijono8ad55352006-02-08 11:16:05 +00002556 tsx->status_code < 200 &&
2557 e->body.tsx_state.type == PJSIP_EVENT_RX_MSG)
2558 {
Benny Prijono70127222006-07-02 14:53:05 +00002559 pjsip_rx_data *rdata = e->body.tsx_state.src.rdata;
Benny Prijono8ad55352006-02-08 11:16:05 +00002560
2561 /*
Benny Prijono70127222006-07-02 14:53:05 +00002562 * Respond BYE with 200/OK
Benny Prijono8ad55352006-02-08 11:16:05 +00002563 */
Benny Prijono70127222006-07-02 14:53:05 +00002564 if (tsx->method.id == PJSIP_BYE_METHOD) {
2565 inv_respond_incoming_bye( inv, tsx, rdata, e );
2566 } else if (tsx->method.id == PJSIP_CANCEL_METHOD) {
2567 /*
2568 * Respond CANCEL with 200/OK too.
2569 */
2570 pjsip_tx_data *tdata;
2571 pj_status_t status;
Benny Prijono8ad55352006-02-08 11:16:05 +00002572
Benny Prijono70127222006-07-02 14:53:05 +00002573 status = pjsip_dlg_create_response(dlg, rdata, 200, NULL, &tdata);
2574 if (status != PJ_SUCCESS) return;
Benny Prijono8ad55352006-02-08 11:16:05 +00002575
Benny Prijono70127222006-07-02 14:53:05 +00002576 status = pjsip_dlg_send_response(dlg, tsx, tdata);
2577 if (status != PJ_SUCCESS) return;
2578
2579 }
Benny Prijono8ad55352006-02-08 11:16:05 +00002580 }
Benny Prijono268ca612006-02-07 12:34:11 +00002581}
2582