FreeCalypso > hg > fc-tourmaline
comparison src/g23m-gprs/grr/grr_pgf.c @ 1:fa8dc04885d8
src/g23m-*: import from Magnetite
author | Mychaela Falconia <falcon@freecalypso.org> |
---|---|
date | Fri, 16 Oct 2020 06:25:50 +0000 |
parents | |
children |
comparison
equal
deleted
inserted
replaced
0:4e78acac3d88 | 1:fa8dc04885d8 |
---|---|
1 /* | |
2 +----------------------------------------------------------------------------- | |
3 | Project : GPRS (8441) | |
4 | Modul : GRR | |
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 module implements local functions for service PG of | |
18 | entity GRR. | |
19 +----------------------------------------------------------------------------- | |
20 */ | |
21 | |
22 #ifndef GRR_PGF_C | |
23 #define GRR_PGF_C | |
24 #endif | |
25 | |
26 #define ENTITY_GRR | |
27 | |
28 /*==== INCLUDES =============================================================*/ | |
29 | |
30 #include "typedefs.h" /* to get Condat data types */ | |
31 | |
32 #include "vsi.h" /* to get a lot of macros */ | |
33 #include "macdef.h" | |
34 #include "gprs.h" | |
35 #include "gsm.h" /* to get a lot of macros */ | |
36 #include "ccdapi.h" /* to get CCD API */ | |
37 #include "cnf_grr.h" /* to get cnf-definitions */ | |
38 #include "mon_grr.h" /* to get mon-definitions */ | |
39 #include "prim.h" /* to get the definitions of used SAP and directions */ | |
40 #include "message.h" | |
41 #include "grr.h" /* to get the global entity definitions */ | |
42 | |
43 #include "grr_f.h" /* */ | |
44 #include "grr_pgf.h" /* */ | |
45 | |
46 #include "grr_ctrls.h" /* */ | |
47 #include "grr_meass.h" | |
48 #include "grr_css.h" | |
49 #include "grr_em.h" /*for Engineering mode*/ | |
50 | |
51 #include <string.h> /* for memcpy*/ | |
52 #include <stdio.h> /* for memcpy*/ | |
53 | |
54 /*==== CONST ================================================================*/ | |
55 | |
56 const USHORT NC2_NON_DRX_TABLE[8] = {0, 240, 480, 720, 960, 1200, 1440, 1920}; | |
57 | |
58 /*==== LOCAL VARS ===========================================================*/ | |
59 static UBYTE NON_DRX_TABLE[] = {0, 1, 2, 4, 8, 16, 32, 64}; | |
60 | |
61 /*==== GLOBAL VARS ===========================================================*/ | |
62 | |
63 | |
64 | |
65 /*==== PRIVATE FUNCTIONS ====================================================*/ | |
66 LOCAL void pg_stop_t_nc2_ndrx( void ); | |
67 LOCAL void pg_stop_non_drx_timer( void ); | |
68 LOCAL void pg_start_non_drx_timer(USHORT milliseconds); | |
69 LOCAL BOOL pg_decode_imsi(T_ms_id ms_id, UBYTE *digits, UBYTE *nr_digits); | |
70 /*==== PUBLIC FUNCTIONS =====================================================*/ | |
71 | |
72 | |
73 | |
74 /* | |
75 +------------------------------------------------------------------------------ | |
76 | Function : pg_init | |
77 +------------------------------------------------------------------------------ | |
78 | Description : The function pg_init() .... | |
79 | | |
80 | Parameters : void | |
81 | | |
82 +------------------------------------------------------------------------------ | |
83 */ | |
84 GLOBAL void pg_init ( void ) | |
85 { | |
86 TRACE_FUNCTION( "pg_init" ); | |
87 | |
88 pg_init_params(); | |
89 | |
90 grr_data->pg.v_emlpp_prio = FALSE; /* valid flag for emlpp_priority*/ | |
91 grr_data->pg.packet_mode = PACKET_MODE_NULL; | |
92 | |
93 INIT_STATE(PG,PG_NULL); | |
94 } /* pg_init() */ | |
95 | |
96 /* | |
97 +------------------------------------------------------------------------------ | |
98 | Function : pg_init_params | |
99 +------------------------------------------------------------------------------ | |
100 | Description : The function pg_init_params() .... | |
101 | | |
102 | Parameters : void | |
103 | | |
104 +------------------------------------------------------------------------------ | |
105 */ | |
106 GLOBAL void pg_init_params ( void ) | |
107 { | |
108 TRACE_FUNCTION( "pg_init_params" ); | |
109 | |
110 psc_db->last_pg_mode = psc_db->network_pg_mode = REORG_PAGING; | |
111 | |
112 } /* pg_init_params() */ | |
113 | |
114 /* | |
115 +------------------------------------------------------------------------------ | |
116 | Function : pg_process_pp_req | |
117 +------------------------------------------------------------------------------ | |
118 | Description : The function pg_process_pp_req() .... | |
119 | | |
120 | Parameters : T_D_PAGING_REQ *paging_req: pointer to the decoded message, | |
121 UBYTE state: state of pg state machine: IDLE, TRANSFER, TIMER_3172 | |
122 | | |
123 +------------------------------------------------------------------------------ | |
124 */ | |
125 GLOBAL void pg_process_pp_req ( T_D_PAGING_REQ *paging_req, UBYTE state ) | |
126 { | |
127 char i; | |
128 ULONG ul; | |
129 | |
130 TRACE_FUNCTION( "pg_process_pp_req" ); | |
131 | |
132 if(paging_req->v_pers_lev) | |
133 { /*copy persistence level parameters into database*/ | |
134 grr_save_persistence_level (&(paging_req->pers_lev)); | |
135 } | |
136 /*if(paging_req->v_nln) | |
137 { message contains at least one page request for RR connection establishment | |
138 04.08: NLN(PCH) Notification List Number | |
139 "The presence of the NLN(PCH) field indicates that if an NCH is present, reduced NCH monitoring can be used, and | |
140 gives the NLN(PCH) value, to be used as specified in 3.3.3." | |
141 This is used by VGCS MSs... | |
142 | |
143 }*/ | |
144 if(paging_req->rep_page_info_trnc_grp.v_rep_page_info) | |
145 { | |
146 for (i=0; i < paging_req->rep_page_info_trnc_grp.c_rep_page_info; i++) | |
147 { | |
148 if(paging_req->rep_page_info_trnc_grp.rep_page_info[i].v_rep_page_s1 | |
149 AND | |
150 grr_t_status( T3172_1 ) EQ 0 /* T3172 is not running*/ | |
151 ) | |
152 { | |
153 /* | |
154 * page request for TBF establishment | |
155 */ | |
156 switch(state) | |
157 { | |
158 case PG_IDLE: | |
159 if(paging_req->rep_page_info_trnc_grp.rep_page_info[i].rep_page_s1.v_ptmsi) | |
160 { | |
161 /* | |
162 *Paging with PTMSI | |
163 */ | |
164 if(grr_data->db.ms_id.new_ptmsi NEQ GMMRR_TMSI_INVALID) | |
165 { | |
166 if(grr_check_ptmsi(&(paging_req->rep_page_info_trnc_grp.rep_page_info[i].rep_page_s1.ptmsi))) | |
167 { | |
168 TRACE_EVENT("Paging with PTIMSI for TBF"); | |
169 sig_pg_ctrl_downlink_transfer ( GMMRR_PTMSI ); | |
170 i = paging_req->rep_page_info_trnc_grp.c_rep_page_info; /* break condition for for loop*/ | |
171 } | |
172 else | |
173 { | |
174 TRACE_EVENT("Paging with wrong PTIMSI for TBF: IGNORE"); | |
175 } | |
176 } | |
177 else | |
178 { | |
179 TRACE_EVENT("Paging with PTMSI, but no valid PTMSI in MS"); | |
180 } | |
181 } | |
182 else | |
183 { | |
184 /* | |
185 *Paging with Mobile Identity (IMSI) | |
186 */ | |
187 UBYTE dummy_digits[MAX_IMSI_DIGITS]; | |
188 UBYTE dummy_nr_digits = 0; | |
189 | |
190 if(pg_decode_imsi(paging_req->rep_page_info_trnc_grp.rep_page_info[i].rep_page_s1.ms_id, | |
191 dummy_digits, &dummy_nr_digits)) | |
192 { | |
193 | |
194 if( dummy_nr_digits | |
195 EQ | |
196 grr_data->db.ms_id.imsi.number_of_digits)/* check the number of digits*/ | |
197 { | |
198 if( !memcmp | |
199 ( grr_data->db.ms_id.imsi.digit, /* IMSI stored in the data_base*/ | |
200 dummy_digits, /* received IMSI*/ | |
201 grr_data->db.ms_id.imsi.number_of_digits | |
202 ) | |
203 ) | |
204 { | |
205 /* The MS was paged with the IMSI*/ | |
206 TRACE_EVENT("Paging with Mobile Identity (IMSI) for TBF"); | |
207 sig_pg_ctrl_downlink_transfer ( GMMRR_IMSI ); | |
208 i = paging_req->rep_page_info_trnc_grp.c_rep_page_info; /* break condition for for loop*/ | |
209 } | |
210 else | |
211 { | |
212 TRACE_EVENT("Paging with wrong IMSI for TBF: IGNORE"); | |
213 } | |
214 | |
215 } | |
216 } | |
217 } | |
218 break; | |
219 default: | |
220 /* | |
221 * ignore paging request for TBF establishment, do | |
222 * not check the content | |
223 */ | |
224 break; | |
225 } | |
226 } | |
227 | |
228 if(paging_req->rep_page_info_trnc_grp.rep_page_info[i].v_rep_page_s2) | |
229 { | |
230 /* | |
231 * page request for RR connection establishment | |
232 */ | |
233 /* | |
234 * In packet access mode, class A or B shall respond to a pp_req for | |
235 * RR connection est. | |
236 * During the time T3172 is running, the mobile station shall ignore all | |
237 * received PACKET PAGING REQUEST messages except paging request to trigger | |
238 * RR connection establishment. | |
239 */ | |
240 switch(state) | |
241 { | |
242 case PG_IDLE: | |
243 case PG_TRANSFER: | |
244 case PG_ACCESS: | |
245 if(paging_req->rep_page_info_trnc_grp.rep_page_info[i].rep_page_s2.v_tmsi_field) | |
246 { | |
247 /* | |
248 *Paging with TMSI | |
249 */ | |
250 if(grr_data->db.ms_id.tmsi NEQ GMMRR_TMSI_INVALID) | |
251 { | |
252 ul = grr_buffer2ulong((BUF_ptmsi *)&(paging_req->rep_page_info_trnc_grp.rep_page_info[i].rep_page_s2.tmsi_field)); | |
253 if(ul EQ grr_data->db.ms_id.tmsi) | |
254 { | |
255 TRACE_EVENT("Paging with Mobile Identity (TMSI) for RR est"); | |
256 | |
257 /* Set some parameters needed RR connection establishment*/ | |
258 grr_data->pg.ch_needed = paging_req->rep_page_info_trnc_grp.rep_page_info[i].rep_page_s2.chan_need; | |
259 grr_data->pg.v_emlpp_prio = paging_req->rep_page_info_trnc_grp.rep_page_info[i].rep_page_s2.v_emlpp_prio; | |
260 grr_data->pg.emlpp_prio = paging_req->rep_page_info_trnc_grp.rep_page_info[i].rep_page_s2.emlpp_prio; | |
261 | |
262 /* MS was paged with TMSI for RR connection establishment*/ | |
263 grr_data->pg.type = ID_TMSI; | |
264 sig_pg_ctrl_rr_est_req ( PG_ON_PCCCH); | |
265 i = paging_req->rep_page_info_trnc_grp.c_rep_page_info; /* break condition for for loop*/ | |
266 } | |
267 else | |
268 { | |
269 TRACE_EVENT("Paging with wrong TMSI for RR: IGNORE"); | |
270 } | |
271 } | |
272 else | |
273 { | |
274 TRACE_EVENT("Paging with TMSI, but no valid TMSI in MS"); | |
275 } | |
276 } | |
277 else | |
278 { | |
279 /* | |
280 *Paging with Mobile Identity (IMSI) | |
281 */ | |
282 UBYTE dummy_digits[MAX_IMSI_DIGITS]; | |
283 UBYTE dummy_nr_digits = 0; | |
284 | |
285 if(pg_decode_imsi(paging_req->rep_page_info_trnc_grp.rep_page_info[i].rep_page_s2.ms_id, | |
286 dummy_digits, &dummy_nr_digits)) | |
287 { | |
288 | |
289 if( dummy_nr_digits | |
290 EQ | |
291 grr_data->db.ms_id.imsi.number_of_digits)/* check the number of digits*/ | |
292 { | |
293 if( !memcmp | |
294 ( grr_data->db.ms_id.imsi.digit, /* IMSI stored in the data_base*/ | |
295 dummy_digits, /* received IMSI*/ | |
296 grr_data->db.ms_id.imsi.number_of_digits | |
297 ) | |
298 ) | |
299 { | |
300 /* The MS was paged with the IMSI*/ | |
301 TRACE_EVENT("Paging with Mobile Identity (IMSI) for RR est"); | |
302 | |
303 /* Set some parameters needed RR connection establishment*/ | |
304 grr_data->pg.ch_needed = paging_req->rep_page_info_trnc_grp.rep_page_info[i].rep_page_s2.chan_need; | |
305 grr_data->pg.v_emlpp_prio = paging_req->rep_page_info_trnc_grp.rep_page_info[i].rep_page_s2.v_emlpp_prio; | |
306 grr_data->pg.emlpp_prio = paging_req->rep_page_info_trnc_grp.rep_page_info[i].rep_page_s2.emlpp_prio; | |
307 | |
308 /* MS was paged with IMSI for RR connection establishment*/ | |
309 grr_data->pg.type = ID_IMSI; | |
310 sig_pg_ctrl_rr_est_req ( PG_ON_PCCCH ); | |
311 i = paging_req->rep_page_info_trnc_grp.c_rep_page_info; /* break condition for for loop*/ | |
312 } | |
313 else | |
314 { | |
315 TRACE_EVENT("Paging with wrong IMSI for RR: IGNORE"); | |
316 } | |
317 } | |
318 } | |
319 } | |
320 break; | |
321 default: | |
322 /* | |
323 * ignore paging request for RR connection establishment, do | |
324 * not check the content | |
325 */ | |
326 break; | |
327 } | |
328 } | |
329 } | |
330 } | |
331 } /* pg_process_pp_req() */ | |
332 | |
333 | |
334 | |
335 /* | |
336 +------------------------------------------------------------------------------ | |
337 | Function : pg_send_stop | |
338 +------------------------------------------------------------------------------ | |
339 | Description : The function pg_send_stop() .... | |
340 | | |
341 | Parameters : void | |
342 | | |
343 +------------------------------------------------------------------------------ | |
344 */ | |
345 GLOBAL void pg_send_stop ( void ) | |
346 { | |
347 TRACE_FUNCTION( "pg_send_stop" ); | |
348 | |
349 if (grr_data->is_pg_started EQ TRUE) | |
350 { | |
351 PALLOC (mphp_stop_pccch_req, MPHP_STOP_PCCCH_REQ); | |
352 PSEND (hCommL1, mphp_stop_pccch_req); | |
353 } | |
354 grr_data->is_pg_started = FALSE; | |
355 | |
356 pg_stop_non_drx_timer(); | |
357 pg_stop_t_nc2_ndrx(); | |
358 | |
359 } /* pg_send_stop() */ | |
360 | |
361 | |
362 | |
363 /* | |
364 +------------------------------------------------------------------------------ | |
365 | Function : pg_send_start | |
366 +------------------------------------------------------------------------------ | |
367 | Description : The function pg_send_start() .... | |
368 | | |
369 | Parameters : UBYTE page_mode | |
370 | | |
371 +------------------------------------------------------------------------------ | |
372 */ | |
373 GLOBAL void pg_send_start ( UBYTE page_mode ) | |
374 { | |
375 UBYTE pccch_group; /* 0 .. KC-1*/ | |
376 USHORT dummy; | |
377 | |
378 TRACE_FUNCTION( "pg_send_start" ); | |
379 | |
380 if(grr_is_non_drx_period()) | |
381 page_mode = REORG_PAGING; | |
382 | |
383 switch(page_mode) | |
384 { | |
385 case NORMAL_PAGING: | |
386 case EXT_PAGING: | |
387 case REORG_PAGING: | |
388 case SAME_PAGING: | |
389 break;/*O.K.*/ | |
390 default: | |
391 /* same as before: nothing to do return*/ | |
392 TRACE_EVENT("page_mode is corrupt: assume same as before: return"); | |
393 return; | |
394 } | |
395 | |
396 if(psc_db->paging_group.kc EQ 0) | |
397 { | |
398 /* | |
399 * No PCCCH description: return | |
400 */ | |
401 TRACE_ERROR("psc_db->paging_group.kc EQ 0"); | |
402 return; | |
403 } | |
404 | |
405 { | |
406 PALLOC( start_pccch, MPHP_START_PCCCH_REQ ); | |
407 | |
408 start_pccch->imsi_mod = grr_imsi_mod( ); | |
409 pccch_group = ( UBYTE )( start_pccch->imsi_mod % psc_db->paging_group.kc ); | |
410 | |
411 if( !grr_get_pccch_freq_par( pccch_group, | |
412 &start_pccch->p_ch_des.p_chan_sel, | |
413 &start_pccch->p_freq_list ) ) | |
414 { | |
415 PFREE( start_pccch ); | |
416 | |
417 return; | |
418 } | |
419 | |
420 start_pccch->kcn = psc_db->paging_group.kc; /*< KC multiplied by N */ | |
421 start_pccch->split_pg_cycle = grr_data->ms.split_pg_cycle; /*< Split paging cycle */ | |
422 start_pccch->bs_pag_blks_res = psc_db->pccch.bs_pag_blks; /*< No paging no PBCCH */ | |
423 start_pccch->bs_pbcch_blks = psc_db->pccch.bs_pbcch_blks; /*< Number of PBCCH per multiframe */ | |
424 start_pccch->pb = psc_db->g_pwr_par.pb; /*< Power reduction value */ | |
425 start_pccch->p_ch_des.tsc = psc_db->paging_group.pccch[pccch_group].tsc; /*< Training sequence code */ | |
426 start_pccch->p_ch_des.tn = psc_db->paging_group.pccch[pccch_group].tn; | |
427 | |
428 if( start_pccch->p_ch_des.p_chan_sel.hopping ) | |
429 { | |
430 TRACE_EVENT_P8( "Hopping PCCCH - ma_num:%d hsn/maio:%ld tsc:%d tn:%d f1:%d f2:%d f3:%d f4:%d", | |
431 psc_db->paging_group.pccch[pccch_group].ma_num, | |
432 start_pccch->p_ch_des.p_chan_sel.p_rf_ch.arfcn, | |
433 start_pccch->p_ch_des.tsc, | |
434 start_pccch->p_ch_des.tn, | |
435 start_pccch->p_freq_list.p_rf_chan_no.p_radio_freq[0], | |
436 start_pccch->p_freq_list.p_rf_chan_no.p_radio_freq[1], | |
437 start_pccch->p_freq_list.p_rf_chan_no.p_radio_freq[2], | |
438 start_pccch->p_freq_list.p_rf_chan_no.p_radio_freq[3] ); | |
439 } | |
440 else | |
441 { | |
442 TRACE_EVENT_P3( "Static PCCCH - arfcn:%ld tsc:%d tn:%d", | |
443 start_pccch->p_ch_des.p_chan_sel.p_rf_ch.arfcn, | |
444 start_pccch->p_ch_des.tsc, | |
445 start_pccch->p_ch_des.tn ); | |
446 | |
447 start_pccch->p_ch_des.p_chan_sel.p_rf_ch.arfcn = | |
448 grr_g23_arfcn_to_l1( start_pccch->p_ch_des.p_chan_sel.p_rf_ch.arfcn ); | |
449 } | |
450 | |
451 /* | |
452 * Calculate downlink signalling counter | |
453 */ | |
454 /* DRX period is used, there is a valid SPILT_PG_CYCLE value */ | |
455 dummy = grr_data->ms.split_pg_cycle; | |
456 | |
457 | |
458 /* Downlink signalling counter*/ | |
459 dummy = ( ( 90 * dummy ) / 64 < 10 ) ? 10 : ( 90 * dummy ) / 64; | |
460 | |
461 if(!grr_data->is_pg_started) | |
462 { | |
463 grr_data->pg.initial_dsc = dummy; | |
464 grr_data->pg.dsc = dummy; | |
465 | |
466 GRR_EM_SET_DSC_VAL(dummy); | |
467 | |
468 } | |
469 if( grr_data->pg.is_l1_ref_tn_changed NEQ psc_db->paging_group.pccch[pccch_group].tn ) | |
470 { | |
471 sig_pg_cs_pause( ); | |
472 } | |
473 | |
474 /*Read PCCCH */ | |
475 { | |
476 start_pccch->page_mode = page_mode; | |
477 PSEND(hCommL1, start_pccch); | |
478 } | |
479 if( grr_data->pg.is_l1_ref_tn_changed NEQ psc_db->paging_group.pccch[pccch_group].tn ) | |
480 { | |
481 sig_pg_cs_resume( ); | |
482 | |
483 grr_data->pg.is_l1_ref_tn_changed = psc_db->paging_group.pccch[pccch_group].tn; | |
484 } | |
485 | |
486 psc_db->last_pg_mode = page_mode; | |
487 | |
488 switch(page_mode) | |
489 { | |
490 case NORMAL_PAGING: | |
491 TRACE_EVENT("page mode NORMAL"); | |
492 break;/*O.K.*/ | |
493 case EXT_PAGING: | |
494 TRACE_EVENT("page mode EXT"); | |
495 break;/*O.K.*/ | |
496 case REORG_PAGING: | |
497 TRACE_EVENT("page mode REORG"); | |
498 break;/*O.K.*/ | |
499 case SAME_PAGING: | |
500 TRACE_EVENT("page mode SAME"); | |
501 break;/*O.K.*/ | |
502 } | |
503 | |
504 } | |
505 grr_data->is_pg_started = TRUE; | |
506 | |
507 } /* pg_send_start() */ | |
508 | |
509 /* | |
510 +------------------------------------------------------------------------------ | |
511 | Function : pg_start_non_drx_timer | |
512 +------------------------------------------------------------------------------ | |
513 | Description : The function pg_start_non_drx_timer() starts the timer for | |
514 | non DRX period | |
515 | | |
516 | Parameters : USHORT milliseconds | |
517 | | |
518 +------------------------------------------------------------------------------ | |
519 */ | |
520 LOCAL void pg_start_non_drx_timer(USHORT milliseconds) | |
521 { | |
522 TRACE_FUNCTION( "pg_start_non_drx_timer"); | |
523 | |
524 psc_db->non_drx_timer_running = TRUE; | |
525 vsi_t_start(GRR_handle,T_TRANS_NON_DRX, milliseconds); | |
526 }/* pg_start_non_drx_timer*/ | |
527 | |
528 /* | |
529 +------------------------------------------------------------------------------ | |
530 | Function : pg_stop_non_drx_timer | |
531 +------------------------------------------------------------------------------ | |
532 | Description : The function pg_stop_non_drx_timer() stops the timer for | |
533 | non DRX period | |
534 | | |
535 | Parameters : void | |
536 | | |
537 +------------------------------------------------------------------------------ | |
538 */ | |
539 GLOBAL void pg_stop_non_drx_timer(void) | |
540 { | |
541 TRACE_FUNCTION( "pg_stop_non_drx_timer"); | |
542 | |
543 if(psc_db->non_drx_timer_running) | |
544 { | |
545 psc_db->non_drx_timer_running = FALSE; | |
546 vsi_t_stop(GRR_handle,T_TRANS_NON_DRX); | |
547 } | |
548 }/* pg_stop_non_drx_timer*/ | |
549 | |
550 /* | |
551 +------------------------------------------------------------------------------ | |
552 | Function : pg_start_t_nc2_ndrx | |
553 +------------------------------------------------------------------------------ | |
554 | Description : ... | |
555 | | |
556 | Parameters : void | |
557 | | |
558 +------------------------------------------------------------------------------ | |
559 */ | |
560 GLOBAL void pg_start_t_nc2_ndrx( UBYTE nc_non_drx_period ) | |
561 { | |
562 TRACE_FUNCTION( "pg_start_t_nc2_ndrx"); | |
563 | |
564 if( nc_non_drx_period NEQ 0 ) | |
565 { | |
566 USHORT nc2_drx_period = NC2_NON_DRX_TABLE[nc_non_drx_period]; | |
567 | |
568 psc_db->nc2_non_drx_period_running = TRUE; | |
569 | |
570 vsi_t_start( GRR_handle, T_NC2_NON_DRX, nc2_drx_period ); | |
571 | |
572 TRACE_EVENT_P1( "Timer T_NC2_NON_DRX started: %d", nc2_drx_period ); | |
573 } | |
574 }/* pg_start_t_nc2_ndrx */ | |
575 | |
576 /* | |
577 +------------------------------------------------------------------------------ | |
578 | Function : pg_stop_t_nc2_ndrx | |
579 +------------------------------------------------------------------------------ | |
580 | Description : ... | |
581 | | |
582 | Parameters : void | |
583 | | |
584 +------------------------------------------------------------------------------ | |
585 */ | |
586 GLOBAL void pg_stop_t_nc2_ndrx( void ) | |
587 { | |
588 TRACE_FUNCTION( "pg_stop_t_nc2_ndrx" ); | |
589 | |
590 if( psc_db->nc2_non_drx_period_running EQ TRUE ) | |
591 { | |
592 psc_db->nc2_non_drx_period_running = FALSE; | |
593 | |
594 vsi_t_stop( GRR_handle, T_NC2_NON_DRX ); | |
595 | |
596 TRACE_EVENT( "Timer T_NC2_NON_DRX stopped" ); | |
597 } | |
598 } /* pg_stop_t_nc2_ndrx */ | |
599 | |
600 /* | |
601 +------------------------------------------------------------------------------ | |
602 | Function : pg_handle_non_drx_timers | |
603 +------------------------------------------------------------------------------ | |
604 | Description : ... | |
605 | | |
606 | Parameters : void | |
607 | | |
608 +------------------------------------------------------------------------------ | |
609 */ | |
610 GLOBAL void pg_handle_non_drx_timers(void) | |
611 { | |
612 USHORT non_drx_timer = 0; | |
613 TRACE_FUNCTION("pg_handle_non_drx_timers"); | |
614 | |
615 non_drx_timer = ( psc_db->non_drx_timer < psc_db->gprs_cell_opt.drx_t_max ) ? | |
616 psc_db->non_drx_timer : psc_db->gprs_cell_opt.drx_t_max; | |
617 | |
618 non_drx_timer = (USHORT)(NON_DRX_TABLE[non_drx_timer]*1000);/* in milliseconds*/ | |
619 | |
620 pg_start_non_drx_timer(non_drx_timer); | |
621 | |
622 }/* pg_handle_non_drx_timers */ | |
623 | |
624 /* | |
625 +------------------------------------------------------------------------------ | |
626 | Function : pg_non_drx | |
627 +------------------------------------------------------------------------------ | |
628 | Description : Handles the timer of non DRX mode | |
629 | | |
630 | Parameters : void | |
631 | | |
632 +------------------------------------------------------------------------------ | |
633 */ | |
634 GLOBAL void pg_non_drx ( void ) | |
635 { | |
636 TRACE_FUNCTION( "pg_non_drx" ); | |
637 | |
638 switch( GET_STATE( PG ) ) | |
639 { | |
640 case PG_IDLE: | |
641 /* | |
642 * To ensure that the paging mode is not set to page normal in other states | |
643 * than packet idle mode, the function grr_is_packet_idle_mode is called in | |
644 * advance. | |
645 * | |
646 * State PG_IDLE not necessarily means that GRR is in packet idle mode. In | |
647 * case GRR is currently executing the RRGRR_STOP_TASK_REQ/CNF procedure, | |
648 * just the services CPAP and TC changes their states. All other services | |
649 * remain in state IDLE, which is not quite correct. A similar scenario | |
650 * occurs during execution of the MPHP_ASSIGNMENT_REQ/CON procedure. | |
651 */ | |
652 if( grr_is_packet_idle_mode( ) EQ TRUE ) | |
653 { | |
654 pg_handle_idle_mode( PTM_IDLE ); | |
655 } | |
656 else | |
657 { | |
658 TRACE_EVENT( "pg_non_drx: Timeout T_TRANS_NON_DRX handled by next packet idle mode" ); | |
659 } | |
660 break; | |
661 default: | |
662 TRACE_EVENT( "pg_non_drx: Timeout T_TRANS_NON_DRX handled by next PG_IDLE" ); | |
663 break; | |
664 } | |
665 }/* pg_non_drx*/ | |
666 | |
667 /* | |
668 +------------------------------------------------------------------------------ | |
669 | Function : pg_decode_imsi | |
670 +------------------------------------------------------------------------------ | |
671 | Description : Handles the timer of non DRX mode | |
672 | | |
673 | Parameters : void | |
674 | | |
675 +------------------------------------------------------------------------------ | |
676 */ | |
677 LOCAL BOOL pg_decode_imsi(T_ms_id ms_id, UBYTE *digits, UBYTE *nr_digits) | |
678 { | |
679 UBYTE odd_even_type, i; | |
680 BOOL is_odd; | |
681 TRACE_FUNCTION("pg_decode_imsi"); | |
682 | |
683 *nr_digits = 0; | |
684 if(ms_id.ms_id_len > 0) | |
685 { | |
686 odd_even_type = (ms_id.ident_digit[0] & 0x0f); | |
687 | |
688 if(odd_even_type & 1) /* Check IMSI or not*/ | |
689 { | |
690 is_odd = odd_even_type & 0x08; /* odd: ung.*/ | |
691 digits[0] = (ms_id.ident_digit[0] >> 4); | |
692 *nr_digits = 1; | |
693 | |
694 for(i=1; i< ms_id.ms_id_len; i++) | |
695 { | |
696 if(!is_odd AND (i EQ (ms_id.ms_id_len-1))) | |
697 { | |
698 digits[*nr_digits] = (ms_id.ident_digit[i] & 0x0f); | |
699 *nr_digits = (*nr_digits) + 1; | |
700 } | |
701 else | |
702 { | |
703 digits[*nr_digits] = (ms_id.ident_digit[i] & 0x0f); | |
704 *nr_digits = (*nr_digits) + 1; | |
705 digits[*nr_digits] = (ms_id.ident_digit[i] >> 4); | |
706 *nr_digits = (*nr_digits) + 1; | |
707 } | |
708 } | |
709 } | |
710 else | |
711 { | |
712 /*No IMSI, ignore*/ | |
713 return FALSE; | |
714 } | |
715 } | |
716 else | |
717 { | |
718 /* No mobile identity: strange*/ | |
719 return FALSE; | |
720 } | |
721 return TRUE; | |
722 }/* pg_decode_imsi*/ | |
723 | |
724 /* | |
725 +------------------------------------------------------------------------------ | |
726 | Function : pg_handle_classB_NMO_III | |
727 +------------------------------------------------------------------------------ | |
728 | Description : Handles NMO III for Class B mobile | |
729 | | |
730 | Parameters : void | |
731 | | |
732 +------------------------------------------------------------------------------ | |
733 */ | |
734 GLOBAL void pg_handle_classB_NMO_III(T_PG_TRANS transition) | |
735 { | |
736 TRACE_FUNCTION("pg_handle_classB_NMO_III"); | |
737 | |
738 switch(transition) | |
739 { | |
740 case T3172_PTM: | |
741 case ACCESS_PTM: | |
742 case IDLE_PTM: | |
743 if(grr_data->ms.reverts_NMO_III) | |
744 { | |
745 /* | |
746 * The MS reverts to CC or CG in NMO III | |
747 * In this state IDLE->TRANSFER it can be only a CG mobile | |
748 * it shall stop PCCCH reading, if available | |
749 * it shall stop also CCCH reading, if running | |
750 */ | |
751 TRACE_EVENT("IDLE/ACCESS->PTM: NMO III Class B reverts. Stop CCCH (and PCCCH)"); | |
752 if(grr_is_pbcch_present()) | |
753 { | |
754 /* | |
755 * Stop monitoring PCCCH | |
756 */ | |
757 pg_send_stop(); | |
758 } | |
759 /* | |
760 * Stop monitoring CCCH | |
761 */ | |
762 sig_pg_ctrl_stop_mon_ccch(); | |
763 } | |
764 else | |
765 { | |
766 /* | |
767 * This is non reverting Class B | |
768 * If PBCCH is present it should stop PCCCH and start CCCH | |
769 * else it should only monitor CCCH | |
770 */ | |
771 TRACE_EVENT("IDLE/ACCESS->PTM: NMO III Class B. Start CCCH (and stop PCCCH)"); | |
772 if(grr_is_pbcch_present()) | |
773 { | |
774 /* | |
775 * Stop monitoring PCCCH | |
776 */ | |
777 pg_send_stop(); | |
778 } | |
779 /* | |
780 * Start monitoring CCCH (Normal Paging) | |
781 */ | |
782 if( grr_data->ms.tbf_mon_ccch | |
783 #ifdef REL99 | |
784 AND | |
785 !psc_db->gprs_cell_opt.gprs_ext_bits.gprs_ext_info.bss_paging_coord | |
786 #endif | |
787 ) | |
788 { | |
789 sig_pg_ctrl_start_mon_ccch(PAG_MODE_PTM_NP); | |
790 } | |
791 } | |
792 break; | |
793 case NULL_IDLE: | |
794 case PTM_IDLE: | |
795 if(grr_data->ms.reverts_NMO_III) | |
796 { | |
797 /* | |
798 * The MS reverts to CC or CG in NMO III | |
799 * In this state TRANSFER/NULL->IDLE it can be only a CG mobile | |
800 * it shall start PCCCH reading, if available | |
801 * it shall start also CCCH reading, if NO PBCCH | |
802 */ | |
803 TRACE_EVENT("PTM/NULL->IDLE: NMO III Class B reverts. Start (P) or CCCH"); | |
804 if(grr_is_pbcch_present()) | |
805 { | |
806 /* | |
807 * Start monitoring PCCCH | |
808 */ | |
809 pg_send_start( psc_db->last_pg_mode ); | |
810 | |
811 } | |
812 else | |
813 { | |
814 /* | |
815 * Start monitoring CCCH (Normal Paging) | |
816 */ | |
817 sig_pg_ctrl_start_mon_ccch(PAG_MODE_DEFAULT); | |
818 } | |
819 } | |
820 else | |
821 { | |
822 /* | |
823 * This is non reverting Class B | |
824 * If PBCCH is present it should start PCCCH and start CCCH | |
825 * else it should only monitor CCCH | |
826 */ | |
827 TRACE_EVENT("PTM/NULL->IDLE: NMO III Class B. Start (P) and CCCH"); | |
828 if(grr_is_pbcch_present()) | |
829 { | |
830 /* | |
831 * Start monitoring PCCCH | |
832 */ | |
833 pg_send_start( psc_db->last_pg_mode ); | |
834 /* | |
835 * Start monitoring CCCH (Normal Paging), before doing this use PAG_MODE_PIM_NP | |
836 * to enter later to idle mode | |
837 * Exceptional situation: if NON DRX is used, REORG is only for PCCCH | |
838 */ | |
839 sig_pg_ctrl_start_mon_ccch(PAG_MODE_PIM_NP); | |
840 } | |
841 else | |
842 { | |
843 /* | |
844 * Start monitoring CCCH (Normal Paging) | |
845 * Normal Paging | |
846 */ | |
847 sig_pg_ctrl_start_mon_ccch(PAG_MODE_DEFAULT); | |
848 } | |
849 } | |
850 break; | |
851 case ACCESS_IDLE: | |
852 if(grr_data->ms.reverts_NMO_III) | |
853 { | |
854 /* | |
855 * The MS reverts to CC or CG in NMO III | |
856 * In this state ACCESS->IDLE it can be only a CG mobile | |
857 * it shall start PCCCH reading, if available | |
858 * it shall start CCCH reading, if NO PBCCH | |
859 */ | |
860 TRACE_EVENT("ACCESS->IDLE: NMO III Class B reverts. Start (P) or CCCH"); | |
861 if(!grr_is_pbcch_present()) | |
862 { | |
863 /* | |
864 * Start monitoring CCCH (Normal Paging) | |
865 */ | |
866 sig_pg_ctrl_start_mon_ccch(PAG_MODE_DEFAULT); | |
867 } | |
868 } | |
869 else | |
870 { | |
871 /* | |
872 * This is non reverting Class B | |
873 * If PBCCH is present it should start PCCCH and start CCCH | |
874 * else it should only monitor CCCH | |
875 */ | |
876 TRACE_EVENT("ACCESS->IDLE: NMO III Class B. Start (P) and CCCH"); | |
877 if(grr_is_pbcch_present()) | |
878 { | |
879 /* | |
880 * Start monitoring PCCCH | |
881 */ | |
882 pg_send_start( psc_db->last_pg_mode ); | |
883 /* | |
884 * Start monitoring CCCH (Normal Paging) | |
885 * Exceptional situation: if NON DRX is used, we RORG is only for PCCCH | |
886 */ | |
887 if(grr_data->ms.tbf_mon_ccch | |
888 #ifdef REL99 | |
889 AND | |
890 !psc_db->gprs_cell_opt.gprs_ext_bits.gprs_ext_info.bss_paging_coord | |
891 #endif | |
892 ) | |
893 { | |
894 sig_pg_ctrl_start_mon_ccch(PAG_MODE_PTM_NP); | |
895 } | |
896 } | |
897 else | |
898 { | |
899 /* | |
900 * Start monitoring CCCH (Normal Paging) | |
901 * Normal Paging | |
902 */ | |
903 sig_pg_ctrl_start_mon_ccch(PAG_MODE_DEFAULT); | |
904 } | |
905 } | |
906 break; | |
907 case PTM_ACCESS: | |
908 case IDLE_ACCESS: | |
909 /* | |
910 * Not necessary | |
911 */ | |
912 break; | |
913 }/*switch*/ | |
914 } | |
915 | |
916 /* | |
917 +------------------------------------------------------------------------------ | |
918 | Function : pg_handle_access_mode | |
919 +------------------------------------------------------------------------------ | |
920 | Description : Handles the paging when entering the packet access mode. | |
921 | | |
922 | Parameters : void | |
923 | | |
924 +------------------------------------------------------------------------------ | |
925 */ | |
926 GLOBAL void pg_handle_access_mode ( void ) | |
927 { | |
928 TRACE_FUNCTION( "pg_handle_access_mode" ); | |
929 | |
930 | |
931 if( grr_is_pbcch_present( ) ) | |
932 { | |
933 if( !grr_data->is_pg_started OR psc_db->last_pg_mode NEQ REORG_PAGING ) | |
934 { | |
935 /* | |
936 * enable the L1 reading USF values | |
937 */ | |
938 pg_send_start( PG_REORG ); | |
939 } | |
940 } | |
941 else | |
942 { | |
943 /* | |
944 * start monitoring CCCH | |
945 */ | |
946 sig_pg_ctrl_start_mon_ccch( PAG_MODE_REORG ); | |
947 } | |
948 } /* pg_handle_access_mode() */ | |
949 | |
950 /* | |
951 +------------------------------------------------------------------------------ | |
952 | Function : pg_set_l1_ref_tn | |
953 +------------------------------------------------------------------------------ | |
954 | Description : The function pg_set_l1_ref_tn () .... | |
955 | | |
956 | Parameters : | |
957 | | |
958 +------------------------------------------------------------------------------ | |
959 */ | |
960 GLOBAL void pg_set_l1_ref_tn ( UBYTE* out_tn, UBYTE in_tn ) | |
961 { | |
962 TRACE_FUNCTION( "pg_set_l1_ref_tn" ); | |
963 | |
964 if( in_tn NEQ *out_tn ) | |
965 { | |
966 grr_data->pg.is_l1_ref_tn_changed = TRUE; | |
967 } | |
968 | |
969 *out_tn = in_tn; | |
970 | |
971 } /* pg_set_l1_ref_tn */ | |
972 | |
973 /* | |
974 +------------------------------------------------------------------------------ | |
975 | Function : pg_handle_idle_mode | |
976 +------------------------------------------------------------------------------ | |
977 | Description : | |
978 | | |
979 | Parameters : | |
980 | | |
981 +------------------------------------------------------------------------------ | |
982 */ | |
983 GLOBAL void pg_handle_idle_mode ( T_PG_TRANS transition ) | |
984 { | |
985 TRACE_FUNCTION( "pg_handle_idle_mode" ); | |
986 | |
987 switch( grr_data->pg.nmo ) | |
988 { | |
989 case GMMRR_NET_MODE_I: | |
990 /* | |
991 * NMOI: | |
992 * ==== | |
993 * Paging on ( CCCH or PCCCH ) and PACCH, GPRS attached MS should | |
994 * monitor one paging channel, so we have to stop monitoring CCCH | |
995 * channel if PCCCH is present. | |
996 */ | |
997 if( grr_is_pbcch_present( ) ) | |
998 { | |
999 pg_send_start( psc_db->last_pg_mode ); | |
1000 } | |
1001 else | |
1002 { | |
1003 sig_pg_ctrl_start_mon_ccch( PAG_MODE_DEFAULT ); | |
1004 } | |
1005 break; | |
1006 | |
1007 case GMMRR_NET_MODE_II: | |
1008 /* | |
1009 * NMOII: | |
1010 * ===== | |
1011 * Paging on CCCH, paging for both circuit switched and packet data | |
1012 * will be sent on CCCH, PCCCH has been not allocated in the cell. | |
1013 */ | |
1014 switch( grr_data->ms.class_of_mode ) | |
1015 { | |
1016 case GMMRR_CLASS_A: | |
1017 /* BEGIN GRR_NMO_III */ | |
1018 case GMMRR_CLASS_B: | |
1019 /* END GRR_NMO_III */ | |
1020 case GMMRR_CLASS_BG: | |
1021 case GMMRR_CLASS_BC: | |
1022 case GMMRR_CLASS_CG: | |
1023 case GMMRR_CLASS_CC: | |
1024 sig_pg_ctrl_start_mon_ccch( PAG_MODE_DEFAULT ); | |
1025 break; | |
1026 } | |
1027 break; | |
1028 | |
1029 case GMMRR_NET_MODE_III: | |
1030 /* | |
1031 * NMOIII: | |
1032 * ======= | |
1033 * If PBCCH not present paging for both circuit switched and packet | |
1034 * data will be sent on CCCH, | |
1035 * if PBCCH is present paging for circuit switched data is sent on | |
1036 * CCCH and for packet data on PCCCH. | |
1037 */ | |
1038 if( grr_is_pbcch_present( ) ) | |
1039 { | |
1040 TRACE_EVENT( "GMMRR_NET_MODE_III PBCCH" ); | |
1041 | |
1042 switch(grr_data->ms.class_of_mode) | |
1043 { | |
1044 case GMMRR_CLASS_A: | |
1045 pg_send_start( psc_db->last_pg_mode ); | |
1046 sig_pg_ctrl_start_mon_ccch( PAG_MODE_PIM_NP ); | |
1047 break; | |
1048 | |
1049 /* | |
1050 * We need to check whether the MS class B mobile | |
1051 * reverts to CG or CC in NMO III. If not it should monitor PCCCH and CCCH | |
1052 * else it needs to monitor either CCCH or PCCCH | |
1053 */ | |
1054 /* BEGIN GRR_NMO_III */ | |
1055 case GMMRR_CLASS_B: | |
1056 /* END GRR_NMO_III */ | |
1057 case GMMRR_CLASS_BG: | |
1058 case GMMRR_CLASS_BC: | |
1059 pg_handle_classB_NMO_III( transition ); | |
1060 break; | |
1061 | |
1062 case GMMRR_CLASS_CG: | |
1063 sig_pg_ctrl_stop_mon_ccch( ); | |
1064 pg_send_start( psc_db->last_pg_mode ); | |
1065 break; | |
1066 | |
1067 case GMMRR_CLASS_CC: | |
1068 sig_pg_ctrl_start_mon_ccch( PAG_MODE_DEFAULT ); | |
1069 break; | |
1070 } | |
1071 } | |
1072 else | |
1073 { | |
1074 TRACE_EVENT ( "GMMRR_NET_MODE_III BCCH" ); | |
1075 | |
1076 sig_pg_ctrl_start_mon_ccch( PAG_MODE_DEFAULT ); | |
1077 } | |
1078 break; | |
1079 } | |
1080 } /* pg_handle_idle_mode() */ | |
1081 | |
1082 /* | |
1083 +------------------------------------------------------------------------------ | |
1084 | Function : pg_handle_transfer_mode | |
1085 +------------------------------------------------------------------------------ | |
1086 | Description : | |
1087 | | |
1088 | Parameters : | |
1089 | | |
1090 +------------------------------------------------------------------------------ | |
1091 */ | |
1092 GLOBAL void pg_handle_transfer_mode ( T_PG_TRANS transition ) | |
1093 { | |
1094 TRACE_FUNCTION( "pg_handle_transfer_mode" ); | |
1095 | |
1096 switch( grr_data->pg.nmo ) | |
1097 { | |
1098 case GMMRR_NET_MODE_I: | |
1099 /* | |
1100 * NMOI: | |
1101 * ==== | |
1102 * Paging on ( CCCH or PCCCH ) and PACCH, GPRS attached MS should | |
1103 * monitor one paging channel, so we have to stop monitoring CCCH | |
1104 * channel if PCCCH is present. | |
1105 */ | |
1106 if( grr_is_pbcch_present( ) ) | |
1107 { | |
1108 pg_send_stop( ); | |
1109 } | |
1110 else | |
1111 { | |
1112 sig_pg_ctrl_stop_mon_ccch( ); | |
1113 } | |
1114 break; | |
1115 | |
1116 case GMMRR_NET_MODE_II: | |
1117 /* | |
1118 * NMOII: | |
1119 * ===== | |
1120 * Paging on CCCH, paging for both circuit switched and packet data | |
1121 * will be sent on CCCH, PCCCH has been not allocated in the cell. | |
1122 * | |
1123 * The MS should monitor CCCH except for the CLASS_CG. | |
1124 */ | |
1125 switch( grr_data->ms.class_of_mode ) | |
1126 { | |
1127 case GMMRR_CLASS_A: | |
1128 /* BEGIN GRR_NMO_III*/ | |
1129 case GMMRR_CLASS_B: | |
1130 /* END GRR_NMO_III*/ | |
1131 case GMMRR_CLASS_BG: | |
1132 case GMMRR_CLASS_BC: | |
1133 if( grr_data->ms.tbf_mon_ccch | |
1134 #ifdef REL99 | |
1135 AND | |
1136 !psc_db->gprs_cell_opt.gprs_ext_bits.gprs_ext_info.bss_paging_coord | |
1137 #endif | |
1138 ) | |
1139 { | |
1140 sig_pg_ctrl_start_mon_ccch( PAG_MODE_PTM_NP ); | |
1141 } | |
1142 else | |
1143 { | |
1144 sig_pg_ctrl_stop_mon_ccch( ); | |
1145 } | |
1146 break; | |
1147 | |
1148 case GMMRR_CLASS_CG: | |
1149 sig_pg_ctrl_stop_mon_ccch( ); | |
1150 break; | |
1151 } | |
1152 break; | |
1153 | |
1154 case GMMRR_NET_MODE_III: | |
1155 /* | |
1156 * NMOIII: | |
1157 * ======= | |
1158 * If PBCCH not present paging for both circuit switched and packet | |
1159 * data will be sent on CCCH, | |
1160 * if PBCCH is present paging for circuit switched data is sent on | |
1161 * CCCH and for packet data on PCCCH. | |
1162 */ | |
1163 switch( grr_data->ms.class_of_mode ) | |
1164 { | |
1165 case GMMRR_CLASS_A: | |
1166 if( grr_is_pbcch_present( ) ) | |
1167 { | |
1168 pg_send_stop( ); | |
1169 } | |
1170 | |
1171 sig_pg_ctrl_start_mon_ccch( PAG_MODE_PTM_NP ); | |
1172 break; | |
1173 | |
1174 /* BEGIN GRR_NMO_III*/ | |
1175 case GMMRR_CLASS_B: | |
1176 /* END GRR_NMO_III*/ | |
1177 case GMMRR_CLASS_BG: | |
1178 case GMMRR_CLASS_BC: | |
1179 pg_handle_classB_NMO_III( transition ); | |
1180 break; | |
1181 | |
1182 case GMMRR_CLASS_CG: | |
1183 if( grr_is_pbcch_present( ) ) | |
1184 { | |
1185 pg_send_stop( ); | |
1186 } | |
1187 else | |
1188 { | |
1189 sig_pg_ctrl_stop_mon_ccch( ); | |
1190 } | |
1191 break; | |
1192 } | |
1193 break; | |
1194 } | |
1195 } /* pg_handle_transfer_mode() */ |