blob: 371abb760b316acc21210008dd5d8fb38523f17c [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
361 /* Set state to CONFIRMED (if we're not in CONFIRMED yet) */
362 if (inv->state != PJSIP_INV_STATE_CONFIRMED) {
363 inv_set_state(inv, PJSIP_INV_STATE_CONFIRMED, e);
364 }
365
Benny Prijono268ca612006-02-07 12:34:11 +0000366 return PJ_SUCCESS;
367}
368
Benny Prijono8ad55352006-02-08 11:16:05 +0000369/*
370 * Module on_rx_request()
371 *
372 * This callback is called for these events:
373 * - endpoint receives request which was unhandled by higher priority
374 * modules (e.g. transaction layer, dialog layer).
375 * - dialog distributes incoming request to its usages.
376 */
377static pj_bool_t mod_inv_on_rx_request(pjsip_rx_data *rdata)
378{
379 pjsip_method *method;
Benny Prijono38998232006-02-08 22:44:25 +0000380 pjsip_dialog *dlg;
381 pjsip_inv_session *inv;
Benny Prijono8ad55352006-02-08 11:16:05 +0000382
383 /* Only wants to receive request from a dialog. */
Benny Prijono38998232006-02-08 22:44:25 +0000384 dlg = pjsip_rdata_get_dlg(rdata);
385 if (dlg == NULL)
Benny Prijono8ad55352006-02-08 11:16:05 +0000386 return PJ_FALSE;
387
Benny Prijonoa1e69682007-05-11 15:14:34 +0000388 inv = (pjsip_inv_session*) dlg->mod_data[mod_inv.mod.id];
Benny Prijono38998232006-02-08 22:44:25 +0000389
Benny Prijono8ad55352006-02-08 11:16:05 +0000390 /* Report to dialog that we handle INVITE, CANCEL, BYE, ACK.
391 * If we need to send response, it will be sent in the state
392 * handlers.
393 */
394 method = &rdata->msg_info.msg->line.req.method;
395
Benny Prijono70127222006-07-02 14:53:05 +0000396 if (method->id == PJSIP_INVITE_METHOD) {
397 return PJ_TRUE;
398 }
399
400 /* BYE and CANCEL must have existing invite session */
401 if (method->id == PJSIP_BYE_METHOD ||
402 method->id == PJSIP_CANCEL_METHOD)
Benny Prijono8ad55352006-02-08 11:16:05 +0000403 {
Benny Prijono70127222006-07-02 14:53:05 +0000404 if (inv == NULL)
405 return PJ_FALSE;
406
Benny Prijono8ad55352006-02-08 11:16:05 +0000407 return PJ_TRUE;
408 }
409
Benny Prijono38998232006-02-08 22:44:25 +0000410 /* On receipt ACK request, when state is CONNECTING,
411 * move state to CONFIRMED.
412 */
Benny Prijono26ff9062006-02-21 23:47:00 +0000413 if (method->id == PJSIP_ACK_METHOD && inv) {
Benny Prijono38998232006-02-08 22:44:25 +0000414
Benny Prijonof521eb02006-08-06 23:07:25 +0000415 /* Ignore ACK if pending INVITE transaction has not finished. */
416 if (inv->invite_tsx &&
417 inv->invite_tsx->state < PJSIP_TSX_STATE_COMPLETED)
418 {
419 return PJ_TRUE;
420 }
421
Benny Prijono5eff0432006-02-09 14:14:21 +0000422 /* Terminate INVITE transaction, if it's still present. */
423 if (inv->invite_tsx &&
424 inv->invite_tsx->state <= PJSIP_TSX_STATE_COMPLETED)
425 {
Benny Prijono7d910092007-06-20 04:19:46 +0000426 /* Before we terminate INVITE transaction, process the SDP
427 * in the ACK request, if any.
428 */
429 inv_check_sdp_in_incoming_msg(inv, inv->invite_tsx, rdata);
430
431 /* Now we can terminate the INVITE transaction */
Benny Prijonof521eb02006-08-06 23:07:25 +0000432 pj_assert(inv->invite_tsx->status_code >= 200);
Benny Prijono5eff0432006-02-09 14:14:21 +0000433 pjsip_tsx_terminate(inv->invite_tsx,
434 inv->invite_tsx->status_code);
435 inv->invite_tsx = NULL;
Benny Prijonodcfc0ba2007-09-30 16:50:27 +0000436 if (inv->last_answer) {
437 pjsip_tx_data_dec_ref(inv->last_answer);
438 inv->last_answer = NULL;
439 }
Benny Prijono5eff0432006-02-09 14:14:21 +0000440 }
441
Benny Prijono32ac8fe2006-03-02 21:16:55 +0000442 /* On receipt of ACK, only set state to confirmed when state
443 * is CONNECTING (e.g. we don't want to set the state to confirmed
444 * when we receive ACK retransmission after sending non-2xx!)
445 */
446 if (inv->state == PJSIP_INV_STATE_CONNECTING) {
Benny Prijono26ff9062006-02-21 23:47:00 +0000447 pjsip_event event;
448
449 PJSIP_EVENT_INIT_RX_MSG(event, rdata);
450 inv_set_state(inv, PJSIP_INV_STATE_CONFIRMED, &event);
451 }
Benny Prijono38998232006-02-08 22:44:25 +0000452 }
453
Benny Prijono8ad55352006-02-08 11:16:05 +0000454 return PJ_FALSE;
455}
456
457/*
458 * Module on_rx_response().
459 *
460 * This callback is called for these events:
461 * - dialog distributes incoming 2xx response to INVITE (outside
462 * transaction) to its usages.
463 * - endpoint distributes strayed responses.
464 */
Benny Prijono268ca612006-02-07 12:34:11 +0000465static pj_bool_t mod_inv_on_rx_response(pjsip_rx_data *rdata)
466{
467 pjsip_dialog *dlg;
468 pjsip_inv_session *inv;
469 pjsip_msg *msg = rdata->msg_info.msg;
470
471 dlg = pjsip_rdata_get_dlg(rdata);
472
473 /* Ignore responses outside dialog */
474 if (dlg == NULL)
475 return PJ_FALSE;
476
477 /* Ignore responses not belonging to invite session */
478 inv = pjsip_dlg_get_inv_session(dlg);
479 if (inv == NULL)
480 return PJ_FALSE;
481
482 /* This MAY be retransmission of 2xx response to INVITE.
483 * If it is, we need to send ACK.
484 */
485 if (msg->type == PJSIP_RESPONSE_MSG && msg->line.status.code/100==2 &&
Benny Prijono26ff9062006-02-21 23:47:00 +0000486 rdata->msg_info.cseq->method.id == PJSIP_INVITE_METHOD &&
487 inv->invite_tsx == NULL)
488 {
Benny Prijonod5f9f422007-11-25 04:40:07 +0000489 pjsip_event e;
Benny Prijono268ca612006-02-07 12:34:11 +0000490
Benny Prijonod5f9f422007-11-25 04:40:07 +0000491 PJSIP_EVENT_INIT_RX_MSG(e, rdata);
492 inv_send_ack(inv, &e);
Benny Prijono268ca612006-02-07 12:34:11 +0000493 return PJ_TRUE;
494
495 }
496
497 /* No other processing needs to be done here. */
498 return PJ_FALSE;
499}
500
Benny Prijono8ad55352006-02-08 11:16:05 +0000501/*
502 * Module on_tsx_state()
503 *
504 * This callback is called by dialog framework for all transactions
505 * inside the dialog for all its dialog usages.
506 */
Benny Prijono268ca612006-02-07 12:34:11 +0000507static void mod_inv_on_tsx_state(pjsip_transaction *tsx, pjsip_event *e)
508{
509 pjsip_dialog *dlg;
510 pjsip_inv_session *inv;
511
512 dlg = pjsip_tsx_get_dlg(tsx);
513 if (dlg == NULL)
514 return;
515
516 inv = pjsip_dlg_get_inv_session(dlg);
517 if (inv == NULL)
518 return;
519
520 /* Call state handler for the invite session. */
521 (*inv_state_handler[inv->state])(inv, e);
522
523 /* Call on_tsx_state */
Benny Prijonod4e0abd2006-03-05 11:53:36 +0000524 if (mod_inv.cb.on_tsx_state_changed && inv->notify)
Benny Prijono268ca612006-02-07 12:34:11 +0000525 (*mod_inv.cb.on_tsx_state_changed)(inv, tsx, e);
526
Benny Prijono46249942007-02-19 22:23:14 +0000527 /* Clear invite transaction when tsx is confirmed.
528 * Previously we set invite_tsx to NULL only when transaction has
529 * terminated, but this didn't work when ACK has the same Via branch
530 * value as the INVITE (see http://www.pjsip.org/trac/ticket/113)
531 */
Benny Prijonodcfc0ba2007-09-30 16:50:27 +0000532 if (tsx->state>=PJSIP_TSX_STATE_CONFIRMED && tsx == inv->invite_tsx) {
Benny Prijono46249942007-02-19 22:23:14 +0000533 inv->invite_tsx = NULL;
Benny Prijonodcfc0ba2007-09-30 16:50:27 +0000534 if (inv->last_answer) {
535 pjsip_tx_data_dec_ref(inv->last_answer);
536 inv->last_answer = NULL;
537 }
538 }
Benny Prijono268ca612006-02-07 12:34:11 +0000539}
540
Benny Prijono8ad55352006-02-08 11:16:05 +0000541
542/*
543 * Initialize the invite module.
544 */
Benny Prijono268ca612006-02-07 12:34:11 +0000545PJ_DEF(pj_status_t) pjsip_inv_usage_init( pjsip_endpoint *endpt,
Benny Prijono268ca612006-02-07 12:34:11 +0000546 const pjsip_inv_callback *cb)
547{
548 pj_status_t status;
549
550 /* Check arguments. */
Benny Prijonoe8b0d3b2006-03-17 17:57:52 +0000551 PJ_ASSERT_RETURN(endpt && cb, PJ_EINVAL);
Benny Prijono268ca612006-02-07 12:34:11 +0000552
553 /* Some callbacks are mandatory */
554 PJ_ASSERT_RETURN(cb->on_state_changed && cb->on_new_session, PJ_EINVAL);
555
556 /* Check if module already registered. */
557 PJ_ASSERT_RETURN(mod_inv.mod.id == -1, PJ_EINVALIDOP);
558
559 /* Copy param. */
560 pj_memcpy(&mod_inv.cb, cb, sizeof(pjsip_inv_callback));
561
562 mod_inv.endpt = endpt;
Benny Prijono268ca612006-02-07 12:34:11 +0000563
564 /* Register the module. */
565 status = pjsip_endpt_register_module(endpt, &mod_inv.mod);
Benny Prijono053f5222006-11-11 16:16:04 +0000566 if (status != PJ_SUCCESS)
567 return status;
Benny Prijono268ca612006-02-07 12:34:11 +0000568
Benny Prijono053f5222006-11-11 16:16:04 +0000569 return PJ_SUCCESS;
Benny Prijono268ca612006-02-07 12:34:11 +0000570}
571
Benny Prijono8ad55352006-02-08 11:16:05 +0000572/*
573 * Get the instance of invite module.
574 */
Benny Prijono268ca612006-02-07 12:34:11 +0000575PJ_DEF(pjsip_module*) pjsip_inv_usage_instance(void)
576{
577 return &mod_inv.mod;
578}
579
580
Benny Prijono632ce712006-02-09 14:01:40 +0000581
Benny Prijono8ad55352006-02-08 11:16:05 +0000582/*
583 * Return the invite session for the specified dialog.
584 */
Benny Prijono268ca612006-02-07 12:34:11 +0000585PJ_DEF(pjsip_inv_session*) pjsip_dlg_get_inv_session(pjsip_dialog *dlg)
586{
Benny Prijonoa1e69682007-05-11 15:14:34 +0000587 return (pjsip_inv_session*) dlg->mod_data[mod_inv.mod.id];
Benny Prijono268ca612006-02-07 12:34:11 +0000588}
589
Benny Prijono8ad55352006-02-08 11:16:05 +0000590
Benny Prijono268ca612006-02-07 12:34:11 +0000591/*
Benny Prijonoe8b0d3b2006-03-17 17:57:52 +0000592 * Get INVITE state name.
593 */
594PJ_DEF(const char *) pjsip_inv_state_name(pjsip_inv_state state)
595{
596 PJ_ASSERT_RETURN(state >= PJSIP_INV_STATE_NULL &&
597 state <= PJSIP_INV_STATE_DISCONNECTED,
598 "??");
599
600 return inv_state_names[state];
601}
602
603/*
Benny Prijono8ad55352006-02-08 11:16:05 +0000604 * Create UAC invite session.
Benny Prijono268ca612006-02-07 12:34:11 +0000605 */
606PJ_DEF(pj_status_t) pjsip_inv_create_uac( pjsip_dialog *dlg,
607 const pjmedia_sdp_session *local_sdp,
608 unsigned options,
609 pjsip_inv_session **p_inv)
610{
611 pjsip_inv_session *inv;
612 pj_status_t status;
613
614 /* Verify arguments. */
615 PJ_ASSERT_RETURN(dlg && p_inv, PJ_EINVAL);
616
Benny Prijono8eae8382006-08-10 21:44:26 +0000617 /* Must lock dialog first */
618 pjsip_dlg_inc_lock(dlg);
619
Benny Prijono268ca612006-02-07 12:34:11 +0000620 /* Normalize options */
621 if (options & PJSIP_INV_REQUIRE_100REL)
622 options |= PJSIP_INV_SUPPORT_100REL;
623
624 if (options & PJSIP_INV_REQUIRE_TIMER)
625 options |= PJSIP_INV_SUPPORT_TIMER;
626
627 /* Create the session */
Benny Prijonoa1e69682007-05-11 15:14:34 +0000628 inv = PJ_POOL_ZALLOC_T(dlg->pool, pjsip_inv_session);
Benny Prijono8eae8382006-08-10 21:44:26 +0000629 pj_assert(inv != NULL);
Benny Prijono268ca612006-02-07 12:34:11 +0000630
631 inv->pool = dlg->pool;
632 inv->role = PJSIP_ROLE_UAC;
633 inv->state = PJSIP_INV_STATE_NULL;
634 inv->dlg = dlg;
635 inv->options = options;
Benny Prijonod4e0abd2006-03-05 11:53:36 +0000636 inv->notify = PJ_TRUE;
Benny Prijonoba5926a2007-05-02 11:29:37 +0000637 inv->cause = (pjsip_status_code) 0;
Benny Prijono268ca612006-02-07 12:34:11 +0000638
Benny Prijonoe4f2abb2006-02-10 14:04:05 +0000639 /* Object name will use the same dialog pointer. */
Benny Prijonoed811d72006-03-10 12:57:12 +0000640 pj_ansi_snprintf(inv->obj_name, PJ_MAX_OBJ_NAME, "inv%p", dlg);
Benny Prijonoe4f2abb2006-02-10 14:04:05 +0000641
Benny Prijono268ca612006-02-07 12:34:11 +0000642 /* Create negotiator if local_sdp is specified. */
643 if (local_sdp) {
644 status = pjmedia_sdp_neg_create_w_local_offer(dlg->pool, local_sdp,
645 &inv->neg);
Benny Prijono8eae8382006-08-10 21:44:26 +0000646 if (status != PJ_SUCCESS) {
647 pjsip_dlg_dec_lock(dlg);
Benny Prijono268ca612006-02-07 12:34:11 +0000648 return status;
Benny Prijono8eae8382006-08-10 21:44:26 +0000649 }
Benny Prijono268ca612006-02-07 12:34:11 +0000650 }
651
652 /* Register invite as dialog usage. */
653 status = pjsip_dlg_add_usage(dlg, &mod_inv.mod, inv);
Benny Prijono8eae8382006-08-10 21:44:26 +0000654 if (status != PJ_SUCCESS) {
655 pjsip_dlg_dec_lock(dlg);
Benny Prijono268ca612006-02-07 12:34:11 +0000656 return status;
Benny Prijono8eae8382006-08-10 21:44:26 +0000657 }
Benny Prijono268ca612006-02-07 12:34:11 +0000658
659 /* Increment dialog session */
Benny Prijonoe4f2abb2006-02-10 14:04:05 +0000660 pjsip_dlg_inc_session(dlg, &mod_inv.mod);
Benny Prijono268ca612006-02-07 12:34:11 +0000661
Benny Prijonodcfc0ba2007-09-30 16:50:27 +0000662 /* Create 100rel handler */
663 pjsip_100rel_attach(inv);
Benny Prijonodcfc0ba2007-09-30 16:50:27 +0000664
Benny Prijono268ca612006-02-07 12:34:11 +0000665 /* Done */
666 *p_inv = inv;
Benny Prijonoe4f2abb2006-02-10 14:04:05 +0000667
Benny Prijono8eae8382006-08-10 21:44:26 +0000668 pjsip_dlg_dec_lock(dlg);
669
Benny Prijonoe4f2abb2006-02-10 14:04:05 +0000670 PJ_LOG(5,(inv->obj_name, "UAC invite session created for dialog %s",
671 dlg->obj_name));
672
Benny Prijono268ca612006-02-07 12:34:11 +0000673 return PJ_SUCCESS;
674}
675
676/*
677 * Verify incoming INVITE request.
678 */
Benny Prijono87a90212008-01-23 20:29:30 +0000679PJ_DEF(pj_status_t) pjsip_inv_verify_request2(pjsip_rx_data *rdata,
680 unsigned *options,
681 const pjmedia_sdp_session *r_sdp,
682 const pjmedia_sdp_session *l_sdp,
683 pjsip_dialog *dlg,
684 pjsip_endpoint *endpt,
685 pjsip_tx_data **p_tdata)
Benny Prijono268ca612006-02-07 12:34:11 +0000686{
687 pjsip_msg *msg;
688 pjsip_allow_hdr *allow;
689 pjsip_supported_hdr *sup_hdr;
690 pjsip_require_hdr *req_hdr;
691 int code = 200;
692 unsigned rem_option = 0;
693 pj_status_t status = PJ_SUCCESS;
694 pjsip_hdr res_hdr_list;
695
696 /* Init return arguments. */
697 if (p_tdata) *p_tdata = NULL;
698
699 /* Verify arguments. */
700 PJ_ASSERT_RETURN(rdata != NULL && options != NULL, PJ_EINVAL);
701
702 /* Normalize options */
703 if (*options & PJSIP_INV_REQUIRE_100REL)
704 *options |= PJSIP_INV_SUPPORT_100REL;
705
706 if (*options & PJSIP_INV_REQUIRE_TIMER)
707 *options |= PJSIP_INV_SUPPORT_TIMER;
708
709 /* Get the message in rdata */
710 msg = rdata->msg_info.msg;
711
712 /* Must be INVITE request. */
713 PJ_ASSERT_RETURN(msg->type == PJSIP_REQUEST_MSG &&
714 msg->line.req.method.id == PJSIP_INVITE_METHOD,
715 PJ_EINVAL);
716
717 /* If tdata is specified, then either dlg or endpt must be specified */
718 PJ_ASSERT_RETURN((!p_tdata) || (endpt || dlg), PJ_EINVAL);
719
720 /* Get the endpoint */
721 endpt = endpt ? endpt : dlg->endpt;
722
723 /* Init response header list */
724 pj_list_init(&res_hdr_list);
725
Benny Prijono87a90212008-01-23 20:29:30 +0000726 /* Check the request body, see if it's something that we support,
727 * only when the body hasn't been parsed before.
Benny Prijono268ca612006-02-07 12:34:11 +0000728 */
Benny Prijono87a90212008-01-23 20:29:30 +0000729 if (r_sdp==NULL && msg->body) {
Benny Prijono268ca612006-02-07 12:34:11 +0000730 pjsip_msg_body *body = msg->body;
731 pj_str_t str_application = {"application", 11};
732 pj_str_t str_sdp = { "sdp", 3 };
733 pjmedia_sdp_session *sdp;
734
735 /* Check content type. */
736 if (pj_stricmp(&body->content_type.type, &str_application) != 0 ||
737 pj_stricmp(&body->content_type.subtype, &str_sdp) != 0)
738 {
739 /* Not "application/sdp" */
740 code = PJSIP_SC_UNSUPPORTED_MEDIA_TYPE;
741 status = PJSIP_ERRNO_FROM_SIP_STATUS(code);
742
743 if (p_tdata) {
744 /* Add Accept header to response */
745 pjsip_accept_hdr *acc;
746
747 acc = pjsip_accept_hdr_create(rdata->tp_info.pool);
748 PJ_ASSERT_RETURN(acc, PJ_ENOMEM);
749 acc->values[acc->count++] = pj_str("application/sdp");
750 pj_list_push_back(&res_hdr_list, acc);
751 }
752
753 goto on_return;
754 }
755
756 /* Parse and validate SDP */
Benny Prijonoa1e69682007-05-11 15:14:34 +0000757 status = pjmedia_sdp_parse(rdata->tp_info.pool,
758 (char*)body->data, body->len, &sdp);
Benny Prijono268ca612006-02-07 12:34:11 +0000759 if (status == PJ_SUCCESS)
760 status = pjmedia_sdp_validate(sdp);
761
762 if (status != PJ_SUCCESS) {
763 /* Unparseable or invalid SDP */
764 code = PJSIP_SC_BAD_REQUEST;
765
766 if (p_tdata) {
767 /* Add Warning header. */
768 pjsip_warning_hdr *w;
769
770 w = pjsip_warning_hdr_create_from_status(rdata->tp_info.pool,
771 pjsip_endpt_name(endpt),
772 status);
773 PJ_ASSERT_RETURN(w, PJ_ENOMEM);
774
775 pj_list_push_back(&res_hdr_list, w);
776 }
777
778 goto on_return;
779 }
780
Benny Prijono87a90212008-01-23 20:29:30 +0000781 r_sdp = sdp;
782 }
783
784 if (r_sdp) {
Benny Prijono268ca612006-02-07 12:34:11 +0000785 /* Negotiate with local SDP */
786 if (l_sdp) {
787 pjmedia_sdp_neg *neg;
788
789 /* Local SDP must be valid! */
790 PJ_ASSERT_RETURN((status=pjmedia_sdp_validate(l_sdp))==PJ_SUCCESS,
791 status);
792
793 /* Create SDP negotiator */
794 status = pjmedia_sdp_neg_create_w_remote_offer(
Benny Prijono87a90212008-01-23 20:29:30 +0000795 rdata->tp_info.pool, l_sdp, r_sdp, &neg);
Benny Prijono268ca612006-02-07 12:34:11 +0000796 PJ_ASSERT_RETURN(status == PJ_SUCCESS, status);
797
798 /* Negotiate SDP */
799 status = pjmedia_sdp_neg_negotiate(rdata->tp_info.pool, neg, 0);
800 if (status != PJ_SUCCESS) {
801
802 /* Incompatible media */
803 code = PJSIP_SC_NOT_ACCEPTABLE_HERE;
Benny Prijono268ca612006-02-07 12:34:11 +0000804
805 if (p_tdata) {
806 pjsip_accept_hdr *acc;
807 pjsip_warning_hdr *w;
808
809 /* Add Warning header. */
810 w = pjsip_warning_hdr_create_from_status(
811 rdata->tp_info.pool,
812 pjsip_endpt_name(endpt), status);
813 PJ_ASSERT_RETURN(w, PJ_ENOMEM);
814
815 pj_list_push_back(&res_hdr_list, w);
816
817 /* Add Accept header to response */
818 acc = pjsip_accept_hdr_create(rdata->tp_info.pool);
819 PJ_ASSERT_RETURN(acc, PJ_ENOMEM);
820 acc->values[acc->count++] = pj_str("application/sdp");
821 pj_list_push_back(&res_hdr_list, acc);
822
823 }
824
825 goto on_return;
826 }
827 }
828 }
829
830 /* Check supported methods, see if peer supports UPDATE.
831 * We just assume that peer supports standard INVITE, ACK, CANCEL, and BYE
832 * implicitly by sending this INVITE.
833 */
Benny Prijonoa1e69682007-05-11 15:14:34 +0000834 allow = (pjsip_allow_hdr*) pjsip_msg_find_hdr(msg, PJSIP_H_ALLOW, NULL);
Benny Prijono268ca612006-02-07 12:34:11 +0000835 if (allow) {
836 unsigned i;
837 const pj_str_t STR_UPDATE = { "UPDATE", 6 };
838
839 for (i=0; i<allow->count; ++i) {
840 if (pj_stricmp(&allow->values[i], &STR_UPDATE)==0)
841 break;
842 }
843
844 if (i != allow->count) {
845 /* UPDATE is present in Allow */
846 rem_option |= PJSIP_INV_SUPPORT_UPDATE;
847 }
848
849 }
850
851 /* Check Supported header */
Benny Prijonoa1e69682007-05-11 15:14:34 +0000852 sup_hdr = (pjsip_supported_hdr*)
853 pjsip_msg_find_hdr(msg, PJSIP_H_SUPPORTED, NULL);
Benny Prijono268ca612006-02-07 12:34:11 +0000854 if (sup_hdr) {
855 unsigned i;
856 pj_str_t STR_100REL = { "100rel", 6};
857 pj_str_t STR_TIMER = { "timer", 5 };
858
859 for (i=0; i<sup_hdr->count; ++i) {
860 if (pj_stricmp(&sup_hdr->values[i], &STR_100REL)==0)
861 rem_option |= PJSIP_INV_SUPPORT_100REL;
862 else if (pj_stricmp(&sup_hdr->values[i], &STR_TIMER)==0)
863 rem_option |= PJSIP_INV_SUPPORT_TIMER;
864 }
865 }
866
867 /* Check Require header */
Benny Prijonoa1e69682007-05-11 15:14:34 +0000868 req_hdr = (pjsip_require_hdr*)
869 pjsip_msg_find_hdr(msg, PJSIP_H_REQUIRE, NULL);
Benny Prijono268ca612006-02-07 12:34:11 +0000870 if (req_hdr) {
871 unsigned i;
Benny Prijono053f5222006-11-11 16:16:04 +0000872 const pj_str_t STR_100REL = { "100rel", 6};
873 const pj_str_t STR_TIMER = { "timer", 5 };
874 const pj_str_t STR_REPLACES = { "replaces", 8 };
Benny Prijono268ca612006-02-07 12:34:11 +0000875 unsigned unsupp_cnt = 0;
876 pj_str_t unsupp_tags[PJSIP_GENERIC_ARRAY_MAX_COUNT];
877
878 for (i=0; i<req_hdr->count; ++i) {
879 if ((*options & PJSIP_INV_SUPPORT_100REL) &&
880 pj_stricmp(&req_hdr->values[i], &STR_100REL)==0)
881 {
882 rem_option |= PJSIP_INV_REQUIRE_100REL;
883
884 } else if ((*options && PJSIP_INV_SUPPORT_TIMER) &&
885 pj_stricmp(&req_hdr->values[i], &STR_TIMER)==0)
886 {
887 rem_option |= PJSIP_INV_REQUIRE_TIMER;
888
Benny Prijono053f5222006-11-11 16:16:04 +0000889 } else if (pj_stricmp(&req_hdr->values[i], &STR_REPLACES)==0) {
890 pj_bool_t supp;
891
892 supp = pjsip_endpt_has_capability(endpt, PJSIP_H_SUPPORTED,
893 NULL, &STR_REPLACES);
894 if (!supp)
895 unsupp_tags[unsupp_cnt++] = req_hdr->values[i];
896
Benny Prijono268ca612006-02-07 12:34:11 +0000897 } else {
898 /* Unknown/unsupported extension tag! */
899 unsupp_tags[unsupp_cnt++] = req_hdr->values[i];
900 }
901 }
902
903 /* Check if there are required tags that we don't support */
904 if (unsupp_cnt) {
905
906 code = PJSIP_SC_BAD_EXTENSION;
907 status = PJSIP_ERRNO_FROM_SIP_STATUS(code);
908
909 if (p_tdata) {
910 pjsip_unsupported_hdr *unsupp_hdr;
911 const pjsip_hdr *h;
912
913 /* Add Unsupported header. */
914 unsupp_hdr = pjsip_unsupported_hdr_create(rdata->tp_info.pool);
915 PJ_ASSERT_RETURN(unsupp_hdr != NULL, PJ_ENOMEM);
916
917 unsupp_hdr->count = unsupp_cnt;
918 for (i=0; i<unsupp_cnt; ++i)
919 unsupp_hdr->values[i] = unsupp_tags[i];
920
921 pj_list_push_back(&res_hdr_list, unsupp_hdr);
922
923 /* Add Supported header. */
924 h = pjsip_endpt_get_capability(endpt, PJSIP_H_SUPPORTED,
925 NULL);
926 pj_assert(h);
927 if (h) {
Benny Prijonoa1e69682007-05-11 15:14:34 +0000928 sup_hdr = (pjsip_supported_hdr*)
929 pjsip_hdr_clone(rdata->tp_info.pool, h);
Benny Prijono268ca612006-02-07 12:34:11 +0000930 pj_list_push_back(&res_hdr_list, sup_hdr);
931 }
932 }
933
934 goto on_return;
935 }
936 }
937
938 /* Check if there are local requirements that are not supported
939 * by peer.
940 */
941 if ( ((*options & PJSIP_INV_REQUIRE_100REL)!=0 &&
942 (rem_option & PJSIP_INV_SUPPORT_100REL)==0) ||
943 ((*options & PJSIP_INV_REQUIRE_TIMER)!=0 &&
944 (rem_option & PJSIP_INV_SUPPORT_TIMER)==0))
945 {
946 code = PJSIP_SC_EXTENSION_REQUIRED;
947 status = PJSIP_ERRNO_FROM_SIP_STATUS(code);
948
949 if (p_tdata) {
950 const pjsip_hdr *h;
951
952 /* Add Require header. */
953 req_hdr = pjsip_require_hdr_create(rdata->tp_info.pool);
954 PJ_ASSERT_RETURN(req_hdr != NULL, PJ_ENOMEM);
955
956 if (*options & PJSIP_INV_REQUIRE_100REL)
957 req_hdr->values[req_hdr->count++] = pj_str("100rel");
958
959 if (*options & PJSIP_INV_REQUIRE_TIMER)
960 req_hdr->values[req_hdr->count++] = pj_str("timer");
961
962 pj_list_push_back(&res_hdr_list, req_hdr);
963
964 /* Add Supported header. */
965 h = pjsip_endpt_get_capability(endpt, PJSIP_H_SUPPORTED,
966 NULL);
967 pj_assert(h);
968 if (h) {
Benny Prijonoa1e69682007-05-11 15:14:34 +0000969 sup_hdr = (pjsip_supported_hdr*)
970 pjsip_hdr_clone(rdata->tp_info.pool, h);
Benny Prijono268ca612006-02-07 12:34:11 +0000971 pj_list_push_back(&res_hdr_list, sup_hdr);
972 }
973
974 }
975
976 goto on_return;
977 }
978
Benny Prijonodcfc0ba2007-09-30 16:50:27 +0000979 /* If remote Require something that we support, make us Require
980 * that feature too.
981 */
982 if (rem_option & PJSIP_INV_REQUIRE_100REL) {
983 pj_assert(*options & PJSIP_INV_SUPPORT_100REL);
984 *options |= PJSIP_INV_REQUIRE_100REL;
985 }
986 if (rem_option & PJSIP_INV_REQUIRE_TIMER) {
987 pj_assert(*options & PJSIP_INV_SUPPORT_TIMER);
988 *options |= PJSIP_INV_REQUIRE_TIMER;
989 }
990
Benny Prijono268ca612006-02-07 12:34:11 +0000991on_return:
992
993 /* Create response if necessary */
994 if (code != 200 && p_tdata) {
995 pjsip_tx_data *tdata;
996 const pjsip_hdr *h;
997
998 if (dlg) {
999 status = pjsip_dlg_create_response(dlg, rdata, code, NULL,
1000 &tdata);
1001 } else {
1002 status = pjsip_endpt_create_response(endpt, rdata, code, NULL,
1003 &tdata);
1004 }
1005
1006 if (status != PJ_SUCCESS)
1007 return status;
1008
1009 /* Add response headers. */
1010 h = res_hdr_list.next;
1011 while (h != &res_hdr_list) {
1012 pjsip_hdr *cloned;
1013
Benny Prijonoa1e69682007-05-11 15:14:34 +00001014 cloned = (pjsip_hdr*) pjsip_hdr_clone(tdata->pool, h);
Benny Prijono268ca612006-02-07 12:34:11 +00001015 PJ_ASSERT_RETURN(cloned, PJ_ENOMEM);
1016
1017 pjsip_msg_add_hdr(tdata->msg, cloned);
1018
1019 h = h->next;
1020 }
1021
1022 *p_tdata = tdata;
Benny Prijono70127222006-07-02 14:53:05 +00001023
1024 /* Can not return PJ_SUCCESS when response message is produced.
1025 * Ref: PROTOS test ~#2490
1026 */
1027 if (status == PJ_SUCCESS)
1028 status = PJSIP_ERRNO_FROM_SIP_STATUS(code);
1029
Benny Prijono268ca612006-02-07 12:34:11 +00001030 }
1031
1032 return status;
1033}
1034
Benny Prijono87a90212008-01-23 20:29:30 +00001035
1036/*
1037 * Verify incoming INVITE request.
1038 */
1039PJ_DEF(pj_status_t) pjsip_inv_verify_request( pjsip_rx_data *rdata,
1040 unsigned *options,
1041 const pjmedia_sdp_session *l_sdp,
1042 pjsip_dialog *dlg,
1043 pjsip_endpoint *endpt,
1044 pjsip_tx_data **p_tdata)
1045{
1046 return pjsip_inv_verify_request2(rdata, options, NULL, l_sdp, dlg,
1047 endpt, p_tdata);
1048}
1049
Benny Prijono268ca612006-02-07 12:34:11 +00001050/*
Benny Prijono8ad55352006-02-08 11:16:05 +00001051 * Create UAS invite session.
Benny Prijono268ca612006-02-07 12:34:11 +00001052 */
1053PJ_DEF(pj_status_t) pjsip_inv_create_uas( pjsip_dialog *dlg,
1054 pjsip_rx_data *rdata,
1055 const pjmedia_sdp_session *local_sdp,
1056 unsigned options,
1057 pjsip_inv_session **p_inv)
1058{
1059 pjsip_inv_session *inv;
Benny Prijonoa66c7152006-02-09 01:26:14 +00001060 struct tsx_inv_data *tsx_inv_data;
Benny Prijono268ca612006-02-07 12:34:11 +00001061 pjsip_msg *msg;
1062 pjmedia_sdp_session *rem_sdp = NULL;
1063 pj_status_t status;
1064
1065 /* Verify arguments. */
1066 PJ_ASSERT_RETURN(dlg && rdata && p_inv, PJ_EINVAL);
1067
1068 /* Dialog MUST have been initialised. */
1069 PJ_ASSERT_RETURN(pjsip_rdata_get_tsx(rdata) != NULL, PJ_EINVALIDOP);
1070
1071 msg = rdata->msg_info.msg;
1072
1073 /* rdata MUST contain INVITE request */
1074 PJ_ASSERT_RETURN(msg->type == PJSIP_REQUEST_MSG &&
1075 msg->line.req.method.id == PJSIP_INVITE_METHOD,
1076 PJ_EINVALIDOP);
1077
Benny Prijono8eae8382006-08-10 21:44:26 +00001078 /* Lock dialog */
1079 pjsip_dlg_inc_lock(dlg);
1080
Benny Prijono268ca612006-02-07 12:34:11 +00001081 /* Normalize options */
1082 if (options & PJSIP_INV_REQUIRE_100REL)
1083 options |= PJSIP_INV_SUPPORT_100REL;
1084
1085 if (options & PJSIP_INV_REQUIRE_TIMER)
1086 options |= PJSIP_INV_SUPPORT_TIMER;
1087
1088 /* Create the session */
Benny Prijonoa1e69682007-05-11 15:14:34 +00001089 inv = PJ_POOL_ZALLOC_T(dlg->pool, pjsip_inv_session);
Benny Prijono8eae8382006-08-10 21:44:26 +00001090 pj_assert(inv != NULL);
Benny Prijono268ca612006-02-07 12:34:11 +00001091
1092 inv->pool = dlg->pool;
1093 inv->role = PJSIP_ROLE_UAS;
Benny Prijono38998232006-02-08 22:44:25 +00001094 inv->state = PJSIP_INV_STATE_NULL;
Benny Prijono268ca612006-02-07 12:34:11 +00001095 inv->dlg = dlg;
1096 inv->options = options;
Benny Prijonod4e0abd2006-03-05 11:53:36 +00001097 inv->notify = PJ_TRUE;
Benny Prijonoba5926a2007-05-02 11:29:37 +00001098 inv->cause = (pjsip_status_code) 0;
Benny Prijono268ca612006-02-07 12:34:11 +00001099
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00001100 /* Object name will use the same dialog pointer. */
Benny Prijonoed811d72006-03-10 12:57:12 +00001101 pj_ansi_snprintf(inv->obj_name, PJ_MAX_OBJ_NAME, "inv%p", dlg);
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00001102
Benny Prijono268ca612006-02-07 12:34:11 +00001103 /* Parse SDP in message body, if present. */
1104 if (msg->body) {
1105 pjsip_msg_body *body = msg->body;
1106
1107 /* Parse and validate SDP */
Benny Prijonoa1e69682007-05-11 15:14:34 +00001108 status = pjmedia_sdp_parse(inv->pool, (char*)body->data, body->len,
Benny Prijono268ca612006-02-07 12:34:11 +00001109 &rem_sdp);
1110 if (status == PJ_SUCCESS)
1111 status = pjmedia_sdp_validate(rem_sdp);
1112
Benny Prijono8eae8382006-08-10 21:44:26 +00001113 if (status != PJ_SUCCESS) {
1114 pjsip_dlg_dec_lock(dlg);
Benny Prijono268ca612006-02-07 12:34:11 +00001115 return status;
Benny Prijono8eae8382006-08-10 21:44:26 +00001116 }
Benny Prijono268ca612006-02-07 12:34:11 +00001117 }
1118
1119 /* Create negotiator. */
1120 if (rem_sdp) {
1121 status = pjmedia_sdp_neg_create_w_remote_offer(inv->pool, local_sdp,
1122 rem_sdp, &inv->neg);
1123
1124 } else if (local_sdp) {
1125 status = pjmedia_sdp_neg_create_w_local_offer(inv->pool, local_sdp,
1126 &inv->neg);
1127 } else {
Benny Prijono95196582006-02-09 00:13:40 +00001128 status = PJ_SUCCESS;
Benny Prijono268ca612006-02-07 12:34:11 +00001129 }
1130
Benny Prijono8eae8382006-08-10 21:44:26 +00001131 if (status != PJ_SUCCESS) {
1132 pjsip_dlg_dec_lock(dlg);
Benny Prijono268ca612006-02-07 12:34:11 +00001133 return status;
Benny Prijono8eae8382006-08-10 21:44:26 +00001134 }
Benny Prijono268ca612006-02-07 12:34:11 +00001135
1136 /* Register invite as dialog usage. */
1137 status = pjsip_dlg_add_usage(dlg, &mod_inv.mod, inv);
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 /* Increment session in the dialog. */
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00001144 pjsip_dlg_inc_session(dlg, &mod_inv.mod);
Benny Prijono268ca612006-02-07 12:34:11 +00001145
1146 /* Save the invite transaction. */
1147 inv->invite_tsx = pjsip_rdata_get_tsx(rdata);
Benny Prijonoa66c7152006-02-09 01:26:14 +00001148
1149 /* Attach our data to the transaction. */
Benny Prijonoa1e69682007-05-11 15:14:34 +00001150 tsx_inv_data = PJ_POOL_ZALLOC_T(inv->invite_tsx->pool, struct tsx_inv_data);
Benny Prijonoa66c7152006-02-09 01:26:14 +00001151 tsx_inv_data->inv = inv;
1152 inv->invite_tsx->mod_data[mod_inv.mod.id] = tsx_inv_data;
Benny Prijono268ca612006-02-07 12:34:11 +00001153
Benny Prijonodcfc0ba2007-09-30 16:50:27 +00001154 /* Create 100rel handler */
1155 if (inv->options & PJSIP_INV_REQUIRE_100REL) {
1156 pjsip_100rel_attach(inv);
1157 }
Benny Prijonodcfc0ba2007-09-30 16:50:27 +00001158
Benny Prijono268ca612006-02-07 12:34:11 +00001159 /* Done */
Benny Prijono8eae8382006-08-10 21:44:26 +00001160 pjsip_dlg_dec_lock(dlg);
Benny Prijono268ca612006-02-07 12:34:11 +00001161 *p_inv = inv;
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00001162
1163 PJ_LOG(5,(inv->obj_name, "UAS invite session created for dialog %s",
1164 dlg->obj_name));
1165
Benny Prijono268ca612006-02-07 12:34:11 +00001166 return PJ_SUCCESS;
1167}
1168
Benny Prijonod4e0abd2006-03-05 11:53:36 +00001169/*
1170 * Forcefully terminate the session.
1171 */
1172PJ_DEF(pj_status_t) pjsip_inv_terminate( pjsip_inv_session *inv,
1173 int st_code,
1174 pj_bool_t notify)
1175{
1176 PJ_ASSERT_RETURN(inv, PJ_EINVAL);
1177
1178 /* Lock dialog. */
1179 pjsip_dlg_inc_lock(inv->dlg);
1180
1181 /* Set callback notify flag. */
1182 inv->notify = notify;
1183
1184 /* If there's pending transaction, terminate the transaction.
1185 * This may subsequently set the INVITE session state to
1186 * disconnected.
1187 */
1188 if (inv->invite_tsx &&
1189 inv->invite_tsx->state <= PJSIP_TSX_STATE_COMPLETED)
1190 {
1191 pjsip_tsx_terminate(inv->invite_tsx, st_code);
1192
1193 }
1194
1195 /* Set cause. */
Benny Prijono0b6340c2006-06-13 22:21:23 +00001196 inv_set_cause(inv, st_code, NULL);
Benny Prijonod4e0abd2006-03-05 11:53:36 +00001197
1198 /* Forcefully terminate the session if state is not DISCONNECTED */
1199 if (inv->state != PJSIP_INV_STATE_DISCONNECTED) {
1200 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, NULL);
1201 }
1202
1203 /* Done.
1204 * The dec_lock() below will actually destroys the dialog if it
1205 * has no other session.
1206 */
1207 pjsip_dlg_dec_lock(inv->dlg);
1208
1209 return PJ_SUCCESS;
1210}
1211
1212
Benny Prijono268ca612006-02-07 12:34:11 +00001213static void *clone_sdp(pj_pool_t *pool, const void *data, unsigned len)
1214{
1215 PJ_UNUSED_ARG(len);
Benny Prijonoa1e69682007-05-11 15:14:34 +00001216 return pjmedia_sdp_session_clone(pool, (const pjmedia_sdp_session*)data);
Benny Prijono268ca612006-02-07 12:34:11 +00001217}
1218
1219static int print_sdp(pjsip_msg_body *body, char *buf, pj_size_t len)
1220{
Benny Prijonoa1e69682007-05-11 15:14:34 +00001221 return pjmedia_sdp_print((const pjmedia_sdp_session*)body->data, buf, len);
Benny Prijono268ca612006-02-07 12:34:11 +00001222}
1223
Benny Prijono56315612006-07-18 14:39:40 +00001224
1225PJ_DEF(pj_status_t) pjsip_create_sdp_body( pj_pool_t *pool,
1226 pjmedia_sdp_session *sdp,
1227 pjsip_msg_body **p_body)
1228{
1229 const pj_str_t STR_APPLICATION = { "application", 11};
1230 const pj_str_t STR_SDP = { "sdp", 3 };
1231 pjsip_msg_body *body;
1232
Benny Prijonoa1e69682007-05-11 15:14:34 +00001233 body = PJ_POOL_ZALLOC_T(pool, pjsip_msg_body);
Benny Prijono56315612006-07-18 14:39:40 +00001234 PJ_ASSERT_RETURN(body != NULL, PJ_ENOMEM);
1235
1236 body->content_type.type = STR_APPLICATION;
1237 body->content_type.subtype = STR_SDP;
1238 body->data = sdp;
1239 body->len = 0;
1240 body->clone_data = &clone_sdp;
1241 body->print_body = &print_sdp;
1242
1243 *p_body = body;
1244
1245 return PJ_SUCCESS;
1246}
1247
Benny Prijono268ca612006-02-07 12:34:11 +00001248static pjsip_msg_body *create_sdp_body(pj_pool_t *pool,
1249 const pjmedia_sdp_session *c_sdp)
1250{
1251 pjsip_msg_body *body;
Benny Prijono56315612006-07-18 14:39:40 +00001252 pj_status_t status;
Benny Prijono268ca612006-02-07 12:34:11 +00001253
Benny Prijono56315612006-07-18 14:39:40 +00001254 status = pjsip_create_sdp_body(pool,
1255 pjmedia_sdp_session_clone(pool, c_sdp),
1256 &body);
Benny Prijono268ca612006-02-07 12:34:11 +00001257
Benny Prijono56315612006-07-18 14:39:40 +00001258 if (status != PJ_SUCCESS)
1259 return NULL;
Benny Prijono268ca612006-02-07 12:34:11 +00001260
1261 return body;
1262}
1263
1264/*
1265 * Create initial INVITE request.
1266 */
1267PJ_DEF(pj_status_t) pjsip_inv_invite( pjsip_inv_session *inv,
1268 pjsip_tx_data **p_tdata )
1269{
1270 pjsip_tx_data *tdata;
1271 const pjsip_hdr *hdr;
Benny Prijono26ff9062006-02-21 23:47:00 +00001272 pj_bool_t has_sdp;
Benny Prijono268ca612006-02-07 12:34:11 +00001273 pj_status_t status;
1274
1275 /* Verify arguments. */
1276 PJ_ASSERT_RETURN(inv && p_tdata, PJ_EINVAL);
1277
Benny Prijono26ff9062006-02-21 23:47:00 +00001278 /* State MUST be NULL or CONFIRMED. */
1279 PJ_ASSERT_RETURN(inv->state == PJSIP_INV_STATE_NULL ||
1280 inv->state == PJSIP_INV_STATE_CONFIRMED,
1281 PJ_EINVALIDOP);
Benny Prijono268ca612006-02-07 12:34:11 +00001282
Benny Prijono64f851e2006-02-23 13:49:28 +00001283 /* Lock dialog. */
1284 pjsip_dlg_inc_lock(inv->dlg);
1285
Benny Prijono268ca612006-02-07 12:34:11 +00001286 /* Create the INVITE request. */
Benny Prijono1f61a8f2007-08-16 10:11:44 +00001287 status = pjsip_dlg_create_request(inv->dlg, pjsip_get_invite_method(), -1,
Benny Prijono268ca612006-02-07 12:34:11 +00001288 &tdata);
1289 if (status != PJ_SUCCESS)
Benny Prijono64f851e2006-02-23 13:49:28 +00001290 goto on_return;
1291
Benny Prijono268ca612006-02-07 12:34:11 +00001292
Benny Prijono26ff9062006-02-21 23:47:00 +00001293 /* If this is the first INVITE, then copy the headers from inv_hdr.
1294 * These are the headers parsed from the request URI when the
1295 * dialog was created.
1296 */
1297 if (inv->state == PJSIP_INV_STATE_NULL) {
1298 hdr = inv->dlg->inv_hdr.next;
1299
1300 while (hdr != &inv->dlg->inv_hdr) {
Benny Prijonoa1e69682007-05-11 15:14:34 +00001301 pjsip_msg_add_hdr(tdata->msg, (pjsip_hdr*)
Benny Prijono26ff9062006-02-21 23:47:00 +00001302 pjsip_hdr_shallow_clone(tdata->pool, hdr));
1303 hdr = hdr->next;
1304 }
1305 }
1306
1307 /* See if we have SDP to send. */
1308 if (inv->neg) {
1309 pjmedia_sdp_neg_state neg_state;
1310
1311 neg_state = pjmedia_sdp_neg_get_state(inv->neg);
1312
1313 has_sdp = (neg_state == PJMEDIA_SDP_NEG_STATE_LOCAL_OFFER ||
1314 (neg_state == PJMEDIA_SDP_NEG_STATE_WAIT_NEGO &&
1315 pjmedia_sdp_neg_has_local_answer(inv->neg)));
1316
1317
1318 } else {
1319 has_sdp = PJ_FALSE;
1320 }
1321
Benny Prijono268ca612006-02-07 12:34:11 +00001322 /* Add SDP, if any. */
Benny Prijono26ff9062006-02-21 23:47:00 +00001323 if (has_sdp) {
Benny Prijono268ca612006-02-07 12:34:11 +00001324 const pjmedia_sdp_session *offer;
1325
1326 status = pjmedia_sdp_neg_get_neg_local(inv->neg, &offer);
1327 if (status != PJ_SUCCESS)
Benny Prijono64f851e2006-02-23 13:49:28 +00001328 goto on_return;
Benny Prijono268ca612006-02-07 12:34:11 +00001329
1330 tdata->msg->body = create_sdp_body(tdata->pool, offer);
1331 }
1332
1333 /* Add Allow header. */
Benny Prijono1f61a8f2007-08-16 10:11:44 +00001334 if (inv->dlg->add_allow) {
Benny Prijono95673f32007-06-26 08:23:18 +00001335 hdr = pjsip_endpt_get_capability(inv->dlg->endpt, PJSIP_H_ALLOW, NULL);
1336 if (hdr) {
1337 pjsip_msg_add_hdr(tdata->msg, (pjsip_hdr*)
1338 pjsip_hdr_shallow_clone(tdata->pool, hdr));
1339 }
Benny Prijono268ca612006-02-07 12:34:11 +00001340 }
1341
1342 /* Add Supported header */
1343 hdr = pjsip_endpt_get_capability(inv->dlg->endpt, PJSIP_H_SUPPORTED, NULL);
1344 if (hdr) {
Benny Prijonoa1e69682007-05-11 15:14:34 +00001345 pjsip_msg_add_hdr(tdata->msg, (pjsip_hdr*)
Benny Prijono268ca612006-02-07 12:34:11 +00001346 pjsip_hdr_shallow_clone(tdata->pool, hdr));
1347 }
1348
1349 /* Add Require header. */
Benny Prijonodcfc0ba2007-09-30 16:50:27 +00001350 if (inv->options & PJSIP_INV_REQUIRE_100REL) {
1351 const pj_str_t HREQ = { "Require", 7 };
1352 const pj_str_t tag_100rel = { "100rel", 6 };
1353 pjsip_generic_string_hdr *hreq;
1354
1355 hreq = pjsip_generic_string_hdr_create(tdata->pool, &HREQ,
1356 &tag_100rel);
1357 pjsip_msg_add_hdr(tdata->msg, (pjsip_hdr*) hreq);
1358 }
Benny Prijono268ca612006-02-07 12:34:11 +00001359
1360 /* Done. */
1361 *p_tdata = tdata;
1362
Benny Prijono64f851e2006-02-23 13:49:28 +00001363
1364on_return:
1365 pjsip_dlg_dec_lock(inv->dlg);
1366 return status;
Benny Prijono268ca612006-02-07 12:34:11 +00001367}
1368
1369
1370/*
Benny Prijono1f7767b2007-10-03 18:28:49 +00001371 * Initiate SDP negotiation in the SDP negotiator.
Benny Prijono95196582006-02-09 00:13:40 +00001372 */
1373static pj_status_t inv_negotiate_sdp( pjsip_inv_session *inv )
1374{
1375 pj_status_t status;
1376
1377 PJ_ASSERT_RETURN(pjmedia_sdp_neg_get_state(inv->neg) ==
1378 PJMEDIA_SDP_NEG_STATE_WAIT_NEGO,
1379 PJMEDIA_SDPNEG_EINSTATE);
1380
1381 status = pjmedia_sdp_neg_negotiate(inv->pool, inv->neg, 0);
1382
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00001383 PJ_LOG(5,(inv->obj_name, "SDP negotiation done, status=%d", status));
1384
Benny Prijonod4e0abd2006-03-05 11:53:36 +00001385 if (mod_inv.cb.on_media_update && inv->notify)
Benny Prijono95196582006-02-09 00:13:40 +00001386 (*mod_inv.cb.on_media_update)(inv, status);
1387
1388 return status;
1389}
1390
1391/*
Benny Prijonoa66c7152006-02-09 01:26:14 +00001392 * Check in incoming message for SDP offer/answer.
1393 */
Benny Prijono26ff9062006-02-21 23:47:00 +00001394static pj_status_t inv_check_sdp_in_incoming_msg( pjsip_inv_session *inv,
1395 pjsip_transaction *tsx,
1396 pjsip_rx_data *rdata)
Benny Prijonoa66c7152006-02-09 01:26:14 +00001397{
1398 struct tsx_inv_data *tsx_inv_data;
1399 static const pj_str_t str_application = { "application", 11 };
1400 static const pj_str_t str_sdp = { "sdp", 3 };
1401 pj_status_t status;
1402 pjsip_msg *msg;
1403 pjmedia_sdp_session *sdp;
1404
1405 /* Get/attach invite session's transaction data */
Benny Prijonoa1e69682007-05-11 15:14:34 +00001406 tsx_inv_data = (struct tsx_inv_data*) tsx->mod_data[mod_inv.mod.id];
Benny Prijonoa66c7152006-02-09 01:26:14 +00001407 if (tsx_inv_data == NULL) {
Benny Prijonoa1e69682007-05-11 15:14:34 +00001408 tsx_inv_data = PJ_POOL_ZALLOC_T(tsx->pool, struct tsx_inv_data);
Benny Prijonoa66c7152006-02-09 01:26:14 +00001409 tsx_inv_data->inv = inv;
1410 tsx->mod_data[mod_inv.mod.id] = tsx_inv_data;
1411 }
1412
1413 /* MUST NOT do multiple SDP offer/answer in a single transaction.
1414 */
1415
Benny Prijono77998ce2007-06-20 10:03:46 +00001416 if (tsx_inv_data->sdp_done) {
1417 if (rdata->msg_info.msg->body) {
1418 PJ_LOG(4,(inv->obj_name, "SDP negotiation done, message "
1419 "body is ignored"));
1420 }
Benny Prijono26ff9062006-02-21 23:47:00 +00001421 return PJ_SUCCESS;
Benny Prijono77998ce2007-06-20 10:03:46 +00001422 }
Benny Prijonoa66c7152006-02-09 01:26:14 +00001423
1424 /* Check if SDP is present in the message. */
1425
1426 msg = rdata->msg_info.msg;
1427 if (msg->body == NULL) {
1428 /* Message doesn't have body. */
Benny Prijono26ff9062006-02-21 23:47:00 +00001429 return PJ_SUCCESS;
Benny Prijonoa66c7152006-02-09 01:26:14 +00001430 }
1431
1432 if (pj_stricmp(&msg->body->content_type.type, &str_application) ||
1433 pj_stricmp(&msg->body->content_type.subtype, &str_sdp))
1434 {
1435 /* Message body is not "application/sdp" */
Benny Prijono26ff9062006-02-21 23:47:00 +00001436 return PJMEDIA_SDP_EINSDP;
Benny Prijonoa66c7152006-02-09 01:26:14 +00001437 }
1438
1439 /* Parse the SDP body. */
1440
Benny Prijonoa1e69682007-05-11 15:14:34 +00001441 status = pjmedia_sdp_parse(rdata->tp_info.pool,
1442 (char*)msg->body->data,
Benny Prijonoa66c7152006-02-09 01:26:14 +00001443 msg->body->len, &sdp);
Benny Prijonoc1b1c0a2007-11-06 08:48:02 +00001444 if (status == PJ_SUCCESS)
1445 status = pjmedia_sdp_validate(sdp);
1446
Benny Prijonoa66c7152006-02-09 01:26:14 +00001447 if (status != PJ_SUCCESS) {
1448 char errmsg[PJ_ERR_MSG_SIZE];
1449 pj_strerror(status, errmsg, sizeof(errmsg));
1450 PJ_LOG(4,(THIS_FILE, "Error parsing SDP in %s: %s",
1451 pjsip_rx_data_get_info(rdata), errmsg));
Benny Prijono26ff9062006-02-21 23:47:00 +00001452 return PJMEDIA_SDP_EINSDP;
Benny Prijonoa66c7152006-02-09 01:26:14 +00001453 }
1454
1455 /* The SDP can be an offer or answer, depending on negotiator's state */
1456
1457 if (inv->neg == NULL ||
1458 pjmedia_sdp_neg_get_state(inv->neg) == PJMEDIA_SDP_NEG_STATE_DONE)
1459 {
1460
1461 /* This is an offer. */
1462
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00001463 PJ_LOG(5,(inv->obj_name, "Got SDP offer in %s",
1464 pjsip_rx_data_get_info(rdata)));
1465
Benny Prijonoa66c7152006-02-09 01:26:14 +00001466 if (inv->neg == NULL) {
1467 status=pjmedia_sdp_neg_create_w_remote_offer(inv->pool, NULL,
1468 sdp, &inv->neg);
1469 } else {
1470 status=pjmedia_sdp_neg_set_remote_offer(inv->pool, inv->neg, sdp);
1471 }
1472
1473 if (status != PJ_SUCCESS) {
1474 char errmsg[PJ_ERR_MSG_SIZE];
1475 pj_strerror(status, errmsg, sizeof(errmsg));
1476 PJ_LOG(4,(THIS_FILE, "Error processing SDP offer in %s: %s",
1477 pjsip_rx_data_get_info(rdata), errmsg));
Benny Prijono26ff9062006-02-21 23:47:00 +00001478 return PJMEDIA_SDP_EINSDP;
Benny Prijonoa66c7152006-02-09 01:26:14 +00001479 }
1480
1481 /* Inform application about remote offer. */
1482
Benny Prijonod4e0abd2006-03-05 11:53:36 +00001483 if (mod_inv.cb.on_rx_offer && inv->notify) {
Benny Prijono26ff9062006-02-21 23:47:00 +00001484
1485 (*mod_inv.cb.on_rx_offer)(inv, sdp);
1486
1487 }
Benny Prijonoa66c7152006-02-09 01:26:14 +00001488
1489 } else if (pjmedia_sdp_neg_get_state(inv->neg) ==
1490 PJMEDIA_SDP_NEG_STATE_LOCAL_OFFER)
1491 {
1492
1493 /* This is an answer.
1494 * Process and negotiate remote answer.
1495 */
1496
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00001497 PJ_LOG(5,(inv->obj_name, "Got SDP answer in %s",
1498 pjsip_rx_data_get_info(rdata)));
1499
Benny Prijonoa66c7152006-02-09 01:26:14 +00001500 status = pjmedia_sdp_neg_set_remote_answer(inv->pool, inv->neg, sdp);
1501
1502 if (status != PJ_SUCCESS) {
1503 char errmsg[PJ_ERR_MSG_SIZE];
1504 pj_strerror(status, errmsg, sizeof(errmsg));
1505 PJ_LOG(4,(THIS_FILE, "Error processing SDP answer in %s: %s",
1506 pjsip_rx_data_get_info(rdata), errmsg));
Benny Prijono26ff9062006-02-21 23:47:00 +00001507 return PJMEDIA_SDP_EINSDP;
Benny Prijonoa66c7152006-02-09 01:26:14 +00001508 }
1509
1510 /* Negotiate SDP */
1511
1512 inv_negotiate_sdp(inv);
1513
1514 /* Mark this transaction has having SDP offer/answer done. */
1515
1516 tsx_inv_data->sdp_done = 1;
1517
1518 } else {
1519
1520 PJ_LOG(5,(THIS_FILE, "Ignored SDP in %s: negotiator state is %s",
1521 pjsip_rx_data_get_info(rdata),
1522 pjmedia_sdp_neg_state_str(pjmedia_sdp_neg_get_state(inv->neg))));
1523 }
1524
Benny Prijono26ff9062006-02-21 23:47:00 +00001525 return PJ_SUCCESS;
Benny Prijonoa66c7152006-02-09 01:26:14 +00001526}
1527
1528
Benny Prijono26ff9062006-02-21 23:47:00 +00001529/*
1530 * Process INVITE answer, for both initial and subsequent re-INVITE
1531 */
1532static pj_status_t process_answer( pjsip_inv_session *inv,
1533 int st_code,
Benny Prijono64f851e2006-02-23 13:49:28 +00001534 pjsip_tx_data *tdata,
1535 const pjmedia_sdp_session *local_sdp)
Benny Prijono26ff9062006-02-21 23:47:00 +00001536{
1537 pj_status_t status;
Benny Prijonoab7399b2006-02-27 00:40:31 +00001538 const pjmedia_sdp_session *sdp = NULL;
Benny Prijono26ff9062006-02-21 23:47:00 +00001539
Benny Prijono64f851e2006-02-23 13:49:28 +00001540 /* If local_sdp is specified, then we MUST NOT have answered the
1541 * offer before.
Benny Prijono26ff9062006-02-21 23:47:00 +00001542 */
Benny Prijono64f851e2006-02-23 13:49:28 +00001543 if (local_sdp && (st_code/100==1 || st_code/100==2)) {
1544
1545 if (inv->neg == NULL) {
1546 status = pjmedia_sdp_neg_create_w_local_offer(inv->pool, local_sdp,
1547 &inv->neg);
1548 } else if (pjmedia_sdp_neg_get_state(inv->neg)==
1549 PJMEDIA_SDP_NEG_STATE_REMOTE_OFFER)
1550 {
1551 status = pjmedia_sdp_neg_set_local_answer(inv->pool, inv->neg,
1552 local_sdp);
1553 } else {
1554
1555 /* Can not specify local SDP at this state. */
1556 pj_assert(0);
1557 status = PJMEDIA_SDPNEG_EINSTATE;
1558 }
1559
1560 if (status != PJ_SUCCESS)
1561 return status;
1562
1563 }
1564
1565
1566 /* If SDP negotiator is ready, start negotiation. */
1567 if (st_code/100==2 || (st_code/10==18 && st_code!=180)) {
Benny Prijono26ff9062006-02-21 23:47:00 +00001568
1569 pjmedia_sdp_neg_state neg_state;
1570
Benny Prijono64f851e2006-02-23 13:49:28 +00001571 /* Start nego when appropriate. */
Benny Prijono26ff9062006-02-21 23:47:00 +00001572 neg_state = inv->neg ? pjmedia_sdp_neg_get_state(inv->neg) :
1573 PJMEDIA_SDP_NEG_STATE_NULL;
1574
1575 if (neg_state == PJMEDIA_SDP_NEG_STATE_LOCAL_OFFER) {
1576
1577 status = pjmedia_sdp_neg_get_neg_local(inv->neg, &sdp);
1578
1579 } else if (neg_state == PJMEDIA_SDP_NEG_STATE_WAIT_NEGO &&
1580 pjmedia_sdp_neg_has_local_answer(inv->neg) )
1581 {
Benny Prijono77998ce2007-06-20 10:03:46 +00001582 struct tsx_inv_data *tsx_inv_data;
1583
1584 /* Get invite session's transaction data */
1585 tsx_inv_data = (struct tsx_inv_data*)
1586 inv->invite_tsx->mod_data[mod_inv.mod.id];
Benny Prijono26ff9062006-02-21 23:47:00 +00001587
1588 status = inv_negotiate_sdp(inv);
1589 if (status != PJ_SUCCESS)
1590 return status;
1591
Benny Prijono77998ce2007-06-20 10:03:46 +00001592 /* Mark this transaction has having SDP offer/answer done. */
1593 tsx_inv_data->sdp_done = 1;
1594
Benny Prijono26ff9062006-02-21 23:47:00 +00001595 status = pjmedia_sdp_neg_get_active_local(inv->neg, &sdp);
1596 }
Benny Prijono26ff9062006-02-21 23:47:00 +00001597 }
1598
Benny Prijono64f851e2006-02-23 13:49:28 +00001599 /* Include SDP when it's available for 2xx and 18x (but not 180) response.
Benny Prijono26ff9062006-02-21 23:47:00 +00001600 * Subsequent response will include this SDP.
Benny Prijono48ab2b72007-11-08 09:24:30 +00001601 *
1602 * Note note:
1603 * - When offer/answer has been completed in reliable 183, we MUST NOT
1604 * send SDP in 2xx response. So if we don't have SDP to send, clear
1605 * the SDP in the message body ONLY if 100rel is active in this
1606 * session.
Benny Prijono26ff9062006-02-21 23:47:00 +00001607 */
1608 if (sdp) {
1609 tdata->msg->body = create_sdp_body(tdata->pool, sdp);
Benny Prijono48ab2b72007-11-08 09:24:30 +00001610 } else {
1611 if (inv->options & PJSIP_INV_REQUIRE_100REL) {
1612 tdata->msg->body = NULL;
1613 }
Benny Prijono26ff9062006-02-21 23:47:00 +00001614 }
1615
Benny Prijono26ff9062006-02-21 23:47:00 +00001616
1617 return PJ_SUCCESS;
1618}
1619
Benny Prijonoa66c7152006-02-09 01:26:14 +00001620
1621/*
Benny Prijono64f851e2006-02-23 13:49:28 +00001622 * Create first response to INVITE
1623 */
1624PJ_DEF(pj_status_t) pjsip_inv_initial_answer( pjsip_inv_session *inv,
1625 pjsip_rx_data *rdata,
1626 int st_code,
1627 const pj_str_t *st_text,
1628 const pjmedia_sdp_session *sdp,
1629 pjsip_tx_data **p_tdata)
1630{
1631 pjsip_tx_data *tdata;
1632 pj_status_t status;
1633
1634 /* Verify arguments. */
1635 PJ_ASSERT_RETURN(inv && p_tdata, PJ_EINVAL);
1636
1637 /* Must have INVITE transaction. */
1638 PJ_ASSERT_RETURN(inv->invite_tsx, PJ_EBUG);
1639
1640 pjsip_dlg_inc_lock(inv->dlg);
1641
1642 /* Create response */
1643 status = pjsip_dlg_create_response(inv->dlg, rdata, st_code, st_text,
1644 &tdata);
1645 if (status != PJ_SUCCESS)
1646 goto on_return;
1647
1648 /* Process SDP in answer */
1649 status = process_answer(inv, st_code, tdata, sdp);
1650 if (status != PJ_SUCCESS) {
1651 pjsip_tx_data_dec_ref(tdata);
1652 goto on_return;
1653 }
1654
Benny Prijonodcfc0ba2007-09-30 16:50:27 +00001655 /* Save this answer */
1656 inv->last_answer = tdata;
1657 pjsip_tx_data_add_ref(inv->last_answer);
1658 PJ_LOG(5,(inv->dlg->obj_name, "Initial answer %s",
1659 pjsip_tx_data_get_info(inv->last_answer)));
1660
Benny Prijono64f851e2006-02-23 13:49:28 +00001661 *p_tdata = tdata;
1662
1663on_return:
1664 pjsip_dlg_dec_lock(inv->dlg);
1665 return status;
1666}
1667
1668
1669/*
Benny Prijono26ff9062006-02-21 23:47:00 +00001670 * Answer initial INVITE
1671 * Re-INVITE will be answered automatically, and will not use this function.
Benny Prijono268ca612006-02-07 12:34:11 +00001672 */
1673PJ_DEF(pj_status_t) pjsip_inv_answer( pjsip_inv_session *inv,
1674 int st_code,
1675 const pj_str_t *st_text,
1676 const pjmedia_sdp_session *local_sdp,
1677 pjsip_tx_data **p_tdata )
1678{
1679 pjsip_tx_data *last_res;
1680 pj_status_t status;
1681
1682 /* Verify arguments. */
1683 PJ_ASSERT_RETURN(inv && p_tdata, PJ_EINVAL);
1684
1685 /* Must have INVITE transaction. */
1686 PJ_ASSERT_RETURN(inv->invite_tsx, PJ_EBUG);
1687
Benny Prijonodcfc0ba2007-09-30 16:50:27 +00001688 /* Must have created an answer before */
1689 PJ_ASSERT_RETURN(inv->last_answer, PJ_EINVALIDOP);
Benny Prijono268ca612006-02-07 12:34:11 +00001690
Benny Prijono64f851e2006-02-23 13:49:28 +00001691 pjsip_dlg_inc_lock(inv->dlg);
Benny Prijono268ca612006-02-07 12:34:11 +00001692
1693 /* Modify last response. */
Benny Prijonodcfc0ba2007-09-30 16:50:27 +00001694 last_res = inv->last_answer;
Benny Prijono268ca612006-02-07 12:34:11 +00001695 status = pjsip_dlg_modify_response(inv->dlg, last_res, st_code, st_text);
1696 if (status != PJ_SUCCESS)
Benny Prijono64f851e2006-02-23 13:49:28 +00001697 goto on_return;
Benny Prijono268ca612006-02-07 12:34:11 +00001698
Benny Prijonodcfc0ba2007-09-30 16:50:27 +00001699 /* For non-2xx final response, strip message body */
1700 if (st_code >= 300) {
1701 last_res->msg->body = NULL;
1702 }
Benny Prijono268ca612006-02-07 12:34:11 +00001703
Benny Prijono26ff9062006-02-21 23:47:00 +00001704 /* Process SDP in answer */
Benny Prijono64f851e2006-02-23 13:49:28 +00001705 status = process_answer(inv, st_code, last_res, local_sdp);
1706 if (status != PJ_SUCCESS) {
1707 pjsip_tx_data_dec_ref(last_res);
1708 goto on_return;
1709 }
Benny Prijono268ca612006-02-07 12:34:11 +00001710
Benny Prijono268ca612006-02-07 12:34:11 +00001711
1712 *p_tdata = last_res;
1713
Benny Prijono64f851e2006-02-23 13:49:28 +00001714on_return:
1715 pjsip_dlg_dec_lock(inv->dlg);
1716 return status;
Benny Prijono268ca612006-02-07 12:34:11 +00001717}
1718
1719
1720/*
Benny Prijono26ff9062006-02-21 23:47:00 +00001721 * Set SDP answer.
1722 */
1723PJ_DEF(pj_status_t) pjsip_inv_set_sdp_answer( pjsip_inv_session *inv,
1724 const pjmedia_sdp_session *sdp )
1725{
1726 pj_status_t status;
1727
1728 PJ_ASSERT_RETURN(inv && sdp, PJ_EINVAL);
1729
1730 pjsip_dlg_inc_lock(inv->dlg);
1731 status = pjmedia_sdp_neg_set_local_answer( inv->pool, inv->neg, sdp);
1732 pjsip_dlg_dec_lock(inv->dlg);
1733
1734 return status;
1735}
1736
1737
1738/*
Benny Prijono268ca612006-02-07 12:34:11 +00001739 * End session.
1740 */
1741PJ_DEF(pj_status_t) pjsip_inv_end_session( pjsip_inv_session *inv,
1742 int st_code,
1743 const pj_str_t *st_text,
1744 pjsip_tx_data **p_tdata )
1745{
1746 pjsip_tx_data *tdata;
1747 pj_status_t status;
1748
1749 /* Verify arguments. */
1750 PJ_ASSERT_RETURN(inv && p_tdata, PJ_EINVAL);
1751
Benny Prijonod4e0abd2006-03-05 11:53:36 +00001752 /* Set cause code. */
Benny Prijono0b6340c2006-06-13 22:21:23 +00001753 inv_set_cause(inv, st_code, st_text);
Benny Prijonod4e0abd2006-03-05 11:53:36 +00001754
Benny Prijono268ca612006-02-07 12:34:11 +00001755 /* Create appropriate message. */
1756 switch (inv->state) {
1757 case PJSIP_INV_STATE_CALLING:
1758 case PJSIP_INV_STATE_EARLY:
1759 case PJSIP_INV_STATE_INCOMING:
1760
1761 if (inv->role == PJSIP_ROLE_UAC) {
1762
1763 /* For UAC when session has not been confirmed, create CANCEL. */
1764
1765 /* MUST have the original UAC INVITE transaction. */
1766 PJ_ASSERT_RETURN(inv->invite_tsx != NULL, PJ_EBUG);
1767
1768 /* But CANCEL should only be called when we have received a
1769 * provisional response. If we haven't received any responses,
1770 * just destroy the transaction.
1771 */
1772 if (inv->invite_tsx->status_code < 100) {
1773
Benny Prijono1dc8be02007-05-30 04:26:40 +00001774 pjsip_tsx_stop_retransmit(inv->invite_tsx);
1775 inv->cancelling = PJ_TRUE;
1776 inv->pending_cancel = PJ_TRUE;
Benny Prijonofccab712006-02-22 22:23:22 +00001777 *p_tdata = NULL;
Benny Prijono1dc8be02007-05-30 04:26:40 +00001778 PJ_LOG(4, (inv->obj_name, "Stopping retransmission, "
1779 "delaying CANCEL"));
Benny Prijonofccab712006-02-22 22:23:22 +00001780 return PJ_SUCCESS;
Benny Prijono268ca612006-02-07 12:34:11 +00001781 }
1782
1783 /* The CSeq here assumes that the dialog is started with an
1784 * INVITE session. This may not be correct; dialog can be
1785 * started as SUBSCRIBE session.
1786 * So fix this!
1787 */
1788 status = pjsip_endpt_create_cancel(inv->dlg->endpt,
1789 inv->invite_tsx->last_tx,
1790 &tdata);
1791
1792 } else {
1793
1794 /* For UAS, send a final response. */
1795 tdata = inv->invite_tsx->last_tx;
1796 PJ_ASSERT_RETURN(tdata != NULL, PJ_EINVALIDOP);
1797
Benny Prijono26ff9062006-02-21 23:47:00 +00001798 //status = pjsip_dlg_modify_response(inv->dlg, tdata, st_code,
1799 // st_text);
1800 status = pjsip_inv_answer(inv, st_code, st_text, NULL, &tdata);
Benny Prijono268ca612006-02-07 12:34:11 +00001801 }
1802 break;
1803
1804 case PJSIP_INV_STATE_CONNECTING:
1805 case PJSIP_INV_STATE_CONFIRMED:
1806 /* For established dialog, send BYE */
Benny Prijono1f61a8f2007-08-16 10:11:44 +00001807 status = pjsip_dlg_create_request(inv->dlg, pjsip_get_bye_method(),
1808 -1, &tdata);
Benny Prijono268ca612006-02-07 12:34:11 +00001809 break;
1810
1811 case PJSIP_INV_STATE_DISCONNECTED:
Benny Prijono268ca612006-02-07 12:34:11 +00001812 /* No need to do anything. */
Benny Prijonod4e0abd2006-03-05 11:53:36 +00001813 return PJSIP_ESESSIONTERMINATED;
Benny Prijono268ca612006-02-07 12:34:11 +00001814
1815 default:
1816 pj_assert("!Invalid operation!");
1817 return PJ_EINVALIDOP;
1818 }
1819
1820 if (status != PJ_SUCCESS)
1821 return status;
1822
1823
1824 /* Done */
1825
Benny Prijono0606e702007-05-22 12:21:40 +00001826 inv->cancelling = PJ_TRUE;
Benny Prijono268ca612006-02-07 12:34:11 +00001827 *p_tdata = tdata;
1828
1829 return PJ_SUCCESS;
1830}
1831
1832
1833/*
1834 * Create re-INVITE.
1835 */
1836PJ_DEF(pj_status_t) pjsip_inv_reinvite( pjsip_inv_session *inv,
1837 const pj_str_t *new_contact,
1838 const pjmedia_sdp_session *new_offer,
1839 pjsip_tx_data **p_tdata )
1840{
Benny Prijono26ff9062006-02-21 23:47:00 +00001841 pj_status_t status;
1842 pjsip_contact_hdr *contact_hdr = NULL;
Benny Prijono268ca612006-02-07 12:34:11 +00001843
Benny Prijono26ff9062006-02-21 23:47:00 +00001844 /* Check arguments. */
1845 PJ_ASSERT_RETURN(inv && p_tdata, PJ_EINVAL);
1846
1847 /* Must NOT have a pending INVITE transaction */
Benny Prijono2b787822007-06-12 15:29:52 +00001848 if (inv->invite_tsx!=NULL)
1849 return PJ_EINVALIDOP;
Benny Prijono26ff9062006-02-21 23:47:00 +00001850
1851
1852 pjsip_dlg_inc_lock(inv->dlg);
1853
1854 if (new_contact) {
1855 pj_str_t tmp;
1856 const pj_str_t STR_CONTACT = { "Contact", 7 };
1857
1858 pj_strdup_with_null(inv->dlg->pool, &tmp, new_contact);
Benny Prijonoa1e69682007-05-11 15:14:34 +00001859 contact_hdr = (pjsip_contact_hdr*)
1860 pjsip_parse_hdr(inv->dlg->pool, &STR_CONTACT,
Benny Prijono26ff9062006-02-21 23:47:00 +00001861 tmp.ptr, tmp.slen, NULL);
1862 if (!contact_hdr) {
1863 status = PJSIP_EINVALIDURI;
1864 goto on_return;
1865 }
1866 }
1867
1868
1869 if (new_offer) {
1870 if (!inv->neg) {
1871 status = pjmedia_sdp_neg_create_w_local_offer(inv->pool, new_offer,
1872 &inv->neg);
1873 if (status != PJ_SUCCESS)
1874 goto on_return;
1875
1876 } else switch (pjmedia_sdp_neg_get_state(inv->neg)) {
1877
1878 case PJMEDIA_SDP_NEG_STATE_NULL:
1879 pj_assert(!"Unexpected SDP neg state NULL");
1880 status = PJ_EBUG;
1881 goto on_return;
1882
1883 case PJMEDIA_SDP_NEG_STATE_LOCAL_OFFER:
1884 PJ_LOG(4,(inv->obj_name,
1885 "pjsip_inv_reinvite: already have an offer, new "
1886 "offer is ignored"));
1887 break;
1888
1889 case PJMEDIA_SDP_NEG_STATE_REMOTE_OFFER:
1890 status = pjmedia_sdp_neg_set_local_answer(inv->pool, inv->neg,
1891 new_offer);
1892 if (status != PJ_SUCCESS)
1893 goto on_return;
1894 break;
1895
1896 case PJMEDIA_SDP_NEG_STATE_WAIT_NEGO:
1897 PJ_LOG(4,(inv->obj_name,
1898 "pjsip_inv_reinvite: SDP in WAIT_NEGO state, new "
1899 "offer is ignored"));
1900 break;
1901
1902 case PJMEDIA_SDP_NEG_STATE_DONE:
1903 status = pjmedia_sdp_neg_modify_local_offer(inv->pool,inv->neg,
1904 new_offer);
1905 if (status != PJ_SUCCESS)
1906 goto on_return;
1907 break;
1908 }
1909 }
1910
1911 if (contact_hdr)
1912 inv->dlg->local.contact = contact_hdr;
1913
1914 status = pjsip_inv_invite(inv, p_tdata);
1915
1916on_return:
1917 pjsip_dlg_dec_lock(inv->dlg);
1918 return status;
Benny Prijono268ca612006-02-07 12:34:11 +00001919}
1920
1921/*
1922 * Create UPDATE.
1923 */
1924PJ_DEF(pj_status_t) pjsip_inv_update ( pjsip_inv_session *inv,
1925 const pj_str_t *new_contact,
Benny Prijono1f7767b2007-10-03 18:28:49 +00001926 const pjmedia_sdp_session *offer,
Benny Prijono268ca612006-02-07 12:34:11 +00001927 pjsip_tx_data **p_tdata )
1928{
Benny Prijono1f7767b2007-10-03 18:28:49 +00001929 pjsip_contact_hdr *contact_hdr = NULL;
1930 pjsip_tx_data *tdata = NULL;
1931 pjmedia_sdp_session *sdp_copy;
1932 pj_status_t status = PJ_SUCCESS;
Benny Prijono268ca612006-02-07 12:34:11 +00001933
Benny Prijono1f7767b2007-10-03 18:28:49 +00001934 /* Verify arguments. */
1935 PJ_ASSERT_RETURN(inv && p_tdata && offer, PJ_EINVAL);
1936
1937 /* Dialog must have been established */
1938 PJ_ASSERT_RETURN(inv->dlg->state == PJSIP_DIALOG_STATE_ESTABLISHED,
1939 PJ_EINVALIDOP);
1940
1941 /* Invite session must not have been disconnected */
1942 PJ_ASSERT_RETURN(inv->state < PJSIP_INV_STATE_DISCONNECTED,
1943 PJ_EINVALIDOP);
1944
1945 /* Lock dialog. */
1946 pjsip_dlg_inc_lock(inv->dlg);
1947
1948 /* Process offer */
1949 if (pjmedia_sdp_neg_get_state(inv->neg)!=PJMEDIA_SDP_NEG_STATE_DONE) {
1950 PJ_LOG(4,(inv->dlg->obj_name,
1951 "Invalid SDP offer/answer state for UPDATE"));
1952 status = PJ_EINVALIDOP;
1953 goto on_error;
1954 }
1955
1956 status = pjmedia_sdp_neg_modify_local_offer(inv->pool,inv->neg,
1957 offer);
1958 if (status != PJ_SUCCESS)
1959 goto on_error;
1960
1961
1962 /* Update Contact if required */
1963 if (new_contact) {
1964 pj_str_t tmp;
1965 const pj_str_t STR_CONTACT = { "Contact", 7 };
1966
1967 pj_strdup_with_null(inv->dlg->pool, &tmp, new_contact);
1968 contact_hdr = (pjsip_contact_hdr*)
1969 pjsip_parse_hdr(inv->dlg->pool, &STR_CONTACT,
1970 tmp.ptr, tmp.slen, NULL);
1971 if (!contact_hdr) {
1972 status = PJSIP_EINVALIDURI;
1973 goto on_error;
1974 }
1975
1976 inv->dlg->local.contact = contact_hdr;
1977 }
1978
1979 /* Create request */
1980 status = pjsip_dlg_create_request(inv->dlg, &pjsip_update_method,
1981 -1, &tdata);
1982 if (status != PJ_SUCCESS)
1983 goto on_error;
1984
1985 /* Attach SDP body */
1986 sdp_copy = pjmedia_sdp_session_clone(tdata->pool, offer);
1987 pjsip_create_sdp_body(tdata->pool, sdp_copy, &tdata->msg->body);
1988
1989 /* Unlock dialog. */
1990 pjsip_dlg_dec_lock(inv->dlg);
1991
1992 *p_tdata = tdata;
1993
1994 return PJ_SUCCESS;
1995
1996on_error:
1997 if (tdata)
1998 pjsip_tx_data_dec_ref(tdata);
1999
2000 /* Unlock dialog. */
2001 pjsip_dlg_dec_lock(inv->dlg);
2002
2003 return status;
Benny Prijono268ca612006-02-07 12:34:11 +00002004}
2005
2006/*
Benny Prijonod5f9f422007-11-25 04:40:07 +00002007 * Create an ACK request.
2008 */
2009PJ_DEF(pj_status_t) pjsip_inv_create_ack(pjsip_inv_session *inv,
2010 int cseq,
2011 pjsip_tx_data **p_tdata)
2012{
2013 const pjmedia_sdp_session *sdp = NULL;
2014 pj_status_t status;
2015
2016 PJ_ASSERT_RETURN(inv && p_tdata, PJ_EINVAL);
2017
2018 /* Lock dialog. */
2019 pjsip_dlg_inc_lock(inv->dlg);
2020
2021 /* Destroy last_ack */
2022 if (inv->last_ack) {
2023 pjsip_tx_data_dec_ref(inv->last_ack);
2024 inv->last_ack = NULL;
2025 }
2026
2027 /* Create new ACK request */
2028 status = pjsip_dlg_create_request(inv->dlg, pjsip_get_ack_method(),
2029 cseq, &inv->last_ack);
2030 if (status != PJ_SUCCESS) {
2031 pjsip_dlg_dec_lock(inv->dlg);
2032 return status;
2033 }
2034
2035 /* See if we have pending SDP answer to send */
2036 sdp = inv_has_pending_answer(inv, inv->invite_tsx);
2037 if (sdp) {
2038 inv->last_ack->msg->body = create_sdp_body(inv->last_ack->pool, sdp);
2039 }
2040
2041 /* Keep this for subsequent response retransmission */
2042 inv->last_ack_cseq = cseq;
2043 pjsip_tx_data_add_ref(inv->last_ack);
2044
2045 /* Done */
2046 *p_tdata = inv->last_ack;
2047
2048 /* Unlock dialog. */
2049 pjsip_dlg_dec_lock(inv->dlg);
2050
2051 return PJ_SUCCESS;
2052}
2053
2054/*
Benny Prijono268ca612006-02-07 12:34:11 +00002055 * Send a request or response message.
2056 */
2057PJ_DEF(pj_status_t) pjsip_inv_send_msg( pjsip_inv_session *inv,
Benny Prijonoe8b0d3b2006-03-17 17:57:52 +00002058 pjsip_tx_data *tdata)
Benny Prijono268ca612006-02-07 12:34:11 +00002059{
2060 pj_status_t status;
2061
2062 /* Verify arguments. */
2063 PJ_ASSERT_RETURN(inv && tdata, PJ_EINVAL);
2064
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00002065 PJ_LOG(5,(inv->obj_name, "Sending %s",
2066 pjsip_tx_data_get_info(tdata)));
2067
Benny Prijono268ca612006-02-07 12:34:11 +00002068 if (tdata->msg->type == PJSIP_REQUEST_MSG) {
Benny Prijonoa66c7152006-02-09 01:26:14 +00002069 struct tsx_inv_data *tsx_inv_data;
Benny Prijono268ca612006-02-07 12:34:11 +00002070
Benny Prijono64158af2006-04-04 11:06:34 +00002071 pjsip_dlg_inc_lock(inv->dlg);
Benny Prijono268ca612006-02-07 12:34:11 +00002072
Benny Prijono22e48c92008-03-20 14:40:50 +00002073 /* Check again that we didn't receive incoming re-INVITE */
Benny Prijono9ae5dfc2008-03-27 17:30:51 +00002074 if (tdata->msg->line.req.method.id==PJSIP_INVITE_METHOD &&
2075 inv->invite_tsx)
2076 {
Benny Prijono22e48c92008-03-20 14:40:50 +00002077 pjsip_tx_data_dec_ref(tdata);
2078 pjsip_dlg_dec_lock(inv->dlg);
2079 return PJ_EINVALIDOP;
2080 }
2081
2082 /* Associate our data in outgoing invite transaction */
Benny Prijonoa1e69682007-05-11 15:14:34 +00002083 tsx_inv_data = PJ_POOL_ZALLOC_T(inv->pool, struct tsx_inv_data);
Benny Prijonoa66c7152006-02-09 01:26:14 +00002084 tsx_inv_data->inv = inv;
2085
Benny Prijono64158af2006-04-04 11:06:34 +00002086 pjsip_dlg_dec_lock(inv->dlg);
2087
2088 status = pjsip_dlg_send_request(inv->dlg, tdata, mod_inv.mod.id,
2089 tsx_inv_data);
2090 if (status != PJ_SUCCESS)
2091 return status;
Benny Prijono268ca612006-02-07 12:34:11 +00002092
2093 } else {
2094 pjsip_cseq_hdr *cseq;
2095
2096 /* Can only do this to send response to original INVITE
2097 * request.
2098 */
Benny Prijonoa1e69682007-05-11 15:14:34 +00002099 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 +00002100 && (cseq->cseq == inv->invite_tsx->cseq),
Benny Prijono268ca612006-02-07 12:34:11 +00002101 PJ_EINVALIDOP);
2102
Benny Prijonodcfc0ba2007-09-30 16:50:27 +00002103 if (inv->options & PJSIP_INV_REQUIRE_100REL) {
Benny Prijono1f7767b2007-10-03 18:28:49 +00002104 status = pjsip_100rel_tx_response(inv, tdata);
Benny Prijonodcfc0ba2007-09-30 16:50:27 +00002105 } else
Benny Prijonodcfc0ba2007-09-30 16:50:27 +00002106 {
Benny Prijono1f7767b2007-10-03 18:28:49 +00002107 status = pjsip_dlg_send_response(inv->dlg, inv->invite_tsx, tdata);
Benny Prijonodcfc0ba2007-09-30 16:50:27 +00002108 }
2109
Benny Prijono268ca612006-02-07 12:34:11 +00002110 if (status != PJ_SUCCESS)
2111 return status;
2112 }
2113
2114 /* Done (?) */
2115 return PJ_SUCCESS;
2116}
2117
2118
Benny Prijono8ad55352006-02-08 11:16:05 +00002119/*
2120 * Respond to incoming CANCEL request.
2121 */
2122static void inv_respond_incoming_cancel(pjsip_inv_session *inv,
2123 pjsip_transaction *cancel_tsx,
2124 pjsip_rx_data *rdata)
2125{
2126 pjsip_tx_data *tdata;
2127 pjsip_transaction *invite_tsx;
2128 pj_str_t key;
2129 pj_status_t status;
2130
2131 /* See if we have matching INVITE server transaction: */
2132
2133 pjsip_tsx_create_key(rdata->tp_info.pool, &key, PJSIP_ROLE_UAS,
Benny Prijono1f61a8f2007-08-16 10:11:44 +00002134 pjsip_get_invite_method(), rdata);
Benny Prijono8ad55352006-02-08 11:16:05 +00002135 invite_tsx = pjsip_tsx_layer_find_tsx(&key, PJ_TRUE);
2136
2137 if (invite_tsx == NULL) {
2138
2139 /* Invite transaction not found!
Benny Prijonoc5145762007-11-23 12:04:40 +00002140 * Respond CANCEL with 481 (RFC 3261 Section 9.2 page 55)
Benny Prijono8ad55352006-02-08 11:16:05 +00002141 */
Benny Prijonoc5145762007-11-23 12:04:40 +00002142 status = pjsip_dlg_create_response( inv->dlg, rdata, 481, NULL,
Benny Prijono8ad55352006-02-08 11:16:05 +00002143 &tdata);
2144
2145 } else {
2146 /* Always answer CANCEL will 200 (OK) regardless of
2147 * the state of the INVITE transaction.
2148 */
2149 status = pjsip_dlg_create_response( inv->dlg, rdata, 200, NULL,
2150 &tdata);
2151 }
2152
2153 /* See if we have created the response successfully. */
2154 if (status != PJ_SUCCESS) return;
2155
2156 /* Send the CANCEL response */
2157 status = pjsip_dlg_send_response(inv->dlg, cancel_tsx, tdata);
2158 if (status != PJ_SUCCESS) return;
2159
2160
2161 /* See if we need to terminate the UAS INVITE transaction
2162 * with 487 (Request Terminated) response.
2163 */
2164 if (invite_tsx && invite_tsx->status_code < 200) {
2165
2166 pj_assert(invite_tsx->last_tx != NULL);
2167
2168 tdata = invite_tsx->last_tx;
2169
2170 status = pjsip_dlg_modify_response(inv->dlg, tdata, 487, NULL);
Benny Prijonofc8bb142007-11-08 09:56:50 +00002171 if (status == PJ_SUCCESS) {
2172 /* Remove the message body */
2173 tdata->msg->body = NULL;
Benny Prijono8ad55352006-02-08 11:16:05 +00002174 pjsip_dlg_send_response(inv->dlg, invite_tsx, tdata);
Benny Prijonofc8bb142007-11-08 09:56:50 +00002175 }
Benny Prijono8ad55352006-02-08 11:16:05 +00002176 }
2177
2178 if (invite_tsx)
2179 pj_mutex_unlock(invite_tsx->mutex);
2180}
2181
2182
2183/*
2184 * Respond to incoming BYE request.
2185 */
2186static void inv_respond_incoming_bye( pjsip_inv_session *inv,
2187 pjsip_transaction *bye_tsx,
2188 pjsip_rx_data *rdata,
2189 pjsip_event *e )
2190{
2191 pj_status_t status;
2192 pjsip_tx_data *tdata;
2193
2194 /* Respond BYE with 200: */
2195
2196 status = pjsip_dlg_create_response(inv->dlg, rdata, 200, NULL, &tdata);
2197 if (status != PJ_SUCCESS) return;
2198
2199 status = pjsip_dlg_send_response(inv->dlg, bye_tsx, tdata);
2200 if (status != PJ_SUCCESS) return;
2201
2202 /* Terminate session: */
2203
Benny Prijonod4e0abd2006-03-05 11:53:36 +00002204 if (inv->state != PJSIP_INV_STATE_DISCONNECTED) {
Benny Prijono0b6340c2006-06-13 22:21:23 +00002205 inv_set_cause(inv, PJSIP_SC_OK, NULL);
Benny Prijono8ad55352006-02-08 11:16:05 +00002206 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
Benny Prijonod4e0abd2006-03-05 11:53:36 +00002207 }
Benny Prijono8ad55352006-02-08 11:16:05 +00002208}
2209
2210/*
Benny Prijono38998232006-02-08 22:44:25 +00002211 * Respond to BYE request.
2212 */
2213static void inv_handle_bye_response( pjsip_inv_session *inv,
2214 pjsip_transaction *tsx,
2215 pjsip_rx_data *rdata,
2216 pjsip_event *e )
2217{
2218 pj_status_t status;
2219
2220 if (e->body.tsx_state.type != PJSIP_EVENT_RX_MSG) {
Benny Prijono0b6340c2006-06-13 22:21:23 +00002221 inv_set_cause(inv, PJSIP_SC_OK, NULL);
Benny Prijono38998232006-02-08 22:44:25 +00002222 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
2223 return;
2224 }
2225
2226 /* Handle 401/407 challenge. */
2227 if (tsx->status_code == 401 || tsx->status_code == 407) {
2228
2229 pjsip_tx_data *tdata;
2230
2231 status = pjsip_auth_clt_reinit_req( &inv->dlg->auth_sess,
2232 rdata,
2233 tsx->last_tx,
2234 &tdata);
2235
2236 if (status != PJ_SUCCESS) {
2237
2238 /* Does not have proper credentials.
2239 * End the session anyway.
2240 */
Benny Prijono0b6340c2006-06-13 22:21:23 +00002241 inv_set_cause(inv, PJSIP_SC_OK, NULL);
Benny Prijono38998232006-02-08 22:44:25 +00002242 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
2243
2244 } else {
2245 /* Re-send BYE. */
Benny Prijonoe8b0d3b2006-03-17 17:57:52 +00002246 status = pjsip_inv_send_msg(inv, tdata);
Benny Prijono38998232006-02-08 22:44:25 +00002247 }
2248
2249 } else {
2250
2251 /* End the session. */
Benny Prijono0b6340c2006-06-13 22:21:23 +00002252 inv_set_cause(inv, PJSIP_SC_OK, NULL);
Benny Prijono38998232006-02-08 22:44:25 +00002253 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
2254 }
2255
2256}
2257
2258/*
Benny Prijono1f7767b2007-10-03 18:28:49 +00002259 * Respond to incoming UPDATE request.
2260 */
2261static void inv_respond_incoming_update(pjsip_inv_session *inv,
2262 pjsip_rx_data *rdata)
2263{
2264 pjmedia_sdp_neg_state neg_state;
2265 pj_status_t status;
2266 pjsip_tx_data *tdata = NULL;
2267
2268 neg_state = pjmedia_sdp_neg_get_state(inv->neg);
2269
2270 /* Send 491 if we receive UPDATE while we're waiting for an answer */
2271 if (neg_state == PJMEDIA_SDP_NEG_STATE_LOCAL_OFFER) {
2272 status = pjsip_dlg_create_response(inv->dlg, rdata,
2273 PJSIP_SC_REQUEST_PENDING, NULL,
2274 &tdata);
2275 }
2276 /* Send 500 with Retry-After header set randomly between 0 and 10 if we
2277 * receive UPDATE while we haven't sent answer.
2278 */
2279 else if (neg_state == PJMEDIA_SDP_NEG_STATE_REMOTE_OFFER ||
2280 neg_state == PJMEDIA_SDP_NEG_STATE_WAIT_NEGO) {
2281 status = pjsip_dlg_create_response(inv->dlg, rdata,
2282 PJSIP_SC_INTERNAL_SERVER_ERROR,
2283 NULL, &tdata);
2284
Benny Prijonoc5cbc052007-11-08 09:44:08 +00002285 /* If UPDATE doesn't contain SDP, just respond with 200/OK.
2286 * This is a valid scenario according to session-timer draft.
2287 */
2288 } else if (rdata->msg_info.msg->body == NULL) {
2289
2290 status = pjsip_dlg_create_response(inv->dlg, rdata,
2291 200, NULL, &tdata);
2292
Benny Prijono1f7767b2007-10-03 18:28:49 +00002293 } else {
2294 /* We receive new offer from remote */
2295 inv_check_sdp_in_incoming_msg(inv, pjsip_rdata_get_tsx(rdata), rdata);
2296
2297 /* Application MUST have supplied the answer by now.
2298 * If so, negotiate the SDP.
2299 */
2300 neg_state = pjmedia_sdp_neg_get_state(inv->neg);
2301 if (neg_state != PJMEDIA_SDP_NEG_STATE_WAIT_NEGO ||
2302 (status=inv_negotiate_sdp(inv)) != PJ_SUCCESS)
2303 {
2304 /* Negotiation has failed */
2305 status = pjsip_dlg_create_response(inv->dlg, rdata,
2306 PJSIP_SC_NOT_ACCEPTABLE_HERE,
2307 NULL, &tdata);
2308 } else {
2309 /* New media has been negotiated successfully, send 200/OK */
2310 status = pjsip_dlg_create_response(inv->dlg, rdata,
2311 PJSIP_SC_OK, NULL, &tdata);
2312 if (status == PJ_SUCCESS) {
Benny Prijono9569a0b2007-10-04 15:35:26 +00002313 const pjmedia_sdp_session *sdp;
Benny Prijono1f7767b2007-10-03 18:28:49 +00002314 status = pjmedia_sdp_neg_get_active_local(inv->neg, &sdp);
2315 if (status == PJ_SUCCESS)
2316 tdata->msg->body = create_sdp_body(tdata->pool, sdp);
2317 }
2318 }
2319 }
2320
2321 if (status != PJ_SUCCESS) {
2322 if (tdata != NULL) {
2323 pjsip_tx_data_dec_ref(tdata);
2324 tdata = NULL;
2325 }
2326 return;
2327 }
2328
2329 pjsip_dlg_send_response(inv->dlg, pjsip_rdata_get_tsx(rdata), tdata);
2330}
2331
2332
2333/*
2334 * Handle incoming response to UAC UPDATE request.
2335 */
2336static void inv_handle_update_response( pjsip_inv_session *inv,
2337 pjsip_event *e)
2338{
2339 pjsip_transaction *tsx = e->body.tsx_state.tsx;
2340 struct tsx_inv_data *tsx_inv_data = NULL;
2341 pj_status_t status = -1;
2342
Benny Prijono48ab2b72007-11-08 09:24:30 +00002343 /* Handle 401/407 challenge. */
Benny Prijono1f7767b2007-10-03 18:28:49 +00002344 if (tsx->state == PJSIP_TSX_STATE_COMPLETED &&
Benny Prijono48ab2b72007-11-08 09:24:30 +00002345 (tsx->status_code == 401 || tsx->status_code == 407)) {
2346
2347 pjsip_tx_data *tdata;
2348
2349 status = pjsip_auth_clt_reinit_req( &inv->dlg->auth_sess,
2350 e->body.tsx_state.src.rdata,
2351 tsx->last_tx,
2352 &tdata);
2353
2354 if (status != PJ_SUCCESS) {
2355
Benny Prijonob2ad04a2008-06-26 13:24:10 +00002356 /* Somehow failed. Probably it's not a good idea to terminate
2357 * the session since this is just a request within dialog. And
2358 * even if we terminate we should send BYE.
Benny Prijono48ab2b72007-11-08 09:24:30 +00002359 */
Benny Prijonob2ad04a2008-06-26 13:24:10 +00002360 /*
Benny Prijono48ab2b72007-11-08 09:24:30 +00002361 inv_set_cause(inv, PJSIP_SC_OK, NULL);
2362 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
Benny Prijonob2ad04a2008-06-26 13:24:10 +00002363 */
Benny Prijono48ab2b72007-11-08 09:24:30 +00002364
2365 } else {
Benny Prijono0b4c57b2008-06-25 10:15:01 +00002366 /* Re-send request. */
Benny Prijono48ab2b72007-11-08 09:24:30 +00002367 status = pjsip_inv_send_msg(inv, tdata);
2368 }
2369
2370 /* Process 2xx response */
2371 } else if (tsx->state == PJSIP_TSX_STATE_COMPLETED &&
Benny Prijono1f7767b2007-10-03 18:28:49 +00002372 tsx->status_code/100 == 2 &&
2373 e->body.tsx_state.src.rdata->msg_info.msg->body)
2374 {
2375 status = inv_check_sdp_in_incoming_msg(inv, tsx,
2376 e->body.tsx_state.src.rdata);
2377
2378 } else {
2379 /* Get/attach invite session's transaction data */
2380 tsx_inv_data = (struct tsx_inv_data*)tsx->mod_data[mod_inv.mod.id];
2381 if (tsx_inv_data == NULL) {
2382 tsx_inv_data=PJ_POOL_ZALLOC_T(tsx->pool, struct tsx_inv_data);
2383 tsx_inv_data->inv = inv;
2384 tsx->mod_data[mod_inv.mod.id] = tsx_inv_data;
2385 }
2386 }
2387
2388 /* Otherwise if we don't get successful response, cancel
2389 * our negotiator.
2390 */
2391 if (status != PJ_SUCCESS &&
2392 pjmedia_sdp_neg_get_state(inv->neg)==PJMEDIA_SDP_NEG_STATE_LOCAL_OFFER &&
2393 tsx_inv_data && tsx_inv_data->sdp_done == PJ_FALSE)
2394 {
2395 pjmedia_sdp_neg_cancel_offer(inv->neg);
2396
2397 /* Prevent from us cancelling different offer! */
2398 tsx_inv_data->sdp_done = PJ_TRUE;
2399 }
2400}
2401
2402
2403/*
2404 * Handle incoming reliable response.
2405 */
2406static void inv_handle_incoming_reliable_response(pjsip_inv_session *inv,
2407 pjsip_rx_data *rdata)
2408{
2409 pjsip_tx_data *tdata;
Benny Prijono9569a0b2007-10-04 15:35:26 +00002410 const pjmedia_sdp_session *sdp;
Benny Prijono1f7767b2007-10-03 18:28:49 +00002411 pj_status_t status;
2412
2413 /* Create PRACK */
2414 status = pjsip_100rel_create_prack(inv, rdata, &tdata);
2415 if (status != PJ_SUCCESS)
2416 return;
2417
2418 /* See if we need to attach SDP answer on the PRACK request */
2419 sdp = inv_has_pending_answer(inv, pjsip_rdata_get_tsx(rdata));
2420 if (sdp) {
2421 tdata->msg->body = create_sdp_body(tdata->pool, sdp);
2422 }
2423
2424 /* Send PRACK (must be using 100rel module!) */
2425 pjsip_100rel_send_prack(inv, tdata);
2426}
2427
2428
2429/*
2430 * Handle incoming PRACK.
2431 */
2432static void inv_respond_incoming_prack(pjsip_inv_session *inv,
2433 pjsip_rx_data *rdata)
2434{
2435 pj_status_t status;
2436
2437 /* Run through 100rel module to see if we can accept this
2438 * PRACK request. The 100rel will send 200/OK to PRACK request.
2439 */
2440 status = pjsip_100rel_on_rx_prack(inv, rdata);
2441 if (status != PJ_SUCCESS)
2442 return;
2443
2444 /* Now check for SDP answer in the PRACK request */
2445 if (rdata->msg_info.msg->body) {
2446 status = inv_check_sdp_in_incoming_msg(inv,
2447 pjsip_rdata_get_tsx(rdata), rdata);
2448 } else {
2449 /* No SDP body */
2450 status = -1;
2451 }
2452
2453 /* If SDP negotiation has been successful, also mark the
2454 * SDP negotiation flag in the invite transaction to be
2455 * done too.
2456 */
2457 if (status == PJ_SUCCESS && inv->invite_tsx) {
2458 struct tsx_inv_data *tsx_inv_data;
2459
2460 /* Get/attach invite session's transaction data */
2461 tsx_inv_data = (struct tsx_inv_data*)
2462 inv->invite_tsx->mod_data[mod_inv.mod.id];
2463 if (tsx_inv_data == NULL) {
2464 tsx_inv_data = PJ_POOL_ZALLOC_T(inv->invite_tsx->pool,
2465 struct tsx_inv_data);
2466 tsx_inv_data->inv = inv;
2467 inv->invite_tsx->mod_data[mod_inv.mod.id] = tsx_inv_data;
2468 }
2469
2470 tsx_inv_data->sdp_done = PJ_TRUE;
2471 }
2472}
2473
2474
2475/*
Benny Prijono8ad55352006-02-08 11:16:05 +00002476 * State NULL is before anything is sent/received.
2477 */
2478static void inv_on_state_null( pjsip_inv_session *inv, pjsip_event *e)
Benny Prijono268ca612006-02-07 12:34:11 +00002479{
2480 pjsip_transaction *tsx = e->body.tsx_state.tsx;
2481 pjsip_dialog *dlg = pjsip_tsx_get_dlg(tsx);
2482
2483 PJ_ASSERT_ON_FAIL(tsx && dlg, return);
2484
2485 if (tsx->method.id == PJSIP_INVITE_METHOD) {
2486
Benny Prijono64f851e2006-02-23 13:49:28 +00002487 /* Keep the initial INVITE transaction. */
2488 if (inv->invite_tsx == NULL)
2489 inv->invite_tsx = tsx;
Benny Prijono268ca612006-02-07 12:34:11 +00002490
Benny Prijono64f851e2006-02-23 13:49:28 +00002491 if (dlg->role == PJSIP_ROLE_UAC) {
Benny Prijono268ca612006-02-07 12:34:11 +00002492
2493 switch (tsx->state) {
2494 case PJSIP_TSX_STATE_CALLING:
Benny Prijono8ad55352006-02-08 11:16:05 +00002495 inv_set_state(inv, PJSIP_INV_STATE_CALLING, e);
Benny Prijono268ca612006-02-07 12:34:11 +00002496 break;
2497 default:
Benny Prijono64f851e2006-02-23 13:49:28 +00002498 inv_on_state_calling(inv, e);
Benny Prijono268ca612006-02-07 12:34:11 +00002499 break;
2500 }
2501
2502 } else {
2503 switch (tsx->state) {
2504 case PJSIP_TSX_STATE_TRYING:
Benny Prijono8ad55352006-02-08 11:16:05 +00002505 inv_set_state(inv, PJSIP_INV_STATE_INCOMING, e);
Benny Prijono268ca612006-02-07 12:34:11 +00002506 break;
Benny Prijono38998232006-02-08 22:44:25 +00002507 case PJSIP_TSX_STATE_PROCEEDING:
2508 inv_set_state(inv, PJSIP_INV_STATE_INCOMING, e);
2509 if (tsx->status_code > 100)
2510 inv_set_state(inv, PJSIP_INV_STATE_EARLY, e);
2511 break;
Benny Prijono268ca612006-02-07 12:34:11 +00002512 default:
Benny Prijono64f851e2006-02-23 13:49:28 +00002513 inv_on_state_incoming(inv, e);
2514 break;
Benny Prijono268ca612006-02-07 12:34:11 +00002515 }
2516 }
2517
2518 } else {
2519 pj_assert(!"Unexpected transaction type");
2520 }
2521}
2522
Benny Prijono8ad55352006-02-08 11:16:05 +00002523/*
Benny Prijono0b4c57b2008-06-25 10:15:01 +00002524 * Generic UAC transaction handler:
2525 * - resend request on 401 or 407 response.
2526 * - terminate dialog on 408 and 481 response.
2527 */
2528static pj_bool_t handle_uac_tsx_response(pjsip_inv_session *inv,
2529 pjsip_event *e)
2530{
2531 /* RFC 3261 Section 12.2.1.2:
2532 * If the response for a request within a dialog is a 481
2533 * (Call/Transaction Does Not Exist) or a 408 (Request Timeout), the UAC
2534 * SHOULD terminate the dialog. A UAC SHOULD also terminate a dialog if
2535 * no response at all is received for the request (the client
2536 * transaction would inform the TU about the timeout.)
2537 *
2538 * For INVITE initiated dialogs, terminating the dialog consists of
2539 * sending a BYE.
2540 *
2541 * Note:
2542 * according to X, this should terminate dialog usage only, not the
2543 * dialog.
2544 */
2545 pjsip_transaction *tsx = e->body.tsx_state.tsx;
2546
2547 pj_assert(tsx->role == PJSIP_UAC_ROLE);
2548
2549 /* Note that 481 response to CANCEL does not terminate dialog usage,
2550 * but only the transaction.
2551 */
Benny Prijono61fc5e62008-06-25 18:35:31 +00002552 if (inv->state != PJSIP_INV_STATE_DISCONNECTED &&
2553 ((tsx->status_code == PJSIP_SC_CALL_TSX_DOES_NOT_EXIST &&
Benny Prijono0b4c57b2008-06-25 10:15:01 +00002554 tsx->method.id != PJSIP_CANCEL_METHOD) ||
Benny Prijono61fc5e62008-06-25 18:35:31 +00002555 tsx->status_code == PJSIP_SC_REQUEST_TIMEOUT ||
2556 tsx->status_code == PJSIP_SC_TSX_TIMEOUT ||
2557 tsx->status_code == PJSIP_SC_TSX_TRANSPORT_ERROR))
Benny Prijono0b4c57b2008-06-25 10:15:01 +00002558 {
2559 pjsip_tx_data *bye;
2560 pj_status_t status;
2561
2562 inv_set_cause(inv, tsx->status_code, &tsx->status_text);
2563 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
2564
2565 /* Send BYE */
2566 status = pjsip_dlg_create_request(inv->dlg, pjsip_get_bye_method(),
2567 -1, &bye);
2568 if (status == PJ_SUCCESS) {
2569 pjsip_inv_send_msg(inv, bye);
2570 }
2571
2572 return PJ_TRUE; /* Handled */
2573
2574 }
2575 /* Handle 401/407 challenge. */
2576 else if (tsx->state == PJSIP_TSX_STATE_COMPLETED &&
2577 (tsx->status_code == PJSIP_SC_UNAUTHORIZED ||
2578 tsx->status_code == PJSIP_SC_PROXY_AUTHENTICATION_REQUIRED))
2579 {
2580
2581 pjsip_tx_data *tdata;
2582 pj_status_t status;
2583
Benny Prijonob2ad04a2008-06-26 13:24:10 +00002584 if (tsx->method.id == PJSIP_INVITE_METHOD)
2585 inv->invite_tsx = NULL;
2586
Benny Prijono0b4c57b2008-06-25 10:15:01 +00002587 status = pjsip_auth_clt_reinit_req( &inv->dlg->auth_sess,
2588 e->body.tsx_state.src.rdata,
2589 tsx->last_tx, &tdata);
2590
2591 if (status != PJ_SUCCESS) {
Benny Prijonob2ad04a2008-06-26 13:24:10 +00002592 /* Somehow failed. Probably it's not a good idea to terminate
2593 * the session since this is just a request within dialog. And
2594 * even if we terminate we should send BYE.
2595 */
2596 /*
Benny Prijono0b4c57b2008-06-25 10:15:01 +00002597 inv_set_cause(inv, PJSIP_SC_OK, NULL);
2598 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
Benny Prijonob2ad04a2008-06-26 13:24:10 +00002599 */
Benny Prijono0b4c57b2008-06-25 10:15:01 +00002600
2601 } else {
2602 /* Re-send request. */
Benny Prijono0b4c57b2008-06-25 10:15:01 +00002603 status = pjsip_inv_send_msg(inv, tdata);
2604 }
2605
2606 return PJ_TRUE; /* Handled */
2607
2608 } else {
2609 return PJ_FALSE; /* Unhandled */
2610 }
2611}
2612
2613
2614/*
Benny Prijono8ad55352006-02-08 11:16:05 +00002615 * State CALLING is after sending initial INVITE request but before
2616 * any response (with tag) is received.
2617 */
2618static void inv_on_state_calling( pjsip_inv_session *inv, pjsip_event *e)
Benny Prijono268ca612006-02-07 12:34:11 +00002619{
2620 pjsip_transaction *tsx = e->body.tsx_state.tsx;
2621 pjsip_dialog *dlg = pjsip_tsx_get_dlg(tsx);
Benny Prijonoccf95622006-02-07 18:48:01 +00002622 pj_status_t status;
Benny Prijono268ca612006-02-07 12:34:11 +00002623
2624 PJ_ASSERT_ON_FAIL(tsx && dlg, return);
2625
Benny Prijono8ad55352006-02-08 11:16:05 +00002626 if (tsx == inv->invite_tsx) {
Benny Prijono268ca612006-02-07 12:34:11 +00002627
2628 switch (tsx->state) {
2629
Benny Prijono64f851e2006-02-23 13:49:28 +00002630 case PJSIP_TSX_STATE_CALLING:
2631 inv_set_state(inv, PJSIP_INV_STATE_CALLING, e);
2632 break;
2633
Benny Prijono268ca612006-02-07 12:34:11 +00002634 case PJSIP_TSX_STATE_PROCEEDING:
Benny Prijono1dc8be02007-05-30 04:26:40 +00002635 if (inv->pending_cancel) {
2636 pjsip_tx_data *cancel;
2637
2638 inv->pending_cancel = PJ_FALSE;
2639
2640 status = pjsip_inv_end_session(inv, 487, NULL, &cancel);
2641 if (status == PJ_SUCCESS && cancel)
2642 status = pjsip_inv_send_msg(inv, cancel);
2643 }
2644
Benny Prijono268ca612006-02-07 12:34:11 +00002645 if (dlg->remote.info->tag.slen) {
Benny Prijonoa66c7152006-02-09 01:26:14 +00002646
Benny Prijono8ad55352006-02-08 11:16:05 +00002647 inv_set_state(inv, PJSIP_INV_STATE_EARLY, e);
Benny Prijonoa66c7152006-02-09 01:26:14 +00002648
2649 inv_check_sdp_in_incoming_msg(inv, tsx,
2650 e->body.tsx_state.src.rdata);
2651
Benny Prijono1f7767b2007-10-03 18:28:49 +00002652 if (pjsip_100rel_is_reliable(e->body.tsx_state.src.rdata)) {
2653 inv_handle_incoming_reliable_response(
2654 inv, e->body.tsx_state.src.rdata);
2655 }
2656
Benny Prijono268ca612006-02-07 12:34:11 +00002657 } else {
2658 /* Ignore 100 (Trying) response, as it doesn't change
2659 * session state. It only ceases retransmissions.
2660 */
2661 }
2662 break;
2663
2664 case PJSIP_TSX_STATE_COMPLETED:
2665 if (tsx->status_code/100 == 2) {
2666
2667 /* This should not happen.
2668 * When transaction receives 2xx, it should be terminated
2669 */
2670 pj_assert(0);
Benny Prijono8ad55352006-02-08 11:16:05 +00002671 inv_set_state(inv, PJSIP_INV_STATE_CONNECTING, e);
Benny Prijonoa66c7152006-02-09 01:26:14 +00002672
2673 inv_check_sdp_in_incoming_msg(inv, tsx,
2674 e->body.tsx_state.src.rdata);
Benny Prijono268ca612006-02-07 12:34:11 +00002675
Benny Prijono0606e702007-05-22 12:21:40 +00002676 } else if ((tsx->status_code==401 || tsx->status_code==407) &&
2677 !inv->cancelling)
2678 {
Benny Prijonoccf95622006-02-07 18:48:01 +00002679
2680 /* Handle authentication failure:
2681 * Resend the request with Authorization header.
2682 */
2683 pjsip_tx_data *tdata;
2684
Benny Prijono8ad55352006-02-08 11:16:05 +00002685 status = pjsip_auth_clt_reinit_req(&inv->dlg->auth_sess,
Benny Prijonoccf95622006-02-07 18:48:01 +00002686 e->body.tsx_state.src.rdata,
2687 tsx->last_tx,
2688 &tdata);
2689
2690 if (status != PJ_SUCCESS) {
2691
2692 /* Does not have proper credentials.
2693 * End the session.
2694 */
Benny Prijono0b6340c2006-06-13 22:21:23 +00002695 inv_set_cause(inv, tsx->status_code, &tsx->status_text);
Benny Prijono8ad55352006-02-08 11:16:05 +00002696 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
Benny Prijonoccf95622006-02-07 18:48:01 +00002697
2698 } else {
2699
2700 /* Restart session. */
Benny Prijono8ad55352006-02-08 11:16:05 +00002701 inv->state = PJSIP_INV_STATE_NULL;
2702 inv->invite_tsx = NULL;
Benny Prijonodcfc0ba2007-09-30 16:50:27 +00002703 if (inv->last_answer) {
2704 pjsip_tx_data_dec_ref(inv->last_answer);
2705 inv->last_answer = NULL;
2706 }
Benny Prijonoccf95622006-02-07 18:48:01 +00002707
2708 /* Send the request. */
Benny Prijonoe8b0d3b2006-03-17 17:57:52 +00002709 status = pjsip_inv_send_msg(inv, tdata);
Benny Prijonoccf95622006-02-07 18:48:01 +00002710 }
2711
Benny Prijono268ca612006-02-07 12:34:11 +00002712 } else {
Benny Prijonoccf95622006-02-07 18:48:01 +00002713
Benny Prijono0b6340c2006-06-13 22:21:23 +00002714 inv_set_cause(inv, tsx->status_code, &tsx->status_text);
Benny Prijono8ad55352006-02-08 11:16:05 +00002715 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
Benny Prijonoccf95622006-02-07 18:48:01 +00002716
Benny Prijono268ca612006-02-07 12:34:11 +00002717 }
2718 break;
2719
2720 case PJSIP_TSX_STATE_TERMINATED:
2721 /* INVITE transaction can be terminated either because UAC
2722 * transaction received 2xx response or because of transport
2723 * error.
2724 */
2725 if (tsx->status_code/100 == 2) {
2726 /* This must be receipt of 2xx response */
2727
2728 /* Set state to CONNECTING */
Benny Prijono8ad55352006-02-08 11:16:05 +00002729 inv_set_state(inv, PJSIP_INV_STATE_CONNECTING, e);
Benny Prijono268ca612006-02-07 12:34:11 +00002730
Benny Prijonoa66c7152006-02-09 01:26:14 +00002731 inv_check_sdp_in_incoming_msg(inv, tsx,
2732 e->body.tsx_state.src.rdata);
2733
Benny Prijono268ca612006-02-07 12:34:11 +00002734 /* Send ACK */
2735 pj_assert(e->body.tsx_state.type == PJSIP_EVENT_RX_MSG);
2736
Benny Prijonod5f9f422007-11-25 04:40:07 +00002737 inv_send_ack(inv, e);
Benny Prijonoa66c7152006-02-09 01:26:14 +00002738
Benny Prijono268ca612006-02-07 12:34:11 +00002739 } else {
Benny Prijono0b6340c2006-06-13 22:21:23 +00002740 inv_set_cause(inv, tsx->status_code, &tsx->status_text);
Benny Prijono8ad55352006-02-08 11:16:05 +00002741 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
Benny Prijono268ca612006-02-07 12:34:11 +00002742 }
2743 break;
2744
Benny Prijono34a404e2006-02-09 14:38:30 +00002745 default:
2746 break;
Benny Prijono268ca612006-02-07 12:34:11 +00002747 }
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00002748
Benny Prijono1f7767b2007-10-03 18:28:49 +00002749 } else if (tsx->role == PJSIP_ROLE_UAC) {
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00002750 /*
Benny Prijono1f7767b2007-10-03 18:28:49 +00002751 * Handle case when outgoing request is answered with 481 (Call/
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00002752 * Transaction Does Not Exist), 408, or when it's timed out. In these
2753 * cases, disconnect session (i.e. dialog usage only).
Benny Prijonoc5145762007-11-23 12:04:40 +00002754 * Note that 481 response to CANCEL does not terminate dialog usage,
2755 * but only the transaction.
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00002756 */
Benny Prijonoc5145762007-11-23 12:04:40 +00002757 if ((tsx->status_code == PJSIP_SC_CALL_TSX_DOES_NOT_EXIST &&
2758 tsx->method.id != PJSIP_CANCEL_METHOD) ||
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00002759 tsx->status_code == PJSIP_SC_REQUEST_TIMEOUT ||
2760 tsx->status_code == PJSIP_SC_TSX_TIMEOUT ||
Benny Prijono0b6340c2006-06-13 22:21:23 +00002761 tsx->status_code == PJSIP_SC_TSX_TRANSPORT_ERROR)
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00002762 {
Benny Prijono0b6340c2006-06-13 22:21:23 +00002763 inv_set_cause(inv, tsx->status_code, &tsx->status_text);
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00002764 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
2765 }
Benny Prijono268ca612006-02-07 12:34:11 +00002766 }
2767}
2768
Benny Prijono8ad55352006-02-08 11:16:05 +00002769/*
2770 * State INCOMING is after we received the request, but before
2771 * responses with tag are sent.
2772 */
2773static void inv_on_state_incoming( pjsip_inv_session *inv, pjsip_event *e)
Benny Prijono268ca612006-02-07 12:34:11 +00002774{
2775 pjsip_transaction *tsx = e->body.tsx_state.tsx;
2776 pjsip_dialog *dlg = pjsip_tsx_get_dlg(tsx);
2777
2778 PJ_ASSERT_ON_FAIL(tsx && dlg, return);
2779
Benny Prijono8ad55352006-02-08 11:16:05 +00002780 if (tsx == inv->invite_tsx) {
2781
2782 /*
2783 * Handle the INVITE state transition.
2784 */
2785
Benny Prijono268ca612006-02-07 12:34:11 +00002786 switch (tsx->state) {
Benny Prijono8ad55352006-02-08 11:16:05 +00002787
Benny Prijono64f851e2006-02-23 13:49:28 +00002788 case PJSIP_TSX_STATE_TRYING:
2789 inv_set_state(inv, PJSIP_INV_STATE_INCOMING, e);
2790 break;
2791
Benny Prijono268ca612006-02-07 12:34:11 +00002792 case PJSIP_TSX_STATE_PROCEEDING:
Benny Prijono8ad55352006-02-08 11:16:05 +00002793 /*
2794 * Transaction sent provisional response.
2795 */
Benny Prijono268ca612006-02-07 12:34:11 +00002796 if (tsx->status_code > 100)
Benny Prijono8ad55352006-02-08 11:16:05 +00002797 inv_set_state(inv, PJSIP_INV_STATE_EARLY, e);
Benny Prijono268ca612006-02-07 12:34:11 +00002798 break;
Benny Prijono8ad55352006-02-08 11:16:05 +00002799
Benny Prijono268ca612006-02-07 12:34:11 +00002800 case PJSIP_TSX_STATE_COMPLETED:
Benny Prijono8ad55352006-02-08 11:16:05 +00002801 /*
2802 * Transaction sent final response.
2803 */
Benny Prijonod4e0abd2006-03-05 11:53:36 +00002804 if (tsx->status_code/100 == 2) {
Benny Prijono8ad55352006-02-08 11:16:05 +00002805 inv_set_state(inv, PJSIP_INV_STATE_CONNECTING, e);
Benny Prijonod4e0abd2006-03-05 11:53:36 +00002806 } else {
Benny Prijono0b6340c2006-06-13 22:21:23 +00002807 inv_set_cause(inv, tsx->status_code, &tsx->status_text);
Benny Prijono8ad55352006-02-08 11:16:05 +00002808 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
Benny Prijonod4e0abd2006-03-05 11:53:36 +00002809 }
Benny Prijono268ca612006-02-07 12:34:11 +00002810 break;
Benny Prijono8ad55352006-02-08 11:16:05 +00002811
Benny Prijono268ca612006-02-07 12:34:11 +00002812 case PJSIP_TSX_STATE_TERMINATED:
Benny Prijono8ad55352006-02-08 11:16:05 +00002813 /*
2814 * This happens on transport error (e.g. failed to send
2815 * response)
2816 */
Benny Prijono0b6340c2006-06-13 22:21:23 +00002817 inv_set_cause(inv, tsx->status_code, &tsx->status_text);
Benny Prijono8ad55352006-02-08 11:16:05 +00002818 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
Benny Prijono268ca612006-02-07 12:34:11 +00002819 break;
Benny Prijono8ad55352006-02-08 11:16:05 +00002820
Benny Prijono268ca612006-02-07 12:34:11 +00002821 default:
Benny Prijono8ad55352006-02-08 11:16:05 +00002822 pj_assert(!"Unexpected INVITE state");
2823 break;
Benny Prijono268ca612006-02-07 12:34:11 +00002824 }
Benny Prijono8ad55352006-02-08 11:16:05 +00002825
2826 } else if (tsx->method.id == PJSIP_CANCEL_METHOD &&
2827 tsx->role == PJSIP_ROLE_UAS &&
2828 tsx->state < PJSIP_TSX_STATE_COMPLETED &&
2829 e->body.tsx_state.type == PJSIP_EVENT_RX_MSG )
2830 {
2831
2832 /*
2833 * Handle incoming CANCEL request.
2834 */
2835
2836 inv_respond_incoming_cancel(inv, tsx, e->body.tsx_state.src.rdata);
2837
Benny Prijono268ca612006-02-07 12:34:11 +00002838 }
2839}
2840
Benny Prijono8ad55352006-02-08 11:16:05 +00002841/*
2842 * State EARLY is for both UAS and UAC, after response with To tag
2843 * is sent/received.
2844 */
2845static void inv_on_state_early( pjsip_inv_session *inv, pjsip_event *e)
Benny Prijono268ca612006-02-07 12:34:11 +00002846{
2847 pjsip_transaction *tsx = e->body.tsx_state.tsx;
2848 pjsip_dialog *dlg = pjsip_tsx_get_dlg(tsx);
2849
2850 PJ_ASSERT_ON_FAIL(tsx && dlg, return);
2851
Benny Prijono8ad55352006-02-08 11:16:05 +00002852 if (tsx == inv->invite_tsx) {
2853
2854 /*
2855 * Handle the INVITE state progress.
2856 */
Benny Prijono268ca612006-02-07 12:34:11 +00002857
2858 switch (tsx->state) {
2859
2860 case PJSIP_TSX_STATE_PROCEEDING:
2861 /* Send/received another provisional response. */
Benny Prijono8ad55352006-02-08 11:16:05 +00002862 inv_set_state(inv, PJSIP_INV_STATE_EARLY, e);
Benny Prijonoa66c7152006-02-09 01:26:14 +00002863
2864 if (e->body.tsx_state.type == PJSIP_EVENT_RX_MSG) {
2865 inv_check_sdp_in_incoming_msg(inv, tsx,
2866 e->body.tsx_state.src.rdata);
Benny Prijono1f7767b2007-10-03 18:28:49 +00002867
2868 if (pjsip_100rel_is_reliable(e->body.tsx_state.src.rdata)) {
2869 inv_handle_incoming_reliable_response(
2870 inv, e->body.tsx_state.src.rdata);
2871 }
Benny Prijonoa66c7152006-02-09 01:26:14 +00002872 }
Benny Prijono268ca612006-02-07 12:34:11 +00002873 break;
2874
2875 case PJSIP_TSX_STATE_COMPLETED:
Benny Prijonoa66c7152006-02-09 01:26:14 +00002876 if (tsx->status_code/100 == 2) {
Benny Prijono8ad55352006-02-08 11:16:05 +00002877 inv_set_state(inv, PJSIP_INV_STATE_CONNECTING, e);
Benny Prijonoa66c7152006-02-09 01:26:14 +00002878 if (e->body.tsx_state.type == PJSIP_EVENT_RX_MSG) {
2879 inv_check_sdp_in_incoming_msg(inv, tsx,
2880 e->body.tsx_state.src.rdata);
2881 }
2882
Benny Prijonod4e0abd2006-03-05 11:53:36 +00002883 } else {
Benny Prijono0b6340c2006-06-13 22:21:23 +00002884 inv_set_cause(inv, tsx->status_code, &tsx->status_text);
Benny Prijono8ad55352006-02-08 11:16:05 +00002885 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
Benny Prijonod4e0abd2006-03-05 11:53:36 +00002886 }
Benny Prijono268ca612006-02-07 12:34:11 +00002887 break;
2888
Benny Prijonof3195072006-02-14 21:15:30 +00002889 case PJSIP_TSX_STATE_CONFIRMED:
Benny Prijono7d910092007-06-20 04:19:46 +00002890 /* For some reason can go here (maybe when ACK for 2xx has
2891 * the same branch value as the INVITE transaction) */
Benny Prijonof3195072006-02-14 21:15:30 +00002892
Benny Prijono268ca612006-02-07 12:34:11 +00002893 case PJSIP_TSX_STATE_TERMINATED:
2894 /* INVITE transaction can be terminated either because UAC
2895 * transaction received 2xx response or because of transport
2896 * error.
2897 */
2898 if (tsx->status_code/100 == 2) {
2899
2900 /* This must be receipt of 2xx response */
2901
2902 /* Set state to CONNECTING */
Benny Prijono8ad55352006-02-08 11:16:05 +00002903 inv_set_state(inv, PJSIP_INV_STATE_CONNECTING, e);
Benny Prijono268ca612006-02-07 12:34:11 +00002904
Benny Prijonoa66c7152006-02-09 01:26:14 +00002905 if (e->body.tsx_state.type == PJSIP_EVENT_RX_MSG) {
2906 inv_check_sdp_in_incoming_msg(inv, tsx,
2907 e->body.tsx_state.src.rdata);
2908 }
2909
Benny Prijono268ca612006-02-07 12:34:11 +00002910 /* if UAC, send ACK and move state to confirmed. */
2911 if (tsx->role == PJSIP_ROLE_UAC) {
2912 pj_assert(e->body.tsx_state.type == PJSIP_EVENT_RX_MSG);
2913
Benny Prijonod5f9f422007-11-25 04:40:07 +00002914 inv_send_ack(inv, e);
Benny Prijono268ca612006-02-07 12:34:11 +00002915 }
2916
2917 } else {
Benny Prijono0b6340c2006-06-13 22:21:23 +00002918 inv_set_cause(inv, tsx->status_code, &tsx->status_text);
Benny Prijono8ad55352006-02-08 11:16:05 +00002919 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
Benny Prijono268ca612006-02-07 12:34:11 +00002920 }
2921 break;
2922
2923 default:
Benny Prijono8ad55352006-02-08 11:16:05 +00002924 pj_assert(!"Unexpected INVITE tsx state");
Benny Prijono268ca612006-02-07 12:34:11 +00002925 }
2926
Benny Prijono8ad55352006-02-08 11:16:05 +00002927 } else if (inv->role == PJSIP_ROLE_UAS &&
2928 tsx->role == PJSIP_ROLE_UAS &&
2929 tsx->method.id == PJSIP_CANCEL_METHOD &&
2930 tsx->state < PJSIP_TSX_STATE_COMPLETED &&
2931 e->body.tsx_state.type == PJSIP_EVENT_RX_MSG )
2932 {
Benny Prijono268ca612006-02-07 12:34:11 +00002933
Benny Prijono8ad55352006-02-08 11:16:05 +00002934 /*
2935 * Handle incoming CANCEL request.
Benny Prijonoccf95622006-02-07 18:48:01 +00002936 */
2937
Benny Prijono8ad55352006-02-08 11:16:05 +00002938 inv_respond_incoming_cancel(inv, tsx, e->body.tsx_state.src.rdata);
2939
Benny Prijono1f7767b2007-10-03 18:28:49 +00002940 } else if (tsx->role == PJSIP_ROLE_UAS &&
2941 tsx->state == PJSIP_TSX_STATE_TRYING &&
2942 pjsip_method_cmp(&tsx->method, &pjsip_update_method)==0)
Benny Prijonoe4f2abb2006-02-10 14:04:05 +00002943 {
2944 /*
Benny Prijono1f7767b2007-10-03 18:28:49 +00002945 * Handle incoming UPDATE
2946 */
2947 inv_respond_incoming_update(inv, e->body.tsx_state.src.rdata);
2948
2949
2950 } else if (tsx->role == PJSIP_ROLE_UAC &&
2951 (tsx->state == PJSIP_TSX_STATE_COMPLETED ||
2952 tsx->state == PJSIP_TSX_STATE_TERMINATED) &&
2953 pjsip_method_cmp(&tsx->method, &pjsip_update_method)==0)
2954 {
2955 /*
2956 * Handle response to outgoing UPDATE request.
2957 */
2958 inv_handle_update_response(inv, e);
2959
2960 } else if (tsx->role == PJSIP_ROLE_UAS &&
2961 tsx->state == PJSIP_TSX_STATE_TRYING &&
2962 pjsip_method_cmp(&tsx->method, &pjsip_prack_method)==0)
2963 {
2964 /*
2965 * Handle incoming PRACK
2966 */
2967 inv_respond_incoming_prack(inv, e->body.tsx_state.src.rdata);
2968
2969 } else if (tsx->role == PJSIP_ROLE_UAC) {
Benny Prijono0b4c57b2008-06-25 10:15:01 +00002970
2971 /* Generic handling for UAC tsx completion */
2972 handle_uac_tsx_response(inv, e);
Benny Prijono268ca612006-02-07 12:34:11 +00002973 }
2974}
2975
Benny Prijono8ad55352006-02-08 11:16:05 +00002976/*
2977 * State CONNECTING is after 2xx response to INVITE is sent/received.
2978 */
2979static void inv_on_state_connecting( pjsip_inv_session *inv, pjsip_event *e)
Benny Prijono268ca612006-02-07 12:34:11 +00002980{
2981 pjsip_transaction *tsx = e->body.tsx_state.tsx;
2982 pjsip_dialog *dlg = pjsip_tsx_get_dlg(tsx);
2983
2984 PJ_ASSERT_ON_FAIL(tsx && dlg, return);
2985
Benny Prijono8ad55352006-02-08 11:16:05 +00002986 if (tsx == inv->invite_tsx) {
Benny Prijono268ca612006-02-07 12:34:11 +00002987
Benny Prijono8ad55352006-02-08 11:16:05 +00002988 /*
2989 * Handle INVITE state progression.
2990 */
Benny Prijono268ca612006-02-07 12:34:11 +00002991 switch (tsx->state) {
2992
2993 case PJSIP_TSX_STATE_CONFIRMED:
Benny Prijono7d910092007-06-20 04:19:46 +00002994 /* It can only go here if incoming ACK request has the same Via
2995 * branch parameter as the INVITE transaction.
2996 */
2997 if (tsx->status_code/100 == 2) {
2998 if (e->body.tsx_state.type == PJSIP_EVENT_RX_MSG) {
2999 inv_check_sdp_in_incoming_msg(inv, tsx,
3000 e->body.tsx_state.src.rdata);
3001 }
3002
Benny Prijono38998232006-02-08 22:44:25 +00003003 inv_set_state(inv, PJSIP_INV_STATE_CONFIRMED, e);
Benny Prijono7d910092007-06-20 04:19:46 +00003004 }
Benny Prijono268ca612006-02-07 12:34:11 +00003005 break;
3006
3007 case PJSIP_TSX_STATE_TERMINATED:
3008 /* INVITE transaction can be terminated either because UAC
3009 * transaction received 2xx response or because of transport
3010 * error.
3011 */
3012 if (tsx->status_code/100 != 2) {
Benny Prijono0b6340c2006-06-13 22:21:23 +00003013 inv_set_cause(inv, tsx->status_code, &tsx->status_text);
Benny Prijono8ad55352006-02-08 11:16:05 +00003014 inv_set_state(inv, PJSIP_INV_STATE_DISCONNECTED, e);
Benny Prijono268ca612006-02-07 12:34:11 +00003015 }
3016 break;
3017
3018 case PJSIP_TSX_STATE_DESTROYED:
3019 /* Do nothing. */
3020 break;
3021
3022 default:
3023 pj_assert(!"Unexpected state");
3024 }
3025
Benny Prijono8ad55352006-02-08 11:16:05 +00003026 } else if (tsx->role == PJSIP_ROLE_UAS &&
3027 tsx->method.id == PJSIP_BYE_METHOD &&
3028 tsx->status_code < 200 &&
3029 e->body.tsx_state.type == PJSIP_EVENT_RX_MSG)
3030 {
3031
3032 /*
3033 * Handle incoming BYE.
3034 */
3035
3036 inv_respond_incoming_bye( inv, tsx, e->body.tsx_state.src.rdata, e );
3037
Benny Prijono38998232006-02-08 22:44:25 +00003038 } else if (tsx->method.id == PJSIP_BYE_METHOD &&
3039 tsx->role == PJSIP_ROLE_UAC &&
Benny Prijonoa66c7152006-02-09 01:26:14 +00003040 (tsx->state == PJSIP_TSX_STATE_COMPLETED ||
3041 tsx->state == PJSIP_TSX_STATE_TERMINATED))
Benny Prijono38998232006-02-08 22:44:25 +00003042 {
3043
3044 /*
3045 * Outgoing BYE
3046 */
3047 inv_handle_bye_response( inv, tsx, e->body.tsx_state.src.rdata, e);
3048
Benny Prijono268ca612006-02-07 12:34:11 +00003049 }
Benny Prijono70127222006-07-02 14:53:05 +00003050 else if (tsx->method.id == PJSIP_CANCEL_METHOD &&
3051 tsx->role == PJSIP_ROLE_UAS &&
3052 tsx->status_code < 200 &&
3053 e->body.tsx_state.type == PJSIP_EVENT_RX_MSG)
3054 {
Benny Prijono38998232006-02-08 22:44:25 +00003055
Benny Prijono70127222006-07-02 14:53:05 +00003056 /*
3057 * Handle strandled incoming CANCEL.
3058 */
3059 pjsip_rx_data *rdata = e->body.tsx_state.src.rdata;
3060 pjsip_tx_data *tdata;
3061 pj_status_t status;
3062
3063 status = pjsip_dlg_create_response(dlg, rdata, 200, NULL, &tdata);
3064 if (status != PJ_SUCCESS) return;
3065
3066 status = pjsip_dlg_send_response(dlg, tsx, tdata);
3067 if (status != PJ_SUCCESS) return;
3068
Benny Prijono1f7767b2007-10-03 18:28:49 +00003069 } else if (tsx->role == PJSIP_ROLE_UAS &&
3070 tsx->state == PJSIP_TSX_STATE_TRYING &&
3071 pjsip_method_cmp(&tsx->method, &pjsip_update_method)==0)
3072 {
3073 /*
3074 * Handle incoming UPDATE
3075 */
3076 inv_respond_incoming_update(inv, e->body.tsx_state.src.rdata);
3077
3078
3079 } else if (tsx->role == PJSIP_ROLE_UAC &&
3080 (tsx->state == PJSIP_TSX_STATE_COMPLETED ||
3081 tsx->state == PJSIP_TSX_STATE_TERMINATED) &&
3082 pjsip_method_cmp(&tsx->method, &pjsip_update_method)==0)
3083 {
3084 /*
3085 * Handle response to outgoing UPDATE request.
3086 */
Benny Prijono0b4c57b2008-06-25 10:15:01 +00003087 if (handle_uac_tsx_response(inv, e) == PJ_FALSE)
Benny Prijono87402382008-02-21 19:28:21 +00003088 inv_handle_update_response(inv, e);
Benny Prijono1f7767b2007-10-03 18:28:49 +00003089
3090 } else if (tsx->role == PJSIP_ROLE_UAS &&
3091 tsx->state == PJSIP_TSX_STATE_TRYING &&
3092 pjsip_method_cmp(&tsx->method, &pjsip_prack_method)==0)
3093 {
3094 /*
3095 * Handle incoming PRACK
3096 */
3097 inv_respond_incoming_prack(inv, e->body.tsx_state.src.rdata);
3098
3099 } else if (tsx->role == PJSIP_ROLE_UAC) {
Benny Prijono87402382008-02-21 19:28:21 +00003100
Benny Prijono0b4c57b2008-06-25 10:15:01 +00003101 /* Generic handling for UAC tsx completion */
3102 handle_uac_tsx_response(inv, e);
Benny Prijono70127222006-07-02 14:53:05 +00003103 }
Benny Prijono1f7767b2007-10-03 18:28:49 +00003104
Benny Prijono268ca612006-02-07 12:34:11 +00003105}
3106
Benny Prijono8ad55352006-02-08 11:16:05 +00003107/*
3108 * State CONFIRMED is after ACK is sent/received.
3109 */
3110static void inv_on_state_confirmed( pjsip_inv_session *inv, pjsip_event *e)
Benny Prijono268ca612006-02-07 12:34:11 +00003111{
3112 pjsip_transaction *tsx = e->body.tsx_state.tsx;
3113 pjsip_dialog *dlg = pjsip_tsx_get_dlg(tsx);
3114
3115 PJ_ASSERT_ON_FAIL(tsx && dlg, return);
3116
Benny Prijono268ca612006-02-07 12:34:11 +00003117
Benny Prijono8ad55352006-02-08 11:16:05 +00003118 if (tsx->method.id == PJSIP_BYE_METHOD &&
3119 tsx->role == PJSIP_ROLE_UAC &&
Benny Prijonoa66c7152006-02-09 01:26:14 +00003120 (tsx->state == PJSIP_TSX_STATE_COMPLETED ||
3121 tsx->state == PJSIP_TSX_STATE_TERMINATED))
Benny Prijono8ad55352006-02-08 11:16:05 +00003122 {
Benny Prijono38998232006-02-08 22:44:25 +00003123
Benny Prijono8ad55352006-02-08 11:16:05 +00003124 /*
Benny Prijono38998232006-02-08 22:44:25 +00003125 * Outgoing BYE
Benny Prijono8ad55352006-02-08 11:16:05 +00003126 */
Benny Prijono8ad55352006-02-08 11:16:05 +00003127
Benny Prijonoa66c7152006-02-09 01:26:14 +00003128 inv_handle_bye_response( inv, tsx, e->body.tsx_state.src.rdata, e);
Benny Prijono268ca612006-02-07 12:34:11 +00003129
Benny Prijono8ad55352006-02-08 11:16:05 +00003130 }
3131 else if (tsx->method.id == PJSIP_BYE_METHOD &&
3132 tsx->role == PJSIP_ROLE_UAS &&
3133 tsx->status_code < 200 &&
3134 e->body.tsx_state.type == PJSIP_EVENT_RX_MSG)
3135 {
Benny Prijonoccf95622006-02-07 18:48:01 +00003136
Benny Prijono8ad55352006-02-08 11:16:05 +00003137 /*
3138 * Handle incoming BYE.
3139 */
Benny Prijono268ca612006-02-07 12:34:11 +00003140
Benny Prijono8ad55352006-02-08 11:16:05 +00003141 inv_respond_incoming_bye( inv, tsx, e->body.tsx_state.src.rdata, e );
3142
Benny Prijono268ca612006-02-07 12:34:11 +00003143 }
Benny Prijono70127222006-07-02 14:53:05 +00003144 else if (tsx->method.id == PJSIP_CANCEL_METHOD &&
3145 tsx->role == PJSIP_ROLE_UAS &&
3146 tsx->status_code < 200 &&
3147 e->body.tsx_state.type == PJSIP_EVENT_RX_MSG)
3148 {
3149
3150 /*
3151 * Handle strandled incoming CANCEL.
3152 */
3153 pjsip_rx_data *rdata = e->body.tsx_state.src.rdata;
3154 pjsip_tx_data *tdata;
3155 pj_status_t status;
3156
3157 status = pjsip_dlg_create_response(dlg, rdata, 200, NULL, &tdata);
3158 if (status != PJ_SUCCESS) return;
3159
3160 status = pjsip_dlg_send_response(dlg, tsx, tdata);
3161 if (status != PJ_SUCCESS) return;
3162
3163 }
Benny Prijono26ff9062006-02-21 23:47:00 +00003164 else if (tsx->method.id == PJSIP_INVITE_METHOD &&
3165 tsx->role == PJSIP_ROLE_UAS)
3166 {
3167
3168 /*
3169 * Handle incoming re-INVITE
3170 */
3171 if (tsx->state == PJSIP_TSX_STATE_TRYING) {
3172
3173 pjsip_rx_data *rdata = e->body.tsx_state.src.rdata;
3174 pjsip_tx_data *tdata;
3175 pj_status_t status;
3176
3177 /* Check if we have INVITE pending. */
3178 if (inv->invite_tsx && inv->invite_tsx!=tsx) {
Benny Prijono46249942007-02-19 22:23:14 +00003179 pj_str_t reason;
3180
3181 reason = pj_str("Another INVITE transaction in progress");
Benny Prijono26ff9062006-02-21 23:47:00 +00003182
3183 /* Can not receive re-INVITE while another one is pending. */
Benny Prijono46249942007-02-19 22:23:14 +00003184 status = pjsip_dlg_create_response( inv->dlg, rdata, 500,
3185 &reason, &tdata);
Benny Prijono26ff9062006-02-21 23:47:00 +00003186 if (status != PJ_SUCCESS)
3187 return;
3188
3189 status = pjsip_dlg_send_response( inv->dlg, tsx, tdata);
3190
3191
3192 return;
3193 }
3194
3195 /* Save the invite transaction. */
3196 inv->invite_tsx = tsx;
3197
3198 /* Process SDP in incoming message. */
3199 status = inv_check_sdp_in_incoming_msg(inv, tsx, rdata);
3200
3201 if (status != PJ_SUCCESS) {
3202
3203 /* Not Acceptable */
3204 const pjsip_hdr *accept;
3205
3206 status = pjsip_dlg_create_response(inv->dlg, rdata,
3207 488, NULL, &tdata);
3208 if (status != PJ_SUCCESS)
3209 return;
3210
3211
3212 accept = pjsip_endpt_get_capability(dlg->endpt, PJSIP_H_ACCEPT,
3213 NULL);
3214 if (accept) {
Benny Prijonoa1e69682007-05-11 15:14:34 +00003215 pjsip_msg_add_hdr(tdata->msg, (pjsip_hdr*)
Benny Prijono26ff9062006-02-21 23:47:00 +00003216 pjsip_hdr_clone(tdata->pool, accept));
3217 }
3218
3219 status = pjsip_dlg_send_response(dlg, tsx, tdata);
3220
3221 return;
3222 }
3223
3224 /* Create 2xx ANSWER */
3225 status = pjsip_dlg_create_response(dlg, rdata, 200, NULL, &tdata);
3226 if (status != PJ_SUCCESS)
3227 return;
3228
Benny Prijono7d910092007-06-20 04:19:46 +00003229 /* If the INVITE request has SDP body, send answer.
3230 * Otherwise generate offer from local active SDP.
3231 */
3232 if (rdata->msg_info.msg->body != NULL) {
3233 status = process_answer(inv, 200, tdata, NULL);
3234 } else {
Benny Prijono77998ce2007-06-20 10:03:46 +00003235 /* INVITE does not have SDP.
3236 * If on_create_offer() callback is implemented, ask app.
3237 * to generate an offer, otherwise just send active local
3238 * SDP to signal that nothing gets modified.
3239 */
3240 pjmedia_sdp_session *sdp = NULL;
3241
3242 if (mod_inv.cb.on_create_offer) {
3243 (*mod_inv.cb.on_create_offer)(inv, &sdp);
3244 if (sdp) {
3245 status = pjmedia_sdp_neg_modify_local_offer(dlg->pool,
3246 inv->neg,
3247 sdp);
3248 }
3249 }
3250
3251 if (sdp == NULL) {
3252 const pjmedia_sdp_session *active_sdp = NULL;
3253 status = pjmedia_sdp_neg_send_local_offer(dlg->pool,
3254 inv->neg,
3255 &active_sdp);
3256 if (status == PJ_SUCCESS)
3257 sdp = (pjmedia_sdp_session*) active_sdp;
3258 }
3259
3260 if (sdp) {
3261 tdata->msg->body = create_sdp_body(tdata->pool, sdp);
Benny Prijono7d910092007-06-20 04:19:46 +00003262 }
3263 }
Benny Prijono26ff9062006-02-21 23:47:00 +00003264
Benny Prijono1d9b9a42006-09-25 13:40:12 +00003265 if (status != PJ_SUCCESS) {
3266 /*
3267 * SDP negotiation has failed.
3268 */
3269 pj_status_t rc;
3270 pj_str_t reason;
3271
3272 /* Delete the 2xx answer */
3273 pjsip_tx_data_dec_ref(tdata);
3274
3275 /* Create 500 response */
3276 reason = pj_str("SDP negotiation failed");
3277 rc = pjsip_dlg_create_response(dlg, rdata, 500, &reason,
3278 &tdata);
3279 if (rc == PJ_SUCCESS) {
3280 pjsip_warning_hdr *w;
3281 const pj_str_t *endpt_name;
3282
3283 endpt_name = pjsip_endpt_name(dlg->endpt);
3284 w = pjsip_warning_hdr_create_from_status(tdata->pool,
3285 endpt_name,
3286 status);
3287 if (w)
3288 pjsip_msg_add_hdr(tdata->msg, (pjsip_hdr*)w);
3289
3290 pjsip_inv_send_msg(inv, tdata);
3291 }
3292 return;
3293 }
3294
3295 /* Send 2xx regardless of the status of negotiation */
Benny Prijonoe8b0d3b2006-03-17 17:57:52 +00003296 status = pjsip_inv_send_msg(inv, tdata);
Benny Prijono26ff9062006-02-21 23:47:00 +00003297
Benny Prijono7d910092007-06-20 04:19:46 +00003298 } else if (tsx->state == PJSIP_TSX_STATE_CONFIRMED) {
3299 /* This is the case where ACK has the same branch as
3300 * the INVITE request.
3301 */
3302 if (tsx->status_code/100 == 2 &&
3303 e->body.tsx_state.type == PJSIP_EVENT_RX_MSG)
3304 {
3305 inv_check_sdp_in_incoming_msg(inv, tsx,
3306 e->body.tsx_state.src.rdata);
3307 }
3308
Benny Prijono26ff9062006-02-21 23:47:00 +00003309 }
3310
3311 }
3312 else if (tsx->method.id == PJSIP_INVITE_METHOD &&
3313 tsx->role == PJSIP_ROLE_UAC)
3314 {
Benny Prijono22e48c92008-03-20 14:40:50 +00003315
Benny Prijono26ff9062006-02-21 23:47:00 +00003316 /*
3317 * Handle outgoing re-INVITE
3318 */
Benny Prijono22e48c92008-03-20 14:40:50 +00003319 if (tsx->state == PJSIP_TSX_STATE_CALLING) {
3320
Benny Prijono61fc5e62008-06-25 18:35:31 +00003321 /* Must not have other pending INVITE transaction */
3322 pj_assert(inv->invite_tsx==NULL || tsx==inv->invite_tsx);
3323
Benny Prijono22e48c92008-03-20 14:40:50 +00003324 /* Save pending invite transaction */
3325 inv->invite_tsx = tsx;
3326
3327 } else if (tsx->state == PJSIP_TSX_STATE_TERMINATED &&
3328 tsx->status_code/100 == 2)
Benny Prijono26ff9062006-02-21 23:47:00 +00003329 {
3330
3331 /* Re-INVITE was accepted. */
3332
3333 /* Process SDP */
3334 inv_check_sdp_in_incoming_msg(inv, tsx,
3335 e->body.tsx_state.src.rdata);
3336
3337 /* Send ACK */
Benny Prijonod5f9f422007-11-25 04:40:07 +00003338 inv_send_ack(inv, e);
Benny Prijono26ff9062006-02-21 23:47:00 +00003339
Benny Prijono0b4c57b2008-06-25 10:15:01 +00003340 } else if (handle_uac_tsx_response(inv, e)) {
Benny Prijono87402382008-02-21 19:28:21 +00003341
3342 /* Handle response that terminates dialog */
3343 /* Nothing to do (already handled) */
3344
Benny Prijono77998ce2007-06-20 10:03:46 +00003345 } else if (tsx->status_code >= 300 && tsx->status_code < 700) {
3346
3347 pjmedia_sdp_neg_state neg_state;
3348
3349 /* Outgoing INVITE transaction has failed, cancel SDP nego */
3350 neg_state = pjmedia_sdp_neg_get_state(inv->neg);
3351 if (neg_state == PJMEDIA_SDP_NEG_STATE_LOCAL_OFFER) {
3352 pjmedia_sdp_neg_cancel_offer(inv->neg);
3353 }
Benny Prijono26ff9062006-02-21 23:47:00 +00003354 }
Benny Prijono1f7767b2007-10-03 18:28:49 +00003355
3356 } else if (tsx->role == PJSIP_ROLE_UAS &&
3357 tsx->state == PJSIP_TSX_STATE_TRYING &&
3358 pjsip_method_cmp(&tsx->method, &pjsip_update_method)==0)
3359 {
3360 /*
3361 * Handle incoming UPDATE
3362 */
3363 inv_respond_incoming_update(inv, e->body.tsx_state.src.rdata);
3364
3365 } else if (tsx->role == PJSIP_ROLE_UAC &&
3366 (tsx->state == PJSIP_TSX_STATE_COMPLETED ||
3367 tsx->state == PJSIP_TSX_STATE_TERMINATED) &&
3368 pjsip_method_cmp(&tsx->method, &pjsip_update_method)==0)
3369 {
3370 /*
3371 * Handle response to outgoing UPDATE request.
3372 */
Benny Prijono0b4c57b2008-06-25 10:15:01 +00003373 if (handle_uac_tsx_response(inv, e) == PJ_FALSE)
Benny Prijono87402382008-02-21 19:28:21 +00003374 inv_handle_update_response(inv, e);
Benny Prijono1f7767b2007-10-03 18:28:49 +00003375
3376 } else if (tsx->role == PJSIP_ROLE_UAS &&
3377 tsx->state == PJSIP_TSX_STATE_TRYING &&
3378 pjsip_method_cmp(&tsx->method, &pjsip_prack_method)==0)
3379 {
3380 /*
3381 * Handle strandled incoming PRACK
3382 */
3383 inv_respond_incoming_prack(inv, e->body.tsx_state.src.rdata);
3384
3385 } else if (tsx->role == PJSIP_ROLE_UAC) {
3386 /*
Benny Prijono0b4c57b2008-06-25 10:15:01 +00003387 * Handle 401/407/408/481 response
Benny Prijono1f7767b2007-10-03 18:28:49 +00003388 */
Benny Prijono0b4c57b2008-06-25 10:15:01 +00003389 handle_uac_tsx_response(inv, e);
Benny Prijono26ff9062006-02-21 23:47:00 +00003390 }
Benny Prijono1f7767b2007-10-03 18:28:49 +00003391
Benny Prijono268ca612006-02-07 12:34:11 +00003392}
3393
Benny Prijono8ad55352006-02-08 11:16:05 +00003394/*
3395 * After session has been terminated, but before dialog is destroyed
3396 * (because dialog has other usages, or because dialog is waiting for
3397 * the last transaction to terminate).
3398 */
3399static void inv_on_state_disconnected( pjsip_inv_session *inv, pjsip_event *e)
Benny Prijono268ca612006-02-07 12:34:11 +00003400{
Benny Prijono8ad55352006-02-08 11:16:05 +00003401 pjsip_transaction *tsx = e->body.tsx_state.tsx;
3402 pjsip_dialog *dlg = pjsip_tsx_get_dlg(tsx);
Benny Prijono268ca612006-02-07 12:34:11 +00003403
Benny Prijono8ad55352006-02-08 11:16:05 +00003404 PJ_ASSERT_ON_FAIL(tsx && dlg, return);
3405
Benny Prijono70127222006-07-02 14:53:05 +00003406 if (tsx->role == PJSIP_ROLE_UAS &&
Benny Prijono8ad55352006-02-08 11:16:05 +00003407 tsx->status_code < 200 &&
3408 e->body.tsx_state.type == PJSIP_EVENT_RX_MSG)
3409 {
Benny Prijono70127222006-07-02 14:53:05 +00003410 pjsip_rx_data *rdata = e->body.tsx_state.src.rdata;
Benny Prijono8ad55352006-02-08 11:16:05 +00003411
3412 /*
Benny Prijono70127222006-07-02 14:53:05 +00003413 * Respond BYE with 200/OK
Benny Prijono8ad55352006-02-08 11:16:05 +00003414 */
Benny Prijono70127222006-07-02 14:53:05 +00003415 if (tsx->method.id == PJSIP_BYE_METHOD) {
3416 inv_respond_incoming_bye( inv, tsx, rdata, e );
3417 } else if (tsx->method.id == PJSIP_CANCEL_METHOD) {
3418 /*
3419 * Respond CANCEL with 200/OK too.
3420 */
3421 pjsip_tx_data *tdata;
3422 pj_status_t status;
Benny Prijono8ad55352006-02-08 11:16:05 +00003423
Benny Prijono70127222006-07-02 14:53:05 +00003424 status = pjsip_dlg_create_response(dlg, rdata, 200, NULL, &tdata);
3425 if (status != PJ_SUCCESS) return;
Benny Prijono8ad55352006-02-08 11:16:05 +00003426
Benny Prijono70127222006-07-02 14:53:05 +00003427 status = pjsip_dlg_send_response(dlg, tsx, tdata);
3428 if (status != PJ_SUCCESS) return;
3429
3430 }
Benny Prijono61fc5e62008-06-25 18:35:31 +00003431
3432 } else if (tsx->role == PJSIP_ROLE_UAC) {
3433 /*
3434 * Handle 401/407/408/481 response
3435 */
3436 handle_uac_tsx_response(inv, e);
Benny Prijono8ad55352006-02-08 11:16:05 +00003437 }
Benny Prijono268ca612006-02-07 12:34:11 +00003438}
3439