1 /****************************************************************************** 2 * 3 * Copyright (C) 2010-2013 Broadcom Corporation 4 * 5 * Licensed under the Apache License, Version 2.0 (the "License"); 6 * you may not use this file except in compliance with the License. 7 * You may obtain a copy of the License at: 8 * 9 * http://www.apache.org/licenses/LICENSE-2.0 10 * 11 * Unless required by applicable law or agreed to in writing, software 12 * distributed under the License is distributed on an "AS IS" BASIS, 13 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 14 * See the License for the specific language governing permissions and 15 * limitations under the License. 16 * 17 ******************************************************************************/ 18 19 20 /****************************************************************************** 21 * 22 * This is the main implementation file for the NFA system manager. 23 * 24 ******************************************************************************/ 25 #include <string.h> 26 #include "nfa_api.h" 27 #include "nfa_sys.h" 28 #include "nfa_sys_int.h" 29 #include "nfa_sys_ptim.h" 30 #include "nfa_dm_int.h" 31 32 /* protocol timer update period, in milliseconds */ 33 #ifndef NFA_SYS_TIMER_PERIOD 34 #define NFA_SYS_TIMER_PERIOD 10 35 #endif 36 37 /* system manager control block definition */ 38 #if NFA_DYNAMIC_MEMORY == FALSE 39 tNFA_SYS_CB nfa_sys_cb = {0}; /* nfa_sys control block. statically initialize 'flags' field to 0 */ 40 #endif 41 42 /******************************************************************************* 43 ** 44 ** Function nfa_sys_init 45 ** 46 ** Description NFA initialization; called from task initialization. 47 ** 48 ** 49 ** Returns void 50 ** 51 *******************************************************************************/ 52 void nfa_sys_init (void) 53 { 54 memset (&nfa_sys_cb, 0, sizeof (tNFA_SYS_CB)); 55 nfa_sys_cb.flags |= NFA_SYS_FL_INITIALIZED; 56 nfa_sys_ptim_init (&nfa_sys_cb.ptim_cb, NFA_SYS_TIMER_PERIOD, p_nfa_sys_cfg->timer); 57 nfa_sys_cb.trace_level = p_nfa_sys_cfg->trace_level; 58 } 59 60 61 62 63 /******************************************************************************* 64 ** 65 ** Function nfa_sys_event 66 ** 67 ** Description BTA event handler; called from task event handler. 68 ** 69 ** 70 ** Returns void 71 ** 72 *******************************************************************************/ 73 void nfa_sys_event (BT_HDR *p_msg) 74 { 75 UINT8 id; 76 BOOLEAN freebuf = TRUE; 77 78 NFA_TRACE_EVENT1 ("NFA got event 0x%04X", p_msg->event); 79 80 /* get subsystem id from event */ 81 id = (UINT8) (p_msg->event >> 8); 82 83 /* verify id and call subsystem event handler */ 84 if ((id < NFA_ID_MAX) && (nfa_sys_cb.is_reg[id])) 85 { 86 freebuf = (*nfa_sys_cb.reg[id]->evt_hdlr) (p_msg); 87 } 88 else 89 { 90 NFA_TRACE_WARNING1 ("NFA got unregistered event id %d", id); 91 } 92 93 if (freebuf) 94 { 95 GKI_freebuf (p_msg); 96 } 97 } 98 99 /******************************************************************************* 100 ** 101 ** Function nfa_sys_timer_update 102 ** 103 ** Description Update the BTA timer list and handle expired timers. 104 ** 105 ** Returns void 106 ** 107 *******************************************************************************/ 108 void nfa_sys_timer_update (void) 109 { 110 if (!nfa_sys_cb.timers_disabled) 111 { 112 nfa_sys_ptim_timer_update (&nfa_sys_cb.ptim_cb); 113 } 114 } 115 116 /******************************************************************************* 117 ** 118 ** Function nfa_sys_register 119 ** 120 ** Description Called by other BTA subsystems to register their event 121 ** handler. 122 ** 123 ** 124 ** Returns void 125 ** 126 *******************************************************************************/ 127 void nfa_sys_register (UINT8 id, const tNFA_SYS_REG *p_reg) 128 { 129 nfa_sys_cb.reg[id] = (tNFA_SYS_REG *) p_reg; 130 nfa_sys_cb.is_reg[id] = TRUE; 131 132 if ((id != NFA_ID_DM) && (id != NFA_ID_SYS)) 133 nfa_sys_cb.enable_cplt_mask |= (0x0001 << id); 134 135 if (id != NFA_ID_SYS) 136 { 137 if (p_reg->proc_nfcc_pwr_mode) 138 nfa_sys_cb.proc_nfcc_pwr_mode_cplt_mask |= (0x0001 << id); 139 } 140 141 NFA_TRACE_DEBUG2 ("nfa_sys_register () id=%i, enable_cplt_mask=0x%x", 142 id, nfa_sys_cb.enable_cplt_mask); 143 } 144 145 146 /******************************************************************************* 147 ** 148 ** Function nfa_sys_check_disabled 149 ** 150 ** Description If all subsystems above DM have been disabled, then 151 ** disable DM. Called during NFA shutdown 152 ** 153 ** Returns void 154 ** 155 *******************************************************************************/ 156 void nfa_sys_check_disabled (void) 157 { 158 UINT8 id; 159 UINT8 done = TRUE; 160 161 /* Check if all subsystems above DM have been disabled. */ 162 for (id = (NFA_ID_DM+1); id < NFA_ID_MAX; id++) 163 { 164 if (nfa_sys_cb.is_reg[id]) 165 { 166 /* as long as one subsystem is not done */ 167 done = FALSE; 168 break; 169 } 170 } 171 172 /* All subsystems disabled. disable DM */ 173 if ((done) && (nfa_sys_cb.is_reg[NFA_ID_DM])) 174 { 175 (*nfa_sys_cb.reg[NFA_ID_DM]->disable) (); 176 } 177 } 178 179 180 /******************************************************************************* 181 ** 182 ** Function nfa_sys_deregister 183 ** 184 ** Description Called by other BTA subsystems to de-register 185 ** handler. 186 ** 187 ** 188 ** Returns void 189 ** 190 *******************************************************************************/ 191 void nfa_sys_deregister (UINT8 id) 192 { 193 NFA_TRACE_DEBUG1 ("nfa_sys: deregistering subsystem %i", id); 194 195 nfa_sys_cb.is_reg[id] = FALSE; 196 197 /* If not deregistering DM, then check if any other subsystems above DM are still */ 198 /* registered. */ 199 if (id != NFA_ID_DM) 200 { 201 /* If all subsystems above NFA_DM have been disabled, then okay to disable DM */ 202 nfa_sys_check_disabled (); 203 } 204 else 205 { 206 /* DM (the final sub-system) is deregistering. Clear pending timer events in nfa_sys. */ 207 nfa_sys_ptim_init (&nfa_sys_cb.ptim_cb, NFA_SYS_TIMER_PERIOD, p_nfa_sys_cfg->timer); 208 } 209 } 210 211 /******************************************************************************* 212 ** 213 ** Function nfa_sys_is_register 214 ** 215 ** Description Called by other BTA subsystems to get registeration 216 ** status. 217 ** 218 ** 219 ** Returns void 220 ** 221 *******************************************************************************/ 222 BOOLEAN nfa_sys_is_register (UINT8 id) 223 { 224 return nfa_sys_cb.is_reg[id]; 225 } 226 227 /******************************************************************************* 228 ** 229 ** Function nfa_sys_is_graceful_disable 230 ** 231 ** Description Called by other BTA subsystems to get disable 232 ** parameter. 233 ** 234 ** 235 ** Returns void 236 ** 237 *******************************************************************************/ 238 BOOLEAN nfa_sys_is_graceful_disable (void) 239 { 240 return nfa_sys_cb.graceful_disable; 241 } 242 243 /******************************************************************************* 244 ** 245 ** Function nfa_sys_enable_subsystems 246 ** 247 ** Description Call on NFA Start up 248 ** 249 ** Returns void 250 ** 251 *******************************************************************************/ 252 void nfa_sys_enable_subsystems (void) 253 { 254 UINT8 id; 255 256 NFA_TRACE_DEBUG0 ("nfa_sys: enabling subsystems"); 257 258 /* Enable all subsystems except SYS */ 259 for (id = NFA_ID_DM; id < NFA_ID_MAX; id++) 260 { 261 if (nfa_sys_cb.is_reg[id]) 262 { 263 if (nfa_sys_cb.reg[id]->enable != NULL) 264 { 265 /* Subsytem has a Disable funciton. Call it now */ 266 (*nfa_sys_cb.reg[id]->enable) (); 267 } 268 else 269 { 270 /* Subsytem does not have a Enable function. Report Enable on behalf of subsystem */ 271 nfa_sys_cback_notify_enable_complete (id); 272 } 273 } 274 } 275 } 276 277 /******************************************************************************* 278 ** 279 ** Function nfa_sys_disable_subsystems 280 ** 281 ** Description Call on NFA shutdown. Disable all subsystems above NFA_DM 282 ** 283 ** Returns void 284 ** 285 *******************************************************************************/ 286 void nfa_sys_disable_subsystems (BOOLEAN graceful) 287 { 288 UINT8 id; 289 BOOLEAN done = TRUE; 290 291 NFA_TRACE_DEBUG1 ("nfa_sys: disabling subsystems:%d", graceful); 292 nfa_sys_cb.graceful_disable = graceful; 293 294 /* Disable all subsystems above NFA_DM. (NFA_DM and NFA_SYS will be disabled last) */ 295 for (id = (NFA_ID_DM+1); id < NFA_ID_MAX; id++) 296 { 297 if (nfa_sys_cb.is_reg[id]) 298 { 299 done = FALSE; 300 if (nfa_sys_cb.reg[id]->disable != NULL) 301 { 302 /* Subsytem has a Disable funciton. Call it now */ 303 (*nfa_sys_cb.reg[id]->disable) (); 304 } 305 else 306 { 307 /* Subsytem does not have a Disable function. Deregister on behalf of subsystem */ 308 nfa_sys_deregister (id); 309 } 310 } 311 } 312 313 /* If All subsystems disabled. disable DM */ 314 if ((done) && (nfa_sys_cb.is_reg[NFA_ID_DM])) 315 { 316 (*nfa_sys_cb.reg[NFA_ID_DM]->disable) (); 317 } 318 } 319 320 /******************************************************************************* 321 ** 322 ** Function nfa_sys_notify_nfcc_power_mode 323 ** 324 ** Description Call to notify NFCC power mode to NFA sub-modules 325 ** 326 ** Returns void 327 ** 328 *******************************************************************************/ 329 void nfa_sys_notify_nfcc_power_mode (UINT8 nfcc_power_mode) 330 { 331 UINT8 id; 332 333 NFA_TRACE_DEBUG1 ("nfa_sys: notify NFCC power mode(%d) to subsystems", nfcc_power_mode); 334 335 /* Notify NFCC power state to all subsystems except NFA_SYS */ 336 for (id = NFA_ID_DM; id < NFA_ID_MAX; id++) 337 { 338 if ((nfa_sys_cb.is_reg[id]) && (nfa_sys_cb.reg[id]->proc_nfcc_pwr_mode)) 339 { 340 /* Subsytem has a funciton for processing NFCC power mode. Call it now */ 341 (*nfa_sys_cb.reg[id]->proc_nfcc_pwr_mode) (nfcc_power_mode); 342 } 343 } 344 } 345 346 /******************************************************************************* 347 ** 348 ** Function nfa_sys_sendmsg 349 ** 350 ** Description Send a GKI message to BTA. This function is designed to 351 ** optimize sending of messages to BTA. It is called by BTA 352 ** API functions and call-in functions. 353 ** 354 ** 355 ** Returns void 356 ** 357 *******************************************************************************/ 358 void nfa_sys_sendmsg (void *p_msg) 359 { 360 GKI_send_msg (NFC_TASK, p_nfa_sys_cfg->mbox, p_msg); 361 } 362 363 /******************************************************************************* 364 ** 365 ** Function nfa_sys_start_timer 366 ** 367 ** Description Start a protocol timer for the specified amount 368 ** of time in milliseconds. 369 ** 370 ** Returns void 371 ** 372 *******************************************************************************/ 373 void nfa_sys_start_timer (TIMER_LIST_ENT *p_tle, UINT16 type, INT32 timeout) 374 { 375 nfa_sys_ptim_start_timer (&nfa_sys_cb.ptim_cb, p_tle, type, timeout); 376 } 377 378 /******************************************************************************* 379 ** 380 ** Function nfa_sys_stop_timer 381 ** 382 ** Description Stop a BTA timer. 383 ** 384 ** Returns void 385 ** 386 *******************************************************************************/ 387 void nfa_sys_stop_timer (TIMER_LIST_ENT *p_tle) 388 { 389 nfa_sys_ptim_stop_timer (&nfa_sys_cb.ptim_cb, p_tle); 390 } 391 392 393 /******************************************************************************* 394 ** 395 ** Function nfa_sys_disable_timers 396 ** 397 ** Description Disable sys timer event handling 398 ** 399 ** Returns void 400 ** 401 *******************************************************************************/ 402 void nfa_sys_disable_timers (void) 403 { 404 nfa_sys_cb.timers_disabled = TRUE; 405 } 406 407 /******************************************************************************* 408 ** 409 ** Function nfa_sys_set_trace_level 410 ** 411 ** Description Set trace level for BTA 412 ** 413 ** Returns void 414 ** 415 *******************************************************************************/ 416 void nfa_sys_set_trace_level (UINT8 level) 417 { 418 nfa_sys_cb.trace_level = level; 419 } 420