1 /******************************************************************************* 2 *Copyright (c) 2014 PMC-Sierra, Inc. All rights reserved. 3 * 4 *Redistribution and use in source and binary forms, with or without modification, are permitted provided 5 *that the following conditions are met: 6 *1. Redistributions of source code must retain the above copyright notice, this list of conditions and the 7 *following disclaimer. 8 *2. Redistributions in binary form must reproduce the above copyright notice, 9 *this list of conditions and the following disclaimer in the documentation and/or other materials provided 10 *with the distribution. 11 * 12 *THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND ANY EXPRESS OR IMPLIED 13 *WARRANTIES,INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS 14 *FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE 15 *FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT 16 *NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR 17 *BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 18 *LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS 19 *SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE 20 21 ********************************************************************************/ 22 #include <sys/cdefs.h> 23 __FBSDID("$FreeBSD$"); 24 #include <dev/pms/config.h> 25 26 #include <dev/pms/freebsd/driver/common/osenv.h> 27 #include <dev/pms/freebsd/driver/common/ostypes.h> 28 #include <dev/pms/freebsd/driver/common/osdebug.h> 29 30 #include <dev/pms/RefTisa/tisa/api/titypes.h> 31 32 #include <dev/pms/RefTisa/sallsdk/api/sa.h> 33 #include <dev/pms/RefTisa/sallsdk/api/saapi.h> 34 #include <dev/pms/RefTisa/sallsdk/api/saosapi.h> 35 36 #include <dev/pms/RefTisa/sat/api/sm.h> 37 #include <dev/pms/RefTisa/sat/api/smapi.h> 38 #include <dev/pms/RefTisa/sat/api/tdsmapi.h> 39 40 #include <dev/pms/RefTisa/sat/src/smdefs.h> 41 #include <dev/pms/RefTisa/sat/src/smproto.h> 42 #include <dev/pms/RefTisa/sat/src/smtypes.h> 43 44 /* start smapi defined APIs */ 45 osGLOBAL bit32 46 smRegisterDevice( 47 smRoot_t *smRoot, 48 agsaDevHandle_t *agDevHandle, 49 smDeviceHandle_t *smDeviceHandle, 50 agsaDevHandle_t *agExpDevHandle, 51 bit32 phyID, 52 bit32 DeviceType 53 ) 54 { 55 smDeviceData_t *oneDeviceData = agNULL; 56 57 SM_DBG2(("smRegisterDevice: start\n")); 58 59 if (smDeviceHandle == agNULL) 60 { 61 SM_DBG1(("smRegisterDevice: smDeviceHandle is NULL!!!\n")); 62 return SM_RC_FAILURE; 63 } 64 65 if (agDevHandle == agNULL) 66 { 67 SM_DBG1(("smRegisterDevice: agDevHandle is NULL!!!\n")); 68 return SM_RC_FAILURE; 69 } 70 71 oneDeviceData = smAddToSharedcontext(smRoot, agDevHandle, smDeviceHandle, agExpDevHandle, phyID); 72 if (oneDeviceData != agNULL) 73 { 74 oneDeviceData->satDeviceType = DeviceType; 75 return SM_RC_SUCCESS; 76 } 77 else 78 { 79 return SM_RC_FAILURE; 80 } 81 82 } 83 84 osGLOBAL bit32 85 smDeregisterDevice( 86 smRoot_t *smRoot, 87 agsaDevHandle_t *agDevHandle, 88 smDeviceHandle_t *smDeviceHandle 89 ) 90 { 91 bit32 status = SM_RC_FAILURE; 92 93 SM_DBG2(("smDeregisterDevice: start\n")); 94 95 if (smDeviceHandle == agNULL) 96 { 97 SM_DBG1(("smDeregisterDevice: smDeviceHandle is NULL!!!\n")); 98 return SM_RC_FAILURE; 99 } 100 101 if (agDevHandle == agNULL) 102 { 103 SM_DBG1(("smDeregisterDevice: agDevHandle is NULL!!!\n")); 104 return SM_RC_FAILURE; 105 } 106 107 status = smRemoveFromSharedcontext(smRoot, agDevHandle, smDeviceHandle); 108 109 return status; 110 } 111 112 osGLOBAL bit32 113 smIOAbort( 114 smRoot_t *smRoot, 115 smIORequest_t *tasktag 116 ) 117 118 { 119 smIntRoot_t *smIntRoot = (smIntRoot_t *)smRoot->smData; 120 smIntContext_t *smAllShared = (smIntContext_t *)&smIntRoot->smAllShared; 121 agsaRoot_t *agRoot; 122 smIORequestBody_t *smIORequestBody = agNULL; 123 smIORequestBody_t *smIONewRequestBody = agNULL; 124 agsaIORequest_t *agIORequest = agNULL; /* IO to be aborted */ 125 bit32 status = SM_RC_FAILURE; 126 agsaIORequest_t *agAbortIORequest; /* abort IO itself */ 127 smIORequestBody_t *smAbortIORequestBody; 128 #if 1 129 bit32 PhysUpper32; 130 bit32 PhysLower32; 131 bit32 memAllocStatus; 132 void *osMemHandle; 133 #endif 134 smSatIOContext_t *satIOContext; 135 smSatInternalIo_t *satIntIo; 136 smSatIOContext_t *satAbortIOContext; 137 138 SM_DBG1(("smIOAbort: start\n")); 139 SM_DBG2(("smIOAbort: tasktag %p\n", tasktag)); 140 /* 141 alloc smIORequestBody for abort itself 142 call saSATAAbort() 143 */ 144 145 agRoot = smAllShared->agRoot; 146 smIORequestBody = (smIORequestBody_t *)tasktag->smData; 147 148 if (smIORequestBody == agNULL) 149 { 150 SM_DBG1(("smIOAbort: smIORequestBody is NULL!!!\n")); 151 return SM_RC_FAILURE; 152 } 153 154 /* needs to distinguish internally generated or externally generated */ 155 satIOContext = &(smIORequestBody->transport.SATA.satIOContext); 156 satIntIo = satIOContext->satIntIoContext; 157 if (satIntIo == agNULL) 158 { 159 SM_DBG2(("smIOAbort: External, OS generated\n")); 160 agIORequest = &(smIORequestBody->agIORequest); 161 } 162 else 163 { 164 SM_DBG2(("smIOAbort: Internal, SM generated\n")); 165 smIONewRequestBody = (smIORequestBody_t *)satIntIo->satIntRequestBody; 166 agIORequest = &(smIONewRequestBody->agIORequest); 167 } 168 169 /* 170 allocate smAbortIORequestBody for abort request itself 171 */ 172 173 #if 1 174 /* allocating agIORequest for abort itself */ 175 memAllocStatus = tdsmAllocMemory( 176 smRoot, 177 &osMemHandle, 178 (void **)&smAbortIORequestBody, 179 &PhysUpper32, 180 &PhysLower32, 181 8, 182 sizeof(smIORequestBody_t), 183 agTRUE 184 ); 185 if (memAllocStatus != SM_RC_SUCCESS) 186 { 187 /* let os process IO */ 188 SM_DBG1(("smIOAbort: tdsmAllocMemory failed...!!!\n")); 189 return SM_RC_FAILURE; 190 } 191 192 if (smAbortIORequestBody == agNULL) 193 { 194 /* let os process IO */ 195 SM_DBG1(("smIOAbort: tdsmAllocMemory returned NULL smAbortIORequestBody!!!\n")); 196 return SM_RC_FAILURE; 197 } 198 199 smIOReInit(smRoot, smAbortIORequestBody); 200 201 /* setup task management structure */ 202 smAbortIORequestBody->IOType.InitiatorTMIO.osMemHandle = osMemHandle; 203 satAbortIOContext = &(smAbortIORequestBody->transport.SATA.satIOContext); 204 satAbortIOContext->smRequestBody = smAbortIORequestBody; 205 206 smAbortIORequestBody->smDevHandle = smIORequestBody->smDevHandle; 207 208 /* initialize agIORequest */ 209 agAbortIORequest = &(smAbortIORequestBody->agIORequest); 210 agAbortIORequest->osData = (void *) smAbortIORequestBody; 211 agAbortIORequest->sdkData = agNULL; /* LL takes care of this */ 212 213 /* remember IO to be aborted */ 214 smAbortIORequestBody->smIOToBeAbortedRequest = tasktag; 215 216 status = saSATAAbort(agRoot, agAbortIORequest, 0, agNULL, 0, agIORequest, smaSATAAbortCB); 217 218 SM_DBG2(("smIOAbort: return status=0x%x\n", status)); 219 220 #endif /* 1 */ 221 222 223 if (status == AGSA_RC_SUCCESS) 224 { 225 return SM_RC_SUCCESS; 226 } 227 else 228 { 229 SM_DBG1(("smIOAbort: failed to call saSATAAbort, status=%d!!!\n", status)); 230 tdsmFreeMemory(smRoot, 231 smAbortIORequestBody->IOType.InitiatorTMIO.osMemHandle, 232 sizeof(smIORequestBody_t) 233 ); 234 return SM_RC_FAILURE; 235 } 236 } 237 238 osGLOBAL bit32 239 smIOAbortAll( 240 smRoot_t *smRoot, 241 smDeviceHandle_t *smDeviceHandle 242 ) 243 { 244 smIntRoot_t *smIntRoot = (smIntRoot_t *)smRoot->smData; 245 smIntContext_t *smAllShared = (smIntContext_t *)&smIntRoot->smAllShared; 246 agsaRoot_t *agRoot; 247 bit32 status = SM_RC_FAILURE; 248 agsaIORequest_t *agAbortIORequest; 249 smIORequestBody_t *smAbortIORequestBody; 250 smSatIOContext_t *satAbortIOContext; 251 smDeviceData_t *oneDeviceData = agNULL; 252 agsaDevHandle_t *agDevHandle; 253 254 bit32 PhysUpper32; 255 bit32 PhysLower32; 256 bit32 memAllocStatus; 257 void *osMemHandle; 258 259 260 SM_DBG2(("smIOAbortAll: start\n")); 261 262 agRoot = smAllShared->agRoot; 263 264 if (smDeviceHandle == agNULL) 265 { 266 SM_DBG1(("smIOAbortAll: smDeviceHandle is NULL!!!\n")); 267 return SM_RC_FAILURE; 268 } 269 270 oneDeviceData = (smDeviceData_t *)smDeviceHandle->smData; 271 if (oneDeviceData == agNULL) 272 { 273 SM_DBG1(("smIOAbortAll: oneDeviceData is NULL!!!\n")); 274 return SM_RC_FAILURE; 275 } 276 if (oneDeviceData->valid == agFALSE) 277 { 278 SM_DBG1(("smIOAbortAll: oneDeviceData is not valid, did %d !!!\n", oneDeviceData->id)); 279 return SM_RC_FAILURE; 280 } 281 282 agDevHandle = oneDeviceData->agDevHandle; 283 if (agDevHandle == agNULL) 284 { 285 SM_DBG1(("smIOAbortAll: agDevHandle is NULL!!!\n")); 286 return SM_RC_FAILURE; 287 } 288 /* 289 smAbortIORequestBody = smDequeueIO(smRoot); 290 if (smAbortIORequestBody == agNULL) 291 { 292 SM_DBG1(("smIOAbortAll: empty freeIOList!!!\n")); 293 return SM_RC_FAILURE; 294 } 295 */ 296 /* allocating agIORequest for abort itself */ 297 memAllocStatus = tdsmAllocMemory( 298 smRoot, 299 &osMemHandle, 300 (void **)&smAbortIORequestBody, 301 &PhysUpper32, 302 &PhysLower32, 303 8, 304 sizeof(smIORequestBody_t), 305 agTRUE 306 ); 307 if (memAllocStatus != SM_RC_SUCCESS) 308 { 309 /* let os process IO */ 310 SM_DBG1(("smIOAbortAll: tdsmAllocMemory failed...!!!\n")); 311 return SM_RC_FAILURE; 312 } 313 314 if (smAbortIORequestBody == agNULL) 315 { 316 /* let os process IO */ 317 SM_DBG1(("smIOAbortAll: tdsmAllocMemory returned NULL smAbortIORequestBody!!!\n")); 318 return SM_RC_FAILURE; 319 } 320 321 smIOReInit(smRoot, smAbortIORequestBody); 322 323 /* setup task management structure */ 324 smAbortIORequestBody->IOType.InitiatorTMIO.osMemHandle = osMemHandle; 325 326 satAbortIOContext = &(smAbortIORequestBody->transport.SATA.satIOContext); 327 satAbortIOContext->smRequestBody = smAbortIORequestBody; 328 smAbortIORequestBody->smDevHandle = smDeviceHandle; 329 330 /* initialize agIORequest */ 331 agAbortIORequest = &(smAbortIORequestBody->agIORequest); 332 agAbortIORequest->osData = (void *) smAbortIORequestBody; 333 agAbortIORequest->sdkData = agNULL; /* LL takes care of this */ 334 335 oneDeviceData->OSAbortAll = agTRUE; 336 /* abort all */ 337 status = saSATAAbort(agRoot, agAbortIORequest, tdsmRotateQnumber(smRoot, smDeviceHandle), agDevHandle, 1, agNULL, smaSATAAbortCB); 338 if (status != AGSA_RC_SUCCESS) 339 { 340 SM_DBG1(("smIOAbortAll: failed to call saSATAAbort, status=%d!!!\n", status)); 341 tdsmFreeMemory(smRoot, 342 smAbortIORequestBody->IOType.InitiatorTMIO.osMemHandle, 343 sizeof(smIORequestBody_t) 344 ); 345 } 346 347 return status; 348 } 349 350 osGLOBAL bit32 351 smSuperIOStart( 352 smRoot_t *smRoot, 353 smIORequest_t *smIORequest, 354 smDeviceHandle_t *smDeviceHandle, 355 smSuperScsiInitiatorRequest_t *smSCSIRequest, 356 bit32 AddrHi, 357 bit32 AddrLo, 358 bit32 interruptContext 359 ) 360 { 361 smDeviceData_t *oneDeviceData = agNULL; 362 smIORequestBody_t *smIORequestBody = agNULL; 363 smSatIOContext_t *satIOContext = agNULL; 364 bit32 status = SM_RC_FAILURE; 365 366 SM_DBG2(("smSuperIOStart: start\n")); 367 368 oneDeviceData = (smDeviceData_t *)smDeviceHandle->smData; 369 if (oneDeviceData == agNULL) 370 { 371 SM_DBG1(("smSuperIOStart: oneDeviceData is NULL!!!\n")); 372 return SM_RC_FAILURE; 373 } 374 if (oneDeviceData->valid == agFALSE) 375 { 376 SM_DBG1(("smSuperIOStart: oneDeviceData is not valid, did %d !!!\n", oneDeviceData->id)); 377 return SM_RC_FAILURE; 378 } 379 smIORequestBody = (smIORequestBody_t*)smIORequest->smData;//smDequeueIO(smRoot); 380 381 if (smIORequestBody == agNULL) 382 { 383 SM_DBG1(("smSuperIOStart: smIORequestBody is NULL!!!\n")); 384 return SM_RC_FAILURE; 385 } 386 387 smIOReInit(smRoot, smIORequestBody); 388 389 SM_DBG3(("smSuperIOStart: io ID %d!!!\n", smIORequestBody->id )); 390 391 oneDeviceData->sasAddressHi = AddrHi; 392 oneDeviceData->sasAddressLo = AddrLo; 393 394 smIORequestBody->smIORequest = smIORequest; 395 smIORequestBody->smDevHandle = smDeviceHandle; 396 397 satIOContext = &(smIORequestBody->transport.SATA.satIOContext); 398 399 /* 400 * Need to initialize all the fields within satIOContext except 401 * reqType and satCompleteCB which will be set later in SM. 402 */ 403 smIORequestBody->transport.SATA.smSenseData.senseData = agNULL; 404 smIORequestBody->transport.SATA.smSenseData.senseLen = 0; 405 satIOContext->pSatDevData = oneDeviceData; 406 satIOContext->pFis = 407 &smIORequestBody->transport.SATA.agSATARequestBody.fis.fisRegHostToDev; 408 satIOContext->pScsiCmnd = &smSCSIRequest->scsiCmnd; 409 satIOContext->pSense = &smIORequestBody->transport.SATA.sensePayload; 410 satIOContext->pSmSenseData = &smIORequestBody->transport.SATA.smSenseData; 411 satIOContext->pSmSenseData->senseData = satIOContext->pSense; 412 /* satIOContext->pSense = (scsiRspSense_t *)satIOContext->pSmSenseData->senseData; */ 413 satIOContext->smRequestBody = smIORequestBody; 414 satIOContext->interruptContext = interruptContext; 415 satIOContext->psmDeviceHandle = smDeviceHandle; 416 satIOContext->smScsiXchg = smSCSIRequest; 417 satIOContext->superIOFlag = agTRUE; 418 // satIOContext->superIOFlag = agFALSE; 419 420 satIOContext->satIntIoContext = agNULL; 421 satIOContext->satOrgIOContext = agNULL; 422 /* satIOContext->tiIORequest = tiIORequest; */ 423 424 /* save context if we need to abort later */ 425 /*smIORequest->smData = smIORequestBody;*/ 426 427 /* followings are used only for internal IO */ 428 satIOContext->currentLBA = 0; 429 satIOContext->OrgTL = 0; 430 431 status = smsatIOStart(smRoot, smIORequest, smDeviceHandle, (smScsiInitiatorRequest_t *)smSCSIRequest, satIOContext); 432 433 return status; 434 } 435 436 /* 437 osGLOBAL bit32 438 tiINIIOStart( 439 tiRoot_t *tiRoot, 440 tiIORequest_t *tiIORequest, 441 tiDeviceHandle_t *tiDeviceHandle, 442 tiScsiInitiatorRequest_t *tiScsiRequest, 443 void *tiRequestBody, 444 bit32 interruptContext 445 ) 446 447 GLOBAL bit32 satIOStart( 448 tiRoot_t *tiRoot, 449 tiIORequest_t *tiIORequest, 450 tiDeviceHandle_t *tiDeviceHandle, 451 tiScsiInitiatorRequest_t *tiScsiRequest, 452 smSatIOContext_t *satIOContext 453 ) 454 smIOStart( 455 smRoot_t *smRoot, 456 smIORequest_t *smIORequest, 457 smDeviceHandle_t *smDeviceHandle, 458 smScsiInitiatorRequest_t *smSCSIRequest, 459 smIORequestBody_t *smRequestBody, 460 bit32 interruptContext 461 ) 462 463 464 */ 465 FORCEINLINE bit32 466 smIOStart( 467 smRoot_t *smRoot, 468 smIORequest_t *smIORequest, 469 smDeviceHandle_t *smDeviceHandle, 470 smScsiInitiatorRequest_t *smSCSIRequest, 471 bit32 interruptContext 472 ) 473 { 474 smDeviceData_t *oneDeviceData = agNULL; 475 smIORequestBody_t *smIORequestBody = agNULL; 476 smSatIOContext_t *satIOContext = agNULL; 477 bit32 status = SM_RC_FAILURE; 478 479 SM_DBG2(("smIOStart: start\n")); 480 481 oneDeviceData = (smDeviceData_t *)smDeviceHandle->smData; 482 if (oneDeviceData == agNULL) 483 { 484 SM_DBG1(("smIOStart: oneDeviceData is NULL!!!\n")); 485 return SM_RC_FAILURE; 486 } 487 if (oneDeviceData->valid == agFALSE) 488 { 489 SM_DBG1(("smIOStart: oneDeviceData is not valid, did %d !!!\n", oneDeviceData->id)); 490 return SM_RC_FAILURE; 491 } 492 smIORequestBody = (smIORequestBody_t*)smIORequest->smData;//smDequeueIO(smRoot); 493 494 if (smIORequestBody == agNULL) 495 { 496 SM_DBG1(("smIOStart: smIORequestBody is NULL!!!\n")); 497 return SM_RC_FAILURE; 498 } 499 500 smIOReInit(smRoot, smIORequestBody); 501 502 SM_DBG3(("smIOStart: io ID %d!!!\n", smIORequestBody->id )); 503 504 smIORequestBody->smIORequest = smIORequest; 505 smIORequestBody->smDevHandle = smDeviceHandle; 506 507 satIOContext = &(smIORequestBody->transport.SATA.satIOContext); 508 509 /* 510 * Need to initialize all the fields within satIOContext except 511 * reqType and satCompleteCB which will be set later in SM. 512 */ 513 smIORequestBody->transport.SATA.smSenseData.senseData = agNULL; 514 smIORequestBody->transport.SATA.smSenseData.senseLen = 0; 515 satIOContext->pSatDevData = oneDeviceData; 516 satIOContext->pFis = 517 &smIORequestBody->transport.SATA.agSATARequestBody.fis.fisRegHostToDev; 518 satIOContext->pScsiCmnd = &smSCSIRequest->scsiCmnd; 519 satIOContext->pSense = &smIORequestBody->transport.SATA.sensePayload; 520 satIOContext->pSmSenseData = &smIORequestBody->transport.SATA.smSenseData; 521 satIOContext->pSmSenseData->senseData = satIOContext->pSense; 522 /* satIOContext->pSense = (scsiRspSense_t *)satIOContext->pSmSenseData->senseData; */ 523 satIOContext->smRequestBody = smIORequestBody; 524 satIOContext->interruptContext = interruptContext; 525 satIOContext->psmDeviceHandle = smDeviceHandle; 526 satIOContext->smScsiXchg = smSCSIRequest; 527 satIOContext->superIOFlag = agFALSE; 528 529 satIOContext->satIntIoContext = agNULL; 530 satIOContext->satOrgIOContext = agNULL; 531 satIOContext->currentLBA = 0; 532 satIOContext->OrgTL = 0; 533 534 status = smsatIOStart(smRoot, smIORequest, smDeviceHandle, smSCSIRequest, satIOContext); 535 536 return status; 537 538 } 539 540 541 542 osGLOBAL bit32 543 smTaskManagement( 544 smRoot_t *smRoot, 545 smDeviceHandle_t *smDeviceHandle, 546 bit32 task, 547 smLUN_t *lun, 548 smIORequest_t *taskTag, /* io to be aborted */ 549 smIORequest_t *currentTaskTag /* task management */ 550 ) 551 { 552 smIntRoot_t *smIntRoot = (smIntRoot_t *)smRoot->smData; 553 smIntContext_t *smAllShared = (smIntContext_t *)&smIntRoot->smAllShared; 554 agsaRoot_t *agRoot = smAllShared->agRoot; 555 smDeviceData_t *oneDeviceData = agNULL; 556 smIORequestBody_t *smIORequestBody = agNULL; 557 bit32 status; 558 agsaContext_t *agContext = agNULL; 559 smSatIOContext_t *satIOContext; 560 561 SM_DBG1(("smTaskManagement: start\n")); 562 oneDeviceData = (smDeviceData_t *)smDeviceHandle->smData; 563 564 if (task == SM_LOGICAL_UNIT_RESET || task == SM_TARGET_WARM_RESET || task == SM_ABORT_TASK) 565 { 566 if (task == AG_LOGICAL_UNIT_RESET) 567 { 568 if ( (lun->lun[0] | lun->lun[1] | lun->lun[2] | lun->lun[3] | 569 lun->lun[4] | lun->lun[5] | lun->lun[6] | lun->lun[7] ) != 0 ) 570 { 571 SM_DBG1(("smTaskManagement: *** REJECT *** LUN not zero, did %d!!!\n", 572 oneDeviceData->id)); 573 return SM_RC_FAILURE; 574 } 575 } 576 577 oneDeviceData->satDriveState = SAT_DEV_STATE_IN_RECOVERY; 578 oneDeviceData->satAbortAfterReset = agFALSE; 579 580 saSetDeviceState(agRoot, 581 agNULL, 582 tdsmRotateQnumber(smRoot, smDeviceHandle), 583 oneDeviceData->agDevHandle, 584 SA_DS_IN_RECOVERY 585 ); 586 587 if (oneDeviceData->directlyAttached == agFALSE) 588 { 589 /* expander attached */ 590 SM_DBG1(("smTaskManagement: LUN reset or device reset expander attached!!!\n")); 591 status = smPhyControlSend(smRoot, 592 oneDeviceData, 593 SMP_PHY_CONTROL_HARD_RESET, 594 currentTaskTag, 595 tdsmRotateQnumber(smRoot, smDeviceHandle) 596 ); 597 return status; 598 } 599 else 600 { 601 SM_DBG1(("smTaskManagement: LUN reset or device reset directly attached\n")); 602 603 smIORequestBody = (smIORequestBody_t*)currentTaskTag->smData;//smDequeueIO(smRoot); 604 605 if (smIORequestBody == agNULL) 606 { 607 SM_DBG1(("smTaskManagement: smIORequestBody is NULL!!!\n")); 608 return SM_RC_FAILURE; 609 } 610 611 smIOReInit(smRoot, smIORequestBody); 612 613 satIOContext = &(smIORequestBody->transport.SATA.satIOContext); 614 satIOContext->smRequestBody = smIORequestBody; 615 smIORequestBody->smDevHandle = smDeviceHandle; 616 617 agContext = &(oneDeviceData->agDeviceResetContext); 618 agContext->osData = currentTaskTag; 619 620 status = saLocalPhyControl(agRoot, 621 agContext, 622 tdsmRotateQnumber(smRoot, smDeviceHandle) &0xFFFF, 623 oneDeviceData->phyID, 624 AGSA_PHY_HARD_RESET, 625 smLocalPhyControlCB 626 ); 627 628 if ( status == AGSA_RC_SUCCESS) 629 { 630 return SM_RC_SUCCESS; 631 } 632 else if (status == AGSA_RC_BUSY) 633 { 634 return SM_RC_BUSY; 635 } 636 else if (status == AGSA_RC_FAILURE) 637 { 638 return SM_RC_FAILURE; 639 } 640 else 641 { 642 SM_DBG1(("smTaskManagement: unknown status %d\n",status)); 643 return SM_RC_FAILURE; 644 } 645 } 646 } 647 else 648 { 649 /* smsatsmTaskManagement() which is satTM() */ 650 smIORequestBody = (smIORequestBody_t*)currentTaskTag->smData;//smDequeueIO(smRoot); 651 652 if (smIORequestBody == agNULL) 653 { 654 SM_DBG1(("smTaskManagement: smIORequestBody is NULL!!!\n")); 655 return SM_RC_FAILURE; 656 } 657 658 smIOReInit(smRoot, smIORequestBody); 659 /*currentTaskTag->smData = smIORequestBody;*/ 660 661 status = smsatTaskManagement(smRoot, 662 smDeviceHandle, 663 task, 664 lun, 665 taskTag, 666 currentTaskTag, 667 smIORequestBody 668 ); 669 670 return status; 671 } 672 return SM_RC_SUCCESS; 673 } 674 675 676 677 /********************************************************* end smapi defined APIS */ 678 /* counterpart is 679 smEnqueueIO(smRoot_t *smRoot, 680 smSatIOContext_t *satIOContext) 681 */ 682 osGLOBAL smIORequestBody_t * 683 smDequeueIO(smRoot_t *smRoot) 684 { 685 smIntRoot_t *smIntRoot = (smIntRoot_t *)smRoot->smData; 686 smIntContext_t *smAllShared = (smIntContext_t *)&smIntRoot->smAllShared; 687 smIORequestBody_t *smIORequestBody = agNULL; 688 smList_t *IOListList; 689 690 SM_DBG2(("smDequeueIO: start\n")); 691 692 tdsmSingleThreadedEnter(smRoot, SM_EXTERNAL_IO_LOCK); 693 if (SMLIST_EMPTY(&(smAllShared->freeIOList))) 694 { 695 SM_DBG1(("smDequeueIO: empty freeIOList!!!\n")); 696 tdsmSingleThreadedLeave(smRoot, SM_EXTERNAL_IO_LOCK); 697 return agNULL; 698 } 699 700 SMLIST_DEQUEUE_FROM_HEAD(&IOListList, &(smAllShared->freeIOList)); 701 smIORequestBody = SMLIST_OBJECT_BASE(smIORequestBody_t, satIoBodyLink, IOListList); 702 SMLIST_DEQUEUE_THIS(&(smIORequestBody->satIoBodyLink)); 703 SMLIST_ENQUEUE_AT_TAIL(&(smIORequestBody->satIoBodyLink), &(smAllShared->mainIOList)); 704 tdsmSingleThreadedLeave(smRoot, SM_EXTERNAL_IO_LOCK); 705 706 if (smIORequestBody->InUse == agTRUE) 707 { 708 SM_DBG1(("smDequeueIO: wrong. already in USE ID %d!!!!\n", smIORequestBody->id)); 709 } 710 smIOReInit(smRoot, smIORequestBody); 711 712 713 SM_DBG2(("smDequeueIO: io ID %d!\n", smIORequestBody->id)); 714 715 /* debugging */ 716 if (smIORequestBody->satIoBodyLink.flink == agNULL) 717 { 718 SM_DBG1(("smDequeueIO: io ID %d, flink is NULL!!!\n", smIORequestBody->id)); 719 } 720 if (smIORequestBody->satIoBodyLink.blink == agNULL) 721 { 722 SM_DBG1(("smDequeueIO: io ID %d, blink is NULL!!!\n", smIORequestBody->id)); 723 } 724 725 return smIORequestBody; 726 } 727 728 //start here 729 //compare with ossaSATAAbortCB() 730 //qqq1 731 osGLOBAL void 732 smsatAbort( 733 smRoot_t *smRoot, 734 agsaRoot_t *agRoot, 735 smSatIOContext_t *satIOContext 736 ) 737 { 738 smIORequestBody_t *smIORequestBody = agNULL; /* abort itself */ 739 smIORequestBody_t *smToBeAbortedIORequestBody; /* io to be aborted */ 740 agsaIORequest_t *agToBeAbortedIORequest; /* io to be aborted */ 741 agsaIORequest_t *agAbortIORequest; /* abort io itself */ 742 smSatIOContext_t *satAbortIOContext; 743 bit32 PhysUpper32; 744 bit32 PhysLower32; 745 bit32 memAllocStatus; 746 void *osMemHandle; 747 748 749 SM_DBG2(("smsatAbort: start\n")); 750 751 if (satIOContext == agNULL) 752 { 753 SM_DBG1(("smsatAbort: satIOContext is NULL, wrong!!!\n")); 754 return; 755 } 756 757 smToBeAbortedIORequestBody = (smIORequestBody_t *)satIOContext->smRequestBody; 758 agToBeAbortedIORequest = (agsaIORequest_t *)&(smToBeAbortedIORequestBody->agIORequest); 759 /* 760 smIORequestBody = smDequeueIO(smRoot); 761 762 if (smIORequestBody == agNULL) 763 { 764 SM_DBG1(("smsatAbort: empty freeIOList!!!\n")); 765 return; 766 } 767 */ 768 /* allocating agIORequest for abort itself */ 769 memAllocStatus = tdsmAllocMemory( 770 smRoot, 771 &osMemHandle, 772 (void **)&smIORequestBody, 773 &PhysUpper32, 774 &PhysLower32, 775 8, 776 sizeof(smIORequestBody_t), 777 agTRUE 778 ); 779 if (memAllocStatus != tiSuccess) 780 { 781 /* let os process IO */ 782 SM_DBG1(("smsatAbort: ostiAllocMemory failed...\n")); 783 return; 784 } 785 786 if (smIORequestBody == agNULL) 787 { 788 /* let os process IO */ 789 SM_DBG1(("smsatAbort: ostiAllocMemory returned NULL smIORequestBody\n")); 790 return; 791 } 792 smIOReInit(smRoot, smIORequestBody); 793 794 smIORequestBody->IOType.InitiatorTMIO.osMemHandle = osMemHandle; 795 smIORequestBody->smDevHandle = smToBeAbortedIORequestBody->smDevHandle; 796 /* initialize agIORequest */ 797 satAbortIOContext = &(smIORequestBody->transport.SATA.satIOContext); 798 satAbortIOContext->smRequestBody = smIORequestBody; 799 800 agAbortIORequest = &(smIORequestBody->agIORequest); 801 agAbortIORequest->osData = (void *) smIORequestBody; 802 agAbortIORequest->sdkData = agNULL; /* LL takes care of this */ 803 804 /* 805 * Issue abort 806 */ 807 saSATAAbort( agRoot, agAbortIORequest, 0, agNULL, 0, agToBeAbortedIORequest, smaSATAAbortCB); 808 809 810 SM_DBG1(("satAbort: end!!!\n")); 811 812 return; 813 } 814 815 osGLOBAL bit32 816 smsatStartCheckPowerMode( 817 smRoot_t *smRoot, 818 smIORequest_t *currentTaskTag, 819 smDeviceHandle_t *smDeviceHandle, 820 smScsiInitiatorRequest_t *smScsiRequest, 821 smSatIOContext_t *satIOContext 822 ) 823 { 824 smSatInternalIo_t *satIntIo = agNULL; 825 smDeviceData_t *oneDeviceData = agNULL; 826 smSatIOContext_t *satNewIOContext; 827 bit32 status; 828 829 SM_DBG1(("smsatStartCheckPowerMode: start\n")); 830 831 oneDeviceData = satIOContext->pSatDevData; 832 833 SM_DBG6(("smsatStartCheckPowerMode: before alloc\n")); 834 835 /* allocate any fis for seting SRT bit in device control */ 836 satIntIo = smsatAllocIntIoResource( smRoot, 837 currentTaskTag, 838 oneDeviceData, 839 0, 840 satIntIo); 841 842 SM_DBG6(("smsatStartCheckPowerMode: before after\n")); 843 844 if (satIntIo == agNULL) 845 { 846 SM_DBG1(("smsatStartCheckPowerMode: can't alloacate!!!\n")); 847 /*smEnqueueIO(smRoot, satIOContext);*/ 848 return SM_RC_FAILURE; 849 } 850 851 satNewIOContext = smsatPrepareNewIO(satIntIo, 852 currentTaskTag, 853 oneDeviceData, 854 agNULL, 855 satIOContext); 856 857 SM_DBG6(("smsatStartCheckPowerMode: TD satIOContext %p \n", satIOContext)); 858 SM_DBG6(("smsatStartCheckPowerMode: SM satNewIOContext %p \n", satNewIOContext)); 859 SM_DBG6(("smsatStartCheckPowerMode: TD smScsiXchg %p \n", satIOContext->smScsiXchg)); 860 SM_DBG6(("smsatStartCheckPowerMode: SM smScsiXchg %p \n", satNewIOContext->smScsiXchg)); 861 862 863 864 SM_DBG2(("smsatStartCheckPowerMode: satNewIOContext %p \n", satNewIOContext)); 865 866 status = smsatCheckPowerMode(smRoot, 867 &satIntIo->satIntSmIORequest, /* New smIORequest */ 868 smDeviceHandle, 869 satNewIOContext->smScsiXchg, /* New tiScsiInitiatorRequest_t *smScsiRequest, */ 870 satNewIOContext); 871 872 if (status != SM_RC_SUCCESS) 873 { 874 SM_DBG1(("smsatStartCheckPowerMode: failed in sending!!!\n")); 875 876 smsatFreeIntIoResource( smRoot, 877 oneDeviceData, 878 satIntIo); 879 880 /*smEnqueueIO(smRoot, satIOContext);*/ 881 882 return SM_RC_FAILURE; 883 } 884 885 886 SM_DBG6(("smsatStartCheckPowerMode: end\n")); 887 888 return status; 889 } 890 891 osGLOBAL bit32 892 smsatStartResetDevice( 893 smRoot_t *smRoot, 894 smIORequest_t *currentTaskTag, 895 smDeviceHandle_t *smDeviceHandle, 896 smScsiInitiatorRequest_t *smScsiRequest, 897 smSatIOContext_t *satIOContext 898 ) 899 { 900 smSatInternalIo_t *satIntIo = agNULL; 901 smDeviceData_t *oneDeviceData = agNULL; 902 smSatIOContext_t *satNewIOContext; 903 bit32 status; 904 905 SM_DBG1(("smsatStartResetDevice: start\n")); 906 907 oneDeviceData = satIOContext->pSatDevData; 908 909 SM_DBG6(("smsatStartResetDevice: before alloc\n")); 910 911 /* allocate any fis for seting SRT bit in device control */ 912 satIntIo = smsatAllocIntIoResource( smRoot, 913 currentTaskTag, 914 oneDeviceData, 915 0, 916 satIntIo); 917 918 SM_DBG6(("smsatStartResetDevice: before after\n")); 919 920 if (satIntIo == agNULL) 921 { 922 SM_DBG1(("smsatStartResetDevice: can't alloacate!!!\n")); 923 /*smEnqueueIO(smRoot, satIOContext);*/ 924 return SM_RC_FAILURE; 925 } 926 927 satNewIOContext = smsatPrepareNewIO(satIntIo, 928 currentTaskTag, 929 oneDeviceData, 930 agNULL, 931 satIOContext); 932 933 SM_DBG6(("smsatStartResetDevice: TD satIOContext %p \n", satIOContext)); 934 SM_DBG6(("smsatStartResetDevice: SM satNewIOContext %p \n", satNewIOContext)); 935 SM_DBG6(("smsatStartResetDevice: TD smScsiXchg %p \n", satIOContext->smScsiXchg)); 936 SM_DBG6(("smsatStartResetDevice: SM smScsiXchg %p \n", satNewIOContext->smScsiXchg)); 937 938 939 940 SM_DBG6(("smsatStartResetDevice: satNewIOContext %p \n", satNewIOContext)); 941 942 if (oneDeviceData->satDeviceType == SATA_ATAPI_DEVICE) 943 { 944 /*if ATAPI device, send DEVICE RESET command to ATAPI device*/ 945 status = smsatDeviceReset(smRoot, 946 &satIntIo->satIntSmIORequest, /* New smIORequest */ 947 smDeviceHandle, 948 satNewIOContext->smScsiXchg, /* New smScsiInitiatorRequest_t *smScsiRequest, NULL */ 949 satNewIOContext); 950 } 951 else 952 { 953 status = smsatResetDevice(smRoot, 954 &satIntIo->satIntSmIORequest, /* New smIORequest */ 955 smDeviceHandle, 956 satNewIOContext->smScsiXchg, /* New smScsiInitiatorRequest_t *smScsiRequest, NULL */ 957 satNewIOContext); 958 } 959 960 if (status != SM_RC_SUCCESS) 961 { 962 SM_DBG1(("smsatStartResetDevice: failed in sending!!!\n")); 963 964 smsatFreeIntIoResource( smRoot, 965 oneDeviceData, 966 satIntIo); 967 968 /*smEnqueueIO(smRoot, satIOContext);*/ 969 970 return SM_RC_FAILURE; 971 } 972 973 974 SM_DBG6(("smsatStartResetDevice: end\n")); 975 976 return status; 977 } 978 979 osGLOBAL bit32 980 smsatTmAbortTask( 981 smRoot_t *smRoot, 982 smIORequest_t *currentTaskTag, /* task management */ 983 smDeviceHandle_t *smDeviceHandle, 984 smScsiInitiatorRequest_t *smScsiRequest, /* NULL */ 985 smSatIOContext_t *satIOContext, /* task management */ 986 smIORequest_t *taskTag) /* io to be aborted */ 987 { 988 smDeviceData_t *oneDeviceData = agNULL; 989 smSatIOContext_t *satTempIOContext = agNULL; 990 smList_t *elementHdr; 991 bit32 found = agFALSE; 992 smIORequestBody_t *smIORequestBody = agNULL; 993 smIORequest_t *smIOReq = agNULL; 994 bit32 status; 995 996 SM_DBG1(("smsatTmAbortTask: start\n")); 997 998 oneDeviceData = (smDeviceData_t *)smDeviceHandle->smData; 999 1000 /* 1001 * Check that the only pending I/O matches taskTag. If not return tiError. 1002 */ 1003 tdsmSingleThreadedEnter(smRoot, SM_EXTERNAL_IO_LOCK); 1004 1005 elementHdr = oneDeviceData->satIoLinkList.flink; 1006 1007 while (elementHdr != &oneDeviceData->satIoLinkList) 1008 { 1009 satTempIOContext = SMLIST_OBJECT_BASE( smSatIOContext_t, 1010 satIoContextLink, 1011 elementHdr ); 1012 1013 if ( satTempIOContext != agNULL) 1014 { 1015 smIORequestBody = (smIORequestBody_t *) satTempIOContext->smRequestBody; 1016 smIOReq = smIORequestBody->smIORequest; 1017 } 1018 1019 elementHdr = elementHdr->flink; /* for the next while loop */ 1020 1021 /* 1022 * Check if the tag matches 1023 */ 1024 if ( smIOReq == taskTag) 1025 { 1026 found = agTRUE; 1027 satIOContext->satToBeAbortedIOContext = satTempIOContext; 1028 SM_DBG1(("smsatTmAbortTask: found matching tag.\n")); 1029 1030 break; 1031 1032 } /* if matching tag */ 1033 1034 } /* while loop */ 1035 1036 tdsmSingleThreadedLeave(smRoot, SM_EXTERNAL_IO_LOCK); 1037 1038 if (found == agFALSE ) 1039 { 1040 SM_DBG1(("smsatTmAbortTask: *** REJECT *** no match!!!\n")); 1041 1042 /*smEnqueueIO(smRoot, satIOContext);*/ 1043 /* clean up TD layer's smIORequestBody */ 1044 if (smIORequestBody) 1045 { 1046 if (smIORequestBody->IOType.InitiatorTMIO.osMemHandle != agNULL) 1047 { 1048 tdsmFreeMemory( 1049 smRoot, 1050 smIORequestBody->IOType.InitiatorTMIO.osMemHandle, 1051 sizeof(smIORequestBody_t) 1052 ); 1053 } 1054 } 1055 else 1056 { 1057 SM_DBG1(("smsatTmAbortTask: smIORequestBody is NULL!!!\n")); 1058 } 1059 1060 return SM_RC_FAILURE; 1061 } 1062 1063 if (satTempIOContext == agNULL) 1064 { 1065 SM_DBG1(("smsatTmAbortTask: satTempIOContext is NULL!!!\n")); 1066 return SM_RC_FAILURE; 1067 } 1068 1069 /* 1070 * Save smIORequest, will be returned at device reset completion to return 1071 * the TM completion. 1072 */ 1073 oneDeviceData->satTmTaskTag = currentTaskTag; 1074 1075 /* 1076 * Set flag to indicate device in recovery mode. 1077 */ 1078 oneDeviceData->satDriveState = SAT_DEV_STATE_IN_RECOVERY; 1079 1080 1081 /* 1082 * Issue SATA device reset or check power mode.. Set flag to to automatically abort 1083 * at the completion of SATA device reset. 1084 * SAT r09 p25 1085 */ 1086 oneDeviceData->satAbortAfterReset = agTRUE; 1087 1088 if ( (satTempIOContext->reqType == AGSA_SATA_PROTOCOL_FPDMA_WRITE) || 1089 (satTempIOContext->reqType == AGSA_SATA_PROTOCOL_FPDMA_READ) 1090 ) 1091 { 1092 SM_DBG1(("smsatTmAbortTask: calling satStartCheckPowerMode!!!\n")); 1093 /* send check power mode */ 1094 status = smsatStartCheckPowerMode( 1095 smRoot, 1096 currentTaskTag, /* currentTaskTag */ 1097 smDeviceHandle, 1098 smScsiRequest, /* NULL */ 1099 satIOContext 1100 ); 1101 } 1102 else 1103 { 1104 SM_DBG1(("smsatTmAbortTask: calling satStartResetDevice!!!\n")); 1105 /* send AGSA_SATA_PROTOCOL_SRST_ASSERT */ 1106 status = smsatStartResetDevice( 1107 smRoot, 1108 currentTaskTag, /* currentTaskTag */ 1109 smDeviceHandle, 1110 smScsiRequest, /* NULL */ 1111 satIOContext 1112 ); 1113 } 1114 return status; 1115 } 1116 1117 /* satTM() */ 1118 osGLOBAL bit32 1119 smsatTaskManagement( 1120 smRoot_t *smRoot, 1121 smDeviceHandle_t *smDeviceHandle, 1122 bit32 task, 1123 smLUN_t *lun, 1124 smIORequest_t *taskTag, /* io to be aborted */ 1125 smIORequest_t *currentTaskTag, /* task management */ 1126 smIORequestBody_t *smIORequestBody 1127 ) 1128 { 1129 smSatIOContext_t *satIOContext = agNULL; 1130 smDeviceData_t *oneDeviceData = agNULL; 1131 bit32 status; 1132 1133 SM_DBG1(("smsatTaskManagement: start\n")); 1134 oneDeviceData = (smDeviceData_t *)smDeviceHandle->smData; 1135 1136 satIOContext = &(smIORequestBody->transport.SATA.satIOContext); 1137 1138 satIOContext->pSatDevData = oneDeviceData; 1139 satIOContext->pFis = 1140 &(smIORequestBody->transport.SATA.agSATARequestBody.fis.fisRegHostToDev); 1141 1142 1143 satIOContext->smRequestBody = smIORequestBody; 1144 satIOContext->psmDeviceHandle = smDeviceHandle; 1145 satIOContext->satIntIoContext = agNULL; 1146 satIOContext->satOrgIOContext = agNULL; 1147 1148 /* followings are used only for internal IO */ 1149 satIOContext->currentLBA = 0; 1150 satIOContext->OrgTL = 0; 1151 1152 /* saving task in satIOContext */ 1153 satIOContext->TMF = task; 1154 1155 satIOContext->satToBeAbortedIOContext = agNULL; 1156 1157 if (task == AG_ABORT_TASK) 1158 { 1159 status = smsatTmAbortTask( smRoot, 1160 currentTaskTag, 1161 smDeviceHandle, 1162 agNULL, 1163 satIOContext, 1164 taskTag); 1165 1166 return status; 1167 } 1168 else 1169 { 1170 SM_DBG1(("smsatTaskManagement: UNSUPPORTED TM task=0x%x!!!\n", task )); 1171 1172 /*smEnqueueIO(smRoot, satIOContext);*/ 1173 1174 return SM_RC_FAILURE; 1175 } 1176 1177 return SM_RC_SUCCESS; 1178 } 1179 1180 1181 osGLOBAL bit32 1182 smPhyControlSend( 1183 smRoot_t *smRoot, 1184 smDeviceData_t *oneDeviceData, /* sata disk itself */ 1185 bit8 phyOp, 1186 smIORequest_t *CurrentTaskTag, 1187 bit32 queueNumber 1188 ) 1189 { 1190 smIntRoot_t *smIntRoot = (smIntRoot_t *)smRoot->smData; 1191 smIntContext_t *smAllShared = (smIntContext_t *)&smIntRoot->smAllShared; 1192 agsaRoot_t *agRoot = smAllShared->agRoot; 1193 agsaDevHandle_t *agExpDevHandle; 1194 smpReqPhyControl_t smpPhyControlReq; 1195 void *osMemHandle; 1196 bit32 PhysUpper32; 1197 bit32 PhysLower32; 1198 bit32 memAllocStatus; 1199 bit32 expectedRspLen = 0; 1200 smSMPRequestBody_t *smSMPRequestBody; 1201 agsaSASRequestBody_t *agSASRequestBody; 1202 agsaSMPFrame_t *agSMPFrame; 1203 agsaIORequest_t *agIORequest; 1204 // agsaDevHandle_t *agDevHandle; 1205 smSMPFrameHeader_t smSMPFrameHeader; 1206 bit32 status; 1207 bit8 *pSmpBody; /* smp payload itself w/o first 4 bytes(header) */ 1208 bit32 smpBodySize; /* smp payload size w/o first 4 bytes(header) */ 1209 bit32 agRequestType; 1210 1211 SM_DBG2(("smPhyControlSend: start\n")); 1212 1213 agExpDevHandle = oneDeviceData->agExpDevHandle; 1214 1215 if (agExpDevHandle == agNULL) 1216 { 1217 SM_DBG1(("smPhyControlSend: agExpDevHandle is NULL!!!\n")); 1218 return SM_RC_FAILURE; 1219 } 1220 1221 SM_DBG5(("smPhyControlSend: phyID %d\n", oneDeviceData->phyID)); 1222 1223 sm_memset(&smpPhyControlReq, 0, sizeof(smpReqPhyControl_t)); 1224 1225 /* fill in SMP payload */ 1226 smpPhyControlReq.phyIdentifier = (bit8)oneDeviceData->phyID; 1227 smpPhyControlReq.phyOperation = phyOp; 1228 1229 /* allocate smp and send it */ 1230 memAllocStatus = tdsmAllocMemory( 1231 smRoot, 1232 &osMemHandle, 1233 (void **)&smSMPRequestBody, 1234 &PhysUpper32, 1235 &PhysLower32, 1236 8, 1237 sizeof(smSMPRequestBody_t), 1238 agTRUE 1239 ); 1240 1241 if (memAllocStatus != SM_RC_SUCCESS) 1242 { 1243 SM_DBG1(("smPhyControlSend: tdsmAllocMemory failed...!!!\n")); 1244 return SM_RC_FAILURE; 1245 } 1246 1247 if (smSMPRequestBody == agNULL) 1248 { 1249 SM_DBG1(("smPhyControlSend: tdsmAllocMemory returned NULL smSMPRequestBody!!!\n")); 1250 return SM_RC_FAILURE; 1251 } 1252 1253 /* saves mem handle for freeing later */ 1254 smSMPRequestBody->osMemHandle = osMemHandle; 1255 1256 /* saves oneDeviceData */ 1257 smSMPRequestBody->smDeviceData = oneDeviceData; /* sata disk */ 1258 1259 /* saves oneDeviceData */ 1260 smSMPRequestBody->smDevHandle = oneDeviceData->smDevHandle; 1261 1262 // agDevHandle = oneDeviceData->agDevHandle; 1263 1264 /* save the callback funtion */ 1265 smSMPRequestBody->SMPCompletionFunc = smSMPCompleted; /* in satcb.c */ 1266 1267 /* for simulate warm target reset */ 1268 smSMPRequestBody->CurrentTaskTag = CurrentTaskTag; 1269 1270 if (CurrentTaskTag != agNULL) 1271 { 1272 CurrentTaskTag->smData = smSMPRequestBody; 1273 } 1274 1275 /* initializes the number of SMP retries */ 1276 smSMPRequestBody->retries = 0; 1277 1278 #ifdef TD_INTERNAL_DEBUG /* debugging */ 1279 SM_DBG4(("smPhyControlSend: SMPRequestbody %p\n", smSMPRequestBody)); 1280 SM_DBG4(("smPhyControlSend: callback fn %p\n", smSMPRequestBody->SMPCompletionFunc)); 1281 #endif 1282 1283 agIORequest = &(smSMPRequestBody->agIORequest); 1284 agIORequest->osData = (void *) smSMPRequestBody; 1285 agIORequest->sdkData = agNULL; /* SALL takes care of this */ 1286 1287 1288 agSASRequestBody = &(smSMPRequestBody->agSASRequestBody); 1289 agSMPFrame = &(agSASRequestBody->smpFrame); 1290 1291 SM_DBG3(("smPhyControlSend: agIORequest %p\n", agIORequest)); 1292 SM_DBG3(("smPhyControlSend: SMPRequestbody %p\n", smSMPRequestBody)); 1293 1294 expectedRspLen = 4; 1295 1296 pSmpBody = (bit8 *)&smpPhyControlReq; 1297 smpBodySize = sizeof(smpReqPhyControl_t); 1298 agRequestType = AGSA_SMP_INIT_REQ; 1299 1300 if (SMIsSPC(agRoot)) 1301 { 1302 if ( (smpBodySize + 4) <= SMP_DIRECT_PAYLOAD_LIMIT) /* 48 */ 1303 { 1304 SM_DBG3(("smPhyControlSend: DIRECT smp payload\n")); 1305 sm_memset(&smSMPFrameHeader, 0, sizeof(smSMPFrameHeader_t)); 1306 sm_memset(smSMPRequestBody->smpPayload, 0, SMP_DIRECT_PAYLOAD_LIMIT); 1307 1308 /* SMP header */ 1309 smSMPFrameHeader.smpFrameType = SMP_REQUEST; /* SMP request */ 1310 smSMPFrameHeader.smpFunction = (bit8)SMP_PHY_CONTROL; 1311 smSMPFrameHeader.smpFunctionResult = 0; 1312 smSMPFrameHeader.smpReserved = 0; 1313 1314 sm_memcpy(smSMPRequestBody->smpPayload, &smSMPFrameHeader, 4); 1315 sm_memcpy((smSMPRequestBody->smpPayload)+4, pSmpBody, smpBodySize); 1316 1317 /* direct SMP payload eg) REPORT_GENERAL, DISCOVER etc */ 1318 agSMPFrame->outFrameBuf = smSMPRequestBody->smpPayload; 1319 agSMPFrame->outFrameLen = smpBodySize + 4; /* without last 4 byte crc */ 1320 /* to specify DIRECT SMP response */ 1321 agSMPFrame->inFrameLen = 0; 1322 1323 /* temporary solution for T2D Combo*/ 1324 #if defined (INITIATOR_DRIVER) && defined (TARGET_DRIVER) 1325 /* force smp repsonse to be direct */ 1326 agSMPFrame->expectedRespLen = 0; 1327 #else 1328 agSMPFrame->expectedRespLen = expectedRspLen; 1329 #endif 1330 // smhexdump("smPhyControlSend", (bit8*)agSMPFrame->outFrameBuf, agSMPFrame->outFrameLen); 1331 // smhexdump("smPhyControlSend new", (bit8*)smSMPRequestBody->smpPayload, agSMPFrame->outFrameLen); 1332 // smhexdump("smPhyControlSend - smSMPRequestBody", (bit8*)smSMPRequestBody, sizeof(smSMPRequestBody_t)); 1333 } 1334 else 1335 { 1336 SM_DBG1(("smPhyControlSend: INDIRECT smp payload, not supported!!!\n")); 1337 tdsmFreeMemory( 1338 smRoot, 1339 osMemHandle, 1340 sizeof(smSMPRequestBody_t) 1341 ); 1342 1343 return SM_RC_FAILURE; 1344 } 1345 } 1346 else /* SPCv controller */ 1347 { 1348 /* only direct mode for both request and response */ 1349 SM_DBG3(("smPhyControlSend: DIRECT smp payload\n")); 1350 agSMPFrame->flag = 0; 1351 sm_memset(&smSMPFrameHeader, 0, sizeof(smSMPFrameHeader_t)); 1352 sm_memset(smSMPRequestBody->smpPayload, 0, SMP_DIRECT_PAYLOAD_LIMIT); 1353 1354 /* SMP header */ 1355 smSMPFrameHeader.smpFrameType = SMP_REQUEST; /* SMP request */ 1356 smSMPFrameHeader.smpFunction = (bit8)SMP_PHY_CONTROL; 1357 smSMPFrameHeader.smpFunctionResult = 0; 1358 smSMPFrameHeader.smpReserved = 0; 1359 1360 sm_memcpy(smSMPRequestBody->smpPayload, &smSMPFrameHeader, 4); 1361 sm_memcpy((smSMPRequestBody->smpPayload)+4, pSmpBody, smpBodySize); 1362 1363 /* direct SMP payload eg) REPORT_GENERAL, DISCOVER etc */ 1364 agSMPFrame->outFrameBuf = smSMPRequestBody->smpPayload; 1365 agSMPFrame->outFrameLen = smpBodySize + 4; /* without last 4 byte crc */ 1366 /* to specify DIRECT SMP response */ 1367 agSMPFrame->inFrameLen = 0; 1368 1369 /* temporary solution for T2D Combo*/ 1370 #if defined (INITIATOR_DRIVER) && defined (TARGET_DRIVER) 1371 /* force smp repsonse to be direct */ 1372 agSMPFrame->expectedRespLen = 0; 1373 #else 1374 agSMPFrame->expectedRespLen = expectedRspLen; 1375 #endif 1376 // smhexdump("smPhyControlSend", (bit8*)agSMPFrame->outFrameBuf, agSMPFrame->outFrameLen); 1377 // smhexdump("smPhyControlSend new", (bit8*)smSMPRequestBody->smpPayload, agSMPFrame->outFrameLen); 1378 // smhexdump("smPhyControlSend - smSMPRequestBody", (bit8*)smSMPRequestBody, sizeof(smSMPRequestBody_t)); 1379 } 1380 1381 status = saSMPStart( 1382 agRoot, 1383 agIORequest, 1384 queueNumber, 1385 agExpDevHandle, 1386 agRequestType, 1387 agSASRequestBody, 1388 &smSMPCompletedCB 1389 ); 1390 1391 if (status == AGSA_RC_SUCCESS) 1392 { 1393 return SM_RC_SUCCESS; 1394 } 1395 else if (status == AGSA_RC_BUSY) 1396 { 1397 SM_DBG1(("smPhyControlSend: saSMPStart is busy!!!\n")); 1398 tdsmFreeMemory( 1399 smRoot, 1400 osMemHandle, 1401 sizeof(smSMPRequestBody_t) 1402 ); 1403 1404 return SM_RC_BUSY; 1405 } 1406 else /* AGSA_RC_FAILURE */ 1407 { 1408 SM_DBG1(("smPhyControlSend: saSMPStart is failed. status %d!!!\n", status)); 1409 tdsmFreeMemory( 1410 smRoot, 1411 osMemHandle, 1412 sizeof(smSMPRequestBody_t) 1413 ); 1414 1415 return SM_RC_FAILURE; 1416 } 1417 } 1418 1419 /* free IO which are internally completed within SM 1420 counterpart is 1421 osGLOBAL smIORequestBody_t * 1422 smDequeueIO(smRoot_t *smRoot) 1423 */ 1424 osGLOBAL void 1425 smEnqueueIO( 1426 smRoot_t *smRoot, 1427 smSatIOContext_t *satIOContext 1428 ) 1429 { 1430 smIntRoot_t *smIntRoot = agNULL; 1431 smIntContext_t *smAllShared = agNULL; 1432 smIORequestBody_t *smIORequestBody; 1433 1434 SM_DBG3(("smEnqueueIO: start\n")); 1435 smIORequestBody = (smIORequestBody_t *)satIOContext->smRequestBody; 1436 smIntRoot = (smIntRoot_t *)smRoot->smData; 1437 smAllShared = (smIntContext_t *)&smIntRoot->smAllShared; 1438 1439 /* enque back to smAllShared->freeIOList */ 1440 if (satIOContext->satIntIoContext == agNULL) 1441 { 1442 SM_DBG2(("smEnqueueIO: external command!!!, io ID %d!!!\n", smIORequestBody->id)); 1443 /* debugging only */ 1444 if (smIORequestBody->satIoBodyLink.flink == agNULL) 1445 { 1446 SM_DBG1(("smEnqueueIO: external command!!!, io ID %d, flink is NULL!!!\n", smIORequestBody->id)); 1447 } 1448 if (smIORequestBody->satIoBodyLink.blink == agNULL) 1449 { 1450 SM_DBG1(("smEnqueueIO: external command!!!, io ID %d, blink is NULL!!!\n", smIORequestBody->id)); 1451 } 1452 } 1453 else 1454 { 1455 SM_DBG2(("smEnqueueIO: internal command!!!, io ID %d!!!\n", smIORequestBody->id)); 1456 /* debugging only */ 1457 if (smIORequestBody->satIoBodyLink.flink == agNULL) 1458 { 1459 SM_DBG1(("smEnqueueIO: internal command!!!, io ID %d, flink is NULL!!!\n", smIORequestBody->id)); 1460 } 1461 if (smIORequestBody->satIoBodyLink.blink == agNULL) 1462 { 1463 SM_DBG1(("smEnqueueIO: internal command!!!, io ID %d, blink is NULL!!!\n", smIORequestBody->id)); 1464 } 1465 } 1466 1467 if (smIORequestBody->smIORequest == agNULL) 1468 { 1469 SM_DBG1(("smEnqueueIO: smIORequest is NULL, io ID %d!!!\n", smIORequestBody->id)); 1470 } 1471 1472 if (smIORequestBody->InUse == agTRUE) 1473 { 1474 smIORequestBody->InUse = agFALSE; 1475 tdsmSingleThreadedEnter(smRoot, SM_EXTERNAL_IO_LOCK); 1476 SMLIST_DEQUEUE_THIS(&(smIORequestBody->satIoBodyLink)); 1477 SMLIST_ENQUEUE_AT_TAIL(&(smIORequestBody->satIoBodyLink), &(smAllShared->freeIOList)); 1478 tdsmSingleThreadedLeave(smRoot, SM_EXTERNAL_IO_LOCK); 1479 } 1480 else 1481 { 1482 SM_DBG2(("smEnqueueIO: check!!!, io ID %d!!!\n", smIORequestBody->id)); 1483 } 1484 1485 1486 return; 1487 } 1488 1489 FORCEINLINE void 1490 smsatFreeIntIoResource( 1491 smRoot_t *smRoot, 1492 smDeviceData_t *satDevData, 1493 smSatInternalIo_t *satIntIo 1494 ) 1495 { 1496 SM_DBG3(("smsatFreeIntIoResource: start\n")); 1497 1498 if (satIntIo == agNULL) 1499 { 1500 SM_DBG2(("smsatFreeIntIoResource: allowed call\n")); 1501 return; 1502 } 1503 1504 /* sets the original smIOrequest to agNULL for internally generated ATA cmnd */ 1505 satIntIo->satOrgSmIORequest = agNULL; 1506 1507 /* 1508 * Free DMA memory if previosly alocated 1509 */ 1510 if (satIntIo->satIntSmScsiXchg.scsiCmnd.expDataLength != 0) 1511 { 1512 SM_DBG3(("smsatFreeIntIoResource: DMA len %d\n", satIntIo->satIntDmaMem.totalLength)); 1513 SM_DBG3(("smsatFreeIntIoResource: pointer %p\n", satIntIo->satIntDmaMem.osHandle)); 1514 1515 tdsmFreeMemory( smRoot, 1516 satIntIo->satIntDmaMem.osHandle, 1517 satIntIo->satIntDmaMem.totalLength); 1518 satIntIo->satIntSmScsiXchg.scsiCmnd.expDataLength = 0; 1519 } 1520 1521 if (satIntIo->satIntReqBodyMem.totalLength != 0) 1522 { 1523 SM_DBG3(("smsatFreeIntIoResource: req body len %d\n", satIntIo->satIntReqBodyMem.totalLength)); 1524 /* 1525 * Free mem allocated for Req body 1526 */ 1527 tdsmFreeMemory( smRoot, 1528 satIntIo->satIntReqBodyMem.osHandle, 1529 satIntIo->satIntReqBodyMem.totalLength); 1530 1531 satIntIo->satIntReqBodyMem.totalLength = 0; 1532 } 1533 1534 SM_DBG3(("smsatFreeIntIoResource: satDevData %p satIntIo id %d\n", satDevData, satIntIo->id)); 1535 /* 1536 * Return satIntIo to the free list 1537 */ 1538 tdsmSingleThreadedEnter(smRoot, SM_INTERNAL_IO_LOCK); 1539 SMLIST_DEQUEUE_THIS (&(satIntIo->satIntIoLink)); 1540 SMLIST_ENQUEUE_AT_TAIL (&(satIntIo->satIntIoLink), &(satDevData->satFreeIntIoLinkList)); 1541 tdsmSingleThreadedLeave(smRoot, SM_INTERNAL_IO_LOCK); 1542 1543 return; 1544 } 1545 //start here 1546 osGLOBAL smSatInternalIo_t * 1547 smsatAllocIntIoResource( 1548 smRoot_t *smRoot, 1549 smIORequest_t *smIORequest, 1550 smDeviceData_t *satDevData, 1551 bit32 dmaAllocLength, 1552 smSatInternalIo_t *satIntIo) 1553 { 1554 smList_t *smList = agNULL; 1555 bit32 memAllocStatus; 1556 1557 SM_DBG3(("smsatAllocIntIoResource: start\n")); 1558 SM_DBG3(("smsatAllocIntIoResource: satIntIo %p\n", satIntIo)); 1559 if (satDevData == agNULL) 1560 { 1561 SM_DBG1(("smsatAllocIntIoResource: ***** ASSERT satDevData is null!!!\n")); 1562 return agNULL; 1563 } 1564 1565 tdsmSingleThreadedEnter(smRoot, SM_INTERNAL_IO_LOCK); 1566 if (!SMLIST_EMPTY(&(satDevData->satFreeIntIoLinkList))) 1567 { 1568 SMLIST_DEQUEUE_FROM_HEAD(&smList, &(satDevData->satFreeIntIoLinkList)); 1569 } 1570 else 1571 { 1572 tdsmSingleThreadedLeave(smRoot, SM_INTERNAL_IO_LOCK); 1573 SM_DBG1(("smsatAllocIntIoResource() no more internal free link!!!\n")); 1574 return agNULL; 1575 } 1576 1577 if (smList == agNULL) 1578 { 1579 tdsmSingleThreadedLeave(smRoot, SM_INTERNAL_IO_LOCK); 1580 SM_DBG1(("smsatAllocIntIoResource() FAIL to alloc satIntIo!!!\n")); 1581 return agNULL; 1582 } 1583 1584 satIntIo = SMLIST_OBJECT_BASE( smSatInternalIo_t, satIntIoLink, smList); 1585 SM_DBG3(("smsatAllocIntIoResource: satDevData %p satIntIo id %d\n", satDevData, satIntIo->id)); 1586 1587 /* Put in active list */ 1588 SMLIST_DEQUEUE_THIS (&(satIntIo->satIntIoLink)); 1589 SMLIST_ENQUEUE_AT_TAIL (&(satIntIo->satIntIoLink), &(satDevData->satActiveIntIoLinkList)); 1590 tdsmSingleThreadedLeave(smRoot, SM_INTERNAL_IO_LOCK); 1591 1592 #ifdef REMOVED 1593 /* Put in active list */ 1594 tdsmSingleThreadedEnter(smRoot, SM_INTERNAL_IO_LOCK); 1595 SMLIST_DEQUEUE_THIS (smList); 1596 SMLIST_ENQUEUE_AT_TAIL (smList, &(satDevData->satActiveIntIoLinkList)); 1597 tdsmSingleThreadedLeave(smRoot, SM_INTERNAL_IO_LOCK); 1598 1599 satIntIo = SMLIST_OBJECT_BASE( smSatInternalIo_t, satIntIoLink, smList); 1600 SM_DBG3(("smsatAllocIntIoResource: satDevData %p satIntIo id %d\n", satDevData, satIntIo->id)); 1601 #endif 1602 1603 /* 1604 typedef struct 1605 { 1606 tdList_t satIntIoLink; 1607 smIORequest_t satIntSmIORequest; 1608 void *satIntRequestBody; 1609 smScsiInitiatorRequest_t satIntSmScsiXchg; 1610 smMem_t satIntDmaMem; 1611 smMem_t satIntReqBodyMem; 1612 bit32 satIntFlag; 1613 } smSatInternalIo_t; 1614 */ 1615 1616 /* 1617 * Allocate mem for Request Body 1618 */ 1619 satIntIo->satIntReqBodyMem.totalLength = sizeof(smIORequestBody_t); 1620 1621 memAllocStatus = tdsmAllocMemory( smRoot, 1622 &satIntIo->satIntReqBodyMem.osHandle, 1623 (void **)&satIntIo->satIntRequestBody, 1624 &satIntIo->satIntReqBodyMem.physAddrUpper, 1625 &satIntIo->satIntReqBodyMem.physAddrLower, 1626 8, 1627 satIntIo->satIntReqBodyMem.totalLength, 1628 agTRUE ); 1629 1630 if (memAllocStatus != SM_RC_SUCCESS) 1631 { 1632 SM_DBG1(("smsatAllocIntIoResource() FAIL to alloc mem for Req Body!!!\n")); 1633 /* 1634 * Return satIntIo to the free list 1635 */ 1636 tdsmSingleThreadedEnter(smRoot, SM_INTERNAL_IO_LOCK); 1637 SMLIST_DEQUEUE_THIS (&satIntIo->satIntIoLink); 1638 SMLIST_ENQUEUE_AT_HEAD(&satIntIo->satIntIoLink, &satDevData->satFreeIntIoLinkList); 1639 tdsmSingleThreadedLeave(smRoot, SM_INTERNAL_IO_LOCK); 1640 1641 return agNULL; 1642 } 1643 1644 /* 1645 * Allocate DMA memory if required 1646 */ 1647 if (dmaAllocLength != 0) 1648 { 1649 satIntIo->satIntDmaMem.totalLength = dmaAllocLength; 1650 1651 memAllocStatus = tdsmAllocMemory( smRoot, 1652 &satIntIo->satIntDmaMem.osHandle, 1653 (void **)&satIntIo->satIntDmaMem.virtPtr, 1654 &satIntIo->satIntDmaMem.physAddrUpper, 1655 &satIntIo->satIntDmaMem.physAddrLower, 1656 8, 1657 satIntIo->satIntDmaMem.totalLength, 1658 agFALSE); 1659 SM_DBG3(("smsatAllocIntIoResource: len %d \n", satIntIo->satIntDmaMem.totalLength)); 1660 SM_DBG3(("smsatAllocIntIoResource: pointer %p \n", satIntIo->satIntDmaMem.osHandle)); 1661 1662 if (memAllocStatus != SM_RC_SUCCESS) 1663 { 1664 SM_DBG1(("smsatAllocIntIoResource() FAIL to alloc mem for DMA mem!!!\n")); 1665 /* 1666 * Return satIntIo to the free list 1667 */ 1668 tdsmSingleThreadedEnter(smRoot, SM_INTERNAL_IO_LOCK); 1669 SMLIST_DEQUEUE_THIS (&satIntIo->satIntIoLink); 1670 SMLIST_ENQUEUE_AT_HEAD(&satIntIo->satIntIoLink, &satDevData->satFreeIntIoLinkList); 1671 tdsmSingleThreadedLeave(smRoot, SM_INTERNAL_IO_LOCK); 1672 1673 /* 1674 * Free mem allocated for Req body 1675 */ 1676 tdsmFreeMemory( smRoot, 1677 satIntIo->satIntReqBodyMem.osHandle, 1678 satIntIo->satIntReqBodyMem.totalLength); 1679 1680 return agNULL; 1681 } 1682 } 1683 1684 /* 1685 typedef struct 1686 { 1687 smList_t satIntIoLink; 1688 smIORequest_t satIntSmIORequest; 1689 void *satIntRequestBody; 1690 smScsiInitiatorRequest_t satIntSmScsiXchg; 1691 smMem_t satIntDmaMem; 1692 smMem_t satIntReqBodyMem; 1693 bit32 satIntFlag; 1694 } smSatInternalIo_t; 1695 */ 1696 1697 /* 1698 * Initialize satIntSmIORequest field 1699 */ 1700 satIntIo->satIntSmIORequest.tdData = agNULL; /* Not used for internal SAT I/O */ 1701 satIntIo->satIntSmIORequest.smData = satIntIo->satIntRequestBody; 1702 1703 /* 1704 * saves the original smIOrequest 1705 */ 1706 satIntIo->satOrgSmIORequest = smIORequest; 1707 /* 1708 typedef struct tiIniScsiCmnd 1709 { 1710 tiLUN_t lun; 1711 bit32 expDataLength; 1712 bit32 taskAttribute; 1713 bit32 crn; 1714 bit8 cdb[16]; 1715 } tiIniScsiCmnd_t; 1716 1717 typedef struct tiScsiInitiatorExchange 1718 { 1719 void *sglVirtualAddr; 1720 tiIniScsiCmnd_t scsiCmnd; 1721 tiSgl_t agSgl1; 1722 tiSgl_t agSgl2; 1723 tiDataDirection_t dataDirection; 1724 } tiScsiInitiatorRequest_t; 1725 1726 */ 1727 1728 /* 1729 * Initialize satIntSmScsiXchg. Since the internal SAT request is NOT 1730 * originated from SCSI request, only the following fields are initialized: 1731 * - sglVirtualAddr if DMA transfer is involved 1732 * - agSgl1 if DMA transfer is involved 1733 * - expDataLength in scsiCmnd since this field is read by smsataLLIOStart() 1734 */ 1735 if (dmaAllocLength != 0) 1736 { 1737 satIntIo->satIntSmScsiXchg.sglVirtualAddr = satIntIo->satIntDmaMem.virtPtr; 1738 1739 OSSA_WRITE_LE_32(agNULL, &satIntIo->satIntSmScsiXchg.smSgl1.len, 0, 1740 satIntIo->satIntDmaMem.totalLength); 1741 satIntIo->satIntSmScsiXchg.smSgl1.lower = satIntIo->satIntDmaMem.physAddrLower; 1742 satIntIo->satIntSmScsiXchg.smSgl1.upper = satIntIo->satIntDmaMem.physAddrUpper; 1743 satIntIo->satIntSmScsiXchg.smSgl1.type = tiSgl; 1744 1745 satIntIo->satIntSmScsiXchg.scsiCmnd.expDataLength = satIntIo->satIntDmaMem.totalLength; 1746 } 1747 else 1748 { 1749 satIntIo->satIntSmScsiXchg.sglVirtualAddr = agNULL; 1750 1751 satIntIo->satIntSmScsiXchg.smSgl1.len = 0; 1752 satIntIo->satIntSmScsiXchg.smSgl1.lower = 0; 1753 satIntIo->satIntSmScsiXchg.smSgl1.upper = 0; 1754 satIntIo->satIntSmScsiXchg.smSgl1.type = tiSgl; 1755 1756 satIntIo->satIntSmScsiXchg.scsiCmnd.expDataLength = 0; 1757 } 1758 1759 SM_DBG5(("smsatAllocIntIoResource: satIntIo->satIntSmScsiXchg.agSgl1.len %d\n", satIntIo->satIntSmScsiXchg.smSgl1.len)); 1760 1761 SM_DBG5(("smsatAllocIntIoResource: satIntIo->satIntSmScsiXchg.agSgl1.upper %d\n", satIntIo->satIntSmScsiXchg.smSgl1.upper)); 1762 1763 SM_DBG5(("smsatAllocIntIoResource: satIntIo->satIntSmScsiXchg.agSgl1.lower %d\n", satIntIo->satIntSmScsiXchg.smSgl1.lower)); 1764 1765 SM_DBG5(("smsatAllocIntIoResource: satIntIo->satIntSmScsiXchg.agSgl1.type %d\n", satIntIo->satIntSmScsiXchg.smSgl1.type)); 1766 SM_DBG5(("smsatAllocIntIoResource: return satIntIo %p\n", satIntIo)); 1767 return satIntIo; 1768 } 1769 1770 osGLOBAL smDeviceData_t * 1771 smAddToSharedcontext( 1772 smRoot_t *smRoot, 1773 agsaDevHandle_t *agDevHandle, 1774 smDeviceHandle_t *smDeviceHandle, 1775 agsaDevHandle_t *agExpDevHandle, 1776 bit32 phyID 1777 ) 1778 { 1779 smIntRoot_t *smIntRoot = (smIntRoot_t *)smRoot->smData; 1780 smIntContext_t *smAllShared = (smIntContext_t *)&smIntRoot->smAllShared; 1781 smDeviceData_t *oneDeviceData = agNULL; 1782 smList_t *DeviceListList; 1783 bit32 new_device = agTRUE; 1784 1785 SM_DBG2(("smAddToSharedcontext: start\n")); 1786 1787 /* find a device's existence */ 1788 DeviceListList = smAllShared->MainDeviceList.flink; 1789 while (DeviceListList != &(smAllShared->MainDeviceList)) 1790 { 1791 oneDeviceData = SMLIST_OBJECT_BASE(smDeviceData_t, MainLink, DeviceListList); 1792 if (oneDeviceData == agNULL) 1793 { 1794 SM_DBG1(("smAddToSharedcontext: oneDeviceData is NULL!!!\n")); 1795 return agNULL; 1796 } 1797 if (oneDeviceData->agDevHandle == agDevHandle) 1798 { 1799 SM_DBG2(("smAddToSharedcontext: did %d\n", oneDeviceData->id)); 1800 new_device = agFALSE; 1801 break; 1802 } 1803 DeviceListList = DeviceListList->flink; 1804 } 1805 1806 /* new device */ 1807 if (new_device == agTRUE) 1808 { 1809 SM_DBG2(("smAddToSharedcontext: new device\n")); 1810 tdsmSingleThreadedEnter(smRoot, SM_DEVICE_LOCK); 1811 if (SMLIST_EMPTY(&(smAllShared->FreeDeviceList))) 1812 { 1813 tdsmSingleThreadedLeave(smRoot, SM_DEVICE_LOCK); 1814 SM_DBG1(("smAddToSharedcontext: empty DeviceData FreeLink!!!\n")); 1815 smDeviceHandle->smData = agNULL; 1816 return agNULL; 1817 } 1818 1819 SMLIST_DEQUEUE_FROM_HEAD(&DeviceListList, &(smAllShared->FreeDeviceList)); 1820 tdsmSingleThreadedLeave(smRoot, SM_DEVICE_LOCK); 1821 oneDeviceData = SMLIST_OBJECT_BASE(smDeviceData_t, FreeLink, DeviceListList); 1822 oneDeviceData->smRoot = smRoot; 1823 oneDeviceData->agDevHandle = agDevHandle; 1824 oneDeviceData->valid = agTRUE; 1825 smDeviceHandle->smData = oneDeviceData; 1826 oneDeviceData->smDevHandle = smDeviceHandle; 1827 if (agExpDevHandle == agNULL) 1828 { 1829 oneDeviceData->directlyAttached = agTRUE; 1830 } 1831 else 1832 { 1833 oneDeviceData->directlyAttached = agFALSE; 1834 } 1835 oneDeviceData->agExpDevHandle = agExpDevHandle; 1836 oneDeviceData->phyID = phyID; 1837 oneDeviceData->satPendingIO = 0; 1838 oneDeviceData->satPendingNCQIO = 0; 1839 oneDeviceData->satPendingNONNCQIO = 0; 1840 /* add the devicedata to the portcontext */ 1841 tdsmSingleThreadedEnter(smRoot, SM_DEVICE_LOCK); 1842 SMLIST_ENQUEUE_AT_TAIL(&(oneDeviceData->MainLink), &(smAllShared->MainDeviceList)); 1843 tdsmSingleThreadedLeave(smRoot, SM_DEVICE_LOCK); 1844 SM_DBG2(("smAddToSharedcontext: new case did %d\n", oneDeviceData->id)); 1845 } 1846 else 1847 { 1848 SM_DBG2(("smAddToSharedcontext: old device\n")); 1849 oneDeviceData->smRoot = smRoot; 1850 oneDeviceData->agDevHandle = agDevHandle; 1851 oneDeviceData->valid = agTRUE; 1852 smDeviceHandle->smData = oneDeviceData; 1853 oneDeviceData->smDevHandle = smDeviceHandle; 1854 if (agExpDevHandle == agNULL) 1855 { 1856 oneDeviceData->directlyAttached = agTRUE; 1857 } 1858 else 1859 { 1860 oneDeviceData->directlyAttached = agFALSE; 1861 } 1862 oneDeviceData->agExpDevHandle = agExpDevHandle; 1863 oneDeviceData->phyID = phyID; 1864 oneDeviceData->satPendingIO = 0; 1865 oneDeviceData->satPendingNCQIO = 0; 1866 oneDeviceData->satPendingNONNCQIO = 0; 1867 SM_DBG2(("smAddToSharedcontext: old case did %d\n", oneDeviceData->id)); 1868 } 1869 1870 return oneDeviceData; 1871 } 1872 1873 osGLOBAL bit32 1874 smRemoveFromSharedcontext( 1875 smRoot_t *smRoot, 1876 agsaDevHandle_t *agDevHandle, 1877 smDeviceHandle_t *smDeviceHandle 1878 ) 1879 { 1880 smIntRoot_t *smIntRoot = (smIntRoot_t *)smRoot->smData; 1881 smIntContext_t *smAllShared = (smIntContext_t *)&smIntRoot->smAllShared; 1882 smDeviceData_t *oneDeviceData = agNULL; 1883 1884 SM_DBG2(("smRemoveFromSharedcontext: start\n")); 1885 1886 //due to device all and completion 1887 //smDeviceHandle->smData = agNULL; 1888 1889 /* find oneDeviceData from MainLink */ 1890 oneDeviceData = smFindInSharedcontext(smRoot, agDevHandle); 1891 1892 if (oneDeviceData == agNULL) 1893 { 1894 return SM_RC_FAILURE; 1895 } 1896 else 1897 { 1898 if (oneDeviceData->valid == agTRUE) 1899 { 1900 smDeviceDataReInit(smRoot, oneDeviceData); 1901 tdsmSingleThreadedEnter(smRoot, SM_DEVICE_LOCK); 1902 SMLIST_DEQUEUE_THIS(&(oneDeviceData->MainLink)); 1903 SMLIST_ENQUEUE_AT_TAIL(&(oneDeviceData->FreeLink), &(smAllShared->FreeDeviceList)); 1904 tdsmSingleThreadedLeave(smRoot, SM_DEVICE_LOCK); 1905 return SM_RC_SUCCESS; 1906 } 1907 else 1908 { 1909 SM_DBG1(("smRemoveFromSharedcontext: did %d bad case!!!\n", oneDeviceData->id)); 1910 return SM_RC_FAILURE; 1911 } 1912 } 1913 1914 } 1915 1916 osGLOBAL smDeviceData_t * 1917 smFindInSharedcontext( 1918 smRoot_t *smRoot, 1919 agsaDevHandle_t *agDevHandle 1920 ) 1921 { 1922 smIntRoot_t *smIntRoot = (smIntRoot_t *)smRoot->smData; 1923 smIntContext_t *smAllShared = (smIntContext_t *)&smIntRoot->smAllShared; 1924 smDeviceData_t *oneDeviceData = agNULL; 1925 smList_t *DeviceListList; 1926 1927 SM_DBG2(("smFindInSharedcontext: start\n")); 1928 1929 tdsmSingleThreadedEnter(smRoot, SM_DEVICE_LOCK); 1930 if (SMLIST_EMPTY(&(smAllShared->MainDeviceList))) 1931 { 1932 SM_DBG1(("smFindInSharedcontext: empty MainDeviceList!!!\n")); 1933 tdsmSingleThreadedLeave(smRoot, SM_DEVICE_LOCK); 1934 return agNULL; 1935 } 1936 else 1937 { 1938 tdsmSingleThreadedLeave(smRoot, SM_DEVICE_LOCK); 1939 } 1940 1941 DeviceListList = smAllShared->MainDeviceList.flink; 1942 while (DeviceListList != &(smAllShared->MainDeviceList)) 1943 { 1944 oneDeviceData = SMLIST_OBJECT_BASE(smDeviceData_t, MainLink, DeviceListList); 1945 if (oneDeviceData == agNULL) 1946 { 1947 SM_DBG1(("smFindInSharedcontext: oneDeviceData is NULL!!!\n")); 1948 return agNULL; 1949 } 1950 if ((oneDeviceData->agDevHandle == agDevHandle) && 1951 (oneDeviceData->valid == agTRUE) 1952 ) 1953 { 1954 SM_DBG2(("smFindInSharedcontext: found, did %d\n", oneDeviceData->id)); 1955 return oneDeviceData; 1956 } 1957 DeviceListList = DeviceListList->flink; 1958 } 1959 SM_DBG2(("smFindInSharedcontext: not found\n")); 1960 return agNULL; 1961 } 1962 1963 osGLOBAL smSatIOContext_t * 1964 smsatPrepareNewIO( 1965 smSatInternalIo_t *satNewIntIo, 1966 smIORequest_t *smOrgIORequest, 1967 smDeviceData_t *satDevData, 1968 smIniScsiCmnd_t *scsiCmnd, 1969 smSatIOContext_t *satOrgIOContext 1970 ) 1971 { 1972 smSatIOContext_t *satNewIOContext; 1973 smIORequestBody_t *smNewIORequestBody; 1974 1975 SM_DBG3(("smsatPrepareNewIO: start\n")); 1976 1977 /* the one to be used; good 8/2/07 */ 1978 satNewIntIo->satOrgSmIORequest = smOrgIORequest; /* this is already done in 1979 smsatAllocIntIoResource() */ 1980 1981 smNewIORequestBody = (smIORequestBody_t *)satNewIntIo->satIntRequestBody; 1982 satNewIOContext = &(smNewIORequestBody->transport.SATA.satIOContext); 1983 1984 satNewIOContext->pSatDevData = satDevData; 1985 satNewIOContext->pFis = &(smNewIORequestBody->transport.SATA.agSATARequestBody.fis.fisRegHostToDev); 1986 satNewIOContext->pScsiCmnd = &(satNewIntIo->satIntSmScsiXchg.scsiCmnd); 1987 if (scsiCmnd != agNULL) 1988 { 1989 /* saves only CBD; not scsi command for LBA and number of blocks */ 1990 sm_memcpy(satNewIOContext->pScsiCmnd->cdb, scsiCmnd->cdb, 16); 1991 } 1992 satNewIOContext->pSense = &(smNewIORequestBody->transport.SATA.sensePayload); 1993 satNewIOContext->pSmSenseData = &(smNewIORequestBody->transport.SATA.smSenseData); 1994 satNewIOContext->pSmSenseData->senseData = satNewIOContext->pSense; 1995 satNewIOContext->smRequestBody = satNewIntIo->satIntRequestBody; 1996 satNewIOContext->interruptContext = satNewIOContext->interruptContext; 1997 satNewIOContext->satIntIoContext = satNewIntIo; 1998 satNewIOContext->psmDeviceHandle = satOrgIOContext->psmDeviceHandle; 1999 satNewIOContext->satOrgIOContext = satOrgIOContext; 2000 /* saves tiScsiXchg; only for writesame10() */ 2001 satNewIOContext->smScsiXchg = satOrgIOContext->smScsiXchg; 2002 2003 return satNewIOContext; 2004 } 2005 2006 2007 osGLOBAL void 2008 smsatSetDevInfo( 2009 smDeviceData_t *oneDeviceData, 2010 agsaSATAIdentifyData_t *SATAIdData 2011 ) 2012 { 2013 SM_DBG3(("smsatSetDevInfo: start\n")); 2014 2015 oneDeviceData->satDriveState = SAT_DEV_STATE_NORMAL; 2016 oneDeviceData->satFormatState = agFALSE; 2017 oneDeviceData->satDeviceFaultState = agFALSE; 2018 oneDeviceData->satTmTaskTag = agNULL; 2019 oneDeviceData->satAbortAfterReset = agFALSE; 2020 oneDeviceData->satAbortCalled = agFALSE; 2021 oneDeviceData->satSectorDone = 0; 2022 2023 /* Qeueu depth, Word 75 */ 2024 oneDeviceData->satNCQMaxIO = SATAIdData->queueDepth + 1; 2025 SM_DBG3(("smsatSetDevInfo: max queue depth %d\n",oneDeviceData->satNCQMaxIO)); 2026 2027 /* Support NCQ, if Word 76 bit 8 is set */ 2028 if (SATAIdData->sataCapabilities & 0x100) 2029 { 2030 SM_DBG3(("smsatSetDevInfo: device supports NCQ\n")); 2031 oneDeviceData->satNCQ = agTRUE; 2032 } 2033 else 2034 { 2035 SM_DBG3(("smsatSetDevInfo: no NCQ\n")); 2036 oneDeviceData->satNCQ = agFALSE; 2037 } 2038 2039 /* Support 48 bit addressing, if Word 83 bit 10 and Word 86 bit 10 are set */ 2040 if ((SATAIdData->commandSetSupported1 & 0x400) && 2041 (SATAIdData->commandSetFeatureEnabled1 & 0x400) ) 2042 { 2043 SM_DBG3(("smsatSetDevInfo: support 48 bit addressing\n")); 2044 oneDeviceData->sat48BitSupport = agTRUE; 2045 } 2046 else 2047 { 2048 SM_DBG3(("smsatSetDevInfo: NO 48 bit addressing\n")); 2049 oneDeviceData->sat48BitSupport = agFALSE; 2050 } 2051 2052 /* Support SMART Self Test, word84 bit 1 */ 2053 if (SATAIdData->commandSetFeatureSupportedExt & 0x02) 2054 { 2055 SM_DBG3(("smsatSetDevInfo: SMART self-test supported \n")); 2056 oneDeviceData->satSMARTSelfTest = agTRUE; 2057 } 2058 else 2059 { 2060 SM_DBG3(("smsatSetDevInfo: no SMART self-test suppored\n")); 2061 oneDeviceData->satSMARTSelfTest = agFALSE; 2062 } 2063 2064 /* Support SMART feature set, word82 bit 0 */ 2065 if (SATAIdData->commandSetSupported & 0x01) 2066 { 2067 SM_DBG3(("smsatSetDevInfo: SMART feature set supported \n")); 2068 oneDeviceData->satSMARTFeatureSet = agTRUE; 2069 } 2070 else 2071 { 2072 SM_DBG3(("smsatSetDevInfo: no SMART feature set suppored\n")); 2073 oneDeviceData->satSMARTFeatureSet = agFALSE; 2074 } 2075 2076 /* Support SMART enabled, word85 bit 0 */ 2077 if (SATAIdData->commandSetFeatureEnabled & 0x01) 2078 { 2079 SM_DBG3(("smsatSetDevInfo: SMART enabled \n")); 2080 oneDeviceData->satSMARTEnabled = agTRUE; 2081 } 2082 else 2083 { 2084 SM_DBG3(("smsatSetDevInfo: no SMART enabled\n")); 2085 oneDeviceData->satSMARTEnabled = agFALSE; 2086 } 2087 2088 oneDeviceData->satVerifyState = 0; 2089 2090 /* Removable Media feature set support, word82 bit 2 */ 2091 if (SATAIdData->commandSetSupported & 0x4) 2092 { 2093 SM_DBG3(("smsatSetDevInfo: Removable Media supported \n")); 2094 oneDeviceData->satRemovableMedia = agTRUE; 2095 } 2096 else 2097 { 2098 SM_DBG3(("smsatSetDevInfo: no Removable Media suppored\n")); 2099 oneDeviceData->satRemovableMedia = agFALSE; 2100 } 2101 2102 /* Removable Media feature set enabled, word 85, bit 2 */ 2103 if (SATAIdData->commandSetFeatureEnabled & 0x4) 2104 { 2105 SM_DBG3(("smsatSetDevInfo: Removable Media enabled\n")); 2106 oneDeviceData->satRemovableMediaEnabled = agTRUE; 2107 } 2108 else 2109 { 2110 SM_DBG3(("smsatSetDevInfo: no Removable Media enabled\n")); 2111 oneDeviceData->satRemovableMediaEnabled = agFALSE; 2112 } 2113 2114 /* DMA Support, word49 bit8 */ 2115 if (SATAIdData->dma_lba_iod_ios_stimer & 0x100) 2116 { 2117 SM_DBG3(("smsatSetDevInfo: DMA supported \n")); 2118 oneDeviceData->satDMASupport = agTRUE; 2119 } 2120 else 2121 { 2122 SM_DBG3(("smsatSetDevInfo: no DMA suppored\n")); 2123 oneDeviceData->satDMASupport = agFALSE; 2124 } 2125 2126 /* Support DMADIR, if Word 62 bit 8 is set */ 2127 if (SATAIdData->word62_74[0] & 0x8000) 2128 { 2129 SM_DBG3(("satSetDevInfo: DMADIR enabled\n")); 2130 oneDeviceData->satDMADIRSupport = agTRUE; 2131 } 2132 else 2133 { 2134 SM_DBG3(("satSetDevInfo: DMADIR disabled\n")); 2135 oneDeviceData->satDMADIRSupport = agFALSE; 2136 } 2137 2138 /* DMA Enabled, word88 bit0-6, bit8-14*/ 2139 /* 0x7F7F = 0111 1111 0111 1111*/ 2140 if (SATAIdData->ultraDMAModes & 0x7F7F) 2141 { 2142 SM_DBG3(("smsatSetDevInfo: DMA enabled \n")); 2143 oneDeviceData->satDMAEnabled = agTRUE; 2144 if (SATAIdData->ultraDMAModes & 0x40) 2145 { 2146 oneDeviceData->satUltraDMAMode = 6; 2147 } 2148 else if (SATAIdData->ultraDMAModes & 0x20) 2149 { 2150 oneDeviceData->satUltraDMAMode = 5; 2151 } 2152 else if (SATAIdData->ultraDMAModes & 0x10) 2153 { 2154 oneDeviceData->satUltraDMAMode = 4; 2155 } 2156 else if (SATAIdData->ultraDMAModes & 0x08) 2157 { 2158 oneDeviceData->satUltraDMAMode = 3; 2159 } 2160 else if (SATAIdData->ultraDMAModes & 0x04) 2161 { 2162 oneDeviceData->satUltraDMAMode = 2; 2163 } 2164 else if (SATAIdData->ultraDMAModes & 0x01) 2165 { 2166 oneDeviceData->satUltraDMAMode = 1; 2167 } 2168 } 2169 else 2170 { 2171 SM_DBG3(("smsatSetDevInfo: no DMA enabled\n")); 2172 oneDeviceData->satDMAEnabled = agFALSE; 2173 oneDeviceData->satUltraDMAMode = 0; 2174 } 2175 2176 /* 2177 setting MaxUserAddrSectors: max user addressable setctors 2178 word60 - 61, should be 0x 0F FF FF FF 2179 */ 2180 oneDeviceData->satMaxUserAddrSectors 2181 = (SATAIdData->numOfUserAddressableSectorsHi << (8*2) ) 2182 + SATAIdData->numOfUserAddressableSectorsLo; 2183 SM_DBG3(("smsatSetDevInfo: MaxUserAddrSectors 0x%x decimal %d\n", oneDeviceData->satMaxUserAddrSectors, oneDeviceData->satMaxUserAddrSectors)); 2184 2185 /* Read Look-ahead is supported */ 2186 if (SATAIdData->commandSetSupported & 0x40) 2187 { 2188 SM_DBG3(("smsatSetDevInfo: Read Look-ahead is supported\n")); 2189 oneDeviceData->satReadLookAheadSupport= agTRUE; 2190 } 2191 else 2192 { 2193 SM_DBG3(("smsatSetDevInfo: Read Look-ahead is not supported\n")); 2194 oneDeviceData->satReadLookAheadSupport= agFALSE; 2195 } 2196 2197 /* Volatile Write Cache is supported */ 2198 if (SATAIdData->commandSetSupported & 0x20) 2199 { 2200 SM_DBG3(("smsatSetDevInfo: Volatile Write Cache is supported\n")); 2201 oneDeviceData->satVolatileWriteCacheSupport = agTRUE; 2202 } 2203 else 2204 { 2205 SM_DBG3(("smsatSetDevInfo: Volatile Write Cache is not supported\n")); 2206 oneDeviceData->satVolatileWriteCacheSupport = agFALSE; 2207 } 2208 2209 /* write cache enabled for caching mode page SAT Table 67 p69, word85 bit5 */ 2210 if (SATAIdData->commandSetFeatureEnabled & 0x20) 2211 { 2212 SM_DBG3(("smsatSetDevInfo: write cache enabled\n")); 2213 oneDeviceData->satWriteCacheEnabled = agTRUE; 2214 } 2215 else 2216 { 2217 SM_DBG3(("smsatSetDevInfo: no write cache enabled\n")); 2218 oneDeviceData->satWriteCacheEnabled = agFALSE; 2219 } 2220 2221 /* look ahead enabled for caching mode page SAT Table 67 p69, word85 bit6 */ 2222 if (SATAIdData->commandSetFeatureEnabled & 0x40) 2223 { 2224 SM_DBG3(("smsatSetDevInfo: look ahead enabled\n")); 2225 oneDeviceData->satLookAheadEnabled = agTRUE; 2226 } 2227 else 2228 { 2229 SM_DBG3(("smsatSetDevInfo: no look ahead enabled\n")); 2230 oneDeviceData->satLookAheadEnabled = agFALSE; 2231 } 2232 2233 /* Support WWN, if Word 87 bit 8 is set */ 2234 if (SATAIdData->commandSetFeatureDefault & 0x100) 2235 { 2236 SM_DBG3(("smsatSetDevInfo: device supports WWN\n")); 2237 oneDeviceData->satWWNSupport = agTRUE; 2238 } 2239 else 2240 { 2241 SM_DBG3(("smsatSetDevInfo: no WWN\n")); 2242 oneDeviceData->satWWNSupport = agFALSE; 2243 } 2244 2245 /* Support DMA Setup Auto-Activate, if Word 78 bit 2 is set */ 2246 if (SATAIdData->sataFeaturesSupported & 0x4) 2247 { 2248 SM_DBG3(("smsatSetDevInfo: device supports DMA Setup Auto-Activate\n")); 2249 oneDeviceData->satDMASetupAA = agTRUE; 2250 } 2251 else 2252 { 2253 SM_DBG3(("smsatSetDevInfo: no DMA Setup Auto-Activate\n")); 2254 oneDeviceData->satDMASetupAA = agFALSE; 2255 } 2256 2257 /* Support NCQ Queue Management Command, if Word 77 bit 5 is set */ 2258 if (SATAIdData->word77 & 0x10) 2259 { 2260 SM_DBG3(("smsatSetDevInfo: device supports NCQ Queue Management Command\n")); 2261 oneDeviceData->satNCQQMgntCmd = agTRUE; 2262 } 2263 else 2264 { 2265 SM_DBG3(("smsatSetDevInfo: no NCQ Queue Management Command\n")); 2266 oneDeviceData->satNCQQMgntCmd = agFALSE; 2267 } 2268 return; 2269 } 2270 2271 2272 osGLOBAL void 2273 smsatInquiryStandard( 2274 bit8 *pInquiry, 2275 agsaSATAIdentifyData_t *pSATAIdData, 2276 smIniScsiCmnd_t *scsiCmnd 2277 ) 2278 { 2279 smLUN_t *pLun; 2280 pLun = &scsiCmnd->lun; 2281 2282 /* 2283 Assumption: Basic Task Mangement is supported 2284 -> BQUE 1 and CMDQUE 0, SPC-4, Table96, p147 2285 */ 2286 /* 2287 See SPC-4, 6.4.2, p 143 2288 and SAT revision 8, 8.1.2, p 28 2289 */ 2290 SM_DBG5(("smsatInquiryStandard: start\n")); 2291 2292 if (pInquiry == agNULL) 2293 { 2294 SM_DBG1(("smsatInquiryStandard: pInquiry is NULL, wrong\n")); 2295 return; 2296 } 2297 else 2298 { 2299 SM_DBG5(("smsatInquiryStandard: pInquiry is NOT NULL\n")); 2300 } 2301 /* 2302 * Reject all other LUN other than LUN 0. 2303 */ 2304 if ( ((pLun->lun[0] | pLun->lun[1] | pLun->lun[2] | pLun->lun[3] | 2305 pLun->lun[4] | pLun->lun[5] | pLun->lun[6] | pLun->lun[7] ) != 0) ) 2306 { 2307 /* SAT Spec Table 8, p27, footnote 'a' */ 2308 pInquiry[0] = 0x7F; 2309 2310 } 2311 else 2312 { 2313 pInquiry[0] = 0x00; 2314 } 2315 2316 if (pSATAIdData->rm_ataDevice & ATA_REMOVABLE_MEDIA_DEVICE_MASK ) 2317 { 2318 pInquiry[1] = 0x80; 2319 } 2320 else 2321 { 2322 pInquiry[1] = 0x00; 2323 } 2324 pInquiry[2] = 0x05; /* SPC-3 */ 2325 pInquiry[3] = 0x12; /* set HiSup 1; resp data format set to 2 */ 2326 pInquiry[4] = 0x1F; /* 35 - 4 = 31; Additional length */ 2327 pInquiry[5] = 0x00; 2328 /* The following two are for task management. SAT Rev8, p20 */ 2329 if (pSATAIdData->sataCapabilities & 0x100) 2330 { 2331 /* NCQ supported; multiple outstanding SCSI IO are supported */ 2332 pInquiry[6] = 0x00; /* BQUE bit is not set */ 2333 pInquiry[7] = 0x02; /* CMDQUE bit is set */ 2334 } 2335 else 2336 { 2337 pInquiry[6] = 0x80; /* BQUE bit is set */ 2338 pInquiry[7] = 0x00; /* CMDQUE bit is not set */ 2339 } 2340 /* 2341 * Vendor ID. 2342 */ 2343 sm_strncpy((char*)&pInquiry[8], AG_SAT_VENDOR_ID_STRING, 8); /* 8 bytes */ 2344 2345 /* 2346 * Product ID 2347 */ 2348 /* when flipped by LL */ 2349 pInquiry[16] = pSATAIdData->modelNumber[1]; 2350 pInquiry[17] = pSATAIdData->modelNumber[0]; 2351 pInquiry[18] = pSATAIdData->modelNumber[3]; 2352 pInquiry[19] = pSATAIdData->modelNumber[2]; 2353 pInquiry[20] = pSATAIdData->modelNumber[5]; 2354 pInquiry[21] = pSATAIdData->modelNumber[4]; 2355 pInquiry[22] = pSATAIdData->modelNumber[7]; 2356 pInquiry[23] = pSATAIdData->modelNumber[6]; 2357 pInquiry[24] = pSATAIdData->modelNumber[9]; 2358 pInquiry[25] = pSATAIdData->modelNumber[8]; 2359 pInquiry[26] = pSATAIdData->modelNumber[11]; 2360 pInquiry[27] = pSATAIdData->modelNumber[10]; 2361 pInquiry[28] = pSATAIdData->modelNumber[13]; 2362 pInquiry[29] = pSATAIdData->modelNumber[12]; 2363 pInquiry[30] = pSATAIdData->modelNumber[15]; 2364 pInquiry[31] = pSATAIdData->modelNumber[14]; 2365 2366 /* when flipped */ 2367 /* 2368 * Product Revision level. 2369 */ 2370 2371 /* 2372 * If the IDENTIFY DEVICE data received in words 25 and 26 from the ATA 2373 * device are ASCII spaces (20h), do this translation. 2374 */ 2375 if ( (pSATAIdData->firmwareVersion[4] == 0x20 ) && 2376 (pSATAIdData->firmwareVersion[5] == 0x20 ) && 2377 (pSATAIdData->firmwareVersion[6] == 0x20 ) && 2378 (pSATAIdData->firmwareVersion[7] == 0x20 ) 2379 ) 2380 { 2381 pInquiry[32] = pSATAIdData->firmwareVersion[1]; 2382 pInquiry[33] = pSATAIdData->firmwareVersion[0]; 2383 pInquiry[34] = pSATAIdData->firmwareVersion[3]; 2384 pInquiry[35] = pSATAIdData->firmwareVersion[2]; 2385 } 2386 else 2387 { 2388 pInquiry[32] = pSATAIdData->firmwareVersion[5]; 2389 pInquiry[33] = pSATAIdData->firmwareVersion[4]; 2390 pInquiry[34] = pSATAIdData->firmwareVersion[7]; 2391 pInquiry[35] = pSATAIdData->firmwareVersion[6]; 2392 } 2393 2394 2395 #ifdef REMOVED 2396 /* 2397 * Product ID 2398 */ 2399 /* when flipped by LL */ 2400 pInquiry[16] = pSATAIdData->modelNumber[0]; 2401 pInquiry[17] = pSATAIdData->modelNumber[1]; 2402 pInquiry[18] = pSATAIdData->modelNumber[2]; 2403 pInquiry[19] = pSATAIdData->modelNumber[3]; 2404 pInquiry[20] = pSATAIdData->modelNumber[4]; 2405 pInquiry[21] = pSATAIdData->modelNumber[5]; 2406 pInquiry[22] = pSATAIdData->modelNumber[6]; 2407 pInquiry[23] = pSATAIdData->modelNumber[7]; 2408 pInquiry[24] = pSATAIdData->modelNumber[8]; 2409 pInquiry[25] = pSATAIdData->modelNumber[9]; 2410 pInquiry[26] = pSATAIdData->modelNumber[10]; 2411 pInquiry[27] = pSATAIdData->modelNumber[11]; 2412 pInquiry[28] = pSATAIdData->modelNumber[12]; 2413 pInquiry[29] = pSATAIdData->modelNumber[13]; 2414 pInquiry[30] = pSATAIdData->modelNumber[14]; 2415 pInquiry[31] = pSATAIdData->modelNumber[15]; 2416 2417 /* when flipped */ 2418 /* 2419 * Product Revision level. 2420 */ 2421 2422 /* 2423 * If the IDENTIFY DEVICE data received in words 25 and 26 from the ATA 2424 * device are ASCII spaces (20h), do this translation. 2425 */ 2426 if ( (pSATAIdData->firmwareVersion[4] == 0x20 ) && 2427 (pSATAIdData->firmwareVersion[5] == 0x20 ) && 2428 (pSATAIdData->firmwareVersion[6] == 0x20 ) && 2429 (pSATAIdData->firmwareVersion[7] == 0x20 ) 2430 ) 2431 { 2432 pInquiry[32] = pSATAIdData->firmwareVersion[0]; 2433 pInquiry[33] = pSATAIdData->firmwareVersion[1]; 2434 pInquiry[34] = pSATAIdData->firmwareVersion[2]; 2435 pInquiry[35] = pSATAIdData->firmwareVersion[3]; 2436 } 2437 else 2438 { 2439 pInquiry[32] = pSATAIdData->firmwareVersion[4]; 2440 pInquiry[33] = pSATAIdData->firmwareVersion[5]; 2441 pInquiry[34] = pSATAIdData->firmwareVersion[6]; 2442 pInquiry[35] = pSATAIdData->firmwareVersion[7]; 2443 } 2444 #endif 2445 2446 SM_DBG5(("smsatInquiryStandard: end\n")); 2447 2448 return; 2449 } 2450 2451 osGLOBAL void 2452 smsatInquiryPage0( 2453 bit8 *pInquiry, 2454 agsaSATAIdentifyData_t *pSATAIdData 2455 ) 2456 { 2457 SM_DBG5(("smsatInquiryPage0: start\n")); 2458 2459 /* 2460 See SPC-4, 7.6.9, p 345 2461 and SAT revision 8, 10.3.2, p 77 2462 */ 2463 pInquiry[0] = 0x00; 2464 pInquiry[1] = 0x00; /* page code */ 2465 pInquiry[2] = 0x00; /* reserved */ 2466 pInquiry[3] = 8 - 3; /* last index(in this case, 6) - 3; page length */ 2467 2468 /* supported vpd page list */ 2469 pInquiry[4] = 0x00; /* page 0x00 supported */ 2470 pInquiry[5] = 0x80; /* page 0x80 supported */ 2471 pInquiry[6] = 0x83; /* page 0x83 supported */ 2472 pInquiry[7] = 0x89; /* page 0x89 supported */ 2473 pInquiry[8] = 0xB1; /* page 0xB1 supported */ 2474 2475 return; 2476 } 2477 2478 osGLOBAL void 2479 smsatInquiryPage83( 2480 bit8 *pInquiry, 2481 agsaSATAIdentifyData_t *pSATAIdData, 2482 smDeviceData_t *oneDeviceData 2483 ) 2484 { 2485 satSimpleSATAIdentifyData_t *pSimpleData; 2486 2487 /* 2488 * When translating the fields, in some cases using the simple form of SATA 2489 * Identify Device Data is easier. So we define it here. 2490 * Both pSimpleData and pSATAIdData points to the same data. 2491 */ 2492 pSimpleData = ( satSimpleSATAIdentifyData_t *)pSATAIdData; 2493 2494 SM_DBG5(("smsatInquiryPage83: start\n")); 2495 2496 pInquiry[0] = 0x00; 2497 pInquiry[1] = 0x83; /* page code */ 2498 pInquiry[2] = 0; /* Reserved */ 2499 /* 2500 * If the ATA device returns word 87 bit 8 set to one in its IDENTIFY DEVICE 2501 * data indicating that it supports the WORLD WIDE NAME field 2502 * (i.e., words 108-111), the SATL shall include an identification descriptor 2503 * containing a logical unit name. 2504 */ 2505 if ( oneDeviceData->satWWNSupport) 2506 { 2507 #ifndef PMC_FREEBSD 2508 /* Fill in SAT Rev8 Table85 */ 2509 /* 2510 * Logical unit name derived from the world wide name. 2511 */ 2512 pInquiry[3] = 12; /* 15-3; page length, no addition ID descriptor assumed*/ 2513 2514 /* 2515 * Identifier descriptor 2516 */ 2517 pInquiry[4] = 0x01; /* Code set: binary codes */ 2518 pInquiry[5] = 0x03; /* Identifier type : NAA */ 2519 pInquiry[6] = 0x00; /* Reserved */ 2520 pInquiry[7] = 0x08; /* Identifier length */ 2521 2522 /* Bit 4-7 NAA field, bit 0-3 MSB of IEEE Company ID */ 2523 pInquiry[8] = (bit8)((pSATAIdData->namingAuthority) >> 8); 2524 pInquiry[9] = (bit8)((pSATAIdData->namingAuthority) & 0xFF); /* IEEE Company ID */ 2525 pInquiry[10] = (bit8)((pSATAIdData->namingAuthority1) >> 8); /* IEEE Company ID */ 2526 /* Bit 4-7 LSB of IEEE Company ID, bit 0-3 MSB of Vendor Specific ID */ 2527 pInquiry[11] = (bit8)((pSATAIdData->namingAuthority1) & 0xFF); 2528 pInquiry[12] = (bit8)((pSATAIdData->uniqueID_bit16_31) >> 8); /* Vendor Specific ID */ 2529 pInquiry[13] = (bit8)((pSATAIdData->uniqueID_bit16_31) & 0xFF); /* Vendor Specific ID */ 2530 pInquiry[14] = (bit8)((pSATAIdData->uniqueID_bit0_15) >> 8); /* Vendor Specific ID */ 2531 pInquiry[15] = (bit8)((pSATAIdData->uniqueID_bit0_15) & 0xFF); /* Vendor Specific ID */ 2532 2533 #else 2534 2535 /* For FreeBSD */ 2536 2537 /* Fill in SAT Rev8 Table85 */ 2538 /* 2539 * Logical unit name derived from the world wide name. 2540 */ 2541 pInquiry[3] = 24; /* 35-3; page length, no addition ID descriptor assumed*/ 2542 /* 2543 * Identifier descriptor 2544 */ 2545 pInquiry[4] = 0x01; /* Code set: binary codes; this is proto_codeset in FreeBSD */ 2546 pInquiry[5] = 0x03; /* Identifier type : NAA ; this is id_type in FreeBSD*/ 2547 pInquiry[6] = 0x00; /* Reserved */ 2548 pInquiry[7] = 0x08; /* Identifier length */ 2549 2550 /* Bit 4-7 NAA field, bit 0-3 MSB of IEEE Company ID */ 2551 pInquiry[8] = (bit8)((pSATAIdData->namingAuthority) >> 8); 2552 pInquiry[9] = (bit8)((pSATAIdData->namingAuthority) & 0xFF); /* IEEE Company ID */ 2553 pInquiry[10] = (bit8)((pSATAIdData->namingAuthority1) >> 8); /* IEEE Company ID */ 2554 /* Bit 4-7 LSB of IEEE Company ID, bit 0-3 MSB of Vendor Specific ID */ 2555 pInquiry[11] = (bit8)((pSATAIdData->namingAuthority1) & 0xFF); 2556 pInquiry[12] = (bit8)((pSATAIdData->uniqueID_bit16_31) >> 8); /* Vendor Specific ID */ 2557 pInquiry[13] = (bit8)((pSATAIdData->uniqueID_bit16_31) & 0xFF); /* Vendor Specific ID */ 2558 pInquiry[14] = (bit8)((pSATAIdData->uniqueID_bit0_15) >> 8); /* Vendor Specific ID */ 2559 pInquiry[15] = (bit8)((pSATAIdData->uniqueID_bit0_15) & 0xFF); /* Vendor Specific ID */ 2560 2561 pInquiry[16] = 0x61; /* Code set: binary codes; this is proto_codeset in FreeBSD; SCSI_PROTO_SAS and SVPD_ID_CODESET_BINARY */ 2562 pInquiry[17] = 0x93; /* Identifier type : NAA ; this is id_type in FreeBSD; PIV set, ASSOCIATION is 01b and NAA (3h) */ 2563 pInquiry[18] = 0x00; /* Reserved */ 2564 pInquiry[19] = 0x08; /* Identifier length */ 2565 2566 SM_DBG5(("smsatInquiryPage83: sasAddressHi 0x%08x\n", oneDeviceData->sasAddressHi)); 2567 SM_DBG5(("smsatInquiryPage83: sasAddressLo 0x%08x\n", oneDeviceData->sasAddressLo)); 2568 2569 /* SAS address of SATA */ 2570 pInquiry[20] = ((oneDeviceData->sasAddressHi) & 0xFF000000 ) >> 24; 2571 pInquiry[21] = ((oneDeviceData->sasAddressHi) & 0xFF0000 ) >> 16; 2572 pInquiry[22] = ((oneDeviceData->sasAddressHi) & 0xFF00 ) >> 8; 2573 pInquiry[23] = (oneDeviceData->sasAddressHi) & 0xFF; 2574 pInquiry[24] = ((oneDeviceData->sasAddressLo) & 0xFF000000 ) >> 24; 2575 pInquiry[25] = ((oneDeviceData->sasAddressLo) & 0xFF0000 ) >> 16; 2576 pInquiry[26] = ((oneDeviceData->sasAddressLo) & 0xFF00 ) >> 8; 2577 pInquiry[27] = (oneDeviceData->sasAddressLo) & 0xFF; 2578 #endif 2579 } 2580 else 2581 { 2582 #ifndef PMC_FREEBSD 2583 /* Fill in SAT Rev8 Table86 */ 2584 /* 2585 * Logical unit name derived from the model number and serial number. 2586 */ 2587 pInquiry[3] = 72; /* 75 - 3; page length */ 2588 2589 /* 2590 * Identifier descriptor 2591 */ 2592 pInquiry[4] = 0x02; /* Code set: ASCII codes */ 2593 pInquiry[5] = 0x01; /* Identifier type : T10 vendor ID based */ 2594 pInquiry[6] = 0x00; /* Reserved */ 2595 pInquiry[7] = 0x44; /* 0x44, 68 Identifier length */ 2596 2597 /* Byte 8 to 15 is the vendor id string 'ATA '. */ 2598 sm_strncpy((char *)&pInquiry[8], AG_SAT_VENDOR_ID_STRING, 8); 2599 2600 2601 /* 2602 * Byte 16 to 75 is vendor specific id 2603 */ 2604 pInquiry[16] = (bit8)((pSimpleData->word[27]) >> 8); 2605 pInquiry[17] = (bit8)((pSimpleData->word[27]) & 0x00ff); 2606 pInquiry[18] = (bit8)((pSimpleData->word[28]) >> 8); 2607 pInquiry[19] = (bit8)((pSimpleData->word[28]) & 0x00ff); 2608 pInquiry[20] = (bit8)((pSimpleData->word[29]) >> 8); 2609 pInquiry[21] = (bit8)((pSimpleData->word[29]) & 0x00ff); 2610 pInquiry[22] = (bit8)((pSimpleData->word[30]) >> 8); 2611 pInquiry[23] = (bit8)((pSimpleData->word[30]) & 0x00ff); 2612 pInquiry[24] = (bit8)((pSimpleData->word[31]) >> 8); 2613 pInquiry[25] = (bit8)((pSimpleData->word[31]) & 0x00ff); 2614 pInquiry[26] = (bit8)((pSimpleData->word[32]) >> 8); 2615 pInquiry[27] = (bit8)((pSimpleData->word[32]) & 0x00ff); 2616 pInquiry[28] = (bit8)((pSimpleData->word[33]) >> 8); 2617 pInquiry[29] = (bit8)((pSimpleData->word[33]) & 0x00ff); 2618 pInquiry[30] = (bit8)((pSimpleData->word[34]) >> 8); 2619 pInquiry[31] = (bit8)((pSimpleData->word[34]) & 0x00ff); 2620 pInquiry[32] = (bit8)((pSimpleData->word[35]) >> 8); 2621 pInquiry[33] = (bit8)((pSimpleData->word[35]) & 0x00ff); 2622 pInquiry[34] = (bit8)((pSimpleData->word[36]) >> 8); 2623 pInquiry[35] = (bit8)((pSimpleData->word[36]) & 0x00ff); 2624 pInquiry[36] = (bit8)((pSimpleData->word[37]) >> 8); 2625 pInquiry[37] = (bit8)((pSimpleData->word[37]) & 0x00ff); 2626 pInquiry[38] = (bit8)((pSimpleData->word[38]) >> 8); 2627 pInquiry[39] = (bit8)((pSimpleData->word[38]) & 0x00ff); 2628 pInquiry[40] = (bit8)((pSimpleData->word[39]) >> 8); 2629 pInquiry[41] = (bit8)((pSimpleData->word[39]) & 0x00ff); 2630 pInquiry[42] = (bit8)((pSimpleData->word[40]) >> 8); 2631 pInquiry[43] = (bit8)((pSimpleData->word[40]) & 0x00ff); 2632 pInquiry[44] = (bit8)((pSimpleData->word[41]) >> 8); 2633 pInquiry[45] = (bit8)((pSimpleData->word[41]) & 0x00ff); 2634 pInquiry[46] = (bit8)((pSimpleData->word[42]) >> 8); 2635 pInquiry[47] = (bit8)((pSimpleData->word[42]) & 0x00ff); 2636 pInquiry[48] = (bit8)((pSimpleData->word[43]) >> 8); 2637 pInquiry[49] = (bit8)((pSimpleData->word[43]) & 0x00ff); 2638 pInquiry[50] = (bit8)((pSimpleData->word[44]) >> 8); 2639 pInquiry[51] = (bit8)((pSimpleData->word[44]) & 0x00ff); 2640 pInquiry[52] = (bit8)((pSimpleData->word[45]) >> 8); 2641 pInquiry[53] = (bit8)((pSimpleData->word[45]) & 0x00ff); 2642 pInquiry[54] = (bit8)((pSimpleData->word[46]) >> 8); 2643 pInquiry[55] = (bit8)((pSimpleData->word[46]) & 0x00ff); 2644 2645 pInquiry[56] = (bit8)((pSimpleData->word[10]) >> 8); 2646 pInquiry[57] = (bit8)((pSimpleData->word[10]) & 0x00ff); 2647 pInquiry[58] = (bit8)((pSimpleData->word[11]) >> 8); 2648 pInquiry[59] = (bit8)((pSimpleData->word[11]) & 0x00ff); 2649 pInquiry[60] = (bit8)((pSimpleData->word[12]) >> 8); 2650 pInquiry[61] = (bit8)((pSimpleData->word[12]) & 0x00ff); 2651 pInquiry[62] = (bit8)((pSimpleData->word[13]) >> 8); 2652 pInquiry[63] = (bit8)((pSimpleData->word[13]) & 0x00ff); 2653 pInquiry[64] = (bit8)((pSimpleData->word[14]) >> 8); 2654 pInquiry[65] = (bit8)((pSimpleData->word[14]) & 0x00ff); 2655 pInquiry[66] = (bit8)((pSimpleData->word[15]) >> 8); 2656 pInquiry[67] = (bit8)((pSimpleData->word[15]) & 0x00ff); 2657 pInquiry[68] = (bit8)((pSimpleData->word[16]) >> 8); 2658 pInquiry[69] = (bit8)((pSimpleData->word[16]) & 0x00ff); 2659 pInquiry[70] = (bit8)((pSimpleData->word[17]) >> 8); 2660 pInquiry[71] = (bit8)((pSimpleData->word[17]) & 0x00ff); 2661 pInquiry[72] = (bit8)((pSimpleData->word[18]) >> 8); 2662 pInquiry[73] = (bit8)((pSimpleData->word[18]) & 0x00ff); 2663 pInquiry[74] = (bit8)((pSimpleData->word[19]) >> 8); 2664 pInquiry[75] = (bit8)((pSimpleData->word[19]) & 0x00ff); 2665 #else 2666 /* for the FreeBSD */ 2667 /* Fill in SAT Rev8 Table86 */ 2668 /* 2669 * Logical unit name derived from the model number and serial number. 2670 */ 2671 pInquiry[3] = 84; /* 87 - 3; page length */ 2672 2673 /* 2674 * Identifier descriptor 2675 */ 2676 pInquiry[4] = 0x02; /* Code set: ASCII codes */ 2677 pInquiry[5] = 0x01; /* Identifier type : T10 vendor ID based */ 2678 pInquiry[6] = 0x00; /* Reserved */ 2679 pInquiry[7] = 0x44; /* 0x44, 68 Identifier length */ 2680 2681 /* Byte 8 to 15 is the vendor id string 'ATA '. */ 2682 sm_strncpy((char *)&pInquiry[8], AG_SAT_VENDOR_ID_STRING, 8); 2683 2684 2685 /* 2686 * Byte 16 to 75 is vendor specific id 2687 */ 2688 pInquiry[16] = (bit8)((pSimpleData->word[27]) >> 8); 2689 pInquiry[17] = (bit8)((pSimpleData->word[27]) & 0x00ff); 2690 pInquiry[18] = (bit8)((pSimpleData->word[28]) >> 8); 2691 pInquiry[19] = (bit8)((pSimpleData->word[28]) & 0x00ff); 2692 pInquiry[20] = (bit8)((pSimpleData->word[29]) >> 8); 2693 pInquiry[21] = (bit8)((pSimpleData->word[29]) & 0x00ff); 2694 pInquiry[22] = (bit8)((pSimpleData->word[30]) >> 8); 2695 pInquiry[23] = (bit8)((pSimpleData->word[30]) & 0x00ff); 2696 pInquiry[24] = (bit8)((pSimpleData->word[31]) >> 8); 2697 pInquiry[25] = (bit8)((pSimpleData->word[31]) & 0x00ff); 2698 pInquiry[26] = (bit8)((pSimpleData->word[32]) >> 8); 2699 pInquiry[27] = (bit8)((pSimpleData->word[32]) & 0x00ff); 2700 pInquiry[28] = (bit8)((pSimpleData->word[33]) >> 8); 2701 pInquiry[29] = (bit8)((pSimpleData->word[33]) & 0x00ff); 2702 pInquiry[30] = (bit8)((pSimpleData->word[34]) >> 8); 2703 pInquiry[31] = (bit8)((pSimpleData->word[34]) & 0x00ff); 2704 pInquiry[32] = (bit8)((pSimpleData->word[35]) >> 8); 2705 pInquiry[33] = (bit8)((pSimpleData->word[35]) & 0x00ff); 2706 pInquiry[34] = (bit8)((pSimpleData->word[36]) >> 8); 2707 pInquiry[35] = (bit8)((pSimpleData->word[36]) & 0x00ff); 2708 pInquiry[36] = (bit8)((pSimpleData->word[37]) >> 8); 2709 pInquiry[37] = (bit8)((pSimpleData->word[37]) & 0x00ff); 2710 pInquiry[38] = (bit8)((pSimpleData->word[38]) >> 8); 2711 pInquiry[39] = (bit8)((pSimpleData->word[38]) & 0x00ff); 2712 pInquiry[40] = (bit8)((pSimpleData->word[39]) >> 8); 2713 pInquiry[41] = (bit8)((pSimpleData->word[39]) & 0x00ff); 2714 pInquiry[42] = (bit8)((pSimpleData->word[40]) >> 8); 2715 pInquiry[43] = (bit8)((pSimpleData->word[40]) & 0x00ff); 2716 pInquiry[44] = (bit8)((pSimpleData->word[41]) >> 8); 2717 pInquiry[45] = (bit8)((pSimpleData->word[41]) & 0x00ff); 2718 pInquiry[46] = (bit8)((pSimpleData->word[42]) >> 8); 2719 pInquiry[47] = (bit8)((pSimpleData->word[42]) & 0x00ff); 2720 pInquiry[48] = (bit8)((pSimpleData->word[43]) >> 8); 2721 pInquiry[49] = (bit8)((pSimpleData->word[43]) & 0x00ff); 2722 pInquiry[50] = (bit8)((pSimpleData->word[44]) >> 8); 2723 pInquiry[51] = (bit8)((pSimpleData->word[44]) & 0x00ff); 2724 pInquiry[52] = (bit8)((pSimpleData->word[45]) >> 8); 2725 pInquiry[53] = (bit8)((pSimpleData->word[45]) & 0x00ff); 2726 pInquiry[54] = (bit8)((pSimpleData->word[46]) >> 8); 2727 pInquiry[55] = (bit8)((pSimpleData->word[46]) & 0x00ff); 2728 2729 pInquiry[56] = (bit8)((pSimpleData->word[10]) >> 8); 2730 pInquiry[57] = (bit8)((pSimpleData->word[10]) & 0x00ff); 2731 pInquiry[58] = (bit8)((pSimpleData->word[11]) >> 8); 2732 pInquiry[59] = (bit8)((pSimpleData->word[11]) & 0x00ff); 2733 pInquiry[60] = (bit8)((pSimpleData->word[12]) >> 8); 2734 pInquiry[61] = (bit8)((pSimpleData->word[12]) & 0x00ff); 2735 pInquiry[62] = (bit8)((pSimpleData->word[13]) >> 8); 2736 pInquiry[63] = (bit8)((pSimpleData->word[13]) & 0x00ff); 2737 pInquiry[64] = (bit8)((pSimpleData->word[14]) >> 8); 2738 pInquiry[65] = (bit8)((pSimpleData->word[14]) & 0x00ff); 2739 pInquiry[66] = (bit8)((pSimpleData->word[15]) >> 8); 2740 pInquiry[67] = (bit8)((pSimpleData->word[15]) & 0x00ff); 2741 pInquiry[68] = (bit8)((pSimpleData->word[16]) >> 8); 2742 pInquiry[69] = (bit8)((pSimpleData->word[16]) & 0x00ff); 2743 pInquiry[70] = (bit8)((pSimpleData->word[17]) >> 8); 2744 pInquiry[71] = (bit8)((pSimpleData->word[17]) & 0x00ff); 2745 pInquiry[72] = (bit8)((pSimpleData->word[18]) >> 8); 2746 pInquiry[73] = (bit8)((pSimpleData->word[18]) & 0x00ff); 2747 pInquiry[74] = (bit8)((pSimpleData->word[19]) >> 8); 2748 pInquiry[75] = (bit8)((pSimpleData->word[19]) & 0x00ff); 2749 2750 pInquiry[76] = 0x61; /* Code set: binary codes; this is proto_codeset in FreeBSD; SCSI_PROTO_SAS and SVPD_ID_CODESET_BINARY */ 2751 pInquiry[77] = 0x93; /* Identifier type : NAA ; this is id_type in FreeBSD; PIV set, ASSOCIATION is 01b and NAA (3h) */ 2752 pInquiry[78] = 0x00; /* Reserved */ 2753 pInquiry[79] = 0x08; /* Identifier length */ 2754 2755 SM_DBG5(("smsatInquiryPage83: NO WWN sasAddressHi 0x%08x\n", oneDeviceData->sasAddressHi)); 2756 SM_DBG5(("smsatInquiryPage83: No WWN sasAddressLo 0x%08x\n", oneDeviceData->sasAddressLo)); 2757 2758 /* SAS address of SATA */ 2759 pInquiry[80] = ((oneDeviceData->sasAddressHi) & 0xFF000000 ) >> 24; 2760 pInquiry[81] = ((oneDeviceData->sasAddressHi) & 0xFF0000 ) >> 16; 2761 pInquiry[82] = ((oneDeviceData->sasAddressHi) & 0xFF00 ) >> 8; 2762 pInquiry[83] = (oneDeviceData->sasAddressHi) & 0xFF; 2763 pInquiry[84] = ((oneDeviceData->sasAddressLo) & 0xFF000000 ) >> 24; 2764 pInquiry[85] = ((oneDeviceData->sasAddressLo) & 0xFF0000 ) >> 16; 2765 pInquiry[86] = ((oneDeviceData->sasAddressLo) & 0xFF00 ) >> 8; 2766 pInquiry[87] = (oneDeviceData->sasAddressLo) & 0xFF; 2767 2768 #endif 2769 } 2770 2771 return; 2772 } 2773 2774 osGLOBAL void 2775 smsatInquiryPage89( 2776 bit8 *pInquiry, 2777 agsaSATAIdentifyData_t *pSATAIdData, 2778 smDeviceData_t *oneDeviceData, 2779 bit32 len 2780 ) 2781 { 2782 /* 2783 SAT revision 8, 10.3.5, p 83 2784 */ 2785 satSimpleSATAIdentifyData_t *pSimpleData; 2786 2787 /* 2788 * When translating the fields, in some cases using the simple form of SATA 2789 * Identify Device Data is easier. So we define it here. 2790 * Both pSimpleData and pSATAIdData points to the same data. 2791 */ 2792 pSimpleData = ( satSimpleSATAIdentifyData_t *)pSATAIdData; 2793 2794 SM_DBG5(("smsatInquiryPage89: start\n")); 2795 2796 pInquiry[0] = 0x00; /* Peripheral Qualifier and Peripheral Device Type */ 2797 pInquiry[1] = 0x89; /* page code */ 2798 2799 /* Page length 0x238 */ 2800 pInquiry[2] = 0x02; 2801 pInquiry[3] = 0x38; 2802 2803 pInquiry[4] = 0x0; /* reserved */ 2804 pInquiry[5] = 0x0; /* reserved */ 2805 pInquiry[6] = 0x0; /* reserved */ 2806 pInquiry[7] = 0x0; /* reserved */ 2807 2808 /* SAT Vendor Identification */ 2809 sm_strncpy((char*)&pInquiry[8], "PMC-SIERRA", 8); /* 8 bytes */ 2810 2811 /* SAT Product Idetification */ 2812 sm_strncpy((char*)&pInquiry[16], "Tachyon-SPC ", 16); /* 16 bytes */ 2813 2814 /* SAT Product Revision Level */ 2815 sm_strncpy((char*)&pInquiry[32], "01", 4); /* 4 bytes */ 2816 2817 /* Signature, SAT revision8, Table88, p85 */ 2818 2819 2820 pInquiry[36] = 0x34; /* FIS type */ 2821 if (oneDeviceData->satDeviceType == SATA_ATA_DEVICE) 2822 { 2823 /* interrupt assume to be 0 */ 2824 pInquiry[37] = (bit8)((oneDeviceData->satPMField) >> (4 * 7)); /* first four bits of PM field */ 2825 } 2826 else 2827 { 2828 /* interrupt assume to be 1 */ 2829 pInquiry[37] = (bit8)(0x40 + (bit8)(((oneDeviceData->satPMField) >> (4 * 7)))); /* first four bits of PM field */ 2830 } 2831 pInquiry[38] = 0; 2832 pInquiry[39] = 0; 2833 2834 if (oneDeviceData->satDeviceType == SATA_ATA_DEVICE) 2835 { 2836 pInquiry[40] = 0x01; /* LBA Low */ 2837 pInquiry[41] = 0x00; /* LBA Mid */ 2838 pInquiry[42] = 0x00; /* LBA High */ 2839 pInquiry[43] = 0x00; /* Device */ 2840 pInquiry[44] = 0x00; /* LBA Low Exp */ 2841 pInquiry[45] = 0x00; /* LBA Mid Exp */ 2842 pInquiry[46] = 0x00; /* LBA High Exp */ 2843 pInquiry[47] = 0x00; /* Reserved */ 2844 pInquiry[48] = 0x01; /* Sector Count */ 2845 pInquiry[49] = 0x00; /* Sector Count Exp */ 2846 } 2847 else 2848 { 2849 pInquiry[40] = 0x01; /* LBA Low */ 2850 pInquiry[41] = 0x00; /* LBA Mid */ 2851 pInquiry[42] = 0x00; /* LBA High */ 2852 pInquiry[43] = 0x00; /* Device */ 2853 pInquiry[44] = 0x00; /* LBA Low Exp */ 2854 pInquiry[45] = 0x00; /* LBA Mid Exp */ 2855 pInquiry[46] = 0x00; /* LBA High Exp */ 2856 pInquiry[47] = 0x00; /* Reserved */ 2857 pInquiry[48] = 0x01; /* Sector Count */ 2858 pInquiry[49] = 0x00; /* Sector Count Exp */ 2859 } 2860 2861 /* Reserved */ 2862 pInquiry[50] = 0x00; 2863 pInquiry[51] = 0x00; 2864 pInquiry[52] = 0x00; 2865 pInquiry[53] = 0x00; 2866 pInquiry[54] = 0x00; 2867 pInquiry[55] = 0x00; 2868 2869 /* Command Code */ 2870 if (oneDeviceData->satDeviceType == SATA_ATA_DEVICE) 2871 { 2872 pInquiry[56] = 0xEC; /* IDENTIFY DEVICE */ 2873 } 2874 else 2875 { 2876 pInquiry[56] = 0xA1; /* IDENTIFY PACKET DEVICE */ 2877 } 2878 /* Reserved */ 2879 pInquiry[57] = 0x0; 2880 pInquiry[58] = 0x0; 2881 pInquiry[59] = 0x0; 2882 2883 /* check the length; len is assumed to be at least 60 */ 2884 if (len < SATA_PAGE89_INQUIRY_SIZE) 2885 { 2886 /* Identify Device */ 2887 sm_memcpy(&pInquiry[60], pSimpleData, MIN((len - 60), sizeof(satSimpleSATAIdentifyData_t))); 2888 } 2889 else 2890 { 2891 /* Identify Device */ 2892 sm_memcpy(&pInquiry[60], pSimpleData, sizeof(satSimpleSATAIdentifyData_t)); 2893 } 2894 2895 return; 2896 } 2897 2898 osGLOBAL void 2899 smsatInquiryPage80( 2900 bit8 *pInquiry, 2901 agsaSATAIdentifyData_t *pSATAIdData 2902 ) 2903 { 2904 SM_DBG5(("smsatInquiryPage89: start\n")); 2905 /* 2906 See SPC-4, 7.6.9, p 345 2907 and SAT revision 8, 10.3.3, p 77 2908 */ 2909 pInquiry[0] = 0x00; 2910 pInquiry[1] = 0x80; /* page code */ 2911 pInquiry[2] = 0x00; /* reserved */ 2912 pInquiry[3] = 0x14; /* page length */ 2913 2914 /* product serial number */ 2915 pInquiry[4] = pSATAIdData->serialNumber[1]; 2916 pInquiry[5] = pSATAIdData->serialNumber[0]; 2917 pInquiry[6] = pSATAIdData->serialNumber[3]; 2918 pInquiry[7] = pSATAIdData->serialNumber[2]; 2919 pInquiry[8] = pSATAIdData->serialNumber[5]; 2920 pInquiry[9] = pSATAIdData->serialNumber[4]; 2921 pInquiry[10] = pSATAIdData->serialNumber[7]; 2922 pInquiry[11] = pSATAIdData->serialNumber[6]; 2923 pInquiry[12] = pSATAIdData->serialNumber[9]; 2924 pInquiry[13] = pSATAIdData->serialNumber[8]; 2925 pInquiry[14] = pSATAIdData->serialNumber[11]; 2926 pInquiry[15] = pSATAIdData->serialNumber[10]; 2927 pInquiry[16] = pSATAIdData->serialNumber[13]; 2928 pInquiry[17] = pSATAIdData->serialNumber[12]; 2929 pInquiry[18] = pSATAIdData->serialNumber[15]; 2930 pInquiry[19] = pSATAIdData->serialNumber[14]; 2931 pInquiry[20] = pSATAIdData->serialNumber[17]; 2932 pInquiry[21] = pSATAIdData->serialNumber[16]; 2933 pInquiry[22] = pSATAIdData->serialNumber[19]; 2934 pInquiry[23] = pSATAIdData->serialNumber[18]; 2935 2936 return; 2937 } 2938 2939 osGLOBAL void 2940 smsatInquiryPageB1( 2941 bit8 *pInquiry, 2942 agsaSATAIdentifyData_t *pSATAIdData 2943 ) 2944 { 2945 bit32 i; 2946 satSimpleSATAIdentifyData_t *pSimpleData; 2947 2948 SM_DBG5(("smsatInquiryPageB1: start\n")); 2949 2950 pSimpleData = ( satSimpleSATAIdentifyData_t *)pSATAIdData; 2951 /* 2952 See SBC-3, revision31, Table193, p273 2953 and SAT-3 revision 3, 10.3.6, p141 2954 */ 2955 pInquiry[0] = 0x00; /* Peripheral Qualifier and Peripheral Device Type */ 2956 pInquiry[1] = 0xB1; /* page code */ 2957 2958 /* page length */ 2959 pInquiry[2] = 0x0; 2960 pInquiry[3] = 0x3C; 2961 2962 /* medium rotation rate */ 2963 pInquiry[4] = (bit8) ((pSimpleData->word[217]) >> 8); 2964 pInquiry[5] = (bit8) ((pSimpleData->word[217]) & 0xFF); 2965 2966 /* reserved */ 2967 pInquiry[6] = 0x0; 2968 2969 /* nominal form factor bits 3:0 */ 2970 pInquiry[7] = (bit8) ((pSimpleData->word[168]) & 0xF); 2971 2972 2973 /* reserved */ 2974 for (i=8;i<64;i++) 2975 { 2976 pInquiry[i] = 0x0; 2977 } 2978 return; 2979 } 2980 2981 osGLOBAL void 2982 smsatDefaultTranslation( 2983 smRoot_t *smRoot, 2984 smIORequest_t *smIORequest, 2985 smSatIOContext_t *satIOContext, 2986 smScsiRspSense_t *pSense, 2987 bit8 ataStatus, 2988 bit8 ataError, 2989 bit32 interruptContext 2990 ) 2991 { 2992 SM_DBG5(("smsatDefaultTranslation: start\n")); 2993 /* 2994 * Check for device fault case 2995 */ 2996 if ( ataStatus & DF_ATA_STATUS_MASK ) 2997 { 2998 smsatSetSensePayload( pSense, 2999 SCSI_SNSKEY_HARDWARE_ERROR, 3000 0, 3001 SCSI_SNSCODE_INTERNAL_TARGET_FAILURE, 3002 satIOContext); 3003 3004 tdsmIOCompletedCB( smRoot, 3005 smIORequest, 3006 smIOSuccess, 3007 SCSI_STAT_CHECK_CONDITION, 3008 satIOContext->pSmSenseData, 3009 interruptContext ); 3010 return; 3011 } 3012 3013 /* 3014 * If status error bit it set, need to check the error register 3015 */ 3016 if ( ataStatus & ERR_ATA_STATUS_MASK ) 3017 { 3018 if ( ataError & NM_ATA_ERROR_MASK ) 3019 { 3020 SM_DBG1(("smsatDefaultTranslation: NM_ATA_ERROR ataError= 0x%x, smIORequest=%p!!!\n", 3021 ataError, smIORequest)); 3022 smsatSetSensePayload( pSense, 3023 SCSI_SNSKEY_NOT_READY, 3024 0, 3025 SCSI_SNSCODE_MEDIUM_NOT_PRESENT, 3026 satIOContext); 3027 } 3028 3029 else if (ataError & UNC_ATA_ERROR_MASK) 3030 { 3031 SM_DBG1(("smsatDefaultTranslation: UNC_ATA_ERROR ataError= 0x%x, smIORequest=%p!!!\n", 3032 ataError, smIORequest)); 3033 smsatSetSensePayload( pSense, 3034 SCSI_SNSKEY_MEDIUM_ERROR, 3035 0, 3036 SCSI_SNSCODE_UNRECOVERED_READ_ERROR, 3037 satIOContext); 3038 } 3039 3040 else if (ataError & IDNF_ATA_ERROR_MASK) 3041 { 3042 SM_DBG1(("smsatDefaultTranslation: IDNF_ATA_ERROR ataError= 0x%x, smIORequest=%p!!!\n", 3043 ataError, smIORequest)); 3044 smsatSetSensePayload( pSense, 3045 SCSI_SNSKEY_MEDIUM_ERROR, 3046 0, 3047 SCSI_SNSCODE_RECORD_NOT_FOUND, 3048 satIOContext); 3049 } 3050 3051 else if (ataError & MC_ATA_ERROR_MASK) 3052 { 3053 SM_DBG1(("smsatDefaultTranslation: MC_ATA_ERROR ataError= 0x%x, smIORequest=%p!!!\n", 3054 ataError, smIORequest)); 3055 smsatSetSensePayload( pSense, 3056 SCSI_SNSKEY_UNIT_ATTENTION, 3057 0, 3058 SCSI_SNSCODE_NOT_READY_TO_READY_CHANGE, 3059 satIOContext); 3060 } 3061 3062 else if (ataError & MCR_ATA_ERROR_MASK) 3063 { 3064 SM_DBG1(("smsatDefaultTranslation: MCR_ATA_ERROR ataError= 0x%x, smIORequest=%p!!!\n", 3065 ataError, smIORequest)); 3066 smsatSetSensePayload( pSense, 3067 SCSI_SNSKEY_UNIT_ATTENTION, 3068 0, 3069 SCSI_SNSCODE_OPERATOR_MEDIUM_REMOVAL_REQUEST, 3070 satIOContext); 3071 } 3072 3073 else if (ataError & ICRC_ATA_ERROR_MASK) 3074 { 3075 SM_DBG1(("smsatDefaultTranslation: ICRC_ATA_ERROR ataError= 0x%x, smIORequest=%p!!!\n", 3076 ataError, smIORequest)); 3077 smsatSetSensePayload( pSense, 3078 SCSI_SNSKEY_ABORTED_COMMAND, 3079 0, 3080 SCSI_SNSCODE_INFORMATION_UNIT_CRC_ERROR, 3081 satIOContext); 3082 } 3083 3084 else if (ataError & ABRT_ATA_ERROR_MASK) 3085 { 3086 SM_DBG1(("smsatDefaultTranslation: ABRT_ATA_ERROR ataError= 0x%x, smIORequest=%p!!!\n", 3087 ataError, smIORequest)); 3088 smsatSetSensePayload( pSense, 3089 SCSI_SNSKEY_ABORTED_COMMAND, 3090 0, 3091 SCSI_SNSCODE_NO_ADDITIONAL_INFO, 3092 satIOContext); 3093 } 3094 3095 else 3096 { 3097 SM_DBG1(("smsatDefaultTranslation: **** UNEXPECTED ATA_ERROR **** ataError= 0x%x, smIORequest=%p!!!\n", 3098 ataError, smIORequest)); 3099 smsatSetSensePayload( pSense, 3100 SCSI_SNSKEY_HARDWARE_ERROR, 3101 0, 3102 SCSI_SNSCODE_INTERNAL_TARGET_FAILURE, 3103 satIOContext); 3104 } 3105 3106 /* Send the completion response now */ 3107 tdsmIOCompletedCB( smRoot, 3108 smIORequest, 3109 smIOSuccess, 3110 SCSI_STAT_CHECK_CONDITION, 3111 satIOContext->pSmSenseData, 3112 interruptContext ); 3113 return; 3114 3115 3116 } 3117 3118 else /* (ataStatus & ERR_ATA_STATUS_MASK ) is false */ 3119 { 3120 /* This case should never happen */ 3121 SM_DBG1(("smsatDefaultTranslation: *** UNEXPECTED ATA status 0x%x *** smIORequest=%p!!!\n", 3122 ataStatus, smIORequest)); 3123 smsatSetSensePayload( pSense, 3124 SCSI_SNSKEY_HARDWARE_ERROR, 3125 0, 3126 SCSI_SNSCODE_INTERNAL_TARGET_FAILURE, 3127 satIOContext); 3128 3129 tdsmIOCompletedCB( smRoot, 3130 smIORequest, 3131 smIOSuccess, 3132 SCSI_STAT_CHECK_CONDITION, 3133 satIOContext->pSmSenseData, 3134 interruptContext ); 3135 return; 3136 3137 } 3138 3139 return; 3140 } 3141 3142 osGLOBAL bit32 3143 smIDStart( 3144 smRoot_t *smRoot, 3145 smIORequest_t *smIORequest, 3146 smDeviceHandle_t *smDeviceHandle 3147 ) 3148 { 3149 smDeviceData_t *oneDeviceData = agNULL; 3150 smIORequestBody_t *smIORequestBody = agNULL; 3151 smSatIOContext_t *satIOContext = agNULL; 3152 bit32 status = SM_RC_FAILURE; 3153 3154 SM_DBG2(("smIDStart: start, smIORequest %p\n", smIORequest)); 3155 3156 oneDeviceData = (smDeviceData_t *)smDeviceHandle->smData; 3157 if (oneDeviceData == agNULL) 3158 { 3159 SM_DBG1(("smIDStart: oneDeviceData is NULL!!!\n")); 3160 return SM_RC_FAILURE; 3161 } 3162 if (oneDeviceData->valid == agFALSE) 3163 { 3164 SM_DBG1(("smIDStart: oneDeviceData is not valid, did %d !!!\n", oneDeviceData->id)); 3165 return SM_RC_FAILURE; 3166 } 3167 3168 smIORequestBody = (smIORequestBody_t*)smIORequest->smData;//smDequeueIO(smRoot); 3169 3170 if (smIORequestBody == agNULL) 3171 { 3172 SM_DBG1(("smIDStart: smIORequestBody is NULL!!!\n")); 3173 return SM_RC_FAILURE; 3174 } 3175 3176 smIOReInit(smRoot, smIORequestBody); 3177 3178 SM_DBG3(("smIDStart: io ID %d!!!\n", smIORequestBody->id )); 3179 3180 smIORequestBody->smIORequest = smIORequest; 3181 smIORequestBody->smDevHandle = smDeviceHandle; 3182 satIOContext = &(smIORequestBody->transport.SATA.satIOContext); 3183 3184 /* setting up satIOContext */ 3185 satIOContext->pSatDevData = oneDeviceData; 3186 satIOContext->pFis = &(smIORequestBody->transport.SATA.agSATARequestBody.fis.fisRegHostToDev); 3187 satIOContext->smRequestBody = smIORequestBody; 3188 satIOContext->psmDeviceHandle = smDeviceHandle; 3189 satIOContext->smScsiXchg = agNULL; 3190 3191 /*smIORequest->smData = smIORequestBody;*/ 3192 SM_DBG3(("smIDStart: smIORequestBody %p smIORequestBody->smIORequest %p!!!\n", smIORequestBody, smIORequestBody->smIORequest)); 3193 SM_DBG1(("smIDStart: did %d\n", oneDeviceData->id)); 3194 3195 status = smsatIDSubStart( smRoot, 3196 smIORequest, 3197 smDeviceHandle, 3198 agNULL, 3199 satIOContext); 3200 3201 if (status != SM_RC_SUCCESS) 3202 { 3203 SM_DBG1(("smIDStart: smsatIDSubStart failure %d!!!\n", status)); 3204 /*smEnqueueIO(smRoot, satIOContext);*/ 3205 } 3206 SM_DBG2(("smIDStart: exit\n")); 3207 3208 return status; 3209 } 3210 3211 /* 3212 SM generated IO, needs to call smsatAllocIntIoResource() 3213 allocating using smsatAllocIntIoResource 3214 */ 3215 osGLOBAL bit32 3216 smsatIDSubStart( 3217 smRoot_t *smRoot, 3218 smIORequest_t *smIORequest, 3219 smDeviceHandle_t *smDeviceHandle, 3220 smScsiInitiatorRequest_t *smSCSIRequest, /* agNULL */ 3221 smSatIOContext_t *satIOContext 3222 ) 3223 { 3224 smSatInternalIo_t *satIntIo = agNULL; 3225 smDeviceData_t *satDevData = agNULL; 3226 smIORequestBody_t *smIORequestBody; 3227 smSatIOContext_t *satNewIOContext; 3228 bit32 status; 3229 SM_DBG2(("smsatIDSubStart: start\n")); 3230 3231 satDevData = satIOContext->pSatDevData; 3232 3233 /* allocate identify device command */ 3234 satIntIo = smsatAllocIntIoResource( smRoot, 3235 smIORequest, 3236 satDevData, 3237 sizeof(agsaSATAIdentifyData_t), /* 512; size of identify device data */ 3238 satIntIo); 3239 3240 if (satIntIo == agNULL) 3241 { 3242 SM_DBG1(("smsatIDSubStart: can't alloacate!!!\n")); 3243 return SM_RC_FAILURE; 3244 } 3245 3246 satIOContext->satIntIoContext = satIntIo; 3247 3248 /* fill in fields */ 3249 /* real ttttttthe one worked and the same; 5/21/07/ */ 3250 satIntIo->satOrgSmIORequest = smIORequest; /* changed */ 3251 smIORequestBody = satIntIo->satIntRequestBody; 3252 satNewIOContext = &(smIORequestBody->transport.SATA.satIOContext); 3253 3254 satNewIOContext->pSatDevData = satDevData; 3255 satNewIOContext->pFis = &(smIORequestBody->transport.SATA.agSATARequestBody.fis.fisRegHostToDev); 3256 satNewIOContext->pScsiCmnd = &(satIntIo->satIntSmScsiXchg.scsiCmnd); 3257 satNewIOContext->pSense = &(smIORequestBody->transport.SATA.sensePayload); 3258 satNewIOContext->pSmSenseData = &(smIORequestBody->transport.SATA.smSenseData); 3259 satNewIOContext->smRequestBody = satIntIo->satIntRequestBody; /* key fix */ 3260 // satNewIOContext->interruptContext = tiInterruptContext; 3261 satNewIOContext->satIntIoContext = satIntIo; 3262 3263 satNewIOContext->psmDeviceHandle = smDeviceHandle; 3264 satNewIOContext->satOrgIOContext = satIOContext; /* changed */ 3265 3266 /* this is valid only for TD layer generated (not triggered by OS at all) IO */ 3267 satNewIOContext->smScsiXchg = &(satIntIo->satIntSmScsiXchg); 3268 3269 3270 SM_DBG6(("smsatIDSubStart: SM satIOContext %p \n", satIOContext)); 3271 SM_DBG6(("smsatIDSubStart: SM satNewIOContext %p \n", satNewIOContext)); 3272 SM_DBG6(("smsatIDSubStart: SM tiScsiXchg %p \n", satIOContext->smScsiXchg)); 3273 SM_DBG6(("smsatIDSubStart: SM tiScsiXchg %p \n", satNewIOContext->smScsiXchg)); 3274 3275 3276 3277 SM_DBG3(("smsatIDSubStart: satNewIOContext %p smIORequestBody %p\n", satNewIOContext, smIORequestBody)); 3278 3279 status = smsatIDStart(smRoot, 3280 &satIntIo->satIntSmIORequest, /* New smIORequest */ 3281 smDeviceHandle, 3282 satNewIOContext->smScsiXchg, /* New smScsiInitiatorRequest_t *smScsiRequest, */ 3283 satNewIOContext); 3284 3285 if (status != SM_RC_SUCCESS) 3286 { 3287 SM_DBG1(("smsatIDSubStart: failed in sending %d!!!\n", status)); 3288 3289 smsatFreeIntIoResource( smRoot, 3290 satDevData, 3291 satIntIo); 3292 3293 return SM_RC_FAILURE; 3294 } 3295 3296 3297 SM_DBG2(("smsatIDSubStart: end\n")); 3298 3299 return status; 3300 3301 } 3302 3303 3304 osGLOBAL bit32 3305 smsatIDStart( 3306 smRoot_t *smRoot, 3307 smIORequest_t *smIORequest, 3308 smDeviceHandle_t *smDeviceHandle, 3309 smScsiInitiatorRequest_t *smSCSIRequest, 3310 smSatIOContext_t *satIOContext 3311 ) 3312 { 3313 bit32 status; 3314 bit32 agRequestType; 3315 smDeviceData_t *pSatDevData; 3316 agsaFisRegHostToDevice_t *fis; 3317 #ifdef SM_INTERNAL_DEBUG 3318 smIORequestBody_t *smIORequestBody; 3319 smSatInternalIo_t *satIntIoContext; 3320 #endif 3321 3322 pSatDevData = satIOContext->pSatDevData; 3323 fis = satIOContext->pFis; 3324 SM_DBG2(("smsatIDStart: start\n")); 3325 #ifdef SM_INTERNAL_DEBUG 3326 satIntIoContext = satIOContext->satIntIoContext; 3327 smIORequestBody = satIntIoContext->satIntRequestBody; 3328 #endif 3329 fis->h.fisType = 0x27; /* Reg host to device */ 3330 fis->h.c_pmPort = 0x80; /* C Bit is set */ 3331 if (pSatDevData->satDeviceType == SATA_ATAPI_DEVICE) 3332 { 3333 SM_DBG2(("smsatIDStart: IDENTIFY_PACKET_DEVICE\n")); 3334 fis->h.command = SAT_IDENTIFY_PACKET_DEVICE; /* 0x40 */ 3335 } 3336 else 3337 { 3338 SM_DBG2(("smsatIDStart: IDENTIFY_DEVICE\n")); 3339 fis->h.command = SAT_IDENTIFY_DEVICE; /* 0xEC */ 3340 } 3341 fis->h.features = 0; /* FIS reserve */ 3342 fis->d.lbaLow = 0; /* FIS LBA (7 :0 ) */ 3343 fis->d.lbaMid = 0; /* FIS LBA (15:8 ) */ 3344 fis->d.lbaHigh = 0; /* FIS LBA (23:16) */ 3345 fis->d.device = 0; /* FIS LBA mode */ 3346 fis->d.lbaLowExp = 0; 3347 fis->d.lbaMidExp = 0; 3348 fis->d.lbaHighExp = 0; 3349 fis->d.featuresExp = 0; 3350 fis->d.sectorCount = 0; /* FIS sector count (7:0) */ 3351 fis->d.sectorCountExp = 0; 3352 fis->d.reserved4 = 0; 3353 fis->d.control = 0; /* FIS HOB bit clear */ 3354 fis->d.reserved5 = 0; 3355 3356 agRequestType = AGSA_SATA_PROTOCOL_PIO_READ; 3357 3358 /* Initialize CB for SATA completion. 3359 */ 3360 satIOContext->satCompleteCB = &smsatIDStartCB; 3361 3362 /* 3363 * Prepare SGL and send FIS to LL layer. 3364 */ 3365 satIOContext->reqType = agRequestType; /* Save it */ 3366 3367 #ifdef SM_INTERNAL_DEBUG 3368 smhexdump("smsatIDStart", (bit8 *)satIOContext->pFis, sizeof(agsaFisRegHostToDevice_t)); 3369 smhexdump("smsatIDStart LL", (bit8 *)&(smIORequestBody->transport.SATA.agSATARequestBody.fis.fisRegHostToDev), sizeof(agsaFisRegHostToDevice_t)); 3370 #endif 3371 status = smsataLLIOStart( smRoot, 3372 smIORequest, 3373 smDeviceHandle, 3374 smSCSIRequest, 3375 satIOContext); 3376 3377 SM_DBG2(("smsatIDStart: end status %d\n", status)); 3378 3379 return status; 3380 } 3381 3382 3383 osGLOBAL FORCEINLINE bit32 3384 smsatIOStart( 3385 smRoot_t *smRoot, 3386 smIORequest_t *smIORequest, 3387 smDeviceHandle_t *smDeviceHandle, 3388 smScsiInitiatorRequest_t *smSCSIRequest, 3389 smSatIOContext_t *satIOContext 3390 ) 3391 { 3392 smDeviceData_t *pSatDevData = satIOContext->pSatDevData; 3393 smScsiRspSense_t *pSense = satIOContext->pSense; 3394 smIniScsiCmnd_t *scsiCmnd = &smSCSIRequest->scsiCmnd; 3395 smLUN_t *pLun = &scsiCmnd->lun; 3396 smSatInternalIo_t *pSatIntIo = agNULL; 3397 bit32 status = SM_RC_FAILURE; 3398 3399 SM_DBG2(("smsatIOStart: start\n")); 3400 3401 /* 3402 * Reject all other LUN other than LUN 0. 3403 */ 3404 if ( ((pLun->lun[0] | pLun->lun[1] | pLun->lun[2] | pLun->lun[3] | 3405 pLun->lun[4] | pLun->lun[5] | pLun->lun[6] | pLun->lun[7] ) != 0) && 3406 (scsiCmnd->cdb[0] != SCSIOPC_INQUIRY) 3407 ) 3408 { 3409 SM_DBG1(("smsatIOStart: *** REJECT *** LUN not zero, cdb[0]=0x%x did %d !!!\n", 3410 scsiCmnd->cdb[0], pSatDevData->id)); 3411 smsatSetSensePayload( pSense, 3412 SCSI_SNSKEY_ILLEGAL_REQUEST, 3413 0, 3414 SCSI_SNSCODE_LOGICAL_NOT_SUPPORTED, 3415 satIOContext); 3416 3417 /*smEnqueueIO(smRoot, satIOContext);*/ 3418 3419 tdsmIOCompletedCB( smRoot, 3420 smIORequest, 3421 smIOSuccess, 3422 SCSI_STAT_CHECK_CONDITION, 3423 satIOContext->pSmSenseData, 3424 satIOContext->interruptContext ); 3425 3426 return SM_RC_SUCCESS; 3427 } 3428 3429 SM_DBG2(("smsatIOStart: satPendingIO %d satNCQMaxIO %d\n",pSatDevData->satPendingIO, pSatDevData->satNCQMaxIO )); 3430 3431 /* this may happen after tiCOMReset until OS sends inquiry */ 3432 if (pSatDevData->IDDeviceValid == agFALSE && (scsiCmnd->cdb[0] != SCSIOPC_INQUIRY)) 3433 { 3434 SM_DBG1(("smsatIOStart: invalid identify device data did %d !!!\n", pSatDevData->id)); 3435 SM_DBG1(("smsatIOStart: satPendingIO %d satNCQMaxIO %d\n", pSatDevData->satPendingIO, pSatDevData->satNCQMaxIO )); 3436 SM_DBG1(("smsatIOStart: satPendingNCQIO %d satPendingNONNCQIO %d\n", pSatDevData->satPendingNCQIO, pSatDevData->satPendingNONNCQIO)); 3437 3438 /*smEnqueueIO(smRoot, satIOContext);*/ 3439 3440 return SM_RC_NODEVICE; 3441 } 3442 3443 /* 3444 * Check if we need to return BUSY, i.e. recovery in progress 3445 */ 3446 if (pSatDevData->satDriveState == SAT_DEV_STATE_IN_RECOVERY) 3447 { 3448 SM_DBG1(("smsatIOStart: IN RECOVERY STATE cdb[0]=0x%x did=%d !!!\n", 3449 scsiCmnd->cdb[0], pSatDevData->id)); 3450 SM_DBG2(("smsatIOStart: device %p satPendingIO %d satNCQMaxIO %d\n", pSatDevData, pSatDevData->satPendingIO, pSatDevData->satNCQMaxIO )); 3451 SM_DBG2(("smsatIOStart: device %p satPendingNCQIO %d satPendingNONNCQIO %d\n",pSatDevData, pSatDevData->satPendingNCQIO, pSatDevData->satPendingNONNCQIO)); 3452 3453 /*smEnqueueIO(smRoot, satIOContext);*/ 3454 3455 // return SM_RC_FAILURE; 3456 return SM_RC_DEVICE_BUSY; 3457 } 3458 3459 if (pSatDevData->satDeviceType == SATA_ATAPI_DEVICE) 3460 { 3461 if (scsiCmnd->cdb[0] == SCSIOPC_REPORT_LUN) 3462 { 3463 return smsatReportLun(smRoot, smIORequest, smDeviceHandle, smSCSIRequest, satIOContext); 3464 } 3465 else 3466 { 3467 return smsatPacket(smRoot, smIORequest, smDeviceHandle, smSCSIRequest, satIOContext); 3468 } 3469 } 3470 else 3471 { 3472 /* Parse CDB */ 3473 switch(scsiCmnd->cdb[0]) 3474 { 3475 case SCSIOPC_READ_10: 3476 status = smsatRead10( smRoot, 3477 smIORequest, 3478 smDeviceHandle, 3479 smSCSIRequest, 3480 satIOContext); 3481 break; 3482 3483 case SCSIOPC_WRITE_10: 3484 status = smsatWrite10( smRoot, 3485 smIORequest, 3486 smDeviceHandle, 3487 smSCSIRequest, 3488 satIOContext); 3489 break; 3490 3491 case SCSIOPC_READ_6: 3492 status = smsatRead6( smRoot, 3493 smIORequest, 3494 smDeviceHandle, 3495 smSCSIRequest, 3496 satIOContext); 3497 break; 3498 3499 case SCSIOPC_READ_12: 3500 SM_DBG5(("smsatIOStart: SCSIOPC_READ_12\n")); 3501 status = smsatRead12( smRoot, 3502 smIORequest, 3503 smDeviceHandle, 3504 smSCSIRequest, 3505 satIOContext); 3506 break; 3507 3508 case SCSIOPC_READ_16: 3509 status = smsatRead16( smRoot, 3510 smIORequest, 3511 smDeviceHandle, 3512 smSCSIRequest, 3513 satIOContext); 3514 break; 3515 3516 case SCSIOPC_WRITE_6: 3517 status = smsatWrite6( smRoot, 3518 smIORequest, 3519 smDeviceHandle, 3520 smSCSIRequest, 3521 satIOContext); 3522 break; 3523 3524 case SCSIOPC_WRITE_12: 3525 SM_DBG5(("smsatIOStart: SCSIOPC_WRITE_12 \n")); 3526 status = smsatWrite12( smRoot, 3527 smIORequest, 3528 smDeviceHandle, 3529 smSCSIRequest, 3530 satIOContext); 3531 break; 3532 3533 case SCSIOPC_WRITE_16: 3534 SM_DBG5(("smsatIOStart: SCSIOPC_WRITE_16 \n")); 3535 status = smsatWrite16( smRoot, 3536 smIORequest, 3537 smDeviceHandle, 3538 smSCSIRequest, 3539 satIOContext); 3540 break; 3541 3542 case SCSIOPC_VERIFY_10: 3543 status = smsatVerify10( smRoot, 3544 smIORequest, 3545 smDeviceHandle, 3546 smSCSIRequest, 3547 satIOContext); 3548 break; 3549 3550 case SCSIOPC_VERIFY_12: 3551 SM_DBG5(("smsatIOStart: SCSIOPC_VERIFY_12\n")); 3552 status = smsatVerify12( smRoot, 3553 smIORequest, 3554 smDeviceHandle, 3555 smSCSIRequest, 3556 satIOContext); 3557 break; 3558 3559 case SCSIOPC_VERIFY_16: 3560 SM_DBG5(("smsatIOStart: SCSIOPC_VERIFY_16\n")); 3561 status = smsatVerify16( smRoot, 3562 smIORequest, 3563 smDeviceHandle, 3564 smSCSIRequest, 3565 satIOContext); 3566 break; 3567 3568 case SCSIOPC_TEST_UNIT_READY: 3569 status = smsatTestUnitReady( smRoot, 3570 smIORequest, 3571 smDeviceHandle, 3572 smSCSIRequest, 3573 satIOContext); 3574 break; 3575 3576 case SCSIOPC_INQUIRY: 3577 status = smsatInquiry( smRoot, 3578 smIORequest, 3579 smDeviceHandle, 3580 smSCSIRequest, 3581 satIOContext); 3582 break; 3583 3584 case SCSIOPC_REQUEST_SENSE: 3585 status = smsatRequestSense( smRoot, 3586 smIORequest, 3587 smDeviceHandle, 3588 smSCSIRequest, 3589 satIOContext); 3590 break; 3591 3592 case SCSIOPC_MODE_SENSE_6: 3593 status = smsatModeSense6( smRoot, 3594 smIORequest, 3595 smDeviceHandle, 3596 smSCSIRequest, 3597 satIOContext); 3598 break; 3599 3600 case SCSIOPC_MODE_SENSE_10: 3601 status = smsatModeSense10( smRoot, 3602 smIORequest, 3603 smDeviceHandle, 3604 smSCSIRequest, 3605 satIOContext); 3606 break; 3607 3608 case SCSIOPC_READ_CAPACITY_10: 3609 status = smsatReadCapacity10( smRoot, 3610 smIORequest, 3611 smDeviceHandle, 3612 smSCSIRequest, 3613 satIOContext); 3614 break; 3615 3616 case SCSIOPC_READ_CAPACITY_16: 3617 status = smsatReadCapacity16( smRoot, 3618 smIORequest, 3619 smDeviceHandle, 3620 smSCSIRequest, 3621 satIOContext); 3622 break; 3623 3624 3625 case SCSIOPC_REPORT_LUN: 3626 status = smsatReportLun( smRoot, 3627 smIORequest, 3628 smDeviceHandle, 3629 smSCSIRequest, 3630 satIOContext); 3631 break; 3632 3633 case SCSIOPC_FORMAT_UNIT: 3634 SM_DBG5(("smsatIOStart: SCSIOPC_FORMAT_UNIT\n")); 3635 status = smsatFormatUnit( smRoot, 3636 smIORequest, 3637 smDeviceHandle, 3638 smSCSIRequest, 3639 satIOContext); 3640 break; 3641 3642 case SCSIOPC_SEND_DIAGNOSTIC: 3643 SM_DBG5(("smsatIOStart: SCSIOPC_SEND_DIAGNOSTIC\n")); 3644 status = smsatSendDiagnostic( smRoot, 3645 smIORequest, 3646 smDeviceHandle, 3647 smSCSIRequest, 3648 satIOContext); 3649 break; 3650 3651 case SCSIOPC_START_STOP_UNIT: 3652 SM_DBG5(("smsatIOStart: SCSIOPC_START_STOP_UNIT\n")); 3653 status = smsatStartStopUnit( smRoot, 3654 smIORequest, 3655 smDeviceHandle, 3656 smSCSIRequest, 3657 satIOContext); 3658 break; 3659 3660 case SCSIOPC_WRITE_SAME_10: 3661 SM_DBG5(("smsatIOStart: SCSIOPC_WRITE_SAME_10\n")); 3662 status = smsatWriteSame10( smRoot, 3663 smIORequest, 3664 smDeviceHandle, 3665 smSCSIRequest, 3666 satIOContext); 3667 break; 3668 3669 case SCSIOPC_WRITE_SAME_16: /* no support due to transfer length(sector count) */ 3670 SM_DBG5(("smsatIOStart: SCSIOPC_WRITE_SAME_16\n")); 3671 status = smsatWriteSame16( smRoot, 3672 smIORequest, 3673 smDeviceHandle, 3674 smSCSIRequest, 3675 satIOContext); 3676 break; 3677 3678 case SCSIOPC_LOG_SENSE: 3679 SM_DBG5(("smsatIOStart: SCSIOPC_LOG_SENSE\n")); 3680 status = smsatLogSense( smRoot, 3681 smIORequest, 3682 smDeviceHandle, 3683 smSCSIRequest, 3684 satIOContext); 3685 break; 3686 3687 case SCSIOPC_MODE_SELECT_6: 3688 SM_DBG5(("smsatIOStart: SCSIOPC_MODE_SELECT_6\n")); 3689 status = smsatModeSelect6( smRoot, 3690 smIORequest, 3691 smDeviceHandle, 3692 smSCSIRequest, 3693 satIOContext); 3694 break; 3695 3696 case SCSIOPC_MODE_SELECT_10: 3697 SM_DBG5(("smsatIOStart: SCSIOPC_MODE_SELECT_10\n")); 3698 status = smsatModeSelect10( smRoot, 3699 smIORequest, 3700 smDeviceHandle, 3701 smSCSIRequest, 3702 satIOContext); 3703 break; 3704 3705 case SCSIOPC_SYNCHRONIZE_CACHE_10: /* on error what to return, sharing CB with 3706 satSynchronizeCache16 */ 3707 SM_DBG5(("smsatIOStart: SCSIOPC_SYNCHRONIZE_CACHE_10\n")); 3708 status = smsatSynchronizeCache10( smRoot, 3709 smIORequest, 3710 smDeviceHandle, 3711 smSCSIRequest, 3712 satIOContext); 3713 break; 3714 3715 case SCSIOPC_SYNCHRONIZE_CACHE_16:/* on error what to return, sharing CB with 3716 satSynchronizeCache16 */ 3717 3718 SM_DBG5(("smsatIOStart: SCSIOPC_SYNCHRONIZE_CACHE_16\n")); 3719 status = smsatSynchronizeCache16( smRoot, 3720 smIORequest, 3721 smDeviceHandle, 3722 smSCSIRequest, 3723 satIOContext); 3724 break; 3725 3726 case SCSIOPC_WRITE_AND_VERIFY_10: /* single write and multiple writes */ 3727 SM_DBG5(("smsatIOStart: SCSIOPC_WRITE_AND_VERIFY_10\n")); 3728 status = smsatWriteAndVerify10( smRoot, 3729 smIORequest, 3730 smDeviceHandle, 3731 smSCSIRequest, 3732 satIOContext); 3733 break; 3734 3735 case SCSIOPC_WRITE_AND_VERIFY_12: 3736 SM_DBG5(("smsatIOStart: SCSIOPC_WRITE_AND_VERIFY_12\n")); 3737 status = smsatWriteAndVerify12( smRoot, 3738 smIORequest, 3739 smDeviceHandle, 3740 smSCSIRequest, 3741 satIOContext); 3742 break; 3743 3744 case SCSIOPC_WRITE_AND_VERIFY_16: 3745 SM_DBG5(("smsatIOStart: SCSIOPC_WRITE_AND_VERIFY_16\n")); 3746 status = smsatWriteAndVerify16( smRoot, 3747 smIORequest, 3748 smDeviceHandle, 3749 smSCSIRequest, 3750 satIOContext); 3751 3752 break; 3753 3754 case SCSIOPC_READ_MEDIA_SERIAL_NUMBER: 3755 SM_DBG5(("smsatIOStart: SCSIOPC_READ_MEDIA_SERIAL_NUMBER\n")); 3756 status = smsatReadMediaSerialNumber( smRoot, 3757 smIORequest, 3758 smDeviceHandle, 3759 smSCSIRequest, 3760 satIOContext); 3761 3762 break; 3763 3764 case SCSIOPC_READ_BUFFER: 3765 SM_DBG5(("smsatIOStart: SCSIOPC_READ_BUFFER\n")); 3766 status = smsatReadBuffer( smRoot, 3767 smIORequest, 3768 smDeviceHandle, 3769 smSCSIRequest, 3770 satIOContext); 3771 3772 break; 3773 3774 case SCSIOPC_WRITE_BUFFER: 3775 SM_DBG5(("smsatIOStart: SCSIOPC_WRITE_BUFFER\n")); 3776 status = smsatWriteBuffer( smRoot, 3777 smIORequest, 3778 smDeviceHandle, 3779 smSCSIRequest, 3780 satIOContext); 3781 3782 break; 3783 3784 case SCSIOPC_REASSIGN_BLOCKS: 3785 SM_DBG5(("smsatIOStart: SCSIOPC_REASSIGN_BLOCKS\n")); 3786 status = smsatReassignBlocks( smRoot, 3787 smIORequest, 3788 smDeviceHandle, 3789 smSCSIRequest, 3790 satIOContext); 3791 3792 break; 3793 3794 case SCSIOPC_ATA_PASS_THROUGH12: /* fall through */ 3795 case SCSIOPC_ATA_PASS_THROUGH16: 3796 SM_DBG5(("smsatIOStart: SCSIOPC_ATA_PASS_THROUGH\n")); 3797 status = smsatPassthrough( smRoot, 3798 smIORequest, 3799 smDeviceHandle, 3800 smSCSIRequest, 3801 satIOContext); 3802 break; 3803 3804 default: 3805 /* Not implemented SCSI cmd, set up error response */ 3806 SM_DBG1(("smsatIOStart: unsupported SCSI cdb[0]=0x%x did=%d !!!\n", 3807 scsiCmnd->cdb[0], pSatDevData->id)); 3808 3809 smsatSetSensePayload( pSense, 3810 SCSI_SNSKEY_ILLEGAL_REQUEST, 3811 0, 3812 SCSI_SNSCODE_INVALID_COMMAND, 3813 satIOContext); 3814 3815 /*smEnqueueIO(smRoot, satIOContext);*/ 3816 3817 tdsmIOCompletedCB( smRoot, 3818 smIORequest, 3819 smIOSuccess, 3820 SCSI_STAT_CHECK_CONDITION, 3821 satIOContext->pSmSenseData, 3822 satIOContext->interruptContext ); 3823 status = SM_RC_SUCCESS; 3824 3825 break; 3826 3827 } /* end switch */ 3828 } 3829 3830 if (status == SM_RC_BUSY || status == SM_RC_DEVICE_BUSY) 3831 { 3832 SM_DBG1(("smsatIOStart: BUSY did %d!!!\n", pSatDevData->id)); 3833 SM_DBG2(("smsatIOStart: LL is busy or target queue is full\n")); 3834 SM_DBG2(("smsatIOStart: device %p satPendingIO %d satNCQMaxIO %d\n",pSatDevData, pSatDevData->satPendingIO, pSatDevData->satNCQMaxIO )); 3835 SM_DBG2(("smsatIOStart: device %p satPendingNCQIO %d satPendingNONNCQIO %d\n",pSatDevData, pSatDevData->satPendingNCQIO, pSatDevData->satPendingNONNCQIO)); 3836 pSatIntIo = satIOContext->satIntIoContext; 3837 3838 /*smEnqueueIO(smRoot, satIOContext);*/ 3839 3840 /* interal structure free */ 3841 smsatFreeIntIoResource( smRoot, 3842 pSatDevData, 3843 pSatIntIo); 3844 } 3845 3846 return status; 3847 } 3848 3849 osGLOBAL void 3850 smsatSetSensePayload( 3851 smScsiRspSense_t *pSense, 3852 bit8 SnsKey, 3853 bit32 SnsInfo, 3854 bit16 SnsCode, 3855 smSatIOContext_t *satIOContext) 3856 { 3857 /* for fixed format sense data, SPC-4, p37 */ 3858 bit32 i; 3859 bit32 senseLength; 3860 bit8 tmp = 0; 3861 3862 SM_DBG2(("smsatSetSensePayload: start\n")); 3863 3864 senseLength = sizeof(smScsiRspSense_t); 3865 3866 /* zero out the data area */ 3867 for (i=0;i< senseLength;i++) 3868 { 3869 ((bit8*)pSense)[i] = 0; 3870 } 3871 3872 /* 3873 * SCSI Sense Data part of response data 3874 */ 3875 pSense->snsRespCode = 0x70; /* 0xC0 == vendor specific */ 3876 /* 0x70 == standard current error */ 3877 pSense->senseKey = SnsKey; 3878 /* 3879 * Put sense info in scsi order format 3880 */ 3881 pSense->info[0] = (bit8)((SnsInfo >> 24) & 0xff); 3882 pSense->info[1] = (bit8)((SnsInfo >> 16) & 0xff); 3883 pSense->info[2] = (bit8)((SnsInfo >> 8) & 0xff); 3884 pSense->info[3] = (bit8)((SnsInfo) & 0xff); 3885 pSense->addSenseLen = 11; /* fixed size of sense data = 18 */ 3886 pSense->addSenseCode = (bit8)((SnsCode >> 8) & 0xFF); 3887 pSense->senseQual = (bit8)(SnsCode & 0xFF); 3888 /* 3889 * Set pointer in scsi status 3890 */ 3891 switch(SnsKey) 3892 { 3893 /* 3894 * set illegal request sense key specific error in cdb, no bit pointer 3895 */ 3896 case SCSI_SNSKEY_ILLEGAL_REQUEST: 3897 pSense->skeySpecific[0] = 0xC8; 3898 break; 3899 3900 default: 3901 break; 3902 } 3903 /* setting sense data length */ 3904 if (satIOContext != agNULL) 3905 { 3906 satIOContext->pSmSenseData->senseLen = 18; 3907 } 3908 else 3909 { 3910 SM_DBG1(("smsatSetSensePayload: satIOContext is NULL!!!\n")); 3911 } 3912 3913 /* Only for SCSI_SNSCODE_ATA_PASS_THROUGH_INFORMATION_AVAILABLE */ 3914 if (SnsCode == SCSI_SNSCODE_ATA_PASS_THROUGH_INFORMATION_AVAILABLE) 3915 { 3916 /* filling in COMMAND-SPECIFIC INFORMATION */ 3917 tmp = satIOContext->extend << 7 | satIOContext->Sector_Cnt_Upper_Nonzero << 6 | satIOContext->LBA_Upper_Nonzero << 5; 3918 SM_DBG3(("smsatSetSensePayload: extend 0x%x Sector_Cnt_Upper_Nonzero 0x%x LBA_Upper_Nonzero 0x%x\n", 3919 satIOContext->extend, satIOContext->Sector_Cnt_Upper_Nonzero, satIOContext->LBA_Upper_Nonzero)); 3920 SM_DBG3(("smsatSetSensePayload: tmp 0x%x\n", tmp)); 3921 pSense->cmdSpecific[0] = tmp; 3922 pSense->cmdSpecific[1] = satIOContext->LBAHigh07; 3923 pSense->cmdSpecific[2] = satIOContext->LBAMid07; 3924 pSense->cmdSpecific[3] = satIOContext->LBALow07; 3925 // smhexdump("smsatSetSensePayload: cmdSpecific",(bit8 *)pSense->cmdSpecific, 4); 3926 // smhexdump("smsatSetSensePayload: info",(bit8 *)pSense->info, 4); 3927 3928 } 3929 return; 3930 } 3931 3932 /***************************************************************************** 3933 *! \brief smsatDecodeSATADeviceType 3934 * 3935 * This routine decodes ATA signature 3936 * 3937 * \param pSignature: ATA signature 3938 * 3939 * 3940 * \return: 3941 * TRUE if ATA signature 3942 * FALSE otherwise 3943 * 3944 *****************************************************************************/ 3945 /* 3946 ATA p65 3947 PM p65 3948 SATAII p79, p80 3949 */ 3950 GLOBAL bit32 3951 smsatDecodeSATADeviceType( 3952 bit8 *pSignature 3953 ) 3954 { 3955 bit32 deviceType = UNKNOWN_DEVICE; 3956 3957 if ( (pSignature)[0] == 0x01 && (pSignature)[1] == 0x01 3958 && (pSignature)[2] == 0x00 && (pSignature)[3] == 0x00 3959 && (pSignature)[4] == 0xA0 ) /* this is the signature of a Hitachi SATA HDD*/ 3960 { 3961 deviceType = SATA_ATA_DEVICE; 3962 } 3963 else if ( (pSignature)[0] == 0x01 && (pSignature)[1] == 0x01 3964 && (pSignature)[2] == 0x00 && (pSignature)[3] == 0x00 3965 && (pSignature)[4] == 0x00 ) 3966 { 3967 deviceType = SATA_ATA_DEVICE; 3968 } 3969 else if ( (pSignature)[0] == 0x01 && (pSignature)[1] == 0x01 3970 && (pSignature)[2] == 0x14 && (pSignature)[3] == 0xEB 3971 && ( (pSignature)[4] == 0x00 || (pSignature)[4] == 0x10) ) 3972 { 3973 deviceType = SATA_ATAPI_DEVICE; 3974 } 3975 else if ( (pSignature)[0] == 0x01 && (pSignature)[1] == 0x01 3976 && (pSignature)[2] == 0x69 && (pSignature)[3] == 0x96 3977 && (pSignature)[4] == 0x00 ) 3978 { 3979 deviceType = SATA_PM_DEVICE; 3980 } 3981 else if ( (pSignature)[0] == 0x01 && (pSignature)[1] == 0x01 3982 && (pSignature)[2] == 0x3C && (pSignature)[3] == 0xC3 3983 && (pSignature)[4] == 0x00 ) 3984 { 3985 deviceType = SATA_SEMB_DEVICE; 3986 } 3987 else if ( (pSignature)[0] == 0xFF && (pSignature)[1] == 0xFF 3988 && (pSignature)[2] == 0xFF && (pSignature)[3] == 0xFF 3989 && (pSignature)[4] == 0xFF ) 3990 { 3991 deviceType = SATA_SEMB_WO_SEP_DEVICE; 3992 } 3993 3994 return deviceType; 3995 } 3996 3997 3998 /*****************************************************************************/ 3999 /*! \brief SAT implementation for ATAPI Packet Command. 4000 * 4001 * SAT implementation for ATAPI Packet and send FIS request to LL layer. 4002 * 4003 * \param tiRoot: Pointer to TISA initiator driver/port instance. 4004 * \param tiIORequest: Pointer to TISA I/O request context for this I/O. 4005 * \param tiDeviceHandle: Pointer to TISA device handle for this I/O. 4006 * \param tiScsiRequest: Pointer to TISA SCSI I/O request and SGL list. 4007 * \param smSatIOContext_t: Pointer to the SAT IO Context 4008 * 4009 * \return If command is started successfully 4010 * - \e smIOSuccess: I/O request successfully initiated. 4011 * - \e smIOBusy: No resources available, try again later. 4012 * - \e smIONoDevice: Invalid device handle. 4013 * - \e smIOError: Other errors. 4014 */ 4015 /*****************************************************************************/ 4016 osGLOBAL bit32 4017 smsatPacket( 4018 smRoot_t *smRoot, 4019 smIORequest_t *smIORequest, 4020 smDeviceHandle_t *smDeviceHandle, 4021 smScsiInitiatorRequest_t *smScsiRequest, 4022 smSatIOContext_t *satIOContext 4023 ) 4024 { 4025 bit32 status; 4026 bit32 agRequestType = AGSA_SATA_PROTOCOL_D2H_PKT; 4027 smDeviceData_t *pSatDevData; 4028 smIniScsiCmnd_t *scsiCmnd; 4029 agsaFisRegHostToDevice_t *fis; 4030 4031 pSatDevData = satIOContext->pSatDevData; 4032 scsiCmnd = &smScsiRequest->scsiCmnd; 4033 fis = satIOContext->pFis; 4034 4035 SM_DBG3(("smsatPacket: start, SCSI CDB is 0x%X %X %X %X %X %X %X %X %X %X %X %X\n", 4036 scsiCmnd->cdb[0],scsiCmnd->cdb[1],scsiCmnd->cdb[2],scsiCmnd->cdb[3], 4037 scsiCmnd->cdb[4],scsiCmnd->cdb[5],scsiCmnd->cdb[6],scsiCmnd->cdb[7], 4038 scsiCmnd->cdb[8],scsiCmnd->cdb[9],scsiCmnd->cdb[10],scsiCmnd->cdb[11])); 4039 4040 fis->h.fisType = 0x27; /* Reg host to device */ 4041 fis->h.c_pmPort = 0x80; /* C Bit is set 1*/ 4042 fis->h.command = SAT_PACKET; /* 0xA0 */ 4043 if (pSatDevData->satDMADIRSupport) /* DMADIR enabled*/ 4044 { 4045 fis->h.features = (smScsiRequest->dataDirection == smDirectionIn)? 0x04 : 0; /* 1 for D2H, 0 for H2D */ 4046 } 4047 else 4048 { 4049 fis->h.features = 0; /* FIS reserve */ 4050 } 4051 4052 if (pSatDevData->satDMASupport == agTRUE && pSatDevData->satDMAEnabled == agTRUE) 4053 { 4054 /*DMA transfer mode*/ 4055 fis->h.features |= 0x01; 4056 } 4057 else 4058 { 4059 /*PIO transfer mode*/ 4060 fis->h.features |= 0x0; 4061 } 4062 /* Byte count low and byte count high */ 4063 if ( scsiCmnd->expDataLength > 0xFFFF ) 4064 { 4065 fis->d.lbaMid = 0xFF; /* FIS LBA (15:8 ) */ 4066 fis->d.lbaHigh = 0xFF; /* FIS LBA (23:16) */ 4067 } 4068 else 4069 { 4070 fis->d.lbaMid = (bit8)scsiCmnd->expDataLength; /* FIS LBA (15:8 ) */ 4071 fis->d.lbaHigh = (bit8)(scsiCmnd->expDataLength>>8); /* FIS LBA (23:16) */ 4072 } 4073 4074 fis->d.lbaLow = 0; /* FIS LBA (7 :0 ) */ 4075 fis->d.device = 0; /* FIS LBA (27:24) and FIS LBA mode */ 4076 fis->d.lbaLowExp = 0; 4077 fis->d.lbaMidExp = 0; 4078 fis->d.lbaHighExp = 0; 4079 fis->d.featuresExp = 0; 4080 fis->d.sectorCount = 0; /* FIS sector count (7:0) */ 4081 fis->d.sectorCountExp = 0; 4082 fis->d.reserved4 = 0; 4083 fis->d.control = 0; /* FIS HOB bit clear */ 4084 fis->d.reserved5 = 0; 4085 4086 satIOContext->ATACmd = SAT_PACKET; 4087 4088 if (smScsiRequest->dataDirection == smDirectionIn) 4089 { 4090 agRequestType = AGSA_SATA_PROTOCOL_D2H_PKT; 4091 } 4092 else 4093 { 4094 agRequestType = AGSA_SATA_PROTOCOL_H2D_PKT; 4095 } 4096 4097 satIOContext->satCompleteCB = &smsatPacketCB; 4098 4099 /* 4100 * Prepare SGL and send FIS to LL layer. 4101 */ 4102 satIOContext->reqType = agRequestType; /* Save it */ 4103 4104 status = smsataLLIOStart(smRoot, 4105 smIORequest, 4106 smDeviceHandle, 4107 smScsiRequest, 4108 satIOContext); 4109 4110 SM_DBG3(("smsatPacket: return\n")); 4111 return (status); 4112 } 4113 4114 /*****************************************************************************/ 4115 /*! \brief SAT implementation for smsatSetFeaturePIO. 4116 * 4117 * This function creates Set Features fis and sends the request to LL layer 4118 * 4119 * \param tiRoot: Pointer to TISA initiator driver/port instance. 4120 * \param tiIORequest: Pointer to TISA I/O request context for this I/O. 4121 * \param tiDeviceHandle: Pointer to TISA device handle for this I/O. 4122 * \param tiScsiRequest: Pointer to TISA SCSI I/O request and SGL list. 4123 * \param smSatIOContext_t: Pointer to the SAT IO Context 4124 * 4125 * \return If command is started successfully 4126 * - \e smIOSuccess: I/O request successfully initiated. 4127 * - \e smIOBusy: No resources available, try again later. 4128 * - \e smIONoDevice: Invalid device handle. 4129 * - \e smIOError: Other errors. 4130 */ 4131 /*****************************************************************************/ 4132 osGLOBAL bit32 4133 smsatSetFeaturesPIO( 4134 smRoot_t *smRoot, 4135 smIORequest_t *smIORequest, 4136 smDeviceHandle_t *smDeviceHandle, 4137 smScsiInitiatorRequest_t *smScsiRequest, 4138 smSatIOContext_t *satIOContext 4139 ) 4140 { 4141 bit32 status = SM_RC_FAILURE; 4142 bit32 agRequestType; 4143 agsaFisRegHostToDevice_t *fis; 4144 4145 fis = satIOContext->pFis; 4146 SM_DBG2(("smsatSetFeaturesPIO: start\n")); 4147 /* 4148 * Send the Set Features command. 4149 */ 4150 fis->h.fisType = 0x27; /* Reg host to device */ 4151 fis->h.c_pmPort = 0x80; /* C Bit is set */ 4152 fis->h.command = SAT_SET_FEATURES; /* 0xEF */ 4153 fis->h.features = 0x03; /* set transfer mode */ 4154 fis->d.lbaLow = 0; 4155 fis->d.lbaMid = 0; 4156 fis->d.lbaHigh = 0; 4157 fis->d.device = 0; 4158 fis->d.lbaLowExp = 0; 4159 fis->d.lbaMidExp = 0; 4160 fis->d.lbaHighExp = 0; 4161 fis->d.featuresExp = 0; 4162 fis->d.sectorCountExp = 0; 4163 fis->d.reserved4 = 0; 4164 fis->d.control = 0; /* FIS HOB bit clear */ 4165 fis->d.reserved5 = 0; 4166 4167 agRequestType = AGSA_SATA_PROTOCOL_NON_DATA; 4168 4169 /* Initialize CB for SATA completion. 4170 */ 4171 fis->d.sectorCount = 0x0C; /*enable PIO transfer mode */ 4172 satIOContext->satCompleteCB = &smsatSetFeaturesPIOCB; 4173 4174 /* 4175 * Prepare SGL and send FIS to LL layer. 4176 */ 4177 satIOContext->reqType = agRequestType; /* Save it */ 4178 4179 status = smsataLLIOStart( smRoot, 4180 smIORequest, 4181 smDeviceHandle, 4182 smScsiRequest, 4183 satIOContext); 4184 4185 SM_DBG2(("smsatSetFeaturesPIO: return\n")); 4186 /* debugging code */ 4187 if (smIORequest->tdData == smIORequest->smData) 4188 { 4189 SM_DBG1(("smsatSetFeaturesPIO: incorrect smIORequest\n")); 4190 } 4191 4192 return status; 4193 } 4194 /*****************************************************************************/ 4195 /*! \brief SAT implementation for SCSI REQUEST SENSE to ATAPI device. 4196 * 4197 * SAT implementation for SCSI REQUEST SENSE. 4198 * 4199 * \param tiRoot: Pointer to TISA initiator driver/port instance. 4200 * \param tiIORequest: Pointer to TISA I/O request context for this I/O. 4201 * \param tiDeviceHandle: Pointer to TISA device handle for this I/O. 4202 * \param tiScsiRequest: Pointer to TISA SCSI I/O request and SGL list. 4203 * \param smSatIOContext_t: Pointer to the SAT IO Context 4204 * 4205 * \return If command is started successfully 4206 * - \e smIOSuccess: I/O request successfully initiated. 4207 * - \e smIOBusy: No resources available, try again later. 4208 * - \e smIONoDevice: Invalid device handle. 4209 * - \e smIOError: Other errors. 4210 */ 4211 /*****************************************************************************/ 4212 osGLOBAL bit32 4213 smsatRequestSenseForATAPI( 4214 smRoot_t *smRoot, 4215 smIORequest_t *smIORequest, 4216 smDeviceHandle_t *smDeviceHandle, 4217 smScsiInitiatorRequest_t *smScsiRequest, 4218 smSatIOContext_t *satIOContext 4219 ) 4220 { 4221 bit32 status; 4222 bit32 agRequestType = AGSA_SATA_PROTOCOL_D2H_PKT; 4223 smDeviceData_t *pSatDevData; 4224 smIniScsiCmnd_t *scsiCmnd; 4225 agsaFisRegHostToDevice_t *fis; 4226 4227 pSatDevData = satIOContext->pSatDevData; 4228 scsiCmnd = &smScsiRequest->scsiCmnd; 4229 fis = satIOContext->pFis; 4230 4231 scsiCmnd->cdb[0] = SCSIOPC_REQUEST_SENSE; 4232 scsiCmnd->cdb[1] = 0; 4233 scsiCmnd->cdb[2] = 0; 4234 scsiCmnd->cdb[3] = 0; 4235 scsiCmnd->cdb[4] = (bit8)scsiCmnd->expDataLength; 4236 scsiCmnd->cdb[5] = 0; 4237 SM_DBG3(("smsatRequestSenseForATAPI: start, SCSI CDB is 0x%X %X %X %X %X %X %X %X %X %X %X %X\n", 4238 scsiCmnd->cdb[0],scsiCmnd->cdb[1],scsiCmnd->cdb[2],scsiCmnd->cdb[3], 4239 scsiCmnd->cdb[4],scsiCmnd->cdb[5],scsiCmnd->cdb[6],scsiCmnd->cdb[7], 4240 scsiCmnd->cdb[8],scsiCmnd->cdb[9],scsiCmnd->cdb[10],scsiCmnd->cdb[11])); 4241 4242 fis->h.fisType = 0x27; /* Reg host to device */ 4243 fis->h.c_pmPort = 0x80; /* C Bit is set 1*/ 4244 fis->h.command = SAT_PACKET; /* 0xA0 */ 4245 if (pSatDevData->satDMADIRSupport) /* DMADIR enabled*/ 4246 { 4247 fis->h.features = (smScsiRequest->dataDirection == smDirectionIn)? 0x04 : 0; /* 1 for D2H, 0 for H2D */ 4248 } 4249 else 4250 { 4251 fis->h.features = 0; /* FIS reserve */ 4252 } 4253 4254 if (pSatDevData->satDMASupport == agTRUE && pSatDevData->satDMAEnabled == agTRUE) 4255 { 4256 fis->h.features |= 0x01; 4257 } 4258 else 4259 { 4260 fis->h.features |= 0x0; 4261 } 4262 4263 fis->d.lbaLow = 0; /* FIS LBA (7 :0 ) */ 4264 fis->d.lbaMid = (bit8)scsiCmnd->expDataLength; /* FIS LBA (15:8 ) */ 4265 fis->d.lbaHigh = (bit8)(scsiCmnd->expDataLength>>8); /* FIS LBA (23:16) */ 4266 fis->d.device = 0; /* FIS LBA (27:24) and FIS LBA mode */ 4267 fis->d.lbaLowExp = 0; 4268 fis->d.lbaMidExp = 0; 4269 fis->d.lbaHighExp = 0; 4270 fis->d.featuresExp = 0; 4271 fis->d.sectorCount = 0; /* FIS sector count (7:0) */ 4272 fis->d.sectorCountExp = 0; 4273 fis->d.reserved4 = 0; 4274 fis->d.control = 0; /* FIS HOB bit clear */ 4275 fis->d.reserved5 = 0; 4276 4277 satIOContext->ATACmd = SAT_PACKET; 4278 4279 agRequestType = AGSA_SATA_PROTOCOL_D2H_PKT; 4280 4281 4282 satIOContext->satCompleteCB = &smsatRequestSenseForATAPICB; 4283 4284 /* 4285 * Prepare SGL and send FIS to LL layer. 4286 */ 4287 satIOContext->reqType = agRequestType; /* Save it */ 4288 4289 status = smsataLLIOStart( smRoot, 4290 smIORequest, 4291 smDeviceHandle, 4292 smScsiRequest, 4293 satIOContext); 4294 4295 SM_DBG3(("smsatRequestSenseForATAPI: return\n")); 4296 return (status); 4297 } 4298 /*****************************************************************************/ 4299 /*! \brief SAT implementation for smsatDeviceReset. 4300 * 4301 * This function creates DEVICE RESET fis and sends the request to LL layer 4302 * 4303 * \param smRoot: Pointer to TISA initiator driver/port instance. 4304 * \param smIORequest: Pointer to TISA I/O request context for this I/O. 4305 * \param smDeviceHandle: Pointer to TISA device handle for this I/O. 4306 * \param smScsiRequest: Pointer to TISA SCSI I/O request and SGL list. 4307 * \param smSatIOContext_t: Pointer to the SAT IO Context 4308 * 4309 * \return If command is started successfully 4310 * - \e smIOSuccess: I/O request successfully initiated. 4311 * - \e smIOBusy: No resources available, try again later. 4312 * - \e smIONoDevice: Invalid device handle. 4313 * - \e smIOError: Other errors. 4314 */ 4315 /*****************************************************************************/ 4316 osGLOBAL bit32 4317 smsatDeviceReset( 4318 smRoot_t *smRoot, 4319 smIORequest_t *smIORequest, 4320 smDeviceHandle_t *smDeviceHandle, 4321 smScsiInitiatorRequest_t *smScsiRequest, 4322 smSatIOContext_t *satIOContext 4323 ) 4324 { 4325 bit32 status; 4326 bit32 agRequestType; 4327 agsaFisRegHostToDevice_t *fis; 4328 4329 fis = satIOContext->pFis; 4330 SM_DBG3(("smsatDeviceReset: start\n")); 4331 /* 4332 * Send the Execute Device Diagnostic command. 4333 */ 4334 fis->h.fisType = 0x27; /* Reg host to device */ 4335 fis->h.c_pmPort = 0x80; /* C Bit is set */ 4336 fis->h.command = SAT_DEVICE_RESET; /* 0x08 */ 4337 fis->h.features = 0; 4338 fis->d.lbaLow = 0; 4339 fis->d.lbaMid = 0; 4340 fis->d.lbaHigh = 0; 4341 fis->d.device = 0; 4342 fis->d.lbaLowExp = 0; 4343 fis->d.lbaMidExp = 0; 4344 fis->d.lbaHighExp = 0; 4345 fis->d.featuresExp = 0; 4346 fis->d.sectorCount = 0; 4347 fis->d.sectorCountExp = 0; 4348 fis->d.reserved4 = 0; 4349 fis->d.control = 0; /* FIS HOB bit clear */ 4350 fis->d.reserved5 = 0; 4351 4352 agRequestType = AGSA_SATA_PROTOCOL_DEV_RESET; 4353 4354 /* Initialize CB for SATA completion. 4355 */ 4356 satIOContext->satCompleteCB = &smsatDeviceResetCB; 4357 4358 /* 4359 * Prepare SGL and send FIS to LL layer. 4360 */ 4361 satIOContext->reqType = agRequestType; /* Save it */ 4362 4363 status = smsataLLIOStart( smRoot, 4364 smIORequest, 4365 smDeviceHandle, 4366 smScsiRequest, 4367 satIOContext); 4368 4369 SM_DBG3(("smsatDeviceReset: return\n")); 4370 4371 return status; 4372 } 4373 4374 4375 /*****************************************************************************/ 4376 /*! \brief SAT implementation for smsatExecuteDeviceDiagnostic. 4377 * 4378 * This function creates Execute Device Diagnostic fis and sends the request to LL layer 4379 * 4380 * \param smRoot: Pointer to TISA initiator driver/port instance. 4381 * \param smIORequest: Pointer to TISA I/O request context for this I/O. 4382 * \param smDeviceHandle: Pointer to TISA device handle for this I/O. 4383 * \param smScsiRequest: Pointer to TISA SCSI I/O request and SGL list. 4384 * \param smSatIOContext_t: Pointer to the SAT IO Context 4385 * 4386 * \return If command is started successfully 4387 * - \e smIOSuccess: I/O request successfully initiated. 4388 * - \e smIOBusy: No resources available, try again later. 4389 * - \e smIONoDevice: Invalid device handle. 4390 * - \e smIOError: Other errors. 4391 */ 4392 /*****************************************************************************/ 4393 osGLOBAL bit32 4394 smsatExecuteDeviceDiagnostic( 4395 smRoot_t *smRoot, 4396 smIORequest_t *smIORequest, 4397 smDeviceHandle_t *smDeviceHandle, 4398 smScsiInitiatorRequest_t *smScsiRequest, 4399 smSatIOContext_t *satIOContext 4400 ) 4401 { 4402 bit32 status; 4403 bit32 agRequestType; 4404 agsaFisRegHostToDevice_t *fis; 4405 4406 fis = satIOContext->pFis; 4407 SM_DBG3(("smsatExecuteDeviceDiagnostic: start\n")); 4408 /* 4409 * Send the Execute Device Diagnostic command. 4410 */ 4411 fis->h.fisType = 0x27; /* Reg host to device */ 4412 fis->h.c_pmPort = 0x80; /* C Bit is set */ 4413 fis->h.command = SAT_EXECUTE_DEVICE_DIAGNOSTIC; /* 0x90 */ 4414 fis->h.features = 0; 4415 fis->d.lbaLow = 0; 4416 fis->d.lbaMid = 0; 4417 fis->d.lbaHigh = 0; 4418 fis->d.device = 0; 4419 fis->d.lbaLowExp = 0; 4420 fis->d.lbaMidExp = 0; 4421 fis->d.lbaHighExp = 0; 4422 fis->d.featuresExp = 0; 4423 fis->d.sectorCount = 0; 4424 fis->d.sectorCountExp = 0; 4425 fis->d.reserved4 = 0; 4426 fis->d.control = 0; /* FIS HOB bit clear */ 4427 fis->d.reserved5 = 0; 4428 4429 agRequestType = AGSA_SATA_PROTOCOL_NON_DATA; 4430 4431 /* Initialize CB for SATA completion. 4432 */ 4433 satIOContext->satCompleteCB = &smsatExecuteDeviceDiagnosticCB; 4434 4435 /* 4436 * Prepare SGL and send FIS to LL layer. 4437 */ 4438 satIOContext->reqType = agRequestType; /* Save it */ 4439 4440 status = smsataLLIOStart( smRoot, 4441 smIORequest, 4442 smDeviceHandle, 4443 smScsiRequest, 4444 satIOContext); 4445 4446 SM_DBG3(("smsatExecuteDeviceDiagnostic: return\n")); 4447 4448 return status; 4449 } 4450 4451 4452 osGLOBAL void 4453 smsatSetDeferredSensePayload( 4454 smScsiRspSense_t *pSense, 4455 bit8 SnsKey, 4456 bit32 SnsInfo, 4457 bit16 SnsCode, 4458 smSatIOContext_t *satIOContext 4459 ) 4460 { 4461 SM_DBG2(("smsatSetDeferredSensePayload: start\n")); 4462 return; 4463 } 4464 4465 4466 GLOBAL bit32 4467 smsatRead6( 4468 smRoot_t *smRoot, 4469 smIORequest_t *smIORequest, 4470 smDeviceHandle_t *smDeviceHandle, 4471 smScsiInitiatorRequest_t *smScsiRequest, 4472 smSatIOContext_t *satIOContext 4473 ) 4474 { 4475 bit32 status; 4476 bit32 agRequestType = AGSA_SATA_PROTOCOL_DMA_READ; 4477 smDeviceData_t *pSatDevData; 4478 smScsiRspSense_t *pSense; 4479 smIniScsiCmnd_t *scsiCmnd; 4480 agsaFisRegHostToDevice_t *fis; 4481 bit32 lba = 0; 4482 bit16 tl = 0; 4483 4484 pSense = satIOContext->pSense; 4485 pSatDevData = satIOContext->pSatDevData; 4486 scsiCmnd = &smScsiRequest->scsiCmnd; 4487 fis = satIOContext->pFis; 4488 4489 SM_DBG2(("smsatRead6: start\n")); 4490 4491 /* no FUA checking since read6 */ 4492 4493 4494 /* checking CONTROL */ 4495 /* NACA == 1 or LINK == 1*/ 4496 if ( (scsiCmnd->cdb[5] & SCSI_NACA_MASK) || (scsiCmnd->cdb[5] & SCSI_LINK_MASK) ) 4497 { 4498 smsatSetSensePayload( pSense, 4499 SCSI_SNSKEY_ILLEGAL_REQUEST, 4500 0, 4501 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 4502 satIOContext); 4503 4504 /*smEnqueueIO(smRoot, satIOContext);*/ 4505 4506 tdsmIOCompletedCB( smRoot, 4507 smIORequest, 4508 smIOSuccess, 4509 SCSI_STAT_CHECK_CONDITION, 4510 satIOContext->pSmSenseData, 4511 satIOContext->interruptContext ); 4512 4513 SM_DBG1(("smsatRead6: return control!!!\n")); 4514 return SM_RC_SUCCESS; 4515 } 4516 4517 /* cbd6; computing LBA and transfer length */ 4518 lba = (((scsiCmnd->cdb[1]) & 0x1f) << (8*2)) 4519 + (scsiCmnd->cdb[2] << 8) + scsiCmnd->cdb[3]; 4520 tl = scsiCmnd->cdb[4]; 4521 4522 /* Table 34, 9.1, p 46 */ 4523 /* 4524 note: As of 2/10/2006, no support for DMA QUEUED 4525 */ 4526 4527 /* 4528 Table 34, 9.1, p 46, b 4529 When no 48-bit addressing support or NCQ, if LBA is beyond (2^28 - 1), 4530 return check condition 4531 */ 4532 if (pSatDevData->satNCQ != agTRUE && 4533 pSatDevData->sat48BitSupport != agTRUE 4534 ) 4535 { 4536 if (lba > SAT_TR_LBA_LIMIT - 1) 4537 { 4538 smsatSetSensePayload( pSense, 4539 SCSI_SNSKEY_ILLEGAL_REQUEST, 4540 0, 4541 SCSI_SNSCODE_LOGICAL_BLOCK_ADDRESS_OUT_OF_RANGE, 4542 satIOContext); 4543 4544 /*smEnqueueIO(smRoot, satIOContext);*/ 4545 4546 tdsmIOCompletedCB( smRoot, 4547 smIORequest, 4548 smIOSuccess, 4549 SCSI_STAT_CHECK_CONDITION, 4550 satIOContext->pSmSenseData, 4551 satIOContext->interruptContext ); 4552 4553 SM_DBG1(("smsatRead6: return LBA out of range!!!\n")); 4554 return SM_RC_SUCCESS; 4555 } 4556 } 4557 4558 /* case 1 and 2 */ 4559 if (lba + tl <= SAT_TR_LBA_LIMIT) 4560 { 4561 if (pSatDevData->satDMASupport == agTRUE && pSatDevData->satDMAEnabled == agTRUE) 4562 { 4563 /* case 2 */ 4564 /* READ DMA*/ 4565 SM_DBG5(("smsatRead6: case 2\n")); 4566 4567 4568 fis->h.fisType = 0x27; /* Reg host to device */ 4569 fis->h.c_pmPort = 0x80; /* C Bit is set */ 4570 fis->h.command = SAT_READ_DMA; /* 0xC8 */ 4571 fis->h.features = 0; /* FIS reserve */ 4572 fis->d.lbaLow = scsiCmnd->cdb[3]; /* FIS LBA (7 :0 ) */ 4573 fis->d.lbaMid = scsiCmnd->cdb[2]; /* FIS LBA (15:8 ) */ 4574 fis->d.lbaHigh = (bit8)((scsiCmnd->cdb[1]) & 0x1f); /* FIS LBA (23:16) */ 4575 fis->d.device = 0x40; /* FIS LBA mode */ 4576 fis->d.lbaLowExp = 0; 4577 fis->d.lbaMidExp = 0; 4578 fis->d.lbaHighExp = 0; 4579 fis->d.featuresExp = 0; 4580 if (tl == 0) 4581 { 4582 /* temporary fix */ 4583 fis->d.sectorCount = 0xff; /* FIS sector count (7:0) */ 4584 } 4585 else 4586 { 4587 fis->d.sectorCount = scsiCmnd->cdb[4]; /* FIS sector count (7:0) */ 4588 } 4589 fis->d.sectorCountExp = 0; 4590 fis->d.reserved4 = 0; 4591 fis->d.control = 0; /* FIS HOB bit clear */ 4592 fis->d.reserved5 = 0; 4593 4594 agRequestType = AGSA_SATA_PROTOCOL_DMA_READ; 4595 } 4596 else 4597 { 4598 /* case 1 */ 4599 /* READ SECTORS for easier implemetation */ 4600 SM_DBG5(("smsatRead6: case 1\n")); 4601 4602 fis->h.fisType = 0x27; /* Reg host to device */ 4603 fis->h.c_pmPort = 0x80; /* C Bit is set */ 4604 fis->h.command = SAT_READ_SECTORS; /* 0x20 */ 4605 fis->h.features = 0; /* FIS reserve */ 4606 fis->d.lbaLow = scsiCmnd->cdb[3]; /* FIS LBA (7 :0 ) */ 4607 fis->d.lbaMid = scsiCmnd->cdb[2]; /* FIS LBA (15:8 ) */ 4608 fis->d.lbaHigh = (bit8)((scsiCmnd->cdb[1]) & 0x1f); /* FIS LBA (23:16) */ 4609 fis->d.device = 0x40; /* FIS LBA mode */ 4610 fis->d.lbaLowExp = 0; 4611 fis->d.lbaMidExp = 0; 4612 fis->d.lbaHighExp = 0; 4613 fis->d.featuresExp = 0; 4614 if (tl == 0) 4615 { 4616 /* temporary fix */ 4617 fis->d.sectorCount = 0xff; /* FIS sector count (7:0) */ 4618 } 4619 else 4620 { 4621 fis->d.sectorCount = scsiCmnd->cdb[4]; /* FIS sector count (7:0) */ 4622 } 4623 fis->d.sectorCountExp = 0; 4624 fis->d.reserved4 = 0; 4625 fis->d.control = 0; /* FIS HOB bit clear */ 4626 fis->d.reserved5 = 0; 4627 4628 agRequestType = AGSA_SATA_PROTOCOL_PIO_READ; 4629 4630 } 4631 } 4632 4633 /* case 3 and 4 */ 4634 if (pSatDevData->sat48BitSupport == agTRUE) 4635 { 4636 if (pSatDevData->satDMASupport == agTRUE && pSatDevData->satDMAEnabled == agTRUE) 4637 { 4638 /* case 3 */ 4639 /* READ DMA EXT only */ 4640 SM_DBG5(("smsatRead6: case 3\n")); 4641 fis->h.fisType = 0x27; /* Reg host to device */ 4642 fis->h.c_pmPort = 0x80; /* C Bit is set */ 4643 fis->h.command = SAT_READ_DMA_EXT; /* 0x25 */ 4644 fis->h.features = 0; /* FIS reserve */ 4645 fis->d.lbaLow = scsiCmnd->cdb[3]; /* FIS LBA (7 :0 ) */ 4646 fis->d.lbaMid = scsiCmnd->cdb[2]; /* FIS LBA (15:8 ) */ 4647 fis->d.lbaHigh = (bit8)((scsiCmnd->cdb[1]) & 0x1f); /* FIS LBA (23:16) */ 4648 fis->d.device = 0x40; /* FIS LBA mode set */ 4649 fis->d.lbaLowExp = 0; /* FIS LBA (31:24) */ 4650 fis->d.lbaMidExp = 0; /* FIS LBA (39:32) */ 4651 fis->d.lbaHighExp = 0; /* FIS LBA (47:40) */ 4652 fis->d.featuresExp = 0; /* FIS reserve */ 4653 if (tl == 0) 4654 { 4655 /* sector count is 256, 0x100*/ 4656 fis->d.sectorCount = 0; /* FIS sector count (7:0) */ 4657 fis->d.sectorCountExp = 0x01; /* FIS sector count (15:8) */ 4658 } 4659 else 4660 { 4661 fis->d.sectorCount = scsiCmnd->cdb[4]; /* FIS sector count (7:0) */ 4662 fis->d.sectorCountExp = 0; /* FIS sector count (15:8) */ 4663 } 4664 fis->d.reserved4 = 0; 4665 fis->d.control = 0; /* FIS HOB bit clear */ 4666 fis->d.reserved5 = 0; 4667 4668 agRequestType = AGSA_SATA_PROTOCOL_DMA_READ; 4669 } 4670 else 4671 { 4672 /* case 4 */ 4673 /* READ SECTORS EXT for easier implemetation */ 4674 SM_DBG5(("smsatRead6: case 4\n")); 4675 4676 fis->h.fisType = 0x27; /* Reg host to device */ 4677 fis->h.c_pmPort = 0x80; /* C Bit is set */ 4678 fis->h.command = SAT_READ_SECTORS_EXT; /* 0x24 */ 4679 fis->h.features = 0; /* FIS reserve */ 4680 fis->d.lbaLow = scsiCmnd->cdb[3]; /* FIS LBA (7 :0 ) */ 4681 fis->d.lbaMid = scsiCmnd->cdb[2]; /* FIS LBA (15:8 ) */ 4682 fis->d.lbaHigh = (bit8)((scsiCmnd->cdb[1]) & 0x1f); /* FIS LBA (23:16) */ 4683 fis->d.device = 0x40; /* FIS LBA mode set */ 4684 fis->d.lbaLowExp = 0; /* FIS LBA (31:24) */ 4685 fis->d.lbaMidExp = 0; /* FIS LBA (39:32) */ 4686 fis->d.lbaHighExp = 0; /* FIS LBA (47:40) */ 4687 fis->d.featuresExp = 0; /* FIS reserve */ 4688 if (tl == 0) 4689 { 4690 /* sector count is 256, 0x100*/ 4691 fis->d.sectorCount = 0; /* FIS sector count (7:0) */ 4692 fis->d.sectorCountExp = 0x01; /* FIS sector count (15:8) */ 4693 } 4694 else 4695 { 4696 fis->d.sectorCount = scsiCmnd->cdb[4]; /* FIS sector count (7:0) */ 4697 fis->d.sectorCountExp = 0; /* FIS sector count (15:8) */ 4698 } 4699 fis->d.reserved4 = 0; 4700 fis->d.control = 0; /* FIS HOB bit clear */ 4701 fis->d.reserved5 = 0; 4702 4703 agRequestType = AGSA_SATA_PROTOCOL_PIO_READ; 4704 } 4705 } 4706 4707 /* case 5 */ 4708 if (pSatDevData->satNCQ == agTRUE) 4709 { 4710 /* READ FPDMA QUEUED */ 4711 if (pSatDevData->sat48BitSupport != agTRUE) 4712 { 4713 /* sanity check */ 4714 SM_DBG1(("smsatRead6: case 5 !!! error NCQ but 28 bit address support!!!\n")); 4715 smsatSetSensePayload( pSense, 4716 SCSI_SNSKEY_ILLEGAL_REQUEST, 4717 0, 4718 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 4719 satIOContext); 4720 4721 /*smEnqueueIO(smRoot, satIOContext);*/ 4722 4723 tdsmIOCompletedCB( smRoot, 4724 smIORequest, 4725 smIOSuccess, 4726 SCSI_STAT_CHECK_CONDITION, 4727 satIOContext->pSmSenseData, 4728 satIOContext->interruptContext ); 4729 return SM_RC_SUCCESS; 4730 } 4731 SM_DBG5(("smsatRead6: case 5\n")); 4732 4733 /* Support 48-bit FPDMA addressing, use READ FPDMA QUEUE command */ 4734 4735 fis->h.fisType = 0x27; /* Reg host to device */ 4736 fis->h.c_pmPort = 0x80; /* C Bit is set */ 4737 fis->h.command = SAT_READ_FPDMA_QUEUED; /* 0x60 */ 4738 fis->d.lbaLow = scsiCmnd->cdb[3]; /* FIS LBA (7 :0 ) */ 4739 fis->d.lbaMid = scsiCmnd->cdb[2]; /* FIS LBA (15:8 ) */ 4740 fis->d.lbaHigh = (bit8)((scsiCmnd->cdb[1]) & 0x1f); /* FIS LBA (23:16) */ 4741 fis->d.device = 0x40; /* FIS FUA clear */ 4742 fis->d.lbaLowExp = 0; /* FIS LBA (31:24) */ 4743 fis->d.lbaMidExp = 0; /* FIS LBA (39:32) */ 4744 fis->d.lbaHighExp = 0; /* FIS LBA (47:40) */ 4745 if (tl == 0) 4746 { 4747 /* sector count is 256, 0x100*/ 4748 fis->h.features = 0; /* FIS sector count (7:0) */ 4749 fis->d.featuresExp = 0x01; /* FIS sector count (15:8) */ 4750 } 4751 else 4752 { 4753 fis->h.features = scsiCmnd->cdb[4]; /* FIS sector count (7:0) */ 4754 fis->d.featuresExp = 0; /* FIS sector count (15:8) */ 4755 } 4756 fis->d.sectorCount = 0; /* Tag (7:3) set by LL layer */ 4757 fis->d.sectorCountExp = 0; 4758 fis->d.reserved4 = 0; 4759 fis->d.control = 0; /* FIS HOB bit clear */ 4760 fis->d.reserved5 = 0; 4761 4762 agRequestType = AGSA_SATA_PROTOCOL_FPDMA_READ; 4763 } 4764 4765 /* Initialize CB for SATA completion. 4766 */ 4767 satIOContext->satCompleteCB = &smsatNonChainedDataIOCB; 4768 4769 /* 4770 * Prepare SGL and send FIS to LL layer. 4771 */ 4772 satIOContext->reqType = agRequestType; /* Save it */ 4773 4774 status = smsataLLIOStart( smRoot, 4775 smIORequest, 4776 smDeviceHandle, 4777 smScsiRequest, 4778 satIOContext); 4779 return (status); 4780 4781 } 4782 4783 osGLOBAL FORCEINLINE bit32 4784 smsatRead10( 4785 smRoot_t *smRoot, 4786 smIORequest_t *smIORequest, 4787 smDeviceHandle_t *smDeviceHandle, 4788 smScsiInitiatorRequest_t *smScsiRequest, 4789 smSatIOContext_t *satIOContext 4790 ) 4791 { 4792 smDeviceData_t *pSatDevData = satIOContext->pSatDevData; 4793 smScsiRspSense_t *pSense = satIOContext->pSense; 4794 smIniScsiCmnd_t *scsiCmnd = &smScsiRequest->scsiCmnd; 4795 agsaFisRegHostToDevice_t *fis = satIOContext->pFis; 4796 4797 bit32 status; 4798 bit32 agRequestType = AGSA_SATA_PROTOCOL_DMA_READ; 4799 bit32 lba = 0; 4800 bit32 tl = 0; 4801 bit32 LoopNum = 1; 4802 bit8 LBA[8]; 4803 bit8 TL[8]; 4804 bit32 AllChk = agFALSE; /* lba, lba+tl check against ATA limit and Disk capacity */ 4805 4806 SM_DBG2(("smsatRead10: start\n")); 4807 SM_DBG2(("smsatRead10: pSatDevData did=%d\n", pSatDevData->id)); 4808 // smhexdump("smsatRead10", (bit8 *)scsiCmnd->cdb, 10); 4809 4810 /* checking FUA_NV */ 4811 if (scsiCmnd->cdb[1] & SCSI_FUA_NV_MASK) 4812 { 4813 smsatSetSensePayload( pSense, 4814 SCSI_SNSKEY_ILLEGAL_REQUEST, 4815 0, 4816 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 4817 satIOContext); 4818 4819 /*smEnqueueIO(smRoot, satIOContext);*/ 4820 4821 tdsmIOCompletedCB( smRoot, 4822 smIORequest, 4823 smIOSuccess, 4824 SCSI_STAT_CHECK_CONDITION, 4825 satIOContext->pSmSenseData, 4826 satIOContext->interruptContext ); 4827 4828 SM_DBG1(("smsatRead10: return FUA_NV!!!\n")); 4829 return SM_RC_SUCCESS; 4830 4831 } 4832 4833 /* checking CONTROL */ 4834 /* NACA == 1 or LINK == 1*/ 4835 if ( (scsiCmnd->cdb[9] & SCSI_NACA_MASK) || (scsiCmnd->cdb[9] & SCSI_LINK_MASK) ) 4836 { 4837 smsatSetSensePayload( pSense, 4838 SCSI_SNSKEY_ILLEGAL_REQUEST, 4839 0, 4840 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 4841 satIOContext); 4842 4843 /*smEnqueueIO(smRoot, satIOContext);*/ 4844 4845 tdsmIOCompletedCB( smRoot, 4846 smIORequest, 4847 smIOSuccess, 4848 SCSI_STAT_CHECK_CONDITION, 4849 satIOContext->pSmSenseData, 4850 satIOContext->interruptContext ); 4851 4852 SM_DBG1(("smsatRead10: return control!!!\n")); 4853 return SM_RC_SUCCESS; 4854 } 4855 /* 4856 sm_memset(LBA, 0, sizeof(LBA)); 4857 sm_memset(TL, 0, sizeof(TL)); 4858 */ 4859 /* do not use memcpy due to indexing in LBA and TL */ 4860 LBA[0] = 0; /* MSB */ 4861 LBA[1] = 0; 4862 LBA[2] = 0; 4863 LBA[3] = 0; 4864 LBA[4] = scsiCmnd->cdb[2]; 4865 LBA[5] = scsiCmnd->cdb[3]; 4866 LBA[6] = scsiCmnd->cdb[4]; 4867 LBA[7] = scsiCmnd->cdb[5]; /* LSB */ 4868 4869 TL[0] = 0; 4870 TL[1] = 0; 4871 TL[2] = 0; 4872 TL[3] = 0; 4873 TL[4] = 0; 4874 TL[5] = 0; 4875 TL[6] = scsiCmnd->cdb[7]; 4876 TL[7] = scsiCmnd->cdb[8]; /* LSB */ 4877 4878 4879 /* cbd10; computing LBA and transfer length */ 4880 lba = (scsiCmnd->cdb[2] << 24) + (scsiCmnd->cdb[3] << 16) 4881 + (scsiCmnd->cdb[4] << 8) + scsiCmnd->cdb[5]; 4882 tl = (scsiCmnd->cdb[7] << 8) + scsiCmnd->cdb[8]; 4883 4884 4885 SM_DBG5(("smsatRead10: lba %d functioned lba %d\n", lba, smsatComputeCDB10LBA(satIOContext))); 4886 SM_DBG5(("smsatRead10: lba 0x%x functioned lba 0x%x\n", lba, smsatComputeCDB10LBA(satIOContext))); 4887 SM_DBG5(("smsatRead10: tl %d functioned tl %d\n", tl, smsatComputeCDB10TL(satIOContext))); 4888 4889 /* Table 34, 9.1, p 46 */ 4890 /* 4891 note: As of 2/10/2006, no support for DMA QUEUED 4892 */ 4893 4894 /* 4895 Table 34, 9.1, p 46, b 4896 When no 48-bit addressing support or NCQ, if LBA is beyond (2^28 - 1), 4897 return check condition 4898 */ 4899 4900 if (pSatDevData->satNCQ != agTRUE && 4901 pSatDevData->sat48BitSupport != agTRUE 4902 ) 4903 { 4904 AllChk = smsatCheckLimit(LBA, TL, agFALSE, pSatDevData); 4905 if (AllChk) 4906 { 4907 SM_DBG1(("smsatRead10: return LBA out of range, not EXT!!!\n")); 4908 smsatSetSensePayload( pSense, 4909 SCSI_SNSKEY_ILLEGAL_REQUEST, 4910 0, 4911 SCSI_SNSCODE_LOGICAL_BLOCK_ADDRESS_OUT_OF_RANGE, 4912 satIOContext); 4913 4914 /*smEnqueueIO(smRoot, satIOContext);*/ 4915 4916 tdsmIOCompletedCB( smRoot, 4917 smIORequest, 4918 smIOSuccess, 4919 SCSI_STAT_CHECK_CONDITION, 4920 satIOContext->pSmSenseData, 4921 satIOContext->interruptContext ); 4922 4923 return SM_RC_SUCCESS; 4924 } 4925 } 4926 else 4927 { 4928 AllChk = smsatCheckLimit(LBA, TL, agTRUE, pSatDevData); 4929 if (AllChk) 4930 { 4931 SM_DBG1(("smsatRead10: return LBA out of range, EXT!!!\n")); 4932 smsatSetSensePayload( pSense, 4933 SCSI_SNSKEY_ILLEGAL_REQUEST, 4934 0, 4935 SCSI_SNSCODE_LOGICAL_BLOCK_ADDRESS_OUT_OF_RANGE, 4936 satIOContext); 4937 4938 /*smEnqueueIO(smRoot, satIOContext);*/ 4939 4940 tdsmIOCompletedCB( smRoot, 4941 smIORequest, 4942 smIOSuccess, 4943 SCSI_STAT_CHECK_CONDITION, 4944 satIOContext->pSmSenseData, 4945 satIOContext->interruptContext ); 4946 4947 return SM_RC_SUCCESS; 4948 } 4949 } 4950 /* case 5 */ 4951 if (pSatDevData->satNCQ == agTRUE) 4952 { 4953 /* READ FPDMA QUEUED */ 4954 if (pSatDevData->sat48BitSupport != agTRUE) 4955 { 4956 SM_DBG1(("smsatRead10: case 5 !!! error NCQ but 28 bit address support!!!\n")); 4957 smsatSetSensePayload( pSense, 4958 SCSI_SNSKEY_ILLEGAL_REQUEST, 4959 0, 4960 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 4961 satIOContext); 4962 4963 /*smEnqueueIO(smRoot, satIOContext);*/ 4964 4965 tdsmIOCompletedCB( smRoot, 4966 smIORequest, 4967 smIOSuccess, 4968 SCSI_STAT_CHECK_CONDITION, 4969 satIOContext->pSmSenseData, 4970 satIOContext->interruptContext ); 4971 return SM_RC_SUCCESS; 4972 } 4973 4974 SM_DBG6(("smsatRead10: case 5\n")); 4975 4976 /* Support 48-bit FPDMA addressing, use READ FPDMA QUEUE command */ 4977 4978 fis->h.fisType = 0x27; /* Reg host to device */ 4979 fis->h.c_pmPort = 0x80; /* C Bit is set */ 4980 fis->h.command = SAT_READ_FPDMA_QUEUED; /* 0x60 */ 4981 fis->h.features = scsiCmnd->cdb[8]; /* FIS sector count (7:0) */ 4982 fis->d.lbaLow = scsiCmnd->cdb[5]; /* FIS LBA (7 :0 ) */ 4983 fis->d.lbaMid = scsiCmnd->cdb[4]; /* FIS LBA (15:8 ) */ 4984 fis->d.lbaHigh = scsiCmnd->cdb[3]; /* FIS LBA (23:16) */ 4985 4986 /* Check FUA bit */ 4987 if (scsiCmnd->cdb[1] & SCSI_READ10_FUA_MASK) 4988 fis->d.device = 0xC0; /* FIS FUA set */ 4989 else 4990 fis->d.device = 0x40; /* FIS FUA clear */ 4991 4992 fis->d.lbaLowExp = scsiCmnd->cdb[2]; /* FIS LBA (31:24) */ 4993 fis->d.lbaMidExp = 0; /* FIS LBA (39:32) */ 4994 fis->d.lbaHighExp = 0; /* FIS LBA (47:40) */ 4995 fis->d.featuresExp = scsiCmnd->cdb[7]; /* FIS sector count (15:8) */ 4996 fis->d.sectorCount = 0; /* Tag (7:3) set by LL layer */ 4997 fis->d.sectorCountExp = 0; 4998 fis->d.reserved4 = 0; 4999 fis->d.control = 0; /* FIS HOB bit clear */ 5000 fis->d.reserved5 = 0; 5001 5002 agRequestType = AGSA_SATA_PROTOCOL_FPDMA_READ; 5003 satIOContext->ATACmd = SAT_READ_FPDMA_QUEUED; 5004 } 5005 else if (pSatDevData->sat48BitSupport == agTRUE) /* case 3 and 4 */ 5006 { 5007 if (pSatDevData->satDMASupport == agTRUE && pSatDevData->satDMAEnabled == agTRUE) 5008 { 5009 /* case 3 */ 5010 /* READ DMA EXT */ 5011 SM_DBG5(("smsatRead10: case 3\n")); 5012 fis->h.fisType = 0x27; /* Reg host to device */ 5013 5014 fis->h.c_pmPort = 0x80; /* C Bit is set */ 5015 fis->h.command = SAT_READ_DMA_EXT; /* 0x25 */ 5016 fis->h.features = 0; /* FIS reserve */ 5017 fis->d.lbaLow = scsiCmnd->cdb[5]; /* FIS LBA (7 :0 ) */ 5018 fis->d.lbaMid = scsiCmnd->cdb[4]; /* FIS LBA (15:8 ) */ 5019 fis->d.lbaHigh = scsiCmnd->cdb[3]; /* FIS LBA (23:16) */ 5020 fis->d.device = 0x40; /* FIS LBA mode set */ 5021 fis->d.lbaLowExp = scsiCmnd->cdb[2]; /* FIS LBA (31:24) */ 5022 fis->d.lbaMidExp = 0; /* FIS LBA (39:32) */ 5023 fis->d.lbaHighExp = 0; /* FIS LBA (47:40) */ 5024 fis->d.featuresExp = 0; /* FIS reserve */ 5025 fis->d.sectorCount = scsiCmnd->cdb[8]; /* FIS sector count (7:0) */ 5026 fis->d.sectorCountExp = scsiCmnd->cdb[7]; /* FIS sector count (15:8) */ 5027 fis->d.reserved4 = 0; 5028 fis->d.control = 0; /* FIS HOB bit clear */ 5029 fis->d.reserved5 = 0; 5030 5031 agRequestType = AGSA_SATA_PROTOCOL_DMA_READ; 5032 satIOContext->ATACmd = SAT_READ_DMA_EXT; 5033 5034 } 5035 else 5036 { 5037 /* case 4 */ 5038 /* READ MULTIPLE EXT or READ SECTOR(S) EXT or READ VERIFY SECTOR(S) EXT*/ 5039 /* READ SECTORS EXT for easier implemetation */ 5040 SM_DBG5(("smsatRead10: case 4\n")); 5041 fis->h.fisType = 0x27; /* Reg host to device */ 5042 fis->h.c_pmPort = 0x80; /* C Bit is set */ 5043 5044 /* Check FUA bit */ 5045 if (scsiCmnd->cdb[1] & SCSI_READ10_FUA_MASK) 5046 { 5047 5048 /* for now, no support for FUA */ 5049 smsatSetSensePayload( pSense, 5050 SCSI_SNSKEY_ILLEGAL_REQUEST, 5051 0, 5052 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 5053 satIOContext); 5054 5055 /*smEnqueueIO(smRoot, satIOContext);*/ 5056 5057 tdsmIOCompletedCB( smRoot, 5058 smIORequest, 5059 smIOSuccess, 5060 SCSI_STAT_CHECK_CONDITION, 5061 satIOContext->pSmSenseData, 5062 satIOContext->interruptContext ); 5063 return SM_RC_SUCCESS; 5064 } 5065 5066 fis->h.command = SAT_READ_SECTORS_EXT; /* 0x24 */ 5067 5068 fis->h.features = 0; /* FIS reserve */ 5069 fis->d.lbaLow = scsiCmnd->cdb[5]; /* FIS LBA (7 :0 ) */ 5070 fis->d.lbaMid = scsiCmnd->cdb[4]; /* FIS LBA (15:8 ) */ 5071 fis->d.lbaHigh = scsiCmnd->cdb[3]; /* FIS LBA (23:16) */ 5072 fis->d.device = 0x40; /* FIS LBA mode set */ 5073 fis->d.lbaLowExp = scsiCmnd->cdb[2]; /* FIS LBA (31:24) */ 5074 fis->d.lbaMidExp = 0; /* FIS LBA (39:32) */ 5075 fis->d.lbaHighExp = 0; /* FIS LBA (47:40) */ 5076 fis->d.featuresExp = 0; /* FIS reserve */ 5077 fis->d.sectorCount = scsiCmnd->cdb[8]; /* FIS sector count (7:0) */ 5078 fis->d.sectorCountExp = scsiCmnd->cdb[7]; /* FIS sector count (15:8) */ 5079 fis->d.reserved4 = 0; 5080 fis->d.control = 0; /* FIS HOB bit clear */ 5081 fis->d.reserved5 = 0; 5082 5083 agRequestType = AGSA_SATA_PROTOCOL_PIO_READ; 5084 satIOContext->ATACmd = SAT_READ_SECTORS_EXT; 5085 } 5086 } 5087 else/* case 1 and 2 */ 5088 { 5089 if (pSatDevData->satDMASupport == agTRUE && pSatDevData->satDMAEnabled == agTRUE) 5090 { 5091 /* case 2 */ 5092 /* READ DMA*/ 5093 /* in case that we can't fit the transfer length, we need to make it fit by sending multiple ATA cmnds */ 5094 SM_DBG5(("smsatRead10: case 2\n")); 5095 5096 5097 fis->h.fisType = 0x27; /* Reg host to device */ 5098 fis->h.c_pmPort = 0x80; /* C Bit is set */ 5099 fis->h.command = SAT_READ_DMA; /* 0xC8 */ 5100 fis->h.features = 0; /* FIS reserve */ 5101 fis->d.lbaLow = scsiCmnd->cdb[5]; /* FIS LBA (7 :0 ) */ 5102 fis->d.lbaMid = scsiCmnd->cdb[4]; /* FIS LBA (15:8 ) */ 5103 fis->d.lbaHigh = scsiCmnd->cdb[3]; /* FIS LBA (23:16) */ 5104 fis->d.device = 5105 (bit8)((0x4 << 4) | (scsiCmnd->cdb[2] & 0xF)); /* FIS LBA (27:24) and FIS LBA mode */ 5106 fis->d.lbaLowExp = 0; 5107 fis->d.lbaMidExp = 0; 5108 fis->d.lbaHighExp = 0; 5109 fis->d.featuresExp = 0; 5110 fis->d.sectorCount = scsiCmnd->cdb[8]; /* FIS sector count (7:0) */ 5111 fis->d.sectorCountExp = 0; 5112 fis->d.reserved4 = 0; 5113 fis->d.control = 0; /* FIS HOB bit clear */ 5114 fis->d.reserved5 = 0; 5115 5116 5117 agRequestType = AGSA_SATA_PROTOCOL_DMA_READ; 5118 satIOContext->ATACmd = SAT_READ_DMA; 5119 } 5120 else 5121 { 5122 /* case 1 */ 5123 /* READ MULTIPLE or READ SECTOR(S) */ 5124 /* READ SECTORS for easier implemetation */ 5125 /* in case that we can't fit the transfer length, we need to make it fit by sending multiple ATA cmnds */ 5126 SM_DBG5(("smsatRead10: case 1\n")); 5127 5128 fis->h.fisType = 0x27; /* Reg host to device */ 5129 fis->h.c_pmPort = 0x80; /* C Bit is set */ 5130 fis->h.command = SAT_READ_SECTORS; /* 0x20 */ 5131 fis->h.features = 0; /* FIS reserve */ 5132 fis->d.lbaLow = scsiCmnd->cdb[5]; /* FIS LBA (7 :0 ) */ 5133 fis->d.lbaMid = scsiCmnd->cdb[4]; /* FIS LBA (15:8 ) */ 5134 fis->d.lbaHigh = scsiCmnd->cdb[3]; /* FIS LBA (23:16) */ 5135 fis->d.device = 5136 (bit8)((0x4 << 4) | (scsiCmnd->cdb[2] & 0xF)); /* FIS LBA (27:24) and FIS LBA mode */ 5137 fis->d.lbaLowExp = 0; 5138 fis->d.lbaMidExp = 0; 5139 fis->d.lbaHighExp = 0; 5140 fis->d.featuresExp = 0; 5141 fis->d.sectorCount = scsiCmnd->cdb[8]; /* FIS sector count (7:0) */ 5142 fis->d.sectorCountExp = 0; 5143 fis->d.reserved4 = 0; 5144 fis->d.control = 0; /* FIS HOB bit clear */ 5145 fis->d.reserved5 = 0; 5146 5147 5148 agRequestType = AGSA_SATA_PROTOCOL_PIO_READ; 5149 satIOContext->ATACmd = SAT_READ_SECTORS; 5150 } 5151 } 5152 // smhexdump("satRead10 final fis", (bit8 *)fis, sizeof(agsaFisRegHostToDevice_t)); 5153 5154 /* saves the current LBA and orginal TL */ 5155 satIOContext->currentLBA = lba; 5156 satIOContext->OrgTL = tl; 5157 5158 /* 5159 computing number of loop and remainder for tl 5160 0xFF in case not ext 5161 0xFFFF in case EXT 5162 */ 5163 if (fis->h.command == SAT_READ_SECTORS || fis->h.command == SAT_READ_DMA) 5164 { 5165 LoopNum = smsatComputeLoopNum(tl, 0x100); 5166 } 5167 else 5168 { 5169 /* SAT_READ_FPDMA_QUEUED */ 5170 /* SAT_READ_SECTORS_EXT, SAT_READ_DMA_EXT */ 5171 LoopNum = smsatComputeLoopNum(tl, 0xFFFF); 5172 } 5173 5174 satIOContext->LoopNum = LoopNum; 5175 5176 /* Initialize CB for SATA completion. 5177 */ 5178 if (LoopNum == 1) 5179 { 5180 SM_DBG5(("smsatRead10: NON CHAINED data\n")); 5181 satIOContext->satCompleteCB = &smsatNonChainedDataIOCB; 5182 } 5183 else 5184 { 5185 SM_DBG2(("smsatRead10: CHAINED data!!!\n")); 5186 5187 /* re-setting tl */ 5188 if (fis->h.command == SAT_READ_SECTORS || fis->h.command == SAT_READ_DMA) 5189 { 5190 fis->d.sectorCount = 0x0; 5191 smsatSplitSGL(smRoot, 5192 smIORequest, 5193 smDeviceHandle, 5194 smScsiRequest, 5195 satIOContext, 5196 NON_BIT48_ADDRESS_TL_LIMIT*SATA_SECTOR_SIZE, /* 0x100 * 0x200 */ 5197 (satIOContext->OrgTL)*SATA_SECTOR_SIZE, 5198 agTRUE); 5199 } 5200 else if (fis->h.command == SAT_READ_SECTORS_EXT || fis->h.command == SAT_READ_DMA_EXT) 5201 { 5202 /* SAT_READ_SECTORS_EXT, SAT_READ_DMA_EXT */ 5203 fis->d.sectorCount = 0xFF; 5204 fis->d.sectorCountExp = 0xFF; 5205 smsatSplitSGL(smRoot, 5206 smIORequest, 5207 smDeviceHandle, 5208 smScsiRequest, 5209 satIOContext, 5210 BIT48_ADDRESS_TL_LIMIT*SATA_SECTOR_SIZE, /* 0xFFFF * 0x200 */ 5211 (satIOContext->OrgTL)*SATA_SECTOR_SIZE, 5212 agTRUE); 5213 } 5214 else 5215 { 5216 /* SAT_READ_FPDMA_QUEUED */ 5217 fis->h.features = 0xFF; 5218 fis->d.featuresExp = 0xFF; 5219 smsatSplitSGL(smRoot, 5220 smIORequest, 5221 smDeviceHandle, 5222 smScsiRequest, 5223 satIOContext, 5224 BIT48_ADDRESS_TL_LIMIT*SATA_SECTOR_SIZE, /* 0xFFFF * 0x200 */ 5225 (satIOContext->OrgTL)*SATA_SECTOR_SIZE, 5226 agTRUE); 5227 } 5228 5229 /* chained data */ 5230 satIOContext->satCompleteCB = &smsatChainedDataIOCB; 5231 5232 } 5233 5234 /* 5235 * Prepare SGL and send FIS to LL layer. 5236 */ 5237 satIOContext->reqType = agRequestType; /* Save it */ 5238 5239 status = smsataLLIOStart( smRoot, 5240 smIORequest, 5241 smDeviceHandle, 5242 smScsiRequest, 5243 satIOContext); 5244 5245 SM_DBG5(("smsatRead10: return\n")); 5246 return (status); 5247 5248 } 5249 5250 osGLOBAL bit32 5251 smsatRead12( 5252 smRoot_t *smRoot, 5253 smIORequest_t *smIORequest, 5254 smDeviceHandle_t *smDeviceHandle, 5255 smScsiInitiatorRequest_t *smScsiRequest, 5256 smSatIOContext_t *satIOContext 5257 ) 5258 { 5259 bit32 status; 5260 bit32 agRequestType = AGSA_SATA_PROTOCOL_DMA_READ; 5261 smDeviceData_t *pSatDevData; 5262 smScsiRspSense_t *pSense; 5263 smIniScsiCmnd_t *scsiCmnd; 5264 agsaFisRegHostToDevice_t *fis; 5265 bit32 lba = 0; 5266 bit32 tl = 0; 5267 bit32 LoopNum = 1; 5268 bit8 LBA[8]; 5269 bit8 TL[8]; 5270 bit32 AllChk = agFALSE; /* lba, lba+tl check against ATA limit and Disk capacity */ 5271 5272 pSense = satIOContext->pSense; 5273 pSatDevData = satIOContext->pSatDevData; 5274 scsiCmnd = &smScsiRequest->scsiCmnd; 5275 fis = satIOContext->pFis; 5276 5277 SM_DBG5(("smsatRead12: start\n")); 5278 5279 /* checking FUA_NV */ 5280 if (scsiCmnd->cdb[1] & SCSI_FUA_NV_MASK) 5281 { 5282 smsatSetSensePayload( pSense, 5283 SCSI_SNSKEY_ILLEGAL_REQUEST, 5284 0, 5285 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 5286 satIOContext); 5287 5288 /*smEnqueueIO(smRoot, satIOContext);*/ 5289 5290 tdsmIOCompletedCB( smRoot, 5291 smIORequest, 5292 smIOSuccess, 5293 SCSI_STAT_CHECK_CONDITION, 5294 satIOContext->pSmSenseData, 5295 satIOContext->interruptContext ); 5296 5297 SM_DBG1(("smsatRead12: return FUA_NV!!!\n")); 5298 return SM_RC_SUCCESS; 5299 5300 } 5301 5302 /* checking CONTROL */ 5303 /* NACA == 1 or LINK == 1*/ 5304 if ( (scsiCmnd->cdb[11] & SCSI_NACA_MASK) || (scsiCmnd->cdb[11] & SCSI_LINK_MASK) ) 5305 { 5306 smsatSetSensePayload( pSense, 5307 SCSI_SNSKEY_ILLEGAL_REQUEST, 5308 0, 5309 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 5310 satIOContext); 5311 5312 /*smEnqueueIO(smRoot, satIOContext);*/ 5313 5314 tdsmIOCompletedCB( smRoot, 5315 smIORequest, 5316 smIOSuccess, 5317 SCSI_STAT_CHECK_CONDITION, 5318 satIOContext->pSmSenseData, 5319 satIOContext->interruptContext ); 5320 5321 SM_DBG1(("smsatRead12: return control!!!\n")); 5322 return SM_RC_SUCCESS; 5323 } 5324 5325 sm_memset(LBA, 0, sizeof(LBA)); 5326 sm_memset(TL, 0, sizeof(TL)); 5327 5328 /* do not use memcpy due to indexing in LBA and TL */ 5329 LBA[0] = 0; /* MSB */ 5330 LBA[1] = 0; 5331 LBA[2] = 0; 5332 LBA[3] = 0; 5333 LBA[4] = scsiCmnd->cdb[2]; 5334 LBA[5] = scsiCmnd->cdb[3]; 5335 LBA[6] = scsiCmnd->cdb[4]; 5336 LBA[7] = scsiCmnd->cdb[5]; /* LSB */ 5337 5338 TL[0] = 0; /* MSB */ 5339 TL[1] = 0; 5340 TL[2] = 0; 5341 TL[3] = 0; 5342 TL[4] = scsiCmnd->cdb[6]; 5343 TL[5] = scsiCmnd->cdb[7]; 5344 TL[6] = scsiCmnd->cdb[8]; 5345 TL[7] = scsiCmnd->cdb[9]; /* LSB */ 5346 5347 5348 lba = smsatComputeCDB12LBA(satIOContext); 5349 tl = smsatComputeCDB12TL(satIOContext); 5350 5351 /* Table 34, 9.1, p 46 */ 5352 /* 5353 note: As of 2/10/2006, no support for DMA QUEUED 5354 */ 5355 5356 /* 5357 Table 34, 9.1, p 46, b 5358 When no 48-bit addressing support or NCQ, if LBA is beyond (2^28 - 1), 5359 return check condition 5360 */ 5361 if (pSatDevData->satNCQ != agTRUE && 5362 pSatDevData->sat48BitSupport != agTRUE 5363 ) 5364 { 5365 5366 AllChk = smsatCheckLimit(LBA, TL, agFALSE, pSatDevData); 5367 if (AllChk) 5368 { 5369 SM_DBG1(("smsatRead12: return LBA out of range, not EXT!!!\n")); 5370 smsatSetSensePayload( pSense, 5371 SCSI_SNSKEY_ILLEGAL_REQUEST, 5372 0, 5373 SCSI_SNSCODE_LOGICAL_BLOCK_ADDRESS_OUT_OF_RANGE, 5374 satIOContext); 5375 5376 /*smEnqueueIO(smRoot, satIOContext);*/ 5377 5378 tdsmIOCompletedCB( smRoot, 5379 smIORequest, 5380 smIOSuccess, 5381 SCSI_STAT_CHECK_CONDITION, 5382 satIOContext->pSmSenseData, 5383 satIOContext->interruptContext ); 5384 5385 return SM_RC_SUCCESS; 5386 } 5387 } 5388 else 5389 { 5390 AllChk = smsatCheckLimit(LBA, TL, agTRUE, pSatDevData); 5391 if (AllChk) 5392 { 5393 SM_DBG1(("smsatRead12: return LBA out of range, EXT!!!\n")); 5394 smsatSetSensePayload( pSense, 5395 SCSI_SNSKEY_ILLEGAL_REQUEST, 5396 0, 5397 SCSI_SNSCODE_LOGICAL_BLOCK_ADDRESS_OUT_OF_RANGE, 5398 satIOContext); 5399 5400 /*smEnqueueIO(smRoot, satIOContext);*/ 5401 5402 tdsmIOCompletedCB( smRoot, 5403 smIORequest, 5404 smIOSuccess, 5405 SCSI_STAT_CHECK_CONDITION, 5406 satIOContext->pSmSenseData, 5407 satIOContext->interruptContext ); 5408 5409 return SM_RC_SUCCESS; 5410 } 5411 } 5412 5413 /* case 1 and 2 */ 5414 if (pSatDevData->satDMASupport == agTRUE && pSatDevData->satDMAEnabled == agTRUE) 5415 { 5416 /* case 2 */ 5417 /* READ DMA*/ 5418 /* in case that we can't fit the transfer length, 5419 we need to make it fit by sending multiple ATA cmnds */ 5420 SM_DBG5(("smsatRead12: case 2\n")); 5421 5422 5423 fis->h.fisType = 0x27; /* Reg host to device */ 5424 fis->h.c_pmPort = 0x80; /* C Bit is set */ 5425 fis->h.command = SAT_READ_DMA; /* 0xC8 */ 5426 fis->h.features = 0; /* FIS reserve */ 5427 fis->d.lbaLow = scsiCmnd->cdb[5]; /* FIS LBA (7 :0 ) */ 5428 fis->d.lbaMid = scsiCmnd->cdb[4]; /* FIS LBA (15:8 ) */ 5429 fis->d.lbaHigh = scsiCmnd->cdb[3]; /* FIS LBA (23:16) */ 5430 fis->d.device = 5431 (bit8)((0x4 << 4) | (scsiCmnd->cdb[2] & 0xF)); /* FIS LBA (27:24) and FIS LBA mode */ 5432 fis->d.lbaLowExp = 0; 5433 fis->d.lbaMidExp = 0; 5434 fis->d.lbaHighExp = 0; 5435 fis->d.featuresExp = 0; 5436 fis->d.sectorCount = scsiCmnd->cdb[9]; /* FIS sector count (7:0) */ 5437 fis->d.sectorCountExp = 0; 5438 fis->d.reserved4 = 0; 5439 fis->d.control = 0; /* FIS HOB bit clear */ 5440 fis->d.reserved5 = 0; 5441 5442 5443 agRequestType = AGSA_SATA_PROTOCOL_DMA_READ; 5444 satIOContext->ATACmd = SAT_READ_DMA; 5445 } 5446 else 5447 { 5448 /* case 1 */ 5449 /* READ MULTIPLE or READ SECTOR(S) */ 5450 /* READ SECTORS for easier implemetation */ 5451 /* can't fit the transfer length but need to make it fit by sending multiple*/ 5452 SM_DBG5(("smsatRead12: case 1\n")); 5453 5454 fis->h.fisType = 0x27; /* Reg host to device */ 5455 fis->h.c_pmPort = 0x80; /* C Bit is set */ 5456 fis->h.command = SAT_READ_SECTORS; /* 0x20 */ 5457 fis->h.features = 0; /* FIS reserve */ 5458 fis->d.lbaLow = scsiCmnd->cdb[5]; /* FIS LBA (7 :0 ) */ 5459 fis->d.lbaMid = scsiCmnd->cdb[4]; /* FIS LBA (15:8 ) */ 5460 fis->d.lbaHigh = scsiCmnd->cdb[3]; /* FIS LBA (23:16) */ 5461 fis->d.device = 5462 (bit8)((0x4 << 4) | (scsiCmnd->cdb[2] & 0xF)); /* FIS LBA (27:24) and FIS LBA mode */ 5463 fis->d.lbaLowExp = 0; 5464 fis->d.lbaMidExp = 0; 5465 fis->d.lbaHighExp = 0; 5466 fis->d.featuresExp = 0; 5467 fis->d.sectorCount = scsiCmnd->cdb[9]; /* FIS sector count (7:0) */ 5468 fis->d.sectorCountExp = 0; 5469 fis->d.reserved4 = 0; 5470 fis->d.control = 0; /* FIS HOB bit clear */ 5471 fis->d.reserved5 = 0; 5472 5473 5474 agRequestType = AGSA_SATA_PROTOCOL_PIO_READ; 5475 satIOContext->ATACmd = SAT_READ_SECTORS; 5476 } 5477 5478 /* case 3 and 4 */ 5479 if (pSatDevData->sat48BitSupport == agTRUE) 5480 { 5481 if (pSatDevData->satDMASupport == agTRUE && pSatDevData->satDMAEnabled == agTRUE) 5482 { 5483 /* case 3 */ 5484 /* READ DMA EXT */ 5485 SM_DBG5(("smsatRead12: case 3\n")); 5486 fis->h.fisType = 0x27; /* Reg host to device */ 5487 5488 fis->h.c_pmPort = 0x80; /* C Bit is set */ 5489 fis->h.command = SAT_READ_DMA_EXT; /* 0x25 */ 5490 fis->h.features = 0; /* FIS reserve */ 5491 fis->d.lbaLow = scsiCmnd->cdb[5]; /* FIS LBA (7 :0 ) */ 5492 fis->d.lbaMid = scsiCmnd->cdb[4]; /* FIS LBA (15:8 ) */ 5493 fis->d.lbaHigh = scsiCmnd->cdb[3]; /* FIS LBA (23:16) */ 5494 fis->d.device = 0x40; /* FIS LBA mode set */ 5495 fis->d.lbaLowExp = scsiCmnd->cdb[2]; /* FIS LBA (31:24) */ 5496 fis->d.lbaMidExp = 0; /* FIS LBA (39:32) */ 5497 fis->d.lbaHighExp = 0; /* FIS LBA (47:40) */ 5498 fis->d.featuresExp = 0; /* FIS reserve */ 5499 fis->d.sectorCount = scsiCmnd->cdb[9]; /* FIS sector count (7:0) */ 5500 fis->d.sectorCountExp = scsiCmnd->cdb[8]; /* FIS sector count (15:8) */ 5501 fis->d.reserved4 = 0; 5502 fis->d.control = 0; /* FIS HOB bit clear */ 5503 fis->d.reserved5 = 0; 5504 5505 agRequestType = AGSA_SATA_PROTOCOL_DMA_READ; 5506 satIOContext->ATACmd = SAT_READ_DMA_EXT; 5507 5508 } 5509 else 5510 { 5511 /* case 4 */ 5512 /* READ MULTIPLE EXT or READ SECTOR(S) EXT or READ VERIFY SECTOR(S) EXT*/ 5513 /* READ SECTORS EXT for easier implemetation */ 5514 SM_DBG5(("smsatRead12: case 4\n")); 5515 fis->h.fisType = 0x27; /* Reg host to device */ 5516 fis->h.c_pmPort = 0x80; /* C Bit is set */ 5517 5518 /* Check FUA bit */ 5519 if (scsiCmnd->cdb[1] & SCSI_READ12_FUA_MASK) 5520 { 5521 5522 /* for now, no support for FUA */ 5523 smsatSetSensePayload( pSense, 5524 SCSI_SNSKEY_ILLEGAL_REQUEST, 5525 0, 5526 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 5527 satIOContext); 5528 5529 /*smEnqueueIO(smRoot, satIOContext);*/ 5530 5531 tdsmIOCompletedCB( smRoot, 5532 smIORequest, 5533 smIOSuccess, 5534 SCSI_STAT_CHECK_CONDITION, 5535 satIOContext->pSmSenseData, 5536 satIOContext->interruptContext ); 5537 return SM_RC_SUCCESS; 5538 } 5539 5540 fis->h.command = SAT_READ_SECTORS_EXT; /* 0x24 */ 5541 5542 fis->h.features = 0; /* FIS reserve */ 5543 fis->d.lbaLow = scsiCmnd->cdb[5]; /* FIS LBA (7 :0 ) */ 5544 fis->d.lbaMid = scsiCmnd->cdb[4]; /* FIS LBA (15:8 ) */ 5545 fis->d.lbaHigh = scsiCmnd->cdb[3]; /* FIS LBA (23:16) */ 5546 fis->d.device = 0x40; /* FIS LBA mode set */ 5547 fis->d.lbaLowExp = scsiCmnd->cdb[2]; /* FIS LBA (31:24) */ 5548 fis->d.lbaMidExp = 0; /* FIS LBA (39:32) */ 5549 fis->d.lbaHighExp = 0; /* FIS LBA (47:40) */ 5550 fis->d.featuresExp = 0; /* FIS reserve */ 5551 fis->d.sectorCount = scsiCmnd->cdb[9]; /* FIS sector count (7:0) */ 5552 fis->d.sectorCountExp = scsiCmnd->cdb[8]; /* FIS sector count (15:8) */ 5553 fis->d.reserved4 = 0; 5554 fis->d.control = 0; /* FIS HOB bit clear */ 5555 fis->d.reserved5 = 0; 5556 5557 agRequestType = AGSA_SATA_PROTOCOL_PIO_READ; 5558 satIOContext->ATACmd = SAT_READ_SECTORS_EXT; 5559 } 5560 } 5561 5562 /* case 5 */ 5563 if (pSatDevData->satNCQ == agTRUE) 5564 { 5565 /* READ FPDMA QUEUED */ 5566 if (pSatDevData->sat48BitSupport != agTRUE) 5567 { 5568 SM_DBG1(("smsatRead12: case 5 !!! error NCQ but 28 bit address support!!!\n")); 5569 smsatSetSensePayload( pSense, 5570 SCSI_SNSKEY_ILLEGAL_REQUEST, 5571 0, 5572 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 5573 satIOContext); 5574 5575 /*smEnqueueIO(smRoot, satIOContext);*/ 5576 5577 tdsmIOCompletedCB( smRoot, 5578 smIORequest, 5579 smIOSuccess, 5580 SCSI_STAT_CHECK_CONDITION, 5581 satIOContext->pSmSenseData, 5582 satIOContext->interruptContext ); 5583 return SM_RC_SUCCESS; 5584 } 5585 5586 SM_DBG6(("smsatRead12: case 5\n")); 5587 5588 /* Support 48-bit FPDMA addressing, use READ FPDMA QUEUE command */ 5589 5590 fis->h.fisType = 0x27; /* Reg host to device */ 5591 fis->h.c_pmPort = 0x80; /* C Bit is set */ 5592 fis->h.command = SAT_READ_FPDMA_QUEUED; /* 0x60 */ 5593 fis->h.features = scsiCmnd->cdb[9]; /* FIS sector count (7:0) */ 5594 fis->d.lbaLow = scsiCmnd->cdb[5]; /* FIS LBA (7 :0 ) */ 5595 fis->d.lbaMid = scsiCmnd->cdb[4]; /* FIS LBA (15:8 ) */ 5596 fis->d.lbaHigh = scsiCmnd->cdb[3]; /* FIS LBA (23:16) */ 5597 5598 /* Check FUA bit */ 5599 if (scsiCmnd->cdb[1] & SCSI_READ12_FUA_MASK) 5600 fis->d.device = 0xC0; /* FIS FUA set */ 5601 else 5602 fis->d.device = 0x40; /* FIS FUA clear */ 5603 5604 fis->d.lbaLowExp = scsiCmnd->cdb[2]; /* FIS LBA (31:24) */ 5605 fis->d.lbaMidExp = 0; /* FIS LBA (39:32) */ 5606 fis->d.lbaHighExp = 0; /* FIS LBA (47:40) */ 5607 fis->d.featuresExp = scsiCmnd->cdb[8]; /* FIS sector count (15:8) */ 5608 fis->d.sectorCount = 0; /* Tag (7:3) set by LL layer */ 5609 fis->d.sectorCountExp = 0; 5610 fis->d.reserved4 = 0; 5611 fis->d.control = 0; /* FIS HOB bit clear */ 5612 fis->d.reserved5 = 0; 5613 5614 agRequestType = AGSA_SATA_PROTOCOL_FPDMA_READ; 5615 satIOContext->ATACmd = SAT_READ_FPDMA_QUEUED; 5616 } 5617 5618 /* saves the current LBA and orginal TL */ 5619 satIOContext->currentLBA = lba; 5620 satIOContext->OrgTL = tl; 5621 5622 /* 5623 computing number of loop and remainder for tl 5624 0xFF in case not ext 5625 0xFFFF in case EXT 5626 */ 5627 if (fis->h.command == SAT_READ_SECTORS || fis->h.command == SAT_READ_DMA) 5628 { 5629 LoopNum = smsatComputeLoopNum(tl, 0xFF); 5630 } 5631 else if (fis->h.command == SAT_READ_SECTORS_EXT || fis->h.command == SAT_READ_DMA_EXT) 5632 { 5633 /* SAT_READ_SECTORS_EXT, SAT_READ_DMA_EXT */ 5634 LoopNum = smsatComputeLoopNum(tl, 0xFFFF); 5635 } 5636 else 5637 { 5638 /* SAT_READ_FPDMA_QUEUEDK */ 5639 LoopNum = smsatComputeLoopNum(tl, 0xFFFF); 5640 } 5641 5642 satIOContext->LoopNum = LoopNum; 5643 5644 if (LoopNum == 1) 5645 { 5646 SM_DBG5(("smsatRead12: NON CHAINED data\n")); 5647 satIOContext->satCompleteCB = &smsatNonChainedDataIOCB; 5648 } 5649 else 5650 { 5651 SM_DBG1(("smsatRead12: CHAINED data\n")); 5652 /* re-setting tl */ 5653 if (fis->h.command == SAT_READ_SECTORS || fis->h.command == SAT_READ_DMA) 5654 { 5655 fis->d.sectorCount = 0xFF; 5656 } 5657 else if (fis->h.command == SAT_READ_SECTORS_EXT || fis->h.command == SAT_READ_DMA_EXT) 5658 { 5659 /* SAT_READ_SECTORS_EXT, SAT_READ_DMA_EXT */ 5660 fis->d.sectorCount = 0xFF; 5661 fis->d.sectorCountExp = 0xFF; 5662 } 5663 else 5664 { 5665 /* SAT_READ_FPDMA_QUEUED */ 5666 fis->h.features = 0xFF; 5667 fis->d.featuresExp = 0xFF; 5668 } 5669 5670 /* chained data */ 5671 satIOContext->satCompleteCB = &smsatChainedDataIOCB; 5672 } 5673 5674 /* 5675 * Prepare SGL and send FIS to LL layer. 5676 */ 5677 satIOContext->reqType = agRequestType; /* Save it */ 5678 5679 status = smsataLLIOStart( smRoot, 5680 smIORequest, 5681 smDeviceHandle, 5682 smScsiRequest, 5683 satIOContext); 5684 5685 SM_DBG5(("smsatRead12: return\n")); 5686 return (status); 5687 } 5688 5689 osGLOBAL bit32 5690 smsatRead16( 5691 smRoot_t *smRoot, 5692 smIORequest_t *smIORequest, 5693 smDeviceHandle_t *smDeviceHandle, 5694 smScsiInitiatorRequest_t *smScsiRequest, 5695 smSatIOContext_t *satIOContext 5696 ) 5697 { 5698 bit32 status; 5699 bit32 agRequestType = AGSA_SATA_PROTOCOL_DMA_READ; 5700 smDeviceData_t *pSatDevData; 5701 smScsiRspSense_t *pSense; 5702 smIniScsiCmnd_t *scsiCmnd; 5703 agsaFisRegHostToDevice_t *fis; 5704 bit32 lba = 0; 5705 bit32 tl = 0; 5706 bit32 LoopNum = 1; 5707 bit8 LBA[8]; 5708 bit8 TL[8]; 5709 bit32 AllChk = agFALSE; /* lba, lba+tl check against ATA limit and Disk capacity */ 5710 // bit32 limitExtChk = agFALSE; /* lba limit check for bit48 addressing check */ 5711 5712 pSense = satIOContext->pSense; 5713 pSatDevData = satIOContext->pSatDevData; 5714 scsiCmnd = &smScsiRequest->scsiCmnd; 5715 fis = satIOContext->pFis; 5716 5717 SM_DBG5(("smsatRead16: start\n")); 5718 5719 /* checking FUA_NV */ 5720 if (scsiCmnd->cdb[1] & SCSI_FUA_NV_MASK) 5721 { 5722 smsatSetSensePayload( pSense, 5723 SCSI_SNSKEY_ILLEGAL_REQUEST, 5724 0, 5725 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 5726 satIOContext); 5727 5728 /*smEnqueueIO(smRoot, satIOContext);*/ 5729 5730 tdsmIOCompletedCB( smRoot, 5731 smIORequest, 5732 smIOSuccess, 5733 SCSI_STAT_CHECK_CONDITION, 5734 satIOContext->pSmSenseData, 5735 satIOContext->interruptContext ); 5736 5737 SM_DBG1(("smsatRead16: return FUA_NV!!!\n")); 5738 return SM_RC_SUCCESS; 5739 5740 } 5741 5742 /* checking CONTROL */ 5743 /* NACA == 1 or LINK == 1*/ 5744 if ( (scsiCmnd->cdb[15] & SCSI_NACA_MASK) || (scsiCmnd->cdb[15] & SCSI_LINK_MASK) ) 5745 { 5746 smsatSetSensePayload( pSense, 5747 SCSI_SNSKEY_ILLEGAL_REQUEST, 5748 0, 5749 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 5750 satIOContext); 5751 5752 /*smEnqueueIO(smRoot, satIOContext);*/ 5753 5754 tdsmIOCompletedCB( smRoot, 5755 smIORequest, 5756 smIOSuccess, 5757 SCSI_STAT_CHECK_CONDITION, 5758 satIOContext->pSmSenseData, 5759 satIOContext->interruptContext ); 5760 5761 SM_DBG1(("smsatRead16: return control!!!\n")); 5762 return SM_RC_SUCCESS; 5763 } 5764 5765 5766 sm_memset(LBA, 0, sizeof(LBA)); 5767 sm_memset(TL, 0, sizeof(TL)); 5768 5769 5770 /* do not use memcpy due to indexing in LBA and TL */ 5771 LBA[0] = scsiCmnd->cdb[2]; /* MSB */ 5772 LBA[1] = scsiCmnd->cdb[3]; 5773 LBA[2] = scsiCmnd->cdb[4]; 5774 LBA[3] = scsiCmnd->cdb[5]; 5775 LBA[4] = scsiCmnd->cdb[6]; 5776 LBA[5] = scsiCmnd->cdb[7]; 5777 LBA[6] = scsiCmnd->cdb[8]; 5778 LBA[7] = scsiCmnd->cdb[9]; /* LSB */ 5779 5780 TL[0] = 0; 5781 TL[1] = 0; 5782 TL[2] = 0; 5783 TL[3] = 0; 5784 TL[4] = scsiCmnd->cdb[10]; /* MSB */ 5785 TL[5] = scsiCmnd->cdb[11]; 5786 TL[6] = scsiCmnd->cdb[12]; 5787 TL[7] = scsiCmnd->cdb[13]; /* LSB */ 5788 5789 5790 5791 5792 lba = smsatComputeCDB16LBA(satIOContext); 5793 tl = smsatComputeCDB16TL(satIOContext); 5794 5795 5796 /* Table 34, 9.1, p 46 */ 5797 /* 5798 note: As of 2/10/2006, no support for DMA QUEUED 5799 */ 5800 5801 /* 5802 Table 34, 9.1, p 46, b 5803 When no 48-bit addressing support or NCQ, if LBA is beyond (2^28 - 1), 5804 return check condition 5805 */ 5806 if (pSatDevData->satNCQ != agTRUE && 5807 pSatDevData->sat48BitSupport != agTRUE 5808 ) 5809 { 5810 AllChk = smsatCheckLimit(LBA, TL, agFALSE, pSatDevData); 5811 if (AllChk) 5812 { 5813 SM_DBG1(("smsatRead16: return LBA out of range, not EXT!!!\n")); 5814 5815 /*smEnqueueIO(smRoot, satIOContext);*/ 5816 5817 5818 smsatSetSensePayload( pSense, 5819 SCSI_SNSKEY_ILLEGAL_REQUEST, 5820 0, 5821 SCSI_SNSCODE_LOGICAL_BLOCK_ADDRESS_OUT_OF_RANGE, 5822 satIOContext); 5823 5824 /*smEnqueueIO(smRoot, satIOContext);*/ 5825 5826 tdsmIOCompletedCB( smRoot, 5827 smIORequest, 5828 smIOSuccess, 5829 SCSI_STAT_CHECK_CONDITION, 5830 satIOContext->pSmSenseData, 5831 satIOContext->interruptContext ); 5832 5833 return SM_RC_SUCCESS; 5834 } 5835 } 5836 else 5837 { 5838 // rangeChk = smsatAddNComparebit64(LBA, TL); 5839 5840 AllChk = smsatCheckLimit(LBA, TL, agTRUE, pSatDevData); 5841 5842 5843 if (AllChk) 5844 { 5845 SM_DBG1(("smsatRead16: return LBA out of range, EXT!!!\n")); 5846 smsatSetSensePayload( pSense, 5847 SCSI_SNSKEY_ILLEGAL_REQUEST, 5848 0, 5849 SCSI_SNSCODE_LOGICAL_BLOCK_ADDRESS_OUT_OF_RANGE, 5850 satIOContext); 5851 5852 /*smEnqueueIO(smRoot, satIOContext);*/ 5853 5854 tdsmIOCompletedCB( smRoot, 5855 smIORequest, 5856 smIOSuccess, 5857 SCSI_STAT_CHECK_CONDITION, 5858 satIOContext->pSmSenseData, 5859 satIOContext->interruptContext ); 5860 5861 return SM_RC_SUCCESS; 5862 } 5863 } 5864 5865 /* case 1 and 2 */ 5866 if (pSatDevData->satDMASupport == agTRUE && pSatDevData->satDMAEnabled == agTRUE) 5867 { 5868 /* case 2 */ 5869 /* READ DMA*/ 5870 /* in case that we can't fit the transfer length, 5871 we need to make it fit by sending multiple ATA cmnds */ 5872 SM_DBG5(("smsatRead16: case 2\n")); 5873 5874 5875 fis->h.fisType = 0x27; /* Reg host to device */ 5876 fis->h.c_pmPort = 0x80; /* C Bit is set */ 5877 fis->h.command = SAT_READ_DMA; /* 0xC8 */ 5878 fis->h.features = 0; /* FIS reserve */ 5879 fis->d.lbaLow = scsiCmnd->cdb[9]; /* FIS LBA (7 :0 ) */ 5880 fis->d.lbaMid = scsiCmnd->cdb[8]; /* FIS LBA (15:8 ) */ 5881 fis->d.lbaHigh = scsiCmnd->cdb[7]; /* FIS LBA (23:16) */ 5882 fis->d.device = 5883 (bit8)((0x4 << 4) | (scsiCmnd->cdb[6] & 0xF)); /* FIS LBA (27:24) and FIS LBA mode */ 5884 fis->d.lbaLowExp = 0; 5885 fis->d.lbaMidExp = 0; 5886 fis->d.lbaHighExp = 0; 5887 fis->d.featuresExp = 0; 5888 fis->d.sectorCount = scsiCmnd->cdb[13]; /* FIS sector count (7:0) */ 5889 fis->d.sectorCountExp = 0; 5890 fis->d.reserved4 = 0; 5891 fis->d.control = 0; /* FIS HOB bit clear */ 5892 fis->d.reserved5 = 0; 5893 5894 5895 agRequestType = AGSA_SATA_PROTOCOL_DMA_READ; 5896 satIOContext->ATACmd = SAT_READ_DMA; 5897 } 5898 else 5899 { 5900 /* case 1 */ 5901 /* READ MULTIPLE or READ SECTOR(S) */ 5902 /* READ SECTORS for easier implemetation */ 5903 /* can't fit the transfer length but need to make it fit by sending multiple*/ 5904 SM_DBG5(("smsatRead16: case 1\n")); 5905 5906 fis->h.fisType = 0x27; /* Reg host to device */ 5907 fis->h.c_pmPort = 0x80; /* C Bit is set */ 5908 fis->h.command = SAT_READ_SECTORS; /* 0x20 */ 5909 fis->h.features = 0; /* FIS reserve */ 5910 fis->d.lbaLow = scsiCmnd->cdb[9]; /* FIS LBA (7 :0 ) */ 5911 fis->d.lbaMid = scsiCmnd->cdb[8]; /* FIS LBA (15:8 ) */ 5912 fis->d.lbaHigh = scsiCmnd->cdb[7]; /* FIS LBA (23:16) */ 5913 fis->d.device = 5914 (bit8)((0x4 << 4) | (scsiCmnd->cdb[6] & 0xF)); /* FIS LBA (27:24) and FIS LBA mode */ 5915 fis->d.lbaLowExp = 0; 5916 fis->d.lbaMidExp = 0; 5917 fis->d.lbaHighExp = 0; 5918 fis->d.featuresExp = 0; 5919 fis->d.sectorCount = scsiCmnd->cdb[13]; /* FIS sector count (7:0) */ 5920 fis->d.sectorCountExp = 0; 5921 fis->d.reserved4 = 0; 5922 fis->d.control = 0; /* FIS HOB bit clear */ 5923 fis->d.reserved5 = 0; 5924 5925 5926 agRequestType = AGSA_SATA_PROTOCOL_PIO_READ; 5927 satIOContext->ATACmd = SAT_READ_SECTORS; 5928 } 5929 5930 /* case 3 and 4 */ 5931 if (pSatDevData->sat48BitSupport == agTRUE) 5932 { 5933 if (pSatDevData->satDMASupport == agTRUE && pSatDevData->satDMAEnabled == agTRUE) 5934 { 5935 /* case 3 */ 5936 /* READ DMA EXT */ 5937 SM_DBG5(("smsatRead16: case 3\n")); 5938 fis->h.fisType = 0x27; /* Reg host to device */ 5939 5940 fis->h.c_pmPort = 0x80; /* C Bit is set */ 5941 fis->h.command = SAT_READ_DMA_EXT; /* 0x25 */ 5942 fis->h.features = 0; /* FIS reserve */ 5943 fis->d.lbaLow = scsiCmnd->cdb[9]; /* FIS LBA (7 :0 ) */ 5944 fis->d.lbaMid = scsiCmnd->cdb[8]; /* FIS LBA (15:8 ) */ 5945 fis->d.lbaHigh = scsiCmnd->cdb[7]; /* FIS LBA (23:16) */ 5946 fis->d.device = 0x40; /* FIS LBA mode set */ 5947 fis->d.lbaLowExp = scsiCmnd->cdb[6]; /* FIS LBA (31:24) */ 5948 fis->d.lbaMidExp = scsiCmnd->cdb[5]; /* FIS LBA (39:32) */ 5949 fis->d.lbaHighExp = scsiCmnd->cdb[4]; /* FIS LBA (47:40) */ 5950 fis->d.featuresExp = 0; /* FIS reserve */ 5951 fis->d.sectorCount = scsiCmnd->cdb[13]; /* FIS sector count (7:0) */ 5952 fis->d.sectorCountExp = scsiCmnd->cdb[12]; /* FIS sector count (15:8) */ 5953 fis->d.reserved4 = 0; 5954 fis->d.control = 0; /* FIS HOB bit clear */ 5955 fis->d.reserved5 = 0; 5956 5957 agRequestType = AGSA_SATA_PROTOCOL_DMA_READ; 5958 satIOContext->ATACmd = SAT_READ_DMA_EXT; 5959 5960 } 5961 else 5962 { 5963 /* case 4 */ 5964 /* READ MULTIPLE EXT or READ SECTOR(S) EXT or READ VERIFY SECTOR(S) EXT*/ 5965 /* READ SECTORS EXT for easier implemetation */ 5966 SM_DBG5(("smsatRead16: case 4\n")); 5967 fis->h.fisType = 0x27; /* Reg host to device */ 5968 fis->h.c_pmPort = 0x80; /* C Bit is set */ 5969 5970 /* Check FUA bit */ 5971 if (scsiCmnd->cdb[1] & SCSI_READ16_FUA_MASK) 5972 { 5973 /* for now, no support for FUA */ 5974 smsatSetSensePayload( pSense, 5975 SCSI_SNSKEY_ILLEGAL_REQUEST, 5976 0, 5977 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 5978 satIOContext); 5979 5980 /*smEnqueueIO(smRoot, satIOContext);*/ 5981 5982 tdsmIOCompletedCB( smRoot, 5983 smIORequest, 5984 smIOSuccess, 5985 SCSI_STAT_CHECK_CONDITION, 5986 satIOContext->pSmSenseData, 5987 satIOContext->interruptContext ); 5988 return SM_RC_SUCCESS; 5989 } 5990 5991 fis->h.command = SAT_READ_SECTORS_EXT; /* 0x24 */ 5992 5993 fis->h.features = 0; /* FIS reserve */ 5994 fis->d.lbaLow = scsiCmnd->cdb[9]; /* FIS LBA (7 :0 ) */ 5995 fis->d.lbaMid = scsiCmnd->cdb[8]; /* FIS LBA (15:8 ) */ 5996 fis->d.lbaHigh = scsiCmnd->cdb[7]; /* FIS LBA (23:16) */ 5997 fis->d.device = 0x40; /* FIS LBA mode set */ 5998 fis->d.lbaLowExp = scsiCmnd->cdb[6]; /* FIS LBA (31:24) */ 5999 fis->d.lbaMidExp = scsiCmnd->cdb[5]; /* FIS LBA (39:32) */ 6000 fis->d.lbaHighExp = scsiCmnd->cdb[4]; /* FIS LBA (47:40) */ 6001 fis->d.featuresExp = 0; /* FIS reserve */ 6002 fis->d.sectorCount = scsiCmnd->cdb[13]; /* FIS sector count (7:0) */ 6003 fis->d.sectorCountExp = scsiCmnd->cdb[12]; /* FIS sector count (15:8) */ 6004 fis->d.reserved4 = 0; 6005 fis->d.control = 0; /* FIS HOB bit clear */ 6006 fis->d.reserved5 = 0; 6007 6008 agRequestType = AGSA_SATA_PROTOCOL_PIO_READ; 6009 satIOContext->ATACmd = SAT_READ_SECTORS_EXT; 6010 } 6011 } 6012 6013 6014 /* case 5 */ 6015 if (pSatDevData->satNCQ == agTRUE) 6016 { 6017 /* READ FPDMA QUEUED */ 6018 if (pSatDevData->sat48BitSupport != agTRUE) 6019 { 6020 SM_DBG1(("smsatRead16: case 5 !!! error NCQ but 28 bit address support!!!\n")); 6021 smsatSetSensePayload( pSense, 6022 SCSI_SNSKEY_ILLEGAL_REQUEST, 6023 0, 6024 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 6025 satIOContext); 6026 6027 /*smEnqueueIO(smRoot, satIOContext);*/ 6028 6029 tdsmIOCompletedCB( smRoot, 6030 smIORequest, 6031 smIOSuccess, 6032 SCSI_STAT_CHECK_CONDITION, 6033 satIOContext->pSmSenseData, 6034 satIOContext->interruptContext ); 6035 return SM_RC_SUCCESS; 6036 } 6037 6038 SM_DBG6(("smsatRead16: case 5\n")); 6039 6040 /* Support 48-bit FPDMA addressing, use READ FPDMA QUEUE command */ 6041 6042 fis->h.fisType = 0x27; /* Reg host to device */ 6043 fis->h.c_pmPort = 0x80; /* C Bit is set */ 6044 fis->h.command = SAT_READ_FPDMA_QUEUED; /* 0x60 */ 6045 fis->h.features = scsiCmnd->cdb[13]; /* FIS sector count (7:0) */ 6046 fis->d.lbaLow = scsiCmnd->cdb[9]; /* FIS LBA (7 :0 ) */ 6047 fis->d.lbaMid = scsiCmnd->cdb[8]; /* FIS LBA (15:8 ) */ 6048 fis->d.lbaHigh = scsiCmnd->cdb[7]; /* FIS LBA (23:16) */ 6049 6050 /* Check FUA bit */ 6051 if (scsiCmnd->cdb[1] & SCSI_READ16_FUA_MASK) 6052 fis->d.device = 0xC0; /* FIS FUA set */ 6053 else 6054 fis->d.device = 0x40; /* FIS FUA clear */ 6055 6056 fis->d.lbaLowExp = scsiCmnd->cdb[6]; /* FIS LBA (31:24) */ 6057 fis->d.lbaMidExp = scsiCmnd->cdb[5]; /* FIS LBA (39:32) */ 6058 fis->d.lbaHighExp = scsiCmnd->cdb[4]; /* FIS LBA (47:40) */ 6059 fis->d.featuresExp = scsiCmnd->cdb[12]; /* FIS sector count (15:8) */ 6060 fis->d.sectorCount = 0; /* Tag (7:3) set by LL layer */ 6061 fis->d.sectorCountExp = 0; 6062 fis->d.reserved4 = 0; 6063 fis->d.control = 0; /* FIS HOB bit clear */ 6064 fis->d.reserved5 = 0; 6065 6066 agRequestType = AGSA_SATA_PROTOCOL_FPDMA_READ; 6067 satIOContext->ATACmd = SAT_READ_FPDMA_QUEUED; 6068 } 6069 6070 /* saves the current LBA and orginal TL */ 6071 satIOContext->currentLBA = lba; 6072 satIOContext->OrgTL = tl; 6073 6074 /* 6075 computing number of loop and remainder for tl 6076 0xFF in case not ext 6077 0xFFFF in case EXT 6078 */ 6079 if (fis->h.command == SAT_READ_SECTORS || fis->h.command == SAT_READ_DMA) 6080 { 6081 LoopNum = smsatComputeLoopNum(tl, 0xFF); 6082 } 6083 else if (fis->h.command == SAT_READ_SECTORS_EXT || fis->h.command == SAT_READ_DMA_EXT) 6084 { 6085 /* SAT_READ_SECTORS_EXT, SAT_READ_DMA_EXT */ 6086 LoopNum = smsatComputeLoopNum(tl, 0xFFFF); 6087 } 6088 else 6089 { 6090 /* SAT_READ_FPDMA_QUEUEDK */ 6091 LoopNum = smsatComputeLoopNum(tl, 0xFFFF); 6092 } 6093 satIOContext->LoopNum = LoopNum; 6094 6095 if (LoopNum == 1) 6096 { 6097 SM_DBG5(("smsatRead16: NON CHAINED data\n")); 6098 satIOContext->satCompleteCB = &smsatNonChainedDataIOCB; 6099 } 6100 else 6101 { 6102 SM_DBG1(("smsatRead16: CHAINED data!!!\n")); 6103 /* re-setting tl */ 6104 if (fis->h.command == SAT_READ_SECTORS || fis->h.command == SAT_READ_DMA) 6105 { 6106 fis->d.sectorCount = 0xFF; 6107 } 6108 else if (fis->h.command == SAT_READ_SECTORS_EXT || fis->h.command == SAT_READ_DMA_EXT) 6109 { 6110 /* SAT_READ_SECTORS_EXT, SAT_READ_DMA_EXT */ 6111 fis->d.sectorCount = 0xFF; 6112 fis->d.sectorCountExp = 0xFF; 6113 } 6114 else 6115 { 6116 /* SAT_READ_FPDMA_QUEUED */ 6117 fis->h.features = 0xFF; 6118 fis->d.featuresExp = 0xFF; 6119 } 6120 6121 /* chained data */ 6122 satIOContext->satCompleteCB = &smsatChainedDataIOCB; 6123 } 6124 6125 /* 6126 * Prepare SGL and send FIS to LL layer. 6127 */ 6128 satIOContext->reqType = agRequestType; /* Save it */ 6129 6130 status = smsataLLIOStart( smRoot, 6131 smIORequest, 6132 smDeviceHandle, 6133 smScsiRequest, 6134 satIOContext); 6135 6136 SM_DBG5(("smsatRead16: return\n")); 6137 return (status); 6138 6139 } 6140 6141 osGLOBAL bit32 6142 smsatWrite6( 6143 smRoot_t *smRoot, 6144 smIORequest_t *smIORequest, 6145 smDeviceHandle_t *smDeviceHandle, 6146 smScsiInitiatorRequest_t *smScsiRequest, 6147 smSatIOContext_t *satIOContext 6148 ) 6149 { 6150 6151 bit32 status; 6152 bit32 agRequestType = AGSA_SATA_PROTOCOL_DMA_WRITE; 6153 smDeviceData_t *pSatDevData; 6154 smScsiRspSense_t *pSense; 6155 smIniScsiCmnd_t *scsiCmnd; 6156 agsaFisRegHostToDevice_t *fis; 6157 bit32 lba = 0; 6158 bit16 tl = 0; 6159 6160 pSense = satIOContext->pSense; 6161 pSatDevData = satIOContext->pSatDevData; 6162 scsiCmnd = &smScsiRequest->scsiCmnd; 6163 fis = satIOContext->pFis; 6164 6165 SM_DBG5(("smsatWrite6: start\n")); 6166 6167 /* checking CONTROL */ 6168 /* NACA == 1 or LINK == 1*/ 6169 if ( (scsiCmnd->cdb[5] & SCSI_NACA_MASK) || (scsiCmnd->cdb[5] & SCSI_LINK_MASK) ) 6170 { 6171 smsatSetSensePayload( pSense, 6172 SCSI_SNSKEY_ILLEGAL_REQUEST, 6173 0, 6174 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 6175 satIOContext); 6176 6177 /*smEnqueueIO(smRoot, satIOContext);*/ 6178 6179 tdsmIOCompletedCB( smRoot, 6180 smIORequest, 6181 smIOSuccess, 6182 SCSI_STAT_CHECK_CONDITION, 6183 satIOContext->pSmSenseData, 6184 satIOContext->interruptContext ); 6185 6186 SM_DBG1(("smsatWrite6: return control!!!\n")); 6187 return SM_RC_SUCCESS; 6188 } 6189 6190 6191 /* cbd6; computing LBA and transfer length */ 6192 lba = (((scsiCmnd->cdb[1]) & 0x1f) << (8*2)) 6193 + (scsiCmnd->cdb[2] << 8) + scsiCmnd->cdb[3]; 6194 tl = scsiCmnd->cdb[4]; 6195 6196 6197 /* Table 34, 9.1, p 46 */ 6198 /* 6199 note: As of 2/10/2006, no support for DMA QUEUED 6200 */ 6201 6202 /* 6203 Table 34, 9.1, p 46, b 6204 When no 48-bit addressing support or NCQ, if LBA is beyond (2^28 - 1), 6205 return check condition 6206 */ 6207 if (pSatDevData->satNCQ != agTRUE && 6208 pSatDevData->sat48BitSupport != agTRUE 6209 ) 6210 { 6211 if (lba > SAT_TR_LBA_LIMIT - 1) 6212 { 6213 smsatSetSensePayload( pSense, 6214 SCSI_SNSKEY_ILLEGAL_REQUEST, 6215 0, 6216 SCSI_SNSCODE_LOGICAL_BLOCK_ADDRESS_OUT_OF_RANGE, 6217 satIOContext); 6218 6219 /*smEnqueueIO(smRoot, satIOContext);*/ 6220 6221 tdsmIOCompletedCB( smRoot, 6222 smIORequest, 6223 smIOSuccess, 6224 SCSI_STAT_CHECK_CONDITION, 6225 satIOContext->pSmSenseData, 6226 satIOContext->interruptContext ); 6227 6228 SM_DBG1(("smsatWrite6: return LBA out of range!!!\n")); 6229 return SM_RC_SUCCESS; 6230 } 6231 } 6232 6233 /* case 1 and 2 */ 6234 if (lba + tl <= SAT_TR_LBA_LIMIT) 6235 { 6236 if (pSatDevData->satDMASupport == agTRUE && pSatDevData->satDMAEnabled == agTRUE) 6237 { 6238 /* case 2 */ 6239 /* WRITE DMA*/ 6240 SM_DBG5(("smsatWrite6: case 2\n")); 6241 6242 6243 fis->h.fisType = 0x27; /* Reg host to device */ 6244 fis->h.c_pmPort = 0x80; /* C Bit is set */ 6245 fis->h.command = SAT_WRITE_DMA; /* 0xCA */ 6246 fis->h.features = 0; /* FIS reserve */ 6247 fis->d.lbaLow = scsiCmnd->cdb[3]; /* FIS LBA (7 :0 ) */ 6248 fis->d.lbaMid = scsiCmnd->cdb[2]; /* FIS LBA (15:8 ) */ 6249 fis->d.lbaHigh = (bit8)((scsiCmnd->cdb[1]) & 0x1f); /* FIS LBA (23:16) */ 6250 fis->d.device = 0x40; /* FIS LBA mode */ 6251 fis->d.lbaLowExp = 0; 6252 fis->d.lbaMidExp = 0; 6253 fis->d.lbaHighExp = 0; 6254 fis->d.featuresExp = 0; 6255 if (tl == 0) 6256 { 6257 /* temporary fix */ 6258 fis->d.sectorCount = 0xff; /* FIS sector count (7:0) */ 6259 } 6260 else 6261 { 6262 fis->d.sectorCount = scsiCmnd->cdb[4]; /* FIS sector count (7:0) */ 6263 } 6264 fis->d.sectorCountExp = 0; 6265 fis->d.reserved4 = 0; 6266 fis->d.control = 0; /* FIS HOB bit clear */ 6267 fis->d.reserved5 = 0; 6268 6269 agRequestType = AGSA_SATA_PROTOCOL_DMA_WRITE; 6270 } 6271 else 6272 { 6273 /* case 1 */ 6274 /* WRITE SECTORS for easier implemetation */ 6275 SM_DBG5(("smsatWrite6: case 1\n")); 6276 6277 fis->h.fisType = 0x27; /* Reg host to device */ 6278 fis->h.c_pmPort = 0x80; /* C Bit is set */ 6279 fis->h.command = SAT_WRITE_SECTORS; /* 0xCA */ 6280 fis->h.features = 0; /* FIS reserve */ 6281 fis->d.lbaLow = scsiCmnd->cdb[3]; /* FIS LBA (7 :0 ) */ 6282 fis->d.lbaMid = scsiCmnd->cdb[2]; /* FIS LBA (15:8 ) */ 6283 fis->d.lbaHigh = (bit8)((scsiCmnd->cdb[1]) & 0x1f); /* FIS LBA (23:16) */ 6284 fis->d.device = 0x40; /* FIS LBA mode */ 6285 fis->d.lbaLowExp = 0; 6286 fis->d.lbaMidExp = 0; 6287 fis->d.lbaHighExp = 0; 6288 fis->d.featuresExp = 0; 6289 if (tl == 0) 6290 { 6291 /* temporary fix */ 6292 fis->d.sectorCount = 0xff; /* FIS sector count (7:0) */ 6293 } 6294 else 6295 { 6296 fis->d.sectorCount = scsiCmnd->cdb[4]; /* FIS sector count (7:0) */ 6297 } 6298 fis->d.sectorCountExp = 0; 6299 fis->d.reserved4 = 0; 6300 fis->d.control = 0; /* FIS HOB bit clear */ 6301 fis->d.reserved5 = 0; 6302 6303 agRequestType = AGSA_SATA_PROTOCOL_PIO_WRITE; 6304 6305 } 6306 } 6307 6308 /* case 3 and 4 */ 6309 if (pSatDevData->sat48BitSupport == agTRUE) 6310 { 6311 if (pSatDevData->satDMASupport == agTRUE && pSatDevData->satDMAEnabled == agTRUE) 6312 { 6313 /* case 3 */ 6314 /* WRITE DMA EXT only */ 6315 SM_DBG5(("smsatWrite6: case 3\n")); 6316 fis->h.fisType = 0x27; /* Reg host to device */ 6317 fis->h.c_pmPort = 0x80; /* C Bit is set */ 6318 fis->h.command = SAT_WRITE_DMA_EXT; /* 0x35 */ 6319 fis->h.features = 0; /* FIS reserve */ 6320 fis->d.lbaLow = scsiCmnd->cdb[3]; /* FIS LBA (7 :0 ) */ 6321 fis->d.lbaMid = scsiCmnd->cdb[2]; /* FIS LBA (15:8 ) */ 6322 fis->d.lbaHigh = (bit8)((scsiCmnd->cdb[1]) & 0x1f); /* FIS LBA (23:16) */ 6323 fis->d.device = 0x40; /* FIS LBA mode set */ 6324 fis->d.lbaLowExp = 0; /* FIS LBA (31:24) */ 6325 fis->d.lbaMidExp = 0; /* FIS LBA (39:32) */ 6326 fis->d.lbaHighExp = 0; /* FIS LBA (47:40) */ 6327 fis->d.featuresExp = 0; /* FIS reserve */ 6328 if (tl == 0) 6329 { 6330 /* sector count is 256, 0x100*/ 6331 fis->d.sectorCount = 0; /* FIS sector count (7:0) */ 6332 fis->d.sectorCountExp = 0x01; /* FIS sector count (15:8) */ 6333 } 6334 else 6335 { 6336 fis->d.sectorCount = scsiCmnd->cdb[4]; /* FIS sector count (7:0) */ 6337 fis->d.sectorCountExp = 0; /* FIS sector count (15:8) */ 6338 } 6339 fis->d.reserved4 = 0; 6340 fis->d.control = 0; /* FIS HOB bit clear */ 6341 fis->d.reserved5 = 0; 6342 6343 agRequestType = AGSA_SATA_PROTOCOL_DMA_WRITE; 6344 } 6345 else 6346 { 6347 /* case 4 */ 6348 /* WRITE SECTORS EXT for easier implemetation */ 6349 SM_DBG5(("smsatWrite6: case 4\n")); 6350 6351 fis->h.fisType = 0x27; /* Reg host to device */ 6352 fis->h.c_pmPort = 0x80; /* C Bit is set */ 6353 fis->h.command = SAT_WRITE_SECTORS_EXT; /* 0x34 */ 6354 fis->h.features = 0; /* FIS reserve */ 6355 fis->d.lbaLow = scsiCmnd->cdb[3]; /* FIS LBA (7 :0 ) */ 6356 fis->d.lbaMid = scsiCmnd->cdb[2]; /* FIS LBA (15:8 ) */ 6357 fis->d.lbaHigh = (bit8)((scsiCmnd->cdb[1]) & 0x1f); /* FIS LBA (23:16) */ 6358 fis->d.device = 0x40; /* FIS LBA mode set */ 6359 fis->d.lbaLowExp = 0; /* FIS LBA (31:24) */ 6360 fis->d.lbaMidExp = 0; /* FIS LBA (39:32) */ 6361 fis->d.lbaHighExp = 0; /* FIS LBA (47:40) */ 6362 fis->d.featuresExp = 0; /* FIS reserve */ 6363 if (tl == 0) 6364 { 6365 /* sector count is 256, 0x100*/ 6366 fis->d.sectorCount = 0; /* FIS sector count (7:0) */ 6367 fis->d.sectorCountExp = 0x01; /* FIS sector count (15:8) */ 6368 } 6369 else 6370 { 6371 fis->d.sectorCount = scsiCmnd->cdb[4]; /* FIS sector count (7:0) */ 6372 fis->d.sectorCountExp = 0; /* FIS sector count (15:8) */ 6373 } 6374 fis->d.reserved4 = 0; 6375 fis->d.control = 0; /* FIS HOB bit clear */ 6376 fis->d.reserved5 = 0; 6377 6378 agRequestType = AGSA_SATA_PROTOCOL_PIO_WRITE; 6379 } 6380 } 6381 6382 /* case 5 */ 6383 if (pSatDevData->satNCQ == agTRUE) 6384 { 6385 /* WRITE FPDMA QUEUED */ 6386 if (pSatDevData->sat48BitSupport != agTRUE) 6387 { 6388 /* sanity check */ 6389 SM_DBG5(("smsatWrite6: case 5 !!! error NCQ but 28 bit address support!!!\n")); 6390 smsatSetSensePayload( pSense, 6391 SCSI_SNSKEY_ILLEGAL_REQUEST, 6392 0, 6393 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 6394 satIOContext); 6395 6396 /*smEnqueueIO(smRoot, satIOContext);*/ 6397 6398 tdsmIOCompletedCB( smRoot, 6399 smIORequest, 6400 smIOSuccess, 6401 SCSI_STAT_CHECK_CONDITION, 6402 satIOContext->pSmSenseData, 6403 satIOContext->interruptContext ); 6404 return SM_RC_SUCCESS; 6405 } 6406 SM_DBG5(("smsatWrite6: case 5\n")); 6407 6408 /* Support 48-bit FPDMA addressing, use WRITE FPDMA QUEUE command */ 6409 6410 fis->h.fisType = 0x27; /* Reg host to device */ 6411 fis->h.c_pmPort = 0x80; /* C Bit is set */ 6412 fis->h.command = SAT_WRITE_FPDMA_QUEUED; /* 0x61 */ 6413 fis->d.lbaLow = scsiCmnd->cdb[3]; /* FIS LBA (7 :0 ) */ 6414 fis->d.lbaMid = scsiCmnd->cdb[2]; /* FIS LBA (15:8 ) */ 6415 fis->d.lbaHigh = (bit8)((scsiCmnd->cdb[1]) & 0x1f); /* FIS LBA (23:16) */ 6416 fis->d.device = 0x40; /* FIS FUA clear */ 6417 fis->d.lbaLowExp = 0; /* FIS LBA (31:24) */ 6418 fis->d.lbaMidExp = 0; /* FIS LBA (39:32) */ 6419 fis->d.lbaHighExp = 0; /* FIS LBA (47:40) */ 6420 if (tl == 0) 6421 { 6422 /* sector count is 256, 0x100*/ 6423 fis->h.features = 0; /* FIS sector count (7:0) */ 6424 fis->d.featuresExp = 0x01; /* FIS sector count (15:8) */ 6425 } 6426 else 6427 { 6428 fis->h.features = scsiCmnd->cdb[4]; /* FIS sector count (7:0) */ 6429 fis->d.featuresExp = 0; /* FIS sector count (15:8) */ 6430 } 6431 fis->d.sectorCount = 0; /* Tag (7:3) set by LL layer */ 6432 fis->d.sectorCountExp = 0; 6433 fis->d.reserved4 = 0; 6434 fis->d.control = 0; /* FIS HOB bit clear */ 6435 fis->d.reserved5 = 0; 6436 6437 agRequestType = AGSA_SATA_PROTOCOL_FPDMA_WRITE; 6438 } 6439 6440 /* Initialize CB for SATA completion. 6441 */ 6442 satIOContext->satCompleteCB = &smsatNonChainedDataIOCB; 6443 6444 /* 6445 * Prepare SGL and send FIS to LL layer. 6446 */ 6447 satIOContext->reqType = agRequestType; /* Save it */ 6448 6449 status = smsataLLIOStart( smRoot, 6450 smIORequest, 6451 smDeviceHandle, 6452 smScsiRequest, 6453 satIOContext); 6454 return (status); 6455 } 6456 6457 osGLOBAL FORCEINLINE bit32 6458 smsatWrite10( 6459 smRoot_t *smRoot, 6460 smIORequest_t *smIORequest, 6461 smDeviceHandle_t *smDeviceHandle, 6462 smScsiInitiatorRequest_t *smScsiRequest, 6463 smSatIOContext_t *satIOContext 6464 ) 6465 { 6466 smDeviceData_t *pSatDevData = satIOContext->pSatDevData; 6467 smScsiRspSense_t *pSense = satIOContext->pSense; 6468 smIniScsiCmnd_t *scsiCmnd = &smScsiRequest->scsiCmnd; 6469 agsaFisRegHostToDevice_t *fis = satIOContext->pFis; 6470 bit32 status = SM_RC_FAILURE; 6471 bit32 agRequestType = AGSA_SATA_PROTOCOL_DMA_WRITE; 6472 bit32 lba = 0; 6473 bit32 tl = 0; 6474 bit32 LoopNum = 1; 6475 bit32 AllChk = agFALSE; /* lba, lba+tl check against ATA limit and Disk capacity */ 6476 bit8 LBA[8]; 6477 bit8 TL[8]; 6478 6479 SM_DBG2(("smsatWrite10: start\n")); 6480 6481 /* checking FUA_NV */ 6482 if (scsiCmnd->cdb[1] & SCSI_FUA_NV_MASK) 6483 { 6484 smsatSetSensePayload( pSense, 6485 SCSI_SNSKEY_ILLEGAL_REQUEST, 6486 0, 6487 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 6488 satIOContext); 6489 6490 /*smEnqueueIO(smRoot, satIOContext);*/ 6491 6492 tdsmIOCompletedCB( smRoot, 6493 smIORequest, 6494 smIOSuccess, 6495 SCSI_STAT_CHECK_CONDITION, 6496 satIOContext->pSmSenseData, 6497 satIOContext->interruptContext ); 6498 6499 SM_DBG1(("smsatWrite10: return FUA_NV!!!\n")); 6500 return SM_RC_SUCCESS; 6501 6502 } 6503 6504 /* checking CONTROL */ 6505 /* NACA == 1 or LINK == 1*/ 6506 if ( (scsiCmnd->cdb[9] & SCSI_NACA_MASK) || (scsiCmnd->cdb[9] & SCSI_LINK_MASK) ) 6507 { 6508 smsatSetSensePayload( pSense, 6509 SCSI_SNSKEY_ILLEGAL_REQUEST, 6510 0, 6511 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 6512 satIOContext); 6513 6514 /*smEnqueueIO(smRoot, satIOContext);*/ 6515 6516 tdsmIOCompletedCB( smRoot, 6517 smIORequest, 6518 smIOSuccess, 6519 SCSI_STAT_CHECK_CONDITION, 6520 satIOContext->pSmSenseData, 6521 satIOContext->interruptContext ); 6522 6523 SM_DBG1(("smsatWrite10: return control!!!\n")); 6524 return SM_RC_SUCCESS; 6525 } 6526 /* 6527 sm_memset(LBA, 0, sizeof(LBA)); 6528 sm_memset(TL, 0, sizeof(TL)); 6529 */ 6530 /* do not use memcpy due to indexing in LBA and TL */ 6531 LBA[0] = 0; /* MSB */ 6532 LBA[1] = 0; 6533 LBA[2] = 0; 6534 LBA[3] = 0; 6535 LBA[4] = scsiCmnd->cdb[2]; 6536 LBA[5] = scsiCmnd->cdb[3]; 6537 LBA[6] = scsiCmnd->cdb[4]; 6538 LBA[7] = scsiCmnd->cdb[5]; /* LSB */ 6539 6540 TL[0] = 0; 6541 TL[1] = 0; 6542 TL[2] = 0; 6543 TL[3] = 0; 6544 TL[4] = 0; 6545 TL[5] = 0; 6546 TL[6] = scsiCmnd->cdb[7]; 6547 TL[7] = scsiCmnd->cdb[8]; /* LSB */ 6548 6549 6550 6551 /* cbd10; computing LBA and transfer length */ 6552 lba = (scsiCmnd->cdb[2] << (24)) + (scsiCmnd->cdb[3] << (16)) 6553 + (scsiCmnd->cdb[4] << 8) + scsiCmnd->cdb[5]; 6554 tl = (scsiCmnd->cdb[7] << 8) + scsiCmnd->cdb[8]; 6555 6556 SM_DBG5(("smsatWrite10: lba %d functioned lba %d\n", lba, smsatComputeCDB10LBA(satIOContext))); 6557 SM_DBG5(("smsatWrite10: tl %d functioned tl %d\n", tl, smsatComputeCDB10TL(satIOContext))); 6558 6559 /* Table 34, 9.1, p 46 */ 6560 /* 6561 note: As of 2/10/2006, no support for DMA QUEUED 6562 */ 6563 6564 /* 6565 Table 34, 9.1, p 46, b 6566 When no 48-bit addressing support or NCQ, if LBA is beyond (2^28 - 1), 6567 return check condition 6568 */ 6569 if (pSatDevData->satNCQ != agTRUE && 6570 pSatDevData->sat48BitSupport != agTRUE 6571 ) 6572 { 6573 AllChk = smsatCheckLimit(LBA, TL, agFALSE, pSatDevData); 6574 if (AllChk) 6575 { 6576 SM_DBG1(("smsatWrite10: return LBA out of range, not EXT!!!\n")); 6577 SM_DBG1(("smsatWrite10: cdb 0x%x 0x%x 0x%x 0x%x!!!\n",scsiCmnd->cdb[2], scsiCmnd->cdb[3], 6578 scsiCmnd->cdb[4], scsiCmnd->cdb[5])); 6579 SM_DBG1(("smsatWrite10: lba 0x%x SAT_TR_LBA_LIMIT 0x%x!!!\n", lba, SAT_TR_LBA_LIMIT)); 6580 smsatSetSensePayload( pSense, 6581 SCSI_SNSKEY_ILLEGAL_REQUEST, 6582 0, 6583 SCSI_SNSCODE_LOGICAL_BLOCK_ADDRESS_OUT_OF_RANGE, 6584 satIOContext); 6585 6586 /*smEnqueueIO(smRoot, satIOContext);*/ 6587 6588 tdsmIOCompletedCB( smRoot, 6589 smIORequest, 6590 smIOSuccess, 6591 SCSI_STAT_CHECK_CONDITION, 6592 satIOContext->pSmSenseData, 6593 satIOContext->interruptContext ); 6594 6595 return SM_RC_SUCCESS; 6596 } 6597 } 6598 else 6599 { 6600 AllChk = smsatCheckLimit(LBA, TL, agTRUE, pSatDevData); 6601 if (AllChk) 6602 { 6603 SM_DBG1(("smsatWrite10: return LBA out of range, EXT!!!\n")); 6604 smsatSetSensePayload( pSense, 6605 SCSI_SNSKEY_ILLEGAL_REQUEST, 6606 0, 6607 SCSI_SNSCODE_LOGICAL_BLOCK_ADDRESS_OUT_OF_RANGE, 6608 satIOContext); 6609 6610 /*smEnqueueIO(smRoot, satIOContext);*/ 6611 6612 tdsmIOCompletedCB( smRoot, 6613 smIORequest, 6614 smIOSuccess, 6615 SCSI_STAT_CHECK_CONDITION, 6616 satIOContext->pSmSenseData, 6617 satIOContext->interruptContext ); 6618 6619 return SM_RC_SUCCESS; 6620 } 6621 6622 } 6623 6624 /* case 5 */ 6625 if (pSatDevData->satNCQ == agTRUE) 6626 { 6627 /* WRITE FPDMA QUEUED */ 6628 if (pSatDevData->sat48BitSupport != agTRUE) 6629 { 6630 SM_DBG1(("smsatWrite10: case 5 !!! error NCQ but 28 bit address support!!!\n")); 6631 smsatSetSensePayload( pSense, 6632 SCSI_SNSKEY_ILLEGAL_REQUEST, 6633 0, 6634 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 6635 satIOContext); 6636 6637 /*smEnqueueIO(smRoot, satIOContext);*/ 6638 6639 tdsmIOCompletedCB( smRoot, 6640 smIORequest, 6641 smIOSuccess, 6642 SCSI_STAT_CHECK_CONDITION, 6643 satIOContext->pSmSenseData, 6644 satIOContext->interruptContext ); 6645 return SM_RC_SUCCESS; 6646 } 6647 SM_DBG6(("smsatWrite10: case 5\n")); 6648 6649 /* Support 48-bit FPDMA addressing, use WRITE FPDMA QUEUE command */ 6650 6651 fis->h.fisType = 0x27; /* Reg host to device */ 6652 fis->h.c_pmPort = 0x80; /* C Bit is set */ 6653 fis->h.command = SAT_WRITE_FPDMA_QUEUED; /* 0x61 */ 6654 fis->h.features = scsiCmnd->cdb[8]; /* FIS sector count (7:0) */ 6655 fis->d.lbaLow = scsiCmnd->cdb[5]; /* FIS LBA (7 :0 ) */ 6656 fis->d.lbaMid = scsiCmnd->cdb[4]; /* FIS LBA (15:8 ) */ 6657 fis->d.lbaHigh = scsiCmnd->cdb[3]; /* FIS LBA (23:16) */ 6658 6659 /* Check FUA bit */ 6660 if (scsiCmnd->cdb[1] & SCSI_WRITE10_FUA_MASK) 6661 fis->d.device = 0xC0; /* FIS FUA set */ 6662 else 6663 fis->d.device = 0x40; /* FIS FUA clear */ 6664 6665 fis->d.lbaLowExp = scsiCmnd->cdb[2]; /* FIS LBA (31:24) */ 6666 fis->d.lbaMidExp = 0; /* FIS LBA (39:32) */ 6667 fis->d.lbaHighExp = 0; /* FIS LBA (47:40) */ 6668 fis->d.featuresExp = scsiCmnd->cdb[7]; /* FIS sector count (15:8) */ 6669 fis->d.sectorCount = 0; /* Tag (7:3) set by LL layer */ 6670 fis->d.sectorCountExp = 0; 6671 fis->d.reserved4 = 0; 6672 fis->d.control = 0; /* FIS HOB bit clear */ 6673 fis->d.reserved5 = 0; 6674 6675 agRequestType = AGSA_SATA_PROTOCOL_FPDMA_WRITE; 6676 satIOContext->ATACmd = SAT_WRITE_FPDMA_QUEUED; 6677 } 6678 /* case 3 and 4 */ 6679 else if (pSatDevData->sat48BitSupport == agTRUE) 6680 { 6681 if (pSatDevData->satDMASupport == agTRUE && pSatDevData->satDMAEnabled == agTRUE) 6682 { 6683 /* case 3 */ 6684 /* WRITE DMA EXT or WRITE DMA FUA EXT */ 6685 SM_DBG5(("smsatWrite10: case 3\n")); 6686 fis->h.fisType = 0x27; /* Reg host to device */ 6687 fis->h.c_pmPort = 0x80; /* C Bit is set */ 6688 6689 /* SAT_WRITE_DMA_FUA_EXT is optional and we don't support it */ 6690 fis->h.command = SAT_WRITE_DMA_EXT; /* 0x35 */ 6691 satIOContext->ATACmd = SAT_WRITE_DMA_EXT; 6692 6693 fis->h.features = 0; /* FIS reserve */ 6694 fis->d.lbaLow = scsiCmnd->cdb[5]; /* FIS LBA (7 :0 ) */ 6695 fis->d.lbaMid = scsiCmnd->cdb[4]; /* FIS LBA (15:8 ) */ 6696 fis->d.lbaHigh = scsiCmnd->cdb[3]; /* FIS LBA (23:16) */ 6697 fis->d.device = 0x40; /* FIS LBA mode set */ 6698 fis->d.lbaLowExp = scsiCmnd->cdb[2]; /* FIS LBA (31:24) */ 6699 fis->d.lbaMidExp = 0; /* FIS LBA (39:32) */ 6700 fis->d.lbaHighExp = 0; /* FIS LBA (47:40) */ 6701 fis->d.featuresExp = 0; /* FIS reserve */ 6702 fis->d.sectorCount = scsiCmnd->cdb[8]; /* FIS sector count (7:0) */ 6703 fis->d.sectorCountExp = scsiCmnd->cdb[7]; /* FIS sector count (15:8) */ 6704 fis->d.reserved4 = 0; 6705 fis->d.control = 0; /* FIS HOB bit clear */ 6706 fis->d.reserved5 = 0; 6707 6708 agRequestType = AGSA_SATA_PROTOCOL_DMA_WRITE; 6709 } 6710 else 6711 { 6712 /* case 4 */ 6713 /* WRITE MULTIPLE EXT or WRITE MULTIPLE FUA EXT or WRITE SECTOR(S) EXT */ 6714 /* WRITE SECTORS EXT for easier implemetation */ 6715 SM_DBG5(("smsatWrite10: case 4\n")); 6716 fis->h.fisType = 0x27; /* Reg host to device */ 6717 fis->h.c_pmPort = 0x80; /* C Bit is set */ 6718 fis->h.command = SAT_WRITE_SECTORS_EXT; /* 0x34 */ 6719 6720 fis->h.features = 0; /* FIS reserve */ 6721 fis->d.lbaLow = scsiCmnd->cdb[5]; /* FIS LBA (7 :0 ) */ 6722 fis->d.lbaMid = scsiCmnd->cdb[4]; /* FIS LBA (15:8 ) */ 6723 fis->d.lbaHigh = scsiCmnd->cdb[3]; /* FIS LBA (23:16) */ 6724 fis->d.device = 0x40; /* FIS LBA mode set */ 6725 fis->d.lbaLowExp = scsiCmnd->cdb[2]; /* FIS LBA (31:24) */ 6726 fis->d.lbaMidExp = 0; /* FIS LBA (39:32) */ 6727 fis->d.lbaHighExp = 0; /* FIS LBA (47:40) */ 6728 fis->d.featuresExp = 0; /* FIS reserve */ 6729 fis->d.sectorCount = scsiCmnd->cdb[8]; /* FIS sector count (7:0) */ 6730 fis->d.sectorCountExp = scsiCmnd->cdb[7]; /* FIS sector count (15:8) */ 6731 fis->d.reserved4 = 0; 6732 fis->d.control = 0; /* FIS HOB bit clear */ 6733 fis->d.reserved5 = 0; 6734 6735 agRequestType = AGSA_SATA_PROTOCOL_PIO_WRITE; 6736 satIOContext->ATACmd = SAT_WRITE_SECTORS_EXT; 6737 } 6738 } 6739 else /* case 1 and 2 */ 6740 { 6741 if (pSatDevData->satDMASupport == agTRUE && pSatDevData->satDMAEnabled == agTRUE) 6742 { 6743 /* case 2 */ 6744 /* WRITE DMA*/ 6745 /* can't fit the transfer length */ 6746 SM_DBG5(("smsatWrite10: case 2\n")); 6747 fis->h.fisType = 0x27; /* Reg host to device */ 6748 fis->h.c_pmPort = 0x80; /* C bit is set */ 6749 fis->h.command = SAT_WRITE_DMA; /* 0xCA */ 6750 fis->h.features = 0; /* FIS reserve */ 6751 fis->d.lbaLow = scsiCmnd->cdb[5]; /* FIS LBA (7 :0 ) */ 6752 fis->d.lbaMid = scsiCmnd->cdb[4]; /* FIS LBA (15:8 ) */ 6753 fis->d.lbaHigh = scsiCmnd->cdb[3]; /* FIS LBA (23:16) */ 6754 6755 /* FIS LBA mode set LBA (27:24) */ 6756 fis->d.device = (bit8)((0x4 << 4) | (scsiCmnd->cdb[2] & 0xF)); 6757 6758 fis->d.lbaLowExp = 0; 6759 fis->d.lbaMidExp = 0; 6760 fis->d.lbaHighExp = 0; 6761 fis->d.featuresExp = 0; 6762 fis->d.sectorCount = scsiCmnd->cdb[8]; /* FIS sector count (7:0) */ 6763 fis->d.sectorCountExp = 0; 6764 fis->d.reserved4 = 0; 6765 fis->d.control = 0; /* FIS HOB bit clear */ 6766 fis->d.reserved5 = 0; 6767 6768 agRequestType = AGSA_SATA_PROTOCOL_DMA_WRITE; 6769 satIOContext->ATACmd = SAT_WRITE_DMA; 6770 } 6771 else 6772 { 6773 /* case 1 */ 6774 /* WRITE MULTIPLE or WRITE SECTOR(S) */ 6775 /* WRITE SECTORS for easier implemetation */ 6776 /* can't fit the transfer length */ 6777 SM_DBG5(("smsatWrite10: case 1\n")); 6778 fis->h.fisType = 0x27; /* Reg host to device */ 6779 fis->h.c_pmPort = 0x80; /* C bit is set */ 6780 fis->h.command = SAT_WRITE_SECTORS; /* 0x30 */ 6781 fis->h.features = 0; /* FIS reserve */ 6782 fis->d.lbaLow = scsiCmnd->cdb[5]; /* FIS LBA (7 :0 ) */ 6783 fis->d.lbaMid = scsiCmnd->cdb[4]; /* FIS LBA (15:8 ) */ 6784 fis->d.lbaHigh = scsiCmnd->cdb[3]; /* FIS LBA (23:16) */ 6785 6786 /* FIS LBA mode set LBA (27:24) */ 6787 fis->d.device = (bit8)((0x4 << 4) | (scsiCmnd->cdb[2] & 0xF)); 6788 6789 fis->d.lbaLowExp = 0; 6790 fis->d.lbaMidExp = 0; 6791 fis->d.lbaHighExp = 0; 6792 fis->d.featuresExp = 0; 6793 fis->d.sectorCount = scsiCmnd->cdb[8]; /* FIS sector count (7:0) */ 6794 fis->d.sectorCountExp = 0; 6795 fis->d.reserved4 = 0; 6796 fis->d.control = 0; /* FIS HOB bit clear */ 6797 fis->d.reserved5 = 0; 6798 6799 agRequestType = AGSA_SATA_PROTOCOL_PIO_WRITE; 6800 satIOContext->ATACmd = SAT_WRITE_SECTORS; 6801 } 6802 } 6803 6804 // smhexdump("satWrite10 final fis", (bit8 *)fis, sizeof(agsaFisRegHostToDevice_t)); 6805 6806 satIOContext->currentLBA = lba; 6807 satIOContext->OrgTL = tl; 6808 6809 /* 6810 computing number of loop and remainder for tl 6811 0xFF in case not ext 6812 0xFFFF in case EXT 6813 */ 6814 if (fis->h.command == SAT_WRITE_SECTORS || fis->h.command == SAT_WRITE_DMA) 6815 { 6816 LoopNum = smsatComputeLoopNum(tl, 0x100); 6817 } 6818 else 6819 { 6820 /* SAT_WRITE_FPDMA_QUEUEDK */ 6821 /* SAT_READ_SECTORS_EXT, SAT_READ_DMA_EXT */ 6822 LoopNum = smsatComputeLoopNum(tl, 0xFFFF); 6823 } 6824 6825 satIOContext->LoopNum = LoopNum; 6826 6827 6828 if (LoopNum == 1) 6829 { 6830 SM_DBG5(("smsatWrite10: NON CHAINED data\n")); 6831 /* Initialize CB for SATA completion. 6832 */ 6833 satIOContext->satCompleteCB = &smsatNonChainedDataIOCB; 6834 } 6835 else 6836 { 6837 SM_DBG2(("smsatWrite10: CHAINED data!!!\n")); 6838 /* re-setting tl */ 6839 if (fis->h.command == SAT_WRITE_SECTORS || fis->h.command == SAT_WRITE_DMA) 6840 { 6841 fis->d.sectorCount = 0x0; 6842 smsatSplitSGL(smRoot, 6843 smIORequest, 6844 smDeviceHandle, 6845 smScsiRequest, 6846 satIOContext, 6847 NON_BIT48_ADDRESS_TL_LIMIT*SATA_SECTOR_SIZE, /* 0x100 * 0x200 */ 6848 (satIOContext->OrgTL)*SATA_SECTOR_SIZE, 6849 agTRUE); 6850 } 6851 else if (fis->h.command == SAT_WRITE_SECTORS_EXT || 6852 fis->h.command == SAT_WRITE_DMA_EXT || 6853 fis->h.command == SAT_WRITE_DMA_FUA_EXT 6854 ) 6855 { 6856 fis->d.sectorCount = 0xFF; 6857 fis->d.sectorCountExp = 0xFF; 6858 smsatSplitSGL(smRoot, 6859 smIORequest, 6860 smDeviceHandle, 6861 smScsiRequest, 6862 satIOContext, 6863 BIT48_ADDRESS_TL_LIMIT*SATA_SECTOR_SIZE, /* 0xFFFF * 0x200 */ 6864 (satIOContext->OrgTL)*SATA_SECTOR_SIZE, 6865 agTRUE); 6866 } 6867 else 6868 { 6869 /* SAT_WRITE_FPDMA_QUEUED */ 6870 fis->h.features = 0xFF; 6871 fis->d.featuresExp = 0xFF; 6872 smsatSplitSGL(smRoot, 6873 smIORequest, 6874 smDeviceHandle, 6875 smScsiRequest, 6876 satIOContext, 6877 BIT48_ADDRESS_TL_LIMIT*SATA_SECTOR_SIZE, /* 0xFFFF * 0x200 */ 6878 (satIOContext->OrgTL)*SATA_SECTOR_SIZE, 6879 agTRUE); 6880 } 6881 6882 /* Initialize CB for SATA completion. 6883 */ 6884 satIOContext->satCompleteCB = &smsatChainedDataIOCB; 6885 } 6886 6887 6888 /* 6889 * Prepare SGL and send FIS to LL layer. 6890 */ 6891 satIOContext->reqType = agRequestType; /* Save it */ 6892 6893 status = smsataLLIOStart( smRoot, 6894 smIORequest, 6895 smDeviceHandle, 6896 smScsiRequest, 6897 satIOContext); 6898 return (status); 6899 } 6900 6901 osGLOBAL bit32 6902 smsatWrite12( 6903 smRoot_t *smRoot, 6904 smIORequest_t *smIORequest, 6905 smDeviceHandle_t *smDeviceHandle, 6906 smScsiInitiatorRequest_t *smScsiRequest, 6907 smSatIOContext_t *satIOContext 6908 ) 6909 { 6910 bit32 status; 6911 bit32 agRequestType = AGSA_SATA_PROTOCOL_DMA_WRITE; 6912 smDeviceData_t *pSatDevData; 6913 smScsiRspSense_t *pSense; 6914 smIniScsiCmnd_t *scsiCmnd; 6915 agsaFisRegHostToDevice_t *fis; 6916 bit32 lba = 0; 6917 bit32 tl = 0; 6918 bit32 LoopNum = 1; 6919 bit8 LBA[8]; 6920 bit8 TL[8]; 6921 bit32 AllChk = agFALSE; /* lba, lba+tl check against ATA limit and Disk capacity */ 6922 6923 pSense = satIOContext->pSense; 6924 pSatDevData = satIOContext->pSatDevData; 6925 scsiCmnd = &smScsiRequest->scsiCmnd; 6926 fis = satIOContext->pFis; 6927 6928 SM_DBG5(("smsatWrite12: start\n")); 6929 6930 /* checking FUA_NV */ 6931 if (scsiCmnd->cdb[1] & SCSI_FUA_NV_MASK) 6932 { 6933 smsatSetSensePayload( pSense, 6934 SCSI_SNSKEY_ILLEGAL_REQUEST, 6935 0, 6936 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 6937 satIOContext); 6938 6939 /*smEnqueueIO(smRoot, satIOContext);*/ 6940 6941 tdsmIOCompletedCB( smRoot, 6942 smIORequest, 6943 smIOSuccess, 6944 SCSI_STAT_CHECK_CONDITION, 6945 satIOContext->pSmSenseData, 6946 satIOContext->interruptContext ); 6947 6948 SM_DBG1(("smsatWrite12: return FUA_NV!!!\n")); 6949 return SM_RC_SUCCESS; 6950 6951 } 6952 6953 6954 /* checking CONTROL */ 6955 /* NACA == 1 or LINK == 1*/ 6956 if ( (scsiCmnd->cdb[11] & SCSI_NACA_MASK) || (scsiCmnd->cdb[11] & SCSI_LINK_MASK) ) 6957 { 6958 smsatSetSensePayload( pSense, 6959 SCSI_SNSKEY_ILLEGAL_REQUEST, 6960 0, 6961 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 6962 satIOContext); 6963 6964 /*smEnqueueIO(smRoot, satIOContext);*/ 6965 6966 tdsmIOCompletedCB( smRoot, 6967 smIORequest, 6968 smIOSuccess, 6969 SCSI_STAT_CHECK_CONDITION, 6970 satIOContext->pSmSenseData, 6971 satIOContext->interruptContext ); 6972 6973 SM_DBG1(("smsatWrite10: return control!!!\n")); 6974 return SM_RC_SUCCESS; 6975 } 6976 6977 6978 sm_memset(LBA, 0, sizeof(LBA)); 6979 sm_memset(TL, 0, sizeof(TL)); 6980 6981 /* do not use memcpy due to indexing in LBA and TL */ 6982 LBA[0] = 0; /* MSB */ 6983 LBA[1] = 0; 6984 LBA[2] = 0; 6985 LBA[3] = 0; 6986 LBA[4] = scsiCmnd->cdb[2]; 6987 LBA[5] = scsiCmnd->cdb[3]; 6988 LBA[6] = scsiCmnd->cdb[4]; 6989 LBA[7] = scsiCmnd->cdb[5]; /* LSB */ 6990 6991 TL[0] = 0; /* MSB */ 6992 TL[1] = 0; 6993 TL[2] = 0; 6994 TL[3] = 0; 6995 TL[4] = scsiCmnd->cdb[6]; 6996 TL[5] = scsiCmnd->cdb[7]; 6997 TL[6] = scsiCmnd->cdb[8]; 6998 TL[7] = scsiCmnd->cdb[9]; /* LSB */ 6999 7000 7001 lba = smsatComputeCDB12LBA(satIOContext); 7002 tl = smsatComputeCDB12TL(satIOContext); 7003 7004 7005 /* Table 34, 9.1, p 46 */ 7006 /* 7007 note: As of 2/10/2006, no support for DMA QUEUED 7008 */ 7009 7010 /* 7011 Table 34, 9.1, p 46, b 7012 When no 48-bit addressing support or NCQ, if LBA is beyond (2^28 - 1), 7013 return check condition 7014 */ 7015 if (pSatDevData->satNCQ != agTRUE && 7016 pSatDevData->sat48BitSupport != agTRUE 7017 ) 7018 { 7019 AllChk = smsatCheckLimit(LBA, TL, agFALSE, pSatDevData); 7020 7021 /*smEnqueueIO(smRoot, satIOContext);*/ 7022 7023 7024 7025 if (AllChk) 7026 { 7027 SM_DBG1(("smsatWrite12: return LBA out of range, not EXT!!!\n")); 7028 smsatSetSensePayload( pSense, 7029 SCSI_SNSKEY_ILLEGAL_REQUEST, 7030 0, 7031 SCSI_SNSCODE_LOGICAL_BLOCK_ADDRESS_OUT_OF_RANGE, 7032 satIOContext); 7033 7034 /*smEnqueueIO(smRoot, satIOContext);*/ 7035 7036 tdsmIOCompletedCB( smRoot, 7037 smIORequest, 7038 smIOSuccess, 7039 SCSI_STAT_CHECK_CONDITION, 7040 satIOContext->pSmSenseData, 7041 satIOContext->interruptContext ); 7042 7043 return SM_RC_SUCCESS; 7044 } 7045 } 7046 else 7047 { 7048 AllChk = smsatCheckLimit(LBA, TL, agTRUE, pSatDevData); 7049 if (AllChk) 7050 { 7051 SM_DBG1(("smsatWrite12: return LBA out of range, EXT!!!\n")); 7052 smsatSetSensePayload( pSense, 7053 SCSI_SNSKEY_ILLEGAL_REQUEST, 7054 0, 7055 SCSI_SNSCODE_LOGICAL_BLOCK_ADDRESS_OUT_OF_RANGE, 7056 satIOContext); 7057 tdsmIOCompletedCB( smRoot, 7058 smIORequest, 7059 smIOSuccess, 7060 SCSI_STAT_CHECK_CONDITION, 7061 satIOContext->pSmSenseData, 7062 satIOContext->interruptContext ); 7063 return SM_RC_SUCCESS; 7064 } 7065 } 7066 if (pSatDevData->satDMASupport == agTRUE && pSatDevData->satDMAEnabled == agTRUE) 7067 { 7068 /* case 2 */ 7069 /* WRITE DMA*/ 7070 /* In case that we can't fit the transfer length, we loop */ 7071 SM_DBG5(("smsatWrite10: case 2\n")); 7072 fis->h.fisType = 0x27; /* Reg host to device */ 7073 fis->h.c_pmPort = 0x80; /* C bit is set */ 7074 fis->h.command = SAT_WRITE_DMA; /* 0xCA */ 7075 fis->h.features = 0; /* FIS reserve */ 7076 fis->d.lbaLow = scsiCmnd->cdb[5]; /* FIS LBA (7 :0 ) */ 7077 fis->d.lbaMid = scsiCmnd->cdb[4]; /* FIS LBA (15:8 ) */ 7078 fis->d.lbaHigh = scsiCmnd->cdb[3]; /* FIS LBA (23:16) */ 7079 7080 /* FIS LBA mode set LBA (27:24) */ 7081 fis->d.device = (bit8)((0x4 << 4) | (scsiCmnd->cdb[2] & 0xF)); 7082 7083 fis->d.lbaLowExp = 0; 7084 fis->d.lbaMidExp = 0; 7085 fis->d.lbaHighExp = 0; 7086 fis->d.featuresExp = 0; 7087 fis->d.sectorCount = scsiCmnd->cdb[9]; /* FIS sector count (7:0) */ 7088 fis->d.sectorCountExp = 0; 7089 fis->d.reserved4 = 0; 7090 fis->d.control = 0; /* FIS HOB bit clear */ 7091 fis->d.reserved5 = 0; 7092 7093 agRequestType = AGSA_SATA_PROTOCOL_DMA_WRITE; 7094 satIOContext->ATACmd = SAT_WRITE_DMA; 7095 } 7096 else 7097 { 7098 /* case 1 */ 7099 /* WRITE MULTIPLE or WRITE SECTOR(S) */ 7100 /* WRITE SECTORS for easier implemetation */ 7101 /* In case that we can't fit the transfer length, we loop */ 7102 SM_DBG5(("smsatWrite10: case 1\n")); 7103 fis->h.fisType = 0x27; /* Reg host to device */ 7104 fis->h.c_pmPort = 0x80; /* C bit is set */ 7105 fis->h.command = SAT_WRITE_SECTORS; /* 0x30 */ 7106 fis->h.features = 0; /* FIS reserve */ 7107 fis->d.lbaLow = scsiCmnd->cdb[5]; /* FIS LBA (7 :0 ) */ 7108 fis->d.lbaMid = scsiCmnd->cdb[4]; /* FIS LBA (15:8 ) */ 7109 fis->d.lbaHigh = scsiCmnd->cdb[3]; /* FIS LBA (23:16) */ 7110 7111 /* FIS LBA mode set LBA (27:24) */ 7112 fis->d.device = (bit8)((0x4 << 4) | (scsiCmnd->cdb[2] & 0xF)); 7113 7114 fis->d.lbaLowExp = 0; 7115 fis->d.lbaMidExp = 0; 7116 fis->d.lbaHighExp = 0; 7117 fis->d.featuresExp = 0; 7118 fis->d.sectorCount = scsiCmnd->cdb[9]; /* FIS sector count (7:0) */ 7119 fis->d.sectorCountExp = 0; 7120 fis->d.reserved4 = 0; 7121 fis->d.control = 0; /* FIS HOB bit clear */ 7122 fis->d.reserved5 = 0; 7123 7124 agRequestType = AGSA_SATA_PROTOCOL_PIO_WRITE; 7125 satIOContext->ATACmd = SAT_WRITE_SECTORS; 7126 } 7127 7128 /* case 3 and 4 */ 7129 if (pSatDevData->sat48BitSupport == agTRUE) 7130 { 7131 if (pSatDevData->satDMASupport == agTRUE && pSatDevData->satDMAEnabled == agTRUE) 7132 { 7133 /* case 3 */ 7134 /* WRITE DMA EXT or WRITE DMA FUA EXT */ 7135 SM_DBG5(("smsatWrite10: case 3\n")); 7136 fis->h.fisType = 0x27; /* Reg host to device */ 7137 fis->h.c_pmPort = 0x80; /* C Bit is set */ 7138 7139 /* SAT_WRITE_DMA_FUA_EXT is optional and we don't support it */ 7140 fis->h.command = SAT_WRITE_DMA_EXT; /* 0x35 */ 7141 7142 fis->h.features = 0; /* FIS reserve */ 7143 fis->d.lbaLow = scsiCmnd->cdb[5]; /* FIS LBA (7 :0 ) */ 7144 fis->d.lbaMid = scsiCmnd->cdb[4]; /* FIS LBA (15:8 ) */ 7145 fis->d.lbaHigh = scsiCmnd->cdb[3]; /* FIS LBA (23:16) */ 7146 fis->d.device = 0x40; /* FIS LBA mode set */ 7147 fis->d.lbaLowExp = scsiCmnd->cdb[2]; /* FIS LBA (31:24) */ 7148 fis->d.lbaMidExp = 0; /* FIS LBA (39:32) */ 7149 fis->d.lbaHighExp = 0; /* FIS LBA (47:40) */ 7150 fis->d.featuresExp = 0; /* FIS reserve */ 7151 fis->d.sectorCount = scsiCmnd->cdb[9]; /* FIS sector count (7:0) */ 7152 fis->d.sectorCountExp = scsiCmnd->cdb[8]; /* FIS sector count (15:8) */ 7153 fis->d.reserved4 = 0; 7154 fis->d.control = 0; /* FIS HOB bit clear */ 7155 fis->d.reserved5 = 0; 7156 7157 agRequestType = AGSA_SATA_PROTOCOL_DMA_WRITE; 7158 satIOContext->ATACmd = SAT_WRITE_DMA_EXT; 7159 } 7160 else 7161 { 7162 /* case 4 */ 7163 /* WRITE MULTIPLE EXT or WRITE MULTIPLE FUA EXT or WRITE SECTOR(S) EXT */ 7164 /* WRITE SECTORS EXT for easier implemetation */ 7165 SM_DBG5(("smsatWrite10: case 4\n")); 7166 fis->h.fisType = 0x27; /* Reg host to device */ 7167 fis->h.c_pmPort = 0x80; /* C Bit is set */ 7168 fis->h.command = SAT_WRITE_SECTORS_EXT; /* 0x34 */ 7169 7170 fis->h.features = 0; /* FIS reserve */ 7171 fis->d.lbaLow = scsiCmnd->cdb[5]; /* FIS LBA (7 :0 ) */ 7172 fis->d.lbaMid = scsiCmnd->cdb[4]; /* FIS LBA (15:8 ) */ 7173 fis->d.lbaHigh = scsiCmnd->cdb[3]; /* FIS LBA (23:16) */ 7174 fis->d.device = 0x40; /* FIS LBA mode set */ 7175 fis->d.lbaLowExp = scsiCmnd->cdb[2]; /* FIS LBA (31:24) */ 7176 fis->d.lbaMidExp = 0; /* FIS LBA (39:32) */ 7177 fis->d.lbaHighExp = 0; /* FIS LBA (47:40) */ 7178 fis->d.featuresExp = 0; /* FIS reserve */ 7179 fis->d.sectorCount = scsiCmnd->cdb[9]; /* FIS sector count (7:0) */ 7180 fis->d.sectorCountExp = scsiCmnd->cdb[8]; /* FIS sector count (15:8) */ 7181 fis->d.reserved4 = 0; 7182 fis->d.control = 0; /* FIS HOB bit clear */ 7183 fis->d.reserved5 = 0; 7184 7185 agRequestType = AGSA_SATA_PROTOCOL_PIO_WRITE; 7186 satIOContext->ATACmd = SAT_WRITE_SECTORS_EXT; 7187 } 7188 } 7189 7190 /* case 5 */ 7191 if (pSatDevData->satNCQ == agTRUE) 7192 { 7193 /* WRITE FPDMA QUEUED */ 7194 if (pSatDevData->sat48BitSupport != agTRUE) 7195 { 7196 SM_DBG5(("smsatWrite10: case 5 !!! error NCQ but 28 bit address support!!!\n")); 7197 smsatSetSensePayload( pSense, 7198 SCSI_SNSKEY_ILLEGAL_REQUEST, 7199 0, 7200 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 7201 satIOContext); 7202 7203 /*smEnqueueIO(smRoot, satIOContext);*/ 7204 7205 tdsmIOCompletedCB( smRoot, 7206 smIORequest, 7207 smIOSuccess, 7208 SCSI_STAT_CHECK_CONDITION, 7209 satIOContext->pSmSenseData, 7210 satIOContext->interruptContext ); 7211 return SM_RC_SUCCESS; 7212 } 7213 SM_DBG6(("smsatWrite10: case 5\n")); 7214 7215 /* Support 48-bit FPDMA addressing, use WRITE FPDMA QUEUE command */ 7216 7217 fis->h.fisType = 0x27; /* Reg host to device */ 7218 fis->h.c_pmPort = 0x80; /* C Bit is set */ 7219 fis->h.command = SAT_WRITE_FPDMA_QUEUED; /* 0x61 */ 7220 fis->h.features = scsiCmnd->cdb[9]; /* FIS sector count (7:0) */ 7221 fis->d.lbaLow = scsiCmnd->cdb[5]; /* FIS LBA (7 :0 ) */ 7222 fis->d.lbaMid = scsiCmnd->cdb[4]; /* FIS LBA (15:8 ) */ 7223 fis->d.lbaHigh = scsiCmnd->cdb[3]; /* FIS LBA (23:16) */ 7224 7225 /* Check FUA bit */ 7226 if (scsiCmnd->cdb[1] & SCSI_WRITE12_FUA_MASK) 7227 fis->d.device = 0xC0; /* FIS FUA set */ 7228 else 7229 fis->d.device = 0x40; /* FIS FUA clear */ 7230 7231 fis->d.lbaLowExp = scsiCmnd->cdb[2]; /* FIS LBA (31:24) */ 7232 fis->d.lbaMidExp = 0; /* FIS LBA (39:32) */ 7233 fis->d.lbaHighExp = 0; /* FIS LBA (47:40) */ 7234 fis->d.featuresExp = scsiCmnd->cdb[8]; /* FIS sector count (15:8) */ 7235 fis->d.sectorCount = 0; /* Tag (7:3) set by LL layer */ 7236 fis->d.sectorCountExp = 0; 7237 fis->d.reserved4 = 0; 7238 fis->d.control = 0; /* FIS HOB bit clear */ 7239 fis->d.reserved5 = 0; 7240 7241 agRequestType = AGSA_SATA_PROTOCOL_FPDMA_WRITE; 7242 satIOContext->ATACmd = SAT_WRITE_FPDMA_QUEUED; 7243 } 7244 7245 satIOContext->currentLBA = lba; 7246 satIOContext->OrgTL = tl; 7247 7248 /* 7249 computing number of loop and remainder for tl 7250 0xFF in case not ext 7251 0xFFFF in case EXT 7252 */ 7253 if (fis->h.command == SAT_WRITE_SECTORS || fis->h.command == SAT_WRITE_DMA) 7254 { 7255 LoopNum = smsatComputeLoopNum(tl, 0xFF); 7256 } 7257 else if (fis->h.command == SAT_WRITE_SECTORS_EXT || 7258 fis->h.command == SAT_WRITE_DMA_EXT || 7259 fis->h.command == SAT_WRITE_DMA_FUA_EXT 7260 ) 7261 { 7262 /* SAT_READ_SECTORS_EXT, SAT_READ_DMA_EXT */ 7263 LoopNum = smsatComputeLoopNum(tl, 0xFFFF); 7264 } 7265 else 7266 { 7267 /* SAT_WRITE_FPDMA_QUEUEDK */ 7268 LoopNum = smsatComputeLoopNum(tl, 0xFFFF); 7269 } 7270 7271 satIOContext->LoopNum = LoopNum; 7272 7273 7274 if (LoopNum == 1) 7275 { 7276 SM_DBG5(("smsatWrite10: NON CHAINED data\n")); 7277 /* Initialize CB for SATA completion. 7278 */ 7279 satIOContext->satCompleteCB = &smsatNonChainedDataIOCB; 7280 } 7281 else 7282 { 7283 SM_DBG1(("smsatWrite10: CHAINED data\n")); 7284 /* re-setting tl */ 7285 if (fis->h.command == SAT_WRITE_SECTORS || fis->h.command == SAT_WRITE_DMA) 7286 { 7287 fis->d.sectorCount = 0xFF; 7288 } 7289 else if (fis->h.command == SAT_WRITE_SECTORS_EXT || 7290 fis->h.command == SAT_WRITE_DMA_EXT || 7291 fis->h.command == SAT_WRITE_DMA_FUA_EXT 7292 ) 7293 { 7294 fis->d.sectorCount = 0xFF; 7295 fis->d.sectorCountExp = 0xFF; 7296 } 7297 else 7298 { 7299 /* SAT_WRITE_FPDMA_QUEUED */ 7300 fis->h.features = 0xFF; 7301 fis->d.featuresExp = 0xFF; 7302 } 7303 7304 /* Initialize CB for SATA completion. 7305 */ 7306 satIOContext->satCompleteCB = &smsatChainedDataIOCB; 7307 } 7308 7309 7310 /* 7311 * Prepare SGL and send FIS to LL layer. 7312 */ 7313 satIOContext->reqType = agRequestType; /* Save it */ 7314 7315 status = smsataLLIOStart( smRoot, 7316 smIORequest, 7317 smDeviceHandle, 7318 smScsiRequest, 7319 satIOContext); 7320 return (status); 7321 } 7322 7323 osGLOBAL bit32 7324 smsatWrite16( 7325 smRoot_t *smRoot, 7326 smIORequest_t *smIORequest, 7327 smDeviceHandle_t *smDeviceHandle, 7328 smScsiInitiatorRequest_t *smScsiRequest, 7329 smSatIOContext_t *satIOContext 7330 ) 7331 { 7332 bit32 status; 7333 bit32 agRequestType = AGSA_SATA_PROTOCOL_DMA_WRITE; 7334 smDeviceData_t *pSatDevData; 7335 smScsiRspSense_t *pSense; 7336 smIniScsiCmnd_t *scsiCmnd; 7337 agsaFisRegHostToDevice_t *fis; 7338 bit32 lba = 0; 7339 bit32 tl = 0; 7340 bit32 LoopNum = 1; 7341 bit8 LBA[8]; 7342 bit8 TL[8]; 7343 bit32 AllChk = agFALSE; /* lba, lba+tl check against ATA limit and Disk capacity */ 7344 7345 pSense = satIOContext->pSense; 7346 pSatDevData = satIOContext->pSatDevData; 7347 scsiCmnd = &smScsiRequest->scsiCmnd; 7348 fis = satIOContext->pFis; 7349 7350 SM_DBG5(("smsatWrite16: start\n")); 7351 7352 /* checking FUA_NV */ 7353 if (scsiCmnd->cdb[1] & SCSI_FUA_NV_MASK) 7354 { 7355 smsatSetSensePayload( pSense, 7356 SCSI_SNSKEY_ILLEGAL_REQUEST, 7357 0, 7358 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 7359 satIOContext); 7360 7361 /*smEnqueueIO(smRoot, satIOContext);*/ 7362 7363 tdsmIOCompletedCB( smRoot, 7364 smIORequest, 7365 smIOSuccess, 7366 SCSI_STAT_CHECK_CONDITION, 7367 satIOContext->pSmSenseData, 7368 satIOContext->interruptContext ); 7369 7370 SM_DBG1(("smsatWrite16: return FUA_NV!!!\n")); 7371 return SM_RC_SUCCESS; 7372 7373 } 7374 7375 /* checking CONTROL */ 7376 /* NACA == 1 or LINK == 1*/ 7377 if ( (scsiCmnd->cdb[15] & SCSI_NACA_MASK) || (scsiCmnd->cdb[15] & SCSI_LINK_MASK) ) 7378 { 7379 smsatSetSensePayload( pSense, 7380 SCSI_SNSKEY_ILLEGAL_REQUEST, 7381 0, 7382 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 7383 satIOContext); 7384 7385 /*smEnqueueIO(smRoot, satIOContext);*/ 7386 7387 tdsmIOCompletedCB( smRoot, 7388 smIORequest, 7389 smIOSuccess, 7390 SCSI_STAT_CHECK_CONDITION, 7391 satIOContext->pSmSenseData, 7392 satIOContext->interruptContext ); 7393 7394 SM_DBG1(("smsatWrite16: return control!!!\n")); 7395 return SM_RC_SUCCESS; 7396 } 7397 7398 7399 sm_memset(LBA, 0, sizeof(LBA)); 7400 sm_memset(TL, 0, sizeof(TL)); 7401 7402 7403 /* do not use memcpy due to indexing in LBA and TL */ 7404 LBA[0] = scsiCmnd->cdb[2]; /* MSB */ 7405 LBA[1] = scsiCmnd->cdb[3]; 7406 LBA[2] = scsiCmnd->cdb[4]; 7407 LBA[3] = scsiCmnd->cdb[5]; 7408 LBA[4] = scsiCmnd->cdb[6]; 7409 LBA[5] = scsiCmnd->cdb[7]; 7410 LBA[6] = scsiCmnd->cdb[8]; 7411 LBA[7] = scsiCmnd->cdb[9]; /* LSB */ 7412 7413 TL[0] = 0; 7414 TL[1] = 0; 7415 TL[2] = 0; 7416 TL[3] = 0; 7417 TL[4] = scsiCmnd->cdb[10]; /* MSB */ 7418 TL[5] = scsiCmnd->cdb[11]; 7419 TL[6] = scsiCmnd->cdb[12]; 7420 TL[7] = scsiCmnd->cdb[13]; /* LSB */ 7421 7422 7423 7424 lba = smsatComputeCDB16LBA(satIOContext); 7425 tl = smsatComputeCDB16TL(satIOContext); 7426 7427 7428 7429 /* Table 34, 9.1, p 46 */ 7430 /* 7431 note: As of 2/10/2006, no support for DMA QUEUED 7432 */ 7433 7434 /* 7435 Table 34, 9.1, p 46, b 7436 When no 48-bit addressing support or NCQ, if LBA is beyond (2^28 - 1), 7437 return check condition 7438 */ 7439 if (pSatDevData->satNCQ != agTRUE && 7440 pSatDevData->sat48BitSupport != agTRUE 7441 ) 7442 { 7443 AllChk = smsatCheckLimit(LBA, TL, agFALSE, pSatDevData); 7444 if (AllChk) 7445 { 7446 SM_DBG1(("smsatWrite16: return LBA out of range, not EXT!!!\n")); 7447 smsatSetSensePayload( pSense, 7448 SCSI_SNSKEY_ILLEGAL_REQUEST, 7449 0, 7450 SCSI_SNSCODE_LOGICAL_BLOCK_ADDRESS_OUT_OF_RANGE, 7451 satIOContext); 7452 7453 /*smEnqueueIO(smRoot, satIOContext);*/ 7454 7455 tdsmIOCompletedCB( smRoot, 7456 smIORequest, 7457 smIOSuccess, 7458 SCSI_STAT_CHECK_CONDITION, 7459 satIOContext->pSmSenseData, 7460 satIOContext->interruptContext ); 7461 7462 return SM_RC_SUCCESS; 7463 } 7464 } 7465 else 7466 { 7467 AllChk = smsatCheckLimit(LBA, TL, agTRUE, pSatDevData); 7468 if (AllChk) 7469 { 7470 SM_DBG1(("smsatWrite16: return LBA out of range, EXT!!!\n")); 7471 smsatSetSensePayload( pSense, 7472 SCSI_SNSKEY_ILLEGAL_REQUEST, 7473 0, 7474 SCSI_SNSCODE_LOGICAL_BLOCK_ADDRESS_OUT_OF_RANGE, 7475 satIOContext); 7476 7477 /*smEnqueueIO(smRoot, satIOContext);*/ 7478 7479 tdsmIOCompletedCB( smRoot, 7480 smIORequest, 7481 smIOSuccess, 7482 SCSI_STAT_CHECK_CONDITION, 7483 satIOContext->pSmSenseData, 7484 satIOContext->interruptContext ); 7485 7486 return SM_RC_SUCCESS; 7487 } 7488 } 7489 7490 /* case 1 and 2 */ 7491 if (pSatDevData->satDMASupport == agTRUE && pSatDevData->satDMAEnabled == agTRUE) 7492 { 7493 /* case 2 */ 7494 /* WRITE DMA*/ 7495 /* In case that we can't fit the transfer length, we loop */ 7496 SM_DBG5(("smsatWrite16: case 2\n")); 7497 fis->h.fisType = 0x27; /* Reg host to device */ 7498 fis->h.c_pmPort = 0x80; /* C bit is set */ 7499 fis->h.command = SAT_WRITE_DMA; /* 0xCA */ 7500 fis->h.features = 0; /* FIS reserve */ 7501 fis->d.lbaLow = scsiCmnd->cdb[9]; /* FIS LBA (7 :0 ) */ 7502 fis->d.lbaMid = scsiCmnd->cdb[8]; /* FIS LBA (15:8 ) */ 7503 fis->d.lbaHigh = scsiCmnd->cdb[7]; /* FIS LBA (23:16) */ 7504 7505 /* FIS LBA mode set LBA (27:24) */ 7506 fis->d.device = (bit8)((0x4 << 4) | (scsiCmnd->cdb[6] & 0xF)); 7507 7508 fis->d.lbaLowExp = 0; 7509 fis->d.lbaMidExp = 0; 7510 fis->d.lbaHighExp = 0; 7511 fis->d.featuresExp = 0; 7512 fis->d.sectorCount = scsiCmnd->cdb[13]; /* FIS sector count (7:0) */ 7513 fis->d.sectorCountExp = 0; 7514 fis->d.reserved4 = 0; 7515 fis->d.control = 0; /* FIS HOB bit clear */ 7516 fis->d.reserved5 = 0; 7517 7518 agRequestType = AGSA_SATA_PROTOCOL_DMA_WRITE; 7519 satIOContext->ATACmd = SAT_WRITE_DMA; 7520 } 7521 else 7522 { 7523 /* case 1 */ 7524 /* WRITE MULTIPLE or WRITE SECTOR(S) */ 7525 /* WRITE SECTORS for easier implemetation */ 7526 /* In case that we can't fit the transfer length, we loop */ 7527 SM_DBG5(("smsatWrite16: case 1\n")); 7528 fis->h.fisType = 0x27; /* Reg host to device */ 7529 fis->h.c_pmPort = 0x80; /* C bit is set */ 7530 fis->h.command = SAT_WRITE_SECTORS; /* 0x30 */ 7531 fis->h.features = 0; /* FIS reserve */ 7532 fis->d.lbaLow = scsiCmnd->cdb[9]; /* FIS LBA (7 :0 ) */ 7533 fis->d.lbaMid = scsiCmnd->cdb[8]; /* FIS LBA (15:8 ) */ 7534 fis->d.lbaHigh = scsiCmnd->cdb[7]; /* FIS LBA (23:16) */ 7535 7536 /* FIS LBA mode set LBA (27:24) */ 7537 fis->d.device = (bit8)((0x4 << 4) | (scsiCmnd->cdb[6] & 0xF)); 7538 7539 fis->d.lbaLowExp = 0; 7540 fis->d.lbaMidExp = 0; 7541 fis->d.lbaHighExp = 0; 7542 fis->d.featuresExp = 0; 7543 fis->d.sectorCount = scsiCmnd->cdb[13]; /* FIS sector count (7:0) */ 7544 fis->d.sectorCountExp = 0; 7545 fis->d.reserved4 = 0; 7546 fis->d.control = 0; /* FIS HOB bit clear */ 7547 fis->d.reserved5 = 0; 7548 7549 agRequestType = AGSA_SATA_PROTOCOL_PIO_WRITE; 7550 satIOContext->ATACmd = SAT_WRITE_SECTORS; 7551 } 7552 7553 /* case 3 and 4 */ 7554 if (pSatDevData->sat48BitSupport == agTRUE) 7555 { 7556 if (pSatDevData->satDMASupport == agTRUE && pSatDevData->satDMAEnabled == agTRUE) 7557 { 7558 /* case 3 */ 7559 /* WRITE DMA EXT or WRITE DMA FUA EXT */ 7560 SM_DBG5(("smsatWrite16: case 3\n")); 7561 fis->h.fisType = 0x27; /* Reg host to device */ 7562 fis->h.c_pmPort = 0x80; /* C Bit is set */ 7563 7564 /* SAT_WRITE_DMA_FUA_EXT is optional and we don't support it */ 7565 fis->h.command = SAT_WRITE_DMA_EXT; /* 0x35 */ 7566 7567 fis->h.features = 0; /* FIS reserve */ 7568 fis->d.lbaLow = scsiCmnd->cdb[9]; /* FIS LBA (7 :0 ) */ 7569 fis->d.lbaMid = scsiCmnd->cdb[8]; /* FIS LBA (15:8 ) */ 7570 fis->d.lbaHigh = scsiCmnd->cdb[7]; /* FIS LBA (23:16) */ 7571 fis->d.device = 0x40; /* FIS LBA mode set */ 7572 fis->d.lbaLowExp = scsiCmnd->cdb[6]; /* FIS LBA (31:24) */ 7573 fis->d.lbaMidExp = scsiCmnd->cdb[5]; /* FIS LBA (39:32) */ 7574 fis->d.lbaHighExp = scsiCmnd->cdb[4]; /* FIS LBA (47:40) */ 7575 fis->d.featuresExp = 0; /* FIS reserve */ 7576 fis->d.sectorCount = scsiCmnd->cdb[13]; /* FIS sector count (7:0) */ 7577 fis->d.sectorCountExp = scsiCmnd->cdb[12]; /* FIS sector count (15:8) */ 7578 fis->d.reserved4 = 0; 7579 fis->d.control = 0; /* FIS HOB bit clear */ 7580 fis->d.reserved5 = 0; 7581 7582 agRequestType = AGSA_SATA_PROTOCOL_DMA_WRITE; 7583 satIOContext->ATACmd = SAT_WRITE_DMA_EXT; 7584 } 7585 else 7586 { 7587 /* case 4 */ 7588 /* WRITE MULTIPLE EXT or WRITE MULTIPLE FUA EXT or WRITE SECTOR(S) EXT */ 7589 /* WRITE SECTORS EXT for easier implemetation */ 7590 SM_DBG5(("smsatWrite16: case 4\n")); 7591 fis->h.fisType = 0x27; /* Reg host to device */ 7592 fis->h.c_pmPort = 0x80; /* C Bit is set */ 7593 fis->h.command = SAT_WRITE_SECTORS_EXT; /* 0x34 */ 7594 7595 fis->h.features = 0; /* FIS reserve */ 7596 fis->d.lbaLow = scsiCmnd->cdb[9]; /* FIS LBA (7 :0 ) */ 7597 fis->d.lbaMid = scsiCmnd->cdb[8]; /* FIS LBA (15:8 ) */ 7598 fis->d.lbaHigh = scsiCmnd->cdb[7]; /* FIS LBA (23:16) */ 7599 fis->d.device = 0x40; /* FIS LBA mode set */ 7600 fis->d.lbaLowExp = scsiCmnd->cdb[6]; /* FIS LBA (31:24) */ 7601 fis->d.lbaMidExp = scsiCmnd->cdb[5]; /* FIS LBA (39:32) */ 7602 fis->d.lbaHighExp = scsiCmnd->cdb[4]; /* FIS LBA (47:40) */ 7603 fis->d.featuresExp = 0; /* FIS reserve */ 7604 fis->d.sectorCount = scsiCmnd->cdb[13]; /* FIS sector count (7:0) */ 7605 fis->d.sectorCountExp = scsiCmnd->cdb[12]; /* FIS sector count (15:8) */ 7606 fis->d.reserved4 = 0; 7607 fis->d.control = 0; /* FIS HOB bit clear */ 7608 fis->d.reserved5 = 0; 7609 7610 agRequestType = AGSA_SATA_PROTOCOL_PIO_WRITE; 7611 satIOContext->ATACmd = SAT_WRITE_SECTORS_EXT; 7612 } 7613 } 7614 7615 /* case 5 */ 7616 if (pSatDevData->satNCQ == agTRUE) 7617 { 7618 /* WRITE FPDMA QUEUED */ 7619 if (pSatDevData->sat48BitSupport != agTRUE) 7620 { 7621 SM_DBG5(("smsatWrite16: case 5 !!! error NCQ but 28 bit address support!!!\n")); 7622 smsatSetSensePayload( pSense, 7623 SCSI_SNSKEY_ILLEGAL_REQUEST, 7624 0, 7625 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 7626 satIOContext); 7627 7628 /*smEnqueueIO(smRoot, satIOContext);*/ 7629 7630 tdsmIOCompletedCB( smRoot, 7631 smIORequest, 7632 smIOSuccess, 7633 SCSI_STAT_CHECK_CONDITION, 7634 satIOContext->pSmSenseData, 7635 satIOContext->interruptContext ); 7636 return SM_RC_SUCCESS; 7637 } 7638 SM_DBG6(("smsatWrite16: case 5\n")); 7639 7640 /* Support 48-bit FPDMA addressing, use WRITE FPDMA QUEUE command */ 7641 7642 fis->h.fisType = 0x27; /* Reg host to device */ 7643 fis->h.c_pmPort = 0x80; /* C Bit is set */ 7644 fis->h.command = SAT_WRITE_FPDMA_QUEUED; /* 0x61 */ 7645 fis->h.features = scsiCmnd->cdb[13]; /* FIS sector count (7:0) */ 7646 fis->d.lbaLow = scsiCmnd->cdb[9]; /* FIS LBA (7 :0 ) */ 7647 fis->d.lbaMid = scsiCmnd->cdb[8]; /* FIS LBA (15:8 ) */ 7648 fis->d.lbaHigh = scsiCmnd->cdb[7]; /* FIS LBA (23:16) */ 7649 7650 /* Check FUA bit */ 7651 if (scsiCmnd->cdb[1] & SCSI_WRITE16_FUA_MASK) 7652 fis->d.device = 0xC0; /* FIS FUA set */ 7653 else 7654 fis->d.device = 0x40; /* FIS FUA clear */ 7655 7656 fis->d.lbaLowExp = scsiCmnd->cdb[6]; /* FIS LBA (31:24) */ 7657 fis->d.lbaMidExp = scsiCmnd->cdb[5]; /* FIS LBA (39:32) */ 7658 fis->d.lbaHighExp = scsiCmnd->cdb[4]; /* FIS LBA (47:40) */ 7659 fis->d.featuresExp = scsiCmnd->cdb[12]; /* FIS sector count (15:8) */ 7660 fis->d.sectorCount = 0; /* Tag (7:3) set by LL layer */ 7661 fis->d.sectorCountExp = 0; 7662 fis->d.reserved4 = 0; 7663 fis->d.control = 0; /* FIS HOB bit clear */ 7664 fis->d.reserved5 = 0; 7665 7666 agRequestType = AGSA_SATA_PROTOCOL_FPDMA_WRITE; 7667 satIOContext->ATACmd = SAT_WRITE_FPDMA_QUEUED; 7668 } 7669 7670 satIOContext->currentLBA = lba; 7671 satIOContext->OrgTL = tl; 7672 7673 /* 7674 computing number of loop and remainder for tl 7675 0xFF in case not ext 7676 0xFFFF in case EXT 7677 */ 7678 if (fis->h.command == SAT_WRITE_SECTORS || fis->h.command == SAT_WRITE_DMA) 7679 { 7680 LoopNum = smsatComputeLoopNum(tl, 0xFF); 7681 } 7682 else if (fis->h.command == SAT_WRITE_SECTORS_EXT || 7683 fis->h.command == SAT_WRITE_DMA_EXT || 7684 fis->h.command == SAT_WRITE_DMA_FUA_EXT 7685 ) 7686 { 7687 /* SAT_READ_SECTORS_EXT, SAT_READ_DMA_EXT */ 7688 LoopNum = smsatComputeLoopNum(tl, 0xFFFF); 7689 } 7690 else 7691 { 7692 /* SAT_WRITE_FPDMA_QUEUEDK */ 7693 LoopNum = smsatComputeLoopNum(tl, 0xFFFF); 7694 } 7695 7696 satIOContext->LoopNum = LoopNum; 7697 7698 7699 if (LoopNum == 1) 7700 { 7701 SM_DBG5(("smsatWrite16: NON CHAINED data\n")); 7702 /* Initialize CB for SATA completion. 7703 */ 7704 satIOContext->satCompleteCB = &smsatNonChainedDataIOCB; 7705 } 7706 else 7707 { 7708 SM_DBG1(("smsatWrite16: CHAINED data!!!\n")); 7709 /* re-setting tl */ 7710 if (fis->h.command == SAT_WRITE_SECTORS || fis->h.command == SAT_WRITE_DMA) 7711 { 7712 fis->d.sectorCount = 0xFF; 7713 } 7714 else if (fis->h.command == SAT_WRITE_SECTORS_EXT || 7715 fis->h.command == SAT_WRITE_DMA_EXT || 7716 fis->h.command == SAT_WRITE_DMA_FUA_EXT 7717 ) 7718 { 7719 fis->d.sectorCount = 0xFF; 7720 fis->d.sectorCountExp = 0xFF; 7721 } 7722 else 7723 { 7724 /* SAT_WRITE_FPDMA_QUEUED */ 7725 fis->h.features = 0xFF; 7726 fis->d.featuresExp = 0xFF; 7727 } 7728 7729 /* Initialize CB for SATA completion. 7730 */ 7731 satIOContext->satCompleteCB = &smsatChainedDataIOCB; 7732 } 7733 7734 7735 /* 7736 * Prepare SGL and send FIS to LL layer. 7737 */ 7738 satIOContext->reqType = agRequestType; /* Save it */ 7739 7740 status = smsataLLIOStart( smRoot, 7741 smIORequest, 7742 smDeviceHandle, 7743 smScsiRequest, 7744 satIOContext); 7745 return (status); 7746 } 7747 7748 7749 osGLOBAL bit32 7750 smsatVerify10( 7751 smRoot_t *smRoot, 7752 smIORequest_t *smIORequest, 7753 smDeviceHandle_t *smDeviceHandle, 7754 smScsiInitiatorRequest_t *smScsiRequest, 7755 smSatIOContext_t *satIOContext 7756 ) 7757 { 7758 /* 7759 For simple implementation, 7760 no byte comparison supported as of 4/5/06 7761 */ 7762 smScsiRspSense_t *pSense; 7763 smIniScsiCmnd_t *scsiCmnd; 7764 smDeviceData_t *pSatDevData; 7765 agsaFisRegHostToDevice_t *fis; 7766 bit32 status; 7767 bit32 agRequestType = AGSA_SATA_PROTOCOL_NON_DATA; 7768 bit32 lba = 0; 7769 bit32 tl = 0; 7770 bit32 LoopNum = 1; 7771 bit8 LBA[8]; 7772 bit8 TL[8]; 7773 bit32 AllChk = agFALSE; /* lba, lba+tl check against ATA limit and Disk capacity */ 7774 7775 pSense = satIOContext->pSense; 7776 scsiCmnd = &smScsiRequest->scsiCmnd; 7777 pSatDevData = satIOContext->pSatDevData; 7778 fis = satIOContext->pFis; 7779 SM_DBG5(("smsatVerify10: start\n")); 7780 /* checking BYTCHK */ 7781 if (scsiCmnd->cdb[1] & SCSI_VERIFY_BYTCHK_MASK) 7782 { 7783 /* 7784 should do the byte check 7785 but not supported in this version 7786 */ 7787 smsatSetSensePayload( pSense, 7788 SCSI_SNSKEY_ILLEGAL_REQUEST, 7789 0, 7790 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 7791 satIOContext); 7792 /*smEnqueueIO(smRoot, satIOContext);*/ 7793 tdsmIOCompletedCB( smRoot, 7794 smIORequest, 7795 smIOSuccess, 7796 SCSI_STAT_CHECK_CONDITION, 7797 satIOContext->pSmSenseData, 7798 satIOContext->interruptContext ); 7799 7800 SM_DBG1(("smsatVerify10: no byte checking!!!\n")); 7801 return SM_RC_SUCCESS; 7802 } 7803 7804 /* checking CONTROL */ 7805 /* NACA == 1 or LINK == 1*/ 7806 if ( (scsiCmnd->cdb[9] & SCSI_NACA_MASK) || (scsiCmnd->cdb[9] & SCSI_LINK_MASK) ) 7807 { 7808 smsatSetSensePayload( pSense, 7809 SCSI_SNSKEY_ILLEGAL_REQUEST, 7810 0, 7811 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 7812 satIOContext); 7813 7814 /*smEnqueueIO(smRoot, satIOContext);*/ 7815 7816 tdsmIOCompletedCB( smRoot, 7817 smIORequest, 7818 smIOSuccess, 7819 SCSI_STAT_CHECK_CONDITION, 7820 satIOContext->pSmSenseData, 7821 satIOContext->interruptContext ); 7822 7823 SM_DBG1(("smsatVerify10: return control!!!\n")); 7824 return SM_RC_SUCCESS; 7825 } 7826 7827 7828 sm_memset(LBA, 0, sizeof(LBA)); 7829 sm_memset(TL, 0, sizeof(TL)); 7830 7831 /* do not use memcpy due to indexing in LBA and TL */ 7832 LBA[0] = 0; /* MSB */ 7833 LBA[1] = 0; 7834 LBA[2] = 0; 7835 LBA[3] = 0; 7836 LBA[4] = scsiCmnd->cdb[2]; 7837 LBA[5] = scsiCmnd->cdb[3]; 7838 LBA[6] = scsiCmnd->cdb[4]; 7839 LBA[7] = scsiCmnd->cdb[5]; /* LSB */ 7840 7841 TL[0] = 0; 7842 TL[1] = 0; 7843 TL[2] = 0; 7844 TL[3] = 0; 7845 TL[4] = 0; 7846 TL[5] = 0; 7847 TL[6] = scsiCmnd->cdb[7]; 7848 TL[7] = scsiCmnd->cdb[8]; /* LSB */ 7849 7850 7851 /* cbd10; computing LBA and transfer length */ 7852 lba = (scsiCmnd->cdb[2] << (8*3)) + (scsiCmnd->cdb[3] << (8*2)) 7853 + (scsiCmnd->cdb[4] << 8) + scsiCmnd->cdb[5]; 7854 tl = (scsiCmnd->cdb[7] << 8) + scsiCmnd->cdb[8]; 7855 7856 if (pSatDevData->satNCQ != agTRUE && 7857 pSatDevData->sat48BitSupport != agTRUE 7858 ) 7859 { 7860 AllChk = smsatCheckLimit(LBA, TL, agFALSE, pSatDevData); 7861 if (AllChk) 7862 { 7863 SM_DBG1(("smsatVerify10: return LBA out of range, not EXT!!!\n")); 7864 SM_DBG1(("smsatVerify10: cdb 0x%x 0x%x 0x%x 0x%x!!!\n",scsiCmnd->cdb[2], scsiCmnd->cdb[3], 7865 scsiCmnd->cdb[4], scsiCmnd->cdb[5])); 7866 SM_DBG1(("smsatVerify10: lba 0x%x SAT_TR_LBA_LIMIT 0x%x!!!\n", lba, SAT_TR_LBA_LIMIT)); 7867 smsatSetSensePayload( pSense, 7868 SCSI_SNSKEY_ILLEGAL_REQUEST, 7869 0, 7870 SCSI_SNSCODE_LOGICAL_BLOCK_ADDRESS_OUT_OF_RANGE, 7871 satIOContext); 7872 7873 /*smEnqueueIO(smRoot, satIOContext);*/ 7874 7875 tdsmIOCompletedCB( smRoot, 7876 smIORequest, 7877 smIOSuccess, 7878 SCSI_STAT_CHECK_CONDITION, 7879 satIOContext->pSmSenseData, 7880 satIOContext->interruptContext ); 7881 7882 return SM_RC_SUCCESS; 7883 } 7884 } 7885 else 7886 { 7887 AllChk = smsatCheckLimit(LBA, TL, agTRUE, pSatDevData); 7888 if (AllChk) 7889 { 7890 SM_DBG1(("smsatVerify10: return LBA out of range, EXT!!!\n")); 7891 smsatSetSensePayload( pSense, 7892 SCSI_SNSKEY_ILLEGAL_REQUEST, 7893 0, 7894 SCSI_SNSCODE_LOGICAL_BLOCK_ADDRESS_OUT_OF_RANGE, 7895 satIOContext); 7896 7897 /*smEnqueueIO(smRoot, satIOContext);*/ 7898 7899 tdsmIOCompletedCB( smRoot, 7900 smIORequest, 7901 smIOSuccess, 7902 SCSI_STAT_CHECK_CONDITION, 7903 satIOContext->pSmSenseData, 7904 satIOContext->interruptContext ); 7905 7906 return SM_RC_SUCCESS; 7907 } 7908 } 7909 7910 if (pSatDevData->sat48BitSupport == agTRUE) 7911 { 7912 SM_DBG5(("smsatVerify10: SAT_READ_VERIFY_SECTORS_EXT\n")); 7913 fis->h.fisType = 0x27; /* Reg host to device */ 7914 fis->h.c_pmPort = 0x80; /* C Bit is set */ 7915 7916 fis->h.command = SAT_READ_VERIFY_SECTORS_EXT;/* 0x42 */ 7917 fis->h.features = 0; /* FIS reserve */ 7918 fis->d.lbaLow = scsiCmnd->cdb[5]; /* FIS LBA (7 :0 ) */ 7919 fis->d.lbaMid = scsiCmnd->cdb[4]; /* FIS LBA (15:8 ) */ 7920 fis->d.lbaHigh = scsiCmnd->cdb[3]; /* FIS LBA (23:16) */ 7921 fis->d.device = 0x40; /* FIS LBA mode set 01000000 */ 7922 fis->d.lbaLowExp = scsiCmnd->cdb[2]; /* FIS LBA (31:24) */ 7923 fis->d.lbaMidExp = 0; /* FIS LBA (39:32) */ 7924 fis->d.lbaHighExp = 0; /* FIS LBA (47:40) */ 7925 fis->d.featuresExp = 0; /* FIS reserve */ 7926 fis->d.sectorCount = scsiCmnd->cdb[8]; /* FIS sector count (7:0) */ 7927 fis->d.sectorCountExp = scsiCmnd->cdb[7]; /* FIS sector count (15:8) */ 7928 7929 fis->d.reserved4 = 0; 7930 fis->d.control = 0; /* FIS HOB bit clear */ 7931 fis->d.reserved5 = 0; 7932 7933 agRequestType = AGSA_SATA_PROTOCOL_NON_DATA; 7934 satIOContext->ATACmd = SAT_READ_VERIFY_SECTORS_EXT; 7935 } 7936 else 7937 { 7938 SM_DBG5(("smsatVerify10: SAT_READ_VERIFY_SECTORS\n")); 7939 fis->h.fisType = 0x27; /* Reg host to device */ 7940 fis->h.c_pmPort = 0x80; /* C bit is set */ 7941 fis->h.command = SAT_READ_VERIFY_SECTORS; /* 0x40 */ 7942 fis->h.features = 0; /* FIS reserve */ 7943 fis->d.lbaLow = scsiCmnd->cdb[5]; /* FIS LBA (7 :0 ) */ 7944 fis->d.lbaMid = scsiCmnd->cdb[4]; /* FIS LBA (15:8 ) */ 7945 fis->d.lbaHigh = scsiCmnd->cdb[3]; /* FIS LBA (23:16) */ 7946 /* FIS LBA mode set LBA (27:24) */ 7947 fis->d.device = (bit8)((0x4 << 4) | (scsiCmnd->cdb[2] & 0xF)); 7948 fis->d.lbaLowExp = 0; 7949 fis->d.lbaMidExp = 0; 7950 fis->d.lbaHighExp = 0; 7951 fis->d.featuresExp = 0; 7952 fis->d.sectorCount = scsiCmnd->cdb[8]; /* FIS sector count (7:0) */ 7953 fis->d.sectorCountExp = 0; 7954 fis->d.reserved4 = 0; 7955 fis->d.control = 0; /* FIS HOB bit clear */ 7956 fis->d.reserved5 = 0; 7957 7958 agRequestType = AGSA_SATA_PROTOCOL_NON_DATA; 7959 satIOContext->ATACmd = SAT_READ_VERIFY_SECTORS; 7960 7961 } 7962 7963 satIOContext->currentLBA = lba; 7964 satIOContext->OrgTL = tl; 7965 7966 /* 7967 computing number of loop and remainder for tl 7968 0xFF in case not ext 7969 0xFFFF in case EXT 7970 */ 7971 if (fis->h.command == SAT_READ_VERIFY_SECTORS) 7972 { 7973 LoopNum = smsatComputeLoopNum(tl, 0xFF); 7974 } 7975 else if (fis->h.command == SAT_READ_VERIFY_SECTORS_EXT) 7976 { 7977 /* SAT_READ_SECTORS_EXT, SAT_READ_DMA_EXT */ 7978 LoopNum = smsatComputeLoopNum(tl, 0xFFFF); 7979 } 7980 else 7981 { 7982 SM_DBG1(("smsatVerify10: error case 1!!!\n")); 7983 LoopNum = 1; 7984 } 7985 7986 satIOContext->LoopNum = LoopNum; 7987 7988 if (LoopNum == 1) 7989 { 7990 SM_DBG5(("smsatVerify10: NON CHAINED data\n")); 7991 /* Initialize CB for SATA completion. 7992 */ 7993 satIOContext->satCompleteCB = &smsatNonChainedVerifyCB; 7994 } 7995 else 7996 { 7997 SM_DBG1(("smsatVerify10: CHAINED data!!!\n")); 7998 /* re-setting tl */ 7999 if (fis->h.command == SAT_READ_VERIFY_SECTORS) 8000 { 8001 fis->d.sectorCount = 0xFF; 8002 } 8003 else if (fis->h.command == SAT_READ_VERIFY_SECTORS_EXT) 8004 { 8005 fis->d.sectorCount = 0xFF; 8006 fis->d.sectorCountExp = 0xFF; 8007 } 8008 else 8009 { 8010 SM_DBG1(("smsatVerify10: error case 2!!!\n")); 8011 } 8012 8013 /* Initialize CB for SATA completion. 8014 */ 8015 satIOContext->satCompleteCB = &smsatChainedVerifyCB; 8016 } 8017 8018 8019 /* 8020 * Prepare SGL and send FIS to LL layer. 8021 */ 8022 satIOContext->reqType = agRequestType; /* Save it */ 8023 8024 status = smsataLLIOStart( smRoot, 8025 smIORequest, 8026 smDeviceHandle, 8027 smScsiRequest, 8028 satIOContext); 8029 return (status); 8030 } 8031 8032 osGLOBAL bit32 8033 smsatVerify12( 8034 smRoot_t *smRoot, 8035 smIORequest_t *smIORequest, 8036 smDeviceHandle_t *smDeviceHandle, 8037 smScsiInitiatorRequest_t *smScsiRequest, 8038 smSatIOContext_t *satIOContext 8039 ) 8040 { 8041 /* 8042 For simple implementation, 8043 no byte comparison supported as of 4/5/06 8044 */ 8045 smScsiRspSense_t *pSense; 8046 smIniScsiCmnd_t *scsiCmnd; 8047 smDeviceData_t *pSatDevData; 8048 agsaFisRegHostToDevice_t *fis; 8049 bit32 status; 8050 bit32 agRequestType = AGSA_SATA_PROTOCOL_NON_DATA; 8051 bit32 lba = 0; 8052 bit32 tl = 0; 8053 bit32 LoopNum = 1; 8054 bit8 LBA[8]; 8055 bit8 TL[8]; 8056 bit32 AllChk = agFALSE; /* lba, lba+tl check against ATA limit and Disk capacity */ 8057 8058 pSense = satIOContext->pSense; 8059 scsiCmnd = &smScsiRequest->scsiCmnd; 8060 pSatDevData = satIOContext->pSatDevData; 8061 fis = satIOContext->pFis; 8062 SM_DBG5(("smsatVerify12: start\n")); 8063 /* checking BYTCHK */ 8064 if (scsiCmnd->cdb[1] & SCSI_VERIFY_BYTCHK_MASK) 8065 { 8066 /* 8067 should do the byte check 8068 but not supported in this version 8069 */ 8070 smsatSetSensePayload( pSense, 8071 SCSI_SNSKEY_ILLEGAL_REQUEST, 8072 0, 8073 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 8074 satIOContext); 8075 8076 /*smEnqueueIO(smRoot, satIOContext);*/ 8077 8078 tdsmIOCompletedCB( smRoot, 8079 smIORequest, 8080 smIOSuccess, 8081 SCSI_STAT_CHECK_CONDITION, 8082 satIOContext->pSmSenseData, 8083 satIOContext->interruptContext ); 8084 8085 SM_DBG1(("smsatVerify12: no byte checking!!!\n")); 8086 return SM_RC_SUCCESS; 8087 } 8088 8089 /* checking CONTROL */ 8090 /* NACA == 1 or LINK == 1*/ 8091 if ( (scsiCmnd->cdb[11] & SCSI_NACA_MASK) || (scsiCmnd->cdb[11] & SCSI_LINK_MASK) ) 8092 { 8093 smsatSetSensePayload( pSense, 8094 SCSI_SNSKEY_ILLEGAL_REQUEST, 8095 0, 8096 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 8097 satIOContext); 8098 8099 /*smEnqueueIO(smRoot, satIOContext);*/ 8100 8101 tdsmIOCompletedCB( smRoot, 8102 smIORequest, 8103 smIOSuccess, 8104 SCSI_STAT_CHECK_CONDITION, 8105 satIOContext->pSmSenseData, 8106 satIOContext->interruptContext ); 8107 8108 SM_DBG1(("smsatVerify12: return control!!!\n")); 8109 return SM_RC_SUCCESS; 8110 } 8111 8112 sm_memset(LBA, 0, sizeof(LBA)); 8113 sm_memset(TL, 0, sizeof(TL)); 8114 8115 /* do not use memcpy due to indexing in LBA and TL */ 8116 LBA[0] = 0; /* MSB */ 8117 LBA[1] = 0; 8118 LBA[2] = 0; 8119 LBA[3] = 0; 8120 LBA[4] = scsiCmnd->cdb[2]; 8121 LBA[5] = scsiCmnd->cdb[3]; 8122 LBA[6] = scsiCmnd->cdb[4]; 8123 LBA[7] = scsiCmnd->cdb[5]; /* LSB */ 8124 8125 TL[0] = 0; /* MSB */ 8126 TL[1] = 0; 8127 TL[2] = 0; 8128 TL[3] = 0; 8129 TL[4] = scsiCmnd->cdb[6]; 8130 TL[5] = scsiCmnd->cdb[7]; 8131 TL[6] = scsiCmnd->cdb[8]; 8132 TL[7] = scsiCmnd->cdb[9]; /* LSB */ 8133 8134 8135 lba = smsatComputeCDB12LBA(satIOContext); 8136 tl = smsatComputeCDB12TL(satIOContext); 8137 8138 if (pSatDevData->satNCQ != agTRUE && 8139 pSatDevData->sat48BitSupport != agTRUE 8140 ) 8141 { 8142 AllChk = smsatCheckLimit(LBA, TL, agFALSE, pSatDevData); 8143 if (AllChk) 8144 { 8145 SM_DBG1(("smsatVerify12: return LBA out of range, not EXT!!!\n")); 8146 SM_DBG1(("smsatVerify12: cdb 0x%x 0x%x 0x%x 0x%x!!!\n",scsiCmnd->cdb[2], scsiCmnd->cdb[3], 8147 scsiCmnd->cdb[4], scsiCmnd->cdb[5])); 8148 SM_DBG1(("smsatVerify12: lba 0x%x SAT_TR_LBA_LIMIT 0x%x!!!\n", lba, SAT_TR_LBA_LIMIT)); 8149 smsatSetSensePayload( pSense, 8150 SCSI_SNSKEY_ILLEGAL_REQUEST, 8151 0, 8152 SCSI_SNSCODE_LOGICAL_BLOCK_ADDRESS_OUT_OF_RANGE, 8153 satIOContext); 8154 8155 /*smEnqueueIO(smRoot, satIOContext);*/ 8156 8157 tdsmIOCompletedCB( smRoot, 8158 smIORequest, 8159 smIOSuccess, 8160 SCSI_STAT_CHECK_CONDITION, 8161 satIOContext->pSmSenseData, 8162 satIOContext->interruptContext ); 8163 8164 return SM_RC_SUCCESS; 8165 } 8166 } 8167 else 8168 { 8169 AllChk = smsatCheckLimit(LBA, TL, agTRUE, pSatDevData); 8170 if (AllChk) 8171 { 8172 SM_DBG1(("smsatVerify12: return LBA out of range, EXT!!!\n")); 8173 smsatSetSensePayload( pSense, 8174 SCSI_SNSKEY_ILLEGAL_REQUEST, 8175 0, 8176 SCSI_SNSCODE_LOGICAL_BLOCK_ADDRESS_OUT_OF_RANGE, 8177 satIOContext); 8178 8179 /*smEnqueueIO(smRoot, satIOContext);*/ 8180 8181 tdsmIOCompletedCB( smRoot, 8182 smIORequest, 8183 smIOSuccess, 8184 SCSI_STAT_CHECK_CONDITION, 8185 satIOContext->pSmSenseData, 8186 satIOContext->interruptContext ); 8187 8188 return SM_RC_SUCCESS; 8189 } 8190 } 8191 8192 if (pSatDevData->sat48BitSupport == agTRUE) 8193 { 8194 SM_DBG5(("smsatVerify12: SAT_READ_VERIFY_SECTORS_EXT\n")); 8195 fis->h.fisType = 0x27; /* Reg host to device */ 8196 fis->h.c_pmPort = 0x80; /* C Bit is set */ 8197 8198 fis->h.command = SAT_READ_VERIFY_SECTORS_EXT;/* 0x42 */ 8199 fis->h.features = 0; /* FIS reserve */ 8200 fis->d.lbaLow = scsiCmnd->cdb[5]; /* FIS LBA (7 :0 ) */ 8201 fis->d.lbaMid = scsiCmnd->cdb[4]; /* FIS LBA (15:8 ) */ 8202 fis->d.lbaHigh = scsiCmnd->cdb[3]; /* FIS LBA (23:16) */ 8203 fis->d.device = 0x40; /* FIS LBA mode set 01000000 */ 8204 fis->d.lbaLowExp = scsiCmnd->cdb[2]; /* FIS LBA (31:24) */ 8205 fis->d.lbaMidExp = 0; /* FIS LBA (39:32) */ 8206 fis->d.lbaHighExp = 0; /* FIS LBA (47:40) */ 8207 fis->d.featuresExp = 0; /* FIS reserve */ 8208 fis->d.sectorCount = scsiCmnd->cdb[9]; /* FIS sector count (7:0) */ 8209 fis->d.sectorCountExp = scsiCmnd->cdb[8]; /* FIS sector count (15:8) */ 8210 8211 fis->d.reserved4 = 0; 8212 fis->d.control = 0; /* FIS HOB bit clear */ 8213 fis->d.reserved5 = 0; 8214 8215 agRequestType = AGSA_SATA_PROTOCOL_NON_DATA; 8216 satIOContext->ATACmd = SAT_READ_VERIFY_SECTORS_EXT; 8217 } 8218 else 8219 { 8220 SM_DBG5(("smsatVerify12: SAT_READ_VERIFY_SECTORS\n")); 8221 fis->h.fisType = 0x27; /* Reg host to device */ 8222 fis->h.c_pmPort = 0x80; /* C bit is set */ 8223 fis->h.command = SAT_READ_VERIFY_SECTORS; /* 0x40 */ 8224 fis->h.features = 0; /* FIS reserve */ 8225 fis->d.lbaLow = scsiCmnd->cdb[5]; /* FIS LBA (7 :0 ) */ 8226 fis->d.lbaMid = scsiCmnd->cdb[4]; /* FIS LBA (15:8 ) */ 8227 fis->d.lbaHigh = scsiCmnd->cdb[3]; /* FIS LBA (23:16) */ 8228 /* FIS LBA mode set LBA (27:24) */ 8229 fis->d.device = (bit8)((0x4 << 4) | (scsiCmnd->cdb[2] & 0xF)); 8230 fis->d.lbaLowExp = 0; 8231 fis->d.lbaMidExp = 0; 8232 fis->d.lbaHighExp = 0; 8233 fis->d.featuresExp = 0; 8234 fis->d.sectorCount = scsiCmnd->cdb[9]; /* FIS sector count (7:0) */ 8235 fis->d.sectorCountExp = 0; 8236 fis->d.reserved4 = 0; 8237 fis->d.control = 0; /* FIS HOB bit clear */ 8238 fis->d.reserved5 = 0; 8239 8240 agRequestType = AGSA_SATA_PROTOCOL_NON_DATA; 8241 satIOContext->ATACmd = SAT_READ_VERIFY_SECTORS; 8242 8243 } 8244 8245 satIOContext->currentLBA = lba; 8246 satIOContext->OrgTL = tl; 8247 8248 /* 8249 computing number of loop and remainder for tl 8250 0xFF in case not ext 8251 0xFFFF in case EXT 8252 */ 8253 if (fis->h.command == SAT_READ_VERIFY_SECTORS) 8254 { 8255 LoopNum = smsatComputeLoopNum(tl, 0xFF); 8256 } 8257 else if (fis->h.command == SAT_READ_VERIFY_SECTORS_EXT) 8258 { 8259 /* SAT_READ_SECTORS_EXT, SAT_READ_DMA_EXT */ 8260 LoopNum = smsatComputeLoopNum(tl, 0xFFFF); 8261 } 8262 else 8263 { 8264 SM_DBG1(("smsatVerify12: error case 1!!!\n")); 8265 LoopNum = 1; 8266 } 8267 8268 satIOContext->LoopNum = LoopNum; 8269 8270 if (LoopNum == 1) 8271 { 8272 SM_DBG5(("smsatVerify12: NON CHAINED data\n")); 8273 /* Initialize CB for SATA completion. 8274 */ 8275 satIOContext->satCompleteCB = &smsatNonChainedVerifyCB; 8276 } 8277 else 8278 { 8279 SM_DBG1(("smsatVerify12: CHAINED data!!!\n")); 8280 /* re-setting tl */ 8281 if (fis->h.command == SAT_READ_VERIFY_SECTORS) 8282 { 8283 fis->d.sectorCount = 0xFF; 8284 } 8285 else if (fis->h.command == SAT_READ_VERIFY_SECTORS_EXT) 8286 { 8287 fis->d.sectorCount = 0xFF; 8288 fis->d.sectorCountExp = 0xFF; 8289 } 8290 else 8291 { 8292 SM_DBG1(("smsatVerify12: error case 2!!!\n")); 8293 } 8294 8295 /* Initialize CB for SATA completion. 8296 */ 8297 satIOContext->satCompleteCB = &smsatChainedVerifyCB; 8298 } 8299 8300 8301 /* 8302 * Prepare SGL and send FIS to LL layer. 8303 */ 8304 satIOContext->reqType = agRequestType; /* Save it */ 8305 8306 status = smsataLLIOStart( smRoot, 8307 smIORequest, 8308 smDeviceHandle, 8309 smScsiRequest, 8310 satIOContext); 8311 return (status); 8312 } 8313 8314 osGLOBAL bit32 8315 smsatVerify16( 8316 smRoot_t *smRoot, 8317 smIORequest_t *smIORequest, 8318 smDeviceHandle_t *smDeviceHandle, 8319 smScsiInitiatorRequest_t *smScsiRequest, 8320 smSatIOContext_t *satIOContext 8321 ) 8322 { 8323 /* 8324 For simple implementation, 8325 no byte comparison supported as of 4/5/06 8326 */ 8327 smScsiRspSense_t *pSense; 8328 smIniScsiCmnd_t *scsiCmnd; 8329 smDeviceData_t *pSatDevData; 8330 agsaFisRegHostToDevice_t *fis; 8331 bit32 status; 8332 bit32 agRequestType = AGSA_SATA_PROTOCOL_NON_DATA; 8333 bit32 lba = 0; 8334 bit32 tl = 0; 8335 bit32 LoopNum = 1; 8336 bit8 LBA[8]; 8337 bit8 TL[8]; 8338 bit32 AllChk = agFALSE; /* lba, lba+tl check against ATA limit and Disk capacity */ 8339 8340 pSense = satIOContext->pSense; 8341 scsiCmnd = &smScsiRequest->scsiCmnd; 8342 pSatDevData = satIOContext->pSatDevData; 8343 fis = satIOContext->pFis; 8344 SM_DBG5(("smsatVerify16: start\n")); 8345 /* checking BYTCHK */ 8346 if (scsiCmnd->cdb[1] & SCSI_VERIFY_BYTCHK_MASK) 8347 { 8348 /* 8349 should do the byte check 8350 but not supported in this version 8351 */ 8352 smsatSetSensePayload( pSense, 8353 SCSI_SNSKEY_ILLEGAL_REQUEST, 8354 0, 8355 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 8356 satIOContext); 8357 /*smEnqueueIO(smRoot, satIOContext);*/ 8358 tdsmIOCompletedCB( smRoot, 8359 smIORequest, 8360 smIOSuccess, 8361 SCSI_STAT_CHECK_CONDITION, 8362 satIOContext->pSmSenseData, 8363 satIOContext->interruptContext ); 8364 SM_DBG1(("smsatVerify16: no byte checking!!!\n")); 8365 return SM_RC_SUCCESS; 8366 } 8367 /* checking CONTROL */ 8368 /* NACA == 1 or LINK == 1*/ 8369 if ( (scsiCmnd->cdb[15] & SCSI_NACA_MASK) || (scsiCmnd->cdb[15] & SCSI_LINK_MASK) ) 8370 { 8371 smsatSetSensePayload( pSense, 8372 SCSI_SNSKEY_ILLEGAL_REQUEST, 8373 0, 8374 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 8375 satIOContext); 8376 /*smEnqueueIO(smRoot, satIOContext);*/ 8377 tdsmIOCompletedCB( smRoot, 8378 smIORequest, 8379 smIOSuccess, 8380 SCSI_STAT_CHECK_CONDITION, 8381 satIOContext->pSmSenseData, 8382 satIOContext->interruptContext ); 8383 SM_DBG1(("smsatVerify16: return control!!!\n")); 8384 return SM_RC_SUCCESS; 8385 } 8386 sm_memset(LBA, 0, sizeof(LBA)); 8387 sm_memset(TL, 0, sizeof(TL)); 8388 8389 /* do not use memcpy due to indexing in LBA and TL */ 8390 LBA[0] = scsiCmnd->cdb[2]; /* MSB */ 8391 LBA[1] = scsiCmnd->cdb[3]; 8392 LBA[2] = scsiCmnd->cdb[4]; 8393 LBA[3] = scsiCmnd->cdb[5]; 8394 LBA[4] = scsiCmnd->cdb[6]; 8395 LBA[5] = scsiCmnd->cdb[7]; 8396 LBA[6] = scsiCmnd->cdb[8]; 8397 LBA[7] = scsiCmnd->cdb[9]; /* LSB */ 8398 8399 TL[0] = 0; 8400 TL[1] = 0; 8401 TL[2] = 0; 8402 TL[3] = 0; 8403 TL[4] = scsiCmnd->cdb[10]; /* MSB */ 8404 TL[5] = scsiCmnd->cdb[11]; 8405 TL[6] = scsiCmnd->cdb[12]; 8406 TL[7] = scsiCmnd->cdb[13]; /* LSB */ 8407 lba = smsatComputeCDB16LBA(satIOContext); 8408 tl = smsatComputeCDB16TL(satIOContext); 8409 8410 if (pSatDevData->satNCQ != agTRUE && 8411 pSatDevData->sat48BitSupport != agTRUE 8412 ) 8413 { 8414 AllChk = smsatCheckLimit(LBA, TL, agFALSE, pSatDevData); 8415 if (AllChk) 8416 { 8417 SM_DBG1(("smsatVerify16: return LBA out of range, not EXT!!!\n")); 8418 smsatSetSensePayload( pSense, 8419 SCSI_SNSKEY_ILLEGAL_REQUEST, 8420 0, 8421 SCSI_SNSCODE_LOGICAL_BLOCK_ADDRESS_OUT_OF_RANGE, 8422 satIOContext); 8423 /*smEnqueueIO(smRoot, satIOContext);*/ 8424 tdsmIOCompletedCB( smRoot, 8425 smIORequest, 8426 smIOSuccess, 8427 SCSI_STAT_CHECK_CONDITION, 8428 satIOContext->pSmSenseData, 8429 satIOContext->interruptContext ); 8430 return SM_RC_SUCCESS; 8431 } 8432 } 8433 else 8434 { 8435 AllChk = smsatCheckLimit(LBA, TL, agTRUE, pSatDevData); 8436 if (AllChk) 8437 { 8438 SM_DBG1(("smsatVerify16: return LBA out of range, EXT!!!\n")); 8439 smsatSetSensePayload( pSense, 8440 SCSI_SNSKEY_ILLEGAL_REQUEST, 8441 0, 8442 SCSI_SNSCODE_LOGICAL_BLOCK_ADDRESS_OUT_OF_RANGE, 8443 satIOContext); 8444 /*smEnqueueIO(smRoot, satIOContext);*/ 8445 tdsmIOCompletedCB( smRoot, 8446 smIORequest, 8447 smIOSuccess, 8448 SCSI_STAT_CHECK_CONDITION, 8449 satIOContext->pSmSenseData, 8450 satIOContext->interruptContext ); 8451 return SM_RC_SUCCESS; 8452 } 8453 } 8454 8455 if (pSatDevData->sat48BitSupport == agTRUE) 8456 { 8457 SM_DBG5(("smsatVerify16: SAT_READ_VERIFY_SECTORS_EXT\n")); 8458 fis->h.fisType = 0x27; /* Reg host to device */ 8459 fis->h.c_pmPort = 0x80; /* C Bit is set */ 8460 fis->h.command = SAT_READ_VERIFY_SECTORS_EXT;/* 0x42 */ 8461 fis->h.features = 0; /* FIS reserve */ 8462 fis->d.lbaLow = scsiCmnd->cdb[9]; /* FIS LBA (7 :0 ) */ 8463 fis->d.lbaMid = scsiCmnd->cdb[8]; /* FIS LBA (15:8 ) */ 8464 fis->d.lbaHigh = scsiCmnd->cdb[7]; /* FIS LBA (23:16) */ 8465 fis->d.device = 0x40; /* FIS LBA mode set 01000000 */ 8466 fis->d.lbaLowExp = scsiCmnd->cdb[6]; /* FIS LBA (31:24) */ 8467 fis->d.lbaMidExp = scsiCmnd->cdb[5]; /* FIS LBA (39:32) */ 8468 fis->d.lbaHighExp = scsiCmnd->cdb[4]; /* FIS LBA (47:40) */ 8469 fis->d.featuresExp = 0; /* FIS reserve */ 8470 fis->d.sectorCount = scsiCmnd->cdb[13]; /* FIS sector count (7:0) */ 8471 fis->d.sectorCountExp = scsiCmnd->cdb[12]; /* FIS sector count (15:8) */ 8472 8473 fis->d.reserved4 = 0; 8474 fis->d.control = 0; /* FIS HOB bit clear */ 8475 fis->d.reserved5 = 0; 8476 8477 agRequestType = AGSA_SATA_PROTOCOL_NON_DATA; 8478 satIOContext->ATACmd = SAT_READ_VERIFY_SECTORS_EXT; 8479 } 8480 else 8481 { 8482 SM_DBG5(("smsatVerify16: SAT_READ_VERIFY_SECTORS\n")); 8483 fis->h.fisType = 0x27; /* Reg host to device */ 8484 fis->h.c_pmPort = 0x80; /* C bit is set */ 8485 fis->h.command = SAT_READ_VERIFY_SECTORS; /* 0x40 */ 8486 fis->h.features = 0; /* FIS reserve */ 8487 fis->d.lbaLow = scsiCmnd->cdb[9]; /* FIS LBA (7 :0 ) */ 8488 fis->d.lbaMid = scsiCmnd->cdb[8]; /* FIS LBA (15:8 ) */ 8489 fis->d.lbaHigh = scsiCmnd->cdb[7]; /* FIS LBA (23:16) */ 8490 /* FIS LBA mode set LBA (27:24) */ 8491 fis->d.device = (bit8)((0x4 << 4) | (scsiCmnd->cdb[6] & 0xF)); 8492 fis->d.lbaLowExp = 0; 8493 fis->d.lbaMidExp = 0; 8494 fis->d.lbaHighExp = 0; 8495 fis->d.featuresExp = 0; 8496 fis->d.sectorCount = scsiCmnd->cdb[13]; /* FIS sector count (7:0) */ 8497 fis->d.sectorCountExp = 0; 8498 fis->d.reserved4 = 0; 8499 fis->d.control = 0; /* FIS HOB bit clear */ 8500 fis->d.reserved5 = 0; 8501 8502 agRequestType = AGSA_SATA_PROTOCOL_NON_DATA; 8503 satIOContext->ATACmd = SAT_READ_VERIFY_SECTORS; 8504 8505 } 8506 8507 satIOContext->currentLBA = lba; 8508 satIOContext->OrgTL = tl; 8509 8510 /* 8511 computing number of loop and remainder for tl 8512 0xFF in case not ext 8513 0xFFFF in case EXT 8514 */ 8515 if (fis->h.command == SAT_READ_VERIFY_SECTORS) 8516 { 8517 LoopNum = smsatComputeLoopNum(tl, 0xFF); 8518 } 8519 else if (fis->h.command == SAT_READ_VERIFY_SECTORS_EXT) 8520 { 8521 /* SAT_READ_SECTORS_EXT, SAT_READ_DMA_EXT */ 8522 LoopNum = smsatComputeLoopNum(tl, 0xFFFF); 8523 } 8524 else 8525 { 8526 SM_DBG1(("smsatVerify16: error case 1!!!\n")); 8527 LoopNum = 1; 8528 } 8529 8530 satIOContext->LoopNum = LoopNum; 8531 8532 if (LoopNum == 1) 8533 { 8534 SM_DBG5(("smsatVerify16: NON CHAINED data\n")); 8535 /* Initialize CB for SATA completion. 8536 */ 8537 satIOContext->satCompleteCB = &smsatNonChainedVerifyCB; 8538 } 8539 else 8540 { 8541 SM_DBG1(("smsatVerify16: CHAINED data!!!\n")); 8542 /* re-setting tl */ 8543 if (fis->h.command == SAT_READ_VERIFY_SECTORS) 8544 { 8545 fis->d.sectorCount = 0xFF; 8546 } 8547 else if (fis->h.command == SAT_READ_VERIFY_SECTORS_EXT) 8548 { 8549 fis->d.sectorCount = 0xFF; 8550 fis->d.sectorCountExp = 0xFF; 8551 } 8552 else 8553 { 8554 SM_DBG1(("smsatVerify16: error case 2!!!\n")); 8555 } 8556 8557 /* Initialize CB for SATA completion. 8558 */ 8559 satIOContext->satCompleteCB = &smsatChainedVerifyCB; 8560 } 8561 8562 8563 /* 8564 * Prepare SGL and send FIS to LL layer. 8565 */ 8566 satIOContext->reqType = agRequestType; /* Save it */ 8567 8568 status = smsataLLIOStart( smRoot, 8569 smIORequest, 8570 smDeviceHandle, 8571 smScsiRequest, 8572 satIOContext); 8573 return (status); 8574 } 8575 8576 osGLOBAL bit32 8577 smsatTestUnitReady( 8578 smRoot_t *smRoot, 8579 smIORequest_t *smIORequest, 8580 smDeviceHandle_t *smDeviceHandle, 8581 smScsiInitiatorRequest_t *smScsiRequest, 8582 smSatIOContext_t *satIOContext 8583 ) 8584 { 8585 bit32 status; 8586 bit32 agRequestType; 8587 smDeviceData_t *pSatDevData; 8588 smScsiRspSense_t *pSense; 8589 smIniScsiCmnd_t *scsiCmnd; 8590 agsaFisRegHostToDevice_t *fis; 8591 8592 pSense = satIOContext->pSense; 8593 pSatDevData = satIOContext->pSatDevData; 8594 scsiCmnd = &smScsiRequest->scsiCmnd; 8595 fis = satIOContext->pFis; 8596 8597 SM_DBG5(("smsatTestUnitReady: start\n")); 8598 8599 /* checking CONTROL */ 8600 /* NACA == 1 or LINK == 1*/ 8601 if ( (scsiCmnd->cdb[5] & SCSI_NACA_MASK) || (scsiCmnd->cdb[5] & SCSI_LINK_MASK) ) 8602 { 8603 smsatSetSensePayload( pSense, 8604 SCSI_SNSKEY_ILLEGAL_REQUEST, 8605 0, 8606 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 8607 satIOContext); 8608 8609 /*smEnqueueIO(smRoot, satIOContext);*/ 8610 8611 tdsmIOCompletedCB( smRoot, 8612 smIORequest, 8613 smIOSuccess, 8614 SCSI_STAT_CHECK_CONDITION, 8615 satIOContext->pSmSenseData, 8616 satIOContext->interruptContext ); 8617 8618 SM_DBG1(("smsatTestUnitReady: return control!!!\n")); 8619 return SM_RC_SUCCESS; 8620 } 8621 8622 /* SAT revision 8, 8.11.2, p42*/ 8623 if (pSatDevData->satStopState == agTRUE) 8624 { 8625 smsatSetSensePayload( pSense, 8626 SCSI_SNSKEY_NOT_READY, 8627 0, 8628 SCSI_SNSCODE_LOGICAL_UNIT_NOT_READY_INITIALIZING_COMMAND_REQUIRED, 8629 satIOContext); 8630 8631 /*smEnqueueIO(smRoot, satIOContext);*/ 8632 8633 tdsmIOCompletedCB( smRoot, 8634 smIORequest, 8635 smIOSuccess, 8636 SCSI_STAT_CHECK_CONDITION, 8637 satIOContext->pSmSenseData, 8638 satIOContext->interruptContext ); 8639 SM_DBG1(("smsatTestUnitReady: stop state!!!\n")); 8640 return SM_RC_SUCCESS; 8641 } 8642 8643 /* 8644 * Check if format is in progress 8645 */ 8646 if (pSatDevData->satDriveState == SAT_DEV_STATE_FORMAT_IN_PROGRESS) 8647 { 8648 SM_DBG1(("smsatTestUnitReady: FORMAT_IN_PROGRESS!!!\n")); 8649 8650 smsatSetSensePayload( pSense, 8651 SCSI_SNSKEY_NOT_READY, 8652 0, 8653 SCSI_SNSCODE_LOGICAL_UNIT_NOT_READY_FORMAT_IN_PROGRESS, 8654 satIOContext); 8655 8656 /*smEnqueueIO(smRoot, satIOContext);*/ 8657 8658 tdsmIOCompletedCB( smRoot, 8659 smIORequest, 8660 smIOSuccess, 8661 SCSI_STAT_CHECK_CONDITION, 8662 satIOContext->pSmSenseData, 8663 satIOContext->interruptContext ); 8664 SM_DBG1(("smsatTestUnitReady: format in progress!!!\n")); 8665 return SM_RC_SUCCESS; 8666 } 8667 8668 /* 8669 check previously issued ATA command 8670 */ 8671 if (pSatDevData->satPendingIO != 0) 8672 { 8673 if (pSatDevData->satDeviceFaultState == agTRUE) 8674 { 8675 smsatSetSensePayload( pSense, 8676 SCSI_SNSKEY_HARDWARE_ERROR, 8677 0, 8678 SCSI_SNSCODE_LOGICAL_UNIT_FAILURE, 8679 satIOContext); 8680 8681 /*smEnqueueIO(smRoot, satIOContext);*/ 8682 8683 tdsmIOCompletedCB( smRoot, 8684 smIORequest, 8685 smIOSuccess, 8686 SCSI_STAT_CHECK_CONDITION, 8687 satIOContext->pSmSenseData, 8688 satIOContext->interruptContext ); 8689 SM_DBG1(("smsatTestUnitReady: previous command ended in error!!!\n")); 8690 return SM_RC_SUCCESS; 8691 } 8692 } 8693 8694 /* 8695 check removalbe media feature set 8696 */ 8697 if(pSatDevData->satRemovableMedia && pSatDevData->satRemovableMediaEnabled) 8698 { 8699 SM_DBG5(("smsatTestUnitReady: sending get media status cmnd\n")); 8700 /* send GET MEDIA STATUS command */ 8701 fis->h.fisType = 0x27; /* Reg host to device */ 8702 fis->h.c_pmPort = 0x80; /* C Bit is set */ 8703 fis->h.command = SAT_GET_MEDIA_STATUS; /* 0xDA */ 8704 fis->h.features = 0; /* FIS features NA */ 8705 fis->d.lbaLow = 0; /* FIS LBA (7 :0 ) */ 8706 fis->d.lbaMid = 0; /* FIS LBA (15:8 ) */ 8707 fis->d.lbaHigh = 0; /* FIS LBA (23:16) */ 8708 fis->d.device = 0; /* FIS DEV is discared in SATA */ 8709 fis->d.lbaLowExp = 0; 8710 fis->d.lbaMidExp = 0; 8711 fis->d.lbaHighExp = 0; 8712 fis->d.featuresExp = 0; 8713 fis->d.sectorCount = 0; /* FIS sector count (7:0) */ 8714 fis->d.sectorCountExp = 0; 8715 fis->d.reserved4 = 0; 8716 fis->d.control = 0; /* FIS HOB bit clear */ 8717 fis->d.reserved5 = 0; 8718 agRequestType = AGSA_SATA_PROTOCOL_NON_DATA; 8719 8720 /* Initialize CB for SATA completion. 8721 */ 8722 satIOContext->satCompleteCB = &smsatTestUnitReadyCB; 8723 8724 /* 8725 * Prepare SGL and send FIS to LL layer. 8726 */ 8727 satIOContext->reqType = agRequestType; /* Save it */ 8728 8729 status = smsataLLIOStart( smRoot, 8730 smIORequest, 8731 smDeviceHandle, 8732 smScsiRequest, 8733 satIOContext); 8734 8735 return (status); 8736 } 8737 /* 8738 number 6) in SAT p42 8739 send ATA CHECK POWER MODE 8740 */ 8741 SM_DBG5(("smsatTestUnitReady: sending check power mode cmnd\n")); 8742 status = smsatTestUnitReady_1( smRoot, 8743 smIORequest, 8744 smDeviceHandle, 8745 smScsiRequest, 8746 satIOContext); 8747 return (status); 8748 } 8749 8750 osGLOBAL bit32 8751 smsatTestUnitReady_1( 8752 smRoot_t *smRoot, 8753 smIORequest_t *smIORequest, 8754 smDeviceHandle_t *smDeviceHandle, 8755 smScsiInitiatorRequest_t *smScsiRequest, 8756 smSatIOContext_t *satIOContext 8757 ) 8758 { 8759 /* 8760 sends SAT_CHECK_POWER_MODE as a part of TESTUNITREADY 8761 internally generated - no directly corresponding scsi 8762 called in satIOCompleted as a part of satTestUnitReady(), SAT, revision8, 8.11.2, p42 8763 */ 8764 bit32 status; 8765 bit32 agRequestType; 8766 agsaFisRegHostToDevice_t *fis; 8767 8768 fis = satIOContext->pFis; 8769 SM_DBG5(("smsatTestUnitReady_1: start\n")); 8770 /* 8771 * Send the ATA CHECK POWER MODE command. 8772 */ 8773 fis->h.fisType = 0x27; /* Reg host to device */ 8774 fis->h.c_pmPort = 0x80; /* C Bit is set */ 8775 fis->h.command = SAT_CHECK_POWER_MODE; /* 0xE5 */ 8776 fis->h.features = 0; 8777 fis->d.lbaLow = 0; 8778 fis->d.lbaMid = 0; 8779 fis->d.lbaHigh = 0; 8780 fis->d.device = 0; 8781 fis->d.lbaLowExp = 0; 8782 fis->d.lbaMidExp = 0; 8783 fis->d.lbaHighExp = 0; 8784 fis->d.featuresExp = 0; 8785 fis->d.sectorCount = 0; 8786 fis->d.sectorCountExp = 0; 8787 fis->d.reserved4 = 0; 8788 fis->d.control = 0; /* FIS HOB bit clear */ 8789 fis->d.reserved5 = 0; 8790 8791 agRequestType = AGSA_SATA_PROTOCOL_NON_DATA; 8792 8793 /* Initialize CB for SATA completion. 8794 */ 8795 satIOContext->satCompleteCB = &smsatTestUnitReadyCB; 8796 8797 /* 8798 * Prepare SGL and send FIS to LL layer. 8799 */ 8800 satIOContext->reqType = agRequestType; /* Save it */ 8801 8802 status = smsataLLIOStart( smRoot, 8803 smIORequest, 8804 smDeviceHandle, 8805 smScsiRequest, 8806 satIOContext); 8807 8808 SM_DBG5(("smsatTestUnitReady_1: return\n")); 8809 8810 return status; 8811 } 8812 8813 osGLOBAL bit32 8814 smsatInquiry( 8815 smRoot_t *smRoot, 8816 smIORequest_t *smIORequest, 8817 smDeviceHandle_t *smDeviceHandle, 8818 smScsiInitiatorRequest_t *smScsiRequest, 8819 smSatIOContext_t *satIOContext 8820 ) 8821 { 8822 /* 8823 CMDDT bit is obsolete in SPC-3 and this is assumed in SAT revision 8 8824 */ 8825 smScsiRspSense_t *pSense; 8826 smIniScsiCmnd_t *scsiCmnd; 8827 smDeviceData_t *pSatDevData; 8828 bit32 status; 8829 8830 pSense = satIOContext->pSense; 8831 scsiCmnd = &smScsiRequest->scsiCmnd; 8832 pSatDevData = satIOContext->pSatDevData; 8833 SM_DBG5(("smsatInquiry: start\n")); 8834 SM_DBG5(("smsatInquiry: pSatDevData did %d\n", pSatDevData->id)); 8835 //smhexdump("smsatInquiry", (bit8 *)scsiCmnd->cdb, 6); 8836 /* checking CONTROL */ 8837 /* NACA == 1 or LINK == 1*/ 8838 if ( (scsiCmnd->cdb[5] & SCSI_NACA_MASK) || (scsiCmnd->cdb[5] & SCSI_LINK_MASK) ) 8839 { 8840 smsatSetSensePayload( pSense, 8841 SCSI_SNSKEY_ILLEGAL_REQUEST, 8842 0, 8843 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 8844 satIOContext); 8845 /*smEnqueueIO(smRoot, satIOContext);*/ 8846 tdsmIOCompletedCB( smRoot, 8847 smIORequest, 8848 smIOSuccess, 8849 SCSI_STAT_CHECK_CONDITION, 8850 satIOContext->pSmSenseData, 8851 satIOContext->interruptContext ); 8852 SM_DBG1(("smsatInquiry: return control!!!\n")); 8853 return SM_RC_SUCCESS; 8854 } 8855 8856 /* checking EVPD and Allocation Length */ 8857 /* SPC-4 spec 6.4 p141 */ 8858 /* EVPD bit == 0 && PAGE CODE != 0 */ 8859 if ( !(scsiCmnd->cdb[1] & SCSI_EVPD_MASK) && 8860 (scsiCmnd->cdb[2] != 0) 8861 ) 8862 { 8863 smsatSetSensePayload( pSense, 8864 SCSI_SNSKEY_ILLEGAL_REQUEST, 8865 0, 8866 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 8867 satIOContext); 8868 /*smEnqueueIO(smRoot, satIOContext);*/ 8869 tdsmIOCompletedCB( smRoot, 8870 smIORequest, 8871 smIOSuccess, 8872 SCSI_STAT_CHECK_CONDITION, 8873 satIOContext->pSmSenseData, 8874 satIOContext->interruptContext ); 8875 SM_DBG1(("smsatInquiry: return EVPD and PAGE CODE!!!\n")); 8876 return SM_RC_SUCCESS; 8877 } 8878 SM_DBG6(("smsatInquiry: allocation length 0x%x %d\n", ((scsiCmnd->cdb[3]) << 8) + scsiCmnd->cdb[4], ((scsiCmnd->cdb[3]) << 8) + scsiCmnd->cdb[4])); 8879 /* convert OS IO to TD internal IO */ 8880 if ( pSatDevData->IDDeviceValid == agFALSE) 8881 { 8882 status = smsatStartIDDev( 8883 smRoot, 8884 smIORequest, 8885 smDeviceHandle, 8886 smScsiRequest, 8887 satIOContext 8888 ); 8889 SM_DBG6(("smsatInquiry: end status %d\n", status)); 8890 return status; 8891 } 8892 else 8893 { 8894 SM_DBG6(("smsatInquiry: calling satInquiryIntCB\n")); 8895 smsatInquiryIntCB( 8896 smRoot, 8897 smIORequest, 8898 smDeviceHandle, 8899 smScsiRequest, 8900 satIOContext 8901 ); 8902 /*smEnqueueIO(smRoot, satIOContext);*/ 8903 return SM_RC_SUCCESS; 8904 } 8905 } 8906 8907 8908 osGLOBAL bit32 8909 smsatStartIDDev( 8910 smRoot_t *smRoot, 8911 smIORequest_t *smIORequest, 8912 smDeviceHandle_t *smDeviceHandle, 8913 smScsiInitiatorRequest_t *smScsiRequest, 8914 smSatIOContext_t *satIOContext 8915 ) 8916 { 8917 smSatInternalIo_t *satIntIo = agNULL; 8918 smDeviceData_t *satDevData = agNULL; 8919 smIORequestBody_t *smIORequestBody; 8920 smSatIOContext_t *satNewIOContext; 8921 bit32 status; 8922 8923 SM_DBG5(("smsatStartIDDev: start\n")); 8924 8925 satDevData = satIOContext->pSatDevData; 8926 8927 SM_DBG6(("smsatStartIDDev: before alloc\n")); 8928 8929 /* allocate identify device command */ 8930 satIntIo = smsatAllocIntIoResource( smRoot, 8931 smIORequest, 8932 satDevData, 8933 sizeof(agsaSATAIdentifyData_t), /* 512; size of identify device data */ 8934 satIntIo); 8935 8936 SM_DBG6(("smsatStartIDDev: before after\n")); 8937 8938 if (satIntIo == agNULL) 8939 { 8940 SM_DBG1(("smsatStartIDDev: can't alloacate!!!\n")); 8941 8942 /*smEnqueueIO(smRoot, satIOContext);*/ 8943 8944 return SM_RC_FAILURE; 8945 } 8946 8947 satIntIo->satOrgSmIORequest = smIORequest; /* changed */ 8948 smIORequestBody = satIntIo->satIntRequestBody; 8949 satNewIOContext = &(smIORequestBody->transport.SATA.satIOContext); 8950 8951 satNewIOContext->pSatDevData = satDevData; 8952 satNewIOContext->pFis = &(smIORequestBody->transport.SATA.agSATARequestBody.fis.fisRegHostToDev); 8953 satNewIOContext->pScsiCmnd = &(satIntIo->satIntSmScsiXchg.scsiCmnd); 8954 satNewIOContext->pSense = &(smIORequestBody->transport.SATA.sensePayload); 8955 satNewIOContext->pSmSenseData = &(smIORequestBody->transport.SATA.smSenseData); 8956 satNewIOContext->smRequestBody = satIntIo->satIntRequestBody; /* key fix */ 8957 satNewIOContext->interruptContext = tiInterruptContext; 8958 satNewIOContext->satIntIoContext = satIntIo; 8959 8960 satNewIOContext->psmDeviceHandle = agNULL; 8961 satNewIOContext->satOrgIOContext = satIOContext; /* changed */ 8962 8963 /* this is valid only for TD layer generated (not triggered by OS at all) IO */ 8964 satNewIOContext->smScsiXchg = &(satIntIo->satIntSmScsiXchg); 8965 8966 8967 SM_DBG6(("smsatStartIDDev: OS satIOContext %p \n", satIOContext)); 8968 SM_DBG6(("smsatStartIDDev: TD satNewIOContext %p \n", satNewIOContext)); 8969 SM_DBG6(("smsatStartIDDev: OS tiScsiXchg %p \n", satIOContext->smScsiXchg)); 8970 SM_DBG6(("smsatStartIDDev: TD tiScsiXchg %p \n", satNewIOContext->smScsiXchg)); 8971 8972 8973 8974 SM_DBG1(("smsatStartIDDev: satNewIOContext %p smIORequestBody %p!!!\n", satNewIOContext, smIORequestBody)); 8975 8976 status = smsatSendIDDev( smRoot, 8977 &satIntIo->satIntSmIORequest, /* New smIORequest */ 8978 smDeviceHandle, 8979 satNewIOContext->smScsiXchg, /* New tiScsiInitiatorRequest_t *tiScsiRequest, */ 8980 satNewIOContext); 8981 8982 if (status != SM_RC_SUCCESS) 8983 { 8984 SM_DBG1(("smsatStartIDDev: failed in sending!!!\n")); 8985 8986 smsatFreeIntIoResource( smRoot, 8987 satDevData, 8988 satIntIo); 8989 /*smEnqueueIO(smRoot, satIOContext);*/ 8990 8991 return SM_RC_FAILURE; 8992 } 8993 8994 8995 SM_DBG6(("smsatStartIDDev: end\n")); 8996 8997 return status; 8998 } 8999 9000 osGLOBAL bit32 9001 smsatSendIDDev( 9002 smRoot_t *smRoot, 9003 smIORequest_t *smIORequest, 9004 smDeviceHandle_t *smDeviceHandle, 9005 smScsiInitiatorRequest_t *smScsiRequest, 9006 smSatIOContext_t *satIOContext 9007 ) 9008 { 9009 bit32 status; 9010 bit32 agRequestType; 9011 smDeviceData_t *pSatDevData; 9012 agsaFisRegHostToDevice_t *fis; 9013 #ifdef SM_INTERNAL_DEBUG 9014 smIORequestBody_t *smIORequestBody; 9015 smSatInternalIo_t *satIntIoContext; 9016 #endif 9017 9018 pSatDevData = satIOContext->pSatDevData; 9019 fis = satIOContext->pFis; 9020 SM_DBG6(("smsatSendIDDev: start\n")); 9021 SM_DBG6(("smsatSendIDDev: did %d\n", pSatDevData->id)); 9022 #ifdef SM_INTERNAL_DEBUG 9023 satIntIoContext = satIOContext->satIntIoContext; 9024 smIORequestBody = satIntIoContext->satIntRequestBody; 9025 #endif 9026 fis->h.fisType = 0x27; /* Reg host to device */ 9027 fis->h.c_pmPort = 0x80; /* C Bit is set */ 9028 if (pSatDevData->satDeviceType == SATA_ATAPI_DEVICE) 9029 fis->h.command = SAT_IDENTIFY_PACKET_DEVICE; /* 0x40 */ 9030 else 9031 fis->h.command = SAT_IDENTIFY_DEVICE; /* 0xEC */ 9032 fis->h.features = 0; /* FIS reserve */ 9033 fis->d.lbaLow = 0; /* FIS LBA (7 :0 ) */ 9034 fis->d.lbaMid = 0; /* FIS LBA (15:8 ) */ 9035 fis->d.lbaHigh = 0; /* FIS LBA (23:16) */ 9036 fis->d.device = 0; /* FIS LBA mode */ 9037 fis->d.lbaLowExp = 0; 9038 fis->d.lbaMidExp = 0; 9039 fis->d.lbaHighExp = 0; 9040 fis->d.featuresExp = 0; 9041 fis->d.sectorCount = 0; /* FIS sector count (7:0) */ 9042 fis->d.sectorCountExp = 0; 9043 fis->d.reserved4 = 0; 9044 fis->d.control = 0; /* FIS HOB bit clear */ 9045 fis->d.reserved5 = 0; 9046 9047 agRequestType = AGSA_SATA_PROTOCOL_PIO_READ; 9048 9049 /* Initialize CB for SATA completion. 9050 */ 9051 satIOContext->satCompleteCB = &smsatInquiryCB; 9052 9053 /* 9054 * Prepare SGL and send FIS to LL layer. 9055 */ 9056 satIOContext->reqType = agRequestType; /* Save it */ 9057 9058 #ifdef SM_INTERNAL_DEBUG 9059 smhexdump("smsatSendIDDev", (bit8 *)satIOContext->pFis, sizeof(agsaFisRegHostToDevice_t)); 9060 smhexdump("smsatSendIDDev LL", (bit8 *)&(smIORequestBody->transport.SATA.agSATARequestBody.fis.fisRegHostToDev), sizeof(agsaFisRegHostToDevice_t)); 9061 #endif 9062 status = smsataLLIOStart( smRoot, 9063 smIORequest, 9064 smDeviceHandle, 9065 smScsiRequest, 9066 satIOContext); 9067 9068 SM_DBG6(("smsatSendIDDev: end status %d\n", status)); 9069 return status; 9070 } 9071 9072 osGLOBAL bit32 9073 smsatRequestSense( 9074 smRoot_t *smRoot, 9075 smIORequest_t *smIORequest, 9076 smDeviceHandle_t *smDeviceHandle, 9077 smScsiInitiatorRequest_t *smScsiRequest, 9078 smSatIOContext_t *satIOContext 9079 ) 9080 { 9081 /* 9082 SAT Rev 8 p38, Table25 9083 sending SMART RETURN STATUS 9084 Checking SMART Treshold Exceeded Condition is done in satRequestSenseCB() 9085 Only fixed format sense data is support. In other words, we don't support DESC bit is set 9086 in Request Sense 9087 */ 9088 bit32 status; 9089 bit32 agRequestType; 9090 smScsiRspSense_t *pSense; 9091 smDeviceData_t *pSatDevData; 9092 smIniScsiCmnd_t *scsiCmnd; 9093 agsaFisRegHostToDevice_t *fis; 9094 smIORequestBody_t *smIORequestBody; 9095 smSatInternalIo_t *satIntIo = agNULL; 9096 smSatIOContext_t *satIOContext2; 9097 bit8 *pDataBuffer = agNULL; 9098 bit32 allocationLen = 0; 9099 9100 pSense = satIOContext->pSense; 9101 pSatDevData = satIOContext->pSatDevData; 9102 scsiCmnd = &smScsiRequest->scsiCmnd; 9103 fis = satIOContext->pFis; 9104 pDataBuffer = (bit8 *) smScsiRequest->sglVirtualAddr; 9105 allocationLen = scsiCmnd->cdb[4]; 9106 allocationLen = MIN(allocationLen, scsiCmnd->expDataLength); 9107 SM_DBG5(("smsatRequestSense: start\n")); 9108 9109 /* checking CONTROL */ 9110 /* NACA == 1 or LINK == 1*/ 9111 if ( (scsiCmnd->cdb[5] & SCSI_NACA_MASK) || (scsiCmnd->cdb[5] & SCSI_LINK_MASK) ) 9112 { 9113 smsatSetSensePayload( pSense, 9114 SCSI_SNSKEY_ILLEGAL_REQUEST, 9115 0, 9116 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 9117 satIOContext); 9118 sm_memcpy(pDataBuffer, pSense, MIN(SENSE_DATA_LENGTH, allocationLen)); 9119 9120 /*smEnqueueIO(smRoot, satIOContext);*/ 9121 9122 tdsmIOCompletedCB( smRoot, 9123 smIORequest, 9124 smIOSuccess, 9125 SCSI_STAT_CHECK_CONDITION, 9126 satIOContext->pSmSenseData, 9127 satIOContext->interruptContext ); 9128 9129 SM_DBG1(("smsatRequestSense: return control!!!\n")); 9130 return SM_RC_SUCCESS; 9131 } 9132 9133 /* 9134 Only fixed format sense data is support. In other words, we don't support DESC bit is set 9135 in Request Sense 9136 */ 9137 if ( scsiCmnd->cdb[1] & ATA_REMOVABLE_MEDIA_DEVICE_MASK ) 9138 { 9139 smsatSetSensePayload( pSense, 9140 SCSI_SNSKEY_ILLEGAL_REQUEST, 9141 0, 9142 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 9143 satIOContext); 9144 sm_memcpy(pDataBuffer, pSense, MIN(SENSE_DATA_LENGTH, allocationLen)); 9145 9146 /*smEnqueueIO(smRoot, satIOContext);*/ 9147 9148 tdsmIOCompletedCB( smRoot, 9149 smIORequest, 9150 smIOSuccess, 9151 SCSI_STAT_CHECK_CONDITION, 9152 satIOContext->pSmSenseData, 9153 satIOContext->interruptContext ); 9154 9155 SM_DBG1(("smsatRequestSense: DESC bit is set, which we don't support!!!\n")); 9156 return SM_RC_SUCCESS; 9157 } 9158 9159 9160 if (pSatDevData->satSMARTEnabled == agTRUE) 9161 { 9162 /* sends SMART RETURN STATUS */ 9163 fis->h.fisType = 0x27; /* Reg host to device */ 9164 fis->h.c_pmPort = 0x80; /* C Bit is set */ 9165 9166 fis->h.command = SAT_SMART; /* 0xB0 */ 9167 fis->h.features = SAT_SMART_RETURN_STATUS; /* FIS features */ 9168 fis->d.featuresExp = 0; /* FIS reserve */ 9169 fis->d.sectorCount = 0; /* FIS sector count (7:0) */ 9170 fis->d.sectorCountExp = 0; /* FIS sector count (15:8) */ 9171 fis->d.lbaLow = 0; /* FIS LBA (7 :0 ) */ 9172 fis->d.lbaLowExp = 0; /* FIS LBA (31:24) */ 9173 fis->d.lbaMid = 0x4F; /* FIS LBA (15:8 ) */ 9174 fis->d.lbaMidExp = 0; /* FIS LBA (39:32) */ 9175 fis->d.lbaHigh = 0xC2; /* FIS LBA (23:16) */ 9176 fis->d.lbaHighExp = 0; /* FIS LBA (47:40) */ 9177 fis->d.device = 0; /* FIS DEV is discared in SATA */ 9178 fis->d.control = 0; /* FIS HOB bit clear */ 9179 fis->d.reserved4 = 0; 9180 fis->d.reserved5 = 0; 9181 9182 agRequestType = AGSA_SATA_PROTOCOL_NON_DATA; 9183 /* Initialize CB for SATA completion. 9184 */ 9185 satIOContext->satCompleteCB = &smsatRequestSenseCB; 9186 9187 /* 9188 * Prepare SGL and send FIS to LL layer. 9189 */ 9190 satIOContext->reqType = agRequestType; /* Save it */ 9191 9192 status = smsataLLIOStart( smRoot, 9193 smIORequest, 9194 smDeviceHandle, 9195 smScsiRequest, 9196 satIOContext); 9197 9198 SM_DBG4(("smsatRequestSense: if return, status %d\n", status)); 9199 return (status); 9200 } 9201 else 9202 { 9203 /*allocate iocontext for xmitting xmit SAT_CHECK_POWER_MODE 9204 then call satRequestSense2 */ 9205 9206 SM_DBG4(("smsatRequestSense: before satIntIo %p\n", satIntIo)); 9207 /* allocate iocontext */ 9208 satIntIo = smsatAllocIntIoResource( smRoot, 9209 smIORequest, /* original request */ 9210 pSatDevData, 9211 smScsiRequest->scsiCmnd.expDataLength, 9212 satIntIo); 9213 9214 SM_DBG4(("smsatRequestSense: after satIntIo %p\n", satIntIo)); 9215 9216 if (satIntIo == agNULL) 9217 { 9218 /* failed during sending SMART RETURN STATUS */ 9219 smsatSetSensePayload( pSense, 9220 SCSI_SNSKEY_NO_SENSE, 9221 0, 9222 SCSI_SNSCODE_HARDWARE_IMPENDING_FAILURE, 9223 satIOContext); 9224 sm_memcpy(pDataBuffer, pSense, MIN(SENSE_DATA_LENGTH, allocationLen)); 9225 9226 /*smEnqueueIO(smRoot, satIOContext);*/ 9227 9228 tdsmIOCompletedCB( smRoot, 9229 smIORequest, 9230 smIOSuccess, 9231 SCSI_STAT_GOOD, 9232 agNULL, 9233 satIOContext->interruptContext ); 9234 9235 SM_DBG1(("smsatRequestSense: else fail 1!!!\n")); 9236 return SM_RC_SUCCESS; 9237 } /* end of memory allocation failure */ 9238 9239 9240 /* 9241 * Need to initialize all the fields within satIOContext except 9242 * reqType and satCompleteCB which will be set depending on cmd. 9243 */ 9244 9245 if (satIntIo == agNULL) 9246 { 9247 SM_DBG4(("smsatRequestSense: satIntIo is NULL\n")); 9248 } 9249 else 9250 { 9251 SM_DBG4(("smsatRequestSense: satIntIo is NOT NULL\n")); 9252 } 9253 /* use this --- tttttthe one the same */ 9254 9255 9256 satIntIo->satOrgSmIORequest = smIORequest; 9257 smIORequestBody = (smIORequestBody_t *)satIntIo->satIntRequestBody; 9258 satIOContext2 = &(smIORequestBody->transport.SATA.satIOContext); 9259 9260 satIOContext2->pSatDevData = pSatDevData; 9261 satIOContext2->pFis = &(smIORequestBody->transport.SATA.agSATARequestBody.fis.fisRegHostToDev); 9262 satIOContext2->pScsiCmnd = &(satIntIo->satIntSmScsiXchg.scsiCmnd); 9263 satIOContext2->pSense = &(smIORequestBody->transport.SATA.sensePayload); 9264 satIOContext2->pSmSenseData = &(smIORequestBody->transport.SATA.smSenseData); 9265 satIOContext2->pSmSenseData->senseData = satIOContext2->pSense; 9266 satIOContext2->smRequestBody = satIntIo->satIntRequestBody; 9267 satIOContext2->interruptContext = satIOContext->interruptContext; 9268 satIOContext2->satIntIoContext = satIntIo; 9269 satIOContext2->psmDeviceHandle = smDeviceHandle; 9270 satIOContext2->satOrgIOContext = satIOContext; 9271 9272 SM_DBG4(("smsatRequestSense: satIntIo->satIntSmScsiXchg.agSgl1.len %d\n", satIntIo->satIntSmScsiXchg.smSgl1.len)); 9273 9274 SM_DBG4(("smsatRequestSense: satIntIo->satIntSmScsiXchg.agSgl1.upper %d\n", satIntIo->satIntSmScsiXchg.smSgl1.upper)); 9275 9276 SM_DBG4(("smsatRequestSense: satIntIo->satIntSmScsiXchg.agSgl1.lower %d\n", satIntIo->satIntSmScsiXchg.smSgl1.lower)); 9277 9278 SM_DBG4(("smsatRequestSense: satIntIo->satIntSmScsiXchg.agSgl1.type %d\n", satIntIo->satIntSmScsiXchg.smSgl1.type)); 9279 9280 status = smsatRequestSense_1( smRoot, 9281 &(satIntIo->satIntSmIORequest), 9282 smDeviceHandle, 9283 &(satIntIo->satIntSmScsiXchg), 9284 satIOContext2); 9285 9286 if (status != SM_RC_SUCCESS) 9287 { 9288 smsatFreeIntIoResource( smRoot, 9289 pSatDevData, 9290 satIntIo); 9291 9292 /* failed during sending SMART RETURN STATUS */ 9293 smsatSetSensePayload( pSense, 9294 SCSI_SNSKEY_NO_SENSE, 9295 0, 9296 SCSI_SNSCODE_HARDWARE_IMPENDING_FAILURE, 9297 satIOContext); 9298 sm_memcpy(pDataBuffer, pSense, MIN(SENSE_DATA_LENGTH, allocationLen)); 9299 9300 /*smEnqueueIO(smRoot, satIOContext);*/ 9301 9302 tdsmIOCompletedCB( smRoot, 9303 smIORequest, 9304 smIOSuccess, 9305 SCSI_STAT_CHECK_CONDITION, 9306 agNULL, 9307 satIOContext->interruptContext ); 9308 9309 SM_DBG1(("smsatRequestSense: else fail 2!!!\n")); 9310 return SM_RC_SUCCESS; 9311 } 9312 SM_DBG4(("smsatRequestSense: else return success\n")); 9313 return SM_RC_SUCCESS; 9314 } 9315 } 9316 9317 osGLOBAL bit32 9318 smsatRequestSense_1( 9319 smRoot_t *smRoot, 9320 smIORequest_t *smIORequest, 9321 smDeviceHandle_t *smDeviceHandle, 9322 smScsiInitiatorRequest_t *smScsiRequest, 9323 smSatIOContext_t *satIOContext 9324 ) 9325 { 9326 /* 9327 sends SAT_CHECK_POWER_MODE 9328 */ 9329 bit32 status; 9330 bit32 agRequestType; 9331 agsaFisRegHostToDevice_t *fis; 9332 9333 fis = satIOContext->pFis; 9334 SM_DBG5(("smsatRequestSense_1: start\n")); 9335 /* 9336 * Send the ATA CHECK POWER MODE command. 9337 */ 9338 fis->h.fisType = 0x27; /* Reg host to device */ 9339 fis->h.c_pmPort = 0x80; /* C Bit is set */ 9340 fis->h.command = SAT_CHECK_POWER_MODE; /* 0xE5 */ 9341 fis->h.features = 0; 9342 fis->d.lbaLow = 0; 9343 fis->d.lbaMid = 0; 9344 fis->d.lbaHigh = 0; 9345 fis->d.device = 0; 9346 fis->d.lbaLowExp = 0; 9347 fis->d.lbaMidExp = 0; 9348 fis->d.lbaHighExp = 0; 9349 fis->d.featuresExp = 0; 9350 fis->d.sectorCount = 0; 9351 fis->d.sectorCountExp = 0; 9352 fis->d.reserved4 = 0; 9353 fis->d.control = 0; /* FIS HOB bit clear */ 9354 fis->d.reserved5 = 0; 9355 9356 agRequestType = AGSA_SATA_PROTOCOL_NON_DATA; 9357 9358 /* Initialize CB for SATA completion. 9359 */ 9360 satIOContext->satCompleteCB = &smsatRequestSenseCB; 9361 9362 /* 9363 * Prepare SGL and send FIS to LL layer. 9364 */ 9365 satIOContext->reqType = agRequestType; /* Save it */ 9366 9367 9368 SM_DBG4(("smsatRequestSense_1: smSgl1.len %d\n", smScsiRequest->smSgl1.len)); 9369 9370 SM_DBG4(("smsatRequestSense_1: smSgl1.upper %d\n", smScsiRequest->smSgl1.upper)); 9371 9372 SM_DBG4(("smsatRequestSense_1: smSgl1.lower %d\n", smScsiRequest->smSgl1.lower)); 9373 9374 SM_DBG4(("smsatRequestSense_1: smSgl1.type %d\n", smScsiRequest->smSgl1.type)); 9375 9376 // smhexdump("smsatRequestSense_1", (bit8 *)fis, sizeof(agsaFisRegHostToDevice_t)); 9377 9378 status = smsataLLIOStart( smRoot, 9379 smIORequest, 9380 smDeviceHandle, 9381 smScsiRequest, 9382 satIOContext); 9383 9384 9385 9386 return status; 9387 } 9388 9389 osGLOBAL bit32 9390 smsatModeSense6( 9391 smRoot_t *smRoot, 9392 smIORequest_t *smIORequest, 9393 smDeviceHandle_t *smDeviceHandle, 9394 smScsiInitiatorRequest_t *smScsiRequest, 9395 smSatIOContext_t *satIOContext 9396 ) 9397 { 9398 smScsiRspSense_t *pSense; 9399 bit32 allocationLen; 9400 smIniScsiCmnd_t *scsiCmnd; 9401 bit32 pageSupported; 9402 bit8 page; 9403 bit8 *pModeSense; /* Mode Sense data buffer */ 9404 smDeviceData_t *pSatDevData; 9405 bit8 PC; 9406 bit8 AllPages[MODE_SENSE6_RETURN_ALL_PAGES_LEN]; 9407 bit8 Control[MODE_SENSE6_CONTROL_PAGE_LEN]; 9408 bit8 RWErrorRecovery[MODE_SENSE6_READ_WRITE_ERROR_RECOVERY_PAGE_LEN]; 9409 bit8 Caching[MODE_SENSE6_CACHING_LEN]; 9410 bit8 InfoExceptionCtrl[MODE_SENSE6_INFORMATION_EXCEPTION_CONTROL_PAGE_LEN]; 9411 bit8 lenRead = 0; 9412 9413 9414 pSense = satIOContext->pSense; 9415 scsiCmnd = &smScsiRequest->scsiCmnd; 9416 pModeSense = (bit8 *) smScsiRequest->sglVirtualAddr; 9417 pSatDevData = satIOContext->pSatDevData; 9418 9419 //smhexdump("smsatModeSense6", (bit8 *)scsiCmnd->cdb, 6); 9420 SM_DBG5(("smsatModeSense6: start\n")); 9421 /* checking CONTROL */ 9422 /* NACA == 1 or LINK == 1*/ 9423 if ( (scsiCmnd->cdb[5] & SCSI_NACA_MASK) || (scsiCmnd->cdb[5] & SCSI_LINK_MASK) ) 9424 { 9425 smsatSetSensePayload( pSense, 9426 SCSI_SNSKEY_ILLEGAL_REQUEST, 9427 0, 9428 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 9429 satIOContext); 9430 /*smEnqueueIO(smRoot, satIOContext);*/ 9431 tdsmIOCompletedCB( smRoot, 9432 smIORequest, 9433 smIOSuccess, 9434 SCSI_STAT_CHECK_CONDITION, 9435 satIOContext->pSmSenseData, 9436 satIOContext->interruptContext ); 9437 SM_DBG1(("smsatModeSense6: return control!!!\n")); 9438 return SM_RC_SUCCESS; 9439 } 9440 /* checking PC(Page Control) 9441 SAT revion 8, 8.5.3 p33 and 10.1.2, p66 9442 */ 9443 PC = (bit8)((scsiCmnd->cdb[2]) & SCSI_MODE_SENSE6_PC_MASK); 9444 if (PC != 0) 9445 { 9446 smsatSetSensePayload( pSense, 9447 SCSI_SNSKEY_ILLEGAL_REQUEST, 9448 0, 9449 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 9450 satIOContext); 9451 /*smEnqueueIO(smRoot, satIOContext);*/ 9452 tdsmIOCompletedCB( smRoot, 9453 smIORequest, 9454 smIOSuccess, 9455 SCSI_STAT_CHECK_CONDITION, 9456 satIOContext->pSmSenseData, 9457 satIOContext->interruptContext ); 9458 SM_DBG1(("smsatModeSense6: return due to PC value pc 0x%x!!!\n", PC >> 6)); 9459 return SM_RC_SUCCESS; 9460 } 9461 /* reading PAGE CODE */ 9462 page = (bit8)((scsiCmnd->cdb[2]) & SCSI_MODE_SENSE6_PAGE_CODE_MASK); 9463 9464 9465 SM_DBG5(("smsatModeSense6: page=0x%x\n", page)); 9466 9467 allocationLen = scsiCmnd->cdb[4]; 9468 allocationLen = MIN(allocationLen, scsiCmnd->expDataLength); 9469 /* 9470 Based on page code value, returns a corresponding mode page 9471 note: no support for subpage 9472 */ 9473 switch(page) 9474 { 9475 case MODESENSE_RETURN_ALL_PAGES: 9476 case MODESENSE_CONTROL_PAGE: /* control */ 9477 case MODESENSE_READ_WRITE_ERROR_RECOVERY_PAGE: /* Read-Write Error Recovery */ 9478 case MODESENSE_CACHING: /* caching */ 9479 case MODESENSE_INFORMATION_EXCEPTION_CONTROL_PAGE: /* informational exceptions control*/ 9480 pageSupported = agTRUE; 9481 break; 9482 case MODESENSE_VENDOR_SPECIFIC_PAGE: /* vendor specific */ 9483 default: 9484 pageSupported = agFALSE; 9485 break; 9486 } 9487 9488 if (pageSupported == agFALSE) 9489 { 9490 9491 SM_DBG1(("smsatModeSense6 *** ERROR *** not supported page 0x%x did %d!!!\n", 9492 page, pSatDevData->id)); 9493 9494 smsatSetSensePayload( pSense, 9495 SCSI_SNSKEY_ILLEGAL_REQUEST, 9496 0, 9497 SCSI_SNSCODE_INVALID_COMMAND, 9498 satIOContext); 9499 9500 /*smEnqueueIO(smRoot, satIOContext);*/ 9501 9502 tdsmIOCompletedCB( smRoot, 9503 smIORequest, 9504 smIOSuccess, 9505 SCSI_STAT_CHECK_CONDITION, 9506 satIOContext->pSmSenseData, 9507 satIOContext->interruptContext ); 9508 return SM_RC_SUCCESS; 9509 } 9510 9511 switch(page) 9512 { 9513 case MODESENSE_RETURN_ALL_PAGES: 9514 lenRead = (bit8)MIN(allocationLen, MODE_SENSE6_RETURN_ALL_PAGES_LEN); 9515 break; 9516 case MODESENSE_CONTROL_PAGE: /* control */ 9517 lenRead = (bit8)MIN(allocationLen, MODE_SENSE6_CONTROL_PAGE_LEN); 9518 break; 9519 case MODESENSE_READ_WRITE_ERROR_RECOVERY_PAGE: /* Read-Write Error Recovery */ 9520 lenRead = (bit8)MIN(allocationLen, MODE_SENSE6_READ_WRITE_ERROR_RECOVERY_PAGE_LEN); 9521 break; 9522 case MODESENSE_CACHING: /* caching */ 9523 lenRead = (bit8)MIN(allocationLen, MODE_SENSE6_CACHING_LEN); 9524 break; 9525 case MODESENSE_INFORMATION_EXCEPTION_CONTROL_PAGE: /* informational exceptions control*/ 9526 lenRead = (bit8)MIN(allocationLen, MODE_SENSE6_INFORMATION_EXCEPTION_CONTROL_PAGE_LEN); 9527 break; 9528 default: 9529 SM_DBG1(("smsatModeSense6: default error page %d!!!\n", page)); 9530 break; 9531 } 9532 9533 if (page == MODESENSE_RETURN_ALL_PAGES) 9534 { 9535 SM_DBG5(("smsatModeSense6: MODESENSE_RETURN_ALL_PAGES\n")); 9536 AllPages[0] = (bit8)(lenRead - 1); 9537 AllPages[1] = 0x00; /* default medium type (currently mounted medium type) */ 9538 AllPages[2] = 0x00; /* no write-protect, no support for DPO-FUA */ 9539 AllPages[3] = 0x08; /* block descriptor length */ 9540 9541 /* 9542 * Fill-up direct-access device block-descriptor, SAT, Table 19 9543 */ 9544 9545 /* density code */ 9546 AllPages[4] = 0x04; /* density-code : reserved for direct-access */ 9547 /* number of blocks */ 9548 AllPages[5] = 0x00; /* unspecified */ 9549 AllPages[6] = 0x00; /* unspecified */ 9550 AllPages[7] = 0x00; /* unspecified */ 9551 /* reserved */ 9552 AllPages[8] = 0x00; /* reserved */ 9553 /* Block size */ 9554 AllPages[9] = 0x00; 9555 AllPages[10] = 0x02; /* Block size is always 512 bytes */ 9556 AllPages[11] = 0x00; 9557 9558 /* MODESENSE_READ_WRITE_ERROR_RECOVERY_PAGE */ 9559 AllPages[12] = 0x01; /* page code */ 9560 AllPages[13] = 0x0A; /* page length */ 9561 AllPages[14] = 0x40; /* ARRE is set */ 9562 AllPages[15] = 0x00; 9563 AllPages[16] = 0x00; 9564 AllPages[17] = 0x00; 9565 AllPages[18] = 0x00; 9566 AllPages[19] = 0x00; 9567 AllPages[20] = 0x00; 9568 AllPages[21] = 0x00; 9569 AllPages[22] = 0x00; 9570 AllPages[23] = 0x00; 9571 /* MODESENSE_CACHING */ 9572 AllPages[24] = 0x08; /* page code */ 9573 AllPages[25] = 0x12; /* page length */ 9574 if (pSatDevData->satWriteCacheEnabled == agTRUE) 9575 { 9576 AllPages[26] = 0x04;/* WCE bit is set */ 9577 } 9578 else 9579 { 9580 AllPages[26] = 0x00;/* WCE bit is NOT set */ 9581 } 9582 9583 AllPages[27] = 0x00; 9584 AllPages[28] = 0x00; 9585 AllPages[29] = 0x00; 9586 AllPages[30] = 0x00; 9587 AllPages[31] = 0x00; 9588 AllPages[32] = 0x00; 9589 AllPages[33] = 0x00; 9590 AllPages[34] = 0x00; 9591 AllPages[35] = 0x00; 9592 if (pSatDevData->satLookAheadEnabled == agTRUE) 9593 { 9594 AllPages[36] = 0x00;/* DRA bit is NOT set */ 9595 } 9596 else 9597 { 9598 AllPages[36] = 0x20;/* DRA bit is set */ 9599 } 9600 AllPages[37] = 0x00; 9601 AllPages[38] = 0x00; 9602 AllPages[39] = 0x00; 9603 AllPages[40] = 0x00; 9604 AllPages[41] = 0x00; 9605 AllPages[42] = 0x00; 9606 AllPages[43] = 0x00; 9607 /* MODESENSE_CONTROL_PAGE */ 9608 AllPages[44] = 0x0A; /* page code */ 9609 AllPages[45] = 0x0A; /* page length */ 9610 AllPages[46] = 0x02; /* only GLTSD bit is set */ 9611 if (pSatDevData->satNCQ == agTRUE) 9612 { 9613 AllPages[47] = 0x12; /* Queue Alogorithm modifier 1b and QErr 01b*/ 9614 } 9615 else 9616 { 9617 AllPages[47] = 0x02; /* Queue Alogorithm modifier 0b and QErr 01b */ 9618 } 9619 AllPages[48] = 0x00; 9620 AllPages[49] = 0x00; 9621 AllPages[50] = 0x00; /* obsolete */ 9622 AllPages[51] = 0x00; /* obsolete */ 9623 AllPages[52] = 0xFF; /* Busy Timeout Period */ 9624 AllPages[53] = 0xFF; /* Busy Timeout Period */ 9625 AllPages[54] = 0x00; /* we don't support non-000b value for the self-test code */ 9626 AllPages[55] = 0x00; /* we don't support non-000b value for the self-test code */ 9627 /* MODESENSE_INFORMATION_EXCEPTION_CONTROL_PAGE */ 9628 AllPages[56] = 0x1C; /* page code */ 9629 AllPages[57] = 0x0A; /* page length */ 9630 if (pSatDevData->satSMARTEnabled == agTRUE) 9631 { 9632 AllPages[58] = 0x00;/* DEXCPT bit is NOT set */ 9633 } 9634 else 9635 { 9636 AllPages[58] = 0x08;/* DEXCPT bit is set */ 9637 } 9638 AllPages[59] = 0x00; /* We don't support MRIE */ 9639 AllPages[60] = 0x00; /* Interval timer vendor-specific */ 9640 AllPages[61] = 0x00; 9641 AllPages[62] = 0x00; 9642 AllPages[63] = 0x00; 9643 AllPages[64] = 0x00; /* REPORT-COUNT */ 9644 AllPages[65] = 0x00; 9645 AllPages[66] = 0x00; 9646 AllPages[67] = 0x00; 9647 9648 sm_memcpy(pModeSense, &AllPages, lenRead); 9649 } 9650 else if (page == MODESENSE_CONTROL_PAGE) 9651 { 9652 SM_DBG5(("smsatModeSense6: MODESENSE_CONTROL_PAGE\n")); 9653 Control[0] = MODE_SENSE6_CONTROL_PAGE_LEN - 1; 9654 Control[1] = 0x00; /* default medium type (currently mounted medium type) */ 9655 Control[2] = 0x00; /* no write-protect, no support for DPO-FUA */ 9656 Control[3] = 0x08; /* block descriptor length */ 9657 /* 9658 * Fill-up direct-access device block-descriptor, SAT, Table 19 9659 */ 9660 9661 /* density code */ 9662 Control[4] = 0x04; /* density-code : reserved for direct-access */ 9663 /* number of blocks */ 9664 Control[5] = 0x00; /* unspecified */ 9665 Control[6] = 0x00; /* unspecified */ 9666 Control[7] = 0x00; /* unspecified */ 9667 /* reserved */ 9668 Control[8] = 0x00; /* reserved */ 9669 /* Block size */ 9670 Control[9] = 0x00; 9671 Control[10] = 0x02; /* Block size is always 512 bytes */ 9672 Control[11] = 0x00; 9673 /* 9674 * Fill-up control mode page, SAT, Table 65 9675 */ 9676 Control[12] = 0x0A; /* page code */ 9677 Control[13] = 0x0A; /* page length */ 9678 Control[14] = 0x02; /* only GLTSD bit is set */ 9679 if (pSatDevData->satNCQ == agTRUE) 9680 { 9681 Control[15] = 0x12; /* Queue Alogorithm modifier 1b and QErr 01b*/ 9682 } 9683 else 9684 { 9685 Control[15] = 0x02; /* Queue Alogorithm modifier 0b and QErr 01b */ 9686 } 9687 Control[16] = 0x00; 9688 Control[17] = 0x00; 9689 Control[18] = 0x00; /* obsolete */ 9690 Control[19] = 0x00; /* obsolete */ 9691 Control[20] = 0xFF; /* Busy Timeout Period */ 9692 Control[21] = 0xFF; /* Busy Timeout Period */ 9693 Control[22] = 0x00; /* we don't support non-000b value for the self-test code */ 9694 Control[23] = 0x00; /* we don't support non-000b value for the self-test code */ 9695 9696 sm_memcpy(pModeSense, &Control, lenRead); 9697 9698 } 9699 else if (page == MODESENSE_READ_WRITE_ERROR_RECOVERY_PAGE) 9700 { 9701 SM_DBG5(("smsatModeSense6: MODESENSE_READ_WRITE_ERROR_RECOVERY_PAGE\n")); 9702 RWErrorRecovery[0] = MODE_SENSE6_READ_WRITE_ERROR_RECOVERY_PAGE_LEN - 1; 9703 RWErrorRecovery[1] = 0x00; /* default medium type (currently mounted medium type) */ 9704 RWErrorRecovery[2] = 0x00; /* no write-protect, no support for DPO-FUA */ 9705 RWErrorRecovery[3] = 0x08; /* block descriptor length */ 9706 /* 9707 * Fill-up direct-access device block-descriptor, SAT, Table 19 9708 */ 9709 9710 /* density code */ 9711 RWErrorRecovery[4] = 0x04; /* density-code : reserved for direct-access */ 9712 /* number of blocks */ 9713 RWErrorRecovery[5] = 0x00; /* unspecified */ 9714 RWErrorRecovery[6] = 0x00; /* unspecified */ 9715 RWErrorRecovery[7] = 0x00; /* unspecified */ 9716 /* reserved */ 9717 RWErrorRecovery[8] = 0x00; /* reserved */ 9718 /* Block size */ 9719 RWErrorRecovery[9] = 0x00; 9720 RWErrorRecovery[10] = 0x02; /* Block size is always 512 bytes */ 9721 RWErrorRecovery[11] = 0x00; 9722 /* 9723 * Fill-up Read-Write Error Recovery mode page, SAT, Table 66 9724 */ 9725 RWErrorRecovery[12] = 0x01; /* page code */ 9726 RWErrorRecovery[13] = 0x0A; /* page length */ 9727 RWErrorRecovery[14] = 0x40; /* ARRE is set */ 9728 RWErrorRecovery[15] = 0x00; 9729 RWErrorRecovery[16] = 0x00; 9730 RWErrorRecovery[17] = 0x00; 9731 RWErrorRecovery[18] = 0x00; 9732 RWErrorRecovery[19] = 0x00; 9733 RWErrorRecovery[20] = 0x00; 9734 RWErrorRecovery[21] = 0x00; 9735 RWErrorRecovery[22] = 0x00; 9736 RWErrorRecovery[23] = 0x00; 9737 9738 sm_memcpy(pModeSense, &RWErrorRecovery, lenRead); 9739 9740 } 9741 else if (page == MODESENSE_CACHING) 9742 { 9743 SM_DBG5(("smsatModeSense6: MODESENSE_CACHING\n")); 9744 /* special case */ 9745 if (allocationLen == 4 && page == MODESENSE_CACHING) 9746 { 9747 SM_DBG5(("smsatModeSense6: linux 2.6.8.24 support\n")); 9748 9749 Caching[0] = 0x20 - 1; /* 32 - 1 */ 9750 Caching[1] = 0x00; /* default medium type (currently mounted medium type) */ 9751 Caching[2] = 0x00; /* no write-protect, no support for DPO-FUA */ 9752 Caching[3] = 0x08; /* block descriptor length */ 9753 9754 sm_memcpy(pModeSense, &Caching, 4); 9755 /*smEnqueueIO(smRoot, satIOContext);*/ 9756 9757 tdsmIOCompletedCB( smRoot, 9758 smIORequest, 9759 smIOSuccess, 9760 SCSI_STAT_GOOD, 9761 agNULL, 9762 satIOContext->interruptContext); 9763 return SM_RC_SUCCESS; 9764 } 9765 Caching[0] = MODE_SENSE6_CACHING_LEN - 1; 9766 Caching[1] = 0x00; /* default medium type (currently mounted medium type) */ 9767 Caching[2] = 0x00; /* no write-protect, no support for DPO-FUA */ 9768 Caching[3] = 0x08; /* block descriptor length */ 9769 /* 9770 * Fill-up direct-access device block-descriptor, SAT, Table 19 9771 */ 9772 9773 /* density code */ 9774 Caching[4] = 0x04; /* density-code : reserved for direct-access */ 9775 /* number of blocks */ 9776 Caching[5] = 0x00; /* unspecified */ 9777 Caching[6] = 0x00; /* unspecified */ 9778 Caching[7] = 0x00; /* unspecified */ 9779 /* reserved */ 9780 Caching[8] = 0x00; /* reserved */ 9781 /* Block size */ 9782 Caching[9] = 0x00; 9783 Caching[10] = 0x02; /* Block size is always 512 bytes */ 9784 Caching[11] = 0x00; 9785 /* 9786 * Fill-up Caching mode page, SAT, Table 67 9787 */ 9788 /* length 20 */ 9789 Caching[12] = 0x08; /* page code */ 9790 Caching[13] = 0x12; /* page length */ 9791 if (pSatDevData->satWriteCacheEnabled == agTRUE) 9792 { 9793 Caching[14] = 0x04;/* WCE bit is set */ 9794 } 9795 else 9796 { 9797 Caching[14] = 0x00;/* WCE bit is NOT set */ 9798 } 9799 9800 Caching[15] = 0x00; 9801 Caching[16] = 0x00; 9802 Caching[17] = 0x00; 9803 Caching[18] = 0x00; 9804 Caching[19] = 0x00; 9805 Caching[20] = 0x00; 9806 Caching[21] = 0x00; 9807 Caching[22] = 0x00; 9808 Caching[23] = 0x00; 9809 if (pSatDevData->satLookAheadEnabled == agTRUE) 9810 { 9811 Caching[24] = 0x00;/* DRA bit is NOT set */ 9812 } 9813 else 9814 { 9815 Caching[24] = 0x20;/* DRA bit is set */ 9816 } 9817 Caching[25] = 0x00; 9818 Caching[26] = 0x00; 9819 Caching[27] = 0x00; 9820 Caching[28] = 0x00; 9821 Caching[29] = 0x00; 9822 Caching[30] = 0x00; 9823 Caching[31] = 0x00; 9824 9825 sm_memcpy(pModeSense, &Caching, lenRead); 9826 9827 } 9828 else if (page == MODESENSE_INFORMATION_EXCEPTION_CONTROL_PAGE) 9829 { 9830 SM_DBG5(("smsatModeSense6: MODESENSE_INFORMATION_EXCEPTION_CONTROL_PAGE\n")); 9831 InfoExceptionCtrl[0] = MODE_SENSE6_INFORMATION_EXCEPTION_CONTROL_PAGE_LEN - 1; 9832 InfoExceptionCtrl[1] = 0x00; /* default medium type (currently mounted medium type) */ 9833 InfoExceptionCtrl[2] = 0x00; /* no write-protect, no support for DPO-FUA */ 9834 InfoExceptionCtrl[3] = 0x08; /* block descriptor length */ 9835 /* 9836 * Fill-up direct-access device block-descriptor, SAT, Table 19 9837 */ 9838 9839 /* density code */ 9840 InfoExceptionCtrl[4] = 0x04; /* density-code : reserved for direct-access */ 9841 /* number of blocks */ 9842 InfoExceptionCtrl[5] = 0x00; /* unspecified */ 9843 InfoExceptionCtrl[6] = 0x00; /* unspecified */ 9844 InfoExceptionCtrl[7] = 0x00; /* unspecified */ 9845 /* reserved */ 9846 InfoExceptionCtrl[8] = 0x00; /* reserved */ 9847 /* Block size */ 9848 InfoExceptionCtrl[9] = 0x00; 9849 InfoExceptionCtrl[10] = 0x02; /* Block size is always 512 bytes */ 9850 InfoExceptionCtrl[11] = 0x00; 9851 /* 9852 * Fill-up informational-exceptions control mode page, SAT, Table 68 9853 */ 9854 InfoExceptionCtrl[12] = 0x1C; /* page code */ 9855 InfoExceptionCtrl[13] = 0x0A; /* page length */ 9856 if (pSatDevData->satSMARTEnabled == agTRUE) 9857 { 9858 InfoExceptionCtrl[14] = 0x00;/* DEXCPT bit is NOT set */ 9859 } 9860 else 9861 { 9862 InfoExceptionCtrl[14] = 0x08;/* DEXCPT bit is set */ 9863 } 9864 InfoExceptionCtrl[15] = 0x00; /* We don't support MRIE */ 9865 InfoExceptionCtrl[16] = 0x00; /* Interval timer vendor-specific */ 9866 InfoExceptionCtrl[17] = 0x00; 9867 InfoExceptionCtrl[18] = 0x00; 9868 InfoExceptionCtrl[19] = 0x00; 9869 InfoExceptionCtrl[20] = 0x00; /* REPORT-COUNT */ 9870 InfoExceptionCtrl[21] = 0x00; 9871 InfoExceptionCtrl[22] = 0x00; 9872 InfoExceptionCtrl[23] = 0x00; 9873 sm_memcpy(pModeSense, &InfoExceptionCtrl, lenRead); 9874 9875 } 9876 else 9877 { 9878 /* Error */ 9879 SM_DBG1(("smsatModeSense6: Error page %d!!!\n", page)); 9880 smsatSetSensePayload( pSense, 9881 SCSI_SNSKEY_ILLEGAL_REQUEST, 9882 0, 9883 SCSI_SNSCODE_INVALID_COMMAND, 9884 satIOContext); 9885 9886 /*smEnqueueIO(smRoot, satIOContext);*/ 9887 9888 tdsmIOCompletedCB( smRoot, 9889 smIORequest, 9890 smIOSuccess, 9891 SCSI_STAT_CHECK_CONDITION, 9892 satIOContext->pSmSenseData, 9893 satIOContext->interruptContext ); 9894 return SM_RC_SUCCESS; 9895 } 9896 9897 /* there can be only underrun not overrun in error case */ 9898 if (allocationLen > lenRead) 9899 { 9900 SM_DBG6(("smsatModeSense6 reporting underrun lenRead=0x%x allocationLen=0x%x\n", lenRead, allocationLen)); 9901 9902 /*smEnqueueIO(smRoot, satIOContext);*/ 9903 9904 tdsmIOCompletedCB( smRoot, 9905 smIORequest, 9906 smIOUnderRun, 9907 allocationLen - lenRead, 9908 agNULL, 9909 satIOContext->interruptContext ); 9910 9911 9912 } 9913 else 9914 { 9915 /*smEnqueueIO(smRoot, satIOContext);*/ 9916 9917 tdsmIOCompletedCB( smRoot, 9918 smIORequest, 9919 smIOSuccess, 9920 SCSI_STAT_GOOD, 9921 agNULL, 9922 satIOContext->interruptContext); 9923 } 9924 9925 return SM_RC_SUCCESS; 9926 9927 } 9928 9929 osGLOBAL bit32 9930 smsatModeSense10( 9931 smRoot_t *smRoot, 9932 smIORequest_t *smIORequest, 9933 smDeviceHandle_t *smDeviceHandle, 9934 smScsiInitiatorRequest_t *smScsiRequest, 9935 smSatIOContext_t *satIOContext 9936 ) 9937 { 9938 smScsiRspSense_t *pSense; 9939 bit32 allocationLen; 9940 smIniScsiCmnd_t *scsiCmnd; 9941 bit32 pageSupported; 9942 bit8 page; 9943 bit8 *pModeSense; /* Mode Sense data buffer */ 9944 smDeviceData_t *pSatDevData; 9945 bit8 PC; /* page control */ 9946 bit8 LLBAA; /* Long LBA Accepted */ 9947 bit32 index; 9948 bit8 AllPages[MODE_SENSE10_RETURN_ALL_PAGES_LLBAA_LEN]; 9949 bit8 Control[MODE_SENSE10_CONTROL_PAGE_LLBAA_LEN]; 9950 bit8 RWErrorRecovery[MODE_SENSE10_READ_WRITE_ERROR_RECOVERY_PAGE_LLBAA_LEN]; 9951 bit8 Caching[MODE_SENSE10_CACHING_LLBAA_LEN]; 9952 bit8 InfoExceptionCtrl[MODE_SENSE10_INFORMATION_EXCEPTION_CONTROL_PAGE_LLBAA_LEN]; 9953 bit8 lenRead = 0; 9954 9955 pSense = satIOContext->pSense; 9956 scsiCmnd = &smScsiRequest->scsiCmnd; 9957 pModeSense = (bit8 *) smScsiRequest->sglVirtualAddr; 9958 pSatDevData = satIOContext->pSatDevData; 9959 SM_DBG5(("smsatModeSense10: start\n")); 9960 /* checking CONTROL */ 9961 /* NACA == 1 or LINK == 1*/ 9962 if ( (scsiCmnd->cdb[9] & SCSI_NACA_MASK) || (scsiCmnd->cdb[9] & SCSI_LINK_MASK) ) 9963 { 9964 smsatSetSensePayload( pSense, 9965 SCSI_SNSKEY_ILLEGAL_REQUEST, 9966 0, 9967 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 9968 satIOContext); 9969 9970 /*smEnqueueIO(smRoot, satIOContext);*/ 9971 9972 tdsmIOCompletedCB( smRoot, 9973 smIORequest, 9974 smIOSuccess, 9975 SCSI_STAT_CHECK_CONDITION, 9976 satIOContext->pSmSenseData, 9977 satIOContext->interruptContext ); 9978 9979 SM_DBG1(("smsatModeSense10: return control!!!\n")); 9980 return SM_RC_SUCCESS; 9981 } 9982 9983 /* checking PC(Page Control) 9984 SAT revion 8, 8.5.3 p33 and 10.1.2, p66 9985 */ 9986 PC = (bit8)((scsiCmnd->cdb[2]) & SCSI_MODE_SENSE10_PC_MASK); 9987 if (PC != 0) 9988 { 9989 smsatSetSensePayload( pSense, 9990 SCSI_SNSKEY_ILLEGAL_REQUEST, 9991 0, 9992 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 9993 satIOContext); 9994 9995 /*smEnqueueIO(smRoot, satIOContext);*/ 9996 9997 tdsmIOCompletedCB( smRoot, 9998 smIORequest, 9999 smIOSuccess, 10000 SCSI_STAT_CHECK_CONDITION, 10001 satIOContext->pSmSenseData, 10002 satIOContext->interruptContext ); 10003 10004 SM_DBG1(("smsatModeSense10: return due to PC value pc 0x%x!!!\n", PC)); 10005 return SM_RC_SUCCESS; 10006 } 10007 10008 /* finding LLBAA bit */ 10009 LLBAA = (bit8)((scsiCmnd->cdb[1]) & SCSI_MODE_SENSE10_LLBAA_MASK); 10010 10011 /* reading PAGE CODE */ 10012 page = (bit8)((scsiCmnd->cdb[2]) & SCSI_MODE_SENSE10_PAGE_CODE_MASK); 10013 SM_DBG5(("smsatModeSense10: page=0x%x, did %d\n", page, pSatDevData->id)); 10014 allocationLen = (scsiCmnd->cdb[7] << 8) + scsiCmnd->cdb[8]; 10015 allocationLen = MIN(allocationLen, scsiCmnd->expDataLength); 10016 10017 /* 10018 Based on page code value, returns a corresponding mode page 10019 note: no support for subpage 10020 */ 10021 switch(page) 10022 { 10023 case MODESENSE_RETURN_ALL_PAGES: /* return all pages */ 10024 case MODESENSE_CONTROL_PAGE: /* control */ 10025 case MODESENSE_READ_WRITE_ERROR_RECOVERY_PAGE: /* Read-Write Error Recovery */ 10026 case MODESENSE_CACHING: /* caching */ 10027 case MODESENSE_INFORMATION_EXCEPTION_CONTROL_PAGE: /* informational exceptions control*/ 10028 pageSupported = agTRUE; 10029 break; 10030 case MODESENSE_VENDOR_SPECIFIC_PAGE: /* vendor specific */ 10031 default: 10032 pageSupported = agFALSE; 10033 break; 10034 } 10035 if (pageSupported == agFALSE) 10036 { 10037 SM_DBG1(("smsatModeSense10 *** ERROR *** not supported page 0x%x did %d!!!\n", page, pSatDevData->id)); 10038 10039 smsatSetSensePayload( pSense, 10040 SCSI_SNSKEY_ILLEGAL_REQUEST, 10041 0, 10042 SCSI_SNSCODE_INVALID_COMMAND, 10043 satIOContext); 10044 /*smEnqueueIO(smRoot, satIOContext);*/ 10045 tdsmIOCompletedCB( smRoot, 10046 smIORequest, 10047 smIOSuccess, 10048 SCSI_STAT_CHECK_CONDITION, 10049 satIOContext->pSmSenseData, 10050 satIOContext->interruptContext ); 10051 return SM_RC_SUCCESS; 10052 } 10053 switch(page) 10054 { 10055 case MODESENSE_RETURN_ALL_PAGES: 10056 if (LLBAA) 10057 { 10058 lenRead = (bit8)MIN(allocationLen, MODE_SENSE10_RETURN_ALL_PAGES_LLBAA_LEN); 10059 } 10060 else 10061 { 10062 lenRead = (bit8)MIN(allocationLen, MODE_SENSE10_RETURN_ALL_PAGES_LEN); 10063 } 10064 break; 10065 case MODESENSE_CONTROL_PAGE: /* control */ 10066 if (LLBAA) 10067 { 10068 lenRead = (bit8)MIN(allocationLen, MODE_SENSE10_CONTROL_PAGE_LLBAA_LEN); 10069 } 10070 else 10071 { 10072 lenRead = (bit8)MIN(allocationLen, MODE_SENSE10_CONTROL_PAGE_LEN); 10073 } 10074 break; 10075 case MODESENSE_READ_WRITE_ERROR_RECOVERY_PAGE: /* Read-Write Error Recovery */ 10076 if (LLBAA) 10077 { 10078 lenRead = (bit8)MIN(allocationLen, MODE_SENSE10_READ_WRITE_ERROR_RECOVERY_PAGE_LLBAA_LEN); 10079 } 10080 else 10081 { 10082 lenRead = (bit8)MIN(allocationLen, MODE_SENSE10_READ_WRITE_ERROR_RECOVERY_PAGE_LEN); 10083 } 10084 break; 10085 case MODESENSE_CACHING: /* caching */ 10086 if (LLBAA) 10087 { 10088 lenRead = (bit8)MIN(allocationLen, MODE_SENSE10_CACHING_LLBAA_LEN); 10089 } 10090 else 10091 { 10092 lenRead = (bit8)MIN(allocationLen, MODE_SENSE10_CACHING_LEN); 10093 } 10094 break; 10095 case MODESENSE_INFORMATION_EXCEPTION_CONTROL_PAGE: /* informational exceptions control*/ 10096 if (LLBAA) 10097 { 10098 lenRead = (bit8)MIN(allocationLen, MODE_SENSE10_INFORMATION_EXCEPTION_CONTROL_PAGE_LLBAA_LEN); 10099 } 10100 else 10101 { 10102 lenRead = (bit8)MIN(allocationLen, MODE_SENSE10_INFORMATION_EXCEPTION_CONTROL_PAGE_LEN); 10103 } 10104 break; 10105 default: 10106 SM_DBG1(("smsatModeSense10: default error page %d!!!\n", page)); 10107 break; 10108 } 10109 10110 if (page == MODESENSE_RETURN_ALL_PAGES) 10111 { 10112 SM_DBG5(("smsatModeSense10: MODESENSE_RETURN_ALL_PAGES\n")); 10113 AllPages[0] = 0; 10114 AllPages[1] = (bit8)(lenRead - 2); 10115 AllPages[2] = 0x00; /* medium type: default medium type (currently mounted medium type) */ 10116 AllPages[3] = 0x00; /* device-specific param: no write-protect, no support for DPO-FUA */ 10117 if (LLBAA) 10118 { 10119 AllPages[4] = 0x00; /* reserved and LONGLBA */ 10120 AllPages[4] = (bit8)(AllPages[4] | 0x1); /* LONGLBA is set */ 10121 } 10122 else 10123 { 10124 AllPages[4] = 0x00; /* reserved and LONGLBA: LONGLBA is not set */ 10125 } 10126 AllPages[5] = 0x00; /* reserved */ 10127 AllPages[6] = 0x00; /* block descriptot length */ 10128 if (LLBAA) 10129 { 10130 AllPages[7] = 0x10; /* block descriptor length: LONGLBA is set. So, length is 16 */ 10131 } 10132 else 10133 { 10134 AllPages[7] = 0x08; /* block descriptor length: LONGLBA is NOT set. So, length is 8 */ 10135 } 10136 10137 /* 10138 * Fill-up direct-access device block-descriptor, SAT, Table 19 10139 */ 10140 10141 if (LLBAA) 10142 { 10143 /* density code */ 10144 AllPages[8] = 0x04; /* density-code : reserved for direct-access */ 10145 /* number of blocks */ 10146 AllPages[9] = 0x00; /* unspecified */ 10147 AllPages[10] = 0x00; /* unspecified */ 10148 AllPages[11] = 0x00; /* unspecified */ 10149 AllPages[12] = 0x00; /* unspecified */ 10150 AllPages[13] = 0x00; /* unspecified */ 10151 AllPages[14] = 0x00; /* unspecified */ 10152 AllPages[15] = 0x00; /* unspecified */ 10153 /* reserved */ 10154 AllPages[16] = 0x00; /* reserved */ 10155 AllPages[17] = 0x00; /* reserved */ 10156 AllPages[18] = 0x00; /* reserved */ 10157 AllPages[19] = 0x00; /* reserved */ 10158 /* Block size */ 10159 AllPages[20] = 0x00; 10160 AllPages[21] = 0x00; 10161 AllPages[22] = 0x02; /* Block size is always 512 bytes */ 10162 AllPages[23] = 0x00; 10163 } 10164 else 10165 { 10166 /* density code */ 10167 AllPages[8] = 0x04; /* density-code : reserved for direct-access */ 10168 /* number of blocks */ 10169 AllPages[9] = 0x00; /* unspecified */ 10170 AllPages[10] = 0x00; /* unspecified */ 10171 AllPages[11] = 0x00; /* unspecified */ 10172 /* reserved */ 10173 AllPages[12] = 0x00; /* reserved */ 10174 /* Block size */ 10175 AllPages[13] = 0x00; 10176 AllPages[14] = 0x02; /* Block size is always 512 bytes */ 10177 AllPages[15] = 0x00; 10178 } 10179 10180 if (LLBAA) 10181 { 10182 index = 24; 10183 } 10184 else 10185 { 10186 index = 16; 10187 } 10188 /* MODESENSE_READ_WRITE_ERROR_RECOVERY_PAGE */ 10189 AllPages[index+0] = 0x01; /* page code */ 10190 AllPages[index+1] = 0x0A; /* page length */ 10191 AllPages[index+2] = 0x40; /* ARRE is set */ 10192 AllPages[index+3] = 0x00; 10193 AllPages[index+4] = 0x00; 10194 AllPages[index+5] = 0x00; 10195 AllPages[index+6] = 0x00; 10196 AllPages[index+7] = 0x00; 10197 AllPages[index+8] = 0x00; 10198 AllPages[index+9] = 0x00; 10199 AllPages[index+10] = 0x00; 10200 AllPages[index+11] = 0x00; 10201 10202 /* MODESENSE_CACHING */ 10203 /* 10204 * Fill-up Caching mode page, SAT, Table 67 10205 */ 10206 /* length 20 */ 10207 AllPages[index+12] = 0x08; /* page code */ 10208 AllPages[index+13] = 0x12; /* page length */ 10209 if (pSatDevData->satWriteCacheEnabled == agTRUE) 10210 { 10211 AllPages[index+14] = 0x04;/* WCE bit is set */ 10212 } 10213 else 10214 { 10215 AllPages[index+14] = 0x00;/* WCE bit is NOT set */ 10216 } 10217 10218 AllPages[index+15] = 0x00; 10219 AllPages[index+16] = 0x00; 10220 AllPages[index+17] = 0x00; 10221 AllPages[index+18] = 0x00; 10222 AllPages[index+19] = 0x00; 10223 AllPages[index+20] = 0x00; 10224 AllPages[index+21] = 0x00; 10225 AllPages[index+22] = 0x00; 10226 AllPages[index+23] = 0x00; 10227 if (pSatDevData->satLookAheadEnabled == agTRUE) 10228 { 10229 AllPages[index+24] = 0x00;/* DRA bit is NOT set */ 10230 } 10231 else 10232 { 10233 AllPages[index+24] = 0x20;/* DRA bit is set */ 10234 } 10235 AllPages[index+25] = 0x00; 10236 AllPages[index+26] = 0x00; 10237 AllPages[index+27] = 0x00; 10238 AllPages[index+28] = 0x00; 10239 AllPages[index+29] = 0x00; 10240 AllPages[index+30] = 0x00; 10241 AllPages[index+31] = 0x00; 10242 10243 /* MODESENSE_CONTROL_PAGE */ 10244 /* 10245 * Fill-up control mode page, SAT, Table 65 10246 */ 10247 AllPages[index+32] = 0x0A; /* page code */ 10248 AllPages[index+33] = 0x0A; /* page length */ 10249 AllPages[index+34] = 0x02; /* only GLTSD bit is set */ 10250 if (pSatDevData->satNCQ == agTRUE) 10251 { 10252 AllPages[index+35] = 0x12; /* Queue Alogorithm modifier 1b and QErr 01b*/ 10253 } 10254 else 10255 { 10256 AllPages[index+35] = 0x02; /* Queue Alogorithm modifier 0b and QErr 01b */ 10257 } 10258 AllPages[index+36] = 0x00; 10259 AllPages[index+37] = 0x00; 10260 AllPages[index+38] = 0x00; /* obsolete */ 10261 AllPages[index+39] = 0x00; /* obsolete */ 10262 AllPages[index+40] = 0xFF; /* Busy Timeout Period */ 10263 AllPages[index+41] = 0xFF; /* Busy Timeout Period */ 10264 AllPages[index+42] = 0x00; /* we don't support non-000b value for the self-test code */ 10265 AllPages[index+43] = 0x00; /* we don't support non-000b value for the self-test code */ 10266 10267 /* MODESENSE_INFORMATION_EXCEPTION_CONTROL_PAGE */ 10268 /* 10269 * Fill-up informational-exceptions control mode page, SAT, Table 68 10270 */ 10271 AllPages[index+44] = 0x1C; /* page code */ 10272 AllPages[index+45] = 0x0A; /* page length */ 10273 if (pSatDevData->satSMARTEnabled == agTRUE) 10274 { 10275 AllPages[index+46] = 0x00;/* DEXCPT bit is NOT set */ 10276 } 10277 else 10278 { 10279 AllPages[index+46] = 0x08;/* DEXCPT bit is set */ 10280 } 10281 AllPages[index+47] = 0x00; /* We don't support MRIE */ 10282 AllPages[index+48] = 0x00; /* Interval timer vendor-specific */ 10283 AllPages[index+49] = 0x00; 10284 AllPages[index+50] = 0x00; 10285 AllPages[index+51] = 0x00; 10286 AllPages[index+52] = 0x00; /* REPORT-COUNT */ 10287 AllPages[index+53] = 0x00; 10288 AllPages[index+54] = 0x00; 10289 AllPages[index+55] = 0x00; 10290 10291 sm_memcpy(pModeSense, &AllPages, lenRead); 10292 } 10293 else if (page == MODESENSE_CONTROL_PAGE) 10294 { 10295 SM_DBG5(("smsatModeSense10: MODESENSE_CONTROL_PAGE\n")); 10296 Control[0] = 0; 10297 Control[1] = (bit8)(lenRead - 2); 10298 Control[2] = 0x00; /* medium type: default medium type (currently mounted medium type) */ 10299 Control[3] = 0x00; /* device-specific param: no write-protect, no support for DPO-FUA */ 10300 if (LLBAA) 10301 { 10302 Control[4] = 0x00; /* reserved and LONGLBA */ 10303 Control[4] = (bit8)(Control[4] | 0x1); /* LONGLBA is set */ 10304 } 10305 else 10306 { 10307 Control[4] = 0x00; /* reserved and LONGLBA: LONGLBA is not set */ 10308 } 10309 Control[5] = 0x00; /* reserved */ 10310 Control[6] = 0x00; /* block descriptot length */ 10311 if (LLBAA) 10312 { 10313 Control[7] = 0x10; /* block descriptor length: LONGLBA is set. So, length is 16 */ 10314 } 10315 else 10316 { 10317 Control[7] = 0x08; /* block descriptor length: LONGLBA is NOT set. So, length is 8 */ 10318 } 10319 10320 /* 10321 * Fill-up direct-access device block-descriptor, SAT, Table 19 10322 */ 10323 10324 if (LLBAA) 10325 { 10326 /* density code */ 10327 Control[8] = 0x04; /* density-code : reserved for direct-access */ 10328 /* number of blocks */ 10329 Control[9] = 0x00; /* unspecified */ 10330 Control[10] = 0x00; /* unspecified */ 10331 Control[11] = 0x00; /* unspecified */ 10332 Control[12] = 0x00; /* unspecified */ 10333 Control[13] = 0x00; /* unspecified */ 10334 Control[14] = 0x00; /* unspecified */ 10335 Control[15] = 0x00; /* unspecified */ 10336 /* reserved */ 10337 Control[16] = 0x00; /* reserved */ 10338 Control[17] = 0x00; /* reserved */ 10339 Control[18] = 0x00; /* reserved */ 10340 Control[19] = 0x00; /* reserved */ 10341 /* Block size */ 10342 Control[20] = 0x00; 10343 Control[21] = 0x00; 10344 Control[22] = 0x02; /* Block size is always 512 bytes */ 10345 Control[23] = 0x00; 10346 } 10347 else 10348 { 10349 /* density code */ 10350 Control[8] = 0x04; /* density-code : reserved for direct-access */ 10351 /* number of blocks */ 10352 Control[9] = 0x00; /* unspecified */ 10353 Control[10] = 0x00; /* unspecified */ 10354 Control[11] = 0x00; /* unspecified */ 10355 /* reserved */ 10356 Control[12] = 0x00; /* reserved */ 10357 /* Block size */ 10358 Control[13] = 0x00; 10359 Control[14] = 0x02; /* Block size is always 512 bytes */ 10360 Control[15] = 0x00; 10361 } 10362 10363 if (LLBAA) 10364 { 10365 index = 24; 10366 } 10367 else 10368 { 10369 index = 16; 10370 } 10371 /* 10372 * Fill-up control mode page, SAT, Table 65 10373 */ 10374 Control[index+0] = 0x0A; /* page code */ 10375 Control[index+1] = 0x0A; /* page length */ 10376 Control[index+2] = 0x02; /* only GLTSD bit is set */ 10377 if (pSatDevData->satNCQ == agTRUE) 10378 { 10379 Control[index+3] = 0x12; /* Queue Alogorithm modifier 1b and QErr 01b*/ 10380 } 10381 else 10382 { 10383 Control[index+3] = 0x02; /* Queue Alogorithm modifier 0b and QErr 01b */ 10384 } 10385 Control[index+4] = 0x00; 10386 Control[index+5] = 0x00; 10387 Control[index+6] = 0x00; /* obsolete */ 10388 Control[index+7] = 0x00; /* obsolete */ 10389 Control[index+8] = 0xFF; /* Busy Timeout Period */ 10390 Control[index+9] = 0xFF; /* Busy Timeout Period */ 10391 Control[index+10] = 0x00; /* we don't support non-000b value for the self-test code */ 10392 Control[index+11] = 0x00; /* we don't support non-000b value for the self-test code */ 10393 10394 sm_memcpy(pModeSense, &Control, lenRead); 10395 } 10396 else if (page == MODESENSE_READ_WRITE_ERROR_RECOVERY_PAGE) 10397 { 10398 SM_DBG5(("smsatModeSense10: MODESENSE_READ_WRITE_ERROR_RECOVERY_PAGE\n")); 10399 RWErrorRecovery[0] = 0; 10400 RWErrorRecovery[1] = (bit8)(lenRead - 2); 10401 RWErrorRecovery[2] = 0x00; /* medium type: default medium type (currently mounted medium type) */ 10402 RWErrorRecovery[3] = 0x00; /* device-specific param: no write-protect, no support for DPO-FUA */ 10403 if (LLBAA) 10404 { 10405 RWErrorRecovery[4] = 0x00; /* reserved and LONGLBA */ 10406 RWErrorRecovery[4] = (bit8)(RWErrorRecovery[4] | 0x1); /* LONGLBA is set */ 10407 } 10408 else 10409 { 10410 RWErrorRecovery[4] = 0x00; /* reserved and LONGLBA: LONGLBA is not set */ 10411 } 10412 RWErrorRecovery[5] = 0x00; /* reserved */ 10413 RWErrorRecovery[6] = 0x00; /* block descriptot length */ 10414 if (LLBAA) 10415 { 10416 RWErrorRecovery[7] = 0x10; /* block descriptor length: LONGLBA is set. So, length is 16 */ 10417 } 10418 else 10419 { 10420 RWErrorRecovery[7] = 0x08; /* block descriptor length: LONGLBA is NOT set. So, length is 8 */ 10421 } 10422 10423 /* 10424 * Fill-up direct-access device block-descriptor, SAT, Table 19 10425 */ 10426 10427 if (LLBAA) 10428 { 10429 /* density code */ 10430 RWErrorRecovery[8] = 0x04; /* density-code : reserved for direct-access */ 10431 /* number of blocks */ 10432 RWErrorRecovery[9] = 0x00; /* unspecified */ 10433 RWErrorRecovery[10] = 0x00; /* unspecified */ 10434 RWErrorRecovery[11] = 0x00; /* unspecified */ 10435 RWErrorRecovery[12] = 0x00; /* unspecified */ 10436 RWErrorRecovery[13] = 0x00; /* unspecified */ 10437 RWErrorRecovery[14] = 0x00; /* unspecified */ 10438 RWErrorRecovery[15] = 0x00; /* unspecified */ 10439 /* reserved */ 10440 RWErrorRecovery[16] = 0x00; /* reserved */ 10441 RWErrorRecovery[17] = 0x00; /* reserved */ 10442 RWErrorRecovery[18] = 0x00; /* reserved */ 10443 RWErrorRecovery[19] = 0x00; /* reserved */ 10444 /* Block size */ 10445 RWErrorRecovery[20] = 0x00; 10446 RWErrorRecovery[21] = 0x00; 10447 RWErrorRecovery[22] = 0x02; /* Block size is always 512 bytes */ 10448 RWErrorRecovery[23] = 0x00; 10449 } 10450 else 10451 { 10452 /* density code */ 10453 RWErrorRecovery[8] = 0x04; /* density-code : reserved for direct-access */ 10454 /* number of blocks */ 10455 RWErrorRecovery[9] = 0x00; /* unspecified */ 10456 RWErrorRecovery[10] = 0x00; /* unspecified */ 10457 RWErrorRecovery[11] = 0x00; /* unspecified */ 10458 /* reserved */ 10459 RWErrorRecovery[12] = 0x00; /* reserved */ 10460 /* Block size */ 10461 RWErrorRecovery[13] = 0x00; 10462 RWErrorRecovery[14] = 0x02; /* Block size is always 512 bytes */ 10463 RWErrorRecovery[15] = 0x00; 10464 } 10465 10466 if (LLBAA) 10467 { 10468 index = 24; 10469 } 10470 else 10471 { 10472 index = 16; 10473 } 10474 /* 10475 * Fill-up Read-Write Error Recovery mode page, SAT, Table 66 10476 */ 10477 RWErrorRecovery[index+0] = 0x01; /* page code */ 10478 RWErrorRecovery[index+1] = 0x0A; /* page length */ 10479 RWErrorRecovery[index+2] = 0x40; /* ARRE is set */ 10480 RWErrorRecovery[index+3] = 0x00; 10481 RWErrorRecovery[index+4] = 0x00; 10482 RWErrorRecovery[index+5] = 0x00; 10483 RWErrorRecovery[index+6] = 0x00; 10484 RWErrorRecovery[index+7] = 0x00; 10485 RWErrorRecovery[index+8] = 0x00; 10486 RWErrorRecovery[index+9] = 0x00; 10487 RWErrorRecovery[index+10] = 0x00; 10488 RWErrorRecovery[index+11] = 0x00; 10489 10490 sm_memcpy(pModeSense, &RWErrorRecovery, lenRead); 10491 } 10492 else if (page == MODESENSE_CACHING) 10493 { 10494 SM_DBG5(("smsatModeSense10: MODESENSE_CACHING\n")); 10495 Caching[0] = 0; 10496 Caching[1] = (bit8)(lenRead - 2); 10497 Caching[2] = 0x00; /* medium type: default medium type (currently mounted medium type) */ 10498 Caching[3] = 0x00; /* device-specific param: no write-protect, no support for DPO-FUA */ 10499 if (LLBAA) 10500 { 10501 Caching[4] = 0x00; /* reserved and LONGLBA */ 10502 Caching[4] = (bit8)(Caching[4] | 0x1); /* LONGLBA is set */ 10503 } 10504 else 10505 { 10506 Caching[4] = 0x00; /* reserved and LONGLBA: LONGLBA is not set */ 10507 } 10508 Caching[5] = 0x00; /* reserved */ 10509 Caching[6] = 0x00; /* block descriptot length */ 10510 if (LLBAA) 10511 { 10512 Caching[7] = 0x10; /* block descriptor length: LONGLBA is set. So, length is 16 */ 10513 } 10514 else 10515 { 10516 Caching[7] = 0x08; /* block descriptor length: LONGLBA is NOT set. So, length is 8 */ 10517 } 10518 10519 /* 10520 * Fill-up direct-access device block-descriptor, SAT, Table 19 10521 */ 10522 10523 if (LLBAA) 10524 { 10525 /* density code */ 10526 Caching[8] = 0x04; /* density-code : reserved for direct-access */ 10527 /* number of blocks */ 10528 Caching[9] = 0x00; /* unspecified */ 10529 Caching[10] = 0x00; /* unspecified */ 10530 Caching[11] = 0x00; /* unspecified */ 10531 Caching[12] = 0x00; /* unspecified */ 10532 Caching[13] = 0x00; /* unspecified */ 10533 Caching[14] = 0x00; /* unspecified */ 10534 Caching[15] = 0x00; /* unspecified */ 10535 /* reserved */ 10536 Caching[16] = 0x00; /* reserved */ 10537 Caching[17] = 0x00; /* reserved */ 10538 Caching[18] = 0x00; /* reserved */ 10539 Caching[19] = 0x00; /* reserved */ 10540 /* Block size */ 10541 Caching[20] = 0x00; 10542 Caching[21] = 0x00; 10543 Caching[22] = 0x02; /* Block size is always 512 bytes */ 10544 Caching[23] = 0x00; 10545 } 10546 else 10547 { 10548 /* density code */ 10549 Caching[8] = 0x04; /* density-code : reserved for direct-access */ 10550 /* number of blocks */ 10551 Caching[9] = 0x00; /* unspecified */ 10552 Caching[10] = 0x00; /* unspecified */ 10553 Caching[11] = 0x00; /* unspecified */ 10554 /* reserved */ 10555 Caching[12] = 0x00; /* reserved */ 10556 /* Block size */ 10557 Caching[13] = 0x00; 10558 Caching[14] = 0x02; /* Block size is always 512 bytes */ 10559 Caching[15] = 0x00; 10560 } 10561 10562 if (LLBAA) 10563 { 10564 index = 24; 10565 } 10566 else 10567 { 10568 index = 16; 10569 } 10570 /* 10571 * Fill-up Caching mode page, SAT, Table 67 10572 */ 10573 /* length 20 */ 10574 Caching[index+0] = 0x08; /* page code */ 10575 Caching[index+1] = 0x12; /* page length */ 10576 if (pSatDevData->satWriteCacheEnabled == agTRUE) 10577 { 10578 Caching[index+2] = 0x04;/* WCE bit is set */ 10579 } 10580 else 10581 { 10582 Caching[index+2] = 0x00;/* WCE bit is NOT set */ 10583 } 10584 10585 Caching[index+3] = 0x00; 10586 Caching[index+4] = 0x00; 10587 Caching[index+5] = 0x00; 10588 Caching[index+6] = 0x00; 10589 Caching[index+7] = 0x00; 10590 Caching[index+8] = 0x00; 10591 Caching[index+9] = 0x00; 10592 Caching[index+10] = 0x00; 10593 Caching[index+11] = 0x00; 10594 if (pSatDevData->satLookAheadEnabled == agTRUE) 10595 { 10596 Caching[index+12] = 0x00;/* DRA bit is NOT set */ 10597 } 10598 else 10599 { 10600 Caching[index+12] = 0x20;/* DRA bit is set */ 10601 } 10602 Caching[index+13] = 0x00; 10603 Caching[index+14] = 0x00; 10604 Caching[index+15] = 0x00; 10605 Caching[index+16] = 0x00; 10606 Caching[index+17] = 0x00; 10607 Caching[index+18] = 0x00; 10608 Caching[index+19] = 0x00; 10609 sm_memcpy(pModeSense, &Caching, lenRead); 10610 10611 } 10612 else if (page == MODESENSE_INFORMATION_EXCEPTION_CONTROL_PAGE) 10613 { 10614 SM_DBG5(("smsatModeSense10: MODESENSE_INFORMATION_EXCEPTION_CONTROL_PAGE\n")); 10615 InfoExceptionCtrl[0] = 0; 10616 InfoExceptionCtrl[1] = (bit8)(lenRead - 2); 10617 InfoExceptionCtrl[2] = 0x00; /* medium type: default medium type (currently mounted medium type) */ 10618 InfoExceptionCtrl[3] = 0x00; /* device-specific param: no write-protect, no support for DPO-FUA */ 10619 if (LLBAA) 10620 { 10621 InfoExceptionCtrl[4] = 0x00; /* reserved and LONGLBA */ 10622 InfoExceptionCtrl[4] = (bit8)(InfoExceptionCtrl[4] | 0x1); /* LONGLBA is set */ 10623 } 10624 else 10625 { 10626 InfoExceptionCtrl[4] = 0x00; /* reserved and LONGLBA: LONGLBA is not set */ 10627 } 10628 InfoExceptionCtrl[5] = 0x00; /* reserved */ 10629 InfoExceptionCtrl[6] = 0x00; /* block descriptot length */ 10630 if (LLBAA) 10631 { 10632 InfoExceptionCtrl[7] = 0x10; /* block descriptor length: LONGLBA is set. So, length is 16 */ 10633 } 10634 else 10635 { 10636 InfoExceptionCtrl[7] = 0x08; /* block descriptor length: LONGLBA is NOT set. So, length is 8 */ 10637 } 10638 10639 /* 10640 * Fill-up direct-access device block-descriptor, SAT, Table 19 10641 */ 10642 10643 if (LLBAA) 10644 { 10645 /* density code */ 10646 InfoExceptionCtrl[8] = 0x04; /* density-code : reserved for direct-access */ 10647 /* number of blocks */ 10648 InfoExceptionCtrl[9] = 0x00; /* unspecified */ 10649 InfoExceptionCtrl[10] = 0x00; /* unspecified */ 10650 InfoExceptionCtrl[11] = 0x00; /* unspecified */ 10651 InfoExceptionCtrl[12] = 0x00; /* unspecified */ 10652 InfoExceptionCtrl[13] = 0x00; /* unspecified */ 10653 InfoExceptionCtrl[14] = 0x00; /* unspecified */ 10654 InfoExceptionCtrl[15] = 0x00; /* unspecified */ 10655 /* reserved */ 10656 InfoExceptionCtrl[16] = 0x00; /* reserved */ 10657 InfoExceptionCtrl[17] = 0x00; /* reserved */ 10658 InfoExceptionCtrl[18] = 0x00; /* reserved */ 10659 InfoExceptionCtrl[19] = 0x00; /* reserved */ 10660 /* Block size */ 10661 InfoExceptionCtrl[20] = 0x00; 10662 InfoExceptionCtrl[21] = 0x00; 10663 InfoExceptionCtrl[22] = 0x02; /* Block size is always 512 bytes */ 10664 InfoExceptionCtrl[23] = 0x00; 10665 } 10666 else 10667 { 10668 /* density code */ 10669 InfoExceptionCtrl[8] = 0x04; /* density-code : reserved for direct-access */ 10670 /* number of blocks */ 10671 InfoExceptionCtrl[9] = 0x00; /* unspecified */ 10672 InfoExceptionCtrl[10] = 0x00; /* unspecified */ 10673 InfoExceptionCtrl[11] = 0x00; /* unspecified */ 10674 /* reserved */ 10675 InfoExceptionCtrl[12] = 0x00; /* reserved */ 10676 /* Block size */ 10677 InfoExceptionCtrl[13] = 0x00; 10678 InfoExceptionCtrl[14] = 0x02; /* Block size is always 512 bytes */ 10679 InfoExceptionCtrl[15] = 0x00; 10680 } 10681 10682 if (LLBAA) 10683 { 10684 index = 24; 10685 } 10686 else 10687 { 10688 index = 16; 10689 } 10690 /* 10691 * Fill-up informational-exceptions control mode page, SAT, Table 68 10692 */ 10693 InfoExceptionCtrl[index+0] = 0x1C; /* page code */ 10694 InfoExceptionCtrl[index+1] = 0x0A; /* page length */ 10695 if (pSatDevData->satSMARTEnabled == agTRUE) 10696 { 10697 InfoExceptionCtrl[index+2] = 0x00;/* DEXCPT bit is NOT set */ 10698 } 10699 else 10700 { 10701 InfoExceptionCtrl[index+2] = 0x08;/* DEXCPT bit is set */ 10702 } 10703 InfoExceptionCtrl[index+3] = 0x00; /* We don't support MRIE */ 10704 InfoExceptionCtrl[index+4] = 0x00; /* Interval timer vendor-specific */ 10705 InfoExceptionCtrl[index+5] = 0x00; 10706 InfoExceptionCtrl[index+6] = 0x00; 10707 InfoExceptionCtrl[index+7] = 0x00; 10708 InfoExceptionCtrl[index+8] = 0x00; /* REPORT-COUNT */ 10709 InfoExceptionCtrl[index+9] = 0x00; 10710 InfoExceptionCtrl[index+10] = 0x00; 10711 InfoExceptionCtrl[index+11] = 0x00; 10712 sm_memcpy(pModeSense, &InfoExceptionCtrl, lenRead); 10713 10714 } 10715 else 10716 { 10717 /* Error */ 10718 SM_DBG1(("smsatModeSense10: Error page %d!!!\n", page)); 10719 smsatSetSensePayload( pSense, 10720 SCSI_SNSKEY_ILLEGAL_REQUEST, 10721 0, 10722 SCSI_SNSCODE_INVALID_COMMAND, 10723 satIOContext); 10724 10725 /*smEnqueueIO(smRoot, satIOContext);*/ 10726 10727 tdsmIOCompletedCB( smRoot, 10728 smIORequest, 10729 smIOSuccess, 10730 SCSI_STAT_CHECK_CONDITION, 10731 satIOContext->pSmSenseData, 10732 satIOContext->interruptContext ); 10733 return SM_RC_SUCCESS; 10734 } 10735 10736 if (allocationLen > lenRead) 10737 { 10738 SM_DBG1(("smsatModeSense10: reporting underrun lenRead=0x%x allocationLen=0x%x smIORequest=%p\n", lenRead, allocationLen, smIORequest)); 10739 10740 /*smEnqueueIO(smRoot, satIOContext);*/ 10741 10742 tdsmIOCompletedCB( smRoot, 10743 smIORequest, 10744 smIOUnderRun, 10745 allocationLen - lenRead, 10746 agNULL, 10747 satIOContext->interruptContext ); 10748 10749 10750 } 10751 else 10752 { 10753 /*smEnqueueIO(smRoot, satIOContext);*/ 10754 10755 tdsmIOCompletedCB( smRoot, 10756 smIORequest, 10757 smIOSuccess, 10758 SCSI_STAT_GOOD, 10759 agNULL, 10760 satIOContext->interruptContext); 10761 } 10762 10763 return SM_RC_SUCCESS; 10764 } 10765 10766 osGLOBAL bit32 10767 smsatReadCapacity10( 10768 smRoot_t *smRoot, 10769 smIORequest_t *smIORequest, 10770 smDeviceHandle_t *smDeviceHandle, 10771 smScsiInitiatorRequest_t *smScsiRequest, 10772 smSatIOContext_t *satIOContext 10773 ) 10774 { 10775 smScsiRspSense_t *pSense; 10776 smIniScsiCmnd_t *scsiCmnd; 10777 bit8 dataBuffer[8] = {0}; 10778 bit32 allocationLen; 10779 bit8 *pVirtAddr = agNULL; 10780 smDeviceData_t *pSatDevData; 10781 agsaSATAIdentifyData_t *pSATAIdData; 10782 bit32 lastLba; 10783 bit32 word117_118; 10784 bit32 word117; 10785 bit32 word118; 10786 10787 pSense = satIOContext->pSense; 10788 pVirtAddr = (bit8 *) smScsiRequest->sglVirtualAddr; 10789 scsiCmnd = &smScsiRequest->scsiCmnd; 10790 pSatDevData = satIOContext->pSatDevData; 10791 pSATAIdData = &pSatDevData->satIdentifyData; 10792 allocationLen = scsiCmnd->expDataLength; 10793 10794 SM_DBG5(("smsatReadCapacity10: start\n")); 10795 10796 /* checking CONTROL */ 10797 /* NACA == 1 or LINK == 1*/ 10798 if ( (scsiCmnd->cdb[9] & SCSI_NACA_MASK) || (scsiCmnd->cdb[9] & SCSI_LINK_MASK) ) 10799 { 10800 smsatSetSensePayload( pSense, 10801 SCSI_SNSKEY_ILLEGAL_REQUEST, 10802 0, 10803 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 10804 satIOContext); 10805 10806 /*smEnqueueIO(smRoot, satIOContext);*/ 10807 10808 tdsmIOCompletedCB( smRoot, 10809 smIORequest, 10810 smIOSuccess, 10811 SCSI_STAT_CHECK_CONDITION, 10812 satIOContext->pSmSenseData, 10813 satIOContext->interruptContext ); 10814 10815 SM_DBG1(("smsatReadCapacity10: return control!!!\n")); 10816 return SM_RC_SUCCESS; 10817 } 10818 10819 10820 /* 10821 * If Logical block address is not set to zero, return error 10822 */ 10823 if ((scsiCmnd->cdb[2] || scsiCmnd->cdb[3] || scsiCmnd->cdb[4] || scsiCmnd->cdb[5])) 10824 { 10825 SM_DBG1(("smsatReadCapacity10: *** ERROR *** logical address non zero, did %d!!!\n", 10826 pSatDevData->id)); 10827 10828 smsatSetSensePayload( pSense, 10829 SCSI_SNSKEY_ILLEGAL_REQUEST, 10830 0, 10831 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 10832 satIOContext); 10833 10834 /*smEnqueueIO(smRoot, satIOContext);*/ 10835 10836 tdsmIOCompletedCB( smRoot, 10837 smIORequest, 10838 smIOSuccess, 10839 SCSI_STAT_CHECK_CONDITION, 10840 satIOContext->pSmSenseData, 10841 satIOContext->interruptContext ); 10842 return SM_RC_SUCCESS; 10843 10844 } 10845 10846 /* 10847 * If PMI bit is not zero, return error 10848 */ 10849 if ( ((scsiCmnd->cdb[8]) & SCSI_READ_CAPACITY10_PMI_MASK) != 0 ) 10850 { 10851 SM_DBG1(("smsatReadCapacity10: *** ERROR *** PMI is not zero, did %d\n", 10852 pSatDevData->id)); 10853 10854 smsatSetSensePayload( pSense, 10855 SCSI_SNSKEY_ILLEGAL_REQUEST, 10856 0, 10857 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 10858 satIOContext); 10859 10860 /*smEnqueueIO(smRoot, satIOContext);*/ 10861 10862 tdsmIOCompletedCB( smRoot, 10863 smIORequest, 10864 smIOSuccess, 10865 SCSI_STAT_CHECK_CONDITION, 10866 satIOContext->pSmSenseData, 10867 satIOContext->interruptContext ); 10868 return SM_RC_SUCCESS; 10869 10870 } 10871 10872 /* 10873 filling in Read Capacity parameter data 10874 saved identify device has been already flipped 10875 See ATA spec p125 and p136 and SBC spec p54 10876 */ 10877 /* 10878 * If 48-bit addressing is supported, set capacity information from Identify 10879 * Device Word 100-103. 10880 */ 10881 if (pSatDevData->sat48BitSupport == agTRUE) 10882 { 10883 /* 10884 * Setting RETURNED LOGICAL BLOCK ADDRESS in READ CAPACITY(10) response data: 10885 * SBC-2 specifies that if the capacity exceeded the 4-byte RETURNED LOGICAL 10886 * BLOCK ADDRESS in READ CAPACITY(10) parameter data, the the RETURNED LOGICAL 10887 * BLOCK ADDRESS should be set to 0xFFFFFFFF so the application client would 10888 * then issue a READ CAPACITY(16) command. 10889 */ 10890 /* ATA Identify Device information word 100 - 103 */ 10891 if ( (pSATAIdData->maxLBA32_47 != 0 ) || (pSATAIdData->maxLBA48_63 != 0)) 10892 { 10893 dataBuffer[0] = 0xFF; /* MSB number of block */ 10894 dataBuffer[1] = 0xFF; 10895 dataBuffer[2] = 0xFF; 10896 dataBuffer[3] = 0xFF; /* LSB number of block */ 10897 SM_DBG1(("smsatReadCapacity10: returns 0xFFFFFFFF!!!\n")); 10898 } 10899 else /* Fit the Readcapacity10 4-bytes response length */ 10900 { 10901 lastLba = (((pSATAIdData->maxLBA16_31) << 16) ) | 10902 (pSATAIdData->maxLBA0_15); 10903 lastLba = lastLba - 1; /* LBA starts from zero */ 10904 10905 /* 10906 for testing 10907 lastLba = lastLba - (512*10) - 1; 10908 */ 10909 10910 10911 dataBuffer[0] = (bit8)((lastLba >> 24) & 0xFF); /* MSB */ 10912 dataBuffer[1] = (bit8)((lastLba >> 16) & 0xFF); 10913 dataBuffer[2] = (bit8)((lastLba >> 8) & 0xFF); 10914 dataBuffer[3] = (bit8)((lastLba ) & 0xFF); /* LSB */ 10915 10916 SM_DBG3(("smsatReadCapacity10: lastLba is 0x%x %d\n", lastLba, lastLba)); 10917 SM_DBG3(("smsatReadCapacity10: LBA 0 is 0x%x %d\n", dataBuffer[0], dataBuffer[0])); 10918 SM_DBG3(("smsatReadCapacity10: LBA 1 is 0x%x %d\n", dataBuffer[1], dataBuffer[1])); 10919 SM_DBG3(("smsatReadCapacity10: LBA 2 is 0x%x %d\n", dataBuffer[2], dataBuffer[2])); 10920 SM_DBG3(("smsatReadCapacity10: LBA 3 is 0x%x %d\n", dataBuffer[3], dataBuffer[3])); 10921 10922 } 10923 } 10924 10925 /* 10926 * For 28-bit addressing, set capacity information from Identify 10927 * Device Word 60-61. 10928 */ 10929 else 10930 { 10931 /* ATA Identify Device information word 60 - 61 */ 10932 lastLba = (((pSATAIdData->numOfUserAddressableSectorsHi) << 16) ) | 10933 (pSATAIdData->numOfUserAddressableSectorsLo); 10934 lastLba = lastLba - 1; /* LBA starts from zero */ 10935 10936 dataBuffer[0] = (bit8)((lastLba >> 24) & 0xFF); /* MSB */ 10937 dataBuffer[1] = (bit8)((lastLba >> 16) & 0xFF); 10938 dataBuffer[2] = (bit8)((lastLba >> 8) & 0xFF); 10939 dataBuffer[3] = (bit8)((lastLba ) & 0xFF); /* LSB */ 10940 } 10941 /* SAT Rev 8d */ 10942 if (((pSATAIdData->word104_107[2]) & 0x1000) == 0) 10943 { 10944 SM_DBG5(("smsatReadCapacity10: Default Block Length is 512\n")); 10945 /* 10946 * Set the block size, fixed at 512 bytes. 10947 */ 10948 dataBuffer[4] = 0x00; /* MSB block size in bytes */ 10949 dataBuffer[5] = 0x00; 10950 dataBuffer[6] = 0x02; 10951 dataBuffer[7] = 0x00; /* LSB block size in bytes */ 10952 } 10953 else 10954 { 10955 word118 = pSATAIdData->word112_126[6]; 10956 word117 = pSATAIdData->word112_126[5]; 10957 10958 word117_118 = (word118 << 16) + word117; 10959 word117_118 = word117_118 * 2; 10960 dataBuffer[4] = (bit8)((word117_118 >> 24) & 0xFF); /* MSB block size in bytes */ 10961 dataBuffer[5] = (bit8)((word117_118 >> 16) & 0xFF); 10962 dataBuffer[6] = (bit8)((word117_118 >> 8) & 0xFF); 10963 dataBuffer[7] = (bit8)(word117_118 & 0xFF); /* LSB block size in bytes */ 10964 10965 SM_DBG1(("smsatReadCapacity10: Nondefault word118 %d 0x%x !!!\n", word118, word118)); 10966 SM_DBG1(("smsatReadCapacity10: Nondefault word117 %d 0x%x !!!\n", word117, word117)); 10967 SM_DBG1(("smsatReadCapacity10: Nondefault Block Length is %d 0x%x !!!\n",word117_118, word117_118)); 10968 10969 } 10970 10971 /* fill in MAX LBA, which is used in satSendDiagnostic_1() */ 10972 pSatDevData->satMaxLBA[0] = 0; /* MSB */ 10973 pSatDevData->satMaxLBA[1] = 0; 10974 pSatDevData->satMaxLBA[2] = 0; 10975 pSatDevData->satMaxLBA[3] = 0; 10976 pSatDevData->satMaxLBA[4] = dataBuffer[0]; 10977 pSatDevData->satMaxLBA[5] = dataBuffer[1]; 10978 pSatDevData->satMaxLBA[6] = dataBuffer[2]; 10979 pSatDevData->satMaxLBA[7] = dataBuffer[3]; /* LSB */ 10980 10981 10982 SM_DBG4(("smsatReadCapacity10: 0x%x 0x%x 0x%x 0x%x 0x%x 0x%x 0x%x 0x%x , did %d\n", 10983 dataBuffer[0], dataBuffer[1], dataBuffer[2], dataBuffer[3], 10984 dataBuffer[4], dataBuffer[5], dataBuffer[6], dataBuffer[7], 10985 pSatDevData->id)); 10986 10987 sm_memcpy(pVirtAddr, dataBuffer, MIN(allocationLen, 8)); 10988 10989 /* 10990 * Send the completion response now. 10991 */ 10992 /*smEnqueueIO(smRoot, satIOContext);*/ 10993 10994 tdsmIOCompletedCB( smRoot, 10995 smIORequest, 10996 smIOSuccess, 10997 SCSI_STAT_GOOD, 10998 agNULL, 10999 satIOContext->interruptContext); 11000 return SM_RC_SUCCESS; 11001 } 11002 11003 osGLOBAL bit32 11004 smsatReadCapacity16( 11005 smRoot_t *smRoot, 11006 smIORequest_t *smIORequest, 11007 smDeviceHandle_t *smDeviceHandle, 11008 smScsiInitiatorRequest_t *smScsiRequest, 11009 smSatIOContext_t *satIOContext 11010 ) 11011 { 11012 smScsiRspSense_t *pSense; 11013 smIniScsiCmnd_t *scsiCmnd; 11014 bit8 dataBuffer[32] = {0}; 11015 bit8 *pVirtAddr = agNULL; 11016 smDeviceData_t *pSatDevData; 11017 agsaSATAIdentifyData_t *pSATAIdData; 11018 bit32 lastLbaLo; 11019 bit32 allocationLen; 11020 bit32 readCapacityLen = 32; 11021 bit32 i = 0; 11022 11023 pSense = satIOContext->pSense; 11024 pVirtAddr = (bit8 *) smScsiRequest->sglVirtualAddr; 11025 scsiCmnd = &smScsiRequest->scsiCmnd; 11026 pSatDevData = satIOContext->pSatDevData; 11027 pSATAIdData = &pSatDevData->satIdentifyData; 11028 11029 SM_DBG5(("smsatReadCapacity16: start\n")); 11030 11031 /* Find the buffer size allocated by Initiator */ 11032 allocationLen = (((bit32)scsiCmnd->cdb[10]) << 24) | 11033 (((bit32)scsiCmnd->cdb[11]) << 16) | 11034 (((bit32)scsiCmnd->cdb[12]) << 8 ) | 11035 (((bit32)scsiCmnd->cdb[13]) ); 11036 allocationLen = MIN(allocationLen, scsiCmnd->expDataLength); 11037 11038 #ifdef REMOVED 11039 if (allocationLen < readCapacityLen) 11040 { 11041 SM_DBG1(("smsatReadCapacity16: *** ERROR *** insufficient len=0x%x readCapacityLen=0x%x!!!\n", allocationLen, readCapacityLen)); 11042 11043 smsatSetSensePayload( pSense, 11044 SCSI_SNSKEY_ILLEGAL_REQUEST, 11045 0, 11046 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 11047 satIOContext); 11048 11049 /*smEnqueueIO(smRoot, satIOContext);*/ 11050 11051 tdsmIOCompletedCB( smRoot, 11052 smIORequest, 11053 smIOSuccess, 11054 SCSI_STAT_CHECK_CONDITION, 11055 satIOContext->pSmSenseData, 11056 satIOContext->interruptContext ); 11057 return SM_RC_SUCCESS; 11058 11059 } 11060 #endif 11061 11062 /* checking CONTROL */ 11063 /* NACA == 1 or LINK == 1*/ 11064 if ( (scsiCmnd->cdb[15] & SCSI_NACA_MASK) || (scsiCmnd->cdb[15] & SCSI_LINK_MASK) ) 11065 { 11066 smsatSetSensePayload( pSense, 11067 SCSI_SNSKEY_ILLEGAL_REQUEST, 11068 0, 11069 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 11070 satIOContext); 11071 11072 /*smEnqueueIO(smRoot, satIOContext);*/ 11073 11074 tdsmIOCompletedCB( smRoot, 11075 smIORequest, 11076 smIOSuccess, 11077 SCSI_STAT_CHECK_CONDITION, 11078 satIOContext->pSmSenseData, 11079 satIOContext->interruptContext ); 11080 11081 SM_DBG1(("smsatReadCapacity16: return control!!!\n")); 11082 return SM_RC_SUCCESS; 11083 } 11084 11085 /* 11086 * If Logical blcok address is not set to zero, return error 11087 */ 11088 if ((scsiCmnd->cdb[2] || scsiCmnd->cdb[3] || scsiCmnd->cdb[4] || scsiCmnd->cdb[5]) || 11089 (scsiCmnd->cdb[6] || scsiCmnd->cdb[7] || scsiCmnd->cdb[8] || scsiCmnd->cdb[9]) ) 11090 { 11091 SM_DBG1(("smsatReadCapacity16: *** ERROR *** logical address non zero, did %d\n", 11092 pSatDevData->id)); 11093 11094 smsatSetSensePayload( pSense, 11095 SCSI_SNSKEY_ILLEGAL_REQUEST, 11096 0, 11097 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 11098 satIOContext); 11099 11100 /*smEnqueueIO(smRoot, satIOContext);*/ 11101 11102 tdsmIOCompletedCB( smRoot, 11103 smIORequest, 11104 smIOSuccess, 11105 SCSI_STAT_CHECK_CONDITION, 11106 satIOContext->pSmSenseData, 11107 satIOContext->interruptContext ); 11108 return SM_RC_SUCCESS; 11109 11110 } 11111 11112 /* 11113 * If PMI bit is not zero, return error 11114 */ 11115 if ( ((scsiCmnd->cdb[14]) & SCSI_READ_CAPACITY16_PMI_MASK) != 0 ) 11116 { 11117 SM_DBG1(("smsatReadCapacity16: *** ERROR *** PMI is not zero, did %d\n", 11118 pSatDevData->id)); 11119 11120 smsatSetSensePayload( pSense, 11121 SCSI_SNSKEY_ILLEGAL_REQUEST, 11122 0, 11123 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 11124 satIOContext); 11125 11126 /*smEnqueueIO(smRoot, satIOContext);*/ 11127 11128 tdsmIOCompletedCB( smRoot, 11129 smIORequest, 11130 smIOSuccess, 11131 SCSI_STAT_CHECK_CONDITION, 11132 satIOContext->pSmSenseData, 11133 satIOContext->interruptContext ); 11134 return SM_RC_SUCCESS; 11135 11136 } 11137 11138 /* 11139 filling in Read Capacity parameter data 11140 */ 11141 11142 /* 11143 * If 48-bit addressing is supported, set capacity information from Identify 11144 * Device Word 100-103. 11145 */ 11146 if (pSatDevData->sat48BitSupport == agTRUE) 11147 { 11148 dataBuffer[0] = (bit8)(((pSATAIdData->maxLBA48_63) >> 8) & 0xff); /* MSB */ 11149 dataBuffer[1] = (bit8)((pSATAIdData->maxLBA48_63) & 0xff); 11150 dataBuffer[2] = (bit8)(((pSATAIdData->maxLBA32_47) >> 8) & 0xff); 11151 dataBuffer[3] = (bit8)((pSATAIdData->maxLBA32_47) & 0xff); 11152 11153 lastLbaLo = (((pSATAIdData->maxLBA16_31) << 16) ) | (pSATAIdData->maxLBA0_15); 11154 lastLbaLo = lastLbaLo - 1; /* LBA starts from zero */ 11155 11156 dataBuffer[4] = (bit8)((lastLbaLo >> 24) & 0xFF); 11157 dataBuffer[5] = (bit8)((lastLbaLo >> 16) & 0xFF); 11158 dataBuffer[6] = (bit8)((lastLbaLo >> 8) & 0xFF); 11159 dataBuffer[7] = (bit8)((lastLbaLo ) & 0xFF); /* LSB */ 11160 11161 } 11162 11163 /* 11164 * For 28-bit addressing, set capacity information from Identify 11165 * Device Word 60-61. 11166 */ 11167 else 11168 { 11169 dataBuffer[0] = 0; /* MSB */ 11170 dataBuffer[1] = 0; 11171 dataBuffer[2] = 0; 11172 dataBuffer[3] = 0; 11173 11174 lastLbaLo = (((pSATAIdData->numOfUserAddressableSectorsHi) << 16) ) | 11175 (pSATAIdData->numOfUserAddressableSectorsLo); 11176 lastLbaLo = lastLbaLo - 1; /* LBA starts from zero */ 11177 11178 dataBuffer[4] = (bit8)((lastLbaLo >> 24) & 0xFF); 11179 dataBuffer[5] = (bit8)((lastLbaLo >> 16) & 0xFF); 11180 dataBuffer[6] = (bit8)((lastLbaLo >> 8) & 0xFF); 11181 dataBuffer[7] = (bit8)((lastLbaLo ) & 0xFF); /* LSB */ 11182 11183 } 11184 11185 /* 11186 * Set the block size, fixed at 512 bytes. 11187 */ 11188 dataBuffer[8] = 0x00; /* MSB block size in bytes */ 11189 dataBuffer[9] = 0x00; 11190 dataBuffer[10] = 0x02; 11191 dataBuffer[11] = 0x00; /* LSB block size in bytes */ 11192 11193 11194 /* fill in MAX LBA, which is used in satSendDiagnostic_1() */ 11195 pSatDevData->satMaxLBA[0] = dataBuffer[0]; /* MSB */ 11196 pSatDevData->satMaxLBA[1] = dataBuffer[1]; 11197 pSatDevData->satMaxLBA[2] = dataBuffer[2]; 11198 pSatDevData->satMaxLBA[3] = dataBuffer[3]; 11199 pSatDevData->satMaxLBA[4] = dataBuffer[4]; 11200 pSatDevData->satMaxLBA[5] = dataBuffer[5]; 11201 pSatDevData->satMaxLBA[6] = dataBuffer[6]; 11202 pSatDevData->satMaxLBA[7] = dataBuffer[7]; /* LSB */ 11203 11204 SM_DBG5(("smsatReadCapacity16: 0x%x 0x%x 0x%x 0x%x 0x%x 0x%x 0x%x 0x%x 0x%x 0x%x 0x%x 0x%x , did %d\n", 11205 dataBuffer[0], dataBuffer[1], dataBuffer[2], dataBuffer[3], 11206 dataBuffer[4], dataBuffer[5], dataBuffer[6], dataBuffer[7], 11207 dataBuffer[8], dataBuffer[9], dataBuffer[10], dataBuffer[11], 11208 pSatDevData->id)); 11209 11210 if (allocationLen > 0xC) /* 0xc = 12 */ 11211 { 11212 for(i=12;i<=31;i++) 11213 { 11214 dataBuffer[i] = 0x00; 11215 } 11216 } 11217 11218 sm_memcpy(pVirtAddr, dataBuffer, MIN(allocationLen, readCapacityLen)); 11219 /* 11220 * Send the completion response now. 11221 */ 11222 if (allocationLen > readCapacityLen) 11223 { 11224 /* underrun */ 11225 SM_DBG1(("smsatReadCapacity16: reporting underrun readCapacityLen=0x%x allocationLen=0x%x !!!\n", readCapacityLen, allocationLen)); 11226 11227 /*smEnqueueIO(smRoot, satIOContext);*/ 11228 11229 tdsmIOCompletedCB( smRoot, 11230 smIORequest, 11231 smIOUnderRun, 11232 allocationLen - readCapacityLen, 11233 agNULL, 11234 satIOContext->interruptContext ); 11235 11236 11237 } 11238 else 11239 { 11240 /*smEnqueueIO(smRoot, satIOContext);*/ 11241 11242 tdsmIOCompletedCB( smRoot, 11243 smIORequest, 11244 smIOSuccess, 11245 SCSI_STAT_GOOD, 11246 agNULL, 11247 satIOContext->interruptContext); 11248 } 11249 return SM_RC_SUCCESS; 11250 } 11251 11252 osGLOBAL bit32 11253 smsatReportLun( 11254 smRoot_t *smRoot, 11255 smIORequest_t *smIORequest, 11256 smDeviceHandle_t *smDeviceHandle, 11257 smScsiInitiatorRequest_t *smScsiRequest, 11258 smSatIOContext_t *satIOContext 11259 ) 11260 { 11261 smScsiRspSense_t *pSense; 11262 bit8 dataBuffer[16] = {0}; 11263 bit32 allocationLen; 11264 bit32 reportLunLen; 11265 smScsiReportLun_t *pReportLun; 11266 smIniScsiCmnd_t *scsiCmnd; 11267 #ifdef TD_DEBUG_ENABLE 11268 smDeviceData_t *pSatDevData; 11269 #endif 11270 11271 pSense = satIOContext->pSense; 11272 pReportLun = (smScsiReportLun_t *) dataBuffer; 11273 scsiCmnd = &smScsiRequest->scsiCmnd; 11274 #ifdef TD_DEBUG_ENABLE 11275 pSatDevData = satIOContext->pSatDevData; 11276 #endif 11277 SM_DBG5(("smsatReportLun: start\n")); 11278 // smhexdump("smsatReportLun: cdb", (bit8 *)scsiCmnd, 16); 11279 /* Find the buffer size allocated by Initiator */ 11280 allocationLen = (((bit32)scsiCmnd->cdb[6]) << 24) | 11281 (((bit32)scsiCmnd->cdb[7]) << 16) | 11282 (((bit32)scsiCmnd->cdb[8]) << 8 ) | 11283 (((bit32)scsiCmnd->cdb[9]) ); 11284 allocationLen = MIN(allocationLen, scsiCmnd->expDataLength); 11285 reportLunLen = 16; /* 8 byte header and 8 bytes of LUN0 */ 11286 if (allocationLen < reportLunLen) 11287 { 11288 SM_DBG1(("smsatReportLun: *** ERROR *** insufficient len=0x%x did %d\n", 11289 reportLunLen, pSatDevData->id)); 11290 smsatSetSensePayload( pSense, 11291 SCSI_SNSKEY_ILLEGAL_REQUEST, 11292 0, 11293 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 11294 satIOContext); 11295 /*smEnqueueIO(smRoot, satIOContext);*/ 11296 tdsmIOCompletedCB( smRoot, 11297 smIORequest, 11298 smIOSuccess, 11299 SCSI_STAT_CHECK_CONDITION, 11300 satIOContext->pSmSenseData, 11301 satIOContext->interruptContext ); 11302 return SM_RC_SUCCESS; 11303 } 11304 /* Set length to one entry */ 11305 pReportLun->len[0] = 0; 11306 pReportLun->len[1] = 0; 11307 pReportLun->len[2] = 0; 11308 pReportLun->len[3] = sizeof (tiLUN_t); 11309 pReportLun->reserved = 0; 11310 /* Set to LUN 0: 11311 * - address method to 0x00: Peripheral device addressing method, 11312 * - bus identifier to 0 11313 */ 11314 pReportLun->lunList[0].lun[0] = 0; 11315 pReportLun->lunList[0].lun[1] = 0; 11316 pReportLun->lunList[0].lun[2] = 0; 11317 pReportLun->lunList[0].lun[3] = 0; 11318 pReportLun->lunList[0].lun[4] = 0; 11319 pReportLun->lunList[0].lun[5] = 0; 11320 pReportLun->lunList[0].lun[6] = 0; 11321 pReportLun->lunList[0].lun[7] = 0; 11322 11323 sm_memcpy(smScsiRequest->sglVirtualAddr, dataBuffer, MIN(allocationLen, reportLunLen)); 11324 if (allocationLen > reportLunLen) 11325 { 11326 /* underrun */ 11327 SM_DBG1(("smsatReportLun: reporting underrun reportLunLen=0x%x allocationLen=0x%x !!!\n", reportLunLen, allocationLen)); 11328 11329 /*smEnqueueIO(smRoot, satIOContext);*/ 11330 11331 tdsmIOCompletedCB( smRoot, 11332 smIORequest, 11333 smIOUnderRun, 11334 allocationLen - reportLunLen, 11335 agNULL, 11336 satIOContext->interruptContext ); 11337 11338 11339 } 11340 else 11341 { 11342 /*smEnqueueIO(smRoot, satIOContext);*/ 11343 11344 tdsmIOCompletedCB( smRoot, 11345 smIORequest, 11346 smIOSuccess, 11347 SCSI_STAT_GOOD, 11348 agNULL, 11349 satIOContext->interruptContext); 11350 } 11351 return SM_RC_SUCCESS; 11352 } 11353 11354 osGLOBAL bit32 11355 smsatFormatUnit( 11356 smRoot_t *smRoot, 11357 smIORequest_t *smIORequest, 11358 smDeviceHandle_t *smDeviceHandle, 11359 smScsiInitiatorRequest_t *smScsiRequest, 11360 smSatIOContext_t *satIOContext 11361 ) 11362 { 11363 /* 11364 note: we don't support media certification in this version and IP bit 11365 satDevData->satFormatState will be agFalse since SAT does not actually sends 11366 any ATA command 11367 */ 11368 11369 smScsiRspSense_t *pSense; 11370 smIniScsiCmnd_t *scsiCmnd; 11371 bit32 index = 0; 11372 11373 pSense = satIOContext->pSense; 11374 scsiCmnd = &smScsiRequest->scsiCmnd; 11375 SM_DBG5(("smsatFormatUnit: start\n")); 11376 /* 11377 checking opcode 11378 1. FMTDATA bit == 0(no defect list header) 11379 2. FMTDATA bit == 1 and DCRT bit == 1(defect list header is provided 11380 with DCRT bit set) 11381 */ 11382 if ( ((scsiCmnd->cdb[1] & SCSI_FORMAT_UNIT_FMTDATA_MASK) == 0) || 11383 ((scsiCmnd->cdb[1] & SCSI_FORMAT_UNIT_FMTDATA_MASK) && 11384 (scsiCmnd->cdb[7] & SCSI_FORMAT_UNIT_DCRT_MASK)) 11385 ) 11386 { 11387 /*smEnqueueIO(smRoot, satIOContext);*/ 11388 11389 tdsmIOCompletedCB( smRoot, 11390 smIORequest, 11391 smIOSuccess, 11392 SCSI_STAT_GOOD, 11393 agNULL, 11394 satIOContext->interruptContext); 11395 11396 SM_DBG1(("smsatFormatUnit: return opcode!!!\n")); 11397 return SM_RC_SUCCESS; 11398 } 11399 11400 /* 11401 checking DEFECT LIST FORMAT and defect list length 11402 */ 11403 if ( (((scsiCmnd->cdb[1] & SCSI_FORMAT_UNIT_DEFECT_LIST_FORMAT_MASK) == 0x00) || 11404 ((scsiCmnd->cdb[1] & SCSI_FORMAT_UNIT_DEFECT_LIST_FORMAT_MASK) == 0x06)) ) 11405 { 11406 /* short parameter header */ 11407 if ((scsiCmnd->cdb[2] & SCSI_FORMAT_UNIT_LONGLIST_MASK) == 0x00) 11408 { 11409 index = 8; 11410 } 11411 /* long parameter header */ 11412 if ((scsiCmnd->cdb[2] & SCSI_FORMAT_UNIT_LONGLIST_MASK) == 0x01) 11413 { 11414 index = 10; 11415 } 11416 /* defect list length */ 11417 if ((scsiCmnd->cdb[index] != 0) || (scsiCmnd->cdb[index+1] != 0)) 11418 { 11419 smsatSetSensePayload( pSense, 11420 SCSI_SNSKEY_ILLEGAL_REQUEST, 11421 0, 11422 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 11423 satIOContext); 11424 11425 /*smEnqueueIO(smRoot, satIOContext);*/ 11426 11427 tdsmIOCompletedCB( smRoot, 11428 smIORequest, 11429 smIOSuccess, 11430 SCSI_STAT_CHECK_CONDITION, 11431 satIOContext->pSmSenseData, 11432 satIOContext->interruptContext ); 11433 11434 SM_DBG1(("smsatFormatUnit: return defect list format!!!\n")); 11435 return SM_RC_SUCCESS; 11436 } 11437 } 11438 11439 if ( (scsiCmnd->cdb[1] & SCSI_FORMAT_UNIT_FMTDATA_MASK) && 11440 (scsiCmnd->cdb[1] & SCSI_FORMAT_UNIT_CMPLIST_MASK) ) 11441 { 11442 smsatSetSensePayload( pSense, 11443 SCSI_SNSKEY_ILLEGAL_REQUEST, 11444 0, 11445 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 11446 satIOContext); 11447 11448 /*smEnqueueIO(smRoot, satIOContext);*/ 11449 11450 tdsmIOCompletedCB( smRoot, 11451 smIORequest, 11452 smIOSuccess, 11453 SCSI_STAT_CHECK_CONDITION, 11454 satIOContext->pSmSenseData, 11455 satIOContext->interruptContext ); 11456 11457 SM_DBG1(("smsatFormatUnit: return cmplist!!!\n")); 11458 return SM_RC_SUCCESS; 11459 11460 } 11461 11462 if ( (scsiCmnd->cdb[5] & SCSI_NACA_MASK) || (scsiCmnd->cdb[5] & SCSI_LINK_MASK) ) 11463 { 11464 smsatSetSensePayload( pSense, 11465 SCSI_SNSKEY_ILLEGAL_REQUEST, 11466 0, 11467 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 11468 satIOContext); 11469 11470 /*smEnqueueIO(smRoot, satIOContext);*/ 11471 11472 tdsmIOCompletedCB( smRoot, 11473 smIORequest, 11474 smIOSuccess, 11475 SCSI_STAT_CHECK_CONDITION, 11476 satIOContext->pSmSenseData, 11477 satIOContext->interruptContext ); 11478 11479 SM_DBG1(("smsatFormatUnit: return control!!!\n")); 11480 return SM_RC_SUCCESS; 11481 } 11482 11483 /* defect list header filed, if exists, SAT rev8, Table 37, p48 */ 11484 if (scsiCmnd->cdb[1] & SCSI_FORMAT_UNIT_FMTDATA_MASK) 11485 { 11486 /* case 1,2,3 */ 11487 /* IMMED 1; FOV 0; FOV 1, DCRT 1, IP 0 */ 11488 if ( (scsiCmnd->cdb[7] & SCSI_FORMAT_UNIT_IMMED_MASK) || 11489 ( !(scsiCmnd->cdb[7] & SCSI_FORMAT_UNIT_FOV_MASK)) || 11490 ( (scsiCmnd->cdb[7] & SCSI_FORMAT_UNIT_FOV_MASK) && 11491 (scsiCmnd->cdb[7] & SCSI_FORMAT_UNIT_DCRT_MASK) && 11492 !(scsiCmnd->cdb[7] & SCSI_FORMAT_UNIT_IP_MASK)) 11493 ) 11494 { 11495 /*smEnqueueIO(smRoot, satIOContext);*/ 11496 11497 tdsmIOCompletedCB( smRoot, 11498 smIORequest, 11499 smIOSuccess, 11500 SCSI_STAT_GOOD, 11501 agNULL, 11502 satIOContext->interruptContext); 11503 11504 SM_DBG5(("smsatFormatUnit: return defect list case 1\n")); 11505 return SM_RC_SUCCESS; 11506 } 11507 /* case 4,5,6 */ 11508 /* 11509 1. IMMED 0, FOV 1, DCRT 0, IP 0 11510 2. IMMED 0, FOV 1, DCRT 0, IP 1 11511 3. IMMED 0, FOV 1, DCRT 1, IP 1 11512 */ 11513 11514 if ( ( !(scsiCmnd->cdb[7] & SCSI_FORMAT_UNIT_IMMED_MASK) && 11515 (scsiCmnd->cdb[7] & SCSI_FORMAT_UNIT_FOV_MASK) && 11516 !(scsiCmnd->cdb[7] & SCSI_FORMAT_UNIT_DCRT_MASK) && 11517 !(scsiCmnd->cdb[7] & SCSI_FORMAT_UNIT_IP_MASK) ) 11518 || 11519 ( !(scsiCmnd->cdb[7] & SCSI_FORMAT_UNIT_IMMED_MASK) && 11520 (scsiCmnd->cdb[7] & SCSI_FORMAT_UNIT_FOV_MASK) && 11521 !(scsiCmnd->cdb[7] & SCSI_FORMAT_UNIT_DCRT_MASK) && 11522 (scsiCmnd->cdb[7] & SCSI_FORMAT_UNIT_IP_MASK) ) 11523 || 11524 ( !(scsiCmnd->cdb[7] & SCSI_FORMAT_UNIT_IMMED_MASK) && 11525 (scsiCmnd->cdb[7] & SCSI_FORMAT_UNIT_FOV_MASK) && 11526 (scsiCmnd->cdb[7] & SCSI_FORMAT_UNIT_DCRT_MASK) && 11527 (scsiCmnd->cdb[7] & SCSI_FORMAT_UNIT_IP_MASK) ) 11528 ) 11529 { 11530 11531 smsatSetSensePayload( pSense, 11532 SCSI_SNSKEY_ILLEGAL_REQUEST, 11533 0, 11534 SCSI_SNSCODE_INVALID_FIELD_PARAMETER_LIST, 11535 satIOContext); 11536 11537 /*smEnqueueIO(smRoot, satIOContext);*/ 11538 11539 tdsmIOCompletedCB( smRoot, 11540 smIORequest, 11541 smIOSuccess, 11542 SCSI_STAT_CHECK_CONDITION, 11543 satIOContext->pSmSenseData, 11544 satIOContext->interruptContext ); 11545 11546 SM_DBG5(("smsatFormatUnit: return defect list case 2\n")); 11547 return SM_RC_SUCCESS; 11548 11549 } 11550 } 11551 11552 11553 /* 11554 * Send the completion response now. 11555 */ 11556 /*smEnqueueIO(smRoot, satIOContext);*/ 11557 11558 tdsmIOCompletedCB( smRoot, 11559 smIORequest, 11560 smIOSuccess, 11561 SCSI_STAT_GOOD, 11562 agNULL, 11563 satIOContext->interruptContext); 11564 11565 SM_DBG5(("smsatFormatUnit: return last\n")); 11566 return SM_RC_SUCCESS; 11567 } 11568 11569 osGLOBAL bit32 11570 smsatSendDiagnostic( 11571 smRoot_t *smRoot, 11572 smIORequest_t *smIORequest, 11573 smDeviceHandle_t *smDeviceHandle, 11574 smScsiInitiatorRequest_t *smScsiRequest, 11575 smSatIOContext_t *satIOContext 11576 ) 11577 { 11578 bit32 status; 11579 bit32 agRequestType; 11580 smDeviceData_t *pSatDevData; 11581 smScsiRspSense_t *pSense; 11582 smIniScsiCmnd_t *scsiCmnd; 11583 agsaFisRegHostToDevice_t *fis; 11584 bit32 parmLen; 11585 11586 pSense = satIOContext->pSense; 11587 pSatDevData = satIOContext->pSatDevData; 11588 scsiCmnd = &smScsiRequest->scsiCmnd; 11589 fis = satIOContext->pFis; 11590 11591 SM_DBG5(("smsatSendDiagnostic: start\n")); 11592 11593 /* reset satVerifyState */ 11594 pSatDevData->satVerifyState = 0; 11595 /* no pending diagnostic in background */ 11596 pSatDevData->satBGPendingDiag = agFALSE; 11597 11598 /* table 27, 8.10 p39 SAT Rev8 */ 11599 /* 11600 1. checking PF == 1 11601 2. checking DEVOFFL == 1 11602 3. checking UNITOFFL == 1 11603 4. checking PARAMETER LIST LENGTH != 0 11604 11605 */ 11606 if ( (scsiCmnd->cdb[1] & SCSI_PF_MASK) || 11607 (scsiCmnd->cdb[1] & SCSI_DEVOFFL_MASK) || 11608 (scsiCmnd->cdb[1] & SCSI_UNITOFFL_MASK) || 11609 ( (scsiCmnd->cdb[3] != 0) || (scsiCmnd->cdb[4] != 0) ) 11610 ) 11611 { 11612 smsatSetSensePayload( pSense, 11613 SCSI_SNSKEY_ILLEGAL_REQUEST, 11614 0, 11615 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 11616 satIOContext); 11617 11618 /*smEnqueueIO(smRoot, satIOContext);*/ 11619 11620 tdsmIOCompletedCB( smRoot, 11621 smIORequest, 11622 smIOSuccess, 11623 SCSI_STAT_CHECK_CONDITION, 11624 satIOContext->pSmSenseData, 11625 satIOContext->interruptContext ); 11626 11627 SM_DBG1(("smsatSendDiagnostic: return PF, DEVOFFL, UNITOFFL, PARAM LIST!!!\n")); 11628 return SM_RC_SUCCESS; 11629 } 11630 11631 /* checking CONTROL */ 11632 /* NACA == 1 or LINK == 1*/ 11633 if ( (scsiCmnd->cdb[5] & SCSI_NACA_MASK) || (scsiCmnd->cdb[5] & SCSI_LINK_MASK) ) 11634 { 11635 smsatSetSensePayload( pSense, 11636 SCSI_SNSKEY_ILLEGAL_REQUEST, 11637 0, 11638 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 11639 satIOContext); 11640 11641 /*smEnqueueIO(smRoot, satIOContext);*/ 11642 11643 tdsmIOCompletedCB( smRoot, 11644 smIORequest, 11645 smIOSuccess, 11646 SCSI_STAT_CHECK_CONDITION, 11647 satIOContext->pSmSenseData, 11648 satIOContext->interruptContext ); 11649 11650 SM_DBG1(("smsatSendDiagnostic: return control!!!\n")); 11651 return SM_RC_SUCCESS; 11652 } 11653 11654 parmLen = (scsiCmnd->cdb[3] << 8) + scsiCmnd->cdb[4]; 11655 11656 /* checking SELFTEST bit*/ 11657 /* table 29, 8.10.3, p41 SAT Rev8 */ 11658 /* case 1 */ 11659 if ( !(scsiCmnd->cdb[1] & SCSI_SEND_DIAGNOSTIC_SELFTEST_MASK) && 11660 (pSatDevData->satSMARTSelfTest == agFALSE) 11661 ) 11662 { 11663 smsatSetSensePayload( pSense, 11664 SCSI_SNSKEY_ILLEGAL_REQUEST, 11665 0, 11666 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 11667 satIOContext); 11668 11669 /*smEnqueueIO(smRoot, satIOContext);*/ 11670 11671 tdsmIOCompletedCB( smRoot, 11672 smIORequest, 11673 smIOSuccess, 11674 SCSI_STAT_CHECK_CONDITION, 11675 satIOContext->pSmSenseData, 11676 satIOContext->interruptContext ); 11677 11678 SM_DBG1(("smsatSendDiagnostic: return Table 29 case 1!!!\n")); 11679 return SM_RC_SUCCESS; 11680 } 11681 11682 /* case 2 */ 11683 if ( !(scsiCmnd->cdb[1] & SCSI_SEND_DIAGNOSTIC_SELFTEST_MASK) && 11684 (pSatDevData->satSMARTSelfTest == agTRUE) && 11685 (pSatDevData->satSMARTEnabled == agFALSE) 11686 ) 11687 { 11688 smsatSetSensePayload( pSense, 11689 SCSI_SNSKEY_ABORTED_COMMAND, 11690 0, 11691 SCSI_SNSCODE_ATA_DEVICE_FEATURE_NOT_ENABLED, 11692 satIOContext); 11693 11694 /*smEnqueueIO(smRoot, satIOContext);*/ 11695 11696 tdsmIOCompletedCB( smRoot, 11697 smIORequest, 11698 smIOSuccess, 11699 SCSI_STAT_CHECK_CONDITION, 11700 satIOContext->pSmSenseData, 11701 satIOContext->interruptContext ); 11702 11703 SM_DBG5(("smsatSendDiagnostic: return Table 29 case 2\n")); 11704 return SM_RC_SUCCESS; 11705 } 11706 /* 11707 case 3 11708 see SELF TEST CODE later 11709 */ 11710 11711 11712 11713 /* case 4 */ 11714 11715 /* 11716 sends three ATA verify commands 11717 11718 */ 11719 if ( ((scsiCmnd->cdb[1] & SCSI_SEND_DIAGNOSTIC_SELFTEST_MASK) && 11720 (pSatDevData->satSMARTSelfTest == agFALSE)) 11721 || 11722 ((scsiCmnd->cdb[1] & SCSI_SEND_DIAGNOSTIC_SELFTEST_MASK) && 11723 (pSatDevData->satSMARTSelfTest == agTRUE) && 11724 (pSatDevData->satSMARTEnabled == agFALSE)) 11725 ) 11726 { 11727 /* 11728 sector count 1, LBA 0 11729 sector count 1, LBA MAX 11730 sector count 1, LBA random 11731 */ 11732 if (pSatDevData->sat48BitSupport == agTRUE) 11733 { 11734 /* sends READ VERIFY SECTOR(S) EXT*/ 11735 fis->h.fisType = 0x27; /* Reg host to device */ 11736 fis->h.c_pmPort = 0x80; /* C Bit is set */ 11737 fis->h.command = SAT_READ_VERIFY_SECTORS_EXT;/* 0x42 */ 11738 fis->h.features = 0; /* FIS reserve */ 11739 fis->d.lbaLow = 0; /* FIS LBA (7 :0 ) */ 11740 fis->d.lbaMid = 0; /* FIS LBA (15:8 ) */ 11741 fis->d.lbaHigh = 0; /* FIS LBA (23:16) */ 11742 fis->d.lbaLowExp = 0; /* FIS LBA (31:24) */ 11743 fis->d.lbaMidExp = 0; /* FIS LBA (39:32) */ 11744 fis->d.lbaHighExp = 0; /* FIS LBA (47:40) */ 11745 fis->d.featuresExp = 0; /* FIS reserve */ 11746 fis->d.sectorCount = 1; /* FIS sector count (7:0) */ 11747 fis->d.sectorCountExp = 0; /* FIS sector count (15:8) */ 11748 fis->d.reserved4 = 0; 11749 fis->d.device = 0x40; /* 01000000 */ 11750 fis->d.control = 0; /* FIS HOB bit clear */ 11751 fis->d.reserved5 = 0; 11752 11753 agRequestType = AGSA_SATA_PROTOCOL_NON_DATA; 11754 } 11755 else 11756 { 11757 /* READ VERIFY SECTOR(S)*/ 11758 fis->h.fisType = 0x27; /* Reg host to device */ 11759 fis->h.c_pmPort = 0x80; /* C Bit is set */ 11760 fis->h.command = SAT_READ_VERIFY_SECTORS;/* 0x40 */ 11761 fis->h.features = 0; /* FIS features NA */ 11762 fis->d.lbaLow = 0; /* FIS LBA (7 :0 ) */ 11763 fis->d.lbaMid = 0; /* FIS LBA (15:8 ) */ 11764 fis->d.lbaHigh = 0; /* FIS LBA (23:16) */ 11765 fis->d.lbaLowExp = 0; 11766 fis->d.lbaMidExp = 0; 11767 fis->d.lbaHighExp = 0; 11768 fis->d.featuresExp = 0; 11769 fis->d.sectorCount = 1; /* FIS sector count (7:0) */ 11770 fis->d.sectorCountExp = 0; 11771 fis->d.reserved4 = 0; 11772 fis->d.device = 0x40; /* 01000000 */ 11773 fis->d.control = 0; /* FIS HOB bit clear */ 11774 fis->d.reserved5 = 0; 11775 11776 agRequestType = AGSA_SATA_PROTOCOL_NON_DATA; 11777 } 11778 11779 /* Initialize CB for SATA completion. 11780 */ 11781 satIOContext->satCompleteCB = &smsatSendDiagnosticCB; 11782 11783 /* 11784 * Prepare SGL and send FIS to LL layer. 11785 */ 11786 satIOContext->reqType = agRequestType; /* Save it */ 11787 11788 status = smsataLLIOStart( smRoot, 11789 smIORequest, 11790 smDeviceHandle, 11791 smScsiRequest, 11792 satIOContext); 11793 11794 11795 SM_DBG5(("smsatSendDiagnostic: return Table 29 case 4\n")); 11796 return (status); 11797 } 11798 /* case 5 */ 11799 if ( (scsiCmnd->cdb[1] & SCSI_SEND_DIAGNOSTIC_SELFTEST_MASK) && 11800 (pSatDevData->satSMARTSelfTest == agTRUE) && 11801 (pSatDevData->satSMARTEnabled == agTRUE) 11802 ) 11803 { 11804 /* sends SMART EXECUTE OFF-LINE IMMEDIATE */ 11805 fis->h.fisType = 0x27; /* Reg host to device */ 11806 fis->h.c_pmPort = 0x80; /* C Bit is set */ 11807 fis->h.command = SAT_SMART; /* 0xB0 */ 11808 fis->h.features = SAT_SMART_EXEUTE_OFF_LINE_IMMEDIATE; /* FIS features NA */ 11809 fis->d.lbaLow = 0x81; /* FIS LBA (7 :0 ) */ 11810 fis->d.lbaMid = 0x4F; /* FIS LBA (15:8 ) */ 11811 fis->d.lbaHigh = 0xC2; /* FIS LBA (23:16) */ 11812 fis->d.lbaLowExp = 0; 11813 fis->d.lbaMidExp = 0; 11814 fis->d.lbaHighExp = 0; 11815 fis->d.featuresExp = 0; 11816 fis->d.sectorCount = 0; /* FIS sector count (7:0) */ 11817 fis->d.sectorCountExp = 0; 11818 fis->d.reserved4 = 0; 11819 fis->d.device = 0; /* FIS DEV is discared in SATA */ 11820 fis->d.control = 0; /* FIS HOB bit clear */ 11821 fis->d.reserved5 = 0; 11822 11823 agRequestType = AGSA_SATA_PROTOCOL_NON_DATA; 11824 11825 /* Initialize CB for SATA completion. 11826 */ 11827 satIOContext->satCompleteCB = &smsatSendDiagnosticCB; 11828 11829 /* 11830 * Prepare SGL and send FIS to LL layer. 11831 */ 11832 satIOContext->reqType = agRequestType; /* Save it */ 11833 11834 status = smsataLLIOStart( smRoot, 11835 smIORequest, 11836 smDeviceHandle, 11837 smScsiRequest, 11838 satIOContext); 11839 11840 11841 SM_DBG5(("smsatSendDiagnostic: return Table 29 case 5\n")); 11842 return (status); 11843 } 11844 11845 11846 11847 11848 /* SAT rev8 Table29 p41 case 3*/ 11849 /* checking SELF TEST CODE*/ 11850 if ( !(scsiCmnd->cdb[1] & SCSI_SEND_DIAGNOSTIC_SELFTEST_MASK) && 11851 (pSatDevData->satSMARTSelfTest == agTRUE) && 11852 (pSatDevData->satSMARTEnabled == agTRUE) 11853 ) 11854 { 11855 /* SAT rev8 Table28 p40 */ 11856 /* finding self-test code */ 11857 switch ((scsiCmnd->cdb[1] & SCSI_SEND_DIAGNOSTIC_TEST_CODE_MASK) >> 5) 11858 { 11859 case 1: 11860 pSatDevData->satBGPendingDiag = agTRUE; 11861 11862 tdsmIOCompletedCB( smRoot, 11863 smIORequest, 11864 smIOSuccess, 11865 SCSI_STAT_GOOD, 11866 agNULL, 11867 satIOContext->interruptContext ); 11868 /* sends SMART EXECUTE OFF-LINE IMMEDIATE */ 11869 fis->h.fisType = 0x27; /* Reg host to device */ 11870 fis->h.c_pmPort = 0x80; /* C Bit is set */ 11871 fis->h.command = SAT_SMART; /* 0x40 */ 11872 fis->h.features = SAT_SMART_EXEUTE_OFF_LINE_IMMEDIATE; /* FIS features NA */ 11873 fis->d.lbaLow = 0x01; /* FIS LBA (7 :0 ) */ 11874 fis->d.lbaMid = 0x4F; /* FIS LBA (15:8 ) */ 11875 fis->d.lbaHigh = 0xC2; /* FIS LBA (23:16) */ 11876 11877 fis->d.lbaLowExp = 0; 11878 fis->d.lbaMidExp = 0; 11879 fis->d.lbaHighExp = 0; 11880 fis->d.featuresExp = 0; 11881 fis->d.sectorCount = 0; /* FIS sector count (7:0) */ 11882 fis->d.sectorCountExp = 0; 11883 fis->d.reserved4 = 0; 11884 fis->d.device = 0; /* FIS DEV is discared in SATA */ 11885 fis->d.control = 0; /* FIS HOB bit clear */ 11886 fis->d.reserved5 = 0; 11887 11888 agRequestType = AGSA_SATA_PROTOCOL_NON_DATA; 11889 11890 /* Initialize CB for SATA completion. 11891 */ 11892 satIOContext->satCompleteCB = &smsatSendDiagnosticCB; 11893 11894 /* 11895 * Prepare SGL and send FIS to LL layer. 11896 */ 11897 satIOContext->reqType = agRequestType; /* Save it */ 11898 11899 status = smsataLLIOStart( smRoot, 11900 smIORequest, 11901 smDeviceHandle, 11902 smScsiRequest, 11903 satIOContext); 11904 11905 11906 SM_DBG5(("smsatSendDiagnostic: return Table 28 case 1\n")); 11907 return (status); 11908 case 2: 11909 pSatDevData->satBGPendingDiag = agTRUE; 11910 11911 tdsmIOCompletedCB( smRoot, 11912 smIORequest, 11913 smIOSuccess, 11914 SCSI_STAT_GOOD, 11915 agNULL, 11916 satIOContext->interruptContext ); 11917 11918 11919 /* issuing SMART EXECUTE OFF-LINE IMMEDIATE */ 11920 fis->h.fisType = 0x27; /* Reg host to device */ 11921 fis->h.c_pmPort = 0x80; /* C Bit is set */ 11922 fis->h.command = SAT_SMART; /* 0x40 */ 11923 fis->h.features = SAT_SMART_EXEUTE_OFF_LINE_IMMEDIATE; /* FIS features NA */ 11924 fis->d.lbaLow = 0x02; /* FIS LBA (7 :0 ) */ 11925 fis->d.lbaMid = 0x4F; /* FIS LBA (15:8 ) */ 11926 fis->d.lbaHigh = 0xC2; /* FIS LBA (23:16) */ 11927 fis->d.lbaLowExp = 0; 11928 fis->d.lbaMidExp = 0; 11929 fis->d.lbaHighExp = 0; 11930 fis->d.featuresExp = 0; 11931 fis->d.sectorCount = 0; /* FIS sector count (7:0) */ 11932 fis->d.sectorCountExp = 0; 11933 fis->d.reserved4 = 0; 11934 fis->d.device = 0; /* FIS DEV is discared in SATA */ 11935 fis->d.control = 0; /* FIS HOB bit clear */ 11936 fis->d.reserved5 = 0; 11937 11938 agRequestType = AGSA_SATA_PROTOCOL_NON_DATA; 11939 11940 /* Initialize CB for SATA completion. 11941 */ 11942 satIOContext->satCompleteCB = &smsatSendDiagnosticCB; 11943 11944 /* 11945 * Prepare SGL and send FIS to LL layer. 11946 */ 11947 satIOContext->reqType = agRequestType; /* Save it */ 11948 11949 status = smsataLLIOStart( smRoot, 11950 smIORequest, 11951 smDeviceHandle, 11952 smScsiRequest, 11953 satIOContext); 11954 11955 11956 SM_DBG5(("smsatSendDiagnostic: return Table 28 case 2\n")); 11957 return (status); 11958 case 4: 11959 11960 if (parmLen != 0) 11961 { 11962 /* check condition */ 11963 smsatSetSensePayload( pSense, 11964 SCSI_SNSKEY_ILLEGAL_REQUEST, 11965 0, 11966 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 11967 satIOContext); 11968 11969 /*smEnqueueIO(smRoot, satIOContext);*/ 11970 11971 tdsmIOCompletedCB( smRoot, 11972 smIORequest, 11973 smIOSuccess, 11974 SCSI_STAT_CHECK_CONDITION, 11975 satIOContext->pSmSenseData, 11976 satIOContext->interruptContext ); 11977 11978 SM_DBG1(("smsatSendDiagnostic: case 4, non zero ParmLen %d!!!\n", parmLen)); 11979 return SM_RC_SUCCESS; 11980 } 11981 if (pSatDevData->satBGPendingDiag == agTRUE) 11982 { 11983 /* sends SMART EXECUTE OFF-LINE IMMEDIATE abort */ 11984 fis->h.fisType = 0x27; /* Reg host to device */ 11985 fis->h.c_pmPort = 0x80; /* C Bit is set */ 11986 fis->h.command = SAT_SMART; /* 0x40 */ 11987 fis->h.features = SAT_SMART_EXEUTE_OFF_LINE_IMMEDIATE; /* FIS features NA */ 11988 fis->d.lbaLow = 0x7F; /* FIS LBA (7 :0 ) */ 11989 fis->d.lbaMid = 0x4F; /* FIS LBA (15:8 ) */ 11990 fis->d.lbaHigh = 0xC2; /* FIS LBA (23:16) */ 11991 11992 fis->d.lbaLowExp = 0; 11993 fis->d.lbaMidExp = 0; 11994 fis->d.lbaHighExp = 0; 11995 fis->d.featuresExp = 0; 11996 fis->d.sectorCount = 0; /* FIS sector count (7:0) */ 11997 fis->d.sectorCountExp = 0; 11998 fis->d.reserved4 = 0; 11999 fis->d.device = 0; /* FIS DEV is discared in SATA */ 12000 fis->d.control = 0; /* FIS HOB bit clear */ 12001 fis->d.reserved5 = 0; 12002 12003 agRequestType = AGSA_SATA_PROTOCOL_NON_DATA; 12004 12005 /* Initialize CB for SATA completion. 12006 */ 12007 satIOContext->satCompleteCB = &smsatSendDiagnosticCB; 12008 12009 /* 12010 * Prepare SGL and send FIS to LL layer. 12011 */ 12012 satIOContext->reqType = agRequestType; /* Save it */ 12013 12014 status = smsataLLIOStart( smRoot, 12015 smIORequest, 12016 smDeviceHandle, 12017 smScsiRequest, 12018 satIOContext); 12019 12020 12021 SM_DBG5(("smsatSendDiagnostic: send SAT_SMART_EXEUTE_OFF_LINE_IMMEDIATE case 3\n")); 12022 SM_DBG5(("smsatSendDiagnostic: Table 28 case 4\n")); 12023 return (status); 12024 } 12025 else 12026 { 12027 /* check condition */ 12028 smsatSetSensePayload( pSense, 12029 SCSI_SNSKEY_ILLEGAL_REQUEST, 12030 0, 12031 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 12032 satIOContext); 12033 12034 /*smEnqueueIO(smRoot, satIOContext);*/ 12035 12036 tdsmIOCompletedCB( smRoot, 12037 smIORequest, 12038 smIOSuccess, 12039 SCSI_STAT_CHECK_CONDITION, 12040 satIOContext->pSmSenseData, 12041 satIOContext->interruptContext ); 12042 12043 SM_DBG1(("smsatSendDiagnostic: case 4, no pending diagnostic in background!!!\n")); 12044 SM_DBG5(("smsatSendDiagnostic: Table 28 case 4\n")); 12045 return SM_RC_SUCCESS; 12046 } 12047 break; 12048 case 5: 12049 /* issuing SMART EXECUTE OFF-LINE IMMEDIATE */ 12050 fis->h.fisType = 0x27; /* Reg host to device */ 12051 fis->h.c_pmPort = 0x80; /* C Bit is set */ 12052 fis->h.command = SAT_SMART; /* 0x40 */ 12053 fis->h.features = SAT_SMART_EXEUTE_OFF_LINE_IMMEDIATE; /* FIS features NA */ 12054 fis->d.lbaLow = 0x81; /* FIS LBA (7 :0 ) */ 12055 fis->d.lbaMid = 0x4F; /* FIS LBA (15:8 ) */ 12056 fis->d.lbaHigh = 0xC2; /* FIS LBA (23:16) */ 12057 fis->d.lbaLowExp = 0; 12058 fis->d.lbaMidExp = 0; 12059 fis->d.lbaHighExp = 0; 12060 fis->d.featuresExp = 0; 12061 fis->d.sectorCount = 0; /* FIS sector count (7:0) */ 12062 fis->d.sectorCountExp = 0; 12063 fis->d.reserved4 = 0; 12064 fis->d.device = 0; /* FIS DEV is discared in SATA */ 12065 fis->d.control = 0; /* FIS HOB bit clear */ 12066 fis->d.reserved5 = 0; 12067 12068 agRequestType = AGSA_SATA_PROTOCOL_NON_DATA; 12069 12070 /* Initialize CB for SATA completion. 12071 */ 12072 satIOContext->satCompleteCB = &smsatSendDiagnosticCB; 12073 12074 /* 12075 * Prepare SGL and send FIS to LL layer. 12076 */ 12077 satIOContext->reqType = agRequestType; /* Save it */ 12078 12079 status = smsataLLIOStart( smRoot, 12080 smIORequest, 12081 smDeviceHandle, 12082 smScsiRequest, 12083 satIOContext); 12084 12085 12086 SM_DBG5(("smsatSendDiagnostic: return Table 28 case 5\n")); 12087 return (status); 12088 case 6: 12089 /* issuing SMART EXECUTE OFF-LINE IMMEDIATE */ 12090 fis->h.fisType = 0x27; /* Reg host to device */ 12091 fis->h.c_pmPort = 0x80; /* C Bit is set */ 12092 fis->h.command = SAT_SMART; /* 0x40 */ 12093 fis->h.features = SAT_SMART_EXEUTE_OFF_LINE_IMMEDIATE; /* FIS features NA */ 12094 fis->d.lbaLow = 0x82; /* FIS LBA (7 :0 ) */ 12095 fis->d.lbaMid = 0x4F; /* FIS LBA (15:8 ) */ 12096 fis->d.lbaHigh = 0xC2; /* FIS LBA (23:16) */ 12097 fis->d.lbaLowExp = 0; 12098 fis->d.lbaMidExp = 0; 12099 fis->d.lbaHighExp = 0; 12100 fis->d.featuresExp = 0; 12101 fis->d.sectorCount = 0; /* FIS sector count (7:0) */ 12102 fis->d.sectorCountExp = 0; 12103 fis->d.reserved4 = 0; 12104 fis->d.device = 0; /* FIS DEV is discared in SATA */ 12105 fis->d.control = 0; /* FIS HOB bit clear */ 12106 fis->d.reserved5 = 0; 12107 12108 agRequestType = AGSA_SATA_PROTOCOL_NON_DATA; 12109 12110 /* Initialize CB for SATA completion. 12111 */ 12112 satIOContext->satCompleteCB = &smsatSendDiagnosticCB; 12113 12114 /* 12115 * Prepare SGL and send FIS to LL layer. 12116 */ 12117 satIOContext->reqType = agRequestType; /* Save it */ 12118 12119 status = smsataLLIOStart( smRoot, 12120 smIORequest, 12121 smDeviceHandle, 12122 smScsiRequest, 12123 satIOContext); 12124 12125 12126 SM_DBG5(("smsatSendDiagnostic: return Table 28 case 6\n")); 12127 return (status); 12128 case 0: 12129 case 3: /* fall through */ 12130 case 7: /* fall through */ 12131 default: 12132 break; 12133 }/* switch */ 12134 12135 /* returns the results of default self-testing, which is good */ 12136 /*smEnqueueIO(smRoot, satIOContext);*/ 12137 12138 tdsmIOCompletedCB( smRoot, 12139 smIORequest, 12140 smIOSuccess, 12141 SCSI_STAT_GOOD, 12142 agNULL, 12143 satIOContext->interruptContext ); 12144 12145 SM_DBG5(("smsatSendDiagnostic: return Table 28 case 0,3,7 and default\n")); 12146 return SM_RC_SUCCESS; 12147 } 12148 12149 12150 /*smEnqueueIO(smRoot, satIOContext);*/ 12151 12152 tdsmIOCompletedCB( smRoot, 12153 smIORequest, 12154 smIOSuccess, 12155 SCSI_STAT_GOOD, 12156 agNULL, 12157 satIOContext->interruptContext ); 12158 12159 12160 SM_DBG5(("smsatSendDiagnostic: return last\n")); 12161 return SM_RC_SUCCESS; 12162 12163 } 12164 12165 osGLOBAL bit32 12166 smsatStartStopUnit( 12167 smRoot_t *smRoot, 12168 smIORequest_t *smIORequest, 12169 smDeviceHandle_t *smDeviceHandle, 12170 smScsiInitiatorRequest_t *smScsiRequest, 12171 smSatIOContext_t *satIOContext 12172 ) 12173 { 12174 bit32 status; 12175 bit32 agRequestType; 12176 smDeviceData_t *pSatDevData; 12177 smScsiRspSense_t *pSense; 12178 smIniScsiCmnd_t *scsiCmnd; 12179 agsaFisRegHostToDevice_t *fis; 12180 12181 pSense = satIOContext->pSense; 12182 pSatDevData = satIOContext->pSatDevData; 12183 scsiCmnd = &smScsiRequest->scsiCmnd; 12184 fis = satIOContext->pFis; 12185 12186 SM_DBG5(("smsatStartStopUnit: start\n")); 12187 12188 /* checking CONTROL */ 12189 /* NACA == 1 or LINK == 1*/ 12190 if ( (scsiCmnd->cdb[5] & SCSI_NACA_MASK) || (scsiCmnd->cdb[5] & SCSI_LINK_MASK) ) 12191 { 12192 smsatSetSensePayload( pSense, 12193 SCSI_SNSKEY_ILLEGAL_REQUEST, 12194 0, 12195 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 12196 satIOContext); 12197 12198 /*smEnqueueIO(smRoot, satIOContext);*/ 12199 12200 tdsmIOCompletedCB( smRoot, 12201 smIORequest, 12202 smIOSuccess, 12203 SCSI_STAT_CHECK_CONDITION, 12204 satIOContext->pSmSenseData, 12205 satIOContext->interruptContext ); 12206 12207 SM_DBG1(("smsatStartStopUnit: return control!!!\n")); 12208 return SM_RC_SUCCESS; 12209 } 12210 12211 /* Spec p55, Table 48 checking START and LOEJ bit */ 12212 /* case 1 */ 12213 if ( !(scsiCmnd->cdb[4] & SCSI_START_MASK) && !(scsiCmnd->cdb[4] & SCSI_LOEJ_MASK) ) 12214 { 12215 if ( (scsiCmnd->cdb[1] & SCSI_IMMED_MASK) ) 12216 { 12217 /* immed bit , SAT rev 8, 9.11.2.1 p 54*/ 12218 /*smEnqueueIO(smRoot, satIOContext);*/ 12219 12220 tdsmIOCompletedCB( smRoot, 12221 smIORequest, 12222 smIOSuccess, 12223 SCSI_STAT_GOOD, 12224 agNULL, 12225 satIOContext->interruptContext ); 12226 SM_DBG5(("smsatStartStopUnit: return table48 case 1-1\n")); 12227 return SM_RC_SUCCESS; 12228 } 12229 /* sends FLUSH CACHE or FLUSH CACHE EXT */ 12230 if (pSatDevData->sat48BitSupport == agTRUE) 12231 { 12232 /* FLUSH CACHE EXT */ 12233 fis->h.fisType = 0x27; /* Reg host to device */ 12234 fis->h.c_pmPort = 0x80; /* C Bit is set */ 12235 12236 fis->h.command = SAT_FLUSH_CACHE_EXT; /* 0xEA */ 12237 fis->h.features = 0; /* FIS reserve */ 12238 fis->d.featuresExp = 0; /* FIS reserve */ 12239 fis->d.sectorCount = 0; /* FIS sector count (7:0) */ 12240 fis->d.sectorCountExp = 0; /* FIS sector count (15:8) */ 12241 fis->d.lbaLow = 0; /* FIS LBA (7 :0 ) */ 12242 fis->d.lbaLowExp = 0; /* FIS LBA (31:24) */ 12243 fis->d.lbaMid = 0; /* FIS LBA (15:8 ) */ 12244 fis->d.lbaMidExp = 0; /* FIS LBA (39:32) */ 12245 fis->d.lbaHigh = 0; /* FIS LBA (23:16) */ 12246 fis->d.lbaHighExp = 0; /* FIS LBA (47:40) */ 12247 fis->d.device = 0; /* FIS DEV is discared in SATA */ 12248 fis->d.control = 0; /* FIS HOB bit clear */ 12249 fis->d.reserved4 = 0; 12250 fis->d.reserved5 = 0; 12251 12252 agRequestType = AGSA_SATA_PROTOCOL_NON_DATA; 12253 } 12254 else 12255 { 12256 /* FLUSH CACHE */ 12257 fis->h.fisType = 0x27; /* Reg host to device */ 12258 fis->h.c_pmPort = 0x80; /* C Bit is set */ 12259 12260 fis->h.command = SAT_FLUSH_CACHE; /* 0xE7 */ 12261 fis->h.features = 0; /* FIS features NA */ 12262 fis->d.lbaLow = 0; /* FIS LBA (7 :0 ) */ 12263 fis->d.lbaMid = 0; /* FIS LBA (15:8 ) */ 12264 fis->d.lbaHigh = 0; /* FIS LBA (23:16) */ 12265 fis->d.lbaLowExp = 0; 12266 fis->d.lbaMidExp = 0; 12267 fis->d.lbaHighExp = 0; 12268 fis->d.featuresExp = 0; 12269 fis->d.sectorCount = 0; /* FIS sector count (7:0) */ 12270 fis->d.sectorCountExp = 0; 12271 fis->d.device = 0; /* FIS DEV is discared in SATA */ 12272 fis->d.control = 0; /* FIS HOB bit clear */ 12273 fis->d.reserved4 = 0; 12274 fis->d.reserved5 = 0; 12275 12276 agRequestType = AGSA_SATA_PROTOCOL_NON_DATA; 12277 } 12278 12279 /* Initialize CB for SATA completion. 12280 */ 12281 satIOContext->satCompleteCB = &smsatStartStopUnitCB; 12282 12283 /* 12284 * Prepare SGL and send FIS to LL layer. 12285 */ 12286 satIOContext->reqType = agRequestType; /* Save it */ 12287 12288 status = smsataLLIOStart( smRoot, 12289 smIORequest, 12290 smDeviceHandle, 12291 smScsiRequest, 12292 satIOContext); 12293 12294 12295 SM_DBG5(("smsatStartStopUnit: return table48 case 1\n")); 12296 return (status); 12297 } 12298 /* case 2 */ 12299 else if ( (scsiCmnd->cdb[4] & SCSI_START_MASK) && !(scsiCmnd->cdb[4] & SCSI_LOEJ_MASK) ) 12300 { 12301 /* immed bit , SAT rev 8, 9.11.2.1 p 54*/ 12302 if ( (scsiCmnd->cdb[1] & SCSI_IMMED_MASK) ) 12303 { 12304 /*smEnqueueIO(smRoot, satIOContext);*/ 12305 12306 tdsmIOCompletedCB( smRoot, 12307 smIORequest, 12308 smIOSuccess, 12309 SCSI_STAT_GOOD, 12310 agNULL, 12311 satIOContext->interruptContext ); 12312 12313 SM_DBG5(("smsatStartStopUnit: return table48 case 2 1\n")); 12314 return SM_RC_SUCCESS; 12315 } 12316 /* 12317 sends READ_VERIFY_SECTORS(_EXT) 12318 sector count 1, any LBA between zero to Maximum 12319 */ 12320 if (pSatDevData->sat48BitSupport == agTRUE) 12321 { 12322 /* READ VERIFY SECTOR(S) EXT*/ 12323 fis->h.fisType = 0x27; /* Reg host to device */ 12324 fis->h.c_pmPort = 0x80; /* C Bit is set */ 12325 12326 fis->h.command = SAT_READ_VERIFY_SECTORS_EXT;/* 0x42 */ 12327 fis->h.features = 0; /* FIS reserve */ 12328 fis->d.lbaLow = 0x01; /* FIS LBA (7 :0 ) */ 12329 fis->d.lbaMid = 0x00; /* FIS LBA (15:8 ) */ 12330 fis->d.lbaHigh = 0x00; /* FIS LBA (23:16) */ 12331 fis->d.lbaLowExp = 0x00; /* FIS LBA (31:24) */ 12332 fis->d.lbaMidExp = 0x00; /* FIS LBA (39:32) */ 12333 fis->d.lbaHighExp = 0x00; /* FIS LBA (47:40) */ 12334 fis->d.featuresExp = 0; /* FIS reserve */ 12335 fis->d.sectorCount = 1; /* FIS sector count (7:0) */ 12336 fis->d.sectorCountExp = 0; /* FIS sector count (15:8) */ 12337 fis->d.reserved4 = 0; 12338 fis->d.device = 0x40; /* 01000000 */ 12339 fis->d.control = 0; /* FIS HOB bit clear */ 12340 fis->d.reserved5 = 0; 12341 12342 } 12343 else 12344 { 12345 /* READ VERIFY SECTOR(S)*/ 12346 fis->h.fisType = 0x27; /* Reg host to device */ 12347 fis->h.c_pmPort = 0x80; /* C Bit is set */ 12348 12349 fis->h.command = SAT_READ_VERIFY_SECTORS;/* 0x40 */ 12350 fis->h.features = 0; /* FIS features NA */ 12351 fis->d.lbaLow = 0x01; /* FIS LBA (7 :0 ) */ 12352 fis->d.lbaMid = 0x00; /* FIS LBA (15:8 ) */ 12353 fis->d.lbaHigh = 0x00; /* FIS LBA (23:16) */ 12354 fis->d.lbaLowExp = 0; 12355 fis->d.lbaMidExp = 0; 12356 fis->d.lbaHighExp = 0; 12357 fis->d.featuresExp = 0; 12358 fis->d.sectorCount = 1; /* FIS sector count (7:0) */ 12359 fis->d.sectorCountExp = 0; 12360 fis->d.reserved4 = 0; 12361 fis->d.device = 0x40; /* 01000000 */ 12362 fis->d.control = 0; /* FIS HOB bit clear */ 12363 fis->d.reserved5 = 0; 12364 12365 } 12366 12367 agRequestType = AGSA_SATA_PROTOCOL_NON_DATA; 12368 12369 /* Initialize CB for SATA completion. 12370 */ 12371 satIOContext->satCompleteCB = &smsatStartStopUnitCB; 12372 12373 /* 12374 * Prepare SGL and send FIS to LL layer. 12375 */ 12376 satIOContext->reqType = agRequestType; /* Save it */ 12377 12378 status = smsataLLIOStart( smRoot, 12379 smIORequest, 12380 smDeviceHandle, 12381 smScsiRequest, 12382 satIOContext); 12383 12384 SM_DBG5(("smsatStartStopUnit: return table48 case 2 2\n")); 12385 return status; 12386 } 12387 /* case 3 */ 12388 else if ( !(scsiCmnd->cdb[4] & SCSI_START_MASK) && (scsiCmnd->cdb[4] & SCSI_LOEJ_MASK) ) 12389 { 12390 if(pSatDevData->satRemovableMedia && pSatDevData->satRemovableMediaEnabled) 12391 { 12392 /* support for removal media */ 12393 /* immed bit , SAT rev 8, 9.11.2.1 p 54*/ 12394 if ( (scsiCmnd->cdb[1] & SCSI_IMMED_MASK) ) 12395 { 12396 /*smEnqueueIO(smRoot, satIOContext);*/ 12397 12398 tdsmIOCompletedCB( smRoot, 12399 smIORequest, 12400 smIOSuccess, 12401 SCSI_STAT_GOOD, 12402 agNULL, 12403 satIOContext->interruptContext ); 12404 12405 SM_DBG5(("smsatStartStopUnit: return table48 case 3 1\n")); 12406 return SM_RC_SUCCESS; 12407 } 12408 /* 12409 sends MEDIA EJECT 12410 */ 12411 /* Media Eject fis */ 12412 fis->h.fisType = 0x27; /* Reg host to device */ 12413 fis->h.c_pmPort = 0x80; /* C Bit is set */ 12414 12415 fis->h.command = SAT_MEDIA_EJECT; /* 0xED */ 12416 fis->h.features = 0; /* FIS features NA */ 12417 fis->d.lbaLow = 0; /* FIS LBA (7 :0 ) */ 12418 fis->d.lbaMid = 0; /* FIS LBA (15:8 ) */ 12419 fis->d.lbaHigh = 0; /* FIS LBA (23:16) */ 12420 fis->d.lbaLowExp = 0; 12421 fis->d.lbaMidExp = 0; 12422 fis->d.lbaHighExp = 0; 12423 fis->d.featuresExp = 0; 12424 /* sector count zero */ 12425 fis->d.sectorCount = 0; /* FIS sector count (7:0) */ 12426 fis->d.sectorCountExp = 0; 12427 fis->d.device = 0; /* FIS DEV is discared in SATA */ 12428 fis->d.control = 0; /* FIS HOB bit clear */ 12429 fis->d.reserved4 = 0; 12430 fis->d.reserved5 = 0; 12431 12432 agRequestType = AGSA_SATA_PROTOCOL_NON_DATA; 12433 12434 /* Initialize CB for SATA completion. 12435 */ 12436 satIOContext->satCompleteCB = &smsatStartStopUnitCB; 12437 12438 /* 12439 * Prepare SGL and send FIS to LL layer. 12440 */ 12441 satIOContext->reqType = agRequestType; /* Save it */ 12442 12443 status = smsataLLIOStart( smRoot, 12444 smIORequest, 12445 smDeviceHandle, 12446 smScsiRequest, 12447 satIOContext); 12448 12449 return status; 12450 } 12451 else 12452 { 12453 /* no support for removal media */ 12454 smsatSetSensePayload( pSense, 12455 SCSI_SNSKEY_ILLEGAL_REQUEST, 12456 0, 12457 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 12458 satIOContext); 12459 12460 /*smEnqueueIO(smRoot, satIOContext);*/ 12461 12462 tdsmIOCompletedCB( smRoot, 12463 smIORequest, 12464 smIOSuccess, 12465 SCSI_STAT_CHECK_CONDITION, 12466 satIOContext->pSmSenseData, 12467 satIOContext->interruptContext ); 12468 12469 SM_DBG5(("smsatStartStopUnit: return Table 29 case 3 2\n")); 12470 return SM_RC_SUCCESS; 12471 } 12472 12473 } 12474 /* case 4 */ 12475 else /* ( (scsiCmnd->cdb[4] & SCSI_START_MASK) && (scsiCmnd->cdb[4] & SCSI_LOEJ_MASK) ) */ 12476 { 12477 smsatSetSensePayload( pSense, 12478 SCSI_SNSKEY_ILLEGAL_REQUEST, 12479 0, 12480 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 12481 satIOContext); 12482 12483 /*smEnqueueIO(smRoot, satIOContext);*/ 12484 12485 tdsmIOCompletedCB( smRoot, 12486 smIORequest, 12487 smIOSuccess, 12488 SCSI_STAT_CHECK_CONDITION, 12489 satIOContext->pSmSenseData, 12490 satIOContext->interruptContext ); 12491 12492 SM_DBG5(("smsatStartStopUnit: return Table 29 case 4\n")); 12493 return SM_RC_SUCCESS; 12494 } 12495 } 12496 12497 osGLOBAL bit32 12498 smsatWriteSame10( 12499 smRoot_t *smRoot, 12500 smIORequest_t *smIORequest, 12501 smDeviceHandle_t *smDeviceHandle, 12502 smScsiInitiatorRequest_t *smScsiRequest, 12503 smSatIOContext_t *satIOContext 12504 ) 12505 { 12506 12507 bit32 status; 12508 bit32 agRequestType = AGSA_SATA_PROTOCOL_DMA_WRITE; 12509 smDeviceData_t *pSatDevData; 12510 smScsiRspSense_t *pSense; 12511 smIniScsiCmnd_t *scsiCmnd; 12512 agsaFisRegHostToDevice_t *fis; 12513 bit32 lba = 0; 12514 bit32 tl = 0; 12515 12516 pSense = satIOContext->pSense; 12517 pSatDevData = satIOContext->pSatDevData; 12518 scsiCmnd = &smScsiRequest->scsiCmnd; 12519 fis = satIOContext->pFis; 12520 12521 SM_DBG5(("smsatWriteSame10: start\n")); 12522 12523 /* checking CONTROL */ 12524 /* NACA == 1 or LINK == 1*/ 12525 if ( (scsiCmnd->cdb[9] & SCSI_NACA_MASK) || (scsiCmnd->cdb[9] & SCSI_LINK_MASK) ) 12526 { 12527 smsatSetSensePayload( pSense, 12528 SCSI_SNSKEY_ILLEGAL_REQUEST, 12529 0, 12530 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 12531 satIOContext); 12532 12533 /*smEnqueueIO(smRoot, satIOContext);*/ 12534 12535 tdsmIOCompletedCB( smRoot, 12536 smIORequest, 12537 smIOSuccess, 12538 SCSI_STAT_CHECK_CONDITION, 12539 satIOContext->pSmSenseData, 12540 satIOContext->interruptContext ); 12541 12542 SM_DBG1(("smsatWriteSame10: return control!!!\n")); 12543 return SM_RC_SUCCESS; 12544 } 12545 12546 12547 /* checking LBDATA and PBDATA */ 12548 /* case 1 */ 12549 if ( !(scsiCmnd->cdb[1] & SCSI_WRITE_SAME_LBDATA_MASK) && 12550 !(scsiCmnd->cdb[1] & SCSI_WRITE_SAME_PBDATA_MASK)) 12551 { 12552 SM_DBG5(("smsatWriteSame10: case 1\n")); 12553 /* spec 9.26.2, Table 62, p64, case 1*/ 12554 /* 12555 normal case 12556 just like write in 9.17.1 12557 */ 12558 12559 if ( pSatDevData->sat48BitSupport != agTRUE ) 12560 { 12561 /* 12562 writeSame10 but no support for 48 bit addressing 12563 -> problem in transfer length. Therefore, return check condition 12564 */ 12565 smsatSetSensePayload( pSense, 12566 SCSI_SNSKEY_ILLEGAL_REQUEST, 12567 0, 12568 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 12569 satIOContext); 12570 12571 /*smEnqueueIO(smRoot, satIOContext);*/ 12572 12573 tdsmIOCompletedCB( smRoot, 12574 smIORequest, 12575 smIOSuccess, 12576 SCSI_STAT_CHECK_CONDITION, 12577 satIOContext->pSmSenseData, 12578 satIOContext->interruptContext ); 12579 12580 SM_DBG1(("smsatWriteSame10: return internal checking!!!\n")); 12581 return SM_RC_SUCCESS; 12582 } 12583 12584 /* cdb10; computing LBA and transfer length */ 12585 lba = (scsiCmnd->cdb[2] << (8*3)) + (scsiCmnd->cdb[3] << (8*2)) 12586 + (scsiCmnd->cdb[4] << 8) + scsiCmnd->cdb[5]; 12587 tl = (scsiCmnd->cdb[7] << 8) + scsiCmnd->cdb[8]; 12588 12589 12590 /* Table 34, 9.1, p 46 */ 12591 /* 12592 note: As of 2/10/2006, no support for DMA QUEUED 12593 */ 12594 12595 /* 12596 Table 34, 9.1, p 46, b (footnote) 12597 When no 48-bit addressing support or NCQ, if LBA is beyond (2^28 - 1), 12598 return check condition 12599 */ 12600 if (pSatDevData->satNCQ != agTRUE && 12601 pSatDevData->sat48BitSupport != agTRUE 12602 ) 12603 { 12604 if (lba > SAT_TR_LBA_LIMIT - 1) /* SAT_TR_LBA_LIMIT is 2^28, 0x10000000 */ 12605 { 12606 smsatSetSensePayload( pSense, 12607 SCSI_SNSKEY_ILLEGAL_REQUEST, 12608 0, 12609 SCSI_SNSCODE_LOGICAL_BLOCK_ADDRESS_OUT_OF_RANGE, 12610 satIOContext); 12611 12612 /*smEnqueueIO(smRoot, satIOContext);*/ 12613 12614 tdsmIOCompletedCB( smRoot, 12615 smIORequest, 12616 smIOSuccess, 12617 SCSI_STAT_CHECK_CONDITION, 12618 satIOContext->pSmSenseData, 12619 satIOContext->interruptContext ); 12620 12621 SM_DBG1(("smsatWriteSame10: return LBA out of range!!!\n")); 12622 return SM_RC_SUCCESS; 12623 } 12624 } 12625 12626 12627 if (lba + tl <= SAT_TR_LBA_LIMIT) 12628 { 12629 if (pSatDevData->satDMASupport == agTRUE && pSatDevData->satDMAEnabled == agTRUE) 12630 { 12631 /* case 2 */ 12632 /* WRITE DMA */ 12633 /* can't fit the transfer length since WRITE DMA has 1 byte for sector count */ 12634 SM_DBG1(("smsatWriteSame10: case 1-2 !!! error due to writesame10!!!\n")); 12635 smsatSetSensePayload( pSense, 12636 SCSI_SNSKEY_ILLEGAL_REQUEST, 12637 0, 12638 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 12639 satIOContext); 12640 12641 /*smEnqueueIO(smRoot, satIOContext);*/ 12642 12643 tdsmIOCompletedCB( smRoot, 12644 smIORequest, 12645 smIOSuccess, 12646 SCSI_STAT_CHECK_CONDITION, 12647 satIOContext->pSmSenseData, 12648 satIOContext->interruptContext ); 12649 return SM_RC_SUCCESS; 12650 } 12651 else 12652 { 12653 /* case 1 */ 12654 /* WRITE MULTIPLE or WRITE SECTOR(S) */ 12655 /* WRITE SECTORS is chosen for easier implemetation */ 12656 /* can't fit the transfer length since WRITE DMA has 1 byte for sector count */ 12657 SM_DBG1(("smsatWriteSame10: case 1-1 !!! error due to writesame10!!!\n")); 12658 smsatSetSensePayload( pSense, 12659 SCSI_SNSKEY_ILLEGAL_REQUEST, 12660 0, 12661 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 12662 satIOContext); 12663 12664 /*smEnqueueIO(smRoot, satIOContext);*/ 12665 12666 tdsmIOCompletedCB( smRoot, 12667 smIORequest, 12668 smIOSuccess, 12669 SCSI_STAT_CHECK_CONDITION, 12670 satIOContext->pSmSenseData, 12671 satIOContext->interruptContext ); 12672 return SM_RC_SUCCESS; 12673 } 12674 } /* end of case 1 and 2 */ 12675 12676 /* case 3 and 4 */ 12677 if (pSatDevData->sat48BitSupport == agTRUE) 12678 { 12679 if (pSatDevData->satDMASupport == agTRUE && pSatDevData->satDMAEnabled == agTRUE) 12680 { 12681 /* case 3 */ 12682 /* WRITE DMA EXT or WRITE DMA FUA EXT */ 12683 /* WRITE DMA EXT is chosen since WRITE SAME does not have FUA bit */ 12684 SM_DBG5(("smsatWriteSame10: case 1-3\n")); 12685 fis->h.fisType = 0x27; /* Reg host to device */ 12686 fis->h.c_pmPort = 0x80; /* C Bit is set */ 12687 12688 fis->h.command = SAT_WRITE_DMA_EXT; /* 0x35 */ 12689 12690 fis->h.features = 0; /* FIS reserve */ 12691 fis->d.lbaLow = scsiCmnd->cdb[5]; /* FIS LBA (7 :0 ) */ 12692 fis->d.lbaMid = scsiCmnd->cdb[4]; /* FIS LBA (15:8 ) */ 12693 fis->d.lbaHigh = scsiCmnd->cdb[3]; /* FIS LBA (23:16) */ 12694 fis->d.device = 0x40; /* FIS LBA mode set */ 12695 fis->d.lbaLowExp = scsiCmnd->cdb[2]; /* FIS LBA (31:24) */ 12696 fis->d.lbaMidExp = 0; /* FIS LBA (39:32) */ 12697 fis->d.lbaHighExp = 0; /* FIS LBA (47:40) */ 12698 fis->d.featuresExp = 0; /* FIS reserve */ 12699 if (tl == 0) 12700 { 12701 /* error check 12702 ATA spec, p125, 6.17.29 12703 pSatDevData->satMaxUserAddrSectors should be 0x0FFFFFFF 12704 and allowed value is 0x0FFFFFFF - 1 12705 */ 12706 if (pSatDevData->satMaxUserAddrSectors > 0x0FFFFFFF) 12707 { 12708 SM_DBG1(("smsatWriteSame10: case 3 !!! warning can't fit sectors!!!\n")); 12709 smsatSetSensePayload( pSense, 12710 SCSI_SNSKEY_ILLEGAL_REQUEST, 12711 0, 12712 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 12713 satIOContext); 12714 12715 /*smEnqueueIO(smRoot, satIOContext);*/ 12716 12717 tdsmIOCompletedCB( smRoot, 12718 smIORequest, 12719 smIOSuccess, 12720 SCSI_STAT_CHECK_CONDITION, 12721 satIOContext->pSmSenseData, 12722 satIOContext->interruptContext ); 12723 return SM_RC_SUCCESS; 12724 } 12725 } 12726 /* one sector at a time */ 12727 fis->d.sectorCount = 1; /* FIS sector count (7:0) */ 12728 fis->d.sectorCountExp = 0; /* FIS sector count (15:8) */ 12729 fis->d.reserved4 = 0; 12730 fis->d.control = 0; /* FIS HOB bit clear */ 12731 fis->d.reserved5 = 0; 12732 12733 agRequestType = AGSA_SATA_PROTOCOL_DMA_WRITE; 12734 } 12735 else 12736 { 12737 /* case 4 */ 12738 /* WRITE MULTIPLE EXT or WRITE MULTIPLE FUA EXT or WRITE SECTOR(S) EXT */ 12739 /* WRITE SECTORS EXT is chosen for easier implemetation */ 12740 SM_DBG5(("smsatWriteSame10: case 1-4\n")); 12741 fis->h.fisType = 0x27; /* Reg host to device */ 12742 fis->h.c_pmPort = 0x80; /* C Bit is set */ 12743 12744 fis->h.command = SAT_WRITE_SECTORS_EXT; /* 0x34 */ 12745 fis->h.features = 0; /* FIS reserve */ 12746 fis->d.lbaLow = scsiCmnd->cdb[5]; /* FIS LBA (7 :0 ) */ 12747 fis->d.lbaMid = scsiCmnd->cdb[4]; /* FIS LBA (15:8 ) */ 12748 fis->d.lbaHigh = scsiCmnd->cdb[3]; /* FIS LBA (23:16) */ 12749 fis->d.device = 0x40; /* FIS LBA mode set */ 12750 fis->d.lbaLowExp = scsiCmnd->cdb[2]; /* FIS LBA (31:24) */ 12751 fis->d.lbaMidExp = 0; /* FIS LBA (39:32) */ 12752 fis->d.lbaHighExp = 0; /* FIS LBA (47:40) */ 12753 fis->d.featuresExp = 0; /* FIS reserve */ 12754 if (tl == 0) 12755 { 12756 /* error check 12757 ATA spec, p125, 6.17.29 12758 pSatDevData->satMaxUserAddrSectors should be 0x0FFFFFFF 12759 and allowed value is 0x0FFFFFFF - 1 12760 */ 12761 if (pSatDevData->satMaxUserAddrSectors > 0x0FFFFFFF) 12762 { 12763 SM_DBG1(("smsatWriteSame10: case 4 !!! warning can't fit sectors!!!\n")); 12764 smsatSetSensePayload( pSense, 12765 SCSI_SNSKEY_ILLEGAL_REQUEST, 12766 0, 12767 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 12768 satIOContext); 12769 12770 /*smEnqueueIO(smRoot, satIOContext);*/ 12771 12772 tdsmIOCompletedCB( smRoot, 12773 smIORequest, 12774 smIOSuccess, 12775 SCSI_STAT_CHECK_CONDITION, 12776 satIOContext->pSmSenseData, 12777 satIOContext->interruptContext ); 12778 return SM_RC_SUCCESS; 12779 } 12780 } 12781 /* one sector at a time */ 12782 fis->d.sectorCount = 1; /* FIS sector count (7:0) */ 12783 fis->d.sectorCountExp = 0; /* FIS sector count (15:8) */ 12784 fis->d.reserved4 = 0; 12785 fis->d.control = 0; /* FIS HOB bit clear */ 12786 fis->d.reserved5 = 0; 12787 12788 agRequestType = AGSA_SATA_PROTOCOL_PIO_WRITE; 12789 } 12790 } 12791 12792 /* case 5 */ 12793 if (pSatDevData->satNCQ == agTRUE) 12794 { 12795 /* WRITE FPDMA QUEUED */ 12796 if (pSatDevData->sat48BitSupport != agTRUE) 12797 { 12798 SM_DBG1(("smsatWriteSame10: case 1-5 !!! error NCQ but 28 bit address support!!!\n")); 12799 smsatSetSensePayload( pSense, 12800 SCSI_SNSKEY_ILLEGAL_REQUEST, 12801 0, 12802 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 12803 satIOContext); 12804 12805 /*smEnqueueIO(smRoot, satIOContext);*/ 12806 12807 tdsmIOCompletedCB( smRoot, 12808 smIORequest, 12809 smIOSuccess, 12810 SCSI_STAT_CHECK_CONDITION, 12811 satIOContext->pSmSenseData, 12812 satIOContext->interruptContext ); 12813 return SM_RC_SUCCESS; 12814 } 12815 SM_DBG5(("smsatWriteSame10: case 1-5\n")); 12816 12817 /* Support 48-bit FPDMA addressing, use WRITE FPDMA QUEUE command */ 12818 12819 fis->h.fisType = 0x27; /* Reg host to device */ 12820 fis->h.c_pmPort = 0x80; /* C Bit is set */ 12821 fis->h.command = SAT_WRITE_FPDMA_QUEUED; /* 0x61 */ 12822 12823 if (tl == 0) 12824 { 12825 /* error check 12826 ATA spec, p125, 6.17.29 12827 pSatDevData->satMaxUserAddrSectors should be 0x0FFFFFFF 12828 and allowed value is 0x0FFFFFFF - 1 12829 */ 12830 if (pSatDevData->satMaxUserAddrSectors > 0x0FFFFFFF) 12831 { 12832 SM_DBG1(("smsatWriteSame10: case 4 !!! warning can't fit sectors!!!\n")); 12833 smsatSetSensePayload( pSense, 12834 SCSI_SNSKEY_ILLEGAL_REQUEST, 12835 0, 12836 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 12837 satIOContext); 12838 12839 /*smEnqueueIO(smRoot, satIOContext);*/ 12840 12841 tdsmIOCompletedCB( smRoot, 12842 smIORequest, 12843 smIOSuccess, 12844 SCSI_STAT_CHECK_CONDITION, 12845 satIOContext->pSmSenseData, 12846 satIOContext->interruptContext ); 12847 return SM_RC_SUCCESS; 12848 } 12849 } 12850 /* one sector at a time */ 12851 fis->h.features = 1; /* FIS sector count (7:0) */ 12852 fis->d.featuresExp = 0; /* FIS sector count (15:8) */ 12853 12854 12855 fis->d.lbaLow = scsiCmnd->cdb[5]; /* FIS LBA (7 :0 ) */ 12856 fis->d.lbaMid = scsiCmnd->cdb[4]; /* FIS LBA (15:8 ) */ 12857 fis->d.lbaHigh = scsiCmnd->cdb[3]; /* FIS LBA (23:16) */ 12858 12859 /* NO FUA bit in the WRITE SAME 10 */ 12860 fis->d.device = 0x40; /* FIS FUA clear */ 12861 12862 fis->d.lbaLowExp = scsiCmnd->cdb[2]; /* FIS LBA (31:24) */ 12863 fis->d.lbaMidExp = 0; /* FIS LBA (39:32) */ 12864 fis->d.lbaHighExp = 0; /* FIS LBA (47:40) */ 12865 fis->d.sectorCount = 0; /* Tag (7:3) set by LL layer */ 12866 fis->d.sectorCountExp = 0; 12867 fis->d.reserved4 = 0; 12868 fis->d.control = 0; /* FIS HOB bit clear */ 12869 fis->d.reserved5 = 0; 12870 12871 agRequestType = AGSA_SATA_PROTOCOL_FPDMA_WRITE; 12872 } 12873 /* Initialize CB for SATA completion. 12874 */ 12875 satIOContext->satCompleteCB = &smsatWriteSame10CB; 12876 12877 /* 12878 * Prepare SGL and send FIS to LL layer. 12879 */ 12880 satIOContext->reqType = agRequestType; /* Save it */ 12881 12882 status = smsataLLIOStart( smRoot, 12883 smIORequest, 12884 smDeviceHandle, 12885 smScsiRequest, 12886 satIOContext); 12887 return (status); 12888 12889 12890 } /* end of case 1 */ 12891 else if ( !(scsiCmnd->cdb[1] & SCSI_WRITE_SAME_LBDATA_MASK) && 12892 (scsiCmnd->cdb[1] & SCSI_WRITE_SAME_PBDATA_MASK)) 12893 { 12894 /* spec 9.26.2, Table 62, p64, case 2*/ 12895 smsatSetSensePayload( pSense, 12896 SCSI_SNSKEY_ILLEGAL_REQUEST, 12897 0, 12898 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 12899 satIOContext); 12900 12901 /*smEnqueueIO(smRoot, satIOContext);*/ 12902 12903 tdsmIOCompletedCB( smRoot, 12904 smIORequest, 12905 smIOSuccess, 12906 SCSI_STAT_CHECK_CONDITION, 12907 satIOContext->pSmSenseData, 12908 satIOContext->interruptContext ); 12909 12910 SM_DBG5(("smsatWriteSame10: return Table 62 case 2\n")); 12911 return SM_RC_SUCCESS; 12912 } 12913 else if ( (scsiCmnd->cdb[1] & SCSI_WRITE_SAME_LBDATA_MASK) && 12914 !(scsiCmnd->cdb[1] & SCSI_WRITE_SAME_PBDATA_MASK)) 12915 { 12916 SM_DBG5(("smsatWriteSame10: Table 62 case 3\n")); 12917 12918 } 12919 else /* ( (scsiCmnd->cdb[1] & SCSI_WRITE_SAME_LBDATA_MASK) && 12920 (scsiCmnd->cdb[1] & SCSI_WRITE_SAME_PBDATA_MASK)) */ 12921 { 12922 12923 /* spec 9.26.2, Table 62, p64, case 4*/ 12924 smsatSetSensePayload( pSense, 12925 SCSI_SNSKEY_ILLEGAL_REQUEST, 12926 0, 12927 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 12928 satIOContext); 12929 12930 /*smEnqueueIO(smRoot, satIOContext);*/ 12931 12932 tdsmIOCompletedCB( smRoot, 12933 smIORequest, 12934 smIOSuccess, 12935 SCSI_STAT_CHECK_CONDITION, 12936 satIOContext->pSmSenseData, 12937 satIOContext->interruptContext ); 12938 12939 SM_DBG5(("smsatWriteSame10: return Table 62 case 4\n")); 12940 return SM_RC_SUCCESS; 12941 } 12942 12943 12944 return SM_RC_SUCCESS; 12945 } 12946 12947 osGLOBAL bit32 12948 smsatWriteSame16( 12949 smRoot_t *smRoot, 12950 smIORequest_t *smIORequest, 12951 smDeviceHandle_t *smDeviceHandle, 12952 smScsiInitiatorRequest_t *smScsiRequest, 12953 smSatIOContext_t *satIOContext 12954 ) 12955 { 12956 smScsiRspSense_t *pSense; 12957 12958 pSense = satIOContext->pSense; 12959 12960 SM_DBG5(("smsatWriteSame16: start\n")); 12961 12962 12963 smsatSetSensePayload( pSense, 12964 SCSI_SNSKEY_NO_SENSE, 12965 0, 12966 SCSI_SNSCODE_NO_ADDITIONAL_INFO, 12967 satIOContext); 12968 12969 /*smEnqueueIO(smRoot, satIOContext);*/ 12970 12971 tdsmIOCompletedCB( smRoot, 12972 smIORequest, /* == &satIntIo->satOrgSmIORequest */ 12973 smIOSuccess, 12974 SCSI_STAT_CHECK_CONDITION, 12975 satIOContext->pSmSenseData, 12976 satIOContext->interruptContext ); 12977 SM_DBG1(("smsatWriteSame16: return internal checking!!!\n")); 12978 return SM_RC_SUCCESS; 12979 } 12980 12981 osGLOBAL bit32 12982 smsatLogSense( 12983 smRoot_t *smRoot, 12984 smIORequest_t *smIORequest, 12985 smDeviceHandle_t *smDeviceHandle, 12986 smScsiInitiatorRequest_t *smScsiRequest, 12987 smSatIOContext_t *satIOContext 12988 ) 12989 { 12990 bit32 status; 12991 bit32 agRequestType; 12992 smDeviceData_t *pSatDevData; 12993 smScsiRspSense_t *pSense; 12994 smIniScsiCmnd_t *scsiCmnd; 12995 agsaFisRegHostToDevice_t *fis; 12996 bit8 *pLogPage; /* Log Page data buffer */ 12997 bit32 flag = 0; 12998 bit16 AllocLen = 0; /* allocation length */ 12999 bit8 AllLogPages[8]; 13000 bit16 lenRead = 0; 13001 13002 pSense = satIOContext->pSense; 13003 pSatDevData = satIOContext->pSatDevData; 13004 scsiCmnd = &smScsiRequest->scsiCmnd; 13005 fis = satIOContext->pFis; 13006 pLogPage = (bit8 *) smScsiRequest->sglVirtualAddr; 13007 13008 SM_DBG5(("smsatLogSense: start\n")); 13009 13010 sm_memset(&AllLogPages, 0, 8); 13011 /* checking CONTROL */ 13012 /* NACA == 1 or LINK == 1*/ 13013 if ( (scsiCmnd->cdb[9] & SCSI_NACA_MASK) || (scsiCmnd->cdb[9] & SCSI_LINK_MASK) ) 13014 { 13015 smsatSetSensePayload( pSense, 13016 SCSI_SNSKEY_ILLEGAL_REQUEST, 13017 0, 13018 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 13019 satIOContext); 13020 13021 /*smEnqueueIO(smRoot, satIOContext);*/ 13022 13023 tdsmIOCompletedCB( smRoot, 13024 smIORequest, 13025 smIOSuccess, 13026 SCSI_STAT_CHECK_CONDITION, 13027 satIOContext->pSmSenseData, 13028 satIOContext->interruptContext ); 13029 13030 SM_DBG1(("smsatLogSense: return control!!!\n")); 13031 return SM_RC_SUCCESS; 13032 } 13033 13034 13035 AllocLen = ((scsiCmnd->cdb[7] << 8) + scsiCmnd->cdb[8]); 13036 AllocLen = MIN(AllocLen, scsiCmnd->expDataLength); 13037 13038 /* checking PC (Page Control) */ 13039 /* nothing */ 13040 13041 /* special cases */ 13042 if (AllocLen == 4) 13043 { 13044 SM_DBG1(("smsatLogSense: AllocLen is 4!!!\n")); 13045 switch (scsiCmnd->cdb[2] & SCSI_LOG_SENSE_PAGE_CODE_MASK) 13046 { 13047 case LOGSENSE_SUPPORTED_LOG_PAGES: 13048 SM_DBG5(("smsatLogSense: case LOGSENSE_SUPPORTED_LOG_PAGES\n")); 13049 13050 if (pSatDevData->satSMARTFeatureSet == agTRUE) 13051 { 13052 /* add informational exception log */ 13053 flag = 1; 13054 if (pSatDevData->satSMARTSelfTest == agTRUE) 13055 { 13056 /* add Self-Test results log page */ 13057 flag = 2; 13058 } 13059 } 13060 else 13061 { 13062 /* only supported, no informational exception log, no Self-Test results log page */ 13063 flag = 0; 13064 } 13065 lenRead = 4; 13066 AllLogPages[0] = LOGSENSE_SUPPORTED_LOG_PAGES; /* page code */ 13067 AllLogPages[1] = 0; /* reserved */ 13068 switch (flag) 13069 { 13070 case 0: 13071 /* only supported */ 13072 AllLogPages[2] = 0; /* page length */ 13073 AllLogPages[3] = 1; /* page length */ 13074 break; 13075 case 1: 13076 /* supported and informational exception log */ 13077 AllLogPages[2] = 0; /* page length */ 13078 AllLogPages[3] = 2; /* page length */ 13079 break; 13080 case 2: 13081 /* supported and informational exception log */ 13082 AllLogPages[2] = 0; /* page length */ 13083 AllLogPages[3] = 3; /* page length */ 13084 break; 13085 default: 13086 SM_DBG1(("smsatLogSense: error unallowed flag value %d!!!\n", flag)); 13087 break; 13088 } 13089 sm_memcpy(pLogPage, &AllLogPages, lenRead); 13090 break; 13091 case LOGSENSE_SELFTEST_RESULTS_PAGE: 13092 SM_DBG5(("smsatLogSense: case LOGSENSE_SUPPORTED_LOG_PAGES\n")); 13093 lenRead = 4; 13094 AllLogPages[0] = LOGSENSE_SELFTEST_RESULTS_PAGE; /* page code */ 13095 AllLogPages[1] = 0; /* reserved */ 13096 /* page length = SELFTEST_RESULTS_LOG_PAGE_LENGTH - 1 - 3 = 400 = 0x190 */ 13097 AllLogPages[2] = 0x01; 13098 AllLogPages[3] = 0x90; /* page length */ 13099 sm_memcpy(pLogPage, &AllLogPages, lenRead); 13100 13101 break; 13102 case LOGSENSE_INFORMATION_EXCEPTIONS_PAGE: 13103 SM_DBG5(("smsatLogSense: case LOGSENSE_SUPPORTED_LOG_PAGES\n")); 13104 lenRead = 4; 13105 AllLogPages[0] = LOGSENSE_INFORMATION_EXCEPTIONS_PAGE; /* page code */ 13106 AllLogPages[1] = 0; /* reserved */ 13107 AllLogPages[2] = 0; /* page length */ 13108 AllLogPages[3] = INFORMATION_EXCEPTIONS_LOG_PAGE_LENGTH - 1 - 3; /* page length */ 13109 sm_memcpy(pLogPage, &AllLogPages, lenRead); 13110 break; 13111 default: 13112 SM_DBG1(("smsatLogSense: default Page Code 0x%x!!!\n", scsiCmnd->cdb[2] & SCSI_LOG_SENSE_PAGE_CODE_MASK)); 13113 smsatSetSensePayload( pSense, 13114 SCSI_SNSKEY_ILLEGAL_REQUEST, 13115 0, 13116 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 13117 satIOContext); 13118 13119 /*smEnqueueIO(smRoot, satIOContext);*/ 13120 13121 tdsmIOCompletedCB( smRoot, 13122 smIORequest, 13123 smIOSuccess, 13124 SCSI_STAT_CHECK_CONDITION, 13125 satIOContext->pSmSenseData, 13126 satIOContext->interruptContext ); 13127 return SM_RC_SUCCESS; 13128 } 13129 /*smEnqueueIO(smRoot, satIOContext);*/ 13130 13131 tdsmIOCompletedCB( smRoot, 13132 smIORequest, 13133 smIOSuccess, 13134 SCSI_STAT_GOOD, 13135 agNULL, 13136 satIOContext->interruptContext); 13137 return SM_RC_SUCCESS; 13138 13139 } /* if */ 13140 13141 /* SAT rev8 Table 11 p30*/ 13142 /* checking Page Code */ 13143 switch (scsiCmnd->cdb[2] & SCSI_LOG_SENSE_PAGE_CODE_MASK) 13144 { 13145 case LOGSENSE_SUPPORTED_LOG_PAGES: 13146 SM_DBG5(("smsatLogSense: case 1\n")); 13147 13148 if (pSatDevData->satSMARTFeatureSet == agTRUE) 13149 { 13150 /* add informational exception log */ 13151 flag = 1; 13152 if (pSatDevData->satSMARTSelfTest == agTRUE) 13153 { 13154 /* add Self-Test results log page */ 13155 flag = 2; 13156 } 13157 } 13158 else 13159 { 13160 /* only supported, no informational exception log, no Self-Test results log page */ 13161 flag = 0; 13162 } 13163 AllLogPages[0] = 0; /* page code */ 13164 AllLogPages[1] = 0; /* reserved */ 13165 switch (flag) 13166 { 13167 case 0: 13168 /* only supported */ 13169 AllLogPages[2] = 0; /* page length */ 13170 AllLogPages[3] = 1; /* page length */ 13171 AllLogPages[4] = 0x00; /* supported page list */ 13172 lenRead = (bit8)(MIN(AllocLen, 5)); 13173 break; 13174 case 1: 13175 /* supported and informational exception log */ 13176 AllLogPages[2] = 0; /* page length */ 13177 AllLogPages[3] = 2; /* page length */ 13178 AllLogPages[4] = 0x00; /* supported page list */ 13179 AllLogPages[5] = 0x10; /* supported page list */ 13180 lenRead = (bit8)(MIN(AllocLen, 6)); 13181 break; 13182 case 2: 13183 /* supported and informational exception log */ 13184 AllLogPages[2] = 0; /* page length */ 13185 AllLogPages[3] = 3; /* page length */ 13186 AllLogPages[4] = 0x00; /* supported page list */ 13187 AllLogPages[5] = 0x10; /* supported page list */ 13188 AllLogPages[6] = 0x2F; /* supported page list */ 13189 lenRead = (bit8)(MIN(AllocLen, 7)); 13190 break; 13191 default: 13192 SM_DBG1(("smsatLogSense: error unallowed flag value %d!!!\n", flag)); 13193 break; 13194 } 13195 13196 sm_memcpy(pLogPage, &AllLogPages, lenRead); 13197 /* comparing allocation length to Log Page byte size */ 13198 /* SPC-4, 4.3.4.6, p28 */ 13199 if (AllocLen > lenRead ) 13200 { 13201 SM_DBG1(("smsatLogSense: reporting underrun lenRead=0x%x AllocLen=0x%x!!!\n", lenRead, AllocLen)); 13202 /*smEnqueueIO(smRoot, satIOContext);*/ 13203 13204 tdsmIOCompletedCB( smRoot, 13205 smIORequest, 13206 smIOUnderRun, 13207 AllocLen - lenRead, 13208 agNULL, 13209 satIOContext->interruptContext ); 13210 } 13211 else 13212 { 13213 /*smEnqueueIO(smRoot, satIOContext);*/ 13214 tdsmIOCompletedCB( smRoot, 13215 smIORequest, 13216 smIOSuccess, 13217 SCSI_STAT_GOOD, 13218 agNULL, 13219 satIOContext->interruptContext); 13220 } 13221 break; 13222 case LOGSENSE_SELFTEST_RESULTS_PAGE: 13223 SM_DBG5(("smsatLogSense: case 2\n")); 13224 /* checking SMART self-test */ 13225 if (pSatDevData->satSMARTSelfTest == agFALSE) 13226 { 13227 SM_DBG5(("smsatLogSense: case 2 no SMART Self Test\n")); 13228 smsatSetSensePayload( pSense, 13229 SCSI_SNSKEY_ILLEGAL_REQUEST, 13230 0, 13231 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 13232 satIOContext); 13233 13234 /*smEnqueueIO(smRoot, satIOContext);*/ 13235 13236 tdsmIOCompletedCB( smRoot, 13237 smIORequest, 13238 smIOSuccess, 13239 SCSI_STAT_CHECK_CONDITION, 13240 satIOContext->pSmSenseData, 13241 satIOContext->interruptContext ); 13242 } 13243 else 13244 { 13245 /* if satSMARTEnabled is false, send SMART_ENABLE_OPERATIONS */ 13246 if (pSatDevData->satSMARTEnabled == agFALSE) 13247 { 13248 SM_DBG5(("smsatLogSense: case 2 calling satSMARTEnable\n")); 13249 status = smsatLogSenseAllocate(smRoot, 13250 smIORequest, 13251 smDeviceHandle, 13252 smScsiRequest, 13253 satIOContext, 13254 0, 13255 LOG_SENSE_0 13256 ); 13257 13258 return status; 13259 13260 } 13261 else 13262 { 13263 /* SAT Rev 8, 10.2.4 p74 */ 13264 if ( pSatDevData->sat48BitSupport == agTRUE ) 13265 { 13266 SM_DBG5(("smsatLogSense: case 2-1 sends READ LOG EXT\n")); 13267 status = smsatLogSenseAllocate(smRoot, 13268 smIORequest, 13269 smDeviceHandle, 13270 smScsiRequest, 13271 satIOContext, 13272 512, 13273 LOG_SENSE_1 13274 ); 13275 13276 return status; 13277 } 13278 else 13279 { 13280 SM_DBG5(("smsatLogSense: case 2-2 sends SMART READ LOG\n")); 13281 status = smsatLogSenseAllocate(smRoot, 13282 smIORequest, 13283 smDeviceHandle, 13284 smScsiRequest, 13285 satIOContext, 13286 512, 13287 LOG_SENSE_2 13288 ); 13289 13290 return status; 13291 } 13292 } 13293 } 13294 break; 13295 case LOGSENSE_INFORMATION_EXCEPTIONS_PAGE: 13296 SM_DBG5(("smsatLogSense: case 3\n")); 13297 /* checking SMART feature set */ 13298 if (pSatDevData->satSMARTFeatureSet == agFALSE) 13299 { 13300 smsatSetSensePayload( pSense, 13301 SCSI_SNSKEY_ILLEGAL_REQUEST, 13302 0, 13303 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 13304 satIOContext); 13305 13306 /*smEnqueueIO(smRoot, satIOContext);*/ 13307 13308 tdsmIOCompletedCB( smRoot, 13309 smIORequest, 13310 smIOSuccess, 13311 SCSI_STAT_CHECK_CONDITION, 13312 satIOContext->pSmSenseData, 13313 satIOContext->interruptContext ); 13314 } 13315 else 13316 { 13317 /* checking SMART feature enabled */ 13318 if (pSatDevData->satSMARTEnabled == agFALSE) 13319 { 13320 smsatSetSensePayload( pSense, 13321 SCSI_SNSKEY_ABORTED_COMMAND, 13322 0, 13323 SCSI_SNSCODE_ATA_DEVICE_FEATURE_NOT_ENABLED, 13324 satIOContext); 13325 13326 /*smEnqueueIO(smRoot, satIOContext);*/ 13327 13328 tdsmIOCompletedCB( smRoot, 13329 smIORequest, 13330 smIOSuccess, 13331 SCSI_STAT_CHECK_CONDITION, 13332 satIOContext->pSmSenseData, 13333 satIOContext->interruptContext ); 13334 } 13335 else 13336 { 13337 /* SAT Rev 8, 10.2.3 p72 */ 13338 SM_DBG5(("smsatLogSense: case 3 sends SMART RETURN STATUS\n")); 13339 13340 /* sends SMART RETURN STATUS */ 13341 fis->h.fisType = 0x27; /* Reg host to device */ 13342 fis->h.c_pmPort = 0x80; /* C Bit is set */ 13343 13344 fis->h.command = SAT_SMART; /* 0xB0 */ 13345 fis->h.features = SAT_SMART_RETURN_STATUS;/* FIS features */ 13346 fis->d.featuresExp = 0; /* FIS reserve */ 13347 fis->d.sectorCount = 0; /* FIS sector count (7:0) */ 13348 fis->d.sectorCountExp = 0; /* FIS sector count (15:8) */ 13349 fis->d.lbaLow = 0; /* FIS LBA (7 :0 ) */ 13350 fis->d.lbaLowExp = 0; /* FIS LBA (31:24) */ 13351 fis->d.lbaMid = 0x4F; /* FIS LBA (15:8 ) */ 13352 fis->d.lbaMidExp = 0; /* FIS LBA (39:32) */ 13353 fis->d.lbaHigh = 0xC2; /* FIS LBA (23:16) */ 13354 fis->d.lbaHighExp = 0; /* FIS LBA (47:40) */ 13355 fis->d.device = 0; /* FIS DEV is discared in SATA */ 13356 fis->d.control = 0; /* FIS HOB bit clear */ 13357 fis->d.reserved4 = 0; 13358 fis->d.reserved5 = 0; 13359 13360 agRequestType = AGSA_SATA_PROTOCOL_NON_DATA; 13361 /* Initialize CB for SATA completion. 13362 */ 13363 satIOContext->satCompleteCB = &smsatLogSenseCB; 13364 13365 /* 13366 * Prepare SGL and send FIS to LL layer. 13367 */ 13368 satIOContext->reqType = agRequestType; /* Save it */ 13369 13370 status = smsataLLIOStart( smRoot, 13371 smIORequest, 13372 smDeviceHandle, 13373 smScsiRequest, 13374 satIOContext); 13375 13376 13377 return status; 13378 } 13379 } 13380 break; 13381 default: 13382 SM_DBG1(("smsatLogSense: default Page Code 0x%x!!!\n", scsiCmnd->cdb[2] & SCSI_LOG_SENSE_PAGE_CODE_MASK)); 13383 smsatSetSensePayload( pSense, 13384 SCSI_SNSKEY_ILLEGAL_REQUEST, 13385 0, 13386 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 13387 satIOContext); 13388 13389 /*smEnqueueIO(smRoot, satIOContext);*/ 13390 13391 tdsmIOCompletedCB( smRoot, 13392 smIORequest, 13393 smIOSuccess, 13394 SCSI_STAT_CHECK_CONDITION, 13395 satIOContext->pSmSenseData, 13396 satIOContext->interruptContext ); 13397 13398 break; 13399 } /* end switch */ 13400 13401 return SM_RC_SUCCESS; 13402 } 13403 13404 osGLOBAL bit32 13405 smsatLogSenseAllocate( 13406 smRoot_t *smRoot, 13407 smIORequest_t *smIORequest, 13408 smDeviceHandle_t *smDeviceHandle, 13409 smScsiInitiatorRequest_t *smSCSIRequest, 13410 smSatIOContext_t *satIOContext, 13411 bit32 payloadSize, 13412 bit32 flag 13413 ) 13414 { 13415 smDeviceData_t *pSatDevData; 13416 smIORequestBody_t *smIORequestBody; 13417 smSatInternalIo_t *satIntIo = agNULL; 13418 smSatIOContext_t *satIOContext2; 13419 bit32 status; 13420 13421 SM_DBG5(("smsatLogSenseAllocate: start\n")); 13422 13423 pSatDevData = satIOContext->pSatDevData; 13424 13425 /* create internal satIOContext */ 13426 satIntIo = smsatAllocIntIoResource( smRoot, 13427 smIORequest, /* original request */ 13428 pSatDevData, 13429 payloadSize, 13430 satIntIo); 13431 13432 if (satIntIo == agNULL) 13433 { 13434 /*smEnqueueIO(smRoot, satIOContext);*/ 13435 13436 tdsmIOCompletedCB( smRoot, 13437 smIORequest, 13438 smIOFailed, 13439 smDetailOtherError, 13440 agNULL, 13441 satIOContext->interruptContext ); 13442 13443 SM_DBG1(("smsatLogSenseAllocate: fail in allocation!!!\n")); 13444 return SM_RC_SUCCESS; 13445 } /* end of memory allocation failure */ 13446 13447 satIntIo->satOrgSmIORequest = smIORequest; 13448 smIORequestBody = (smIORequestBody_t *)satIntIo->satIntRequestBody; 13449 satIOContext2 = &(smIORequestBody->transport.SATA.satIOContext); 13450 13451 satIOContext2->pSatDevData = pSatDevData; 13452 satIOContext2->pFis = &(smIORequestBody->transport.SATA.agSATARequestBody.fis.fisRegHostToDev); 13453 satIOContext2->pScsiCmnd = &(satIntIo->satIntSmScsiXchg.scsiCmnd); 13454 satIOContext2->pSense = &(smIORequestBody->transport.SATA.sensePayload); 13455 satIOContext2->pSmSenseData = &(smIORequestBody->transport.SATA.smSenseData); 13456 satIOContext2->pSmSenseData->senseData = satIOContext2->pSense; 13457 satIOContext2->smRequestBody = satIntIo->satIntRequestBody; 13458 satIOContext2->interruptContext = satIOContext->interruptContext; 13459 satIOContext2->satIntIoContext = satIntIo; 13460 satIOContext2->psmDeviceHandle = smDeviceHandle; 13461 satIOContext2->satOrgIOContext = satIOContext; 13462 13463 if (flag == LOG_SENSE_0) 13464 { 13465 /* SAT_SMART_ENABLE_OPERATIONS */ 13466 status = smsatSMARTEnable( smRoot, 13467 &(satIntIo->satIntSmIORequest), 13468 smDeviceHandle, 13469 &(satIntIo->satIntSmScsiXchg), 13470 satIOContext2); 13471 } 13472 else if (flag == LOG_SENSE_1) 13473 { 13474 /* SAT_READ_LOG_EXT */ 13475 status = smsatLogSense_2( smRoot, 13476 &(satIntIo->satIntSmIORequest), 13477 smDeviceHandle, 13478 &(satIntIo->satIntSmScsiXchg), 13479 satIOContext2); 13480 } 13481 else 13482 { 13483 /* SAT_SMART_READ_LOG */ 13484 /* SAT_READ_LOG_EXT */ 13485 status = smsatLogSense_3( smRoot, 13486 &(satIntIo->satIntSmIORequest), 13487 smDeviceHandle, 13488 &(satIntIo->satIntSmScsiXchg), 13489 satIOContext2); 13490 13491 } 13492 if (status != SM_RC_SUCCESS) 13493 { 13494 smsatFreeIntIoResource( smRoot, 13495 pSatDevData, 13496 satIntIo); 13497 13498 /*smEnqueueIO(smRoot, satIOContext);*/ 13499 13500 tdsmIOCompletedCB( smRoot, 13501 smIORequest, 13502 smIOFailed, 13503 smDetailOtherError, 13504 agNULL, 13505 satIOContext->interruptContext ); 13506 return SM_RC_SUCCESS; 13507 } 13508 13509 13510 return SM_RC_SUCCESS; 13511 } 13512 13513 osGLOBAL bit32 13514 smsatSMARTEnable( 13515 smRoot_t *smRoot, 13516 smIORequest_t *smIORequest, 13517 smDeviceHandle_t *smDeviceHandle, 13518 smScsiInitiatorRequest_t *smScsiRequest, 13519 smSatIOContext_t *satIOContext 13520 ) 13521 { 13522 bit32 status; 13523 bit32 agRequestType; 13524 agsaFisRegHostToDevice_t *fis; 13525 13526 fis = satIOContext->pFis; 13527 SM_DBG5(("smsatSMARTEnable: start\n")); 13528 /* 13529 * Send the SAT_SMART_ENABLE_OPERATIONS command. 13530 */ 13531 fis->h.fisType = 0x27; /* Reg host to device */ 13532 fis->h.c_pmPort = 0x80; /* C Bit is set */ 13533 fis->h.command = SAT_SMART; /* 0xB0 */ 13534 fis->h.features = SAT_SMART_ENABLE_OPERATIONS; 13535 fis->d.lbaLow = 0; 13536 fis->d.lbaMid = 0x4F; 13537 fis->d.lbaHigh = 0xC2; 13538 fis->d.device = 0; 13539 fis->d.lbaLowExp = 0; 13540 fis->d.lbaMidExp = 0; 13541 fis->d.lbaHighExp = 0; 13542 fis->d.featuresExp = 0; 13543 fis->d.sectorCount = 0; 13544 fis->d.sectorCountExp = 0; 13545 fis->d.reserved4 = 0; 13546 fis->d.control = 0; /* FIS HOB bit clear */ 13547 fis->d.reserved5 = 0; 13548 13549 agRequestType = AGSA_SATA_PROTOCOL_NON_DATA; 13550 13551 /* Initialize CB for SATA completion. 13552 */ 13553 satIOContext->satCompleteCB = &smsatSMARTEnableCB; 13554 13555 /* 13556 * Prepare SGL and send FIS to LL layer. 13557 */ 13558 satIOContext->reqType = agRequestType; /* Save it */ 13559 13560 status = smsataLLIOStart( smRoot, 13561 smIORequest, 13562 smDeviceHandle, 13563 smScsiRequest, 13564 satIOContext); 13565 13566 13567 return status; 13568 } 13569 13570 osGLOBAL bit32 13571 smsatLogSense_2( 13572 smRoot_t *smRoot, 13573 smIORequest_t *smIORequest, 13574 smDeviceHandle_t *smDeviceHandle, 13575 smScsiInitiatorRequest_t *smScsiRequest, 13576 smSatIOContext_t *satIOContext 13577 ) 13578 { 13579 bit32 status; 13580 bit32 agRequestType; 13581 agsaFisRegHostToDevice_t *fis; 13582 13583 fis = satIOContext->pFis; 13584 SM_DBG5(("smsatLogSense_2: start\n")); 13585 13586 /* sends READ LOG EXT */ 13587 fis->h.fisType = 0x27; /* Reg host to device */ 13588 fis->h.c_pmPort = 0x80; /* C Bit is set */ 13589 13590 fis->h.command = SAT_READ_LOG_EXT; /* 0x2F */ 13591 fis->h.features = 0; /* FIS reserve */ 13592 fis->d.lbaLow = 0x07; /* 0x07 */ 13593 fis->d.lbaMid = 0; /* */ 13594 fis->d.lbaHigh = 0; /* */ 13595 fis->d.device = 0; /* */ 13596 fis->d.lbaLowExp = 0; /* */ 13597 fis->d.lbaMidExp = 0; /* */ 13598 fis->d.lbaHighExp = 0; /* */ 13599 fis->d.featuresExp = 0; /* FIS reserve */ 13600 fis->d.sectorCount = 0x01; /* 1 sector counts */ 13601 fis->d.sectorCountExp = 0x00; /* 1 sector counts */ 13602 fis->d.reserved4 = 0; 13603 fis->d.control = 0; /* FIS HOB bit clear */ 13604 fis->d.reserved5 = 0; 13605 13606 agRequestType = AGSA_SATA_PROTOCOL_PIO_READ; 13607 13608 /* Initialize CB for SATA completion. 13609 */ 13610 satIOContext->satCompleteCB = &smsatLogSenseCB; 13611 13612 /* 13613 * Prepare SGL and send FIS to LL layer. 13614 */ 13615 satIOContext->reqType = agRequestType; /* Save it */ 13616 13617 status = smsataLLIOStart( smRoot, 13618 smIORequest, 13619 smDeviceHandle, 13620 smScsiRequest, 13621 satIOContext); 13622 return status; 13623 } 13624 13625 osGLOBAL bit32 13626 smsatLogSense_3( 13627 smRoot_t *smRoot, 13628 smIORequest_t *smIORequest, 13629 smDeviceHandle_t *smDeviceHandle, 13630 smScsiInitiatorRequest_t *smScsiRequest, 13631 smSatIOContext_t *satIOContext 13632 ) 13633 { 13634 bit32 status; 13635 bit32 agRequestType; 13636 agsaFisRegHostToDevice_t *fis; 13637 13638 fis = satIOContext->pFis; 13639 SM_DBG5(("smsatLogSense_3: start\n")); 13640 /* sends READ LOG EXT */ 13641 fis->h.fisType = 0x27; /* Reg host to device */ 13642 fis->h.c_pmPort = 0x80; /* C Bit is set */ 13643 fis->h.command = SAT_SMART; /* 0x2F */ 13644 fis->h.features = SAT_SMART_READ_LOG; /* 0xd5 */ 13645 fis->d.lbaLow = 0x06; /* 0x06 */ 13646 fis->d.lbaMid = 0x4F; /* 0x4f */ 13647 fis->d.lbaHigh = 0xC2; /* 0xc2 */ 13648 fis->d.device = 0; /* */ 13649 fis->d.lbaLowExp = 0; /* */ 13650 fis->d.lbaMidExp = 0; /* */ 13651 fis->d.lbaHighExp = 0; /* */ 13652 fis->d.featuresExp = 0; /* FIS reserve */ 13653 fis->d.sectorCount = 0x01; /* 1 sector counts */ 13654 fis->d.sectorCountExp = 0x00; /* 1 sector counts */ 13655 fis->d.reserved4 = 0; 13656 fis->d.control = 0; /* FIS HOB bit clear */ 13657 fis->d.reserved5 = 0; 13658 agRequestType = AGSA_SATA_PROTOCOL_PIO_READ; 13659 /* Initialize CB for SATA completion. 13660 */ 13661 satIOContext->satCompleteCB = &smsatLogSenseCB; 13662 /* 13663 * Prepare SGL and send FIS to LL layer. 13664 */ 13665 satIOContext->reqType = agRequestType; /* Save it */ 13666 status = smsataLLIOStart( smRoot, 13667 smIORequest, 13668 smDeviceHandle, 13669 smScsiRequest, 13670 satIOContext); 13671 return status; 13672 } 13673 13674 13675 osGLOBAL bit32 13676 smsatModeSelect6( 13677 smRoot_t *smRoot, 13678 smIORequest_t *smIORequest, 13679 smDeviceHandle_t *smDeviceHandle, 13680 smScsiInitiatorRequest_t *smScsiRequest, 13681 smSatIOContext_t *satIOContext 13682 ) 13683 { 13684 bit32 status; 13685 bit32 agRequestType; 13686 smDeviceData_t *pSatDevData; 13687 smScsiRspSense_t *pSense; 13688 smIniScsiCmnd_t *scsiCmnd; 13689 agsaFisRegHostToDevice_t *fis; 13690 bit8 *pLogPage; /* Log Page data buffer */ 13691 bit32 StartingIndex = 0; 13692 bit8 PageCode = 0; 13693 bit32 chkCnd = agFALSE; 13694 bit32 parameterListLen = 0; 13695 13696 pSense = satIOContext->pSense; 13697 pSatDevData = satIOContext->pSatDevData; 13698 scsiCmnd = &smScsiRequest->scsiCmnd; 13699 fis = satIOContext->pFis; 13700 pLogPage = (bit8 *) smScsiRequest->sglVirtualAddr; 13701 13702 SM_DBG5(("smsatModeSelect6: start\n")); 13703 13704 /* checking CONTROL */ 13705 /* NACA == 1 or LINK == 1*/ 13706 if ( (scsiCmnd->cdb[5] & SCSI_NACA_MASK) || (scsiCmnd->cdb[5] & SCSI_LINK_MASK) ) 13707 { 13708 smsatSetSensePayload( pSense, 13709 SCSI_SNSKEY_ILLEGAL_REQUEST, 13710 0, 13711 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 13712 satIOContext); 13713 13714 /*smEnqueueIO(smRoot, satIOContext);*/ 13715 13716 tdsmIOCompletedCB( smRoot, 13717 smIORequest, 13718 smIOSuccess, 13719 SCSI_STAT_CHECK_CONDITION, 13720 satIOContext->pSmSenseData, 13721 satIOContext->interruptContext ); 13722 13723 SM_DBG1(("smsatModeSelect6: return control!!!\n")); 13724 return SM_RC_SUCCESS; 13725 } 13726 13727 /* checking PF bit */ 13728 if ( !(scsiCmnd->cdb[1] & SCSI_MODE_SELECT6_PF_MASK)) 13729 { 13730 smsatSetSensePayload( pSense, 13731 SCSI_SNSKEY_ILLEGAL_REQUEST, 13732 0, 13733 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 13734 satIOContext); 13735 13736 /*smEnqueueIO(smRoot, satIOContext);*/ 13737 13738 tdsmIOCompletedCB( smRoot, 13739 smIORequest, 13740 smIOSuccess, 13741 SCSI_STAT_CHECK_CONDITION, 13742 satIOContext->pSmSenseData, 13743 satIOContext->interruptContext ); 13744 13745 SM_DBG1(("smsatModeSelect6: PF bit check!!!\n")); 13746 return SM_RC_SUCCESS; 13747 } 13748 13749 parameterListLen = scsiCmnd->cdb[4]; 13750 parameterListLen = MIN(parameterListLen, scsiCmnd->expDataLength); 13751 if ((0 == parameterListLen) || (agNULL == pLogPage)) 13752 { 13753 tdsmIOCompletedCB( smRoot, 13754 smIORequest, 13755 smIOSuccess, 13756 SCSI_STAT_GOOD, 13757 agNULL, 13758 satIOContext->interruptContext); 13759 return SM_RC_SUCCESS; 13760 } 13761 13762 /* checking Block Descriptor Length on Mode parameter header(6)*/ 13763 if (pLogPage[3] == 8) 13764 { 13765 /* mode parameter block descriptor exists */ 13766 PageCode = (bit8)(pLogPage[12] & 0x3F); /* page code and index is 4 + 8 */ 13767 StartingIndex = 12; 13768 } 13769 else if (pLogPage[3] == 0) 13770 { 13771 /* mode parameter block descriptor does not exist */ 13772 PageCode = (bit8)(pLogPage[4] & 0x3F); /* page code and index is 4 + 0 */ 13773 StartingIndex = 4; 13774 /*smEnqueueIO(smRoot, satIOContext);*/ 13775 13776 tdsmIOCompletedCB( smRoot, 13777 smIORequest, 13778 smIOSuccess, 13779 SCSI_STAT_GOOD, 13780 agNULL, 13781 satIOContext->interruptContext); 13782 return SM_RC_SUCCESS; 13783 } 13784 else 13785 { 13786 SM_DBG1(("smsatModeSelect6: return mode parameter block descriptor 0x%x!!!\n", pLogPage[3])); 13787 13788 smsatSetSensePayload( pSense, 13789 SCSI_SNSKEY_NO_SENSE, 13790 0, 13791 SCSI_SNSCODE_NO_ADDITIONAL_INFO, 13792 satIOContext); 13793 13794 /*smEnqueueIO(smRoot, satIOContext);*/ 13795 13796 tdsmIOCompletedCB( smRoot, 13797 smIORequest, 13798 smIOSuccess, 13799 SCSI_STAT_CHECK_CONDITION, 13800 satIOContext->pSmSenseData, 13801 satIOContext->interruptContext ); 13802 return SM_RC_SUCCESS; 13803 } 13804 13805 13806 13807 switch (PageCode) /* page code */ 13808 { 13809 case MODESELECT_CONTROL_PAGE: 13810 SM_DBG1(("smsatModeSelect6: Control mode page!!!\n")); 13811 13812 if ( pLogPage[StartingIndex+1] != 0x0A || 13813 pLogPage[StartingIndex+2] != 0x02 || 13814 (pSatDevData->satNCQ == agTRUE && pLogPage[StartingIndex+3] != 0x12) || 13815 (pSatDevData->satNCQ == agFALSE && pLogPage[StartingIndex+3] != 0x02) || 13816 (pLogPage[StartingIndex+4] & BIT3_MASK) != 0x00 || /* SWP bit */ 13817 (pLogPage[StartingIndex+4] & BIT4_MASK) != 0x00 || /* UA_INTLCK_CTRL */ 13818 (pLogPage[StartingIndex+4] & BIT5_MASK) != 0x00 || /* UA_INTLCK_CTRL */ 13819 13820 (pLogPage[StartingIndex+5] & BIT0_MASK) != 0x00 || /* AUTOLOAD MODE */ 13821 (pLogPage[StartingIndex+5] & BIT1_MASK) != 0x00 || /* AUTOLOAD MODE */ 13822 (pLogPage[StartingIndex+5] & BIT2_MASK) != 0x00 || /* AUTOLOAD MODE */ 13823 (pLogPage[StartingIndex+5] & BIT6_MASK) != 0x00 || /* TAS bit */ 13824 13825 pLogPage[StartingIndex+8] != 0xFF || 13826 pLogPage[StartingIndex+9] != 0xFF || 13827 pLogPage[StartingIndex+10] != 0x00 || 13828 pLogPage[StartingIndex+11] != 0x00 13829 ) 13830 { 13831 chkCnd = agTRUE; 13832 } 13833 if (chkCnd == agTRUE) 13834 { 13835 smsatSetSensePayload( pSense, 13836 SCSI_SNSKEY_ILLEGAL_REQUEST, 13837 0, 13838 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 13839 satIOContext); 13840 13841 /*smEnqueueIO(smRoot, satIOContext);*/ 13842 13843 tdsmIOCompletedCB( smRoot, 13844 smIORequest, 13845 smIOSuccess, 13846 SCSI_STAT_CHECK_CONDITION, 13847 satIOContext->pSmSenseData, 13848 satIOContext->interruptContext ); 13849 13850 SM_DBG1(("smsatModeSelect6: unexpected values!!!\n")); 13851 } 13852 else 13853 { 13854 /*smEnqueueIO(smRoot, satIOContext);*/ 13855 13856 tdsmIOCompletedCB( smRoot, 13857 smIORequest, 13858 smIOSuccess, 13859 SCSI_STAT_GOOD, 13860 agNULL, 13861 satIOContext->interruptContext); 13862 } 13863 return SM_RC_SUCCESS; 13864 break; 13865 case MODESELECT_READ_WRITE_ERROR_RECOVERY_PAGE: 13866 SM_DBG1(("smsatModeSelect6: Read-Write Error Recovery mode page!!!\n")); 13867 13868 if ( (pLogPage[StartingIndex + 2] & SCSI_MODE_SELECT6_AWRE_MASK) || 13869 (pLogPage[StartingIndex + 2] & SCSI_MODE_SELECT6_RC_MASK) || 13870 (pLogPage[StartingIndex + 2] & SCSI_MODE_SELECT6_EER_MASK) || 13871 (pLogPage[StartingIndex + 2] & SCSI_MODE_SELECT6_PER_MASK) || 13872 (pLogPage[StartingIndex + 2] & SCSI_MODE_SELECT6_DTE_MASK) || 13873 (pLogPage[StartingIndex + 2] & SCSI_MODE_SELECT6_DCR_MASK) || 13874 (pLogPage[StartingIndex + 10]) || 13875 (pLogPage[StartingIndex + 11]) 13876 ) 13877 { 13878 SM_DBG5(("smsatModeSelect6: return check condition\n")); 13879 13880 smsatSetSensePayload( pSense, 13881 SCSI_SNSKEY_ILLEGAL_REQUEST, 13882 0, 13883 SCSI_SNSCODE_INVALID_FIELD_PARAMETER_LIST, 13884 satIOContext); 13885 13886 /*smEnqueueIO(smRoot, satIOContext);*/ 13887 13888 tdsmIOCompletedCB( smRoot, 13889 smIORequest, 13890 smIOSuccess, 13891 SCSI_STAT_CHECK_CONDITION, 13892 satIOContext->pSmSenseData, 13893 satIOContext->interruptContext ); 13894 return SM_RC_SUCCESS; 13895 } 13896 else 13897 { 13898 SM_DBG5(("smsatModeSelect6: return GOOD \n")); 13899 /*smEnqueueIO(smRoot, satIOContext);*/ 13900 13901 tdsmIOCompletedCB( smRoot, 13902 smIORequest, 13903 smIOSuccess, 13904 SCSI_STAT_GOOD, 13905 agNULL, 13906 satIOContext->interruptContext); 13907 return SM_RC_SUCCESS; 13908 } 13909 13910 break; 13911 case MODESELECT_CACHING: 13912 /* SAT rev8 Table67, p69*/ 13913 SM_DBG5(("smsatModeSelect6: Caching mode page\n")); 13914 if ( (pLogPage[StartingIndex + 2] & 0xFB) || /* 1111 1011 */ 13915 (pLogPage[StartingIndex + 3]) || 13916 (pLogPage[StartingIndex + 4]) || 13917 (pLogPage[StartingIndex + 5]) || 13918 (pLogPage[StartingIndex + 6]) || 13919 (pLogPage[StartingIndex + 7]) || 13920 (pLogPage[StartingIndex + 8]) || 13921 (pLogPage[StartingIndex + 9]) || 13922 (pLogPage[StartingIndex + 10]) || 13923 (pLogPage[StartingIndex + 11]) || 13924 13925 (pLogPage[StartingIndex + 12] & 0xC1) || /* 1100 0001 */ 13926 (pLogPage[StartingIndex + 13]) || 13927 (pLogPage[StartingIndex + 14]) || 13928 (pLogPage[StartingIndex + 15]) 13929 ) 13930 { 13931 SM_DBG1(("smsatModeSelect6: return check condition!!!\n")); 13932 13933 smsatSetSensePayload( pSense, 13934 SCSI_SNSKEY_ILLEGAL_REQUEST, 13935 0, 13936 SCSI_SNSCODE_INVALID_FIELD_PARAMETER_LIST, 13937 satIOContext); 13938 13939 /*smEnqueueIO(smRoot, satIOContext);*/ 13940 13941 tdsmIOCompletedCB( smRoot, 13942 smIORequest, 13943 smIOSuccess, 13944 SCSI_STAT_CHECK_CONDITION, 13945 satIOContext->pSmSenseData, 13946 satIOContext->interruptContext ); 13947 return SM_RC_SUCCESS; 13948 13949 } 13950 else 13951 { 13952 /* sends ATA SET FEATURES based on WCE bit */ 13953 if ( !(pLogPage[StartingIndex + 2] & SCSI_MODE_SELECT6_WCE_MASK) ) 13954 { 13955 SM_DBG5(("smsatModeSelect6: disable write cache\n")); 13956 /* sends SET FEATURES */ 13957 fis->h.fisType = 0x27; /* Reg host to device */ 13958 fis->h.c_pmPort = 0x80; /* C Bit is set */ 13959 13960 fis->h.command = SAT_SET_FEATURES; /* 0xEF */ 13961 fis->h.features = 0x82; /* disable write cache */ 13962 fis->d.lbaLow = 0; /* */ 13963 fis->d.lbaMid = 0; /* */ 13964 fis->d.lbaHigh = 0; /* */ 13965 fis->d.device = 0; /* */ 13966 fis->d.lbaLowExp = 0; /* */ 13967 fis->d.lbaMidExp = 0; /* */ 13968 fis->d.lbaHighExp = 0; /* */ 13969 fis->d.featuresExp = 0; /* */ 13970 fis->d.sectorCount = 0; /* */ 13971 fis->d.sectorCountExp = 0; /* */ 13972 fis->d.reserved4 = 0; 13973 fis->d.control = 0; /* FIS HOB bit clear */ 13974 fis->d.reserved5 = 0; 13975 13976 agRequestType = AGSA_SATA_PROTOCOL_NON_DATA; 13977 13978 /* Initialize CB for SATA completion. 13979 */ 13980 satIOContext->satCompleteCB = &smsatModeSelect6n10CB; 13981 13982 /* 13983 * Prepare SGL and send FIS to LL layer. 13984 */ 13985 satIOContext->reqType = agRequestType; /* Save it */ 13986 13987 status = smsataLLIOStart( smRoot, 13988 smIORequest, 13989 smDeviceHandle, 13990 smScsiRequest, 13991 satIOContext); 13992 return status; 13993 } 13994 else 13995 { 13996 SM_DBG5(("smsatModeSelect6: enable write cache\n")); 13997 /* sends SET FEATURES */ 13998 fis->h.fisType = 0x27; /* Reg host to device */ 13999 fis->h.c_pmPort = 0x80; /* C Bit is set */ 14000 14001 fis->h.command = SAT_SET_FEATURES; /* 0xEF */ 14002 fis->h.features = 0x02; /* enable write cache */ 14003 fis->d.lbaLow = 0; /* */ 14004 fis->d.lbaMid = 0; /* */ 14005 fis->d.lbaHigh = 0; /* */ 14006 fis->d.device = 0; /* */ 14007 fis->d.lbaLowExp = 0; /* */ 14008 fis->d.lbaMidExp = 0; /* */ 14009 fis->d.lbaHighExp = 0; /* */ 14010 fis->d.featuresExp = 0; /* */ 14011 fis->d.sectorCount = 0; /* */ 14012 fis->d.sectorCountExp = 0; /* */ 14013 fis->d.reserved4 = 0; 14014 fis->d.control = 0; /* FIS HOB bit clear */ 14015 fis->d.reserved5 = 0; 14016 14017 agRequestType = AGSA_SATA_PROTOCOL_NON_DATA; 14018 14019 /* Initialize CB for SATA completion. 14020 */ 14021 satIOContext->satCompleteCB = &smsatModeSelect6n10CB; 14022 14023 /* 14024 * Prepare SGL and send FIS to LL layer. 14025 */ 14026 satIOContext->reqType = agRequestType; /* Save it */ 14027 14028 status = smsataLLIOStart( smRoot, 14029 smIORequest, 14030 smDeviceHandle, 14031 smScsiRequest, 14032 satIOContext); 14033 return status; 14034 14035 } 14036 } 14037 break; 14038 case MODESELECT_INFORMATION_EXCEPTION_CONTROL_PAGE: 14039 SM_DBG5(("smsatModeSelect6: Informational Exception Control mode page\n")); 14040 14041 if ( (pLogPage[StartingIndex + 2] & SCSI_MODE_SELECT6_PERF_MASK) || 14042 (pLogPage[StartingIndex + 2] & SCSI_MODE_SELECT6_TEST_MASK) 14043 ) 14044 { 14045 SM_DBG1(("smsatModeSelect6: return check condition!!! \n")); 14046 14047 smsatSetSensePayload( pSense, 14048 SCSI_SNSKEY_ILLEGAL_REQUEST, 14049 0, 14050 SCSI_SNSCODE_INVALID_FIELD_PARAMETER_LIST, 14051 satIOContext); 14052 14053 /*smEnqueueIO(smRoot, satIOContext);*/ 14054 14055 tdsmIOCompletedCB( smRoot, 14056 smIORequest, 14057 smIOSuccess, 14058 SCSI_STAT_CHECK_CONDITION, 14059 satIOContext->pSmSenseData, 14060 satIOContext->interruptContext ); 14061 return SM_RC_SUCCESS; 14062 } 14063 else 14064 { 14065 /* sends either ATA SMART ENABLE/DISABLE OPERATIONS based on DEXCPT bit */ 14066 if ( !(pLogPage[StartingIndex + 2] & 0x08) ) 14067 { 14068 SM_DBG5(("smsatModeSelect6: enable information exceptions reporting\n")); 14069 /* sends SMART ENABLE OPERATIONS */ 14070 fis->h.fisType = 0x27; /* Reg host to device */ 14071 fis->h.c_pmPort = 0x80; /* C Bit is set */ 14072 14073 fis->h.command = SAT_SMART; /* 0xB0 */ 14074 fis->h.features = SAT_SMART_ENABLE_OPERATIONS; /* enable */ 14075 fis->d.lbaLow = 0; /* */ 14076 fis->d.lbaMid = 0x4F; /* 0x4F */ 14077 fis->d.lbaHigh = 0xC2; /* 0xC2 */ 14078 fis->d.device = 0; /* */ 14079 fis->d.lbaLowExp = 0; /* */ 14080 fis->d.lbaMidExp = 0; /* */ 14081 fis->d.lbaHighExp = 0; /* */ 14082 fis->d.featuresExp = 0; /* */ 14083 fis->d.sectorCount = 0; /* */ 14084 fis->d.sectorCountExp = 0; /* */ 14085 fis->d.reserved4 = 0; 14086 fis->d.control = 0; /* FIS HOB bit clear */ 14087 fis->d.reserved5 = 0; 14088 14089 agRequestType = AGSA_SATA_PROTOCOL_NON_DATA; 14090 14091 /* Initialize CB for SATA completion. 14092 */ 14093 satIOContext->satCompleteCB = &smsatModeSelect6n10CB; 14094 14095 /* 14096 * Prepare SGL and send FIS to LL layer. 14097 */ 14098 satIOContext->reqType = agRequestType; /* Save it */ 14099 14100 status = smsataLLIOStart( smRoot, 14101 smIORequest, 14102 smDeviceHandle, 14103 smScsiRequest, 14104 satIOContext); 14105 return status; 14106 } 14107 else 14108 { 14109 SM_DBG5(("smsatModeSelect6: disable information exceptions reporting\n")); 14110 /* sends SMART DISABLE OPERATIONS */ 14111 fis->h.fisType = 0x27; /* Reg host to device */ 14112 fis->h.c_pmPort = 0x80; /* C Bit is set */ 14113 14114 fis->h.command = SAT_SMART; /* 0xB0 */ 14115 fis->h.features = SAT_SMART_DISABLE_OPERATIONS; /* disable */ 14116 fis->d.lbaLow = 0; /* */ 14117 fis->d.lbaMid = 0x4F; /* 0x4F */ 14118 fis->d.lbaHigh = 0xC2; /* 0xC2 */ 14119 fis->d.device = 0; /* */ 14120 fis->d.lbaLowExp = 0; /* */ 14121 fis->d.lbaMidExp = 0; /* */ 14122 fis->d.lbaHighExp = 0; /* */ 14123 fis->d.featuresExp = 0; /* */ 14124 fis->d.sectorCount = 0; /* */ 14125 fis->d.sectorCountExp = 0; /* */ 14126 fis->d.reserved4 = 0; 14127 fis->d.control = 0; /* FIS HOB bit clear */ 14128 fis->d.reserved5 = 0; 14129 14130 agRequestType = AGSA_SATA_PROTOCOL_NON_DATA; 14131 14132 /* Initialize CB for SATA completion. 14133 */ 14134 satIOContext->satCompleteCB = &smsatModeSelect6n10CB; 14135 14136 /* 14137 * Prepare SGL and send FIS to LL layer. 14138 */ 14139 satIOContext->reqType = agRequestType; /* Save it */ 14140 14141 status = smsataLLIOStart( smRoot, 14142 smIORequest, 14143 smDeviceHandle, 14144 smScsiRequest, 14145 satIOContext); 14146 return status; 14147 14148 } 14149 } 14150 break; 14151 default: 14152 SM_DBG1(("smsatModeSelect6: Error unknown page code 0x%x!!!\n", pLogPage[12])); 14153 smsatSetSensePayload( pSense, 14154 SCSI_SNSKEY_NO_SENSE, 14155 0, 14156 SCSI_SNSCODE_NO_ADDITIONAL_INFO, 14157 satIOContext); 14158 14159 /*smEnqueueIO(smRoot, satIOContext);*/ 14160 14161 tdsmIOCompletedCB( smRoot, 14162 smIORequest, 14163 smIOSuccess, 14164 SCSI_STAT_CHECK_CONDITION, 14165 satIOContext->pSmSenseData, 14166 satIOContext->interruptContext ); 14167 return SM_RC_SUCCESS; 14168 } 14169 } 14170 14171 14172 osGLOBAL bit32 14173 smsatModeSelect10( 14174 smRoot_t *smRoot, 14175 smIORequest_t *smIORequest, 14176 smDeviceHandle_t *smDeviceHandle, 14177 smScsiInitiatorRequest_t *smScsiRequest, 14178 smSatIOContext_t *satIOContext 14179 ) 14180 { 14181 bit32 status; 14182 bit32 agRequestType; 14183 smDeviceData_t *pSatDevData; 14184 smScsiRspSense_t *pSense; 14185 smIniScsiCmnd_t *scsiCmnd; 14186 agsaFisRegHostToDevice_t *fis; 14187 bit8 *pLogPage; /* Log Page data buffer */ 14188 bit16 BlkDescLen = 0; /* Block Descriptor Length */ 14189 bit32 StartingIndex = 0; 14190 bit8 PageCode = 0; 14191 bit32 chkCnd = agFALSE; 14192 bit32 parameterListLen = 0; 14193 14194 pSense = satIOContext->pSense; 14195 pSatDevData = satIOContext->pSatDevData; 14196 scsiCmnd = &smScsiRequest->scsiCmnd; 14197 fis = satIOContext->pFis; 14198 pLogPage = (bit8 *) smScsiRequest->sglVirtualAddr; 14199 14200 SM_DBG5(("smsatModeSelect10: start\n")); 14201 14202 /* checking CONTROL */ 14203 /* NACA == 1 or LINK == 1*/ 14204 if ( (scsiCmnd->cdb[9] & SCSI_NACA_MASK) || (scsiCmnd->cdb[9] & SCSI_LINK_MASK) ) 14205 { 14206 smsatSetSensePayload( pSense, 14207 SCSI_SNSKEY_ILLEGAL_REQUEST, 14208 0, 14209 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 14210 satIOContext); 14211 14212 /*smEnqueueIO(smRoot, satIOContext);*/ 14213 14214 tdsmIOCompletedCB( smRoot, 14215 smIORequest, 14216 smIOSuccess, 14217 SCSI_STAT_CHECK_CONDITION, 14218 satIOContext->pSmSenseData, 14219 satIOContext->interruptContext ); 14220 14221 SM_DBG1(("smsatModeSelect10: return control!!!\n")); 14222 return SM_RC_SUCCESS; 14223 } 14224 14225 /* checking PF bit */ 14226 if ( !(scsiCmnd->cdb[1] & SCSI_MODE_SELECT10_PF_MASK)) 14227 { 14228 smsatSetSensePayload( pSense, 14229 SCSI_SNSKEY_ILLEGAL_REQUEST, 14230 0, 14231 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 14232 satIOContext); 14233 14234 /*smEnqueueIO(smRoot, satIOContext);*/ 14235 14236 tdsmIOCompletedCB( smRoot, 14237 smIORequest, 14238 smIOSuccess, 14239 SCSI_STAT_CHECK_CONDITION, 14240 satIOContext->pSmSenseData, 14241 satIOContext->interruptContext ); 14242 14243 SM_DBG1(("smsatModeSelect10: PF bit check!!!\n")); 14244 return SM_RC_SUCCESS; 14245 } 14246 14247 parameterListLen = ((scsiCmnd->cdb[7]) << 8) + scsiCmnd->cdb[8]; 14248 parameterListLen = MIN(parameterListLen, scsiCmnd->expDataLength); 14249 if ((0 == parameterListLen) || (agNULL == pLogPage)) 14250 { 14251 tdsmIOCompletedCB( smRoot, 14252 smIORequest, 14253 smIOSuccess, 14254 SCSI_STAT_GOOD, 14255 agNULL, 14256 satIOContext->interruptContext); 14257 return SM_RC_SUCCESS; 14258 } 14259 14260 BlkDescLen = (bit8)((pLogPage[6] << 8) + pLogPage[7]); 14261 14262 /* checking Block Descriptor Length on Mode parameter header(10) and LONGLBA bit*/ 14263 if ( (BlkDescLen == 8) && !(pLogPage[4] & SCSI_MODE_SELECT10_LONGLBA_MASK) ) 14264 { 14265 /* mode parameter block descriptor exists and length is 8 byte */ 14266 PageCode = (bit8)(pLogPage[16] & 0x3F); /* page code and index is 8 + 8 */ 14267 StartingIndex = 16; 14268 } 14269 else if ( (BlkDescLen == 16) && (pLogPage[4] & SCSI_MODE_SELECT10_LONGLBA_MASK) ) 14270 { 14271 /* mode parameter block descriptor exists and length is 16 byte */ 14272 PageCode = (bit8)(pLogPage[24] & 0x3F); /* page code and index is 8 + 16 */ 14273 StartingIndex = 24; 14274 } 14275 else if (BlkDescLen == 0) 14276 { 14277 PageCode = (bit8)(pLogPage[8] & 0x3F); /* page code and index is 8 + 0 */ 14278 StartingIndex = 8; 14279 /*smEnqueueIO(smRoot, satIOContext);*/ 14280 14281 tdsmIOCompletedCB( smRoot, 14282 smIORequest, 14283 smIOSuccess, 14284 SCSI_STAT_GOOD, 14285 agNULL, 14286 satIOContext->interruptContext); 14287 return SM_RC_SUCCESS; 14288 } 14289 else 14290 { 14291 SM_DBG1(("smsatModeSelect10: return mode parameter block descriptor 0x%x!!!\n", BlkDescLen)); 14292 /* no more than one mode parameter block descriptor shall be supported */ 14293 smsatSetSensePayload( pSense, 14294 SCSI_SNSKEY_NO_SENSE, 14295 0, 14296 SCSI_SNSCODE_NO_ADDITIONAL_INFO, 14297 satIOContext); 14298 14299 /*smEnqueueIO(smRoot, satIOContext);*/ 14300 14301 tdsmIOCompletedCB( smRoot, 14302 smIORequest, 14303 smIOSuccess, 14304 SCSI_STAT_CHECK_CONDITION, 14305 satIOContext->pSmSenseData, 14306 satIOContext->interruptContext ); 14307 return SM_RC_SUCCESS; 14308 } 14309 /* 14310 for debugging only 14311 */ 14312 if (StartingIndex == 8) 14313 { 14314 smhexdump("startingindex 8", (bit8 *)pLogPage, 8); 14315 } 14316 else if(StartingIndex == 16) 14317 { 14318 if (PageCode == MODESELECT_CACHING) 14319 { 14320 smhexdump("startingindex 16", (bit8 *)pLogPage, 16+20); 14321 } 14322 else 14323 { 14324 smhexdump("startingindex 16", (bit8 *)pLogPage, 16+12); 14325 } 14326 } 14327 else 14328 { 14329 if (PageCode == MODESELECT_CACHING) 14330 { 14331 smhexdump("startingindex 24", (bit8 *)pLogPage, 24+20); 14332 } 14333 else 14334 { 14335 smhexdump("startingindex 24", (bit8 *)pLogPage, 24+12); 14336 } 14337 } 14338 switch (PageCode) /* page code */ 14339 { 14340 case MODESELECT_CONTROL_PAGE: 14341 SM_DBG5(("smsatModeSelect10: Control mode page\n")); 14342 /* 14343 compare pLogPage to expected value (SAT Table 65, p67) 14344 If not match, return check condition 14345 */ 14346 if ( pLogPage[StartingIndex+1] != 0x0A || 14347 pLogPage[StartingIndex+2] != 0x02 || 14348 (pSatDevData->satNCQ == agTRUE && pLogPage[StartingIndex+3] != 0x12) || 14349 (pSatDevData->satNCQ == agFALSE && pLogPage[StartingIndex+3] != 0x02) || 14350 (pLogPage[StartingIndex+4] & BIT3_MASK) != 0x00 || /* SWP bit */ 14351 (pLogPage[StartingIndex+4] & BIT4_MASK) != 0x00 || /* UA_INTLCK_CTRL */ 14352 (pLogPage[StartingIndex+4] & BIT5_MASK) != 0x00 || /* UA_INTLCK_CTRL */ 14353 14354 (pLogPage[StartingIndex+5] & BIT0_MASK) != 0x00 || /* AUTOLOAD MODE */ 14355 (pLogPage[StartingIndex+5] & BIT1_MASK) != 0x00 || /* AUTOLOAD MODE */ 14356 (pLogPage[StartingIndex+5] & BIT2_MASK) != 0x00 || /* AUTOLOAD MODE */ 14357 (pLogPage[StartingIndex+5] & BIT6_MASK) != 0x00 || /* TAS bit */ 14358 14359 pLogPage[StartingIndex+8] != 0xFF || 14360 pLogPage[StartingIndex+9] != 0xFF || 14361 pLogPage[StartingIndex+10] != 0x00 || 14362 pLogPage[StartingIndex+11] != 0x00 14363 ) 14364 { 14365 chkCnd = agTRUE; 14366 } 14367 if (chkCnd == agTRUE) 14368 { 14369 smsatSetSensePayload( pSense, 14370 SCSI_SNSKEY_ILLEGAL_REQUEST, 14371 0, 14372 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 14373 satIOContext); 14374 14375 /*smEnqueueIO(smRoot, satIOContext);*/ 14376 14377 tdsmIOCompletedCB( smRoot, 14378 smIORequest, 14379 smIOSuccess, 14380 SCSI_STAT_CHECK_CONDITION, 14381 satIOContext->pSmSenseData, 14382 satIOContext->interruptContext ); 14383 14384 SM_DBG1(("smsatModeSelect10: unexpected values!!!\n")); 14385 } 14386 else 14387 { 14388 /*smEnqueueIO(smRoot, satIOContext);*/ 14389 14390 tdsmIOCompletedCB( smRoot, 14391 smIORequest, 14392 smIOSuccess, 14393 SCSI_STAT_GOOD, 14394 agNULL, 14395 satIOContext->interruptContext); 14396 } 14397 return SM_RC_SUCCESS; 14398 break; 14399 case MODESELECT_READ_WRITE_ERROR_RECOVERY_PAGE: 14400 SM_DBG5(("smsatModeSelect10: Read-Write Error Recovery mode page\n")); 14401 14402 if ( (pLogPage[StartingIndex + 2] & SCSI_MODE_SELECT10_AWRE_MASK) || 14403 (pLogPage[StartingIndex + 2] & SCSI_MODE_SELECT10_RC_MASK) || 14404 (pLogPage[StartingIndex + 2] & SCSI_MODE_SELECT10_EER_MASK) || 14405 (pLogPage[StartingIndex + 2] & SCSI_MODE_SELECT10_PER_MASK) || 14406 (pLogPage[StartingIndex + 2] & SCSI_MODE_SELECT10_DTE_MASK) || 14407 (pLogPage[StartingIndex + 2] & SCSI_MODE_SELECT10_DCR_MASK) || 14408 (pLogPage[StartingIndex + 10]) || 14409 (pLogPage[StartingIndex + 11]) 14410 ) 14411 { 14412 SM_DBG1(("smsatModeSelect10: return check condition!!!\n")); 14413 14414 smsatSetSensePayload( pSense, 14415 SCSI_SNSKEY_ILLEGAL_REQUEST, 14416 0, 14417 SCSI_SNSCODE_INVALID_FIELD_PARAMETER_LIST, 14418 satIOContext); 14419 14420 /*smEnqueueIO(smRoot, satIOContext);*/ 14421 14422 tdsmIOCompletedCB( smRoot, 14423 smIORequest, 14424 smIOSuccess, 14425 SCSI_STAT_CHECK_CONDITION, 14426 satIOContext->pSmSenseData, 14427 satIOContext->interruptContext ); 14428 return SM_RC_SUCCESS; 14429 } 14430 else 14431 { 14432 SM_DBG2(("smsatModeSelect10: return GOOD \n")); 14433 /*smEnqueueIO(smRoot, satIOContext);*/ 14434 14435 tdsmIOCompletedCB( smRoot, 14436 smIORequest, 14437 smIOSuccess, 14438 SCSI_STAT_GOOD, 14439 agNULL, 14440 satIOContext->interruptContext); 14441 return SM_RC_SUCCESS; 14442 } 14443 14444 break; 14445 case MODESELECT_CACHING: 14446 /* SAT rev8 Table67, p69*/ 14447 SM_DBG5(("smsatModeSelect10: Caching mode page\n")); 14448 if ( (pLogPage[StartingIndex + 2] & 0xFB) || /* 1111 1011 */ 14449 (pLogPage[StartingIndex + 3]) || 14450 (pLogPage[StartingIndex + 4]) || 14451 (pLogPage[StartingIndex + 5]) || 14452 (pLogPage[StartingIndex + 6]) || 14453 (pLogPage[StartingIndex + 7]) || 14454 (pLogPage[StartingIndex + 8]) || 14455 (pLogPage[StartingIndex + 9]) || 14456 (pLogPage[StartingIndex + 10]) || 14457 (pLogPage[StartingIndex + 11]) || 14458 14459 (pLogPage[StartingIndex + 12] & 0xC1) || /* 1100 0001 */ 14460 (pLogPage[StartingIndex + 13]) || 14461 (pLogPage[StartingIndex + 14]) || 14462 (pLogPage[StartingIndex + 15]) 14463 ) 14464 { 14465 SM_DBG1(("smsatModeSelect10: return check condition!!!\n")); 14466 14467 smsatSetSensePayload( pSense, 14468 SCSI_SNSKEY_ILLEGAL_REQUEST, 14469 0, 14470 SCSI_SNSCODE_INVALID_FIELD_PARAMETER_LIST, 14471 satIOContext); 14472 14473 /*smEnqueueIO(smRoot, satIOContext);*/ 14474 14475 tdsmIOCompletedCB( smRoot, 14476 smIORequest, 14477 smIOSuccess, 14478 SCSI_STAT_CHECK_CONDITION, 14479 satIOContext->pSmSenseData, 14480 satIOContext->interruptContext ); 14481 return SM_RC_SUCCESS; 14482 14483 } 14484 else 14485 { 14486 /* sends ATA SET FEATURES based on WCE bit */ 14487 if ( !(pLogPage[StartingIndex + 2] & SCSI_MODE_SELECT10_WCE_MASK) ) 14488 { 14489 SM_DBG5(("smsatModeSelect10: disable write cache\n")); 14490 /* sends SET FEATURES */ 14491 fis->h.fisType = 0x27; /* Reg host to device */ 14492 fis->h.c_pmPort = 0x80; /* C Bit is set */ 14493 14494 fis->h.command = SAT_SET_FEATURES; /* 0xEF */ 14495 fis->h.features = 0x82; /* disable write cache */ 14496 fis->d.lbaLow = 0; /* */ 14497 fis->d.lbaMid = 0; /* */ 14498 fis->d.lbaHigh = 0; /* */ 14499 fis->d.device = 0; /* */ 14500 fis->d.lbaLowExp = 0; /* */ 14501 fis->d.lbaMidExp = 0; /* */ 14502 fis->d.lbaHighExp = 0; /* */ 14503 fis->d.featuresExp = 0; /* */ 14504 fis->d.sectorCount = 0; /* */ 14505 fis->d.sectorCountExp = 0; /* */ 14506 fis->d.reserved4 = 0; 14507 fis->d.control = 0; /* FIS HOB bit clear */ 14508 fis->d.reserved5 = 0; 14509 14510 agRequestType = AGSA_SATA_PROTOCOL_NON_DATA; 14511 14512 /* Initialize CB for SATA completion. 14513 */ 14514 satIOContext->satCompleteCB = &smsatModeSelect6n10CB; 14515 14516 /* 14517 * Prepare SGL and send FIS to LL layer. 14518 */ 14519 satIOContext->reqType = agRequestType; /* Save it */ 14520 14521 status = smsataLLIOStart( smRoot, 14522 smIORequest, 14523 smDeviceHandle, 14524 smScsiRequest, 14525 satIOContext); 14526 return status; 14527 } 14528 else 14529 { 14530 SM_DBG5(("smsatModeSelect10: enable write cache\n")); 14531 /* sends SET FEATURES */ 14532 fis->h.fisType = 0x27; /* Reg host to device */ 14533 fis->h.c_pmPort = 0x80; /* C Bit is set */ 14534 14535 fis->h.command = SAT_SET_FEATURES; /* 0xEF */ 14536 fis->h.features = 0x02; /* enable write cache */ 14537 fis->d.lbaLow = 0; /* */ 14538 fis->d.lbaMid = 0; /* */ 14539 fis->d.lbaHigh = 0; /* */ 14540 fis->d.device = 0; /* */ 14541 fis->d.lbaLowExp = 0; /* */ 14542 fis->d.lbaMidExp = 0; /* */ 14543 fis->d.lbaHighExp = 0; /* */ 14544 fis->d.featuresExp = 0; /* */ 14545 fis->d.sectorCount = 0; /* */ 14546 fis->d.sectorCountExp = 0; /* */ 14547 fis->d.reserved4 = 0; 14548 fis->d.control = 0; /* FIS HOB bit clear */ 14549 fis->d.reserved5 = 0; 14550 14551 agRequestType = AGSA_SATA_PROTOCOL_NON_DATA; 14552 14553 /* Initialize CB for SATA completion. 14554 */ 14555 satIOContext->satCompleteCB = &smsatModeSelect6n10CB; 14556 14557 /* 14558 * Prepare SGL and send FIS to LL layer. 14559 */ 14560 satIOContext->reqType = agRequestType; /* Save it */ 14561 14562 status = smsataLLIOStart( smRoot, 14563 smIORequest, 14564 smDeviceHandle, 14565 smScsiRequest, 14566 satIOContext); 14567 return status; 14568 14569 } 14570 } 14571 break; 14572 case MODESELECT_INFORMATION_EXCEPTION_CONTROL_PAGE: 14573 SM_DBG5(("smsatModeSelect10: Informational Exception Control mode page\n")); 14574 14575 if ( (pLogPage[StartingIndex + 2] & SCSI_MODE_SELECT10_PERF_MASK) || 14576 (pLogPage[StartingIndex + 2] & SCSI_MODE_SELECT10_TEST_MASK) 14577 ) 14578 { 14579 SM_DBG1(("smsatModeSelect10: return check condition!!!\n")); 14580 14581 smsatSetSensePayload( pSense, 14582 SCSI_SNSKEY_ILLEGAL_REQUEST, 14583 0, 14584 SCSI_SNSCODE_INVALID_FIELD_PARAMETER_LIST, 14585 satIOContext); 14586 14587 /*smEnqueueIO(smRoot, satIOContext);*/ 14588 14589 tdsmIOCompletedCB( smRoot, 14590 smIORequest, 14591 smIOSuccess, 14592 SCSI_STAT_CHECK_CONDITION, 14593 satIOContext->pSmSenseData, 14594 satIOContext->interruptContext ); 14595 return SM_RC_SUCCESS; 14596 } 14597 else 14598 { 14599 /* sends either ATA SMART ENABLE/DISABLE OPERATIONS based on DEXCPT bit */ 14600 if ( !(pLogPage[StartingIndex + 2] & SCSI_MODE_SELECT10_DEXCPT_MASK) ) 14601 { 14602 SM_DBG5(("smsatModeSelect10: enable information exceptions reporting\n")); 14603 /* sends SMART ENABLE OPERATIONS */ 14604 fis->h.fisType = 0x27; /* Reg host to device */ 14605 fis->h.c_pmPort = 0x80; /* C Bit is set */ 14606 14607 fis->h.command = SAT_SMART; /* 0xB0 */ 14608 fis->h.features = SAT_SMART_ENABLE_OPERATIONS; /* enable */ 14609 fis->d.lbaLow = 0; /* */ 14610 fis->d.lbaMid = 0x4F; /* 0x4F */ 14611 fis->d.lbaHigh = 0xC2; /* 0xC2 */ 14612 fis->d.device = 0; /* */ 14613 fis->d.lbaLowExp = 0; /* */ 14614 fis->d.lbaMidExp = 0; /* */ 14615 fis->d.lbaHighExp = 0; /* */ 14616 fis->d.featuresExp = 0; /* */ 14617 fis->d.sectorCount = 0; /* */ 14618 fis->d.sectorCountExp = 0; /* */ 14619 fis->d.reserved4 = 0; 14620 fis->d.control = 0; /* FIS HOB bit clear */ 14621 fis->d.reserved5 = 0; 14622 14623 agRequestType = AGSA_SATA_PROTOCOL_NON_DATA; 14624 14625 /* Initialize CB for SATA completion. 14626 */ 14627 satIOContext->satCompleteCB = &smsatModeSelect6n10CB; 14628 14629 /* 14630 * Prepare SGL and send FIS to LL layer. 14631 */ 14632 satIOContext->reqType = agRequestType; /* Save it */ 14633 14634 status = smsataLLIOStart( smRoot, 14635 smIORequest, 14636 smDeviceHandle, 14637 smScsiRequest, 14638 satIOContext); 14639 return status; 14640 } 14641 else 14642 { 14643 SM_DBG5(("smsatModeSelect10: disable information exceptions reporting\n")); 14644 /* sends SMART DISABLE OPERATIONS */ 14645 fis->h.fisType = 0x27; /* Reg host to device */ 14646 fis->h.c_pmPort = 0x80; /* C Bit is set */ 14647 14648 fis->h.command = SAT_SMART; /* 0xB0 */ 14649 fis->h.features = SAT_SMART_DISABLE_OPERATIONS; /* disable */ 14650 fis->d.lbaLow = 0; /* */ 14651 fis->d.lbaMid = 0x4F; /* 0x4F */ 14652 fis->d.lbaHigh = 0xC2; /* 0xC2 */ 14653 fis->d.device = 0; /* */ 14654 fis->d.lbaLowExp = 0; /* */ 14655 fis->d.lbaMidExp = 0; /* */ 14656 fis->d.lbaHighExp = 0; /* */ 14657 fis->d.featuresExp = 0; /* */ 14658 fis->d.sectorCount = 0; /* */ 14659 fis->d.sectorCountExp = 0; /* */ 14660 fis->d.reserved4 = 0; 14661 fis->d.control = 0; /* FIS HOB bit clear */ 14662 fis->d.reserved5 = 0; 14663 14664 agRequestType = AGSA_SATA_PROTOCOL_NON_DATA; 14665 14666 /* Initialize CB for SATA completion. 14667 */ 14668 satIOContext->satCompleteCB = &smsatModeSelect6n10CB; 14669 14670 /* 14671 * Prepare SGL and send FIS to LL layer. 14672 */ 14673 satIOContext->reqType = agRequestType; /* Save it */ 14674 14675 status = smsataLLIOStart( smRoot, 14676 smIORequest, 14677 smDeviceHandle, 14678 smScsiRequest, 14679 satIOContext); 14680 return status; 14681 14682 } 14683 } 14684 break; 14685 default: 14686 SM_DBG1(("smsatModeSelect10: Error unknown page code 0x%x!!!\n", pLogPage[12])); 14687 smsatSetSensePayload( pSense, 14688 SCSI_SNSKEY_NO_SENSE, 14689 0, 14690 SCSI_SNSCODE_NO_ADDITIONAL_INFO, 14691 satIOContext); 14692 14693 /*smEnqueueIO(smRoot, satIOContext);*/ 14694 14695 tdsmIOCompletedCB( smRoot, 14696 smIORequest, 14697 smIOSuccess, 14698 SCSI_STAT_CHECK_CONDITION, 14699 satIOContext->pSmSenseData, 14700 satIOContext->interruptContext ); 14701 return SM_RC_SUCCESS; 14702 } 14703 } 14704 14705 osGLOBAL bit32 14706 smsatSynchronizeCache10( 14707 smRoot_t *smRoot, 14708 smIORequest_t *smIORequest, 14709 smDeviceHandle_t *smDeviceHandle, 14710 smScsiInitiatorRequest_t *smScsiRequest, 14711 smSatIOContext_t *satIOContext 14712 ) 14713 { 14714 bit32 status; 14715 bit32 agRequestType; 14716 smDeviceData_t *pSatDevData; 14717 smScsiRspSense_t *pSense; 14718 smIniScsiCmnd_t *scsiCmnd; 14719 agsaFisRegHostToDevice_t *fis; 14720 14721 pSense = satIOContext->pSense; 14722 pSatDevData = satIOContext->pSatDevData; 14723 scsiCmnd = &smScsiRequest->scsiCmnd; 14724 fis = satIOContext->pFis; 14725 14726 SM_DBG5(("smsatSynchronizeCache10: start\n")); 14727 14728 /* checking CONTROL */ 14729 /* NACA == 1 or LINK == 1*/ 14730 if ( (scsiCmnd->cdb[9] & SCSI_NACA_MASK) || (scsiCmnd->cdb[9] & SCSI_LINK_MASK) ) 14731 { 14732 smsatSetSensePayload( pSense, 14733 SCSI_SNSKEY_ILLEGAL_REQUEST, 14734 0, 14735 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 14736 satIOContext); 14737 14738 /*smEnqueueIO(smRoot, satIOContext);*/ 14739 14740 tdsmIOCompletedCB( smRoot, 14741 smIORequest, 14742 smIOSuccess, 14743 SCSI_STAT_CHECK_CONDITION, 14744 satIOContext->pSmSenseData, 14745 satIOContext->interruptContext ); 14746 14747 SM_DBG1(("smsatSynchronizeCache10: return control!!!\n")); 14748 return SM_RC_SUCCESS; 14749 } 14750 14751 /* checking IMMED bit */ 14752 if (scsiCmnd->cdb[1] & SCSI_SYNC_CACHE_IMMED_MASK) 14753 { 14754 SM_DBG1(("smsatSynchronizeCache10: GOOD status due to IMMED bit!!!\n")); 14755 14756 /* return GOOD status first here */ 14757 tdsmIOCompletedCB( smRoot, 14758 smIORequest, 14759 smIOSuccess, 14760 SCSI_STAT_GOOD, 14761 agNULL, 14762 satIOContext->interruptContext); 14763 } 14764 14765 /* sends FLUSH CACHE or FLUSH CACHE EXT */ 14766 if (pSatDevData->sat48BitSupport == agTRUE) 14767 { 14768 SM_DBG5(("smsatSynchronizeCache10: sends FLUSH CACHE EXT\n")); 14769 /* FLUSH CACHE EXT */ 14770 fis->h.fisType = 0x27; /* Reg host to device */ 14771 fis->h.c_pmPort = 0x80; /* C Bit is set */ 14772 14773 fis->h.command = SAT_FLUSH_CACHE_EXT; /* 0xEA */ 14774 fis->h.features = 0; /* FIS reserve */ 14775 fis->d.featuresExp = 0; /* FIS reserve */ 14776 fis->d.sectorCount = 0; /* FIS sector count (7:0) */ 14777 fis->d.sectorCountExp = 0; /* FIS sector count (15:8) */ 14778 fis->d.lbaLow = 0; /* FIS LBA (7 :0 ) */ 14779 fis->d.lbaLowExp = 0; /* FIS LBA (31:24) */ 14780 fis->d.lbaMid = 0; /* FIS LBA (15:8 ) */ 14781 fis->d.lbaMidExp = 0; /* FIS LBA (39:32) */ 14782 fis->d.lbaHigh = 0; /* FIS LBA (23:16) */ 14783 fis->d.lbaHighExp = 0; /* FIS LBA (47:40) */ 14784 fis->d.device = 0; /* FIS DEV is discared in SATA */ 14785 fis->d.control = 0; /* FIS HOB bit clear */ 14786 fis->d.reserved4 = 0; 14787 fis->d.reserved5 = 0; 14788 14789 } 14790 else 14791 { 14792 SM_DBG5(("smsatSynchronizeCache10: sends FLUSH CACHE\n")); 14793 /* FLUSH CACHE */ 14794 fis->h.fisType = 0x27; /* Reg host to device */ 14795 fis->h.c_pmPort = 0x80; /* C Bit is set */ 14796 14797 fis->h.command = SAT_FLUSH_CACHE; /* 0xE7 */ 14798 fis->h.features = 0; /* FIS features NA */ 14799 fis->d.lbaLow = 0; /* FIS LBA (7 :0 ) */ 14800 fis->d.lbaMid = 0; /* FIS LBA (15:8 ) */ 14801 fis->d.lbaHigh = 0; /* FIS LBA (23:16) */ 14802 fis->d.lbaLowExp = 0; 14803 fis->d.lbaMidExp = 0; 14804 fis->d.lbaHighExp = 0; 14805 fis->d.featuresExp = 0; 14806 fis->d.sectorCount = 0; /* FIS sector count (7:0) */ 14807 fis->d.sectorCountExp = 0; 14808 fis->d.device = 0; /* FIS DEV is discared in SATA */ 14809 fis->d.control = 0; /* FIS HOB bit clear */ 14810 fis->d.reserved4 = 0; 14811 fis->d.reserved5 = 0; 14812 14813 } 14814 14815 agRequestType = AGSA_SATA_PROTOCOL_NON_DATA; 14816 14817 /* Initialize CB for SATA completion. 14818 */ 14819 satIOContext->satCompleteCB = &smsatSynchronizeCache10n16CB; 14820 14821 /* 14822 * Prepare SGL and send FIS to LL layer. 14823 */ 14824 satIOContext->reqType = agRequestType; /* Save it */ 14825 14826 status = smsataLLIOStart( smRoot, 14827 smIORequest, 14828 smDeviceHandle, 14829 smScsiRequest, 14830 satIOContext); 14831 14832 14833 return (status); 14834 } 14835 14836 osGLOBAL bit32 14837 smsatSynchronizeCache16( 14838 smRoot_t *smRoot, 14839 smIORequest_t *smIORequest, 14840 smDeviceHandle_t *smDeviceHandle, 14841 smScsiInitiatorRequest_t *smScsiRequest, 14842 smSatIOContext_t *satIOContext 14843 ) 14844 { 14845 bit32 status; 14846 bit32 agRequestType; 14847 smDeviceData_t *pSatDevData; 14848 smScsiRspSense_t *pSense; 14849 smIniScsiCmnd_t *scsiCmnd; 14850 agsaFisRegHostToDevice_t *fis; 14851 14852 pSense = satIOContext->pSense; 14853 pSatDevData = satIOContext->pSatDevData; 14854 scsiCmnd = &smScsiRequest->scsiCmnd; 14855 fis = satIOContext->pFis; 14856 14857 SM_DBG5(("smsatSynchronizeCache10: start\n")); 14858 14859 /* checking CONTROL */ 14860 /* NACA == 1 or LINK == 1*/ 14861 if ( (scsiCmnd->cdb[15] & SCSI_NACA_MASK) || (scsiCmnd->cdb[15] & SCSI_LINK_MASK) ) 14862 { 14863 smsatSetSensePayload( pSense, 14864 SCSI_SNSKEY_ILLEGAL_REQUEST, 14865 0, 14866 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 14867 satIOContext); 14868 14869 /*smEnqueueIO(smRoot, satIOContext);*/ 14870 14871 tdsmIOCompletedCB( smRoot, 14872 smIORequest, 14873 smIOSuccess, 14874 SCSI_STAT_CHECK_CONDITION, 14875 satIOContext->pSmSenseData, 14876 satIOContext->interruptContext ); 14877 14878 SM_DBG1(("smsatSynchronizeCache10: return control!!!\n")); 14879 return SM_RC_SUCCESS; 14880 } 14881 14882 14883 /* checking IMMED bit */ 14884 if (scsiCmnd->cdb[1] & SCSI_SYNC_CACHE_IMMED_MASK) 14885 { 14886 SM_DBG1(("smsatSynchronizeCache10: GOOD status due to IMMED bit!!!\n")); 14887 14888 /* return GOOD status first here */ 14889 tdsmIOCompletedCB( smRoot, 14890 smIORequest, 14891 smIOSuccess, 14892 SCSI_STAT_GOOD, 14893 agNULL, 14894 satIOContext->interruptContext); 14895 } 14896 14897 /* sends FLUSH CACHE or FLUSH CACHE EXT */ 14898 if (pSatDevData->sat48BitSupport == agTRUE) 14899 { 14900 SM_DBG5(("smsatSynchronizeCache10: sends FLUSH CACHE EXT\n")); 14901 /* FLUSH CACHE EXT */ 14902 fis->h.fisType = 0x27; /* Reg host to device */ 14903 fis->h.c_pmPort = 0x80; /* C Bit is set */ 14904 14905 fis->h.command = SAT_FLUSH_CACHE_EXT; /* 0xEA */ 14906 fis->h.features = 0; /* FIS reserve */ 14907 fis->d.featuresExp = 0; /* FIS reserve */ 14908 fis->d.sectorCount = 0; /* FIS sector count (7:0) */ 14909 fis->d.sectorCountExp = 0; /* FIS sector count (15:8) */ 14910 fis->d.lbaLow = 0; /* FIS LBA (7 :0 ) */ 14911 fis->d.lbaLowExp = 0; /* FIS LBA (31:24) */ 14912 fis->d.lbaMid = 0; /* FIS LBA (15:8 ) */ 14913 fis->d.lbaMidExp = 0; /* FIS LBA (39:32) */ 14914 fis->d.lbaHigh = 0; /* FIS LBA (23:16) */ 14915 fis->d.lbaHighExp = 0; /* FIS LBA (47:40) */ 14916 fis->d.device = 0; /* FIS DEV is discared in SATA */ 14917 fis->d.control = 0; /* FIS HOB bit clear */ 14918 fis->d.reserved4 = 0; 14919 fis->d.reserved5 = 0; 14920 14921 } 14922 else 14923 { 14924 SM_DBG5(("smsatSynchronizeCache10: sends FLUSH CACHE\n")); 14925 /* FLUSH CACHE */ 14926 fis->h.fisType = 0x27; /* Reg host to device */ 14927 fis->h.c_pmPort = 0x80; /* C Bit is set */ 14928 14929 fis->h.command = SAT_FLUSH_CACHE; /* 0xE7 */ 14930 fis->h.features = 0; /* FIS features NA */ 14931 fis->d.lbaLow = 0; /* FIS LBA (7 :0 ) */ 14932 fis->d.lbaMid = 0; /* FIS LBA (15:8 ) */ 14933 fis->d.lbaHigh = 0; /* FIS LBA (23:16) */ 14934 fis->d.lbaLowExp = 0; 14935 fis->d.lbaMidExp = 0; 14936 fis->d.lbaHighExp = 0; 14937 fis->d.featuresExp = 0; 14938 fis->d.sectorCount = 0; /* FIS sector count (7:0) */ 14939 fis->d.sectorCountExp = 0; 14940 fis->d.device = 0; /* FIS DEV is discared in SATA */ 14941 fis->d.control = 0; /* FIS HOB bit clear */ 14942 fis->d.reserved4 = 0; 14943 fis->d.reserved5 = 0; 14944 14945 } 14946 14947 agRequestType = AGSA_SATA_PROTOCOL_NON_DATA; 14948 14949 /* Initialize CB for SATA completion. 14950 */ 14951 satIOContext->satCompleteCB = &smsatSynchronizeCache10n16CB; 14952 14953 /* 14954 * Prepare SGL and send FIS to LL layer. 14955 */ 14956 satIOContext->reqType = agRequestType; /* Save it */ 14957 14958 status = smsataLLIOStart( smRoot, 14959 smIORequest, 14960 smDeviceHandle, 14961 smScsiRequest, 14962 satIOContext); 14963 14964 14965 return (status); 14966 } 14967 14968 osGLOBAL bit32 14969 smsatWriteAndVerify10( 14970 smRoot_t *smRoot, 14971 smIORequest_t *smIORequest, 14972 smDeviceHandle_t *smDeviceHandle, 14973 smScsiInitiatorRequest_t *smScsiRequest, 14974 smSatIOContext_t *satIOContext 14975 ) 14976 { 14977 /* 14978 combination of write10 and verify10 14979 */ 14980 14981 bit32 status; 14982 bit32 agRequestType = AGSA_SATA_PROTOCOL_DMA_WRITE; 14983 smDeviceData_t *pSatDevData; 14984 smScsiRspSense_t *pSense; 14985 smIniScsiCmnd_t *scsiCmnd; 14986 agsaFisRegHostToDevice_t *fis; 14987 bit32 lba = 0; 14988 bit32 tl = 0; 14989 bit32 LoopNum = 1; 14990 bit8 LBA[8]; 14991 bit8 TL[8]; 14992 bit32 AllChk = agFALSE; /* lba, lba+tl check against ATA limit and Disk capacity */ 14993 14994 pSense = satIOContext->pSense; 14995 pSatDevData = satIOContext->pSatDevData; 14996 scsiCmnd = &smScsiRequest->scsiCmnd; 14997 fis = satIOContext->pFis; 14998 14999 SM_DBG5(("smsatWriteAndVerify10: start\n")); 15000 15001 /* checking BYTCHK bit */ 15002 if (scsiCmnd->cdb[1] & SCSI_WRITE_N_VERIFY_BYTCHK_MASK) 15003 { 15004 smsatSetSensePayload( pSense, 15005 SCSI_SNSKEY_ILLEGAL_REQUEST, 15006 0, 15007 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 15008 satIOContext); 15009 15010 /*smEnqueueIO(smRoot, satIOContext);*/ 15011 15012 tdsmIOCompletedCB( smRoot, 15013 smIORequest, 15014 smIOSuccess, 15015 SCSI_STAT_CHECK_CONDITION, 15016 satIOContext->pSmSenseData, 15017 satIOContext->interruptContext ); 15018 15019 SM_DBG1(("smsatWriteAndVerify10: BYTCHK bit checking!!!\n")); 15020 return SM_RC_SUCCESS; 15021 } 15022 15023 15024 /* checking CONTROL */ 15025 /* NACA == 1 or LINK == 1*/ 15026 if ( (scsiCmnd->cdb[9] & SCSI_NACA_MASK) || (scsiCmnd->cdb[9] & SCSI_LINK_MASK) ) 15027 { 15028 smsatSetSensePayload( pSense, 15029 SCSI_SNSKEY_ILLEGAL_REQUEST, 15030 0, 15031 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 15032 satIOContext); 15033 15034 /*smEnqueueIO(smRoot, satIOContext);*/ 15035 15036 tdsmIOCompletedCB( smRoot, 15037 smIORequest, 15038 smIOSuccess, 15039 SCSI_STAT_CHECK_CONDITION, 15040 satIOContext->pSmSenseData, 15041 satIOContext->interruptContext ); 15042 15043 SM_DBG1(("smsatWriteAndVerify10: return control!!!\n")); 15044 return SM_RC_SUCCESS; 15045 } 15046 15047 sm_memset(LBA, 0, sizeof(LBA)); 15048 sm_memset(TL, 0, sizeof(TL)); 15049 15050 /* do not use memcpy due to indexing in LBA and TL */ 15051 LBA[0] = 0; /* MSB */ 15052 LBA[1] = 0; 15053 LBA[2] = 0; 15054 LBA[3] = 0; 15055 LBA[4] = scsiCmnd->cdb[2]; 15056 LBA[5] = scsiCmnd->cdb[3]; 15057 LBA[6] = scsiCmnd->cdb[4]; 15058 LBA[7] = scsiCmnd->cdb[5]; /* LSB */ 15059 15060 TL[0] = 0; 15061 TL[1] = 0; 15062 TL[2] = 0; 15063 TL[3] = 0; 15064 TL[4] = 0; 15065 TL[5] = 0; 15066 TL[6] = scsiCmnd->cdb[7]; 15067 TL[7] = scsiCmnd->cdb[8]; /* LSB */ 15068 15069 15070 /* cbd10; computing LBA and transfer length */ 15071 lba = (scsiCmnd->cdb[2] << (8*3)) + (scsiCmnd->cdb[3] << (8*2)) 15072 + (scsiCmnd->cdb[4] << 8) + scsiCmnd->cdb[5]; 15073 tl = (scsiCmnd->cdb[7] << 8) + scsiCmnd->cdb[8]; 15074 15075 15076 /* Table 34, 9.1, p 46 */ 15077 /* 15078 note: As of 2/10/2006, no support for DMA QUEUED 15079 */ 15080 15081 /* 15082 Table 34, 9.1, p 46, b 15083 When no 48-bit addressing support or NCQ, if LBA is beyond (2^28 - 1), 15084 return check condition 15085 */ 15086 if (pSatDevData->satNCQ != agTRUE && 15087 pSatDevData->sat48BitSupport != agTRUE 15088 ) 15089 { 15090 AllChk = smsatCheckLimit(LBA, TL, agFALSE, pSatDevData); 15091 if (AllChk) 15092 { 15093 SM_DBG1(("smsatWriteAndVerify10: return LBA out of range!!!\n")); 15094 smsatSetSensePayload( pSense, 15095 SCSI_SNSKEY_ILLEGAL_REQUEST, 15096 0, 15097 SCSI_SNSCODE_LOGICAL_BLOCK_ADDRESS_OUT_OF_RANGE, 15098 satIOContext); 15099 15100 /*smEnqueueIO(smRoot, satIOContext);*/ 15101 15102 tdsmIOCompletedCB( smRoot, 15103 smIORequest, 15104 smIOSuccess, 15105 SCSI_STAT_CHECK_CONDITION, 15106 satIOContext->pSmSenseData, 15107 satIOContext->interruptContext ); 15108 15109 return SM_RC_SUCCESS; 15110 } 15111 } 15112 else 15113 { 15114 AllChk = smsatCheckLimit(LBA, TL, agTRUE, pSatDevData); 15115 if (AllChk) 15116 { 15117 SM_DBG1(("smsatWriteAndVerify10: return LBA out of range, EXT!!!\n")); 15118 smsatSetSensePayload( pSense, 15119 SCSI_SNSKEY_ILLEGAL_REQUEST, 15120 0, 15121 SCSI_SNSCODE_LOGICAL_BLOCK_ADDRESS_OUT_OF_RANGE, 15122 satIOContext); 15123 15124 /*smEnqueueIO(smRoot, satIOContext);*/ 15125 15126 tdsmIOCompletedCB( smRoot, 15127 smIORequest, 15128 smIOSuccess, 15129 SCSI_STAT_CHECK_CONDITION, 15130 satIOContext->pSmSenseData, 15131 satIOContext->interruptContext ); 15132 15133 return SM_RC_SUCCESS; 15134 } 15135 } 15136 15137 15138 /* case 1 and 2 */ 15139 if (pSatDevData->satDMASupport == agTRUE && pSatDevData->satDMAEnabled == agTRUE) 15140 { 15141 /* case 2 */ 15142 /* WRITE DMA*/ 15143 /* can't fit the transfer length */ 15144 SM_DBG5(("smsatWriteAndVerify10: case 2\n")); 15145 fis->h.fisType = 0x27; /* Reg host to device */ 15146 fis->h.c_pmPort = 0x80; /* C bit is set */ 15147 fis->h.command = SAT_WRITE_DMA; /* 0xCA */ 15148 fis->h.features = 0; /* FIS reserve */ 15149 fis->d.lbaLow = scsiCmnd->cdb[5]; /* FIS LBA (7 :0 ) */ 15150 fis->d.lbaMid = scsiCmnd->cdb[4]; /* FIS LBA (15:8 ) */ 15151 fis->d.lbaHigh = scsiCmnd->cdb[3]; /* FIS LBA (23:16) */ 15152 15153 /* FIS LBA mode set LBA (27:24) */ 15154 fis->d.device = (bit8)((0x4 << 4) | (scsiCmnd->cdb[2] & 0xF)); 15155 15156 fis->d.lbaLowExp = 0; 15157 fis->d.lbaMidExp = 0; 15158 fis->d.lbaHighExp = 0; 15159 fis->d.featuresExp = 0; 15160 fis->d.sectorCount = scsiCmnd->cdb[8]; /* FIS sector count (7:0) */ 15161 fis->d.sectorCountExp = 0; 15162 fis->d.reserved4 = 0; 15163 fis->d.control = 0; /* FIS HOB bit clear */ 15164 fis->d.reserved5 = 0; 15165 15166 agRequestType = AGSA_SATA_PROTOCOL_DMA_WRITE; 15167 satIOContext->ATACmd = SAT_WRITE_DMA; 15168 } 15169 else 15170 { 15171 /* case 1 */ 15172 /* WRITE MULTIPLE or WRITE SECTOR(S) */ 15173 /* WRITE SECTORS for easier implemetation */ 15174 /* can't fit the transfer length */ 15175 SM_DBG5(("smsatWriteAndVerify10: case 1\n")); 15176 fis->h.fisType = 0x27; /* Reg host to device */ 15177 fis->h.c_pmPort = 0x80; /* C bit is set */ 15178 fis->h.command = SAT_WRITE_SECTORS; /* 0x30 */ 15179 fis->h.features = 0; /* FIS reserve */ 15180 fis->d.lbaLow = scsiCmnd->cdb[5]; /* FIS LBA (7 :0 ) */ 15181 fis->d.lbaMid = scsiCmnd->cdb[4]; /* FIS LBA (15:8 ) */ 15182 fis->d.lbaHigh = scsiCmnd->cdb[3]; /* FIS LBA (23:16) */ 15183 15184 /* FIS LBA mode set LBA (27:24) */ 15185 fis->d.device = (bit8)((0x4 << 4) | (scsiCmnd->cdb[2] & 0xF)); 15186 15187 fis->d.lbaLowExp = 0; 15188 fis->d.lbaMidExp = 0; 15189 fis->d.lbaHighExp = 0; 15190 fis->d.featuresExp = 0; 15191 fis->d.sectorCount = scsiCmnd->cdb[8]; /* FIS sector count (7:0) */ 15192 fis->d.sectorCountExp = 0; 15193 fis->d.reserved4 = 0; 15194 fis->d.control = 0; /* FIS HOB bit clear */ 15195 fis->d.reserved5 = 0; 15196 15197 agRequestType = AGSA_SATA_PROTOCOL_PIO_WRITE; 15198 satIOContext->ATACmd = SAT_WRITE_SECTORS; 15199 15200 } 15201 15202 /* case 3 and 4 */ 15203 if (pSatDevData->sat48BitSupport == agTRUE) 15204 { 15205 if (pSatDevData->satDMASupport == agTRUE && pSatDevData->satDMAEnabled == agTRUE) 15206 { 15207 /* case 3 */ 15208 /* WRITE DMA EXT or WRITE DMA FUA EXT */ 15209 SM_DBG5(("smsatWriteAndVerify10: case 3\n")); 15210 fis->h.fisType = 0x27; /* Reg host to device */ 15211 fis->h.c_pmPort = 0x80; /* C Bit is set */ 15212 15213 /* SAT_WRITE_DMA_FUA_EXT is optional and we don't support it */ 15214 fis->h.command = SAT_WRITE_DMA_EXT; /* 0x35 */ 15215 15216 fis->h.features = 0; /* FIS reserve */ 15217 fis->d.lbaLow = scsiCmnd->cdb[5]; /* FIS LBA (7 :0 ) */ 15218 fis->d.lbaMid = scsiCmnd->cdb[4]; /* FIS LBA (15:8 ) */ 15219 fis->d.lbaHigh = scsiCmnd->cdb[3]; /* FIS LBA (23:16) */ 15220 fis->d.device = 0x40; /* FIS LBA mode set */ 15221 fis->d.lbaLowExp = scsiCmnd->cdb[2]; /* FIS LBA (31:24) */ 15222 fis->d.lbaMidExp = 0; /* FIS LBA (39:32) */ 15223 fis->d.lbaHighExp = 0; /* FIS LBA (47:40) */ 15224 fis->d.featuresExp = 0; /* FIS reserve */ 15225 fis->d.sectorCount = scsiCmnd->cdb[8]; /* FIS sector count (7:0) */ 15226 fis->d.sectorCountExp = scsiCmnd->cdb[7]; /* FIS sector count (15:8) */ 15227 fis->d.reserved4 = 0; 15228 fis->d.control = 0; /* FIS HOB bit clear */ 15229 fis->d.reserved5 = 0; 15230 15231 agRequestType = AGSA_SATA_PROTOCOL_DMA_WRITE; 15232 satIOContext->ATACmd = SAT_WRITE_DMA_EXT; 15233 } 15234 else 15235 { 15236 /* case 4 */ 15237 /* WRITE MULTIPLE EXT or WRITE MULTIPLE FUA EXT or WRITE SECTOR(S) EXT */ 15238 /* WRITE SECTORS EXT for easier implemetation */ 15239 SM_DBG5(("smsatWriteAndVerify10: case 4\n")); 15240 fis->h.fisType = 0x27; /* Reg host to device */ 15241 fis->h.c_pmPort = 0x80; /* C Bit is set */ 15242 fis->h.command = SAT_WRITE_SECTORS_EXT; /* 0x34 */ 15243 15244 fis->h.features = 0; /* FIS reserve */ 15245 fis->d.lbaLow = scsiCmnd->cdb[5]; /* FIS LBA (7 :0 ) */ 15246 fis->d.lbaMid = scsiCmnd->cdb[4]; /* FIS LBA (15:8 ) */ 15247 fis->d.lbaHigh = scsiCmnd->cdb[3]; /* FIS LBA (23:16) */ 15248 fis->d.device = 0x40; /* FIS LBA mode set */ 15249 fis->d.lbaLowExp = scsiCmnd->cdb[2]; /* FIS LBA (31:24) */ 15250 fis->d.lbaMidExp = 0; /* FIS LBA (39:32) */ 15251 fis->d.lbaHighExp = 0; /* FIS LBA (47:40) */ 15252 fis->d.featuresExp = 0; /* FIS reserve */ 15253 fis->d.sectorCount = scsiCmnd->cdb[8]; /* FIS sector count (7:0) */ 15254 fis->d.sectorCountExp = scsiCmnd->cdb[7]; /* FIS sector count (15:8) */ 15255 fis->d.reserved4 = 0; 15256 fis->d.control = 0; /* FIS HOB bit clear */ 15257 fis->d.reserved5 = 0; 15258 15259 agRequestType = AGSA_SATA_PROTOCOL_PIO_WRITE; 15260 satIOContext->ATACmd = SAT_WRITE_SECTORS_EXT; 15261 } 15262 } 15263 /* case 5 */ 15264 if (pSatDevData->satNCQ == agTRUE) 15265 { 15266 /* WRITE FPDMA QUEUED */ 15267 if (pSatDevData->sat48BitSupport != agTRUE) 15268 { 15269 SM_DBG1(("smsatWriteAndVerify10: case 5 !!! error NCQ but 28 bit address support!!!\n")); 15270 smsatSetSensePayload( pSense, 15271 SCSI_SNSKEY_ILLEGAL_REQUEST, 15272 0, 15273 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 15274 satIOContext); 15275 15276 /*smEnqueueIO(smRoot, satIOContext);*/ 15277 15278 tdsmIOCompletedCB( smRoot, 15279 smIORequest, 15280 smIOSuccess, 15281 SCSI_STAT_CHECK_CONDITION, 15282 satIOContext->pSmSenseData, 15283 satIOContext->interruptContext ); 15284 return SM_RC_SUCCESS; 15285 } 15286 SM_DBG5(("smsatWriteAndVerify10: case 5\n")); 15287 15288 /* Support 48-bit FPDMA addressing, use WRITE FPDMA QUEUE command */ 15289 15290 fis->h.fisType = 0x27; /* Reg host to device */ 15291 fis->h.c_pmPort = 0x80; /* C Bit is set */ 15292 fis->h.command = SAT_WRITE_FPDMA_QUEUED; /* 0x61 */ 15293 fis->h.features = scsiCmnd->cdb[8]; /* FIS sector count (7:0) */ 15294 fis->d.lbaLow = scsiCmnd->cdb[5]; /* FIS LBA (7 :0 ) */ 15295 fis->d.lbaMid = scsiCmnd->cdb[4]; /* FIS LBA (15:8 ) */ 15296 fis->d.lbaHigh = scsiCmnd->cdb[3]; /* FIS LBA (23:16) */ 15297 15298 /* Check FUA bit */ 15299 if (scsiCmnd->cdb[1] & SCSI_WRITE_N_VERIFY10_FUA_MASK) 15300 fis->d.device = 0xC0; /* FIS FUA set */ 15301 else 15302 fis->d.device = 0x40; /* FIS FUA clear */ 15303 15304 fis->d.lbaLowExp = scsiCmnd->cdb[2]; /* FIS LBA (31:24) */ 15305 fis->d.lbaMidExp = 0; /* FIS LBA (39:32) */ 15306 fis->d.lbaHighExp = 0; /* FIS LBA (47:40) */ 15307 fis->d.featuresExp = scsiCmnd->cdb[7]; /* FIS sector count (15:8) */ 15308 fis->d.sectorCount = 0; /* Tag (7:3) set by LL layer */ 15309 fis->d.sectorCountExp = 0; 15310 fis->d.reserved4 = 0; 15311 fis->d.control = 0; /* FIS HOB bit clear */ 15312 fis->d.reserved5 = 0; 15313 15314 agRequestType = AGSA_SATA_PROTOCOL_FPDMA_WRITE; 15315 satIOContext->ATACmd = SAT_WRITE_FPDMA_QUEUED; 15316 } 15317 15318 satIOContext->currentLBA = lba; 15319 satIOContext->OrgTL = tl; 15320 15321 /* 15322 computing number of loop and remainder for tl 15323 0xFF in case not ext 15324 0xFFFF in case EXT 15325 */ 15326 if (fis->h.command == SAT_WRITE_SECTORS || fis->h.command == SAT_WRITE_DMA) 15327 { 15328 LoopNum = smsatComputeLoopNum(tl, 0xFF); 15329 } 15330 else if (fis->h.command == SAT_WRITE_SECTORS_EXT || 15331 fis->h.command == SAT_WRITE_DMA_EXT || 15332 fis->h.command == SAT_WRITE_DMA_FUA_EXT 15333 ) 15334 { 15335 /* SAT_READ_SECTORS_EXT, SAT_READ_DMA_EXT */ 15336 LoopNum = smsatComputeLoopNum(tl, 0xFFFF); 15337 } 15338 else 15339 { 15340 /* SAT_WRITE_FPDMA_QUEUED */ 15341 LoopNum = smsatComputeLoopNum(tl, 0xFFFF); 15342 } 15343 15344 satIOContext->LoopNum = LoopNum; 15345 15346 15347 if (LoopNum == 1) 15348 { 15349 SM_DBG5(("smsatWriteAndVerify10: NON CHAINED data\n")); 15350 /* Initialize CB for SATA completion. 15351 */ 15352 satIOContext->satCompleteCB = &smsatNonChainedWriteNVerifyCB; 15353 } 15354 else 15355 { 15356 SM_DBG1(("smsatWriteAndVerify10: CHAINED data!!!\n")); 15357 /* re-setting tl */ 15358 if (fis->h.command == SAT_WRITE_SECTORS || fis->h.command == SAT_WRITE_DMA) 15359 { 15360 fis->d.sectorCount = 0xFF; 15361 } 15362 else if (fis->h.command == SAT_WRITE_SECTORS_EXT || 15363 fis->h.command == SAT_WRITE_DMA_EXT || 15364 fis->h.command == SAT_WRITE_DMA_FUA_EXT 15365 ) 15366 { 15367 fis->d.sectorCount = 0xFF; 15368 fis->d.sectorCountExp = 0xFF; 15369 } 15370 else 15371 { 15372 /* SAT_WRITE_FPDMA_QUEUED */ 15373 fis->h.features = 0xFF; 15374 fis->d.featuresExp = 0xFF; 15375 } 15376 15377 /* Initialize CB for SATA completion. 15378 */ 15379 satIOContext->satCompleteCB = &smsatChainedWriteNVerifyCB; 15380 } 15381 15382 15383 /* 15384 * Prepare SGL and send FIS to LL layer. 15385 */ 15386 satIOContext->reqType = agRequestType; /* Save it */ 15387 15388 status = smsataLLIOStart( smRoot, 15389 smIORequest, 15390 smDeviceHandle, 15391 smScsiRequest, 15392 satIOContext); 15393 return (status); 15394 15395 } 15396 15397 osGLOBAL bit32 15398 smsatWriteAndVerify12( 15399 smRoot_t *smRoot, 15400 smIORequest_t *smIORequest, 15401 smDeviceHandle_t *smDeviceHandle, 15402 smScsiInitiatorRequest_t *smScsiRequest, 15403 smSatIOContext_t *satIOContext 15404 ) 15405 { 15406 /* 15407 combination of write12 and verify12 15408 temp: since write12 is not support (due to internal checking), no support 15409 */ 15410 bit32 status; 15411 bit32 agRequestType = AGSA_SATA_PROTOCOL_DMA_WRITE; 15412 smDeviceData_t *pSatDevData; 15413 smScsiRspSense_t *pSense; 15414 smIniScsiCmnd_t *scsiCmnd; 15415 agsaFisRegHostToDevice_t *fis; 15416 bit32 lba = 0; 15417 bit32 tl = 0; 15418 bit32 LoopNum = 1; 15419 bit8 LBA[8]; 15420 bit8 TL[8]; 15421 bit32 AllChk = agFALSE; /* lba, lba+tl check against ATA limit and Disk capacity */ 15422 15423 pSense = satIOContext->pSense; 15424 pSatDevData = satIOContext->pSatDevData; 15425 scsiCmnd = &smScsiRequest->scsiCmnd; 15426 fis = satIOContext->pFis; 15427 15428 SM_DBG5(("smsatWriteAndVerify12: start\n")); 15429 15430 /* checking BYTCHK bit */ 15431 if (scsiCmnd->cdb[1] & SCSI_WRITE_N_VERIFY_BYTCHK_MASK) 15432 { 15433 smsatSetSensePayload( pSense, 15434 SCSI_SNSKEY_ILLEGAL_REQUEST, 15435 0, 15436 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 15437 satIOContext); 15438 15439 /*smEnqueueIO(smRoot, satIOContext);*/ 15440 15441 tdsmIOCompletedCB( smRoot, 15442 smIORequest, 15443 smIOSuccess, 15444 SCSI_STAT_CHECK_CONDITION, 15445 satIOContext->pSmSenseData, 15446 satIOContext->interruptContext ); 15447 15448 SM_DBG1(("smsatWriteAndVerify12: BYTCHK bit checking!!!\n")); 15449 return SM_RC_SUCCESS; 15450 } 15451 15452 /* checking CONTROL */ 15453 /* NACA == 1 or LINK == 1*/ 15454 if ( (scsiCmnd->cdb[11] & SCSI_NACA_MASK) || (scsiCmnd->cdb[11] & SCSI_LINK_MASK) ) 15455 { 15456 smsatSetSensePayload( pSense, 15457 SCSI_SNSKEY_ILLEGAL_REQUEST, 15458 0, 15459 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 15460 satIOContext); 15461 15462 /*smEnqueueIO(smRoot, satIOContext);*/ 15463 15464 tdsmIOCompletedCB( smRoot, 15465 smIORequest, 15466 smIOSuccess, 15467 SCSI_STAT_CHECK_CONDITION, 15468 satIOContext->pSmSenseData, 15469 satIOContext->interruptContext ); 15470 15471 SM_DBG1(("smsatWriteAndVerify12: return control!!!\n")); 15472 return SM_RC_SUCCESS; 15473 } 15474 15475 sm_memset(LBA, 0, sizeof(LBA)); 15476 sm_memset(TL, 0, sizeof(TL)); 15477 15478 /* do not use memcpy due to indexing in LBA and TL */ 15479 LBA[0] = 0; /* MSB */ 15480 LBA[1] = 0; 15481 LBA[2] = 0; 15482 LBA[3] = 0; 15483 LBA[4] = scsiCmnd->cdb[2]; 15484 LBA[5] = scsiCmnd->cdb[3]; 15485 LBA[6] = scsiCmnd->cdb[4]; 15486 LBA[7] = scsiCmnd->cdb[5]; /* LSB */ 15487 15488 TL[0] = 0; /* MSB */ 15489 TL[1] = 0; 15490 TL[2] = 0; 15491 TL[3] = 0; 15492 TL[4] = scsiCmnd->cdb[6]; 15493 TL[5] = scsiCmnd->cdb[7]; 15494 TL[6] = scsiCmnd->cdb[8]; 15495 TL[7] = scsiCmnd->cdb[9]; /* LSB */ 15496 15497 15498 lba = smsatComputeCDB12LBA(satIOContext); 15499 tl = smsatComputeCDB12TL(satIOContext); 15500 15501 15502 /* Table 34, 9.1, p 46 */ 15503 /* 15504 note: As of 2/10/2006, no support for DMA QUEUED 15505 */ 15506 15507 /* 15508 Table 34, 9.1, p 46, b 15509 When no 48-bit addressing support or NCQ, if LBA is beyond (2^28 - 1), 15510 return check condition 15511 */ 15512 if (pSatDevData->satNCQ != agTRUE && 15513 pSatDevData->sat48BitSupport != agTRUE 15514 ) 15515 { 15516 AllChk = smsatCheckLimit(LBA, TL, agFALSE, pSatDevData); 15517 if (AllChk) 15518 { 15519 15520 /*smEnqueueIO(smRoot, satIOContext);*/ 15521 15522 15523 SM_DBG1(("smsatWriteAndVerify12: return LBA out of range, not EXT!!!\n")); 15524 15525 smsatSetSensePayload( pSense, 15526 SCSI_SNSKEY_ILLEGAL_REQUEST, 15527 0, 15528 SCSI_SNSCODE_LOGICAL_BLOCK_ADDRESS_OUT_OF_RANGE, 15529 satIOContext); 15530 15531 /*smEnqueueIO(smRoot, satIOContext);*/ 15532 15533 tdsmIOCompletedCB( smRoot, 15534 smIORequest, 15535 smIOSuccess, 15536 SCSI_STAT_CHECK_CONDITION, 15537 satIOContext->pSmSenseData, 15538 satIOContext->interruptContext ); 15539 15540 return SM_RC_SUCCESS; 15541 } 15542 } 15543 else 15544 { 15545 AllChk = smsatCheckLimit(LBA, TL, agTRUE, pSatDevData); 15546 if (AllChk) 15547 { 15548 SM_DBG1(("smsatWriteAndVerify12: return LBA out of range, EXT!!!\n")); 15549 smsatSetSensePayload( pSense, 15550 SCSI_SNSKEY_ILLEGAL_REQUEST, 15551 0, 15552 SCSI_SNSCODE_LOGICAL_BLOCK_ADDRESS_OUT_OF_RANGE, 15553 satIOContext); 15554 tdsmIOCompletedCB( smRoot, 15555 smIORequest, 15556 smIOSuccess, 15557 SCSI_STAT_CHECK_CONDITION, 15558 satIOContext->pSmSenseData, 15559 satIOContext->interruptContext ); 15560 return SM_RC_SUCCESS; 15561 } 15562 } 15563 if (pSatDevData->satDMASupport == agTRUE && pSatDevData->satDMAEnabled == agTRUE) 15564 { 15565 /* case 2 */ 15566 /* WRITE DMA*/ 15567 /* In case that we can't fit the transfer length, we loop */ 15568 SM_DBG5(("smsatWriteAndVerify12: case 2\n")); 15569 fis->h.fisType = 0x27; /* Reg host to device */ 15570 fis->h.c_pmPort = 0x80; /* C bit is set */ 15571 fis->h.command = SAT_WRITE_DMA; /* 0xCA */ 15572 fis->h.features = 0; /* FIS reserve */ 15573 fis->d.lbaLow = scsiCmnd->cdb[5]; /* FIS LBA (7 :0 ) */ 15574 fis->d.lbaMid = scsiCmnd->cdb[4]; /* FIS LBA (15:8 ) */ 15575 fis->d.lbaHigh = scsiCmnd->cdb[3]; /* FIS LBA (23:16) */ 15576 15577 /* FIS LBA mode set LBA (27:24) */ 15578 fis->d.device = (bit8)((0x4 << 4) | (scsiCmnd->cdb[2] & 0xF)); 15579 15580 fis->d.lbaLowExp = 0; 15581 fis->d.lbaMidExp = 0; 15582 fis->d.lbaHighExp = 0; 15583 fis->d.featuresExp = 0; 15584 fis->d.sectorCount = scsiCmnd->cdb[9]; /* FIS sector count (7:0) */ 15585 fis->d.sectorCountExp = 0; 15586 fis->d.reserved4 = 0; 15587 fis->d.control = 0; /* FIS HOB bit clear */ 15588 fis->d.reserved5 = 0; 15589 15590 agRequestType = AGSA_SATA_PROTOCOL_DMA_WRITE; 15591 satIOContext->ATACmd = SAT_WRITE_DMA; 15592 } 15593 else 15594 { 15595 /* case 1 */ 15596 /* WRITE MULTIPLE or WRITE SECTOR(S) */ 15597 /* WRITE SECTORS for easier implemetation */ 15598 /* In case that we can't fit the transfer length, we loop */ 15599 SM_DBG5(("smsatWriteAndVerify12: case 1\n")); 15600 fis->h.fisType = 0x27; /* Reg host to device */ 15601 fis->h.c_pmPort = 0x80; /* C bit is set */ 15602 fis->h.command = SAT_WRITE_SECTORS; /* 0x30 */ 15603 fis->h.features = 0; /* FIS reserve */ 15604 fis->d.lbaLow = scsiCmnd->cdb[5]; /* FIS LBA (7 :0 ) */ 15605 fis->d.lbaMid = scsiCmnd->cdb[4]; /* FIS LBA (15:8 ) */ 15606 fis->d.lbaHigh = scsiCmnd->cdb[3]; /* FIS LBA (23:16) */ 15607 15608 /* FIS LBA mode set LBA (27:24) */ 15609 fis->d.device = (bit8)((0x4 << 4) | (scsiCmnd->cdb[2] & 0xF)); 15610 15611 fis->d.lbaLowExp = 0; 15612 fis->d.lbaMidExp = 0; 15613 fis->d.lbaHighExp = 0; 15614 fis->d.featuresExp = 0; 15615 fis->d.sectorCount = scsiCmnd->cdb[9]; /* FIS sector count (7:0) */ 15616 fis->d.sectorCountExp = 0; 15617 fis->d.reserved4 = 0; 15618 fis->d.control = 0; /* FIS HOB bit clear */ 15619 fis->d.reserved5 = 0; 15620 15621 agRequestType = AGSA_SATA_PROTOCOL_PIO_WRITE; 15622 satIOContext->ATACmd = SAT_WRITE_SECTORS; 15623 } 15624 15625 /* case 3 and 4 */ 15626 if (pSatDevData->sat48BitSupport == agTRUE) 15627 { 15628 if (pSatDevData->satDMASupport == agTRUE && pSatDevData->satDMAEnabled == agTRUE) 15629 { 15630 /* case 3 */ 15631 /* WRITE DMA EXT or WRITE DMA FUA EXT */ 15632 SM_DBG5(("smsatWriteAndVerify12: case 3\n")); 15633 fis->h.fisType = 0x27; /* Reg host to device */ 15634 fis->h.c_pmPort = 0x80; /* C Bit is set */ 15635 15636 /* SAT_WRITE_DMA_FUA_EXT is optional and we don't support it */ 15637 fis->h.command = SAT_WRITE_DMA_EXT; /* 0x35 */ 15638 15639 fis->h.features = 0; /* FIS reserve */ 15640 fis->d.lbaLow = scsiCmnd->cdb[5]; /* FIS LBA (7 :0 ) */ 15641 fis->d.lbaMid = scsiCmnd->cdb[4]; /* FIS LBA (15:8 ) */ 15642 fis->d.lbaHigh = scsiCmnd->cdb[3]; /* FIS LBA (23:16) */ 15643 fis->d.device = 0x40; /* FIS LBA mode set */ 15644 fis->d.lbaLowExp = scsiCmnd->cdb[2]; /* FIS LBA (31:24) */ 15645 fis->d.lbaMidExp = 0; /* FIS LBA (39:32) */ 15646 fis->d.lbaHighExp = 0; /* FIS LBA (47:40) */ 15647 fis->d.featuresExp = 0; /* FIS reserve */ 15648 fis->d.sectorCount = scsiCmnd->cdb[9]; /* FIS sector count (7:0) */ 15649 fis->d.sectorCountExp = scsiCmnd->cdb[8]; /* FIS sector count (15:8) */ 15650 fis->d.reserved4 = 0; 15651 fis->d.control = 0; /* FIS HOB bit clear */ 15652 fis->d.reserved5 = 0; 15653 15654 agRequestType = AGSA_SATA_PROTOCOL_DMA_WRITE; 15655 satIOContext->ATACmd = SAT_WRITE_DMA_EXT; 15656 } 15657 else 15658 { 15659 /* case 4 */ 15660 /* WRITE MULTIPLE EXT or WRITE MULTIPLE FUA EXT or WRITE SECTOR(S) EXT */ 15661 /* WRITE SECTORS EXT for easier implemetation */ 15662 SM_DBG5(("smsatWriteAndVerify12: case 4\n")); 15663 fis->h.fisType = 0x27; /* Reg host to device */ 15664 fis->h.c_pmPort = 0x80; /* C Bit is set */ 15665 fis->h.command = SAT_WRITE_SECTORS_EXT; /* 0x34 */ 15666 15667 fis->h.features = 0; /* FIS reserve */ 15668 fis->d.lbaLow = scsiCmnd->cdb[5]; /* FIS LBA (7 :0 ) */ 15669 fis->d.lbaMid = scsiCmnd->cdb[4]; /* FIS LBA (15:8 ) */ 15670 fis->d.lbaHigh = scsiCmnd->cdb[3]; /* FIS LBA (23:16) */ 15671 fis->d.device = 0x40; /* FIS LBA mode set */ 15672 fis->d.lbaLowExp = scsiCmnd->cdb[2]; /* FIS LBA (31:24) */ 15673 fis->d.lbaMidExp = 0; /* FIS LBA (39:32) */ 15674 fis->d.lbaHighExp = 0; /* FIS LBA (47:40) */ 15675 fis->d.featuresExp = 0; /* FIS reserve */ 15676 fis->d.sectorCount = scsiCmnd->cdb[9]; /* FIS sector count (7:0) */ 15677 fis->d.sectorCountExp = scsiCmnd->cdb[8]; /* FIS sector count (15:8) */ 15678 fis->d.reserved4 = 0; 15679 fis->d.control = 0; /* FIS HOB bit clear */ 15680 fis->d.reserved5 = 0; 15681 15682 agRequestType = AGSA_SATA_PROTOCOL_PIO_WRITE; 15683 satIOContext->ATACmd = SAT_WRITE_SECTORS_EXT; 15684 } 15685 } 15686 15687 /* case 5 */ 15688 if (pSatDevData->satNCQ == agTRUE) 15689 { 15690 /* WRITE FPDMA QUEUED */ 15691 if (pSatDevData->sat48BitSupport != agTRUE) 15692 { 15693 SM_DBG1(("smsatWriteAndVerify12: case 5 !!! error NCQ but 28 bit address support!!!\n")); 15694 smsatSetSensePayload( pSense, 15695 SCSI_SNSKEY_ILLEGAL_REQUEST, 15696 0, 15697 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 15698 satIOContext); 15699 15700 /*smEnqueueIO(smRoot, satIOContext);*/ 15701 15702 tdsmIOCompletedCB( smRoot, 15703 smIORequest, 15704 smIOSuccess, 15705 SCSI_STAT_CHECK_CONDITION, 15706 satIOContext->pSmSenseData, 15707 satIOContext->interruptContext ); 15708 return SM_RC_SUCCESS; 15709 } 15710 SM_DBG6(("smsatWriteAndVerify12: case 5\n")); 15711 15712 /* Support 48-bit FPDMA addressing, use WRITE FPDMA QUEUE command */ 15713 15714 fis->h.fisType = 0x27; /* Reg host to device */ 15715 fis->h.c_pmPort = 0x80; /* C Bit is set */ 15716 fis->h.command = SAT_WRITE_FPDMA_QUEUED; /* 0x61 */ 15717 fis->h.features = scsiCmnd->cdb[9]; /* FIS sector count (7:0) */ 15718 fis->d.lbaLow = scsiCmnd->cdb[5]; /* FIS LBA (7 :0 ) */ 15719 fis->d.lbaMid = scsiCmnd->cdb[4]; /* FIS LBA (15:8 ) */ 15720 fis->d.lbaHigh = scsiCmnd->cdb[3]; /* FIS LBA (23:16) */ 15721 15722 /* Check FUA bit */ 15723 if (scsiCmnd->cdb[1] & SCSI_WRITE12_FUA_MASK) 15724 fis->d.device = 0xC0; /* FIS FUA set */ 15725 else 15726 fis->d.device = 0x40; /* FIS FUA clear */ 15727 15728 fis->d.lbaLowExp = scsiCmnd->cdb[2]; /* FIS LBA (31:24) */ 15729 fis->d.lbaMidExp = 0; /* FIS LBA (39:32) */ 15730 fis->d.lbaHighExp = 0; /* FIS LBA (47:40) */ 15731 fis->d.featuresExp = scsiCmnd->cdb[8]; /* FIS sector count (15:8) */ 15732 fis->d.sectorCount = 0; /* Tag (7:3) set by LL layer */ 15733 fis->d.sectorCountExp = 0; 15734 fis->d.reserved4 = 0; 15735 fis->d.control = 0; /* FIS HOB bit clear */ 15736 fis->d.reserved5 = 0; 15737 15738 agRequestType = AGSA_SATA_PROTOCOL_FPDMA_WRITE; 15739 satIOContext->ATACmd = SAT_WRITE_FPDMA_QUEUED; 15740 } 15741 15742 satIOContext->currentLBA = lba; 15743 // satIOContext->OrgLBA = lba; 15744 satIOContext->OrgTL = tl; 15745 15746 /* 15747 computing number of loop and remainder for tl 15748 0xFF in case not ext 15749 0xFFFF in case EXT 15750 */ 15751 if (fis->h.command == SAT_WRITE_SECTORS || fis->h.command == SAT_WRITE_DMA) 15752 { 15753 LoopNum = smsatComputeLoopNum(tl, 0xFF); 15754 } 15755 else if (fis->h.command == SAT_WRITE_SECTORS_EXT || 15756 fis->h.command == SAT_WRITE_DMA_EXT || 15757 fis->h.command == SAT_WRITE_DMA_FUA_EXT 15758 ) 15759 { 15760 /* SAT_READ_SECTORS_EXT, SAT_READ_DMA_EXT */ 15761 LoopNum = smsatComputeLoopNum(tl, 0xFFFF); 15762 } 15763 else 15764 { 15765 /* SAT_WRITE_FPDMA_QUEUEDK */ 15766 LoopNum = smsatComputeLoopNum(tl, 0xFFFF); 15767 } 15768 15769 satIOContext->LoopNum = LoopNum; 15770 satIOContext->LoopNum2 = LoopNum; 15771 15772 15773 if (LoopNum == 1) 15774 { 15775 SM_DBG5(("smsatWriteAndVerify12: NON CHAINED data\n")); 15776 /* Initialize CB for SATA completion. 15777 */ 15778 satIOContext->satCompleteCB = &smsatNonChainedWriteNVerifyCB; 15779 } 15780 else 15781 { 15782 SM_DBG1(("smsatWriteAndVerify12: CHAINED data!!!\n")); 15783 /* re-setting tl */ 15784 if (fis->h.command == SAT_WRITE_SECTORS || fis->h.command == SAT_WRITE_DMA) 15785 { 15786 fis->d.sectorCount = 0xFF; 15787 } 15788 else if (fis->h.command == SAT_WRITE_SECTORS_EXT || 15789 fis->h.command == SAT_WRITE_DMA_EXT || 15790 fis->h.command == SAT_WRITE_DMA_FUA_EXT 15791 ) 15792 { 15793 fis->d.sectorCount = 0xFF; 15794 fis->d.sectorCountExp = 0xFF; 15795 } 15796 else 15797 { 15798 /* SAT_WRITE_FPDMA_QUEUED */ 15799 fis->h.features = 0xFF; 15800 fis->d.featuresExp = 0xFF; 15801 } 15802 15803 /* Initialize CB for SATA completion. 15804 */ 15805 satIOContext->satCompleteCB = &smsatChainedWriteNVerifyCB; 15806 } 15807 15808 15809 /* 15810 * Prepare SGL and send FIS to LL layer. 15811 */ 15812 satIOContext->reqType = agRequestType; /* Save it */ 15813 15814 status = smsataLLIOStart( smRoot, 15815 smIORequest, 15816 smDeviceHandle, 15817 smScsiRequest, 15818 satIOContext); 15819 return (status); 15820 } 15821 15822 osGLOBAL bit32 15823 smsatWriteAndVerify16( 15824 smRoot_t *smRoot, 15825 smIORequest_t *smIORequest, 15826 smDeviceHandle_t *smDeviceHandle, 15827 smScsiInitiatorRequest_t *smScsiRequest, 15828 smSatIOContext_t *satIOContext 15829 ) 15830 { 15831 /* 15832 combination of write16 and verify16 15833 since write16 has 8 bytes LBA -> problem ATA LBA(upto 6 bytes), no support 15834 */ 15835 bit32 status; 15836 bit32 agRequestType = AGSA_SATA_PROTOCOL_DMA_WRITE; 15837 smDeviceData_t *pSatDevData; 15838 smScsiRspSense_t *pSense; 15839 smIniScsiCmnd_t *scsiCmnd; 15840 agsaFisRegHostToDevice_t *fis; 15841 bit32 lba = 0; 15842 bit32 tl = 0; 15843 bit32 LoopNum = 1; 15844 bit8 LBA[8]; 15845 bit8 TL[8]; 15846 bit32 AllChk = agFALSE; /* lba, lba+tl check against ATA limit and Disk capacity */ 15847 15848 pSense = satIOContext->pSense; 15849 pSatDevData = satIOContext->pSatDevData; 15850 scsiCmnd = &smScsiRequest->scsiCmnd; 15851 fis = satIOContext->pFis; 15852 15853 SM_DBG5(("smsatWriteAndVerify16: start\n")); 15854 15855 /* checking BYTCHK bit */ 15856 if (scsiCmnd->cdb[1] & SCSI_WRITE_N_VERIFY_BYTCHK_MASK) 15857 { 15858 smsatSetSensePayload( pSense, 15859 SCSI_SNSKEY_ILLEGAL_REQUEST, 15860 0, 15861 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 15862 satIOContext); 15863 15864 /*smEnqueueIO(smRoot, satIOContext);*/ 15865 15866 tdsmIOCompletedCB( smRoot, 15867 smIORequest, 15868 smIOSuccess, 15869 SCSI_STAT_CHECK_CONDITION, 15870 satIOContext->pSmSenseData, 15871 satIOContext->interruptContext ); 15872 15873 SM_DBG1(("smsatWriteAndVerify16: BYTCHK bit checking!!!\n")); 15874 return SM_RC_SUCCESS; 15875 } 15876 15877 15878 /* checking CONTROL */ 15879 /* NACA == 1 or LINK == 1*/ 15880 if ( (scsiCmnd->cdb[15] & SCSI_NACA_MASK) || (scsiCmnd->cdb[15] & SCSI_LINK_MASK) ) 15881 { 15882 smsatSetSensePayload( pSense, 15883 SCSI_SNSKEY_ILLEGAL_REQUEST, 15884 0, 15885 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 15886 satIOContext); 15887 15888 /*smEnqueueIO(smRoot, satIOContext);*/ 15889 15890 tdsmIOCompletedCB( smRoot, 15891 smIORequest, 15892 smIOSuccess, 15893 SCSI_STAT_CHECK_CONDITION, 15894 satIOContext->pSmSenseData, 15895 satIOContext->interruptContext ); 15896 15897 SM_DBG1(("smsatWriteAndVerify16: return control!!!\n")); 15898 return SM_RC_SUCCESS; 15899 } 15900 15901 sm_memset(LBA, 0, sizeof(LBA)); 15902 sm_memset(TL, 0, sizeof(TL)); 15903 15904 15905 /* do not use memcpy due to indexing in LBA and TL */ 15906 LBA[0] = scsiCmnd->cdb[2]; /* MSB */ 15907 LBA[1] = scsiCmnd->cdb[3]; 15908 LBA[2] = scsiCmnd->cdb[4]; 15909 LBA[3] = scsiCmnd->cdb[5]; 15910 LBA[4] = scsiCmnd->cdb[6]; 15911 LBA[5] = scsiCmnd->cdb[7]; 15912 LBA[6] = scsiCmnd->cdb[8]; 15913 LBA[7] = scsiCmnd->cdb[9]; /* LSB */ 15914 15915 TL[0] = 0; 15916 TL[1] = 0; 15917 TL[2] = 0; 15918 TL[3] = 0; 15919 TL[4] = scsiCmnd->cdb[10]; /* MSB */ 15920 TL[5] = scsiCmnd->cdb[11]; 15921 TL[6] = scsiCmnd->cdb[12]; 15922 TL[7] = scsiCmnd->cdb[13]; /* LSB */ 15923 15924 15925 15926 lba = smsatComputeCDB16LBA(satIOContext); 15927 tl = smsatComputeCDB16TL(satIOContext); 15928 15929 15930 /* Table 34, 9.1, p 46 */ 15931 /* 15932 note: As of 2/10/2006, no support for DMA QUEUED 15933 */ 15934 15935 /* 15936 Table 34, 9.1, p 46, b 15937 When no 48-bit addressing support or NCQ, if LBA is beyond (2^28 - 1), 15938 return check condition 15939 */ 15940 if (pSatDevData->satNCQ != agTRUE && 15941 pSatDevData->sat48BitSupport != agTRUE 15942 ) 15943 { 15944 AllChk = smsatCheckLimit(LBA, TL, agFALSE, pSatDevData); 15945 if (AllChk) 15946 { 15947 SM_DBG1(("smsatWriteAndVerify16: return LBA out of range, not EXT!!!\n")); 15948 smsatSetSensePayload( pSense, 15949 SCSI_SNSKEY_ILLEGAL_REQUEST, 15950 0, 15951 SCSI_SNSCODE_LOGICAL_BLOCK_ADDRESS_OUT_OF_RANGE, 15952 satIOContext); 15953 15954 /*smEnqueueIO(smRoot, satIOContext);*/ 15955 15956 tdsmIOCompletedCB( smRoot, 15957 smIORequest, 15958 smIOSuccess, 15959 SCSI_STAT_CHECK_CONDITION, 15960 satIOContext->pSmSenseData, 15961 satIOContext->interruptContext ); 15962 15963 return SM_RC_SUCCESS; 15964 } 15965 } 15966 else 15967 { 15968 AllChk = smsatCheckLimit(LBA, TL, agTRUE, pSatDevData); 15969 if (AllChk) 15970 { 15971 SM_DBG1(("smsatWriteAndVerify16: return LBA out of range, EXT!!!\n")); 15972 smsatSetSensePayload( pSense, 15973 SCSI_SNSKEY_ILLEGAL_REQUEST, 15974 0, 15975 SCSI_SNSCODE_LOGICAL_BLOCK_ADDRESS_OUT_OF_RANGE, 15976 satIOContext); 15977 15978 /*smEnqueueIO(smRoot, satIOContext);*/ 15979 15980 tdsmIOCompletedCB( smRoot, 15981 smIORequest, 15982 smIOSuccess, 15983 SCSI_STAT_CHECK_CONDITION, 15984 satIOContext->pSmSenseData, 15985 satIOContext->interruptContext ); 15986 15987 return SM_RC_SUCCESS; 15988 } 15989 } 15990 15991 15992 /* case 1 and 2 */ 15993 if (pSatDevData->satDMASupport == agTRUE && pSatDevData->satDMAEnabled == agTRUE) 15994 { 15995 /* case 2 */ 15996 /* WRITE DMA*/ 15997 /* In case that we can't fit the transfer length, we loop */ 15998 SM_DBG5(("smsatWriteAndVerify16: case 2\n")); 15999 fis->h.fisType = 0x27; /* Reg host to device */ 16000 fis->h.c_pmPort = 0x80; /* C bit is set */ 16001 fis->h.command = SAT_WRITE_DMA; /* 0xCA */ 16002 fis->h.features = 0; /* FIS reserve */ 16003 fis->d.lbaLow = scsiCmnd->cdb[9]; /* FIS LBA (7 :0 ) */ 16004 fis->d.lbaMid = scsiCmnd->cdb[8]; /* FIS LBA (15:8 ) */ 16005 fis->d.lbaHigh = scsiCmnd->cdb[7]; /* FIS LBA (23:16) */ 16006 16007 /* FIS LBA mode set LBA (27:24) */ 16008 fis->d.device = (bit8)((0x4 << 4) | (scsiCmnd->cdb[6] & 0xF)); 16009 16010 fis->d.lbaLowExp = 0; 16011 fis->d.lbaMidExp = 0; 16012 fis->d.lbaHighExp = 0; 16013 fis->d.featuresExp = 0; 16014 fis->d.sectorCount = scsiCmnd->cdb[13]; /* FIS sector count (7:0) */ 16015 fis->d.sectorCountExp = 0; 16016 fis->d.reserved4 = 0; 16017 fis->d.control = 0; /* FIS HOB bit clear */ 16018 fis->d.reserved5 = 0; 16019 16020 agRequestType = AGSA_SATA_PROTOCOL_DMA_WRITE; 16021 satIOContext->ATACmd = SAT_WRITE_DMA; 16022 } 16023 else 16024 { 16025 /* case 1 */ 16026 /* WRITE MULTIPLE or WRITE SECTOR(S) */ 16027 /* WRITE SECTORS for easier implemetation */ 16028 /* In case that we can't fit the transfer length, we loop */ 16029 SM_DBG5(("smsatWriteAndVerify16: case 1\n")); 16030 fis->h.fisType = 0x27; /* Reg host to device */ 16031 fis->h.c_pmPort = 0x80; /* C bit is set */ 16032 fis->h.command = SAT_WRITE_SECTORS; /* 0x30 */ 16033 fis->h.features = 0; /* FIS reserve */ 16034 fis->d.lbaLow = scsiCmnd->cdb[9]; /* FIS LBA (7 :0 ) */ 16035 fis->d.lbaMid = scsiCmnd->cdb[8]; /* FIS LBA (15:8 ) */ 16036 fis->d.lbaHigh = scsiCmnd->cdb[7]; /* FIS LBA (23:16) */ 16037 16038 /* FIS LBA mode set LBA (27:24) */ 16039 fis->d.device = (bit8)((0x4 << 4) | (scsiCmnd->cdb[6] & 0xF)); 16040 16041 fis->d.lbaLowExp = 0; 16042 fis->d.lbaMidExp = 0; 16043 fis->d.lbaHighExp = 0; 16044 fis->d.featuresExp = 0; 16045 fis->d.sectorCount = scsiCmnd->cdb[13]; /* FIS sector count (7:0) */ 16046 fis->d.sectorCountExp = 0; 16047 fis->d.reserved4 = 0; 16048 fis->d.control = 0; /* FIS HOB bit clear */ 16049 fis->d.reserved5 = 0; 16050 16051 agRequestType = AGSA_SATA_PROTOCOL_PIO_WRITE; 16052 satIOContext->ATACmd = SAT_WRITE_SECTORS; 16053 } 16054 16055 /* case 3 and 4 */ 16056 if (pSatDevData->sat48BitSupport == agTRUE) 16057 { 16058 if (pSatDevData->satDMASupport == agTRUE && pSatDevData->satDMAEnabled == agTRUE) 16059 { 16060 /* case 3 */ 16061 /* WRITE DMA EXT or WRITE DMA FUA EXT */ 16062 SM_DBG5(("smsatWriteAndVerify16: case 3\n")); 16063 fis->h.fisType = 0x27; /* Reg host to device */ 16064 fis->h.c_pmPort = 0x80; /* C Bit is set */ 16065 16066 /* SAT_WRITE_DMA_FUA_EXT is optional and we don't support it */ 16067 fis->h.command = SAT_WRITE_DMA_EXT; /* 0x35 */ 16068 16069 fis->h.features = 0; /* FIS reserve */ 16070 fis->d.lbaLow = scsiCmnd->cdb[9]; /* FIS LBA (7 :0 ) */ 16071 fis->d.lbaMid = scsiCmnd->cdb[8]; /* FIS LBA (15:8 ) */ 16072 fis->d.lbaHigh = scsiCmnd->cdb[7]; /* FIS LBA (23:16) */ 16073 fis->d.device = 0x40; /* FIS LBA mode set */ 16074 fis->d.lbaLowExp = scsiCmnd->cdb[6]; /* FIS LBA (31:24) */ 16075 fis->d.lbaMidExp = scsiCmnd->cdb[5]; /* FIS LBA (39:32) */ 16076 fis->d.lbaHighExp = scsiCmnd->cdb[4]; /* FIS LBA (47:40) */ 16077 fis->d.featuresExp = 0; /* FIS reserve */ 16078 fis->d.sectorCount = scsiCmnd->cdb[13]; /* FIS sector count (7:0) */ 16079 fis->d.sectorCountExp = scsiCmnd->cdb[12]; /* FIS sector count (15:8) */ 16080 fis->d.reserved4 = 0; 16081 fis->d.control = 0; /* FIS HOB bit clear */ 16082 fis->d.reserved5 = 0; 16083 16084 agRequestType = AGSA_SATA_PROTOCOL_DMA_WRITE; 16085 satIOContext->ATACmd = SAT_WRITE_DMA_EXT; 16086 } 16087 else 16088 { 16089 /* case 4 */ 16090 /* WRITE MULTIPLE EXT or WRITE MULTIPLE FUA EXT or WRITE SECTOR(S) EXT */ 16091 /* WRITE SECTORS EXT for easier implemetation */ 16092 SM_DBG5(("smsatWriteAndVerify16: case 4\n")); 16093 fis->h.fisType = 0x27; /* Reg host to device */ 16094 fis->h.c_pmPort = 0x80; /* C Bit is set */ 16095 fis->h.command = SAT_WRITE_SECTORS_EXT; /* 0x34 */ 16096 16097 fis->h.features = 0; /* FIS reserve */ 16098 fis->d.lbaLow = scsiCmnd->cdb[9]; /* FIS LBA (7 :0 ) */ 16099 fis->d.lbaMid = scsiCmnd->cdb[8]; /* FIS LBA (15:8 ) */ 16100 fis->d.lbaHigh = scsiCmnd->cdb[7]; /* FIS LBA (23:16) */ 16101 fis->d.device = 0x40; /* FIS LBA mode set */ 16102 fis->d.lbaLowExp = scsiCmnd->cdb[6]; /* FIS LBA (31:24) */ 16103 fis->d.lbaMidExp = scsiCmnd->cdb[5]; /* FIS LBA (39:32) */ 16104 fis->d.lbaHighExp = scsiCmnd->cdb[4]; /* FIS LBA (47:40) */ 16105 fis->d.featuresExp = 0; /* FIS reserve */ 16106 fis->d.sectorCount = scsiCmnd->cdb[13]; /* FIS sector count (7:0) */ 16107 fis->d.sectorCountExp = scsiCmnd->cdb[12]; /* FIS sector count (15:8) */ 16108 fis->d.reserved4 = 0; 16109 fis->d.control = 0; /* FIS HOB bit clear */ 16110 fis->d.reserved5 = 0; 16111 16112 agRequestType = AGSA_SATA_PROTOCOL_PIO_WRITE; 16113 satIOContext->ATACmd = SAT_WRITE_SECTORS_EXT; 16114 } 16115 } 16116 16117 /* case 5 */ 16118 if (pSatDevData->satNCQ == agTRUE) 16119 { 16120 /* WRITE FPDMA QUEUED */ 16121 if (pSatDevData->sat48BitSupport != agTRUE) 16122 { 16123 SM_DBG1(("smsatWriteAndVerify16: case 5 !!! error NCQ but 28 bit address support!!!\n")); 16124 smsatSetSensePayload( pSense, 16125 SCSI_SNSKEY_ILLEGAL_REQUEST, 16126 0, 16127 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 16128 satIOContext); 16129 16130 /*smEnqueueIO(smRoot, satIOContext);*/ 16131 16132 tdsmIOCompletedCB( smRoot, 16133 smIORequest, 16134 smIOSuccess, 16135 SCSI_STAT_CHECK_CONDITION, 16136 satIOContext->pSmSenseData, 16137 satIOContext->interruptContext ); 16138 return SM_RC_SUCCESS; 16139 } 16140 SM_DBG6(("smsatWriteAndVerify16: case 5\n")); 16141 16142 /* Support 48-bit FPDMA addressing, use WRITE FPDMA QUEUE command */ 16143 16144 fis->h.fisType = 0x27; /* Reg host to device */ 16145 fis->h.c_pmPort = 0x80; /* C Bit is set */ 16146 fis->h.command = SAT_WRITE_FPDMA_QUEUED; /* 0x61 */ 16147 fis->h.features = scsiCmnd->cdb[13]; /* FIS sector count (7:0) */ 16148 fis->d.lbaLow = scsiCmnd->cdb[9]; /* FIS LBA (7 :0 ) */ 16149 fis->d.lbaMid = scsiCmnd->cdb[8]; /* FIS LBA (15:8 ) */ 16150 fis->d.lbaHigh = scsiCmnd->cdb[7]; /* FIS LBA (23:16) */ 16151 16152 /* Check FUA bit */ 16153 if (scsiCmnd->cdb[1] & SCSI_WRITE16_FUA_MASK) 16154 fis->d.device = 0xC0; /* FIS FUA set */ 16155 else 16156 fis->d.device = 0x40; /* FIS FUA clear */ 16157 16158 fis->d.lbaLowExp = scsiCmnd->cdb[6]; /* FIS LBA (31:24) */ 16159 fis->d.lbaMidExp = scsiCmnd->cdb[5]; /* FIS LBA (39:32) */ 16160 fis->d.lbaHighExp = scsiCmnd->cdb[4]; /* FIS LBA (47:40) */ 16161 fis->d.featuresExp = scsiCmnd->cdb[12]; /* FIS sector count (15:8) */ 16162 fis->d.sectorCount = 0; /* Tag (7:3) set by LL layer */ 16163 fis->d.sectorCountExp = 0; 16164 fis->d.reserved4 = 0; 16165 fis->d.control = 0; /* FIS HOB bit clear */ 16166 fis->d.reserved5 = 0; 16167 16168 agRequestType = AGSA_SATA_PROTOCOL_FPDMA_WRITE; 16169 satIOContext->ATACmd = SAT_WRITE_FPDMA_QUEUED; 16170 } 16171 16172 satIOContext->currentLBA = lba; 16173 satIOContext->OrgTL = tl; 16174 16175 /* 16176 computing number of loop and remainder for tl 16177 0xFF in case not ext 16178 0xFFFF in case EXT 16179 */ 16180 if (fis->h.command == SAT_WRITE_SECTORS || fis->h.command == SAT_WRITE_DMA) 16181 { 16182 LoopNum = smsatComputeLoopNum(tl, 0xFF); 16183 } 16184 else if (fis->h.command == SAT_WRITE_SECTORS_EXT || 16185 fis->h.command == SAT_WRITE_DMA_EXT || 16186 fis->h.command == SAT_WRITE_DMA_FUA_EXT 16187 ) 16188 { 16189 /* SAT_READ_SECTORS_EXT, SAT_READ_DMA_EXT */ 16190 LoopNum = smsatComputeLoopNum(tl, 0xFFFF); 16191 } 16192 else 16193 { 16194 /* SAT_WRITE_FPDMA_QUEUEDK */ 16195 LoopNum = smsatComputeLoopNum(tl, 0xFFFF); 16196 } 16197 16198 satIOContext->LoopNum = LoopNum; 16199 16200 16201 if (LoopNum == 1) 16202 { 16203 SM_DBG5(("smsatWriteAndVerify16: NON CHAINED data\n")); 16204 /* Initialize CB for SATA completion. 16205 */ 16206 satIOContext->satCompleteCB = &smsatNonChainedWriteNVerifyCB; 16207 } 16208 else 16209 { 16210 SM_DBG1(("smsatWriteAndVerify16: CHAINED data!!!\n")); 16211 /* re-setting tl */ 16212 if (fis->h.command == SAT_WRITE_SECTORS || fis->h.command == SAT_WRITE_DMA) 16213 { 16214 fis->d.sectorCount = 0xFF; 16215 } 16216 else if (fis->h.command == SAT_WRITE_SECTORS_EXT || 16217 fis->h.command == SAT_WRITE_DMA_EXT || 16218 fis->h.command == SAT_WRITE_DMA_FUA_EXT 16219 ) 16220 { 16221 fis->d.sectorCount = 0xFF; 16222 fis->d.sectorCountExp = 0xFF; 16223 } 16224 else 16225 { 16226 /* SAT_WRITE_FPDMA_QUEUED */ 16227 fis->h.features = 0xFF; 16228 fis->d.featuresExp = 0xFF; 16229 } 16230 16231 /* Initialize CB for SATA completion. 16232 */ 16233 satIOContext->satCompleteCB = &smsatChainedWriteNVerifyCB; 16234 } 16235 16236 16237 /* 16238 * Prepare SGL and send FIS to LL layer. 16239 */ 16240 satIOContext->reqType = agRequestType; /* Save it */ 16241 16242 status = smsataLLIOStart( smRoot, 16243 smIORequest, 16244 smDeviceHandle, 16245 smScsiRequest, 16246 satIOContext); 16247 return (status); 16248 } 16249 16250 osGLOBAL bit32 16251 smsatReadMediaSerialNumber( 16252 smRoot_t *smRoot, 16253 smIORequest_t *smIORequest, 16254 smDeviceHandle_t *smDeviceHandle, 16255 smScsiInitiatorRequest_t *smScsiRequest, 16256 smSatIOContext_t *satIOContext 16257 ) 16258 { 16259 bit32 status; 16260 bit32 agRequestType = AGSA_SATA_PROTOCOL_PIO_READ; 16261 smDeviceData_t *pSatDevData; 16262 smScsiRspSense_t *pSense; 16263 smIniScsiCmnd_t *scsiCmnd; 16264 agsaFisRegHostToDevice_t *fis; 16265 agsaSATAIdentifyData_t *pSATAIdData; 16266 bit8 *pSerialNumber; 16267 bit8 MediaSerialNumber[64] = {0}; 16268 bit32 allocationLen = 0; 16269 16270 pSense = satIOContext->pSense; 16271 pSatDevData = satIOContext->pSatDevData; 16272 scsiCmnd = &smScsiRequest->scsiCmnd; 16273 fis = satIOContext->pFis; 16274 pSATAIdData = &(pSatDevData->satIdentifyData); 16275 pSerialNumber = (bit8 *) smScsiRequest->sglVirtualAddr; 16276 16277 SM_DBG5(("smsatReadMediaSerialNumber: start\n")); 16278 16279 /* checking CONTROL */ 16280 /* NACA == 1 or LINK == 1*/ 16281 if ( (scsiCmnd->cdb[11] & SCSI_NACA_MASK) || (scsiCmnd->cdb[11] & SCSI_LINK_MASK) ) 16282 { 16283 smsatSetSensePayload( pSense, 16284 SCSI_SNSKEY_ILLEGAL_REQUEST, 16285 0, 16286 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 16287 satIOContext); 16288 16289 /*smEnqueueIO(smRoot, satIOContext);*/ 16290 16291 tdsmIOCompletedCB( smRoot, 16292 smIORequest, 16293 smIOSuccess, 16294 SCSI_STAT_CHECK_CONDITION, 16295 satIOContext->pSmSenseData, 16296 satIOContext->interruptContext ); 16297 16298 SM_DBG1(("smsatReadMediaSerialNumber: return control!!!\n")); 16299 return SM_RC_SUCCESS; 16300 } 16301 16302 allocationLen = (((bit32)scsiCmnd->cdb[6]) << 24) | 16303 (((bit32)scsiCmnd->cdb[7]) << 16) | 16304 (((bit32)scsiCmnd->cdb[8]) << 8 ) | 16305 (((bit32)scsiCmnd->cdb[9])); 16306 allocationLen = MIN(allocationLen, scsiCmnd->expDataLength); 16307 if (allocationLen == 4) 16308 { 16309 if (pSATAIdData->commandSetFeatureDefault & 0x4) 16310 { 16311 SM_DBG1(("smsatReadMediaSerialNumber: Media serial number returning only length!!!\n")); 16312 /* SPC-3 6.16 p192; filling in length */ 16313 MediaSerialNumber[0] = 0; 16314 MediaSerialNumber[1] = 0; 16315 MediaSerialNumber[2] = 0; 16316 MediaSerialNumber[3] = 0x3C; 16317 } 16318 else 16319 { 16320 /* 1 sector - 4 = 512 - 4 to avoid underflow; 0x1fc*/ 16321 MediaSerialNumber[0] = 0; 16322 MediaSerialNumber[1] = 0; 16323 MediaSerialNumber[2] = 0x1; 16324 MediaSerialNumber[3] = 0xfc; 16325 } 16326 16327 sm_memcpy(pSerialNumber, MediaSerialNumber, 4); 16328 /*smEnqueueIO(smRoot, satIOContext);*/ 16329 16330 tdsmIOCompletedCB( smRoot, 16331 smIORequest, 16332 smIOSuccess, 16333 SCSI_STAT_GOOD, 16334 agNULL, 16335 satIOContext->interruptContext); 16336 16337 return SM_RC_SUCCESS; 16338 } 16339 16340 if ( pSatDevData->IDDeviceValid == agTRUE) 16341 { 16342 if (pSATAIdData->commandSetFeatureDefault & 0x4) 16343 { 16344 /* word87 bit2 Media serial number is valid */ 16345 /* read word 176 to 205; length is 2*30 = 60 = 0x3C*/ 16346 #ifdef LOG_ENABLE 16347 smhexdump("ID smsatReadMediaSerialNumber", (bit8*)pSATAIdData->currentMediaSerialNumber, 2*30); 16348 #endif 16349 /* SPC-3 6.16 p192; filling in length */ 16350 MediaSerialNumber[0] = 0; 16351 MediaSerialNumber[1] = 0; 16352 MediaSerialNumber[2] = 0; 16353 MediaSerialNumber[3] = 0x3C; 16354 sm_memcpy(&MediaSerialNumber[4], (void *)pSATAIdData->currentMediaSerialNumber, 60); 16355 #ifdef LOG_ENABLE 16356 smhexdump("smsatReadMediaSerialNumber", (bit8*)MediaSerialNumber, 2*30 + 4); 16357 #endif 16358 sm_memcpy(pSerialNumber, MediaSerialNumber, MIN(allocationLen, 64)); 16359 /*smEnqueueIO(smRoot, satIOContext);*/ 16360 16361 tdsmIOCompletedCB( smRoot, 16362 smIORequest, 16363 smIOSuccess, 16364 SCSI_STAT_GOOD, 16365 agNULL, 16366 satIOContext->interruptContext); 16367 return SM_RC_SUCCESS; 16368 16369 16370 } 16371 else 16372 { 16373 /* word87 bit2 Media serial number is NOT valid */ 16374 SM_DBG1(("smsatReadMediaSerialNumber: Media serial number is NOT valid!!!\n")); 16375 16376 if (pSatDevData->sat48BitSupport == agTRUE) 16377 { 16378 /* READ VERIFY SECTORS EXT */ 16379 fis->h.fisType = 0x27; /* Reg host to device */ 16380 fis->h.c_pmPort = 0x80; /* C Bit is set */ 16381 fis->h.command = SAT_READ_SECTORS_EXT; /* 0x24 */ 16382 16383 fis->h.features = 0; /* FIS reserve */ 16384 fis->d.lbaLow = 0; /* FIS LBA (7 :0 ) */ 16385 fis->d.lbaMid = 0; /* FIS LBA (15:8 ) */ 16386 fis->d.lbaHigh = 0; /* FIS LBA (23:16) */ 16387 fis->d.device = 0x40; /* FIS LBA mode set */ 16388 fis->d.lbaLowExp = 0; /* FIS LBA (31:24) */ 16389 fis->d.lbaMidExp = 0; /* FIS LBA (39:32) */ 16390 fis->d.lbaHighExp = 0; /* FIS LBA (47:40) */ 16391 fis->d.featuresExp = 0; /* FIS reserve */ 16392 fis->d.sectorCount = 1; /* FIS sector count (7:0) */ 16393 fis->d.sectorCountExp = 0; /* FIS sector count (15:8) */ 16394 fis->d.reserved4 = 0; 16395 fis->d.control = 0; /* FIS HOB bit clear */ 16396 fis->d.reserved5 = 0; 16397 16398 agRequestType = AGSA_SATA_PROTOCOL_PIO_READ; 16399 } 16400 else 16401 { 16402 /* READ VERIFY SECTORS */ 16403 fis->h.fisType = 0x27; /* Reg host to device */ 16404 fis->h.c_pmPort = 0x80; /* C Bit is set */ 16405 fis->h.command = SAT_READ_SECTORS; /* 0x20 */ 16406 fis->h.features = 0; /* FIS reserve */ 16407 fis->d.lbaLow = 0; /* FIS LBA (7 :0 ) */ 16408 fis->d.lbaMid = 0; /* FIS LBA (15:8 ) */ 16409 fis->d.lbaHigh = 0; /* FIS LBA (23:16) */ 16410 fis->d.device = 0x40; /* FIS LBA (27:24) and FIS LBA mode */ 16411 fis->d.lbaLowExp = 0; 16412 fis->d.lbaMidExp = 0; 16413 fis->d.lbaHighExp = 0; 16414 fis->d.featuresExp = 0; 16415 fis->d.sectorCount = 1; /* FIS sector count (7:0) */ 16416 fis->d.sectorCountExp = 0; 16417 fis->d.reserved4 = 0; 16418 fis->d.control = 0; /* FIS HOB bit clear */ 16419 fis->d.reserved5 = 0; 16420 16421 16422 agRequestType = AGSA_SATA_PROTOCOL_PIO_READ; 16423 } 16424 satIOContext->satCompleteCB = &smsatReadMediaSerialNumberCB; 16425 satIOContext->reqType = agRequestType; /* Save it */ 16426 status = smsataLLIOStart( smRoot, 16427 smIORequest, 16428 smDeviceHandle, 16429 smScsiRequest, 16430 satIOContext); 16431 16432 return status; 16433 } 16434 } 16435 else 16436 { 16437 16438 tdsmIOCompletedCB( smRoot, 16439 smIORequest, 16440 smIOFailed, 16441 smDetailOtherError, 16442 agNULL, 16443 satIOContext->interruptContext); 16444 16445 return SM_RC_SUCCESS; 16446 16447 } 16448 } 16449 16450 osGLOBAL bit32 16451 smsatReadBuffer( 16452 smRoot_t *smRoot, 16453 smIORequest_t *smIORequest, 16454 smDeviceHandle_t *smDeviceHandle, 16455 smScsiInitiatorRequest_t *smScsiRequest, 16456 smSatIOContext_t *satIOContext 16457 ) 16458 { 16459 bit32 status = SM_RC_SUCCESS; 16460 bit32 agRequestType = AGSA_SATA_PROTOCOL_PIO_READ; 16461 smScsiRspSense_t *pSense; 16462 smIniScsiCmnd_t *scsiCmnd; 16463 agsaFisRegHostToDevice_t *fis; 16464 bit32 bufferOffset; 16465 bit32 tl; 16466 bit8 mode; 16467 bit8 bufferID; 16468 bit8 *pBuff; 16469 16470 pSense = satIOContext->pSense; 16471 scsiCmnd = &smScsiRequest->scsiCmnd; 16472 fis = satIOContext->pFis; 16473 pBuff = (bit8 *) smScsiRequest->sglVirtualAddr; 16474 16475 SM_DBG5(("smsatReadBuffer: start\n")); 16476 16477 /* checking CONTROL */ 16478 /* NACA == 1 or LINK == 1*/ 16479 if ( (scsiCmnd->cdb[9] & SCSI_NACA_MASK) || (scsiCmnd->cdb[9] & SCSI_LINK_MASK) ) 16480 { 16481 smsatSetSensePayload( pSense, 16482 SCSI_SNSKEY_ILLEGAL_REQUEST, 16483 0, 16484 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 16485 satIOContext); 16486 16487 /*smEnqueueIO(smRoot, satIOContext);*/ 16488 16489 tdsmIOCompletedCB( smRoot, 16490 smIORequest, 16491 smIOSuccess, 16492 SCSI_STAT_CHECK_CONDITION, 16493 satIOContext->pSmSenseData, 16494 satIOContext->interruptContext ); 16495 16496 SM_DBG1(("smsatReadBuffer: return control!!!\n")); 16497 return SM_RC_SUCCESS; 16498 } 16499 16500 bufferOffset = (scsiCmnd->cdb[3] << (8*2)) + (scsiCmnd->cdb[4] << 8) + scsiCmnd->cdb[5]; 16501 tl = (scsiCmnd->cdb[6] << (8*2)) + (scsiCmnd->cdb[7] << 8) + scsiCmnd->cdb[8]; 16502 16503 mode = (bit8)(scsiCmnd->cdb[1] & SCSI_READ_BUFFER_MODE_MASK); 16504 bufferID = scsiCmnd->cdb[2]; 16505 16506 if (mode == READ_BUFFER_DATA_MODE) /* 2 */ 16507 { 16508 if (bufferID == 0 && bufferOffset == 0 && tl == 512) 16509 { 16510 /* send ATA READ BUFFER */ 16511 fis->h.fisType = 0x27; /* Reg host to device */ 16512 fis->h.c_pmPort = 0x80; /* C Bit is set */ 16513 fis->h.command = SAT_READ_BUFFER; /* 0xE4 */ 16514 fis->h.features = 0; /* FIS reserve */ 16515 fis->d.lbaLow = 0; /* FIS LBA (7 :0 ) */ 16516 fis->d.lbaMid = 0; /* FIS LBA (15:8 ) */ 16517 fis->d.lbaHigh = 0; /* FIS LBA (23:16) */ 16518 fis->d.device = 0x40; /* FIS LBA (27:24) and FIS LBA mode */ 16519 fis->d.lbaLowExp = 0; 16520 fis->d.lbaMidExp = 0; 16521 fis->d.lbaHighExp = 0; 16522 fis->d.featuresExp = 0; 16523 fis->d.sectorCount = 0; /* FIS sector count (7:0) */ 16524 fis->d.sectorCountExp = 0; 16525 fis->d.reserved4 = 0; 16526 fis->d.control = 0; /* FIS HOB bit clear */ 16527 fis->d.reserved5 = 0; 16528 16529 16530 agRequestType = AGSA_SATA_PROTOCOL_PIO_READ; 16531 16532 satIOContext->satCompleteCB = &smsatReadBufferCB; 16533 16534 satIOContext->reqType = agRequestType; /* Save it */ 16535 16536 status = smsataLLIOStart( smRoot, 16537 smIORequest, 16538 smDeviceHandle, 16539 smScsiRequest, 16540 satIOContext); 16541 return status; 16542 } 16543 16544 if (bufferID == 0 && bufferOffset == 0 && tl != 512) 16545 { 16546 smsatSetSensePayload( pSense, 16547 SCSI_SNSKEY_ILLEGAL_REQUEST, 16548 0, 16549 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 16550 satIOContext); 16551 16552 /*smEnqueueIO(smRoot, satIOContext);*/ 16553 16554 tdsmIOCompletedCB( smRoot, 16555 smIORequest, 16556 smIOSuccess, 16557 SCSI_STAT_CHECK_CONDITION, 16558 satIOContext->pSmSenseData, 16559 satIOContext->interruptContext ); 16560 16561 SM_DBG1(("smsatReadBuffer: allocation length is not 512; it is %d!!!\n", tl)); 16562 return SM_RC_SUCCESS; 16563 } 16564 16565 if (bufferID == 0 && bufferOffset != 0) 16566 { 16567 smsatSetSensePayload( pSense, 16568 SCSI_SNSKEY_ILLEGAL_REQUEST, 16569 0, 16570 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 16571 satIOContext); 16572 16573 /*smEnqueueIO(smRoot, satIOContext);*/ 16574 16575 tdsmIOCompletedCB( smRoot, 16576 smIORequest, 16577 smIOSuccess, 16578 SCSI_STAT_CHECK_CONDITION, 16579 satIOContext->pSmSenseData, 16580 satIOContext->interruptContext ); 16581 16582 SM_DBG1(("smsatReadBuffer: buffer offset is not 0; it is %d!!!\n", bufferOffset)); 16583 return SM_RC_SUCCESS; 16584 } 16585 /* all other cases unsupported */ 16586 SM_DBG1(("smsatReadBuffer: unsupported case 1!!!\n")); 16587 smsatSetSensePayload( pSense, 16588 SCSI_SNSKEY_ILLEGAL_REQUEST, 16589 0, 16590 SCSI_SNSCODE_INVALID_COMMAND, 16591 satIOContext); 16592 16593 /*smEnqueueIO(smRoot, satIOContext);*/ 16594 16595 tdsmIOCompletedCB( smRoot, 16596 smIORequest, 16597 smIOSuccess, 16598 SCSI_STAT_CHECK_CONDITION, 16599 satIOContext->pSmSenseData, 16600 satIOContext->interruptContext ); 16601 16602 return SM_RC_SUCCESS; 16603 16604 } 16605 else if (mode == READ_BUFFER_DESCRIPTOR_MODE) /* 3 */ 16606 { 16607 if (tl < READ_BUFFER_DESCRIPTOR_MODE_DATA_LEN) /* 4 */ 16608 { 16609 smsatSetSensePayload( pSense, 16610 SCSI_SNSKEY_ILLEGAL_REQUEST, 16611 0, 16612 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 16613 satIOContext); 16614 16615 /*smEnqueueIO(smRoot, satIOContext);*/ 16616 16617 tdsmIOCompletedCB( smRoot, 16618 smIORequest, 16619 smIOSuccess, 16620 SCSI_STAT_CHECK_CONDITION, 16621 satIOContext->pSmSenseData, 16622 satIOContext->interruptContext ); 16623 16624 SM_DBG1(("smsatReadBuffer: tl < 4; tl is %d!!!\n", tl)); 16625 return SM_RC_SUCCESS; 16626 } 16627 if (bufferID == 0) 16628 { 16629 /* SPC-4, 6.15.5, p189; SAT-2 Rev00, 8.7.2.3, p41*/ 16630 pBuff[0] = 0xFF; 16631 pBuff[1] = 0x00; 16632 pBuff[2] = 0x02; 16633 pBuff[3] = 0x00; 16634 if (READ_BUFFER_DESCRIPTOR_MODE_DATA_LEN < tl) 16635 { 16636 /* underrrun */ 16637 SM_DBG1(("smsatReadBuffer: underrun tl %d data %d!!!\n", tl, READ_BUFFER_DESCRIPTOR_MODE_DATA_LEN)); 16638 /*smEnqueueIO(smRoot, satIOContext);*/ 16639 16640 tdsmIOCompletedCB( smRoot, 16641 smIORequest, 16642 smIOUnderRun, 16643 tl - READ_BUFFER_DESCRIPTOR_MODE_DATA_LEN, 16644 agNULL, 16645 satIOContext->interruptContext ); 16646 16647 return SM_RC_SUCCESS; 16648 } 16649 else 16650 { 16651 /*smEnqueueIO(smRoot, satIOContext);*/ 16652 16653 tdsmIOCompletedCB( smRoot, 16654 smIORequest, 16655 smIOSuccess, 16656 SCSI_STAT_GOOD, 16657 agNULL, 16658 satIOContext->interruptContext); 16659 return SM_RC_SUCCESS; 16660 } 16661 } 16662 else 16663 { 16664 /* We don't support other than bufferID 0 */ 16665 smsatSetSensePayload( pSense, 16666 SCSI_SNSKEY_ILLEGAL_REQUEST, 16667 0, 16668 SCSI_SNSCODE_INVALID_COMMAND, 16669 satIOContext); 16670 16671 /*smEnqueueIO(smRoot, satIOContext);*/ 16672 16673 tdsmIOCompletedCB( smRoot, 16674 smIORequest, 16675 smIOSuccess, 16676 SCSI_STAT_CHECK_CONDITION, 16677 satIOContext->pSmSenseData, 16678 satIOContext->interruptContext ); 16679 16680 return SM_RC_SUCCESS; 16681 } 16682 } 16683 else 16684 { 16685 /* We don't support any other mode */ 16686 SM_DBG1(("smsatReadBuffer: unsupported mode %d!!!\n", mode)); 16687 smsatSetSensePayload( pSense, 16688 SCSI_SNSKEY_ILLEGAL_REQUEST, 16689 0, 16690 SCSI_SNSCODE_INVALID_COMMAND, 16691 satIOContext); 16692 16693 /*smEnqueueIO(smRoot, satIOContext);*/ 16694 16695 tdsmIOCompletedCB( smRoot, 16696 smIORequest, 16697 smIOSuccess, 16698 SCSI_STAT_CHECK_CONDITION, 16699 satIOContext->pSmSenseData, 16700 satIOContext->interruptContext ); 16701 16702 return SM_RC_SUCCESS; 16703 } 16704 } 16705 16706 osGLOBAL bit32 16707 smsatWriteBuffer( 16708 smRoot_t *smRoot, 16709 smIORequest_t *smIORequest, 16710 smDeviceHandle_t *smDeviceHandle, 16711 smScsiInitiatorRequest_t *smScsiRequest, 16712 smSatIOContext_t *satIOContext 16713 ) 16714 { 16715 #ifdef NOT_YET 16716 bit32 agRequestType = AGSA_SATA_PROTOCOL_PIO_READ; 16717 #endif 16718 smScsiRspSense_t *pSense; 16719 smIniScsiCmnd_t *scsiCmnd; 16720 #ifdef NOT_YET 16721 agsaFisRegHostToDevice_t *fis; 16722 #endif 16723 bit32 bufferOffset; 16724 bit32 parmLen; 16725 bit8 mode; 16726 bit8 bufferID; 16727 bit8 *pBuff; 16728 16729 pSense = satIOContext->pSense; 16730 scsiCmnd = &smScsiRequest->scsiCmnd; 16731 #ifdef NOT_YET 16732 fis = satIOContext->pFis; 16733 #endif 16734 pBuff = (bit8 *) smScsiRequest->sglVirtualAddr; 16735 16736 SM_DBG5(("smsatWriteBuffer: start\n")); 16737 16738 /* checking CONTROL */ 16739 /* NACA == 1 or LINK == 1*/ 16740 if ( (scsiCmnd->cdb[9] & SCSI_NACA_MASK) || (scsiCmnd->cdb[9] & SCSI_LINK_MASK) ) 16741 { 16742 smsatSetSensePayload( pSense, 16743 SCSI_SNSKEY_ILLEGAL_REQUEST, 16744 0, 16745 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 16746 satIOContext); 16747 16748 /*smEnqueueIO(smRoot, satIOContext);*/ 16749 16750 tdsmIOCompletedCB( smRoot, 16751 smIORequest, 16752 smIOSuccess, 16753 SCSI_STAT_CHECK_CONDITION, 16754 satIOContext->pSmSenseData, 16755 satIOContext->interruptContext ); 16756 16757 SM_DBG1(("smsatWriteBuffer: return control!!!\n")); 16758 return SM_RC_SUCCESS; 16759 } 16760 16761 bufferOffset = (scsiCmnd->cdb[3] << (8*2)) + (scsiCmnd->cdb[4] << 8) + scsiCmnd->cdb[5]; 16762 parmLen = (scsiCmnd->cdb[6] << (8*2)) + (scsiCmnd->cdb[7] << 8) + scsiCmnd->cdb[8]; 16763 16764 mode = (bit8)(scsiCmnd->cdb[1] & SCSI_READ_BUFFER_MODE_MASK); 16765 bufferID = scsiCmnd->cdb[2]; 16766 16767 /* for debugging only */ 16768 smhexdump("smsatWriteBuffer pBuff", (bit8 *)pBuff, 24); 16769 16770 if (mode == WRITE_BUFFER_DATA_MODE) /* 2 */ 16771 { 16772 if (bufferID == 0 && bufferOffset == 0 && parmLen == 512) 16773 { 16774 SM_DBG1(("smsatWriteBuffer: sending ATA WRITE BUFFER!!!\n")); 16775 /* send ATA WRITE BUFFER */ 16776 #ifdef NOT_YET 16777 fis->h.fisType = 0x27; /* Reg host to device */ 16778 fis->h.c_pmPort = 0x80; /* C Bit is set */ 16779 fis->h.command = SAT_WRITE_BUFFER; /* 0xE8 */ 16780 fis->h.features = 0; /* FIS reserve */ 16781 fis->d.lbaLow = 0; /* FIS LBA (7 :0 ) */ 16782 fis->d.lbaMid = 0; /* FIS LBA (15:8 ) */ 16783 fis->d.lbaHigh = 0; /* FIS LBA (23:16) */ 16784 fis->d.device = 0x40; /* FIS LBA (27:24) and FIS LBA mode */ 16785 fis->d.lbaLowExp = 0; 16786 fis->d.lbaMidExp = 0; 16787 fis->d.lbaHighExp = 0; 16788 fis->d.featuresExp = 0; 16789 fis->d.sectorCount = 0; /* FIS sector count (7:0) */ 16790 fis->d.sectorCountExp = 0; 16791 fis->d.reserved4 = 0; 16792 fis->d.control = 0; /* FIS HOB bit clear */ 16793 fis->d.reserved5 = 0; 16794 16795 16796 agRequestType = AGSA_SATA_PROTOCOL_PIO_WRITE; 16797 16798 satIOContext->satCompleteCB = &smsatWriteBufferCB; 16799 16800 satIOContext->reqType = agRequestType; /* Save it */ 16801 16802 status = smsataLLIOStart( smRoot, 16803 smIORequest, 16804 smDeviceHandle, 16805 smScsiRequest, 16806 satIOContext); 16807 return status; 16808 #endif 16809 /* temp */ 16810 /*smEnqueueIO(smRoot, satIOContext);*/ 16811 16812 tdsmIOCompletedCB( smRoot, 16813 smIORequest, 16814 smIOSuccess, 16815 SCSI_STAT_GOOD, 16816 agNULL, 16817 satIOContext->interruptContext); 16818 return SM_RC_SUCCESS; 16819 } 16820 if ( (bufferID == 0 && bufferOffset != 0) || 16821 (bufferID == 0 && parmLen != 512) 16822 ) 16823 { 16824 smsatSetSensePayload( pSense, 16825 SCSI_SNSKEY_ILLEGAL_REQUEST, 16826 0, 16827 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 16828 satIOContext); 16829 16830 /*smEnqueueIO(smRoot, satIOContext);*/ 16831 16832 tdsmIOCompletedCB( smRoot, 16833 smIORequest, 16834 smIOSuccess, 16835 SCSI_STAT_CHECK_CONDITION, 16836 satIOContext->pSmSenseData, 16837 satIOContext->interruptContext ); 16838 16839 SM_DBG1(("smsatWriteBuffer: wrong buffer offset %d or parameter length parmLen %d!!!\n", bufferOffset, parmLen)); 16840 return SM_RC_SUCCESS; 16841 } 16842 16843 /* all other cases unsupported */ 16844 SM_DBG1(("smsatWriteBuffer: unsupported case 1!!!\n")); 16845 smsatSetSensePayload( pSense, 16846 SCSI_SNSKEY_ILLEGAL_REQUEST, 16847 0, 16848 SCSI_SNSCODE_INVALID_COMMAND, 16849 satIOContext); 16850 16851 /*smEnqueueIO(smRoot, satIOContext);*/ 16852 16853 tdsmIOCompletedCB( smRoot, 16854 smIORequest, 16855 smIOSuccess, 16856 SCSI_STAT_CHECK_CONDITION, 16857 satIOContext->pSmSenseData, 16858 satIOContext->interruptContext ); 16859 16860 return SM_RC_SUCCESS; 16861 16862 } 16863 else if (mode == WRITE_BUFFER_DL_MICROCODE_SAVE_MODE) /* 5 */ 16864 { 16865 /* temporary */ 16866 SM_DBG1(("smsatWriteBuffer: not yet supported mode %d!!!\n", mode)); 16867 smsatSetSensePayload( pSense, 16868 SCSI_SNSKEY_ILLEGAL_REQUEST, 16869 0, 16870 SCSI_SNSCODE_INVALID_COMMAND, 16871 satIOContext); 16872 16873 16874 tdsmIOCompletedCB( smRoot, 16875 smIORequest, 16876 smIOSuccess, 16877 SCSI_STAT_CHECK_CONDITION, 16878 satIOContext->pSmSenseData, 16879 satIOContext->interruptContext ); 16880 16881 return SM_RC_SUCCESS; 16882 } 16883 else 16884 { 16885 /* We don't support any other mode */ 16886 SM_DBG1(("smsatWriteBuffer: unsupported mode %d!!!\n", mode)); 16887 smsatSetSensePayload( pSense, 16888 SCSI_SNSKEY_ILLEGAL_REQUEST, 16889 0, 16890 SCSI_SNSCODE_INVALID_COMMAND, 16891 satIOContext); 16892 16893 /*smEnqueueIO(smRoot, satIOContext);*/ 16894 16895 tdsmIOCompletedCB( smRoot, 16896 smIORequest, 16897 smIOSuccess, 16898 SCSI_STAT_CHECK_CONDITION, 16899 satIOContext->pSmSenseData, 16900 satIOContext->interruptContext ); 16901 16902 return SM_RC_SUCCESS; 16903 } 16904 16905 } 16906 16907 osGLOBAL bit32 16908 smsatReassignBlocks( 16909 smRoot_t *smRoot, 16910 smIORequest_t *smIORequest, 16911 smDeviceHandle_t *smDeviceHandle, 16912 smScsiInitiatorRequest_t *smScsiRequest, 16913 smSatIOContext_t *satIOContext 16914 ) 16915 { 16916 /* 16917 assumes all LBA fits in ATA command; no boundary condition is checked here yet 16918 */ 16919 bit32 status; 16920 bit32 agRequestType; 16921 smDeviceData_t *pSatDevData; 16922 smScsiRspSense_t *pSense; 16923 smIniScsiCmnd_t *scsiCmnd; 16924 agsaFisRegHostToDevice_t *fis; 16925 bit8 *pParmList; /* Log Page data buffer */ 16926 bit8 LongLBA; 16927 bit8 LongList; 16928 bit32 defectListLen; 16929 bit8 LBA[8]; 16930 bit32 startingIndex; 16931 16932 pSense = satIOContext->pSense; 16933 pSatDevData = satIOContext->pSatDevData; 16934 scsiCmnd = &smScsiRequest->scsiCmnd; 16935 fis = satIOContext->pFis; 16936 pParmList = (bit8 *) smScsiRequest->sglVirtualAddr; 16937 16938 SM_DBG5(("smsatReassignBlocks: start\n")); 16939 16940 /* checking CONTROL */ 16941 /* NACA == 1 or LINK == 1*/ 16942 if ( (scsiCmnd->cdb[5] & SCSI_NACA_MASK) || (scsiCmnd->cdb[5] & SCSI_LINK_MASK) ) 16943 { 16944 smsatSetSensePayload( pSense, 16945 SCSI_SNSKEY_ILLEGAL_REQUEST, 16946 0, 16947 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 16948 satIOContext); 16949 16950 /*smEnqueueIO(smRoot, satIOContext);*/ 16951 16952 tdsmIOCompletedCB( smRoot, 16953 smIORequest, 16954 smIOSuccess, 16955 SCSI_STAT_CHECK_CONDITION, 16956 satIOContext->pSmSenseData, 16957 satIOContext->interruptContext ); 16958 16959 SM_DBG1(("smsatReassignBlocks: return control!!!\n")); 16960 return SM_RC_SUCCESS; 16961 } 16962 16963 sm_memset(satIOContext->LBA, 0, 8); 16964 satIOContext->ParmIndex = 0; 16965 satIOContext->ParmLen = 0; 16966 16967 LongList = (bit8)(scsiCmnd->cdb[1] & SCSI_REASSIGN_BLOCKS_LONGLIST_MASK); 16968 LongLBA = (bit8)(scsiCmnd->cdb[1] & SCSI_REASSIGN_BLOCKS_LONGLBA_MASK); 16969 sm_memset(LBA, 0, sizeof(LBA)); 16970 16971 if (LongList == 0) 16972 { 16973 defectListLen = (pParmList[2] << 8) + pParmList[3]; 16974 } 16975 else 16976 { 16977 defectListLen = (pParmList[0] << (8*3)) + (pParmList[1] << (8*2)) 16978 + (pParmList[2] << 8) + pParmList[3]; 16979 } 16980 /* SBC 5.16.2, p61*/ 16981 satIOContext->ParmLen = defectListLen + 4 /* header size */; 16982 16983 startingIndex = 4; 16984 16985 if (LongLBA == 0) 16986 { 16987 LBA[4] = pParmList[startingIndex]; /* MSB */ 16988 LBA[5] = pParmList[startingIndex+1]; 16989 LBA[6] = pParmList[startingIndex+2]; 16990 LBA[7] = pParmList[startingIndex+3]; /* LSB */ 16991 startingIndex = startingIndex + 4; 16992 } 16993 else 16994 { 16995 LBA[0] = pParmList[startingIndex]; /* MSB */ 16996 LBA[1] = pParmList[startingIndex+1]; 16997 LBA[2] = pParmList[startingIndex+2]; 16998 LBA[3] = pParmList[startingIndex+3]; 16999 LBA[4] = pParmList[startingIndex+4]; 17000 LBA[5] = pParmList[startingIndex+5]; 17001 LBA[6] = pParmList[startingIndex+6]; 17002 LBA[7] = pParmList[startingIndex+7]; /* LSB */ 17003 startingIndex = startingIndex + 8; 17004 } 17005 17006 smhexdump("smsatReassignBlocks Parameter list", (bit8 *)pParmList, 4 + defectListLen); 17007 17008 if (pSatDevData->sat48BitSupport == agTRUE) 17009 { 17010 /* sends READ VERIFY SECTOR(S) EXT*/ 17011 fis->h.fisType = 0x27; /* Reg host to device */ 17012 fis->h.c_pmPort = 0x80; /* C Bit is set */ 17013 fis->h.command = SAT_READ_VERIFY_SECTORS_EXT;/* 0x42 */ 17014 fis->h.features = 0; /* FIS reserve */ 17015 fis->d.lbaLow = LBA[7]; /* FIS LBA (7 :0 ) */ 17016 fis->d.lbaMid = LBA[6]; /* FIS LBA (15:8 ) */ 17017 fis->d.lbaHigh = LBA[5]; /* FIS LBA (23:16) */ 17018 fis->d.lbaLowExp = LBA[4]; /* FIS LBA (31:24) */ 17019 fis->d.lbaMidExp = LBA[3]; /* FIS LBA (39:32) */ 17020 fis->d.lbaHighExp = LBA[2]; /* FIS LBA (47:40) */ 17021 fis->d.featuresExp = 0; /* FIS reserve */ 17022 fis->d.sectorCount = 1; /* FIS sector count (7:0) */ 17023 fis->d.sectorCountExp = 0; /* FIS sector count (15:8) */ 17024 fis->d.reserved4 = 0; 17025 fis->d.device = 0x40; /* 01000000 */ 17026 fis->d.control = 0; /* FIS HOB bit clear */ 17027 fis->d.reserved5 = 0; 17028 } 17029 else 17030 { 17031 /* READ VERIFY SECTOR(S)*/ 17032 fis->h.fisType = 0x27; /* Reg host to device */ 17033 fis->h.c_pmPort = 0x80; /* C Bit is set */ 17034 fis->h.command = SAT_READ_VERIFY_SECTORS;/* 0x40 */ 17035 fis->h.features = 0; /* FIS features NA */ 17036 fis->d.lbaLow = LBA[7]; /* FIS LBA (7 :0 ) */ 17037 fis->d.lbaMid = LBA[6]; /* FIS LBA (15:8 ) */ 17038 fis->d.lbaHigh = LBA[5]; /* FIS LBA (23:16) */ 17039 fis->d.lbaLowExp = 0; 17040 fis->d.lbaMidExp = 0; 17041 fis->d.lbaHighExp = 0; 17042 fis->d.featuresExp = 0; 17043 fis->d.sectorCount = 1; /* FIS sector count (7:0) */ 17044 fis->d.sectorCountExp = 0; 17045 fis->d.reserved4 = 0; 17046 fis->d.device = (bit8)((0x4 << 4) | (LBA[4] & 0xF)); 17047 /* DEV and LBA 27:24 */ 17048 fis->d.control = 0; /* FIS HOB bit clear */ 17049 fis->d.reserved5 = 0; 17050 } 17051 17052 sm_memcpy(satIOContext->LBA, LBA, 8); 17053 satIOContext->ParmIndex = startingIndex; 17054 17055 agRequestType = AGSA_SATA_PROTOCOL_NON_DATA; 17056 17057 /* Initialize CB for SATA completion. 17058 */ 17059 satIOContext->satCompleteCB = &smsatReassignBlocksCB; 17060 17061 /* 17062 * Prepare SGL and send FIS to LL layer. 17063 */ 17064 satIOContext->reqType = agRequestType; /* Save it */ 17065 17066 status = smsataLLIOStart( smRoot, 17067 smIORequest, 17068 smDeviceHandle, 17069 smScsiRequest, 17070 satIOContext); 17071 17072 return status; 17073 } 17074 17075 osGLOBAL bit32 17076 smsatRead_1( 17077 smRoot_t *smRoot, 17078 smIORequest_t *smIORequest, 17079 smDeviceHandle_t *smDeviceHandle, 17080 smScsiInitiatorRequest_t *smScsiRequest, 17081 smSatIOContext_t *satIOContext 17082 ) 17083 { 17084 /* 17085 Assumption: error check on lba and tl has been done in satRead*() 17086 lba = lba + tl; 17087 */ 17088 bit32 status; 17089 smSatIOContext_t *satOrgIOContext = agNULL; 17090 smIniScsiCmnd_t *scsiCmnd; 17091 agsaFisRegHostToDevice_t *fis; 17092 bit32 agRequestType = AGSA_SATA_PROTOCOL_DMA_READ; 17093 bit32 lba = 0; 17094 bit32 DenomTL = 0xFF; 17095 bit32 Remainder = 0; 17096 bit8 LBA[4]; /* 0 MSB, 3 LSB */ 17097 17098 SM_DBG2(("smsatRead_1: start\n")); 17099 17100 fis = satIOContext->pFis; 17101 satOrgIOContext = satIOContext->satOrgIOContext; 17102 scsiCmnd = satOrgIOContext->pScsiCmnd; 17103 17104 sm_memset(LBA,0, sizeof(LBA)); 17105 17106 switch (satOrgIOContext->ATACmd) 17107 { 17108 case SAT_READ_DMA: 17109 DenomTL = 0x100; 17110 break; 17111 case SAT_READ_SECTORS: 17112 DenomTL = 0x100; 17113 break; 17114 case SAT_READ_DMA_EXT: 17115 DenomTL = 0xFFFF; 17116 break; 17117 case SAT_READ_SECTORS_EXT: 17118 DenomTL = 0xFFFF; 17119 break; 17120 case SAT_READ_FPDMA_QUEUED: 17121 DenomTL = 0xFFFF; 17122 break; 17123 default: 17124 SM_DBG1(("smsatRead_1: error incorrect ata command 0x%x!!!\n", satIOContext->ATACmd)); 17125 return SM_RC_FAILURE; 17126 break; 17127 } 17128 17129 Remainder = satOrgIOContext->OrgTL % DenomTL; 17130 satOrgIOContext->currentLBA = satOrgIOContext->currentLBA + DenomTL; 17131 lba = satOrgIOContext->currentLBA; 17132 17133 LBA[0] = (bit8)((lba & 0xFF000000) >> (8 * 3)); 17134 LBA[1] = (bit8)((lba & 0xFF0000) >> (8 * 2)); 17135 LBA[2] = (bit8)((lba & 0xFF00) >> 8); 17136 LBA[3] = (bit8)(lba & 0xFF); 17137 17138 switch (satOrgIOContext->ATACmd) 17139 { 17140 case SAT_READ_DMA: 17141 fis->h.fisType = 0x27; /* Reg host to device */ 17142 fis->h.c_pmPort = 0x80; /* C Bit is set */ 17143 fis->h.command = SAT_READ_DMA; /* 0xC8 */ 17144 fis->h.features = 0; /* FIS reserve */ 17145 fis->d.lbaLow = LBA[3]; /* FIS LBA (7 :0 ) */ 17146 fis->d.lbaMid = LBA[2]; /* FIS LBA (15:8 ) */ 17147 fis->d.lbaHigh = LBA[1]; /* FIS LBA (23:16) */ 17148 fis->d.device = 17149 (bit8)((0x4 << 4) | (LBA[0] & 0xF)); /* FIS LBA (27:24) and FIS LBA mode */ 17150 fis->d.lbaLowExp = 0; 17151 fis->d.lbaMidExp = 0; 17152 fis->d.lbaHighExp = 0; 17153 fis->d.featuresExp = 0; 17154 17155 if (satOrgIOContext->LoopNum == 1) 17156 { 17157 /* last loop */ 17158 fis->d.sectorCount = (bit8)Remainder; /* FIS sector count (7:0) */ 17159 } 17160 else 17161 { 17162 fis->d.sectorCount = 0x0; /* FIS sector count (7:0) */ 17163 } 17164 17165 fis->d.sectorCountExp = 0; 17166 fis->d.reserved4 = 0; 17167 fis->d.control = 0; /* FIS HOB bit clear */ 17168 fis->d.reserved5 = 0; 17169 17170 agRequestType = AGSA_SATA_PROTOCOL_DMA_READ; 17171 17172 break; 17173 case SAT_READ_SECTORS: 17174 fis->h.fisType = 0x27; /* Reg host to device */ 17175 fis->h.c_pmPort = 0x80; /* C Bit is set */ 17176 fis->h.command = SAT_READ_SECTORS; /* 0x20 */ 17177 fis->h.features = 0; /* FIS reserve */ 17178 fis->d.lbaLow = LBA[3]; /* FIS LBA (7 :0 ) */ 17179 fis->d.lbaMid = LBA[2]; /* FIS LBA (15:8 ) */ 17180 fis->d.lbaHigh = LBA[1]; /* FIS LBA (23:16) */ 17181 fis->d.device = 17182 (bit8)((0x4 << 4) | (LBA[0] & 0xF)); /* FIS LBA (27:24) and FIS LBA mode */ 17183 fis->d.lbaLowExp = 0; 17184 fis->d.lbaMidExp = 0; 17185 fis->d.lbaHighExp = 0; 17186 fis->d.featuresExp = 0; 17187 if (satOrgIOContext->LoopNum == 1) 17188 { 17189 /* last loop */ 17190 fis->d.sectorCount = (bit8)Remainder; /* FIS sector count (7:0) */ 17191 } 17192 else 17193 { 17194 fis->d.sectorCount = 0x0; /* FIS sector count (7:0) */ 17195 } 17196 fis->d.sectorCountExp = 0; 17197 fis->d.reserved4 = 0; 17198 fis->d.control = 0; /* FIS HOB bit clear */ 17199 fis->d.reserved5 = 0; 17200 17201 agRequestType = AGSA_SATA_PROTOCOL_PIO_READ; 17202 17203 break; 17204 case SAT_READ_DMA_EXT: 17205 fis->h.fisType = 0x27; /* Reg host to device */ 17206 fis->h.c_pmPort = 0x80; /* C Bit is set */ 17207 fis->h.command = SAT_READ_DMA_EXT; /* 0x25 */ 17208 fis->h.features = 0; /* FIS reserve */ 17209 fis->d.lbaLow = LBA[3]; /* FIS LBA (7 :0 ) */ 17210 fis->d.lbaMid = LBA[2]; /* FIS LBA (15:8 ) */ 17211 fis->d.lbaHigh = LBA[1]; /* FIS LBA (23:16) */ 17212 fis->d.device = 0x40; /* FIS LBA mode set */ 17213 fis->d.lbaLowExp = LBA[0]; /* FIS LBA (31:24) */ 17214 fis->d.lbaMidExp = 0; /* FIS LBA (39:32) */ 17215 fis->d.lbaHighExp = 0; /* FIS LBA (47:40) */ 17216 fis->d.featuresExp = 0; /* FIS reserve */ 17217 if (satOrgIOContext->LoopNum == 1) 17218 { 17219 /* last loop */ 17220 fis->d.sectorCount = (bit8)(Remainder & 0xFF); /* FIS sector count (7:0) */ 17221 fis->d.sectorCountExp = (bit8)((Remainder & 0xFF00) >> 8); /* FIS sector count (15:8) */ 17222 17223 } 17224 else 17225 { 17226 fis->d.sectorCount = 0xFF; /* FIS sector count (7:0) */ 17227 fis->d.sectorCountExp = 0xFF; /* FIS sector count (15:8) */ 17228 } 17229 fis->d.reserved4 = 0; 17230 fis->d.control = 0; /* FIS HOB bit clear */ 17231 fis->d.reserved5 = 0; 17232 17233 agRequestType = AGSA_SATA_PROTOCOL_DMA_READ; 17234 17235 break; 17236 case SAT_READ_SECTORS_EXT: 17237 fis->h.fisType = 0x27; /* Reg host to device */ 17238 fis->h.c_pmPort = 0x80; /* C Bit is set */ 17239 fis->h.command = SAT_READ_SECTORS_EXT; /* 0x24 */ 17240 fis->h.features = 0; /* FIS reserve */ 17241 fis->d.lbaLow = LBA[3]; /* FIS LBA (7 :0 ) */ 17242 fis->d.lbaMid = LBA[2]; /* FIS LBA (15:8 ) */ 17243 fis->d.lbaHigh = LBA[1]; /* FIS LBA (23:16) */ 17244 fis->d.device = 0x40; /* FIS LBA mode set */ 17245 fis->d.lbaLowExp = LBA[0]; /* FIS LBA (31:24) */ 17246 fis->d.lbaMidExp = 0; /* FIS LBA (39:32) */ 17247 fis->d.lbaHighExp = 0; /* FIS LBA (47:40) */ 17248 fis->d.featuresExp = 0; /* FIS reserve */ 17249 if (satOrgIOContext->LoopNum == 1) 17250 { 17251 /* last loop */ 17252 fis->d.sectorCount = (bit8)(Remainder & 0xFF); /* FIS sector count (7:0) */ 17253 fis->d.sectorCountExp = (bit8)((Remainder & 0xFF00) >> 8); /* FIS sector count (15:8) */ 17254 } 17255 else 17256 { 17257 fis->d.sectorCount = 0xFF; /* FIS sector count (7:0) */ 17258 fis->d.sectorCountExp = 0xFF; /* FIS sector count (15:8) */ 17259 } 17260 fis->d.reserved4 = 0; 17261 fis->d.control = 0; /* FIS HOB bit clear */ 17262 fis->d.reserved5 = 0; 17263 17264 agRequestType = AGSA_SATA_PROTOCOL_PIO_READ; 17265 break; 17266 case SAT_READ_FPDMA_QUEUED: 17267 fis->h.fisType = 0x27; /* Reg host to device */ 17268 fis->h.c_pmPort = 0x80; /* C Bit is set */ 17269 fis->h.command = SAT_READ_FPDMA_QUEUED; /* 0x60 */ 17270 fis->d.lbaLow = LBA[3]; /* FIS LBA (7 :0 ) */ 17271 fis->d.lbaMid = LBA[2]; /* FIS LBA (15:8 ) */ 17272 fis->d.lbaHigh = LBA[1]; /* FIS LBA (23:16) */ 17273 17274 /* Check FUA bit */ 17275 if (scsiCmnd->cdb[1] & SCSI_READ10_FUA_MASK) 17276 fis->d.device = 0xC0; /* FIS FUA set */ 17277 else 17278 fis->d.device = 0x40; /* FIS FUA clear */ 17279 17280 fis->d.lbaLowExp = LBA[0]; /* FIS LBA (31:24) */ 17281 fis->d.lbaMidExp = 0; /* FIS LBA (39:32) */ 17282 fis->d.lbaHighExp = 0; /* FIS LBA (47:40) */ 17283 if (satOrgIOContext->LoopNum == 1) 17284 { 17285 /* last loop */ 17286 fis->h.features = (bit8)(Remainder & 0xFF); /* FIS sector count (7:0) */ 17287 fis->d.featuresExp = (bit8)((Remainder & 0xFF00) >> 8); /* FIS sector count (15:8) */ 17288 } 17289 else 17290 { 17291 fis->h.features = 0xFF; /* FIS sector count (7:0) */ 17292 fis->d.featuresExp = 0xFF; /* FIS sector count (15:8) */ 17293 } 17294 fis->d.sectorCount = 0; /* Tag (7:3) set by LL layer */ 17295 fis->d.sectorCountExp = 0; 17296 fis->d.reserved4 = 0; 17297 fis->d.control = 0; /* FIS HOB bit clear */ 17298 fis->d.reserved5 = 0; 17299 17300 agRequestType = AGSA_SATA_PROTOCOL_FPDMA_READ; 17301 break; 17302 default: 17303 SM_DBG1(("smsatRead_1: error incorrect ata command 0x%x!!!\n", satIOContext->ATACmd)); 17304 return SM_RC_FAILURE; 17305 break; 17306 } 17307 17308 /* Initialize CB for SATA completion. 17309 */ 17310 /* chained data */ 17311 satIOContext->satCompleteCB = &smsatChainedDataIOCB; 17312 17313 if (satOrgIOContext->ATACmd == SAT_READ_DMA || satOrgIOContext->ATACmd == SAT_READ_SECTORS) 17314 { 17315 smsatSplitSGL(smRoot, 17316 smIORequest, 17317 smDeviceHandle, 17318 (smScsiInitiatorRequest_t *)satOrgIOContext->smScsiXchg, 17319 satOrgIOContext, 17320 NON_BIT48_ADDRESS_TL_LIMIT * SATA_SECTOR_SIZE, /* 0x100 * 0x200*/ 17321 (satOrgIOContext->OrgTL) * SATA_SECTOR_SIZE, 17322 agFALSE); 17323 } 17324 else 17325 { 17326 smsatSplitSGL(smRoot, 17327 smIORequest, 17328 smDeviceHandle, 17329 (smScsiInitiatorRequest_t *)satOrgIOContext->smScsiXchg, 17330 satOrgIOContext, 17331 BIT48_ADDRESS_TL_LIMIT * SATA_SECTOR_SIZE, /* 0xFFFF * 0x200*/ 17332 (satOrgIOContext->OrgTL) * SATA_SECTOR_SIZE, 17333 agFALSE); 17334 } 17335 17336 /* 17337 * Prepare SGL and send FIS to LL layer. 17338 */ 17339 satIOContext->reqType = agRequestType; /* Save it */ 17340 17341 status = smsataLLIOStart( smRoot, 17342 smIORequest, 17343 smDeviceHandle, 17344 (smScsiInitiatorRequest_t *)satOrgIOContext->smScsiXchg, //smScsiRequest, 17345 satIOContext); 17346 17347 SM_DBG5(("smsatRead_1: return\n")); 17348 return (status); 17349 } 17350 17351 osGLOBAL bit32 17352 smsatWrite_1( 17353 smRoot_t *smRoot, 17354 smIORequest_t *smIORequest, 17355 smDeviceHandle_t *smDeviceHandle, 17356 smScsiInitiatorRequest_t *smScsiRequest, 17357 smSatIOContext_t *satIOContext 17358 ) 17359 { 17360 /* 17361 Assumption: error check on lba and tl has been done in satWrite*() 17362 lba = lba + tl; 17363 */ 17364 bit32 status; 17365 smSatIOContext_t *satOrgIOContext = agNULL; 17366 smIniScsiCmnd_t *scsiCmnd; 17367 agsaFisRegHostToDevice_t *fis; 17368 bit32 agRequestType = AGSA_SATA_PROTOCOL_DMA_WRITE; 17369 bit32 lba = 0; 17370 bit32 DenomTL = 0xFF; 17371 bit32 Remainder = 0; 17372 bit8 LBA[4]; /* 0 MSB, 3 LSB */ 17373 17374 SM_DBG2(("smsatWrite_1: start\n")); 17375 17376 fis = satIOContext->pFis; 17377 satOrgIOContext = satIOContext->satOrgIOContext; 17378 scsiCmnd = satOrgIOContext->pScsiCmnd; 17379 17380 sm_memset(LBA,0, sizeof(LBA)); 17381 17382 switch (satOrgIOContext->ATACmd) 17383 { 17384 case SAT_WRITE_DMA: 17385 DenomTL = 0x100; 17386 break; 17387 case SAT_WRITE_SECTORS: 17388 DenomTL = 0x100; 17389 break; 17390 case SAT_WRITE_DMA_EXT: 17391 DenomTL = 0xFFFF; 17392 break; 17393 case SAT_WRITE_DMA_FUA_EXT: 17394 DenomTL = 0xFFFF; 17395 break; 17396 case SAT_WRITE_SECTORS_EXT: 17397 DenomTL = 0xFFFF; 17398 break; 17399 case SAT_WRITE_FPDMA_QUEUED: 17400 DenomTL = 0xFFFF; 17401 break; 17402 default: 17403 SM_DBG1(("smsatWrite_1: error incorrect ata command 0x%x!!!\n", satIOContext->ATACmd)); 17404 return SM_RC_FAILURE; 17405 break; 17406 } 17407 17408 Remainder = satOrgIOContext->OrgTL % DenomTL; 17409 satOrgIOContext->currentLBA = satOrgIOContext->currentLBA + DenomTL; 17410 lba = satOrgIOContext->currentLBA; 17411 17412 17413 LBA[0] = (bit8)((lba & 0xFF000000) >> (8 * 3)); 17414 LBA[1] = (bit8)((lba & 0xFF0000) >> (8 * 2)); 17415 LBA[2] = (bit8)((lba & 0xFF00) >> 8); 17416 LBA[3] = (bit8)(lba & 0xFF); 17417 17418 switch (satOrgIOContext->ATACmd) 17419 { 17420 case SAT_WRITE_DMA: 17421 fis->h.fisType = 0x27; /* Reg host to device */ 17422 fis->h.c_pmPort = 0x80; /* C bit is set */ 17423 fis->h.command = SAT_WRITE_DMA; /* 0xCA */ 17424 fis->h.features = 0; /* FIS reserve */ 17425 fis->d.lbaLow = LBA[3]; /* FIS LBA (7 :0 ) */ 17426 fis->d.lbaMid = LBA[2]; /* FIS LBA (15:8 ) */ 17427 fis->d.lbaHigh = LBA[1]; /* FIS LBA (23:16) */ 17428 17429 /* FIS LBA mode set LBA (27:24) */ 17430 fis->d.device = (bit8)((0x4 << 4) | (LBA[0] & 0xF)); 17431 17432 fis->d.lbaLowExp = 0; 17433 fis->d.lbaMidExp = 0; 17434 fis->d.lbaHighExp = 0; 17435 fis->d.featuresExp = 0; 17436 if (satOrgIOContext->LoopNum == 1) 17437 { 17438 /* last loop */ 17439 fis->d.sectorCount = (bit8)Remainder; /* FIS sector count (7:0) */ 17440 } 17441 else 17442 { 17443 fis->d.sectorCount = 0x0; /* FIS sector count (7:0) */ 17444 } 17445 fis->d.sectorCountExp = 0; 17446 fis->d.reserved4 = 0; 17447 fis->d.control = 0; /* FIS HOB bit clear */ 17448 fis->d.reserved5 = 0; 17449 17450 agRequestType = AGSA_SATA_PROTOCOL_DMA_WRITE; 17451 17452 break; 17453 case SAT_WRITE_SECTORS: 17454 fis->h.fisType = 0x27; /* Reg host to device */ 17455 fis->h.c_pmPort = 0x80; /* C bit is set */ 17456 fis->h.command = SAT_WRITE_SECTORS; /* 0x30 */ 17457 fis->h.features = 0; /* FIS reserve */ 17458 fis->d.lbaLow = LBA[3]; /* FIS LBA (7 :0 ) */ 17459 fis->d.lbaMid = LBA[2]; /* FIS LBA (15:8 ) */ 17460 fis->d.lbaHigh = LBA[1]; /* FIS LBA (23:16) */ 17461 17462 /* FIS LBA mode set LBA (27:24) */ 17463 fis->d.device = (bit8)((0x4 << 4) | (LBA[0] & 0xF)); 17464 17465 fis->d.lbaLowExp = 0; 17466 fis->d.lbaMidExp = 0; 17467 fis->d.lbaHighExp = 0; 17468 fis->d.featuresExp = 0; 17469 if (satOrgIOContext->LoopNum == 1) 17470 { 17471 /* last loop */ 17472 fis->d.sectorCount = (bit8)Remainder; /* FIS sector count (7:0) */ 17473 } 17474 else 17475 { 17476 fis->d.sectorCount = 0x0; /* FIS sector count (7:0) */ 17477 } 17478 fis->d.sectorCountExp = 0; 17479 fis->d.reserved4 = 0; 17480 fis->d.control = 0; /* FIS HOB bit clear */ 17481 fis->d.reserved5 = 0; 17482 17483 agRequestType = AGSA_SATA_PROTOCOL_PIO_WRITE; 17484 17485 break; 17486 case SAT_WRITE_DMA_EXT: 17487 fis->h.fisType = 0x27; /* Reg host to device */ 17488 fis->h.c_pmPort = 0x80; /* C Bit is set */ 17489 fis->h.command = SAT_WRITE_DMA_EXT; /* 0x3D */ 17490 fis->h.features = 0; /* FIS reserve */ 17491 fis->d.lbaLow = LBA[3]; /* FIS LBA (7 :0 ) */ 17492 fis->d.lbaMid = LBA[2]; /* FIS LBA (15:8 ) */ 17493 fis->d.lbaHigh = LBA[1]; /* FIS LBA (23:16) */ 17494 fis->d.device = 0x40; /* FIS LBA mode set */ 17495 fis->d.lbaLowExp = LBA[0]; /* FIS LBA (31:24) */ 17496 fis->d.lbaMidExp = 0; /* FIS LBA (39:32) */ 17497 fis->d.lbaHighExp = 0; /* FIS LBA (47:40) */ 17498 fis->d.featuresExp = 0; /* FIS reserve */ 17499 if (satOrgIOContext->LoopNum == 1) 17500 { 17501 /* last loop */ 17502 fis->d.sectorCount = (bit8)(Remainder & 0xFF); /* FIS sector count (7:0) */ 17503 fis->d.sectorCountExp = (bit8)((Remainder & 0xFF00) >> 8); /* FIS sector count (15:8) */ 17504 } 17505 else 17506 { 17507 fis->d.sectorCount = 0xFF; /* FIS sector count (7:0) */ 17508 fis->d.sectorCountExp = 0xFF; /* FIS sector count (15:8) */ 17509 } 17510 fis->d.reserved4 = 0; 17511 fis->d.control = 0; /* FIS HOB bit clear */ 17512 fis->d.reserved5 = 0; 17513 17514 agRequestType = AGSA_SATA_PROTOCOL_DMA_WRITE; 17515 17516 break; 17517 case SAT_WRITE_SECTORS_EXT: 17518 fis->h.fisType = 0x27; /* Reg host to device */ 17519 fis->h.c_pmPort = 0x80; /* C Bit is set */ 17520 fis->h.command = SAT_WRITE_SECTORS_EXT; /* 0x34 */ 17521 17522 fis->h.features = 0; /* FIS reserve */ 17523 fis->d.lbaLow = LBA[3]; /* FIS LBA (7 :0 ) */ 17524 fis->d.lbaMid = LBA[2]; /* FIS LBA (15:8 ) */ 17525 fis->d.lbaHigh = LBA[1]; /* FIS LBA (23:16) */ 17526 fis->d.device = 0x40; /* FIS LBA mode set */ 17527 fis->d.lbaLowExp = LBA[0]; /* FIS LBA (31:24) */ 17528 fis->d.lbaMidExp = 0; /* FIS LBA (39:32) */ 17529 fis->d.lbaHighExp = 0; /* FIS LBA (47:40) */ 17530 fis->d.featuresExp = 0; /* FIS reserve */ 17531 if (satOrgIOContext->LoopNum == 1) 17532 { 17533 /* last loop */ 17534 fis->d.sectorCount = (bit8)(Remainder & 0xFF); /* FIS sector count (7:0) */ 17535 fis->d.sectorCountExp = (bit8)((Remainder & 0xFF00) >> 8); /* FIS sector count (15:8) */ 17536 } 17537 else 17538 { 17539 fis->d.sectorCount = 0xFF; /* FIS sector count (7:0) */ 17540 fis->d.sectorCountExp = 0xFF; /* FIS sector count (15:8) */ 17541 } 17542 fis->d.reserved4 = 0; 17543 fis->d.control = 0; /* FIS HOB bit clear */ 17544 fis->d.reserved5 = 0; 17545 17546 agRequestType = AGSA_SATA_PROTOCOL_PIO_WRITE; 17547 17548 break; 17549 case SAT_WRITE_FPDMA_QUEUED: 17550 fis->h.fisType = 0x27; /* Reg host to device */ 17551 fis->h.c_pmPort = 0x80; /* C Bit is set */ 17552 fis->h.command = SAT_WRITE_FPDMA_QUEUED; /* 0x61 */ 17553 fis->d.lbaLow = LBA[3]; /* FIS LBA (7 :0 ) */ 17554 fis->d.lbaMid = LBA[2]; /* FIS LBA (15:8 ) */ 17555 fis->d.lbaHigh = LBA[1]; /* FIS LBA (23:16) */ 17556 17557 /* Check FUA bit */ 17558 if (scsiCmnd->cdb[1] & SCSI_WRITE10_FUA_MASK) 17559 fis->d.device = 0xC0; /* FIS FUA set */ 17560 else 17561 fis->d.device = 0x40; /* FIS FUA clear */ 17562 17563 fis->d.lbaLowExp = LBA[0];; /* FIS LBA (31:24) */ 17564 fis->d.lbaMidExp = 0; /* FIS LBA (39:32) */ 17565 fis->d.lbaHighExp = 0; /* FIS LBA (47:40) */ 17566 if (satOrgIOContext->LoopNum == 1) 17567 { 17568 /* last loop */ 17569 fis->h.features = (bit8)(Remainder & 0xFF); /* FIS sector count (7:0) */ 17570 fis->d.featuresExp = (bit8)((Remainder & 0xFF00) >> 8); /* FIS sector count (15:8) */ 17571 } 17572 else 17573 { 17574 fis->h.features = 0xFF; /* FIS sector count (7:0) */ 17575 fis->d.featuresExp = 0xFF; /* FIS sector count (15:8) */ 17576 } 17577 fis->d.sectorCount = 0; /* Tag (7:3) set by LL layer */ 17578 fis->d.sectorCountExp = 0; 17579 fis->d.reserved4 = 0; 17580 fis->d.control = 0; /* FIS HOB bit clear */ 17581 fis->d.reserved5 = 0; 17582 17583 agRequestType = AGSA_SATA_PROTOCOL_FPDMA_WRITE; 17584 break; 17585 17586 default: 17587 SM_DBG1(("smsatWrite_1: error incorrect ata command 0x%x!!!\n", satIOContext->ATACmd)); 17588 return SM_RC_FAILURE; 17589 break; 17590 } 17591 17592 /* Initialize CB for SATA completion. 17593 */ 17594 /* chained data */ 17595 satIOContext->satCompleteCB = &smsatChainedDataIOCB; 17596 17597 if (satOrgIOContext->ATACmd == SAT_WRITE_DMA || satOrgIOContext->ATACmd == SAT_WRITE_SECTORS) 17598 { 17599 smsatSplitSGL(smRoot, 17600 smIORequest, 17601 smDeviceHandle, 17602 (smScsiInitiatorRequest_t *)satOrgIOContext->smScsiXchg, 17603 satOrgIOContext, 17604 NON_BIT48_ADDRESS_TL_LIMIT * SATA_SECTOR_SIZE, /* 0x100 * 0x200*/ 17605 (satOrgIOContext->OrgTL) * SATA_SECTOR_SIZE, 17606 agFALSE); 17607 } 17608 else 17609 { 17610 smsatSplitSGL(smRoot, 17611 smIORequest, 17612 smDeviceHandle, 17613 (smScsiInitiatorRequest_t *)satOrgIOContext->smScsiXchg, 17614 satOrgIOContext, 17615 BIT48_ADDRESS_TL_LIMIT * SATA_SECTOR_SIZE, /* 0xFFFF * 0x200*/ 17616 (satOrgIOContext->OrgTL) * SATA_SECTOR_SIZE, 17617 agFALSE); 17618 } 17619 17620 /* 17621 * Prepare SGL and send FIS to LL layer. 17622 */ 17623 satIOContext->reqType = agRequestType; /* Save it */ 17624 17625 status = smsataLLIOStart( smRoot, 17626 smIORequest, 17627 smDeviceHandle, 17628 (smScsiInitiatorRequest_t *)satOrgIOContext->smScsiXchg, //smScsiRequest, 17629 satIOContext); 17630 17631 SM_DBG5(("smsatWrite_1: return\n")); 17632 return (status); 17633 } 17634 17635 osGLOBAL bit32 17636 smsatPassthrough( 17637 smRoot_t *smRoot, 17638 smIORequest_t *smIORequest, 17639 smDeviceHandle_t *smDeviceHandle, 17640 smScsiInitiatorRequest_t *smScsiRequest, 17641 smSatIOContext_t *satIOContext 17642 ) 17643 { 17644 smScsiRspSense_t *pSense; 17645 smIniScsiCmnd_t *scsiCmnd; 17646 smDeviceData_t *pSatDevData; 17647 agsaFisRegHostToDevice_t *fis; 17648 bit32 status; 17649 bit32 agRequestType; 17650 smAtaPassThroughHdr_t ataPassThroughHdr; 17651 17652 17653 pSense = satIOContext->pSense; 17654 scsiCmnd = &smScsiRequest->scsiCmnd; 17655 pSatDevData = satIOContext->pSatDevData; 17656 fis = satIOContext->pFis; 17657 17658 SM_DBG1(("smsatPassthrough: START!!!\n")); 17659 17660 osti_memset(&ataPassThroughHdr, 0 , sizeof(smAtaPassThroughHdr_t)); 17661 17662 ataPassThroughHdr.opc = scsiCmnd->cdb[0]; 17663 ataPassThroughHdr.mulCount = scsiCmnd->cdb[1] >> 5; 17664 ataPassThroughHdr.proto = (scsiCmnd->cdb[1] >> 1) & 0x0F; 17665 ataPassThroughHdr.extend = scsiCmnd->cdb[1] & 1; 17666 ataPassThroughHdr.offline = scsiCmnd->cdb[2] >> 6; 17667 ataPassThroughHdr.ckCond = (scsiCmnd->cdb[2] >> 5) & 1; 17668 ataPassThroughHdr.tType = (scsiCmnd->cdb[2] >> 4) & 1; 17669 ataPassThroughHdr.tDir = (scsiCmnd->cdb[2] >> 3) & 1; 17670 ataPassThroughHdr.byteBlock = (scsiCmnd->cdb[2] >> 2) & 1; 17671 ataPassThroughHdr.tlength = scsiCmnd->cdb[2] & 0x3; 17672 17673 switch(ataPassThroughHdr.proto) 17674 { 17675 case 0: 17676 case 9: 17677 agRequestType = AGSA_SATA_PROTOCOL_DEV_RESET; //Device Reset 17678 break; 17679 case 1: 17680 agRequestType = AGSA_SATA_PROTOCOL_SRST_ASSERT; //Software reset 17681 break; 17682 case 3: 17683 agRequestType = AGSA_SATA_PROTOCOL_NON_DATA; //Non Data mode 17684 break; 17685 case 4: 17686 agRequestType = AGSA_SATA_PROTOCOL_PIO_READ; //IO_Data_In mode 17687 break; 17688 case 5: 17689 agRequestType = AGSA_SATA_PROTOCOL_PIO_WRITE; //PIO_Data_out 17690 break; 17691 case 6: 17692 agRequestType = AGSA_SATA_PROTOCOL_DMA_WRITE; //DMA READ and WRITE 17693 break; 17694 case 8: 17695 agRequestType = AGSA_SATA_ATAP_EXECDEVDIAG; //device diagnostic 17696 break; 17697 case 12: 17698 agRequestType = AGSA_SATA_PROTOCOL_FPDMA_WRITE; //FPDMA Read and Write 17699 break; 17700 default: 17701 agRequestType = AGSA_SATA_PROTOCOL_NON_DATA; //Default Non Data Mode 17702 break; 17703 } 17704 17705 17706 if((ataPassThroughHdr.tlength == 0) && (agRequestType != AGSA_SATA_PROTOCOL_NON_DATA)) 17707 { 17708 SM_DBG1(("smsatPassthrough SCSI_SNSCODE_INVALID_FIELD_IN_CDB\n")); 17709 17710 smsatSetSensePayload( pSense, 17711 SCSI_SNSKEY_ILLEGAL_REQUEST, 17712 0, 17713 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 17714 satIOContext); 17715 17716 tdsmIOCompletedCB( smRoot, 17717 smIORequest, 17718 smIOSuccess, 17719 SCSI_STAT_CHECK_CONDITION, 17720 satIOContext->pSmSenseData, 17721 satIOContext->interruptContext ); 17722 17723 return SM_RC_SUCCESS; 17724 } 17725 17726 if(scsiCmnd->cdb[0] == 0xA1) 17727 { 17728 SM_DBG1(("smsatPassthrough A1h: COMMAND: %x FEATURE: %x \n",scsiCmnd->cdb[9],scsiCmnd->cdb[3])); 17729 17730 fis->h.fisType = 0x27; /* Reg host to device */ 17731 fis->h.c_pmPort = 0x80; /* C Bit is set */ 17732 fis->h.features = scsiCmnd->cdb[3]; 17733 fis->d.sectorCount = scsiCmnd->cdb[4]; /* 0x01 FIS sector count (7:0) */ 17734 fis->d.lbaLow = scsiCmnd->cdb[5]; /* Reading LBA FIS LBA (7 :0 ) */ 17735 fis->d.lbaMid = scsiCmnd->cdb[6]; 17736 fis->d.lbaHigh = scsiCmnd->cdb[7]; 17737 fis->d.device = scsiCmnd->cdb[8]; 17738 fis->h.command = scsiCmnd->cdb[9]; 17739 fis->d.featuresExp = 0; 17740 fis->d.sectorCountExp = 0; 17741 fis->d.lbaLowExp = 0; 17742 fis->d.lbaMidExp = 0; 17743 fis->d.lbaHighExp = 0; 17744 fis->d.reserved4 = 0; 17745 fis->d.control = 0; /* FIS HOB bit clear */ 17746 fis->d.reserved5 = 0; 17747 17748 /* Initialize CB for SATA completion*/ 17749 satIOContext->satCompleteCB = &smsatPassthroughCB; 17750 17751 /* 17752 * Prepare SGL and send FIS to LL layer. 17753 */ 17754 17755 satIOContext->reqType = agRequestType; 17756 status = smsataLLIOStart( smRoot, 17757 smIORequest, 17758 smDeviceHandle, 17759 smScsiRequest, 17760 satIOContext); 17761 return status; 17762 17763 } 17764 else if(scsiCmnd->cdb[0] == 0x85) 17765 { 17766 SM_DBG1(("smsatPassthrough 85h: COMMAND: %x FEATURE: %x \n",scsiCmnd->cdb[14],scsiCmnd->cdb[4])); 17767 17768 fis->h.fisType = 0x27; /* Reg host to device */ 17769 fis->h.c_pmPort = 0x80; /* C Bit is set */ 17770 17771 if(1 == ataPassThroughHdr.extend) 17772 { 17773 fis->d.featuresExp = scsiCmnd->cdb[3]; 17774 fis->d.sectorCountExp = scsiCmnd->cdb[5]; 17775 fis->d.lbaMidExp = scsiCmnd->cdb[9]; 17776 fis->d.lbaHighExp = scsiCmnd->cdb[11]; 17777 fis->d.lbaLowExp = scsiCmnd->cdb[7]; 17778 } 17779 fis->h.features = scsiCmnd->cdb[4]; 17780 fis->d.sectorCount = scsiCmnd->cdb[6]; 17781 fis->d.lbaLow = scsiCmnd->cdb[8]; 17782 fis->d.lbaMid = scsiCmnd->cdb[10]; 17783 fis->d.lbaHigh = scsiCmnd->cdb[12]; 17784 fis->d.device = scsiCmnd->cdb[13]; 17785 fis->h.command = scsiCmnd->cdb[14]; 17786 fis->d.reserved4 = 0; 17787 fis->d.control = 0; 17788 fis->d.reserved5 = 0; 17789 17790 17791 /* Initialize CB for SATA completion. 17792 */ 17793 17794 satIOContext->satCompleteCB = &smsatPassthroughCB; 17795 17796 /* 17797 * Prepare SGL and send FIS to LL layer. 17798 */ 17799 satIOContext->reqType = agRequestType; 17800 status = smsataLLIOStart( smRoot, 17801 smIORequest, 17802 smDeviceHandle, 17803 smScsiRequest, 17804 satIOContext); 17805 return status; 17806 17807 } 17808 else 17809 { 17810 SM_DBG1(("smsatPassthrough : INVALD PASSTHROUGH!!!\n")); 17811 smsatSetSensePayload( pSense, 17812 SCSI_SNSKEY_ILLEGAL_REQUEST, 17813 0, 17814 SCSI_SNSCODE_INVALID_FIELD_IN_CDB, 17815 satIOContext); 17816 tdsmIOCompletedCB( smRoot, 17817 smIORequest, 17818 smIOSuccess, 17819 SCSI_STAT_CHECK_CONDITION, 17820 satIOContext->pSmSenseData, 17821 satIOContext->interruptContext ); 17822 17823 SM_DBG1(("smsatPassthrough : return control!!!\n")); 17824 17825 return SM_RC_SUCCESS; 17826 } 17827 } 17828 17829 osGLOBAL bit32 17830 smsatNonChainedWriteNVerify_Verify( 17831 smRoot_t *smRoot, 17832 smIORequest_t *smIORequest, 17833 smDeviceHandle_t *smDeviceHandle, 17834 smScsiInitiatorRequest_t *smScsiRequest, 17835 smSatIOContext_t *satIOContext 17836 ) 17837 { 17838 bit32 status; 17839 bit32 agRequestType = AGSA_SATA_PROTOCOL_DMA_WRITE; 17840 smDeviceData_t *pSatDevData; 17841 smIniScsiCmnd_t *scsiCmnd; 17842 agsaFisRegHostToDevice_t *fis; 17843 17844 pSatDevData = satIOContext->pSatDevData; 17845 scsiCmnd = &smScsiRequest->scsiCmnd; 17846 fis = satIOContext->pFis; 17847 SM_DBG5(("smsatNonChainedWriteNVerify_Verify: start\n")); 17848 if (pSatDevData->sat48BitSupport == agTRUE) 17849 { 17850 fis->h.fisType = 0x27; /* Reg host to device */ 17851 fis->h.c_pmPort = 0x80; /* C Bit is set */ 17852 17853 fis->h.command = SAT_READ_VERIFY_SECTORS_EXT;/* 0x42 */ 17854 fis->h.features = 0; /* FIS reserve */ 17855 fis->d.lbaLow = scsiCmnd->cdb[5]; /* FIS LBA (7 :0 ) */ 17856 fis->d.lbaMid = scsiCmnd->cdb[4]; /* FIS LBA (15:8 ) */ 17857 fis->d.lbaHigh = scsiCmnd->cdb[3]; /* FIS LBA (23:16) */ 17858 fis->d.device = 0x40; /* FIS LBA mode set 01000000 */ 17859 fis->d.lbaLowExp = scsiCmnd->cdb[2]; /* FIS LBA (31:24) */ 17860 fis->d.lbaMidExp = 0; /* FIS LBA (39:32) */ 17861 fis->d.lbaHighExp = 0; /* FIS LBA (47:40) */ 17862 fis->d.featuresExp = 0; /* FIS reserve */ 17863 fis->d.sectorCount = scsiCmnd->cdb[8]; /* FIS sector count (7:0) */ 17864 fis->d.sectorCountExp = scsiCmnd->cdb[7]; /* FIS sector count (15:8) */ 17865 17866 fis->d.reserved4 = 0; 17867 fis->d.control = 0; /* FIS HOB bit clear */ 17868 fis->d.reserved5 = 0; 17869 17870 agRequestType = AGSA_SATA_PROTOCOL_NON_DATA; 17871 17872 /* Initialize CB for SATA completion. 17873 */ 17874 satIOContext->satCompleteCB = &smsatNonChainedWriteNVerifyCB; 17875 17876 /* 17877 * Prepare SGL and send FIS to LL layer. 17878 */ 17879 satIOContext->reqType = agRequestType; /* Save it */ 17880 17881 status = smsataLLIOStart( smRoot, 17882 smIORequest, 17883 smDeviceHandle, 17884 smScsiRequest, 17885 satIOContext); 17886 17887 17888 SM_DBG1(("smsatNonChainedWriteNVerify_Verify: return status %d!!!\n", status)); 17889 return (status); 17890 } 17891 else 17892 { 17893 /* can't fit in SAT_READ_VERIFY_SECTORS becasue of Sector Count and LBA */ 17894 SM_DBG1(("smsatNonChainedWriteNVerify_Verify: can't fit in SAT_READ_VERIFY_SECTORS!!!\n")); 17895 return SM_RC_FAILURE; 17896 } 17897 } 17898 17899 osGLOBAL bit32 17900 smsatChainedWriteNVerify_Start_Verify( 17901 smRoot_t *smRoot, 17902 smIORequest_t *smIORequest, 17903 smDeviceHandle_t *smDeviceHandle, 17904 smScsiInitiatorRequest_t *smScsiRequest, 17905 smSatIOContext_t *satIOContext 17906 ) 17907 { 17908 /* 17909 deal with transfer length; others have been handled previously at this point; 17910 no LBA check; no range check; 17911 */ 17912 bit32 status; 17913 bit32 agRequestType = AGSA_SATA_PROTOCOL_NON_DATA; 17914 smDeviceData_t *pSatDevData; 17915 smIniScsiCmnd_t *scsiCmnd; 17916 agsaFisRegHostToDevice_t *fis; 17917 bit32 lba = 0; 17918 bit32 tl = 0; 17919 bit32 LoopNum = 1; 17920 bit8 LBA[4]; 17921 bit8 TL[4]; 17922 17923 pSatDevData = satIOContext->pSatDevData; 17924 scsiCmnd = &smScsiRequest->scsiCmnd; 17925 fis = satIOContext->pFis; 17926 17927 SM_DBG5(("smsatChainedWriteNVerify_Start_Verify: start\n")); 17928 sm_memset(LBA, 0, sizeof(LBA)); 17929 sm_memset(TL, 0, sizeof(TL)); 17930 /* do not use memcpy due to indexing in LBA and TL */ 17931 LBA[0] = scsiCmnd->cdb[2]; /* MSB */ 17932 LBA[1] = scsiCmnd->cdb[3]; 17933 LBA[2] = scsiCmnd->cdb[4]; 17934 LBA[3] = scsiCmnd->cdb[5]; /* LSB */ 17935 TL[0] = scsiCmnd->cdb[6]; /* MSB */ 17936 TL[1] = scsiCmnd->cdb[7]; 17937 TL[2] = scsiCmnd->cdb[7]; 17938 TL[3] = scsiCmnd->cdb[8]; /* LSB */ 17939 lba = smsatComputeCDB12LBA(satIOContext); 17940 tl = smsatComputeCDB12TL(satIOContext); 17941 if (pSatDevData->sat48BitSupport == agTRUE) 17942 { 17943 SM_DBG5(("smsatChainedWriteNVerify_Start_Verify: SAT_READ_VERIFY_SECTORS_EXT\n")); 17944 fis->h.fisType = 0x27; /* Reg host to device */ 17945 fis->h.c_pmPort = 0x80; /* C Bit is set */ 17946 17947 fis->h.command = SAT_READ_VERIFY_SECTORS_EXT;/* 0x42 */ 17948 fis->h.features = 0; /* FIS reserve */ 17949 fis->d.lbaLow = scsiCmnd->cdb[5]; /* FIS LBA (7 :0 ) */ 17950 fis->d.lbaMid = scsiCmnd->cdb[4]; /* FIS LBA (15:8 ) */ 17951 fis->d.lbaHigh = scsiCmnd->cdb[3]; /* FIS LBA (23:16) */ 17952 fis->d.device = 0x40; /* FIS LBA mode set 01000000 */ 17953 fis->d.lbaLowExp = scsiCmnd->cdb[2]; /* FIS LBA (31:24) */ 17954 fis->d.lbaMidExp = 0; /* FIS LBA (39:32) */ 17955 fis->d.lbaHighExp = 0; /* FIS LBA (47:40) */ 17956 fis->d.featuresExp = 0; /* FIS reserve */ 17957 fis->d.sectorCount = scsiCmnd->cdb[8]; /* FIS sector count (7:0) */ 17958 fis->d.sectorCountExp = scsiCmnd->cdb[7]; /* FIS sector count (15:8) */ 17959 17960 fis->d.reserved4 = 0; 17961 fis->d.control = 0; /* FIS HOB bit clear */ 17962 fis->d.reserved5 = 0; 17963 17964 agRequestType = AGSA_SATA_PROTOCOL_NON_DATA; 17965 satIOContext->ATACmd = SAT_READ_VERIFY_SECTORS_EXT; 17966 } 17967 else 17968 { 17969 SM_DBG5(("smsatChainedWriteNVerify_Start_Verify: SAT_READ_VERIFY_SECTORS\n")); 17970 fis->h.fisType = 0x27; /* Reg host to device */ 17971 fis->h.c_pmPort = 0x80; /* C bit is set */ 17972 fis->h.command = SAT_READ_VERIFY_SECTORS; /* 0x40 */ 17973 fis->h.features = 0; /* FIS reserve */ 17974 fis->d.lbaLow = scsiCmnd->cdb[5]; /* FIS LBA (7 :0 ) */ 17975 fis->d.lbaMid = scsiCmnd->cdb[4]; /* FIS LBA (15:8 ) */ 17976 fis->d.lbaHigh = scsiCmnd->cdb[3]; /* FIS LBA (23:16) */ 17977 /* FIS LBA mode set LBA (27:24) */ 17978 fis->d.device = (bit8)((0x4 << 4) | (scsiCmnd->cdb[2] & 0xF)); 17979 fis->d.lbaLowExp = 0; 17980 fis->d.lbaMidExp = 0; 17981 fis->d.lbaHighExp = 0; 17982 fis->d.featuresExp = 0; 17983 fis->d.sectorCount = scsiCmnd->cdb[8]; /* FIS sector count (7:0) */ 17984 fis->d.sectorCountExp = 0; 17985 fis->d.reserved4 = 0; 17986 fis->d.control = 0; /* FIS HOB bit clear */ 17987 fis->d.reserved5 = 0; 17988 17989 agRequestType = AGSA_SATA_PROTOCOL_NON_DATA; 17990 satIOContext->ATACmd = SAT_READ_VERIFY_SECTORS; 17991 17992 } 17993 17994 satIOContext->currentLBA = lba; 17995 satIOContext->OrgTL = tl; 17996 17997 /* 17998 computing number of loop and remainder for tl 17999 0xFF in case not ext 18000 0xFFFF in case EXT 18001 */ 18002 if (fis->h.command == SAT_READ_VERIFY_SECTORS) 18003 { 18004 LoopNum = smsatComputeLoopNum(tl, 0xFF); 18005 } 18006 else if (fis->h.command == SAT_READ_VERIFY_SECTORS_EXT) 18007 { 18008 /* SAT_READ_SECTORS_EXT, SAT_READ_DMA_EXT */ 18009 LoopNum = smsatComputeLoopNum(tl, 0xFFFF); 18010 } 18011 else 18012 { 18013 SM_DBG1(("smsatChainedWriteNVerify_Start_Verify: error case 1!!!\n")); 18014 LoopNum = 1; 18015 } 18016 18017 satIOContext->LoopNum = LoopNum; 18018 18019 if (LoopNum == 1) 18020 { 18021 SM_DBG5(("smsatChainedWriteNVerify_Start_Verify: NON CHAINED data\n")); 18022 /* Initialize CB for SATA completion. 18023 */ 18024 satIOContext->satCompleteCB = &smsatNonChainedWriteNVerifyCB; 18025 } 18026 else 18027 { 18028 SM_DBG1(("smsatChainedWriteNVerify_Start_Verify: CHAINED data!!!\n")); 18029 /* re-setting tl */ 18030 if (fis->h.command == SAT_READ_VERIFY_SECTORS) 18031 { 18032 fis->d.sectorCount = 0xFF; 18033 } 18034 else if (fis->h.command == SAT_READ_VERIFY_SECTORS_EXT) 18035 { 18036 fis->d.sectorCount = 0xFF; 18037 fis->d.sectorCountExp = 0xFF; 18038 } 18039 else 18040 { 18041 SM_DBG1(("smsatChainedWriteNVerify_Start_Verify: error case 2!!!\n")); 18042 } 18043 18044 /* Initialize CB for SATA completion. 18045 */ 18046 satIOContext->satCompleteCB = &smsatChainedWriteNVerifyCB; 18047 } 18048 18049 18050 /* 18051 * Prepare SGL and send FIS to LL layer. 18052 */ 18053 satIOContext->reqType = agRequestType; /* Save it */ 18054 18055 status = smsataLLIOStart( smRoot, 18056 smIORequest, 18057 smDeviceHandle, 18058 smScsiRequest, 18059 satIOContext); 18060 return (status); 18061 18062 18063 } 18064 18065 osGLOBAL bit32 18066 smsatChainedWriteNVerify_Write( 18067 smRoot_t *smRoot, 18068 smIORequest_t *smIORequest, 18069 smDeviceHandle_t *smDeviceHandle, 18070 smScsiInitiatorRequest_t *smScsiRequest, 18071 smSatIOContext_t *satIOContext 18072 ) 18073 { 18074 /* 18075 Assumption: error check on lba and tl has been done in satWrite*() 18076 lba = lba + tl; 18077 */ 18078 bit32 status; 18079 smSatIOContext_t *satOrgIOContext = agNULL; 18080 smIniScsiCmnd_t *scsiCmnd; 18081 agsaFisRegHostToDevice_t *fis; 18082 bit32 agRequestType = AGSA_SATA_PROTOCOL_DMA_WRITE; 18083 bit32 lba = 0; 18084 bit32 DenomTL = 0xFF; 18085 bit32 Remainder = 0; 18086 bit8 LBA[4]; /* 0 MSB, 3 LSB */ 18087 18088 SM_DBG1(("smsatChainedWriteNVerify_Write: start\n")); 18089 18090 fis = satIOContext->pFis; 18091 satOrgIOContext = satIOContext->satOrgIOContext; 18092 scsiCmnd = satOrgIOContext->pScsiCmnd; 18093 18094 18095 sm_memset(LBA,0, sizeof(LBA)); 18096 18097 switch (satOrgIOContext->ATACmd) 18098 { 18099 case SAT_WRITE_DMA: 18100 DenomTL = 0xFF; 18101 break; 18102 case SAT_WRITE_SECTORS: 18103 DenomTL = 0xFF; 18104 break; 18105 case SAT_WRITE_DMA_EXT: 18106 DenomTL = 0xFFFF; 18107 break; 18108 case SAT_WRITE_DMA_FUA_EXT: 18109 DenomTL = 0xFFFF; 18110 break; 18111 case SAT_WRITE_SECTORS_EXT: 18112 DenomTL = 0xFFFF; 18113 break; 18114 case SAT_WRITE_FPDMA_QUEUED: 18115 DenomTL = 0xFFFF; 18116 break; 18117 default: 18118 SM_DBG1(("satChainedWriteNVerify_Write: error incorrect ata command 0x%x!!!\n", satIOContext->ATACmd)); 18119 return SM_RC_FAILURE; 18120 break; 18121 } 18122 18123 Remainder = satOrgIOContext->OrgTL % DenomTL; 18124 satOrgIOContext->currentLBA = satOrgIOContext->currentLBA + DenomTL; 18125 lba = satOrgIOContext->currentLBA; 18126 18127 LBA[0] = (bit8)((lba & 0xF000) >> (8 * 3)); /* MSB */ 18128 LBA[1] = (bit8)((lba & 0xF00) >> (8 * 2)); 18129 LBA[2] = (bit8)((lba & 0xF0) >> 8); 18130 LBA[3] = (bit8)(lba & 0xF); /* LSB */ 18131 18132 switch (satOrgIOContext->ATACmd) 18133 { 18134 case SAT_WRITE_DMA: 18135 fis->h.fisType = 0x27; /* Reg host to device */ 18136 fis->h.c_pmPort = 0x80; /* C bit is set */ 18137 fis->h.command = SAT_WRITE_DMA; /* 0xCA */ 18138 fis->h.features = 0; /* FIS reserve */ 18139 fis->d.lbaLow = LBA[3]; /* FIS LBA (7 :0 ) */ 18140 fis->d.lbaMid = LBA[2]; /* FIS LBA (15:8 ) */ 18141 fis->d.lbaHigh = LBA[1]; /* FIS LBA (23:16) */ 18142 18143 /* FIS LBA mode set LBA (27:24) */ 18144 fis->d.device = (bit8)((0x4 << 4) | (LBA[0] & 0xF)); 18145 18146 fis->d.lbaLowExp = 0; 18147 fis->d.lbaMidExp = 0; 18148 fis->d.lbaHighExp = 0; 18149 fis->d.featuresExp = 0; 18150 if (satOrgIOContext->LoopNum == 1) 18151 { 18152 /* last loop */ 18153 fis->d.sectorCount = (bit8)Remainder; /* FIS sector count (7:0) */ 18154 } 18155 else 18156 { 18157 fis->d.sectorCount = 0xFF; /* FIS sector count (7:0) */ 18158 } 18159 fis->d.sectorCountExp = 0; 18160 fis->d.reserved4 = 0; 18161 fis->d.control = 0; /* FIS HOB bit clear */ 18162 fis->d.reserved5 = 0; 18163 18164 agRequestType = AGSA_SATA_PROTOCOL_DMA_WRITE; 18165 18166 break; 18167 case SAT_WRITE_SECTORS: 18168 fis->h.fisType = 0x27; /* Reg host to device */ 18169 fis->h.c_pmPort = 0x80; /* C bit is set */ 18170 fis->h.command = SAT_WRITE_SECTORS; /* 0x30 */ 18171 fis->h.features = 0; /* FIS reserve */ 18172 fis->d.lbaLow = LBA[3]; /* FIS LBA (7 :0 ) */ 18173 fis->d.lbaMid = LBA[2]; /* FIS LBA (15:8 ) */ 18174 fis->d.lbaHigh = LBA[1]; /* FIS LBA (23:16) */ 18175 18176 /* FIS LBA mode set LBA (27:24) */ 18177 fis->d.device = (bit8)((0x4 << 4) | (LBA[0] & 0xF)); 18178 18179 fis->d.lbaLowExp = 0; 18180 fis->d.lbaMidExp = 0; 18181 fis->d.lbaHighExp = 0; 18182 fis->d.featuresExp = 0; 18183 if (satOrgIOContext->LoopNum == 1) 18184 { 18185 /* last loop */ 18186 fis->d.sectorCount = (bit8)Remainder; /* FIS sector count (7:0) */ 18187 } 18188 else 18189 { 18190 fis->d.sectorCount = 0xFF; /* FIS sector count (7:0) */ 18191 } 18192 fis->d.sectorCountExp = 0; 18193 fis->d.reserved4 = 0; 18194 fis->d.control = 0; /* FIS HOB bit clear */ 18195 fis->d.reserved5 = 0; 18196 18197 agRequestType = AGSA_SATA_PROTOCOL_PIO_WRITE; 18198 18199 break; 18200 case SAT_WRITE_DMA_EXT: 18201 fis->h.fisType = 0x27; /* Reg host to device */ 18202 fis->h.c_pmPort = 0x80; /* C Bit is set */ 18203 fis->h.command = SAT_WRITE_DMA_EXT; /* 0x3D */ 18204 fis->h.features = 0; /* FIS reserve */ 18205 fis->d.lbaLow = LBA[3]; /* FIS LBA (7 :0 ) */ 18206 fis->d.lbaMid = LBA[2]; /* FIS LBA (15:8 ) */ 18207 fis->d.lbaHigh = LBA[1]; /* FIS LBA (23:16) */ 18208 fis->d.device = 0x40; /* FIS LBA mode set */ 18209 fis->d.lbaLowExp = LBA[0]; /* FIS LBA (31:24) */ 18210 fis->d.lbaMidExp = 0; /* FIS LBA (39:32) */ 18211 fis->d.lbaHighExp = 0; /* FIS LBA (47:40) */ 18212 fis->d.featuresExp = 0; /* FIS reserve */ 18213 if (satOrgIOContext->LoopNum == 1) 18214 { 18215 /* last loop */ 18216 fis->d.sectorCount = (bit8)(Remainder & 0xFF); /* FIS sector count (7:0) */ 18217 fis->d.sectorCountExp = (bit8)((Remainder & 0xFF00) >> 8); /* FIS sector count (15:8) */ 18218 } 18219 else 18220 { 18221 fis->d.sectorCount = 0xFF; /* FIS sector count (7:0) */ 18222 fis->d.sectorCountExp = 0xFF; /* FIS sector count (15:8) */ 18223 } 18224 fis->d.reserved4 = 0; 18225 fis->d.control = 0; /* FIS HOB bit clear */ 18226 fis->d.reserved5 = 0; 18227 18228 agRequestType = AGSA_SATA_PROTOCOL_DMA_WRITE; 18229 18230 break; 18231 case SAT_WRITE_SECTORS_EXT: 18232 fis->h.fisType = 0x27; /* Reg host to device */ 18233 fis->h.c_pmPort = 0x80; /* C Bit is set */ 18234 fis->h.command = SAT_WRITE_SECTORS_EXT; /* 0x34 */ 18235 18236 fis->h.features = 0; /* FIS reserve */ 18237 fis->d.lbaLow = LBA[3]; /* FIS LBA (7 :0 ) */ 18238 fis->d.lbaMid = LBA[2]; /* FIS LBA (15:8 ) */ 18239 fis->d.lbaHigh = LBA[1]; /* FIS LBA (23:16) */ 18240 fis->d.device = 0x40; /* FIS LBA mode set */ 18241 fis->d.lbaLowExp = LBA[0]; /* FIS LBA (31:24) */ 18242 fis->d.lbaMidExp = 0; /* FIS LBA (39:32) */ 18243 fis->d.lbaHighExp = 0; /* FIS LBA (47:40) */ 18244 fis->d.featuresExp = 0; /* FIS reserve */ 18245 if (satOrgIOContext->LoopNum == 1) 18246 { 18247 /* last loop */ 18248 fis->d.sectorCount = (bit8)(Remainder & 0xFF); /* FIS sector count (7:0) */ 18249 fis->d.sectorCountExp = (bit8)((Remainder & 0xFF00) >> 8); /* FIS sector count (15:8) */ 18250 } 18251 else 18252 { 18253 fis->d.sectorCount = 0xFF; /* FIS sector count (7:0) */ 18254 fis->d.sectorCountExp = 0xFF; /* FIS sector count (15:8) */ 18255 } 18256 fis->d.reserved4 = 0; 18257 fis->d.control = 0; /* FIS HOB bit clear */ 18258 fis->d.reserved5 = 0; 18259 18260 agRequestType = AGSA_SATA_PROTOCOL_PIO_WRITE; 18261 18262 break; 18263 case SAT_WRITE_FPDMA_QUEUED: 18264 fis->h.fisType = 0x27; /* Reg host to device */ 18265 fis->h.c_pmPort = 0x80; /* C Bit is set */ 18266 fis->h.command = SAT_WRITE_FPDMA_QUEUED; /* 0x61 */ 18267 fis->d.lbaLow = LBA[3]; /* FIS LBA (7 :0 ) */ 18268 fis->d.lbaMid = LBA[2]; /* FIS LBA (15:8 ) */ 18269 fis->d.lbaHigh = LBA[1]; /* FIS LBA (23:16) */ 18270 18271 /* Check FUA bit */ 18272 if (scsiCmnd->cdb[1] & SCSI_WRITE10_FUA_MASK) 18273 fis->d.device = 0xC0; /* FIS FUA set */ 18274 else 18275 fis->d.device = 0x40; /* FIS FUA clear */ 18276 18277 fis->d.lbaLowExp = LBA[0];; /* FIS LBA (31:24) */ 18278 fis->d.lbaMidExp = 0; /* FIS LBA (39:32) */ 18279 fis->d.lbaHighExp = 0; /* FIS LBA (47:40) */ 18280 if (satOrgIOContext->LoopNum == 1) 18281 { 18282 /* last loop */ 18283 fis->h.features = (bit8)(Remainder & 0xFF); /* FIS sector count (7:0) */ 18284 fis->d.featuresExp = (bit8)((Remainder & 0xFF00) >> 8); /* FIS sector count (15:8) */ 18285 } 18286 else 18287 { 18288 fis->h.features = 0xFF; /* FIS sector count (7:0) */ 18289 fis->d.featuresExp = 0xFF; /* FIS sector count (15:8) */ 18290 } 18291 fis->d.sectorCount = 0; /* Tag (7:3) set by LL layer */ 18292 fis->d.sectorCountExp = 0; 18293 fis->d.reserved4 = 0; 18294 fis->d.control = 0; /* FIS HOB bit clear */ 18295 fis->d.reserved5 = 0; 18296 18297 agRequestType = AGSA_SATA_PROTOCOL_FPDMA_WRITE; 18298 break; 18299 18300 default: 18301 SM_DBG1(("satChainedWriteNVerify_Write: error incorrect ata command 0x%x!!!\n", satIOContext->ATACmd)); 18302 return SM_RC_FAILURE; 18303 break; 18304 } 18305 18306 /* Initialize CB for SATA completion. 18307 */ 18308 /* chained data */ 18309 satIOContext->satCompleteCB = &smsatChainedWriteNVerifyCB; 18310 18311 18312 /* 18313 * Prepare SGL and send FIS to LL layer. 18314 */ 18315 satIOContext->reqType = agRequestType; /* Save it */ 18316 18317 status = smsataLLIOStart( smRoot, 18318 smIORequest, 18319 smDeviceHandle, 18320 smScsiRequest, 18321 satIOContext); 18322 18323 SM_DBG5(("satChainedWriteNVerify_Write: return\n")); 18324 return (status); 18325 } 18326 18327 osGLOBAL bit32 18328 smsatChainedWriteNVerify_Verify( 18329 smRoot_t *smRoot, 18330 smIORequest_t *smIORequest, 18331 smDeviceHandle_t *smDeviceHandle, 18332 smScsiInitiatorRequest_t *smScsiRequest, 18333 smSatIOContext_t *satIOContext 18334 ) 18335 { 18336 bit32 status; 18337 smSatIOContext_t *satOrgIOContext = agNULL; 18338 agsaFisRegHostToDevice_t *fis; 18339 bit32 agRequestType = AGSA_SATA_PROTOCOL_NON_DATA; 18340 bit32 lba = 0; 18341 bit32 DenomTL = 0xFF; 18342 bit32 Remainder = 0; 18343 bit8 LBA[4]; /* 0 MSB, 3 LSB */ 18344 18345 SM_DBG2(("smsatChainedWriteNVerify_Verify: start\n")); 18346 fis = satIOContext->pFis; 18347 satOrgIOContext = satIOContext->satOrgIOContext; 18348 sm_memset(LBA,0, sizeof(LBA)); 18349 switch (satOrgIOContext->ATACmd) 18350 { 18351 case SAT_READ_VERIFY_SECTORS: 18352 DenomTL = 0xFF; 18353 break; 18354 case SAT_READ_VERIFY_SECTORS_EXT: 18355 DenomTL = 0xFFFF; 18356 break; 18357 default: 18358 SM_DBG1(("smsatChainedWriteNVerify_Verify: error incorrect ata command 0x%x!!!\n", satIOContext->ATACmd)); 18359 return SM_RC_FAILURE; 18360 break; 18361 } 18362 18363 Remainder = satOrgIOContext->OrgTL % DenomTL; 18364 satOrgIOContext->currentLBA = satOrgIOContext->currentLBA + DenomTL; 18365 lba = satOrgIOContext->currentLBA; 18366 18367 LBA[0] = (bit8)((lba & 0xF000) >> (8 * 3)); /* MSB */ 18368 LBA[1] = (bit8)((lba & 0xF00) >> (8 * 2)); 18369 LBA[2] = (bit8)((lba & 0xF0) >> 8); 18370 LBA[3] = (bit8)(lba & 0xF); /* LSB */ 18371 18372 switch (satOrgIOContext->ATACmd) 18373 { 18374 case SAT_READ_VERIFY_SECTORS: 18375 fis->h.fisType = 0x27; /* Reg host to device */ 18376 fis->h.c_pmPort = 0x80; /* C bit is set */ 18377 fis->h.command = SAT_READ_VERIFY_SECTORS; /* 0x40 */ 18378 fis->h.features = 0; /* FIS reserve */ 18379 fis->d.lbaLow = LBA[3]; /* FIS LBA (7 :0 ) */ 18380 fis->d.lbaMid = LBA[2]; /* FIS LBA (15:8 ) */ 18381 fis->d.lbaHigh = LBA[1]; /* FIS LBA (23:16) */ 18382 18383 /* FIS LBA mode set LBA (27:24) */ 18384 fis->d.device = (bit8)((0x4 << 4) | (LBA[0] & 0xF)); 18385 18386 fis->d.lbaLowExp = 0; 18387 fis->d.lbaMidExp = 0; 18388 fis->d.lbaHighExp = 0; 18389 fis->d.featuresExp = 0; 18390 if (satOrgIOContext->LoopNum == 1) 18391 { 18392 /* last loop */ 18393 fis->d.sectorCount = (bit8)Remainder; /* FIS sector count (7:0) */ 18394 } 18395 else 18396 { 18397 fis->d.sectorCount = 0xFF; /* FIS sector count (7:0) */ 18398 } 18399 fis->d.sectorCountExp = 0; 18400 fis->d.reserved4 = 0; 18401 fis->d.control = 0; /* FIS HOB bit clear */ 18402 fis->d.reserved5 = 0; 18403 18404 agRequestType = AGSA_SATA_PROTOCOL_NON_DATA; 18405 18406 break; 18407 case SAT_READ_VERIFY_SECTORS_EXT: 18408 fis->h.fisType = 0x27; /* Reg host to device */ 18409 fis->h.c_pmPort = 0x80; /* C Bit is set */ 18410 fis->h.command = SAT_READ_VERIFY_SECTORS_EXT; /* 0x42 */ 18411 fis->h.features = 0; /* FIS reserve */ 18412 fis->d.lbaLow = LBA[3]; /* FIS LBA (7 :0 ) */ 18413 fis->d.lbaMid = LBA[2]; /* FIS LBA (15:8 ) */ 18414 fis->d.lbaHigh = LBA[1]; /* FIS LBA (23:16) */ 18415 fis->d.device = 0x40; /* FIS LBA mode set */ 18416 fis->d.lbaLowExp = LBA[0]; /* FIS LBA (31:24) */ 18417 fis->d.lbaMidExp = 0; /* FIS LBA (39:32) */ 18418 fis->d.lbaHighExp = 0; /* FIS LBA (47:40) */ 18419 fis->d.featuresExp = 0; /* FIS reserve */ 18420 if (satOrgIOContext->LoopNum == 1) 18421 { 18422 /* last loop */ 18423 fis->d.sectorCount = (bit8)(Remainder & 0xFF); /* FIS sector count (7:0) */ 18424 fis->d.sectorCountExp = (bit8)((Remainder & 0xFF00) >> 8); /* FIS sector count (15:8) */ 18425 } 18426 else 18427 { 18428 fis->d.sectorCount = 0xFF; /* FIS sector count (7:0) */ 18429 fis->d.sectorCountExp = 0xFF; /* FIS sector count (15:8) */ 18430 } 18431 fis->d.reserved4 = 0; 18432 fis->d.control = 0; /* FIS HOB bit clear */ 18433 fis->d.reserved5 = 0; 18434 18435 agRequestType = AGSA_SATA_PROTOCOL_NON_DATA; 18436 18437 break; 18438 18439 default: 18440 SM_DBG1(("smsatChainedWriteNVerify_Verify: error incorrect ata command 0x%x!!!\n", satIOContext->ATACmd)); 18441 return SM_RC_FAILURE; 18442 break; 18443 } 18444 18445 /* Initialize CB for SATA completion. 18446 */ 18447 /* chained data */ 18448 satIOContext->satCompleteCB = &smsatChainedWriteNVerifyCB; 18449 18450 18451 /* 18452 * Prepare SGL and send FIS to LL layer. 18453 */ 18454 satIOContext->reqType = agRequestType; /* Save it */ 18455 18456 status = smsataLLIOStart( smRoot, 18457 smIORequest, 18458 smDeviceHandle, 18459 smScsiRequest, 18460 satIOContext); 18461 18462 SM_DBG5(("smsatChainedWriteNVerify_Verify: return\n")); 18463 return (status); 18464 } 18465 18466 osGLOBAL bit32 18467 smsatChainedVerify( 18468 smRoot_t *smRoot, 18469 smIORequest_t *smIORequest, 18470 smDeviceHandle_t *smDeviceHandle, 18471 smScsiInitiatorRequest_t *smScsiRequest, 18472 smSatIOContext_t *satIOContext 18473 ) 18474 { 18475 bit32 status; 18476 smSatIOContext_t *satOrgIOContext = agNULL; 18477 agsaFisRegHostToDevice_t *fis; 18478 bit32 agRequestType = AGSA_SATA_PROTOCOL_NON_DATA; 18479 bit32 lba = 0; 18480 bit32 DenomTL = 0xFF; 18481 bit32 Remainder = 0; 18482 bit8 LBA[4]; /* 0 MSB, 3 LSB */ 18483 18484 SM_DBG2(("smsatChainedVerify: start\n")); 18485 fis = satIOContext->pFis; 18486 satOrgIOContext = satIOContext->satOrgIOContext; 18487 sm_memset(LBA,0, sizeof(LBA)); 18488 switch (satOrgIOContext->ATACmd) 18489 { 18490 case SAT_READ_VERIFY_SECTORS: 18491 DenomTL = 0xFF; 18492 break; 18493 case SAT_READ_VERIFY_SECTORS_EXT: 18494 DenomTL = 0xFFFF; 18495 break; 18496 default: 18497 SM_DBG1(("satChainedVerify: error incorrect ata command 0x%x!!!\n", satIOContext->ATACmd)); 18498 return tiError; 18499 break; 18500 } 18501 18502 Remainder = satOrgIOContext->OrgTL % DenomTL; 18503 satOrgIOContext->currentLBA = satOrgIOContext->currentLBA + DenomTL; 18504 lba = satOrgIOContext->currentLBA; 18505 18506 LBA[0] = (bit8)((lba & 0xF000) >> (8 * 3)); /* MSB */ 18507 LBA[1] = (bit8)((lba & 0xF00) >> (8 * 2)); 18508 LBA[2] = (bit8)((lba & 0xF0) >> 8); 18509 LBA[3] = (bit8)(lba & 0xF); /* LSB */ 18510 18511 switch (satOrgIOContext->ATACmd) 18512 { 18513 case SAT_READ_VERIFY_SECTORS: 18514 fis->h.fisType = 0x27; /* Reg host to device */ 18515 fis->h.c_pmPort = 0x80; /* C bit is set */ 18516 fis->h.command = SAT_READ_VERIFY_SECTORS; /* 0x40 */ 18517 fis->h.features = 0; /* FIS reserve */ 18518 fis->d.lbaLow = LBA[3]; /* FIS LBA (7 :0 ) */ 18519 fis->d.lbaMid = LBA[2]; /* FIS LBA (15:8 ) */ 18520 fis->d.lbaHigh = LBA[1]; /* FIS LBA (23:16) */ 18521 18522 /* FIS LBA mode set LBA (27:24) */ 18523 fis->d.device = (bit8)((0x4 << 4) | (LBA[0] & 0xF)); 18524 18525 fis->d.lbaLowExp = 0; 18526 fis->d.lbaMidExp = 0; 18527 fis->d.lbaHighExp = 0; 18528 fis->d.featuresExp = 0; 18529 if (satOrgIOContext->LoopNum == 1) 18530 { 18531 /* last loop */ 18532 fis->d.sectorCount = (bit8)Remainder; /* FIS sector count (7:0) */ 18533 } 18534 else 18535 { 18536 fis->d.sectorCount = 0xFF; /* FIS sector count (7:0) */ 18537 } 18538 fis->d.sectorCountExp = 0; 18539 fis->d.reserved4 = 0; 18540 fis->d.control = 0; /* FIS HOB bit clear */ 18541 fis->d.reserved5 = 0; 18542 18543 agRequestType = AGSA_SATA_PROTOCOL_NON_DATA; 18544 18545 break; 18546 case SAT_READ_VERIFY_SECTORS_EXT: 18547 fis->h.fisType = 0x27; /* Reg host to device */ 18548 fis->h.c_pmPort = 0x80; /* C Bit is set */ 18549 fis->h.command = SAT_READ_VERIFY_SECTORS_EXT; /* 0x42 */ 18550 fis->h.features = 0; /* FIS reserve */ 18551 fis->d.lbaLow = LBA[3]; /* FIS LBA (7 :0 ) */ 18552 fis->d.lbaMid = LBA[2]; /* FIS LBA (15:8 ) */ 18553 fis->d.lbaHigh = LBA[1]; /* FIS LBA (23:16) */ 18554 fis->d.device = 0x40; /* FIS LBA mode set */ 18555 fis->d.lbaLowExp = LBA[0]; /* FIS LBA (31:24) */ 18556 fis->d.lbaMidExp = 0; /* FIS LBA (39:32) */ 18557 fis->d.lbaHighExp = 0; /* FIS LBA (47:40) */ 18558 fis->d.featuresExp = 0; /* FIS reserve */ 18559 if (satOrgIOContext->LoopNum == 1) 18560 { 18561 /* last loop */ 18562 fis->d.sectorCount = (bit8)(Remainder & 0xFF); /* FIS sector count (7:0) */ 18563 fis->d.sectorCountExp = (bit8)((Remainder & 0xFF00) >> 8); /* FIS sector count (15:8) */ 18564 } 18565 else 18566 { 18567 fis->d.sectorCount = 0xFF; /* FIS sector count (7:0) */ 18568 fis->d.sectorCountExp = 0xFF; /* FIS sector count (15:8) */ 18569 } 18570 fis->d.reserved4 = 0; 18571 fis->d.control = 0; /* FIS HOB bit clear */ 18572 fis->d.reserved5 = 0; 18573 18574 agRequestType = AGSA_SATA_PROTOCOL_NON_DATA; 18575 18576 break; 18577 18578 default: 18579 SM_DBG1(("satChainedVerify: error incorrect ata command 0x%x!!!\n", satIOContext->ATACmd)); 18580 return tiError; 18581 break; 18582 } 18583 18584 /* Initialize CB for SATA completion. 18585 */ 18586 /* chained data */ 18587 satIOContext->satCompleteCB = &smsatChainedVerifyCB; 18588 18589 18590 /* 18591 * Prepare SGL and send FIS to LL layer. 18592 */ 18593 satIOContext->reqType = agRequestType; /* Save it */ 18594 18595 status = smsataLLIOStart( smRoot, 18596 smIORequest, 18597 smDeviceHandle, 18598 smScsiRequest, 18599 satIOContext); 18600 18601 SM_DBG5(("satChainedVerify: return\n")); 18602 return (status); 18603 } 18604 18605 osGLOBAL bit32 18606 smsatWriteSame10_1( 18607 smRoot_t *smRoot, 18608 smIORequest_t *smIORequest, 18609 smDeviceHandle_t *smDeviceHandle, 18610 smScsiInitiatorRequest_t *smScsiRequest, 18611 smSatIOContext_t *satIOContext, 18612 bit32 lba 18613 ) 18614 { 18615 /* 18616 sends SAT_WRITE_DMA_EXT 18617 */ 18618 18619 bit32 status; 18620 bit32 agRequestType; 18621 agsaFisRegHostToDevice_t *fis; 18622 bit8 lba1, lba2 ,lba3, lba4; 18623 18624 SM_DBG5(("smsatWriteSame10_1: start\n")); 18625 fis = satIOContext->pFis; 18626 /* MSB */ 18627 lba1 = (bit8)((lba & 0xFF000000) >> (8*3)); 18628 lba2 = (bit8)((lba & 0x00FF0000) >> (8*2)); 18629 lba3 = (bit8)((lba & 0x0000FF00) >> (8*1)); 18630 /* LSB */ 18631 lba4 = (bit8)(lba & 0x000000FF); 18632 /* SAT_WRITE_DMA_EXT */ 18633 fis->h.fisType = 0x27; /* Reg host to device */ 18634 fis->h.c_pmPort = 0x80; /* C Bit is set */ 18635 fis->h.command = SAT_WRITE_DMA_EXT; /* 0x35 */ 18636 fis->h.features = 0; /* FIS reserve */ 18637 fis->d.lbaLow = lba4; /* FIS LBA (7 :0 ) */ 18638 fis->d.lbaMid = lba3; /* FIS LBA (15:8 ) */ 18639 fis->d.lbaHigh = lba2; /* FIS LBA (23:16) */ 18640 fis->d.device = 0x40; /* FIS LBA mode set */ 18641 fis->d.lbaLowExp = lba1; /* FIS LBA (31:24) */ 18642 fis->d.lbaMidExp = 0; /* FIS LBA (39:32) */ 18643 fis->d.lbaHighExp = 0; /* FIS LBA (47:40) */ 18644 fis->d.featuresExp = 0; /* FIS reserve */ 18645 /* one sector at a time */ 18646 fis->d.sectorCount = 1; /* FIS sector count (7:0) */ 18647 fis->d.sectorCountExp = 0; /* FIS sector count (15:8) */ 18648 fis->d.reserved4 = 0; 18649 fis->d.control = 0; /* FIS HOB bit clear */ 18650 fis->d.reserved5 = 0; 18651 agRequestType = AGSA_SATA_PROTOCOL_DMA_WRITE; 18652 /* Initialize CB for SATA completion. 18653 */ 18654 satIOContext->satCompleteCB = &smsatWriteSame10CB; 18655 /* 18656 * Prepare SGL and send FIS to LL layer. 18657 */ 18658 satIOContext->reqType = agRequestType; /* Save it */ 18659 status = smsataLLIOStart( smRoot, 18660 smIORequest, 18661 smDeviceHandle, 18662 smScsiRequest, 18663 satIOContext); 18664 SM_DBG5(("smsatWriteSame10_1 return status %d\n", status)); 18665 return status; 18666 } 18667 18668 18669 osGLOBAL bit32 18670 smsatWriteSame10_2( 18671 smRoot_t *smRoot, 18672 smIORequest_t *smIORequest, 18673 smDeviceHandle_t *smDeviceHandle, 18674 smScsiInitiatorRequest_t *smScsiRequest, 18675 smSatIOContext_t *satIOContext, 18676 bit32 lba 18677 ) 18678 { 18679 /* 18680 sends SAT_WRITE_SECTORS_EXT 18681 */ 18682 18683 bit32 status; 18684 bit32 agRequestType; 18685 agsaFisRegHostToDevice_t *fis; 18686 bit8 lba1, lba2 ,lba3, lba4; 18687 18688 SM_DBG5(("smsatWriteSame10_2: start\n")); 18689 fis = satIOContext->pFis; 18690 /* MSB */ 18691 lba1 = (bit8)((lba & 0xFF000000) >> (8*3)); 18692 lba2 = (bit8)((lba & 0x00FF0000) >> (8*2)); 18693 lba3 = (bit8)((lba & 0x0000FF00) >> (8*1)); 18694 /* LSB */ 18695 lba4 = (bit8)(lba & 0x000000FF); 18696 /* SAT_WRITE_SECTORS_EXT */ 18697 fis->h.fisType = 0x27; /* Reg host to device */ 18698 fis->h.c_pmPort = 0x80; /* C Bit is set */ 18699 fis->h.command = SAT_WRITE_SECTORS_EXT; /* 0x34 */ 18700 fis->h.features = 0; /* FIS reserve */ 18701 fis->d.lbaLow = lba4; /* FIS LBA (7 :0 ) */ 18702 fis->d.lbaMid = lba3; /* FIS LBA (15:8 ) */ 18703 fis->d.lbaHigh = lba2; /* FIS LBA (23:16) */ 18704 fis->d.device = 0x40; /* FIS LBA mode set */ 18705 fis->d.lbaLowExp = lba1; /* FIS LBA (31:24) */ 18706 fis->d.lbaMidExp = 0; /* FIS LBA (39:32) */ 18707 fis->d.lbaHighExp = 0; /* FIS LBA (47:40) */ 18708 fis->d.featuresExp = 0; /* FIS reserve */ 18709 /* one sector at a time */ 18710 fis->d.sectorCount = 1; /* FIS sector count (7:0) */ 18711 fis->d.sectorCountExp = 0; /* FIS sector count (15:8) */ 18712 fis->d.reserved4 = 0; 18713 fis->d.control = 0; /* FIS HOB bit clear */ 18714 fis->d.reserved5 = 0; 18715 agRequestType = AGSA_SATA_PROTOCOL_PIO_WRITE; 18716 /* Initialize CB for SATA completion. 18717 */ 18718 satIOContext->satCompleteCB = &smsatWriteSame10CB; 18719 /* 18720 * Prepare SGL and send FIS to LL layer. 18721 */ 18722 satIOContext->reqType = agRequestType; /* Save it */ 18723 status = smsataLLIOStart( smRoot, 18724 smIORequest, 18725 smDeviceHandle, 18726 smScsiRequest, 18727 satIOContext); 18728 SM_DBG5(("smsatWriteSame10_2 return status %d\n", status)); 18729 return status; 18730 } 18731 18732 18733 osGLOBAL bit32 18734 smsatWriteSame10_3( 18735 smRoot_t *smRoot, 18736 smIORequest_t *smIORequest, 18737 smDeviceHandle_t *smDeviceHandle, 18738 smScsiInitiatorRequest_t *smScsiRequest, 18739 smSatIOContext_t *satIOContext, 18740 bit32 lba 18741 ) 18742 { 18743 /* 18744 sends SAT_WRITE_FPDMA_QUEUED 18745 */ 18746 18747 bit32 status; 18748 bit32 agRequestType; 18749 agsaFisRegHostToDevice_t *fis; 18750 bit8 lba1, lba2 ,lba3, lba4; 18751 18752 SM_DBG5(("smsatWriteSame10_3: start\n")); 18753 fis = satIOContext->pFis; 18754 /* MSB */ 18755 lba1 = (bit8)((lba & 0xFF000000) >> (8*3)); 18756 lba2 = (bit8)((lba & 0x00FF0000) >> (8*2)); 18757 lba3 = (bit8)((lba & 0x0000FF00) >> (8*1)); 18758 /* LSB */ 18759 lba4 = (bit8)(lba & 0x000000FF); 18760 18761 /* SAT_WRITE_FPDMA_QUEUED */ 18762 fis->h.fisType = 0x27; /* Reg host to device */ 18763 fis->h.c_pmPort = 0x80; /* C Bit is set */ 18764 fis->h.command = SAT_WRITE_FPDMA_QUEUED; /* 0x61 */ 18765 18766 18767 /* one sector at a time */ 18768 fis->h.features = 1; /* FIS sector count (7:0) */ 18769 fis->d.featuresExp = 0; /* FIS sector count (15:8) */ 18770 18771 fis->d.lbaLow = lba4; /* FIS LBA (7 :0 ) */ 18772 fis->d.lbaMid = lba3; /* FIS LBA (15:8 ) */ 18773 fis->d.lbaHigh = lba2; /* FIS LBA (23:16) */ 18774 /* NO FUA bit in the WRITE SAME 10 */ 18775 fis->d.device = 0x40; /* FIS FUA clear */ 18776 fis->d.lbaLowExp = lba1; /* FIS LBA (31:24) */ 18777 fis->d.lbaMidExp = 0; /* FIS LBA (39:32) */ 18778 fis->d.lbaHighExp = 0; /* FIS LBA (47:40) */ 18779 fis->d.sectorCount = 0; /* Tag (7:3) set by LL layer */ 18780 fis->d.sectorCountExp = 0; 18781 fis->d.reserved4 = 0; 18782 fis->d.control = 0; /* FIS HOB bit clear */ 18783 fis->d.reserved5 = 0; 18784 agRequestType = AGSA_SATA_PROTOCOL_FPDMA_WRITE; 18785 18786 /* Initialize CB for SATA completion. 18787 */ 18788 satIOContext->satCompleteCB = &smsatWriteSame10CB; 18789 /* 18790 * Prepare SGL and send FIS to LL layer. 18791 */ 18792 satIOContext->reqType = agRequestType; /* Save it */ 18793 status = smsataLLIOStart( smRoot, 18794 smIORequest, 18795 smDeviceHandle, 18796 smScsiRequest, 18797 satIOContext); 18798 18799 SM_DBG5(("smsatWriteSame10_3 return status %d\n", status)); 18800 return status; 18801 } 18802 18803 osGLOBAL bit32 18804 smsatStartStopUnit_1( 18805 smRoot_t *smRoot, 18806 smIORequest_t *smIORequest, 18807 smDeviceHandle_t *smDeviceHandle, 18808 smScsiInitiatorRequest_t *smScsiRequest, 18809 smSatIOContext_t *satIOContext 18810 ) 18811 { 18812 /* 18813 SAT Rev 8, Table 48, 9.11.3 p55 18814 sends STANDBY 18815 */ 18816 bit32 status; 18817 bit32 agRequestType; 18818 agsaFisRegHostToDevice_t *fis; 18819 18820 SM_DBG5(("smsatStartStopUnit_1: start\n")); 18821 fis = satIOContext->pFis; 18822 /* STANDBY */ 18823 fis->h.fisType = 0x27; /* Reg host to device */ 18824 fis->h.c_pmPort = 0x80; /* C Bit is set */ 18825 fis->h.command = SAT_STANDBY; /* 0xE2 */ 18826 fis->h.features = 0; /* FIS features NA */ 18827 fis->d.lbaLow = 0; /* FIS LBA (7 :0 ) */ 18828 fis->d.lbaMid = 0; /* FIS LBA (15:8 ) */ 18829 fis->d.lbaHigh = 0; /* FIS LBA (23:16) */ 18830 fis->d.lbaLowExp = 0; 18831 fis->d.lbaMidExp = 0; 18832 fis->d.lbaHighExp = 0; 18833 fis->d.featuresExp = 0; 18834 fis->d.sectorCount = 0; /* FIS sector count (7:0) */ 18835 fis->d.sectorCountExp = 0; 18836 fis->d.reserved4 = 0; 18837 fis->d.device = 0; /* 0 */ 18838 fis->d.control = 0; /* FIS HOB bit clear */ 18839 fis->d.reserved5 = 0; 18840 18841 agRequestType = AGSA_SATA_PROTOCOL_NON_DATA; 18842 18843 /* Initialize CB for SATA completion. 18844 */ 18845 satIOContext->satCompleteCB = &smsatStartStopUnitCB; 18846 18847 /* 18848 * Prepare SGL and send FIS to LL layer. 18849 */ 18850 satIOContext->reqType = agRequestType; /* Save it */ 18851 18852 status = smsataLLIOStart( smRoot, 18853 smIORequest, 18854 smDeviceHandle, 18855 smScsiRequest, 18856 satIOContext); 18857 18858 SM_DBG5(("smsatStartStopUnit_1 return status %d\n", status)); 18859 return status; 18860 } 18861 18862 osGLOBAL bit32 18863 smsatSendDiagnostic_1( 18864 smRoot_t *smRoot, 18865 smIORequest_t *smIORequest, 18866 smDeviceHandle_t *smDeviceHandle, 18867 smScsiInitiatorRequest_t *smScsiRequest, 18868 smSatIOContext_t *satIOContext 18869 ) 18870 { 18871 /* 18872 SAT Rev9, Table29, p41 18873 send 2nd SAT_READ_VERIFY_SECTORS(_EXT) 18874 */ 18875 bit32 status; 18876 bit32 agRequestType; 18877 smDeviceData_t *pSatDevData; 18878 agsaFisRegHostToDevice_t *fis; 18879 18880 SM_DBG5(("smsatSendDiagnostic_1: start\n")); 18881 pSatDevData = satIOContext->pSatDevData; 18882 fis = satIOContext->pFis; 18883 /* 18884 sector count 1, LBA MAX 18885 */ 18886 if (pSatDevData->sat48BitSupport == agTRUE) 18887 { 18888 /* sends READ VERIFY SECTOR(S) EXT*/ 18889 fis->h.fisType = 0x27; /* Reg host to device */ 18890 fis->h.c_pmPort = 0x80; /* C Bit is set */ 18891 fis->h.command = SAT_READ_VERIFY_SECTORS_EXT;/* 0x42 */ 18892 fis->h.features = 0; /* FIS reserve */ 18893 fis->d.lbaLow = pSatDevData->satMaxLBA[7]; /* FIS LBA (7 :0 ) */ 18894 fis->d.lbaMid = pSatDevData->satMaxLBA[6]; /* FIS LBA (15:8 ) */ 18895 fis->d.lbaHigh = pSatDevData->satMaxLBA[5]; /* FIS LBA (23:16) */ 18896 fis->d.lbaLowExp = pSatDevData->satMaxLBA[4]; /* FIS LBA (31:24) */ 18897 fis->d.lbaMidExp = pSatDevData->satMaxLBA[3]; /* FIS LBA (39:32) */ 18898 fis->d.lbaHighExp = pSatDevData->satMaxLBA[2]; /* FIS LBA (47:40) */ 18899 fis->d.featuresExp = 0; /* FIS reserve */ 18900 fis->d.sectorCount = 1; /* FIS sector count (7:0) */ 18901 fis->d.sectorCountExp = 0; /* FIS sector count (15:8) */ 18902 fis->d.reserved4 = 0; 18903 fis->d.device = 0x40; /* 01000000 */ 18904 fis->d.control = 0; /* FIS HOB bit clear */ 18905 fis->d.reserved5 = 0; 18906 18907 } 18908 else 18909 { 18910 /* READ VERIFY SECTOR(S)*/ 18911 fis->h.fisType = 0x27; /* Reg host to device */ 18912 fis->h.c_pmPort = 0x80; /* C Bit is set */ 18913 fis->h.command = SAT_READ_VERIFY_SECTORS;/* 0x40 */ 18914 fis->h.features = 0; /* FIS features NA */ 18915 fis->d.lbaLow = pSatDevData->satMaxLBA[7]; /* FIS LBA (7 :0 ) */ 18916 fis->d.lbaMid = pSatDevData->satMaxLBA[6]; /* FIS LBA (15:8 ) */ 18917 fis->d.lbaHigh = pSatDevData->satMaxLBA[5]; /* FIS LBA (23:16) */ 18918 fis->d.lbaLowExp = 0; 18919 fis->d.lbaMidExp = 0; 18920 fis->d.lbaHighExp = 0; 18921 fis->d.featuresExp = 0; 18922 fis->d.sectorCount = 1; /* FIS sector count (7:0) */ 18923 fis->d.sectorCountExp = 0; 18924 fis->d.reserved4 = 0; 18925 fis->d.device = (bit8)((0x4 << 4) | (pSatDevData->satMaxLBA[4] & 0xF)); 18926 /* DEV and LBA 27:24 */ 18927 fis->d.control = 0; /* FIS HOB bit clear */ 18928 fis->d.reserved5 = 0; 18929 18930 } 18931 18932 agRequestType = AGSA_SATA_PROTOCOL_NON_DATA; 18933 18934 /* Initialize CB for SATA completion. 18935 */ 18936 satIOContext->satCompleteCB = &smsatSendDiagnosticCB; 18937 18938 /* 18939 * Prepare SGL and send FIS to LL layer. 18940 */ 18941 satIOContext->reqType = agRequestType; /* Save it */ 18942 18943 status = smsataLLIOStart( smRoot, 18944 smIORequest, 18945 smDeviceHandle, 18946 smScsiRequest, 18947 satIOContext); 18948 18949 18950 return status; 18951 } 18952 18953 osGLOBAL bit32 18954 smsatSendDiagnostic_2( 18955 smRoot_t *smRoot, 18956 smIORequest_t *smIORequest, 18957 smDeviceHandle_t *smDeviceHandle, 18958 smScsiInitiatorRequest_t *smScsiRequest, 18959 smSatIOContext_t *satIOContext 18960 ) 18961 { 18962 /* 18963 SAT Rev9, Table29, p41 18964 send 3rd SAT_READ_VERIFY_SECTORS(_EXT) 18965 */ 18966 bit32 status; 18967 bit32 agRequestType; 18968 smDeviceData_t *pSatDevData; 18969 agsaFisRegHostToDevice_t *fis; 18970 18971 SM_DBG5(("smsatSendDiagnostic_2: start\n")); 18972 18973 pSatDevData = satIOContext->pSatDevData; 18974 fis = satIOContext->pFis; 18975 /* 18976 sector count 1, LBA Random 18977 */ 18978 if (pSatDevData->sat48BitSupport == agTRUE) 18979 { 18980 /* sends READ VERIFY SECTOR(S) EXT*/ 18981 fis->h.fisType = 0x27; /* Reg host to device */ 18982 fis->h.c_pmPort = 0x80; /* C Bit is set */ 18983 fis->h.command = SAT_READ_VERIFY_SECTORS_EXT;/* 0x42 */ 18984 fis->h.features = 0; /* FIS reserve */ 18985 fis->d.lbaLow = 0x7F; /* FIS LBA (7 :0 ) */ 18986 fis->d.lbaMid = 0; /* FIS LBA (15:8 ) */ 18987 fis->d.lbaHigh = 0; /* FIS LBA (23:16) */ 18988 fis->d.lbaLowExp = 0; /* FIS LBA (31:24) */ 18989 fis->d.lbaMidExp = 0; /* FIS LBA (39:32) */ 18990 fis->d.lbaHighExp = 0; /* FIS LBA (47:40) */ 18991 fis->d.featuresExp = 0; /* FIS reserve */ 18992 fis->d.sectorCount = 1; /* FIS sector count (7:0) */ 18993 fis->d.sectorCountExp = 0; /* FIS sector count (15:8) */ 18994 fis->d.reserved4 = 0; 18995 fis->d.device = 0x40; /* 01000000 */ 18996 fis->d.control = 0; /* FIS HOB bit clear */ 18997 fis->d.reserved5 = 0; 18998 18999 } 19000 else 19001 { 19002 /* READ VERIFY SECTOR(S)*/ 19003 fis->h.fisType = 0x27; /* Reg host to device */ 19004 fis->h.c_pmPort = 0x80; /* C Bit is set */ 19005 fis->h.command = SAT_READ_VERIFY_SECTORS;/* 0x40 */ 19006 fis->h.features = 0; /* FIS features NA */ 19007 fis->d.lbaLow = 0x7F; /* FIS LBA (7 :0 ) */ 19008 fis->d.lbaMid = 0; /* FIS LBA (15:8 ) */ 19009 fis->d.lbaHigh = 0; /* FIS LBA (23:16) */ 19010 fis->d.lbaLowExp = 0; 19011 fis->d.lbaMidExp = 0; 19012 fis->d.lbaHighExp = 0; 19013 fis->d.featuresExp = 0; 19014 fis->d.sectorCount = 1; /* FIS sector count (7:0) */ 19015 fis->d.sectorCountExp = 0; 19016 fis->d.reserved4 = 0; 19017 fis->d.device = 0x40; /* FIS LBA mode set 01000000 */ 19018 fis->d.control = 0; /* FIS HOB bit clear */ 19019 fis->d.reserved5 = 0; 19020 19021 } 19022 19023 agRequestType = AGSA_SATA_PROTOCOL_NON_DATA; 19024 19025 /* Initialize CB for SATA completion. 19026 */ 19027 satIOContext->satCompleteCB = &smsatSendDiagnosticCB; 19028 19029 /* 19030 * Prepare SGL and send FIS to LL layer. 19031 */ 19032 satIOContext->reqType = agRequestType; /* Save it */ 19033 19034 status = smsataLLIOStart( smRoot, 19035 smIORequest, 19036 smDeviceHandle, 19037 smScsiRequest, 19038 satIOContext); 19039 19040 19041 return status; 19042 } 19043 19044 osGLOBAL bit32 19045 smsatModeSelect6n10_1( 19046 smRoot_t *smRoot, 19047 smIORequest_t *smIORequest, 19048 smDeviceHandle_t *smDeviceHandle, 19049 smScsiInitiatorRequest_t *smScsiRequest, 19050 smSatIOContext_t *satIOContext 19051 ) 19052 { 19053 /* sends either ATA SET FEATURES based on DRA bit */ 19054 bit32 status; 19055 bit32 agRequestType; 19056 agsaFisRegHostToDevice_t *fis; 19057 bit8 *pLogPage; /* Log Page data buffer */ 19058 bit32 StartingIndex = 0; 19059 19060 fis = satIOContext->pFis; 19061 pLogPage = (bit8 *) smScsiRequest->sglVirtualAddr; 19062 SM_DBG5(("smsatModeSelect6n10_1: start\n")); 19063 19064 if (pLogPage[3] == 8) 19065 { 19066 /* mode parameter block descriptor exists */ 19067 StartingIndex = 12; 19068 } 19069 else 19070 { 19071 /* mode parameter block descriptor does not exist */ 19072 StartingIndex = 4; 19073 } 19074 19075 /* sends ATA SET FEATURES based on DRA bit */ 19076 if ( !(pLogPage[StartingIndex + 12] & SCSI_MODE_SELECT6_DRA_MASK) ) 19077 { 19078 SM_DBG5(("smsatModeSelect6n10_1: enable read look-ahead feature\n")); 19079 /* sends SET FEATURES */ 19080 fis->h.fisType = 0x27; /* Reg host to device */ 19081 fis->h.c_pmPort = 0x80; /* C Bit is set */ 19082 19083 fis->h.command = SAT_SET_FEATURES; /* 0xEF */ 19084 fis->h.features = 0xAA; /* enable read look-ahead */ 19085 fis->d.lbaLow = 0; /* */ 19086 fis->d.lbaMid = 0; /* */ 19087 fis->d.lbaHigh = 0; /* */ 19088 fis->d.device = 0; /* */ 19089 fis->d.lbaLowExp = 0; /* */ 19090 fis->d.lbaMidExp = 0; /* */ 19091 fis->d.lbaHighExp = 0; /* */ 19092 fis->d.featuresExp = 0; /* */ 19093 fis->d.sectorCount = 0; /* */ 19094 fis->d.sectorCountExp = 0; /* */ 19095 fis->d.reserved4 = 0; 19096 fis->d.control = 0; /* FIS HOB bit clear */ 19097 fis->d.reserved5 = 0; 19098 19099 agRequestType = AGSA_SATA_PROTOCOL_NON_DATA; 19100 19101 /* Initialize CB for SATA completion. 19102 */ 19103 satIOContext->satCompleteCB = &smsatModeSelect6n10CB; 19104 19105 /* 19106 * Prepare SGL and send FIS to LL layer. 19107 */ 19108 satIOContext->reqType = agRequestType; /* Save it */ 19109 19110 status = smsataLLIOStart( smRoot, 19111 smIORequest, 19112 smDeviceHandle, 19113 smScsiRequest, 19114 satIOContext); 19115 return status; 19116 } 19117 else 19118 { 19119 SM_DBG5(("smsatModeSelect6n10_1: disable read look-ahead feature\n")); 19120 /* sends SET FEATURES */ 19121 fis->h.fisType = 0x27; /* Reg host to device */ 19122 fis->h.c_pmPort = 0x80; /* C Bit is set */ 19123 19124 fis->h.command = SAT_SET_FEATURES; /* 0xEF */ 19125 fis->h.features = 0x55; /* disable read look-ahead */ 19126 fis->d.lbaLow = 0; /* */ 19127 fis->d.lbaMid = 0; /* */ 19128 fis->d.lbaHigh = 0; /* */ 19129 fis->d.device = 0; /* */ 19130 fis->d.lbaLowExp = 0; /* */ 19131 fis->d.lbaMidExp = 0; /* */ 19132 fis->d.lbaHighExp = 0; /* */ 19133 fis->d.featuresExp = 0; /* */ 19134 fis->d.sectorCount = 0; /* */ 19135 fis->d.sectorCountExp = 0; /* */ 19136 fis->d.reserved4 = 0; 19137 fis->d.control = 0; /* FIS HOB bit clear */ 19138 fis->d.reserved5 = 0; 19139 19140 agRequestType = AGSA_SATA_PROTOCOL_NON_DATA; 19141 19142 /* Initialize CB for SATA completion. 19143 */ 19144 satIOContext->satCompleteCB = &smsatModeSelect6n10CB; 19145 19146 /* 19147 * Prepare SGL and send FIS to LL layer. 19148 */ 19149 satIOContext->reqType = agRequestType; /* Save it */ 19150 19151 status = smsataLLIOStart( smRoot, 19152 smIORequest, 19153 smDeviceHandle, 19154 smScsiRequest, 19155 satIOContext); 19156 return status; 19157 } 19158 } 19159 19160 19161 osGLOBAL bit32 19162 smsatLogSense_1( 19163 smRoot_t *smRoot, 19164 smIORequest_t *smIORequest, 19165 smDeviceHandle_t *smDeviceHandle, 19166 smScsiInitiatorRequest_t *smScsiRequest, 19167 smSatIOContext_t *satIOContext 19168 ) 19169 { 19170 bit32 status; 19171 bit32 agRequestType; 19172 smDeviceData_t *pSatDevData; 19173 agsaFisRegHostToDevice_t *fis; 19174 19175 pSatDevData = satIOContext->pSatDevData; 19176 fis = satIOContext->pFis; 19177 19178 SM_DBG5(("smsatLogSense_1: start\n")); 19179 19180 /* SAT Rev 8, 10.2.4 p74 */ 19181 if ( pSatDevData->sat48BitSupport == agTRUE ) 19182 { 19183 SM_DBG5(("smsatLogSense_1: case 2-1 sends READ LOG EXT\n")); 19184 /* sends READ LOG EXT */ 19185 fis->h.fisType = 0x27; /* Reg host to device */ 19186 fis->h.c_pmPort = 0x80; /* C Bit is set */ 19187 19188 fis->h.command = SAT_READ_LOG_EXT; /* 0x2F */ 19189 fis->h.features = 0; /* FIS reserve */ 19190 fis->d.lbaLow = 0x07; /* 0x07 */ 19191 fis->d.lbaMid = 0; /* */ 19192 fis->d.lbaHigh = 0; /* */ 19193 fis->d.device = 0; /* */ 19194 fis->d.lbaLowExp = 0; /* */ 19195 fis->d.lbaMidExp = 0; /* */ 19196 fis->d.lbaHighExp = 0; /* */ 19197 fis->d.featuresExp = 0; /* FIS reserve */ 19198 fis->d.sectorCount = 0x01; /* 1 sector counts */ 19199 fis->d.sectorCountExp = 0x00; /* 1 sector counts */ 19200 fis->d.reserved4 = 0; 19201 fis->d.control = 0; /* FIS HOB bit clear */ 19202 fis->d.reserved5 = 0; 19203 19204 agRequestType = AGSA_SATA_PROTOCOL_PIO_READ; 19205 19206 /* Initialize CB for SATA completion. 19207 */ 19208 satIOContext->satCompleteCB = &smsatLogSenseCB; 19209 19210 /* 19211 * Prepare SGL and send FIS to LL layer. 19212 */ 19213 satIOContext->reqType = agRequestType; /* Save it */ 19214 19215 status = smsataLLIOStart( smRoot, 19216 smIORequest, 19217 smDeviceHandle, 19218 smScsiRequest, 19219 satIOContext); 19220 return status; 19221 19222 } 19223 else 19224 { 19225 SM_DBG5(("smsatLogSense_1: case 2-2 sends SMART READ LOG\n")); 19226 /* sends SMART READ LOG */ 19227 fis->h.fisType = 0x27; /* Reg host to device */ 19228 fis->h.c_pmPort = 0x80; /* C Bit is set */ 19229 19230 fis->h.command = SAT_SMART; /* 0x2F */ 19231 fis->h.features = SAT_SMART_READ_LOG; /* 0xd5 */ 19232 fis->d.lbaLow = 0x06; /* 0x06 */ 19233 fis->d.lbaMid = 0x00; /* 0x4f */ 19234 fis->d.lbaHigh = 0x00; /* 0xc2 */ 19235 fis->d.device = 0; /* */ 19236 fis->d.lbaLowExp = 0; /* */ 19237 fis->d.lbaMidExp = 0; /* */ 19238 fis->d.lbaHighExp = 0; /* */ 19239 fis->d.featuresExp = 0; /* FIS reserve */ 19240 fis->d.sectorCount = 0x01; /* */ 19241 fis->d.sectorCountExp = 0x00; /* */ 19242 fis->d.reserved4 = 0; 19243 fis->d.control = 0; /* FIS HOB bit clear */ 19244 fis->d.reserved5 = 0; 19245 19246 agRequestType = AGSA_SATA_PROTOCOL_PIO_READ; 19247 19248 /* Initialize CB for SATA completion. 19249 */ 19250 satIOContext->satCompleteCB = &smsatLogSenseCB; 19251 19252 /* 19253 * Prepare SGL and send FIS to LL layer. 19254 */ 19255 satIOContext->reqType = agRequestType; /* Save it */ 19256 19257 status = smsataLLIOStart( smRoot, 19258 smIORequest, 19259 smDeviceHandle, 19260 smScsiRequest, 19261 satIOContext); 19262 return status; 19263 19264 } 19265 } 19266 19267 osGLOBAL bit32 19268 smsatReassignBlocks_2( 19269 smRoot_t *smRoot, 19270 smIORequest_t *smIORequest, 19271 smDeviceHandle_t *smDeviceHandle, 19272 smScsiInitiatorRequest_t *smScsiRequest, 19273 smSatIOContext_t *satIOContext, 19274 bit8 *LBA 19275 ) 19276 { 19277 /* 19278 assumes all LBA fits in ATA command; no boundary condition is checked here yet 19279 tiScsiRequest is TD generated for writing 19280 */ 19281 bit32 status; 19282 bit32 agRequestType; 19283 smDeviceData_t *pSatDevData; 19284 smScsiRspSense_t *pSense; 19285 agsaFisRegHostToDevice_t *fis; 19286 19287 pSense = satIOContext->pSense; 19288 pSatDevData = satIOContext->pSatDevData; 19289 fis = satIOContext->pFis; 19290 SM_DBG5(("smsatReassignBlocks_2: start\n")); 19291 19292 if (pSatDevData->satDMASupport == agTRUE && pSatDevData->satDMAEnabled == agTRUE) 19293 { 19294 /* case 2 */ 19295 /* WRITE DMA*/ 19296 /* can't fit the transfer length */ 19297 SM_DBG5(("smsatReassignBlocks_2: case 2\n")); 19298 fis->h.fisType = 0x27; /* Reg host to device */ 19299 fis->h.c_pmPort = 0x80; /* C bit is set */ 19300 fis->h.command = SAT_WRITE_DMA; /* 0xCA */ 19301 fis->h.features = 0; /* FIS reserve */ 19302 fis->d.lbaLow = LBA[7]; /* FIS LBA (7 :0 ) */ 19303 fis->d.lbaMid = LBA[6]; /* FIS LBA (15:8 ) */ 19304 fis->d.lbaHigh = LBA[5]; /* FIS LBA (23:16) */ 19305 19306 /* FIS LBA mode set LBA (27:24) */ 19307 fis->d.device = (bit8)((0x4 << 4) | (LBA[4] & 0xF)); 19308 19309 fis->d.lbaLowExp = 0; 19310 fis->d.lbaMidExp = 0; 19311 fis->d.lbaHighExp = 0; 19312 fis->d.featuresExp = 0; 19313 fis->d.sectorCount = 1; /* FIS sector count (7:0) */ 19314 fis->d.sectorCountExp = 0; 19315 fis->d.reserved4 = 0; 19316 fis->d.control = 0; /* FIS HOB bit clear */ 19317 fis->d.reserved5 = 0; 19318 19319 agRequestType = AGSA_SATA_PROTOCOL_DMA_WRITE; 19320 satIOContext->ATACmd = SAT_WRITE_DMA; 19321 } 19322 else 19323 { 19324 /* case 1 */ 19325 /* WRITE MULTIPLE or WRITE SECTOR(S) */ 19326 /* WRITE SECTORS for easier implemetation */ 19327 /* can't fit the transfer length */ 19328 SM_DBG5(("smsatReassignBlocks_2: case 1\n")); 19329 fis->h.fisType = 0x27; /* Reg host to device */ 19330 fis->h.c_pmPort = 0x80; /* C bit is set */ 19331 fis->h.command = SAT_WRITE_SECTORS; /* 0x30 */ 19332 fis->h.features = 0; /* FIS reserve */ 19333 fis->d.lbaLow = LBA[7]; /* FIS LBA (7 :0 ) */ 19334 fis->d.lbaMid = LBA[6]; /* FIS LBA (15:8 ) */ 19335 fis->d.lbaHigh = LBA[7]; /* FIS LBA (23:16) */ 19336 19337 /* FIS LBA mode set LBA (27:24) */ 19338 fis->d.device = (bit8)((0x4 << 4) | (LBA[4] & 0xF)); 19339 19340 fis->d.lbaLowExp = 0; 19341 fis->d.lbaMidExp = 0; 19342 fis->d.lbaHighExp = 0; 19343 fis->d.featuresExp = 0; 19344 fis->d.sectorCount = 1; /* FIS sector count (7:0) */ 19345 fis->d.sectorCountExp = 0; 19346 fis->d.reserved4 = 0; 19347 fis->d.control = 0; /* FIS HOB bit clear */ 19348 fis->d.reserved5 = 0; 19349 19350 agRequestType = AGSA_SATA_PROTOCOL_PIO_WRITE; 19351 satIOContext->ATACmd = SAT_WRITE_SECTORS; 19352 } 19353 19354 /* case 3 and 4 */ 19355 if (pSatDevData->sat48BitSupport == agTRUE) 19356 { 19357 if (pSatDevData->satDMASupport == agTRUE && pSatDevData->satDMAEnabled == agTRUE) 19358 { 19359 /* case 3 */ 19360 /* WRITE DMA EXT or WRITE DMA FUA EXT */ 19361 SM_DBG5(("smsatReassignBlocks_2: case 3\n")); 19362 fis->h.fisType = 0x27; /* Reg host to device */ 19363 fis->h.c_pmPort = 0x80; /* C Bit is set */ 19364 19365 /* SAT_WRITE_DMA_FUA_EXT is optional and we don't support it */ 19366 fis->h.command = SAT_WRITE_DMA_EXT; /* 0x35 */ 19367 satIOContext->ATACmd = SAT_WRITE_DMA_EXT; 19368 19369 fis->h.features = 0; /* FIS reserve */ 19370 fis->d.lbaLow = LBA[7]; /* FIS LBA (7 :0 ) */ 19371 fis->d.lbaMid = LBA[6]; /* FIS LBA (15:8 ) */ 19372 fis->d.lbaHigh = LBA[5]; /* FIS LBA (23:16) */ 19373 fis->d.device = 0x40; /* FIS LBA mode set */ 19374 fis->d.lbaLowExp = LBA[4]; /* FIS LBA (31:24) */ 19375 fis->d.lbaMidExp = LBA[3]; /* FIS LBA (39:32) */ 19376 fis->d.lbaHighExp = LBA[2]; /* FIS LBA (47:40) */ 19377 fis->d.featuresExp = 0; /* FIS reserve */ 19378 fis->d.sectorCount = 1; /* FIS sector count (7:0) */ 19379 fis->d.sectorCountExp = 0; /* FIS sector count (15:8) */ 19380 fis->d.reserved4 = 0; 19381 fis->d.control = 0; /* FIS HOB bit clear */ 19382 fis->d.reserved5 = 0; 19383 19384 agRequestType = AGSA_SATA_PROTOCOL_DMA_WRITE; 19385 } 19386 else 19387 { 19388 /* case 4 */ 19389 /* WRITE MULTIPLE EXT or WRITE MULTIPLE FUA EXT or WRITE SECTOR(S) EXT */ 19390 /* WRITE SECTORS EXT for easier implemetation */ 19391 SM_DBG5(("smsatReassignBlocks_2: case 4\n")); 19392 fis->h.fisType = 0x27; /* Reg host to device */ 19393 fis->h.c_pmPort = 0x80; /* C Bit is set */ 19394 fis->h.command = SAT_WRITE_SECTORS_EXT; /* 0x34 */ 19395 19396 fis->h.features = 0; /* FIS reserve */ 19397 fis->d.lbaLow = LBA[7]; /* FIS LBA (7 :0 ) */ 19398 fis->d.lbaMid = LBA[6]; /* FIS LBA (15:8 ) */ 19399 fis->d.lbaHigh = LBA[5]; /* FIS LBA (23:16) */ 19400 fis->d.device = 0x40; /* FIS LBA mode set */ 19401 fis->d.lbaLowExp = LBA[4]; /* FIS LBA (31:24) */ 19402 fis->d.lbaMidExp = LBA[3]; /* FIS LBA (39:32) */ 19403 fis->d.lbaHighExp = LBA[2]; /* FIS LBA (47:40) */ 19404 fis->d.featuresExp = 0; /* FIS reserve */ 19405 fis->d.sectorCount = 1; /* FIS sector count (7:0) */ 19406 fis->d.sectorCountExp = 0; /* FIS sector count (15:8) */ 19407 fis->d.reserved4 = 0; 19408 fis->d.control = 0; /* FIS HOB bit clear */ 19409 fis->d.reserved5 = 0; 19410 19411 agRequestType = AGSA_SATA_PROTOCOL_PIO_WRITE; 19412 satIOContext->ATACmd = SAT_WRITE_SECTORS_EXT; 19413 } 19414 } 19415 /* case 5 */ 19416 if (pSatDevData->satNCQ == agTRUE) 19417 { 19418 /* WRITE FPDMA QUEUED */ 19419 if (pSatDevData->sat48BitSupport != agTRUE) 19420 { 19421 SM_DBG5(("smsatReassignBlocks_2: case 5 !!! error NCQ but 28 bit address support \n")); 19422 smsatSetSensePayload( pSense, 19423 SCSI_SNSKEY_HARDWARE_ERROR, 19424 0, 19425 SCSI_SNSCODE_WRITE_ERROR_AUTO_REALLOCATION_FAILED, 19426 satIOContext); 19427 19428 /*smEnqueueIO(smRoot, satIOContext);*/ 19429 19430 tdsmIOCompletedCB( smRoot, 19431 smIORequest, 19432 smIOSuccess, 19433 SCSI_STAT_CHECK_CONDITION, 19434 satIOContext->pSmSenseData, 19435 satIOContext->interruptContext ); 19436 return SM_RC_SUCCESS; 19437 } 19438 SM_DBG6(("satWrite10: case 5\n")); 19439 19440 /* Support 48-bit FPDMA addressing, use WRITE FPDMA QUEUE command */ 19441 19442 fis->h.fisType = 0x27; /* Reg host to device */ 19443 fis->h.c_pmPort = 0x80; /* C Bit is set */ 19444 fis->h.command = SAT_WRITE_FPDMA_QUEUED; /* 0x61 */ 19445 fis->h.features = 1; /* FIS sector count (7:0) */ 19446 fis->d.lbaLow = LBA[7]; /* FIS LBA (7 :0 ) */ 19447 fis->d.lbaMid = LBA[6]; /* FIS LBA (15:8 ) */ 19448 fis->d.lbaHigh = LBA[5]; /* FIS LBA (23:16) */ 19449 19450 /* Check FUA bit */ 19451 fis->d.device = 0x40; /* FIS FUA clear */ 19452 19453 fis->d.lbaLowExp = LBA[4]; /* FIS LBA (31:24) */ 19454 fis->d.lbaMidExp = LBA[3]; /* FIS LBA (39:32) */ 19455 fis->d.lbaHighExp = LBA[2]; /* FIS LBA (47:40) */ 19456 fis->d.featuresExp = 0; /* FIS sector count (15:8) */ 19457 fis->d.sectorCount = 0; /* Tag (7:3) set by LL layer */ 19458 fis->d.sectorCountExp = 0; 19459 fis->d.reserved4 = 0; 19460 fis->d.control = 0; /* FIS HOB bit clear */ 19461 fis->d.reserved5 = 0; 19462 19463 agRequestType = AGSA_SATA_PROTOCOL_FPDMA_WRITE; 19464 satIOContext->ATACmd = SAT_WRITE_FPDMA_QUEUED; 19465 } 19466 19467 satIOContext->satCompleteCB = &smsatReassignBlocksCB; 19468 19469 /* 19470 * Prepare SGL and send FIS to LL layer. 19471 */ 19472 satIOContext->reqType = agRequestType; /* Save it */ 19473 19474 status = smsataLLIOStart( smRoot, 19475 smIORequest, 19476 smDeviceHandle, 19477 /* not the original, should be the TD generated one */ 19478 smScsiRequest, 19479 satIOContext); 19480 return (status); 19481 } 19482 19483 osGLOBAL bit32 19484 smsatReassignBlocks_1( 19485 smRoot_t *smRoot, 19486 smIORequest_t *smIORequest, 19487 smDeviceHandle_t *smDeviceHandle, 19488 smScsiInitiatorRequest_t *smScsiRequest, 19489 smSatIOContext_t *satIOContext, 19490 smSatIOContext_t *satOrgIOContext 19491 ) 19492 { 19493 /* 19494 assumes all LBA fits in ATA command; no boundary condition is checked here yet 19495 tiScsiRequest is OS generated; needs for accessing parameter list 19496 */ 19497 bit32 agRequestType; 19498 smDeviceData_t *pSatDevData; 19499 smIniScsiCmnd_t *scsiCmnd; 19500 agsaFisRegHostToDevice_t *fis; 19501 bit8 *pParmList; /* Log Page data buffer */ 19502 bit8 LongLBA; 19503 bit8 LBA[8]; 19504 bit32 startingIndex; 19505 19506 pSatDevData = satIOContext->pSatDevData; 19507 scsiCmnd = &smScsiRequest->scsiCmnd; 19508 fis = satIOContext->pFis; 19509 pParmList = (bit8 *) smScsiRequest->sglVirtualAddr; 19510 SM_DBG5(("smsatReassignBlocks_1: start\n")); 19511 LongLBA = (bit8)(scsiCmnd->cdb[1] & SCSI_REASSIGN_BLOCKS_LONGLBA_MASK); 19512 sm_memset(LBA, 0, sizeof(LBA)); 19513 startingIndex = satOrgIOContext->ParmIndex; 19514 if (LongLBA == 0) 19515 { 19516 LBA[4] = pParmList[startingIndex]; 19517 LBA[5] = pParmList[startingIndex+1]; 19518 LBA[6] = pParmList[startingIndex+2]; 19519 LBA[7] = pParmList[startingIndex+3]; 19520 startingIndex = startingIndex + 4; 19521 } 19522 else 19523 { 19524 LBA[0] = pParmList[startingIndex]; 19525 LBA[1] = pParmList[startingIndex+1]; 19526 LBA[2] = pParmList[startingIndex+2]; 19527 LBA[3] = pParmList[startingIndex+3]; 19528 LBA[4] = pParmList[startingIndex+4]; 19529 LBA[5] = pParmList[startingIndex+5]; 19530 LBA[6] = pParmList[startingIndex+6]; 19531 LBA[7] = pParmList[startingIndex+7]; 19532 startingIndex = startingIndex + 8; 19533 } 19534 19535 if (pSatDevData->sat48BitSupport == agTRUE) 19536 { 19537 /* sends READ VERIFY SECTOR(S) EXT*/ 19538 fis->h.fisType = 0x27; /* Reg host to device */ 19539 fis->h.c_pmPort = 0x80; /* C Bit is set */ 19540 fis->h.command = SAT_READ_VERIFY_SECTORS_EXT;/* 0x42 */ 19541 fis->h.features = 0; /* FIS reserve */ 19542 fis->d.lbaLow = LBA[7]; /* FIS LBA (7 :0 ) */ 19543 fis->d.lbaMid = LBA[6]; /* FIS LBA (15:8 ) */ 19544 fis->d.lbaHigh = LBA[5]; /* FIS LBA (23:16) */ 19545 fis->d.lbaLowExp = LBA[4]; /* FIS LBA (31:24) */ 19546 fis->d.lbaMidExp = LBA[3]; /* FIS LBA (39:32) */ 19547 fis->d.lbaHighExp = LBA[2]; /* FIS LBA (47:40) */ 19548 fis->d.featuresExp = 0; /* FIS reserve */ 19549 fis->d.sectorCount = 1; /* FIS sector count (7:0) */ 19550 fis->d.sectorCountExp = 0; /* FIS sector count (15:8) */ 19551 fis->d.reserved4 = 0; 19552 fis->d.device = 0x40; /* 01000000 */ 19553 fis->d.control = 0; /* FIS HOB bit clear */ 19554 fis->d.reserved5 = 0; 19555 } 19556 else 19557 { 19558 /* READ VERIFY SECTOR(S)*/ 19559 fis->h.fisType = 0x27; /* Reg host to device */ 19560 fis->h.c_pmPort = 0x80; /* C Bit is set */ 19561 fis->h.command = SAT_READ_VERIFY_SECTORS;/* 0x40 */ 19562 fis->h.features = 0; /* FIS features NA */ 19563 fis->d.lbaLow = LBA[7]; /* FIS LBA (7 :0 ) */ 19564 fis->d.lbaMid = LBA[6]; /* FIS LBA (15:8 ) */ 19565 fis->d.lbaHigh = LBA[5]; /* FIS LBA (23:16) */ 19566 fis->d.lbaLowExp = 0; 19567 fis->d.lbaMidExp = 0; 19568 fis->d.lbaHighExp = 0; 19569 fis->d.featuresExp = 0; 19570 fis->d.sectorCount = 1; /* FIS sector count (7:0) */ 19571 fis->d.sectorCountExp = 0; 19572 fis->d.reserved4 = 0; 19573 fis->d.device = (bit8)((0x4 << 4) | (LBA[4] & 0xF)); 19574 /* DEV and LBA 27:24 */ 19575 fis->d.control = 0; /* FIS HOB bit clear */ 19576 fis->d.reserved5 = 0; 19577 } 19578 19579 sm_memcpy(satOrgIOContext->LBA, LBA, 8); 19580 satOrgIOContext->ParmIndex = startingIndex; 19581 19582 agRequestType = AGSA_SATA_PROTOCOL_NON_DATA; 19583 19584 /* Initialize CB for SATA completion. 19585 */ 19586 satIOContext->satCompleteCB = &smsatReassignBlocksCB; 19587 19588 /* 19589 * Prepare SGL and send FIS to LL layer. 19590 */ 19591 satIOContext->reqType = agRequestType; /* Save it */ 19592 19593 smsataLLIOStart( smRoot, 19594 smIORequest, 19595 smDeviceHandle, 19596 smScsiRequest, 19597 satIOContext); 19598 19599 return SM_RC_SUCCESS; 19600 } 19601 19602 osGLOBAL bit32 19603 smsatSendReadLogExt( 19604 smRoot_t *smRoot, 19605 smIORequest_t *smIORequest, 19606 smDeviceHandle_t *smDeviceHandle, 19607 smScsiInitiatorRequest_t *smScsiRequest, 19608 smSatIOContext_t *satIOContext 19609 ) 19610 { 19611 bit32 status; 19612 bit32 agRequestType; 19613 agsaFisRegHostToDevice_t *fis; 19614 19615 fis = satIOContext->pFis; 19616 SM_DBG1(("smsatSendReadLogExt: start\n")); 19617 fis->h.fisType = 0x27; /* Reg host to device */ 19618 fis->h.c_pmPort = 0x80; /* C Bit is set */ 19619 fis->h.command = SAT_READ_LOG_EXT; /* 0x2F */ 19620 fis->h.features = 0; /* FIS reserve */ 19621 fis->d.lbaLow = 0x10; /* Page number */ 19622 fis->d.lbaMid = 0; /* */ 19623 fis->d.lbaHigh = 0; /* */ 19624 fis->d.device = 0; /* DEV is ignored in SATA */ 19625 fis->d.lbaLowExp = 0; /* */ 19626 fis->d.lbaMidExp = 0; /* */ 19627 fis->d.lbaHighExp = 0; /* */ 19628 fis->d.featuresExp = 0; /* FIS reserve */ 19629 fis->d.sectorCount = 0x01; /* 1 sector counts*/ 19630 fis->d.sectorCountExp = 0x00; /* 1 sector counts */ 19631 fis->d.reserved4 = 0; 19632 fis->d.control = 0; /* FIS HOB bit clear */ 19633 fis->d.reserved5 = 0; 19634 19635 agRequestType = AGSA_SATA_PROTOCOL_PIO_READ; 19636 19637 /* Initialize CB for SATA completion. 19638 */ 19639 satIOContext->satCompleteCB = &smsatReadLogExtCB; 19640 19641 /* 19642 * Prepare SGL and send FIS to LL layer. 19643 */ 19644 satIOContext->reqType = agRequestType; /* Save it */ 19645 19646 status = smsataLLIOStart( smRoot, 19647 smIORequest, 19648 smDeviceHandle, 19649 smScsiRequest, 19650 satIOContext); 19651 19652 SM_DBG1(("smsatSendReadLogExt: end status %d!!!\n", status)); 19653 19654 return (status); 19655 } 19656 19657 osGLOBAL bit32 19658 smsatCheckPowerMode( 19659 smRoot_t *smRoot, 19660 smIORequest_t *smIORequest, 19661 smDeviceHandle_t *smDeviceHandle, 19662 smScsiInitiatorRequest_t *smScsiRequest, 19663 smSatIOContext_t *satIOContext 19664 ) 19665 { 19666 /* 19667 sends SAT_CHECK_POWER_MODE as a part of ABORT TASKMANGEMENT for NCQ commands 19668 internally generated - no directly corresponding scsi 19669 */ 19670 bit32 status; 19671 bit32 agRequestType; 19672 agsaFisRegHostToDevice_t *fis; 19673 19674 fis = satIOContext->pFis; 19675 SM_DBG1(("smsatCheckPowerMode: start\n")); 19676 /* 19677 * Send the ATA CHECK POWER MODE command. 19678 */ 19679 fis->h.fisType = 0x27; /* Reg host to device */ 19680 fis->h.c_pmPort = 0x80; /* C Bit is set */ 19681 fis->h.command = SAT_CHECK_POWER_MODE; /* 0xE5 */ 19682 fis->h.features = 0; 19683 fis->d.lbaLow = 0; 19684 fis->d.lbaMid = 0; 19685 fis->d.lbaHigh = 0; 19686 fis->d.device = 0; 19687 fis->d.lbaLowExp = 0; 19688 fis->d.lbaMidExp = 0; 19689 fis->d.lbaHighExp = 0; 19690 fis->d.featuresExp = 0; 19691 fis->d.sectorCount = 0; 19692 fis->d.sectorCountExp = 0; 19693 fis->d.reserved4 = 0; 19694 fis->d.control = 0; /* FIS HOB bit clear */ 19695 fis->d.reserved5 = 0; 19696 19697 agRequestType = AGSA_SATA_PROTOCOL_NON_DATA; 19698 19699 /* Initialize CB for SATA completion. 19700 */ 19701 satIOContext->satCompleteCB = &smsatCheckPowerModeCB; 19702 19703 /* 19704 * Prepare SGL and send FIS to LL layer. 19705 */ 19706 satIOContext->reqType = agRequestType; /* Save it */ 19707 19708 status = smsataLLIOStart( smRoot, 19709 smIORequest, 19710 smDeviceHandle, 19711 smScsiRequest, 19712 satIOContext); 19713 19714 SM_DBG5(("smsatCheckPowerMode: return\n")); 19715 19716 return status; 19717 } 19718 19719 osGLOBAL bit32 19720 smsatResetDevice( 19721 smRoot_t *smRoot, 19722 smIORequest_t *smIORequest, 19723 smDeviceHandle_t *smDeviceHandle, 19724 smScsiInitiatorRequest_t *smScsiRequest, /* NULL */ 19725 smSatIOContext_t *satIOContext 19726 ) 19727 { 19728 bit32 status; 19729 bit32 agRequestType; 19730 agsaFisRegHostToDevice_t *fis; 19731 #ifdef TD_DEBUG_ENABLE 19732 smIORequestBody_t *smIORequestBody; 19733 smSatInternalIo_t *satIntIoContext; 19734 #endif 19735 19736 fis = satIOContext->pFis; 19737 SM_DBG1(("smsatResetDevice: start\n")); 19738 #ifdef TD_DEBUG_ENABLE 19739 satIntIoContext = satIOContext->satIntIoContext; 19740 smIORequestBody = satIntIoContext->satIntRequestBody; 19741 #endif 19742 SM_DBG5(("smsatResetDevice: satIOContext %p smIORequestBody %p\n", satIOContext, smIORequestBody)); 19743 /* any fis should work */ 19744 fis->h.fisType = 0x27; /* Reg host to device */ 19745 fis->h.c_pmPort = 0; /* C Bit is not set */ 19746 fis->h.command = 0; /* any command */ 19747 fis->h.features = 0; /* FIS reserve */ 19748 fis->d.lbaLow = 0; /* FIS LBA (7 :0 ) */ 19749 fis->d.lbaMid = 0; /* FIS LBA (15:8 ) */ 19750 fis->d.lbaHigh = 0; /* FIS LBA (23:16) */ 19751 fis->d.device = 0; /* FIS LBA mode */ 19752 fis->d.lbaLowExp = 0; 19753 fis->d.lbaMidExp = 0; 19754 fis->d.lbaHighExp = 0; 19755 fis->d.featuresExp = 0; 19756 fis->d.sectorCount = 0; /* FIS sector count (7:0) */ 19757 fis->d.sectorCountExp = 0; 19758 fis->d.reserved4 = 0; 19759 fis->d.control = 0x4; /* SRST bit is set */ 19760 fis->d.reserved5 = 0; 19761 19762 agRequestType = AGSA_SATA_PROTOCOL_SRST_ASSERT; 19763 19764 /* Initialize CB for SATA completion. 19765 */ 19766 satIOContext->satCompleteCB = &smsatResetDeviceCB; 19767 19768 /* 19769 * Prepare SGL and send FIS to LL layer. 19770 */ 19771 satIOContext->reqType = agRequestType; /* Save it */ 19772 19773 #ifdef SM_INTERNAL_DEBUG 19774 smhexdump("smsatResetDevice", (bit8 *)satIOContext->pFis, sizeof(agsaFisRegHostToDevice_t)); 19775 #ifdef TD_DEBUG_ENABLE 19776 smhexdump("smsatResetDevice LL", (bit8 *)&(smIORequestBody->transport.SATA.agSATARequestBody.fis.fisRegHostToDev), sizeof(agsaFisRegHostToDevice_t)); 19777 #endif 19778 #endif 19779 19780 status = smsataLLIOStart( smRoot, 19781 smIORequest, 19782 smDeviceHandle, 19783 smScsiRequest, 19784 satIOContext); 19785 19786 SM_DBG6(("smsatResetDevice: end status %d\n", status)); 19787 return status; 19788 } 19789 19790 osGLOBAL bit32 19791 smsatDeResetDevice( 19792 smRoot_t *smRoot, 19793 smIORequest_t *smIORequest, 19794 smDeviceHandle_t *smDeviceHandle, 19795 smScsiInitiatorRequest_t *smScsiRequest, 19796 smSatIOContext_t *satIOContext 19797 ) 19798 { 19799 bit32 status; 19800 bit32 agRequestType; 19801 agsaFisRegHostToDevice_t *fis; 19802 #ifdef TD_DEBUG_ENABLE 19803 smIORequestBody_t *smIORequestBody; 19804 smSatInternalIo_t *satIntIoContext; 19805 #endif 19806 19807 fis = satIOContext->pFis; 19808 SM_DBG1(("smsatDeResetDevice: start\n")); 19809 #ifdef TD_DEBUG_ENABLE 19810 satIntIoContext = satIOContext->satIntIoContext; 19811 smIORequestBody = satIntIoContext->satIntRequestBody; 19812 #endif 19813 SM_DBG5(("smsatDeResetDevice: satIOContext %p smIORequestBody %p\n", satIOContext, smIORequestBody)); 19814 /* any fis should work */ 19815 fis->h.fisType = 0x27; /* Reg host to device */ 19816 fis->h.c_pmPort = 0; /* C Bit is not set */ 19817 fis->h.command = 0; /* any command */ 19818 fis->h.features = 0; /* FIS reserve */ 19819 fis->d.lbaLow = 0; /* FIS LBA (7 :0 ) */ 19820 fis->d.lbaMid = 0; /* FIS LBA (15:8 ) */ 19821 fis->d.lbaHigh = 0; /* FIS LBA (23:16) */ 19822 fis->d.device = 0; /* FIS LBA mode */ 19823 fis->d.lbaLowExp = 0; 19824 fis->d.lbaMidExp = 0; 19825 fis->d.lbaHighExp = 0; 19826 fis->d.featuresExp = 0; 19827 fis->d.sectorCount = 0; /* FIS sector count (7:0) */ 19828 fis->d.sectorCountExp = 0; 19829 fis->d.reserved4 = 0; 19830 fis->d.control = 0; /* SRST bit is not set */ 19831 fis->d.reserved5 = 0; 19832 19833 agRequestType = AGSA_SATA_PROTOCOL_SRST_DEASSERT; 19834 19835 /* Initialize CB for SATA completion. 19836 */ 19837 satIOContext->satCompleteCB = &smsatDeResetDeviceCB; 19838 19839 /* 19840 * Prepare SGL and send FIS to LL layer. 19841 */ 19842 satIOContext->reqType = agRequestType; /* Save it */ 19843 19844 #ifdef SM_INTERNAL_DEBUG 19845 smhexdump("smsatDeResetDevice", (bit8 *)satIOContext->pFis, sizeof(agsaFisRegHostToDevice_t)); 19846 #ifdef TD_DEBUG_ENABLE 19847 smhexdump("smsatDeResetDevice LL", (bit8 *)&(smIORequestBody->transport.SATA.agSATARequestBody.fis.fisRegHostToDev), sizeof(agsaFisRegHostToDevice_t)); 19848 #endif 19849 #endif 19850 19851 status = smsataLLIOStart( smRoot, 19852 smIORequest, 19853 smDeviceHandle, 19854 smScsiRequest, 19855 satIOContext); 19856 19857 SM_DBG6(("smsatDeResetDevice: end status %d\n", status)); 19858 return status; 19859 } 19860 19861 /* set feature for auto activate */ 19862 osGLOBAL bit32 19863 smsatSetFeaturesAA( 19864 smRoot_t *smRoot, 19865 smIORequest_t *smIORequest, 19866 smDeviceHandle_t *smDeviceHandle, 19867 smScsiInitiatorRequest_t *smScsiRequest, 19868 smSatIOContext_t *satIOContext 19869 ) 19870 { 19871 bit32 status = SM_RC_FAILURE; 19872 bit32 agRequestType; 19873 agsaFisRegHostToDevice_t *fis; 19874 19875 fis = satIOContext->pFis; 19876 SM_DBG2(("smsatSetFeaturesAA: start\n")); 19877 /* 19878 * Send the Set Features command. 19879 * See SATA II 1.0a spec 19880 */ 19881 fis->h.fisType = 0x27; /* Reg host to device */ 19882 fis->h.c_pmPort = 0x80; /* C Bit is set */ 19883 fis->h.command = SAT_SET_FEATURES; /* 0xEF */ 19884 fis->h.features = 0x10; /* enable SATA feature */ 19885 fis->d.lbaLow = 0; 19886 fis->d.lbaMid = 0; 19887 fis->d.lbaHigh = 0; 19888 fis->d.device = 0; 19889 fis->d.lbaLowExp = 0; 19890 fis->d.lbaMidExp = 0; 19891 fis->d.lbaHighExp = 0; 19892 fis->d.featuresExp = 0; 19893 fis->d.sectorCount = 0x02; /* DMA Setup FIS Auto-Activate */ 19894 fis->d.sectorCountExp = 0; 19895 fis->d.reserved4 = 0; 19896 fis->d.control = 0; /* FIS HOB bit clear */ 19897 fis->d.reserved5 = 0; 19898 19899 agRequestType = AGSA_SATA_PROTOCOL_NON_DATA; 19900 19901 /* Initialize CB for SATA completion. 19902 */ 19903 satIOContext->satCompleteCB = &smsatSetFeaturesAACB; 19904 19905 /* 19906 * Prepare SGL and send FIS to LL layer. 19907 */ 19908 satIOContext->reqType = agRequestType; /* Save it */ 19909 19910 status = smsataLLIOStart( smRoot, 19911 smIORequest, 19912 smDeviceHandle, 19913 smScsiRequest, 19914 satIOContext); 19915 19916 /* debugging code */ 19917 if (smIORequest->tdData == smIORequest->smData) 19918 { 19919 SM_DBG1(("smsatSetFeaturesAA: incorrect smIORequest\n")); 19920 } 19921 SM_DBG2(("smsatSetFeatures: return\n")); 19922 return status; 19923 } 19924 19925 19926 /* set feature for DMA transfer mode*/ 19927 osGLOBAL bit32 19928 smsatSetFeaturesDMA( 19929 smRoot_t *smRoot, 19930 smIORequest_t *smIORequest, 19931 smDeviceHandle_t *smDeviceHandle, 19932 smScsiInitiatorRequest_t *smScsiRequest, 19933 smSatIOContext_t *satIOContext 19934 ) 19935 { 19936 bit32 status = SM_RC_FAILURE; 19937 bit32 agRequestType; 19938 smDeviceData_t *pSatDevData; 19939 agsaFisRegHostToDevice_t *fis; 19940 19941 pSatDevData = satIOContext->pSatDevData; 19942 fis = satIOContext->pFis; 19943 SM_DBG2(("smsatSetFeaturesDMA: start\n")); 19944 /* 19945 * Send the Set Features command. 19946 * See SATA II 1.0a spec 19947 */ 19948 fis->h.fisType = 0x27; /* Reg host to device */ 19949 fis->h.c_pmPort = 0x80; /* C Bit is set */ 19950 fis->h.command = SAT_SET_FEATURES; /* 0xEF */ 19951 fis->h.features = 0x03; /* enable ATA transfer mode */ 19952 fis->d.lbaLow = 0; 19953 fis->d.lbaMid = 0; 19954 fis->d.lbaHigh = 0; 19955 fis->d.device = 0; 19956 fis->d.lbaLowExp = 0; 19957 fis->d.lbaMidExp = 0; 19958 fis->d.lbaHighExp = 0; 19959 fis->d.featuresExp = 0; 19960 fis->d.sectorCount = 0x40 |(bit8)pSatDevData->satUltraDMAMode; /* enable Ultra DMA mode */ 19961 fis->d.sectorCountExp = 0; 19962 fis->d.reserved4 = 0; 19963 fis->d.control = 0; /* FIS HOB bit clear */ 19964 fis->d.reserved5 = 0; 19965 19966 agRequestType = AGSA_SATA_PROTOCOL_NON_DATA; 19967 19968 /* Initialize CB for SATA completion. 19969 */ 19970 satIOContext->satCompleteCB = &smsatSetFeaturesDMACB; 19971 19972 /* 19973 * Prepare SGL and send FIS to LL layer. 19974 */ 19975 satIOContext->reqType = agRequestType; /* Save it */ 19976 19977 status = smsataLLIOStart( smRoot, 19978 smIORequest, 19979 smDeviceHandle, 19980 smScsiRequest, 19981 satIOContext); 19982 19983 /* debugging code */ 19984 if (smIORequest->tdData == smIORequest->smData) 19985 { 19986 SM_DBG1(("smsatSetFeaturesDMA: incorrect smIORequest\n")); 19987 } 19988 19989 SM_DBG2(("smsatSetFeaturesDMA: return\n")); 19990 19991 return status; 19992 } 19993 19994 /* set feature for Read Look Ahead*/ 19995 osGLOBAL bit32 19996 smsatSetFeaturesReadLookAhead( 19997 smRoot_t *smRoot, 19998 smIORequest_t *smIORequest, 19999 smDeviceHandle_t *smDeviceHandle, 20000 smScsiInitiatorRequest_t *smScsiRequest, 20001 smSatIOContext_t *satIOContext 20002 ) 20003 { 20004 bit32 status = SM_RC_FAILURE; 20005 bit32 agRequestType; 20006 agsaFisRegHostToDevice_t *fis; 20007 20008 fis = satIOContext->pFis; 20009 SM_DBG2(("smsatSetFeaturesReadLookAhead: start\n")); 20010 /* 20011 * Send the Set Features command. 20012 * See SATA II 1.0a spec 20013 */ 20014 fis->h.fisType = 0x27; /* Reg host to device */ 20015 fis->h.c_pmPort = 0x80; /* C Bit is set */ 20016 fis->h.command = SAT_SET_FEATURES; /* 0xEF */ 20017 fis->h.features = 0xAA; /* Enable read look-ahead feature */ 20018 fis->d.lbaLow = 0; 20019 fis->d.lbaMid = 0; 20020 fis->d.lbaHigh = 0; 20021 fis->d.device = 0; 20022 fis->d.lbaLowExp = 0; 20023 fis->d.lbaMidExp = 0; 20024 fis->d.lbaHighExp = 0; 20025 fis->d.featuresExp = 0; 20026 fis->d.sectorCount = 0; 20027 fis->d.sectorCountExp = 0; 20028 fis->d.reserved4 = 0; 20029 fis->d.control = 0; /* FIS HOB bit clear */ 20030 fis->d.reserved5 = 0; 20031 20032 agRequestType = AGSA_SATA_PROTOCOL_NON_DATA; 20033 20034 /* Initialize CB for SATA completion. 20035 */ 20036 satIOContext->satCompleteCB = &smsatSetFeaturesReadLookAheadCB; 20037 20038 /* 20039 * Prepare SGL and send FIS to LL layer. 20040 */ 20041 satIOContext->reqType = agRequestType; /* Save it */ 20042 20043 status = smsataLLIOStart( smRoot, 20044 smIORequest, 20045 smDeviceHandle, 20046 smScsiRequest, 20047 satIOContext); 20048 20049 /* debugging code */ 20050 if (smIORequest->tdData == smIORequest->smData) 20051 { 20052 SM_DBG1(("smsatSetFeaturesReadLookAhead: incorrect smIORequest\n")); 20053 } 20054 20055 SM_DBG2(("smsatSetFeaturesReadLookAhead: return\n")); 20056 20057 return status; 20058 } 20059 20060 /* set feature for Volatile Write Cache*/ 20061 osGLOBAL bit32 20062 smsatSetFeaturesVolatileWriteCache( 20063 smRoot_t *smRoot, 20064 smIORequest_t *smIORequest, 20065 smDeviceHandle_t *smDeviceHandle, 20066 smScsiInitiatorRequest_t *smScsiRequest, 20067 smSatIOContext_t *satIOContext 20068 ) 20069 { 20070 bit32 status = SM_RC_FAILURE; 20071 bit32 agRequestType; 20072 agsaFisRegHostToDevice_t *fis; 20073 20074 fis = satIOContext->pFis; 20075 SM_DBG2(("smsatSetFeaturesVolatileWriteCache: start\n")); 20076 /* 20077 * Send the Set Features command. 20078 * See SATA II 1.0a spec 20079 */ 20080 fis->h.fisType = 0x27; /* Reg host to device */ 20081 fis->h.c_pmPort = 0x80; /* C Bit is set */ 20082 fis->h.command = SAT_SET_FEATURES; /* 0xEF */ 20083 fis->h.features = 0x02; /* Enable Volatile Write Cache feature */ 20084 fis->d.lbaLow = 0; 20085 fis->d.lbaMid = 0; 20086 fis->d.lbaHigh = 0; 20087 fis->d.device = 0; 20088 fis->d.lbaLowExp = 0; 20089 fis->d.lbaMidExp = 0; 20090 fis->d.lbaHighExp = 0; 20091 fis->d.featuresExp = 0; 20092 fis->d.sectorCount = 0; 20093 fis->d.sectorCountExp = 0; 20094 fis->d.reserved4 = 0; 20095 fis->d.control = 0; /* FIS HOB bit clear */ 20096 fis->d.reserved5 = 0; 20097 20098 agRequestType = AGSA_SATA_PROTOCOL_NON_DATA; 20099 20100 /* Initialize CB for SATA completion. 20101 */ 20102 satIOContext->satCompleteCB = &smsatSetFeaturesVolatileWriteCacheCB; 20103 /* 20104 * Prepare SGL and send FIS to LL layer. 20105 */ 20106 satIOContext->reqType = agRequestType; /* Save it */ 20107 20108 status = smsataLLIOStart( smRoot, 20109 smIORequest, 20110 smDeviceHandle, 20111 smScsiRequest, 20112 satIOContext); 20113 /* debugging code */ 20114 if (smIORequest->tdData == smIORequest->smData) 20115 { 20116 SM_DBG1(("smsatSetFeaturesVolatileWriteCache: incorrect smIORequest\n")); 20117 } 20118 SM_DBG2(("smsatSetFeaturesVolatileWriteCache: return\n")); 20119 20120 return status; 20121 } 20122 20123 20124 20125 /******************************** start of utils ***********************************************************/ 20126 osGLOBAL FORCEINLINE void 20127 smsatBitSet(smRoot_t *smRoot, bit8 *data, bit32 index) 20128 { 20129 data[index>>3] |= (1 << (index&7)); 20130 } 20131 20132 osGLOBAL FORCEINLINE void 20133 smsatBitClear(smRoot_t *smRoot, bit8 *data, bit32 index) 20134 { 20135 data[index>>3] &= ~(1 << (index&7)); 20136 } 20137 20138 osGLOBAL FORCEINLINE BOOLEAN 20139 smsatBitTest(smRoot_t *smRoot, bit8 *data, bit32 index) 20140 { 20141 return ( (BOOLEAN)((data[index>>3] & (1 << (index&7)) ) ? 1: 0)); 20142 } 20143 20144 20145 FORCEINLINE bit32 20146 smsatTagAlloc( 20147 smRoot_t *smRoot, 20148 smDeviceData_t *pSatDevData, 20149 bit8 *pTag 20150 ) 20151 { 20152 bit32 retCode = agFALSE; 20153 bit32 i; 20154 20155 tdsmSingleThreadedEnter(smRoot, SM_NCQ_TAG_LOCK); 20156 20157 #ifdef CCFLAG_OPTIMIZE_SAT_LOCK 20158 20159 if (tdsmBitScanForward(smRoot, &i, ~(pSatDevData->freeSATAFDMATagBitmap))) 20160 { 20161 smsatBitSet(smRoot, (bit8*)&pSatDevData->freeSATAFDMATagBitmap, i); 20162 *pTag = (bit8)i; 20163 retCode = agTRUE; 20164 } 20165 20166 #else 20167 20168 for ( i = 0; i < pSatDevData->satNCQMaxIO; i ++ ) 20169 { 20170 if ( 0 == smsatBitTest(smRoot, (bit8 *)&pSatDevData->freeSATAFDMATagBitmap, i) ) 20171 { 20172 smsatBitSet(smRoot, (bit8*)&pSatDevData->freeSATAFDMATagBitmap, i); 20173 *pTag = (bit8) i; 20174 retCode = agTRUE; 20175 break; 20176 } 20177 } 20178 20179 #endif 20180 20181 tdsmSingleThreadedLeave(smRoot, SM_NCQ_TAG_LOCK); 20182 20183 return retCode; 20184 } 20185 20186 FORCEINLINE bit32 20187 smsatTagRelease( 20188 smRoot_t *smRoot, 20189 smDeviceData_t *pSatDevData, 20190 bit8 tag 20191 ) 20192 { 20193 bit32 retCode = agFALSE; 20194 20195 if ( tag < pSatDevData->satNCQMaxIO ) 20196 { 20197 tdsmSingleThreadedEnter(smRoot, SM_NCQ_TAG_LOCK); 20198 smsatBitClear(smRoot, (bit8 *)&pSatDevData->freeSATAFDMATagBitmap, (bit32)tag); 20199 tdsmSingleThreadedLeave(smRoot, SM_NCQ_TAG_LOCK); 20200 /*tdsmInterlockedAnd(smRoot, (volatile LONG *)(&pSatDevData->freeSATAFDMATagBitmap), ~(1 << (tag&31)));*/ 20201 retCode = agTRUE; 20202 } 20203 else 20204 { 20205 SM_DBG1(("smsatTagRelease: tag %d >= satNCQMaxIO %d!!!!\n", tag, pSatDevData->satNCQMaxIO)); 20206 } 20207 return retCode; 20208 } 20209 20210 20211 20212 osGLOBAL bit32 20213 smsatComputeCDB10LBA(smSatIOContext_t *satIOContext) 20214 { 20215 smIniScsiCmnd_t *scsiCmnd; 20216 smScsiInitiatorRequest_t *smScsiRequest; 20217 bit32 lba = 0; 20218 20219 SM_DBG5(("smsatComputeCDB10LBA: start\n")); 20220 smScsiRequest = satIOContext->smScsiXchg; 20221 scsiCmnd = &(smScsiRequest->scsiCmnd); 20222 20223 lba = (scsiCmnd->cdb[2] << (8*3)) + (scsiCmnd->cdb[3] << (8*2)) 20224 + (scsiCmnd->cdb[4] << 8) + scsiCmnd->cdb[5]; 20225 20226 return lba; 20227 } 20228 20229 osGLOBAL bit32 20230 smsatComputeCDB10TL(smSatIOContext_t *satIOContext) 20231 { 20232 20233 smIniScsiCmnd_t *scsiCmnd; 20234 smScsiInitiatorRequest_t *smScsiRequest; 20235 bit32 tl = 0; 20236 20237 SM_DBG5(("smsatComputeCDB10TL: start\n")); 20238 smScsiRequest = satIOContext->smScsiXchg; 20239 scsiCmnd = &(smScsiRequest->scsiCmnd); 20240 20241 tl = (scsiCmnd->cdb[7] << 8) + scsiCmnd->cdb[8]; 20242 return tl; 20243 } 20244 20245 osGLOBAL bit32 20246 smsatComputeCDB12LBA(smSatIOContext_t *satIOContext) 20247 { 20248 smIniScsiCmnd_t *scsiCmnd; 20249 smScsiInitiatorRequest_t *smScsiRequest; 20250 bit32 lba = 0; 20251 20252 SM_DBG5(("smsatComputeCDB12LBA: start\n")); 20253 smScsiRequest = satIOContext->smScsiXchg; 20254 scsiCmnd = &(smScsiRequest->scsiCmnd); 20255 20256 lba = (scsiCmnd->cdb[2] << (8*3)) + (scsiCmnd->cdb[3] << (8*2)) 20257 + (scsiCmnd->cdb[4] << 8) + scsiCmnd->cdb[5]; 20258 20259 return lba; 20260 } 20261 20262 osGLOBAL bit32 20263 smsatComputeCDB12TL(smSatIOContext_t *satIOContext) 20264 { 20265 20266 smIniScsiCmnd_t *scsiCmnd; 20267 smScsiInitiatorRequest_t *smScsiRequest; 20268 bit32 tl = 0; 20269 20270 SM_DBG5(("smsatComputeCDB12TL: start\n")); 20271 smScsiRequest = satIOContext->smScsiXchg; 20272 scsiCmnd = &(smScsiRequest->scsiCmnd); 20273 20274 tl = (scsiCmnd->cdb[6] << (8*3)) + (scsiCmnd->cdb[7] << (8*2)) 20275 + (scsiCmnd->cdb[8] << 8) + scsiCmnd->cdb[9]; 20276 return tl; 20277 } 20278 20279 /* 20280 CBD16 has bit64 LBA 20281 But it has to be less than (2^28 - 1) 20282 Therefore, use last four bytes to compute LBA is OK 20283 */ 20284 osGLOBAL bit32 20285 smsatComputeCDB16LBA(smSatIOContext_t *satIOContext) 20286 { 20287 smIniScsiCmnd_t *scsiCmnd; 20288 smScsiInitiatorRequest_t *smScsiRequest; 20289 bit32 lba = 0; 20290 20291 SM_DBG5(("smsatComputeCDB16LBA: start\n")); 20292 smScsiRequest = satIOContext->smScsiXchg; 20293 scsiCmnd = &(smScsiRequest->scsiCmnd); 20294 20295 lba = (scsiCmnd->cdb[6] << (8*3)) + (scsiCmnd->cdb[7] << (8*2)) 20296 + (scsiCmnd->cdb[8] << 8) + scsiCmnd->cdb[9]; 20297 20298 return lba; 20299 } 20300 20301 osGLOBAL bit32 20302 smsatComputeCDB16TL(smSatIOContext_t *satIOContext) 20303 { 20304 20305 smIniScsiCmnd_t *scsiCmnd; 20306 smScsiInitiatorRequest_t *smScsiRequest; 20307 bit32 tl = 0; 20308 20309 SM_DBG5(("smsatComputeCDB16TL: start\n")); 20310 smScsiRequest = satIOContext->smScsiXchg; 20311 scsiCmnd = &(smScsiRequest->scsiCmnd); 20312 20313 tl = (scsiCmnd->cdb[10] << (8*3)) + (scsiCmnd->cdb[11] << (8*2)) 20314 + (scsiCmnd->cdb[12] << 8) + scsiCmnd->cdb[13]; 20315 return tl; 20316 } 20317 20318 /* 20319 (tl, denom) 20320 tl can be upto bit32 because CDB16 has bit32 tl 20321 Therefore, fine 20322 either (tl, 0xFF) or (tl, 0xFFFF) 20323 */ 20324 osGLOBAL FORCEINLINE bit32 20325 smsatComputeLoopNum(bit32 a, bit32 b) 20326 { 20327 bit32 LoopNum = 0; 20328 20329 SM_DBG5(("smsatComputeLoopNum: start\n")); 20330 20331 if (a < b || a == 0) 20332 { 20333 LoopNum = 1; 20334 } 20335 else 20336 { 20337 if (a == b || a == 0) 20338 { 20339 LoopNum = a/b; 20340 } 20341 else 20342 { 20343 LoopNum = a/b + 1; 20344 } 20345 } 20346 20347 return LoopNum; 20348 } 20349 20350 /* 20351 Generic new function for checking 20352 LBA itself, LBA+TL < SAT_TR_LBA_LIMIT or SAT_EXT_TR_LBA_LIMIT 20353 and LBA+TL < Read Capacity Limit 20354 flag: false - not 48BitSupport; true - 48BitSupport 20355 returns TRUE when over the limit 20356 20357 */ 20358 osGLOBAL FORCEINLINE bit32 20359 smsatCheckLimit(bit8 *lba, bit8 *tl, int flag, smDeviceData_t *pSatDevData) 20360 { 20361 bit32 lbaCheck = agFALSE; 20362 int i; 20363 bit8 limit[8]; 20364 bit32 rangeCheck = agFALSE; 20365 bit16 ans[8]; // 0 MSB, 8 LSB 20366 bit8 final_ans[9]; // 0 MSB, 9 LSB 20367 bit8 Bit28max[8]; 20368 bit8 Bit48max[8]; 20369 bit32 ReadCapCheck = agFALSE; 20370 bit32 ret; 20371 20372 bit8 final_satMaxLBA[9]; 20373 bit8 oneTL[8]; 20374 bit8 temp_satMaxLBA[8]; // 0 MSB, 8 LSB 20375 /* 20376 check LBA 20377 */ 20378 if (flag == agFALSE) 20379 { 20380 /* limit is 0xF FF FF = 2^28 - 1 */ 20381 limit[0] = 0x0; /* MSB */ 20382 limit[1] = 0x0; 20383 limit[2] = 0x0; 20384 limit[3] = 0x0; 20385 limit[4] = 0xF; 20386 limit[5] = 0xFF; 20387 limit[6] = 0xFF; 20388 limit[7] = 0xFF; /* LSB */ 20389 } 20390 else 20391 { 20392 /* limit is 0xF FF FF = 2^48 - 1 */ 20393 limit[0] = 0x0; /* MSB */ 20394 limit[1] = 0x0; 20395 limit[2] = 0xFF; 20396 limit[3] = 0xFF; 20397 limit[4] = 0xFF; 20398 limit[5] = 0xFF; 20399 limit[6] = 0xFF; 20400 limit[7] = 0xFF; /* LSB */ 20401 } 20402 //compare lba to limit 20403 for(i=0;i<8;i++) 20404 { 20405 if (lba[i] > limit[i]) 20406 { 20407 SM_DBG1(("smsatCheckLimit: LBA check True at %d\n", i)); 20408 lbaCheck = agTRUE; 20409 break; 20410 } 20411 else if (lba[i] < limit[i]) 20412 { 20413 SM_DBG5(("smsatCheckLimit: LBA check False at %d\n", i)); 20414 lbaCheck = agFALSE; 20415 break; 20416 } 20417 else 20418 { 20419 continue; 20420 } 20421 } 20422 20423 if (lbaCheck == agTRUE) 20424 { 20425 SM_DBG1(("smsatCheckLimit: return LBA check True\n")); 20426 return agTRUE; 20427 } 20428 20429 /* 20430 check LBA+TL < SAT_TR_LBA_LIMIT or SAT_EXT_TR_LBA_LIMIT 20431 */ 20432 sm_memset(ans, 0, sizeof(ans)); 20433 sm_memset(final_ans, 0, sizeof(final_ans)); 20434 20435 // adding from LSB to MSB 20436 for(i=7;i>=0;i--) 20437 { 20438 ans[i] = (bit16)(lba[i] + tl[i]); 20439 if (i != 7) 20440 { 20441 ans[i] = (bit16)(ans[i] + ((ans[i+1] & 0xFF00) >> 8)); 20442 } 20443 } 20444 20445 /* 20446 filling in the final answer 20447 */ 20448 final_ans[0] = (bit8)(((ans[0] & 0xFF00) >> 8)); 20449 20450 for(i=1;i<=8;i++) 20451 { 20452 final_ans[i] = (bit8)(ans[i-1] & 0xFF); 20453 } 20454 20455 20456 if (flag == agFALSE) 20457 { 20458 sm_memset(Bit28max, 0, sizeof(Bit28max)); 20459 Bit28max[4] = 0x10; // max =0x1000 0000 20460 20461 //compare final_ans to max 20462 if (final_ans[0] != 0 || final_ans[1] != 0 || final_ans[2] != 0 20463 || final_ans[3] != 0 || final_ans[4] != 0) 20464 { 20465 SM_DBG1(("smsatCheckLimit: before 28Bit addressing TRUE\n")); 20466 rangeCheck = agTRUE; 20467 } 20468 else 20469 { 20470 for(i=5;i<=8;i++) 20471 { 20472 if (final_ans[i] > Bit28max[i-1]) 20473 { 20474 SM_DBG1(("smsatCheckLimit: 28Bit addressing TRUE at %d\n", i)); 20475 rangeCheck = agTRUE; 20476 break; 20477 } 20478 else if (final_ans[i] < Bit28max[i-1]) 20479 { 20480 SM_DBG5(("smsatCheckLimit: 28Bit addressing FALSE at %d\n", i)); 20481 rangeCheck = agFALSE; 20482 break; 20483 } 20484 else 20485 { 20486 continue; 20487 } 20488 } 20489 } 20490 } 20491 else 20492 { 20493 sm_memset(Bit48max, 0, sizeof(Bit48max)); 20494 Bit48max[1] = 0x1; //max = 0x1 0000 0000 0000 20495 20496 //compare final_ans to max 20497 if (final_ans[0] != 0 || final_ans[1] != 0) 20498 { 20499 SM_DBG1(("smsatCheckLimit: before 48Bit addressing TRUE\n")); 20500 rangeCheck = agTRUE; 20501 } 20502 else 20503 { 20504 for(i=2;i<=8;i++) 20505 { 20506 if (final_ans[i] > Bit48max[i-1]) 20507 { 20508 SM_DBG1(("smsatCheckLimit: 48Bit addressing TRUE at %d\n", i)); 20509 rangeCheck = agTRUE; 20510 break; 20511 } 20512 else if (final_ans[i] < Bit48max[i-1]) 20513 { 20514 SM_DBG5(("smsatCheckLimit: 48Bit addressing FALSE at %d\n", i)); 20515 rangeCheck = agFALSE; 20516 break; 20517 } 20518 else 20519 { 20520 continue; 20521 } 20522 } 20523 } 20524 } 20525 if (rangeCheck == agTRUE) 20526 { 20527 SM_DBG1(("smsatCheckLimit: return rangeCheck True\n")); 20528 return agTRUE; 20529 } 20530 20531 /* 20532 LBA+TL < Read Capacity Limit 20533 */ 20534 sm_memset(temp_satMaxLBA, 0, sizeof(temp_satMaxLBA)); 20535 sm_memset(oneTL, 0, sizeof(oneTL)); 20536 sm_memset(final_satMaxLBA, 0, sizeof(final_satMaxLBA)); 20537 sm_memset(ans, 0, sizeof(ans)); 20538 20539 sm_memcpy(&temp_satMaxLBA, &pSatDevData->satMaxLBA, sizeof(temp_satMaxLBA)); 20540 oneTL[7] = 1; 20541 20542 // adding temp_satMaxLBA to oneTL 20543 for(i=7;i>=0;i--) 20544 { 20545 ans[i] = (bit16)(temp_satMaxLBA[i] + oneTL[i]); 20546 if (i != 7) 20547 { 20548 ans[i] = (bit16)(ans[i] + ((ans[i+1] & 0xFF00) >> 8)); 20549 } 20550 } 20551 20552 /* 20553 filling in the final answer 20554 */ 20555 final_satMaxLBA[0] = (bit8)(((ans[0] & 0xFF00) >> 8)); 20556 20557 for(i=1;i<=8;i++) 20558 { 20559 final_satMaxLBA[i] = (bit8)(ans[i-1] & 0xFF); 20560 } 20561 if ( pSatDevData->ReadCapacity == 10) 20562 { 20563 for (i=0;i<=8;i++) 20564 { 20565 if (final_ans[i] > final_satMaxLBA[i]) 20566 { 20567 SM_DBG1(("smsatCheckLimit: Read Capacity 10 TRUE at %d\n", i)); 20568 ReadCapCheck = agTRUE; 20569 break; 20570 } 20571 else if (final_ans[i] < final_satMaxLBA[i]) 20572 { 20573 SM_DBG5(("smsatCheckLimit: Read Capacity 10 FALSE at %d\n", i)); 20574 ReadCapCheck = agFALSE; 20575 break; 20576 } 20577 else 20578 { 20579 continue; 20580 } 20581 } 20582 if ( ReadCapCheck) 20583 { 20584 SM_DBG1(("smsatCheckLimit: after Read Capacity 10 TRUE\n")); 20585 } 20586 else 20587 { 20588 SM_DBG5(("smsatCheckLimit: after Read Capacity 10 FALSE\n")); 20589 } 20590 } 20591 else if ( pSatDevData->ReadCapacity == 16) 20592 { 20593 for (i=0;i<=8;i++) 20594 { 20595 if (final_ans[i] > final_satMaxLBA[i]) 20596 { 20597 SM_DBG1(("smsatCheckLimit: Read Capacity 16 TRUE at %d\n", i)); 20598 ReadCapCheck = agTRUE; 20599 break; 20600 } 20601 else if (final_ans[i] < final_satMaxLBA[i]) 20602 { 20603 SM_DBG5(("smsatCheckLimit: Read Capacity 16 FALSE at %d\n", i)); 20604 ReadCapCheck = agFALSE; 20605 break; 20606 } 20607 else 20608 { 20609 continue; 20610 } 20611 } 20612 if ( ReadCapCheck) 20613 { 20614 SM_DBG1(("smsatCheckLimit: after Read Capacity 16 TRUE\n")); 20615 } 20616 else 20617 { 20618 SM_DBG5(("smsatCheckLimit: after Read Capacity 16 FALSE\n")); 20619 } 20620 } 20621 else 20622 { 20623 SM_DBG5(("smsatCheckLimit: unknown pSatDevData->ReadCapacity %d\n", pSatDevData->ReadCapacity)); 20624 } 20625 20626 if (ReadCapCheck == agTRUE) 20627 { 20628 SM_DBG1(("smsatCheckLimit: return ReadCapCheck True\n")); 20629 return agTRUE; 20630 } 20631 20632 20633 ret = (lbaCheck | rangeCheck | ReadCapCheck); 20634 if (ret == agTRUE) 20635 { 20636 SM_DBG1(("smsatCheckLimit: final check TRUE\n")); 20637 } 20638 else 20639 { 20640 SM_DBG5(("smsatCheckLimit: final check FALSE\n")); 20641 } 20642 return ret; 20643 } 20644 20645 20646 20647 osGLOBAL void 20648 smsatPrintSgl( 20649 smRoot_t *smRoot, 20650 agsaEsgl_t *agEsgl, 20651 bit32 idx 20652 ) 20653 { 20654 bit32 i=0; 20655 #ifdef TD_DEBUG_ENABLE 20656 agsaSgl_t *agSgl; 20657 #endif 20658 20659 for (i=0;i<idx;i++) 20660 { 20661 #ifdef TD_DEBUG_ENABLE 20662 agSgl = &(agEsgl->descriptor[i]); 20663 #endif 20664 SM_DBG3(("smsatPrintSgl: agSgl %d upperAddr 0x%08x lowerAddr 0x%08x len 0x%08x ext 0x%08x\n", 20665 i, agSgl->sgUpper, agSgl->sgLower, agSgl->len, agSgl->extReserved)); 20666 } 20667 20668 return; 20669 } 20670 20671 20672 osGLOBAL void 20673 smsatSplitSGL( 20674 smRoot_t *smRoot, 20675 smIORequest_t *smIORequest, 20676 smDeviceHandle_t *smDeviceHandle, 20677 smScsiInitiatorRequest_t *smScsiRequest, 20678 smSatIOContext_t *satIOContext, 20679 bit32 split, /*in sector number, depeding on IO value */ 20680 bit32 tl, /* in sector number */ 20681 bit32 flag 20682 ) 20683 { 20684 agsaSgl_t *agSgl; 20685 agsaEsgl_t *agEsgl; 20686 bit32 i=0; 20687 smIniScsiCmnd_t *scsiCmnd; 20688 bit32 totalLen=0; /* in bytes */ 20689 bit32 splitLen=0; /* in bytes */ 20690 bit32 splitDiffByte = 0; /* in bytes */ 20691 bit32 splitDiffExtra = 0; /* in bytes */ 20692 bit32 splitIdx = 0; 20693 bit32 UpperAddr, LowerAddr; 20694 bit32 tmpLowerAddr; 20695 void *sglVirtualAddr; 20696 void *sglSplitVirtualAddr; 20697 20698 scsiCmnd = &smScsiRequest->scsiCmnd; 20699 SM_DBG3(("smsatSplitSGL: start\n")); 20700 20701 if (smScsiRequest->smSgl1.type == 0x80000000) /* esgl */ 20702 { 20703 if (flag == agFALSE) 20704 { 20705 SM_DBG3(("smsatSplitSGL: Not first time\n")); 20706 SM_DBG3(("smsatSplitSGL: UpperAddr 0x%08x LowerAddr 0x%08x\n", satIOContext->UpperAddr, satIOContext->LowerAddr)); 20707 SM_DBG3(("smsatSplitSGL: SplitIdx %d AdjustBytes 0x%08x\n", satIOContext->SplitIdx, satIOContext->AdjustBytes)); 20708 20709 sglVirtualAddr = smScsiRequest->sglVirtualAddr; 20710 20711 agEsgl = (agsaEsgl_t *)smScsiRequest->sglVirtualAddr; 20712 20713 sglSplitVirtualAddr = &(agEsgl->descriptor[satIOContext->SplitIdx]); 20714 20715 agEsgl = (agsaEsgl_t *)sglSplitVirtualAddr; 20716 20717 if (agEsgl == agNULL) 20718 { 20719 SM_DBG1(("smsatSplitSGL: error!\n")); 20720 return; 20721 } 20722 /* first sgl ajustment */ 20723 agSgl = &(agEsgl->descriptor[0]); 20724 agSgl->sgUpper = satIOContext->UpperAddr; 20725 agSgl->sgLower = satIOContext->LowerAddr; 20726 agSgl->len = satIOContext->AdjustBytes; 20727 sm_memcpy(sglVirtualAddr, sglSplitVirtualAddr, (satIOContext->EsglLen) * sizeof(agsaSgl_t)); 20728 agEsgl = (agsaEsgl_t *)smScsiRequest->sglVirtualAddr; 20729 smsatPrintSgl(smRoot, (agsaEsgl_t *)sglVirtualAddr, satIOContext->EsglLen); 20730 } 20731 else 20732 { 20733 /* first time */ 20734 SM_DBG3(("smsatSplitSGL: first time\n")); 20735 satIOContext->EsglLen = smScsiRequest->smSgl1.len; 20736 agEsgl = (agsaEsgl_t *)smScsiRequest->sglVirtualAddr; 20737 if (agEsgl == agNULL) 20738 { 20739 return; 20740 } 20741 smsatPrintSgl(smRoot, agEsgl, satIOContext->EsglLen); 20742 } 20743 20744 if (tl > split) 20745 { 20746 /* split */ 20747 SM_DBG3(("smsatSplitSGL: split case\n")); 20748 i = 0; 20749 while (1) 20750 { 20751 agSgl = &(agEsgl->descriptor[i]); 20752 splitLen = splitLen + agSgl->len; 20753 if (splitLen >= split) 20754 { 20755 splitDiffExtra = splitLen - split; 20756 splitDiffByte = agSgl->len - splitDiffExtra; 20757 splitIdx = i; 20758 break; 20759 } 20760 i++; 20761 } 20762 SM_DBG3(("smsatSplitSGL: splitIdx %d\n", splitIdx)); 20763 SM_DBG3(("smsatSplitSGL: splitDiffByte 0x%8x\n", splitDiffByte)); 20764 SM_DBG3(("smsatSplitSGL: splitDiffExtra 0x%8x \n", splitDiffExtra)); 20765 20766 20767 agSgl = &(agEsgl->descriptor[splitIdx]); 20768 UpperAddr = agSgl->sgUpper; 20769 LowerAddr = agSgl->sgLower; 20770 tmpLowerAddr = LowerAddr + splitDiffByte; 20771 if (tmpLowerAddr < LowerAddr) 20772 { 20773 UpperAddr = UpperAddr + 1; 20774 } 20775 SM_DBG3(("smsatSplitSGL: UpperAddr 0x%08x tmpLowerAddr 0x%08x\n", UpperAddr, tmpLowerAddr)); 20776 agSgl->len = splitDiffByte; 20777 /* Esgl len adjustment */ 20778 smScsiRequest->smSgl1.len = splitIdx; 20779 /* expected data lent adjustment */ 20780 scsiCmnd->expDataLength = 0x20000; 20781 /* remeber for the next round */ 20782 satIOContext->UpperAddr = UpperAddr; 20783 satIOContext->LowerAddr = tmpLowerAddr; 20784 satIOContext->SplitIdx = splitIdx; 20785 satIOContext->AdjustBytes = splitDiffExtra; 20786 satIOContext->EsglLen = satIOContext->EsglLen - smScsiRequest->smSgl1.len; 20787 satIOContext->OrgTL = satIOContext->OrgTL - 0x100; 20788 // smsatPrintSgl(smRoot, agEsgl, satIOContext->EsglLen); 20789 20790 } 20791 else 20792 { 20793 /* no split */ 20794 SM_DBG3(("smsatSplitSGL: no split case\n")); 20795 /* Esgl len adjustment */ 20796 smScsiRequest->smSgl1.len = satIOContext->EsglLen; 20797 for (i=0;i< smScsiRequest->smSgl1.len;i++) 20798 { 20799 agSgl = &(agEsgl->descriptor[i]); 20800 totalLen = totalLen + (agSgl->len); 20801 } 20802 /* expected data lent adjustment */ 20803 scsiCmnd->expDataLength = totalLen; 20804 // smsatPrintSgl(smRoot, agEsgl, satIOContext->EsglLen); 20805 } 20806 } 20807 else 20808 { 20809 SM_DBG1(("not exntened esgl\n")); 20810 20811 } 20812 20813 return; 20814 } 20815 20816 20817 /******************************** end of utils ***********************************************************/ 20818 20819 20820 20821