comparison gsm-fw/g23m-gsm/cc/cc_act.c @ 673:2f7df7a314f8

gsm-fw/g23m-gsm subtree: initial import from LoCosto source
author Michael Spacefalcon <msokolov@ivan.Harhan.ORG>
date Sun, 28 Sep 2014 23:20:04 +0000
parents
children 75488f27acdd
comparison
equal deleted inserted replaced
672:0dc6f9e8e980 673:2f7df7a314f8
1 /*
2 +-----------------------------------------------------------------------------
3 | Project : GSM-PS (6147)
4 | Modul : CC_ACT
5 +-----------------------------------------------------------------------------
6 | Copyright 2002 Texas Instruments Berlin, AG
7 | All rights reserved.
8 |
9 | This file is confidential and a trade secret of Texas
10 | Instruments Berlin, AG
11 | The receipt of or possession of this file does not convey
12 | any rights to reproduce or disclose its contents or to
13 | manufacture, use, or sell anything it may describe, in
14 | whole, or in part, without the specific written consent of
15 | Texas Instruments Berlin, AG.
16 +-----------------------------------------------------------------------------
17 | Purpose : This Modul defines the primitive processing functions
18 | of the SDL process Call Control for the call active
19 | phase of the component CC of the mobile station
20 +-----------------------------------------------------------------------------
21 */
22
23 #ifndef CC_ACT_C
24 #define CC_ACT_C
25
26 #define ENTITY_CC
27 /*==== INCLUDES ===================================================*/
28
29 #include <string.h>
30 #include "typedefs.h"
31 #include "pcm.h"
32 #include "vsi.h"
33 #include "custom.h"
34 #include "gsm.h"
35 #include "message.h"
36 #include "ccdapi.h"
37 #include "prim.h"
38 #include "cnf_cc.h"
39 #include "mon_cc.h"
40 #include "pei.h"
41 #include "tok.h"
42 #include "cc.h"
43 #include "cc_em.h"
44
45 /*==== EXPORT =====================================================*/
46
47 /*==== PROTOTYPES =================================================*/
48 LOCAL void cc_mncc_hold_retrieve_req (T_PRIM * prim);
49
50 /*==== PRIVAT =====================================================*/
51
52 /*==== VARIABLES ==================================================*/
53
54 /*==== FUNCTIONS ==================================================*/
55
56 /*
57 +--------------------------------------------------------------------+
58 | PROJECT : GSM-PS (6147) MODULE : CC_ACT |
59 | STATE : code ROUTINE : cc_dtmf_full |
60 +--------------------------------------------------------------------+
61
62 PURPOSE : Check whether DTMF buffer is full
63
64 */
65
66 LOCAL BOOL cc_dtmf_full (const T_DTMF *p_dtmf)
67 {
68 TRACE_FUNCTION ("cc_dtmf_full()");
69
70 return ((p_dtmf->read EQ p_dtmf->write + 1) OR
71 (p_dtmf->read EQ 0 AND p_dtmf->write EQ DTMF_BUF_SIZE - 1));
72 }
73
74
75 /*
76 +--------------------------------------------------------------------+
77 | PROJECT : GSM-PS (6147) MODULE : CC_ACT |
78 | STATE : code ROUTINE : cc_dtmf_empty |
79 +--------------------------------------------------------------------+
80
81 PURPOSE : Check whether DTMF buffer is empty
82
83 */
84
85 LOCAL BOOL cc_dtmf_empty (const T_DTMF *p_dtmf)
86 {
87 TRACE_FUNCTION ("cc_dtmf_empty()");
88
89 return (p_dtmf->read EQ p_dtmf->write);
90 }
91
92
93 /*
94 +--------------------------------------------------------------------+
95 | PROJECT : GSM-PS (6147) MODULE : CC_ACT |
96 | STATE : code ROUTINE : cc_dtmf_write |
97 +--------------------------------------------------------------------+
98
99 PURPOSE : Write next entry into DTMF buffer.
100 No checks are done (e.g. buffer full)
101
102 */
103
104 LOCAL void cc_dtmf_write (T_DTMF *p_dtmf, UBYTE val)
105 {
106 TRACE_FUNCTION ("cc_dtmf_write()");
107
108 p_dtmf->buf[p_dtmf->write++] = val;
109 if (p_dtmf->write EQ DTMF_BUF_SIZE)
110 p_dtmf->write = 0;
111 }
112
113 /*
114 +--------------------------------------------------------------------+
115 | PROJECT : GSM-PS (6147) MODULE : CC_ACT |
116 | STATE : code ROUTINE : cc_dtmf_write |
117 +--------------------------------------------------------------------+
118
119 PURPOSE : Read next entry from DTMF buffer.
120 No checks are done (e.g. buffer empty)
121
122 */
123
124 LOCAL UBYTE cc_dtmf_read (T_DTMF *p_dtmf)
125 {
126 UBYTE val;
127
128 TRACE_FUNCTION ("cc_dtmf_read()");
129
130 val = p_dtmf->buf[p_dtmf->read++];
131 if (p_dtmf->read EQ DTMF_BUF_SIZE)
132 p_dtmf->read = 0;
133 return val;
134 }
135
136
137 /*
138 +--------------------------------------------------------------------+
139 | PROJECT : GSM-PS (6147) MODULE : CC_ACT |
140 | STATE : code ROUTINE : cc_dtmf_allowed |
141 +--------------------------------------------------------------------+
142
143 PURPOSE : This functions returns TRUE if DTMF is allowed in the
144 current state, otherwise FALSE.
145
146 */
147
148 LOCAL BOOL cc_dtmf_allowed (void)
149 {
150 GET_INSTANCE_DATA;
151 TRACE_FUNCTION ("cc_dtmf_allowed()");
152
153 /*
154 * "The mobile station shall be capable of transmitting DTMF messages if
155 * and only if the mobile station has the user connection for speech
156 * attached and an appropriate channel is available.
157 *
158 * ".."
159 *
160 * NOTE 1: This specification means that DTMF messages can generally be
161 * sent in the active state of a call in speech transmission
162 * mode or when a traffic channel is available during setup or
163 * release and the progress indicator IE has been received."
164 *
165 * [TS 24.008 subclause 5.5.7]
166 *
167 * As of GSM 04.08 the progress indicator IE can be received by
168 * CALL PROCEEDING, PROGRESS, CONNECT, SETUP, ALERTING and DISCONNECT.
169 */
170 switch (cc_data->state[cc_data->index_ti])
171 {
172 case M_CC_CS_3: /* mobile originating call proceeding */
173 case M_CC_CS_4: /* call delivered */
174 case M_CC_CS_8: /* connect request */
175 case M_CC_CS_11: /* disconnect request */
176 case M_CC_CS_12: /* disconnect indication */
177 case M_CC_CS_19: /* release request */
178 case CS_101: /* substate of CS_10, reestablish requested */
179 case M_CC_CS_26: /* mobile originating modify */
180 case CS_261: /* substate of CS_26, reestablish requested */
181 if (cc_data->progress_desc[cc_data->index_ti] EQ NOT_PRESENT_8BIT)
182 return FALSE; /* Progress indicator IE has not been received */
183 /*FALLTHROUGH*/ /*lint -fallthrough*/
184 case M_CC_CS_10: /* active */
185 if (cc_data->channel_type EQ NAS_CH_SDCCH)
186 return FALSE; /* No TCH assigned */
187
188 if ((cc_data->channel_mode NEQ NAS_CHM_SPEECH) AND
189 (cc_data->channel_mode NEQ NAS_CHM_SPEECH_V2) AND
190 (cc_data->channel_mode NEQ NAS_CHM_SPEECH_V3))
191 return FALSE; /* Channel mode is not speech */
192
193 return TRUE;
194
195 default:
196 return FALSE;
197 }
198 }
199
200 /*==== VARIABLES ==================================================*/
201
202 /*==== FUNCTIONS ==================================================*/
203
204 /*
205 +--------------------------------------------------------------------+
206 | PROJECT : GSM-PS (6147) MODULE : CC_ACT |
207 | STATE : code ROUTINE : cc_mncc_start_dtmf_req |
208 +--------------------------------------------------------------------+
209
210 PURPOSE : Sending of DTMF tones to the infrastructure side.
211
212 */
213
214 GLOBAL void cc_mncc_start_dtmf_req (T_MNCC_START_DTMF_REQ * dtmf)
215 {
216 GET_INSTANCE_DATA;
217 T_DTMF * p_dtmf;
218
219 TRACE_FUNCTION ("cc_mncc_start_dtmf_req()");
220
221
222 if ((cc_data->index_ti = srv_convert_ti (dtmf->ti))
223 EQ NOT_PRESENT_8BIT)
224 {
225 PFREE (dtmf);
226 return;
227 }
228
229 switch (cc_data->state[cc_data->index_ti])
230 {
231 default:
232 if (!cc_dtmf_allowed ())
233 {
234 /* Tone dropped. Confirm DTMF request with negative cause */
235 PALLOC (dtmf_cnf, MNCC_START_DTMF_CNF);
236 dtmf_cnf->ti = dtmf->ti;
237 dtmf_cnf->key = dtmf->key;
238 dtmf_cnf->cause = MNCC_CAUSE_DTMF_NOT_ALLOWED;
239 dtmf_cnf->dtmf_mod = dtmf->dtmf_mod;
240 PSENDX (MMI, dtmf_cnf);
241
242 PFREE (dtmf);
243 return;
244 }
245
246 p_dtmf = &cc_data->dtmf [cc_data->index_ti];
247
248 if ( dtmf->dtmf_mod EQ MNCC_DTMF_MOD_AUTO OR
249 dtmf->dtmf_mod EQ MNCC_DTMF_MOD_MAN_START )
250 {
251 if (p_dtmf->state EQ DTMF_IDLE)
252 {
253 /*
254 * If sending of DTMF tones is not in
255 * progress send the tone.
256 */
257 CCD_START;
258 {
259 MCAST (start_dtmf, U_START_DTMF);
260
261 cc_build_start_dtmf (dtmf->key, start_dtmf);
262 for_start_dtmf (start_dtmf);
263 }
264 CCD_END;
265
266 /* On sending a START DTMF message the MS shall start T336 */
267 TIMERSTART (T336, T336_VALUE);
268
269 p_dtmf->state = DTMF_SEND_REQUEST;
270 p_dtmf->mode = (dtmf->dtmf_mod EQ MNCC_DTMF_MOD_AUTO)?DTMF_AUTO:DTMF_MAN;
271 p_dtmf->key = dtmf->key;
272 }
273 else
274 {
275 /*
276 * Not DTMF_IDLE.
277 */
278
279 if (dtmf->dtmf_mod EQ MNCC_DTMF_MOD_AUTO)
280 {
281 /*
282 * If DTMF_MOD_AUTO, store the tone if the buffer is not full.
283 */
284 if (!cc_dtmf_full (p_dtmf))
285 cc_dtmf_write (p_dtmf, (UBYTE)SET_DTMF_MOD(dtmf->key, DTMF_AUTO));
286 // PATCH HM 13.04.00 >>>
287 else
288 {
289 /* Tone dropped. Confirm DTMF request with negative cause */
290 PALLOC (dtmf_cnf, MNCC_START_DTMF_CNF);
291 dtmf_cnf->ti = dtmf->ti;
292 dtmf_cnf->key = dtmf->key;
293 dtmf_cnf->cause = MNCC_CAUSE_DTMF_BUFFER_FULL;
294 dtmf_cnf->dtmf_mod = dtmf->dtmf_mod;
295 PSENDX(MMI, dtmf_cnf);
296 }
297 // PATCH HM 13.04.00 <<<
298 }
299 else
300 {
301 /*
302 * If DTMF_MOD_MAN_START, two conditions have to be fullfilled
303 * that the tone is stored and not dropped:
304 *
305 * - The buffer is not full.
306 * - There is enough room to store the stop which
307 * will follow the tone.
308 */
309 if (!cc_dtmf_full (p_dtmf) AND
310 (p_dtmf->read NEQ (p_dtmf->write + 2) % DTMF_BUF_SIZE))
311 {
312 cc_dtmf_write (p_dtmf, (UBYTE)SET_DTMF_MOD(dtmf->key, DTMF_MAN));
313 }
314 // PATCH HM 13.04.00 >>>
315 else
316 {
317 /* Tone dropped. Confirm DTMF request with negative cause */
318 PALLOC (dtmf_cnf, MNCC_START_DTMF_CNF);
319 dtmf_cnf->ti = dtmf->ti;
320 dtmf_cnf->key = dtmf->key;
321 dtmf_cnf->cause = MNCC_CAUSE_DTMF_BUFFER_FULL;
322 dtmf_cnf->dtmf_mod = dtmf->dtmf_mod;
323 PSENDX (MMI, dtmf_cnf);
324 }
325 // PATCH HM 13.04.00 <<<
326 }
327 }
328 }
329 else
330 {
331 /*
332 * This is DTMF_MOD_MAN_STOP
333 */
334 if ( p_dtmf->state EQ DTMF_SEND_ACKNOWLEDGE AND
335 cc_dtmf_empty (p_dtmf))
336 {
337 /*
338 * If sending of DTMF tones is in
339 * progress stop the tone.
340 */
341 for_stop_dtmf ();
342
343 /* On sending a STOP DTMF message the MS shall start T337 */
344 TIMERSTART (T337, T337_VALUE);
345
346 p_dtmf->state = DTMF_STOP_REQUEST;
347 }
348 else
349 {
350 /*
351 * If DTMF_MOD_MAN_STOP, the stop mark is stored if
352 * the buffer is not full. If DTMF STOP ACK is received
353 * and there are unexpected stops in the buffer due to
354 * a loss of the respective tones, the stops are dropped
355 * until a tone is found.
356 */
357 if (!cc_dtmf_full(p_dtmf))
358 {
359 cc_dtmf_write (p_dtmf, NOT_PRESENT_8BIT);
360 }
361 }
362 }
363 PFREE (dtmf)
364 break;
365
366 case M_CC_CS_26: /* mobile originating modify */
367 case CS_101:
368 case CS_261:
369 srv_store_prim ((T_PRIM *)D2P(dtmf));
370 break;
371 }
372 }
373
374
375 /*
376 +--------------------------------------------------------------------+
377 | PROJECT : GSM-PS (6147) MODULE : CC_ACT |
378 | STATE : code ROUTINE : cc_reset_dtmf |
379 +--------------------------------------------------------------------+
380
381 PURPOSE : Reset DTMF machine. If used before CC leaves any state in
382 which DTMF is possible MMI is informed about DTMF failure
383 if there was still a DTMF request outstanding.
384
385 */
386
387 GLOBAL void cc_reset_dtmf (void)
388 {
389 GET_INSTANCE_DATA;
390 T_DTMF * p_dtmf;
391 BOOL dtmf_aborted;
392
393 TRACE_FUNCTION ("cc_reset_dtmf");
394
395 p_dtmf = &cc_data->dtmf[cc_data->index_ti];
396
397 if (cc_dtmf_allowed())
398 {
399 /* Stop all possibly running DTMF timers */
400 TIMERSTOP (TIMER_DTMF);
401
402 /*
403 * A DTMF acknowledge maybe outstanding for ACI either if the
404 * DTMF state is DTMF_SEND_REQ or
405 * a tone was found in the DTMF send buffer
406 */
407 dtmf_aborted = (p_dtmf->state EQ DTMF_SEND_REQUEST);
408 while (!dtmf_aborted AND !cc_dtmf_empty(p_dtmf))
409 dtmf_aborted = (cc_dtmf_read (p_dtmf) NEQ NOT_PRESENT_8BIT);
410
411 if (dtmf_aborted)
412 {
413 /* Tone dropped. Confirm DTMF request with negative cause */
414 PALLOC (dtmf_cnf, MNCC_START_DTMF_CNF);
415 dtmf_cnf->ti = cc_data->ti;
416 dtmf_cnf->key = NOT_PRESENT_8BIT;
417 dtmf_cnf->cause = MNCC_CAUSE_DTMF_ABORTED;
418 dtmf_cnf->dtmf_mod = NOT_PRESENT_8BIT;
419 PSENDX (MMI, dtmf_cnf);
420 }
421
422 }
423 p_dtmf->state = DTMF_IDLE;
424 p_dtmf->write = 0;
425 p_dtmf->read = 0;
426 }
427
428
429 /*
430 +--------------------------------------------------------------------+
431 | PROJECT : GSM-PS (6147) MODULE : CC_ACT |
432 | STATE : code ROUTINE : cc_start_dtmf_ack |
433 +--------------------------------------------------------------------+
434
435 PURPOSE : Processing an incoming start dtmf acknowledge message.
436
437 */
438
439 GLOBAL void cc_start_dtmf_ack (T_D_START_DTMF_ACK * start_dtmf_ack)
440 {
441 GET_INSTANCE_DATA;
442 T_DTMF * p_dtmf;
443
444 TRACE_FUNCTION ("cc_start_dtmf_ack()");
445
446 if (cc_dtmf_allowed ())
447 {
448 if (cc_check_error_flag ())
449 {
450 p_dtmf = &cc_data->dtmf[cc_data->index_ti];
451
452 /* Stop T336 */
453 TIMERSTOP (TIMER_DTMF);
454
455 if (p_dtmf->state EQ DTMF_SEND_REQUEST)
456 {
457 PALLOC (start, MNCC_START_DTMF_CNF);
458
459 start->ti = cc_data->ti;
460 start->key = start_dtmf_ack->key_facility.key;
461 start->cause = MNCC_CAUSE_DTMF_START_SUCCESS;
462 start->dtmf_mod = (p_dtmf->mode EQ DTMF_AUTO) ?
463 MNCC_DTMF_MOD_AUTO : MNCC_DTMF_MOD_MAN_START;
464 PSENDX (MMI, start);
465
466 EM_CC_START_DTMF_ACKNOWLEDGE;
467
468 if( p_dtmf->mode EQ DTMF_AUTO )
469 {
470 for_stop_dtmf ();
471
472 /* On sending a STOP DTMF message the MS shall start T337 */
473 TIMERSTART (T337, T337_VALUE);
474
475 p_dtmf->state = DTMF_STOP_REQUEST;
476 }
477 else
478 {
479 /* This is DTMF_MAN */
480 if (!cc_dtmf_empty (p_dtmf))
481 {
482 UBYTE tone;
483
484 tone = cc_dtmf_read (p_dtmf);
485 if (tone EQ NOT_PRESENT_8BIT)
486 {
487 /* Should be always true */
488 p_dtmf->mode = GET_DTMF_MOD(tone);
489 for_stop_dtmf ();
490
491 /* On sending a STOP DTMF message the MS shall start T337 */
492 TIMERSTART (T337, T337_VALUE);
493
494 p_dtmf->state = DTMF_STOP_REQUEST;
495 }
496 while (!cc_dtmf_empty(p_dtmf) AND
497 p_dtmf->buf[p_dtmf->read] EQ NOT_PRESENT_8BIT)
498 {
499 /* Consume all unwanted stop marks */
500 (void)cc_dtmf_read(p_dtmf);
501 }
502 }
503 else
504 p_dtmf->state = DTMF_SEND_ACKNOWLEDGE;
505 }
506 /* Implements Measure# 3 and streamline encoding */
507 CCD_END;
508 }
509 else
510 {
511
512 /* Implements Measure# 3 and streamline encoding */
513 CCD_END;
514 cc_send_status (M_CC_CAUSE_MESSAGE_TYPE_INCOMPAT);
515 }
516 /* Implements Measure# 3 and streamline encoding */
517 } /* if (cc_check_error_flag (cc_data)) */
518 } /* if (cc_dtmf_allowed (cc_data)) */
519 else
520 {
521 /*
522 * DTMF not allowed for some reason, e.g. wrong state,
523 * call is not a speech call, no TCH assigned etc.
524 */
525 CCD_END;
526 /* Implements Measure# 3 and streamline encoding */
527 cc_send_status (M_CC_CAUSE_MESSAGE_TYPE_INCOMPAT);
528 }
529 }
530
531
532 /*
533 +--------------------------------------------------------------------+
534 | PROJECT : GSM-PS (6147) MODULE : CC_ACT |
535 | STATE : code ROUTINE : cc_start_dtmf_rej |
536 +--------------------------------------------------------------------+
537
538 PURPOSE : Processing an incoming start dtmf reject message.
539
540 */
541
542 GLOBAL void cc_start_dtmf_rej (T_D_START_DTMF_REJ * start_dtmf_rej)
543 {
544 GET_INSTANCE_DATA;
545 T_DTMF * p_dtmf;
546
547 TRACE_FUNCTION ("cc_start_dtmf_rej()");
548
549 if (cc_dtmf_allowed ())
550 {
551 if (cc_check_error_flag ())
552 {
553 /* Stop T336 */
554 TIMERSTOP (TIMER_DTMF);
555 p_dtmf = &cc_data->dtmf[cc_data->index_ti];
556 if (p_dtmf->state EQ DTMF_SEND_REQUEST)
557 {
558 PALLOC (start, MNCC_START_DTMF_CNF);
559
560 start->ti = cc_data->ti;
561 start->key = NOT_PRESENT_8BIT;
562 start->dtmf_mod = NOT_PRESENT_8BIT;
563 /*
564 * cause IE is mandatory and the check for presence has been done already
565 * thus there is no need to check for validity of cause IE here anymore
566 */
567 start->cause = CAUSE_MAKE(DEFBY_STD,
568 ORIGSIDE_NET,
569 MNCC_CC_ORIGINATING_ENTITY,
570 start_dtmf_rej->cc_cause.cause);
571 PSENDX (MMI, start);
572
573 p_dtmf->state = DTMF_IDLE;
574 p_dtmf->write = 0;
575 p_dtmf->read = 0;
576 /* Implements Measure# 3 and streamline encoding */
577 CCD_END;
578 }
579 else
580 {
581 /* Implements Measure# 3 and streamline encoding */
582 CCD_END;
583 cc_send_status (M_CC_CAUSE_MESSAGE_TYPE_INCOMPAT);
584 }
585 /* Implements Measure# 3 and streamline encoding */
586 }
587 }
588 else
589 {
590 /*
591 * DTMF not allowed for some reason, e.g. wrong state,
592 * call is not a speech call, no TCH assigned etc.
593 */
594 CCD_END;
595 /* Implements Measure# 3 and streamline encoding */
596 cc_send_status (M_CC_CAUSE_MESSAGE_TYPE_INCOMPAT);
597 }
598 }
599
600
601 /*
602 +--------------------------------------------------------------------+
603 | PROJECT : GSM-PS (6147) MODULE : CC_ACT |
604 | STATE : code ROUTINE : cc_stop_dtmf |
605 +--------------------------------------------------------------------+
606
607 PURPOSE : Stops DTMF, back to DTMF_IDLE state.
608 Reasons for call:
609 1.) Reception of STOP DTMF ACK message
610 2.) Expiry of T336
611 3.) Expiry of T337
612
613 */
614
615 GLOBAL void cc_stop_dtmf (void)
616 {
617 GET_INSTANCE_DATA;
618 T_DTMF * p_dtmf;
619
620 TRACE_FUNCTION ("cc_stop_dtmf()");
621
622 p_dtmf = &cc_data->dtmf[cc_data->index_ti];
623
624 while (!cc_dtmf_empty(p_dtmf))
625 {
626 UBYTE tone;
627
628 tone = cc_dtmf_read (p_dtmf);
629 if (tone NEQ NOT_PRESENT_8BIT)
630 {
631 /*
632 * another DTMF tone has to be sent
633 */
634 p_dtmf->mode = GET_DTMF_MOD (tone);
635 p_dtmf->key = tone & 0x7f;
636 CCD_START;
637 {
638 MCAST (start, U_START_DTMF);
639
640 cc_build_start_dtmf (p_dtmf->key, start);
641 for_start_dtmf (start);
642 }
643 CCD_END;
644
645 /* On sending a START DTMF message the MS shall start T336 */
646 TIMERSTART (T336, T336_VALUE);
647
648 p_dtmf->state = DTMF_SEND_REQUEST;
649 return;
650 }
651 }
652 p_dtmf->state = DTMF_IDLE;
653 srv_use_stored_prim ();
654 }
655
656
657 /*
658 +--------------------------------------------------------------------+
659 | PROJECT : GSM-PS (6147) MODULE : CC_ACT |
660 | STATE : code ROUTINE : cc_stop_dtmf_ack |
661 +--------------------------------------------------------------------+
662
663 PURPOSE : Processing an incoming stop dtmf ack message.
664
665 */
666
667 GLOBAL void cc_stop_dtmf_ack (void)
668 {
669 GET_INSTANCE_DATA;
670 T_DTMF * p_dtmf;
671
672 TRACE_FUNCTION ("cc_stop_dtmf_ack()");
673
674 if (cc_dtmf_allowed ())
675 {
676 if (cc_check_error_flag ())
677 {
678 CCD_END;
679 p_dtmf = &cc_data->dtmf[cc_data->index_ti];
680 if (p_dtmf->state EQ DTMF_STOP_REQUEST)
681 {
682 if (p_dtmf->mode EQ DTMF_MAN)
683 {
684 PALLOC (start, MNCC_START_DTMF_CNF);
685 start->ti = cc_data->ti;
686 start->key = NOT_PRESENT_8BIT;
687 start->cause = MNCC_CAUSE_DTMF_STOP_SUCCESS;
688 start->dtmf_mod = MNCC_DTMF_MOD_MAN_STOP;
689 PSENDX (MMI, start);
690 }
691 /* Stop T337 */
692 TIMERSTOP (TIMER_DTMF);
693 cc_stop_dtmf ();
694 }
695 else
696 {
697 /*
698 * The STOP DTMF ACK message was not expected as we did not sent
699 * previously a START DTMF request.
700 */
701 /* Implements Measure# 3 and streamline encoding */
702 cc_send_status (M_CC_CAUSE_MESSAGE_TYPE_INCOMPAT);
703 }
704 }
705 }
706 else
707 {
708 /*
709 * DTMF not allowed in the current state
710 */
711 CCD_END;
712 /* Implements Measure# 3 and streamline encoding */
713 cc_send_status (M_CC_CAUSE_MESSAGE_TYPE_INCOMPAT);
714 }
715 }
716
717
718 /*
719 +--------------------------------------------------------------------+
720 | PROJECT : GSM-PS (6147) MODULE : CC_ACT |
721 | STATE : code ROUTINE : cc_mncc_facility_req |
722 +--------------------------------------------------------------------+
723
724 PURPOSE : Sending of Facility Information by the mobile side.
725
726 */
727
728 GLOBAL void cc_mncc_facility_req (T_MNCC_FACILITY_REQ * facility)
729 {
730 GET_INSTANCE_DATA;
731
732 TRACE_FUNCTION ("cc_mncc_facility_req()");
733
734 if ((cc_data->index_ti = srv_convert_ti (facility->ti))
735 EQ NOT_PRESENT_8BIT)
736 {
737 PFREE (facility);
738 return;
739 }
740
741 switch (cc_data->state[cc_data->index_ti])
742 {
743 case M_CC_CS_19:
744 PFREE (facility);
745 break;
746
747 case M_CC_CS_01:
748 case CS_101:
749 case CS_261:
750 srv_store_prim ((T_PRIM *)D2P(facility));
751 break;
752
753 default:
754 CCD_START;
755 {
756 MCAST (facility_msg, U_FACILITY);
757 cc_build_facility (facility, facility_msg);
758 for_facility (facility_msg);
759
760 EM_CC_FACILITY_TO_THE_NETWORK;
761
762 }
763 CCD_END;
764 PFREE (facility);
765 break;
766 }
767 }
768
769
770 /*
771 +--------------------------------------------------------------------+
772 | PROJECT : GSM-PS (6147) MODULE : CC_ACT |
773 | STATE : code ROUTINE : cc_mncc_modify_req |
774 +--------------------------------------------------------------------+
775
776 PURPOSE : Start of a mobile originated in-call modification.
777
778 */
779
780 GLOBAL void cc_mncc_modify_req (T_MNCC_MODIFY_REQ * modify)
781 {
782 GET_INSTANCE_DATA;
783
784 TRACE_FUNCTION ("cc_mncc_modify_req()");
785
786 if ((cc_data->index_ti = srv_convert_ti (modify->ti))
787 EQ NOT_PRESENT_8BIT)
788 {
789 PFREE (modify);
790 return;
791 }
792
793 switch (cc_data->state[cc_data->index_ti])
794 {
795 case M_CC_CS_10:
796 if (((cc_data->hold_state[cc_data->index_ti] EQ M_CC_HLD_IDLE)) AND
797 (cc_data->dtmf[cc_data->index_ti].state EQ DTMF_IDLE))
798 {
799 /*
800 * If no other procedure is running
801 */
802 CCD_START;
803 {
804 MCAST (modify_msg, B_MODIFY);
805 cc_build_modify (modify, modify_msg);
806 for_modify (modify_msg);
807 }
808 CCD_END;
809 PFREE (modify);
810 cc_set_state (M_CC_CS_26);
811 TIMERSTART (T323, T323_VALUE);
812
813 EM_CC_MO_IN_CALL_MODIFICATION;
814
815 }
816 else
817 srv_store_prim ((T_PRIM *)D2P(modify));
818 break;
819
820 default:
821 PFREE (modify);
822 break;
823 }
824 }
825
826
827 /*
828 +--------------------------------------------------------------------+
829 | PROJECT : GSM-PS (6147) MODULE : CC_ACT |
830 | STATE : code ROUTINE : cc_mncc_user_req |
831 +--------------------------------------------------------------------+
832
833 PURPOSE : Send user information or congestion control.
834
835 */
836
837 GLOBAL void cc_mncc_user_req (T_MNCC_USER_REQ * user)
838 {
839 GET_INSTANCE_DATA;
840
841 TRACE_FUNCTION ("cc_mncc_user_req()");
842
843 if ((cc_data->index_ti = srv_convert_ti (user->ti))
844 EQ NOT_PRESENT_8BIT)
845 {
846 PFREE (user);
847 return;
848 }
849
850 switch (cc_data->state[cc_data->index_ti])
851 {
852 case M_CC_CS_10:
853 switch (user->congest_lev)
854 {
855 case M_CC_CL_RECEIVER_READY:
856 case M_CC_CL_RECEIVER_NOT_READY:
857 /*
858 * Send Congestion Control
859 */
860 CCD_START;
861 {
862 MCAST (cong_ctrl, B_CONGEST_CTRL);
863
864 cc_build_congestion_control (user, cong_ctrl);
865 for_congestion_control (cong_ctrl);
866 }
867 CCD_END;
868 PFREE (user);
869 break;
870 default:
871 /*
872 * Send User Information
873 */
874 CCD_START;
875 {
876 MCAST (user_msg, B_USER_INFO);
877
878 cc_build_user_information (user, user_msg);
879 for_user_information (user_msg);
880 }
881 CCD_END;
882 PFREE (user);
883 break;
884 }
885 break;
886
887 case CS_101:
888 srv_store_prim ((T_PRIM *)D2P(user));
889 break;
890
891 default:
892 PFREE (user);
893 break;
894 }
895 }
896
897
898 /*
899 +--------------------------------------------------------------------+
900 | PROJECT : GSM-PS (6147) MODULE : CC_ACT |
901 | STATE : code ROUTINE : cc_mncc_hold_req |
902 +--------------------------------------------------------------------+
903
904 PURPOSE : Send hold message.
905
906 */
907
908 GLOBAL void cc_mncc_hold_req (T_MNCC_HOLD_REQ * hold)
909 {
910
911 TRACE_FUNCTION ("cc_mncc_hold_req()");
912
913 /* Implements Measure# 18 */
914 cc_mncc_hold_retrieve_req ((T_PRIM *)D2P(hold));
915 }
916
917
918 /*
919 +--------------------------------------------------------------------+
920 | PROJECT : GSM-PS (6147) MODULE : CC_ACT |
921 | STATE : code ROUTINE : cc_mncc_retrieve_req |
922 +--------------------------------------------------------------------+
923
924 PURPOSE : Send retrieve message.
925
926 */
927 GLOBAL void cc_mncc_retrieve_req (T_MNCC_RETRIEVE_REQ * retrieve)
928 {
929
930 TRACE_FUNCTION ("cc_mncc_retrieve_req()");
931
932 /* Implements Measure# 18 */
933 cc_mncc_hold_retrieve_req ((T_PRIM *)D2P(retrieve));
934 }
935
936
937 /*
938 +--------------------------------------------------------------------+
939 | PROJECT : GSM-PS (6147) MODULE : CC_ACT |
940 | STATE : code ROUTINE : cc_call_is_active |
941 +--------------------------------------------------------------------+
942
943 PURPOSE : Return TRUE if the respective CC instance is in an active
944 state, otherwise return FALSE.
945
946 */
947
948 LOCAL BOOL cc_call_is_active (UBYTE index_ti)
949 {
950 GET_INSTANCE_DATA;
951 TRACE_FUNCTION ("cc_call_is_active()");
952
953 switch (cc_data->state[index_ti])
954 {
955 case M_CC_CS_10:
956 case M_CC_CS_26:
957 case CS_101:
958 case CS_261:
959 return TRUE;
960
961 default:
962 return FALSE;
963 }
964 }
965
966
967 /*
968 +--------------------------------------------------------------------+
969 | PROJECT : GSM-PS (6147) MODULE : CC_ACT |
970 | STATE : code ROUTINE : cc_mncc_sync_req |
971 +--------------------------------------------------------------------+
972
973 PURPOSE : Processing a MNCC_SYNC_REQ primitive. To be able to
974 understand some details of this function, see first
975 GSM 03.84, GSM 04.84 and GSM 11.10 chapters 31.4.1 and 31.4.2.
976
977 */
978
979 GLOBAL void cc_mncc_sync_req (T_MNCC_SYNC_REQ * mncc_sync_req)
980 {
981 GET_INSTANCE_DATA;
982 UBYTE i;
983
984 TRACE_FUNCTION ("cc_mncc_sync_req()");
985
986 if ((cc_data->index_ti = srv_convert_ti (mncc_sync_req->ti))
987 EQ NOT_PRESENT_8BIT)
988 {
989 PFREE (mncc_sync_req);
990 return;
991 }
992
993 switch (cc_data->state[cc_data->index_ti])
994 {
995 /*
996 * Note: MPTY and HOLD only has a meaning in the following states,
997 * in other states these aux states have absolutely no meaning.
998 */
999 case M_CC_CS_10:
1000 case M_CC_CS_26:
1001 case CS_101:
1002 case CS_261:
1003 switch (mncc_sync_req->synccs)
1004 {
1005 case MNCC_SYNCCS_MPTY_EVENT:
1006
1007 TRACE_EVENT_P2 (" ti = %d, mpty_event = %d",
1008 mncc_sync_req->ti,
1009 mncc_sync_req->mpty_event);
1010
1011 switch (mncc_sync_req->mpty_event)
1012 {
1013 case MNCC_MPTY_BUILD_SENT:
1014 for (i = 0; i < MAX_CC_CALLS; i++)
1015 {
1016 if (cc_call_is_active(i) AND
1017 cc_data->mpty_state[i] EQ M_CC_MPTY_IDLE)
1018 {
1019 cc_data->mpty_state[i] = M_CC_MPTY_REQ;
1020 }
1021 }
1022 break;
1023
1024 case MNCC_MPTY_BUILD_FAIL:
1025 for (i = 0; i < MAX_CC_CALLS; i++)
1026 {
1027 if (cc_call_is_active(i) AND
1028 cc_data->mpty_state[i] EQ M_CC_MPTY_REQ)
1029 {
1030 cc_data->mpty_state[i] = M_CC_MPTY_IDLE;
1031 }
1032 }
1033 break;
1034
1035 case MNCC_MPTY_BUILD_SUCCESS:
1036 for (i = 0; i < MAX_CC_CALLS; i++)
1037 {
1038 if (cc_call_is_active(i))
1039 {
1040 cc_data->mpty_state[i] = M_CC_MPTY_CALL;
1041 cc_data->hold_state[i] = M_CC_HLD_IDLE;
1042 }
1043 }
1044 break;
1045
1046 /* -- */
1047
1048 case MNCC_MPTY_SPLIT_SENT:
1049 if (cc_data->mpty_state[cc_data->index_ti] EQ M_CC_MPTY_CALL)
1050 {
1051 cc_data->mpty_state[cc_data->index_ti] = M_CC_MPTY_SPLIT;
1052 }
1053 break;
1054
1055 case MNCC_MPTY_SPLIT_FAIL:
1056 if (cc_data->mpty_state[cc_data->index_ti] EQ M_CC_MPTY_SPLIT)
1057 {
1058 cc_data->mpty_state[cc_data->index_ti] = M_CC_MPTY_CALL;
1059 }
1060 break;
1061
1062 case MNCC_MPTY_SPLIT_SUCCESS:
1063 cc_data->mpty_state[cc_data->index_ti] = M_CC_MPTY_IDLE;
1064 cc_data->hold_state[cc_data->index_ti] = M_CC_HLD_IDLE;
1065 for (i = 0; i < MAX_CC_CALLS; i++)
1066 {
1067 if (cc_call_is_active(i) AND
1068 cc_data->mpty_state[i] NEQ M_CC_MPTY_IDLE)
1069 {
1070 cc_data->hold_state[i] = M_CC_HLD_CALL_HELD;
1071 }
1072 }
1073 break;
1074
1075 /* -- */
1076
1077 case MNCC_MPTY_HOLD_SENT:
1078 for (i = 0; i < MAX_CC_CALLS; i++)
1079 {
1080 if (cc_call_is_active(i) AND
1081 cc_data->mpty_state[i] NEQ M_CC_MPTY_IDLE)
1082 {
1083 cc_data->hold_state[i] = M_CC_HLD_HOLD_REQ;
1084 }
1085 }
1086 break;
1087
1088 case MNCC_MPTY_HOLD_FAIL:
1089 for (i = 0; i < MAX_CC_CALLS; i++)
1090 {
1091 if (cc_call_is_active(i) AND
1092 cc_data->mpty_state[i] NEQ M_CC_MPTY_IDLE)
1093 {
1094 cc_data->hold_state[i] = M_CC_HLD_IDLE;
1095 }
1096 }
1097 break;
1098
1099 case MNCC_MPTY_HOLD_SUCCESS:
1100 for (i = 0; i < MAX_CC_CALLS; i++)
1101 {
1102 if (cc_call_is_active(i) AND
1103 cc_data->mpty_state[i] NEQ M_CC_MPTY_IDLE)
1104 {
1105 cc_data->hold_state[i] = M_CC_HLD_CALL_HELD;
1106 }
1107 }
1108 break;
1109
1110 /* -- */
1111
1112 case MNCC_MPTY_RETRIEVE_SENT:
1113 for (i = 0; i < MAX_CC_CALLS; i++)
1114 {
1115 if (cc_call_is_active(i) AND
1116 cc_data->mpty_state[i] NEQ M_CC_MPTY_IDLE)
1117 {
1118 cc_data->hold_state[i] = M_CC_HLD_RETRIEVE_REQ;
1119 }
1120 }
1121 break;
1122
1123 case MNCC_MPTY_RETRIEVE_FAIL:
1124 for (i = 0; i < MAX_CC_CALLS; i++)
1125 {
1126 if (cc_call_is_active(i) AND
1127 cc_data->mpty_state[i] NEQ M_CC_MPTY_IDLE)
1128 {
1129 cc_data->hold_state[i] = M_CC_HLD_CALL_HELD;
1130 }
1131 }
1132 break;
1133
1134 case MNCC_MPTY_RETRIEVE_SUCCESS:
1135 for (i = 0; i < MAX_CC_CALLS; i++)
1136 {
1137 if (cc_call_is_active(i) AND
1138 cc_data->mpty_state[i] NEQ M_CC_MPTY_IDLE)
1139 {
1140 cc_data->hold_state[i] = M_CC_HLD_IDLE;
1141 }
1142 }
1143 break;
1144
1145 /* -- */
1146
1147 default:
1148 TRACE_ERROR ("Illegal MPTY event");
1149 break;
1150 } /* switch (mncc_sync_req->mpty_event) */
1151 break;
1152
1153 default:
1154 TRACE_ERROR ("Illegal SYNCCS");
1155 } /* switch (mncc_sync_req->synccs) */
1156 break;
1157
1158 default:
1159 break; /* Ignore the event */
1160 } /* switch (cc_data->state[cc_data->index_ti]) */
1161
1162 #ifndef NTRACE
1163 for (i = 0; i < MAX_CC_CALLS; i++)
1164 {
1165 if (cc_call_is_active(i))
1166 {
1167 TRACE_EVENT_P4 (" hold[%d] = %d, mpty[%d] = %d",
1168 cc_data->stored_ti_values[i],
1169 cc_data->hold_state[i],
1170 cc_data->stored_ti_values[i],
1171 cc_data->mpty_state[i]);
1172 }
1173 }
1174 #endif /* #ifndef NTRACE */
1175
1176 PFREE (mncc_sync_req);
1177 }
1178
1179
1180 /*
1181 +--------------------------------------------------------------------+
1182 | PROJECT : GSM-PS (6147) MODULE : CC_ACT |
1183 | STATE : code ROUTINE : cc_mncc_status_res |
1184 +--------------------------------------------------------------------+
1185
1186 PURPOSE : Processing a MNCC_STATUS_RES primitive. There is an
1187 outstanding STATUS message which still has to be sent to
1188 the network after having received a STATUS ENQUIRY in U10.
1189
1190 */
1191
1192 GLOBAL void cc_mncc_status_res (T_MNCC_STATUS_RES * mncc_status_res)
1193 {
1194 GET_INSTANCE_DATA;
1195
1196 TRACE_FUNCTION ("cc_mncc_status_res()");
1197
1198 if ((cc_data->index_ti = srv_convert_ti (mncc_status_res->ti))
1199 EQ NOT_PRESENT_8BIT)
1200 {
1201 PFREE (mncc_status_res);
1202 return;
1203 }
1204
1205 switch (cc_data->state[cc_data->index_ti])
1206 {
1207 case M_CC_CS_01: /* MM connection pending */
1208 case CS_101: /* No network connection, reestablishment in progress */
1209 case CS_261:
1210 break;
1211
1212 default:
1213 /* Implements Measure# 3 and streamline encoding */
1214 cc_send_status (M_CC_CAUSE_STATUS_ENQUIRY);
1215 break;
1216 }
1217
1218 PFREE (mncc_status_res);
1219 }
1220
1221
1222 /*
1223 +--------------------------------------------------------------------+
1224 | PROJECT : GSM-PS (6147) MODULE : CC_ACT |
1225 | STATE : code ROUTINE : cc_facility |
1226 +--------------------------------------------------------------------+
1227
1228 PURPOSE : Processing of an incoming facility message.
1229
1230 */
1231
1232 GLOBAL void cc_facility (T_D_FACILITY * facility)
1233 {
1234 GET_INSTANCE_DATA;
1235 TRACE_FUNCTION ("cc_facility()");
1236
1237 /*
1238 * A facility message can be received in every state where CC has a
1239 * MM connection, no need to check states here.
1240 */
1241 if (cc_check_error_flag ())
1242 {
1243 cc_build_facility_ind (MNCC_FAC_IN_FACILITY, TRUE, &facility->facility);
1244 CCD_END;
1245 }
1246
1247 EM_CC_FACILITY_FROM_THE_NETWORK;
1248
1249 }
1250
1251
1252 /*
1253 +--------------------------------------------------------------------+
1254 | PROJECT : GSM-PS (6147) MODULE : CC_ACT |
1255 | STATE : code ROUTINE : cc_hold_ack |
1256 +--------------------------------------------------------------------+
1257
1258 PURPOSE : Processing an incoming hold ack message.
1259
1260 */
1261
1262 GLOBAL void cc_hold_ack (void)
1263 {
1264 GET_INSTANCE_DATA;
1265 CCD_END;
1266
1267 TRACE_FUNCTION ("cc_hold_ack()");
1268
1269 switch (cc_data->state[cc_data->index_ti])
1270 {
1271 /*
1272 * CS_0 is handled by the formatter
1273 */
1274 case M_CC_CS_10:
1275 if (cc_check_error_flag ())
1276 {
1277 if (cc_data->hold_state[cc_data->index_ti] EQ M_CC_HLD_HOLD_REQ)
1278 {
1279 PALLOC (hold_cnf, MNCC_HOLD_CNF);
1280
1281 cc_data->hold_state[cc_data->index_ti] = M_CC_HLD_CALL_HELD;
1282 hold_cnf->ti = cc_data->ti;
1283 hold_cnf->cause = MNCC_CAUSE_HOLD_SUCCESS;
1284 PSENDX (MMI, hold_cnf);
1285
1286 EM_CC_CALL_HOLD_ACKNOWLEDGE;
1287
1288 }
1289 else
1290 {
1291 /* Implements Measure# 3 and streamline encoding */
1292 cc_send_status (M_CC_CAUSE_MESSAGE_TYPE_INCOMPAT);
1293 }
1294 }
1295 break;
1296
1297 default:
1298 /* Implements Measure# 3 and streamline encoding */
1299 cc_send_status (M_CC_CAUSE_MESSAGE_TYPE_INCOMPAT);
1300 break;
1301 }
1302 }
1303
1304 /*
1305 +--------------------------------------------------------------------+
1306 | PROJECT : GSM-PS (6147) MODULE : CC_ACT |
1307 | STATE : code ROUTINE : cc_hold_rej |
1308 +--------------------------------------------------------------------+
1309
1310 PURPOSE : Processing an incoming hold rej message.
1311
1312 */
1313
1314 GLOBAL void cc_hold_rej (T_D_HOLD_REJ * hold_rej)
1315 {
1316 GET_INSTANCE_DATA;
1317 TRACE_FUNCTION ("cc_hold_rej()");
1318
1319 switch (cc_data->state[cc_data->index_ti])
1320 {
1321 /*
1322 * CS_0 is handled by the formatter
1323 */
1324 case M_CC_CS_10:
1325 if (cc_data->hold_state[cc_data->index_ti] EQ M_CC_HLD_HOLD_REQ)
1326 {
1327 PALLOC (hold_cnf, MNCC_HOLD_CNF);
1328
1329 cc_data->hold_state[cc_data->index_ti] = M_CC_HLD_IDLE;
1330 hold_cnf->ti = cc_data->ti;
1331
1332 /*
1333 * It is possible for the network to send a REJ without a Cause IE.
1334 * This causes the decoder to flag a Madatory IE error. According to
1335 * 04.08 Sec: 8.5.3.e, the REJ should still be managed as if a correctly
1336 * formatted REJ had been received.
1337 */
1338 if (!hold_rej->cc_cause.v_cause)
1339 {
1340 hold_cnf->cause = MNCC_CAUSE_NO_NET_CAUSE;
1341 }
1342 else
1343 {
1344 hold_cnf->cause = CAUSE_MAKE(DEFBY_STD,
1345 ORIGSIDE_NET,
1346 MNCC_CC_ORIGINATING_ENTITY,
1347 hold_rej->cc_cause.cause);
1348 }
1349 PSENDX (MMI, hold_cnf);
1350 srv_use_stored_prim ();
1351
1352 EM_CC_CALL_HOLD_REJECT;
1353
1354 CCD_END;
1355 }
1356 else
1357 {
1358 CCD_END;
1359 /* Implements Measure# 3 and streamline encoding */
1360 cc_send_status (M_CC_CAUSE_MESSAGE_TYPE_INCOMPAT);
1361 }
1362 break;
1363
1364 default:
1365 CCD_END;
1366 /* Implements Measure# 3 and streamline encoding */
1367 cc_send_status (M_CC_CAUSE_MESSAGE_TYPE_INCOMPAT);
1368 break;
1369 }
1370 }
1371
1372 /*
1373 +--------------------------------------------------------------------+
1374 | PROJECT : GSM-PS (6147) MODULE : CC_ACT |
1375 | STATE : code ROUTINE : cc_retrieve_ack |
1376 +--------------------------------------------------------------------+
1377
1378 PURPOSE : Processing an incoming retrieve ack message.
1379
1380 */
1381
1382 GLOBAL void cc_retrieve_ack (void)
1383 {
1384 GET_INSTANCE_DATA;
1385
1386 TRACE_FUNCTION ("cc_retrieve_ack()");
1387
1388 switch (cc_data->state[cc_data->index_ti])
1389 {
1390 /*
1391 * CS_0 is handled by the formatter
1392 */
1393 case M_CC_CS_10:
1394 if (cc_check_error_flag ())
1395 {
1396 CCD_END;
1397 if (cc_data->hold_state[cc_data->index_ti] EQ M_CC_HLD_RETRIEVE_REQ)
1398 {
1399 PALLOC (retrieve_cnf, MNCC_RETRIEVE_CNF);
1400
1401 cc_data->hold_state[cc_data->index_ti] = M_CC_HLD_IDLE;
1402 retrieve_cnf->ti = cc_data->ti;
1403 retrieve_cnf->cause = MNCC_CAUSE_RETRIEVE_SUCCESS;
1404 PSENDX (MMI, retrieve_cnf);
1405
1406 EM_CC_CALL_RETRIEVE_ACKNOWLEDGE;
1407
1408 }
1409 else
1410 {
1411 /* Implements Measure# 3 and streamline encoding */
1412 cc_send_status (M_CC_CAUSE_MESSAGE_TYPE_INCOMPAT);
1413 }
1414 }
1415 break;
1416
1417 default:
1418 CCD_END;
1419 /* Implements Measure# 3 and streamline encoding */
1420 cc_send_status (M_CC_CAUSE_MESSAGE_TYPE_INCOMPAT);
1421 break;
1422 }
1423 }
1424
1425 /*
1426 +--------------------------------------------------------------------+
1427 | PROJECT : GSM-PS (6147) MODULE : CC_ACT |
1428 | STATE : code ROUTINE : cc_retrieve_rej |
1429 +--------------------------------------------------------------------+
1430
1431 PURPOSE : Processing an incoming retrieve rej message.
1432
1433 */
1434
1435 GLOBAL void cc_retrieve_rej (T_D_RETRIEVE_REJ * retrieve_rej)
1436 {
1437 GET_INSTANCE_DATA;
1438 TRACE_FUNCTION ("cc_retrieve_rej()");
1439
1440 switch (cc_data->state[cc_data->index_ti])
1441 {
1442 /*
1443 * CS_0 is handled by the formatter
1444 */
1445 case M_CC_CS_10:
1446 if (cc_data->hold_state[cc_data->index_ti] EQ M_CC_HLD_RETRIEVE_REQ)
1447 {
1448 PALLOC (retrieve_cnf, MNCC_RETRIEVE_CNF);
1449
1450 cc_data->hold_state[cc_data->index_ti] = M_CC_HLD_CALL_HELD;
1451 retrieve_cnf->ti = cc_data->ti;
1452
1453 /*
1454 * It is possible for the network to send a REJ without a Cause IE.
1455 * This causes the decoder to flag a Madatory IE error. According to
1456 * 04.08 Sec: 8.5.3.e, the REJ should still be managed as if a correctly
1457 * formatted REJ had been received.
1458 */
1459 if (!retrieve_rej->cc_cause.v_cause)
1460 {
1461 retrieve_cnf->cause = MNCC_CAUSE_NO_NET_CAUSE;
1462 }
1463 else
1464 {
1465 retrieve_cnf->cause = CAUSE_MAKE(DEFBY_STD,
1466 ORIGSIDE_NET,
1467 MNCC_CC_ORIGINATING_ENTITY,
1468 retrieve_rej->cc_cause.cause);
1469 }
1470 PSENDX (MMI, retrieve_cnf);
1471 srv_use_stored_prim ();
1472
1473 EM_CC_CALL_RETRIEVE_REJECT;
1474
1475 CCD_END;
1476 }
1477 else
1478 {
1479 CCD_END;
1480 /* Implements Measure# 3 and streamline encoding */
1481 cc_send_status (M_CC_CAUSE_MESSAGE_TYPE_INCOMPAT);
1482 }
1483 break;
1484
1485 default:
1486 CCD_END;
1487 /* Implements Measure# 3 and streamline encoding */
1488 cc_send_status (M_CC_CAUSE_MESSAGE_TYPE_INCOMPAT);
1489 break;
1490 }
1491 }
1492
1493 /*
1494 +--------------------------------------------------------------------+
1495 | PROJECT : GSM-PS (6147) MODULE : CC_ACT |
1496 | STATE : code ROUTINE : cc_congestion_control |
1497 +--------------------------------------------------------------------+
1498
1499 PURPOSE : Processing an incoming congestion control message.
1500
1501 */
1502
1503 GLOBAL void cc_congestion_control (T_B_CONGEST_CTRL * cong)
1504 {
1505 GET_INSTANCE_DATA;
1506 TRACE_FUNCTION ("cc_congestion_control()");
1507
1508 switch (cc_data->state[cc_data->index_ti])
1509 {
1510 /*
1511 * CS_0 is handled by the formatter
1512 */
1513 case M_CC_CS_10:
1514 if (cc_check_error_flag ())
1515 {
1516 PALLOC (user, MNCC_USER_IND);
1517
1518 cc_build_user_ind_from_cong (user, cong);
1519 PSENDX (MMI, user);
1520 CCD_END;
1521 }
1522 break;
1523
1524 default:
1525 CCD_END;
1526 /* Implements Measure# 3 and streamline encoding*/
1527 cc_send_status (M_CC_CAUSE_MESSAGE_TYPE_INCOMPAT);
1528 break;
1529 }
1530 }
1531
1532 /*
1533 +--------------------------------------------------------------------+
1534 | PROJECT : GSM-PS (6147) MODULE : CC_ACT |
1535 | STATE : code ROUTINE : cc_user_information |
1536 +--------------------------------------------------------------------+
1537
1538 PURPOSE : Processing an incoming user information message.
1539
1540 */
1541
1542 GLOBAL void cc_user_information ( T_B_USER_INFO * user)
1543 {
1544 GET_INSTANCE_DATA;
1545 TRACE_FUNCTION ("cc_user_information()");
1546
1547 switch (cc_data->state[cc_data->index_ti])
1548 {
1549 /*
1550 * CS_0 is handled by the formatter
1551 */
1552 case M_CC_CS_10:
1553 if (cc_check_error_flag ())
1554 {
1555 PALLOC (user_ind, MNCC_USER_IND);
1556
1557 cc_build_user_ind_from_user (user_ind, user);
1558 CCD_END;
1559 PSENDX (MMI, user_ind);
1560
1561 EM_CC_USER_TO_USER_DATA_RECEIVED;
1562
1563 }
1564 break;
1565
1566 default:
1567 CCD_END;
1568 /* Implements Measure# 4 and streamline encoding*/
1569 cc_send_status (M_CC_CAUSE_MESSAGE_TYPE_INCOMPAT);
1570 break;
1571 }
1572 }
1573
1574 /*
1575 +--------------------------------------------------------------------+
1576 | PROJECT : GSM-PS (6147) MODULE : CC_ACT |
1577 | STATE : code ROUTINE : cc_modify |
1578 +--------------------------------------------------------------------+
1579
1580 PURPOSE : Processing an incoming modify message.
1581
1582 */
1583
1584 GLOBAL void cc_modify (T_B_MODIFY * modify)
1585 {
1586 GET_INSTANCE_DATA;
1587
1588 TRACE_FUNCTION ("cc_modify()");
1589
1590 switch (cc_data->state[cc_data->index_ti])
1591 {
1592 /*
1593 * CS_0 is handled by the formatter
1594 */
1595 case M_CC_CS_10:
1596 if (cc_check_error_flag ())
1597 {
1598 switch (cc_data->neg_ri)
1599 {
1600 case M_CC_REPEAT_CIRCULAR:
1601 if (cc_data->active_service EQ cc_data->neg_serv1)
1602 {
1603 if (cc_bcs_compatible (&modify->bearer_cap,
1604 &cc_data->neg_bc2, TRUE) EQ TRUE)
1605 {
1606 CCD_END;
1607 cc_mod_complete ();
1608 }
1609 else
1610 {
1611 CCD_END;
1612 cc_mod_reject ();
1613 }
1614 }
1615 else
1616 {
1617 if (cc_bcs_compatible (&modify->bearer_cap,
1618 &cc_data->neg_bc1, TRUE) EQ TRUE)
1619 {
1620 CCD_END;
1621 cc_mod_complete ();
1622 }
1623 else
1624 {
1625 CCD_END;
1626 cc_mod_reject ();
1627 }
1628 }
1629 break;
1630 #ifndef REL99
1631 case M_CC_REPEAT_SEQUENTIAL:
1632 if (cc_data->active_service EQ cc_data->neg_serv1)
1633 {
1634 if (cc_bcs_compatible (&modify->bearer_cap,
1635 &cc_data->neg_bc2, TRUE) EQ TRUE)
1636 {
1637 CCD_END;
1638 cc_mod_complete ();
1639 }
1640 else
1641 {
1642 CCD_END;
1643 cc_mod_reject ();
1644 }
1645 }
1646 else
1647 {
1648 CCD_END;
1649 cc_mod_reject ();
1650 }
1651 break;
1652 #endif /* #ifndef REL99 */
1653 default:
1654 CCD_END;
1655 cc_mod_reject ();
1656 break;
1657 }
1658 }
1659 break;
1660
1661 default:
1662 CCD_END;
1663 /* Implements Measure# 5 and streamline encoding*/
1664 cc_send_status (M_CC_CAUSE_MESSAGE_TYPE_INCOMPAT);
1665 break;
1666 }
1667 }
1668
1669 /*
1670 +--------------------------------------------------------------------+
1671 | PROJECT : GSM-PS (6147) MODULE : CC_ACT |
1672 | STATE : code ROUTINE : cc_mod_reject |
1673 +--------------------------------------------------------------------+
1674
1675 PURPOSE : Negative end of a mobile terminated in-call modification.
1676
1677 */
1678
1679 GLOBAL void cc_mod_reject (void)
1680 {
1681 GET_INSTANCE_DATA;
1682 TRACE_FUNCTION ("cc_mod_reject()");
1683
1684 CCD_START;
1685 {
1686 MCAST (modify_rej, B_MODIFY_REJ);
1687
1688
1689 cc_build_modify_reject (modify_rej,
1690 CAUSE_MAKE(DEFBY_STD,
1691 ORIGSIDE_MS,
1692 MNCC_CC_ORIGINATING_ENTITY,
1693 M_CC_CAUSE_BEARER_CAP_AUTHORIZ));
1694 for_modify_reject (modify_rej);
1695 }
1696
1697 EM_CC_MT_IN_CALL_MODIFICATION_FAILED;
1698
1699 CCD_END;
1700 }
1701
1702 /*
1703 +--------------------------------------------------------------------+
1704 | PROJECT : GSM-PS (6147) MODULE : CC_ACT |
1705 | STATE : code ROUTINE : cc_mod_complete |
1706 +--------------------------------------------------------------------+
1707
1708 PURPOSE : Positive end of a mobile terminated in-call modification.
1709
1710 */
1711
1712 GLOBAL void cc_mod_complete (void)
1713 {
1714 GET_INSTANCE_DATA;
1715 TRACE_FUNCTION ("cc_mod_complete()");
1716
1717 CCD_START;
1718 {
1719 MCAST (modify_com, B_MODIFY_COMP);
1720 PALLOC (mod_ind, MNCC_MODIFY_IND);
1721
1722
1723 if (cc_data->active_service EQ cc_data->neg_serv1)
1724 cc_data->active_service = cc_data->neg_serv2;
1725 else
1726 cc_data->active_service = cc_data->neg_serv1;
1727
1728 mod_ind->ti = cc_data->ti;
1729 mod_ind->serv = cc_data->active_service;
1730 PSENDX (MMI, mod_ind);
1731
1732 EM_CC_MT_IN_CALL_MODIFICATION_PASSED;
1733
1734 cc_build_modify_complete (modify_com);
1735 for_modify_complete (modify_com);
1736 }
1737 CCD_END;
1738 }
1739
1740 /*
1741 +--------------------------------------------------------------------+
1742 | PROJECT : GSM-PS (6147) MODULE : CC_ACT |
1743 | STATE : code ROUTINE : cc_modify_complete |
1744 +--------------------------------------------------------------------+
1745
1746 PURPOSE : Processing an incoming modify complete message.
1747
1748 */
1749
1750 GLOBAL void cc_modify_complete ( T_B_MODIFY_COMP * mod_com)
1751 {
1752 GET_INSTANCE_DATA;
1753 TRACE_FUNCTION ("cc_modify_complete()");
1754
1755 switch (cc_data->state[cc_data->index_ti])
1756 {
1757 /*
1758 * CS_0 is handled by the formatter
1759 */
1760 case M_CC_CS_26:
1761 if (cc_check_error_flag ())
1762 {
1763 if (cc_data->new_itc EQ mod_com->bearer_cap.trans_cap)
1764 {
1765 PALLOC (mod_cnf, MNCC_MODIFY_CNF);
1766
1767 TIMERSTOP (TIMER_CC);
1768 if (cc_data->active_service EQ cc_data->serv1)
1769 cc_data->active_service = cc_data->serv2;
1770 else
1771 cc_data->active_service = cc_data->serv1;
1772 mod_cnf->ti = cc_data->ti;
1773 mod_cnf->cause = MNCC_CAUSE_MODIFY_SUCCESS;
1774 /* Setting raw_cause to empty as we are not sending any cause from network*/
1775 mod_cnf->c_raw_cause = 0;
1776
1777 EM_CC_MO_IN_CALL_MODIFICATION_PASSED;
1778
1779 PSENDX (MMI, mod_cnf);
1780 cc_set_state (M_CC_CS_10);
1781 }
1782 CCD_END;
1783 }
1784 break;
1785
1786 default:
1787 CCD_END;
1788 /* Implements Measure# 6 and streamline encoding*/
1789 cc_send_status (M_CC_CAUSE_MESSAGE_TYPE_INCOMPAT);
1790 break;
1791 }
1792 }
1793
1794 /*
1795 +--------------------------------------------------------------------+
1796 | PROJECT : GSM-PS (6147) MODULE : CC_ACT |
1797 | STATE : code ROUTINE : cc_modify_reject |
1798 +--------------------------------------------------------------------+
1799
1800 PURPOSE : Processing an incoming modify reject message.
1801
1802 */
1803
1804 GLOBAL void cc_modify_reject (T_B_MODIFY_REJ * mod_rej)
1805 {
1806 GET_INSTANCE_DATA;
1807 TRACE_FUNCTION ("cc_modify_reject()");
1808
1809 switch (cc_data->state[cc_data->index_ti])
1810 {
1811 /*
1812 * CS_0 is handled by the formatter
1813 */
1814 case M_CC_CS_26:
1815 if (cc_check_error_flag ())
1816 {
1817 if (cc_data->old_itc EQ mod_rej->bearer_cap.trans_cap)
1818 {
1819 PALLOC (mod_cnf, MNCC_MODIFY_CNF);
1820
1821 TIMERSTOP (TIMER_CC);
1822 mod_cnf->ti = cc_data->ti;
1823 /*
1824 * cause IE is mandatory and the check for presence has been done
1825 * already thus there is no need to check for validity of cause IE
1826 * here anymore
1827 */
1828 mod_cnf->cause = CAUSE_MAKE(DEFBY_STD,
1829 ORIGSIDE_NET,
1830 MNCC_CC_ORIGINATING_ENTITY,
1831 mod_rej->cc_cause.cause);
1832 mod_cnf->c_raw_cause = cc_build_cause (&mod_rej->cc_cause,
1833 mod_cnf->raw_cause);
1834 PSENDX (MMI, mod_cnf);
1835 cc_set_state (M_CC_CS_10);
1836 }
1837
1838 EM_CC_MO_IN_CALL_MODIFICATION_FAILED;
1839
1840 CCD_END;
1841 }
1842 break;
1843
1844 default:
1845 CCD_END;
1846 /* Implements Measure# 7 and streamline encoding*/
1847 cc_send_status (M_CC_CAUSE_MESSAGE_TYPE_INCOMPAT);
1848 break;
1849 }
1850 }
1851
1852
1853 /*
1854 +--------------------------------------------------------------------+
1855 | PROJECT : GSM-PS (6147) MODULE : CC_ACT |
1856 | STATE : code ROUTINE : cc_status |
1857 +--------------------------------------------------------------------+
1858
1859 PURPOSE : Processing an incoming status message.
1860
1861 */
1862
1863 GLOBAL void cc_status (T_B_STATUS * status)
1864 {
1865 GET_INSTANCE_DATA;
1866 TRACE_FUNCTION ("cc_status()");
1867
1868 /* The cause is in STATUS a mandatory IE (given intact message) */
1869 TRACE_EVENT_P1 ("STATUS cause: %02x", status->cc_cause.cause);
1870
1871 switch (cc_data->state[cc_data->index_ti])
1872 {
1873 case M_CC_CS_0:
1874 CCD_END;
1875 if (status->call_state.state NEQ M_CC_CS_0)
1876 {
1877 CCD_START;
1878 {
1879 MCAST (rel_com, U_RELEASE_COMP);
1880
1881 cc_build_release_complete (rel_com,
1882 CAUSE_MAKE(DEFBY_STD,
1883 ORIGSIDE_MS,
1884 MNCC_CC_ORIGINATING_ENTITY,
1885 M_CC_CAUSE_MESSAGE_INCOMPAT));
1886 for_release_complete (rel_com);
1887 }
1888 CCD_END;
1889 }
1890 for_rel_req ();
1891 break;
1892
1893 case M_CC_CS_1:
1894 case M_CC_CS_3:
1895 case M_CC_CS_4:
1896 case M_CC_CS_6:
1897 case M_CC_CS_7:
1898 case M_CC_CS_8:
1899 case M_CC_CS_9:
1900 case M_CC_CS_10:
1901 case M_CC_CS_26:
1902 if (cc_check_error_flag ())
1903 {
1904 /* check for incompatible state */
1905 if (status->call_state.state NEQ cc_data->state[cc_data->index_ti])
1906 {
1907 PALLOC (rel_ind, MNCC_RELEASE_IND);
1908
1909 cc_reset_dtmf (); // China change HM 11.07.00
1910
1911 TIMERSTOP (TIMER_CC);
1912
1913 rel_ind->ti = cc_data->ti;
1914
1915 rel_ind->cause = CAUSE_MAKE(DEFBY_STD,
1916 ORIGSIDE_MS,
1917 MNCC_CC_ORIGINATING_ENTITY,
1918 M_CC_CAUSE_MESSAGE_INCOMPAT);
1919 rel_ind->c_raw_cause = 0;
1920 PSENDX (MMI, rel_ind);
1921
1922 CCD_END;
1923 CCD_START;
1924 {
1925 MCAST (rel_com, U_RELEASE_COMP);
1926
1927 cc_build_release_complete (rel_com,
1928 CAUSE_MAKE(DEFBY_STD,
1929 ORIGSIDE_MS,
1930 MNCC_CC_ORIGINATING_ENTITY,
1931 M_CC_CAUSE_MESSAGE_INCOMPAT));
1932 for_release_complete (rel_com);
1933 }
1934 CCD_END;
1935 for_rel_req ();
1936 cc_set_state (M_CC_CS_0);
1937 }
1938 else
1939 /*
1940 * compatible, perform checks/ actions of 04.08, ch. 5.5.3.2.2;
1941 * Condat action (option!) is to initiate normal call clearing
1942 * (with DISCONNECT)
1943 */
1944 {
1945 if ((status->cc_cause.cause >= M_CC_CAUSE_INCORRECT_MESSAGE)
1946 AND
1947 (status->cc_cause.cause <= M_CC_CAUSE_COND_INFO_ELEM))
1948 {
1949 /*
1950 * Block entered for
1951 * CAUSE_INCORRECT_MESSAGE, CAUSE_INVALID_MAND_INFO,
1952 * CAUSE_MESSAGE_TYPE_NOT_IMPLEM, CAUSE_MESSAGE_TYPE_INCOMPAT,
1953 * CAUSE_INFO_ELEM_NOT_IMPLEM, CAUSE_COND_INFO_ELEM. (0x5f..0x64)
1954 */
1955 if ((cc_data->state[cc_data->index_ti] EQ M_CC_CS_10) AND
1956 (cc_data->hold_state[cc_data->index_ti] EQ M_CC_HLD_HOLD_REQ))
1957 {
1958 /*
1959 * Compatible state, last CC message was a hold request which is
1960 * not supported by the network. Handled like HOLD REJECT.
1961 */
1962 PALLOC (hold_cnf, MNCC_HOLD_CNF); /* T_MNCC_HOLD_CNF */
1963 cc_data->hold_state[cc_data->index_ti] = M_CC_HLD_IDLE;
1964 hold_cnf->ti = cc_data->ti;
1965 hold_cnf->cause = CAUSE_MAKE(DEFBY_STD,
1966 ORIGSIDE_NET,
1967 MNCC_CC_ORIGINATING_ENTITY,
1968 status->cc_cause.cause);
1969 PSENDX (MMI, hold_cnf);
1970 CCD_END;
1971 srv_use_stored_prim ();
1972 }
1973 else if (status->cc_cause.cause NEQ M_CC_CAUSE_INFO_ELEM_NOT_IMPLEM)
1974 {
1975 /*
1976 * Compatible state, special condition for HOLD REQUEST not met.
1977 * The network is also not complaining about an optional
1978 * information element which can be simply ignored.
1979 * Disconnect the call.
1980 */
1981 PALLOC (disc_ind, MNCC_DISCONNECT_IND);
1982
1983 TIMERSTOP (TIMER_CC);
1984
1985 disc_ind->ti = cc_data->ti;
1986 disc_ind->cause = CAUSE_MAKE(DEFBY_STD,
1987 ORIGSIDE_NET,
1988 MNCC_CC_ORIGINATING_ENTITY,
1989 status->cc_cause.cause);
1990 disc_ind->c_raw_cause = cc_build_cause (&status->cc_cause,
1991 disc_ind->raw_cause);
1992 disc_ind->diagnostic = NOT_PRESENT_8BIT;
1993 disc_ind->progress_desc = MNCC_PROG_NOT_PRES;
1994 PSENDX (MMI, disc_ind);
1995
1996 CCD_END;
1997 CCD_START;
1998 {
1999 MCAST (disconnect, U_DISCONNECT);
2000
2001 cc_build_disconnect (disconnect,
2002 CAUSE_MAKE(DEFBY_STD,
2003 ORIGSIDE_MS,
2004 MNCC_CC_ORIGINATING_ENTITY,
2005 M_CC_CAUSE_CALL_CLEAR),
2006 NULL, MNCC_SS_VER_NOT_PRES);
2007 for_disconnect (disconnect);
2008 }
2009 TIMERSTART (T305, T305_VALUE);
2010 cc_set_state (M_CC_CS_11);
2011 }
2012 }
2013 else
2014 {
2015 /*
2016 * STATUS message but neither incompatible state nor indicating
2017 * incorrect message / IE; check if we received a STATUS with cause
2018 * "message not compatible with protocol state". GSM 04.08 does not
2019 * define what to do and also does not define that this case has to
2020 * be handled at all;
2021 * Condat (HM) decided to "clear DTMF" if it was running to try to
2022 * recover from the misaligned states scenario.
2023 */
2024 T_DTMF *p_dtmf;
2025
2026 p_dtmf = &cc_data->dtmf[cc_data->index_ti];
2027 if ((p_dtmf->state EQ DTMF_SEND_REQUEST) AND
2028 (status->cc_cause.cause EQ M_CC_CAUSE_MESSAGE_INCOMPAT))
2029 {
2030 /* Send negative acknowledge for DTMF send request to MMI */
2031 PALLOC (mncc_start_dtmf_cnf, MNCC_START_DTMF_CNF);
2032 mncc_start_dtmf_cnf->ti = cc_data->ti;
2033 mncc_start_dtmf_cnf->key = NOT_PRESENT_8BIT;
2034 mncc_start_dtmf_cnf->dtmf_mod = NOT_PRESENT_8BIT;
2035 mncc_start_dtmf_cnf->cause = CAUSE_MAKE(DEFBY_STD,
2036 ORIGSIDE_NET,
2037 MNCC_CC_ORIGINATING_ENTITY,
2038 M_CC_CAUSE_MESSAGE_INCOMPAT);
2039 PSENDX (MMI, mncc_start_dtmf_cnf);
2040 }
2041
2042 /* Bring screwed up network back to work */
2043 for_stop_dtmf ();
2044
2045 /* Reset screwed up DTMF state machine */
2046 p_dtmf->state = DTMF_IDLE;
2047 p_dtmf->read = 0;
2048 p_dtmf->write = 0;
2049 }
2050 CCD_END;
2051 }
2052 }
2053 break;
2054
2055 case M_CC_CS_11:
2056 case M_CC_CS_12:
2057 if (cc_check_error_flag ())
2058 {
2059 if (status->call_state.state NEQ cc_data->state[cc_data->index_ti])
2060 {
2061 /*
2062 * STATUS message with incompatible state
2063 * On receipt of a STATUS message reporting an incompatible call
2064 * control state, the receiving entity shall clear the call by
2065 * sending a RELEASE COMPLETE message with cause # 101 "message
2066 * not compatible with protocol state". The reported call control
2067 * state is incompatible if the combination of call control states
2068 * at the sender and receiver side cannot occur, do not match or cannot
2069 * be aligned by actions of the receiver; the exact definition is
2070 * implementation dependent. [04.08 clause 5.5.3.2.1]
2071 */
2072 cc_reset_dtmf ();
2073
2074 TIMERSTOP (TIMER_CC);
2075
2076 /* Send MNCC_RELEASE_IND to MMI */
2077 {
2078 PALLOC (rel_ind, MNCC_RELEASE_IND); /* T_MNCC_RELEASE_IND */
2079 rel_ind->ti = cc_data->ti;
2080 rel_ind->cause = CAUSE_MAKE(DEFBY_STD,
2081 ORIGSIDE_MS,
2082 MNCC_CC_ORIGINATING_ENTITY,
2083 M_CC_CAUSE_MESSAGE_INCOMPAT);
2084 rel_ind->c_raw_cause = 0;
2085 PSENDX (MMI, rel_ind);
2086 }
2087
2088 CCD_END;
2089 CCD_START;
2090 {
2091 MCAST (rel_com, U_RELEASE_COMP); /* T_U_RELEASE_COMP */
2092 cc_build_release_complete (rel_com,
2093 CAUSE_MAKE(DEFBY_STD,
2094 ORIGSIDE_MS,
2095 MNCC_CC_ORIGINATING_ENTITY,
2096 M_CC_CAUSE_MESSAGE_INCOMPAT));
2097 for_release_complete (rel_com);
2098 for_rel_req ();
2099 cc_set_state (M_CC_CS_0);
2100 }
2101 }
2102 CCD_END;
2103 }
2104 break;
2105
2106 case M_CC_CS_19:
2107 if (cc_check_error_flag ())
2108 {
2109 if (status->call_state.state NEQ cc_data->state[cc_data->index_ti])
2110 {
2111 TIMERSTOP (TIMER_CC);
2112
2113 cc_reset_dtmf ();
2114
2115 {
2116 PALLOC (rel_cnf, MNCC_RELEASE_CNF); /* T_MNCC_RELEASE_CNF */
2117 rel_cnf->ti = cc_data->ti;
2118 rel_cnf->cause = CAUSE_MAKE(DEFBY_STD,
2119 ORIGSIDE_MS,
2120 MNCC_CC_ORIGINATING_ENTITY,
2121 M_CC_CAUSE_MESSAGE_INCOMPAT);
2122 /* Setting raw_cause to empty as this is a local release
2123 * of MM connection,CC is not receiving any cause value
2124 * from Network
2125 */
2126 rel_cnf->c_raw_cause = 0;
2127 PSENDX (MMI, rel_cnf);
2128 }
2129
2130 CCD_END;
2131 CCD_START;
2132 {
2133 MCAST (rel_com, U_RELEASE_COMP); /* T_U_RELEASE_COMP */
2134 cc_build_release_complete (rel_com,
2135 CAUSE_MAKE(DEFBY_STD,
2136 ORIGSIDE_MS,
2137 MNCC_CC_ORIGINATING_ENTITY,
2138 M_CC_CAUSE_MESSAGE_INCOMPAT));
2139 for_release_complete (rel_com);
2140 for_rel_req ();
2141 cc_set_state (M_CC_CS_0);
2142 }
2143 }
2144 CCD_END;
2145 }
2146 break;
2147
2148 default:
2149 CCD_END;
2150 break;
2151 }
2152
2153 EM_CC_STATUS_RECEIVED;
2154
2155 }
2156
2157 /*
2158 +--------------------------------------------------------------------+
2159 | PROJECT : GSM-PS (6147) MODULE : CC_ACT |
2160 | STATE : code ROUTINE : cc_status_enquiry |
2161 +--------------------------------------------------------------------+
2162
2163 PURPOSE : Processing an incoming status enquiry message.
2164
2165 */
2166
2167 GLOBAL void cc_status_enquiry (void)
2168 {
2169 GET_INSTANCE_DATA;
2170
2171 TRACE_FUNCTION ("cc_status_enquiry()");
2172
2173 /*
2174 * CS_0 is handled by the formatter
2175 */
2176 switch (cc_data->state[cc_data->index_ti])
2177 {
2178 case M_CC_CS_10:
2179 /*
2180 * Ensure synchronization of auxiliary states with ACI.
2181 */
2182 if (cc_check_error_flag ())
2183 {
2184 CCD_END;
2185 {
2186 PALLOC (mncc_status_ind, MNCC_STATUS_IND);
2187 mncc_status_ind->ti = cc_data->ti;
2188 PSENDX (MMI, mncc_status_ind);
2189 }
2190
2191 EM_CC_STATUS_ENQUIRY_RECEIVED;
2192
2193 }
2194 break;
2195
2196 default:
2197 /*
2198 * No auxiliary states in non-active states. Immediate response is safe.
2199 */
2200 if (cc_check_error_flag ())
2201 {
2202 CCD_END;
2203 /* EM call moved above cc_send_status should not have any impact */
2204 EM_CC_STATUS_ENQUIRY_RECEIVED;
2205
2206 /* Implements Measure# 7 and streamline encoding*/
2207 cc_send_status (M_CC_CAUSE_STATUS_ENQUIRY);
2208 }
2209 break;
2210 }
2211 }
2212
2213
2214 /*
2215 +--------------------------------------------------------------------+
2216 | PROJECT : GSM-PS (6147) MODULE : CC_ACT |
2217 | STATE : code ROUTINE : cc_unknown_message |
2218 +--------------------------------------------------------------------+
2219
2220 PURPOSE : Processing of an incoming unknown message.
2221
2222 */
2223
2224 GLOBAL void cc_unknown_message (void)
2225 {
2226
2227 CCD_END;
2228 CCD_START;
2229 {
2230 MCAST (status, B_STATUS);
2231
2232 TRACE_FUNCTION ("cc_unknown_message()");
2233
2234 cc_build_status (status,
2235 CAUSE_MAKE(DEFBY_STD,
2236 ORIGSIDE_MS,
2237 MNCC_CC_ORIGINATING_ENTITY,
2238 M_CC_CAUSE_MESSAGE_TYPE_NOT_IMPLEM));
2239 for_status (status);
2240 }
2241 CCD_END;
2242 }
2243
2244 /* Implements Measure# 18 */
2245 /*
2246 +--------------------------------------------------------------------+
2247 | PROJECT : GSM-PS (6147) MODULE : CC_ACT |
2248 | STATE : code ROUTINE : cc_mncc_hold_retrieve_req |
2249 +--------------------------------------------------------------------+
2250
2251 PURPOSE : Send hold REQ/retrieve REQ.
2252
2253 */
2254
2255 LOCAL void cc_mncc_hold_retrieve_req (T_PRIM * prim)
2256 {
2257 GET_INSTANCE_DATA;
2258 UBYTE ti;
2259
2260 TRACE_FUNCTION ("cc_mncc_hold_retrieve_req()");
2261
2262 if(prim->custom.opc EQ MNCC_HOLD_REQ)
2263 {
2264 ti = ((T_MNCC_HOLD_REQ *)P2D(prim))->ti;
2265 }
2266 else
2267 {
2268 ti = ((T_MNCC_RETRIEVE_REQ *)P2D(prim))->ti;
2269 }
2270 if ((cc_data->index_ti = srv_convert_ti (ti)) EQ NOT_PRESENT_8BIT)
2271 {
2272 PFREE (P2D(prim));
2273 return;
2274 }
2275
2276 switch (cc_data->state[cc_data->index_ti])
2277 {
2278 case M_CC_CS_10:
2279 if(prim->custom.opc EQ MNCC_HOLD_REQ)
2280 {
2281 /* We should think about storing the hold request
2282 until DTMF idle state reached. ...
2283 This causes some other changes, we have to add some
2284 calls to serv_use_stored_prim() at some places after
2285 state transition of the DTMF state machine to DTMF idle.*/
2286 cc_data->hold_state[cc_data->index_ti] = M_CC_HLD_HOLD_REQ;
2287 for_hold ();
2288 EM_CC_CALL_HOLD;
2289 }
2290 else
2291 {
2292 cc_data->hold_state[cc_data->index_ti] = M_CC_HLD_RETRIEVE_REQ;
2293 for_retrieve ();
2294 EM_CC_CALL_RETRIEVE;
2295 }
2296 PFREE (P2D(prim));
2297
2298 break;
2299
2300 case CS_101:
2301 case M_CC_CS_26:
2302 case CS_261:
2303 srv_store_prim (prim);
2304 break;
2305
2306 default:
2307 PFREE (P2D(prim));
2308 break;
2309 }
2310 }
2311
2312 #endif