blob: 593a914e539596083ee73a83a77464a24711474f [file] [log] [blame]
Benny Prijono268ca612006-02-07 12:34:11 +00001/* $Id$ */
2/*
3 * Copyright (C) 2003-2006 Benny Prijono <benny@prijono.org>
4 *
5 * This program is free software; you can redistribute it and/or modify
6 * it under the terms of the GNU General Public License as published by
7 * the Free Software Foundation; either version 2 of the License, or
8 * (at your option) any later version.
9 *
10 * This program is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 * GNU General Public License for more details.
14 *
15 * You should have received a copy of the GNU General Public License
16 * along with this program; if not, write to the Free Software
17 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
18 */
19#include <pjsip-ua/sip_inv.h>
20#include <pjsip/sip_module.h>
21#include <pjsip/sip_endpoint.h>
22#include <pjsip/sip_event.h>
23#include <pjsip/sip_transaction.h>
24#include <pjmedia/sdp.h>
25#include <pjmedia/sdp_neg.h>
Benny Prijono95196582006-02-09 00:13:40 +000026#include <pjmedia/errno.h>
Benny Prijono268ca612006-02-07 12:34:11 +000027#include <pj/string.h>
28#include <pj/pool.h>
29#include <pj/assert.h>
Benny Prijono8ad55352006-02-08 11:16:05 +000030#include <pj/os.h>
Benny Prijonoa66c7152006-02-09 01:26:14 +000031#include <pj/log.h>
32
Benny Prijono268ca612006-02-07 12:34:11 +000033
34#define THIS_FILE "sip_invite_session.c"
35
Benny Prijonoe8b0d3b2006-03-17 17:57:52 +000036static const char *inv_state_names[] =
37{
38 "NULL ",
39 "CALLING ",
40 "INCOMING ",
41 "EARLY ",
42 "CONNECTING",
43 "CONFIRMED ",
44 "DISCONNCTD",
45 "TERMINATED",
46};
47
Benny Prijono268ca612006-02-07 12:34:11 +000048/*
49 * Static prototypes.
50 */
51static pj_status_t mod_inv_load(pjsip_endpoint *endpt);
52static pj_status_t mod_inv_unload(void);
53static pj_bool_t mod_inv_on_rx_request(pjsip_rx_data *rdata);
54static pj_bool_t mod_inv_on_rx_response(pjsip_rx_data *rdata);
55static void mod_inv_on_tsx_state(pjsip_transaction*, pjsip_event*);
56
Benny Prijono8ad55352006-02-08 11:16:05 +000057static void inv_on_state_null( pjsip_inv_session *inv, pjsip_event *e);
58static void inv_on_state_calling( pjsip_inv_session *inv, pjsip_event *e);
59static void inv_on_state_incoming( pjsip_inv_session *inv, pjsip_event *e);
60static void inv_on_state_early( pjsip_inv_session *inv, pjsip_event *e);
61static void inv_on_state_connecting( pjsip_inv_session *inv, pjsip_event *e);
62static void inv_on_state_confirmed( pjsip_inv_session *inv, pjsip_event *e);
63static void inv_on_state_disconnected( pjsip_inv_session *inv, pjsip_event *e);
Benny Prijono268ca612006-02-07 12:34:11 +000064
Benny Prijono8ad55352006-02-08 11:16:05 +000065static void (*inv_state_handler[])( pjsip_inv_session *inv, pjsip_event *e) =
Benny Prijono268ca612006-02-07 12:34:11 +000066{
67 &inv_on_state_null,
68 &inv_on_state_calling,
69 &inv_on_state_incoming,
70 &inv_on_state_early,
71 &inv_on_state_connecting,
72 &inv_on_state_confirmed,
73 &inv_on_state_disconnected,
Benny Prijono268ca612006-02-07 12:34:11 +000074};
75
76static struct mod_inv
77{
78 pjsip_module mod;
79 pjsip_endpoint *endpt;
80 pjsip_inv_callback cb;
Benny Prijono268ca612006-02-07 12:34:11 +000081} mod_inv =
82{
83 {
Benny Prijono2f8992b2006-02-25 21:16:36 +000084 NULL, NULL, /* prev, next. */
85 { "mod-invite", 10 }, /* Name. */
86 -1, /* Id */
87 PJSIP_MOD_PRIORITY_DIALOG_USAGE, /* Priority */
88 &mod_inv_load, /* load() */
89 NULL, /* start() */
90 NULL, /* stop() */
91 &mod_inv_unload, /* unload() */
92 &mod_inv_on_rx_request, /* on_rx_request() */
93 &mod_inv_on_rx_response, /* on_rx_response() */
94 NULL, /* on_tx_request. */
95 NULL, /* on_tx_response() */
96 &mod_inv_on_tsx_state, /* on_tsx_state() */
Benny Prijono268ca612006-02-07 12:34:11 +000097 }
98};
99
100
Benny Prijonoa66c7152006-02-09 01:26:14 +0000101/* Invite session data to be attached to transaction. */
102struct tsx_inv_data
103{
104 pjsip_inv_session *inv;
105 pj_bool_t sdp_done;
106};
107
108
Benny Prijono8ad55352006-02-08 11:16:05 +0000109/*
110 * Module load()
111 */
Benny Prijono268ca612006-02-07 12:34:11 +0000112static pj_status_t mod_inv_load(pjsip_endpoint *endpt)
113{
114 pj_str_t allowed[] = {{"INVITE", 6}, {"ACK",3}, {"BYE",3}, {"CANCEL",6}};
115
116 /* Register supported methods: INVITE, ACK, BYE, CANCEL */
117 pjsip_endpt_add_capability(endpt, &mod_inv.mod, PJSIP_H_ALLOW, NULL,
118 PJ_ARRAY_SIZE(allowed), allowed);
119
120 return PJ_SUCCESS;
121}
122
Benny Prijono8ad55352006-02-08 11:16:05 +0000123/*
124 * Module unload()
125 */
Benny Prijono268ca612006-02-07 12:34:11 +0000126static pj_status_t mod_inv_unload(void)
127{
128 /* Should remove capability here */
129 return PJ_SUCCESS;
130}
131
Benny Prijono8ad55352006-02-08 11:16:05 +0000132/*
Benny Prijono38998232006-02-08 22:44:25 +0000133 * Set session state.
134 */
135void inv_set_state(pjsip_inv_session *inv, pjsip_inv_state state,
136 pjsip_event *e)
137{
Benny Prijonod4e0abd2006-03-05 11:53:36 +0000138 pjsip_inv_state prev_state = inv->state;
139
140 /* Set state. */
Benny Prijono38998232006-02-08 22:44:25 +0000141 inv->state = state;
Benny Prijonod4e0abd2006-03-05 11:53:36 +0000142
143 /* If state is DISCONNECTED, cause code MUST have been set. */
144 pj_assert(inv->state != PJSIP_INV_STATE_DISCONNECTED ||
145 inv->cause != 0);
146
147 /* Call on_state_changed() callback. */
148 if (mod_inv.cb.on_state_changed && inv->notify)
Benny Prijono38998232006-02-08 22:44:25 +0000149 (*mod_inv.cb.on_state_changed)(inv, e);
150
Benny Prijonod4e0abd2006-03-05 11:53:36 +0000151 /* Only decrement when previous state is not already DISCONNECTED */
152 if (inv->state == PJSIP_INV_STATE_DISCONNECTED &&
153 prev_state != PJSIP_INV_STATE_DISCONNECTED)
154 {
Benny Prijonoe4f2abb2006-02-10 14:04:05 +0000155 pjsip_dlg_dec_session(inv->dlg, &mod_inv.mod);
Benny Prijonod4e0abd2006-03-05 11:53:36 +0000156 }
Benny Prijono38998232006-02-08 22:44:25 +0000157}
158
159
160/*
Benny Prijono8ad55352006-02-08 11:16:05 +0000161 * Send ACK for 2xx response.
162 */
163static pj_status_t inv_send_ack(pjsip_inv_session *inv, pjsip_rx_data *rdata)
Benny Prijono268ca612006-02-07 12:34:11 +0000164{
165 pjsip_tx_data *tdata;
166 pj_status_t status;
167
Benny Prijonoe4f2abb2006-02-10 14:04:05 +0000168 PJ_LOG(5,(inv->obj_name, "Received %s, sending ACK",
169 pjsip_rx_data_get_info(rdata)));
170
Benny Prijono268ca612006-02-07 12:34:11 +0000171 status = pjsip_dlg_create_request(inv->dlg, &pjsip_ack_method,
172 rdata->msg_info.cseq->cseq, &tdata);
173 if (status != PJ_SUCCESS) {
174 /* Better luck next time */
175 pj_assert(!"Unable to create ACK!");
176 return status;
177 }
178
Benny Prijono64158af2006-04-04 11:06:34 +0000179 status = pjsip_dlg_send_request(inv->dlg, tdata, -1, NULL);
Benny Prijono268ca612006-02-07 12:34:11 +0000180 if (status != PJ_SUCCESS) {
181 /* Better luck next time */
182 pj_assert(!"Unable to send ACK!");
183 return status;
184 }
185
186 return PJ_SUCCESS;
187}
188
Benny Prijono8ad55352006-02-08 11:16:05 +0000189/*
190 * Module on_rx_request()
191 *
192 * This callback is called for these events:
193 * - endpoint receives request which was unhandled by higher priority
194 * modules (e.g. transaction layer, dialog layer).
195 * - dialog distributes incoming request to its usages.
196 */
197static pj_bool_t mod_inv_on_rx_request(pjsip_rx_data *rdata)
198{
199 pjsip_method *method;
Benny Prijono38998232006-02-08 22:44:25 +0000200 pjsip_dialog *dlg;
201 pjsip_inv_session *inv;
Benny Prijono8ad55352006-02-08 11:16:05 +0000202
203 /* Only wants to receive request from a dialog. */
Benny Prijono38998232006-02-08 22:44:25 +0000204 dlg = pjsip_rdata_get_dlg(rdata);
205 if (dlg == NULL)
Benny Prijono8ad55352006-02-08 11:16:05 +0000206 return PJ_FALSE;
207
Benny Prijono38998232006-02-08 22:44:25 +0000208 inv = dlg->mod_data[mod_inv.mod.id];
209
Benny Prijono8ad55352006-02-08 11:16:05 +0000210 /* Report to dialog that we handle INVITE, CANCEL, BYE, ACK.
211 * If we need to send response, it will be sent in the state
212 * handlers.
213 */
214 method = &rdata->msg_info.msg->line.req.method;
215
216 if (method->id == PJSIP_INVITE_METHOD ||
217 method->id == PJSIP_CANCEL_METHOD ||
Benny Prijono8ad55352006-02-08 11:16:05 +0000218 method->id == PJSIP_BYE_METHOD)
219 {
220 return PJ_TRUE;
221 }
222
Benny Prijono38998232006-02-08 22:44:25 +0000223 /* On receipt ACK request, when state is CONNECTING,
224 * move state to CONFIRMED.
225 */
Benny Prijono26ff9062006-02-21 23:47:00 +0000226 if (method->id == PJSIP_ACK_METHOD && inv) {
Benny Prijono38998232006-02-08 22:44:25 +0000227
Benny Prijono5eff0432006-02-09 14:14:21 +0000228 /* Terminate INVITE transaction, if it's still present. */
229 if (inv->invite_tsx &&
230 inv->invite_tsx->state <= PJSIP_TSX_STATE_COMPLETED)
231 {
232 pjsip_tsx_terminate(inv->invite_tsx,
233 inv->invite_tsx->status_code);
234 inv->invite_tsx = NULL;
235 }
236
Benny Prijono32ac8fe2006-03-02 21:16:55 +0000237 /* On receipt of ACK, only set state to confirmed when state
238 * is CONNECTING (e.g. we don't want to set the state to confirmed
239 * when we receive ACK retransmission after sending non-2xx!)
240 */
241 if (inv->state == PJSIP_INV_STATE_CONNECTING) {
Benny Prijono26ff9062006-02-21 23:47:00 +0000242 pjsip_event event;
243
244 PJSIP_EVENT_INIT_RX_MSG(event, rdata);
245 inv_set_state(inv, PJSIP_INV_STATE_CONFIRMED, &event);
246 }
Benny Prijono38998232006-02-08 22:44:25 +0000247 }
248
Benny Prijono8ad55352006-02-08 11:16:05 +0000249 return PJ_FALSE;
250}
251
252/*
253 * Module on_rx_response().
254 *
255 * This callback is called for these events:
256 * - dialog distributes incoming 2xx response to INVITE (outside
257 * transaction) to its usages.
258 * - endpoint distributes strayed responses.
259 */
Benny Prijono268ca612006-02-07 12:34:11 +0000260static pj_bool_t mod_inv_on_rx_response(pjsip_rx_data *rdata)
261{
262 pjsip_dialog *dlg;
263 pjsip_inv_session *inv;
264 pjsip_msg *msg = rdata->msg_info.msg;
265
266 dlg = pjsip_rdata_get_dlg(rdata);
267
268 /* Ignore responses outside dialog */
269 if (dlg == NULL)
270 return PJ_FALSE;
271
272 /* Ignore responses not belonging to invite session */
273 inv = pjsip_dlg_get_inv_session(dlg);
274 if (inv == NULL)
275 return PJ_FALSE;
276
277 /* This MAY be retransmission of 2xx response to INVITE.
278 * If it is, we need to send ACK.
279 */
280 if (msg->type == PJSIP_RESPONSE_MSG && msg->line.status.code/100==2 &&
Benny Prijono26ff9062006-02-21 23:47:00 +0000281 rdata->msg_info.cseq->method.id == PJSIP_INVITE_METHOD &&
282 inv->invite_tsx == NULL)
283 {
Benny Prijono268ca612006-02-07 12:34:11 +0000284
Benny Prijono8ad55352006-02-08 11:16:05 +0000285 inv_send_ack(inv, rdata);
Benny Prijono268ca612006-02-07 12:34:11 +0000286 return PJ_TRUE;
287
288 }
289
290 /* No other processing needs to be done here. */
291 return PJ_FALSE;
292}
293
Benny Prijono8ad55352006-02-08 11:16:05 +0000294/*
295 * Module on_tsx_state()
296 *
297 * This callback is called by dialog framework for all transactions
298 * inside the dialog for all its dialog usages.
299 */
Benny Prijono268ca612006-02-07 12:34:11 +0000300static void mod_inv_on_tsx_state(pjsip_transaction *tsx, pjsip_event *e)
301{
302 pjsip_dialog *dlg;
303 pjsip_inv_session *inv;
304
305 dlg = pjsip_tsx_get_dlg(tsx);
306 if (dlg == NULL)
307 return;
308
309 inv = pjsip_dlg_get_inv_session(dlg);
310 if (inv == NULL)
311 return;
312
313 /* Call state handler for the invite session. */
314 (*inv_state_handler[inv->state])(inv, e);
315
316 /* Call on_tsx_state */
Benny Prijonod4e0abd2006-03-05 11:53:36 +0000317 if (mod_inv.cb.on_tsx_state_changed && inv->notify)
Benny Prijono268ca612006-02-07 12:34:11 +0000318 (*mod_inv.cb.on_tsx_state_changed)(inv, tsx, e);
319
320 /* Clear invite transaction when tsx is terminated. */
321 if (tsx->state==PJSIP_TSX_STATE_TERMINATED && tsx == inv->invite_tsx)
322 inv->invite_tsx = NULL;
323}
324
Benny Prijono8ad55352006-02-08 11:16:05 +0000325
326/*
327 * Initialize the invite module.
328 */
Benny Prijono268ca612006-02-07 12:34:11 +0000329PJ_DEF(pj_status_t) pjsip_inv_usage_init( pjsip_endpoint *endpt,
Benny Prijono268ca612006-02-07 12:34:11 +0000330 const pjsip_inv_callback *cb)
331{
332 pj_status_t status;
333
334 /* Check arguments. */
Benny Prijonoe8b0d3b2006-03-17 17:57:52 +0000335 PJ_ASSERT_RETURN(endpt && cb, PJ_EINVAL);
Benny Prijono268ca612006-02-07 12:34:11 +0000336
337 /* Some callbacks are mandatory */
338 PJ_ASSERT_RETURN(cb->on_state_changed && cb->on_new_session, PJ_EINVAL);
339
340 /* Check if module already registered. */
341 PJ_ASSERT_RETURN(mod_inv.mod.id == -1, PJ_EINVALIDOP);
342
343 /* Copy param. */
344 pj_memcpy(&mod_inv.cb, cb, sizeof(pjsip_inv_callback));
345
346 mod_inv.endpt = endpt;
Benny Prijono268ca612006-02-07 12:34:11 +0000347
348 /* Register the module. */
349 status = pjsip_endpt_register_module(endpt, &mod_inv.mod);
350
351 return status;
352}
353
Benny Prijono8ad55352006-02-08 11:16:05 +0000354/*
355 * Get the instance of invite module.
356 */
Benny Prijono268ca612006-02-07 12:34:11 +0000357PJ_DEF(pjsip_module*) pjsip_inv_usage_instance(void)
358{
359 return &mod_inv.mod;
360}
361
362
Benny Prijono632ce712006-02-09 14:01:40 +0000363
Benny Prijono8ad55352006-02-08 11:16:05 +0000364/*
365 * Return the invite session for the specified dialog.
366 */
Benny Prijono268ca612006-02-07 12:34:11 +0000367PJ_DEF(pjsip_inv_session*) pjsip_dlg_get_inv_session(pjsip_dialog *dlg)
368{
369 return dlg->mod_data[mod_inv.mod.id];
370}
371
Benny Prijono8ad55352006-02-08 11:16:05 +0000372
Benny Prijono268ca612006-02-07 12:34:11 +0000373/*
Benny Prijonoe8b0d3b2006-03-17 17:57:52 +0000374 * Get INVITE state name.
375 */
376PJ_DEF(const char *) pjsip_inv_state_name(pjsip_inv_state state)
377{
378 PJ_ASSERT_RETURN(state >= PJSIP_INV_STATE_NULL &&
379 state <= PJSIP_INV_STATE_DISCONNECTED,
380 "??");
381
382 return inv_state_names[state];
383}
384
385/*
Benny Prijono8ad55352006-02-08 11:16:05 +0000386 * Create UAC invite session.
Benny Prijono268ca612006-02-07 12:34:11 +0000387 */
388PJ_DEF(pj_status_t) pjsip_inv_create_uac( pjsip_dialog *dlg,
389 const pjmedia_sdp_session *local_sdp,
390 unsigned options,
391 pjsip_inv_session **p_inv)
392{
393 pjsip_inv_session *inv;
394 pj_status_t status;
395
396 /* Verify arguments. */
397 PJ_ASSERT_RETURN(dlg && p_inv, PJ_EINVAL);
398
399 /* Normalize options */
400 if (options & PJSIP_INV_REQUIRE_100REL)
401 options |= PJSIP_INV_SUPPORT_100REL;
402
403 if (options & PJSIP_INV_REQUIRE_TIMER)
404 options |= PJSIP_INV_SUPPORT_TIMER;
405
406 /* Create the session */
407 inv = pj_pool_zalloc(dlg->pool, sizeof(pjsip_inv_session));
408 PJ_ASSERT_RETURN(inv != NULL, PJ_ENOMEM);
409
410 inv->pool = dlg->pool;
411 inv->role = PJSIP_ROLE_UAC;
412 inv->state = PJSIP_INV_STATE_NULL;
413 inv->dlg = dlg;
414 inv->options = options;
Benny Prijonod4e0abd2006-03-05 11:53:36 +0000415 inv->notify = PJ_TRUE;
416 inv->cause = 0;
Benny Prijono268ca612006-02-07 12:34:11 +0000417
Benny Prijonoe4f2abb2006-02-10 14:04:05 +0000418 /* Object name will use the same dialog pointer. */
Benny Prijonoed811d72006-03-10 12:57:12 +0000419 pj_ansi_snprintf(inv->obj_name, PJ_MAX_OBJ_NAME, "inv%p", dlg);
Benny Prijonoe4f2abb2006-02-10 14:04:05 +0000420
Benny Prijono268ca612006-02-07 12:34:11 +0000421 /* Create negotiator if local_sdp is specified. */
422 if (local_sdp) {
423 status = pjmedia_sdp_neg_create_w_local_offer(dlg->pool, local_sdp,
424 &inv->neg);
425 if (status != PJ_SUCCESS)
426 return status;
427 }
428
429 /* Register invite as dialog usage. */
430 status = pjsip_dlg_add_usage(dlg, &mod_inv.mod, inv);
431 if (status != PJ_SUCCESS)
432 return status;
433
434 /* Increment dialog session */
Benny Prijonoe4f2abb2006-02-10 14:04:05 +0000435 pjsip_dlg_inc_session(dlg, &mod_inv.mod);
Benny Prijono268ca612006-02-07 12:34:11 +0000436
437 /* Done */
438 *p_inv = inv;
Benny Prijonoe4f2abb2006-02-10 14:04:05 +0000439
440 PJ_LOG(5,(inv->obj_name, "UAC invite session created for dialog %s",
441 dlg->obj_name));
442
Benny Prijono268ca612006-02-07 12:34:11 +0000443 return PJ_SUCCESS;
444}
445
446/*
447 * Verify incoming INVITE request.
448 */
449PJ_DEF(pj_status_t) pjsip_inv_verify_request(pjsip_rx_data *rdata,
450 unsigned *options,
451 const pjmedia_sdp_session *l_sdp,
452 pjsip_dialog *dlg,
453 pjsip_endpoint *endpt,
454 pjsip_tx_data **p_tdata)
455{
456 pjsip_msg *msg;
457 pjsip_allow_hdr *allow;
458 pjsip_supported_hdr *sup_hdr;
459 pjsip_require_hdr *req_hdr;
460 int code = 200;
461 unsigned rem_option = 0;
462 pj_status_t status = PJ_SUCCESS;
463 pjsip_hdr res_hdr_list;
464
465 /* Init return arguments. */
466 if (p_tdata) *p_tdata = NULL;
467
468 /* Verify arguments. */
469 PJ_ASSERT_RETURN(rdata != NULL && options != NULL, PJ_EINVAL);
470
471 /* Normalize options */
472 if (*options & PJSIP_INV_REQUIRE_100REL)
473 *options |= PJSIP_INV_SUPPORT_100REL;
474
475 if (*options & PJSIP_INV_REQUIRE_TIMER)
476 *options |= PJSIP_INV_SUPPORT_TIMER;
477
478 /* Get the message in rdata */
479 msg = rdata->msg_info.msg;
480
481 /* Must be INVITE request. */
482 PJ_ASSERT_RETURN(msg->type == PJSIP_REQUEST_MSG &&
483 msg->line.req.method.id == PJSIP_INVITE_METHOD,
484 PJ_EINVAL);
485
486 /* If tdata is specified, then either dlg or endpt must be specified */
487 PJ_ASSERT_RETURN((!p_tdata) || (endpt || dlg), PJ_EINVAL);
488
489 /* Get the endpoint */
490 endpt = endpt ? endpt : dlg->endpt;
491
492 /* Init response header list */
493 pj_list_init(&res_hdr_list);
494
Benny Prijono8ad55352006-02-08 11:16:05 +0000495 /* Check the request body, see if it'inv something that we support
Benny Prijono268ca612006-02-07 12:34:11 +0000496 * (i.e. SDP).
497 */
498 if (msg->body) {
499 pjsip_msg_body *body = msg->body;
500 pj_str_t str_application = {"application", 11};
501 pj_str_t str_sdp = { "sdp", 3 };
502 pjmedia_sdp_session *sdp;
503
504 /* Check content type. */
505 if (pj_stricmp(&body->content_type.type, &str_application) != 0 ||
506 pj_stricmp(&body->content_type.subtype, &str_sdp) != 0)
507 {
508 /* Not "application/sdp" */
509 code = PJSIP_SC_UNSUPPORTED_MEDIA_TYPE;
510 status = PJSIP_ERRNO_FROM_SIP_STATUS(code);
511
512 if (p_tdata) {
513 /* Add Accept header to response */
514 pjsip_accept_hdr *acc;
515
516 acc = pjsip_accept_hdr_create(rdata->tp_info.pool);
517 PJ_ASSERT_RETURN(acc, PJ_ENOMEM);
518 acc->values[acc->count++] = pj_str("application/sdp");
519 pj_list_push_back(&res_hdr_list, acc);
520 }
521
522 goto on_return;
523 }
524
525 /* Parse and validate SDP */
526 status = pjmedia_sdp_parse(rdata->tp_info.pool, body->data, body->len,
527 &sdp);
528 if (status == PJ_SUCCESS)
529 status = pjmedia_sdp_validate(sdp);
530
531 if (status != PJ_SUCCESS) {
532 /* Unparseable or invalid SDP */
533 code = PJSIP_SC_BAD_REQUEST;
534
535 if (p_tdata) {
536 /* Add Warning header. */
537 pjsip_warning_hdr *w;
538
539 w = pjsip_warning_hdr_create_from_status(rdata->tp_info.pool,
540 pjsip_endpt_name(endpt),
541 status);
542 PJ_ASSERT_RETURN(w, PJ_ENOMEM);
543
544 pj_list_push_back(&res_hdr_list, w);
545 }
546
547 goto on_return;
548 }
549
550 /* Negotiate with local SDP */
551 if (l_sdp) {
552 pjmedia_sdp_neg *neg;
553
554 /* Local SDP must be valid! */
555 PJ_ASSERT_RETURN((status=pjmedia_sdp_validate(l_sdp))==PJ_SUCCESS,
556 status);
557
558 /* Create SDP negotiator */
559 status = pjmedia_sdp_neg_create_w_remote_offer(
560 rdata->tp_info.pool, l_sdp, sdp, &neg);
561 PJ_ASSERT_RETURN(status == PJ_SUCCESS, status);
562
563 /* Negotiate SDP */
564 status = pjmedia_sdp_neg_negotiate(rdata->tp_info.pool, neg, 0);
565 if (status != PJ_SUCCESS) {
566
567 /* Incompatible media */
568 code = PJSIP_SC_NOT_ACCEPTABLE_HERE;
569 status = PJSIP_ERRNO_FROM_SIP_STATUS(code);
570
571 if (p_tdata) {
572 pjsip_accept_hdr *acc;
573 pjsip_warning_hdr *w;
574
575 /* Add Warning header. */
576 w = pjsip_warning_hdr_create_from_status(
577 rdata->tp_info.pool,
578 pjsip_endpt_name(endpt), status);
579 PJ_ASSERT_RETURN(w, PJ_ENOMEM);
580
581 pj_list_push_back(&res_hdr_list, w);
582
583 /* Add Accept header to response */
584 acc = pjsip_accept_hdr_create(rdata->tp_info.pool);
585 PJ_ASSERT_RETURN(acc, PJ_ENOMEM);
586 acc->values[acc->count++] = pj_str("application/sdp");
587 pj_list_push_back(&res_hdr_list, acc);
588
589 }
590
591 goto on_return;
592 }
593 }
594 }
595
596 /* Check supported methods, see if peer supports UPDATE.
597 * We just assume that peer supports standard INVITE, ACK, CANCEL, and BYE
598 * implicitly by sending this INVITE.
599 */
600 allow = pjsip_msg_find_hdr(msg, PJSIP_H_ALLOW, NULL);
601 if (allow) {
602 unsigned i;
603 const pj_str_t STR_UPDATE = { "UPDATE", 6 };
604
605 for (i=0; i<allow->count; ++i) {
606 if (pj_stricmp(&allow->values[i], &STR_UPDATE)==0)
607 break;
608 }
609
610 if (i != allow->count) {
611 /* UPDATE is present in Allow */
612 rem_option |= PJSIP_INV_SUPPORT_UPDATE;
613 }
614
615 }
616
617 /* Check Supported header */
618 sup_hdr = pjsip_msg_find_hdr(msg, PJSIP_H_SUPPORTED, NULL);
619 if (sup_hdr) {
620 unsigned i;
621 pj_str_t STR_100REL = { "100rel", 6};
622 pj_str_t STR_TIMER = { "timer", 5 };
623
624 for (i=0; i<sup_hdr->count; ++i) {
625 if (pj_stricmp(&sup_hdr->values[i], &STR_100REL)==0)
626 rem_option |= PJSIP_INV_SUPPORT_100REL;
627 else if (pj_stricmp(&sup_hdr->values[i], &STR_TIMER)==0)
628 rem_option |= PJSIP_INV_SUPPORT_TIMER;
629 }
630 }
631
632 /* Check Require header */
633 req_hdr = pjsip_msg_find_hdr(msg, PJSIP_H_REQUIRE, NULL);
634 if (req_hdr) {
635 unsigned i;
636 pj_str_t STR_100REL = { "100rel", 6};
637 pj_str_t STR_TIMER = { "timer", 5 };
638 unsigned unsupp_cnt = 0;
639 pj_str_t unsupp_tags[PJSIP_GENERIC_ARRAY_MAX_COUNT];
640
641 for (i=0; i<req_hdr->count; ++i) {
642 if ((*options & PJSIP_INV_SUPPORT_100REL) &&
643 pj_stricmp(&req_hdr->values[i], &STR_100REL)==0)
644 {
645 rem_option |= PJSIP_INV_REQUIRE_100REL;
646
647 } else if ((*options && PJSIP_INV_SUPPORT_TIMER) &&
648 pj_stricmp(&req_hdr->values[i], &STR_TIMER)==0)
649 {
650 rem_option |= PJSIP_INV_REQUIRE_TIMER;
651
652 } else {
653 /* Unknown/unsupported extension tag! */
654 unsupp_tags[unsupp_cnt++] = req_hdr->values[i];
655 }
656 }
657
658 /* Check if there are required tags that we don't support */
659 if (unsupp_cnt) {
660
661 code = PJSIP_SC_BAD_EXTENSION;
662 status = PJSIP_ERRNO_FROM_SIP_STATUS(code);
663
664 if (p_tdata) {
665 pjsip_unsupported_hdr *unsupp_hdr;
666 const pjsip_hdr *h;
667
668 /* Add Unsupported header. */
669 unsupp_hdr = pjsip_unsupported_hdr_create(rdata->tp_info.pool);
670 PJ_ASSERT_RETURN(unsupp_hdr != NULL, PJ_ENOMEM);
671
672 unsupp_hdr->count = unsupp_cnt;
673 for (i=0; i<unsupp_cnt; ++i)
674 unsupp_hdr->values[i] = unsupp_tags[i];
675
676 pj_list_push_back(&res_hdr_list, unsupp_hdr);
677
678 /* Add Supported header. */
679 h = pjsip_endpt_get_capability(endpt, PJSIP_H_SUPPORTED,
680 NULL);
681 pj_assert(h);
682 if (h) {
683 sup_hdr = pjsip_hdr_clone(rdata->tp_info.pool, h);
684 pj_list_push_back(&res_hdr_list, sup_hdr);
685 }
686 }
687
688 goto on_return;
689 }
690 }
691
692 /* Check if there are local requirements that are not supported
693 * by peer.
694 */
695 if ( ((*options & PJSIP_INV_REQUIRE_100REL)!=0 &&
696 (rem_option & PJSIP_INV_SUPPORT_100REL)==0) ||
697 ((*options & PJSIP_INV_REQUIRE_TIMER)!=0 &&
698 (rem_option & PJSIP_INV_SUPPORT_TIMER)==0))
699 {
700 code = PJSIP_SC_EXTENSION_REQUIRED;
701 status = PJSIP_ERRNO_FROM_SIP_STATUS(code);
702
703 if (p_tdata) {
704 const pjsip_hdr *h;
705
706 /* Add Require header. */
707 req_hdr = pjsip_require_hdr_create(rdata->tp_info.pool);
708 PJ_ASSERT_RETURN(req_hdr != NULL, PJ_ENOMEM);
709
710 if (*options & PJSIP_INV_REQUIRE_100REL)
711 req_hdr->values[req_hdr->count++] = pj_str("100rel");
712
713 if (*options & PJSIP_INV_REQUIRE_TIMER)
714 req_hdr->values[req_hdr->count++] = pj_str("timer");
715
716 pj_list_push_back(&res_hdr_list, req_hdr);
717
718 /* Add Supported header. */
719 h = pjsip_endpt_get_capability(endpt, PJSIP_H_SUPPORTED,
720 NULL);
721 pj_assert(h);
722 if (h) {
723 sup_hdr = pjsip_hdr_clone(rdata->tp_info.pool, h);
724 pj_list_push_back(&res_hdr_list, sup_hdr);
725 }
726
727 }
728
729 goto on_return;
730 }
731
732on_return:
733
734 /* Create response if necessary */
735 if (code != 200 && p_tdata) {
736 pjsip_tx_data *tdata;
737 const pjsip_hdr *h;
738
739 if (dlg) {
740 status = pjsip_dlg_create_response(dlg, rdata, code, NULL,
741 &tdata);
742 } else {
743 status = pjsip_endpt_create_response(endpt, rdata, code, NULL,
744 &tdata);
745 }
746
747 if (status != PJ_SUCCESS)
748 return status;
749
750 /* Add response headers. */
751 h = res_hdr_list.next;
752 while (h != &res_hdr_list) {
753 pjsip_hdr *cloned;
754
755 cloned = pjsip_hdr_clone(tdata->pool, h);
756 PJ_ASSERT_RETURN(cloned, PJ_ENOMEM);
757
758 pjsip_msg_add_hdr(tdata->msg, cloned);
759
760 h = h->next;
761 }
762
763 *p_tdata = tdata;
764 }
765
766 return status;
767}
768
769/*
Benny Prijono8ad55352006-02-08 11:16:05 +0000770 * Create UAS invite session.
Benny Prijono268ca612006-02-07 12:34:11 +0000771 */
772PJ_DEF(pj_status_t) pjsip_inv_create_uas( pjsip_dialog *dlg,
773 pjsip_rx_data *rdata,
774 const pjmedia_sdp_session *local_sdp,
775 unsigned options,
776 pjsip_inv_session **p_inv)
777{
778 pjsip_inv_session *inv;
Benny Prijonoa66c7152006-02-09 01:26:14 +0000779 struct tsx_inv_data *tsx_inv_data;
Benny Prijono268ca612006-02-07 12:34:11 +0000780 pjsip_msg *msg;
781 pjmedia_sdp_session *rem_sdp = NULL;
782 pj_status_t status;
783
784 /* Verify arguments. */
785 PJ_ASSERT_RETURN(dlg && rdata && p_inv, PJ_EINVAL);
786
787 /* Dialog MUST have been initialised. */
788 PJ_ASSERT_RETURN(pjsip_rdata_get_tsx(rdata) != NULL, PJ_EINVALIDOP);
789
790 msg = rdata->msg_info.msg;
791
792 /* rdata MUST contain INVITE request */
793 PJ_ASSERT_RETURN(msg->type == PJSIP_REQUEST_MSG &&
794 msg->line.req.method.id == PJSIP_INVITE_METHOD,
795 PJ_EINVALIDOP);
796
797 /* Normalize options */
798 if (options & PJSIP_INV_REQUIRE_100REL)
799 options |= PJSIP_INV_SUPPORT_100REL;
800
801 if (options & PJSIP_INV_REQUIRE_TIMER)
802 options |= PJSIP_INV_SUPPORT_TIMER;
803
804 /* Create the session */
805 inv = pj_pool_zalloc(dlg->pool, sizeof(pjsip_inv_session));
806 PJ_ASSERT_RETURN(inv != NULL, PJ_ENOMEM);
807
808 inv->pool = dlg->pool;
809 inv->role = PJSIP_ROLE_UAS;
Benny Prijono38998232006-02-08 22:44:25 +0000810 inv->state = PJSIP_INV_STATE_NULL;
Benny Prijono268ca612006-02-07 12:34:11 +0000811 inv->dlg = dlg;
812 inv->options = options;
Benny Prijonod4e0abd2006-03-05 11:53:36 +0000813 inv->notify = PJ_TRUE;
814 inv->cause = 0;
Benny Prijono268ca612006-02-07 12:34:11 +0000815
Benny Prijonoe4f2abb2006-02-10 14:04:05 +0000816 /* Object name will use the same dialog pointer. */
Benny Prijonoed811d72006-03-10 12:57:12 +0000817 pj_ansi_snprintf(inv->obj_name, PJ_MAX_OBJ_NAME, "inv%p", dlg);
Benny Prijonoe4f2abb2006-02-10 14:04:05 +0000818
Benny Prijono268ca612006-02-07 12:34:11 +0000819 /* Parse SDP in message body, if present. */
820 if (msg->body) {
821 pjsip_msg_body *body = msg->body;
822
823 /* Parse and validate SDP */
824 status = pjmedia_sdp_parse(inv->pool, body->data, body->len,
825 &rem_sdp);
826 if (status == PJ_SUCCESS)
827 status = pjmedia_sdp_validate(rem_sdp);
828
829 if (status != PJ_SUCCESS)
830 return status;
831 }
832
833 /* Create negotiator. */
834 if (rem_sdp) {
835 status = pjmedia_sdp_neg_create_w_remote_offer(inv->pool, local_sdp,
836 rem_sdp, &inv->neg);
837
838 } else if (local_sdp) {
839 status = pjmedia_sdp_neg_create_w_local_offer(inv->pool, local_sdp,
840 &inv->neg);
841 } else {
Benny Prijono95196582006-02-09 00:13:40 +0000842 status = PJ_SUCCESS;
Benny Prijono268ca612006-02-07 12:34:11 +0000843 }
844
845 if (status != PJ_SUCCESS)
846 return status;
847
848 /* Register invite as dialog usage. */
849 status = pjsip_dlg_add_usage(dlg, &mod_inv.mod, inv);
850 if (status != PJ_SUCCESS)
851 return status;
852
853 /* Increment session in the dialog. */
Benny Prijonoe4f2abb2006-02-10 14:04:05 +0000854 pjsip_dlg_inc_session(dlg, &mod_inv.mod);
Benny Prijono268ca612006-02-07 12:34:11 +0000855
856 /* Save the invite transaction. */
857 inv->invite_tsx = pjsip_rdata_get_tsx(rdata);
Benny Prijonoa66c7152006-02-09 01:26:14 +0000858
859 /* Attach our data to the transaction. */
860 tsx_inv_data = pj_pool_zalloc(inv->invite_tsx->pool,
861 sizeof(struct tsx_inv_data));
862 tsx_inv_data->inv = inv;
863 inv->invite_tsx->mod_data[mod_inv.mod.id] = tsx_inv_data;
Benny Prijono268ca612006-02-07 12:34:11 +0000864
865 /* Done */
866 *p_inv = inv;
Benny Prijonoe4f2abb2006-02-10 14:04:05 +0000867
868 PJ_LOG(5,(inv->obj_name, "UAS invite session created for dialog %s",
869 dlg->obj_name));
870
Benny Prijono268ca612006-02-07 12:34:11 +0000871 return PJ_SUCCESS;
872}
873
Benny Prijonod4e0abd2006-03-05 11:53:36 +0000874/*
875 * Forcefully terminate the session.
876 */
877PJ_DEF(pj_status_t) pjsip_inv_terminate( pjsip_inv_session *inv,
878 int st_code,
879 pj_bool_t notify)
880{
881 PJ_ASSERT_RETURN(inv, PJ_EINVAL);
882
883 /* Lock dialog. */
884 pjsip_dlg_inc_lock(inv->dlg);
885
886 /* Set callback notify flag. */
887 inv->notify = notify;
888
889 /* If there's pending transaction, terminate the transaction.
890 * This may subsequently set the INVITE session state to
891 * disconnected.
892 */
893 if (inv->invite_tsx &&
894 inv->invite_tsx->state <= PJSIP_TSX_STATE_COMPLETED)
895 {
896 pjsip_tsx_terminate(inv->invite_tsx, st_code);
897
898 }
899
900 /* Set cause. */
901 inv->cause = st_code;
902
903 /* Forcefully terminate the session if state is not DISCONNECTED */
904 if (inv->state != PJSIP_INV_STATE_DISCONNECTED) {
905 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, NULL);
906 }
907
908 /* Done.
909 * The dec_lock() below will actually destroys the dialog if it
910 * has no other session.
911 */
912 pjsip_dlg_dec_lock(inv->dlg);
913
914 return PJ_SUCCESS;
915}
916
917
Benny Prijono268ca612006-02-07 12:34:11 +0000918static void *clone_sdp(pj_pool_t *pool, const void *data, unsigned len)
919{
920 PJ_UNUSED_ARG(len);
921 return pjmedia_sdp_session_clone(pool, data);
922}
923
924static int print_sdp(pjsip_msg_body *body, char *buf, pj_size_t len)
925{
926 return pjmedia_sdp_print(body->data, buf, len);
927}
928
929static pjsip_msg_body *create_sdp_body(pj_pool_t *pool,
930 const pjmedia_sdp_session *c_sdp)
931{
932 pjsip_msg_body *body;
933
934
935 body = pj_pool_zalloc(pool, sizeof(pjsip_msg_body));
936 PJ_ASSERT_RETURN(body != NULL, NULL);
937
938 body->content_type.type = pj_str("application");
939 body->content_type.subtype = pj_str("sdp");
940 body->data = pjmedia_sdp_session_clone(pool, c_sdp);
941 body->len = 0;
942 body->clone_data = &clone_sdp;
943 body->print_body = &print_sdp;
944
945 return body;
946}
947
948/*
949 * Create initial INVITE request.
950 */
951PJ_DEF(pj_status_t) pjsip_inv_invite( pjsip_inv_session *inv,
952 pjsip_tx_data **p_tdata )
953{
954 pjsip_tx_data *tdata;
955 const pjsip_hdr *hdr;
Benny Prijono26ff9062006-02-21 23:47:00 +0000956 pj_bool_t has_sdp;
Benny Prijono268ca612006-02-07 12:34:11 +0000957 pj_status_t status;
958
959 /* Verify arguments. */
960 PJ_ASSERT_RETURN(inv && p_tdata, PJ_EINVAL);
961
Benny Prijono26ff9062006-02-21 23:47:00 +0000962 /* State MUST be NULL or CONFIRMED. */
963 PJ_ASSERT_RETURN(inv->state == PJSIP_INV_STATE_NULL ||
964 inv->state == PJSIP_INV_STATE_CONFIRMED,
965 PJ_EINVALIDOP);
Benny Prijono268ca612006-02-07 12:34:11 +0000966
Benny Prijono64f851e2006-02-23 13:49:28 +0000967 /* Lock dialog. */
968 pjsip_dlg_inc_lock(inv->dlg);
969
Benny Prijono268ca612006-02-07 12:34:11 +0000970 /* Create the INVITE request. */
971 status = pjsip_dlg_create_request(inv->dlg, &pjsip_invite_method, -1,
972 &tdata);
973 if (status != PJ_SUCCESS)
Benny Prijono64f851e2006-02-23 13:49:28 +0000974 goto on_return;
975
Benny Prijono268ca612006-02-07 12:34:11 +0000976
Benny Prijono26ff9062006-02-21 23:47:00 +0000977 /* If this is the first INVITE, then copy the headers from inv_hdr.
978 * These are the headers parsed from the request URI when the
979 * dialog was created.
980 */
981 if (inv->state == PJSIP_INV_STATE_NULL) {
982 hdr = inv->dlg->inv_hdr.next;
983
984 while (hdr != &inv->dlg->inv_hdr) {
985 pjsip_msg_add_hdr(tdata->msg,
986 pjsip_hdr_shallow_clone(tdata->pool, hdr));
987 hdr = hdr->next;
988 }
989 }
990
991 /* See if we have SDP to send. */
992 if (inv->neg) {
993 pjmedia_sdp_neg_state neg_state;
994
995 neg_state = pjmedia_sdp_neg_get_state(inv->neg);
996
997 has_sdp = (neg_state == PJMEDIA_SDP_NEG_STATE_LOCAL_OFFER ||
998 (neg_state == PJMEDIA_SDP_NEG_STATE_WAIT_NEGO &&
999 pjmedia_sdp_neg_has_local_answer(inv->neg)));
1000
1001
1002 } else {
1003 has_sdp = PJ_FALSE;
1004 }
1005
Benny Prijono268ca612006-02-07 12:34:11 +00001006 /* Add SDP, if any. */
Benny Prijono26ff9062006-02-21 23:47:00 +00001007 if (has_sdp) {
Benny Prijono268ca612006-02-07 12:34:11 +00001008 const pjmedia_sdp_session *offer;
1009
1010 status = pjmedia_sdp_neg_get_neg_local(inv->neg, &offer);
1011 if (status != PJ_SUCCESS)
Benny Prijono64f851e2006-02-23 13:49:28 +00001012 goto on_return;
Benny Prijono268ca612006-02-07 12:34:11 +00001013
1014 tdata->msg->body = create_sdp_body(tdata->pool, offer);
1015 }
1016
1017 /* Add Allow header. */
1018 hdr = pjsip_endpt_get_capability(inv->dlg->endpt, PJSIP_H_ALLOW, NULL);
1019 if (hdr) {
1020 pjsip_msg_add_hdr(tdata->msg,
1021 pjsip_hdr_shallow_clone(tdata->pool, hdr));
1022 }
1023
1024 /* Add Supported header */
1025 hdr = pjsip_endpt_get_capability(inv->dlg->endpt, PJSIP_H_SUPPORTED, NULL);
1026 if (hdr) {
1027 pjsip_msg_add_hdr(tdata->msg,
1028 pjsip_hdr_shallow_clone(tdata->pool, hdr));
1029 }
1030
1031 /* Add Require header. */
1032 PJ_TODO(INVITE_ADD_REQUIRE_HEADER);
1033
1034 /* Done. */
1035 *p_tdata = tdata;
1036
Benny Prijono64f851e2006-02-23 13:49:28 +00001037
1038on_return:
1039 pjsip_dlg_dec_lock(inv->dlg);
1040 return status;
Benny Prijono268ca612006-02-07 12:34:11 +00001041}
1042
1043
1044/*
Benny Prijono95196582006-02-09 00:13:40 +00001045 * Negotiate SDP.
1046 */
1047static pj_status_t inv_negotiate_sdp( pjsip_inv_session *inv )
1048{
1049 pj_status_t status;
1050
1051 PJ_ASSERT_RETURN(pjmedia_sdp_neg_get_state(inv->neg) ==
1052 PJMEDIA_SDP_NEG_STATE_WAIT_NEGO,
1053 PJMEDIA_SDPNEG_EINSTATE);
1054
1055 status = pjmedia_sdp_neg_negotiate(inv->pool, inv->neg, 0);
1056
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00001057 PJ_LOG(5,(inv->obj_name, "SDP negotiation done, status=%d", status));
1058
Benny Prijonod4e0abd2006-03-05 11:53:36 +00001059 if (mod_inv.cb.on_media_update && inv->notify)
Benny Prijono95196582006-02-09 00:13:40 +00001060 (*mod_inv.cb.on_media_update)(inv, status);
1061
1062 return status;
1063}
1064
1065/*
Benny Prijonoa66c7152006-02-09 01:26:14 +00001066 * Check in incoming message for SDP offer/answer.
1067 */
Benny Prijono26ff9062006-02-21 23:47:00 +00001068static pj_status_t inv_check_sdp_in_incoming_msg( pjsip_inv_session *inv,
1069 pjsip_transaction *tsx,
1070 pjsip_rx_data *rdata)
Benny Prijonoa66c7152006-02-09 01:26:14 +00001071{
1072 struct tsx_inv_data *tsx_inv_data;
1073 static const pj_str_t str_application = { "application", 11 };
1074 static const pj_str_t str_sdp = { "sdp", 3 };
1075 pj_status_t status;
1076 pjsip_msg *msg;
1077 pjmedia_sdp_session *sdp;
1078
1079 /* Get/attach invite session's transaction data */
1080 tsx_inv_data = tsx->mod_data[mod_inv.mod.id];
1081 if (tsx_inv_data == NULL) {
1082 tsx_inv_data = pj_pool_zalloc(tsx->pool, sizeof(struct tsx_inv_data));
1083 tsx_inv_data->inv = inv;
1084 tsx->mod_data[mod_inv.mod.id] = tsx_inv_data;
1085 }
1086
1087 /* MUST NOT do multiple SDP offer/answer in a single transaction.
1088 */
1089
1090 if (tsx_inv_data->sdp_done)
Benny Prijono26ff9062006-02-21 23:47:00 +00001091 return PJ_SUCCESS;
Benny Prijonoa66c7152006-02-09 01:26:14 +00001092
1093 /* Check if SDP is present in the message. */
1094
1095 msg = rdata->msg_info.msg;
1096 if (msg->body == NULL) {
1097 /* Message doesn't have body. */
Benny Prijono26ff9062006-02-21 23:47:00 +00001098 return PJ_SUCCESS;
Benny Prijonoa66c7152006-02-09 01:26:14 +00001099 }
1100
1101 if (pj_stricmp(&msg->body->content_type.type, &str_application) ||
1102 pj_stricmp(&msg->body->content_type.subtype, &str_sdp))
1103 {
1104 /* Message body is not "application/sdp" */
Benny Prijono26ff9062006-02-21 23:47:00 +00001105 return PJMEDIA_SDP_EINSDP;
Benny Prijonoa66c7152006-02-09 01:26:14 +00001106 }
1107
1108 /* Parse the SDP body. */
1109
1110 status = pjmedia_sdp_parse(rdata->tp_info.pool, msg->body->data,
1111 msg->body->len, &sdp);
1112 if (status != PJ_SUCCESS) {
1113 char errmsg[PJ_ERR_MSG_SIZE];
1114 pj_strerror(status, errmsg, sizeof(errmsg));
1115 PJ_LOG(4,(THIS_FILE, "Error parsing SDP in %s: %s",
1116 pjsip_rx_data_get_info(rdata), errmsg));
Benny Prijono26ff9062006-02-21 23:47:00 +00001117 return PJMEDIA_SDP_EINSDP;
Benny Prijonoa66c7152006-02-09 01:26:14 +00001118 }
1119
1120 /* The SDP can be an offer or answer, depending on negotiator's state */
1121
1122 if (inv->neg == NULL ||
1123 pjmedia_sdp_neg_get_state(inv->neg) == PJMEDIA_SDP_NEG_STATE_DONE)
1124 {
1125
1126 /* This is an offer. */
1127
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00001128 PJ_LOG(5,(inv->obj_name, "Got SDP offer in %s",
1129 pjsip_rx_data_get_info(rdata)));
1130
Benny Prijonoa66c7152006-02-09 01:26:14 +00001131 if (inv->neg == NULL) {
1132 status=pjmedia_sdp_neg_create_w_remote_offer(inv->pool, NULL,
1133 sdp, &inv->neg);
1134 } else {
1135 status=pjmedia_sdp_neg_set_remote_offer(inv->pool, inv->neg, sdp);
1136 }
1137
1138 if (status != PJ_SUCCESS) {
1139 char errmsg[PJ_ERR_MSG_SIZE];
1140 pj_strerror(status, errmsg, sizeof(errmsg));
1141 PJ_LOG(4,(THIS_FILE, "Error processing SDP offer in %s: %s",
1142 pjsip_rx_data_get_info(rdata), errmsg));
Benny Prijono26ff9062006-02-21 23:47:00 +00001143 return PJMEDIA_SDP_EINSDP;
Benny Prijonoa66c7152006-02-09 01:26:14 +00001144 }
1145
1146 /* Inform application about remote offer. */
1147
Benny Prijonod4e0abd2006-03-05 11:53:36 +00001148 if (mod_inv.cb.on_rx_offer && inv->notify) {
Benny Prijono26ff9062006-02-21 23:47:00 +00001149
1150 (*mod_inv.cb.on_rx_offer)(inv, sdp);
1151
1152 }
Benny Prijonoa66c7152006-02-09 01:26:14 +00001153
1154 } else if (pjmedia_sdp_neg_get_state(inv->neg) ==
1155 PJMEDIA_SDP_NEG_STATE_LOCAL_OFFER)
1156 {
1157
1158 /* This is an answer.
1159 * Process and negotiate remote answer.
1160 */
1161
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00001162 PJ_LOG(5,(inv->obj_name, "Got SDP answer in %s",
1163 pjsip_rx_data_get_info(rdata)));
1164
Benny Prijonoa66c7152006-02-09 01:26:14 +00001165 status = pjmedia_sdp_neg_set_remote_answer(inv->pool, inv->neg, sdp);
1166
1167 if (status != PJ_SUCCESS) {
1168 char errmsg[PJ_ERR_MSG_SIZE];
1169 pj_strerror(status, errmsg, sizeof(errmsg));
1170 PJ_LOG(4,(THIS_FILE, "Error processing SDP answer in %s: %s",
1171 pjsip_rx_data_get_info(rdata), errmsg));
Benny Prijono26ff9062006-02-21 23:47:00 +00001172 return PJMEDIA_SDP_EINSDP;
Benny Prijonoa66c7152006-02-09 01:26:14 +00001173 }
1174
1175 /* Negotiate SDP */
1176
1177 inv_negotiate_sdp(inv);
1178
1179 /* Mark this transaction has having SDP offer/answer done. */
1180
1181 tsx_inv_data->sdp_done = 1;
1182
1183 } else {
1184
1185 PJ_LOG(5,(THIS_FILE, "Ignored SDP in %s: negotiator state is %s",
1186 pjsip_rx_data_get_info(rdata),
1187 pjmedia_sdp_neg_state_str(pjmedia_sdp_neg_get_state(inv->neg))));
1188 }
1189
Benny Prijono26ff9062006-02-21 23:47:00 +00001190 return PJ_SUCCESS;
Benny Prijonoa66c7152006-02-09 01:26:14 +00001191}
1192
1193
Benny Prijono26ff9062006-02-21 23:47:00 +00001194/*
1195 * Process INVITE answer, for both initial and subsequent re-INVITE
1196 */
1197static pj_status_t process_answer( pjsip_inv_session *inv,
1198 int st_code,
Benny Prijono64f851e2006-02-23 13:49:28 +00001199 pjsip_tx_data *tdata,
1200 const pjmedia_sdp_session *local_sdp)
Benny Prijono26ff9062006-02-21 23:47:00 +00001201{
1202 pj_status_t status;
Benny Prijonoab7399b2006-02-27 00:40:31 +00001203 const pjmedia_sdp_session *sdp = NULL;
Benny Prijono26ff9062006-02-21 23:47:00 +00001204
Benny Prijono64f851e2006-02-23 13:49:28 +00001205 /* If local_sdp is specified, then we MUST NOT have answered the
1206 * offer before.
Benny Prijono26ff9062006-02-21 23:47:00 +00001207 */
Benny Prijono64f851e2006-02-23 13:49:28 +00001208 if (local_sdp && (st_code/100==1 || st_code/100==2)) {
1209
1210 if (inv->neg == NULL) {
1211 status = pjmedia_sdp_neg_create_w_local_offer(inv->pool, local_sdp,
1212 &inv->neg);
1213 } else if (pjmedia_sdp_neg_get_state(inv->neg)==
1214 PJMEDIA_SDP_NEG_STATE_REMOTE_OFFER)
1215 {
1216 status = pjmedia_sdp_neg_set_local_answer(inv->pool, inv->neg,
1217 local_sdp);
1218 } else {
1219
1220 /* Can not specify local SDP at this state. */
1221 pj_assert(0);
1222 status = PJMEDIA_SDPNEG_EINSTATE;
1223 }
1224
1225 if (status != PJ_SUCCESS)
1226 return status;
1227
1228 }
1229
1230
1231 /* If SDP negotiator is ready, start negotiation. */
1232 if (st_code/100==2 || (st_code/10==18 && st_code!=180)) {
Benny Prijono26ff9062006-02-21 23:47:00 +00001233
1234 pjmedia_sdp_neg_state neg_state;
1235
Benny Prijono64f851e2006-02-23 13:49:28 +00001236 /* Start nego when appropriate. */
Benny Prijono26ff9062006-02-21 23:47:00 +00001237 neg_state = inv->neg ? pjmedia_sdp_neg_get_state(inv->neg) :
1238 PJMEDIA_SDP_NEG_STATE_NULL;
1239
1240 if (neg_state == PJMEDIA_SDP_NEG_STATE_LOCAL_OFFER) {
1241
1242 status = pjmedia_sdp_neg_get_neg_local(inv->neg, &sdp);
1243
1244 } else if (neg_state == PJMEDIA_SDP_NEG_STATE_WAIT_NEGO &&
1245 pjmedia_sdp_neg_has_local_answer(inv->neg) )
1246 {
1247
1248 status = inv_negotiate_sdp(inv);
1249 if (status != PJ_SUCCESS)
1250 return status;
1251
1252 status = pjmedia_sdp_neg_get_active_local(inv->neg, &sdp);
1253 }
Benny Prijono26ff9062006-02-21 23:47:00 +00001254 }
1255
Benny Prijono64f851e2006-02-23 13:49:28 +00001256 /* Include SDP when it's available for 2xx and 18x (but not 180) response.
Benny Prijono26ff9062006-02-21 23:47:00 +00001257 * Subsequent response will include this SDP.
1258 */
1259 if (sdp) {
1260 tdata->msg->body = create_sdp_body(tdata->pool, sdp);
1261 }
1262
Benny Prijono26ff9062006-02-21 23:47:00 +00001263
1264 return PJ_SUCCESS;
1265}
1266
Benny Prijonoa66c7152006-02-09 01:26:14 +00001267
1268/*
Benny Prijono64f851e2006-02-23 13:49:28 +00001269 * Create first response to INVITE
1270 */
1271PJ_DEF(pj_status_t) pjsip_inv_initial_answer( pjsip_inv_session *inv,
1272 pjsip_rx_data *rdata,
1273 int st_code,
1274 const pj_str_t *st_text,
1275 const pjmedia_sdp_session *sdp,
1276 pjsip_tx_data **p_tdata)
1277{
1278 pjsip_tx_data *tdata;
1279 pj_status_t status;
1280
1281 /* Verify arguments. */
1282 PJ_ASSERT_RETURN(inv && p_tdata, PJ_EINVAL);
1283
1284 /* Must have INVITE transaction. */
1285 PJ_ASSERT_RETURN(inv->invite_tsx, PJ_EBUG);
1286
1287 pjsip_dlg_inc_lock(inv->dlg);
1288
1289 /* Create response */
1290 status = pjsip_dlg_create_response(inv->dlg, rdata, st_code, st_text,
1291 &tdata);
1292 if (status != PJ_SUCCESS)
1293 goto on_return;
1294
1295 /* Process SDP in answer */
1296 status = process_answer(inv, st_code, tdata, sdp);
1297 if (status != PJ_SUCCESS) {
1298 pjsip_tx_data_dec_ref(tdata);
1299 goto on_return;
1300 }
1301
1302 *p_tdata = tdata;
1303
1304on_return:
1305 pjsip_dlg_dec_lock(inv->dlg);
1306 return status;
1307}
1308
1309
1310/*
Benny Prijono26ff9062006-02-21 23:47:00 +00001311 * Answer initial INVITE
1312 * Re-INVITE will be answered automatically, and will not use this function.
Benny Prijono268ca612006-02-07 12:34:11 +00001313 */
1314PJ_DEF(pj_status_t) pjsip_inv_answer( pjsip_inv_session *inv,
1315 int st_code,
1316 const pj_str_t *st_text,
1317 const pjmedia_sdp_session *local_sdp,
1318 pjsip_tx_data **p_tdata )
1319{
1320 pjsip_tx_data *last_res;
1321 pj_status_t status;
1322
1323 /* Verify arguments. */
1324 PJ_ASSERT_RETURN(inv && p_tdata, PJ_EINVAL);
1325
1326 /* Must have INVITE transaction. */
1327 PJ_ASSERT_RETURN(inv->invite_tsx, PJ_EBUG);
1328
1329 /* INVITE transaction MUST have transmitted a response (e.g. 100) */
1330 PJ_ASSERT_RETURN(inv->invite_tsx->last_tx, PJ_EINVALIDOP);
1331
Benny Prijono64f851e2006-02-23 13:49:28 +00001332 pjsip_dlg_inc_lock(inv->dlg);
Benny Prijono268ca612006-02-07 12:34:11 +00001333
1334 /* Modify last response. */
Benny Prijono26ff9062006-02-21 23:47:00 +00001335 last_res = inv->invite_tsx->last_tx;
Benny Prijono268ca612006-02-07 12:34:11 +00001336 status = pjsip_dlg_modify_response(inv->dlg, last_res, st_code, st_text);
1337 if (status != PJ_SUCCESS)
Benny Prijono64f851e2006-02-23 13:49:28 +00001338 goto on_return;
Benny Prijono268ca612006-02-07 12:34:11 +00001339
Benny Prijono268ca612006-02-07 12:34:11 +00001340
Benny Prijono26ff9062006-02-21 23:47:00 +00001341 /* Process SDP in answer */
Benny Prijono64f851e2006-02-23 13:49:28 +00001342 status = process_answer(inv, st_code, last_res, local_sdp);
1343 if (status != PJ_SUCCESS) {
1344 pjsip_tx_data_dec_ref(last_res);
1345 goto on_return;
1346 }
Benny Prijono268ca612006-02-07 12:34:11 +00001347
Benny Prijono268ca612006-02-07 12:34:11 +00001348
1349 *p_tdata = last_res;
1350
Benny Prijono64f851e2006-02-23 13:49:28 +00001351on_return:
1352 pjsip_dlg_dec_lock(inv->dlg);
1353 return status;
Benny Prijono268ca612006-02-07 12:34:11 +00001354}
1355
1356
1357/*
Benny Prijono26ff9062006-02-21 23:47:00 +00001358 * Set SDP answer.
1359 */
1360PJ_DEF(pj_status_t) pjsip_inv_set_sdp_answer( pjsip_inv_session *inv,
1361 const pjmedia_sdp_session *sdp )
1362{
1363 pj_status_t status;
1364
1365 PJ_ASSERT_RETURN(inv && sdp, PJ_EINVAL);
1366
1367 pjsip_dlg_inc_lock(inv->dlg);
1368 status = pjmedia_sdp_neg_set_local_answer( inv->pool, inv->neg, sdp);
1369 pjsip_dlg_dec_lock(inv->dlg);
1370
1371 return status;
1372}
1373
1374
1375/*
Benny Prijono268ca612006-02-07 12:34:11 +00001376 * End session.
1377 */
1378PJ_DEF(pj_status_t) pjsip_inv_end_session( pjsip_inv_session *inv,
1379 int st_code,
1380 const pj_str_t *st_text,
1381 pjsip_tx_data **p_tdata )
1382{
1383 pjsip_tx_data *tdata;
1384 pj_status_t status;
1385
1386 /* Verify arguments. */
1387 PJ_ASSERT_RETURN(inv && p_tdata, PJ_EINVAL);
1388
Benny Prijonod4e0abd2006-03-05 11:53:36 +00001389 /* Set cause code. */
1390 if (inv->cause==0) inv->cause = st_code;
1391
Benny Prijono268ca612006-02-07 12:34:11 +00001392 /* Create appropriate message. */
1393 switch (inv->state) {
1394 case PJSIP_INV_STATE_CALLING:
1395 case PJSIP_INV_STATE_EARLY:
1396 case PJSIP_INV_STATE_INCOMING:
1397
1398 if (inv->role == PJSIP_ROLE_UAC) {
1399
1400 /* For UAC when session has not been confirmed, create CANCEL. */
1401
1402 /* MUST have the original UAC INVITE transaction. */
1403 PJ_ASSERT_RETURN(inv->invite_tsx != NULL, PJ_EBUG);
1404
1405 /* But CANCEL should only be called when we have received a
1406 * provisional response. If we haven't received any responses,
1407 * just destroy the transaction.
1408 */
1409 if (inv->invite_tsx->status_code < 100) {
1410
1411 pjsip_tsx_terminate(inv->invite_tsx, 487);
Benny Prijonofccab712006-02-22 22:23:22 +00001412 *p_tdata = NULL;
1413 return PJ_SUCCESS;
Benny Prijono268ca612006-02-07 12:34:11 +00001414 }
1415
1416 /* The CSeq here assumes that the dialog is started with an
1417 * INVITE session. This may not be correct; dialog can be
1418 * started as SUBSCRIBE session.
1419 * So fix this!
1420 */
1421 status = pjsip_endpt_create_cancel(inv->dlg->endpt,
1422 inv->invite_tsx->last_tx,
1423 &tdata);
1424
1425 } else {
1426
1427 /* For UAS, send a final response. */
1428 tdata = inv->invite_tsx->last_tx;
1429 PJ_ASSERT_RETURN(tdata != NULL, PJ_EINVALIDOP);
1430
Benny Prijono26ff9062006-02-21 23:47:00 +00001431 //status = pjsip_dlg_modify_response(inv->dlg, tdata, st_code,
1432 // st_text);
1433 status = pjsip_inv_answer(inv, st_code, st_text, NULL, &tdata);
Benny Prijono268ca612006-02-07 12:34:11 +00001434 }
1435 break;
1436
1437 case PJSIP_INV_STATE_CONNECTING:
1438 case PJSIP_INV_STATE_CONFIRMED:
1439 /* For established dialog, send BYE */
1440 status = pjsip_dlg_create_request(inv->dlg, &pjsip_bye_method, -1,
1441 &tdata);
1442 break;
1443
1444 case PJSIP_INV_STATE_DISCONNECTED:
Benny Prijono268ca612006-02-07 12:34:11 +00001445 /* No need to do anything. */
Benny Prijonod4e0abd2006-03-05 11:53:36 +00001446 return PJSIP_ESESSIONTERMINATED;
Benny Prijono268ca612006-02-07 12:34:11 +00001447
1448 default:
1449 pj_assert("!Invalid operation!");
1450 return PJ_EINVALIDOP;
1451 }
1452
1453 if (status != PJ_SUCCESS)
1454 return status;
1455
1456
1457 /* Done */
1458
1459 *p_tdata = tdata;
1460
1461 return PJ_SUCCESS;
1462}
1463
1464
1465/*
1466 * Create re-INVITE.
1467 */
1468PJ_DEF(pj_status_t) pjsip_inv_reinvite( pjsip_inv_session *inv,
1469 const pj_str_t *new_contact,
1470 const pjmedia_sdp_session *new_offer,
1471 pjsip_tx_data **p_tdata )
1472{
Benny Prijono26ff9062006-02-21 23:47:00 +00001473 pj_status_t status;
1474 pjsip_contact_hdr *contact_hdr = NULL;
Benny Prijono268ca612006-02-07 12:34:11 +00001475
Benny Prijono26ff9062006-02-21 23:47:00 +00001476 /* Check arguments. */
1477 PJ_ASSERT_RETURN(inv && p_tdata, PJ_EINVAL);
1478
1479 /* Must NOT have a pending INVITE transaction */
1480 PJ_ASSERT_RETURN(inv->invite_tsx==NULL, PJ_EINVALIDOP);
1481
1482
1483 pjsip_dlg_inc_lock(inv->dlg);
1484
1485 if (new_contact) {
1486 pj_str_t tmp;
1487 const pj_str_t STR_CONTACT = { "Contact", 7 };
1488
1489 pj_strdup_with_null(inv->dlg->pool, &tmp, new_contact);
1490 contact_hdr = pjsip_parse_hdr(inv->dlg->pool, &STR_CONTACT,
1491 tmp.ptr, tmp.slen, NULL);
1492 if (!contact_hdr) {
1493 status = PJSIP_EINVALIDURI;
1494 goto on_return;
1495 }
1496 }
1497
1498
1499 if (new_offer) {
1500 if (!inv->neg) {
1501 status = pjmedia_sdp_neg_create_w_local_offer(inv->pool, new_offer,
1502 &inv->neg);
1503 if (status != PJ_SUCCESS)
1504 goto on_return;
1505
1506 } else switch (pjmedia_sdp_neg_get_state(inv->neg)) {
1507
1508 case PJMEDIA_SDP_NEG_STATE_NULL:
1509 pj_assert(!"Unexpected SDP neg state NULL");
1510 status = PJ_EBUG;
1511 goto on_return;
1512
1513 case PJMEDIA_SDP_NEG_STATE_LOCAL_OFFER:
1514 PJ_LOG(4,(inv->obj_name,
1515 "pjsip_inv_reinvite: already have an offer, new "
1516 "offer is ignored"));
1517 break;
1518
1519 case PJMEDIA_SDP_NEG_STATE_REMOTE_OFFER:
1520 status = pjmedia_sdp_neg_set_local_answer(inv->pool, inv->neg,
1521 new_offer);
1522 if (status != PJ_SUCCESS)
1523 goto on_return;
1524 break;
1525
1526 case PJMEDIA_SDP_NEG_STATE_WAIT_NEGO:
1527 PJ_LOG(4,(inv->obj_name,
1528 "pjsip_inv_reinvite: SDP in WAIT_NEGO state, new "
1529 "offer is ignored"));
1530 break;
1531
1532 case PJMEDIA_SDP_NEG_STATE_DONE:
1533 status = pjmedia_sdp_neg_modify_local_offer(inv->pool,inv->neg,
1534 new_offer);
1535 if (status != PJ_SUCCESS)
1536 goto on_return;
1537 break;
1538 }
1539 }
1540
1541 if (contact_hdr)
1542 inv->dlg->local.contact = contact_hdr;
1543
1544 status = pjsip_inv_invite(inv, p_tdata);
1545
1546on_return:
1547 pjsip_dlg_dec_lock(inv->dlg);
1548 return status;
Benny Prijono268ca612006-02-07 12:34:11 +00001549}
1550
1551/*
1552 * Create UPDATE.
1553 */
1554PJ_DEF(pj_status_t) pjsip_inv_update ( pjsip_inv_session *inv,
1555 const pj_str_t *new_contact,
1556 const pjmedia_sdp_session *new_offer,
1557 pjsip_tx_data **p_tdata )
1558{
1559 PJ_UNUSED_ARG(inv);
1560 PJ_UNUSED_ARG(new_contact);
1561 PJ_UNUSED_ARG(new_offer);
1562 PJ_UNUSED_ARG(p_tdata);
1563
1564 PJ_TODO(CREATE_UPDATE_REQUEST);
1565 return PJ_ENOTSUP;
1566}
1567
1568/*
1569 * Send a request or response message.
1570 */
1571PJ_DEF(pj_status_t) pjsip_inv_send_msg( pjsip_inv_session *inv,
Benny Prijonoe8b0d3b2006-03-17 17:57:52 +00001572 pjsip_tx_data *tdata)
Benny Prijono268ca612006-02-07 12:34:11 +00001573{
1574 pj_status_t status;
1575
1576 /* Verify arguments. */
1577 PJ_ASSERT_RETURN(inv && tdata, PJ_EINVAL);
1578
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00001579 PJ_LOG(5,(inv->obj_name, "Sending %s",
1580 pjsip_tx_data_get_info(tdata)));
1581
Benny Prijono268ca612006-02-07 12:34:11 +00001582 if (tdata->msg->type == PJSIP_REQUEST_MSG) {
Benny Prijonoa66c7152006-02-09 01:26:14 +00001583 struct tsx_inv_data *tsx_inv_data;
Benny Prijono268ca612006-02-07 12:34:11 +00001584
Benny Prijono64158af2006-04-04 11:06:34 +00001585 pjsip_dlg_inc_lock(inv->dlg);
Benny Prijono268ca612006-02-07 12:34:11 +00001586
Benny Prijono64158af2006-04-04 11:06:34 +00001587 tsx_inv_data = pj_pool_zalloc(inv->pool, sizeof(struct tsx_inv_data));
Benny Prijonoa66c7152006-02-09 01:26:14 +00001588 tsx_inv_data->inv = inv;
1589
Benny Prijono64158af2006-04-04 11:06:34 +00001590 pjsip_dlg_dec_lock(inv->dlg);
1591
1592 status = pjsip_dlg_send_request(inv->dlg, tdata, mod_inv.mod.id,
1593 tsx_inv_data);
1594 if (status != PJ_SUCCESS)
1595 return status;
Benny Prijono268ca612006-02-07 12:34:11 +00001596
1597 } else {
1598 pjsip_cseq_hdr *cseq;
1599
1600 /* Can only do this to send response to original INVITE
1601 * request.
1602 */
Benny Prijono64f851e2006-02-23 13:49:28 +00001603 PJ_ASSERT_RETURN((cseq=pjsip_msg_find_hdr(tdata->msg, PJSIP_H_CSEQ, NULL))!=NULL
1604 && (cseq->cseq == inv->invite_tsx->cseq),
Benny Prijono268ca612006-02-07 12:34:11 +00001605 PJ_EINVALIDOP);
1606
1607 status = pjsip_dlg_send_response(inv->dlg, inv->invite_tsx, tdata);
1608 if (status != PJ_SUCCESS)
1609 return status;
1610 }
1611
1612 /* Done (?) */
1613 return PJ_SUCCESS;
1614}
1615
1616
Benny Prijono8ad55352006-02-08 11:16:05 +00001617/*
1618 * Respond to incoming CANCEL request.
1619 */
1620static void inv_respond_incoming_cancel(pjsip_inv_session *inv,
1621 pjsip_transaction *cancel_tsx,
1622 pjsip_rx_data *rdata)
1623{
1624 pjsip_tx_data *tdata;
1625 pjsip_transaction *invite_tsx;
1626 pj_str_t key;
1627 pj_status_t status;
1628
1629 /* See if we have matching INVITE server transaction: */
1630
1631 pjsip_tsx_create_key(rdata->tp_info.pool, &key, PJSIP_ROLE_UAS,
1632 &pjsip_invite_method, rdata);
1633 invite_tsx = pjsip_tsx_layer_find_tsx(&key, PJ_TRUE);
1634
1635 if (invite_tsx == NULL) {
1636
1637 /* Invite transaction not found!
1638 * Respond CANCEL with 491 (RFC 3261 Section 9.2 page 42)
1639 */
1640 status = pjsip_dlg_create_response( inv->dlg, rdata, 200, NULL,
1641 &tdata);
1642
1643 } else {
1644 /* Always answer CANCEL will 200 (OK) regardless of
1645 * the state of the INVITE transaction.
1646 */
1647 status = pjsip_dlg_create_response( inv->dlg, rdata, 200, NULL,
1648 &tdata);
1649 }
1650
1651 /* See if we have created the response successfully. */
1652 if (status != PJ_SUCCESS) return;
1653
1654 /* Send the CANCEL response */
1655 status = pjsip_dlg_send_response(inv->dlg, cancel_tsx, tdata);
1656 if (status != PJ_SUCCESS) return;
1657
1658
1659 /* See if we need to terminate the UAS INVITE transaction
1660 * with 487 (Request Terminated) response.
1661 */
1662 if (invite_tsx && invite_tsx->status_code < 200) {
1663
1664 pj_assert(invite_tsx->last_tx != NULL);
1665
1666 tdata = invite_tsx->last_tx;
1667
1668 status = pjsip_dlg_modify_response(inv->dlg, tdata, 487, NULL);
1669 if (status == PJ_SUCCESS)
1670 pjsip_dlg_send_response(inv->dlg, invite_tsx, tdata);
1671 }
1672
1673 if (invite_tsx)
1674 pj_mutex_unlock(invite_tsx->mutex);
1675}
1676
1677
1678/*
1679 * Respond to incoming BYE request.
1680 */
1681static void inv_respond_incoming_bye( pjsip_inv_session *inv,
1682 pjsip_transaction *bye_tsx,
1683 pjsip_rx_data *rdata,
1684 pjsip_event *e )
1685{
1686 pj_status_t status;
1687 pjsip_tx_data *tdata;
1688
1689 /* Respond BYE with 200: */
1690
1691 status = pjsip_dlg_create_response(inv->dlg, rdata, 200, NULL, &tdata);
1692 if (status != PJ_SUCCESS) return;
1693
1694 status = pjsip_dlg_send_response(inv->dlg, bye_tsx, tdata);
1695 if (status != PJ_SUCCESS) return;
1696
1697 /* Terminate session: */
1698
Benny Prijonod4e0abd2006-03-05 11:53:36 +00001699 if (inv->state != PJSIP_INV_STATE_DISCONNECTED) {
1700 if (inv->cause==0) inv->cause=PJSIP_SC_OK;
Benny Prijono8ad55352006-02-08 11:16:05 +00001701 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
Benny Prijonod4e0abd2006-03-05 11:53:36 +00001702 }
Benny Prijono8ad55352006-02-08 11:16:05 +00001703}
1704
1705/*
Benny Prijono38998232006-02-08 22:44:25 +00001706 * Respond to BYE request.
1707 */
1708static void inv_handle_bye_response( pjsip_inv_session *inv,
1709 pjsip_transaction *tsx,
1710 pjsip_rx_data *rdata,
1711 pjsip_event *e )
1712{
1713 pj_status_t status;
1714
1715 if (e->body.tsx_state.type != PJSIP_EVENT_RX_MSG) {
Benny Prijonod4e0abd2006-03-05 11:53:36 +00001716 if (inv->cause==0) inv->cause=PJSIP_SC_OK;
Benny Prijono38998232006-02-08 22:44:25 +00001717 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
1718 return;
1719 }
1720
1721 /* Handle 401/407 challenge. */
1722 if (tsx->status_code == 401 || tsx->status_code == 407) {
1723
1724 pjsip_tx_data *tdata;
1725
1726 status = pjsip_auth_clt_reinit_req( &inv->dlg->auth_sess,
1727 rdata,
1728 tsx->last_tx,
1729 &tdata);
1730
1731 if (status != PJ_SUCCESS) {
1732
1733 /* Does not have proper credentials.
1734 * End the session anyway.
1735 */
Benny Prijonod4e0abd2006-03-05 11:53:36 +00001736 if (inv->cause==0) inv->cause=PJSIP_SC_OK;
Benny Prijono38998232006-02-08 22:44:25 +00001737 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
1738
1739 } else {
1740 /* Re-send BYE. */
Benny Prijonoe8b0d3b2006-03-17 17:57:52 +00001741 status = pjsip_inv_send_msg(inv, tdata);
Benny Prijono38998232006-02-08 22:44:25 +00001742 }
1743
1744 } else {
1745
1746 /* End the session. */
Benny Prijonod4e0abd2006-03-05 11:53:36 +00001747 if (inv->cause==0) inv->cause=PJSIP_SC_OK;
Benny Prijono38998232006-02-08 22:44:25 +00001748 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
1749 }
1750
1751}
1752
1753/*
Benny Prijono8ad55352006-02-08 11:16:05 +00001754 * State NULL is before anything is sent/received.
1755 */
1756static void inv_on_state_null( pjsip_inv_session *inv, pjsip_event *e)
Benny Prijono268ca612006-02-07 12:34:11 +00001757{
1758 pjsip_transaction *tsx = e->body.tsx_state.tsx;
1759 pjsip_dialog *dlg = pjsip_tsx_get_dlg(tsx);
1760
1761 PJ_ASSERT_ON_FAIL(tsx && dlg, return);
1762
1763 if (tsx->method.id == PJSIP_INVITE_METHOD) {
1764
Benny Prijono64f851e2006-02-23 13:49:28 +00001765 /* Keep the initial INVITE transaction. */
1766 if (inv->invite_tsx == NULL)
1767 inv->invite_tsx = tsx;
Benny Prijono268ca612006-02-07 12:34:11 +00001768
Benny Prijono64f851e2006-02-23 13:49:28 +00001769 if (dlg->role == PJSIP_ROLE_UAC) {
Benny Prijono268ca612006-02-07 12:34:11 +00001770
1771 switch (tsx->state) {
1772 case PJSIP_TSX_STATE_CALLING:
Benny Prijono8ad55352006-02-08 11:16:05 +00001773 inv_set_state(inv, PJSIP_INV_STATE_CALLING, e);
Benny Prijono268ca612006-02-07 12:34:11 +00001774 break;
1775 default:
Benny Prijono64f851e2006-02-23 13:49:28 +00001776 inv_on_state_calling(inv, e);
Benny Prijono268ca612006-02-07 12:34:11 +00001777 break;
1778 }
1779
1780 } else {
1781 switch (tsx->state) {
1782 case PJSIP_TSX_STATE_TRYING:
Benny Prijono8ad55352006-02-08 11:16:05 +00001783 inv_set_state(inv, PJSIP_INV_STATE_INCOMING, e);
Benny Prijono268ca612006-02-07 12:34:11 +00001784 break;
Benny Prijono38998232006-02-08 22:44:25 +00001785 case PJSIP_TSX_STATE_PROCEEDING:
1786 inv_set_state(inv, PJSIP_INV_STATE_INCOMING, e);
1787 if (tsx->status_code > 100)
1788 inv_set_state(inv, PJSIP_INV_STATE_EARLY, e);
1789 break;
Benny Prijono268ca612006-02-07 12:34:11 +00001790 default:
Benny Prijono64f851e2006-02-23 13:49:28 +00001791 inv_on_state_incoming(inv, e);
1792 break;
Benny Prijono268ca612006-02-07 12:34:11 +00001793 }
1794 }
1795
1796 } else {
1797 pj_assert(!"Unexpected transaction type");
1798 }
1799}
1800
Benny Prijono8ad55352006-02-08 11:16:05 +00001801/*
1802 * State CALLING is after sending initial INVITE request but before
1803 * any response (with tag) is received.
1804 */
1805static void inv_on_state_calling( pjsip_inv_session *inv, pjsip_event *e)
Benny Prijono268ca612006-02-07 12:34:11 +00001806{
1807 pjsip_transaction *tsx = e->body.tsx_state.tsx;
1808 pjsip_dialog *dlg = pjsip_tsx_get_dlg(tsx);
Benny Prijonoccf95622006-02-07 18:48:01 +00001809 pj_status_t status;
Benny Prijono268ca612006-02-07 12:34:11 +00001810
1811 PJ_ASSERT_ON_FAIL(tsx && dlg, return);
1812
Benny Prijono8ad55352006-02-08 11:16:05 +00001813 if (tsx == inv->invite_tsx) {
Benny Prijono268ca612006-02-07 12:34:11 +00001814
1815 switch (tsx->state) {
1816
Benny Prijono64f851e2006-02-23 13:49:28 +00001817 case PJSIP_TSX_STATE_CALLING:
1818 inv_set_state(inv, PJSIP_INV_STATE_CALLING, e);
1819 break;
1820
Benny Prijono268ca612006-02-07 12:34:11 +00001821 case PJSIP_TSX_STATE_PROCEEDING:
1822 if (dlg->remote.info->tag.slen) {
Benny Prijonoa66c7152006-02-09 01:26:14 +00001823
Benny Prijono8ad55352006-02-08 11:16:05 +00001824 inv_set_state(inv, PJSIP_INV_STATE_EARLY, e);
Benny Prijonoa66c7152006-02-09 01:26:14 +00001825
1826 inv_check_sdp_in_incoming_msg(inv, tsx,
1827 e->body.tsx_state.src.rdata);
1828
Benny Prijono268ca612006-02-07 12:34:11 +00001829 } else {
1830 /* Ignore 100 (Trying) response, as it doesn't change
1831 * session state. It only ceases retransmissions.
1832 */
1833 }
1834 break;
1835
1836 case PJSIP_TSX_STATE_COMPLETED:
1837 if (tsx->status_code/100 == 2) {
1838
1839 /* This should not happen.
1840 * When transaction receives 2xx, it should be terminated
1841 */
1842 pj_assert(0);
Benny Prijono8ad55352006-02-08 11:16:05 +00001843 inv_set_state(inv, PJSIP_INV_STATE_CONNECTING, e);
Benny Prijonoa66c7152006-02-09 01:26:14 +00001844
1845 inv_check_sdp_in_incoming_msg(inv, tsx,
1846 e->body.tsx_state.src.rdata);
Benny Prijono268ca612006-02-07 12:34:11 +00001847
Benny Prijonoccf95622006-02-07 18:48:01 +00001848 } else if (tsx->status_code==401 || tsx->status_code==407) {
1849
1850 /* Handle authentication failure:
1851 * Resend the request with Authorization header.
1852 */
1853 pjsip_tx_data *tdata;
1854
Benny Prijono8ad55352006-02-08 11:16:05 +00001855 status = pjsip_auth_clt_reinit_req(&inv->dlg->auth_sess,
Benny Prijonoccf95622006-02-07 18:48:01 +00001856 e->body.tsx_state.src.rdata,
1857 tsx->last_tx,
1858 &tdata);
1859
1860 if (status != PJ_SUCCESS) {
1861
1862 /* Does not have proper credentials.
1863 * End the session.
1864 */
Benny Prijonod4e0abd2006-03-05 11:53:36 +00001865 if (inv->cause==0) inv->cause = tsx->status_code;
Benny Prijono8ad55352006-02-08 11:16:05 +00001866 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
Benny Prijonoccf95622006-02-07 18:48:01 +00001867
1868 } else {
1869
1870 /* Restart session. */
Benny Prijono8ad55352006-02-08 11:16:05 +00001871 inv->state = PJSIP_INV_STATE_NULL;
1872 inv->invite_tsx = NULL;
Benny Prijonoccf95622006-02-07 18:48:01 +00001873
1874 /* Send the request. */
Benny Prijonoe8b0d3b2006-03-17 17:57:52 +00001875 status = pjsip_inv_send_msg(inv, tdata);
Benny Prijonoccf95622006-02-07 18:48:01 +00001876 }
1877
Benny Prijono268ca612006-02-07 12:34:11 +00001878 } else {
Benny Prijonoccf95622006-02-07 18:48:01 +00001879
Benny Prijonod4e0abd2006-03-05 11:53:36 +00001880 if (inv->cause==0) inv->cause = tsx->status_code;
Benny Prijono8ad55352006-02-08 11:16:05 +00001881 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
Benny Prijonoccf95622006-02-07 18:48:01 +00001882
Benny Prijono268ca612006-02-07 12:34:11 +00001883 }
1884 break;
1885
1886 case PJSIP_TSX_STATE_TERMINATED:
1887 /* INVITE transaction can be terminated either because UAC
1888 * transaction received 2xx response or because of transport
1889 * error.
1890 */
1891 if (tsx->status_code/100 == 2) {
1892 /* This must be receipt of 2xx response */
1893
1894 /* Set state to CONNECTING */
Benny Prijono8ad55352006-02-08 11:16:05 +00001895 inv_set_state(inv, PJSIP_INV_STATE_CONNECTING, e);
Benny Prijono268ca612006-02-07 12:34:11 +00001896
Benny Prijonoa66c7152006-02-09 01:26:14 +00001897 inv_check_sdp_in_incoming_msg(inv, tsx,
1898 e->body.tsx_state.src.rdata);
1899
Benny Prijono268ca612006-02-07 12:34:11 +00001900 /* Send ACK */
1901 pj_assert(e->body.tsx_state.type == PJSIP_EVENT_RX_MSG);
1902
Benny Prijono8ad55352006-02-08 11:16:05 +00001903 inv_send_ack(inv, e->body.tsx_state.src.rdata);
Benny Prijono5eff0432006-02-09 14:14:21 +00001904 inv_set_state(inv, PJSIP_INV_STATE_CONFIRMED, e);
Benny Prijono268ca612006-02-07 12:34:11 +00001905
Benny Prijonoa66c7152006-02-09 01:26:14 +00001906
Benny Prijono268ca612006-02-07 12:34:11 +00001907 } else {
Benny Prijonod4e0abd2006-03-05 11:53:36 +00001908 if (inv->cause==0) inv->cause = tsx->status_code;
Benny Prijono8ad55352006-02-08 11:16:05 +00001909 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
Benny Prijono268ca612006-02-07 12:34:11 +00001910 }
1911 break;
1912
Benny Prijono34a404e2006-02-09 14:38:30 +00001913 default:
1914 break;
Benny Prijono268ca612006-02-07 12:34:11 +00001915 }
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00001916
1917 } else if (inv->role == PJSIP_ROLE_UAC &&
1918 tsx->role == PJSIP_ROLE_UAC &&
1919 tsx->method.id == PJSIP_CANCEL_METHOD)
1920 {
1921 /*
1922 * Handle case when outgoing CANCEL is answered with 481 (Call/
1923 * Transaction Does Not Exist), 408, or when it's timed out. In these
1924 * cases, disconnect session (i.e. dialog usage only).
1925 */
1926 if (tsx->status_code == PJSIP_SC_CALL_TSX_DOES_NOT_EXIST ||
1927 tsx->status_code == PJSIP_SC_REQUEST_TIMEOUT ||
1928 tsx->status_code == PJSIP_SC_TSX_TIMEOUT ||
1929 PJSIP_SC_TSX_TRANSPORT_ERROR)
1930 {
Benny Prijonod4e0abd2006-03-05 11:53:36 +00001931 if (inv->cause==0) inv->cause = tsx->status_code;
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00001932 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
1933 }
Benny Prijono268ca612006-02-07 12:34:11 +00001934 }
1935}
1936
Benny Prijono8ad55352006-02-08 11:16:05 +00001937/*
1938 * State INCOMING is after we received the request, but before
1939 * responses with tag are sent.
1940 */
1941static void inv_on_state_incoming( pjsip_inv_session *inv, pjsip_event *e)
Benny Prijono268ca612006-02-07 12:34:11 +00001942{
1943 pjsip_transaction *tsx = e->body.tsx_state.tsx;
1944 pjsip_dialog *dlg = pjsip_tsx_get_dlg(tsx);
1945
1946 PJ_ASSERT_ON_FAIL(tsx && dlg, return);
1947
Benny Prijono8ad55352006-02-08 11:16:05 +00001948 if (tsx == inv->invite_tsx) {
1949
1950 /*
1951 * Handle the INVITE state transition.
1952 */
1953
Benny Prijono268ca612006-02-07 12:34:11 +00001954 switch (tsx->state) {
Benny Prijono8ad55352006-02-08 11:16:05 +00001955
Benny Prijono64f851e2006-02-23 13:49:28 +00001956 case PJSIP_TSX_STATE_TRYING:
1957 inv_set_state(inv, PJSIP_INV_STATE_INCOMING, e);
1958 break;
1959
Benny Prijono268ca612006-02-07 12:34:11 +00001960 case PJSIP_TSX_STATE_PROCEEDING:
Benny Prijono8ad55352006-02-08 11:16:05 +00001961 /*
1962 * Transaction sent provisional response.
1963 */
Benny Prijono268ca612006-02-07 12:34:11 +00001964 if (tsx->status_code > 100)
Benny Prijono8ad55352006-02-08 11:16:05 +00001965 inv_set_state(inv, PJSIP_INV_STATE_EARLY, e);
Benny Prijono268ca612006-02-07 12:34:11 +00001966 break;
Benny Prijono8ad55352006-02-08 11:16:05 +00001967
Benny Prijono268ca612006-02-07 12:34:11 +00001968 case PJSIP_TSX_STATE_COMPLETED:
Benny Prijono8ad55352006-02-08 11:16:05 +00001969 /*
1970 * Transaction sent final response.
1971 */
Benny Prijonod4e0abd2006-03-05 11:53:36 +00001972 if (tsx->status_code/100 == 2) {
Benny Prijono8ad55352006-02-08 11:16:05 +00001973 inv_set_state(inv, PJSIP_INV_STATE_CONNECTING, e);
Benny Prijonod4e0abd2006-03-05 11:53:36 +00001974 } else {
1975 if (inv->cause==0) inv->cause = tsx->status_code;
Benny Prijono8ad55352006-02-08 11:16:05 +00001976 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
Benny Prijonod4e0abd2006-03-05 11:53:36 +00001977 }
Benny Prijono268ca612006-02-07 12:34:11 +00001978 break;
Benny Prijono8ad55352006-02-08 11:16:05 +00001979
Benny Prijono268ca612006-02-07 12:34:11 +00001980 case PJSIP_TSX_STATE_TERMINATED:
Benny Prijono8ad55352006-02-08 11:16:05 +00001981 /*
1982 * This happens on transport error (e.g. failed to send
1983 * response)
1984 */
Benny Prijonod4e0abd2006-03-05 11:53:36 +00001985 if (inv->cause==0) inv->cause = tsx->status_code;
Benny Prijono8ad55352006-02-08 11:16:05 +00001986 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
Benny Prijono268ca612006-02-07 12:34:11 +00001987 break;
Benny Prijono8ad55352006-02-08 11:16:05 +00001988
Benny Prijono268ca612006-02-07 12:34:11 +00001989 default:
Benny Prijono8ad55352006-02-08 11:16:05 +00001990 pj_assert(!"Unexpected INVITE state");
1991 break;
Benny Prijono268ca612006-02-07 12:34:11 +00001992 }
Benny Prijono8ad55352006-02-08 11:16:05 +00001993
1994 } else if (tsx->method.id == PJSIP_CANCEL_METHOD &&
1995 tsx->role == PJSIP_ROLE_UAS &&
1996 tsx->state < PJSIP_TSX_STATE_COMPLETED &&
1997 e->body.tsx_state.type == PJSIP_EVENT_RX_MSG )
1998 {
1999
2000 /*
2001 * Handle incoming CANCEL request.
2002 */
2003
2004 inv_respond_incoming_cancel(inv, tsx, e->body.tsx_state.src.rdata);
2005
Benny Prijono268ca612006-02-07 12:34:11 +00002006 }
2007}
2008
Benny Prijono8ad55352006-02-08 11:16:05 +00002009/*
2010 * State EARLY is for both UAS and UAC, after response with To tag
2011 * is sent/received.
2012 */
2013static void inv_on_state_early( pjsip_inv_session *inv, pjsip_event *e)
Benny Prijono268ca612006-02-07 12:34:11 +00002014{
2015 pjsip_transaction *tsx = e->body.tsx_state.tsx;
2016 pjsip_dialog *dlg = pjsip_tsx_get_dlg(tsx);
2017
2018 PJ_ASSERT_ON_FAIL(tsx && dlg, return);
2019
Benny Prijono8ad55352006-02-08 11:16:05 +00002020 if (tsx == inv->invite_tsx) {
2021
2022 /*
2023 * Handle the INVITE state progress.
2024 */
Benny Prijono268ca612006-02-07 12:34:11 +00002025
2026 switch (tsx->state) {
2027
2028 case PJSIP_TSX_STATE_PROCEEDING:
2029 /* Send/received another provisional response. */
Benny Prijono8ad55352006-02-08 11:16:05 +00002030 inv_set_state(inv, PJSIP_INV_STATE_EARLY, e);
Benny Prijonoa66c7152006-02-09 01:26:14 +00002031
2032 if (e->body.tsx_state.type == PJSIP_EVENT_RX_MSG) {
2033 inv_check_sdp_in_incoming_msg(inv, tsx,
2034 e->body.tsx_state.src.rdata);
2035 }
Benny Prijono268ca612006-02-07 12:34:11 +00002036 break;
2037
2038 case PJSIP_TSX_STATE_COMPLETED:
Benny Prijonoa66c7152006-02-09 01:26:14 +00002039 if (tsx->status_code/100 == 2) {
Benny Prijono8ad55352006-02-08 11:16:05 +00002040 inv_set_state(inv, PJSIP_INV_STATE_CONNECTING, e);
Benny Prijonoa66c7152006-02-09 01:26:14 +00002041 if (e->body.tsx_state.type == PJSIP_EVENT_RX_MSG) {
2042 inv_check_sdp_in_incoming_msg(inv, tsx,
2043 e->body.tsx_state.src.rdata);
2044 }
2045
Benny Prijonod4e0abd2006-03-05 11:53:36 +00002046 } else {
2047 if (inv->cause==0) inv->cause = tsx->status_code;
Benny Prijono8ad55352006-02-08 11:16:05 +00002048 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
Benny Prijonod4e0abd2006-03-05 11:53:36 +00002049 }
Benny Prijono268ca612006-02-07 12:34:11 +00002050 break;
2051
Benny Prijonof3195072006-02-14 21:15:30 +00002052 case PJSIP_TSX_STATE_CONFIRMED:
2053 /* For some reason can go here */
2054
Benny Prijono268ca612006-02-07 12:34:11 +00002055 case PJSIP_TSX_STATE_TERMINATED:
2056 /* INVITE transaction can be terminated either because UAC
2057 * transaction received 2xx response or because of transport
2058 * error.
2059 */
2060 if (tsx->status_code/100 == 2) {
2061
2062 /* This must be receipt of 2xx response */
2063
2064 /* Set state to CONNECTING */
Benny Prijono8ad55352006-02-08 11:16:05 +00002065 inv_set_state(inv, PJSIP_INV_STATE_CONNECTING, e);
Benny Prijono268ca612006-02-07 12:34:11 +00002066
Benny Prijonoa66c7152006-02-09 01:26:14 +00002067 if (e->body.tsx_state.type == PJSIP_EVENT_RX_MSG) {
2068 inv_check_sdp_in_incoming_msg(inv, tsx,
2069 e->body.tsx_state.src.rdata);
2070 }
2071
Benny Prijono268ca612006-02-07 12:34:11 +00002072 /* if UAC, send ACK and move state to confirmed. */
2073 if (tsx->role == PJSIP_ROLE_UAC) {
2074 pj_assert(e->body.tsx_state.type == PJSIP_EVENT_RX_MSG);
2075
Benny Prijono8ad55352006-02-08 11:16:05 +00002076 inv_send_ack(inv, e->body.tsx_state.src.rdata);
Benny Prijono8ad55352006-02-08 11:16:05 +00002077 inv_set_state(inv, PJSIP_INV_STATE_CONFIRMED, e);
Benny Prijono268ca612006-02-07 12:34:11 +00002078 }
2079
2080 } else {
Benny Prijonod4e0abd2006-03-05 11:53:36 +00002081 if (inv->cause==0) inv->cause = tsx->status_code;
Benny Prijono8ad55352006-02-08 11:16:05 +00002082 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
Benny Prijono268ca612006-02-07 12:34:11 +00002083 }
2084 break;
2085
2086 default:
Benny Prijono8ad55352006-02-08 11:16:05 +00002087 pj_assert(!"Unexpected INVITE tsx state");
Benny Prijono268ca612006-02-07 12:34:11 +00002088 }
2089
Benny Prijono8ad55352006-02-08 11:16:05 +00002090 } else if (inv->role == PJSIP_ROLE_UAS &&
2091 tsx->role == PJSIP_ROLE_UAS &&
2092 tsx->method.id == PJSIP_CANCEL_METHOD &&
2093 tsx->state < PJSIP_TSX_STATE_COMPLETED &&
2094 e->body.tsx_state.type == PJSIP_EVENT_RX_MSG )
2095 {
Benny Prijono268ca612006-02-07 12:34:11 +00002096
Benny Prijono8ad55352006-02-08 11:16:05 +00002097 /*
2098 * Handle incoming CANCEL request.
Benny Prijonoccf95622006-02-07 18:48:01 +00002099 */
2100
Benny Prijono8ad55352006-02-08 11:16:05 +00002101 inv_respond_incoming_cancel(inv, tsx, e->body.tsx_state.src.rdata);
2102
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00002103 } else if (inv->role == PJSIP_ROLE_UAC &&
2104 tsx->role == PJSIP_ROLE_UAC &&
2105 tsx->method.id == PJSIP_CANCEL_METHOD)
2106 {
2107 /*
2108 * Handle case when outgoing CANCEL is answered with 481 (Call/
2109 * Transaction Does Not Exist), 408, or when it's timed out. In these
2110 * cases, disconnect session (i.e. dialog usage only).
2111 */
2112 if (tsx->status_code == PJSIP_SC_CALL_TSX_DOES_NOT_EXIST ||
2113 tsx->status_code == PJSIP_SC_REQUEST_TIMEOUT ||
2114 tsx->status_code == PJSIP_SC_TSX_TIMEOUT ||
2115 PJSIP_SC_TSX_TRANSPORT_ERROR)
2116 {
Benny Prijonod4e0abd2006-03-05 11:53:36 +00002117 if (inv->cause==0) inv->cause = tsx->status_code;
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00002118 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
2119 }
Benny Prijono268ca612006-02-07 12:34:11 +00002120 }
2121}
2122
Benny Prijono8ad55352006-02-08 11:16:05 +00002123/*
2124 * State CONNECTING is after 2xx response to INVITE is sent/received.
2125 */
2126static void inv_on_state_connecting( pjsip_inv_session *inv, pjsip_event *e)
Benny Prijono268ca612006-02-07 12:34:11 +00002127{
2128 pjsip_transaction *tsx = e->body.tsx_state.tsx;
2129 pjsip_dialog *dlg = pjsip_tsx_get_dlg(tsx);
2130
2131 PJ_ASSERT_ON_FAIL(tsx && dlg, return);
2132
Benny Prijono8ad55352006-02-08 11:16:05 +00002133 if (tsx == inv->invite_tsx) {
Benny Prijono268ca612006-02-07 12:34:11 +00002134
Benny Prijono8ad55352006-02-08 11:16:05 +00002135 /*
2136 * Handle INVITE state progression.
2137 */
Benny Prijono268ca612006-02-07 12:34:11 +00002138 switch (tsx->state) {
2139
2140 case PJSIP_TSX_STATE_CONFIRMED:
Benny Prijono38998232006-02-08 22:44:25 +00002141 if (tsx->status_code/100 == 2)
2142 inv_set_state(inv, PJSIP_INV_STATE_CONFIRMED, e);
Benny Prijono268ca612006-02-07 12:34:11 +00002143 break;
2144
2145 case PJSIP_TSX_STATE_TERMINATED:
2146 /* INVITE transaction can be terminated either because UAC
2147 * transaction received 2xx response or because of transport
2148 * error.
2149 */
2150 if (tsx->status_code/100 != 2) {
Benny Prijonod4e0abd2006-03-05 11:53:36 +00002151 if (inv->cause==0) inv->cause = tsx->status_code;
Benny Prijono8ad55352006-02-08 11:16:05 +00002152 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
Benny Prijono268ca612006-02-07 12:34:11 +00002153 }
2154 break;
2155
2156 case PJSIP_TSX_STATE_DESTROYED:
2157 /* Do nothing. */
2158 break;
2159
2160 default:
2161 pj_assert(!"Unexpected state");
2162 }
2163
Benny Prijono8ad55352006-02-08 11:16:05 +00002164 } else if (tsx->role == PJSIP_ROLE_UAS &&
2165 tsx->method.id == PJSIP_BYE_METHOD &&
2166 tsx->status_code < 200 &&
2167 e->body.tsx_state.type == PJSIP_EVENT_RX_MSG)
2168 {
2169
2170 /*
2171 * Handle incoming BYE.
2172 */
2173
2174 inv_respond_incoming_bye( inv, tsx, e->body.tsx_state.src.rdata, e );
2175
Benny Prijono38998232006-02-08 22:44:25 +00002176 } else if (tsx->method.id == PJSIP_BYE_METHOD &&
2177 tsx->role == PJSIP_ROLE_UAC &&
Benny Prijonoa66c7152006-02-09 01:26:14 +00002178 (tsx->state == PJSIP_TSX_STATE_COMPLETED ||
2179 tsx->state == PJSIP_TSX_STATE_TERMINATED))
Benny Prijono38998232006-02-08 22:44:25 +00002180 {
2181
2182 /*
2183 * Outgoing BYE
2184 */
2185 inv_handle_bye_response( inv, tsx, e->body.tsx_state.src.rdata, e);
2186
Benny Prijono268ca612006-02-07 12:34:11 +00002187 }
Benny Prijono38998232006-02-08 22:44:25 +00002188
Benny Prijono268ca612006-02-07 12:34:11 +00002189}
2190
Benny Prijono8ad55352006-02-08 11:16:05 +00002191/*
2192 * State CONFIRMED is after ACK is sent/received.
2193 */
2194static void inv_on_state_confirmed( pjsip_inv_session *inv, pjsip_event *e)
Benny Prijono268ca612006-02-07 12:34:11 +00002195{
2196 pjsip_transaction *tsx = e->body.tsx_state.tsx;
2197 pjsip_dialog *dlg = pjsip_tsx_get_dlg(tsx);
2198
2199 PJ_ASSERT_ON_FAIL(tsx && dlg, return);
2200
Benny Prijono268ca612006-02-07 12:34:11 +00002201
Benny Prijono8ad55352006-02-08 11:16:05 +00002202 if (tsx->method.id == PJSIP_BYE_METHOD &&
2203 tsx->role == PJSIP_ROLE_UAC &&
Benny Prijonoa66c7152006-02-09 01:26:14 +00002204 (tsx->state == PJSIP_TSX_STATE_COMPLETED ||
2205 tsx->state == PJSIP_TSX_STATE_TERMINATED))
Benny Prijono8ad55352006-02-08 11:16:05 +00002206 {
Benny Prijono38998232006-02-08 22:44:25 +00002207
Benny Prijono8ad55352006-02-08 11:16:05 +00002208 /*
Benny Prijono38998232006-02-08 22:44:25 +00002209 * Outgoing BYE
Benny Prijono8ad55352006-02-08 11:16:05 +00002210 */
Benny Prijono8ad55352006-02-08 11:16:05 +00002211
Benny Prijonoa66c7152006-02-09 01:26:14 +00002212 inv_handle_bye_response( inv, tsx, e->body.tsx_state.src.rdata, e);
Benny Prijono268ca612006-02-07 12:34:11 +00002213
Benny Prijono8ad55352006-02-08 11:16:05 +00002214 }
2215 else if (tsx->method.id == PJSIP_BYE_METHOD &&
2216 tsx->role == PJSIP_ROLE_UAS &&
2217 tsx->status_code < 200 &&
2218 e->body.tsx_state.type == PJSIP_EVENT_RX_MSG)
2219 {
Benny Prijonoccf95622006-02-07 18:48:01 +00002220
Benny Prijono8ad55352006-02-08 11:16:05 +00002221 /*
2222 * Handle incoming BYE.
2223 */
Benny Prijono268ca612006-02-07 12:34:11 +00002224
Benny Prijono8ad55352006-02-08 11:16:05 +00002225 inv_respond_incoming_bye( inv, tsx, e->body.tsx_state.src.rdata, e );
2226
Benny Prijono268ca612006-02-07 12:34:11 +00002227 }
Benny Prijono26ff9062006-02-21 23:47:00 +00002228 else if (tsx->method.id == PJSIP_INVITE_METHOD &&
2229 tsx->role == PJSIP_ROLE_UAS)
2230 {
2231
2232 /*
2233 * Handle incoming re-INVITE
2234 */
2235 if (tsx->state == PJSIP_TSX_STATE_TRYING) {
2236
2237 pjsip_rx_data *rdata = e->body.tsx_state.src.rdata;
2238 pjsip_tx_data *tdata;
2239 pj_status_t status;
2240
2241 /* Check if we have INVITE pending. */
2242 if (inv->invite_tsx && inv->invite_tsx!=tsx) {
2243
2244 /* Can not receive re-INVITE while another one is pending. */
2245 status = pjsip_dlg_create_response( inv->dlg, rdata, 500, NULL,
2246 &tdata);
2247 if (status != PJ_SUCCESS)
2248 return;
2249
2250 status = pjsip_dlg_send_response( inv->dlg, tsx, tdata);
2251
2252
2253 return;
2254 }
2255
2256 /* Save the invite transaction. */
2257 inv->invite_tsx = tsx;
2258
2259 /* Process SDP in incoming message. */
2260 status = inv_check_sdp_in_incoming_msg(inv, tsx, rdata);
2261
2262 if (status != PJ_SUCCESS) {
2263
2264 /* Not Acceptable */
2265 const pjsip_hdr *accept;
2266
2267 status = pjsip_dlg_create_response(inv->dlg, rdata,
2268 488, NULL, &tdata);
2269 if (status != PJ_SUCCESS)
2270 return;
2271
2272
2273 accept = pjsip_endpt_get_capability(dlg->endpt, PJSIP_H_ACCEPT,
2274 NULL);
2275 if (accept) {
2276 pjsip_msg_add_hdr(tdata->msg,
2277 pjsip_hdr_clone(tdata->pool, accept));
2278 }
2279
2280 status = pjsip_dlg_send_response(dlg, tsx, tdata);
2281
2282 return;
2283 }
2284
2285 /* Create 2xx ANSWER */
2286 status = pjsip_dlg_create_response(dlg, rdata, 200, NULL, &tdata);
2287 if (status != PJ_SUCCESS)
2288 return;
2289
2290 /* Process SDP in the answer */
Benny Prijono64f851e2006-02-23 13:49:28 +00002291 status = process_answer(inv, 200, tdata, NULL);
Benny Prijono26ff9062006-02-21 23:47:00 +00002292 if (status != PJ_SUCCESS)
2293 return;
2294
Benny Prijonoe8b0d3b2006-03-17 17:57:52 +00002295 status = pjsip_inv_send_msg(inv, tdata);
Benny Prijono26ff9062006-02-21 23:47:00 +00002296
2297 }
2298
2299 }
2300 else if (tsx->method.id == PJSIP_INVITE_METHOD &&
2301 tsx->role == PJSIP_ROLE_UAC)
2302 {
2303 /*
2304 * Handle outgoing re-INVITE
2305 */
2306 if (tsx->state == PJSIP_TSX_STATE_TERMINATED &&
2307 tsx->status_code/100 == 2)
2308 {
2309
2310 /* Re-INVITE was accepted. */
2311
2312 /* Process SDP */
2313 inv_check_sdp_in_incoming_msg(inv, tsx,
2314 e->body.tsx_state.src.rdata);
2315
2316 /* Send ACK */
2317 inv_send_ack(inv, e->body.tsx_state.src.rdata);
2318
2319 } else if (tsx->state == PJSIP_TSX_STATE_COMPLETED &&
2320 (tsx->status_code==401 || tsx->status_code==407))
2321 {
2322 pjsip_tx_data *tdata;
2323 pj_status_t status;
2324
2325 /* Handle authentication challenge. */
2326 status = pjsip_auth_clt_reinit_req( &dlg->auth_sess,
2327 e->body.tsx_state.src.rdata,
2328 tsx->last_tx,
2329 &tdata);
2330 if (status != PJ_SUCCESS)
2331 return;
2332
2333 /* Send re-INVITE */
Benny Prijonoe8b0d3b2006-03-17 17:57:52 +00002334 status = pjsip_inv_send_msg( inv, tdata);
Benny Prijono26ff9062006-02-21 23:47:00 +00002335
2336 } else if (tsx->status_code==PJSIP_SC_CALL_TSX_DOES_NOT_EXIST ||
2337 tsx->status_code==PJSIP_SC_REQUEST_TIMEOUT ||
2338 tsx->status_code >= 700)
2339 {
2340 /*
2341 * Handle responses that terminates dialog.
2342 */
Benny Prijonod4e0abd2006-03-05 11:53:36 +00002343 if (inv->cause==0) inv->cause = tsx->status_code;
Benny Prijono26ff9062006-02-21 23:47:00 +00002344 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
2345 }
2346 }
Benny Prijono268ca612006-02-07 12:34:11 +00002347}
2348
Benny Prijono8ad55352006-02-08 11:16:05 +00002349/*
2350 * After session has been terminated, but before dialog is destroyed
2351 * (because dialog has other usages, or because dialog is waiting for
2352 * the last transaction to terminate).
2353 */
2354static void inv_on_state_disconnected( pjsip_inv_session *inv, pjsip_event *e)
Benny Prijono268ca612006-02-07 12:34:11 +00002355{
Benny Prijono8ad55352006-02-08 11:16:05 +00002356 pjsip_transaction *tsx = e->body.tsx_state.tsx;
2357 pjsip_dialog *dlg = pjsip_tsx_get_dlg(tsx);
Benny Prijono268ca612006-02-07 12:34:11 +00002358
Benny Prijono8ad55352006-02-08 11:16:05 +00002359 PJ_ASSERT_ON_FAIL(tsx && dlg, return);
2360
2361 if (tsx->method.id == PJSIP_BYE_METHOD &&
2362 tsx->role == PJSIP_ROLE_UAS &&
2363 tsx->status_code < 200 &&
2364 e->body.tsx_state.type == PJSIP_EVENT_RX_MSG)
2365 {
2366
2367 /*
2368 * Be nice, handle incoming BYE.
2369 */
2370
2371 inv_respond_incoming_bye( inv, tsx, e->body.tsx_state.src.rdata, e );
2372
2373 }
Benny Prijono268ca612006-02-07 12:34:11 +00002374}
2375