79                              uint32_t lun_id, 
const Callback &transfer_cb,
 
   82     flashDisk(
p.image[lun_id]),
 
   83     flashDevice(
p.internalflash[lun_id]),
 
   84     blkSize(
p.img_blk_size),
 
   85     lunAvail(
p.image.size()),
 
   86     diskSize(flashDisk->size()),
 
   87     capacityLower((diskSize - 1) & 0xffffffff),
 
   90     transferCompleted(false),
 
   94     amountOfWriteTransfers(0),
 
   95     amountOfReadTransfers(0)
 
  111     uint32_t temp_id = ((lun_id | 0x30) << 24) | 0x3A4449;
 
  136                                                 {0x01400A0A, 0x00000000,
 
  139                                                 {0x03800A01, 0x00000000,
 
  142                                                 {0x00011208, 0x00000000,
 
  143                                                  0x00000000, 0x00000020,
 
  173     statusCheck(SCSIGood, scsi_out.
senseCode);
 
  175     scsi_out.
LUN = lunID;
 
  176     scsi_out.
status = SCSIGood;
 
  181     switch (SCSI_msg[4] & 0xFF) {
 
  192                   (
reinterpret_cast<uint32_t*
> (&lunInfo))[
count];
 
  202           uint8_t* tempptr = 
reinterpret_cast<uint8_t*
>(&SCSI_msg[4]);
 
  208           uint32_t tmp = *
reinterpret_cast<uint32_t*
>(tempptr);
 
  209           uint64_t read_offset = 
betoh(tmp) & 0x1FFFFF;
 
  211           uint32_t read_size = tempptr[4];
 
  214           scsi_out.
msgSize =  read_size * blkSize;
 
  215           scsi_out.
offset = read_offset * blkSize;
 
  217           if ((read_offset + read_size) > diskSize)
 
  218               scsi_out.
status = SCSIIllegalRequest;
 
  221                   read_offset, read_size);
 
  228           scsi_out.
status = (scsi_out.
status == SCSIGood) ? SCSIGood :
 
  237           uint8_t* tempptr = 
reinterpret_cast<uint8_t*
>(&SCSI_msg[4]);
 
  240           uint32_t tmp = *
reinterpret_cast<uint32_t*
>(&tempptr[2]);
 
  241           uint64_t read_offset = 
betoh(tmp);
 
  243           uint16_t tmpsize = *
reinterpret_cast<uint16_t*
>(&tempptr[7]);
 
  244           uint32_t read_size = 
betoh(tmpsize);
 
  246           scsi_out.
msgSize =  read_size * blkSize;
 
  247           scsi_out.
offset = read_offset * blkSize;
 
  249           if ((read_offset + read_size) > diskSize)
 
  250               scsi_out.
status = SCSIIllegalRequest;
 
  253                   read_offset, read_size);
 
  260           scsi_out.
status = (scsi_out.
status == SCSIGood) ? SCSIGood :
 
  269           uint8_t* tempptr = 
reinterpret_cast<uint8_t*
>(&SCSI_msg[4]);
 
  272           uint32_t tmp = *
reinterpret_cast<uint32_t*
>(&tempptr[2]);
 
  273           uint64_t read_offset = 
betoh(tmp);
 
  275           tmp = *
reinterpret_cast<uint32_t*
>(&tempptr[6]);
 
  276           read_offset = (read_offset << 32) | 
betoh(tmp);
 
  278           tmp = *
reinterpret_cast<uint32_t*
>(&tempptr[10]);
 
  279           uint32_t read_size = 
betoh(tmp);
 
  281           scsi_out.
msgSize =  read_size * blkSize;
 
  282           scsi_out.
offset = read_offset * blkSize;
 
  284           if ((read_offset + read_size) > diskSize)
 
  285               scsi_out.
status = SCSIIllegalRequest;
 
  288                   read_offset, read_size);
 
  295           scsi_out.
status = (scsi_out.
status == SCSIGood) ? SCSIGood :
 
  300       case SCSIReadCapacity10: {
 
  307               betoh(capacityLower);
 
  311       case SCSIReadCapacity16: {
 
  315               betoh(capacityUpper);
 
  317               betoh(capacityLower);
 
  327       case SCSIReportLUNs: {
 
  344       case SCSIStartStop: {
 
  350       case SCSITestUnitReady: {
 
  363           uint8_t* tempptr = 
reinterpret_cast<uint8_t*
>(&SCSI_msg[4]);
 
  366           uint32_t tmp = *
reinterpret_cast<uint32_t*
>(&tempptr[2]);
 
  367           uint64_t read_offset = 
betoh(tmp);
 
  369           uint16_t tmpsize = *
reinterpret_cast<uint16_t*
>(&tempptr[7]);
 
  370           uint32_t read_size = 
betoh(tmpsize);
 
  372           if ((read_offset + read_size) > diskSize)
 
  373               scsi_out.
status = SCSIIllegalRequest;
 
  380           scsi_out.
status = (scsi_out.
status == SCSIGood) ? SCSIGood :
 
  390           uint8_t* tempptr = 
reinterpret_cast<uint8_t*
>(&SCSI_msg[4]);
 
  396           uint32_t tmp = *
reinterpret_cast<uint32_t*
>(tempptr);
 
  397           uint64_t write_offset = 
betoh(tmp) & 0x1FFFFF;
 
  399           uint32_t write_size = tempptr[4];
 
  401           scsi_out.
msgSize =  write_size * blkSize;
 
  402           scsi_out.
offset = write_offset * blkSize;
 
  405           if ((write_offset + write_size) > diskSize)
 
  406               scsi_out.
status = SCSIIllegalRequest;
 
  409                   write_offset, write_size);
 
  416           scsi_out.
status = (scsi_out.
status == SCSIGood) ? SCSIGood :
 
  422           uint8_t* tempptr = 
reinterpret_cast<uint8_t*
>(&SCSI_msg[4]);
 
  425           uint32_t tmp = *
reinterpret_cast<uint32_t*
>(&tempptr[2]);
 
  426           uint64_t write_offset = 
betoh(tmp);
 
  428           uint16_t tmpsize = *
reinterpret_cast<uint16_t*
>(&tempptr[7]);
 
  429           uint32_t write_size = 
betoh(tmpsize);
 
  431           scsi_out.
msgSize =  write_size * blkSize;
 
  432           scsi_out.
offset = write_offset * blkSize;
 
  435           if ((write_offset + write_size) > diskSize)
 
  436               scsi_out.
status = SCSIIllegalRequest;
 
  439                   write_offset, write_size);
 
  446           scsi_out.
status = (scsi_out.
status == SCSIGood) ? SCSIGood :
 
  452           uint8_t* tempptr = 
reinterpret_cast<uint8_t*
>(&SCSI_msg[4]);
 
  455           uint32_t tmp = *
reinterpret_cast<uint32_t*
>(&tempptr[2]);
 
  456           uint64_t write_offset = 
betoh(tmp);
 
  458           tmp = *
reinterpret_cast<uint32_t*
>(&tempptr[6]);
 
  459           write_offset = (write_offset << 32) | 
betoh(tmp);
 
  461           tmp = *
reinterpret_cast<uint32_t*
>(&tempptr[10]);
 
  462           uint32_t write_size = 
betoh(tmp);
 
  464           scsi_out.
msgSize = write_size * blkSize;
 
  465           scsi_out.
offset = write_offset * blkSize;
 
  468           if ((write_offset + write_size) > diskSize)
 
  469               scsi_out.
status = SCSIIllegalRequest;
 
  472                   write_offset, write_size);
 
  479           scsi_out.
status = (scsi_out.
status == SCSIGood) ? SCSIGood :
 
  484       case SCSIFormatUnit: {
 
  490       case SCSISendDiagnostic: {
 
  495       case SCSISynchronizeCache: {
 
  503       case SCSIModeSelect10:
 
  509       case SCSIModeSense6: 
case SCSIModeSense10: {
 
  514           if ((SCSI_msg[4] & 0x3F0000) >> 16 == 0x0A) {
 
  525           } 
else if ((SCSI_msg[4] & 0x3F0000) >> 16 == 0x01) {
 
  538           } 
else if ((SCSI_msg[4] & 0x3F0000) >> 16 == 0x08) {
 
  550           } 
else if ((SCSI_msg[4] & 0x3F0000) >> 16 == 0x3F) {
 
  553                                                 sizeof(recoveryPage) +
 
  554                                                 sizeof(cachingPage)) >> 2)
 
  571           } 
else inform(
"Wrong mode page requested\n");
 
  576       case SCSIRequestSense: {
 
  581       case SCSIUnmap:
break;
 
  583       case SCSIWriteBuffer: {
 
  586           uint8_t* tempptr = 
reinterpret_cast<uint8_t*
>(&SCSI_msg[4]);
 
  589           uint32_t tmp = *
reinterpret_cast<uint32_t*
>(&tempptr[2]);
 
  590           uint64_t write_offset = 
betoh(tmp) & 0xFFFFFF;
 
  592           tmp = *
reinterpret_cast<uint32_t*
>(&tempptr[5]);
 
  593           uint32_t write_size = 
betoh(tmp) & 0xFFFFFF;
 
  596           scsi_out.
offset = write_offset;
 
  600       case SCSIReadBuffer: {
 
  608           uint8_t* tempptr = 
reinterpret_cast<uint8_t*
>(&SCSI_msg[4]);
 
  611           uint32_t tmp = *
reinterpret_cast<uint32_t*
>(&tempptr[2]);
 
  612           uint64_t read_offset = 
betoh(tmp) & 0xFFFFFF;
 
  614           tmp = *
reinterpret_cast<uint32_t*
>(&tempptr[5]);
 
  615           uint32_t read_size = 
betoh(tmp) & 0xFFFFFF;
 
  618           scsi_out.
offset = read_offset;
 
  620           if ((read_offset + read_size) > capacityLower * blkSize)
 
  621               scsi_out.
status = SCSIIllegalRequest;
 
  629           scsi_out.
status = (scsi_out.
status == SCSIGood) ? SCSIGood :
 
  634       case SCSIMaintenanceIn: {
 
  642           statusCheck(SCSIIllegalRequest, scsi_out.
senseCode);
 
  644           scsi_out.
status = (scsi_out.
status == SCSIGood) ? SCSIGood :
 
  650           statusCheck(SCSIIllegalRequest, scsi_out.
senseCode);
 
  652           scsi_out.
status = (scsi_out.
status == SCSIGood) ? SCSIGood :
 
  655           inform(
"Unsupported scsi message type: %2x\n", SCSI_msg[4] & 0xFF);
 
  656           inform(
"0x%8x\n", SCSI_msg[0]);
 
  657           inform(
"0x%8x\n", SCSI_msg[1]);
 
  658           inform(
"0x%8x\n", SCSI_msg[2]);
 
  659           inform(
"0x%8x\n", SCSI_msg[3]);
 
  660           inform(
"0x%8x\n", SCSI_msg[4]);
 
  675                                 uint8_t* sensecodelist)
 
  678         sensecodelist[
count] = 0;
 
  680     sensecodelist[0] = 18; 
 
  681     sensecodelist[1] = 0x70; 
 
  682     sensecodelist[3] = 
status & 0xF; 
 
  683     sensecodelist[8] = 0x1F; 
 
  737     UTPEvent([
this]{ finalUTP(); }, 
name())
 
  739     DPRINTF(UFSHostDevice, 
"The hostcontroller hosts %d Logic units\n",
 
  741     UFSDevice.resize(lunAvail);
 
  745                 [
this]() { LUNSignal(); },
 
  746                 [
this]() { readCallback(); });
 
  750         warn(
"UFSSlots = %d, this will results in %d command slots",
 
  751              UFSSlots, (UFSSlots & 0x1F));
 
  753     if ((UFSSlots & 0x1F) == 0)
 
  754         fatal(
"Number of UFS command slots should be between 1 and 32.");
 
  761     : statistics::
Group(parent, 
"UFSDiskHost"),
 
  762       ADD_STAT(currentSCSIQueue, statistics::units::Count::get(),
 
  763                "Most up to date length of the command queue"),
 
  764       ADD_STAT(currentReadSSDQueue, statistics::units::Count::get(),
 
  765                "Most up to date length of the read SSD queue"),
 
  766       ADD_STAT(currentWriteSSDQueue, statistics::units::Count::get(),
 
  767                "Most up to date length of the write SSD queue"),
 
  769       ADD_STAT(totalReadSSD, statistics::units::Byte::get(),
 
  770                "Number of bytes read from SSD"),
 
  771       ADD_STAT(totalWrittenSSD, statistics::units::Byte::get(),
 
  772                "Number of bytes written to SSD"),
 
  773       ADD_STAT(totalReadDiskTransactions, statistics::units::Count::get(),
 
  774                "Number of transactions from disk"),
 
  775       ADD_STAT(totalWriteDiskTransactions, statistics::units::Count::get(),
 
  776                "Number of transactions to disk"),
 
  777       ADD_STAT(totalReadUFSTransactions, statistics::units::Count::get(),
 
  778                "Number of transactions from device"),
 
  779       ADD_STAT(totalWriteUFSTransactions, statistics::units::Count::get(),
 
  780                "Number of transactions to device"),
 
  782       ADD_STAT(averageReadSSDBW, statistics::units::Rate<
 
  783                     statistics::units::Byte, statistics::units::Second>::get(),
 
  784                "Average read bandwidth",
 
  786       ADD_STAT(averageWriteSSDBW, statistics::units::Rate<
 
  787                     statistics::units::Byte, statistics::units::Second>::get(),
 
  788                "Average write bandwidth",
 
  790       ADD_STAT(averageSCSIQueue, statistics::units::Rate<
 
  791                     statistics::units::Count, statistics::units::
Tick>::get(),
 
  792                "Average command queue length"),
 
  793       ADD_STAT(averageReadSSDQueue, statistics::units::Rate<
 
  794                     statistics::units::Count, statistics::units::
Tick>::get(),
 
  795                "Average read queue length"),
 
  796       ADD_STAT(averageWriteSSDQueue, statistics::units::Rate<
 
  797                     statistics::units::Count, statistics::units::
Tick>::get(),
 
  798                "Average write queue length"),
 
  800       ADD_STAT(curDoorbell, statistics::units::Count::get(),
 
  801                "Most up to date number of doorbells used",
 
  802                parent->activeDoorbells),
 
  803       ADD_STAT(maxDoorbell, statistics::units::Count::get(),
 
  804                "Maximum number of doorbells utilized"),
 
  805       ADD_STAT(averageDoorbell, statistics::units::Rate<
 
  806                     statistics::units::Count, statistics::units::
Tick>::get(),
 
  807                "Average number of Doorbells used"),
 
  809       ADD_STAT(transactionLatency, statistics::units::
Tick::get(),
 
  810                "Histogram of transaction times"),
 
  811       ADD_STAT(idleTimes, statistics::units::
Tick::get(), 
"Histogram of idle times")
 
  813     using namespace statistics;
 
 1058     const uint32_t 
data = pkt->
getUintX(ByteOrder::little);
 
 1060     switch (pkt->
getAddr() & 0xFF)
 
 1203     Addr address = 0x00;
 
 1209     transferstart_info.
done = 0;
 
 1241             address = (
count * size) + (address << 32) +
 
 1245             inform(
"UFSmodel received a task from the system; this might" 
 1246                    " lead to untested behaviour.\n");
 
 1250             task_info.
size = size;
 
 1256                         reinterpret_cast<uint8_t*
> 
 1257                         (&
taskInfo.back().destination), 0, 0);
 
 1284             stats.averageDoorbell = 
stats.maxDoorbell.value();
 
 1292             transferstart_info.
address = address;
 
 1293             transferstart_info.
size = size;
 
 1301                     transferstart_info.
done);
 
 1307                             address, size, 
reinterpret_cast<uint8_t*
> 
 1369                            uint32_t req_pos, 
Addr finaladdress, uint32_t
 
 1387     readDevice(
true, finaladdress, finalsize, 
reinterpret_cast<uint8_t*
> 
 1388                (request_in), 
true, NULL);
 
 1405                                int req_pos, 
Addr finaladdress, uint32_t
 
 1406                                finalsize, uint32_t done)
 
 1409     Addr cmd_desc_addr = 0x00;
 
 1428     cmd_desc_addr = (cmd_desc_addr << 32) |
 
 1457     if (
UFSDevice[LUN]->SCSIInfoQueue.size() < 2) 
 
 1460     else if (
UFSDevice[LUN]->SCSIInfoQueue.size() > 32)
 
 1462               SCSIInfoQueue.size());
 
 1495     if (
UFSDevice[lun_id]->SCSIInfoQueue.empty())
 
 1496         panic(
"No SCSI message scheduled lun:%d Doorbell: 0x%8x", lun_id,
 
 1501         SCSIInfoQueue.front().RequestIn;
 
 1503     uint32_t req_pos = 
UFSDevice[lun_id]->SCSIInfoQueue.front().reqPos;
 
 1505     Addr finaladdress = 
UFSDevice[lun_id]->SCSIInfoQueue.front().
 
 1508     uint32_t finalsize = 
UFSDevice[lun_id]->SCSIInfoQueue.front().finalSize;
 
 1510     uint32_t* transfercommand = 
reinterpret_cast<uint32_t*
> 
 1511         (&(
UFSDevice[lun_id]->SCSIInfoQueue.front().destination[0]));
 
 1516         SCSICMDHandle(transfercommand);
 
 1526     UFSDevice[lun_id]->transferInfo.requestOut.header.dWord0 =
 
 1528         | (transfercommand[0] & 0xFF000000);
 
 1530     UFSDevice[lun_id]->transferInfo.requestOut.header.dWord1 = 0x00000000 |
 
 1533     UFSDevice[lun_id]->transferInfo.requestOut.header.dWord2 = 0x00000000 |
 
 1536     UFSDevice[lun_id]->transferInfo.requestOut.senseDataLen =
 
 1557     response_addr = (response_addr << 32) |
 
 1562     UFSDevice[lun_id]->transferInfo.responseStartAddr = response_addr;
 
 1563     UFSDevice[lun_id]->transferInfo.reqPos = req_pos;
 
 1564     UFSDevice[lun_id]->transferInfo.size = finalsize;
 
 1565     UFSDevice[lun_id]->transferInfo.address = finaladdress;
 
 1566     UFSDevice[lun_id]->transferInfo.destination = 
reinterpret_cast<uint8_t*
> 
 1567         (
UFSDevice[lun_id]->SCSIInfoQueue.front().RequestIn);
 
 1568     UFSDevice[lun_id]->transferInfo.finished = 
true;
 
 1589         uint32_t size_accum = 0;
 
 1594         while ((length > 
count) && size_accum
 
 1598             SCSI_start = (SCSI_start << 32) |
 
 1602                     (sglist[
count].size + 1));
 
 1610             uint32_t size_to_send = sglist[
count].
size + 1;
 
 1616                        reinterpret_cast<uint8_t*
> 
 1620             size_accum += size_to_send;
 
 1632                      reinterpret_cast<uint8_t*
>(request_in), 
true, lun_id);
 
 1646     uint8_t this_lun = 0;
 
 1657                      UFSDevice[this_lun]->transferInfo.reqPos,
 
 1658                      UFSDevice[this_lun]->transferInfo.requestOut,
 
 1660                      UFSDevice[this_lun]->transferInfo.address,
 
 1661                      UFSDevice[this_lun]->transferInfo.destination,
 
 1662                      UFSDevice[this_lun]->transferInfo.finished,
 
 1663                      UFSDevice[this_lun]->transferInfo.lunID);
 
 1677                             struct UTPUPIURSP request_out, uint32_t size,
 
 1679                             bool finished, uint32_t lun_id)
 
 1682     if (
UFSDevice[lun_id]->SCSIInfoQueue.empty())
 
 1683         panic(
"No SCSI message scheduled lun:%d Doorbell: 0x%8x", lun_id,
 
 1687             responseStartAddr, 
sizeof(request_out));
 
 1690     lastinfo.
mask = req_pos;
 
 1691     lastinfo.
done = finished;
 
 1693     lastinfo.
size = size;
 
 1696     lastinfo.
lun_id = lun_id;
 
 1702     readDevice(
false, responseStartAddr, 
sizeof(request_out),
 
 1703                reinterpret_cast<uint8_t*
> 
 1719     UFSDevice[lun_id]->SCSIInfoQueue.pop_front();
 
 1721             UFSDevice[lun_id]->SCSIInfoQueue.size(), lun_id);
 
 1734                transferEnd.front().size, 
reinterpret_cast<uint8_t*
> 
 1746     stats.averageDoorbell = 
stats.maxDoorbell.value();
 
 1753     if (!
UFSDevice[lun_id]->SCSIInfoQueue.empty())
 
 1852                            SCSIDiskOffset, uint32_t lun_id)
 
 1872         new_transfer.
offset = SCSIDiskOffset;
 
 1873         new_transfer.
size = size;
 
 1874         new_transfer.
lunID = lun_id;
 
 1889         assert(!additional_action->
scheduled());
 
 1910     UFSDevice[LUN]->setTotalWrite(sg_table_length);
 
 1918         next_packet.
start = (next_packet.
start << 32) |
 
 1920         next_packet.
LUN = LUN;
 
 1924                 (sglist[
count].size + 1));
 
 1925         assert(sglist[
count].size > 0);
 
 1952     ++
stats.totalWriteUFSTransactions;
 
 1971             UFSDevice[lun]->SSDWriteDoneInfo.size());
 
 1990     stats.currentWriteSSDQueue = 
UFSDevice[lun]->SSDWriteDoneInfo.size();
 
 1991     stats.averageWriteSSDQueue = 
UFSDevice[lun]->SSDWriteDoneInfo.size();
 
 1992     ++
stats.totalWriteDiskTransactions;
 
 2009     assert(SSDWriteDoneInfo.size() > 0);
 
 2010     flashDevice->writeMemory(
 
 2011         SSDWriteDoneInfo.front().offset,
 
 2012         SSDWriteDoneInfo.front().size, memWriteCallback);
 
 2014     SSDWriteDoneInfo.pop_front();
 
 2017             SSDWriteDoneInfo.size());
 
 2029             totalWrite, amountOfWriteTransfers);
 
 2032     ++amountOfWriteTransfers;
 
 2035     assert(totalWrite >= amountOfWriteTransfers && totalWrite != 0);
 
 2038     if (totalWrite == amountOfWriteTransfers) {
 
 2041         amountOfWriteTransfers = 0;
 
 2061             start, size, (
reinterpret_cast<uint32_t *
>(
destination))[0]);
 
 2075         if (additional_action != NULL)
 
 2076             assert(!additional_action->
scheduled());
 
 2092                                   offset, uint32_t sg_table_length,
 
 2095     uint32_t size_accum = 0;
 
 2110         new_transfer.
