blob: c0e241d10babd9be28319f4ee73b180420695932 [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
36/*
37 * Static prototypes.
38 */
39static pj_status_t mod_inv_load(pjsip_endpoint *endpt);
40static pj_status_t mod_inv_unload(void);
41static pj_bool_t mod_inv_on_rx_request(pjsip_rx_data *rdata);
42static pj_bool_t mod_inv_on_rx_response(pjsip_rx_data *rdata);
43static void mod_inv_on_tsx_state(pjsip_transaction*, pjsip_event*);
44
Benny Prijono8ad55352006-02-08 11:16:05 +000045static void inv_on_state_null( pjsip_inv_session *inv, pjsip_event *e);
46static void inv_on_state_calling( pjsip_inv_session *inv, pjsip_event *e);
47static void inv_on_state_incoming( pjsip_inv_session *inv, pjsip_event *e);
48static void inv_on_state_early( pjsip_inv_session *inv, pjsip_event *e);
49static void inv_on_state_connecting( pjsip_inv_session *inv, pjsip_event *e);
50static void inv_on_state_confirmed( pjsip_inv_session *inv, pjsip_event *e);
51static void inv_on_state_disconnected( pjsip_inv_session *inv, pjsip_event *e);
Benny Prijono268ca612006-02-07 12:34:11 +000052
Benny Prijono8ad55352006-02-08 11:16:05 +000053static void (*inv_state_handler[])( pjsip_inv_session *inv, pjsip_event *e) =
Benny Prijono268ca612006-02-07 12:34:11 +000054{
55 &inv_on_state_null,
56 &inv_on_state_calling,
57 &inv_on_state_incoming,
58 &inv_on_state_early,
59 &inv_on_state_connecting,
60 &inv_on_state_confirmed,
61 &inv_on_state_disconnected,
Benny Prijono268ca612006-02-07 12:34:11 +000062};
63
64static struct mod_inv
65{
66 pjsip_module mod;
67 pjsip_endpoint *endpt;
68 pjsip_inv_callback cb;
69 pjsip_module *app_user;
70} mod_inv =
71{
72 {
Benny Prijono2f8992b2006-02-25 21:16:36 +000073 NULL, NULL, /* prev, next. */
74 { "mod-invite", 10 }, /* Name. */
75 -1, /* Id */
76 PJSIP_MOD_PRIORITY_DIALOG_USAGE, /* Priority */
77 &mod_inv_load, /* load() */
78 NULL, /* start() */
79 NULL, /* stop() */
80 &mod_inv_unload, /* unload() */
81 &mod_inv_on_rx_request, /* on_rx_request() */
82 &mod_inv_on_rx_response, /* on_rx_response() */
83 NULL, /* on_tx_request. */
84 NULL, /* on_tx_response() */
85 &mod_inv_on_tsx_state, /* on_tsx_state() */
Benny Prijono268ca612006-02-07 12:34:11 +000086 }
87};
88
89
Benny Prijonoa66c7152006-02-09 01:26:14 +000090/* Invite session data to be attached to transaction. */
91struct tsx_inv_data
92{
93 pjsip_inv_session *inv;
94 pj_bool_t sdp_done;
95};
96
97
Benny Prijono8ad55352006-02-08 11:16:05 +000098/*
99 * Module load()
100 */
Benny Prijono268ca612006-02-07 12:34:11 +0000101static pj_status_t mod_inv_load(pjsip_endpoint *endpt)
102{
103 pj_str_t allowed[] = {{"INVITE", 6}, {"ACK",3}, {"BYE",3}, {"CANCEL",6}};
104
105 /* Register supported methods: INVITE, ACK, BYE, CANCEL */
106 pjsip_endpt_add_capability(endpt, &mod_inv.mod, PJSIP_H_ALLOW, NULL,
107 PJ_ARRAY_SIZE(allowed), allowed);
108
109 return PJ_SUCCESS;
110}
111
Benny Prijono8ad55352006-02-08 11:16:05 +0000112/*
113 * Module unload()
114 */
Benny Prijono268ca612006-02-07 12:34:11 +0000115static pj_status_t mod_inv_unload(void)
116{
117 /* Should remove capability here */
118 return PJ_SUCCESS;
119}
120
Benny Prijono8ad55352006-02-08 11:16:05 +0000121/*
Benny Prijono38998232006-02-08 22:44:25 +0000122 * Set session state.
123 */
124void inv_set_state(pjsip_inv_session *inv, pjsip_inv_state state,
125 pjsip_event *e)
126{
Benny Prijonod4e0abd2006-03-05 11:53:36 +0000127 pjsip_inv_state prev_state = inv->state;
128
129 /* Set state. */
Benny Prijono38998232006-02-08 22:44:25 +0000130 inv->state = state;
Benny Prijonod4e0abd2006-03-05 11:53:36 +0000131
132 /* If state is DISCONNECTED, cause code MUST have been set. */
133 pj_assert(inv->state != PJSIP_INV_STATE_DISCONNECTED ||
134 inv->cause != 0);
135
136 /* Call on_state_changed() callback. */
137 if (mod_inv.cb.on_state_changed && inv->notify)
Benny Prijono38998232006-02-08 22:44:25 +0000138 (*mod_inv.cb.on_state_changed)(inv, e);
139
Benny Prijonod4e0abd2006-03-05 11:53:36 +0000140 /* Only decrement when previous state is not already DISCONNECTED */
141 if (inv->state == PJSIP_INV_STATE_DISCONNECTED &&
142 prev_state != PJSIP_INV_STATE_DISCONNECTED)
143 {
Benny Prijonoe4f2abb2006-02-10 14:04:05 +0000144 pjsip_dlg_dec_session(inv->dlg, &mod_inv.mod);
Benny Prijonod4e0abd2006-03-05 11:53:36 +0000145 }
Benny Prijono38998232006-02-08 22:44:25 +0000146}
147
148
149/*
Benny Prijono8ad55352006-02-08 11:16:05 +0000150 * Send ACK for 2xx response.
151 */
152static pj_status_t inv_send_ack(pjsip_inv_session *inv, pjsip_rx_data *rdata)
Benny Prijono268ca612006-02-07 12:34:11 +0000153{
154 pjsip_tx_data *tdata;
155 pj_status_t status;
156
Benny Prijonoe4f2abb2006-02-10 14:04:05 +0000157 PJ_LOG(5,(inv->obj_name, "Received %s, sending ACK",
158 pjsip_rx_data_get_info(rdata)));
159
Benny Prijono268ca612006-02-07 12:34:11 +0000160 status = pjsip_dlg_create_request(inv->dlg, &pjsip_ack_method,
161 rdata->msg_info.cseq->cseq, &tdata);
162 if (status != PJ_SUCCESS) {
163 /* Better luck next time */
164 pj_assert(!"Unable to create ACK!");
165 return status;
166 }
167
168 status = pjsip_dlg_send_request(inv->dlg, tdata, NULL);
169 if (status != PJ_SUCCESS) {
170 /* Better luck next time */
171 pj_assert(!"Unable to send ACK!");
172 return status;
173 }
174
175 return PJ_SUCCESS;
176}
177
Benny Prijono8ad55352006-02-08 11:16:05 +0000178/*
179 * Module on_rx_request()
180 *
181 * This callback is called for these events:
182 * - endpoint receives request which was unhandled by higher priority
183 * modules (e.g. transaction layer, dialog layer).
184 * - dialog distributes incoming request to its usages.
185 */
186static pj_bool_t mod_inv_on_rx_request(pjsip_rx_data *rdata)
187{
188 pjsip_method *method;
Benny Prijono38998232006-02-08 22:44:25 +0000189 pjsip_dialog *dlg;
190 pjsip_inv_session *inv;
Benny Prijono8ad55352006-02-08 11:16:05 +0000191
192 /* Only wants to receive request from a dialog. */
Benny Prijono38998232006-02-08 22:44:25 +0000193 dlg = pjsip_rdata_get_dlg(rdata);
194 if (dlg == NULL)
Benny Prijono8ad55352006-02-08 11:16:05 +0000195 return PJ_FALSE;
196
Benny Prijono38998232006-02-08 22:44:25 +0000197 inv = dlg->mod_data[mod_inv.mod.id];
198
Benny Prijono8ad55352006-02-08 11:16:05 +0000199 /* Report to dialog that we handle INVITE, CANCEL, BYE, ACK.
200 * If we need to send response, it will be sent in the state
201 * handlers.
202 */
203 method = &rdata->msg_info.msg->line.req.method;
204
205 if (method->id == PJSIP_INVITE_METHOD ||
206 method->id == PJSIP_CANCEL_METHOD ||
Benny Prijono8ad55352006-02-08 11:16:05 +0000207 method->id == PJSIP_BYE_METHOD)
208 {
209 return PJ_TRUE;
210 }
211
Benny Prijono38998232006-02-08 22:44:25 +0000212 /* On receipt ACK request, when state is CONNECTING,
213 * move state to CONFIRMED.
214 */
Benny Prijono26ff9062006-02-21 23:47:00 +0000215 if (method->id == PJSIP_ACK_METHOD && inv) {
Benny Prijono38998232006-02-08 22:44:25 +0000216
Benny Prijono5eff0432006-02-09 14:14:21 +0000217 /* Terminate INVITE transaction, if it's still present. */
218 if (inv->invite_tsx &&
219 inv->invite_tsx->state <= PJSIP_TSX_STATE_COMPLETED)
220 {
221 pjsip_tsx_terminate(inv->invite_tsx,
222 inv->invite_tsx->status_code);
223 inv->invite_tsx = NULL;
224 }
225
Benny Prijono32ac8fe2006-03-02 21:16:55 +0000226 /* On receipt of ACK, only set state to confirmed when state
227 * is CONNECTING (e.g. we don't want to set the state to confirmed
228 * when we receive ACK retransmission after sending non-2xx!)
229 */
230 if (inv->state == PJSIP_INV_STATE_CONNECTING) {
Benny Prijono26ff9062006-02-21 23:47:00 +0000231 pjsip_event event;
232
233 PJSIP_EVENT_INIT_RX_MSG(event, rdata);
234 inv_set_state(inv, PJSIP_INV_STATE_CONFIRMED, &event);
235 }
Benny Prijono38998232006-02-08 22:44:25 +0000236 }
237
Benny Prijono8ad55352006-02-08 11:16:05 +0000238 return PJ_FALSE;
239}
240
241/*
242 * Module on_rx_response().
243 *
244 * This callback is called for these events:
245 * - dialog distributes incoming 2xx response to INVITE (outside
246 * transaction) to its usages.
247 * - endpoint distributes strayed responses.
248 */
Benny Prijono268ca612006-02-07 12:34:11 +0000249static pj_bool_t mod_inv_on_rx_response(pjsip_rx_data *rdata)
250{
251 pjsip_dialog *dlg;
252 pjsip_inv_session *inv;
253 pjsip_msg *msg = rdata->msg_info.msg;
254
255 dlg = pjsip_rdata_get_dlg(rdata);
256
257 /* Ignore responses outside dialog */
258 if (dlg == NULL)
259 return PJ_FALSE;
260
261 /* Ignore responses not belonging to invite session */
262 inv = pjsip_dlg_get_inv_session(dlg);
263 if (inv == NULL)
264 return PJ_FALSE;
265
266 /* This MAY be retransmission of 2xx response to INVITE.
267 * If it is, we need to send ACK.
268 */
269 if (msg->type == PJSIP_RESPONSE_MSG && msg->line.status.code/100==2 &&
Benny Prijono26ff9062006-02-21 23:47:00 +0000270 rdata->msg_info.cseq->method.id == PJSIP_INVITE_METHOD &&
271 inv->invite_tsx == NULL)
272 {
Benny Prijono268ca612006-02-07 12:34:11 +0000273
Benny Prijono8ad55352006-02-08 11:16:05 +0000274 inv_send_ack(inv, rdata);
Benny Prijono268ca612006-02-07 12:34:11 +0000275 return PJ_TRUE;
276
277 }
278
279 /* No other processing needs to be done here. */
280 return PJ_FALSE;
281}
282
Benny Prijono8ad55352006-02-08 11:16:05 +0000283/*
284 * Module on_tsx_state()
285 *
286 * This callback is called by dialog framework for all transactions
287 * inside the dialog for all its dialog usages.
288 */
Benny Prijono268ca612006-02-07 12:34:11 +0000289static void mod_inv_on_tsx_state(pjsip_transaction *tsx, pjsip_event *e)
290{
291 pjsip_dialog *dlg;
292 pjsip_inv_session *inv;
293
294 dlg = pjsip_tsx_get_dlg(tsx);
295 if (dlg == NULL)
296 return;
297
298 inv = pjsip_dlg_get_inv_session(dlg);
299 if (inv == NULL)
300 return;
301
302 /* Call state handler for the invite session. */
303 (*inv_state_handler[inv->state])(inv, e);
304
305 /* Call on_tsx_state */
Benny Prijonod4e0abd2006-03-05 11:53:36 +0000306 if (mod_inv.cb.on_tsx_state_changed && inv->notify)
Benny Prijono268ca612006-02-07 12:34:11 +0000307 (*mod_inv.cb.on_tsx_state_changed)(inv, tsx, e);
308
309 /* Clear invite transaction when tsx is terminated. */
310 if (tsx->state==PJSIP_TSX_STATE_TERMINATED && tsx == inv->invite_tsx)
311 inv->invite_tsx = NULL;
312}
313
Benny Prijono8ad55352006-02-08 11:16:05 +0000314
315/*
316 * Initialize the invite module.
317 */
Benny Prijono268ca612006-02-07 12:34:11 +0000318PJ_DEF(pj_status_t) pjsip_inv_usage_init( pjsip_endpoint *endpt,
319 pjsip_module *app_module,
320 const pjsip_inv_callback *cb)
321{
322 pj_status_t status;
323
324 /* Check arguments. */
325 PJ_ASSERT_RETURN(endpt && app_module && cb, PJ_EINVAL);
326
327 /* Some callbacks are mandatory */
328 PJ_ASSERT_RETURN(cb->on_state_changed && cb->on_new_session, PJ_EINVAL);
329
330 /* Check if module already registered. */
331 PJ_ASSERT_RETURN(mod_inv.mod.id == -1, PJ_EINVALIDOP);
332
333 /* Copy param. */
334 pj_memcpy(&mod_inv.cb, cb, sizeof(pjsip_inv_callback));
335
336 mod_inv.endpt = endpt;
337 mod_inv.app_user = app_module;
338
339 /* Register the module. */
340 status = pjsip_endpt_register_module(endpt, &mod_inv.mod);
341
342 return status;
343}
344
Benny Prijono8ad55352006-02-08 11:16:05 +0000345/*
346 * Get the instance of invite module.
347 */
Benny Prijono268ca612006-02-07 12:34:11 +0000348PJ_DEF(pjsip_module*) pjsip_inv_usage_instance(void)
349{
350 return &mod_inv.mod;
351}
352
353
Benny Prijono632ce712006-02-09 14:01:40 +0000354
Benny Prijono8ad55352006-02-08 11:16:05 +0000355/*
356 * Return the invite session for the specified dialog.
357 */
Benny Prijono268ca612006-02-07 12:34:11 +0000358PJ_DEF(pjsip_inv_session*) pjsip_dlg_get_inv_session(pjsip_dialog *dlg)
359{
360 return dlg->mod_data[mod_inv.mod.id];
361}
362
Benny Prijono8ad55352006-02-08 11:16:05 +0000363
Benny Prijono268ca612006-02-07 12:34:11 +0000364/*
Benny Prijono8ad55352006-02-08 11:16:05 +0000365 * Create UAC invite session.
Benny Prijono268ca612006-02-07 12:34:11 +0000366 */
367PJ_DEF(pj_status_t) pjsip_inv_create_uac( pjsip_dialog *dlg,
368 const pjmedia_sdp_session *local_sdp,
369 unsigned options,
370 pjsip_inv_session **p_inv)
371{
372 pjsip_inv_session *inv;
373 pj_status_t status;
374
375 /* Verify arguments. */
376 PJ_ASSERT_RETURN(dlg && p_inv, PJ_EINVAL);
377
378 /* Normalize options */
379 if (options & PJSIP_INV_REQUIRE_100REL)
380 options |= PJSIP_INV_SUPPORT_100REL;
381
382 if (options & PJSIP_INV_REQUIRE_TIMER)
383 options |= PJSIP_INV_SUPPORT_TIMER;
384
385 /* Create the session */
386 inv = pj_pool_zalloc(dlg->pool, sizeof(pjsip_inv_session));
387 PJ_ASSERT_RETURN(inv != NULL, PJ_ENOMEM);
388
389 inv->pool = dlg->pool;
390 inv->role = PJSIP_ROLE_UAC;
391 inv->state = PJSIP_INV_STATE_NULL;
392 inv->dlg = dlg;
393 inv->options = options;
Benny Prijonod4e0abd2006-03-05 11:53:36 +0000394 inv->notify = PJ_TRUE;
395 inv->cause = 0;
Benny Prijono268ca612006-02-07 12:34:11 +0000396
Benny Prijonoe4f2abb2006-02-10 14:04:05 +0000397 /* Object name will use the same dialog pointer. */
Benny Prijonoed811d72006-03-10 12:57:12 +0000398 pj_ansi_snprintf(inv->obj_name, PJ_MAX_OBJ_NAME, "inv%p", dlg);
Benny Prijonoe4f2abb2006-02-10 14:04:05 +0000399
Benny Prijono268ca612006-02-07 12:34:11 +0000400 /* Create negotiator if local_sdp is specified. */
401 if (local_sdp) {
402 status = pjmedia_sdp_neg_create_w_local_offer(dlg->pool, local_sdp,
403 &inv->neg);
404 if (status != PJ_SUCCESS)
405 return status;
406 }
407
408 /* Register invite as dialog usage. */
409 status = pjsip_dlg_add_usage(dlg, &mod_inv.mod, inv);
410 if (status != PJ_SUCCESS)
411 return status;
412
413 /* Increment dialog session */
Benny Prijonoe4f2abb2006-02-10 14:04:05 +0000414 pjsip_dlg_inc_session(dlg, &mod_inv.mod);
Benny Prijono268ca612006-02-07 12:34:11 +0000415
416 /* Done */
417 *p_inv = inv;
Benny Prijonoe4f2abb2006-02-10 14:04:05 +0000418
419 PJ_LOG(5,(inv->obj_name, "UAC invite session created for dialog %s",
420 dlg->obj_name));
421
Benny Prijono268ca612006-02-07 12:34:11 +0000422 return PJ_SUCCESS;
423}
424
425/*
426 * Verify incoming INVITE request.
427 */
428PJ_DEF(pj_status_t) pjsip_inv_verify_request(pjsip_rx_data *rdata,
429 unsigned *options,
430 const pjmedia_sdp_session *l_sdp,
431 pjsip_dialog *dlg,
432 pjsip_endpoint *endpt,
433 pjsip_tx_data **p_tdata)
434{
435 pjsip_msg *msg;
436 pjsip_allow_hdr *allow;
437 pjsip_supported_hdr *sup_hdr;
438 pjsip_require_hdr *req_hdr;
439 int code = 200;
440 unsigned rem_option = 0;
441 pj_status_t status = PJ_SUCCESS;
442 pjsip_hdr res_hdr_list;
443
444 /* Init return arguments. */
445 if (p_tdata) *p_tdata = NULL;
446
447 /* Verify arguments. */
448 PJ_ASSERT_RETURN(rdata != NULL && options != NULL, PJ_EINVAL);
449
450 /* Normalize options */
451 if (*options & PJSIP_INV_REQUIRE_100REL)
452 *options |= PJSIP_INV_SUPPORT_100REL;
453
454 if (*options & PJSIP_INV_REQUIRE_TIMER)
455 *options |= PJSIP_INV_SUPPORT_TIMER;
456
457 /* Get the message in rdata */
458 msg = rdata->msg_info.msg;
459
460 /* Must be INVITE request. */
461 PJ_ASSERT_RETURN(msg->type == PJSIP_REQUEST_MSG &&
462 msg->line.req.method.id == PJSIP_INVITE_METHOD,
463 PJ_EINVAL);
464
465 /* If tdata is specified, then either dlg or endpt must be specified */
466 PJ_ASSERT_RETURN((!p_tdata) || (endpt || dlg), PJ_EINVAL);
467
468 /* Get the endpoint */
469 endpt = endpt ? endpt : dlg->endpt;
470
471 /* Init response header list */
472 pj_list_init(&res_hdr_list);
473
Benny Prijono8ad55352006-02-08 11:16:05 +0000474 /* Check the request body, see if it'inv something that we support
Benny Prijono268ca612006-02-07 12:34:11 +0000475 * (i.e. SDP).
476 */
477 if (msg->body) {
478 pjsip_msg_body *body = msg->body;
479 pj_str_t str_application = {"application", 11};
480 pj_str_t str_sdp = { "sdp", 3 };
481 pjmedia_sdp_session *sdp;
482
483 /* Check content type. */
484 if (pj_stricmp(&body->content_type.type, &str_application) != 0 ||
485 pj_stricmp(&body->content_type.subtype, &str_sdp) != 0)
486 {
487 /* Not "application/sdp" */
488 code = PJSIP_SC_UNSUPPORTED_MEDIA_TYPE;
489 status = PJSIP_ERRNO_FROM_SIP_STATUS(code);
490
491 if (p_tdata) {
492 /* Add Accept header to response */
493 pjsip_accept_hdr *acc;
494
495 acc = pjsip_accept_hdr_create(rdata->tp_info.pool);
496 PJ_ASSERT_RETURN(acc, PJ_ENOMEM);
497 acc->values[acc->count++] = pj_str("application/sdp");
498 pj_list_push_back(&res_hdr_list, acc);
499 }
500
501 goto on_return;
502 }
503
504 /* Parse and validate SDP */
505 status = pjmedia_sdp_parse(rdata->tp_info.pool, body->data, body->len,
506 &sdp);
507 if (status == PJ_SUCCESS)
508 status = pjmedia_sdp_validate(sdp);
509
510 if (status != PJ_SUCCESS) {
511 /* Unparseable or invalid SDP */
512 code = PJSIP_SC_BAD_REQUEST;
513
514 if (p_tdata) {
515 /* Add Warning header. */
516 pjsip_warning_hdr *w;
517
518 w = pjsip_warning_hdr_create_from_status(rdata->tp_info.pool,
519 pjsip_endpt_name(endpt),
520 status);
521 PJ_ASSERT_RETURN(w, PJ_ENOMEM);
522
523 pj_list_push_back(&res_hdr_list, w);
524 }
525
526 goto on_return;
527 }
528
529 /* Negotiate with local SDP */
530 if (l_sdp) {
531 pjmedia_sdp_neg *neg;
532
533 /* Local SDP must be valid! */
534 PJ_ASSERT_RETURN((status=pjmedia_sdp_validate(l_sdp))==PJ_SUCCESS,
535 status);
536
537 /* Create SDP negotiator */
538 status = pjmedia_sdp_neg_create_w_remote_offer(
539 rdata->tp_info.pool, l_sdp, sdp, &neg);
540 PJ_ASSERT_RETURN(status == PJ_SUCCESS, status);
541
542 /* Negotiate SDP */
543 status = pjmedia_sdp_neg_negotiate(rdata->tp_info.pool, neg, 0);
544 if (status != PJ_SUCCESS) {
545
546 /* Incompatible media */
547 code = PJSIP_SC_NOT_ACCEPTABLE_HERE;
548 status = PJSIP_ERRNO_FROM_SIP_STATUS(code);
549
550 if (p_tdata) {
551 pjsip_accept_hdr *acc;
552 pjsip_warning_hdr *w;
553
554 /* Add Warning header. */
555 w = pjsip_warning_hdr_create_from_status(
556 rdata->tp_info.pool,
557 pjsip_endpt_name(endpt), status);
558 PJ_ASSERT_RETURN(w, PJ_ENOMEM);
559
560 pj_list_push_back(&res_hdr_list, w);
561
562 /* Add Accept header to response */
563 acc = pjsip_accept_hdr_create(rdata->tp_info.pool);
564 PJ_ASSERT_RETURN(acc, PJ_ENOMEM);
565 acc->values[acc->count++] = pj_str("application/sdp");
566 pj_list_push_back(&res_hdr_list, acc);
567
568 }
569
570 goto on_return;
571 }
572 }
573 }
574
575 /* Check supported methods, see if peer supports UPDATE.
576 * We just assume that peer supports standard INVITE, ACK, CANCEL, and BYE
577 * implicitly by sending this INVITE.
578 */
579 allow = pjsip_msg_find_hdr(msg, PJSIP_H_ALLOW, NULL);
580 if (allow) {
581 unsigned i;
582 const pj_str_t STR_UPDATE = { "UPDATE", 6 };
583
584 for (i=0; i<allow->count; ++i) {
585 if (pj_stricmp(&allow->values[i], &STR_UPDATE)==0)
586 break;
587 }
588
589 if (i != allow->count) {
590 /* UPDATE is present in Allow */
591 rem_option |= PJSIP_INV_SUPPORT_UPDATE;
592 }
593
594 }
595
596 /* Check Supported header */
597 sup_hdr = pjsip_msg_find_hdr(msg, PJSIP_H_SUPPORTED, NULL);
598 if (sup_hdr) {
599 unsigned i;
600 pj_str_t STR_100REL = { "100rel", 6};
601 pj_str_t STR_TIMER = { "timer", 5 };
602
603 for (i=0; i<sup_hdr->count; ++i) {
604 if (pj_stricmp(&sup_hdr->values[i], &STR_100REL)==0)
605 rem_option |= PJSIP_INV_SUPPORT_100REL;
606 else if (pj_stricmp(&sup_hdr->values[i], &STR_TIMER)==0)
607 rem_option |= PJSIP_INV_SUPPORT_TIMER;
608 }
609 }
610
611 /* Check Require header */
612 req_hdr = pjsip_msg_find_hdr(msg, PJSIP_H_REQUIRE, NULL);
613 if (req_hdr) {
614 unsigned i;
615 pj_str_t STR_100REL = { "100rel", 6};
616 pj_str_t STR_TIMER = { "timer", 5 };
617 unsigned unsupp_cnt = 0;
618 pj_str_t unsupp_tags[PJSIP_GENERIC_ARRAY_MAX_COUNT];
619
620 for (i=0; i<req_hdr->count; ++i) {
621 if ((*options & PJSIP_INV_SUPPORT_100REL) &&
622 pj_stricmp(&req_hdr->values[i], &STR_100REL)==0)
623 {
624 rem_option |= PJSIP_INV_REQUIRE_100REL;
625
626 } else if ((*options && PJSIP_INV_SUPPORT_TIMER) &&
627 pj_stricmp(&req_hdr->values[i], &STR_TIMER)==0)
628 {
629 rem_option |= PJSIP_INV_REQUIRE_TIMER;
630
631 } else {
632 /* Unknown/unsupported extension tag! */
633 unsupp_tags[unsupp_cnt++] = req_hdr->values[i];
634 }
635 }
636
637 /* Check if there are required tags that we don't support */
638 if (unsupp_cnt) {
639
640 code = PJSIP_SC_BAD_EXTENSION;
641 status = PJSIP_ERRNO_FROM_SIP_STATUS(code);
642
643 if (p_tdata) {
644 pjsip_unsupported_hdr *unsupp_hdr;
645 const pjsip_hdr *h;
646
647 /* Add Unsupported header. */
648 unsupp_hdr = pjsip_unsupported_hdr_create(rdata->tp_info.pool);
649 PJ_ASSERT_RETURN(unsupp_hdr != NULL, PJ_ENOMEM);
650
651 unsupp_hdr->count = unsupp_cnt;
652 for (i=0; i<unsupp_cnt; ++i)
653 unsupp_hdr->values[i] = unsupp_tags[i];
654
655 pj_list_push_back(&res_hdr_list, unsupp_hdr);
656
657 /* Add Supported header. */
658 h = pjsip_endpt_get_capability(endpt, PJSIP_H_SUPPORTED,
659 NULL);
660 pj_assert(h);
661 if (h) {
662 sup_hdr = pjsip_hdr_clone(rdata->tp_info.pool, h);
663 pj_list_push_back(&res_hdr_list, sup_hdr);
664 }
665 }
666
667 goto on_return;
668 }
669 }
670
671 /* Check if there are local requirements that are not supported
672 * by peer.
673 */
674 if ( ((*options & PJSIP_INV_REQUIRE_100REL)!=0 &&
675 (rem_option & PJSIP_INV_SUPPORT_100REL)==0) ||
676 ((*options & PJSIP_INV_REQUIRE_TIMER)!=0 &&
677 (rem_option & PJSIP_INV_SUPPORT_TIMER)==0))
678 {
679 code = PJSIP_SC_EXTENSION_REQUIRED;
680 status = PJSIP_ERRNO_FROM_SIP_STATUS(code);
681
682 if (p_tdata) {
683 const pjsip_hdr *h;
684
685 /* Add Require header. */
686 req_hdr = pjsip_require_hdr_create(rdata->tp_info.pool);
687 PJ_ASSERT_RETURN(req_hdr != NULL, PJ_ENOMEM);
688
689 if (*options & PJSIP_INV_REQUIRE_100REL)
690 req_hdr->values[req_hdr->count++] = pj_str("100rel");
691
692 if (*options & PJSIP_INV_REQUIRE_TIMER)
693 req_hdr->values[req_hdr->count++] = pj_str("timer");
694
695 pj_list_push_back(&res_hdr_list, req_hdr);
696
697 /* Add Supported header. */
698 h = pjsip_endpt_get_capability(endpt, PJSIP_H_SUPPORTED,
699 NULL);
700 pj_assert(h);
701 if (h) {
702 sup_hdr = pjsip_hdr_clone(rdata->tp_info.pool, h);
703 pj_list_push_back(&res_hdr_list, sup_hdr);
704 }
705
706 }
707
708 goto on_return;
709 }
710
711on_return:
712
713 /* Create response if necessary */
714 if (code != 200 && p_tdata) {
715 pjsip_tx_data *tdata;
716 const pjsip_hdr *h;
717
718 if (dlg) {
719 status = pjsip_dlg_create_response(dlg, rdata, code, NULL,
720 &tdata);
721 } else {
722 status = pjsip_endpt_create_response(endpt, rdata, code, NULL,
723 &tdata);
724 }
725
726 if (status != PJ_SUCCESS)
727 return status;
728
729 /* Add response headers. */
730 h = res_hdr_list.next;
731 while (h != &res_hdr_list) {
732 pjsip_hdr *cloned;
733
734 cloned = pjsip_hdr_clone(tdata->pool, h);
735 PJ_ASSERT_RETURN(cloned, PJ_ENOMEM);
736
737 pjsip_msg_add_hdr(tdata->msg, cloned);
738
739 h = h->next;
740 }
741
742 *p_tdata = tdata;
743 }
744
745 return status;
746}
747
748/*
Benny Prijono8ad55352006-02-08 11:16:05 +0000749 * Create UAS invite session.
Benny Prijono268ca612006-02-07 12:34:11 +0000750 */
751PJ_DEF(pj_status_t) pjsip_inv_create_uas( pjsip_dialog *dlg,
752 pjsip_rx_data *rdata,
753 const pjmedia_sdp_session *local_sdp,
754 unsigned options,
755 pjsip_inv_session **p_inv)
756{
757 pjsip_inv_session *inv;
Benny Prijonoa66c7152006-02-09 01:26:14 +0000758 struct tsx_inv_data *tsx_inv_data;
Benny Prijono268ca612006-02-07 12:34:11 +0000759 pjsip_msg *msg;
760 pjmedia_sdp_session *rem_sdp = NULL;
761 pj_status_t status;
762
763 /* Verify arguments. */
764 PJ_ASSERT_RETURN(dlg && rdata && p_inv, PJ_EINVAL);
765
766 /* Dialog MUST have been initialised. */
767 PJ_ASSERT_RETURN(pjsip_rdata_get_tsx(rdata) != NULL, PJ_EINVALIDOP);
768
769 msg = rdata->msg_info.msg;
770
771 /* rdata MUST contain INVITE request */
772 PJ_ASSERT_RETURN(msg->type == PJSIP_REQUEST_MSG &&
773 msg->line.req.method.id == PJSIP_INVITE_METHOD,
774 PJ_EINVALIDOP);
775
776 /* Normalize options */
777 if (options & PJSIP_INV_REQUIRE_100REL)
778 options |= PJSIP_INV_SUPPORT_100REL;
779
780 if (options & PJSIP_INV_REQUIRE_TIMER)
781 options |= PJSIP_INV_SUPPORT_TIMER;
782
783 /* Create the session */
784 inv = pj_pool_zalloc(dlg->pool, sizeof(pjsip_inv_session));
785 PJ_ASSERT_RETURN(inv != NULL, PJ_ENOMEM);
786
787 inv->pool = dlg->pool;
788 inv->role = PJSIP_ROLE_UAS;
Benny Prijono38998232006-02-08 22:44:25 +0000789 inv->state = PJSIP_INV_STATE_NULL;
Benny Prijono268ca612006-02-07 12:34:11 +0000790 inv->dlg = dlg;
791 inv->options = options;
Benny Prijonod4e0abd2006-03-05 11:53:36 +0000792 inv->notify = PJ_TRUE;
793 inv->cause = 0;
Benny Prijono268ca612006-02-07 12:34:11 +0000794
Benny Prijonoe4f2abb2006-02-10 14:04:05 +0000795 /* Object name will use the same dialog pointer. */
Benny Prijonoed811d72006-03-10 12:57:12 +0000796 pj_ansi_snprintf(inv->obj_name, PJ_MAX_OBJ_NAME, "inv%p", dlg);
Benny Prijonoe4f2abb2006-02-10 14:04:05 +0000797
Benny Prijono268ca612006-02-07 12:34:11 +0000798 /* Parse SDP in message body, if present. */
799 if (msg->body) {
800 pjsip_msg_body *body = msg->body;
801
802 /* Parse and validate SDP */
803 status = pjmedia_sdp_parse(inv->pool, body->data, body->len,
804 &rem_sdp);
805 if (status == PJ_SUCCESS)
806 status = pjmedia_sdp_validate(rem_sdp);
807
808 if (status != PJ_SUCCESS)
809 return status;
810 }
811
812 /* Create negotiator. */
813 if (rem_sdp) {
814 status = pjmedia_sdp_neg_create_w_remote_offer(inv->pool, local_sdp,
815 rem_sdp, &inv->neg);
816
817 } else if (local_sdp) {
818 status = pjmedia_sdp_neg_create_w_local_offer(inv->pool, local_sdp,
819 &inv->neg);
820 } else {
Benny Prijono95196582006-02-09 00:13:40 +0000821 status = PJ_SUCCESS;
Benny Prijono268ca612006-02-07 12:34:11 +0000822 }
823
824 if (status != PJ_SUCCESS)
825 return status;
826
827 /* Register invite as dialog usage. */
828 status = pjsip_dlg_add_usage(dlg, &mod_inv.mod, inv);
829 if (status != PJ_SUCCESS)
830 return status;
831
832 /* Increment session in the dialog. */
Benny Prijonoe4f2abb2006-02-10 14:04:05 +0000833 pjsip_dlg_inc_session(dlg, &mod_inv.mod);
Benny Prijono268ca612006-02-07 12:34:11 +0000834
835 /* Save the invite transaction. */
836 inv->invite_tsx = pjsip_rdata_get_tsx(rdata);
Benny Prijonoa66c7152006-02-09 01:26:14 +0000837
838 /* Attach our data to the transaction. */
839 tsx_inv_data = pj_pool_zalloc(inv->invite_tsx->pool,
840 sizeof(struct tsx_inv_data));
841 tsx_inv_data->inv = inv;
842 inv->invite_tsx->mod_data[mod_inv.mod.id] = tsx_inv_data;
Benny Prijono268ca612006-02-07 12:34:11 +0000843
844 /* Done */
845 *p_inv = inv;
Benny Prijonoe4f2abb2006-02-10 14:04:05 +0000846
847 PJ_LOG(5,(inv->obj_name, "UAS invite session created for dialog %s",
848 dlg->obj_name));
849
Benny Prijono268ca612006-02-07 12:34:11 +0000850 return PJ_SUCCESS;
851}
852
Benny Prijonod4e0abd2006-03-05 11:53:36 +0000853/*
854 * Forcefully terminate the session.
855 */
856PJ_DEF(pj_status_t) pjsip_inv_terminate( pjsip_inv_session *inv,
857 int st_code,
858 pj_bool_t notify)
859{
860 PJ_ASSERT_RETURN(inv, PJ_EINVAL);
861
862 /* Lock dialog. */
863 pjsip_dlg_inc_lock(inv->dlg);
864
865 /* Set callback notify flag. */
866 inv->notify = notify;
867
868 /* If there's pending transaction, terminate the transaction.
869 * This may subsequently set the INVITE session state to
870 * disconnected.
871 */
872 if (inv->invite_tsx &&
873 inv->invite_tsx->state <= PJSIP_TSX_STATE_COMPLETED)
874 {
875 pjsip_tsx_terminate(inv->invite_tsx, st_code);
876
877 }
878
879 /* Set cause. */
880 inv->cause = st_code;
881
882 /* Forcefully terminate the session if state is not DISCONNECTED */
883 if (inv->state != PJSIP_INV_STATE_DISCONNECTED) {
884 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, NULL);
885 }
886
887 /* Done.
888 * The dec_lock() below will actually destroys the dialog if it
889 * has no other session.
890 */
891 pjsip_dlg_dec_lock(inv->dlg);
892
893 return PJ_SUCCESS;
894}
895
896
Benny Prijono268ca612006-02-07 12:34:11 +0000897static void *clone_sdp(pj_pool_t *pool, const void *data, unsigned len)
898{
899 PJ_UNUSED_ARG(len);
900 return pjmedia_sdp_session_clone(pool, data);
901}
902
903static int print_sdp(pjsip_msg_body *body, char *buf, pj_size_t len)
904{
905 return pjmedia_sdp_print(body->data, buf, len);
906}
907
908static pjsip_msg_body *create_sdp_body(pj_pool_t *pool,
909 const pjmedia_sdp_session *c_sdp)
910{
911 pjsip_msg_body *body;
912
913
914 body = pj_pool_zalloc(pool, sizeof(pjsip_msg_body));
915 PJ_ASSERT_RETURN(body != NULL, NULL);
916
917 body->content_type.type = pj_str("application");
918 body->content_type.subtype = pj_str("sdp");
919 body->data = pjmedia_sdp_session_clone(pool, c_sdp);
920 body->len = 0;
921 body->clone_data = &clone_sdp;
922 body->print_body = &print_sdp;
923
924 return body;
925}
926
927/*
928 * Create initial INVITE request.
929 */
930PJ_DEF(pj_status_t) pjsip_inv_invite( pjsip_inv_session *inv,
931 pjsip_tx_data **p_tdata )
932{
933 pjsip_tx_data *tdata;
934 const pjsip_hdr *hdr;
Benny Prijono26ff9062006-02-21 23:47:00 +0000935 pj_bool_t has_sdp;
Benny Prijono268ca612006-02-07 12:34:11 +0000936 pj_status_t status;
937
938 /* Verify arguments. */
939 PJ_ASSERT_RETURN(inv && p_tdata, PJ_EINVAL);
940
Benny Prijono26ff9062006-02-21 23:47:00 +0000941 /* State MUST be NULL or CONFIRMED. */
942 PJ_ASSERT_RETURN(inv->state == PJSIP_INV_STATE_NULL ||
943 inv->state == PJSIP_INV_STATE_CONFIRMED,
944 PJ_EINVALIDOP);
Benny Prijono268ca612006-02-07 12:34:11 +0000945
Benny Prijono64f851e2006-02-23 13:49:28 +0000946 /* Lock dialog. */
947 pjsip_dlg_inc_lock(inv->dlg);
948
Benny Prijono268ca612006-02-07 12:34:11 +0000949 /* Create the INVITE request. */
950 status = pjsip_dlg_create_request(inv->dlg, &pjsip_invite_method, -1,
951 &tdata);
952 if (status != PJ_SUCCESS)
Benny Prijono64f851e2006-02-23 13:49:28 +0000953 goto on_return;
954
Benny Prijono268ca612006-02-07 12:34:11 +0000955
Benny Prijono26ff9062006-02-21 23:47:00 +0000956 /* If this is the first INVITE, then copy the headers from inv_hdr.
957 * These are the headers parsed from the request URI when the
958 * dialog was created.
959 */
960 if (inv->state == PJSIP_INV_STATE_NULL) {
961 hdr = inv->dlg->inv_hdr.next;
962
963 while (hdr != &inv->dlg->inv_hdr) {
964 pjsip_msg_add_hdr(tdata->msg,
965 pjsip_hdr_shallow_clone(tdata->pool, hdr));
966 hdr = hdr->next;
967 }
968 }
969
970 /* See if we have SDP to send. */
971 if (inv->neg) {
972 pjmedia_sdp_neg_state neg_state;
973
974 neg_state = pjmedia_sdp_neg_get_state(inv->neg);
975
976 has_sdp = (neg_state == PJMEDIA_SDP_NEG_STATE_LOCAL_OFFER ||
977 (neg_state == PJMEDIA_SDP_NEG_STATE_WAIT_NEGO &&
978 pjmedia_sdp_neg_has_local_answer(inv->neg)));
979
980
981 } else {
982 has_sdp = PJ_FALSE;
983 }
984
Benny Prijono268ca612006-02-07 12:34:11 +0000985 /* Add SDP, if any. */
Benny Prijono26ff9062006-02-21 23:47:00 +0000986 if (has_sdp) {
Benny Prijono268ca612006-02-07 12:34:11 +0000987 const pjmedia_sdp_session *offer;
988
989 status = pjmedia_sdp_neg_get_neg_local(inv->neg, &offer);
990 if (status != PJ_SUCCESS)
Benny Prijono64f851e2006-02-23 13:49:28 +0000991 goto on_return;
Benny Prijono268ca612006-02-07 12:34:11 +0000992
993 tdata->msg->body = create_sdp_body(tdata->pool, offer);
994 }
995
996 /* Add Allow header. */
997 hdr = pjsip_endpt_get_capability(inv->dlg->endpt, PJSIP_H_ALLOW, NULL);
998 if (hdr) {
999 pjsip_msg_add_hdr(tdata->msg,
1000 pjsip_hdr_shallow_clone(tdata->pool, hdr));
1001 }
1002
1003 /* Add Supported header */
1004 hdr = pjsip_endpt_get_capability(inv->dlg->endpt, PJSIP_H_SUPPORTED, NULL);
1005 if (hdr) {
1006 pjsip_msg_add_hdr(tdata->msg,
1007 pjsip_hdr_shallow_clone(tdata->pool, hdr));
1008 }
1009
1010 /* Add Require header. */
1011 PJ_TODO(INVITE_ADD_REQUIRE_HEADER);
1012
1013 /* Done. */
1014 *p_tdata = tdata;
1015
Benny Prijono64f851e2006-02-23 13:49:28 +00001016
1017on_return:
1018 pjsip_dlg_dec_lock(inv->dlg);
1019 return status;
Benny Prijono268ca612006-02-07 12:34:11 +00001020}
1021
1022
1023/*
Benny Prijono95196582006-02-09 00:13:40 +00001024 * Negotiate SDP.
1025 */
1026static pj_status_t inv_negotiate_sdp( pjsip_inv_session *inv )
1027{
1028 pj_status_t status;
1029
1030 PJ_ASSERT_RETURN(pjmedia_sdp_neg_get_state(inv->neg) ==
1031 PJMEDIA_SDP_NEG_STATE_WAIT_NEGO,
1032 PJMEDIA_SDPNEG_EINSTATE);
1033
1034 status = pjmedia_sdp_neg_negotiate(inv->pool, inv->neg, 0);
1035
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00001036 PJ_LOG(5,(inv->obj_name, "SDP negotiation done, status=%d", status));
1037
Benny Prijonod4e0abd2006-03-05 11:53:36 +00001038 if (mod_inv.cb.on_media_update && inv->notify)
Benny Prijono95196582006-02-09 00:13:40 +00001039 (*mod_inv.cb.on_media_update)(inv, status);
1040
1041 return status;
1042}
1043
1044/*
Benny Prijonoa66c7152006-02-09 01:26:14 +00001045 * Check in incoming message for SDP offer/answer.
1046 */
Benny Prijono26ff9062006-02-21 23:47:00 +00001047static pj_status_t inv_check_sdp_in_incoming_msg( pjsip_inv_session *inv,
1048 pjsip_transaction *tsx,
1049 pjsip_rx_data *rdata)
Benny Prijonoa66c7152006-02-09 01:26:14 +00001050{
1051 struct tsx_inv_data *tsx_inv_data;
1052 static const pj_str_t str_application = { "application", 11 };
1053 static const pj_str_t str_sdp = { "sdp", 3 };
1054 pj_status_t status;
1055 pjsip_msg *msg;
1056 pjmedia_sdp_session *sdp;
1057
1058 /* Get/attach invite session's transaction data */
1059 tsx_inv_data = tsx->mod_data[mod_inv.mod.id];
1060 if (tsx_inv_data == NULL) {
1061 tsx_inv_data = pj_pool_zalloc(tsx->pool, sizeof(struct tsx_inv_data));
1062 tsx_inv_data->inv = inv;
1063 tsx->mod_data[mod_inv.mod.id] = tsx_inv_data;
1064 }
1065
1066 /* MUST NOT do multiple SDP offer/answer in a single transaction.
1067 */
1068
1069 if (tsx_inv_data->sdp_done)
Benny Prijono26ff9062006-02-21 23:47:00 +00001070 return PJ_SUCCESS;
Benny Prijonoa66c7152006-02-09 01:26:14 +00001071
1072 /* Check if SDP is present in the message. */
1073
1074 msg = rdata->msg_info.msg;
1075 if (msg->body == NULL) {
1076 /* Message doesn't have body. */
Benny Prijono26ff9062006-02-21 23:47:00 +00001077 return PJ_SUCCESS;
Benny Prijonoa66c7152006-02-09 01:26:14 +00001078 }
1079
1080 if (pj_stricmp(&msg->body->content_type.type, &str_application) ||
1081 pj_stricmp(&msg->body->content_type.subtype, &str_sdp))
1082 {
1083 /* Message body is not "application/sdp" */
Benny Prijono26ff9062006-02-21 23:47:00 +00001084 return PJMEDIA_SDP_EINSDP;
Benny Prijonoa66c7152006-02-09 01:26:14 +00001085 }
1086
1087 /* Parse the SDP body. */
1088
1089 status = pjmedia_sdp_parse(rdata->tp_info.pool, msg->body->data,
1090 msg->body->len, &sdp);
1091 if (status != PJ_SUCCESS) {
1092 char errmsg[PJ_ERR_MSG_SIZE];
1093 pj_strerror(status, errmsg, sizeof(errmsg));
1094 PJ_LOG(4,(THIS_FILE, "Error parsing SDP in %s: %s",
1095 pjsip_rx_data_get_info(rdata), errmsg));
Benny Prijono26ff9062006-02-21 23:47:00 +00001096 return PJMEDIA_SDP_EINSDP;
Benny Prijonoa66c7152006-02-09 01:26:14 +00001097 }
1098
1099 /* The SDP can be an offer or answer, depending on negotiator's state */
1100
1101 if (inv->neg == NULL ||
1102 pjmedia_sdp_neg_get_state(inv->neg) == PJMEDIA_SDP_NEG_STATE_DONE)
1103 {
1104
1105 /* This is an offer. */
1106
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00001107 PJ_LOG(5,(inv->obj_name, "Got SDP offer in %s",
1108 pjsip_rx_data_get_info(rdata)));
1109
Benny Prijonoa66c7152006-02-09 01:26:14 +00001110 if (inv->neg == NULL) {
1111 status=pjmedia_sdp_neg_create_w_remote_offer(inv->pool, NULL,
1112 sdp, &inv->neg);
1113 } else {
1114 status=pjmedia_sdp_neg_set_remote_offer(inv->pool, inv->neg, sdp);
1115 }
1116
1117 if (status != PJ_SUCCESS) {
1118 char errmsg[PJ_ERR_MSG_SIZE];
1119 pj_strerror(status, errmsg, sizeof(errmsg));
1120 PJ_LOG(4,(THIS_FILE, "Error processing SDP offer in %s: %s",
1121 pjsip_rx_data_get_info(rdata), errmsg));
Benny Prijono26ff9062006-02-21 23:47:00 +00001122 return PJMEDIA_SDP_EINSDP;
Benny Prijonoa66c7152006-02-09 01:26:14 +00001123 }
1124
1125 /* Inform application about remote offer. */
1126
Benny Prijonod4e0abd2006-03-05 11:53:36 +00001127 if (mod_inv.cb.on_rx_offer && inv->notify) {
Benny Prijono26ff9062006-02-21 23:47:00 +00001128
1129 (*mod_inv.cb.on_rx_offer)(inv, sdp);
1130
1131 }
Benny Prijonoa66c7152006-02-09 01:26:14 +00001132
1133 } else if (pjmedia_sdp_neg_get_state(inv->neg) ==
1134 PJMEDIA_SDP_NEG_STATE_LOCAL_OFFER)
1135 {
1136
1137 /* This is an answer.
1138 * Process and negotiate remote answer.
1139 */
1140
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00001141 PJ_LOG(5,(inv->obj_name, "Got SDP answer in %s",
1142 pjsip_rx_data_get_info(rdata)));
1143
Benny Prijonoa66c7152006-02-09 01:26:14 +00001144 status = pjmedia_sdp_neg_set_remote_answer(inv->pool, inv->neg, sdp);
1145
1146 if (status != PJ_SUCCESS) {
1147 char errmsg[PJ_ERR_MSG_SIZE];
1148 pj_strerror(status, errmsg, sizeof(errmsg));
1149 PJ_LOG(4,(THIS_FILE, "Error processing SDP answer in %s: %s",
1150 pjsip_rx_data_get_info(rdata), errmsg));
Benny Prijono26ff9062006-02-21 23:47:00 +00001151 return PJMEDIA_SDP_EINSDP;
Benny Prijonoa66c7152006-02-09 01:26:14 +00001152 }
1153
1154 /* Negotiate SDP */
1155
1156 inv_negotiate_sdp(inv);
1157
1158 /* Mark this transaction has having SDP offer/answer done. */
1159
1160 tsx_inv_data->sdp_done = 1;
1161
1162 } else {
1163
1164 PJ_LOG(5,(THIS_FILE, "Ignored SDP in %s: negotiator state is %s",
1165 pjsip_rx_data_get_info(rdata),
1166 pjmedia_sdp_neg_state_str(pjmedia_sdp_neg_get_state(inv->neg))));
1167 }
1168
Benny Prijono26ff9062006-02-21 23:47:00 +00001169 return PJ_SUCCESS;
Benny Prijonoa66c7152006-02-09 01:26:14 +00001170}
1171
1172
Benny Prijono26ff9062006-02-21 23:47:00 +00001173/*
1174 * Process INVITE answer, for both initial and subsequent re-INVITE
1175 */
1176static pj_status_t process_answer( pjsip_inv_session *inv,
1177 int st_code,
Benny Prijono64f851e2006-02-23 13:49:28 +00001178 pjsip_tx_data *tdata,
1179 const pjmedia_sdp_session *local_sdp)
Benny Prijono26ff9062006-02-21 23:47:00 +00001180{
1181 pj_status_t status;
Benny Prijonoab7399b2006-02-27 00:40:31 +00001182 const pjmedia_sdp_session *sdp = NULL;
Benny Prijono26ff9062006-02-21 23:47:00 +00001183
Benny Prijono64f851e2006-02-23 13:49:28 +00001184 /* If local_sdp is specified, then we MUST NOT have answered the
1185 * offer before.
Benny Prijono26ff9062006-02-21 23:47:00 +00001186 */
Benny Prijono64f851e2006-02-23 13:49:28 +00001187 if (local_sdp && (st_code/100==1 || st_code/100==2)) {
1188
1189 if (inv->neg == NULL) {
1190 status = pjmedia_sdp_neg_create_w_local_offer(inv->pool, local_sdp,
1191 &inv->neg);
1192 } else if (pjmedia_sdp_neg_get_state(inv->neg)==
1193 PJMEDIA_SDP_NEG_STATE_REMOTE_OFFER)
1194 {
1195 status = pjmedia_sdp_neg_set_local_answer(inv->pool, inv->neg,
1196 local_sdp);
1197 } else {
1198
1199 /* Can not specify local SDP at this state. */
1200 pj_assert(0);
1201 status = PJMEDIA_SDPNEG_EINSTATE;
1202 }
1203
1204 if (status != PJ_SUCCESS)
1205 return status;
1206
1207 }
1208
1209
1210 /* If SDP negotiator is ready, start negotiation. */
1211 if (st_code/100==2 || (st_code/10==18 && st_code!=180)) {
Benny Prijono26ff9062006-02-21 23:47:00 +00001212
1213 pjmedia_sdp_neg_state neg_state;
1214
Benny Prijono64f851e2006-02-23 13:49:28 +00001215 /* Start nego when appropriate. */
Benny Prijono26ff9062006-02-21 23:47:00 +00001216 neg_state = inv->neg ? pjmedia_sdp_neg_get_state(inv->neg) :
1217 PJMEDIA_SDP_NEG_STATE_NULL;
1218
1219 if (neg_state == PJMEDIA_SDP_NEG_STATE_LOCAL_OFFER) {
1220
1221 status = pjmedia_sdp_neg_get_neg_local(inv->neg, &sdp);
1222
1223 } else if (neg_state == PJMEDIA_SDP_NEG_STATE_WAIT_NEGO &&
1224 pjmedia_sdp_neg_has_local_answer(inv->neg) )
1225 {
1226
1227 status = inv_negotiate_sdp(inv);
1228 if (status != PJ_SUCCESS)
1229 return status;
1230
1231 status = pjmedia_sdp_neg_get_active_local(inv->neg, &sdp);
1232 }
Benny Prijono26ff9062006-02-21 23:47:00 +00001233 }
1234
Benny Prijono64f851e2006-02-23 13:49:28 +00001235 /* Include SDP when it's available for 2xx and 18x (but not 180) response.
Benny Prijono26ff9062006-02-21 23:47:00 +00001236 * Subsequent response will include this SDP.
1237 */
1238 if (sdp) {
1239 tdata->msg->body = create_sdp_body(tdata->pool, sdp);
1240 }
1241
Benny Prijono26ff9062006-02-21 23:47:00 +00001242
1243 return PJ_SUCCESS;
1244}
1245
Benny Prijonoa66c7152006-02-09 01:26:14 +00001246
1247/*
Benny Prijono64f851e2006-02-23 13:49:28 +00001248 * Create first response to INVITE
1249 */
1250PJ_DEF(pj_status_t) pjsip_inv_initial_answer( pjsip_inv_session *inv,
1251 pjsip_rx_data *rdata,
1252 int st_code,
1253 const pj_str_t *st_text,
1254 const pjmedia_sdp_session *sdp,
1255 pjsip_tx_data **p_tdata)
1256{
1257 pjsip_tx_data *tdata;
1258 pj_status_t status;
1259
1260 /* Verify arguments. */
1261 PJ_ASSERT_RETURN(inv && p_tdata, PJ_EINVAL);
1262
1263 /* Must have INVITE transaction. */
1264 PJ_ASSERT_RETURN(inv->invite_tsx, PJ_EBUG);
1265
1266 pjsip_dlg_inc_lock(inv->dlg);
1267
1268 /* Create response */
1269 status = pjsip_dlg_create_response(inv->dlg, rdata, st_code, st_text,
1270 &tdata);
1271 if (status != PJ_SUCCESS)
1272 goto on_return;
1273
1274 /* Process SDP in answer */
1275 status = process_answer(inv, st_code, tdata, sdp);
1276 if (status != PJ_SUCCESS) {
1277 pjsip_tx_data_dec_ref(tdata);
1278 goto on_return;
1279 }
1280
1281 *p_tdata = tdata;
1282
1283on_return:
1284 pjsip_dlg_dec_lock(inv->dlg);
1285 return status;
1286}
1287
1288
1289/*
Benny Prijono26ff9062006-02-21 23:47:00 +00001290 * Answer initial INVITE
1291 * Re-INVITE will be answered automatically, and will not use this function.
Benny Prijono268ca612006-02-07 12:34:11 +00001292 */
1293PJ_DEF(pj_status_t) pjsip_inv_answer( pjsip_inv_session *inv,
1294 int st_code,
1295 const pj_str_t *st_text,
1296 const pjmedia_sdp_session *local_sdp,
1297 pjsip_tx_data **p_tdata )
1298{
1299 pjsip_tx_data *last_res;
1300 pj_status_t status;
1301
1302 /* Verify arguments. */
1303 PJ_ASSERT_RETURN(inv && p_tdata, PJ_EINVAL);
1304
1305 /* Must have INVITE transaction. */
1306 PJ_ASSERT_RETURN(inv->invite_tsx, PJ_EBUG);
1307
1308 /* INVITE transaction MUST have transmitted a response (e.g. 100) */
1309 PJ_ASSERT_RETURN(inv->invite_tsx->last_tx, PJ_EINVALIDOP);
1310
Benny Prijono64f851e2006-02-23 13:49:28 +00001311 pjsip_dlg_inc_lock(inv->dlg);
Benny Prijono268ca612006-02-07 12:34:11 +00001312
1313 /* Modify last response. */
Benny Prijono26ff9062006-02-21 23:47:00 +00001314 last_res = inv->invite_tsx->last_tx;
Benny Prijono268ca612006-02-07 12:34:11 +00001315 status = pjsip_dlg_modify_response(inv->dlg, last_res, st_code, st_text);
1316 if (status != PJ_SUCCESS)
Benny Prijono64f851e2006-02-23 13:49:28 +00001317 goto on_return;
Benny Prijono268ca612006-02-07 12:34:11 +00001318
Benny Prijono268ca612006-02-07 12:34:11 +00001319
Benny Prijono26ff9062006-02-21 23:47:00 +00001320 /* Process SDP in answer */
Benny Prijono64f851e2006-02-23 13:49:28 +00001321 status = process_answer(inv, st_code, last_res, local_sdp);
1322 if (status != PJ_SUCCESS) {
1323 pjsip_tx_data_dec_ref(last_res);
1324 goto on_return;
1325 }
Benny Prijono268ca612006-02-07 12:34:11 +00001326
Benny Prijono268ca612006-02-07 12:34:11 +00001327
1328 *p_tdata = last_res;
1329
Benny Prijono64f851e2006-02-23 13:49:28 +00001330on_return:
1331 pjsip_dlg_dec_lock(inv->dlg);
1332 return status;
Benny Prijono268ca612006-02-07 12:34:11 +00001333}
1334
1335
1336/*
Benny Prijono26ff9062006-02-21 23:47:00 +00001337 * Set SDP answer.
1338 */
1339PJ_DEF(pj_status_t) pjsip_inv_set_sdp_answer( pjsip_inv_session *inv,
1340 const pjmedia_sdp_session *sdp )
1341{
1342 pj_status_t status;
1343
1344 PJ_ASSERT_RETURN(inv && sdp, PJ_EINVAL);
1345
1346 pjsip_dlg_inc_lock(inv->dlg);
1347 status = pjmedia_sdp_neg_set_local_answer( inv->pool, inv->neg, sdp);
1348 pjsip_dlg_dec_lock(inv->dlg);
1349
1350 return status;
1351}
1352
1353
1354/*
Benny Prijono268ca612006-02-07 12:34:11 +00001355 * End session.
1356 */
1357PJ_DEF(pj_status_t) pjsip_inv_end_session( pjsip_inv_session *inv,
1358 int st_code,
1359 const pj_str_t *st_text,
1360 pjsip_tx_data **p_tdata )
1361{
1362 pjsip_tx_data *tdata;
1363 pj_status_t status;
1364
1365 /* Verify arguments. */
1366 PJ_ASSERT_RETURN(inv && p_tdata, PJ_EINVAL);
1367
Benny Prijonod4e0abd2006-03-05 11:53:36 +00001368 /* Set cause code. */
1369 if (inv->cause==0) inv->cause = st_code;
1370
Benny Prijono268ca612006-02-07 12:34:11 +00001371 /* Create appropriate message. */
1372 switch (inv->state) {
1373 case PJSIP_INV_STATE_CALLING:
1374 case PJSIP_INV_STATE_EARLY:
1375 case PJSIP_INV_STATE_INCOMING:
1376
1377 if (inv->role == PJSIP_ROLE_UAC) {
1378
1379 /* For UAC when session has not been confirmed, create CANCEL. */
1380
1381 /* MUST have the original UAC INVITE transaction. */
1382 PJ_ASSERT_RETURN(inv->invite_tsx != NULL, PJ_EBUG);
1383
1384 /* But CANCEL should only be called when we have received a
1385 * provisional response. If we haven't received any responses,
1386 * just destroy the transaction.
1387 */
1388 if (inv->invite_tsx->status_code < 100) {
1389
1390 pjsip_tsx_terminate(inv->invite_tsx, 487);
Benny Prijonofccab712006-02-22 22:23:22 +00001391 *p_tdata = NULL;
1392 return PJ_SUCCESS;
Benny Prijono268ca612006-02-07 12:34:11 +00001393 }
1394
1395 /* The CSeq here assumes that the dialog is started with an
1396 * INVITE session. This may not be correct; dialog can be
1397 * started as SUBSCRIBE session.
1398 * So fix this!
1399 */
1400 status = pjsip_endpt_create_cancel(inv->dlg->endpt,
1401 inv->invite_tsx->last_tx,
1402 &tdata);
1403
1404 } else {
1405
1406 /* For UAS, send a final response. */
1407 tdata = inv->invite_tsx->last_tx;
1408 PJ_ASSERT_RETURN(tdata != NULL, PJ_EINVALIDOP);
1409
Benny Prijono26ff9062006-02-21 23:47:00 +00001410 //status = pjsip_dlg_modify_response(inv->dlg, tdata, st_code,
1411 // st_text);
1412 status = pjsip_inv_answer(inv, st_code, st_text, NULL, &tdata);
Benny Prijono268ca612006-02-07 12:34:11 +00001413 }
1414 break;
1415
1416 case PJSIP_INV_STATE_CONNECTING:
1417 case PJSIP_INV_STATE_CONFIRMED:
1418 /* For established dialog, send BYE */
1419 status = pjsip_dlg_create_request(inv->dlg, &pjsip_bye_method, -1,
1420 &tdata);
1421 break;
1422
1423 case PJSIP_INV_STATE_DISCONNECTED:
Benny Prijono268ca612006-02-07 12:34:11 +00001424 /* No need to do anything. */
Benny Prijonod4e0abd2006-03-05 11:53:36 +00001425 return PJSIP_ESESSIONTERMINATED;
Benny Prijono268ca612006-02-07 12:34:11 +00001426
1427 default:
1428 pj_assert("!Invalid operation!");
1429 return PJ_EINVALIDOP;
1430 }
1431
1432 if (status != PJ_SUCCESS)
1433 return status;
1434
1435
1436 /* Done */
1437
1438 *p_tdata = tdata;
1439
1440 return PJ_SUCCESS;
1441}
1442
1443
1444/*
1445 * Create re-INVITE.
1446 */
1447PJ_DEF(pj_status_t) pjsip_inv_reinvite( pjsip_inv_session *inv,
1448 const pj_str_t *new_contact,
1449 const pjmedia_sdp_session *new_offer,
1450 pjsip_tx_data **p_tdata )
1451{
Benny Prijono26ff9062006-02-21 23:47:00 +00001452 pj_status_t status;
1453 pjsip_contact_hdr *contact_hdr = NULL;
Benny Prijono268ca612006-02-07 12:34:11 +00001454
Benny Prijono26ff9062006-02-21 23:47:00 +00001455 /* Check arguments. */
1456 PJ_ASSERT_RETURN(inv && p_tdata, PJ_EINVAL);
1457
1458 /* Must NOT have a pending INVITE transaction */
1459 PJ_ASSERT_RETURN(inv->invite_tsx==NULL, PJ_EINVALIDOP);
1460
1461
1462 pjsip_dlg_inc_lock(inv->dlg);
1463
1464 if (new_contact) {
1465 pj_str_t tmp;
1466 const pj_str_t STR_CONTACT = { "Contact", 7 };
1467
1468 pj_strdup_with_null(inv->dlg->pool, &tmp, new_contact);
1469 contact_hdr = pjsip_parse_hdr(inv->dlg->pool, &STR_CONTACT,
1470 tmp.ptr, tmp.slen, NULL);
1471 if (!contact_hdr) {
1472 status = PJSIP_EINVALIDURI;
1473 goto on_return;
1474 }
1475 }
1476
1477
1478 if (new_offer) {
1479 if (!inv->neg) {
1480 status = pjmedia_sdp_neg_create_w_local_offer(inv->pool, new_offer,
1481 &inv->neg);
1482 if (status != PJ_SUCCESS)
1483 goto on_return;
1484
1485 } else switch (pjmedia_sdp_neg_get_state(inv->neg)) {
1486
1487 case PJMEDIA_SDP_NEG_STATE_NULL:
1488 pj_assert(!"Unexpected SDP neg state NULL");
1489 status = PJ_EBUG;
1490 goto on_return;
1491
1492 case PJMEDIA_SDP_NEG_STATE_LOCAL_OFFER:
1493 PJ_LOG(4,(inv->obj_name,
1494 "pjsip_inv_reinvite: already have an offer, new "
1495 "offer is ignored"));
1496 break;
1497
1498 case PJMEDIA_SDP_NEG_STATE_REMOTE_OFFER:
1499 status = pjmedia_sdp_neg_set_local_answer(inv->pool, inv->neg,
1500 new_offer);
1501 if (status != PJ_SUCCESS)
1502 goto on_return;
1503 break;
1504
1505 case PJMEDIA_SDP_NEG_STATE_WAIT_NEGO:
1506 PJ_LOG(4,(inv->obj_name,
1507 "pjsip_inv_reinvite: SDP in WAIT_NEGO state, new "
1508 "offer is ignored"));
1509 break;
1510
1511 case PJMEDIA_SDP_NEG_STATE_DONE:
1512 status = pjmedia_sdp_neg_modify_local_offer(inv->pool,inv->neg,
1513 new_offer);
1514 if (status != PJ_SUCCESS)
1515 goto on_return;
1516 break;
1517 }
1518 }
1519
1520 if (contact_hdr)
1521 inv->dlg->local.contact = contact_hdr;
1522
1523 status = pjsip_inv_invite(inv, p_tdata);
1524
1525on_return:
1526 pjsip_dlg_dec_lock(inv->dlg);
1527 return status;
Benny Prijono268ca612006-02-07 12:34:11 +00001528}
1529
1530/*
1531 * Create UPDATE.
1532 */
1533PJ_DEF(pj_status_t) pjsip_inv_update ( pjsip_inv_session *inv,
1534 const pj_str_t *new_contact,
1535 const pjmedia_sdp_session *new_offer,
1536 pjsip_tx_data **p_tdata )
1537{
1538 PJ_UNUSED_ARG(inv);
1539 PJ_UNUSED_ARG(new_contact);
1540 PJ_UNUSED_ARG(new_offer);
1541 PJ_UNUSED_ARG(p_tdata);
1542
1543 PJ_TODO(CREATE_UPDATE_REQUEST);
1544 return PJ_ENOTSUP;
1545}
1546
1547/*
1548 * Send a request or response message.
1549 */
1550PJ_DEF(pj_status_t) pjsip_inv_send_msg( pjsip_inv_session *inv,
1551 pjsip_tx_data *tdata,
1552 void *token )
1553{
1554 pj_status_t status;
1555
1556 /* Verify arguments. */
1557 PJ_ASSERT_RETURN(inv && tdata, PJ_EINVAL);
1558
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00001559 PJ_LOG(5,(inv->obj_name, "Sending %s",
1560 pjsip_tx_data_get_info(tdata)));
1561
Benny Prijono268ca612006-02-07 12:34:11 +00001562 if (tdata->msg->type == PJSIP_REQUEST_MSG) {
1563 pjsip_transaction *tsx;
Benny Prijonoa66c7152006-02-09 01:26:14 +00001564 struct tsx_inv_data *tsx_inv_data;
Benny Prijono268ca612006-02-07 12:34:11 +00001565
1566 status = pjsip_dlg_send_request(inv->dlg, tdata, &tsx);
1567 if (status != PJ_SUCCESS)
1568 return status;
1569
Benny Prijonoa66c7152006-02-09 01:26:14 +00001570 tsx_inv_data = pj_pool_zalloc(tsx->pool, sizeof(struct tsx_inv_data));
1571 tsx_inv_data->inv = inv;
1572
1573 tsx->mod_data[mod_inv.mod.id] = tsx_inv_data;
Benny Prijono268ca612006-02-07 12:34:11 +00001574 tsx->mod_data[mod_inv.app_user->id] = token;
1575
1576 } else {
1577 pjsip_cseq_hdr *cseq;
1578
1579 /* Can only do this to send response to original INVITE
1580 * request.
1581 */
Benny Prijono64f851e2006-02-23 13:49:28 +00001582 PJ_ASSERT_RETURN((cseq=pjsip_msg_find_hdr(tdata->msg, PJSIP_H_CSEQ, NULL))!=NULL
1583 && (cseq->cseq == inv->invite_tsx->cseq),
Benny Prijono268ca612006-02-07 12:34:11 +00001584 PJ_EINVALIDOP);
1585
1586 status = pjsip_dlg_send_response(inv->dlg, inv->invite_tsx, tdata);
1587 if (status != PJ_SUCCESS)
1588 return status;
1589 }
1590
1591 /* Done (?) */
1592 return PJ_SUCCESS;
1593}
1594
1595
Benny Prijono8ad55352006-02-08 11:16:05 +00001596/*
1597 * Respond to incoming CANCEL request.
1598 */
1599static void inv_respond_incoming_cancel(pjsip_inv_session *inv,
1600 pjsip_transaction *cancel_tsx,
1601 pjsip_rx_data *rdata)
1602{
1603 pjsip_tx_data *tdata;
1604 pjsip_transaction *invite_tsx;
1605 pj_str_t key;
1606 pj_status_t status;
1607
1608 /* See if we have matching INVITE server transaction: */
1609
1610 pjsip_tsx_create_key(rdata->tp_info.pool, &key, PJSIP_ROLE_UAS,
1611 &pjsip_invite_method, rdata);
1612 invite_tsx = pjsip_tsx_layer_find_tsx(&key, PJ_TRUE);
1613
1614 if (invite_tsx == NULL) {
1615
1616 /* Invite transaction not found!
1617 * Respond CANCEL with 491 (RFC 3261 Section 9.2 page 42)
1618 */
1619 status = pjsip_dlg_create_response( inv->dlg, rdata, 200, NULL,
1620 &tdata);
1621
1622 } else {
1623 /* Always answer CANCEL will 200 (OK) regardless of
1624 * the state of the INVITE transaction.
1625 */
1626 status = pjsip_dlg_create_response( inv->dlg, rdata, 200, NULL,
1627 &tdata);
1628 }
1629
1630 /* See if we have created the response successfully. */
1631 if (status != PJ_SUCCESS) return;
1632
1633 /* Send the CANCEL response */
1634 status = pjsip_dlg_send_response(inv->dlg, cancel_tsx, tdata);
1635 if (status != PJ_SUCCESS) return;
1636
1637
1638 /* See if we need to terminate the UAS INVITE transaction
1639 * with 487 (Request Terminated) response.
1640 */
1641 if (invite_tsx && invite_tsx->status_code < 200) {
1642
1643 pj_assert(invite_tsx->last_tx != NULL);
1644
1645 tdata = invite_tsx->last_tx;
1646
1647 status = pjsip_dlg_modify_response(inv->dlg, tdata, 487, NULL);
1648 if (status == PJ_SUCCESS)
1649 pjsip_dlg_send_response(inv->dlg, invite_tsx, tdata);
1650 }
1651
1652 if (invite_tsx)
1653 pj_mutex_unlock(invite_tsx->mutex);
1654}
1655
1656
1657/*
1658 * Respond to incoming BYE request.
1659 */
1660static void inv_respond_incoming_bye( pjsip_inv_session *inv,
1661 pjsip_transaction *bye_tsx,
1662 pjsip_rx_data *rdata,
1663 pjsip_event *e )
1664{
1665 pj_status_t status;
1666 pjsip_tx_data *tdata;
1667
1668 /* Respond BYE with 200: */
1669
1670 status = pjsip_dlg_create_response(inv->dlg, rdata, 200, NULL, &tdata);
1671 if (status != PJ_SUCCESS) return;
1672
1673 status = pjsip_dlg_send_response(inv->dlg, bye_tsx, tdata);
1674 if (status != PJ_SUCCESS) return;
1675
1676 /* Terminate session: */
1677
Benny Prijonod4e0abd2006-03-05 11:53:36 +00001678 if (inv->state != PJSIP_INV_STATE_DISCONNECTED) {
1679 if (inv->cause==0) inv->cause=PJSIP_SC_OK;
Benny Prijono8ad55352006-02-08 11:16:05 +00001680 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
Benny Prijonod4e0abd2006-03-05 11:53:36 +00001681 }
Benny Prijono8ad55352006-02-08 11:16:05 +00001682}
1683
1684/*
Benny Prijono38998232006-02-08 22:44:25 +00001685 * Respond to BYE request.
1686 */
1687static void inv_handle_bye_response( pjsip_inv_session *inv,
1688 pjsip_transaction *tsx,
1689 pjsip_rx_data *rdata,
1690 pjsip_event *e )
1691{
1692 pj_status_t status;
1693
1694 if (e->body.tsx_state.type != PJSIP_EVENT_RX_MSG) {
Benny Prijonod4e0abd2006-03-05 11:53:36 +00001695 if (inv->cause==0) inv->cause=PJSIP_SC_OK;
Benny Prijono38998232006-02-08 22:44:25 +00001696 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
1697 return;
1698 }
1699
1700 /* Handle 401/407 challenge. */
1701 if (tsx->status_code == 401 || tsx->status_code == 407) {
1702
1703 pjsip_tx_data *tdata;
1704
1705 status = pjsip_auth_clt_reinit_req( &inv->dlg->auth_sess,
1706 rdata,
1707 tsx->last_tx,
1708 &tdata);
1709
1710 if (status != PJ_SUCCESS) {
1711
1712 /* Does not have proper credentials.
1713 * End the session anyway.
1714 */
Benny Prijonod4e0abd2006-03-05 11:53:36 +00001715 if (inv->cause==0) inv->cause=PJSIP_SC_OK;
Benny Prijono38998232006-02-08 22:44:25 +00001716 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
1717
1718 } else {
1719 /* Re-send BYE. */
1720 status = pjsip_inv_send_msg(inv, tdata, NULL );
1721 }
1722
1723 } else {
1724
1725 /* End the session. */
Benny Prijonod4e0abd2006-03-05 11:53:36 +00001726 if (inv->cause==0) inv->cause=PJSIP_SC_OK;
Benny Prijono38998232006-02-08 22:44:25 +00001727 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
1728 }
1729
1730}
1731
1732/*
Benny Prijono8ad55352006-02-08 11:16:05 +00001733 * State NULL is before anything is sent/received.
1734 */
1735static void inv_on_state_null( pjsip_inv_session *inv, pjsip_event *e)
Benny Prijono268ca612006-02-07 12:34:11 +00001736{
1737 pjsip_transaction *tsx = e->body.tsx_state.tsx;
1738 pjsip_dialog *dlg = pjsip_tsx_get_dlg(tsx);
1739
1740 PJ_ASSERT_ON_FAIL(tsx && dlg, return);
1741
1742 if (tsx->method.id == PJSIP_INVITE_METHOD) {
1743
Benny Prijono64f851e2006-02-23 13:49:28 +00001744 /* Keep the initial INVITE transaction. */
1745 if (inv->invite_tsx == NULL)
1746 inv->invite_tsx = tsx;
Benny Prijono268ca612006-02-07 12:34:11 +00001747
Benny Prijono64f851e2006-02-23 13:49:28 +00001748 if (dlg->role == PJSIP_ROLE_UAC) {
Benny Prijono268ca612006-02-07 12:34:11 +00001749
1750 switch (tsx->state) {
1751 case PJSIP_TSX_STATE_CALLING:
Benny Prijono8ad55352006-02-08 11:16:05 +00001752 inv_set_state(inv, PJSIP_INV_STATE_CALLING, e);
Benny Prijono268ca612006-02-07 12:34:11 +00001753 break;
1754 default:
Benny Prijono64f851e2006-02-23 13:49:28 +00001755 inv_on_state_calling(inv, e);
Benny Prijono268ca612006-02-07 12:34:11 +00001756 break;
1757 }
1758
1759 } else {
1760 switch (tsx->state) {
1761 case PJSIP_TSX_STATE_TRYING:
Benny Prijono8ad55352006-02-08 11:16:05 +00001762 inv_set_state(inv, PJSIP_INV_STATE_INCOMING, e);
Benny Prijono268ca612006-02-07 12:34:11 +00001763 break;
Benny Prijono38998232006-02-08 22:44:25 +00001764 case PJSIP_TSX_STATE_PROCEEDING:
1765 inv_set_state(inv, PJSIP_INV_STATE_INCOMING, e);
1766 if (tsx->status_code > 100)
1767 inv_set_state(inv, PJSIP_INV_STATE_EARLY, e);
1768 break;
Benny Prijono268ca612006-02-07 12:34:11 +00001769 default:
Benny Prijono64f851e2006-02-23 13:49:28 +00001770 inv_on_state_incoming(inv, e);
1771 break;
Benny Prijono268ca612006-02-07 12:34:11 +00001772 }
1773 }
1774
1775 } else {
1776 pj_assert(!"Unexpected transaction type");
1777 }
1778}
1779
Benny Prijono8ad55352006-02-08 11:16:05 +00001780/*
1781 * State CALLING is after sending initial INVITE request but before
1782 * any response (with tag) is received.
1783 */
1784static void inv_on_state_calling( pjsip_inv_session *inv, pjsip_event *e)
Benny Prijono268ca612006-02-07 12:34:11 +00001785{
1786 pjsip_transaction *tsx = e->body.tsx_state.tsx;
1787 pjsip_dialog *dlg = pjsip_tsx_get_dlg(tsx);
Benny Prijonoccf95622006-02-07 18:48:01 +00001788 pj_status_t status;
Benny Prijono268ca612006-02-07 12:34:11 +00001789
1790 PJ_ASSERT_ON_FAIL(tsx && dlg, return);
1791
Benny Prijono8ad55352006-02-08 11:16:05 +00001792 if (tsx == inv->invite_tsx) {
Benny Prijono268ca612006-02-07 12:34:11 +00001793
1794 switch (tsx->state) {
1795
Benny Prijono64f851e2006-02-23 13:49:28 +00001796 case PJSIP_TSX_STATE_CALLING:
1797 inv_set_state(inv, PJSIP_INV_STATE_CALLING, e);
1798 break;
1799
Benny Prijono268ca612006-02-07 12:34:11 +00001800 case PJSIP_TSX_STATE_PROCEEDING:
1801 if (dlg->remote.info->tag.slen) {
Benny Prijonoa66c7152006-02-09 01:26:14 +00001802
Benny Prijono8ad55352006-02-08 11:16:05 +00001803 inv_set_state(inv, PJSIP_INV_STATE_EARLY, e);
Benny Prijonoa66c7152006-02-09 01:26:14 +00001804
1805 inv_check_sdp_in_incoming_msg(inv, tsx,
1806 e->body.tsx_state.src.rdata);
1807
Benny Prijono268ca612006-02-07 12:34:11 +00001808 } else {
1809 /* Ignore 100 (Trying) response, as it doesn't change
1810 * session state. It only ceases retransmissions.
1811 */
1812 }
1813 break;
1814
1815 case PJSIP_TSX_STATE_COMPLETED:
1816 if (tsx->status_code/100 == 2) {
1817
1818 /* This should not happen.
1819 * When transaction receives 2xx, it should be terminated
1820 */
1821 pj_assert(0);
Benny Prijono8ad55352006-02-08 11:16:05 +00001822 inv_set_state(inv, PJSIP_INV_STATE_CONNECTING, e);
Benny Prijonoa66c7152006-02-09 01:26:14 +00001823
1824 inv_check_sdp_in_incoming_msg(inv, tsx,
1825 e->body.tsx_state.src.rdata);
Benny Prijono268ca612006-02-07 12:34:11 +00001826
Benny Prijonoccf95622006-02-07 18:48:01 +00001827 } else if (tsx->status_code==401 || tsx->status_code==407) {
1828
1829 /* Handle authentication failure:
1830 * Resend the request with Authorization header.
1831 */
1832 pjsip_tx_data *tdata;
1833
Benny Prijono8ad55352006-02-08 11:16:05 +00001834 status = pjsip_auth_clt_reinit_req(&inv->dlg->auth_sess,
Benny Prijonoccf95622006-02-07 18:48:01 +00001835 e->body.tsx_state.src.rdata,
1836 tsx->last_tx,
1837 &tdata);
1838
1839 if (status != PJ_SUCCESS) {
1840
1841 /* Does not have proper credentials.
1842 * End the session.
1843 */
Benny Prijonod4e0abd2006-03-05 11:53:36 +00001844 if (inv->cause==0) inv->cause = tsx->status_code;
Benny Prijono8ad55352006-02-08 11:16:05 +00001845 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
Benny Prijonoccf95622006-02-07 18:48:01 +00001846
1847 } else {
1848
1849 /* Restart session. */
Benny Prijono8ad55352006-02-08 11:16:05 +00001850 inv->state = PJSIP_INV_STATE_NULL;
1851 inv->invite_tsx = NULL;
Benny Prijonoccf95622006-02-07 18:48:01 +00001852
1853 /* Send the request. */
Benny Prijono8ad55352006-02-08 11:16:05 +00001854 status = pjsip_inv_send_msg(inv, tdata, NULL );
Benny Prijonoccf95622006-02-07 18:48:01 +00001855 }
1856
Benny Prijono268ca612006-02-07 12:34:11 +00001857 } else {
Benny Prijonoccf95622006-02-07 18:48:01 +00001858
Benny Prijonod4e0abd2006-03-05 11:53:36 +00001859 if (inv->cause==0) inv->cause = tsx->status_code;
Benny Prijono8ad55352006-02-08 11:16:05 +00001860 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
Benny Prijonoccf95622006-02-07 18:48:01 +00001861
Benny Prijono268ca612006-02-07 12:34:11 +00001862 }
1863 break;
1864
1865 case PJSIP_TSX_STATE_TERMINATED:
1866 /* INVITE transaction can be terminated either because UAC
1867 * transaction received 2xx response or because of transport
1868 * error.
1869 */
1870 if (tsx->status_code/100 == 2) {
1871 /* This must be receipt of 2xx response */
1872
1873 /* Set state to CONNECTING */
Benny Prijono8ad55352006-02-08 11:16:05 +00001874 inv_set_state(inv, PJSIP_INV_STATE_CONNECTING, e);
Benny Prijono268ca612006-02-07 12:34:11 +00001875
Benny Prijonoa66c7152006-02-09 01:26:14 +00001876 inv_check_sdp_in_incoming_msg(inv, tsx,
1877 e->body.tsx_state.src.rdata);
1878
Benny Prijono268ca612006-02-07 12:34:11 +00001879 /* Send ACK */
1880 pj_assert(e->body.tsx_state.type == PJSIP_EVENT_RX_MSG);
1881
Benny Prijono8ad55352006-02-08 11:16:05 +00001882 inv_send_ack(inv, e->body.tsx_state.src.rdata);
Benny Prijono5eff0432006-02-09 14:14:21 +00001883 inv_set_state(inv, PJSIP_INV_STATE_CONFIRMED, e);
Benny Prijono268ca612006-02-07 12:34:11 +00001884
Benny Prijonoa66c7152006-02-09 01:26:14 +00001885
Benny Prijono268ca612006-02-07 12:34:11 +00001886 } else {
Benny Prijonod4e0abd2006-03-05 11:53:36 +00001887 if (inv->cause==0) inv->cause = tsx->status_code;
Benny Prijono8ad55352006-02-08 11:16:05 +00001888 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
Benny Prijono268ca612006-02-07 12:34:11 +00001889 }
1890 break;
1891
Benny Prijono34a404e2006-02-09 14:38:30 +00001892 default:
1893 break;
Benny Prijono268ca612006-02-07 12:34:11 +00001894 }
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00001895
1896 } else if (inv->role == PJSIP_ROLE_UAC &&
1897 tsx->role == PJSIP_ROLE_UAC &&
1898 tsx->method.id == PJSIP_CANCEL_METHOD)
1899 {
1900 /*
1901 * Handle case when outgoing CANCEL is answered with 481 (Call/
1902 * Transaction Does Not Exist), 408, or when it's timed out. In these
1903 * cases, disconnect session (i.e. dialog usage only).
1904 */
1905 if (tsx->status_code == PJSIP_SC_CALL_TSX_DOES_NOT_EXIST ||
1906 tsx->status_code == PJSIP_SC_REQUEST_TIMEOUT ||
1907 tsx->status_code == PJSIP_SC_TSX_TIMEOUT ||
1908 PJSIP_SC_TSX_TRANSPORT_ERROR)
1909 {
Benny Prijonod4e0abd2006-03-05 11:53:36 +00001910 if (inv->cause==0) inv->cause = tsx->status_code;
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00001911 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
1912 }
Benny Prijono268ca612006-02-07 12:34:11 +00001913 }
1914}
1915
Benny Prijono8ad55352006-02-08 11:16:05 +00001916/*
1917 * State INCOMING is after we received the request, but before
1918 * responses with tag are sent.
1919 */
1920static void inv_on_state_incoming( pjsip_inv_session *inv, pjsip_event *e)
Benny Prijono268ca612006-02-07 12:34:11 +00001921{
1922 pjsip_transaction *tsx = e->body.tsx_state.tsx;
1923 pjsip_dialog *dlg = pjsip_tsx_get_dlg(tsx);
1924
1925 PJ_ASSERT_ON_FAIL(tsx && dlg, return);
1926
Benny Prijono8ad55352006-02-08 11:16:05 +00001927 if (tsx == inv->invite_tsx) {
1928
1929 /*
1930 * Handle the INVITE state transition.
1931 */
1932
Benny Prijono268ca612006-02-07 12:34:11 +00001933 switch (tsx->state) {
Benny Prijono8ad55352006-02-08 11:16:05 +00001934
Benny Prijono64f851e2006-02-23 13:49:28 +00001935 case PJSIP_TSX_STATE_TRYING:
1936 inv_set_state(inv, PJSIP_INV_STATE_INCOMING, e);
1937 break;
1938
Benny Prijono268ca612006-02-07 12:34:11 +00001939 case PJSIP_TSX_STATE_PROCEEDING:
Benny Prijono8ad55352006-02-08 11:16:05 +00001940 /*
1941 * Transaction sent provisional response.
1942 */
Benny Prijono268ca612006-02-07 12:34:11 +00001943 if (tsx->status_code > 100)
Benny Prijono8ad55352006-02-08 11:16:05 +00001944 inv_set_state(inv, PJSIP_INV_STATE_EARLY, e);
Benny Prijono268ca612006-02-07 12:34:11 +00001945 break;
Benny Prijono8ad55352006-02-08 11:16:05 +00001946
Benny Prijono268ca612006-02-07 12:34:11 +00001947 case PJSIP_TSX_STATE_COMPLETED:
Benny Prijono8ad55352006-02-08 11:16:05 +00001948 /*
1949 * Transaction sent final response.
1950 */
Benny Prijonod4e0abd2006-03-05 11:53:36 +00001951 if (tsx->status_code/100 == 2) {
Benny Prijono8ad55352006-02-08 11:16:05 +00001952 inv_set_state(inv, PJSIP_INV_STATE_CONNECTING, e);
Benny Prijonod4e0abd2006-03-05 11:53:36 +00001953 } else {
1954 if (inv->cause==0) inv->cause = tsx->status_code;
Benny Prijono8ad55352006-02-08 11:16:05 +00001955 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
Benny Prijonod4e0abd2006-03-05 11:53:36 +00001956 }
Benny Prijono268ca612006-02-07 12:34:11 +00001957 break;
Benny Prijono8ad55352006-02-08 11:16:05 +00001958
Benny Prijono268ca612006-02-07 12:34:11 +00001959 case PJSIP_TSX_STATE_TERMINATED:
Benny Prijono8ad55352006-02-08 11:16:05 +00001960 /*
1961 * This happens on transport error (e.g. failed to send
1962 * response)
1963 */
Benny Prijonod4e0abd2006-03-05 11:53:36 +00001964 if (inv->cause==0) inv->cause = tsx->status_code;
Benny Prijono8ad55352006-02-08 11:16:05 +00001965 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, 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 default:
Benny Prijono8ad55352006-02-08 11:16:05 +00001969 pj_assert(!"Unexpected INVITE state");
1970 break;
Benny Prijono268ca612006-02-07 12:34:11 +00001971 }
Benny Prijono8ad55352006-02-08 11:16:05 +00001972
1973 } else if (tsx->method.id == PJSIP_CANCEL_METHOD &&
1974 tsx->role == PJSIP_ROLE_UAS &&
1975 tsx->state < PJSIP_TSX_STATE_COMPLETED &&
1976 e->body.tsx_state.type == PJSIP_EVENT_RX_MSG )
1977 {
1978
1979 /*
1980 * Handle incoming CANCEL request.
1981 */
1982
1983 inv_respond_incoming_cancel(inv, tsx, e->body.tsx_state.src.rdata);
1984
Benny Prijono268ca612006-02-07 12:34:11 +00001985 }
1986}
1987
Benny Prijono8ad55352006-02-08 11:16:05 +00001988/*
1989 * State EARLY is for both UAS and UAC, after response with To tag
1990 * is sent/received.
1991 */
1992static void inv_on_state_early( pjsip_inv_session *inv, pjsip_event *e)
Benny Prijono268ca612006-02-07 12:34:11 +00001993{
1994 pjsip_transaction *tsx = e->body.tsx_state.tsx;
1995 pjsip_dialog *dlg = pjsip_tsx_get_dlg(tsx);
1996
1997 PJ_ASSERT_ON_FAIL(tsx && dlg, return);
1998
Benny Prijono8ad55352006-02-08 11:16:05 +00001999 if (tsx == inv->invite_tsx) {
2000
2001 /*
2002 * Handle the INVITE state progress.
2003 */
Benny Prijono268ca612006-02-07 12:34:11 +00002004
2005 switch (tsx->state) {
2006
2007 case PJSIP_TSX_STATE_PROCEEDING:
2008 /* Send/received another provisional response. */
Benny Prijono8ad55352006-02-08 11:16:05 +00002009 inv_set_state(inv, PJSIP_INV_STATE_EARLY, e);
Benny Prijonoa66c7152006-02-09 01:26:14 +00002010
2011 if (e->body.tsx_state.type == PJSIP_EVENT_RX_MSG) {
2012 inv_check_sdp_in_incoming_msg(inv, tsx,
2013 e->body.tsx_state.src.rdata);
2014 }
Benny Prijono268ca612006-02-07 12:34:11 +00002015 break;
2016
2017 case PJSIP_TSX_STATE_COMPLETED:
Benny Prijonoa66c7152006-02-09 01:26:14 +00002018 if (tsx->status_code/100 == 2) {
Benny Prijono8ad55352006-02-08 11:16:05 +00002019 inv_set_state(inv, PJSIP_INV_STATE_CONNECTING, e);
Benny Prijonoa66c7152006-02-09 01:26:14 +00002020 if (e->body.tsx_state.type == PJSIP_EVENT_RX_MSG) {
2021 inv_check_sdp_in_incoming_msg(inv, tsx,
2022 e->body.tsx_state.src.rdata);
2023 }
2024
Benny Prijonod4e0abd2006-03-05 11:53:36 +00002025 } else {
2026 if (inv->cause==0) inv->cause = tsx->status_code;
Benny Prijono8ad55352006-02-08 11:16:05 +00002027 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
Benny Prijonod4e0abd2006-03-05 11:53:36 +00002028 }
Benny Prijono268ca612006-02-07 12:34:11 +00002029 break;
2030
Benny Prijonof3195072006-02-14 21:15:30 +00002031 case PJSIP_TSX_STATE_CONFIRMED:
2032 /* For some reason can go here */
2033
Benny Prijono268ca612006-02-07 12:34:11 +00002034 case PJSIP_TSX_STATE_TERMINATED:
2035 /* INVITE transaction can be terminated either because UAC
2036 * transaction received 2xx response or because of transport
2037 * error.
2038 */
2039 if (tsx->status_code/100 == 2) {
2040
2041 /* This must be receipt of 2xx response */
2042
2043 /* Set state to CONNECTING */
Benny Prijono8ad55352006-02-08 11:16:05 +00002044 inv_set_state(inv, PJSIP_INV_STATE_CONNECTING, e);
Benny Prijono268ca612006-02-07 12:34:11 +00002045
Benny Prijonoa66c7152006-02-09 01:26:14 +00002046 if (e->body.tsx_state.type == PJSIP_EVENT_RX_MSG) {
2047 inv_check_sdp_in_incoming_msg(inv, tsx,
2048 e->body.tsx_state.src.rdata);
2049 }
2050
Benny Prijono268ca612006-02-07 12:34:11 +00002051 /* if UAC, send ACK and move state to confirmed. */
2052 if (tsx->role == PJSIP_ROLE_UAC) {
2053 pj_assert(e->body.tsx_state.type == PJSIP_EVENT_RX_MSG);
2054
Benny Prijono8ad55352006-02-08 11:16:05 +00002055 inv_send_ack(inv, e->body.tsx_state.src.rdata);
Benny Prijono8ad55352006-02-08 11:16:05 +00002056 inv_set_state(inv, PJSIP_INV_STATE_CONFIRMED, e);
Benny Prijono268ca612006-02-07 12:34:11 +00002057 }
2058
2059 } else {
Benny Prijonod4e0abd2006-03-05 11:53:36 +00002060 if (inv->cause==0) inv->cause = tsx->status_code;
Benny Prijono8ad55352006-02-08 11:16:05 +00002061 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
Benny Prijono268ca612006-02-07 12:34:11 +00002062 }
2063 break;
2064
2065 default:
Benny Prijono8ad55352006-02-08 11:16:05 +00002066 pj_assert(!"Unexpected INVITE tsx state");
Benny Prijono268ca612006-02-07 12:34:11 +00002067 }
2068
Benny Prijono8ad55352006-02-08 11:16:05 +00002069 } else if (inv->role == PJSIP_ROLE_UAS &&
2070 tsx->role == PJSIP_ROLE_UAS &&
2071 tsx->method.id == PJSIP_CANCEL_METHOD &&
2072 tsx->state < PJSIP_TSX_STATE_COMPLETED &&
2073 e->body.tsx_state.type == PJSIP_EVENT_RX_MSG )
2074 {
Benny Prijono268ca612006-02-07 12:34:11 +00002075
Benny Prijono8ad55352006-02-08 11:16:05 +00002076 /*
2077 * Handle incoming CANCEL request.
Benny Prijonoccf95622006-02-07 18:48:01 +00002078 */
2079
Benny Prijono8ad55352006-02-08 11:16:05 +00002080 inv_respond_incoming_cancel(inv, tsx, e->body.tsx_state.src.rdata);
2081
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00002082 } else if (inv->role == PJSIP_ROLE_UAC &&
2083 tsx->role == PJSIP_ROLE_UAC &&
2084 tsx->method.id == PJSIP_CANCEL_METHOD)
2085 {
2086 /*
2087 * Handle case when outgoing CANCEL is answered with 481 (Call/
2088 * Transaction Does Not Exist), 408, or when it's timed out. In these
2089 * cases, disconnect session (i.e. dialog usage only).
2090 */
2091 if (tsx->status_code == PJSIP_SC_CALL_TSX_DOES_NOT_EXIST ||
2092 tsx->status_code == PJSIP_SC_REQUEST_TIMEOUT ||
2093 tsx->status_code == PJSIP_SC_TSX_TIMEOUT ||
2094 PJSIP_SC_TSX_TRANSPORT_ERROR)
2095 {
Benny Prijonod4e0abd2006-03-05 11:53:36 +00002096 if (inv->cause==0) inv->cause = tsx->status_code;
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00002097 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
2098 }
Benny Prijono268ca612006-02-07 12:34:11 +00002099 }
2100}
2101
Benny Prijono8ad55352006-02-08 11:16:05 +00002102/*
2103 * State CONNECTING is after 2xx response to INVITE is sent/received.
2104 */
2105static void inv_on_state_connecting( pjsip_inv_session *inv, pjsip_event *e)
Benny Prijono268ca612006-02-07 12:34:11 +00002106{
2107 pjsip_transaction *tsx = e->body.tsx_state.tsx;
2108 pjsip_dialog *dlg = pjsip_tsx_get_dlg(tsx);
2109
2110 PJ_ASSERT_ON_FAIL(tsx && dlg, return);
2111
Benny Prijono8ad55352006-02-08 11:16:05 +00002112 if (tsx == inv->invite_tsx) {
Benny Prijono268ca612006-02-07 12:34:11 +00002113
Benny Prijono8ad55352006-02-08 11:16:05 +00002114 /*
2115 * Handle INVITE state progression.
2116 */
Benny Prijono268ca612006-02-07 12:34:11 +00002117 switch (tsx->state) {
2118
2119 case PJSIP_TSX_STATE_CONFIRMED:
Benny Prijono38998232006-02-08 22:44:25 +00002120 if (tsx->status_code/100 == 2)
2121 inv_set_state(inv, PJSIP_INV_STATE_CONFIRMED, e);
Benny Prijono268ca612006-02-07 12:34:11 +00002122 break;
2123
2124 case PJSIP_TSX_STATE_TERMINATED:
2125 /* INVITE transaction can be terminated either because UAC
2126 * transaction received 2xx response or because of transport
2127 * error.
2128 */
2129 if (tsx->status_code/100 != 2) {
Benny Prijonod4e0abd2006-03-05 11:53:36 +00002130 if (inv->cause==0) inv->cause = tsx->status_code;
Benny Prijono8ad55352006-02-08 11:16:05 +00002131 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
Benny Prijono268ca612006-02-07 12:34:11 +00002132 }
2133 break;
2134
2135 case PJSIP_TSX_STATE_DESTROYED:
2136 /* Do nothing. */
2137 break;
2138
2139 default:
2140 pj_assert(!"Unexpected state");
2141 }
2142
Benny Prijono8ad55352006-02-08 11:16:05 +00002143 } else if (tsx->role == PJSIP_ROLE_UAS &&
2144 tsx->method.id == PJSIP_BYE_METHOD &&
2145 tsx->status_code < 200 &&
2146 e->body.tsx_state.type == PJSIP_EVENT_RX_MSG)
2147 {
2148
2149 /*
2150 * Handle incoming BYE.
2151 */
2152
2153 inv_respond_incoming_bye( inv, tsx, e->body.tsx_state.src.rdata, e );
2154
Benny Prijono38998232006-02-08 22:44:25 +00002155 } else if (tsx->method.id == PJSIP_BYE_METHOD &&
2156 tsx->role == PJSIP_ROLE_UAC &&
Benny Prijonoa66c7152006-02-09 01:26:14 +00002157 (tsx->state == PJSIP_TSX_STATE_COMPLETED ||
2158 tsx->state == PJSIP_TSX_STATE_TERMINATED))
Benny Prijono38998232006-02-08 22:44:25 +00002159 {
2160
2161 /*
2162 * Outgoing BYE
2163 */
2164 inv_handle_bye_response( inv, tsx, e->body.tsx_state.src.rdata, e);
2165
Benny Prijono268ca612006-02-07 12:34:11 +00002166 }
Benny Prijono38998232006-02-08 22:44:25 +00002167
Benny Prijono268ca612006-02-07 12:34:11 +00002168}
2169
Benny Prijono8ad55352006-02-08 11:16:05 +00002170/*
2171 * State CONFIRMED is after ACK is sent/received.
2172 */
2173static void inv_on_state_confirmed( pjsip_inv_session *inv, pjsip_event *e)
Benny Prijono268ca612006-02-07 12:34:11 +00002174{
2175 pjsip_transaction *tsx = e->body.tsx_state.tsx;
2176 pjsip_dialog *dlg = pjsip_tsx_get_dlg(tsx);
2177
2178 PJ_ASSERT_ON_FAIL(tsx && dlg, return);
2179
Benny Prijono268ca612006-02-07 12:34:11 +00002180
Benny Prijono8ad55352006-02-08 11:16:05 +00002181 if (tsx->method.id == PJSIP_BYE_METHOD &&
2182 tsx->role == PJSIP_ROLE_UAC &&
Benny Prijonoa66c7152006-02-09 01:26:14 +00002183 (tsx->state == PJSIP_TSX_STATE_COMPLETED ||
2184 tsx->state == PJSIP_TSX_STATE_TERMINATED))
Benny Prijono8ad55352006-02-08 11:16:05 +00002185 {
Benny Prijono38998232006-02-08 22:44:25 +00002186
Benny Prijono8ad55352006-02-08 11:16:05 +00002187 /*
Benny Prijono38998232006-02-08 22:44:25 +00002188 * Outgoing BYE
Benny Prijono8ad55352006-02-08 11:16:05 +00002189 */
Benny Prijono8ad55352006-02-08 11:16:05 +00002190
Benny Prijonoa66c7152006-02-09 01:26:14 +00002191 inv_handle_bye_response( inv, tsx, e->body.tsx_state.src.rdata, e);
Benny Prijono268ca612006-02-07 12:34:11 +00002192
Benny Prijono8ad55352006-02-08 11:16:05 +00002193 }
2194 else if (tsx->method.id == PJSIP_BYE_METHOD &&
2195 tsx->role == PJSIP_ROLE_UAS &&
2196 tsx->status_code < 200 &&
2197 e->body.tsx_state.type == PJSIP_EVENT_RX_MSG)
2198 {
Benny Prijonoccf95622006-02-07 18:48:01 +00002199
Benny Prijono8ad55352006-02-08 11:16:05 +00002200 /*
2201 * Handle incoming BYE.
2202 */
Benny Prijono268ca612006-02-07 12:34:11 +00002203
Benny Prijono8ad55352006-02-08 11:16:05 +00002204 inv_respond_incoming_bye( inv, tsx, e->body.tsx_state.src.rdata, e );
2205
Benny Prijono268ca612006-02-07 12:34:11 +00002206 }
Benny Prijono26ff9062006-02-21 23:47:00 +00002207 else if (tsx->method.id == PJSIP_INVITE_METHOD &&
2208 tsx->role == PJSIP_ROLE_UAS)
2209 {
2210
2211 /*
2212 * Handle incoming re-INVITE
2213 */
2214 if (tsx->state == PJSIP_TSX_STATE_TRYING) {
2215
2216 pjsip_rx_data *rdata = e->body.tsx_state.src.rdata;
2217 pjsip_tx_data *tdata;
2218 pj_status_t status;
2219
2220 /* Check if we have INVITE pending. */
2221 if (inv->invite_tsx && inv->invite_tsx!=tsx) {
2222
2223 /* Can not receive re-INVITE while another one is pending. */
2224 status = pjsip_dlg_create_response( inv->dlg, rdata, 500, NULL,
2225 &tdata);
2226 if (status != PJ_SUCCESS)
2227 return;
2228
2229 status = pjsip_dlg_send_response( inv->dlg, tsx, tdata);
2230
2231
2232 return;
2233 }
2234
2235 /* Save the invite transaction. */
2236 inv->invite_tsx = tsx;
2237
2238 /* Process SDP in incoming message. */
2239 status = inv_check_sdp_in_incoming_msg(inv, tsx, rdata);
2240
2241 if (status != PJ_SUCCESS) {
2242
2243 /* Not Acceptable */
2244 const pjsip_hdr *accept;
2245
2246 status = pjsip_dlg_create_response(inv->dlg, rdata,
2247 488, NULL, &tdata);
2248 if (status != PJ_SUCCESS)
2249 return;
2250
2251
2252 accept = pjsip_endpt_get_capability(dlg->endpt, PJSIP_H_ACCEPT,
2253 NULL);
2254 if (accept) {
2255 pjsip_msg_add_hdr(tdata->msg,
2256 pjsip_hdr_clone(tdata->pool, accept));
2257 }
2258
2259 status = pjsip_dlg_send_response(dlg, tsx, tdata);
2260
2261 return;
2262 }
2263
2264 /* Create 2xx ANSWER */
2265 status = pjsip_dlg_create_response(dlg, rdata, 200, NULL, &tdata);
2266 if (status != PJ_SUCCESS)
2267 return;
2268
2269 /* Process SDP in the answer */
Benny Prijono64f851e2006-02-23 13:49:28 +00002270 status = process_answer(inv, 200, tdata, NULL);
Benny Prijono26ff9062006-02-21 23:47:00 +00002271 if (status != PJ_SUCCESS)
2272 return;
2273
2274 status = pjsip_inv_send_msg(inv, tdata, NULL);
2275
2276 }
2277
2278 }
2279 else if (tsx->method.id == PJSIP_INVITE_METHOD &&
2280 tsx->role == PJSIP_ROLE_UAC)
2281 {
2282 /*
2283 * Handle outgoing re-INVITE
2284 */
2285 if (tsx->state == PJSIP_TSX_STATE_TERMINATED &&
2286 tsx->status_code/100 == 2)
2287 {
2288
2289 /* Re-INVITE was accepted. */
2290
2291 /* Process SDP */
2292 inv_check_sdp_in_incoming_msg(inv, tsx,
2293 e->body.tsx_state.src.rdata);
2294
2295 /* Send ACK */
2296 inv_send_ack(inv, e->body.tsx_state.src.rdata);
2297
2298 } else if (tsx->state == PJSIP_TSX_STATE_COMPLETED &&
2299 (tsx->status_code==401 || tsx->status_code==407))
2300 {
2301 pjsip_tx_data *tdata;
2302 pj_status_t status;
2303
2304 /* Handle authentication challenge. */
2305 status = pjsip_auth_clt_reinit_req( &dlg->auth_sess,
2306 e->body.tsx_state.src.rdata,
2307 tsx->last_tx,
2308 &tdata);
2309 if (status != PJ_SUCCESS)
2310 return;
2311
2312 /* Send re-INVITE */
2313 status = pjsip_inv_send_msg( inv, tdata, NULL);
2314
2315 } else if (tsx->status_code==PJSIP_SC_CALL_TSX_DOES_NOT_EXIST ||
2316 tsx->status_code==PJSIP_SC_REQUEST_TIMEOUT ||
2317 tsx->status_code >= 700)
2318 {
2319 /*
2320 * Handle responses that terminates dialog.
2321 */
Benny Prijonod4e0abd2006-03-05 11:53:36 +00002322 if (inv->cause==0) inv->cause = tsx->status_code;
Benny Prijono26ff9062006-02-21 23:47:00 +00002323 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
2324 }
2325 }
Benny Prijono268ca612006-02-07 12:34:11 +00002326}
2327
Benny Prijono8ad55352006-02-08 11:16:05 +00002328/*
2329 * After session has been terminated, but before dialog is destroyed
2330 * (because dialog has other usages, or because dialog is waiting for
2331 * the last transaction to terminate).
2332 */
2333static void inv_on_state_disconnected( pjsip_inv_session *inv, pjsip_event *e)
Benny Prijono268ca612006-02-07 12:34:11 +00002334{
Benny Prijono8ad55352006-02-08 11:16:05 +00002335 pjsip_transaction *tsx = e->body.tsx_state.tsx;
2336 pjsip_dialog *dlg = pjsip_tsx_get_dlg(tsx);
Benny Prijono268ca612006-02-07 12:34:11 +00002337
Benny Prijono8ad55352006-02-08 11:16:05 +00002338 PJ_ASSERT_ON_FAIL(tsx && dlg, return);
2339
2340 if (tsx->method.id == PJSIP_BYE_METHOD &&
2341 tsx->role == PJSIP_ROLE_UAS &&
2342 tsx->status_code < 200 &&
2343 e->body.tsx_state.type == PJSIP_EVENT_RX_MSG)
2344 {
2345
2346 /*
2347 * Be nice, handle incoming BYE.
2348 */
2349
2350 inv_respond_incoming_bye( inv, tsx, e->body.tsx_state.src.rdata, e );
2351
2352 }
Benny Prijono268ca612006-02-07 12:34:11 +00002353}
2354