blob: c775d0195afa901e38d21423f9bd2e6979a6428e [file] [log] [blame]
Benny Prijono268ca612006-02-07 12:34:11 +00001/* $Id$ */
2/*
Benny Prijono32177c02008-06-20 22:44:47 +00003 * Copyright (C) 2003-2008 Benny Prijono <benny@prijono.org>
Benny Prijono268ca612006-02-07 12:34:11 +00004 *
5 * This program is free software; you can redistribute it and/or modify
6 * it under the terms of the GNU General Public License as published by
7 * the Free Software Foundation; either version 2 of the License, or
8 * (at your option) any later version.
9 *
10 * This program is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 * GNU General Public License for more details.
14 *
15 * You should have received a copy of the GNU General Public License
16 * along with this program; if not, write to the Free Software
17 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
18 */
19#include <pjsip-ua/sip_inv.h>
Benny Prijonodcfc0ba2007-09-30 16:50:27 +000020#include <pjsip-ua/sip_100rel.h>
Benny Prijono268ca612006-02-07 12:34:11 +000021#include <pjsip/sip_module.h>
22#include <pjsip/sip_endpoint.h>
23#include <pjsip/sip_event.h>
24#include <pjsip/sip_transaction.h>
25#include <pjmedia/sdp.h>
26#include <pjmedia/sdp_neg.h>
Benny Prijono95196582006-02-09 00:13:40 +000027#include <pjmedia/errno.h>
Benny Prijono268ca612006-02-07 12:34:11 +000028#include <pj/string.h>
29#include <pj/pool.h>
30#include <pj/assert.h>
Benny Prijono8ad55352006-02-08 11:16:05 +000031#include <pj/os.h>
Benny Prijonoa66c7152006-02-09 01:26:14 +000032#include <pj/log.h>
33
Benny Prijono1f7767b2007-10-03 18:28:49 +000034/*
35 * Note on offer/answer:
36 *
37 * The offer/answer framework in this implementation assumes the occurence
38 * of SDP in a particular request/response according to this table:
39
40 offer answer Note:
41 ========================================================================
42 INVITE X INVITE may contain offer
43 18x/INVITE X X Response may contain offer or answer
44 2xx/INVITE X X Response may contain offer or answer
45 ACK X ACK may contain answer
46
47 PRACK X PRACK can only contain answer
48 2xx/PRACK Response may not have offer nor answer
49
50 UPDATE X UPDATE may only contain offer
51 2xx/UPDATE X Response may only contain answer
52 ========================================================================
53
54 *
55 */
Benny Prijono268ca612006-02-07 12:34:11 +000056
Benny Prijonodcfc0ba2007-09-30 16:50:27 +000057#define THIS_FILE "sip_inv.c"
Benny Prijono268ca612006-02-07 12:34:11 +000058
Benny Prijonoe8b0d3b2006-03-17 17:57:52 +000059static const char *inv_state_names[] =
60{
Benny Prijono4be63b52006-11-25 14:50:25 +000061 "NULL",
62 "CALLING",
63 "INCOMING",
64 "EARLY",
Benny Prijonoe8b0d3b2006-03-17 17:57:52 +000065 "CONNECTING",
Benny Prijonoc5055702007-01-13 23:20:18 +000066 "CONFIRMED",
Benny Prijonoe8b0d3b2006-03-17 17:57:52 +000067 "DISCONNCTD",
68 "TERMINATED",
69};
70
Benny Prijono1f7767b2007-10-03 18:28:49 +000071/* UPDATE method */
72const pjsip_method pjsip_update_method =
73{
74 PJSIP_OTHER_METHOD,
75 { "UPDATE", 6 }
76};
77
Benny Prijono268ca612006-02-07 12:34:11 +000078/*
79 * Static prototypes.
80 */
81static pj_status_t mod_inv_load(pjsip_endpoint *endpt);
82static pj_status_t mod_inv_unload(void);
83static pj_bool_t mod_inv_on_rx_request(pjsip_rx_data *rdata);
84static pj_bool_t mod_inv_on_rx_response(pjsip_rx_data *rdata);
85static void mod_inv_on_tsx_state(pjsip_transaction*, pjsip_event*);
86
Benny Prijono8ad55352006-02-08 11:16:05 +000087static void inv_on_state_null( pjsip_inv_session *inv, pjsip_event *e);
88static void inv_on_state_calling( pjsip_inv_session *inv, pjsip_event *e);
89static void inv_on_state_incoming( pjsip_inv_session *inv, pjsip_event *e);
90static void inv_on_state_early( pjsip_inv_session *inv, pjsip_event *e);
91static void inv_on_state_connecting( pjsip_inv_session *inv, pjsip_event *e);
92static void inv_on_state_confirmed( pjsip_inv_session *inv, pjsip_event *e);
93static void inv_on_state_disconnected( pjsip_inv_session *inv, pjsip_event *e);
Benny Prijono268ca612006-02-07 12:34:11 +000094
Benny Prijono7d910092007-06-20 04:19:46 +000095static pj_status_t inv_check_sdp_in_incoming_msg( pjsip_inv_session *inv,
96 pjsip_transaction *tsx,
97 pjsip_rx_data *rdata);
Benny Prijono1f7767b2007-10-03 18:28:49 +000098static pj_status_t inv_negotiate_sdp( pjsip_inv_session *inv );
99static pjsip_msg_body *create_sdp_body(pj_pool_t *pool,
100 const pjmedia_sdp_session *c_sdp);
Benny Prijono7d910092007-06-20 04:19:46 +0000101static pj_status_t process_answer( pjsip_inv_session *inv,
102 int st_code,
103 pjsip_tx_data *tdata,
104 const pjmedia_sdp_session *local_sdp);
105
Benny Prijono8ad55352006-02-08 11:16:05 +0000106static void (*inv_state_handler[])( pjsip_inv_session *inv, pjsip_event *e) =
Benny Prijono268ca612006-02-07 12:34:11 +0000107{
108 &inv_on_state_null,
109 &inv_on_state_calling,
110 &inv_on_state_incoming,
111 &inv_on_state_early,
112 &inv_on_state_connecting,
113 &inv_on_state_confirmed,
114 &inv_on_state_disconnected,
Benny Prijono268ca612006-02-07 12:34:11 +0000115};
116
117static struct mod_inv
118{
119 pjsip_module mod;
120 pjsip_endpoint *endpt;
121 pjsip_inv_callback cb;
Benny Prijono268ca612006-02-07 12:34:11 +0000122} mod_inv =
123{
124 {
Benny Prijono2f8992b2006-02-25 21:16:36 +0000125 NULL, NULL, /* prev, next. */
126 { "mod-invite", 10 }, /* Name. */
127 -1, /* Id */
128 PJSIP_MOD_PRIORITY_DIALOG_USAGE, /* Priority */
129 &mod_inv_load, /* load() */
130 NULL, /* start() */
131 NULL, /* stop() */
132 &mod_inv_unload, /* unload() */
133 &mod_inv_on_rx_request, /* on_rx_request() */
134 &mod_inv_on_rx_response, /* on_rx_response() */
135 NULL, /* on_tx_request. */
136 NULL, /* on_tx_response() */
137 &mod_inv_on_tsx_state, /* on_tsx_state() */
Benny Prijono268ca612006-02-07 12:34:11 +0000138 }
139};
140
141
Benny Prijonoa66c7152006-02-09 01:26:14 +0000142/* Invite session data to be attached to transaction. */
143struct tsx_inv_data
144{
145 pjsip_inv_session *inv;
146 pj_bool_t sdp_done;
147};
148
Benny Prijono8ad55352006-02-08 11:16:05 +0000149/*
150 * Module load()
151 */
Benny Prijono268ca612006-02-07 12:34:11 +0000152static pj_status_t mod_inv_load(pjsip_endpoint *endpt)
153{
Benny Prijono1f7767b2007-10-03 18:28:49 +0000154 pj_str_t allowed[] = {{"INVITE", 6}, {"ACK",3}, {"BYE",3}, {"CANCEL",6},
155 { "UPDATE", 6}};
Benny Prijono56315612006-07-18 14:39:40 +0000156 pj_str_t accepted = { "application/sdp", 15 };
Benny Prijono268ca612006-02-07 12:34:11 +0000157
Benny Prijono1f7767b2007-10-03 18:28:49 +0000158 /* Register supported methods: INVITE, ACK, BYE, CANCEL, UPDATE */
Benny Prijono268ca612006-02-07 12:34:11 +0000159 pjsip_endpt_add_capability(endpt, &mod_inv.mod, PJSIP_H_ALLOW, NULL,
160 PJ_ARRAY_SIZE(allowed), allowed);
161
Benny Prijono56315612006-07-18 14:39:40 +0000162 /* Register "application/sdp" in Accept header */
163 pjsip_endpt_add_capability(endpt, &mod_inv.mod, PJSIP_H_ACCEPT, NULL,
164 1, &accepted);
165
Benny Prijono268ca612006-02-07 12:34:11 +0000166 return PJ_SUCCESS;
167}
168
Benny Prijono8ad55352006-02-08 11:16:05 +0000169/*
170 * Module unload()
171 */
Benny Prijono268ca612006-02-07 12:34:11 +0000172static pj_status_t mod_inv_unload(void)
173{
174 /* Should remove capability here */
175 return PJ_SUCCESS;
176}
177
Benny Prijono8ad55352006-02-08 11:16:05 +0000178/*
Benny Prijono38998232006-02-08 22:44:25 +0000179 * Set session state.
180 */
181void inv_set_state(pjsip_inv_session *inv, pjsip_inv_state state,
182 pjsip_event *e)
183{
Benny Prijonod4e0abd2006-03-05 11:53:36 +0000184 pjsip_inv_state prev_state = inv->state;
Benny Prijono7d910092007-06-20 04:19:46 +0000185 pj_status_t status;
186
187
188 /* If state is confirmed, check that SDP negotiation is done,
189 * otherwise disconnect the session.
190 */
191 if (state == PJSIP_INV_STATE_CONFIRMED) {
192 if (pjmedia_sdp_neg_get_state(inv->neg)!=PJMEDIA_SDP_NEG_STATE_DONE) {
193 pjsip_tx_data *bye;
194
195 PJ_LOG(4,(inv->obj_name, "SDP offer/answer incomplete, ending the "
196 "session"));
197
198 status = pjsip_inv_end_session(inv, PJSIP_SC_NOT_ACCEPTABLE,
199 NULL, &bye);
200 if (status == PJ_SUCCESS && bye)
201 status = pjsip_inv_send_msg(inv, bye);
202
203 return;
204 }
205 }
Benny Prijonod4e0abd2006-03-05 11:53:36 +0000206
207 /* Set state. */
Benny Prijono38998232006-02-08 22:44:25 +0000208 inv->state = state;
Benny Prijonod4e0abd2006-03-05 11:53:36 +0000209
210 /* If state is DISCONNECTED, cause code MUST have been set. */
211 pj_assert(inv->state != PJSIP_INV_STATE_DISCONNECTED ||
212 inv->cause != 0);
213
214 /* Call on_state_changed() callback. */
215 if (mod_inv.cb.on_state_changed && inv->notify)
Benny Prijono38998232006-02-08 22:44:25 +0000216 (*mod_inv.cb.on_state_changed)(inv, e);
217
Benny Prijonod4e0abd2006-03-05 11:53:36 +0000218 /* Only decrement when previous state is not already DISCONNECTED */
219 if (inv->state == PJSIP_INV_STATE_DISCONNECTED &&
220 prev_state != PJSIP_INV_STATE_DISCONNECTED)
221 {
Benny Prijono1f7767b2007-10-03 18:28:49 +0000222 if (inv->last_ack) {
223 pjsip_tx_data_dec_ref(inv->last_ack);
224 inv->last_ack = NULL;
225 }
226 pjsip_100rel_end_session(inv);
Benny Prijonoe4f2abb2006-02-10 14:04:05 +0000227 pjsip_dlg_dec_session(inv->dlg, &mod_inv.mod);
Benny Prijonod4e0abd2006-03-05 11:53:36 +0000228 }
Benny Prijono38998232006-02-08 22:44:25 +0000229}
230
231
232/*
Benny Prijono0b6340c2006-06-13 22:21:23 +0000233 * Set cause code.
234 */
235void inv_set_cause(pjsip_inv_session *inv, int cause_code,
236 const pj_str_t *cause_text)
237{
238 if (cause_code > inv->cause) {
Benny Prijonoba5926a2007-05-02 11:29:37 +0000239 inv->cause = (pjsip_status_code) cause_code;
Benny Prijono0b6340c2006-06-13 22:21:23 +0000240 if (cause_text)
241 pj_strdup(inv->pool, &inv->cause_text, cause_text);
242 else if (cause_code/100 == 2)
243 inv->cause_text = pj_str("Normal call clearing");
244 else
245 inv->cause_text = *pjsip_get_status_text(cause_code);
246 }
247}
248
249
Benny Prijono1f7767b2007-10-03 18:28:49 +0000250/*
251 * Check if outgoing request needs to have SDP answer.
252 * This applies for both ACK and PRACK requests.
253 */
Benny Prijono9569a0b2007-10-04 15:35:26 +0000254static const pjmedia_sdp_session *inv_has_pending_answer(pjsip_inv_session *inv,
255 pjsip_transaction *tsx)
Benny Prijono1f7767b2007-10-03 18:28:49 +0000256{
257 pjmedia_sdp_neg_state neg_state;
Benny Prijono9569a0b2007-10-04 15:35:26 +0000258 const pjmedia_sdp_session *sdp = NULL;
Benny Prijono1f7767b2007-10-03 18:28:49 +0000259 pj_status_t status;
260
261 /* If SDP negotiator is ready, start negotiation. */
262
263 /* Start nego when appropriate. */
264 neg_state = inv->neg ? pjmedia_sdp_neg_get_state(inv->neg) :
265 PJMEDIA_SDP_NEG_STATE_NULL;
266
267 if (neg_state == PJMEDIA_SDP_NEG_STATE_DONE) {
268
269 /* Nothing to do */
270
271 } else if (neg_state == PJMEDIA_SDP_NEG_STATE_WAIT_NEGO &&
272 pjmedia_sdp_neg_has_local_answer(inv->neg) )
273 {
274 struct tsx_inv_data *tsx_inv_data;
Benny Prijonod5f9f422007-11-25 04:40:07 +0000275 struct tsx_inv_data dummy;
Benny Prijono1f7767b2007-10-03 18:28:49 +0000276
Benny Prijonod5f9f422007-11-25 04:40:07 +0000277 /* Get invite session's transaction data.
278 * Note that tsx may be NULL, for example when application sends
279 * delayed ACK request (at this time, the original INVITE
280 * transaction may have been destroyed.
281 */
282 if (tsx) {
283 tsx_inv_data = (struct tsx_inv_data*)tsx->mod_data[mod_inv.mod.id];
284 } else {
285 tsx_inv_data = &dummy;
286 pj_bzero(&dummy, sizeof(dummy));
287 dummy.inv = inv;
288 }
Benny Prijono1f7767b2007-10-03 18:28:49 +0000289
290 status = inv_negotiate_sdp(inv);
291 if (status != PJ_SUCCESS)
292 return NULL;
293
294 /* Mark this transaction has having SDP offer/answer done. */
295 tsx_inv_data->sdp_done = 1;
296
297 status = pjmedia_sdp_neg_get_active_local(inv->neg, &sdp);
298
299 } else {
300 /* This remark is only valid for ACK.
301 PJ_LOG(4,(inv->dlg->obj_name,
302 "FYI, the SDP negotiator state (%s) is in a mess "
303 "when sending this ACK/PRACK request",
304 pjmedia_sdp_neg_state_str(neg_state)));
305 */
306 }
307
308 return sdp;
309}
310
Benny Prijono0b6340c2006-06-13 22:21:23 +0000311
312/*
Benny Prijono8ad55352006-02-08 11:16:05 +0000313 * Send ACK for 2xx response.
314 */
Benny Prijonod5f9f422007-11-25 04:40:07 +0000315static pj_status_t inv_send_ack(pjsip_inv_session *inv, pjsip_event *e)
Benny Prijono268ca612006-02-07 12:34:11 +0000316{
Benny Prijonod5f9f422007-11-25 04:40:07 +0000317 pjsip_rx_data *rdata;
Benny Prijono268ca612006-02-07 12:34:11 +0000318 pj_status_t status;
319
Benny Prijonod5f9f422007-11-25 04:40:07 +0000320 if (e->type == PJSIP_EVENT_TSX_STATE)
321 rdata = e->body.tsx_state.src.rdata;
322 else if (e->type == PJSIP_EVENT_RX_MSG)
323 rdata = e->body.rx_msg.rdata;
324 else {
325 pj_assert(!"Unsupported event type");
326 return PJ_EBUG;
327 }
328
Benny Prijonoe4f2abb2006-02-10 14:04:05 +0000329 PJ_LOG(5,(inv->obj_name, "Received %s, sending ACK",
330 pjsip_rx_data_get_info(rdata)));
331
Benny Prijono1f7767b2007-10-03 18:28:49 +0000332 /* Check if we have cached ACK request */
333 if (inv->last_ack && rdata->msg_info.cseq->cseq == inv->last_ack_cseq) {
Benny Prijonod5f9f422007-11-25 04:40:07 +0000334
Benny Prijono1f7767b2007-10-03 18:28:49 +0000335 pjsip_tx_data_add_ref(inv->last_ack);
Benny Prijonod5f9f422007-11-25 04:40:07 +0000336
337 } else if (mod_inv.cb.on_send_ack) {
338 /* If application handles ACK transmission manually, just notify the
339 * callback
340 */
341 PJ_LOG(5,(inv->obj_name, "Received %s, notifying application callback",
342 pjsip_rx_data_get_info(rdata)));
343
344 (*mod_inv.cb.on_send_ack)(inv, rdata);
345 return PJ_SUCCESS;
346
Benny Prijono1f7767b2007-10-03 18:28:49 +0000347 } else {
Benny Prijonod5f9f422007-11-25 04:40:07 +0000348 status = pjsip_inv_create_ack(inv, rdata->msg_info.cseq->cseq,
349 &inv->last_ack);
Benny Prijono268ca612006-02-07 12:34:11 +0000350 }
351
Benny Prijono1f7767b2007-10-03 18:28:49 +0000352 /* Send ACK */
353 status = pjsip_dlg_send_request(inv->dlg, inv->last_ack, -1, NULL);
Benny Prijono268ca612006-02-07 12:34:11 +0000354 if (status != PJ_SUCCESS) {
355 /* Better luck next time */
356 pj_assert(!"Unable to send ACK!");
357 return status;
358 }
359
Benny Prijonod5f9f422007-11-25 04:40:07 +0000360
Benny Prijonoe6da48a2008-09-22 14:36:00 +0000361 /* Set state to CONFIRMED (if we're not in CONFIRMED yet).
362 * But don't set it to CONFIRMED if we're already DISCONNECTED
363 * (this may have been a late 200/OK response.
364 */
365 if (inv->state < PJSIP_INV_STATE_CONFIRMED) {
Benny Prijonod5f9f422007-11-25 04:40:07 +0000366 inv_set_state(inv, PJSIP_INV_STATE_CONFIRMED, e);
367 }
368
Benny Prijono268ca612006-02-07 12:34:11 +0000369 return PJ_SUCCESS;
370}
371
Benny Prijono8ad55352006-02-08 11:16:05 +0000372/*
373 * Module on_rx_request()
374 *
375 * This callback is called for these events:
376 * - endpoint receives request which was unhandled by higher priority
377 * modules (e.g. transaction layer, dialog layer).
378 * - dialog distributes incoming request to its usages.
379 */
380static pj_bool_t mod_inv_on_rx_request(pjsip_rx_data *rdata)
381{
382 pjsip_method *method;
Benny Prijono38998232006-02-08 22:44:25 +0000383 pjsip_dialog *dlg;
384 pjsip_inv_session *inv;
Benny Prijono8ad55352006-02-08 11:16:05 +0000385
386 /* Only wants to receive request from a dialog. */
Benny Prijono38998232006-02-08 22:44:25 +0000387 dlg = pjsip_rdata_get_dlg(rdata);
388 if (dlg == NULL)
Benny Prijono8ad55352006-02-08 11:16:05 +0000389 return PJ_FALSE;
390
Benny Prijonoa1e69682007-05-11 15:14:34 +0000391 inv = (pjsip_inv_session*) dlg->mod_data[mod_inv.mod.id];
Benny Prijono38998232006-02-08 22:44:25 +0000392
Benny Prijono8ad55352006-02-08 11:16:05 +0000393 /* Report to dialog that we handle INVITE, CANCEL, BYE, ACK.
394 * If we need to send response, it will be sent in the state
395 * handlers.
396 */
397 method = &rdata->msg_info.msg->line.req.method;
398
Benny Prijono70127222006-07-02 14:53:05 +0000399 if (method->id == PJSIP_INVITE_METHOD) {
400 return PJ_TRUE;
401 }
402
403 /* BYE and CANCEL must have existing invite session */
404 if (method->id == PJSIP_BYE_METHOD ||
405 method->id == PJSIP_CANCEL_METHOD)
Benny Prijono8ad55352006-02-08 11:16:05 +0000406 {
Benny Prijono70127222006-07-02 14:53:05 +0000407 if (inv == NULL)
408 return PJ_FALSE;
409
Benny Prijono8ad55352006-02-08 11:16:05 +0000410 return PJ_TRUE;
411 }
412
Benny Prijono38998232006-02-08 22:44:25 +0000413 /* On receipt ACK request, when state is CONNECTING,
414 * move state to CONFIRMED.
415 */
Benny Prijono26ff9062006-02-21 23:47:00 +0000416 if (method->id == PJSIP_ACK_METHOD && inv) {
Benny Prijono38998232006-02-08 22:44:25 +0000417
Benny Prijonof521eb02006-08-06 23:07:25 +0000418 /* Ignore ACK if pending INVITE transaction has not finished. */
419 if (inv->invite_tsx &&
420 inv->invite_tsx->state < PJSIP_TSX_STATE_COMPLETED)
421 {
422 return PJ_TRUE;
423 }
424
Benny Prijono5eff0432006-02-09 14:14:21 +0000425 /* Terminate INVITE transaction, if it's still present. */
426 if (inv->invite_tsx &&
427 inv->invite_tsx->state <= PJSIP_TSX_STATE_COMPLETED)
428 {
Benny Prijono7d910092007-06-20 04:19:46 +0000429 /* Before we terminate INVITE transaction, process the SDP
Benny Prijono59e9e952008-09-21 22:55:43 +0000430 * in the ACK request, if any.
431 * Only do this when invite state is not already disconnected
432 * (http://trac.pjsip.org/repos/ticket/640).
Benny Prijono7d910092007-06-20 04:19:46 +0000433 */
Benny Prijono59e9e952008-09-21 22:55:43 +0000434 if (inv->state < PJSIP_INV_STATE_DISCONNECTED) {
435 inv_check_sdp_in_incoming_msg(inv, inv->invite_tsx, rdata);
436 }
Benny Prijono7d910092007-06-20 04:19:46 +0000437
438 /* Now we can terminate the INVITE transaction */
Benny Prijonof521eb02006-08-06 23:07:25 +0000439 pj_assert(inv->invite_tsx->status_code >= 200);
Benny Prijono5eff0432006-02-09 14:14:21 +0000440 pjsip_tsx_terminate(inv->invite_tsx,
441 inv->invite_tsx->status_code);
442 inv->invite_tsx = NULL;
Benny Prijonodcfc0ba2007-09-30 16:50:27 +0000443 if (inv->last_answer) {
444 pjsip_tx_data_dec_ref(inv->last_answer);
445 inv->last_answer = NULL;
446 }
Benny Prijono5eff0432006-02-09 14:14:21 +0000447 }
448
Benny Prijono32ac8fe2006-03-02 21:16:55 +0000449 /* On receipt of ACK, only set state to confirmed when state
450 * is CONNECTING (e.g. we don't want to set the state to confirmed
451 * when we receive ACK retransmission after sending non-2xx!)
452 */
453 if (inv->state == PJSIP_INV_STATE_CONNECTING) {
Benny Prijono26ff9062006-02-21 23:47:00 +0000454 pjsip_event event;
455
456 PJSIP_EVENT_INIT_RX_MSG(event, rdata);
457 inv_set_state(inv, PJSIP_INV_STATE_CONFIRMED, &event);
458 }
Benny Prijono38998232006-02-08 22:44:25 +0000459 }
460
Benny Prijono8ad55352006-02-08 11:16:05 +0000461 return PJ_FALSE;
462}
463
464/*
465 * Module on_rx_response().
466 *
467 * This callback is called for these events:
468 * - dialog distributes incoming 2xx response to INVITE (outside
469 * transaction) to its usages.
470 * - endpoint distributes strayed responses.
471 */
Benny Prijono268ca612006-02-07 12:34:11 +0000472static pj_bool_t mod_inv_on_rx_response(pjsip_rx_data *rdata)
473{
474 pjsip_dialog *dlg;
475 pjsip_inv_session *inv;
476 pjsip_msg *msg = rdata->msg_info.msg;
477
478 dlg = pjsip_rdata_get_dlg(rdata);
479
480 /* Ignore responses outside dialog */
481 if (dlg == NULL)
482 return PJ_FALSE;
483
484 /* Ignore responses not belonging to invite session */
485 inv = pjsip_dlg_get_inv_session(dlg);
486 if (inv == NULL)
487 return PJ_FALSE;
488
489 /* This MAY be retransmission of 2xx response to INVITE.
490 * If it is, we need to send ACK.
491 */
492 if (msg->type == PJSIP_RESPONSE_MSG && msg->line.status.code/100==2 &&
Benny Prijono26ff9062006-02-21 23:47:00 +0000493 rdata->msg_info.cseq->method.id == PJSIP_INVITE_METHOD &&
494 inv->invite_tsx == NULL)
495 {
Benny Prijonod5f9f422007-11-25 04:40:07 +0000496 pjsip_event e;
Benny Prijono268ca612006-02-07 12:34:11 +0000497
Benny Prijonod5f9f422007-11-25 04:40:07 +0000498 PJSIP_EVENT_INIT_RX_MSG(e, rdata);
499 inv_send_ack(inv, &e);
Benny Prijono268ca612006-02-07 12:34:11 +0000500 return PJ_TRUE;
501
502 }
503
504 /* No other processing needs to be done here. */
505 return PJ_FALSE;
506}
507
Benny Prijono8ad55352006-02-08 11:16:05 +0000508/*
509 * Module on_tsx_state()
510 *
511 * This callback is called by dialog framework for all transactions
512 * inside the dialog for all its dialog usages.
513 */
Benny Prijono268ca612006-02-07 12:34:11 +0000514static void mod_inv_on_tsx_state(pjsip_transaction *tsx, pjsip_event *e)
515{
516 pjsip_dialog *dlg;
517 pjsip_inv_session *inv;
518
519 dlg = pjsip_tsx_get_dlg(tsx);
520 if (dlg == NULL)
521 return;
522
523 inv = pjsip_dlg_get_inv_session(dlg);
524 if (inv == NULL)
525 return;
526
527 /* Call state handler for the invite session. */
528 (*inv_state_handler[inv->state])(inv, e);
529
530 /* Call on_tsx_state */
Benny Prijonod4e0abd2006-03-05 11:53:36 +0000531 if (mod_inv.cb.on_tsx_state_changed && inv->notify)
Benny Prijono268ca612006-02-07 12:34:11 +0000532 (*mod_inv.cb.on_tsx_state_changed)(inv, tsx, e);
533
Benny Prijono46249942007-02-19 22:23:14 +0000534 /* Clear invite transaction when tsx is confirmed.
535 * Previously we set invite_tsx to NULL only when transaction has
536 * terminated, but this didn't work when ACK has the same Via branch
537 * value as the INVITE (see http://www.pjsip.org/trac/ticket/113)
538 */
Benny Prijonodcfc0ba2007-09-30 16:50:27 +0000539 if (tsx->state>=PJSIP_TSX_STATE_CONFIRMED && tsx == inv->invite_tsx) {
Benny Prijono46249942007-02-19 22:23:14 +0000540 inv->invite_tsx = NULL;
Benny Prijonodcfc0ba2007-09-30 16:50:27 +0000541 if (inv->last_answer) {
542 pjsip_tx_data_dec_ref(inv->last_answer);
543 inv->last_answer = NULL;
544 }
545 }
Benny Prijono268ca612006-02-07 12:34:11 +0000546}
547
Benny Prijono8ad55352006-02-08 11:16:05 +0000548
549/*
550 * Initialize the invite module.
551 */
Benny Prijono268ca612006-02-07 12:34:11 +0000552PJ_DEF(pj_status_t) pjsip_inv_usage_init( pjsip_endpoint *endpt,
Benny Prijono268ca612006-02-07 12:34:11 +0000553 const pjsip_inv_callback *cb)
554{
555 pj_status_t status;
556
557 /* Check arguments. */
Benny Prijonoe8b0d3b2006-03-17 17:57:52 +0000558 PJ_ASSERT_RETURN(endpt && cb, PJ_EINVAL);
Benny Prijono268ca612006-02-07 12:34:11 +0000559
560 /* Some callbacks are mandatory */
561 PJ_ASSERT_RETURN(cb->on_state_changed && cb->on_new_session, PJ_EINVAL);
562
563 /* Check if module already registered. */
564 PJ_ASSERT_RETURN(mod_inv.mod.id == -1, PJ_EINVALIDOP);
565
566 /* Copy param. */
567 pj_memcpy(&mod_inv.cb, cb, sizeof(pjsip_inv_callback));
568
569 mod_inv.endpt = endpt;
Benny Prijono268ca612006-02-07 12:34:11 +0000570
571 /* Register the module. */
572 status = pjsip_endpt_register_module(endpt, &mod_inv.mod);
Benny Prijono053f5222006-11-11 16:16:04 +0000573 if (status != PJ_SUCCESS)
574 return status;
Benny Prijono268ca612006-02-07 12:34:11 +0000575
Benny Prijono053f5222006-11-11 16:16:04 +0000576 return PJ_SUCCESS;
Benny Prijono268ca612006-02-07 12:34:11 +0000577}
578
Benny Prijono8ad55352006-02-08 11:16:05 +0000579/*
580 * Get the instance of invite module.
581 */
Benny Prijono268ca612006-02-07 12:34:11 +0000582PJ_DEF(pjsip_module*) pjsip_inv_usage_instance(void)
583{
584 return &mod_inv.mod;
585}
586
587
Benny Prijono632ce712006-02-09 14:01:40 +0000588
Benny Prijono8ad55352006-02-08 11:16:05 +0000589/*
590 * Return the invite session for the specified dialog.
591 */
Benny Prijono268ca612006-02-07 12:34:11 +0000592PJ_DEF(pjsip_inv_session*) pjsip_dlg_get_inv_session(pjsip_dialog *dlg)
593{
Benny Prijonoa1e69682007-05-11 15:14:34 +0000594 return (pjsip_inv_session*) dlg->mod_data[mod_inv.mod.id];
Benny Prijono268ca612006-02-07 12:34:11 +0000595}
596
Benny Prijono8ad55352006-02-08 11:16:05 +0000597
Benny Prijono268ca612006-02-07 12:34:11 +0000598/*
Benny Prijonoe8b0d3b2006-03-17 17:57:52 +0000599 * Get INVITE state name.
600 */
601PJ_DEF(const char *) pjsip_inv_state_name(pjsip_inv_state state)
602{
603 PJ_ASSERT_RETURN(state >= PJSIP_INV_STATE_NULL &&
604 state <= PJSIP_INV_STATE_DISCONNECTED,
605 "??");
606
607 return inv_state_names[state];
608}
609
610/*
Benny Prijono8ad55352006-02-08 11:16:05 +0000611 * Create UAC invite session.
Benny Prijono268ca612006-02-07 12:34:11 +0000612 */
613PJ_DEF(pj_status_t) pjsip_inv_create_uac( pjsip_dialog *dlg,
614 const pjmedia_sdp_session *local_sdp,
615 unsigned options,
616 pjsip_inv_session **p_inv)
617{
618 pjsip_inv_session *inv;
619 pj_status_t status;
620
621 /* Verify arguments. */
622 PJ_ASSERT_RETURN(dlg && p_inv, PJ_EINVAL);
623
Benny Prijono8eae8382006-08-10 21:44:26 +0000624 /* Must lock dialog first */
625 pjsip_dlg_inc_lock(dlg);
626
Benny Prijono268ca612006-02-07 12:34:11 +0000627 /* Normalize options */
628 if (options & PJSIP_INV_REQUIRE_100REL)
629 options |= PJSIP_INV_SUPPORT_100REL;
630
631 if (options & PJSIP_INV_REQUIRE_TIMER)
632 options |= PJSIP_INV_SUPPORT_TIMER;
633
634 /* Create the session */
Benny Prijonoa1e69682007-05-11 15:14:34 +0000635 inv = PJ_POOL_ZALLOC_T(dlg->pool, pjsip_inv_session);
Benny Prijono8eae8382006-08-10 21:44:26 +0000636 pj_assert(inv != NULL);
Benny Prijono268ca612006-02-07 12:34:11 +0000637
638 inv->pool = dlg->pool;
639 inv->role = PJSIP_ROLE_UAC;
640 inv->state = PJSIP_INV_STATE_NULL;
641 inv->dlg = dlg;
642 inv->options = options;
Benny Prijonod4e0abd2006-03-05 11:53:36 +0000643 inv->notify = PJ_TRUE;
Benny Prijonoba5926a2007-05-02 11:29:37 +0000644 inv->cause = (pjsip_status_code) 0;
Benny Prijono268ca612006-02-07 12:34:11 +0000645
Benny Prijonoe4f2abb2006-02-10 14:04:05 +0000646 /* Object name will use the same dialog pointer. */
Benny Prijonoed811d72006-03-10 12:57:12 +0000647 pj_ansi_snprintf(inv->obj_name, PJ_MAX_OBJ_NAME, "inv%p", dlg);
Benny Prijonoe4f2abb2006-02-10 14:04:05 +0000648
Benny Prijono268ca612006-02-07 12:34:11 +0000649 /* Create negotiator if local_sdp is specified. */
650 if (local_sdp) {
651 status = pjmedia_sdp_neg_create_w_local_offer(dlg->pool, local_sdp,
652 &inv->neg);
Benny Prijono8eae8382006-08-10 21:44:26 +0000653 if (status != PJ_SUCCESS) {
654 pjsip_dlg_dec_lock(dlg);
Benny Prijono268ca612006-02-07 12:34:11 +0000655 return status;
Benny Prijono8eae8382006-08-10 21:44:26 +0000656 }
Benny Prijono268ca612006-02-07 12:34:11 +0000657 }
658
659 /* Register invite as dialog usage. */
660 status = pjsip_dlg_add_usage(dlg, &mod_inv.mod, inv);
Benny Prijono8eae8382006-08-10 21:44:26 +0000661 if (status != PJ_SUCCESS) {
662 pjsip_dlg_dec_lock(dlg);
Benny Prijono268ca612006-02-07 12:34:11 +0000663 return status;
Benny Prijono8eae8382006-08-10 21:44:26 +0000664 }
Benny Prijono268ca612006-02-07 12:34:11 +0000665
666 /* Increment dialog session */
Benny Prijonoe4f2abb2006-02-10 14:04:05 +0000667 pjsip_dlg_inc_session(dlg, &mod_inv.mod);
Benny Prijono268ca612006-02-07 12:34:11 +0000668
Benny Prijonodcfc0ba2007-09-30 16:50:27 +0000669 /* Create 100rel handler */
670 pjsip_100rel_attach(inv);
Benny Prijonodcfc0ba2007-09-30 16:50:27 +0000671
Benny Prijono268ca612006-02-07 12:34:11 +0000672 /* Done */
673 *p_inv = inv;
Benny Prijonoe4f2abb2006-02-10 14:04:05 +0000674
Benny Prijono8eae8382006-08-10 21:44:26 +0000675 pjsip_dlg_dec_lock(dlg);
676
Benny Prijonoe4f2abb2006-02-10 14:04:05 +0000677 PJ_LOG(5,(inv->obj_name, "UAC invite session created for dialog %s",
678 dlg->obj_name));
679
Benny Prijono268ca612006-02-07 12:34:11 +0000680 return PJ_SUCCESS;
681}
682
683/*
684 * Verify incoming INVITE request.
685 */
Benny Prijono87a90212008-01-23 20:29:30 +0000686PJ_DEF(pj_status_t) pjsip_inv_verify_request2(pjsip_rx_data *rdata,
687 unsigned *options,
688 const pjmedia_sdp_session *r_sdp,
689 const pjmedia_sdp_session *l_sdp,
690 pjsip_dialog *dlg,
691 pjsip_endpoint *endpt,
692 pjsip_tx_data **p_tdata)
Benny Prijono268ca612006-02-07 12:34:11 +0000693{
694 pjsip_msg *msg;
695 pjsip_allow_hdr *allow;
696 pjsip_supported_hdr *sup_hdr;
697 pjsip_require_hdr *req_hdr;
698 int code = 200;
699 unsigned rem_option = 0;
700 pj_status_t status = PJ_SUCCESS;
701 pjsip_hdr res_hdr_list;
702
703 /* Init return arguments. */
704 if (p_tdata) *p_tdata = NULL;
705
706 /* Verify arguments. */
707 PJ_ASSERT_RETURN(rdata != NULL && options != NULL, PJ_EINVAL);
708
709 /* Normalize options */
710 if (*options & PJSIP_INV_REQUIRE_100REL)
711 *options |= PJSIP_INV_SUPPORT_100REL;
712
713 if (*options & PJSIP_INV_REQUIRE_TIMER)
714 *options |= PJSIP_INV_SUPPORT_TIMER;
715
716 /* Get the message in rdata */
717 msg = rdata->msg_info.msg;
718
719 /* Must be INVITE request. */
720 PJ_ASSERT_RETURN(msg->type == PJSIP_REQUEST_MSG &&
721 msg->line.req.method.id == PJSIP_INVITE_METHOD,
722 PJ_EINVAL);
723
724 /* If tdata is specified, then either dlg or endpt must be specified */
725 PJ_ASSERT_RETURN((!p_tdata) || (endpt || dlg), PJ_EINVAL);
726
727 /* Get the endpoint */
728 endpt = endpt ? endpt : dlg->endpt;
729
730 /* Init response header list */
731 pj_list_init(&res_hdr_list);
732
Benny Prijono87a90212008-01-23 20:29:30 +0000733 /* Check the request body, see if it's something that we support,
734 * only when the body hasn't been parsed before.
Benny Prijono268ca612006-02-07 12:34:11 +0000735 */
Benny Prijono87a90212008-01-23 20:29:30 +0000736 if (r_sdp==NULL && msg->body) {
Benny Prijono268ca612006-02-07 12:34:11 +0000737 pjsip_msg_body *body = msg->body;
738 pj_str_t str_application = {"application", 11};
739 pj_str_t str_sdp = { "sdp", 3 };
740 pjmedia_sdp_session *sdp;
741
742 /* Check content type. */
743 if (pj_stricmp(&body->content_type.type, &str_application) != 0 ||
744 pj_stricmp(&body->content_type.subtype, &str_sdp) != 0)
745 {
746 /* Not "application/sdp" */
747 code = PJSIP_SC_UNSUPPORTED_MEDIA_TYPE;
748 status = PJSIP_ERRNO_FROM_SIP_STATUS(code);
749
750 if (p_tdata) {
751 /* Add Accept header to response */
752 pjsip_accept_hdr *acc;
753
754 acc = pjsip_accept_hdr_create(rdata->tp_info.pool);
755 PJ_ASSERT_RETURN(acc, PJ_ENOMEM);
756 acc->values[acc->count++] = pj_str("application/sdp");
757 pj_list_push_back(&res_hdr_list, acc);
758 }
759
760 goto on_return;
761 }
762
763 /* Parse and validate SDP */
Benny Prijonoa1e69682007-05-11 15:14:34 +0000764 status = pjmedia_sdp_parse(rdata->tp_info.pool,
765 (char*)body->data, body->len, &sdp);
Benny Prijono268ca612006-02-07 12:34:11 +0000766 if (status == PJ_SUCCESS)
767 status = pjmedia_sdp_validate(sdp);
768
769 if (status != PJ_SUCCESS) {
770 /* Unparseable or invalid SDP */
771 code = PJSIP_SC_BAD_REQUEST;
772
773 if (p_tdata) {
774 /* Add Warning header. */
775 pjsip_warning_hdr *w;
776
777 w = pjsip_warning_hdr_create_from_status(rdata->tp_info.pool,
778 pjsip_endpt_name(endpt),
779 status);
780 PJ_ASSERT_RETURN(w, PJ_ENOMEM);
781
782 pj_list_push_back(&res_hdr_list, w);
783 }
784
785 goto on_return;
786 }
787
Benny Prijono87a90212008-01-23 20:29:30 +0000788 r_sdp = sdp;
789 }
790
791 if (r_sdp) {
Benny Prijono268ca612006-02-07 12:34:11 +0000792 /* Negotiate with local SDP */
793 if (l_sdp) {
794 pjmedia_sdp_neg *neg;
795
796 /* Local SDP must be valid! */
797 PJ_ASSERT_RETURN((status=pjmedia_sdp_validate(l_sdp))==PJ_SUCCESS,
798 status);
799
800 /* Create SDP negotiator */
801 status = pjmedia_sdp_neg_create_w_remote_offer(
Benny Prijono87a90212008-01-23 20:29:30 +0000802 rdata->tp_info.pool, l_sdp, r_sdp, &neg);
Benny Prijono268ca612006-02-07 12:34:11 +0000803 PJ_ASSERT_RETURN(status == PJ_SUCCESS, status);
804
805 /* Negotiate SDP */
806 status = pjmedia_sdp_neg_negotiate(rdata->tp_info.pool, neg, 0);
807 if (status != PJ_SUCCESS) {
808
809 /* Incompatible media */
810 code = PJSIP_SC_NOT_ACCEPTABLE_HERE;
Benny Prijono268ca612006-02-07 12:34:11 +0000811
812 if (p_tdata) {
813 pjsip_accept_hdr *acc;
814 pjsip_warning_hdr *w;
815
816 /* Add Warning header. */
817 w = pjsip_warning_hdr_create_from_status(
818 rdata->tp_info.pool,
819 pjsip_endpt_name(endpt), status);
820 PJ_ASSERT_RETURN(w, PJ_ENOMEM);
821
822 pj_list_push_back(&res_hdr_list, w);
823
824 /* Add Accept header to response */
825 acc = pjsip_accept_hdr_create(rdata->tp_info.pool);
826 PJ_ASSERT_RETURN(acc, PJ_ENOMEM);
827 acc->values[acc->count++] = pj_str("application/sdp");
828 pj_list_push_back(&res_hdr_list, acc);
829
830 }
831
832 goto on_return;
833 }
834 }
835 }
836
837 /* Check supported methods, see if peer supports UPDATE.
838 * We just assume that peer supports standard INVITE, ACK, CANCEL, and BYE
839 * implicitly by sending this INVITE.
840 */
Benny Prijonoa1e69682007-05-11 15:14:34 +0000841 allow = (pjsip_allow_hdr*) pjsip_msg_find_hdr(msg, PJSIP_H_ALLOW, NULL);
Benny Prijono268ca612006-02-07 12:34:11 +0000842 if (allow) {
843 unsigned i;
844 const pj_str_t STR_UPDATE = { "UPDATE", 6 };
845
846 for (i=0; i<allow->count; ++i) {
847 if (pj_stricmp(&allow->values[i], &STR_UPDATE)==0)
848 break;
849 }
850
851 if (i != allow->count) {
852 /* UPDATE is present in Allow */
853 rem_option |= PJSIP_INV_SUPPORT_UPDATE;
854 }
855
856 }
857
858 /* Check Supported header */
Benny Prijonoa1e69682007-05-11 15:14:34 +0000859 sup_hdr = (pjsip_supported_hdr*)
860 pjsip_msg_find_hdr(msg, PJSIP_H_SUPPORTED, NULL);
Benny Prijono268ca612006-02-07 12:34:11 +0000861 if (sup_hdr) {
862 unsigned i;
863 pj_str_t STR_100REL = { "100rel", 6};
864 pj_str_t STR_TIMER = { "timer", 5 };
865
866 for (i=0; i<sup_hdr->count; ++i) {
867 if (pj_stricmp(&sup_hdr->values[i], &STR_100REL)==0)
868 rem_option |= PJSIP_INV_SUPPORT_100REL;
869 else if (pj_stricmp(&sup_hdr->values[i], &STR_TIMER)==0)
870 rem_option |= PJSIP_INV_SUPPORT_TIMER;
871 }
872 }
873
874 /* Check Require header */
Benny Prijonoa1e69682007-05-11 15:14:34 +0000875 req_hdr = (pjsip_require_hdr*)
876 pjsip_msg_find_hdr(msg, PJSIP_H_REQUIRE, NULL);
Benny Prijono268ca612006-02-07 12:34:11 +0000877 if (req_hdr) {
878 unsigned i;
Benny Prijono053f5222006-11-11 16:16:04 +0000879 const pj_str_t STR_100REL = { "100rel", 6};
880 const pj_str_t STR_TIMER = { "timer", 5 };
881 const pj_str_t STR_REPLACES = { "replaces", 8 };
Benny Prijono268ca612006-02-07 12:34:11 +0000882 unsigned unsupp_cnt = 0;
883 pj_str_t unsupp_tags[PJSIP_GENERIC_ARRAY_MAX_COUNT];
884
885 for (i=0; i<req_hdr->count; ++i) {
886 if ((*options & PJSIP_INV_SUPPORT_100REL) &&
887 pj_stricmp(&req_hdr->values[i], &STR_100REL)==0)
888 {
889 rem_option |= PJSIP_INV_REQUIRE_100REL;
890
891 } else if ((*options && PJSIP_INV_SUPPORT_TIMER) &&
892 pj_stricmp(&req_hdr->values[i], &STR_TIMER)==0)
893 {
894 rem_option |= PJSIP_INV_REQUIRE_TIMER;
895
Benny Prijono053f5222006-11-11 16:16:04 +0000896 } else if (pj_stricmp(&req_hdr->values[i], &STR_REPLACES)==0) {
897 pj_bool_t supp;
898
899 supp = pjsip_endpt_has_capability(endpt, PJSIP_H_SUPPORTED,
900 NULL, &STR_REPLACES);
901 if (!supp)
902 unsupp_tags[unsupp_cnt++] = req_hdr->values[i];
903
Benny Prijono268ca612006-02-07 12:34:11 +0000904 } else {
905 /* Unknown/unsupported extension tag! */
906 unsupp_tags[unsupp_cnt++] = req_hdr->values[i];
907 }
908 }
909
910 /* Check if there are required tags that we don't support */
911 if (unsupp_cnt) {
912
913 code = PJSIP_SC_BAD_EXTENSION;
914 status = PJSIP_ERRNO_FROM_SIP_STATUS(code);
915
916 if (p_tdata) {
917 pjsip_unsupported_hdr *unsupp_hdr;
918 const pjsip_hdr *h;
919
920 /* Add Unsupported header. */
921 unsupp_hdr = pjsip_unsupported_hdr_create(rdata->tp_info.pool);
922 PJ_ASSERT_RETURN(unsupp_hdr != NULL, PJ_ENOMEM);
923
924 unsupp_hdr->count = unsupp_cnt;
925 for (i=0; i<unsupp_cnt; ++i)
926 unsupp_hdr->values[i] = unsupp_tags[i];
927
928 pj_list_push_back(&res_hdr_list, unsupp_hdr);
929
930 /* Add Supported header. */
931 h = pjsip_endpt_get_capability(endpt, PJSIP_H_SUPPORTED,
932 NULL);
933 pj_assert(h);
934 if (h) {
Benny Prijonoa1e69682007-05-11 15:14:34 +0000935 sup_hdr = (pjsip_supported_hdr*)
936 pjsip_hdr_clone(rdata->tp_info.pool, h);
Benny Prijono268ca612006-02-07 12:34:11 +0000937 pj_list_push_back(&res_hdr_list, sup_hdr);
938 }
939 }
940
941 goto on_return;
942 }
943 }
944
945 /* Check if there are local requirements that are not supported
946 * by peer.
947 */
948 if ( ((*options & PJSIP_INV_REQUIRE_100REL)!=0 &&
949 (rem_option & PJSIP_INV_SUPPORT_100REL)==0) ||
950 ((*options & PJSIP_INV_REQUIRE_TIMER)!=0 &&
951 (rem_option & PJSIP_INV_SUPPORT_TIMER)==0))
952 {
953 code = PJSIP_SC_EXTENSION_REQUIRED;
954 status = PJSIP_ERRNO_FROM_SIP_STATUS(code);
955
956 if (p_tdata) {
957 const pjsip_hdr *h;
958
959 /* Add Require header. */
960 req_hdr = pjsip_require_hdr_create(rdata->tp_info.pool);
961 PJ_ASSERT_RETURN(req_hdr != NULL, PJ_ENOMEM);
962
963 if (*options & PJSIP_INV_REQUIRE_100REL)
964 req_hdr->values[req_hdr->count++] = pj_str("100rel");
965
966 if (*options & PJSIP_INV_REQUIRE_TIMER)
967 req_hdr->values[req_hdr->count++] = pj_str("timer");
968
969 pj_list_push_back(&res_hdr_list, req_hdr);
970
971 /* Add Supported header. */
972 h = pjsip_endpt_get_capability(endpt, PJSIP_H_SUPPORTED,
973 NULL);
974 pj_assert(h);
975 if (h) {
Benny Prijonoa1e69682007-05-11 15:14:34 +0000976 sup_hdr = (pjsip_supported_hdr*)
977 pjsip_hdr_clone(rdata->tp_info.pool, h);
Benny Prijono268ca612006-02-07 12:34:11 +0000978 pj_list_push_back(&res_hdr_list, sup_hdr);
979 }
980
981 }
982
983 goto on_return;
984 }
985
Benny Prijonodcfc0ba2007-09-30 16:50:27 +0000986 /* If remote Require something that we support, make us Require
987 * that feature too.
988 */
989 if (rem_option & PJSIP_INV_REQUIRE_100REL) {
990 pj_assert(*options & PJSIP_INV_SUPPORT_100REL);
991 *options |= PJSIP_INV_REQUIRE_100REL;
992 }
993 if (rem_option & PJSIP_INV_REQUIRE_TIMER) {
994 pj_assert(*options & PJSIP_INV_SUPPORT_TIMER);
995 *options |= PJSIP_INV_REQUIRE_TIMER;
996 }
997
Benny Prijono268ca612006-02-07 12:34:11 +0000998on_return:
999
1000 /* Create response if necessary */
1001 if (code != 200 && p_tdata) {
1002 pjsip_tx_data *tdata;
1003 const pjsip_hdr *h;
1004
1005 if (dlg) {
1006 status = pjsip_dlg_create_response(dlg, rdata, code, NULL,
1007 &tdata);
1008 } else {
1009 status = pjsip_endpt_create_response(endpt, rdata, code, NULL,
1010 &tdata);
1011 }
1012
1013 if (status != PJ_SUCCESS)
1014 return status;
1015
1016 /* Add response headers. */
1017 h = res_hdr_list.next;
1018 while (h != &res_hdr_list) {
1019 pjsip_hdr *cloned;
1020
Benny Prijonoa1e69682007-05-11 15:14:34 +00001021 cloned = (pjsip_hdr*) pjsip_hdr_clone(tdata->pool, h);
Benny Prijono268ca612006-02-07 12:34:11 +00001022 PJ_ASSERT_RETURN(cloned, PJ_ENOMEM);
1023
1024 pjsip_msg_add_hdr(tdata->msg, cloned);
1025
1026 h = h->next;
1027 }
1028
1029 *p_tdata = tdata;
Benny Prijono70127222006-07-02 14:53:05 +00001030
1031 /* Can not return PJ_SUCCESS when response message is produced.
1032 * Ref: PROTOS test ~#2490
1033 */
1034 if (status == PJ_SUCCESS)
1035 status = PJSIP_ERRNO_FROM_SIP_STATUS(code);
1036
Benny Prijono268ca612006-02-07 12:34:11 +00001037 }
1038
1039 return status;
1040}
1041
Benny Prijono87a90212008-01-23 20:29:30 +00001042
1043/*
1044 * Verify incoming INVITE request.
1045 */
1046PJ_DEF(pj_status_t) pjsip_inv_verify_request( pjsip_rx_data *rdata,
1047 unsigned *options,
1048 const pjmedia_sdp_session *l_sdp,
1049 pjsip_dialog *dlg,
1050 pjsip_endpoint *endpt,
1051 pjsip_tx_data **p_tdata)
1052{
1053 return pjsip_inv_verify_request2(rdata, options, NULL, l_sdp, dlg,
1054 endpt, p_tdata);
1055}
1056
Benny Prijono268ca612006-02-07 12:34:11 +00001057/*
Benny Prijono8ad55352006-02-08 11:16:05 +00001058 * Create UAS invite session.
Benny Prijono268ca612006-02-07 12:34:11 +00001059 */
1060PJ_DEF(pj_status_t) pjsip_inv_create_uas( pjsip_dialog *dlg,
1061 pjsip_rx_data *rdata,
1062 const pjmedia_sdp_session *local_sdp,
1063 unsigned options,
1064 pjsip_inv_session **p_inv)
1065{
1066 pjsip_inv_session *inv;
Benny Prijonoa66c7152006-02-09 01:26:14 +00001067 struct tsx_inv_data *tsx_inv_data;
Benny Prijono268ca612006-02-07 12:34:11 +00001068 pjsip_msg *msg;
1069 pjmedia_sdp_session *rem_sdp = NULL;
1070 pj_status_t status;
1071
1072 /* Verify arguments. */
1073 PJ_ASSERT_RETURN(dlg && rdata && p_inv, PJ_EINVAL);
1074
1075 /* Dialog MUST have been initialised. */
1076 PJ_ASSERT_RETURN(pjsip_rdata_get_tsx(rdata) != NULL, PJ_EINVALIDOP);
1077
1078 msg = rdata->msg_info.msg;
1079
1080 /* rdata MUST contain INVITE request */
1081 PJ_ASSERT_RETURN(msg->type == PJSIP_REQUEST_MSG &&
1082 msg->line.req.method.id == PJSIP_INVITE_METHOD,
1083 PJ_EINVALIDOP);
1084
Benny Prijono8eae8382006-08-10 21:44:26 +00001085 /* Lock dialog */
1086 pjsip_dlg_inc_lock(dlg);
1087
Benny Prijono268ca612006-02-07 12:34:11 +00001088 /* Normalize options */
1089 if (options & PJSIP_INV_REQUIRE_100REL)
1090 options |= PJSIP_INV_SUPPORT_100REL;
1091
1092 if (options & PJSIP_INV_REQUIRE_TIMER)
1093 options |= PJSIP_INV_SUPPORT_TIMER;
1094
1095 /* Create the session */
Benny Prijonoa1e69682007-05-11 15:14:34 +00001096 inv = PJ_POOL_ZALLOC_T(dlg->pool, pjsip_inv_session);
Benny Prijono8eae8382006-08-10 21:44:26 +00001097 pj_assert(inv != NULL);
Benny Prijono268ca612006-02-07 12:34:11 +00001098
1099 inv->pool = dlg->pool;
1100 inv->role = PJSIP_ROLE_UAS;
Benny Prijono38998232006-02-08 22:44:25 +00001101 inv->state = PJSIP_INV_STATE_NULL;
Benny Prijono268ca612006-02-07 12:34:11 +00001102 inv->dlg = dlg;
1103 inv->options = options;
Benny Prijonod4e0abd2006-03-05 11:53:36 +00001104 inv->notify = PJ_TRUE;
Benny Prijonoba5926a2007-05-02 11:29:37 +00001105 inv->cause = (pjsip_status_code) 0;
Benny Prijono268ca612006-02-07 12:34:11 +00001106
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00001107 /* Object name will use the same dialog pointer. */
Benny Prijonoed811d72006-03-10 12:57:12 +00001108 pj_ansi_snprintf(inv->obj_name, PJ_MAX_OBJ_NAME, "inv%p", dlg);
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00001109
Benny Prijono268ca612006-02-07 12:34:11 +00001110 /* Parse SDP in message body, if present. */
1111 if (msg->body) {
1112 pjsip_msg_body *body = msg->body;
1113
1114 /* Parse and validate SDP */
Benny Prijonoa1e69682007-05-11 15:14:34 +00001115 status = pjmedia_sdp_parse(inv->pool, (char*)body->data, body->len,
Benny Prijono268ca612006-02-07 12:34:11 +00001116 &rem_sdp);
1117 if (status == PJ_SUCCESS)
1118 status = pjmedia_sdp_validate(rem_sdp);
1119
Benny Prijono8eae8382006-08-10 21:44:26 +00001120 if (status != PJ_SUCCESS) {
1121 pjsip_dlg_dec_lock(dlg);
Benny Prijono268ca612006-02-07 12:34:11 +00001122 return status;
Benny Prijono8eae8382006-08-10 21:44:26 +00001123 }
Benny Prijono268ca612006-02-07 12:34:11 +00001124 }
1125
1126 /* Create negotiator. */
1127 if (rem_sdp) {
1128 status = pjmedia_sdp_neg_create_w_remote_offer(inv->pool, local_sdp,
1129 rem_sdp, &inv->neg);
1130
1131 } else if (local_sdp) {
1132 status = pjmedia_sdp_neg_create_w_local_offer(inv->pool, local_sdp,
1133 &inv->neg);
1134 } else {
Benny Prijono95196582006-02-09 00:13:40 +00001135 status = PJ_SUCCESS;
Benny Prijono268ca612006-02-07 12:34:11 +00001136 }
1137
Benny Prijono8eae8382006-08-10 21:44:26 +00001138 if (status != PJ_SUCCESS) {
1139 pjsip_dlg_dec_lock(dlg);
Benny Prijono268ca612006-02-07 12:34:11 +00001140 return status;
Benny Prijono8eae8382006-08-10 21:44:26 +00001141 }
Benny Prijono268ca612006-02-07 12:34:11 +00001142
1143 /* Register invite as dialog usage. */
1144 status = pjsip_dlg_add_usage(dlg, &mod_inv.mod, inv);
Benny Prijono8eae8382006-08-10 21:44:26 +00001145 if (status != PJ_SUCCESS) {
1146 pjsip_dlg_dec_lock(dlg);
Benny Prijono268ca612006-02-07 12:34:11 +00001147 return status;
Benny Prijono8eae8382006-08-10 21:44:26 +00001148 }
Benny Prijono268ca612006-02-07 12:34:11 +00001149
1150 /* Increment session in the dialog. */
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00001151 pjsip_dlg_inc_session(dlg, &mod_inv.mod);
Benny Prijono268ca612006-02-07 12:34:11 +00001152
1153 /* Save the invite transaction. */
1154 inv->invite_tsx = pjsip_rdata_get_tsx(rdata);
Benny Prijonoa66c7152006-02-09 01:26:14 +00001155
1156 /* Attach our data to the transaction. */
Benny Prijonoa1e69682007-05-11 15:14:34 +00001157 tsx_inv_data = PJ_POOL_ZALLOC_T(inv->invite_tsx->pool, struct tsx_inv_data);
Benny Prijonoa66c7152006-02-09 01:26:14 +00001158 tsx_inv_data->inv = inv;
1159 inv->invite_tsx->mod_data[mod_inv.mod.id] = tsx_inv_data;
Benny Prijono268ca612006-02-07 12:34:11 +00001160
Benny Prijonodcfc0ba2007-09-30 16:50:27 +00001161 /* Create 100rel handler */
1162 if (inv->options & PJSIP_INV_REQUIRE_100REL) {
1163 pjsip_100rel_attach(inv);
1164 }
Benny Prijonodcfc0ba2007-09-30 16:50:27 +00001165
Benny Prijono268ca612006-02-07 12:34:11 +00001166 /* Done */
Benny Prijono8eae8382006-08-10 21:44:26 +00001167 pjsip_dlg_dec_lock(dlg);
Benny Prijono268ca612006-02-07 12:34:11 +00001168 *p_inv = inv;
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00001169
1170 PJ_LOG(5,(inv->obj_name, "UAS invite session created for dialog %s",
1171 dlg->obj_name));
1172
Benny Prijono268ca612006-02-07 12:34:11 +00001173 return PJ_SUCCESS;
1174}
1175
Benny Prijonod4e0abd2006-03-05 11:53:36 +00001176/*
1177 * Forcefully terminate the session.
1178 */
1179PJ_DEF(pj_status_t) pjsip_inv_terminate( pjsip_inv_session *inv,
1180 int st_code,
1181 pj_bool_t notify)
1182{
1183 PJ_ASSERT_RETURN(inv, PJ_EINVAL);
1184
1185 /* Lock dialog. */
1186 pjsip_dlg_inc_lock(inv->dlg);
1187
1188 /* Set callback notify flag. */
1189 inv->notify = notify;
1190
1191 /* If there's pending transaction, terminate the transaction.
1192 * This may subsequently set the INVITE session state to
1193 * disconnected.
1194 */
1195 if (inv->invite_tsx &&
1196 inv->invite_tsx->state <= PJSIP_TSX_STATE_COMPLETED)
1197 {
1198 pjsip_tsx_terminate(inv->invite_tsx, st_code);
1199
1200 }
1201
1202 /* Set cause. */
Benny Prijono0b6340c2006-06-13 22:21:23 +00001203 inv_set_cause(inv, st_code, NULL);
Benny Prijonod4e0abd2006-03-05 11:53:36 +00001204
1205 /* Forcefully terminate the session if state is not DISCONNECTED */
1206 if (inv->state != PJSIP_INV_STATE_DISCONNECTED) {
1207 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, NULL);
1208 }
1209
1210 /* Done.
1211 * The dec_lock() below will actually destroys the dialog if it
1212 * has no other session.
1213 */
1214 pjsip_dlg_dec_lock(inv->dlg);
1215
1216 return PJ_SUCCESS;
1217}
1218
1219
Benny Prijono268ca612006-02-07 12:34:11 +00001220static void *clone_sdp(pj_pool_t *pool, const void *data, unsigned len)
1221{
1222 PJ_UNUSED_ARG(len);
Benny Prijonoa1e69682007-05-11 15:14:34 +00001223 return pjmedia_sdp_session_clone(pool, (const pjmedia_sdp_session*)data);
Benny Prijono268ca612006-02-07 12:34:11 +00001224}
1225
1226static int print_sdp(pjsip_msg_body *body, char *buf, pj_size_t len)
1227{
Benny Prijonoa1e69682007-05-11 15:14:34 +00001228 return pjmedia_sdp_print((const pjmedia_sdp_session*)body->data, buf, len);
Benny Prijono268ca612006-02-07 12:34:11 +00001229}
1230
Benny Prijono56315612006-07-18 14:39:40 +00001231
1232PJ_DEF(pj_status_t) pjsip_create_sdp_body( pj_pool_t *pool,
1233 pjmedia_sdp_session *sdp,
1234 pjsip_msg_body **p_body)
1235{
1236 const pj_str_t STR_APPLICATION = { "application", 11};
1237 const pj_str_t STR_SDP = { "sdp", 3 };
1238 pjsip_msg_body *body;
1239
Benny Prijonoa1e69682007-05-11 15:14:34 +00001240 body = PJ_POOL_ZALLOC_T(pool, pjsip_msg_body);
Benny Prijono56315612006-07-18 14:39:40 +00001241 PJ_ASSERT_RETURN(body != NULL, PJ_ENOMEM);
1242
1243 body->content_type.type = STR_APPLICATION;
1244 body->content_type.subtype = STR_SDP;
1245 body->data = sdp;
1246 body->len = 0;
1247 body->clone_data = &clone_sdp;
1248 body->print_body = &print_sdp;
1249
1250 *p_body = body;
1251
1252 return PJ_SUCCESS;
1253}
1254
Benny Prijono268ca612006-02-07 12:34:11 +00001255static pjsip_msg_body *create_sdp_body(pj_pool_t *pool,
1256 const pjmedia_sdp_session *c_sdp)
1257{
1258 pjsip_msg_body *body;
Benny Prijono56315612006-07-18 14:39:40 +00001259 pj_status_t status;
Benny Prijono268ca612006-02-07 12:34:11 +00001260
Benny Prijono56315612006-07-18 14:39:40 +00001261 status = pjsip_create_sdp_body(pool,
1262 pjmedia_sdp_session_clone(pool, c_sdp),
1263 &body);
Benny Prijono268ca612006-02-07 12:34:11 +00001264
Benny Prijono56315612006-07-18 14:39:40 +00001265 if (status != PJ_SUCCESS)
1266 return NULL;
Benny Prijono268ca612006-02-07 12:34:11 +00001267
1268 return body;
1269}
1270
1271/*
1272 * Create initial INVITE request.
1273 */
1274PJ_DEF(pj_status_t) pjsip_inv_invite( pjsip_inv_session *inv,
1275 pjsip_tx_data **p_tdata )
1276{
1277 pjsip_tx_data *tdata;
1278 const pjsip_hdr *hdr;
Benny Prijono26ff9062006-02-21 23:47:00 +00001279 pj_bool_t has_sdp;
Benny Prijono268ca612006-02-07 12:34:11 +00001280 pj_status_t status;
1281
1282 /* Verify arguments. */
1283 PJ_ASSERT_RETURN(inv && p_tdata, PJ_EINVAL);
1284
Benny Prijono26ff9062006-02-21 23:47:00 +00001285 /* State MUST be NULL or CONFIRMED. */
1286 PJ_ASSERT_RETURN(inv->state == PJSIP_INV_STATE_NULL ||
1287 inv->state == PJSIP_INV_STATE_CONFIRMED,
1288 PJ_EINVALIDOP);
Benny Prijono268ca612006-02-07 12:34:11 +00001289
Benny Prijono64f851e2006-02-23 13:49:28 +00001290 /* Lock dialog. */
1291 pjsip_dlg_inc_lock(inv->dlg);
1292
Benny Prijono268ca612006-02-07 12:34:11 +00001293 /* Create the INVITE request. */
Benny Prijono1f61a8f2007-08-16 10:11:44 +00001294 status = pjsip_dlg_create_request(inv->dlg, pjsip_get_invite_method(), -1,
Benny Prijono268ca612006-02-07 12:34:11 +00001295 &tdata);
1296 if (status != PJ_SUCCESS)
Benny Prijono64f851e2006-02-23 13:49:28 +00001297 goto on_return;
1298
Benny Prijono268ca612006-02-07 12:34:11 +00001299
Benny Prijono26ff9062006-02-21 23:47:00 +00001300 /* If this is the first INVITE, then copy the headers from inv_hdr.
1301 * These are the headers parsed from the request URI when the
1302 * dialog was created.
1303 */
1304 if (inv->state == PJSIP_INV_STATE_NULL) {
1305 hdr = inv->dlg->inv_hdr.next;
1306
1307 while (hdr != &inv->dlg->inv_hdr) {
Benny Prijonoa1e69682007-05-11 15:14:34 +00001308 pjsip_msg_add_hdr(tdata->msg, (pjsip_hdr*)
Benny Prijono26ff9062006-02-21 23:47:00 +00001309 pjsip_hdr_shallow_clone(tdata->pool, hdr));
1310 hdr = hdr->next;
1311 }
1312 }
1313
1314 /* See if we have SDP to send. */
1315 if (inv->neg) {
1316 pjmedia_sdp_neg_state neg_state;
1317
1318 neg_state = pjmedia_sdp_neg_get_state(inv->neg);
1319
1320 has_sdp = (neg_state == PJMEDIA_SDP_NEG_STATE_LOCAL_OFFER ||
1321 (neg_state == PJMEDIA_SDP_NEG_STATE_WAIT_NEGO &&
1322 pjmedia_sdp_neg_has_local_answer(inv->neg)));
1323
1324
1325 } else {
1326 has_sdp = PJ_FALSE;
1327 }
1328
Benny Prijono268ca612006-02-07 12:34:11 +00001329 /* Add SDP, if any. */
Benny Prijono26ff9062006-02-21 23:47:00 +00001330 if (has_sdp) {
Benny Prijono268ca612006-02-07 12:34:11 +00001331 const pjmedia_sdp_session *offer;
1332
1333 status = pjmedia_sdp_neg_get_neg_local(inv->neg, &offer);
1334 if (status != PJ_SUCCESS)
Benny Prijono64f851e2006-02-23 13:49:28 +00001335 goto on_return;
Benny Prijono268ca612006-02-07 12:34:11 +00001336
1337 tdata->msg->body = create_sdp_body(tdata->pool, offer);
1338 }
1339
1340 /* Add Allow header. */
Benny Prijono1f61a8f2007-08-16 10:11:44 +00001341 if (inv->dlg->add_allow) {
Benny Prijono95673f32007-06-26 08:23:18 +00001342 hdr = pjsip_endpt_get_capability(inv->dlg->endpt, PJSIP_H_ALLOW, NULL);
1343 if (hdr) {
1344 pjsip_msg_add_hdr(tdata->msg, (pjsip_hdr*)
1345 pjsip_hdr_shallow_clone(tdata->pool, hdr));
1346 }
Benny Prijono268ca612006-02-07 12:34:11 +00001347 }
1348
1349 /* Add Supported header */
1350 hdr = pjsip_endpt_get_capability(inv->dlg->endpt, PJSIP_H_SUPPORTED, NULL);
1351 if (hdr) {
Benny Prijonoa1e69682007-05-11 15:14:34 +00001352 pjsip_msg_add_hdr(tdata->msg, (pjsip_hdr*)
Benny Prijono268ca612006-02-07 12:34:11 +00001353 pjsip_hdr_shallow_clone(tdata->pool, hdr));
1354 }
1355
1356 /* Add Require header. */
Benny Prijonodcfc0ba2007-09-30 16:50:27 +00001357 if (inv->options & PJSIP_INV_REQUIRE_100REL) {
1358 const pj_str_t HREQ = { "Require", 7 };
1359 const pj_str_t tag_100rel = { "100rel", 6 };
1360 pjsip_generic_string_hdr *hreq;
1361
1362 hreq = pjsip_generic_string_hdr_create(tdata->pool, &HREQ,
1363 &tag_100rel);
1364 pjsip_msg_add_hdr(tdata->msg, (pjsip_hdr*) hreq);
1365 }
Benny Prijono268ca612006-02-07 12:34:11 +00001366
1367 /* Done. */
1368 *p_tdata = tdata;
1369
Benny Prijono64f851e2006-02-23 13:49:28 +00001370
1371on_return:
1372 pjsip_dlg_dec_lock(inv->dlg);
1373 return status;
Benny Prijono268ca612006-02-07 12:34:11 +00001374}
1375
1376
1377/*
Benny Prijono1f7767b2007-10-03 18:28:49 +00001378 * Initiate SDP negotiation in the SDP negotiator.
Benny Prijono95196582006-02-09 00:13:40 +00001379 */
1380static pj_status_t inv_negotiate_sdp( pjsip_inv_session *inv )
1381{
1382 pj_status_t status;
1383
1384 PJ_ASSERT_RETURN(pjmedia_sdp_neg_get_state(inv->neg) ==
1385 PJMEDIA_SDP_NEG_STATE_WAIT_NEGO,
1386 PJMEDIA_SDPNEG_EINSTATE);
1387
1388 status = pjmedia_sdp_neg_negotiate(inv->pool, inv->neg, 0);
1389
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00001390 PJ_LOG(5,(inv->obj_name, "SDP negotiation done, status=%d", status));
1391
Benny Prijonod4e0abd2006-03-05 11:53:36 +00001392 if (mod_inv.cb.on_media_update && inv->notify)
Benny Prijono95196582006-02-09 00:13:40 +00001393 (*mod_inv.cb.on_media_update)(inv, status);
1394
1395 return status;
1396}
1397
1398/*
Benny Prijonoa66c7152006-02-09 01:26:14 +00001399 * Check in incoming message for SDP offer/answer.
1400 */
Benny Prijono26ff9062006-02-21 23:47:00 +00001401static pj_status_t inv_check_sdp_in_incoming_msg( pjsip_inv_session *inv,
1402 pjsip_transaction *tsx,
1403 pjsip_rx_data *rdata)
Benny Prijonoa66c7152006-02-09 01:26:14 +00001404{
1405 struct tsx_inv_data *tsx_inv_data;
1406 static const pj_str_t str_application = { "application", 11 };
1407 static const pj_str_t str_sdp = { "sdp", 3 };
1408 pj_status_t status;
1409 pjsip_msg *msg;
1410 pjmedia_sdp_session *sdp;
1411
1412 /* Get/attach invite session's transaction data */
Benny Prijonoa1e69682007-05-11 15:14:34 +00001413 tsx_inv_data = (struct tsx_inv_data*) tsx->mod_data[mod_inv.mod.id];
Benny Prijonoa66c7152006-02-09 01:26:14 +00001414 if (tsx_inv_data == NULL) {
Benny Prijonoa1e69682007-05-11 15:14:34 +00001415 tsx_inv_data = PJ_POOL_ZALLOC_T(tsx->pool, struct tsx_inv_data);
Benny Prijonoa66c7152006-02-09 01:26:14 +00001416 tsx_inv_data->inv = inv;
1417 tsx->mod_data[mod_inv.mod.id] = tsx_inv_data;
1418 }
1419
1420 /* MUST NOT do multiple SDP offer/answer in a single transaction.
1421 */
1422
Benny Prijono77998ce2007-06-20 10:03:46 +00001423 if (tsx_inv_data->sdp_done) {
1424 if (rdata->msg_info.msg->body) {
1425 PJ_LOG(4,(inv->obj_name, "SDP negotiation done, message "
1426 "body is ignored"));
1427 }
Benny Prijono26ff9062006-02-21 23:47:00 +00001428 return PJ_SUCCESS;
Benny Prijono77998ce2007-06-20 10:03:46 +00001429 }
Benny Prijonoa66c7152006-02-09 01:26:14 +00001430
1431 /* Check if SDP is present in the message. */
1432
1433 msg = rdata->msg_info.msg;
1434 if (msg->body == NULL) {
1435 /* Message doesn't have body. */
Benny Prijono26ff9062006-02-21 23:47:00 +00001436 return PJ_SUCCESS;
Benny Prijonoa66c7152006-02-09 01:26:14 +00001437 }
1438
1439 if (pj_stricmp(&msg->body->content_type.type, &str_application) ||
1440 pj_stricmp(&msg->body->content_type.subtype, &str_sdp))
1441 {
1442 /* Message body is not "application/sdp" */
Benny Prijono26ff9062006-02-21 23:47:00 +00001443 return PJMEDIA_SDP_EINSDP;
Benny Prijonoa66c7152006-02-09 01:26:14 +00001444 }
1445
1446 /* Parse the SDP body. */
1447
Benny Prijonoa1e69682007-05-11 15:14:34 +00001448 status = pjmedia_sdp_parse(rdata->tp_info.pool,
1449 (char*)msg->body->data,
Benny Prijonoa66c7152006-02-09 01:26:14 +00001450 msg->body->len, &sdp);
Benny Prijonoc1b1c0a2007-11-06 08:48:02 +00001451 if (status == PJ_SUCCESS)
1452 status = pjmedia_sdp_validate(sdp);
1453
Benny Prijonoa66c7152006-02-09 01:26:14 +00001454 if (status != PJ_SUCCESS) {
1455 char errmsg[PJ_ERR_MSG_SIZE];
1456 pj_strerror(status, errmsg, sizeof(errmsg));
1457 PJ_LOG(4,(THIS_FILE, "Error parsing SDP in %s: %s",
1458 pjsip_rx_data_get_info(rdata), errmsg));
Benny Prijono26ff9062006-02-21 23:47:00 +00001459 return PJMEDIA_SDP_EINSDP;
Benny Prijonoa66c7152006-02-09 01:26:14 +00001460 }
1461
1462 /* The SDP can be an offer or answer, depending on negotiator's state */
1463
1464 if (inv->neg == NULL ||
1465 pjmedia_sdp_neg_get_state(inv->neg) == PJMEDIA_SDP_NEG_STATE_DONE)
1466 {
1467
1468 /* This is an offer. */
1469
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00001470 PJ_LOG(5,(inv->obj_name, "Got SDP offer in %s",
1471 pjsip_rx_data_get_info(rdata)));
1472
Benny Prijonoa66c7152006-02-09 01:26:14 +00001473 if (inv->neg == NULL) {
1474 status=pjmedia_sdp_neg_create_w_remote_offer(inv->pool, NULL,
1475 sdp, &inv->neg);
1476 } else {
1477 status=pjmedia_sdp_neg_set_remote_offer(inv->pool, inv->neg, sdp);
1478 }
1479
1480 if (status != PJ_SUCCESS) {
1481 char errmsg[PJ_ERR_MSG_SIZE];
1482 pj_strerror(status, errmsg, sizeof(errmsg));
1483 PJ_LOG(4,(THIS_FILE, "Error processing SDP offer in %s: %s",
1484 pjsip_rx_data_get_info(rdata), errmsg));
Benny Prijono26ff9062006-02-21 23:47:00 +00001485 return PJMEDIA_SDP_EINSDP;
Benny Prijonoa66c7152006-02-09 01:26:14 +00001486 }
1487
1488 /* Inform application about remote offer. */
1489
Benny Prijonod4e0abd2006-03-05 11:53:36 +00001490 if (mod_inv.cb.on_rx_offer && inv->notify) {
Benny Prijono26ff9062006-02-21 23:47:00 +00001491
1492 (*mod_inv.cb.on_rx_offer)(inv, sdp);
1493
1494 }
Benny Prijonoa66c7152006-02-09 01:26:14 +00001495
1496 } else if (pjmedia_sdp_neg_get_state(inv->neg) ==
1497 PJMEDIA_SDP_NEG_STATE_LOCAL_OFFER)
1498 {
1499
1500 /* This is an answer.
1501 * Process and negotiate remote answer.
1502 */
1503
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00001504 PJ_LOG(5,(inv->obj_name, "Got SDP answer in %s",
1505 pjsip_rx_data_get_info(rdata)));
1506
Benny Prijonoa66c7152006-02-09 01:26:14 +00001507 status = pjmedia_sdp_neg_set_remote_answer(inv->pool, inv->neg, sdp);
1508
1509 if (status != PJ_SUCCESS) {
1510 char errmsg[PJ_ERR_MSG_SIZE];
1511 pj_strerror(status, errmsg, sizeof(errmsg));
1512 PJ_LOG(4,(THIS_FILE, "Error processing SDP answer in %s: %s",
1513 pjsip_rx_data_get_info(rdata), errmsg));
Benny Prijono26ff9062006-02-21 23:47:00 +00001514 return PJMEDIA_SDP_EINSDP;
Benny Prijonoa66c7152006-02-09 01:26:14 +00001515 }
1516
1517 /* Negotiate SDP */
1518
1519 inv_negotiate_sdp(inv);
1520
1521 /* Mark this transaction has having SDP offer/answer done. */
1522
1523 tsx_inv_data->sdp_done = 1;
1524
1525 } else {
1526
1527 PJ_LOG(5,(THIS_FILE, "Ignored SDP in %s: negotiator state is %s",
1528 pjsip_rx_data_get_info(rdata),
1529 pjmedia_sdp_neg_state_str(pjmedia_sdp_neg_get_state(inv->neg))));
1530 }
1531
Benny Prijono26ff9062006-02-21 23:47:00 +00001532 return PJ_SUCCESS;
Benny Prijonoa66c7152006-02-09 01:26:14 +00001533}
1534
1535
Benny Prijono26ff9062006-02-21 23:47:00 +00001536/*
1537 * Process INVITE answer, for both initial and subsequent re-INVITE
1538 */
1539static pj_status_t process_answer( pjsip_inv_session *inv,
1540 int st_code,
Benny Prijono64f851e2006-02-23 13:49:28 +00001541 pjsip_tx_data *tdata,
1542 const pjmedia_sdp_session *local_sdp)
Benny Prijono26ff9062006-02-21 23:47:00 +00001543{
1544 pj_status_t status;
Benny Prijonoab7399b2006-02-27 00:40:31 +00001545 const pjmedia_sdp_session *sdp = NULL;
Benny Prijono26ff9062006-02-21 23:47:00 +00001546
Benny Prijono64f851e2006-02-23 13:49:28 +00001547 /* If local_sdp is specified, then we MUST NOT have answered the
1548 * offer before.
Benny Prijono26ff9062006-02-21 23:47:00 +00001549 */
Benny Prijono64f851e2006-02-23 13:49:28 +00001550 if (local_sdp && (st_code/100==1 || st_code/100==2)) {
1551
1552 if (inv->neg == NULL) {
1553 status = pjmedia_sdp_neg_create_w_local_offer(inv->pool, local_sdp,
1554 &inv->neg);
1555 } else if (pjmedia_sdp_neg_get_state(inv->neg)==
1556 PJMEDIA_SDP_NEG_STATE_REMOTE_OFFER)
1557 {
1558 status = pjmedia_sdp_neg_set_local_answer(inv->pool, inv->neg,
1559 local_sdp);
1560 } else {
1561
1562 /* Can not specify local SDP at this state. */
1563 pj_assert(0);
1564 status = PJMEDIA_SDPNEG_EINSTATE;
1565 }
1566
1567 if (status != PJ_SUCCESS)
1568 return status;
1569
1570 }
1571
1572
1573 /* If SDP negotiator is ready, start negotiation. */
1574 if (st_code/100==2 || (st_code/10==18 && st_code!=180)) {
Benny Prijono26ff9062006-02-21 23:47:00 +00001575
1576 pjmedia_sdp_neg_state neg_state;
1577
Benny Prijono64f851e2006-02-23 13:49:28 +00001578 /* Start nego when appropriate. */
Benny Prijono26ff9062006-02-21 23:47:00 +00001579 neg_state = inv->neg ? pjmedia_sdp_neg_get_state(inv->neg) :
1580 PJMEDIA_SDP_NEG_STATE_NULL;
1581
1582 if (neg_state == PJMEDIA_SDP_NEG_STATE_LOCAL_OFFER) {
1583
1584 status = pjmedia_sdp_neg_get_neg_local(inv->neg, &sdp);
1585
1586 } else if (neg_state == PJMEDIA_SDP_NEG_STATE_WAIT_NEGO &&
1587 pjmedia_sdp_neg_has_local_answer(inv->neg) )
1588 {
Benny Prijono77998ce2007-06-20 10:03:46 +00001589 struct tsx_inv_data *tsx_inv_data;
1590
1591 /* Get invite session's transaction data */
1592 tsx_inv_data = (struct tsx_inv_data*)
1593 inv->invite_tsx->mod_data[mod_inv.mod.id];
Benny Prijono26ff9062006-02-21 23:47:00 +00001594
1595 status = inv_negotiate_sdp(inv);
1596 if (status != PJ_SUCCESS)
1597 return status;
1598
Benny Prijono77998ce2007-06-20 10:03:46 +00001599 /* Mark this transaction has having SDP offer/answer done. */
1600 tsx_inv_data->sdp_done = 1;
1601
Benny Prijono26ff9062006-02-21 23:47:00 +00001602 status = pjmedia_sdp_neg_get_active_local(inv->neg, &sdp);
1603 }
Benny Prijono26ff9062006-02-21 23:47:00 +00001604 }
1605
Benny Prijono64f851e2006-02-23 13:49:28 +00001606 /* Include SDP when it's available for 2xx and 18x (but not 180) response.
Benny Prijono26ff9062006-02-21 23:47:00 +00001607 * Subsequent response will include this SDP.
Benny Prijono48ab2b72007-11-08 09:24:30 +00001608 *
1609 * Note note:
1610 * - When offer/answer has been completed in reliable 183, we MUST NOT
1611 * send SDP in 2xx response. So if we don't have SDP to send, clear
1612 * the SDP in the message body ONLY if 100rel is active in this
1613 * session.
Benny Prijono26ff9062006-02-21 23:47:00 +00001614 */
1615 if (sdp) {
1616 tdata->msg->body = create_sdp_body(tdata->pool, sdp);
Benny Prijono48ab2b72007-11-08 09:24:30 +00001617 } else {
1618 if (inv->options & PJSIP_INV_REQUIRE_100REL) {
1619 tdata->msg->body = NULL;
1620 }
Benny Prijono26ff9062006-02-21 23:47:00 +00001621 }
1622
Benny Prijono26ff9062006-02-21 23:47:00 +00001623
1624 return PJ_SUCCESS;
1625}
1626
Benny Prijonoa66c7152006-02-09 01:26:14 +00001627
1628/*
Benny Prijono64f851e2006-02-23 13:49:28 +00001629 * Create first response to INVITE
1630 */
1631PJ_DEF(pj_status_t) pjsip_inv_initial_answer( pjsip_inv_session *inv,
1632 pjsip_rx_data *rdata,
1633 int st_code,
1634 const pj_str_t *st_text,
1635 const pjmedia_sdp_session *sdp,
1636 pjsip_tx_data **p_tdata)
1637{
1638 pjsip_tx_data *tdata;
1639 pj_status_t status;
1640
1641 /* Verify arguments. */
1642 PJ_ASSERT_RETURN(inv && p_tdata, PJ_EINVAL);
1643
1644 /* Must have INVITE transaction. */
1645 PJ_ASSERT_RETURN(inv->invite_tsx, PJ_EBUG);
1646
1647 pjsip_dlg_inc_lock(inv->dlg);
1648
1649 /* Create response */
1650 status = pjsip_dlg_create_response(inv->dlg, rdata, st_code, st_text,
1651 &tdata);
1652 if (status != PJ_SUCCESS)
1653 goto on_return;
1654
1655 /* Process SDP in answer */
1656 status = process_answer(inv, st_code, tdata, sdp);
1657 if (status != PJ_SUCCESS) {
1658 pjsip_tx_data_dec_ref(tdata);
1659 goto on_return;
1660 }
1661
Benny Prijonodcfc0ba2007-09-30 16:50:27 +00001662 /* Save this answer */
1663 inv->last_answer = tdata;
1664 pjsip_tx_data_add_ref(inv->last_answer);
1665 PJ_LOG(5,(inv->dlg->obj_name, "Initial answer %s",
1666 pjsip_tx_data_get_info(inv->last_answer)));
1667
Benny Prijono64f851e2006-02-23 13:49:28 +00001668 *p_tdata = tdata;
1669
1670on_return:
1671 pjsip_dlg_dec_lock(inv->dlg);
1672 return status;
1673}
1674
1675
1676/*
Benny Prijono26ff9062006-02-21 23:47:00 +00001677 * Answer initial INVITE
1678 * Re-INVITE will be answered automatically, and will not use this function.
Benny Prijono268ca612006-02-07 12:34:11 +00001679 */
1680PJ_DEF(pj_status_t) pjsip_inv_answer( pjsip_inv_session *inv,
1681 int st_code,
1682 const pj_str_t *st_text,
1683 const pjmedia_sdp_session *local_sdp,
1684 pjsip_tx_data **p_tdata )
1685{
1686 pjsip_tx_data *last_res;
1687 pj_status_t status;
1688
1689 /* Verify arguments. */
1690 PJ_ASSERT_RETURN(inv && p_tdata, PJ_EINVAL);
1691
1692 /* Must have INVITE transaction. */
1693 PJ_ASSERT_RETURN(inv->invite_tsx, PJ_EBUG);
1694
Benny Prijonodcfc0ba2007-09-30 16:50:27 +00001695 /* Must have created an answer before */
1696 PJ_ASSERT_RETURN(inv->last_answer, PJ_EINVALIDOP);
Benny Prijono268ca612006-02-07 12:34:11 +00001697
Benny Prijono64f851e2006-02-23 13:49:28 +00001698 pjsip_dlg_inc_lock(inv->dlg);
Benny Prijono268ca612006-02-07 12:34:11 +00001699
1700 /* Modify last response. */
Benny Prijonodcfc0ba2007-09-30 16:50:27 +00001701 last_res = inv->last_answer;
Benny Prijono268ca612006-02-07 12:34:11 +00001702 status = pjsip_dlg_modify_response(inv->dlg, last_res, st_code, st_text);
1703 if (status != PJ_SUCCESS)
Benny Prijono64f851e2006-02-23 13:49:28 +00001704 goto on_return;
Benny Prijono268ca612006-02-07 12:34:11 +00001705
Benny Prijonodcfc0ba2007-09-30 16:50:27 +00001706 /* For non-2xx final response, strip message body */
1707 if (st_code >= 300) {
1708 last_res->msg->body = NULL;
1709 }
Benny Prijono268ca612006-02-07 12:34:11 +00001710
Benny Prijono26ff9062006-02-21 23:47:00 +00001711 /* Process SDP in answer */
Benny Prijono64f851e2006-02-23 13:49:28 +00001712 status = process_answer(inv, st_code, last_res, local_sdp);
1713 if (status != PJ_SUCCESS) {
1714 pjsip_tx_data_dec_ref(last_res);
1715 goto on_return;
1716 }
Benny Prijono268ca612006-02-07 12:34:11 +00001717
Benny Prijono268ca612006-02-07 12:34:11 +00001718
1719 *p_tdata = last_res;
1720
Benny Prijono64f851e2006-02-23 13:49:28 +00001721on_return:
1722 pjsip_dlg_dec_lock(inv->dlg);
1723 return status;
Benny Prijono268ca612006-02-07 12:34:11 +00001724}
1725
1726
1727/*
Benny Prijono26ff9062006-02-21 23:47:00 +00001728 * Set SDP answer.
1729 */
1730PJ_DEF(pj_status_t) pjsip_inv_set_sdp_answer( pjsip_inv_session *inv,
1731 const pjmedia_sdp_session *sdp )
1732{
1733 pj_status_t status;
1734
1735 PJ_ASSERT_RETURN(inv && sdp, PJ_EINVAL);
1736
1737 pjsip_dlg_inc_lock(inv->dlg);
1738 status = pjmedia_sdp_neg_set_local_answer( inv->pool, inv->neg, sdp);
1739 pjsip_dlg_dec_lock(inv->dlg);
1740
1741 return status;
1742}
1743
1744
1745/*
Benny Prijono268ca612006-02-07 12:34:11 +00001746 * End session.
1747 */
1748PJ_DEF(pj_status_t) pjsip_inv_end_session( pjsip_inv_session *inv,
1749 int st_code,
1750 const pj_str_t *st_text,
1751 pjsip_tx_data **p_tdata )
1752{
1753 pjsip_tx_data *tdata;
1754 pj_status_t status;
1755
1756 /* Verify arguments. */
1757 PJ_ASSERT_RETURN(inv && p_tdata, PJ_EINVAL);
1758
Benny Prijonod4e0abd2006-03-05 11:53:36 +00001759 /* Set cause code. */
Benny Prijono0b6340c2006-06-13 22:21:23 +00001760 inv_set_cause(inv, st_code, st_text);
Benny Prijonod4e0abd2006-03-05 11:53:36 +00001761
Benny Prijono268ca612006-02-07 12:34:11 +00001762 /* Create appropriate message. */
1763 switch (inv->state) {
1764 case PJSIP_INV_STATE_CALLING:
1765 case PJSIP_INV_STATE_EARLY:
1766 case PJSIP_INV_STATE_INCOMING:
1767
1768 if (inv->role == PJSIP_ROLE_UAC) {
1769
1770 /* For UAC when session has not been confirmed, create CANCEL. */
1771
1772 /* MUST have the original UAC INVITE transaction. */
1773 PJ_ASSERT_RETURN(inv->invite_tsx != NULL, PJ_EBUG);
1774
1775 /* But CANCEL should only be called when we have received a
1776 * provisional response. If we haven't received any responses,
1777 * just destroy the transaction.
1778 */
1779 if (inv->invite_tsx->status_code < 100) {
1780
Benny Prijono1dc8be02007-05-30 04:26:40 +00001781 pjsip_tsx_stop_retransmit(inv->invite_tsx);
1782 inv->cancelling = PJ_TRUE;
1783 inv->pending_cancel = PJ_TRUE;
Benny Prijonofccab712006-02-22 22:23:22 +00001784 *p_tdata = NULL;
Benny Prijono1dc8be02007-05-30 04:26:40 +00001785 PJ_LOG(4, (inv->obj_name, "Stopping retransmission, "
1786 "delaying CANCEL"));
Benny Prijonofccab712006-02-22 22:23:22 +00001787 return PJ_SUCCESS;
Benny Prijono268ca612006-02-07 12:34:11 +00001788 }
1789
1790 /* The CSeq here assumes that the dialog is started with an
1791 * INVITE session. This may not be correct; dialog can be
1792 * started as SUBSCRIBE session.
1793 * So fix this!
1794 */
1795 status = pjsip_endpt_create_cancel(inv->dlg->endpt,
1796 inv->invite_tsx->last_tx,
1797 &tdata);
1798
1799 } else {
1800
1801 /* For UAS, send a final response. */
1802 tdata = inv->invite_tsx->last_tx;
1803 PJ_ASSERT_RETURN(tdata != NULL, PJ_EINVALIDOP);
1804
Benny Prijono26ff9062006-02-21 23:47:00 +00001805 //status = pjsip_dlg_modify_response(inv->dlg, tdata, st_code,
1806 // st_text);
1807 status = pjsip_inv_answer(inv, st_code, st_text, NULL, &tdata);
Benny Prijono268ca612006-02-07 12:34:11 +00001808 }
1809 break;
1810
1811 case PJSIP_INV_STATE_CONNECTING:
1812 case PJSIP_INV_STATE_CONFIRMED:
1813 /* For established dialog, send BYE */
Benny Prijono1f61a8f2007-08-16 10:11:44 +00001814 status = pjsip_dlg_create_request(inv->dlg, pjsip_get_bye_method(),
1815 -1, &tdata);
Benny Prijono268ca612006-02-07 12:34:11 +00001816 break;
1817
1818 case PJSIP_INV_STATE_DISCONNECTED:
Benny Prijono268ca612006-02-07 12:34:11 +00001819 /* No need to do anything. */
Benny Prijonod4e0abd2006-03-05 11:53:36 +00001820 return PJSIP_ESESSIONTERMINATED;
Benny Prijono268ca612006-02-07 12:34:11 +00001821
1822 default:
1823 pj_assert("!Invalid operation!");
1824 return PJ_EINVALIDOP;
1825 }
1826
1827 if (status != PJ_SUCCESS)
1828 return status;
1829
1830
1831 /* Done */
1832
Benny Prijono0606e702007-05-22 12:21:40 +00001833 inv->cancelling = PJ_TRUE;
Benny Prijono268ca612006-02-07 12:34:11 +00001834 *p_tdata = tdata;
1835
1836 return PJ_SUCCESS;
1837}
1838
1839
1840/*
1841 * Create re-INVITE.
1842 */
1843PJ_DEF(pj_status_t) pjsip_inv_reinvite( pjsip_inv_session *inv,
1844 const pj_str_t *new_contact,
1845 const pjmedia_sdp_session *new_offer,
1846 pjsip_tx_data **p_tdata )
1847{
Benny Prijono26ff9062006-02-21 23:47:00 +00001848 pj_status_t status;
1849 pjsip_contact_hdr *contact_hdr = NULL;
Benny Prijono268ca612006-02-07 12:34:11 +00001850
Benny Prijono26ff9062006-02-21 23:47:00 +00001851 /* Check arguments. */
1852 PJ_ASSERT_RETURN(inv && p_tdata, PJ_EINVAL);
1853
1854 /* Must NOT have a pending INVITE transaction */
Benny Prijono2b787822007-06-12 15:29:52 +00001855 if (inv->invite_tsx!=NULL)
1856 return PJ_EINVALIDOP;
Benny Prijono26ff9062006-02-21 23:47:00 +00001857
1858
1859 pjsip_dlg_inc_lock(inv->dlg);
1860
1861 if (new_contact) {
1862 pj_str_t tmp;
1863 const pj_str_t STR_CONTACT = { "Contact", 7 };
1864
1865 pj_strdup_with_null(inv->dlg->pool, &tmp, new_contact);
Benny Prijonoa1e69682007-05-11 15:14:34 +00001866 contact_hdr = (pjsip_contact_hdr*)
1867 pjsip_parse_hdr(inv->dlg->pool, &STR_CONTACT,
Benny Prijono26ff9062006-02-21 23:47:00 +00001868 tmp.ptr, tmp.slen, NULL);
1869 if (!contact_hdr) {
1870 status = PJSIP_EINVALIDURI;
1871 goto on_return;
1872 }
1873 }
1874
1875
1876 if (new_offer) {
1877 if (!inv->neg) {
1878 status = pjmedia_sdp_neg_create_w_local_offer(inv->pool, new_offer,
1879 &inv->neg);
1880 if (status != PJ_SUCCESS)
1881 goto on_return;
1882
1883 } else switch (pjmedia_sdp_neg_get_state(inv->neg)) {
1884
1885 case PJMEDIA_SDP_NEG_STATE_NULL:
1886 pj_assert(!"Unexpected SDP neg state NULL");
1887 status = PJ_EBUG;
1888 goto on_return;
1889
1890 case PJMEDIA_SDP_NEG_STATE_LOCAL_OFFER:
1891 PJ_LOG(4,(inv->obj_name,
1892 "pjsip_inv_reinvite: already have an offer, new "
1893 "offer is ignored"));
1894 break;
1895
1896 case PJMEDIA_SDP_NEG_STATE_REMOTE_OFFER:
1897 status = pjmedia_sdp_neg_set_local_answer(inv->pool, inv->neg,
1898 new_offer);
1899 if (status != PJ_SUCCESS)
1900 goto on_return;
1901 break;
1902
1903 case PJMEDIA_SDP_NEG_STATE_WAIT_NEGO:
1904 PJ_LOG(4,(inv->obj_name,
1905 "pjsip_inv_reinvite: SDP in WAIT_NEGO state, new "
1906 "offer is ignored"));
1907 break;
1908
1909 case PJMEDIA_SDP_NEG_STATE_DONE:
1910 status = pjmedia_sdp_neg_modify_local_offer(inv->pool,inv->neg,
1911 new_offer);
1912 if (status != PJ_SUCCESS)
1913 goto on_return;
1914 break;
1915 }
1916 }
1917
1918 if (contact_hdr)
1919 inv->dlg->local.contact = contact_hdr;
1920
1921 status = pjsip_inv_invite(inv, p_tdata);
1922
1923on_return:
1924 pjsip_dlg_dec_lock(inv->dlg);
1925 return status;
Benny Prijono268ca612006-02-07 12:34:11 +00001926}
1927
1928/*
1929 * Create UPDATE.
1930 */
1931PJ_DEF(pj_status_t) pjsip_inv_update ( pjsip_inv_session *inv,
1932 const pj_str_t *new_contact,
Benny Prijono1f7767b2007-10-03 18:28:49 +00001933 const pjmedia_sdp_session *offer,
Benny Prijono268ca612006-02-07 12:34:11 +00001934 pjsip_tx_data **p_tdata )
1935{
Benny Prijono1f7767b2007-10-03 18:28:49 +00001936 pjsip_contact_hdr *contact_hdr = NULL;
1937 pjsip_tx_data *tdata = NULL;
1938 pjmedia_sdp_session *sdp_copy;
1939 pj_status_t status = PJ_SUCCESS;
Benny Prijono268ca612006-02-07 12:34:11 +00001940
Benny Prijono1f7767b2007-10-03 18:28:49 +00001941 /* Verify arguments. */
1942 PJ_ASSERT_RETURN(inv && p_tdata && offer, PJ_EINVAL);
1943
1944 /* Dialog must have been established */
1945 PJ_ASSERT_RETURN(inv->dlg->state == PJSIP_DIALOG_STATE_ESTABLISHED,
1946 PJ_EINVALIDOP);
1947
1948 /* Invite session must not have been disconnected */
1949 PJ_ASSERT_RETURN(inv->state < PJSIP_INV_STATE_DISCONNECTED,
1950 PJ_EINVALIDOP);
1951
1952 /* Lock dialog. */
1953 pjsip_dlg_inc_lock(inv->dlg);
1954
1955 /* Process offer */
1956 if (pjmedia_sdp_neg_get_state(inv->neg)!=PJMEDIA_SDP_NEG_STATE_DONE) {
1957 PJ_LOG(4,(inv->dlg->obj_name,
1958 "Invalid SDP offer/answer state for UPDATE"));
1959 status = PJ_EINVALIDOP;
1960 goto on_error;
1961 }
1962
1963 status = pjmedia_sdp_neg_modify_local_offer(inv->pool,inv->neg,
1964 offer);
1965 if (status != PJ_SUCCESS)
1966 goto on_error;
1967
1968
1969 /* Update Contact if required */
1970 if (new_contact) {
1971 pj_str_t tmp;
1972 const pj_str_t STR_CONTACT = { "Contact", 7 };
1973
1974 pj_strdup_with_null(inv->dlg->pool, &tmp, new_contact);
1975 contact_hdr = (pjsip_contact_hdr*)
1976 pjsip_parse_hdr(inv->dlg->pool, &STR_CONTACT,
1977 tmp.ptr, tmp.slen, NULL);
1978 if (!contact_hdr) {
1979 status = PJSIP_EINVALIDURI;
1980 goto on_error;
1981 }
1982
1983 inv->dlg->local.contact = contact_hdr;
1984 }
1985
1986 /* Create request */
1987 status = pjsip_dlg_create_request(inv->dlg, &pjsip_update_method,
1988 -1, &tdata);
1989 if (status != PJ_SUCCESS)
1990 goto on_error;
1991
1992 /* Attach SDP body */
1993 sdp_copy = pjmedia_sdp_session_clone(tdata->pool, offer);
1994 pjsip_create_sdp_body(tdata->pool, sdp_copy, &tdata->msg->body);
1995
1996 /* Unlock dialog. */
1997 pjsip_dlg_dec_lock(inv->dlg);
1998
1999 *p_tdata = tdata;
2000
2001 return PJ_SUCCESS;
2002
2003on_error:
2004 if (tdata)
2005 pjsip_tx_data_dec_ref(tdata);
2006
2007 /* Unlock dialog. */
2008 pjsip_dlg_dec_lock(inv->dlg);
2009
2010 return status;
Benny Prijono268ca612006-02-07 12:34:11 +00002011}
2012
2013/*
Benny Prijonod5f9f422007-11-25 04:40:07 +00002014 * Create an ACK request.
2015 */
2016PJ_DEF(pj_status_t) pjsip_inv_create_ack(pjsip_inv_session *inv,
2017 int cseq,
2018 pjsip_tx_data **p_tdata)
2019{
2020 const pjmedia_sdp_session *sdp = NULL;
2021 pj_status_t status;
2022
2023 PJ_ASSERT_RETURN(inv && p_tdata, PJ_EINVAL);
2024
2025 /* Lock dialog. */
2026 pjsip_dlg_inc_lock(inv->dlg);
2027
2028 /* Destroy last_ack */
2029 if (inv->last_ack) {
2030 pjsip_tx_data_dec_ref(inv->last_ack);
2031 inv->last_ack = NULL;
2032 }
2033
2034 /* Create new ACK request */
2035 status = pjsip_dlg_create_request(inv->dlg, pjsip_get_ack_method(),
2036 cseq, &inv->last_ack);
2037 if (status != PJ_SUCCESS) {
2038 pjsip_dlg_dec_lock(inv->dlg);
2039 return status;
2040 }
2041
2042 /* See if we have pending SDP answer to send */
2043 sdp = inv_has_pending_answer(inv, inv->invite_tsx);
2044 if (sdp) {
2045 inv->last_ack->msg->body = create_sdp_body(inv->last_ack->pool, sdp);
2046 }
2047
2048 /* Keep this for subsequent response retransmission */
2049 inv->last_ack_cseq = cseq;
2050 pjsip_tx_data_add_ref(inv->last_ack);
2051
2052 /* Done */
2053 *p_tdata = inv->last_ack;
2054
2055 /* Unlock dialog. */
2056 pjsip_dlg_dec_lock(inv->dlg);
2057
2058 return PJ_SUCCESS;
2059}
2060
2061/*
Benny Prijono268ca612006-02-07 12:34:11 +00002062 * Send a request or response message.
2063 */
2064PJ_DEF(pj_status_t) pjsip_inv_send_msg( pjsip_inv_session *inv,
Benny Prijonoe8b0d3b2006-03-17 17:57:52 +00002065 pjsip_tx_data *tdata)
Benny Prijono268ca612006-02-07 12:34:11 +00002066{
2067 pj_status_t status;
2068
2069 /* Verify arguments. */
2070 PJ_ASSERT_RETURN(inv && tdata, PJ_EINVAL);
2071
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00002072 PJ_LOG(5,(inv->obj_name, "Sending %s",
2073 pjsip_tx_data_get_info(tdata)));
2074
Benny Prijono268ca612006-02-07 12:34:11 +00002075 if (tdata->msg->type == PJSIP_REQUEST_MSG) {
Benny Prijonoa66c7152006-02-09 01:26:14 +00002076 struct tsx_inv_data *tsx_inv_data;
Benny Prijono268ca612006-02-07 12:34:11 +00002077
Benny Prijono64158af2006-04-04 11:06:34 +00002078 pjsip_dlg_inc_lock(inv->dlg);
Benny Prijono268ca612006-02-07 12:34:11 +00002079
Benny Prijono22e48c92008-03-20 14:40:50 +00002080 /* Check again that we didn't receive incoming re-INVITE */
Benny Prijono9ae5dfc2008-03-27 17:30:51 +00002081 if (tdata->msg->line.req.method.id==PJSIP_INVITE_METHOD &&
2082 inv->invite_tsx)
2083 {
Benny Prijono22e48c92008-03-20 14:40:50 +00002084 pjsip_tx_data_dec_ref(tdata);
2085 pjsip_dlg_dec_lock(inv->dlg);
2086 return PJ_EINVALIDOP;
2087 }
2088
2089 /* Associate our data in outgoing invite transaction */
Benny Prijonoa1e69682007-05-11 15:14:34 +00002090 tsx_inv_data = PJ_POOL_ZALLOC_T(inv->pool, struct tsx_inv_data);
Benny Prijonoa66c7152006-02-09 01:26:14 +00002091 tsx_inv_data->inv = inv;
2092
Benny Prijono64158af2006-04-04 11:06:34 +00002093 pjsip_dlg_dec_lock(inv->dlg);
2094
2095 status = pjsip_dlg_send_request(inv->dlg, tdata, mod_inv.mod.id,
2096 tsx_inv_data);
2097 if (status != PJ_SUCCESS)
2098 return status;
Benny Prijono268ca612006-02-07 12:34:11 +00002099
2100 } else {
2101 pjsip_cseq_hdr *cseq;
2102
2103 /* Can only do this to send response to original INVITE
2104 * request.
2105 */
Benny Prijonoa1e69682007-05-11 15:14:34 +00002106 PJ_ASSERT_RETURN((cseq=(pjsip_cseq_hdr*)pjsip_msg_find_hdr(tdata->msg, PJSIP_H_CSEQ, NULL))!=NULL
Benny Prijono64f851e2006-02-23 13:49:28 +00002107 && (cseq->cseq == inv->invite_tsx->cseq),
Benny Prijono268ca612006-02-07 12:34:11 +00002108 PJ_EINVALIDOP);
2109
Benny Prijonodcfc0ba2007-09-30 16:50:27 +00002110 if (inv->options & PJSIP_INV_REQUIRE_100REL) {
Benny Prijono1f7767b2007-10-03 18:28:49 +00002111 status = pjsip_100rel_tx_response(inv, tdata);
Benny Prijonodcfc0ba2007-09-30 16:50:27 +00002112 } else
Benny Prijonodcfc0ba2007-09-30 16:50:27 +00002113 {
Benny Prijono1f7767b2007-10-03 18:28:49 +00002114 status = pjsip_dlg_send_response(inv->dlg, inv->invite_tsx, tdata);
Benny Prijonodcfc0ba2007-09-30 16:50:27 +00002115 }
2116
Benny Prijono268ca612006-02-07 12:34:11 +00002117 if (status != PJ_SUCCESS)
2118 return status;
2119 }
2120
2121 /* Done (?) */
2122 return PJ_SUCCESS;
2123}
2124
2125
Benny Prijono8ad55352006-02-08 11:16:05 +00002126/*
2127 * Respond to incoming CANCEL request.
2128 */
2129static void inv_respond_incoming_cancel(pjsip_inv_session *inv,
2130 pjsip_transaction *cancel_tsx,
2131 pjsip_rx_data *rdata)
2132{
2133 pjsip_tx_data *tdata;
2134 pjsip_transaction *invite_tsx;
2135 pj_str_t key;
2136 pj_status_t status;
2137
2138 /* See if we have matching INVITE server transaction: */
2139
2140 pjsip_tsx_create_key(rdata->tp_info.pool, &key, PJSIP_ROLE_UAS,
Benny Prijono1f61a8f2007-08-16 10:11:44 +00002141 pjsip_get_invite_method(), rdata);
Benny Prijono8ad55352006-02-08 11:16:05 +00002142 invite_tsx = pjsip_tsx_layer_find_tsx(&key, PJ_TRUE);
2143
2144 if (invite_tsx == NULL) {
2145
2146 /* Invite transaction not found!
Benny Prijonoc5145762007-11-23 12:04:40 +00002147 * Respond CANCEL with 481 (RFC 3261 Section 9.2 page 55)
Benny Prijono8ad55352006-02-08 11:16:05 +00002148 */
Benny Prijonoc5145762007-11-23 12:04:40 +00002149 status = pjsip_dlg_create_response( inv->dlg, rdata, 481, NULL,
Benny Prijono8ad55352006-02-08 11:16:05 +00002150 &tdata);
2151
2152 } else {
2153 /* Always answer CANCEL will 200 (OK) regardless of
2154 * the state of the INVITE transaction.
2155 */
2156 status = pjsip_dlg_create_response( inv->dlg, rdata, 200, NULL,
2157 &tdata);
2158 }
2159
2160 /* See if we have created the response successfully. */
2161 if (status != PJ_SUCCESS) return;
2162
2163 /* Send the CANCEL response */
2164 status = pjsip_dlg_send_response(inv->dlg, cancel_tsx, tdata);
2165 if (status != PJ_SUCCESS) return;
2166
2167
2168 /* See if we need to terminate the UAS INVITE transaction
2169 * with 487 (Request Terminated) response.
2170 */
2171 if (invite_tsx && invite_tsx->status_code < 200) {
2172
2173 pj_assert(invite_tsx->last_tx != NULL);
2174
2175 tdata = invite_tsx->last_tx;
2176
2177 status = pjsip_dlg_modify_response(inv->dlg, tdata, 487, NULL);
Benny Prijonofc8bb142007-11-08 09:56:50 +00002178 if (status == PJ_SUCCESS) {
2179 /* Remove the message body */
2180 tdata->msg->body = NULL;
Benny Prijono8ad55352006-02-08 11:16:05 +00002181 pjsip_dlg_send_response(inv->dlg, invite_tsx, tdata);
Benny Prijonofc8bb142007-11-08 09:56:50 +00002182 }
Benny Prijono8ad55352006-02-08 11:16:05 +00002183 }
2184
2185 if (invite_tsx)
2186 pj_mutex_unlock(invite_tsx->mutex);
2187}
2188
2189
2190/*
2191 * Respond to incoming BYE request.
2192 */
2193static void inv_respond_incoming_bye( pjsip_inv_session *inv,
2194 pjsip_transaction *bye_tsx,
2195 pjsip_rx_data *rdata,
2196 pjsip_event *e )
2197{
2198 pj_status_t status;
2199 pjsip_tx_data *tdata;
2200
2201 /* Respond BYE with 200: */
2202
2203 status = pjsip_dlg_create_response(inv->dlg, rdata, 200, NULL, &tdata);
2204 if (status != PJ_SUCCESS) return;
2205
2206 status = pjsip_dlg_send_response(inv->dlg, bye_tsx, tdata);
2207 if (status != PJ_SUCCESS) return;
2208
2209 /* Terminate session: */
2210
Benny Prijonod4e0abd2006-03-05 11:53:36 +00002211 if (inv->state != PJSIP_INV_STATE_DISCONNECTED) {
Benny Prijono0b6340c2006-06-13 22:21:23 +00002212 inv_set_cause(inv, PJSIP_SC_OK, NULL);
Benny Prijono8ad55352006-02-08 11:16:05 +00002213 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
Benny Prijonod4e0abd2006-03-05 11:53:36 +00002214 }
Benny Prijono8ad55352006-02-08 11:16:05 +00002215}
2216
2217/*
Benny Prijono38998232006-02-08 22:44:25 +00002218 * Respond to BYE request.
2219 */
2220static void inv_handle_bye_response( pjsip_inv_session *inv,
2221 pjsip_transaction *tsx,
2222 pjsip_rx_data *rdata,
2223 pjsip_event *e )
2224{
2225 pj_status_t status;
2226
2227 if (e->body.tsx_state.type != PJSIP_EVENT_RX_MSG) {
Benny Prijono0b6340c2006-06-13 22:21:23 +00002228 inv_set_cause(inv, PJSIP_SC_OK, NULL);
Benny Prijono38998232006-02-08 22:44:25 +00002229 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
2230 return;
2231 }
2232
2233 /* Handle 401/407 challenge. */
2234 if (tsx->status_code == 401 || tsx->status_code == 407) {
2235
2236 pjsip_tx_data *tdata;
2237
2238 status = pjsip_auth_clt_reinit_req( &inv->dlg->auth_sess,
2239 rdata,
2240 tsx->last_tx,
2241 &tdata);
2242
2243 if (status != PJ_SUCCESS) {
2244
2245 /* Does not have proper credentials.
2246 * End the session anyway.
2247 */
Benny Prijono0b6340c2006-06-13 22:21:23 +00002248 inv_set_cause(inv, PJSIP_SC_OK, NULL);
Benny Prijono38998232006-02-08 22:44:25 +00002249 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
2250
2251 } else {
2252 /* Re-send BYE. */
Benny Prijonoe8b0d3b2006-03-17 17:57:52 +00002253 status = pjsip_inv_send_msg(inv, tdata);
Benny Prijono38998232006-02-08 22:44:25 +00002254 }
2255
2256 } else {
2257
2258 /* End the session. */
Benny Prijono0b6340c2006-06-13 22:21:23 +00002259 inv_set_cause(inv, PJSIP_SC_OK, NULL);
Benny Prijono38998232006-02-08 22:44:25 +00002260 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
2261 }
2262
2263}
2264
2265/*
Benny Prijono1f7767b2007-10-03 18:28:49 +00002266 * Respond to incoming UPDATE request.
2267 */
2268static void inv_respond_incoming_update(pjsip_inv_session *inv,
2269 pjsip_rx_data *rdata)
2270{
2271 pjmedia_sdp_neg_state neg_state;
2272 pj_status_t status;
2273 pjsip_tx_data *tdata = NULL;
2274
2275 neg_state = pjmedia_sdp_neg_get_state(inv->neg);
2276
2277 /* Send 491 if we receive UPDATE while we're waiting for an answer */
2278 if (neg_state == PJMEDIA_SDP_NEG_STATE_LOCAL_OFFER) {
2279 status = pjsip_dlg_create_response(inv->dlg, rdata,
2280 PJSIP_SC_REQUEST_PENDING, NULL,
2281 &tdata);
2282 }
2283 /* Send 500 with Retry-After header set randomly between 0 and 10 if we
2284 * receive UPDATE while we haven't sent answer.
2285 */
2286 else if (neg_state == PJMEDIA_SDP_NEG_STATE_REMOTE_OFFER ||
2287 neg_state == PJMEDIA_SDP_NEG_STATE_WAIT_NEGO) {
2288 status = pjsip_dlg_create_response(inv->dlg, rdata,
2289 PJSIP_SC_INTERNAL_SERVER_ERROR,
2290 NULL, &tdata);
2291
Benny Prijonoc5cbc052007-11-08 09:44:08 +00002292 /* If UPDATE doesn't contain SDP, just respond with 200/OK.
2293 * This is a valid scenario according to session-timer draft.
2294 */
2295 } else if (rdata->msg_info.msg->body == NULL) {
2296
2297 status = pjsip_dlg_create_response(inv->dlg, rdata,
2298 200, NULL, &tdata);
2299
Benny Prijono1f7767b2007-10-03 18:28:49 +00002300 } else {
2301 /* We receive new offer from remote */
2302 inv_check_sdp_in_incoming_msg(inv, pjsip_rdata_get_tsx(rdata), rdata);
2303
2304 /* Application MUST have supplied the answer by now.
2305 * If so, negotiate the SDP.
2306 */
2307 neg_state = pjmedia_sdp_neg_get_state(inv->neg);
2308 if (neg_state != PJMEDIA_SDP_NEG_STATE_WAIT_NEGO ||
2309 (status=inv_negotiate_sdp(inv)) != PJ_SUCCESS)
2310 {
2311 /* Negotiation has failed */
2312 status = pjsip_dlg_create_response(inv->dlg, rdata,
2313 PJSIP_SC_NOT_ACCEPTABLE_HERE,
2314 NULL, &tdata);
2315 } else {
2316 /* New media has been negotiated successfully, send 200/OK */
2317 status = pjsip_dlg_create_response(inv->dlg, rdata,
2318 PJSIP_SC_OK, NULL, &tdata);
2319 if (status == PJ_SUCCESS) {
Benny Prijono9569a0b2007-10-04 15:35:26 +00002320 const pjmedia_sdp_session *sdp;
Benny Prijono1f7767b2007-10-03 18:28:49 +00002321 status = pjmedia_sdp_neg_get_active_local(inv->neg, &sdp);
2322 if (status == PJ_SUCCESS)
2323 tdata->msg->body = create_sdp_body(tdata->pool, sdp);
2324 }
2325 }
2326 }
2327
2328 if (status != PJ_SUCCESS) {
2329 if (tdata != NULL) {
2330 pjsip_tx_data_dec_ref(tdata);
2331 tdata = NULL;
2332 }
2333 return;
2334 }
2335
2336 pjsip_dlg_send_response(inv->dlg, pjsip_rdata_get_tsx(rdata), tdata);
2337}
2338
2339
2340/*
2341 * Handle incoming response to UAC UPDATE request.
2342 */
2343static void inv_handle_update_response( pjsip_inv_session *inv,
2344 pjsip_event *e)
2345{
2346 pjsip_transaction *tsx = e->body.tsx_state.tsx;
2347 struct tsx_inv_data *tsx_inv_data = NULL;
2348 pj_status_t status = -1;
2349
Benny Prijono48ab2b72007-11-08 09:24:30 +00002350 /* Handle 401/407 challenge. */
Benny Prijono1f7767b2007-10-03 18:28:49 +00002351 if (tsx->state == PJSIP_TSX_STATE_COMPLETED &&
Benny Prijono48ab2b72007-11-08 09:24:30 +00002352 (tsx->status_code == 401 || tsx->status_code == 407)) {
2353
2354 pjsip_tx_data *tdata;
2355
2356 status = pjsip_auth_clt_reinit_req( &inv->dlg->auth_sess,
2357 e->body.tsx_state.src.rdata,
2358 tsx->last_tx,
2359 &tdata);
2360
2361 if (status != PJ_SUCCESS) {
2362
Benny Prijonob2ad04a2008-06-26 13:24:10 +00002363 /* Somehow failed. Probably it's not a good idea to terminate
2364 * the session since this is just a request within dialog. And
2365 * even if we terminate we should send BYE.
Benny Prijono48ab2b72007-11-08 09:24:30 +00002366 */
Benny Prijonob2ad04a2008-06-26 13:24:10 +00002367 /*
Benny Prijono48ab2b72007-11-08 09:24:30 +00002368 inv_set_cause(inv, PJSIP_SC_OK, NULL);
2369 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
Benny Prijonob2ad04a2008-06-26 13:24:10 +00002370 */
Benny Prijono48ab2b72007-11-08 09:24:30 +00002371
2372 } else {
Benny Prijono0b4c57b2008-06-25 10:15:01 +00002373 /* Re-send request. */
Benny Prijono48ab2b72007-11-08 09:24:30 +00002374 status = pjsip_inv_send_msg(inv, tdata);
2375 }
2376
2377 /* Process 2xx response */
2378 } else if (tsx->state == PJSIP_TSX_STATE_COMPLETED &&
Benny Prijono1f7767b2007-10-03 18:28:49 +00002379 tsx->status_code/100 == 2 &&
2380 e->body.tsx_state.src.rdata->msg_info.msg->body)
2381 {
2382 status = inv_check_sdp_in_incoming_msg(inv, tsx,
2383 e->body.tsx_state.src.rdata);
2384
2385 } else {
2386 /* Get/attach invite session's transaction data */
2387 tsx_inv_data = (struct tsx_inv_data*)tsx->mod_data[mod_inv.mod.id];
2388 if (tsx_inv_data == NULL) {
2389 tsx_inv_data=PJ_POOL_ZALLOC_T(tsx->pool, struct tsx_inv_data);
2390 tsx_inv_data->inv = inv;
2391 tsx->mod_data[mod_inv.mod.id] = tsx_inv_data;
2392 }
2393 }
2394
2395 /* Otherwise if we don't get successful response, cancel
2396 * our negotiator.
2397 */
2398 if (status != PJ_SUCCESS &&
2399 pjmedia_sdp_neg_get_state(inv->neg)==PJMEDIA_SDP_NEG_STATE_LOCAL_OFFER &&
2400 tsx_inv_data && tsx_inv_data->sdp_done == PJ_FALSE)
2401 {
2402 pjmedia_sdp_neg_cancel_offer(inv->neg);
2403
2404 /* Prevent from us cancelling different offer! */
2405 tsx_inv_data->sdp_done = PJ_TRUE;
2406 }
2407}
2408
2409
2410/*
2411 * Handle incoming reliable response.
2412 */
2413static void inv_handle_incoming_reliable_response(pjsip_inv_session *inv,
2414 pjsip_rx_data *rdata)
2415{
2416 pjsip_tx_data *tdata;
Benny Prijono9569a0b2007-10-04 15:35:26 +00002417 const pjmedia_sdp_session *sdp;
Benny Prijono1f7767b2007-10-03 18:28:49 +00002418 pj_status_t status;
2419
2420 /* Create PRACK */
2421 status = pjsip_100rel_create_prack(inv, rdata, &tdata);
2422 if (status != PJ_SUCCESS)
2423 return;
2424
2425 /* See if we need to attach SDP answer on the PRACK request */
2426 sdp = inv_has_pending_answer(inv, pjsip_rdata_get_tsx(rdata));
2427 if (sdp) {
2428 tdata->msg->body = create_sdp_body(tdata->pool, sdp);
2429 }
2430
2431 /* Send PRACK (must be using 100rel module!) */
2432 pjsip_100rel_send_prack(inv, tdata);
2433}
2434
2435
2436/*
2437 * Handle incoming PRACK.
2438 */
2439static void inv_respond_incoming_prack(pjsip_inv_session *inv,
2440 pjsip_rx_data *rdata)
2441{
2442 pj_status_t status;
2443
2444 /* Run through 100rel module to see if we can accept this
2445 * PRACK request. The 100rel will send 200/OK to PRACK request.
2446 */
2447 status = pjsip_100rel_on_rx_prack(inv, rdata);
2448 if (status != PJ_SUCCESS)
2449 return;
2450
2451 /* Now check for SDP answer in the PRACK request */
2452 if (rdata->msg_info.msg->body) {
2453 status = inv_check_sdp_in_incoming_msg(inv,
2454 pjsip_rdata_get_tsx(rdata), rdata);
2455 } else {
2456 /* No SDP body */
2457 status = -1;
2458 }
2459
2460 /* If SDP negotiation has been successful, also mark the
2461 * SDP negotiation flag in the invite transaction to be
2462 * done too.
2463 */
2464 if (status == PJ_SUCCESS && inv->invite_tsx) {
2465 struct tsx_inv_data *tsx_inv_data;
2466
2467 /* Get/attach invite session's transaction data */
2468 tsx_inv_data = (struct tsx_inv_data*)
2469 inv->invite_tsx->mod_data[mod_inv.mod.id];
2470 if (tsx_inv_data == NULL) {
2471 tsx_inv_data = PJ_POOL_ZALLOC_T(inv->invite_tsx->pool,
2472 struct tsx_inv_data);
2473 tsx_inv_data->inv = inv;
2474 inv->invite_tsx->mod_data[mod_inv.mod.id] = tsx_inv_data;
2475 }
2476
2477 tsx_inv_data->sdp_done = PJ_TRUE;
2478 }
2479}
2480
2481
2482/*
Benny Prijono8ad55352006-02-08 11:16:05 +00002483 * State NULL is before anything is sent/received.
2484 */
2485static void inv_on_state_null( pjsip_inv_session *inv, pjsip_event *e)
Benny Prijono268ca612006-02-07 12:34:11 +00002486{
2487 pjsip_transaction *tsx = e->body.tsx_state.tsx;
2488 pjsip_dialog *dlg = pjsip_tsx_get_dlg(tsx);
2489
2490 PJ_ASSERT_ON_FAIL(tsx && dlg, return);
2491
2492 if (tsx->method.id == PJSIP_INVITE_METHOD) {
2493
Benny Prijono64f851e2006-02-23 13:49:28 +00002494 /* Keep the initial INVITE transaction. */
2495 if (inv->invite_tsx == NULL)
2496 inv->invite_tsx = tsx;
Benny Prijono268ca612006-02-07 12:34:11 +00002497
Benny Prijono64f851e2006-02-23 13:49:28 +00002498 if (dlg->role == PJSIP_ROLE_UAC) {
Benny Prijono268ca612006-02-07 12:34:11 +00002499
2500 switch (tsx->state) {
2501 case PJSIP_TSX_STATE_CALLING:
Benny Prijono8ad55352006-02-08 11:16:05 +00002502 inv_set_state(inv, PJSIP_INV_STATE_CALLING, e);
Benny Prijono268ca612006-02-07 12:34:11 +00002503 break;
2504 default:
Benny Prijono64f851e2006-02-23 13:49:28 +00002505 inv_on_state_calling(inv, e);
Benny Prijono268ca612006-02-07 12:34:11 +00002506 break;
2507 }
2508
2509 } else {
2510 switch (tsx->state) {
2511 case PJSIP_TSX_STATE_TRYING:
Benny Prijono8ad55352006-02-08 11:16:05 +00002512 inv_set_state(inv, PJSIP_INV_STATE_INCOMING, e);
Benny Prijono268ca612006-02-07 12:34:11 +00002513 break;
Benny Prijono38998232006-02-08 22:44:25 +00002514 case PJSIP_TSX_STATE_PROCEEDING:
2515 inv_set_state(inv, PJSIP_INV_STATE_INCOMING, e);
2516 if (tsx->status_code > 100)
2517 inv_set_state(inv, PJSIP_INV_STATE_EARLY, e);
2518 break;
Benny Prijono268ca612006-02-07 12:34:11 +00002519 default:
Benny Prijono64f851e2006-02-23 13:49:28 +00002520 inv_on_state_incoming(inv, e);
2521 break;
Benny Prijono268ca612006-02-07 12:34:11 +00002522 }
2523 }
2524
2525 } else {
2526 pj_assert(!"Unexpected transaction type");
2527 }
2528}
2529
Benny Prijono8ad55352006-02-08 11:16:05 +00002530/*
Benny Prijono0b4c57b2008-06-25 10:15:01 +00002531 * Generic UAC transaction handler:
2532 * - resend request on 401 or 407 response.
2533 * - terminate dialog on 408 and 481 response.
2534 */
2535static pj_bool_t handle_uac_tsx_response(pjsip_inv_session *inv,
2536 pjsip_event *e)
2537{
2538 /* RFC 3261 Section 12.2.1.2:
2539 * If the response for a request within a dialog is a 481
2540 * (Call/Transaction Does Not Exist) or a 408 (Request Timeout), the UAC
2541 * SHOULD terminate the dialog. A UAC SHOULD also terminate a dialog if
2542 * no response at all is received for the request (the client
2543 * transaction would inform the TU about the timeout.)
2544 *
2545 * For INVITE initiated dialogs, terminating the dialog consists of
2546 * sending a BYE.
2547 *
2548 * Note:
2549 * according to X, this should terminate dialog usage only, not the
2550 * dialog.
2551 */
2552 pjsip_transaction *tsx = e->body.tsx_state.tsx;
2553
2554 pj_assert(tsx->role == PJSIP_UAC_ROLE);
2555
2556 /* Note that 481 response to CANCEL does not terminate dialog usage,
2557 * but only the transaction.
2558 */
Benny Prijono61fc5e62008-06-25 18:35:31 +00002559 if (inv->state != PJSIP_INV_STATE_DISCONNECTED &&
2560 ((tsx->status_code == PJSIP_SC_CALL_TSX_DOES_NOT_EXIST &&
Benny Prijono0b4c57b2008-06-25 10:15:01 +00002561 tsx->method.id != PJSIP_CANCEL_METHOD) ||
Benny Prijono61fc5e62008-06-25 18:35:31 +00002562 tsx->status_code == PJSIP_SC_REQUEST_TIMEOUT ||
2563 tsx->status_code == PJSIP_SC_TSX_TIMEOUT ||
2564 tsx->status_code == PJSIP_SC_TSX_TRANSPORT_ERROR))
Benny Prijono0b4c57b2008-06-25 10:15:01 +00002565 {
2566 pjsip_tx_data *bye;
2567 pj_status_t status;
2568
2569 inv_set_cause(inv, tsx->status_code, &tsx->status_text);
2570 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
2571
2572 /* Send BYE */
2573 status = pjsip_dlg_create_request(inv->dlg, pjsip_get_bye_method(),
2574 -1, &bye);
2575 if (status == PJ_SUCCESS) {
2576 pjsip_inv_send_msg(inv, bye);
2577 }
2578
2579 return PJ_TRUE; /* Handled */
2580
2581 }
2582 /* Handle 401/407 challenge. */
2583 else if (tsx->state == PJSIP_TSX_STATE_COMPLETED &&
2584 (tsx->status_code == PJSIP_SC_UNAUTHORIZED ||
2585 tsx->status_code == PJSIP_SC_PROXY_AUTHENTICATION_REQUIRED))
2586 {
2587
2588 pjsip_tx_data *tdata;
2589 pj_status_t status;
2590
Benny Prijonob2ad04a2008-06-26 13:24:10 +00002591 if (tsx->method.id == PJSIP_INVITE_METHOD)
2592 inv->invite_tsx = NULL;
2593
Benny Prijono0b4c57b2008-06-25 10:15:01 +00002594 status = pjsip_auth_clt_reinit_req( &inv->dlg->auth_sess,
2595 e->body.tsx_state.src.rdata,
2596 tsx->last_tx, &tdata);
2597
2598 if (status != PJ_SUCCESS) {
Benny Prijonob2ad04a2008-06-26 13:24:10 +00002599 /* Somehow failed. Probably it's not a good idea to terminate
2600 * the session since this is just a request within dialog. And
2601 * even if we terminate we should send BYE.
2602 */
2603 /*
Benny Prijono0b4c57b2008-06-25 10:15:01 +00002604 inv_set_cause(inv, PJSIP_SC_OK, NULL);
2605 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
Benny Prijonob2ad04a2008-06-26 13:24:10 +00002606 */
Benny Prijono0b4c57b2008-06-25 10:15:01 +00002607
2608 } else {
2609 /* Re-send request. */
Benny Prijono0b4c57b2008-06-25 10:15:01 +00002610 status = pjsip_inv_send_msg(inv, tdata);
2611 }
2612
2613 return PJ_TRUE; /* Handled */
2614
2615 } else {
2616 return PJ_FALSE; /* Unhandled */
2617 }
2618}
2619
2620
2621/*
Benny Prijono8ad55352006-02-08 11:16:05 +00002622 * State CALLING is after sending initial INVITE request but before
2623 * any response (with tag) is received.
2624 */
2625static void inv_on_state_calling( pjsip_inv_session *inv, pjsip_event *e)
Benny Prijono268ca612006-02-07 12:34:11 +00002626{
2627 pjsip_transaction *tsx = e->body.tsx_state.tsx;
2628 pjsip_dialog *dlg = pjsip_tsx_get_dlg(tsx);
Benny Prijonoccf95622006-02-07 18:48:01 +00002629 pj_status_t status;
Benny Prijono268ca612006-02-07 12:34:11 +00002630
2631 PJ_ASSERT_ON_FAIL(tsx && dlg, return);
2632
Benny Prijono8ad55352006-02-08 11:16:05 +00002633 if (tsx == inv->invite_tsx) {
Benny Prijono268ca612006-02-07 12:34:11 +00002634
2635 switch (tsx->state) {
2636
Benny Prijono64f851e2006-02-23 13:49:28 +00002637 case PJSIP_TSX_STATE_CALLING:
2638 inv_set_state(inv, PJSIP_INV_STATE_CALLING, e);
2639 break;
2640
Benny Prijono268ca612006-02-07 12:34:11 +00002641 case PJSIP_TSX_STATE_PROCEEDING:
Benny Prijono1dc8be02007-05-30 04:26:40 +00002642 if (inv->pending_cancel) {
2643 pjsip_tx_data *cancel;
2644
2645 inv->pending_cancel = PJ_FALSE;
2646
2647 status = pjsip_inv_end_session(inv, 487, NULL, &cancel);
2648 if (status == PJ_SUCCESS && cancel)
2649 status = pjsip_inv_send_msg(inv, cancel);
2650 }
2651
Benny Prijono268ca612006-02-07 12:34:11 +00002652 if (dlg->remote.info->tag.slen) {
Benny Prijonoa66c7152006-02-09 01:26:14 +00002653
Benny Prijono8ad55352006-02-08 11:16:05 +00002654 inv_set_state(inv, PJSIP_INV_STATE_EARLY, e);
Benny Prijonoa66c7152006-02-09 01:26:14 +00002655
2656 inv_check_sdp_in_incoming_msg(inv, tsx,
2657 e->body.tsx_state.src.rdata);
2658
Benny Prijono1f7767b2007-10-03 18:28:49 +00002659 if (pjsip_100rel_is_reliable(e->body.tsx_state.src.rdata)) {
2660 inv_handle_incoming_reliable_response(
2661 inv, e->body.tsx_state.src.rdata);
2662 }
2663
Benny Prijono268ca612006-02-07 12:34:11 +00002664 } else {
2665 /* Ignore 100 (Trying) response, as it doesn't change
2666 * session state. It only ceases retransmissions.
2667 */
2668 }
2669 break;
2670
2671 case PJSIP_TSX_STATE_COMPLETED:
2672 if (tsx->status_code/100 == 2) {
2673
2674 /* This should not happen.
2675 * When transaction receives 2xx, it should be terminated
2676 */
2677 pj_assert(0);
Benny Prijono8ad55352006-02-08 11:16:05 +00002678 inv_set_state(inv, PJSIP_INV_STATE_CONNECTING, e);
Benny Prijonoa66c7152006-02-09 01:26:14 +00002679
2680 inv_check_sdp_in_incoming_msg(inv, tsx,
2681 e->body.tsx_state.src.rdata);
Benny Prijono268ca612006-02-07 12:34:11 +00002682
Benny Prijono0606e702007-05-22 12:21:40 +00002683 } else if ((tsx->status_code==401 || tsx->status_code==407) &&
2684 !inv->cancelling)
2685 {
Benny Prijonoccf95622006-02-07 18:48:01 +00002686
2687 /* Handle authentication failure:
2688 * Resend the request with Authorization header.
2689 */
2690 pjsip_tx_data *tdata;
2691
Benny Prijono8ad55352006-02-08 11:16:05 +00002692 status = pjsip_auth_clt_reinit_req(&inv->dlg->auth_sess,
Benny Prijonoccf95622006-02-07 18:48:01 +00002693 e->body.tsx_state.src.rdata,
2694 tsx->last_tx,
2695 &tdata);
2696
2697 if (status != PJ_SUCCESS) {
2698
2699 /* Does not have proper credentials.
2700 * End the session.
2701 */
Benny Prijono0b6340c2006-06-13 22:21:23 +00002702 inv_set_cause(inv, tsx->status_code, &tsx->status_text);
Benny Prijono8ad55352006-02-08 11:16:05 +00002703 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
Benny Prijonoccf95622006-02-07 18:48:01 +00002704
2705 } else {
2706
2707 /* Restart session. */
Benny Prijono8ad55352006-02-08 11:16:05 +00002708 inv->state = PJSIP_INV_STATE_NULL;
2709 inv->invite_tsx = NULL;
Benny Prijonodcfc0ba2007-09-30 16:50:27 +00002710 if (inv->last_answer) {
2711 pjsip_tx_data_dec_ref(inv->last_answer);
2712 inv->last_answer = NULL;
2713 }
Benny Prijonoccf95622006-02-07 18:48:01 +00002714
2715 /* Send the request. */
Benny Prijonoe8b0d3b2006-03-17 17:57:52 +00002716 status = pjsip_inv_send_msg(inv, tdata);
Benny Prijonoccf95622006-02-07 18:48:01 +00002717 }
2718
Benny Prijono268ca612006-02-07 12:34:11 +00002719 } else {
Benny Prijonoccf95622006-02-07 18:48:01 +00002720
Benny Prijono0b6340c2006-06-13 22:21:23 +00002721 inv_set_cause(inv, tsx->status_code, &tsx->status_text);
Benny Prijono8ad55352006-02-08 11:16:05 +00002722 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
Benny Prijonoccf95622006-02-07 18:48:01 +00002723
Benny Prijono268ca612006-02-07 12:34:11 +00002724 }
2725 break;
2726
2727 case PJSIP_TSX_STATE_TERMINATED:
2728 /* INVITE transaction can be terminated either because UAC
2729 * transaction received 2xx response or because of transport
2730 * error.
2731 */
2732 if (tsx->status_code/100 == 2) {
2733 /* This must be receipt of 2xx response */
2734
2735 /* Set state to CONNECTING */
Benny Prijono8ad55352006-02-08 11:16:05 +00002736 inv_set_state(inv, PJSIP_INV_STATE_CONNECTING, e);
Benny Prijono268ca612006-02-07 12:34:11 +00002737
Benny Prijonoa66c7152006-02-09 01:26:14 +00002738 inv_check_sdp_in_incoming_msg(inv, tsx,
2739 e->body.tsx_state.src.rdata);
2740
Benny Prijono268ca612006-02-07 12:34:11 +00002741 /* Send ACK */
2742 pj_assert(e->body.tsx_state.type == PJSIP_EVENT_RX_MSG);
2743
Benny Prijonod5f9f422007-11-25 04:40:07 +00002744 inv_send_ack(inv, e);
Benny Prijonoa66c7152006-02-09 01:26:14 +00002745
Benny Prijono268ca612006-02-07 12:34:11 +00002746 } else {
Benny Prijono0b6340c2006-06-13 22:21:23 +00002747 inv_set_cause(inv, tsx->status_code, &tsx->status_text);
Benny Prijono8ad55352006-02-08 11:16:05 +00002748 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
Benny Prijono268ca612006-02-07 12:34:11 +00002749 }
2750 break;
2751
Benny Prijono34a404e2006-02-09 14:38:30 +00002752 default:
2753 break;
Benny Prijono268ca612006-02-07 12:34:11 +00002754 }
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00002755
Benny Prijono1f7767b2007-10-03 18:28:49 +00002756 } else if (tsx->role == PJSIP_ROLE_UAC) {
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00002757 /*
Benny Prijono1f7767b2007-10-03 18:28:49 +00002758 * Handle case when outgoing request is answered with 481 (Call/
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00002759 * Transaction Does Not Exist), 408, or when it's timed out. In these
2760 * cases, disconnect session (i.e. dialog usage only).
Benny Prijonoc5145762007-11-23 12:04:40 +00002761 * Note that 481 response to CANCEL does not terminate dialog usage,
2762 * but only the transaction.
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00002763 */
Benny Prijonoc5145762007-11-23 12:04:40 +00002764 if ((tsx->status_code == PJSIP_SC_CALL_TSX_DOES_NOT_EXIST &&
2765 tsx->method.id != PJSIP_CANCEL_METHOD) ||
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00002766 tsx->status_code == PJSIP_SC_REQUEST_TIMEOUT ||
2767 tsx->status_code == PJSIP_SC_TSX_TIMEOUT ||
Benny Prijono0b6340c2006-06-13 22:21:23 +00002768 tsx->status_code == PJSIP_SC_TSX_TRANSPORT_ERROR)
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00002769 {
Benny Prijono0b6340c2006-06-13 22:21:23 +00002770 inv_set_cause(inv, tsx->status_code, &tsx->status_text);
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00002771 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
2772 }
Benny Prijono268ca612006-02-07 12:34:11 +00002773 }
2774}
2775
Benny Prijono8ad55352006-02-08 11:16:05 +00002776/*
2777 * State INCOMING is after we received the request, but before
2778 * responses with tag are sent.
2779 */
2780static void inv_on_state_incoming( pjsip_inv_session *inv, pjsip_event *e)
Benny Prijono268ca612006-02-07 12:34:11 +00002781{
2782 pjsip_transaction *tsx = e->body.tsx_state.tsx;
2783 pjsip_dialog *dlg = pjsip_tsx_get_dlg(tsx);
2784
2785 PJ_ASSERT_ON_FAIL(tsx && dlg, return);
2786
Benny Prijono8ad55352006-02-08 11:16:05 +00002787 if (tsx == inv->invite_tsx) {
2788
2789 /*
2790 * Handle the INVITE state transition.
2791 */
2792
Benny Prijono268ca612006-02-07 12:34:11 +00002793 switch (tsx->state) {
Benny Prijono8ad55352006-02-08 11:16:05 +00002794
Benny Prijono64f851e2006-02-23 13:49:28 +00002795 case PJSIP_TSX_STATE_TRYING:
2796 inv_set_state(inv, PJSIP_INV_STATE_INCOMING, e);
2797 break;
2798
Benny Prijono268ca612006-02-07 12:34:11 +00002799 case PJSIP_TSX_STATE_PROCEEDING:
Benny Prijono8ad55352006-02-08 11:16:05 +00002800 /*
2801 * Transaction sent provisional response.
2802 */
Benny Prijono268ca612006-02-07 12:34:11 +00002803 if (tsx->status_code > 100)
Benny Prijono8ad55352006-02-08 11:16:05 +00002804 inv_set_state(inv, PJSIP_INV_STATE_EARLY, e);
Benny Prijono268ca612006-02-07 12:34:11 +00002805 break;
Benny Prijono8ad55352006-02-08 11:16:05 +00002806
Benny Prijono268ca612006-02-07 12:34:11 +00002807 case PJSIP_TSX_STATE_COMPLETED:
Benny Prijono8ad55352006-02-08 11:16:05 +00002808 /*
2809 * Transaction sent final response.
2810 */
Benny Prijonod4e0abd2006-03-05 11:53:36 +00002811 if (tsx->status_code/100 == 2) {
Benny Prijono8ad55352006-02-08 11:16:05 +00002812 inv_set_state(inv, PJSIP_INV_STATE_CONNECTING, e);
Benny Prijonod4e0abd2006-03-05 11:53:36 +00002813 } else {
Benny Prijono0b6340c2006-06-13 22:21:23 +00002814 inv_set_cause(inv, tsx->status_code, &tsx->status_text);
Benny Prijono8ad55352006-02-08 11:16:05 +00002815 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
Benny Prijonod4e0abd2006-03-05 11:53:36 +00002816 }
Benny Prijono268ca612006-02-07 12:34:11 +00002817 break;
Benny Prijono8ad55352006-02-08 11:16:05 +00002818
Benny Prijono268ca612006-02-07 12:34:11 +00002819 case PJSIP_TSX_STATE_TERMINATED:
Benny Prijono8ad55352006-02-08 11:16:05 +00002820 /*
2821 * This happens on transport error (e.g. failed to send
2822 * response)
2823 */
Benny Prijono0b6340c2006-06-13 22:21:23 +00002824 inv_set_cause(inv, tsx->status_code, &tsx->status_text);
Benny Prijono8ad55352006-02-08 11:16:05 +00002825 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
Benny Prijono268ca612006-02-07 12:34:11 +00002826 break;
Benny Prijono8ad55352006-02-08 11:16:05 +00002827
Benny Prijono268ca612006-02-07 12:34:11 +00002828 default:
Benny Prijono8ad55352006-02-08 11:16:05 +00002829 pj_assert(!"Unexpected INVITE state");
2830 break;
Benny Prijono268ca612006-02-07 12:34:11 +00002831 }
Benny Prijono8ad55352006-02-08 11:16:05 +00002832
2833 } else if (tsx->method.id == PJSIP_CANCEL_METHOD &&
2834 tsx->role == PJSIP_ROLE_UAS &&
2835 tsx->state < PJSIP_TSX_STATE_COMPLETED &&
2836 e->body.tsx_state.type == PJSIP_EVENT_RX_MSG )
2837 {
2838
2839 /*
2840 * Handle incoming CANCEL request.
2841 */
2842
2843 inv_respond_incoming_cancel(inv, tsx, e->body.tsx_state.src.rdata);
2844
Benny Prijono268ca612006-02-07 12:34:11 +00002845 }
2846}
2847
Benny Prijono8ad55352006-02-08 11:16:05 +00002848/*
2849 * State EARLY is for both UAS and UAC, after response with To tag
2850 * is sent/received.
2851 */
2852static void inv_on_state_early( pjsip_inv_session *inv, pjsip_event *e)
Benny Prijono268ca612006-02-07 12:34:11 +00002853{
2854 pjsip_transaction *tsx = e->body.tsx_state.tsx;
2855 pjsip_dialog *dlg = pjsip_tsx_get_dlg(tsx);
2856
2857 PJ_ASSERT_ON_FAIL(tsx && dlg, return);
2858
Benny Prijono8ad55352006-02-08 11:16:05 +00002859 if (tsx == inv->invite_tsx) {
2860
2861 /*
2862 * Handle the INVITE state progress.
2863 */
Benny Prijono268ca612006-02-07 12:34:11 +00002864
2865 switch (tsx->state) {
2866
2867 case PJSIP_TSX_STATE_PROCEEDING:
2868 /* Send/received another provisional response. */
Benny Prijono8ad55352006-02-08 11:16:05 +00002869 inv_set_state(inv, PJSIP_INV_STATE_EARLY, e);
Benny Prijonoa66c7152006-02-09 01:26:14 +00002870
2871 if (e->body.tsx_state.type == PJSIP_EVENT_RX_MSG) {
2872 inv_check_sdp_in_incoming_msg(inv, tsx,
2873 e->body.tsx_state.src.rdata);
Benny Prijono1f7767b2007-10-03 18:28:49 +00002874
2875 if (pjsip_100rel_is_reliable(e->body.tsx_state.src.rdata)) {
2876 inv_handle_incoming_reliable_response(
2877 inv, e->body.tsx_state.src.rdata);
2878 }
Benny Prijonoa66c7152006-02-09 01:26:14 +00002879 }
Benny Prijono268ca612006-02-07 12:34:11 +00002880 break;
2881
2882 case PJSIP_TSX_STATE_COMPLETED:
Benny Prijonoa66c7152006-02-09 01:26:14 +00002883 if (tsx->status_code/100 == 2) {
Benny Prijono8ad55352006-02-08 11:16:05 +00002884 inv_set_state(inv, PJSIP_INV_STATE_CONNECTING, e);
Benny Prijonoa66c7152006-02-09 01:26:14 +00002885 if (e->body.tsx_state.type == PJSIP_EVENT_RX_MSG) {
2886 inv_check_sdp_in_incoming_msg(inv, tsx,
2887 e->body.tsx_state.src.rdata);
2888 }
2889
Benny Prijonod4e0abd2006-03-05 11:53:36 +00002890 } else {
Benny Prijono0b6340c2006-06-13 22:21:23 +00002891 inv_set_cause(inv, tsx->status_code, &tsx->status_text);
Benny Prijono8ad55352006-02-08 11:16:05 +00002892 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
Benny Prijonod4e0abd2006-03-05 11:53:36 +00002893 }
Benny Prijono268ca612006-02-07 12:34:11 +00002894 break;
2895
Benny Prijonof3195072006-02-14 21:15:30 +00002896 case PJSIP_TSX_STATE_CONFIRMED:
Benny Prijono7d910092007-06-20 04:19:46 +00002897 /* For some reason can go here (maybe when ACK for 2xx has
2898 * the same branch value as the INVITE transaction) */
Benny Prijonof3195072006-02-14 21:15:30 +00002899
Benny Prijono268ca612006-02-07 12:34:11 +00002900 case PJSIP_TSX_STATE_TERMINATED:
2901 /* INVITE transaction can be terminated either because UAC
2902 * transaction received 2xx response or because of transport
2903 * error.
2904 */
2905 if (tsx->status_code/100 == 2) {
2906
2907 /* This must be receipt of 2xx response */
2908
2909 /* Set state to CONNECTING */
Benny Prijono8ad55352006-02-08 11:16:05 +00002910 inv_set_state(inv, PJSIP_INV_STATE_CONNECTING, e);
Benny Prijono268ca612006-02-07 12:34:11 +00002911
Benny Prijonoa66c7152006-02-09 01:26:14 +00002912 if (e->body.tsx_state.type == PJSIP_EVENT_RX_MSG) {
2913 inv_check_sdp_in_incoming_msg(inv, tsx,
2914 e->body.tsx_state.src.rdata);
2915 }
2916
Benny Prijono268ca612006-02-07 12:34:11 +00002917 /* if UAC, send ACK and move state to confirmed. */
2918 if (tsx->role == PJSIP_ROLE_UAC) {
2919 pj_assert(e->body.tsx_state.type == PJSIP_EVENT_RX_MSG);
2920
Benny Prijonod5f9f422007-11-25 04:40:07 +00002921 inv_send_ack(inv, e);
Benny Prijono268ca612006-02-07 12:34:11 +00002922 }
2923
2924 } else {
Benny Prijono0b6340c2006-06-13 22:21:23 +00002925 inv_set_cause(inv, tsx->status_code, &tsx->status_text);
Benny Prijono8ad55352006-02-08 11:16:05 +00002926 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
Benny Prijono268ca612006-02-07 12:34:11 +00002927 }
2928 break;
2929
2930 default:
Benny Prijono8ad55352006-02-08 11:16:05 +00002931 pj_assert(!"Unexpected INVITE tsx state");
Benny Prijono268ca612006-02-07 12:34:11 +00002932 }
2933
Benny Prijono8ad55352006-02-08 11:16:05 +00002934 } else if (inv->role == PJSIP_ROLE_UAS &&
2935 tsx->role == PJSIP_ROLE_UAS &&
2936 tsx->method.id == PJSIP_CANCEL_METHOD &&
2937 tsx->state < PJSIP_TSX_STATE_COMPLETED &&
2938 e->body.tsx_state.type == PJSIP_EVENT_RX_MSG )
2939 {
Benny Prijono268ca612006-02-07 12:34:11 +00002940
Benny Prijono8ad55352006-02-08 11:16:05 +00002941 /*
2942 * Handle incoming CANCEL request.
Benny Prijonoccf95622006-02-07 18:48:01 +00002943 */
2944
Benny Prijono8ad55352006-02-08 11:16:05 +00002945 inv_respond_incoming_cancel(inv, tsx, e->body.tsx_state.src.rdata);
2946
Benny Prijono1f7767b2007-10-03 18:28:49 +00002947 } else if (tsx->role == PJSIP_ROLE_UAS &&
2948 tsx->state == PJSIP_TSX_STATE_TRYING &&
2949 pjsip_method_cmp(&tsx->method, &pjsip_update_method)==0)
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00002950 {
2951 /*
Benny Prijono1f7767b2007-10-03 18:28:49 +00002952 * Handle incoming UPDATE
2953 */
2954 inv_respond_incoming_update(inv, e->body.tsx_state.src.rdata);
2955
2956
2957 } else if (tsx->role == PJSIP_ROLE_UAC &&
2958 (tsx->state == PJSIP_TSX_STATE_COMPLETED ||
2959 tsx->state == PJSIP_TSX_STATE_TERMINATED) &&
2960 pjsip_method_cmp(&tsx->method, &pjsip_update_method)==0)
2961 {
2962 /*
2963 * Handle response to outgoing UPDATE request.
2964 */
2965 inv_handle_update_response(inv, e);
2966
2967 } else if (tsx->role == PJSIP_ROLE_UAS &&
2968 tsx->state == PJSIP_TSX_STATE_TRYING &&
2969 pjsip_method_cmp(&tsx->method, &pjsip_prack_method)==0)
2970 {
2971 /*
2972 * Handle incoming PRACK
2973 */
2974 inv_respond_incoming_prack(inv, e->body.tsx_state.src.rdata);
2975
2976 } else if (tsx->role == PJSIP_ROLE_UAC) {
Benny Prijono0b4c57b2008-06-25 10:15:01 +00002977
2978 /* Generic handling for UAC tsx completion */
2979 handle_uac_tsx_response(inv, e);
Benny Prijono268ca612006-02-07 12:34:11 +00002980 }
2981}
2982
Benny Prijono8ad55352006-02-08 11:16:05 +00002983/*
2984 * State CONNECTING is after 2xx response to INVITE is sent/received.
2985 */
2986static void inv_on_state_connecting( pjsip_inv_session *inv, pjsip_event *e)
Benny Prijono268ca612006-02-07 12:34:11 +00002987{
2988 pjsip_transaction *tsx = e->body.tsx_state.tsx;
2989 pjsip_dialog *dlg = pjsip_tsx_get_dlg(tsx);
2990
2991 PJ_ASSERT_ON_FAIL(tsx && dlg, return);
2992
Benny Prijono8ad55352006-02-08 11:16:05 +00002993 if (tsx == inv->invite_tsx) {
Benny Prijono268ca612006-02-07 12:34:11 +00002994
Benny Prijono8ad55352006-02-08 11:16:05 +00002995 /*
2996 * Handle INVITE state progression.
2997 */
Benny Prijono268ca612006-02-07 12:34:11 +00002998 switch (tsx->state) {
2999
3000 case PJSIP_TSX_STATE_CONFIRMED:
Benny Prijono7d910092007-06-20 04:19:46 +00003001 /* It can only go here if incoming ACK request has the same Via
3002 * branch parameter as the INVITE transaction.
3003 */
3004 if (tsx->status_code/100 == 2) {
3005 if (e->body.tsx_state.type == PJSIP_EVENT_RX_MSG) {
3006 inv_check_sdp_in_incoming_msg(inv, tsx,
3007 e->body.tsx_state.src.rdata);
3008 }
3009
Benny Prijono38998232006-02-08 22:44:25 +00003010 inv_set_state(inv, PJSIP_INV_STATE_CONFIRMED, e);
Benny Prijono7d910092007-06-20 04:19:46 +00003011 }
Benny Prijono268ca612006-02-07 12:34:11 +00003012 break;
3013
3014 case PJSIP_TSX_STATE_TERMINATED:
3015 /* INVITE transaction can be terminated either because UAC
3016 * transaction received 2xx response or because of transport
3017 * error.
3018 */
3019 if (tsx->status_code/100 != 2) {
Benny Prijono0b6340c2006-06-13 22:21:23 +00003020 inv_set_cause(inv, tsx->status_code, &tsx->status_text);
Benny Prijono8ad55352006-02-08 11:16:05 +00003021 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
Benny Prijono268ca612006-02-07 12:34:11 +00003022 }
3023 break;
3024
3025 case PJSIP_TSX_STATE_DESTROYED:
3026 /* Do nothing. */
3027 break;
3028
3029 default:
3030 pj_assert(!"Unexpected state");
3031 }
3032
Benny Prijono8ad55352006-02-08 11:16:05 +00003033 } else if (tsx->role == PJSIP_ROLE_UAS &&
3034 tsx->method.id == PJSIP_BYE_METHOD &&
3035 tsx->status_code < 200 &&
3036 e->body.tsx_state.type == PJSIP_EVENT_RX_MSG)
3037 {
3038
3039 /*
3040 * Handle incoming BYE.
3041 */
3042
3043 inv_respond_incoming_bye( inv, tsx, e->body.tsx_state.src.rdata, e );
3044
Benny Prijono38998232006-02-08 22:44:25 +00003045 } else if (tsx->method.id == PJSIP_BYE_METHOD &&
3046 tsx->role == PJSIP_ROLE_UAC &&
Benny Prijonoa66c7152006-02-09 01:26:14 +00003047 (tsx->state == PJSIP_TSX_STATE_COMPLETED ||
3048 tsx->state == PJSIP_TSX_STATE_TERMINATED))
Benny Prijono38998232006-02-08 22:44:25 +00003049 {
3050
3051 /*
3052 * Outgoing BYE
3053 */
3054 inv_handle_bye_response( inv, tsx, e->body.tsx_state.src.rdata, e);
3055
Benny Prijono268ca612006-02-07 12:34:11 +00003056 }
Benny Prijono70127222006-07-02 14:53:05 +00003057 else if (tsx->method.id == PJSIP_CANCEL_METHOD &&
3058 tsx->role == PJSIP_ROLE_UAS &&
3059 tsx->status_code < 200 &&
3060 e->body.tsx_state.type == PJSIP_EVENT_RX_MSG)
3061 {
Benny Prijono38998232006-02-08 22:44:25 +00003062
Benny Prijono70127222006-07-02 14:53:05 +00003063 /*
3064 * Handle strandled incoming CANCEL.
3065 */
3066 pjsip_rx_data *rdata = e->body.tsx_state.src.rdata;
3067 pjsip_tx_data *tdata;
3068 pj_status_t status;
3069
3070 status = pjsip_dlg_create_response(dlg, rdata, 200, NULL, &tdata);
3071 if (status != PJ_SUCCESS) return;
3072
3073 status = pjsip_dlg_send_response(dlg, tsx, tdata);
3074 if (status != PJ_SUCCESS) return;
3075
Benny Prijono1f7767b2007-10-03 18:28:49 +00003076 } else if (tsx->role == PJSIP_ROLE_UAS &&
3077 tsx->state == PJSIP_TSX_STATE_TRYING &&
3078 pjsip_method_cmp(&tsx->method, &pjsip_update_method)==0)
3079 {
3080 /*
3081 * Handle incoming UPDATE
3082 */
3083 inv_respond_incoming_update(inv, e->body.tsx_state.src.rdata);
3084
3085
3086 } else if (tsx->role == PJSIP_ROLE_UAC &&
3087 (tsx->state == PJSIP_TSX_STATE_COMPLETED ||
3088 tsx->state == PJSIP_TSX_STATE_TERMINATED) &&
3089 pjsip_method_cmp(&tsx->method, &pjsip_update_method)==0)
3090 {
3091 /*
3092 * Handle response to outgoing UPDATE request.
3093 */
Benny Prijono0b4c57b2008-06-25 10:15:01 +00003094 if (handle_uac_tsx_response(inv, e) == PJ_FALSE)
Benny Prijono87402382008-02-21 19:28:21 +00003095 inv_handle_update_response(inv, e);
Benny Prijono1f7767b2007-10-03 18:28:49 +00003096
3097 } else if (tsx->role == PJSIP_ROLE_UAS &&
3098 tsx->state == PJSIP_TSX_STATE_TRYING &&
3099 pjsip_method_cmp(&tsx->method, &pjsip_prack_method)==0)
3100 {
3101 /*
3102 * Handle incoming PRACK
3103 */
3104 inv_respond_incoming_prack(inv, e->body.tsx_state.src.rdata);
3105
3106 } else if (tsx->role == PJSIP_ROLE_UAC) {
Benny Prijono87402382008-02-21 19:28:21 +00003107
Benny Prijono0b4c57b2008-06-25 10:15:01 +00003108 /* Generic handling for UAC tsx completion */
3109 handle_uac_tsx_response(inv, e);
Benny Prijono70127222006-07-02 14:53:05 +00003110 }
Benny Prijono1f7767b2007-10-03 18:28:49 +00003111
Benny Prijono268ca612006-02-07 12:34:11 +00003112}
3113
Benny Prijono8ad55352006-02-08 11:16:05 +00003114/*
3115 * State CONFIRMED is after ACK is sent/received.
3116 */
3117static void inv_on_state_confirmed( pjsip_inv_session *inv, pjsip_event *e)
Benny Prijono268ca612006-02-07 12:34:11 +00003118{
3119 pjsip_transaction *tsx = e->body.tsx_state.tsx;
3120 pjsip_dialog *dlg = pjsip_tsx_get_dlg(tsx);
3121
3122 PJ_ASSERT_ON_FAIL(tsx && dlg, return);
3123
Benny Prijono268ca612006-02-07 12:34:11 +00003124
Benny Prijono8ad55352006-02-08 11:16:05 +00003125 if (tsx->method.id == PJSIP_BYE_METHOD &&
3126 tsx->role == PJSIP_ROLE_UAC &&
Benny Prijonoa66c7152006-02-09 01:26:14 +00003127 (tsx->state == PJSIP_TSX_STATE_COMPLETED ||
3128 tsx->state == PJSIP_TSX_STATE_TERMINATED))
Benny Prijono8ad55352006-02-08 11:16:05 +00003129 {
Benny Prijono38998232006-02-08 22:44:25 +00003130
Benny Prijono8ad55352006-02-08 11:16:05 +00003131 /*
Benny Prijono38998232006-02-08 22:44:25 +00003132 * Outgoing BYE
Benny Prijono8ad55352006-02-08 11:16:05 +00003133 */
Benny Prijono8ad55352006-02-08 11:16:05 +00003134
Benny Prijonoa66c7152006-02-09 01:26:14 +00003135 inv_handle_bye_response( inv, tsx, e->body.tsx_state.src.rdata, e);
Benny Prijono268ca612006-02-07 12:34:11 +00003136
Benny Prijono8ad55352006-02-08 11:16:05 +00003137 }
3138 else if (tsx->method.id == PJSIP_BYE_METHOD &&
3139 tsx->role == PJSIP_ROLE_UAS &&
3140 tsx->status_code < 200 &&
3141 e->body.tsx_state.type == PJSIP_EVENT_RX_MSG)
3142 {
Benny Prijonoccf95622006-02-07 18:48:01 +00003143
Benny Prijono8ad55352006-02-08 11:16:05 +00003144 /*
3145 * Handle incoming BYE.
3146 */
Benny Prijono268ca612006-02-07 12:34:11 +00003147
Benny Prijono8ad55352006-02-08 11:16:05 +00003148 inv_respond_incoming_bye( inv, tsx, e->body.tsx_state.src.rdata, e );
3149
Benny Prijono268ca612006-02-07 12:34:11 +00003150 }
Benny Prijono70127222006-07-02 14:53:05 +00003151 else if (tsx->method.id == PJSIP_CANCEL_METHOD &&
3152 tsx->role == PJSIP_ROLE_UAS &&
3153 tsx->status_code < 200 &&
3154 e->body.tsx_state.type == PJSIP_EVENT_RX_MSG)
3155 {
3156
3157 /*
3158 * Handle strandled incoming CANCEL.
3159 */
3160 pjsip_rx_data *rdata = e->body.tsx_state.src.rdata;
3161 pjsip_tx_data *tdata;
3162 pj_status_t status;
3163
3164 status = pjsip_dlg_create_response(dlg, rdata, 200, NULL, &tdata);
3165 if (status != PJ_SUCCESS) return;
3166
3167 status = pjsip_dlg_send_response(dlg, tsx, tdata);
3168 if (status != PJ_SUCCESS) return;
3169
3170 }
Benny Prijono26ff9062006-02-21 23:47:00 +00003171 else if (tsx->method.id == PJSIP_INVITE_METHOD &&
3172 tsx->role == PJSIP_ROLE_UAS)
3173 {
3174
3175 /*
3176 * Handle incoming re-INVITE
3177 */
3178 if (tsx->state == PJSIP_TSX_STATE_TRYING) {
3179
3180 pjsip_rx_data *rdata = e->body.tsx_state.src.rdata;
3181 pjsip_tx_data *tdata;
3182 pj_status_t status;
3183
3184 /* Check if we have INVITE pending. */
3185 if (inv->invite_tsx && inv->invite_tsx!=tsx) {
Benny Prijono46249942007-02-19 22:23:14 +00003186 pj_str_t reason;
3187
3188 reason = pj_str("Another INVITE transaction in progress");
Benny Prijono26ff9062006-02-21 23:47:00 +00003189
3190 /* Can not receive re-INVITE while another one is pending. */
Benny Prijono46249942007-02-19 22:23:14 +00003191 status = pjsip_dlg_create_response( inv->dlg, rdata, 500,
3192 &reason, &tdata);
Benny Prijono26ff9062006-02-21 23:47:00 +00003193 if (status != PJ_SUCCESS)
3194 return;
3195
3196 status = pjsip_dlg_send_response( inv->dlg, tsx, tdata);
3197
3198
3199 return;
3200 }
3201
3202 /* Save the invite transaction. */
3203 inv->invite_tsx = tsx;
3204
3205 /* Process SDP in incoming message. */
3206 status = inv_check_sdp_in_incoming_msg(inv, tsx, rdata);
3207
3208 if (status != PJ_SUCCESS) {
3209
3210 /* Not Acceptable */
3211 const pjsip_hdr *accept;
3212
3213 status = pjsip_dlg_create_response(inv->dlg, rdata,
3214 488, NULL, &tdata);
3215 if (status != PJ_SUCCESS)
3216 return;
3217
3218
3219 accept = pjsip_endpt_get_capability(dlg->endpt, PJSIP_H_ACCEPT,
3220 NULL);
3221 if (accept) {
Benny Prijonoa1e69682007-05-11 15:14:34 +00003222 pjsip_msg_add_hdr(tdata->msg, (pjsip_hdr*)
Benny Prijono26ff9062006-02-21 23:47:00 +00003223 pjsip_hdr_clone(tdata->pool, accept));
3224 }
3225
3226 status = pjsip_dlg_send_response(dlg, tsx, tdata);
3227
3228 return;
3229 }
3230
3231 /* Create 2xx ANSWER */
3232 status = pjsip_dlg_create_response(dlg, rdata, 200, NULL, &tdata);
3233 if (status != PJ_SUCCESS)
3234 return;
3235
Benny Prijono7d910092007-06-20 04:19:46 +00003236 /* If the INVITE request has SDP body, send answer.
3237 * Otherwise generate offer from local active SDP.
3238 */
3239 if (rdata->msg_info.msg->body != NULL) {
3240 status = process_answer(inv, 200, tdata, NULL);
3241 } else {
Benny Prijono77998ce2007-06-20 10:03:46 +00003242 /* INVITE does not have SDP.
3243 * If on_create_offer() callback is implemented, ask app.
3244 * to generate an offer, otherwise just send active local
3245 * SDP to signal that nothing gets modified.
3246 */
3247 pjmedia_sdp_session *sdp = NULL;
3248
3249 if (mod_inv.cb.on_create_offer) {
3250 (*mod_inv.cb.on_create_offer)(inv, &sdp);
3251 if (sdp) {
3252 status = pjmedia_sdp_neg_modify_local_offer(dlg->pool,
3253 inv->neg,
3254 sdp);
3255 }
3256 }
3257
3258 if (sdp == NULL) {
3259 const pjmedia_sdp_session *active_sdp = NULL;
3260 status = pjmedia_sdp_neg_send_local_offer(dlg->pool,
3261 inv->neg,
3262 &active_sdp);
3263 if (status == PJ_SUCCESS)
3264 sdp = (pjmedia_sdp_session*) active_sdp;
3265 }
3266
3267 if (sdp) {
3268 tdata->msg->body = create_sdp_body(tdata->pool, sdp);
Benny Prijono7d910092007-06-20 04:19:46 +00003269 }
3270 }
Benny Prijono26ff9062006-02-21 23:47:00 +00003271
Benny Prijono1d9b9a42006-09-25 13:40:12 +00003272 if (status != PJ_SUCCESS) {
3273 /*
3274 * SDP negotiation has failed.
3275 */
3276 pj_status_t rc;
3277 pj_str_t reason;
3278
3279 /* Delete the 2xx answer */
3280 pjsip_tx_data_dec_ref(tdata);
3281
3282 /* Create 500 response */
3283 reason = pj_str("SDP negotiation failed");
3284 rc = pjsip_dlg_create_response(dlg, rdata, 500, &reason,
3285 &tdata);
3286 if (rc == PJ_SUCCESS) {
3287 pjsip_warning_hdr *w;
3288 const pj_str_t *endpt_name;
3289
3290 endpt_name = pjsip_endpt_name(dlg->endpt);
3291 w = pjsip_warning_hdr_create_from_status(tdata->pool,
3292 endpt_name,
3293 status);
3294 if (w)
3295 pjsip_msg_add_hdr(tdata->msg, (pjsip_hdr*)w);
3296
3297 pjsip_inv_send_msg(inv, tdata);
3298 }
3299 return;
3300 }
3301
3302 /* Send 2xx regardless of the status of negotiation */
Benny Prijonoe8b0d3b2006-03-17 17:57:52 +00003303 status = pjsip_inv_send_msg(inv, tdata);
Benny Prijono26ff9062006-02-21 23:47:00 +00003304
Benny Prijono7d910092007-06-20 04:19:46 +00003305 } else if (tsx->state == PJSIP_TSX_STATE_CONFIRMED) {
3306 /* This is the case where ACK has the same branch as
3307 * the INVITE request.
3308 */
3309 if (tsx->status_code/100 == 2 &&
3310 e->body.tsx_state.type == PJSIP_EVENT_RX_MSG)
3311 {
3312 inv_check_sdp_in_incoming_msg(inv, tsx,
3313 e->body.tsx_state.src.rdata);
3314 }
3315
Benny Prijono26ff9062006-02-21 23:47:00 +00003316 }
3317
3318 }
3319 else if (tsx->method.id == PJSIP_INVITE_METHOD &&
3320 tsx->role == PJSIP_ROLE_UAC)
3321 {
Benny Prijono22e48c92008-03-20 14:40:50 +00003322
Benny Prijono26ff9062006-02-21 23:47:00 +00003323 /*
3324 * Handle outgoing re-INVITE
3325 */
Benny Prijono22e48c92008-03-20 14:40:50 +00003326 if (tsx->state == PJSIP_TSX_STATE_CALLING) {
3327
Benny Prijono61fc5e62008-06-25 18:35:31 +00003328 /* Must not have other pending INVITE transaction */
3329 pj_assert(inv->invite_tsx==NULL || tsx==inv->invite_tsx);
3330
Benny Prijono22e48c92008-03-20 14:40:50 +00003331 /* Save pending invite transaction */
3332 inv->invite_tsx = tsx;
3333
3334 } else if (tsx->state == PJSIP_TSX_STATE_TERMINATED &&
3335 tsx->status_code/100 == 2)
Benny Prijono26ff9062006-02-21 23:47:00 +00003336 {
3337
3338 /* Re-INVITE was accepted. */
3339
3340 /* Process SDP */
3341 inv_check_sdp_in_incoming_msg(inv, tsx,
3342 e->body.tsx_state.src.rdata);
3343
3344 /* Send ACK */
Benny Prijonod5f9f422007-11-25 04:40:07 +00003345 inv_send_ack(inv, e);
Benny Prijono26ff9062006-02-21 23:47:00 +00003346
Benny Prijono0b4c57b2008-06-25 10:15:01 +00003347 } else if (handle_uac_tsx_response(inv, e)) {
Benny Prijono87402382008-02-21 19:28:21 +00003348
3349 /* Handle response that terminates dialog */
3350 /* Nothing to do (already handled) */
3351
Benny Prijono77998ce2007-06-20 10:03:46 +00003352 } else if (tsx->status_code >= 300 && tsx->status_code < 700) {
3353
3354 pjmedia_sdp_neg_state neg_state;
3355
3356 /* Outgoing INVITE transaction has failed, cancel SDP nego */
3357 neg_state = pjmedia_sdp_neg_get_state(inv->neg);
3358 if (neg_state == PJMEDIA_SDP_NEG_STATE_LOCAL_OFFER) {
3359 pjmedia_sdp_neg_cancel_offer(inv->neg);
3360 }
Benny Prijono26ff9062006-02-21 23:47:00 +00003361 }
Benny Prijono1f7767b2007-10-03 18:28:49 +00003362
3363 } else if (tsx->role == PJSIP_ROLE_UAS &&
3364 tsx->state == PJSIP_TSX_STATE_TRYING &&
3365 pjsip_method_cmp(&tsx->method, &pjsip_update_method)==0)
3366 {
3367 /*
3368 * Handle incoming UPDATE
3369 */
3370 inv_respond_incoming_update(inv, e->body.tsx_state.src.rdata);
3371
3372 } else if (tsx->role == PJSIP_ROLE_UAC &&
3373 (tsx->state == PJSIP_TSX_STATE_COMPLETED ||
3374 tsx->state == PJSIP_TSX_STATE_TERMINATED) &&
3375 pjsip_method_cmp(&tsx->method, &pjsip_update_method)==0)
3376 {
3377 /*
3378 * Handle response to outgoing UPDATE request.
3379 */
Benny Prijono0b4c57b2008-06-25 10:15:01 +00003380 if (handle_uac_tsx_response(inv, e) == PJ_FALSE)
Benny Prijono87402382008-02-21 19:28:21 +00003381 inv_handle_update_response(inv, e);
Benny Prijono1f7767b2007-10-03 18:28:49 +00003382
3383 } else if (tsx->role == PJSIP_ROLE_UAS &&
3384 tsx->state == PJSIP_TSX_STATE_TRYING &&
3385 pjsip_method_cmp(&tsx->method, &pjsip_prack_method)==0)
3386 {
3387 /*
3388 * Handle strandled incoming PRACK
3389 */
3390 inv_respond_incoming_prack(inv, e->body.tsx_state.src.rdata);
3391
3392 } else if (tsx->role == PJSIP_ROLE_UAC) {
3393 /*
Benny Prijono0b4c57b2008-06-25 10:15:01 +00003394 * Handle 401/407/408/481 response
Benny Prijono1f7767b2007-10-03 18:28:49 +00003395 */
Benny Prijono0b4c57b2008-06-25 10:15:01 +00003396 handle_uac_tsx_response(inv, e);
Benny Prijono26ff9062006-02-21 23:47:00 +00003397 }
Benny Prijono1f7767b2007-10-03 18:28:49 +00003398
Benny Prijono268ca612006-02-07 12:34:11 +00003399}
3400
Benny Prijono8ad55352006-02-08 11:16:05 +00003401/*
3402 * After session has been terminated, but before dialog is destroyed
3403 * (because dialog has other usages, or because dialog is waiting for
3404 * the last transaction to terminate).
3405 */
3406static void inv_on_state_disconnected( pjsip_inv_session *inv, pjsip_event *e)
Benny Prijono268ca612006-02-07 12:34:11 +00003407{
Benny Prijono8ad55352006-02-08 11:16:05 +00003408 pjsip_transaction *tsx = e->body.tsx_state.tsx;
3409 pjsip_dialog *dlg = pjsip_tsx_get_dlg(tsx);
Benny Prijono268ca612006-02-07 12:34:11 +00003410
Benny Prijono8ad55352006-02-08 11:16:05 +00003411 PJ_ASSERT_ON_FAIL(tsx && dlg, return);
3412
Benny Prijono70127222006-07-02 14:53:05 +00003413 if (tsx->role == PJSIP_ROLE_UAS &&
Benny Prijono8ad55352006-02-08 11:16:05 +00003414 tsx->status_code < 200 &&
3415 e->body.tsx_state.type == PJSIP_EVENT_RX_MSG)
3416 {
Benny Prijono70127222006-07-02 14:53:05 +00003417 pjsip_rx_data *rdata = e->body.tsx_state.src.rdata;
Benny Prijono8ad55352006-02-08 11:16:05 +00003418
3419 /*
Benny Prijono70127222006-07-02 14:53:05 +00003420 * Respond BYE with 200/OK
Benny Prijono8ad55352006-02-08 11:16:05 +00003421 */
Benny Prijono70127222006-07-02 14:53:05 +00003422 if (tsx->method.id == PJSIP_BYE_METHOD) {
3423 inv_respond_incoming_bye( inv, tsx, rdata, e );
3424 } else if (tsx->method.id == PJSIP_CANCEL_METHOD) {
3425 /*
3426 * Respond CANCEL with 200/OK too.
3427 */
3428 pjsip_tx_data *tdata;
3429 pj_status_t status;
Benny Prijono8ad55352006-02-08 11:16:05 +00003430
Benny Prijono70127222006-07-02 14:53:05 +00003431 status = pjsip_dlg_create_response(dlg, rdata, 200, NULL, &tdata);
3432 if (status != PJ_SUCCESS) return;
Benny Prijono8ad55352006-02-08 11:16:05 +00003433
Benny Prijono70127222006-07-02 14:53:05 +00003434 status = pjsip_dlg_send_response(dlg, tsx, tdata);
3435 if (status != PJ_SUCCESS) return;
3436
3437 }
Benny Prijono61fc5e62008-06-25 18:35:31 +00003438
3439 } else if (tsx->role == PJSIP_ROLE_UAC) {
3440 /*
3441 * Handle 401/407/408/481 response
3442 */
3443 handle_uac_tsx_response(inv, e);
Benny Prijono8ad55352006-02-08 11:16:05 +00003444 }
Benny Prijono268ca612006-02-07 12:34:11 +00003445}
3446