1 /****************************************************************************** 2 3 � 1995-2003, 2004, 2005-2011 Freescale Semiconductor, Inc. 4 All rights reserved. 5 6 This is proprietary source code of Freescale Semiconductor Inc., 7 and its use is subject to the NetComm Device Drivers EULA. 8 The copyright notice above does not evidence any actual or intended 9 publication of such source code. 10 11 ALTERNATIVELY, redistribution and use in source and binary forms, with 12 or without modification, are permitted provided that the following 13 conditions are met: 14 * Redistributions of source code must retain the above copyright 15 notice, this list of conditions and the following disclaimer. 16 * Redistributions in binary form must reproduce the above copyright 17 notice, this list of conditions and the following disclaimer in the 18 documentation and/or other materials provided with the distribution. 19 * Neither the name of Freescale Semiconductor nor the 20 names of its contributors may be used to endorse or promote products 21 derived from this software without specific prior written permission. 22 23 THIS SOFTWARE IS PROVIDED BY Freescale Semiconductor ``AS IS'' AND ANY 24 EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED 25 WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE 26 DISCLAIMED. IN NO EVENT SHALL Freescale Semiconductor BE LIABLE FOR ANY 27 DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES 28 (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; 29 LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND 30 ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT 31 (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS 32 SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 33 * 34 35 36 **************************************************************************/ 37 /****************************************************************************** 38 @File bm.c 39 40 @Description BM 41 *//***************************************************************************/ 42 #include "error_ext.h" 43 #include "std_ext.h" 44 #include "string_ext.h" 45 #include "sprint_ext.h" 46 #include "debug_ext.h" 47 #include "mm_ext.h" 48 49 #include "bm.h" 50 51 52 t_Error BM_ConfigException(t_Handle h_Bm, e_BmExceptions exception, bool enable); 53 54 55 /****************************************/ 56 /* static functions */ 57 /****************************************/ 58 59 static volatile bool blockingFlag = FALSE; 60 static void BmIpcMsgCompletionCB(t_Handle h_Module, 61 uint8_t *p_Msg, 62 uint8_t *p_Reply, 63 uint32_t replyLength, 64 t_Error status) 65 { 66 SANITY_CHECK_RETURN(h_Module, E_INVALID_HANDLE); 67 68 #ifdef DISABLE_SANITY_CHECKS 69 UNUSED(h_Module); 70 #endif /* DISABLE_SANITY_CHECKS */ 71 UNUSED(p_Msg);UNUSED(p_Reply);UNUSED(replyLength);UNUSED(status); 72 73 blockingFlag = FALSE; 74 } 75 76 static t_Error BmHandleIpcMsgCB(t_Handle h_Bm, 77 uint8_t *p_Msg, 78 uint32_t msgLength, 79 uint8_t *p_Reply, 80 uint32_t *p_ReplyLength) 81 { 82 t_Bm *p_Bm = (t_Bm*)h_Bm; 83 t_BmIpcMsg *p_IpcMsg = (t_BmIpcMsg*)p_Msg; 84 t_BmIpcReply *p_IpcReply = (t_BmIpcReply *)p_Reply; 85 86 SANITY_CHECK_RETURN_ERROR(p_Bm, E_INVALID_HANDLE); 87 SANITY_CHECK_RETURN_ERROR((msgLength >= sizeof(uint32_t)), E_INVALID_VALUE); 88 89 #ifdef DISABLE_SANITY_CHECKS 90 UNUSED(msgLength); 91 #endif /* DISABLE_SANITY_CHECKS */ 92 93 ASSERT_COND(p_IpcMsg); 94 95 memset(p_IpcReply, 0, (sizeof(uint8_t) * BM_IPC_MAX_REPLY_SIZE)); 96 *p_ReplyLength = 0; 97 98 switch(p_IpcMsg->msgId) 99 { 100 case (BM_MASTER_IS_ALIVE): 101 *(uint8_t*)p_IpcReply->replyBody = 1; 102 *p_ReplyLength = sizeof(uint32_t) + sizeof(uint8_t); 103 break; 104 case (BM_SET_POOL_THRESH): 105 { 106 t_Error err; 107 t_BmIpcPoolThreshParams ipcPoolThresh; 108 109 memcpy((uint8_t*)&ipcPoolThresh, p_IpcMsg->msgBody, sizeof(t_BmIpcPoolThreshParams)); 110 if ((err = BmSetPoolThresholds(p_Bm, 111 ipcPoolThresh.bpid, 112 ipcPoolThresh.thresholds)) != E_OK) 113 REPORT_ERROR(MINOR, err, NO_MSG); 114 break; 115 } 116 case (BM_UNSET_POOL_THRESH): 117 { 118 t_Error err; 119 t_BmIpcPoolThreshParams ipcPoolThresh; 120 121 memcpy((uint8_t*)&ipcPoolThresh, p_IpcMsg->msgBody, sizeof(t_BmIpcPoolThreshParams)); 122 if ((err = BmUnSetPoolThresholds(p_Bm, 123 ipcPoolThresh.bpid)) != E_OK) 124 REPORT_ERROR(MINOR, err, NO_MSG); 125 break; 126 } 127 case (BM_GET_COUNTER): 128 { 129 t_BmIpcGetCounter ipcCounter; 130 uint32_t count; 131 132 memcpy((uint8_t*)&ipcCounter, p_IpcMsg->msgBody, sizeof(t_BmIpcGetCounter)); 133 count = BmGetCounter(p_Bm, 134 (e_BmInterModuleCounters)ipcCounter.enumId, 135 ipcCounter.bpid); 136 memcpy(p_IpcReply->replyBody, (uint8_t*)&count, sizeof(uint32_t)); 137 *p_ReplyLength = sizeof(uint32_t) + sizeof(uint32_t); 138 break; 139 } 140 case (BM_GET_REVISION): 141 { 142 t_BmRevisionInfo revInfo; 143 t_BmIpcRevisionInfo ipcRevInfo; 144 145 p_IpcReply->error = (uint32_t)BmGetRevision(h_Bm, &revInfo); 146 ipcRevInfo.majorRev = revInfo.majorRev; 147 ipcRevInfo.minorRev = revInfo.minorRev; 148 memcpy(p_IpcReply->replyBody, (uint8_t*)&ipcRevInfo, sizeof(t_BmIpcRevisionInfo)); 149 *p_ReplyLength = sizeof(uint32_t) + sizeof(t_BmIpcRevisionInfo); 150 break; 151 } 152 case (BM_FORCE_BPID): 153 { 154 t_BmIpcBpidParams ipcBpid; 155 uint32_t tmp; 156 157 memcpy((uint8_t*)&ipcBpid, p_IpcMsg->msgBody, sizeof(t_BmIpcBpidParams)); 158 tmp = BmBpidGet(p_Bm, TRUE, ipcBpid.bpid); 159 memcpy(p_IpcReply->replyBody, (uint8_t*)&tmp, sizeof(uint32_t)); 160 *p_ReplyLength = sizeof(uint32_t) + sizeof(uint32_t); 161 break; 162 } 163 case (BM_PUT_BPID): 164 { 165 t_Error err; 166 t_BmIpcBpidParams ipcBpid; 167 168 memcpy((uint8_t*)&ipcBpid, p_IpcMsg->msgBody, sizeof(t_BmIpcBpidParams)); 169 if ((err = BmBpidPut(p_Bm, ipcBpid.bpid)) != E_OK) 170 REPORT_ERROR(MINOR, err, NO_MSG); 171 break; 172 } 173 default: 174 *p_ReplyLength = 0; 175 RETURN_ERROR(MINOR, E_INVALID_SELECTION, ("command not found!!!")); 176 } 177 178 return E_OK; 179 } 180 181 static t_Error CheckBmParameters(t_Bm *p_Bm) 182 { 183 if ((p_Bm->p_BmDriverParams->partBpidBase + p_Bm->p_BmDriverParams->partNumOfPools) > BM_MAX_NUM_OF_POOLS) 184 RETURN_ERROR(MAJOR, E_INVALID_VALUE, ("partBpidBase+partNumOfPools out of range!!!")); 185 186 if (p_Bm->guestId == NCSW_MASTER_ID) 187 { 188 if (!p_Bm->p_BmDriverParams->totalNumOfBuffers) 189 RETURN_ERROR(MAJOR, E_INVALID_VALUE, ("totalNumOfBuffers must be larger than '0'!!!")); 190 if (p_Bm->p_BmDriverParams->totalNumOfBuffers > (128*MEGABYTE)) 191 RETURN_ERROR(MAJOR, E_INVALID_VALUE, ("totalNumOfBuffers must be equal or smaller than 128M!!!")); 192 if(!p_Bm->f_Exception) 193 RETURN_ERROR(MAJOR, E_INVALID_VALUE, ("Exceptions callback not provided")); 194 } 195 196 return E_OK; 197 } 198 199 static __inline__ uint32_t GenerateThresh(uint32_t val, int roundup) 200 { 201 uint32_t e = 0; /* co-efficient, exponent */ 202 uint32_t oddbit = 0; 203 while(val > 0xff) { 204 oddbit = val & 1; 205 val >>= 1; 206 e++; 207 if(roundup && oddbit) 208 val++; 209 } 210 return (val | (e << 8)); 211 } 212 213 static t_Error BmSetPool(t_Handle h_Bm, 214 uint8_t bpid, 215 uint32_t swdet, 216 uint32_t swdxt, 217 uint32_t hwdet, 218 uint32_t hwdxt) 219 { 220 t_Bm *p_Bm = (t_Bm*)h_Bm; 221 222 SANITY_CHECK_RETURN_ERROR(p_Bm, E_INVALID_HANDLE); 223 SANITY_CHECK_RETURN_ERROR(bpid < BM_MAX_NUM_OF_POOLS, E_INVALID_VALUE); 224 225 WRITE_UINT32(p_Bm->p_BmRegs->swdet[bpid], GenerateThresh(swdet, 0)); 226 WRITE_UINT32(p_Bm->p_BmRegs->swdxt[bpid], GenerateThresh(swdxt, 1)); 227 WRITE_UINT32(p_Bm->p_BmRegs->hwdet[bpid], GenerateThresh(hwdet, 0)); 228 WRITE_UINT32(p_Bm->p_BmRegs->hwdxt[bpid], GenerateThresh(hwdxt, 1)); 229 230 return E_OK; 231 } 232 233 /****************************************/ 234 /* Inter-Module functions */ 235 /****************************************/ 236 237 t_Error BmSetPoolThresholds(t_Handle h_Bm, uint8_t bpid, const uint32_t *thresholds) 238 { 239 t_Bm *p_Bm = (t_Bm*)h_Bm; 240 241 SANITY_CHECK_RETURN_ERROR(p_Bm, E_INVALID_HANDLE); 242 SANITY_CHECK_RETURN_ERROR(bpid < BM_MAX_NUM_OF_POOLS, E_INVALID_VALUE); 243 244 if (p_Bm->guestId == NCSW_MASTER_ID) 245 { 246 return BmSetPool(h_Bm, 247 bpid, 248 thresholds[0], 249 thresholds[1], 250 thresholds[2], 251 thresholds[3]); 252 } 253 else if (p_Bm->h_Session) 254 { 255 t_BmIpcMsg msg; 256 t_BmIpcPoolThreshParams ipcPoolThresh; 257 t_Error errCode = E_OK; 258 259 memset(&msg, 0, sizeof(t_BmIpcMsg)); 260 ipcPoolThresh.bpid = bpid; 261 memcpy(ipcPoolThresh.thresholds, thresholds, sizeof(uintptr_t) * MAX_DEPLETION_THRESHOLDS); 262 msg.msgId = BM_SET_POOL_THRESH; 263 memcpy(msg.msgBody, &ipcPoolThresh, sizeof(t_BmIpcPoolThreshParams)); 264 if ((errCode = XX_IpcSendMessage(p_Bm->h_Session, 265 (uint8_t*)&msg, 266 sizeof(msg.msgId) + sizeof(t_BmIpcPoolThreshParams), 267 NULL, 268 NULL, 269 NULL, 270 NULL)) != E_OK) 271 RETURN_ERROR(MAJOR, errCode, NO_MSG); 272 return E_OK; 273 } 274 else 275 RETURN_ERROR(WARNING, E_NOT_SUPPORTED, ("IPC")); 276 } 277 278 t_Error BmUnSetPoolThresholds(t_Handle h_Bm, uint8_t bpid) 279 { 280 t_Bm *p_Bm = (t_Bm*)h_Bm; 281 282 SANITY_CHECK_RETURN_ERROR(p_Bm, E_INVALID_HANDLE); 283 SANITY_CHECK_RETURN_ERROR(bpid < BM_MAX_NUM_OF_POOLS, E_INVALID_VALUE); 284 285 if (p_Bm->guestId == NCSW_MASTER_ID) 286 { 287 return BmSetPool(h_Bm, 288 bpid, 289 0, 290 0, 291 0, 292 0); 293 } 294 else if (p_Bm->h_Session) 295 { 296 t_BmIpcMsg msg; 297 t_BmIpcPoolThreshParams ipcPoolThresh; 298 t_Error errCode = E_OK; 299 300 memset(&msg, 0, sizeof(t_BmIpcMsg)); 301 memset(&ipcPoolThresh, 0, sizeof(t_BmIpcPoolThreshParams)); 302 ipcPoolThresh.bpid = bpid; 303 msg.msgId = BM_UNSET_POOL_THRESH; 304 memcpy(msg.msgBody, &ipcPoolThresh, sizeof(t_BmIpcPoolThreshParams)); 305 if ((errCode = XX_IpcSendMessage(p_Bm->h_Session, 306 (uint8_t*)&msg, 307 sizeof(msg.msgId) + sizeof(t_BmIpcPoolThreshParams), 308 NULL, 309 NULL, 310 NULL, 311 NULL)) != E_OK) 312 RETURN_ERROR(MAJOR, errCode, NO_MSG); 313 return E_OK; 314 } 315 else 316 RETURN_ERROR(WARNING, E_NOT_SUPPORTED, ("IPC")); 317 } 318 319 uint32_t BmGetCounter(t_Handle h_Bm, e_BmInterModuleCounters counter, uint8_t bpid) 320 { 321 t_Bm *p_Bm = (t_Bm*)h_Bm; 322 323 SANITY_CHECK_RETURN_VALUE(p_Bm, E_INVALID_HANDLE, 0); 324 SANITY_CHECK_RETURN_VALUE(bpid < BM_MAX_NUM_OF_POOLS, E_INVALID_VALUE, 0); 325 SANITY_CHECK_RETURN_VALUE((((p_Bm->guestId == NCSW_MASTER_ID) && p_Bm->p_BmRegs) || 326 (p_Bm->guestId != NCSW_MASTER_ID)), E_INVALID_STATE, 0); 327 328 if ((p_Bm->guestId == NCSW_MASTER_ID) || 329 (!p_Bm->h_Session && p_Bm->p_BmRegs)) 330 { 331 switch(counter) 332 { 333 case(e_BM_IM_COUNTERS_POOL_CONTENT): 334 return GET_UINT32(p_Bm->p_BmRegs->content[bpid]); 335 case(e_BM_IM_COUNTERS_POOL_SW_DEPLETION): 336 return GET_UINT32(p_Bm->p_BmRegs->sdcnt[bpid]); 337 case(e_BM_IM_COUNTERS_POOL_HW_DEPLETION): 338 return GET_UINT32(p_Bm->p_BmRegs->hdcnt[bpid]); 339 case(e_BM_IM_COUNTERS_FBPR): 340 return GET_UINT32(p_Bm->p_BmRegs->fbpr_fpc); 341 default: 342 break; 343 } 344 /* should never get here */ 345 ASSERT_COND(FALSE); 346 } 347 else if (p_Bm->h_Session) 348 { 349 t_BmIpcMsg msg; 350 t_BmIpcReply reply; 351 t_BmIpcGetCounter ipcCounter; 352 uint32_t replyLength; 353 uint32_t count; 354 t_Error errCode = E_OK; 355 356 memset(&msg, 0, sizeof(t_BmIpcMsg)); 357 memset(&reply, 0, sizeof(t_BmIpcReply)); 358 ipcCounter.bpid = bpid; 359 ipcCounter.enumId = (uint32_t)counter; 360 msg.msgId = BM_GET_COUNTER; 361 memcpy(msg.msgBody, &ipcCounter, sizeof(t_BmIpcGetCounter)); 362 replyLength = sizeof(uint32_t) + sizeof(uint32_t); 363 if ((errCode = XX_IpcSendMessage(p_Bm->h_Session, 364 (uint8_t*)&msg, 365 sizeof(msg.msgId) + sizeof(t_BmIpcGetCounter), 366 (uint8_t*)&reply, 367 &replyLength, 368 NULL, 369 NULL)) != E_OK) 370 REPORT_ERROR(MAJOR, errCode, NO_MSG); 371 if (replyLength != (sizeof(uint32_t) + sizeof(uint32_t))) 372 { 373 REPORT_ERROR(MAJOR, E_INVALID_VALUE, ("IPC reply length mismatch")); 374 errCode = E_INVALID_VALUE; 375 } 376 if (errCode == E_OK) 377 { 378 memcpy((uint8_t*)&count, reply.replyBody, sizeof(uint32_t)); 379 return count; 380 } 381 } 382 else 383 REPORT_ERROR(WARNING, E_NOT_SUPPORTED, 384 ("In 'guest', either IPC or 'baseAddress' is required!")); 385 386 return 0; 387 } 388 389 t_Error BmGetRevision(t_Handle h_Bm, t_BmRevisionInfo *p_BmRevisionInfo) 390 { 391 t_Bm *p_Bm = (t_Bm*)h_Bm; 392 uint32_t tmpReg; 393 394 SANITY_CHECK_RETURN_ERROR(p_Bm, E_INVALID_HANDLE); 395 SANITY_CHECK_RETURN_ERROR(p_BmRevisionInfo, E_NULL_POINTER); 396 SANITY_CHECK_RETURN_ERROR((((p_Bm->guestId == NCSW_MASTER_ID) && p_Bm->p_BmRegs) || 397 (p_Bm->guestId != NCSW_MASTER_ID)), E_INVALID_STATE); 398 399 if ((p_Bm->guestId == NCSW_MASTER_ID) || 400 (!p_Bm->h_Session && p_Bm->p_BmRegs)) 401 { 402 /* read revision register 1 */ 403 tmpReg = GET_UINT32(p_Bm->p_BmRegs->ip_rev_1); 404 p_BmRevisionInfo->majorRev = (uint8_t)((tmpReg & REV1_MAJOR_MASK) >> REV1_MAJOR_SHIFT); 405 p_BmRevisionInfo->minorRev = (uint8_t)((tmpReg & REV1_MINOR_MASK) >> REV1_MINOR_SHIFT); 406 } 407 else if (p_Bm->h_Session) 408 { 409 t_BmIpcMsg msg; 410 t_BmIpcReply reply; 411 t_BmIpcRevisionInfo ipcRevInfo; 412 uint32_t replyLength; 413 t_Error errCode = E_OK; 414 415 memset(&msg, 0, sizeof(t_BmIpcMsg)); 416 memset(&reply, 0, sizeof(t_BmIpcReply)); 417 msg.msgId = BM_GET_REVISION; 418 replyLength = sizeof(uint32_t) + sizeof(t_BmIpcRevisionInfo); 419 if ((errCode = XX_IpcSendMessage(p_Bm->h_Session, 420 (uint8_t*)&msg, 421 sizeof(msg.msgId), 422 (uint8_t*)&reply, 423 &replyLength, 424 NULL, 425 NULL)) != E_OK) 426 RETURN_ERROR(MAJOR, errCode, NO_MSG); 427 if (replyLength != (sizeof(uint32_t) + sizeof(t_BmIpcRevisionInfo))) 428 RETURN_ERROR(MAJOR, E_INVALID_VALUE, ("IPC reply length mismatch")); 429 430 memcpy((uint8_t*)&ipcRevInfo, reply.replyBody, sizeof(t_BmIpcRevisionInfo)); 431 p_BmRevisionInfo->majorRev = ipcRevInfo.majorRev; 432 p_BmRevisionInfo->minorRev = ipcRevInfo.minorRev; 433 return (t_Error)(reply.error); 434 } 435 else 436 RETURN_ERROR(WARNING, E_NOT_SUPPORTED, 437 ("In 'guest', either IPC or 'baseAddress' is required!")); 438 439 return E_OK; 440 } 441 442 static void FreeInitResources(t_Bm *p_Bm) 443 { 444 if (p_Bm->p_FbprBase) 445 XX_FreeSmart(p_Bm->p_FbprBase); 446 if (p_Bm->h_Session) 447 XX_IpcFreeSession(p_Bm->h_Session); 448 if (p_Bm->h_BpidMm) 449 MM_Free(p_Bm->h_BpidMm); 450 } 451 452 /****************************************/ 453 /* API Init unit functions */ 454 /****************************************/ 455 456 t_Handle BM_Config(t_BmParam *p_BmParam) 457 { 458 t_Bm *p_Bm; 459 460 SANITY_CHECK_RETURN_VALUE(p_BmParam, E_INVALID_HANDLE, NULL); 461 462 p_Bm = (t_Bm *)XX_Malloc(sizeof(t_Bm)); 463 if (!p_Bm) 464 { 465 REPORT_ERROR(MAJOR, E_NO_MEMORY, ("BM obj!!!")); 466 return NULL; 467 } 468 memset(p_Bm, 0, sizeof(t_Bm)); 469 470 p_Bm->p_BmDriverParams = (t_BmDriverParams *)XX_Malloc(sizeof(t_BmDriverParams)); 471 if (!p_Bm->p_BmDriverParams) 472 { 473 XX_Free(p_Bm); 474 REPORT_ERROR(MAJOR, E_NO_MEMORY, ("Bm driver parameters")); 475 return NULL; 476 } 477 memset(p_Bm->p_BmDriverParams, 0, sizeof(t_BmDriverParams)); 478 479 p_Bm->guestId = p_BmParam->guestId; 480 p_Bm->p_BmDriverParams->partNumOfPools = p_BmParam->partNumOfPools; 481 p_Bm->p_BmDriverParams->partBpidBase = p_BmParam->partBpidBase; 482 p_Bm->p_BmRegs = (t_BmRegs *)UINT_TO_PTR(p_BmParam->baseAddress); 483 484 if (p_Bm->guestId == NCSW_MASTER_ID) 485 { 486 p_Bm->exceptions = DEFAULT_exceptions; 487 p_Bm->f_Exception = p_BmParam->f_Exception; 488 p_Bm->h_App = p_BmParam->h_App; 489 p_Bm->errIrq = p_BmParam->errIrq; 490 p_Bm->p_BmDriverParams->totalNumOfBuffers = p_BmParam->totalNumOfBuffers; 491 p_Bm->p_BmDriverParams->fbprMemPartitionId = p_BmParam->fbprMemPartitionId; 492 p_Bm->p_BmDriverParams->fbprThreshold = DEFAULT_fbprThreshold; 493 p_Bm->p_BmDriverParams->liodn = p_BmParam->liodn; 494 495 } 496 /* build the BM partition IPC address */ 497 memset(p_Bm->moduleName, 0, MODULE_NAME_SIZE); 498 if(Sprint (p_Bm->moduleName, "BM_0_%d",p_Bm->guestId) != (p_Bm->guestId<10 ? 6:7)) 499 { 500 XX_Free(p_Bm->p_BmDriverParams); 501 XX_Free(p_Bm); 502 REPORT_ERROR(MAJOR, E_INVALID_STATE, ("Sprint failed")); 503 return NULL; 504 } 505 return p_Bm; 506 } 507 508 t_Error BM_Init(t_Handle h_Bm) 509 { 510 t_Bm *p_Bm = (t_Bm *)h_Bm; 511 t_Error err; 512 513 SANITY_CHECK_RETURN_ERROR(p_Bm, E_INVALID_HANDLE); 514 SANITY_CHECK_RETURN_ERROR(p_Bm->p_BmDriverParams, E_INVALID_HANDLE); 515 516 CHECK_INIT_PARAMETERS(p_Bm, CheckBmParameters); 517 518 if (p_Bm->p_BmDriverParams->partNumOfPools) 519 if (MM_Init(&p_Bm->h_BpidMm, p_Bm->p_BmDriverParams->partBpidBase, p_Bm->p_BmDriverParams->partNumOfPools) != E_OK) 520 { 521 FreeInitResources(p_Bm); 522 RETURN_ERROR(MAJOR, E_INVALID_HANDLE, ("BM-BPIDS-MEM partition!!!")); 523 } 524 525 if (p_Bm->guestId == NCSW_MASTER_ID) 526 { 527 uint64_t phyAddr; 528 t_BmRevisionInfo revInfo; 529 uint32_t dsSize, exp; 530 531 BmGetRevision(p_Bm, &revInfo); 532 DBG(TRACE, ("Bman ver:%02x,%02x", revInfo.majorRev, revInfo.minorRev)); 533 534 WRITE_UINT32(p_Bm->p_BmRegs->liodnr, (uint16_t)p_Bm->p_BmDriverParams->liodn); 535 536 /* FBPR memory */ 537 dsSize = (uint32_t)(p_Bm->p_BmDriverParams->totalNumOfBuffers * (FBPR_ENTRY_SIZE / 8)); 538 LOG2(dsSize, exp); 539 if (!POWER_OF_2(dsSize)) (exp++); 540 dsSize = (uint32_t)(1 << exp); 541 if (dsSize < (4*KILOBYTE)) 542 { 543 dsSize = (4*KILOBYTE); 544 LOG2(dsSize, exp); 545 } 546 p_Bm->p_FbprBase = XX_MallocSmart(dsSize, (int)p_Bm->p_BmDriverParams->fbprMemPartitionId, dsSize); 547 if (!p_Bm->p_FbprBase) 548 { 549 FreeInitResources(p_Bm); 550 RETURN_ERROR(MAJOR, E_NO_MEMORY, ("FBPR obj!!!")); 551 } 552 phyAddr = XX_VirtToPhys(p_Bm->p_FbprBase); 553 WRITE_UINT32(p_Bm->p_BmRegs->fbpr_bare, ((uint32_t)(phyAddr >> 32) & 0xffff)); 554 WRITE_UINT32(p_Bm->p_BmRegs->fbpr_bar, (uint32_t)phyAddr); 555 WRITE_UINT32(p_Bm->p_BmRegs->fbpr_ar, (exp - 1)); 556 557 WRITE_UINT32(p_Bm->p_BmRegs->fbpr_fp_lwit, p_Bm->p_BmDriverParams->fbprThreshold); 558 WRITE_UINT32(p_Bm->p_BmRegs->err_isr, p_Bm->exceptions); 559 WRITE_UINT32(p_Bm->p_BmRegs->err_ier, p_Bm->exceptions); 560 WRITE_UINT32(p_Bm->p_BmRegs->err_isdr, 0x0); 561 if (p_Bm->errIrq != NO_IRQ) 562 { 563 XX_SetIntr(p_Bm->errIrq, BM_ErrorIsr, p_Bm); 564 XX_EnableIntr(p_Bm->errIrq); 565 } 566 567 if ((err = XX_IpcRegisterMsgHandler(p_Bm->moduleName, BmHandleIpcMsgCB, p_Bm, BM_IPC_MAX_REPLY_SIZE)) != E_OK) 568 { 569 FreeInitResources(p_Bm); 570 RETURN_ERROR(MAJOR, err, NO_MSG); 571 } 572 } 573 else /* guest mode */ 574 { 575 char masterModuleName[MODULE_NAME_SIZE]; 576 577 memset(masterModuleName, 0, MODULE_NAME_SIZE); 578 if(Sprint (masterModuleName, "BM_0_%d", NCSW_MASTER_ID) != (NCSW_MASTER_ID<10 ? 6:7)) 579 { 580 FreeInitResources(p_Bm); 581 RETURN_ERROR(MAJOR, E_INVALID_STATE, ("Sprint failed")); 582 } 583 584 p_Bm->h_Session = XX_IpcInitSession(masterModuleName, p_Bm->moduleName); 585 if (p_Bm->h_Session) 586 { 587 t_BmIpcMsg msg; 588 uint8_t isMasterAlive = 0; 589 t_BmIpcReply reply; 590 uint32_t replyLength; 591 592 memset(&msg, 0, sizeof(t_BmIpcMsg)); 593 memset(&reply, 0, sizeof(t_BmIpcReply)); 594 msg.msgId = BM_MASTER_IS_ALIVE; 595 replyLength = sizeof(uint32_t) + sizeof(uint8_t); 596 do 597 { 598 blockingFlag = TRUE; 599 if ((err = XX_IpcSendMessage(p_Bm->h_Session, 600 (uint8_t*)&msg, 601 sizeof(msg.msgId), 602 (uint8_t*)&reply, 603 &replyLength, 604 BmIpcMsgCompletionCB, 605 p_Bm)) != E_OK) 606 REPORT_ERROR(MAJOR, err, NO_MSG); 607 while(blockingFlag) ; 608 if(replyLength != (sizeof(uint32_t) + sizeof(uint8_t))) 609 REPORT_ERROR(MAJOR, E_INVALID_VALUE, ("IPC reply length mismatch")); 610 isMasterAlive = *(uint8_t*)(reply.replyBody); 611 } while (!isMasterAlive); 612 } 613 } 614 615 XX_Free(p_Bm->p_BmDriverParams); 616 p_Bm->p_BmDriverParams = NULL; 617 618 return E_OK; 619 } 620 621 t_Error BM_Free(t_Handle h_Bm) 622 { 623 t_Bm *p_Bm = (t_Bm *)h_Bm; 624 625 if (!p_Bm) 626 return ERROR_CODE(E_INVALID_HANDLE); 627 628 if (p_Bm->guestId == NCSW_MASTER_ID) 629 { 630 XX_IpcUnregisterMsgHandler(p_Bm->moduleName); 631 if (p_Bm->errIrq != NO_IRQ) 632 { 633 XX_DisableIntr(p_Bm->errIrq); 634 XX_FreeIntr(p_Bm->errIrq); 635 } 636 } 637 FreeInitResources(p_Bm); 638 639 if(p_Bm->p_BmDriverParams) 640 XX_Free(p_Bm->p_BmDriverParams); 641 642 XX_Free(p_Bm); 643 return E_OK; 644 } 645 646 t_Error BM_ConfigException(t_Handle h_Bm, e_BmExceptions exception, bool enable) 647 { 648 t_Bm *p_Bm = (t_Bm*)h_Bm; 649 uint32_t bitMask = 0; 650 651 SANITY_CHECK_RETURN_ERROR(p_Bm, E_INVALID_HANDLE); 652 SANITY_CHECK_RETURN_ERROR(p_Bm->p_BmDriverParams, E_INVALID_HANDLE); 653 654 GET_EXCEPTION_FLAG(bitMask, exception); 655 if(bitMask) 656 { 657 if (enable) 658 p_Bm->exceptions |= bitMask; 659 else 660 p_Bm->exceptions &= ~bitMask; 661 } 662 else 663 RETURN_ERROR(MAJOR, E_INVALID_VALUE, ("Undefined exception")); 664 665 return E_OK; 666 } 667 668 t_Error BM_ConfigFbprThreshold(t_Handle h_Bm, uint32_t threshold) 669 { 670 t_Bm *p_Bm = (t_Bm *)h_Bm; 671 672 SANITY_CHECK_RETURN_ERROR(p_Bm, E_INVALID_HANDLE); 673 SANITY_CHECK_RETURN_ERROR(p_Bm->p_BmDriverParams, E_INVALID_HANDLE); 674 675 p_Bm->p_BmDriverParams->fbprThreshold = threshold; 676 677 return E_OK; 678 } 679 680 void BM_ErrorIsr(t_Handle h_Bm) 681 { 682 t_Bm *p_Bm = (t_Bm *)h_Bm; 683 uint32_t tmpReg; 684 685 SANITY_CHECK_RETURN(p_Bm, E_INVALID_HANDLE); 686 687 if (p_Bm->guestId != NCSW_MASTER_ID) 688 { 689 REPORT_ERROR(WARNING, E_INVALID_OPERATION, ("Master Only")); 690 return; 691 } 692 693 tmpReg = GET_UINT32(p_Bm->p_BmRegs->err_isr); 694 tmpReg &= GET_UINT32(p_Bm->p_BmRegs->err_ier); 695 WRITE_UINT32(p_Bm->p_BmRegs->err_isr, tmpReg); 696 697 if (tmpReg & BM_EX_INVALID_COMMAND) 698 p_Bm->f_Exception(p_Bm->h_App, e_BM_EX_INVALID_COMMAND); 699 if (tmpReg & BM_EX_FBPR_THRESHOLD) 700 p_Bm->f_Exception(p_Bm->h_App, e_BM_EX_FBPR_THRESHOLD); 701 if (tmpReg & BM_EX_MULTI_ECC) 702 p_Bm->f_Exception(p_Bm->h_App, e_BM_EX_MULTI_ECC); 703 if (tmpReg & BM_EX_SINGLE_ECC) 704 p_Bm->f_Exception(p_Bm->h_App, e_BM_EX_SINGLE_ECC); 705 } 706 707 uint32_t BM_GetCounter(t_Handle h_Bm, e_BmCounters counter) 708 { 709 t_Bm *p_Bm = (t_Bm*)h_Bm; 710 711 SANITY_CHECK_RETURN_VALUE(p_Bm, E_INVALID_HANDLE, 0); 712 SANITY_CHECK_RETURN_VALUE(!p_Bm->p_BmDriverParams, E_INVALID_STATE, 0); 713 714 switch(counter) 715 { 716 case(e_BM_COUNTERS_FBPR): 717 return BmGetCounter(p_Bm, e_BM_IM_COUNTERS_FBPR, 0); 718 default: 719 break; 720 } 721 /* should never get here */ 722 ASSERT_COND(FALSE); 723 724 return 0; 725 } 726 727 t_Error BM_SetException(t_Handle h_Bm, e_BmExceptions exception, bool enable) 728 { 729 t_Bm *p_Bm = (t_Bm*)h_Bm; 730 uint32_t tmpReg, bitMask = 0; 731 732 SANITY_CHECK_RETURN_ERROR(p_Bm, E_INVALID_HANDLE); 733 734 if (p_Bm->guestId != NCSW_MASTER_ID) 735 RETURN_ERROR(WARNING, E_INVALID_OPERATION, ("Master Only")); 736 737 BM_ConfigException(p_Bm, exception, enable); 738 739 tmpReg = GET_UINT32(p_Bm->p_BmRegs->err_ier); 740 741 if(enable) 742 tmpReg |= bitMask; 743 else 744 tmpReg &= ~bitMask; 745 WRITE_UINT32(p_Bm->p_BmRegs->err_ier, tmpReg); 746 747 return E_OK; 748 } 749 750 t_Error BM_GetRevision(t_Handle h_Bm, t_BmRevisionInfo *p_BmRevisionInfo) 751 { 752 t_Bm *p_Bm = (t_Bm*)h_Bm; 753 754 SANITY_CHECK_RETURN_ERROR(p_Bm, E_INVALID_HANDLE); 755 SANITY_CHECK_RETURN_ERROR(p_BmRevisionInfo, E_NULL_POINTER); 756 757 return BmGetRevision(p_Bm, p_BmRevisionInfo); 758 } 759 760 #if (defined(DEBUG_ERRORS) && (DEBUG_ERRORS > 0)) 761 t_Error BM_DumpRegs(t_Handle h_Bm) 762 { 763 t_Bm *p_Bm = (t_Bm *)h_Bm; 764 765 DECLARE_DUMP; 766 767 if (p_Bm->guestId != NCSW_MASTER_ID) 768 RETURN_ERROR(WARNING, E_INVALID_OPERATION, ("Master Only")); 769 770 SANITY_CHECK_RETURN_ERROR(p_Bm, E_INVALID_HANDLE); 771 SANITY_CHECK_RETURN_ERROR(!p_Bm->p_BmDriverParams, E_INVALID_STATE); 772 773 DUMP_SUBTITLE(("\n")); 774 775 DUMP_TITLE(p_Bm->p_BmRegs, ("BmRegs Regs")); 776 777 DUMP_ARR(p_Bm->p_BmRegs, swdet); 778 DUMP_ARR(p_Bm->p_BmRegs, hwdet); 779 DUMP_ARR(p_Bm->p_BmRegs, swdxt); 780 DUMP_ARR(p_Bm->p_BmRegs, hwdxt); 781 DUMP_ARR(p_Bm->p_BmRegs, sdcnt); 782 DUMP_ARR(p_Bm->p_BmRegs, hdcnt); 783 DUMP_ARR(p_Bm->p_BmRegs, content); 784 DUMP_ARR(p_Bm->p_BmRegs, hdptr); 785 786 DUMP_VAR(p_Bm->p_BmRegs,fbpr_fpc); 787 DUMP_VAR(p_Bm->p_BmRegs,fbpr_fp_lwit); 788 789 DUMP_ARR(p_Bm->p_BmRegs, cmd_pm_cfg); 790 DUMP_ARR(p_Bm->p_BmRegs, fl_pm_cfg); 791 DUMP_VAR(p_Bm->p_BmRegs, ecsr); 792 DUMP_VAR(p_Bm->p_BmRegs, ecir); 793 DUMP_VAR(p_Bm->p_BmRegs, eadr); 794 DUMP_ARR(p_Bm->p_BmRegs, edata); 795 DUMP_VAR(p_Bm->p_BmRegs,sbet); 796 DUMP_VAR(p_Bm->p_BmRegs,efcr); 797 DUMP_VAR(p_Bm->p_BmRegs,efar); 798 DUMP_VAR(p_Bm->p_BmRegs,sbec0); 799 DUMP_VAR(p_Bm->p_BmRegs,sbec1); 800 DUMP_VAR(p_Bm->p_BmRegs,ip_rev_1); 801 DUMP_VAR(p_Bm->p_BmRegs,ip_rev_2); 802 DUMP_VAR(p_Bm->p_BmRegs,fbpr_bare); 803 DUMP_VAR(p_Bm->p_BmRegs,fbpr_bar); 804 DUMP_VAR(p_Bm->p_BmRegs,fbpr_ar); 805 DUMP_VAR(p_Bm->p_BmRegs,srcidr); 806 DUMP_VAR(p_Bm->p_BmRegs,liodnr); 807 DUMP_VAR(p_Bm->p_BmRegs,err_isr); 808 DUMP_VAR(p_Bm->p_BmRegs,err_ier); 809 DUMP_VAR(p_Bm->p_BmRegs,err_isdr); 810 DUMP_VAR(p_Bm->p_BmRegs,err_iir); 811 DUMP_VAR(p_Bm->p_BmRegs,err_ifr); 812 813 return E_OK; 814 } 815 #endif /* (defined(DEBUG_ERRORS) && ... */ 816