blob: 7d1814a040958ddec6adca2b8635102040eff734 [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 {
73 NULL, NULL, /* prev, next. */
74 { "mod-invite", 10 }, /* Name. */
75 -1, /* Id */
76 PJSIP_MOD_PRIORITY_APPLICATION-1, /* Priority */
77 NULL, /* User data. */
78 &mod_inv_load, /* load() */
79 NULL, /* start() */
80 NULL, /* stop() */
81 &mod_inv_unload, /* unload() */
82 &mod_inv_on_rx_request, /* on_rx_request() */
83 &mod_inv_on_rx_response, /* on_rx_response() */
84 NULL, /* on_tx_request. */
85 NULL, /* on_tx_response() */
86 &mod_inv_on_tsx_state, /* on_tsx_state() */
87 }
88};
89
90
Benny Prijonoa66c7152006-02-09 01:26:14 +000091/* Invite session data to be attached to transaction. */
92struct tsx_inv_data
93{
94 pjsip_inv_session *inv;
95 pj_bool_t sdp_done;
96};
97
98
Benny Prijono8ad55352006-02-08 11:16:05 +000099/*
100 * Module load()
101 */
Benny Prijono268ca612006-02-07 12:34:11 +0000102static pj_status_t mod_inv_load(pjsip_endpoint *endpt)
103{
104 pj_str_t allowed[] = {{"INVITE", 6}, {"ACK",3}, {"BYE",3}, {"CANCEL",6}};
105
106 /* Register supported methods: INVITE, ACK, BYE, CANCEL */
107 pjsip_endpt_add_capability(endpt, &mod_inv.mod, PJSIP_H_ALLOW, NULL,
108 PJ_ARRAY_SIZE(allowed), allowed);
109
110 return PJ_SUCCESS;
111}
112
Benny Prijono8ad55352006-02-08 11:16:05 +0000113/*
114 * Module unload()
115 */
Benny Prijono268ca612006-02-07 12:34:11 +0000116static pj_status_t mod_inv_unload(void)
117{
118 /* Should remove capability here */
119 return PJ_SUCCESS;
120}
121
Benny Prijono8ad55352006-02-08 11:16:05 +0000122/*
Benny Prijono38998232006-02-08 22:44:25 +0000123 * Set session state.
124 */
125void inv_set_state(pjsip_inv_session *inv, pjsip_inv_state state,
126 pjsip_event *e)
127{
128 inv->state = state;
129 if (mod_inv.cb.on_state_changed)
130 (*mod_inv.cb.on_state_changed)(inv, e);
131
132 if (inv->state == PJSIP_INV_STATE_DISCONNECTED)
Benny Prijonoe4f2abb2006-02-10 14:04:05 +0000133 pjsip_dlg_dec_session(inv->dlg, &mod_inv.mod);
Benny Prijono38998232006-02-08 22:44:25 +0000134}
135
136
137/*
Benny Prijono8ad55352006-02-08 11:16:05 +0000138 * Send ACK for 2xx response.
139 */
140static pj_status_t inv_send_ack(pjsip_inv_session *inv, pjsip_rx_data *rdata)
Benny Prijono268ca612006-02-07 12:34:11 +0000141{
142 pjsip_tx_data *tdata;
143 pj_status_t status;
144
Benny Prijonoe4f2abb2006-02-10 14:04:05 +0000145 PJ_LOG(5,(inv->obj_name, "Received %s, sending ACK",
146 pjsip_rx_data_get_info(rdata)));
147
Benny Prijono268ca612006-02-07 12:34:11 +0000148 status = pjsip_dlg_create_request(inv->dlg, &pjsip_ack_method,
149 rdata->msg_info.cseq->cseq, &tdata);
150 if (status != PJ_SUCCESS) {
151 /* Better luck next time */
152 pj_assert(!"Unable to create ACK!");
153 return status;
154 }
155
156 status = pjsip_dlg_send_request(inv->dlg, tdata, NULL);
157 if (status != PJ_SUCCESS) {
158 /* Better luck next time */
159 pj_assert(!"Unable to send ACK!");
160 return status;
161 }
162
163 return PJ_SUCCESS;
164}
165
Benny Prijono8ad55352006-02-08 11:16:05 +0000166/*
167 * Module on_rx_request()
168 *
169 * This callback is called for these events:
170 * - endpoint receives request which was unhandled by higher priority
171 * modules (e.g. transaction layer, dialog layer).
172 * - dialog distributes incoming request to its usages.
173 */
174static pj_bool_t mod_inv_on_rx_request(pjsip_rx_data *rdata)
175{
176 pjsip_method *method;
Benny Prijono38998232006-02-08 22:44:25 +0000177 pjsip_dialog *dlg;
178 pjsip_inv_session *inv;
Benny Prijono8ad55352006-02-08 11:16:05 +0000179
180 /* Only wants to receive request from a dialog. */
Benny Prijono38998232006-02-08 22:44:25 +0000181 dlg = pjsip_rdata_get_dlg(rdata);
182 if (dlg == NULL)
Benny Prijono8ad55352006-02-08 11:16:05 +0000183 return PJ_FALSE;
184
Benny Prijono38998232006-02-08 22:44:25 +0000185 inv = dlg->mod_data[mod_inv.mod.id];
186
Benny Prijono8ad55352006-02-08 11:16:05 +0000187 /* Report to dialog that we handle INVITE, CANCEL, BYE, ACK.
188 * If we need to send response, it will be sent in the state
189 * handlers.
190 */
191 method = &rdata->msg_info.msg->line.req.method;
192
193 if (method->id == PJSIP_INVITE_METHOD ||
194 method->id == PJSIP_CANCEL_METHOD ||
Benny Prijono8ad55352006-02-08 11:16:05 +0000195 method->id == PJSIP_BYE_METHOD)
196 {
197 return PJ_TRUE;
198 }
199
Benny Prijono38998232006-02-08 22:44:25 +0000200 /* On receipt ACK request, when state is CONNECTING,
201 * move state to CONFIRMED.
202 */
Benny Prijono26ff9062006-02-21 23:47:00 +0000203 if (method->id == PJSIP_ACK_METHOD && inv) {
Benny Prijono38998232006-02-08 22:44:25 +0000204
Benny Prijono5eff0432006-02-09 14:14:21 +0000205 /* Terminate INVITE transaction, if it's still present. */
206 if (inv->invite_tsx &&
207 inv->invite_tsx->state <= PJSIP_TSX_STATE_COMPLETED)
208 {
209 pjsip_tsx_terminate(inv->invite_tsx,
210 inv->invite_tsx->status_code);
211 inv->invite_tsx = NULL;
212 }
213
Benny Prijono26ff9062006-02-21 23:47:00 +0000214 if (inv->state != PJSIP_INV_STATE_CONFIRMED) {
215 pjsip_event event;
216
217 PJSIP_EVENT_INIT_RX_MSG(event, rdata);
218 inv_set_state(inv, PJSIP_INV_STATE_CONFIRMED, &event);
219 }
Benny Prijono38998232006-02-08 22:44:25 +0000220 }
221
Benny Prijono8ad55352006-02-08 11:16:05 +0000222 return PJ_FALSE;
223}
224
225/*
226 * Module on_rx_response().
227 *
228 * This callback is called for these events:
229 * - dialog distributes incoming 2xx response to INVITE (outside
230 * transaction) to its usages.
231 * - endpoint distributes strayed responses.
232 */
Benny Prijono268ca612006-02-07 12:34:11 +0000233static pj_bool_t mod_inv_on_rx_response(pjsip_rx_data *rdata)
234{
235 pjsip_dialog *dlg;
236 pjsip_inv_session *inv;
237 pjsip_msg *msg = rdata->msg_info.msg;
238
239 dlg = pjsip_rdata_get_dlg(rdata);
240
241 /* Ignore responses outside dialog */
242 if (dlg == NULL)
243 return PJ_FALSE;
244
245 /* Ignore responses not belonging to invite session */
246 inv = pjsip_dlg_get_inv_session(dlg);
247 if (inv == NULL)
248 return PJ_FALSE;
249
250 /* This MAY be retransmission of 2xx response to INVITE.
251 * If it is, we need to send ACK.
252 */
253 if (msg->type == PJSIP_RESPONSE_MSG && msg->line.status.code/100==2 &&
Benny Prijono26ff9062006-02-21 23:47:00 +0000254 rdata->msg_info.cseq->method.id == PJSIP_INVITE_METHOD &&
255 inv->invite_tsx == NULL)
256 {
Benny Prijono268ca612006-02-07 12:34:11 +0000257
Benny Prijono8ad55352006-02-08 11:16:05 +0000258 inv_send_ack(inv, rdata);
Benny Prijono268ca612006-02-07 12:34:11 +0000259 return PJ_TRUE;
260
261 }
262
263 /* No other processing needs to be done here. */
264 return PJ_FALSE;
265}
266
Benny Prijono8ad55352006-02-08 11:16:05 +0000267/*
268 * Module on_tsx_state()
269 *
270 * This callback is called by dialog framework for all transactions
271 * inside the dialog for all its dialog usages.
272 */
Benny Prijono268ca612006-02-07 12:34:11 +0000273static void mod_inv_on_tsx_state(pjsip_transaction *tsx, pjsip_event *e)
274{
275 pjsip_dialog *dlg;
276 pjsip_inv_session *inv;
277
278 dlg = pjsip_tsx_get_dlg(tsx);
279 if (dlg == NULL)
280 return;
281
282 inv = pjsip_dlg_get_inv_session(dlg);
283 if (inv == NULL)
284 return;
285
286 /* Call state handler for the invite session. */
287 (*inv_state_handler[inv->state])(inv, e);
288
289 /* Call on_tsx_state */
290 if (mod_inv.cb.on_tsx_state_changed)
291 (*mod_inv.cb.on_tsx_state_changed)(inv, tsx, e);
292
293 /* Clear invite transaction when tsx is terminated. */
294 if (tsx->state==PJSIP_TSX_STATE_TERMINATED && tsx == inv->invite_tsx)
295 inv->invite_tsx = NULL;
296}
297
Benny Prijono8ad55352006-02-08 11:16:05 +0000298
299/*
300 * Initialize the invite module.
301 */
Benny Prijono268ca612006-02-07 12:34:11 +0000302PJ_DEF(pj_status_t) pjsip_inv_usage_init( pjsip_endpoint *endpt,
303 pjsip_module *app_module,
304 const pjsip_inv_callback *cb)
305{
306 pj_status_t status;
307
308 /* Check arguments. */
309 PJ_ASSERT_RETURN(endpt && app_module && cb, PJ_EINVAL);
310
311 /* Some callbacks are mandatory */
312 PJ_ASSERT_RETURN(cb->on_state_changed && cb->on_new_session, PJ_EINVAL);
313
314 /* Check if module already registered. */
315 PJ_ASSERT_RETURN(mod_inv.mod.id == -1, PJ_EINVALIDOP);
316
317 /* Copy param. */
318 pj_memcpy(&mod_inv.cb, cb, sizeof(pjsip_inv_callback));
319
320 mod_inv.endpt = endpt;
321 mod_inv.app_user = app_module;
322
323 /* Register the module. */
324 status = pjsip_endpt_register_module(endpt, &mod_inv.mod);
325
326 return status;
327}
328
Benny Prijono8ad55352006-02-08 11:16:05 +0000329/*
330 * Get the instance of invite module.
331 */
Benny Prijono268ca612006-02-07 12:34:11 +0000332PJ_DEF(pjsip_module*) pjsip_inv_usage_instance(void)
333{
334 return &mod_inv.mod;
335}
336
337
Benny Prijono632ce712006-02-09 14:01:40 +0000338
Benny Prijono8ad55352006-02-08 11:16:05 +0000339/*
340 * Return the invite session for the specified dialog.
341 */
Benny Prijono268ca612006-02-07 12:34:11 +0000342PJ_DEF(pjsip_inv_session*) pjsip_dlg_get_inv_session(pjsip_dialog *dlg)
343{
344 return dlg->mod_data[mod_inv.mod.id];
345}
346
Benny Prijono8ad55352006-02-08 11:16:05 +0000347
Benny Prijono268ca612006-02-07 12:34:11 +0000348/*
Benny Prijono8ad55352006-02-08 11:16:05 +0000349 * Create UAC invite session.
Benny Prijono268ca612006-02-07 12:34:11 +0000350 */
351PJ_DEF(pj_status_t) pjsip_inv_create_uac( pjsip_dialog *dlg,
352 const pjmedia_sdp_session *local_sdp,
353 unsigned options,
354 pjsip_inv_session **p_inv)
355{
356 pjsip_inv_session *inv;
357 pj_status_t status;
358
359 /* Verify arguments. */
360 PJ_ASSERT_RETURN(dlg && p_inv, PJ_EINVAL);
361
362 /* Normalize options */
363 if (options & PJSIP_INV_REQUIRE_100REL)
364 options |= PJSIP_INV_SUPPORT_100REL;
365
366 if (options & PJSIP_INV_REQUIRE_TIMER)
367 options |= PJSIP_INV_SUPPORT_TIMER;
368
369 /* Create the session */
370 inv = pj_pool_zalloc(dlg->pool, sizeof(pjsip_inv_session));
371 PJ_ASSERT_RETURN(inv != NULL, PJ_ENOMEM);
372
373 inv->pool = dlg->pool;
374 inv->role = PJSIP_ROLE_UAC;
375 inv->state = PJSIP_INV_STATE_NULL;
376 inv->dlg = dlg;
377 inv->options = options;
378
Benny Prijonoe4f2abb2006-02-10 14:04:05 +0000379 /* Object name will use the same dialog pointer. */
380 pj_snprintf(inv->obj_name, PJ_MAX_OBJ_NAME, "inv%p", dlg);
381
Benny Prijono268ca612006-02-07 12:34:11 +0000382 /* Create negotiator if local_sdp is specified. */
383 if (local_sdp) {
384 status = pjmedia_sdp_neg_create_w_local_offer(dlg->pool, local_sdp,
385 &inv->neg);
386 if (status != PJ_SUCCESS)
387 return status;
388 }
389
390 /* Register invite as dialog usage. */
391 status = pjsip_dlg_add_usage(dlg, &mod_inv.mod, inv);
392 if (status != PJ_SUCCESS)
393 return status;
394
395 /* Increment dialog session */
Benny Prijonoe4f2abb2006-02-10 14:04:05 +0000396 pjsip_dlg_inc_session(dlg, &mod_inv.mod);
Benny Prijono268ca612006-02-07 12:34:11 +0000397
398 /* Done */
399 *p_inv = inv;
Benny Prijonoe4f2abb2006-02-10 14:04:05 +0000400
401 PJ_LOG(5,(inv->obj_name, "UAC invite session created for dialog %s",
402 dlg->obj_name));
403
Benny Prijono268ca612006-02-07 12:34:11 +0000404 return PJ_SUCCESS;
405}
406
407/*
408 * Verify incoming INVITE request.
409 */
410PJ_DEF(pj_status_t) pjsip_inv_verify_request(pjsip_rx_data *rdata,
411 unsigned *options,
412 const pjmedia_sdp_session *l_sdp,
413 pjsip_dialog *dlg,
414 pjsip_endpoint *endpt,
415 pjsip_tx_data **p_tdata)
416{
417 pjsip_msg *msg;
418 pjsip_allow_hdr *allow;
419 pjsip_supported_hdr *sup_hdr;
420 pjsip_require_hdr *req_hdr;
421 int code = 200;
422 unsigned rem_option = 0;
423 pj_status_t status = PJ_SUCCESS;
424 pjsip_hdr res_hdr_list;
425
426 /* Init return arguments. */
427 if (p_tdata) *p_tdata = NULL;
428
429 /* Verify arguments. */
430 PJ_ASSERT_RETURN(rdata != NULL && options != NULL, PJ_EINVAL);
431
432 /* Normalize options */
433 if (*options & PJSIP_INV_REQUIRE_100REL)
434 *options |= PJSIP_INV_SUPPORT_100REL;
435
436 if (*options & PJSIP_INV_REQUIRE_TIMER)
437 *options |= PJSIP_INV_SUPPORT_TIMER;
438
439 /* Get the message in rdata */
440 msg = rdata->msg_info.msg;
441
442 /* Must be INVITE request. */
443 PJ_ASSERT_RETURN(msg->type == PJSIP_REQUEST_MSG &&
444 msg->line.req.method.id == PJSIP_INVITE_METHOD,
445 PJ_EINVAL);
446
447 /* If tdata is specified, then either dlg or endpt must be specified */
448 PJ_ASSERT_RETURN((!p_tdata) || (endpt || dlg), PJ_EINVAL);
449
450 /* Get the endpoint */
451 endpt = endpt ? endpt : dlg->endpt;
452
453 /* Init response header list */
454 pj_list_init(&res_hdr_list);
455
Benny Prijono8ad55352006-02-08 11:16:05 +0000456 /* Check the request body, see if it'inv something that we support
Benny Prijono268ca612006-02-07 12:34:11 +0000457 * (i.e. SDP).
458 */
459 if (msg->body) {
460 pjsip_msg_body *body = msg->body;
461 pj_str_t str_application = {"application", 11};
462 pj_str_t str_sdp = { "sdp", 3 };
463 pjmedia_sdp_session *sdp;
464
465 /* Check content type. */
466 if (pj_stricmp(&body->content_type.type, &str_application) != 0 ||
467 pj_stricmp(&body->content_type.subtype, &str_sdp) != 0)
468 {
469 /* Not "application/sdp" */
470 code = PJSIP_SC_UNSUPPORTED_MEDIA_TYPE;
471 status = PJSIP_ERRNO_FROM_SIP_STATUS(code);
472
473 if (p_tdata) {
474 /* Add Accept header to response */
475 pjsip_accept_hdr *acc;
476
477 acc = pjsip_accept_hdr_create(rdata->tp_info.pool);
478 PJ_ASSERT_RETURN(acc, PJ_ENOMEM);
479 acc->values[acc->count++] = pj_str("application/sdp");
480 pj_list_push_back(&res_hdr_list, acc);
481 }
482
483 goto on_return;
484 }
485
486 /* Parse and validate SDP */
487 status = pjmedia_sdp_parse(rdata->tp_info.pool, body->data, body->len,
488 &sdp);
489 if (status == PJ_SUCCESS)
490 status = pjmedia_sdp_validate(sdp);
491
492 if (status != PJ_SUCCESS) {
493 /* Unparseable or invalid SDP */
494 code = PJSIP_SC_BAD_REQUEST;
495
496 if (p_tdata) {
497 /* Add Warning header. */
498 pjsip_warning_hdr *w;
499
500 w = pjsip_warning_hdr_create_from_status(rdata->tp_info.pool,
501 pjsip_endpt_name(endpt),
502 status);
503 PJ_ASSERT_RETURN(w, PJ_ENOMEM);
504
505 pj_list_push_back(&res_hdr_list, w);
506 }
507
508 goto on_return;
509 }
510
511 /* Negotiate with local SDP */
512 if (l_sdp) {
513 pjmedia_sdp_neg *neg;
514
515 /* Local SDP must be valid! */
516 PJ_ASSERT_RETURN((status=pjmedia_sdp_validate(l_sdp))==PJ_SUCCESS,
517 status);
518
519 /* Create SDP negotiator */
520 status = pjmedia_sdp_neg_create_w_remote_offer(
521 rdata->tp_info.pool, l_sdp, sdp, &neg);
522 PJ_ASSERT_RETURN(status == PJ_SUCCESS, status);
523
524 /* Negotiate SDP */
525 status = pjmedia_sdp_neg_negotiate(rdata->tp_info.pool, neg, 0);
526 if (status != PJ_SUCCESS) {
527
528 /* Incompatible media */
529 code = PJSIP_SC_NOT_ACCEPTABLE_HERE;
530 status = PJSIP_ERRNO_FROM_SIP_STATUS(code);
531
532 if (p_tdata) {
533 pjsip_accept_hdr *acc;
534 pjsip_warning_hdr *w;
535
536 /* Add Warning header. */
537 w = pjsip_warning_hdr_create_from_status(
538 rdata->tp_info.pool,
539 pjsip_endpt_name(endpt), status);
540 PJ_ASSERT_RETURN(w, PJ_ENOMEM);
541
542 pj_list_push_back(&res_hdr_list, w);
543
544 /* Add Accept header to response */
545 acc = pjsip_accept_hdr_create(rdata->tp_info.pool);
546 PJ_ASSERT_RETURN(acc, PJ_ENOMEM);
547 acc->values[acc->count++] = pj_str("application/sdp");
548 pj_list_push_back(&res_hdr_list, acc);
549
550 }
551
552 goto on_return;
553 }
554 }
555 }
556
557 /* Check supported methods, see if peer supports UPDATE.
558 * We just assume that peer supports standard INVITE, ACK, CANCEL, and BYE
559 * implicitly by sending this INVITE.
560 */
561 allow = pjsip_msg_find_hdr(msg, PJSIP_H_ALLOW, NULL);
562 if (allow) {
563 unsigned i;
564 const pj_str_t STR_UPDATE = { "UPDATE", 6 };
565
566 for (i=0; i<allow->count; ++i) {
567 if (pj_stricmp(&allow->values[i], &STR_UPDATE)==0)
568 break;
569 }
570
571 if (i != allow->count) {
572 /* UPDATE is present in Allow */
573 rem_option |= PJSIP_INV_SUPPORT_UPDATE;
574 }
575
576 }
577
578 /* Check Supported header */
579 sup_hdr = pjsip_msg_find_hdr(msg, PJSIP_H_SUPPORTED, NULL);
580 if (sup_hdr) {
581 unsigned i;
582 pj_str_t STR_100REL = { "100rel", 6};
583 pj_str_t STR_TIMER = { "timer", 5 };
584
585 for (i=0; i<sup_hdr->count; ++i) {
586 if (pj_stricmp(&sup_hdr->values[i], &STR_100REL)==0)
587 rem_option |= PJSIP_INV_SUPPORT_100REL;
588 else if (pj_stricmp(&sup_hdr->values[i], &STR_TIMER)==0)
589 rem_option |= PJSIP_INV_SUPPORT_TIMER;
590 }
591 }
592
593 /* Check Require header */
594 req_hdr = pjsip_msg_find_hdr(msg, PJSIP_H_REQUIRE, NULL);
595 if (req_hdr) {
596 unsigned i;
597 pj_str_t STR_100REL = { "100rel", 6};
598 pj_str_t STR_TIMER = { "timer", 5 };
599 unsigned unsupp_cnt = 0;
600 pj_str_t unsupp_tags[PJSIP_GENERIC_ARRAY_MAX_COUNT];
601
602 for (i=0; i<req_hdr->count; ++i) {
603 if ((*options & PJSIP_INV_SUPPORT_100REL) &&
604 pj_stricmp(&req_hdr->values[i], &STR_100REL)==0)
605 {
606 rem_option |= PJSIP_INV_REQUIRE_100REL;
607
608 } else if ((*options && PJSIP_INV_SUPPORT_TIMER) &&
609 pj_stricmp(&req_hdr->values[i], &STR_TIMER)==0)
610 {
611 rem_option |= PJSIP_INV_REQUIRE_TIMER;
612
613 } else {
614 /* Unknown/unsupported extension tag! */
615 unsupp_tags[unsupp_cnt++] = req_hdr->values[i];
616 }
617 }
618
619 /* Check if there are required tags that we don't support */
620 if (unsupp_cnt) {
621
622 code = PJSIP_SC_BAD_EXTENSION;
623 status = PJSIP_ERRNO_FROM_SIP_STATUS(code);
624
625 if (p_tdata) {
626 pjsip_unsupported_hdr *unsupp_hdr;
627 const pjsip_hdr *h;
628
629 /* Add Unsupported header. */
630 unsupp_hdr = pjsip_unsupported_hdr_create(rdata->tp_info.pool);
631 PJ_ASSERT_RETURN(unsupp_hdr != NULL, PJ_ENOMEM);
632
633 unsupp_hdr->count = unsupp_cnt;
634 for (i=0; i<unsupp_cnt; ++i)
635 unsupp_hdr->values[i] = unsupp_tags[i];
636
637 pj_list_push_back(&res_hdr_list, unsupp_hdr);
638
639 /* Add Supported header. */
640 h = pjsip_endpt_get_capability(endpt, PJSIP_H_SUPPORTED,
641 NULL);
642 pj_assert(h);
643 if (h) {
644 sup_hdr = pjsip_hdr_clone(rdata->tp_info.pool, h);
645 pj_list_push_back(&res_hdr_list, sup_hdr);
646 }
647 }
648
649 goto on_return;
650 }
651 }
652
653 /* Check if there are local requirements that are not supported
654 * by peer.
655 */
656 if ( ((*options & PJSIP_INV_REQUIRE_100REL)!=0 &&
657 (rem_option & PJSIP_INV_SUPPORT_100REL)==0) ||
658 ((*options & PJSIP_INV_REQUIRE_TIMER)!=0 &&
659 (rem_option & PJSIP_INV_SUPPORT_TIMER)==0))
660 {
661 code = PJSIP_SC_EXTENSION_REQUIRED;
662 status = PJSIP_ERRNO_FROM_SIP_STATUS(code);
663
664 if (p_tdata) {
665 const pjsip_hdr *h;
666
667 /* Add Require header. */
668 req_hdr = pjsip_require_hdr_create(rdata->tp_info.pool);
669 PJ_ASSERT_RETURN(req_hdr != NULL, PJ_ENOMEM);
670
671 if (*options & PJSIP_INV_REQUIRE_100REL)
672 req_hdr->values[req_hdr->count++] = pj_str("100rel");
673
674 if (*options & PJSIP_INV_REQUIRE_TIMER)
675 req_hdr->values[req_hdr->count++] = pj_str("timer");
676
677 pj_list_push_back(&res_hdr_list, req_hdr);
678
679 /* Add Supported header. */
680 h = pjsip_endpt_get_capability(endpt, PJSIP_H_SUPPORTED,
681 NULL);
682 pj_assert(h);
683 if (h) {
684 sup_hdr = pjsip_hdr_clone(rdata->tp_info.pool, h);
685 pj_list_push_back(&res_hdr_list, sup_hdr);
686 }
687
688 }
689
690 goto on_return;
691 }
692
693on_return:
694
695 /* Create response if necessary */
696 if (code != 200 && p_tdata) {
697 pjsip_tx_data *tdata;
698 const pjsip_hdr *h;
699
700 if (dlg) {
701 status = pjsip_dlg_create_response(dlg, rdata, code, NULL,
702 &tdata);
703 } else {
704 status = pjsip_endpt_create_response(endpt, rdata, code, NULL,
705 &tdata);
706 }
707
708 if (status != PJ_SUCCESS)
709 return status;
710
711 /* Add response headers. */
712 h = res_hdr_list.next;
713 while (h != &res_hdr_list) {
714 pjsip_hdr *cloned;
715
716 cloned = pjsip_hdr_clone(tdata->pool, h);
717 PJ_ASSERT_RETURN(cloned, PJ_ENOMEM);
718
719 pjsip_msg_add_hdr(tdata->msg, cloned);
720
721 h = h->next;
722 }
723
724 *p_tdata = tdata;
725 }
726
727 return status;
728}
729
730/*
Benny Prijono8ad55352006-02-08 11:16:05 +0000731 * Create UAS invite session.
Benny Prijono268ca612006-02-07 12:34:11 +0000732 */
733PJ_DEF(pj_status_t) pjsip_inv_create_uas( pjsip_dialog *dlg,
734 pjsip_rx_data *rdata,
735 const pjmedia_sdp_session *local_sdp,
736 unsigned options,
737 pjsip_inv_session **p_inv)
738{
739 pjsip_inv_session *inv;
Benny Prijonoa66c7152006-02-09 01:26:14 +0000740 struct tsx_inv_data *tsx_inv_data;
Benny Prijono268ca612006-02-07 12:34:11 +0000741 pjsip_msg *msg;
742 pjmedia_sdp_session *rem_sdp = NULL;
743 pj_status_t status;
744
745 /* Verify arguments. */
746 PJ_ASSERT_RETURN(dlg && rdata && p_inv, PJ_EINVAL);
747
748 /* Dialog MUST have been initialised. */
749 PJ_ASSERT_RETURN(pjsip_rdata_get_tsx(rdata) != NULL, PJ_EINVALIDOP);
750
751 msg = rdata->msg_info.msg;
752
753 /* rdata MUST contain INVITE request */
754 PJ_ASSERT_RETURN(msg->type == PJSIP_REQUEST_MSG &&
755 msg->line.req.method.id == PJSIP_INVITE_METHOD,
756 PJ_EINVALIDOP);
757
758 /* Normalize options */
759 if (options & PJSIP_INV_REQUIRE_100REL)
760 options |= PJSIP_INV_SUPPORT_100REL;
761
762 if (options & PJSIP_INV_REQUIRE_TIMER)
763 options |= PJSIP_INV_SUPPORT_TIMER;
764
765 /* Create the session */
766 inv = pj_pool_zalloc(dlg->pool, sizeof(pjsip_inv_session));
767 PJ_ASSERT_RETURN(inv != NULL, PJ_ENOMEM);
768
769 inv->pool = dlg->pool;
770 inv->role = PJSIP_ROLE_UAS;
Benny Prijono38998232006-02-08 22:44:25 +0000771 inv->state = PJSIP_INV_STATE_NULL;
Benny Prijono268ca612006-02-07 12:34:11 +0000772 inv->dlg = dlg;
773 inv->options = options;
774
Benny Prijonoe4f2abb2006-02-10 14:04:05 +0000775 /* Object name will use the same dialog pointer. */
776 pj_snprintf(inv->obj_name, PJ_MAX_OBJ_NAME, "inv%p", dlg);
777
Benny Prijono268ca612006-02-07 12:34:11 +0000778 /* Parse SDP in message body, if present. */
779 if (msg->body) {
780 pjsip_msg_body *body = msg->body;
781
782 /* Parse and validate SDP */
783 status = pjmedia_sdp_parse(inv->pool, body->data, body->len,
784 &rem_sdp);
785 if (status == PJ_SUCCESS)
786 status = pjmedia_sdp_validate(rem_sdp);
787
788 if (status != PJ_SUCCESS)
789 return status;
790 }
791
792 /* Create negotiator. */
793 if (rem_sdp) {
794 status = pjmedia_sdp_neg_create_w_remote_offer(inv->pool, local_sdp,
795 rem_sdp, &inv->neg);
796
797 } else if (local_sdp) {
798 status = pjmedia_sdp_neg_create_w_local_offer(inv->pool, local_sdp,
799 &inv->neg);
800 } else {
Benny Prijono95196582006-02-09 00:13:40 +0000801 status = PJ_SUCCESS;
Benny Prijono268ca612006-02-07 12:34:11 +0000802 }
803
804 if (status != PJ_SUCCESS)
805 return status;
806
807 /* Register invite as dialog usage. */
808 status = pjsip_dlg_add_usage(dlg, &mod_inv.mod, inv);
809 if (status != PJ_SUCCESS)
810 return status;
811
812 /* Increment session in the dialog. */
Benny Prijonoe4f2abb2006-02-10 14:04:05 +0000813 pjsip_dlg_inc_session(dlg, &mod_inv.mod);
Benny Prijono268ca612006-02-07 12:34:11 +0000814
815 /* Save the invite transaction. */
816 inv->invite_tsx = pjsip_rdata_get_tsx(rdata);
Benny Prijonoa66c7152006-02-09 01:26:14 +0000817
818 /* Attach our data to the transaction. */
819 tsx_inv_data = pj_pool_zalloc(inv->invite_tsx->pool,
820 sizeof(struct tsx_inv_data));
821 tsx_inv_data->inv = inv;
822 inv->invite_tsx->mod_data[mod_inv.mod.id] = tsx_inv_data;
Benny Prijono268ca612006-02-07 12:34:11 +0000823
824 /* Done */
825 *p_inv = inv;
Benny Prijonoe4f2abb2006-02-10 14:04:05 +0000826
827 PJ_LOG(5,(inv->obj_name, "UAS invite session created for dialog %s",
828 dlg->obj_name));
829
Benny Prijono268ca612006-02-07 12:34:11 +0000830 return PJ_SUCCESS;
831}
832
833static void *clone_sdp(pj_pool_t *pool, const void *data, unsigned len)
834{
835 PJ_UNUSED_ARG(len);
836 return pjmedia_sdp_session_clone(pool, data);
837}
838
839static int print_sdp(pjsip_msg_body *body, char *buf, pj_size_t len)
840{
841 return pjmedia_sdp_print(body->data, buf, len);
842}
843
844static pjsip_msg_body *create_sdp_body(pj_pool_t *pool,
845 const pjmedia_sdp_session *c_sdp)
846{
847 pjsip_msg_body *body;
848
849
850 body = pj_pool_zalloc(pool, sizeof(pjsip_msg_body));
851 PJ_ASSERT_RETURN(body != NULL, NULL);
852
853 body->content_type.type = pj_str("application");
854 body->content_type.subtype = pj_str("sdp");
855 body->data = pjmedia_sdp_session_clone(pool, c_sdp);
856 body->len = 0;
857 body->clone_data = &clone_sdp;
858 body->print_body = &print_sdp;
859
860 return body;
861}
862
863/*
864 * Create initial INVITE request.
865 */
866PJ_DEF(pj_status_t) pjsip_inv_invite( pjsip_inv_session *inv,
867 pjsip_tx_data **p_tdata )
868{
869 pjsip_tx_data *tdata;
870 const pjsip_hdr *hdr;
Benny Prijono26ff9062006-02-21 23:47:00 +0000871 pj_bool_t has_sdp;
Benny Prijono268ca612006-02-07 12:34:11 +0000872 pj_status_t status;
873
874 /* Verify arguments. */
875 PJ_ASSERT_RETURN(inv && p_tdata, PJ_EINVAL);
876
Benny Prijono26ff9062006-02-21 23:47:00 +0000877 /* State MUST be NULL or CONFIRMED. */
878 PJ_ASSERT_RETURN(inv->state == PJSIP_INV_STATE_NULL ||
879 inv->state == PJSIP_INV_STATE_CONFIRMED,
880 PJ_EINVALIDOP);
Benny Prijono268ca612006-02-07 12:34:11 +0000881
Benny Prijono64f851e2006-02-23 13:49:28 +0000882 /* Lock dialog. */
883 pjsip_dlg_inc_lock(inv->dlg);
884
Benny Prijono268ca612006-02-07 12:34:11 +0000885 /* Create the INVITE request. */
886 status = pjsip_dlg_create_request(inv->dlg, &pjsip_invite_method, -1,
887 &tdata);
888 if (status != PJ_SUCCESS)
Benny Prijono64f851e2006-02-23 13:49:28 +0000889 goto on_return;
890
Benny Prijono268ca612006-02-07 12:34:11 +0000891
Benny Prijono26ff9062006-02-21 23:47:00 +0000892 /* If this is the first INVITE, then copy the headers from inv_hdr.
893 * These are the headers parsed from the request URI when the
894 * dialog was created.
895 */
896 if (inv->state == PJSIP_INV_STATE_NULL) {
897 hdr = inv->dlg->inv_hdr.next;
898
899 while (hdr != &inv->dlg->inv_hdr) {
900 pjsip_msg_add_hdr(tdata->msg,
901 pjsip_hdr_shallow_clone(tdata->pool, hdr));
902 hdr = hdr->next;
903 }
904 }
905
906 /* See if we have SDP to send. */
907 if (inv->neg) {
908 pjmedia_sdp_neg_state neg_state;
909
910 neg_state = pjmedia_sdp_neg_get_state(inv->neg);
911
912 has_sdp = (neg_state == PJMEDIA_SDP_NEG_STATE_LOCAL_OFFER ||
913 (neg_state == PJMEDIA_SDP_NEG_STATE_WAIT_NEGO &&
914 pjmedia_sdp_neg_has_local_answer(inv->neg)));
915
916
917 } else {
918 has_sdp = PJ_FALSE;
919 }
920
Benny Prijono268ca612006-02-07 12:34:11 +0000921 /* Add SDP, if any. */
Benny Prijono26ff9062006-02-21 23:47:00 +0000922 if (has_sdp) {
Benny Prijono268ca612006-02-07 12:34:11 +0000923 const pjmedia_sdp_session *offer;
924
925 status = pjmedia_sdp_neg_get_neg_local(inv->neg, &offer);
926 if (status != PJ_SUCCESS)
Benny Prijono64f851e2006-02-23 13:49:28 +0000927 goto on_return;
Benny Prijono268ca612006-02-07 12:34:11 +0000928
929 tdata->msg->body = create_sdp_body(tdata->pool, offer);
930 }
931
932 /* Add Allow header. */
933 hdr = pjsip_endpt_get_capability(inv->dlg->endpt, PJSIP_H_ALLOW, NULL);
934 if (hdr) {
935 pjsip_msg_add_hdr(tdata->msg,
936 pjsip_hdr_shallow_clone(tdata->pool, hdr));
937 }
938
939 /* Add Supported header */
940 hdr = pjsip_endpt_get_capability(inv->dlg->endpt, PJSIP_H_SUPPORTED, NULL);
941 if (hdr) {
942 pjsip_msg_add_hdr(tdata->msg,
943 pjsip_hdr_shallow_clone(tdata->pool, hdr));
944 }
945
946 /* Add Require header. */
947 PJ_TODO(INVITE_ADD_REQUIRE_HEADER);
948
949 /* Done. */
950 *p_tdata = tdata;
951
Benny Prijono64f851e2006-02-23 13:49:28 +0000952
953on_return:
954 pjsip_dlg_dec_lock(inv->dlg);
955 return status;
Benny Prijono268ca612006-02-07 12:34:11 +0000956}
957
958
959/*
Benny Prijono95196582006-02-09 00:13:40 +0000960 * Negotiate SDP.
961 */
962static pj_status_t inv_negotiate_sdp( pjsip_inv_session *inv )
963{
964 pj_status_t status;
965
966 PJ_ASSERT_RETURN(pjmedia_sdp_neg_get_state(inv->neg) ==
967 PJMEDIA_SDP_NEG_STATE_WAIT_NEGO,
968 PJMEDIA_SDPNEG_EINSTATE);
969
970 status = pjmedia_sdp_neg_negotiate(inv->pool, inv->neg, 0);
971
Benny Prijonoe4f2abb2006-02-10 14:04:05 +0000972 PJ_LOG(5,(inv->obj_name, "SDP negotiation done, status=%d", status));
973
Benny Prijono95196582006-02-09 00:13:40 +0000974 if (mod_inv.cb.on_media_update)
975 (*mod_inv.cb.on_media_update)(inv, status);
976
977 return status;
978}
979
980/*
Benny Prijonoa66c7152006-02-09 01:26:14 +0000981 * Check in incoming message for SDP offer/answer.
982 */
Benny Prijono26ff9062006-02-21 23:47:00 +0000983static pj_status_t inv_check_sdp_in_incoming_msg( pjsip_inv_session *inv,
984 pjsip_transaction *tsx,
985 pjsip_rx_data *rdata)
Benny Prijonoa66c7152006-02-09 01:26:14 +0000986{
987 struct tsx_inv_data *tsx_inv_data;
988 static const pj_str_t str_application = { "application", 11 };
989 static const pj_str_t str_sdp = { "sdp", 3 };
990 pj_status_t status;
991 pjsip_msg *msg;
992 pjmedia_sdp_session *sdp;
993
994 /* Get/attach invite session's transaction data */
995 tsx_inv_data = tsx->mod_data[mod_inv.mod.id];
996 if (tsx_inv_data == NULL) {
997 tsx_inv_data = pj_pool_zalloc(tsx->pool, sizeof(struct tsx_inv_data));
998 tsx_inv_data->inv = inv;
999 tsx->mod_data[mod_inv.mod.id] = tsx_inv_data;
1000 }
1001
1002 /* MUST NOT do multiple SDP offer/answer in a single transaction.
1003 */
1004
1005 if (tsx_inv_data->sdp_done)
Benny Prijono26ff9062006-02-21 23:47:00 +00001006 return PJ_SUCCESS;
Benny Prijonoa66c7152006-02-09 01:26:14 +00001007
1008 /* Check if SDP is present in the message. */
1009
1010 msg = rdata->msg_info.msg;
1011 if (msg->body == NULL) {
1012 /* Message doesn't have body. */
Benny Prijono26ff9062006-02-21 23:47:00 +00001013 return PJ_SUCCESS;
Benny Prijonoa66c7152006-02-09 01:26:14 +00001014 }
1015
1016 if (pj_stricmp(&msg->body->content_type.type, &str_application) ||
1017 pj_stricmp(&msg->body->content_type.subtype, &str_sdp))
1018 {
1019 /* Message body is not "application/sdp" */
Benny Prijono26ff9062006-02-21 23:47:00 +00001020 return PJMEDIA_SDP_EINSDP;
Benny Prijonoa66c7152006-02-09 01:26:14 +00001021 }
1022
1023 /* Parse the SDP body. */
1024
1025 status = pjmedia_sdp_parse(rdata->tp_info.pool, msg->body->data,
1026 msg->body->len, &sdp);
1027 if (status != PJ_SUCCESS) {
1028 char errmsg[PJ_ERR_MSG_SIZE];
1029 pj_strerror(status, errmsg, sizeof(errmsg));
1030 PJ_LOG(4,(THIS_FILE, "Error parsing SDP in %s: %s",
1031 pjsip_rx_data_get_info(rdata), errmsg));
Benny Prijono26ff9062006-02-21 23:47:00 +00001032 return PJMEDIA_SDP_EINSDP;
Benny Prijonoa66c7152006-02-09 01:26:14 +00001033 }
1034
1035 /* The SDP can be an offer or answer, depending on negotiator's state */
1036
1037 if (inv->neg == NULL ||
1038 pjmedia_sdp_neg_get_state(inv->neg) == PJMEDIA_SDP_NEG_STATE_DONE)
1039 {
1040
1041 /* This is an offer. */
1042
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00001043 PJ_LOG(5,(inv->obj_name, "Got SDP offer in %s",
1044 pjsip_rx_data_get_info(rdata)));
1045
Benny Prijonoa66c7152006-02-09 01:26:14 +00001046 if (inv->neg == NULL) {
1047 status=pjmedia_sdp_neg_create_w_remote_offer(inv->pool, NULL,
1048 sdp, &inv->neg);
1049 } else {
1050 status=pjmedia_sdp_neg_set_remote_offer(inv->pool, inv->neg, sdp);
1051 }
1052
1053 if (status != PJ_SUCCESS) {
1054 char errmsg[PJ_ERR_MSG_SIZE];
1055 pj_strerror(status, errmsg, sizeof(errmsg));
1056 PJ_LOG(4,(THIS_FILE, "Error processing SDP offer in %s: %s",
1057 pjsip_rx_data_get_info(rdata), errmsg));
Benny Prijono26ff9062006-02-21 23:47:00 +00001058 return PJMEDIA_SDP_EINSDP;
Benny Prijonoa66c7152006-02-09 01:26:14 +00001059 }
1060
1061 /* Inform application about remote offer. */
1062
Benny Prijono26ff9062006-02-21 23:47:00 +00001063 if (mod_inv.cb.on_rx_offer) {
1064
1065 (*mod_inv.cb.on_rx_offer)(inv, sdp);
1066
1067 }
Benny Prijonoa66c7152006-02-09 01:26:14 +00001068
1069 } else if (pjmedia_sdp_neg_get_state(inv->neg) ==
1070 PJMEDIA_SDP_NEG_STATE_LOCAL_OFFER)
1071 {
1072
1073 /* This is an answer.
1074 * Process and negotiate remote answer.
1075 */
1076
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00001077 PJ_LOG(5,(inv->obj_name, "Got SDP answer in %s",
1078 pjsip_rx_data_get_info(rdata)));
1079
Benny Prijonoa66c7152006-02-09 01:26:14 +00001080 status = pjmedia_sdp_neg_set_remote_answer(inv->pool, inv->neg, sdp);
1081
1082 if (status != PJ_SUCCESS) {
1083 char errmsg[PJ_ERR_MSG_SIZE];
1084 pj_strerror(status, errmsg, sizeof(errmsg));
1085 PJ_LOG(4,(THIS_FILE, "Error processing SDP answer in %s: %s",
1086 pjsip_rx_data_get_info(rdata), errmsg));
Benny Prijono26ff9062006-02-21 23:47:00 +00001087 return PJMEDIA_SDP_EINSDP;
Benny Prijonoa66c7152006-02-09 01:26:14 +00001088 }
1089
1090 /* Negotiate SDP */
1091
1092 inv_negotiate_sdp(inv);
1093
1094 /* Mark this transaction has having SDP offer/answer done. */
1095
1096 tsx_inv_data->sdp_done = 1;
1097
1098 } else {
1099
1100 PJ_LOG(5,(THIS_FILE, "Ignored SDP in %s: negotiator state is %s",
1101 pjsip_rx_data_get_info(rdata),
1102 pjmedia_sdp_neg_state_str(pjmedia_sdp_neg_get_state(inv->neg))));
1103 }
1104
Benny Prijono26ff9062006-02-21 23:47:00 +00001105 return PJ_SUCCESS;
Benny Prijonoa66c7152006-02-09 01:26:14 +00001106}
1107
1108
Benny Prijono26ff9062006-02-21 23:47:00 +00001109/*
1110 * Process INVITE answer, for both initial and subsequent re-INVITE
1111 */
1112static pj_status_t process_answer( pjsip_inv_session *inv,
1113 int st_code,
Benny Prijono64f851e2006-02-23 13:49:28 +00001114 pjsip_tx_data *tdata,
1115 const pjmedia_sdp_session *local_sdp)
Benny Prijono26ff9062006-02-21 23:47:00 +00001116{
1117 pj_status_t status;
1118 pjmedia_sdp_session *sdp = NULL;
1119
Benny Prijono64f851e2006-02-23 13:49:28 +00001120 /* If local_sdp is specified, then we MUST NOT have answered the
1121 * offer before.
Benny Prijono26ff9062006-02-21 23:47:00 +00001122 */
Benny Prijono64f851e2006-02-23 13:49:28 +00001123 if (local_sdp && (st_code/100==1 || st_code/100==2)) {
1124
1125 if (inv->neg == NULL) {
1126 status = pjmedia_sdp_neg_create_w_local_offer(inv->pool, local_sdp,
1127 &inv->neg);
1128 } else if (pjmedia_sdp_neg_get_state(inv->neg)==
1129 PJMEDIA_SDP_NEG_STATE_REMOTE_OFFER)
1130 {
1131 status = pjmedia_sdp_neg_set_local_answer(inv->pool, inv->neg,
1132 local_sdp);
1133 } else {
1134
1135 /* Can not specify local SDP at this state. */
1136 pj_assert(0);
1137 status = PJMEDIA_SDPNEG_EINSTATE;
1138 }
1139
1140 if (status != PJ_SUCCESS)
1141 return status;
1142
1143 }
1144
1145
1146 /* If SDP negotiator is ready, start negotiation. */
1147 if (st_code/100==2 || (st_code/10==18 && st_code!=180)) {
Benny Prijono26ff9062006-02-21 23:47:00 +00001148
1149 pjmedia_sdp_neg_state neg_state;
1150
Benny Prijono64f851e2006-02-23 13:49:28 +00001151 /* Start nego when appropriate. */
Benny Prijono26ff9062006-02-21 23:47:00 +00001152 neg_state = inv->neg ? pjmedia_sdp_neg_get_state(inv->neg) :
1153 PJMEDIA_SDP_NEG_STATE_NULL;
1154
1155 if (neg_state == PJMEDIA_SDP_NEG_STATE_LOCAL_OFFER) {
1156
1157 status = pjmedia_sdp_neg_get_neg_local(inv->neg, &sdp);
1158
1159 } else if (neg_state == PJMEDIA_SDP_NEG_STATE_WAIT_NEGO &&
1160 pjmedia_sdp_neg_has_local_answer(inv->neg) )
1161 {
1162
1163 status = inv_negotiate_sdp(inv);
1164 if (status != PJ_SUCCESS)
1165 return status;
1166
1167 status = pjmedia_sdp_neg_get_active_local(inv->neg, &sdp);
1168 }
Benny Prijono26ff9062006-02-21 23:47:00 +00001169 }
1170
Benny Prijono64f851e2006-02-23 13:49:28 +00001171 /* Include SDP when it's available for 2xx and 18x (but not 180) response.
Benny Prijono26ff9062006-02-21 23:47:00 +00001172 * Subsequent response will include this SDP.
1173 */
1174 if (sdp) {
1175 tdata->msg->body = create_sdp_body(tdata->pool, sdp);
1176 }
1177
Benny Prijono26ff9062006-02-21 23:47:00 +00001178
1179 return PJ_SUCCESS;
1180}
1181
Benny Prijonoa66c7152006-02-09 01:26:14 +00001182
1183/*
Benny Prijono64f851e2006-02-23 13:49:28 +00001184 * Create first response to INVITE
1185 */
1186PJ_DEF(pj_status_t) pjsip_inv_initial_answer( pjsip_inv_session *inv,
1187 pjsip_rx_data *rdata,
1188 int st_code,
1189 const pj_str_t *st_text,
1190 const pjmedia_sdp_session *sdp,
1191 pjsip_tx_data **p_tdata)
1192{
1193 pjsip_tx_data *tdata;
1194 pj_status_t status;
1195
1196 /* Verify arguments. */
1197 PJ_ASSERT_RETURN(inv && p_tdata, PJ_EINVAL);
1198
1199 /* Must have INVITE transaction. */
1200 PJ_ASSERT_RETURN(inv->invite_tsx, PJ_EBUG);
1201
1202 pjsip_dlg_inc_lock(inv->dlg);
1203
1204 /* Create response */
1205 status = pjsip_dlg_create_response(inv->dlg, rdata, st_code, st_text,
1206 &tdata);
1207 if (status != PJ_SUCCESS)
1208 goto on_return;
1209
1210 /* Process SDP in answer */
1211 status = process_answer(inv, st_code, tdata, sdp);
1212 if (status != PJ_SUCCESS) {
1213 pjsip_tx_data_dec_ref(tdata);
1214 goto on_return;
1215 }
1216
1217 *p_tdata = tdata;
1218
1219on_return:
1220 pjsip_dlg_dec_lock(inv->dlg);
1221 return status;
1222}
1223
1224
1225/*
Benny Prijono26ff9062006-02-21 23:47:00 +00001226 * Answer initial INVITE
1227 * Re-INVITE will be answered automatically, and will not use this function.
Benny Prijono268ca612006-02-07 12:34:11 +00001228 */
1229PJ_DEF(pj_status_t) pjsip_inv_answer( pjsip_inv_session *inv,
1230 int st_code,
1231 const pj_str_t *st_text,
1232 const pjmedia_sdp_session *local_sdp,
1233 pjsip_tx_data **p_tdata )
1234{
1235 pjsip_tx_data *last_res;
1236 pj_status_t status;
1237
1238 /* Verify arguments. */
1239 PJ_ASSERT_RETURN(inv && p_tdata, PJ_EINVAL);
1240
1241 /* Must have INVITE transaction. */
1242 PJ_ASSERT_RETURN(inv->invite_tsx, PJ_EBUG);
1243
1244 /* INVITE transaction MUST have transmitted a response (e.g. 100) */
1245 PJ_ASSERT_RETURN(inv->invite_tsx->last_tx, PJ_EINVALIDOP);
1246
Benny Prijono64f851e2006-02-23 13:49:28 +00001247 pjsip_dlg_inc_lock(inv->dlg);
Benny Prijono268ca612006-02-07 12:34:11 +00001248
1249 /* Modify last response. */
Benny Prijono26ff9062006-02-21 23:47:00 +00001250 last_res = inv->invite_tsx->last_tx;
Benny Prijono268ca612006-02-07 12:34:11 +00001251 status = pjsip_dlg_modify_response(inv->dlg, last_res, st_code, st_text);
1252 if (status != PJ_SUCCESS)
Benny Prijono64f851e2006-02-23 13:49:28 +00001253 goto on_return;
Benny Prijono268ca612006-02-07 12:34:11 +00001254
Benny Prijono268ca612006-02-07 12:34:11 +00001255
Benny Prijono26ff9062006-02-21 23:47:00 +00001256 /* Process SDP in answer */
Benny Prijono64f851e2006-02-23 13:49:28 +00001257 status = process_answer(inv, st_code, last_res, local_sdp);
1258 if (status != PJ_SUCCESS) {
1259 pjsip_tx_data_dec_ref(last_res);
1260 goto on_return;
1261 }
Benny Prijono268ca612006-02-07 12:34:11 +00001262
Benny Prijono268ca612006-02-07 12:34:11 +00001263
1264 *p_tdata = last_res;
1265
Benny Prijono64f851e2006-02-23 13:49:28 +00001266on_return:
1267 pjsip_dlg_dec_lock(inv->dlg);
1268 return status;
Benny Prijono268ca612006-02-07 12:34:11 +00001269}
1270
1271
1272/*
Benny Prijono26ff9062006-02-21 23:47:00 +00001273 * Set SDP answer.
1274 */
1275PJ_DEF(pj_status_t) pjsip_inv_set_sdp_answer( pjsip_inv_session *inv,
1276 const pjmedia_sdp_session *sdp )
1277{
1278 pj_status_t status;
1279
1280 PJ_ASSERT_RETURN(inv && sdp, PJ_EINVAL);
1281
1282 pjsip_dlg_inc_lock(inv->dlg);
1283 status = pjmedia_sdp_neg_set_local_answer( inv->pool, inv->neg, sdp);
1284 pjsip_dlg_dec_lock(inv->dlg);
1285
1286 return status;
1287}
1288
1289
1290/*
Benny Prijono268ca612006-02-07 12:34:11 +00001291 * End session.
1292 */
1293PJ_DEF(pj_status_t) pjsip_inv_end_session( pjsip_inv_session *inv,
1294 int st_code,
1295 const pj_str_t *st_text,
1296 pjsip_tx_data **p_tdata )
1297{
1298 pjsip_tx_data *tdata;
1299 pj_status_t status;
1300
1301 /* Verify arguments. */
1302 PJ_ASSERT_RETURN(inv && p_tdata, PJ_EINVAL);
1303
1304 /* Create appropriate message. */
1305 switch (inv->state) {
1306 case PJSIP_INV_STATE_CALLING:
1307 case PJSIP_INV_STATE_EARLY:
1308 case PJSIP_INV_STATE_INCOMING:
1309
1310 if (inv->role == PJSIP_ROLE_UAC) {
1311
1312 /* For UAC when session has not been confirmed, create CANCEL. */
1313
1314 /* MUST have the original UAC INVITE transaction. */
1315 PJ_ASSERT_RETURN(inv->invite_tsx != NULL, PJ_EBUG);
1316
1317 /* But CANCEL should only be called when we have received a
1318 * provisional response. If we haven't received any responses,
1319 * just destroy the transaction.
1320 */
1321 if (inv->invite_tsx->status_code < 100) {
1322
1323 pjsip_tsx_terminate(inv->invite_tsx, 487);
Benny Prijonofccab712006-02-22 22:23:22 +00001324 *p_tdata = NULL;
1325 return PJ_SUCCESS;
Benny Prijono268ca612006-02-07 12:34:11 +00001326 }
1327
1328 /* The CSeq here assumes that the dialog is started with an
1329 * INVITE session. This may not be correct; dialog can be
1330 * started as SUBSCRIBE session.
1331 * So fix this!
1332 */
1333 status = pjsip_endpt_create_cancel(inv->dlg->endpt,
1334 inv->invite_tsx->last_tx,
1335 &tdata);
1336
1337 } else {
1338
1339 /* For UAS, send a final response. */
1340 tdata = inv->invite_tsx->last_tx;
1341 PJ_ASSERT_RETURN(tdata != NULL, PJ_EINVALIDOP);
1342
Benny Prijono26ff9062006-02-21 23:47:00 +00001343 //status = pjsip_dlg_modify_response(inv->dlg, tdata, st_code,
1344 // st_text);
1345 status = pjsip_inv_answer(inv, st_code, st_text, NULL, &tdata);
Benny Prijono268ca612006-02-07 12:34:11 +00001346 }
1347 break;
1348
1349 case PJSIP_INV_STATE_CONNECTING:
1350 case PJSIP_INV_STATE_CONFIRMED:
1351 /* For established dialog, send BYE */
1352 status = pjsip_dlg_create_request(inv->dlg, &pjsip_bye_method, -1,
1353 &tdata);
1354 break;
1355
1356 case PJSIP_INV_STATE_DISCONNECTED:
Benny Prijono268ca612006-02-07 12:34:11 +00001357 /* No need to do anything. */
1358 PJ_TODO(RETURN_A_PROPER_STATUS_CODE_HERE);
1359 return PJ_EINVALIDOP;
1360
1361 default:
1362 pj_assert("!Invalid operation!");
1363 return PJ_EINVALIDOP;
1364 }
1365
1366 if (status != PJ_SUCCESS)
1367 return status;
1368
1369
1370 /* Done */
1371
1372 *p_tdata = tdata;
1373
1374 return PJ_SUCCESS;
1375}
1376
1377
1378/*
1379 * Create re-INVITE.
1380 */
1381PJ_DEF(pj_status_t) pjsip_inv_reinvite( pjsip_inv_session *inv,
1382 const pj_str_t *new_contact,
1383 const pjmedia_sdp_session *new_offer,
1384 pjsip_tx_data **p_tdata )
1385{
Benny Prijono26ff9062006-02-21 23:47:00 +00001386 pj_status_t status;
1387 pjsip_contact_hdr *contact_hdr = NULL;
Benny Prijono268ca612006-02-07 12:34:11 +00001388
Benny Prijono26ff9062006-02-21 23:47:00 +00001389 /* Check arguments. */
1390 PJ_ASSERT_RETURN(inv && p_tdata, PJ_EINVAL);
1391
1392 /* Must NOT have a pending INVITE transaction */
1393 PJ_ASSERT_RETURN(inv->invite_tsx==NULL, PJ_EINVALIDOP);
1394
1395
1396 pjsip_dlg_inc_lock(inv->dlg);
1397
1398 if (new_contact) {
1399 pj_str_t tmp;
1400 const pj_str_t STR_CONTACT = { "Contact", 7 };
1401
1402 pj_strdup_with_null(inv->dlg->pool, &tmp, new_contact);
1403 contact_hdr = pjsip_parse_hdr(inv->dlg->pool, &STR_CONTACT,
1404 tmp.ptr, tmp.slen, NULL);
1405 if (!contact_hdr) {
1406 status = PJSIP_EINVALIDURI;
1407 goto on_return;
1408 }
1409 }
1410
1411
1412 if (new_offer) {
1413 if (!inv->neg) {
1414 status = pjmedia_sdp_neg_create_w_local_offer(inv->pool, new_offer,
1415 &inv->neg);
1416 if (status != PJ_SUCCESS)
1417 goto on_return;
1418
1419 } else switch (pjmedia_sdp_neg_get_state(inv->neg)) {
1420
1421 case PJMEDIA_SDP_NEG_STATE_NULL:
1422 pj_assert(!"Unexpected SDP neg state NULL");
1423 status = PJ_EBUG;
1424 goto on_return;
1425
1426 case PJMEDIA_SDP_NEG_STATE_LOCAL_OFFER:
1427 PJ_LOG(4,(inv->obj_name,
1428 "pjsip_inv_reinvite: already have an offer, new "
1429 "offer is ignored"));
1430 break;
1431
1432 case PJMEDIA_SDP_NEG_STATE_REMOTE_OFFER:
1433 status = pjmedia_sdp_neg_set_local_answer(inv->pool, inv->neg,
1434 new_offer);
1435 if (status != PJ_SUCCESS)
1436 goto on_return;
1437 break;
1438
1439 case PJMEDIA_SDP_NEG_STATE_WAIT_NEGO:
1440 PJ_LOG(4,(inv->obj_name,
1441 "pjsip_inv_reinvite: SDP in WAIT_NEGO state, new "
1442 "offer is ignored"));
1443 break;
1444
1445 case PJMEDIA_SDP_NEG_STATE_DONE:
1446 status = pjmedia_sdp_neg_modify_local_offer(inv->pool,inv->neg,
1447 new_offer);
1448 if (status != PJ_SUCCESS)
1449 goto on_return;
1450 break;
1451 }
1452 }
1453
1454 if (contact_hdr)
1455 inv->dlg->local.contact = contact_hdr;
1456
1457 status = pjsip_inv_invite(inv, p_tdata);
1458
1459on_return:
1460 pjsip_dlg_dec_lock(inv->dlg);
1461 return status;
Benny Prijono268ca612006-02-07 12:34:11 +00001462}
1463
1464/*
1465 * Create UPDATE.
1466 */
1467PJ_DEF(pj_status_t) pjsip_inv_update ( pjsip_inv_session *inv,
1468 const pj_str_t *new_contact,
1469 const pjmedia_sdp_session *new_offer,
1470 pjsip_tx_data **p_tdata )
1471{
1472 PJ_UNUSED_ARG(inv);
1473 PJ_UNUSED_ARG(new_contact);
1474 PJ_UNUSED_ARG(new_offer);
1475 PJ_UNUSED_ARG(p_tdata);
1476
1477 PJ_TODO(CREATE_UPDATE_REQUEST);
1478 return PJ_ENOTSUP;
1479}
1480
1481/*
1482 * Send a request or response message.
1483 */
1484PJ_DEF(pj_status_t) pjsip_inv_send_msg( pjsip_inv_session *inv,
1485 pjsip_tx_data *tdata,
1486 void *token )
1487{
1488 pj_status_t status;
1489
1490 /* Verify arguments. */
1491 PJ_ASSERT_RETURN(inv && tdata, PJ_EINVAL);
1492
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00001493 PJ_LOG(5,(inv->obj_name, "Sending %s",
1494 pjsip_tx_data_get_info(tdata)));
1495
Benny Prijono268ca612006-02-07 12:34:11 +00001496 if (tdata->msg->type == PJSIP_REQUEST_MSG) {
1497 pjsip_transaction *tsx;
Benny Prijonoa66c7152006-02-09 01:26:14 +00001498 struct tsx_inv_data *tsx_inv_data;
Benny Prijono268ca612006-02-07 12:34:11 +00001499
1500 status = pjsip_dlg_send_request(inv->dlg, tdata, &tsx);
1501 if (status != PJ_SUCCESS)
1502 return status;
1503
Benny Prijonoa66c7152006-02-09 01:26:14 +00001504 tsx_inv_data = pj_pool_zalloc(tsx->pool, sizeof(struct tsx_inv_data));
1505 tsx_inv_data->inv = inv;
1506
1507 tsx->mod_data[mod_inv.mod.id] = tsx_inv_data;
Benny Prijono268ca612006-02-07 12:34:11 +00001508 tsx->mod_data[mod_inv.app_user->id] = token;
1509
1510 } else {
1511 pjsip_cseq_hdr *cseq;
1512
1513 /* Can only do this to send response to original INVITE
1514 * request.
1515 */
Benny Prijono64f851e2006-02-23 13:49:28 +00001516 PJ_ASSERT_RETURN((cseq=pjsip_msg_find_hdr(tdata->msg, PJSIP_H_CSEQ, NULL))!=NULL
1517 && (cseq->cseq == inv->invite_tsx->cseq),
Benny Prijono268ca612006-02-07 12:34:11 +00001518 PJ_EINVALIDOP);
1519
1520 status = pjsip_dlg_send_response(inv->dlg, inv->invite_tsx, tdata);
1521 if (status != PJ_SUCCESS)
1522 return status;
1523 }
1524
1525 /* Done (?) */
1526 return PJ_SUCCESS;
1527}
1528
1529
Benny Prijono8ad55352006-02-08 11:16:05 +00001530/*
1531 * Respond to incoming CANCEL request.
1532 */
1533static void inv_respond_incoming_cancel(pjsip_inv_session *inv,
1534 pjsip_transaction *cancel_tsx,
1535 pjsip_rx_data *rdata)
1536{
1537 pjsip_tx_data *tdata;
1538 pjsip_transaction *invite_tsx;
1539 pj_str_t key;
1540 pj_status_t status;
1541
1542 /* See if we have matching INVITE server transaction: */
1543
1544 pjsip_tsx_create_key(rdata->tp_info.pool, &key, PJSIP_ROLE_UAS,
1545 &pjsip_invite_method, rdata);
1546 invite_tsx = pjsip_tsx_layer_find_tsx(&key, PJ_TRUE);
1547
1548 if (invite_tsx == NULL) {
1549
1550 /* Invite transaction not found!
1551 * Respond CANCEL with 491 (RFC 3261 Section 9.2 page 42)
1552 */
1553 status = pjsip_dlg_create_response( inv->dlg, rdata, 200, NULL,
1554 &tdata);
1555
1556 } else {
1557 /* Always answer CANCEL will 200 (OK) regardless of
1558 * the state of the INVITE transaction.
1559 */
1560 status = pjsip_dlg_create_response( inv->dlg, rdata, 200, NULL,
1561 &tdata);
1562 }
1563
1564 /* See if we have created the response successfully. */
1565 if (status != PJ_SUCCESS) return;
1566
1567 /* Send the CANCEL response */
1568 status = pjsip_dlg_send_response(inv->dlg, cancel_tsx, tdata);
1569 if (status != PJ_SUCCESS) return;
1570
1571
1572 /* See if we need to terminate the UAS INVITE transaction
1573 * with 487 (Request Terminated) response.
1574 */
1575 if (invite_tsx && invite_tsx->status_code < 200) {
1576
1577 pj_assert(invite_tsx->last_tx != NULL);
1578
1579 tdata = invite_tsx->last_tx;
1580
1581 status = pjsip_dlg_modify_response(inv->dlg, tdata, 487, NULL);
1582 if (status == PJ_SUCCESS)
1583 pjsip_dlg_send_response(inv->dlg, invite_tsx, tdata);
1584 }
1585
1586 if (invite_tsx)
1587 pj_mutex_unlock(invite_tsx->mutex);
1588}
1589
1590
1591/*
1592 * Respond to incoming BYE request.
1593 */
1594static void inv_respond_incoming_bye( pjsip_inv_session *inv,
1595 pjsip_transaction *bye_tsx,
1596 pjsip_rx_data *rdata,
1597 pjsip_event *e )
1598{
1599 pj_status_t status;
1600 pjsip_tx_data *tdata;
1601
1602 /* Respond BYE with 200: */
1603
1604 status = pjsip_dlg_create_response(inv->dlg, rdata, 200, NULL, &tdata);
1605 if (status != PJ_SUCCESS) return;
1606
1607 status = pjsip_dlg_send_response(inv->dlg, bye_tsx, tdata);
1608 if (status != PJ_SUCCESS) return;
1609
1610 /* Terminate session: */
1611
1612 if (inv->state != PJSIP_INV_STATE_DISCONNECTED)
1613 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
1614}
1615
1616/*
Benny Prijono38998232006-02-08 22:44:25 +00001617 * Respond to BYE request.
1618 */
1619static void inv_handle_bye_response( pjsip_inv_session *inv,
1620 pjsip_transaction *tsx,
1621 pjsip_rx_data *rdata,
1622 pjsip_event *e )
1623{
1624 pj_status_t status;
1625
1626 if (e->body.tsx_state.type != PJSIP_EVENT_RX_MSG) {
1627 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
1628 return;
1629 }
1630
1631 /* Handle 401/407 challenge. */
1632 if (tsx->status_code == 401 || tsx->status_code == 407) {
1633
1634 pjsip_tx_data *tdata;
1635
1636 status = pjsip_auth_clt_reinit_req( &inv->dlg->auth_sess,
1637 rdata,
1638 tsx->last_tx,
1639 &tdata);
1640
1641 if (status != PJ_SUCCESS) {
1642
1643 /* Does not have proper credentials.
1644 * End the session anyway.
1645 */
1646 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
1647
1648 } else {
1649 /* Re-send BYE. */
1650 status = pjsip_inv_send_msg(inv, tdata, NULL );
1651 }
1652
1653 } else {
1654
1655 /* End the session. */
1656
1657 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
1658 }
1659
1660}
1661
1662/*
Benny Prijono8ad55352006-02-08 11:16:05 +00001663 * State NULL is before anything is sent/received.
1664 */
1665static void inv_on_state_null( pjsip_inv_session *inv, pjsip_event *e)
Benny Prijono268ca612006-02-07 12:34:11 +00001666{
1667 pjsip_transaction *tsx = e->body.tsx_state.tsx;
1668 pjsip_dialog *dlg = pjsip_tsx_get_dlg(tsx);
1669
1670 PJ_ASSERT_ON_FAIL(tsx && dlg, return);
1671
1672 if (tsx->method.id == PJSIP_INVITE_METHOD) {
1673
Benny Prijono64f851e2006-02-23 13:49:28 +00001674 /* Keep the initial INVITE transaction. */
1675 if (inv->invite_tsx == NULL)
1676 inv->invite_tsx = tsx;
Benny Prijono268ca612006-02-07 12:34:11 +00001677
Benny Prijono64f851e2006-02-23 13:49:28 +00001678 if (dlg->role == PJSIP_ROLE_UAC) {
Benny Prijono268ca612006-02-07 12:34:11 +00001679
1680 switch (tsx->state) {
1681 case PJSIP_TSX_STATE_CALLING:
Benny Prijono8ad55352006-02-08 11:16:05 +00001682 inv_set_state(inv, PJSIP_INV_STATE_CALLING, e);
Benny Prijono268ca612006-02-07 12:34:11 +00001683 break;
1684 default:
Benny Prijono64f851e2006-02-23 13:49:28 +00001685 inv_on_state_calling(inv, e);
Benny Prijono268ca612006-02-07 12:34:11 +00001686 break;
1687 }
1688
1689 } else {
1690 switch (tsx->state) {
1691 case PJSIP_TSX_STATE_TRYING:
Benny Prijono8ad55352006-02-08 11:16:05 +00001692 inv_set_state(inv, PJSIP_INV_STATE_INCOMING, e);
Benny Prijono268ca612006-02-07 12:34:11 +00001693 break;
Benny Prijono38998232006-02-08 22:44:25 +00001694 case PJSIP_TSX_STATE_PROCEEDING:
1695 inv_set_state(inv, PJSIP_INV_STATE_INCOMING, e);
1696 if (tsx->status_code > 100)
1697 inv_set_state(inv, PJSIP_INV_STATE_EARLY, e);
1698 break;
Benny Prijono268ca612006-02-07 12:34:11 +00001699 default:
Benny Prijono64f851e2006-02-23 13:49:28 +00001700 inv_on_state_incoming(inv, e);
1701 break;
Benny Prijono268ca612006-02-07 12:34:11 +00001702 }
1703 }
1704
1705 } else {
1706 pj_assert(!"Unexpected transaction type");
1707 }
1708}
1709
Benny Prijono8ad55352006-02-08 11:16:05 +00001710/*
1711 * State CALLING is after sending initial INVITE request but before
1712 * any response (with tag) is received.
1713 */
1714static void inv_on_state_calling( pjsip_inv_session *inv, pjsip_event *e)
Benny Prijono268ca612006-02-07 12:34:11 +00001715{
1716 pjsip_transaction *tsx = e->body.tsx_state.tsx;
1717 pjsip_dialog *dlg = pjsip_tsx_get_dlg(tsx);
Benny Prijonoccf95622006-02-07 18:48:01 +00001718 pj_status_t status;
Benny Prijono268ca612006-02-07 12:34:11 +00001719
1720 PJ_ASSERT_ON_FAIL(tsx && dlg, return);
1721
Benny Prijono8ad55352006-02-08 11:16:05 +00001722 if (tsx == inv->invite_tsx) {
Benny Prijono268ca612006-02-07 12:34:11 +00001723
1724 switch (tsx->state) {
1725
Benny Prijono64f851e2006-02-23 13:49:28 +00001726 case PJSIP_TSX_STATE_CALLING:
1727 inv_set_state(inv, PJSIP_INV_STATE_CALLING, e);
1728 break;
1729
Benny Prijono268ca612006-02-07 12:34:11 +00001730 case PJSIP_TSX_STATE_PROCEEDING:
1731 if (dlg->remote.info->tag.slen) {
Benny Prijonoa66c7152006-02-09 01:26:14 +00001732
Benny Prijono8ad55352006-02-08 11:16:05 +00001733 inv_set_state(inv, PJSIP_INV_STATE_EARLY, e);
Benny Prijonoa66c7152006-02-09 01:26:14 +00001734
1735 inv_check_sdp_in_incoming_msg(inv, tsx,
1736 e->body.tsx_state.src.rdata);
1737
Benny Prijono268ca612006-02-07 12:34:11 +00001738 } else {
1739 /* Ignore 100 (Trying) response, as it doesn't change
1740 * session state. It only ceases retransmissions.
1741 */
1742 }
1743 break;
1744
1745 case PJSIP_TSX_STATE_COMPLETED:
1746 if (tsx->status_code/100 == 2) {
1747
1748 /* This should not happen.
1749 * When transaction receives 2xx, it should be terminated
1750 */
1751 pj_assert(0);
Benny Prijono8ad55352006-02-08 11:16:05 +00001752 inv_set_state(inv, PJSIP_INV_STATE_CONNECTING, e);
Benny Prijonoa66c7152006-02-09 01:26:14 +00001753
1754 inv_check_sdp_in_incoming_msg(inv, tsx,
1755 e->body.tsx_state.src.rdata);
Benny Prijono268ca612006-02-07 12:34:11 +00001756
Benny Prijonoccf95622006-02-07 18:48:01 +00001757 } else if (tsx->status_code==401 || tsx->status_code==407) {
1758
1759 /* Handle authentication failure:
1760 * Resend the request with Authorization header.
1761 */
1762 pjsip_tx_data *tdata;
1763
Benny Prijono8ad55352006-02-08 11:16:05 +00001764 status = pjsip_auth_clt_reinit_req(&inv->dlg->auth_sess,
Benny Prijonoccf95622006-02-07 18:48:01 +00001765 e->body.tsx_state.src.rdata,
1766 tsx->last_tx,
1767 &tdata);
1768
1769 if (status != PJ_SUCCESS) {
1770
1771 /* Does not have proper credentials.
1772 * End the session.
1773 */
Benny Prijono8ad55352006-02-08 11:16:05 +00001774 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
Benny Prijonoccf95622006-02-07 18:48:01 +00001775
1776 } else {
1777
1778 /* Restart session. */
Benny Prijono8ad55352006-02-08 11:16:05 +00001779 inv->state = PJSIP_INV_STATE_NULL;
1780 inv->invite_tsx = NULL;
Benny Prijonoccf95622006-02-07 18:48:01 +00001781
1782 /* Send the request. */
Benny Prijono8ad55352006-02-08 11:16:05 +00001783 status = pjsip_inv_send_msg(inv, tdata, NULL );
Benny Prijonoccf95622006-02-07 18:48:01 +00001784 }
1785
Benny Prijono268ca612006-02-07 12:34:11 +00001786 } else {
Benny Prijonoccf95622006-02-07 18:48:01 +00001787
Benny Prijono8ad55352006-02-08 11:16:05 +00001788 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
Benny Prijonoccf95622006-02-07 18:48:01 +00001789
Benny Prijono268ca612006-02-07 12:34:11 +00001790 }
1791 break;
1792
1793 case PJSIP_TSX_STATE_TERMINATED:
1794 /* INVITE transaction can be terminated either because UAC
1795 * transaction received 2xx response or because of transport
1796 * error.
1797 */
1798 if (tsx->status_code/100 == 2) {
1799 /* This must be receipt of 2xx response */
1800
1801 /* Set state to CONNECTING */
Benny Prijono8ad55352006-02-08 11:16:05 +00001802 inv_set_state(inv, PJSIP_INV_STATE_CONNECTING, e);
Benny Prijono268ca612006-02-07 12:34:11 +00001803
Benny Prijonoa66c7152006-02-09 01:26:14 +00001804 inv_check_sdp_in_incoming_msg(inv, tsx,
1805 e->body.tsx_state.src.rdata);
1806
Benny Prijono268ca612006-02-07 12:34:11 +00001807 /* Send ACK */
1808 pj_assert(e->body.tsx_state.type == PJSIP_EVENT_RX_MSG);
1809
Benny Prijono8ad55352006-02-08 11:16:05 +00001810 inv_send_ack(inv, e->body.tsx_state.src.rdata);
Benny Prijono5eff0432006-02-09 14:14:21 +00001811 inv_set_state(inv, PJSIP_INV_STATE_CONFIRMED, e);
Benny Prijono268ca612006-02-07 12:34:11 +00001812
Benny Prijonoa66c7152006-02-09 01:26:14 +00001813
Benny Prijono268ca612006-02-07 12:34:11 +00001814 } else {
Benny Prijono8ad55352006-02-08 11:16:05 +00001815 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
Benny Prijono268ca612006-02-07 12:34:11 +00001816 }
1817 break;
1818
Benny Prijono34a404e2006-02-09 14:38:30 +00001819 default:
1820 break;
Benny Prijono268ca612006-02-07 12:34:11 +00001821 }
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00001822
1823 } else if (inv->role == PJSIP_ROLE_UAC &&
1824 tsx->role == PJSIP_ROLE_UAC &&
1825 tsx->method.id == PJSIP_CANCEL_METHOD)
1826 {
1827 /*
1828 * Handle case when outgoing CANCEL is answered with 481 (Call/
1829 * Transaction Does Not Exist), 408, or when it's timed out. In these
1830 * cases, disconnect session (i.e. dialog usage only).
1831 */
1832 if (tsx->status_code == PJSIP_SC_CALL_TSX_DOES_NOT_EXIST ||
1833 tsx->status_code == PJSIP_SC_REQUEST_TIMEOUT ||
1834 tsx->status_code == PJSIP_SC_TSX_TIMEOUT ||
1835 PJSIP_SC_TSX_TRANSPORT_ERROR)
1836 {
1837 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
1838 }
Benny Prijono268ca612006-02-07 12:34:11 +00001839 }
1840}
1841
Benny Prijono8ad55352006-02-08 11:16:05 +00001842/*
1843 * State INCOMING is after we received the request, but before
1844 * responses with tag are sent.
1845 */
1846static void inv_on_state_incoming( pjsip_inv_session *inv, pjsip_event *e)
Benny Prijono268ca612006-02-07 12:34:11 +00001847{
1848 pjsip_transaction *tsx = e->body.tsx_state.tsx;
1849 pjsip_dialog *dlg = pjsip_tsx_get_dlg(tsx);
1850
1851 PJ_ASSERT_ON_FAIL(tsx && dlg, return);
1852
Benny Prijono8ad55352006-02-08 11:16:05 +00001853 if (tsx == inv->invite_tsx) {
1854
1855 /*
1856 * Handle the INVITE state transition.
1857 */
1858
Benny Prijono268ca612006-02-07 12:34:11 +00001859 switch (tsx->state) {
Benny Prijono8ad55352006-02-08 11:16:05 +00001860
Benny Prijono64f851e2006-02-23 13:49:28 +00001861 case PJSIP_TSX_STATE_TRYING:
1862 inv_set_state(inv, PJSIP_INV_STATE_INCOMING, e);
1863 break;
1864
Benny Prijono268ca612006-02-07 12:34:11 +00001865 case PJSIP_TSX_STATE_PROCEEDING:
Benny Prijono8ad55352006-02-08 11:16:05 +00001866 /*
1867 * Transaction sent provisional response.
1868 */
Benny Prijono268ca612006-02-07 12:34:11 +00001869 if (tsx->status_code > 100)
Benny Prijono8ad55352006-02-08 11:16:05 +00001870 inv_set_state(inv, PJSIP_INV_STATE_EARLY, e);
Benny Prijono268ca612006-02-07 12:34:11 +00001871 break;
Benny Prijono8ad55352006-02-08 11:16:05 +00001872
Benny Prijono268ca612006-02-07 12:34:11 +00001873 case PJSIP_TSX_STATE_COMPLETED:
Benny Prijono8ad55352006-02-08 11:16:05 +00001874 /*
1875 * Transaction sent final response.
1876 */
Benny Prijono268ca612006-02-07 12:34:11 +00001877 if (tsx->status_code/100 == 2)
Benny Prijono8ad55352006-02-08 11:16:05 +00001878 inv_set_state(inv, PJSIP_INV_STATE_CONNECTING, e);
Benny Prijono268ca612006-02-07 12:34:11 +00001879 else
Benny Prijono8ad55352006-02-08 11:16:05 +00001880 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
Benny Prijono268ca612006-02-07 12:34:11 +00001881 break;
Benny Prijono8ad55352006-02-08 11:16:05 +00001882
Benny Prijono268ca612006-02-07 12:34:11 +00001883 case PJSIP_TSX_STATE_TERMINATED:
Benny Prijono8ad55352006-02-08 11:16:05 +00001884 /*
1885 * This happens on transport error (e.g. failed to send
1886 * response)
1887 */
1888 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
Benny Prijono268ca612006-02-07 12:34:11 +00001889 break;
Benny Prijono8ad55352006-02-08 11:16:05 +00001890
Benny Prijono268ca612006-02-07 12:34:11 +00001891 default:
Benny Prijono8ad55352006-02-08 11:16:05 +00001892 pj_assert(!"Unexpected INVITE state");
1893 break;
Benny Prijono268ca612006-02-07 12:34:11 +00001894 }
Benny Prijono8ad55352006-02-08 11:16:05 +00001895
1896 } else if (tsx->method.id == PJSIP_CANCEL_METHOD &&
1897 tsx->role == PJSIP_ROLE_UAS &&
1898 tsx->state < PJSIP_TSX_STATE_COMPLETED &&
1899 e->body.tsx_state.type == PJSIP_EVENT_RX_MSG )
1900 {
1901
1902 /*
1903 * Handle incoming CANCEL request.
1904 */
1905
1906 inv_respond_incoming_cancel(inv, tsx, e->body.tsx_state.src.rdata);
1907
Benny Prijono268ca612006-02-07 12:34:11 +00001908 }
1909}
1910
Benny Prijono8ad55352006-02-08 11:16:05 +00001911/*
1912 * State EARLY is for both UAS and UAC, after response with To tag
1913 * is sent/received.
1914 */
1915static void inv_on_state_early( pjsip_inv_session *inv, pjsip_event *e)
Benny Prijono268ca612006-02-07 12:34:11 +00001916{
1917 pjsip_transaction *tsx = e->body.tsx_state.tsx;
1918 pjsip_dialog *dlg = pjsip_tsx_get_dlg(tsx);
1919
1920 PJ_ASSERT_ON_FAIL(tsx && dlg, return);
1921
Benny Prijono8ad55352006-02-08 11:16:05 +00001922 if (tsx == inv->invite_tsx) {
1923
1924 /*
1925 * Handle the INVITE state progress.
1926 */
Benny Prijono268ca612006-02-07 12:34:11 +00001927
1928 switch (tsx->state) {
1929
1930 case PJSIP_TSX_STATE_PROCEEDING:
1931 /* Send/received another provisional response. */
Benny Prijono8ad55352006-02-08 11:16:05 +00001932 inv_set_state(inv, PJSIP_INV_STATE_EARLY, e);
Benny Prijonoa66c7152006-02-09 01:26:14 +00001933
1934 if (e->body.tsx_state.type == PJSIP_EVENT_RX_MSG) {
1935 inv_check_sdp_in_incoming_msg(inv, tsx,
1936 e->body.tsx_state.src.rdata);
1937 }
Benny Prijono268ca612006-02-07 12:34:11 +00001938 break;
1939
1940 case PJSIP_TSX_STATE_COMPLETED:
Benny Prijonoa66c7152006-02-09 01:26:14 +00001941 if (tsx->status_code/100 == 2) {
Benny Prijono8ad55352006-02-08 11:16:05 +00001942 inv_set_state(inv, PJSIP_INV_STATE_CONNECTING, e);
Benny Prijonoa66c7152006-02-09 01:26:14 +00001943 if (e->body.tsx_state.type == PJSIP_EVENT_RX_MSG) {
1944 inv_check_sdp_in_incoming_msg(inv, tsx,
1945 e->body.tsx_state.src.rdata);
1946 }
1947
1948 } else
Benny Prijono8ad55352006-02-08 11:16:05 +00001949 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
Benny Prijono268ca612006-02-07 12:34:11 +00001950 break;
1951
Benny Prijonof3195072006-02-14 21:15:30 +00001952 case PJSIP_TSX_STATE_CONFIRMED:
1953 /* For some reason can go here */
1954
Benny Prijono268ca612006-02-07 12:34:11 +00001955 case PJSIP_TSX_STATE_TERMINATED:
1956 /* INVITE transaction can be terminated either because UAC
1957 * transaction received 2xx response or because of transport
1958 * error.
1959 */
1960 if (tsx->status_code/100 == 2) {
1961
1962 /* This must be receipt of 2xx response */
1963
1964 /* Set state to CONNECTING */
Benny Prijono8ad55352006-02-08 11:16:05 +00001965 inv_set_state(inv, PJSIP_INV_STATE_CONNECTING, e);
Benny Prijono268ca612006-02-07 12:34:11 +00001966
Benny Prijonoa66c7152006-02-09 01:26:14 +00001967 if (e->body.tsx_state.type == PJSIP_EVENT_RX_MSG) {
1968 inv_check_sdp_in_incoming_msg(inv, tsx,
1969 e->body.tsx_state.src.rdata);
1970 }
1971
Benny Prijono268ca612006-02-07 12:34:11 +00001972 /* if UAC, send ACK and move state to confirmed. */
1973 if (tsx->role == PJSIP_ROLE_UAC) {
1974 pj_assert(e->body.tsx_state.type == PJSIP_EVENT_RX_MSG);
1975
Benny Prijono8ad55352006-02-08 11:16:05 +00001976 inv_send_ack(inv, e->body.tsx_state.src.rdata);
Benny Prijono8ad55352006-02-08 11:16:05 +00001977 inv_set_state(inv, PJSIP_INV_STATE_CONFIRMED, e);
Benny Prijono268ca612006-02-07 12:34:11 +00001978 }
1979
1980 } else {
Benny Prijono8ad55352006-02-08 11:16:05 +00001981 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
Benny Prijono268ca612006-02-07 12:34:11 +00001982 }
1983 break;
1984
1985 default:
Benny Prijono8ad55352006-02-08 11:16:05 +00001986 pj_assert(!"Unexpected INVITE tsx state");
Benny Prijono268ca612006-02-07 12:34:11 +00001987 }
1988
Benny Prijono8ad55352006-02-08 11:16:05 +00001989 } else if (inv->role == PJSIP_ROLE_UAS &&
1990 tsx->role == PJSIP_ROLE_UAS &&
1991 tsx->method.id == PJSIP_CANCEL_METHOD &&
1992 tsx->state < PJSIP_TSX_STATE_COMPLETED &&
1993 e->body.tsx_state.type == PJSIP_EVENT_RX_MSG )
1994 {
Benny Prijono268ca612006-02-07 12:34:11 +00001995
Benny Prijono8ad55352006-02-08 11:16:05 +00001996 /*
1997 * Handle incoming CANCEL request.
Benny Prijonoccf95622006-02-07 18:48:01 +00001998 */
1999
Benny Prijono8ad55352006-02-08 11:16:05 +00002000 inv_respond_incoming_cancel(inv, tsx, e->body.tsx_state.src.rdata);
2001
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00002002 } else if (inv->role == PJSIP_ROLE_UAC &&
2003 tsx->role == PJSIP_ROLE_UAC &&
2004 tsx->method.id == PJSIP_CANCEL_METHOD)
2005 {
2006 /*
2007 * Handle case when outgoing CANCEL is answered with 481 (Call/
2008 * Transaction Does Not Exist), 408, or when it's timed out. In these
2009 * cases, disconnect session (i.e. dialog usage only).
2010 */
2011 if (tsx->status_code == PJSIP_SC_CALL_TSX_DOES_NOT_EXIST ||
2012 tsx->status_code == PJSIP_SC_REQUEST_TIMEOUT ||
2013 tsx->status_code == PJSIP_SC_TSX_TIMEOUT ||
2014 PJSIP_SC_TSX_TRANSPORT_ERROR)
2015 {
2016 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
2017 }
Benny Prijono268ca612006-02-07 12:34:11 +00002018 }
2019}
2020
Benny Prijono8ad55352006-02-08 11:16:05 +00002021/*
2022 * State CONNECTING is after 2xx response to INVITE is sent/received.
2023 */
2024static void inv_on_state_connecting( pjsip_inv_session *inv, pjsip_event *e)
Benny Prijono268ca612006-02-07 12:34:11 +00002025{
2026 pjsip_transaction *tsx = e->body.tsx_state.tsx;
2027 pjsip_dialog *dlg = pjsip_tsx_get_dlg(tsx);
2028
2029 PJ_ASSERT_ON_FAIL(tsx && dlg, return);
2030
Benny Prijono8ad55352006-02-08 11:16:05 +00002031 if (tsx == inv->invite_tsx) {
Benny Prijono268ca612006-02-07 12:34:11 +00002032
Benny Prijono8ad55352006-02-08 11:16:05 +00002033 /*
2034 * Handle INVITE state progression.
2035 */
Benny Prijono268ca612006-02-07 12:34:11 +00002036 switch (tsx->state) {
2037
2038 case PJSIP_TSX_STATE_CONFIRMED:
Benny Prijono38998232006-02-08 22:44:25 +00002039 if (tsx->status_code/100 == 2)
2040 inv_set_state(inv, PJSIP_INV_STATE_CONFIRMED, e);
Benny Prijono268ca612006-02-07 12:34:11 +00002041 break;
2042
2043 case PJSIP_TSX_STATE_TERMINATED:
2044 /* INVITE transaction can be terminated either because UAC
2045 * transaction received 2xx response or because of transport
2046 * error.
2047 */
2048 if (tsx->status_code/100 != 2) {
Benny Prijono8ad55352006-02-08 11:16:05 +00002049 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
Benny Prijono268ca612006-02-07 12:34:11 +00002050 }
2051 break;
2052
2053 case PJSIP_TSX_STATE_DESTROYED:
2054 /* Do nothing. */
2055 break;
2056
2057 default:
2058 pj_assert(!"Unexpected state");
2059 }
2060
Benny Prijono8ad55352006-02-08 11:16:05 +00002061 } else if (tsx->role == PJSIP_ROLE_UAS &&
2062 tsx->method.id == PJSIP_BYE_METHOD &&
2063 tsx->status_code < 200 &&
2064 e->body.tsx_state.type == PJSIP_EVENT_RX_MSG)
2065 {
2066
2067 /*
2068 * Handle incoming BYE.
2069 */
2070
2071 inv_respond_incoming_bye( inv, tsx, e->body.tsx_state.src.rdata, e );
2072
Benny Prijono38998232006-02-08 22:44:25 +00002073 } else if (tsx->method.id == PJSIP_BYE_METHOD &&
2074 tsx->role == PJSIP_ROLE_UAC &&
Benny Prijonoa66c7152006-02-09 01:26:14 +00002075 (tsx->state == PJSIP_TSX_STATE_COMPLETED ||
2076 tsx->state == PJSIP_TSX_STATE_TERMINATED))
Benny Prijono38998232006-02-08 22:44:25 +00002077 {
2078
2079 /*
2080 * Outgoing BYE
2081 */
2082 inv_handle_bye_response( inv, tsx, e->body.tsx_state.src.rdata, e);
2083
Benny Prijono268ca612006-02-07 12:34:11 +00002084 }
Benny Prijono38998232006-02-08 22:44:25 +00002085
Benny Prijono268ca612006-02-07 12:34:11 +00002086}
2087
Benny Prijono8ad55352006-02-08 11:16:05 +00002088/*
2089 * State CONFIRMED is after ACK is sent/received.
2090 */
2091static void inv_on_state_confirmed( pjsip_inv_session *inv, pjsip_event *e)
Benny Prijono268ca612006-02-07 12:34:11 +00002092{
2093 pjsip_transaction *tsx = e->body.tsx_state.tsx;
2094 pjsip_dialog *dlg = pjsip_tsx_get_dlg(tsx);
2095
2096 PJ_ASSERT_ON_FAIL(tsx && dlg, return);
2097
Benny Prijono268ca612006-02-07 12:34:11 +00002098
Benny Prijono8ad55352006-02-08 11:16:05 +00002099 if (tsx->method.id == PJSIP_BYE_METHOD &&
2100 tsx->role == PJSIP_ROLE_UAC &&
Benny Prijonoa66c7152006-02-09 01:26:14 +00002101 (tsx->state == PJSIP_TSX_STATE_COMPLETED ||
2102 tsx->state == PJSIP_TSX_STATE_TERMINATED))
Benny Prijono8ad55352006-02-08 11:16:05 +00002103 {
Benny Prijono38998232006-02-08 22:44:25 +00002104
Benny Prijono8ad55352006-02-08 11:16:05 +00002105 /*
Benny Prijono38998232006-02-08 22:44:25 +00002106 * Outgoing BYE
Benny Prijono8ad55352006-02-08 11:16:05 +00002107 */
Benny Prijono8ad55352006-02-08 11:16:05 +00002108
Benny Prijonoa66c7152006-02-09 01:26:14 +00002109 inv_handle_bye_response( inv, tsx, e->body.tsx_state.src.rdata, e);
Benny Prijono268ca612006-02-07 12:34:11 +00002110
Benny Prijono8ad55352006-02-08 11:16:05 +00002111 }
2112 else if (tsx->method.id == PJSIP_BYE_METHOD &&
2113 tsx->role == PJSIP_ROLE_UAS &&
2114 tsx->status_code < 200 &&
2115 e->body.tsx_state.type == PJSIP_EVENT_RX_MSG)
2116 {
Benny Prijonoccf95622006-02-07 18:48:01 +00002117
Benny Prijono8ad55352006-02-08 11:16:05 +00002118 /*
2119 * Handle incoming BYE.
2120 */
Benny Prijono268ca612006-02-07 12:34:11 +00002121
Benny Prijono8ad55352006-02-08 11:16:05 +00002122 inv_respond_incoming_bye( inv, tsx, e->body.tsx_state.src.rdata, e );
2123
Benny Prijono268ca612006-02-07 12:34:11 +00002124 }
Benny Prijono26ff9062006-02-21 23:47:00 +00002125 else if (tsx->method.id == PJSIP_INVITE_METHOD &&
2126 tsx->role == PJSIP_ROLE_UAS)
2127 {
2128
2129 /*
2130 * Handle incoming re-INVITE
2131 */
2132 if (tsx->state == PJSIP_TSX_STATE_TRYING) {
2133
2134 pjsip_rx_data *rdata = e->body.tsx_state.src.rdata;
2135 pjsip_tx_data *tdata;
2136 pj_status_t status;
2137
2138 /* Check if we have INVITE pending. */
2139 if (inv->invite_tsx && inv->invite_tsx!=tsx) {
2140
2141 /* Can not receive re-INVITE while another one is pending. */
2142 status = pjsip_dlg_create_response( inv->dlg, rdata, 500, NULL,
2143 &tdata);
2144 if (status != PJ_SUCCESS)
2145 return;
2146
2147 status = pjsip_dlg_send_response( inv->dlg, tsx, tdata);
2148
2149
2150 return;
2151 }
2152
2153 /* Save the invite transaction. */
2154 inv->invite_tsx = tsx;
2155
2156 /* Process SDP in incoming message. */
2157 status = inv_check_sdp_in_incoming_msg(inv, tsx, rdata);
2158
2159 if (status != PJ_SUCCESS) {
2160
2161 /* Not Acceptable */
2162 const pjsip_hdr *accept;
2163
2164 status = pjsip_dlg_create_response(inv->dlg, rdata,
2165 488, NULL, &tdata);
2166 if (status != PJ_SUCCESS)
2167 return;
2168
2169
2170 accept = pjsip_endpt_get_capability(dlg->endpt, PJSIP_H_ACCEPT,
2171 NULL);
2172 if (accept) {
2173 pjsip_msg_add_hdr(tdata->msg,
2174 pjsip_hdr_clone(tdata->pool, accept));
2175 }
2176
2177 status = pjsip_dlg_send_response(dlg, tsx, tdata);
2178
2179 return;
2180 }
2181
2182 /* Create 2xx ANSWER */
2183 status = pjsip_dlg_create_response(dlg, rdata, 200, NULL, &tdata);
2184 if (status != PJ_SUCCESS)
2185 return;
2186
2187 /* Process SDP in the answer */
Benny Prijono64f851e2006-02-23 13:49:28 +00002188 status = process_answer(inv, 200, tdata, NULL);
Benny Prijono26ff9062006-02-21 23:47:00 +00002189 if (status != PJ_SUCCESS)
2190 return;
2191
2192 status = pjsip_inv_send_msg(inv, tdata, NULL);
2193
2194 }
2195
2196 }
2197 else if (tsx->method.id == PJSIP_INVITE_METHOD &&
2198 tsx->role == PJSIP_ROLE_UAC)
2199 {
2200 /*
2201 * Handle outgoing re-INVITE
2202 */
2203 if (tsx->state == PJSIP_TSX_STATE_TERMINATED &&
2204 tsx->status_code/100 == 2)
2205 {
2206
2207 /* Re-INVITE was accepted. */
2208
2209 /* Process SDP */
2210 inv_check_sdp_in_incoming_msg(inv, tsx,
2211 e->body.tsx_state.src.rdata);
2212
2213 /* Send ACK */
2214 inv_send_ack(inv, e->body.tsx_state.src.rdata);
2215
2216 } else if (tsx->state == PJSIP_TSX_STATE_COMPLETED &&
2217 (tsx->status_code==401 || tsx->status_code==407))
2218 {
2219 pjsip_tx_data *tdata;
2220 pj_status_t status;
2221
2222 /* Handle authentication challenge. */
2223 status = pjsip_auth_clt_reinit_req( &dlg->auth_sess,
2224 e->body.tsx_state.src.rdata,
2225 tsx->last_tx,
2226 &tdata);
2227 if (status != PJ_SUCCESS)
2228 return;
2229
2230 /* Send re-INVITE */
2231 status = pjsip_inv_send_msg( inv, tdata, NULL);
2232
2233 } else if (tsx->status_code==PJSIP_SC_CALL_TSX_DOES_NOT_EXIST ||
2234 tsx->status_code==PJSIP_SC_REQUEST_TIMEOUT ||
2235 tsx->status_code >= 700)
2236 {
2237 /*
2238 * Handle responses that terminates dialog.
2239 */
2240 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
2241 }
2242 }
Benny Prijono268ca612006-02-07 12:34:11 +00002243}
2244
Benny Prijono8ad55352006-02-08 11:16:05 +00002245/*
2246 * After session has been terminated, but before dialog is destroyed
2247 * (because dialog has other usages, or because dialog is waiting for
2248 * the last transaction to terminate).
2249 */
2250static void inv_on_state_disconnected( pjsip_inv_session *inv, pjsip_event *e)
Benny Prijono268ca612006-02-07 12:34:11 +00002251{
Benny Prijono8ad55352006-02-08 11:16:05 +00002252 pjsip_transaction *tsx = e->body.tsx_state.tsx;
2253 pjsip_dialog *dlg = pjsip_tsx_get_dlg(tsx);
Benny Prijono268ca612006-02-07 12:34:11 +00002254
Benny Prijono8ad55352006-02-08 11:16:05 +00002255 PJ_ASSERT_ON_FAIL(tsx && dlg, return);
2256
2257 if (tsx->method.id == PJSIP_BYE_METHOD &&
2258 tsx->role == PJSIP_ROLE_UAS &&
2259 tsx->status_code < 200 &&
2260 e->body.tsx_state.type == PJSIP_EVENT_RX_MSG)
2261 {
2262
2263 /*
2264 * Be nice, handle incoming BYE.
2265 */
2266
2267 inv_respond_incoming_bye( inv, tsx, e->body.tsx_state.src.rdata, e );
2268
2269 }
Benny Prijono268ca612006-02-07 12:34:11 +00002270}
2271