lunID = LUN;
 
 2115         UFSDevice[LUN]->SSDReadInfo.push_back(new_transfer);
 
 2116         UFSDevice[LUN]->SSDReadInfo.back().buffer.resize(sglist[
count].size
 
 2125                                   SSDReadInfo.back().buffer[0],
 
 2132                 " 0x%8x\n", (
count + 1), (size-size_accum), size_accum);
 
 2140     UFSDevice[LUN]->SSDReadStart(sg_table_length);
 
 2143     ++
stats.totalReadUFSTransactions;
 
 2158     totalRead = total_read;
 
 2159     for (uint32_t number_handled = 0; number_handled < SSDReadInfo.size();
 
 2165         flashDevice->readMemory(SSDReadInfo.front().filePointer,
 
 2166                                 SSDReadInfo.front().size, memReadCallback);
 
 2180             " %d so far\n", lunID, totalRead, amountOfReadTransfers);
 
 2182     if (totalRead == amountOfReadTransfers) {
 
 2184         amountOfReadTransfers = 0;
 
 2200     ++amountOfReadTransfers;
 
 2206     deviceReadCallback();
 
 2221     uint8_t this_lun = 0;
 
 2234         UFSDevice[this_lun]->SSDReadInfo.pop_front();
 
 2245         ++
stats.totalReadDiskTransactions;
 
 2272     const uint8_t* temp_HCI_mem = 
reinterpret_cast<const uint8_t*
>(&
UFSHCIMem);
 
 2288     uint8_t* temp_HCI_mem = 
reinterpret_cast<uint8_t*
>(&
UFSHCIMem);