gem5  v20.0.0.3
ufs_device.cc
Go to the documentation of this file.
1 /*
2  * Copyright (c) 2013-2015 ARM Limited
3  * All rights reserved
4  *
5  * The license below extends only to copyright in the software and shall
6  * not be construed as granting a license to any other intellectual
7  * property including but not limited to intellectual property relating
8  * to a hardware implementation of the functionality of the software
9  * licensed hereunder. You may use the software subject to the license
10  * terms below provided that you ensure that this notice is replicated
11  * unmodified and in its entirety in all distributions of the software,
12  * modified or unmodified, in source code or in binary form.
13  *
14  * Redistribution and use in source and binary forms, with or without
15  * modification, are permitted provided that the following conditions are
16  * met: redistributions of source code must retain the above copyright
17  * notice, this list of conditions and the following disclaimer;
18  * redistributions in binary form must reproduce the above copyright
19  * notice, this list of conditions and the following disclaimer in the
20  * documentation and/or other materials provided with the distribution;
21  * neither the name of the copyright holders nor the names of its
22  * contributors may be used to endorse or promote products derived from
23  * this software without specific prior written permission.
24  *
25  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
26  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
27  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
28  * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
29  * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
30  * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
31  * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
32  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
33  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
34  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
35  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
36  */
37 
70 #include "dev/arm/ufs_device.hh"
71 
76  uint32_t lun_id, Callback *transfer_cb,
77  Callback *read_cb):
78  SimObject(p),
79  flashDisk(p->image[lun_id]),
80  flashDevice(p->internalflash[lun_id]),
81  blkSize(p->img_blk_size),
82  lunAvail(p->image.size()),
83  diskSize(flashDisk->size()),
84  capacityLower((diskSize - 1) & 0xffffffff),
85  capacityUpper((diskSize - SectorSize) >> 32),
86  lunID(lun_id),
87  transferCompleted(false),
88  readCompleted(false),
89  totalRead(0),
90  totalWrite(0),
91  amountOfWriteTransfers(0),
92  amountOfReadTransfers(0)
93 {
99  signalDone = transfer_cb;
102  deviceReadCallback = read_cb;
105 
110  uint32_t temp_id = ((lun_id | 0x30) << 24) | 0x3A4449;
111  lunInfo.dWord0 = 0x02060000; //data
112  lunInfo.dWord1 = 0x0200001F;
113  lunInfo.vendor0 = 0x484D5241; //ARMH (HMRA)
114  lunInfo.vendor1 = 0x424D4143; //CAMB (BMAC)
115  lunInfo.product0 = 0x356D6567; //gem5 (5meg)
116  lunInfo.product1 = 0x4D534655; //UFSM (MSFU)
117  lunInfo.product2 = 0x4C45444F; //ODEL (LEDO)
118  lunInfo.product3 = temp_id; // ID:"lun_id" ("lun_id":DI)
119  lunInfo.productRevision = 0x01000000; //0x01
120 
121  DPRINTF(UFSHostDevice, "Logic unit %d assumes that %d logic units are"
122  " present in the system\n", lunID, lunAvail);
123  DPRINTF(UFSHostDevice,"The disksize of lun: %d should be %d blocks\n",
124  lunID, diskSize);
126 }
127 
128 
134 const unsigned int UFSHostDevice::UFSSCSIDevice::controlPage[3] =
135  {0x01400A0A, 0x00000000,
136  0x0000FFFF};
137 const unsigned int UFSHostDevice::UFSSCSIDevice::recoveryPage[3] =
138  {0x03800A01, 0x00000000,
139  0xFFFF0003};
140 const unsigned int UFSHostDevice::UFSSCSIDevice::cachingPage[5] =
141  {0x00011208, 0x00000000,
142  0x00000000, 0x00000020,
143  0x00000000};
144 
146 
160 {
161  struct SCSIReply scsi_out;
162  scsi_out.reset();
163 
169  lunID << 16;
172  statusCheck(SCSIGood, scsi_out.senseCode);
173  scsi_out.senseSize = scsi_out.senseCode[0];
174  scsi_out.LUN = lunID;
175  scsi_out.status = SCSIGood;
176 
177  DPRINTF(UFSHostDevice, "SCSI command:%2x\n", SCSI_msg[4]);
180  switch (SCSI_msg[4] & 0xFF) {
181 
182  case SCSIInquiry: {
186  scsi_out.msgSize = 36;
187  scsi_out.message.dataMsg.resize(9);
188 
189  for (uint8_t count = 0; count < 9; count++)
190  scsi_out.message.dataMsg[count] =
191  (reinterpret_cast<uint32_t*> (&lunInfo))[count];
192  } break;
193 
194  case SCSIRead6: {
198  scsi_out.expectMore = 0x02;
199  scsi_out.msgSize = 0;
200 
201  uint8_t* tempptr = reinterpret_cast<uint8_t*>(&SCSI_msg[4]);
202 
207  uint32_t tmp = *reinterpret_cast<uint32_t*>(tempptr);
208  uint64_t read_offset = betoh(tmp) & 0x1FFFFF;
209 
210  uint32_t read_size = tempptr[4];
211 
212 
213  scsi_out.msgSize = read_size * blkSize;
214  scsi_out.offset = read_offset * blkSize;
215 
216  if ((read_offset + read_size) > diskSize)
217  scsi_out.status = SCSIIllegalRequest;
218 
219  DPRINTF(UFSHostDevice, "Read6 offset: 0x%8x, for %d blocks\n",
220  read_offset, read_size);
221 
225  statusCheck(scsi_out.status, scsi_out.senseCode);
226  scsi_out.senseSize = scsi_out.senseCode[0];
227  scsi_out.status = (scsi_out.status == SCSIGood) ? SCSIGood :
229 
230  } break;
231 
232  case SCSIRead10: {
233  scsi_out.expectMore = 0x02;
234  scsi_out.msgSize = 0;
235 
236  uint8_t* tempptr = reinterpret_cast<uint8_t*>(&SCSI_msg[4]);
237 
239  uint32_t tmp = *reinterpret_cast<uint32_t*>(&tempptr[2]);
240  uint64_t read_offset = betoh(tmp);
241 
242  uint16_t tmpsize = *reinterpret_cast<uint16_t*>(&tempptr[7]);
243  uint32_t read_size = betoh(tmpsize);
244 
245  scsi_out.msgSize = read_size * blkSize;
246  scsi_out.offset = read_offset * blkSize;
247 
248  if ((read_offset + read_size) > diskSize)
249  scsi_out.status = SCSIIllegalRequest;
250 
251  DPRINTF(UFSHostDevice, "Read10 offset: 0x%8x, for %d blocks\n",
252  read_offset, read_size);
253 
257  statusCheck(scsi_out.status, scsi_out.senseCode);
258  scsi_out.senseSize = scsi_out.senseCode[0];
259  scsi_out.status = (scsi_out.status == SCSIGood) ? SCSIGood :
261 
262  } break;
263 
264  case SCSIRead16: {
265  scsi_out.expectMore = 0x02;
266  scsi_out.msgSize = 0;
267 
268  uint8_t* tempptr = reinterpret_cast<uint8_t*>(&SCSI_msg[4]);
269 
271  uint32_t tmp = *reinterpret_cast<uint32_t*>(&tempptr[2]);
272  uint64_t read_offset = betoh(tmp);
273 
274  tmp = *reinterpret_cast<uint32_t*>(&tempptr[6]);
275  read_offset = (read_offset << 32) | betoh(tmp);
276 
277  tmp = *reinterpret_cast<uint32_t*>(&tempptr[10]);
278  uint32_t read_size = betoh(tmp);
279 
280  scsi_out.msgSize = read_size * blkSize;
281  scsi_out.offset = read_offset * blkSize;
282 
283  if ((read_offset + read_size) > diskSize)
284  scsi_out.status = SCSIIllegalRequest;
285 
286  DPRINTF(UFSHostDevice, "Read16 offset: 0x%8x, for %d blocks\n",
287  read_offset, read_size);
288 
292  statusCheck(scsi_out.status, scsi_out.senseCode);
293  scsi_out.senseSize = scsi_out.senseCode[0];
294  scsi_out.status = (scsi_out.status == SCSIGood) ? SCSIGood :
296 
297  } break;
298 
299  case SCSIReadCapacity10: {
303  scsi_out.msgSize = 8;
304  scsi_out.message.dataMsg.resize(2);
305  scsi_out.message.dataMsg[0] =
306  betoh(capacityLower);//last block
307  scsi_out.message.dataMsg[1] = betoh(blkSize);//blocksize
308 
309  } break;
310  case SCSIReadCapacity16: {
311  scsi_out.msgSize = 32;
312  scsi_out.message.dataMsg.resize(8);
313  scsi_out.message.dataMsg[0] =
314  betoh(capacityUpper);//last block
315  scsi_out.message.dataMsg[1] =
316  betoh(capacityLower);//last block
317  scsi_out.message.dataMsg[2] = betoh(blkSize);//blocksize
318  scsi_out.message.dataMsg[3] = 0x00;//
319  scsi_out.message.dataMsg[4] = 0x00;//reserved
320  scsi_out.message.dataMsg[5] = 0x00;//reserved
321  scsi_out.message.dataMsg[6] = 0x00;//reserved
322  scsi_out.message.dataMsg[7] = 0x00;//reserved
323 
324  } break;
325 
326  case SCSIReportLUNs: {
330  scsi_out.msgSize = (lunAvail * 8) + 8;//list + overhead
331  scsi_out.message.dataMsg.resize(2 * lunAvail + 2);
332  scsi_out.message.dataMsg[0] = (lunAvail * 8) << 24;//LUN listlength
333  scsi_out.message.dataMsg[1] = 0x00;
334 
335  for (uint8_t count = 0; count < lunAvail; count++) {
336  //LUN "count"
337  scsi_out.message.dataMsg[2 + 2 * count] = (count & 0x7F) << 8;
338  scsi_out.message.dataMsg[3 + 2 * count] = 0x00;
339  }
340 
341  } break;
342 
343  case SCSIStartStop: {
344  //Just acknowledge; not deemed relevant ATM
345  scsi_out.msgSize = 0;
346 
347  } break;
348 
349  case SCSITestUnitReady: {
350  //Just acknowledge; not deemed relevant ATM
351  scsi_out.msgSize = 0;
352 
353  } break;
354 
355  case SCSIVerify10: {
360  scsi_out.msgSize = 0;
361 
362  uint8_t* tempptr = reinterpret_cast<uint8_t*>(&SCSI_msg[4]);
363 
365  uint32_t tmp = *reinterpret_cast<uint32_t*>(&tempptr[2]);
366  uint64_t read_offset = betoh(tmp);
367 
368  uint16_t tmpsize = *reinterpret_cast<uint16_t*>(&tempptr[7]);
369  uint32_t read_size = betoh(tmpsize);
370 
371  if ((read_offset + read_size) > diskSize)
372  scsi_out.status = SCSIIllegalRequest;
373 
377  statusCheck(scsi_out.status, scsi_out.senseCode);
378  scsi_out.senseSize = scsi_out.senseCode[0];
379  scsi_out.status = (scsi_out.status == SCSIGood) ? SCSIGood :
381 
382  } break;
383 
384  case SCSIWrite6: {
389  uint8_t* tempptr = reinterpret_cast<uint8_t*>(&SCSI_msg[4]);
390 
395  uint32_t tmp = *reinterpret_cast<uint32_t*>(tempptr);
396  uint64_t write_offset = betoh(tmp) & 0x1FFFFF;
397 
398  uint32_t write_size = tempptr[4];
399 
400  scsi_out.msgSize = write_size * blkSize;
401  scsi_out.offset = write_offset * blkSize;
402  scsi_out.expectMore = 0x01;
403 
404  if ((write_offset + write_size) > diskSize)
405  scsi_out.status = SCSIIllegalRequest;
406 
407  DPRINTF(UFSHostDevice, "Write6 offset: 0x%8x, for %d blocks\n",
408  write_offset, write_size);
409 
413  statusCheck(scsi_out.status, scsi_out.senseCode);
414  scsi_out.senseSize = scsi_out.senseCode[0];
415  scsi_out.status = (scsi_out.status == SCSIGood) ? SCSIGood :
417 
418  } break;
419 
420  case SCSIWrite10: {
421  uint8_t* tempptr = reinterpret_cast<uint8_t*>(&SCSI_msg[4]);
422 
424  uint32_t tmp = *reinterpret_cast<uint32_t*>(&tempptr[2]);
425  uint64_t write_offset = betoh(tmp);
426 
427  uint16_t tmpsize = *reinterpret_cast<uint16_t*>(&tempptr[7]);
428  uint32_t write_size = betoh(tmpsize);
429 
430  scsi_out.msgSize = write_size * blkSize;
431  scsi_out.offset = write_offset * blkSize;
432  scsi_out.expectMore = 0x01;
433 
434  if ((write_offset + write_size) > diskSize)
435  scsi_out.status = SCSIIllegalRequest;
436 
437  DPRINTF(UFSHostDevice, "Write10 offset: 0x%8x, for %d blocks\n",
438  write_offset, write_size);
439 
443  statusCheck(scsi_out.status, scsi_out.senseCode);
444  scsi_out.senseSize = scsi_out.senseCode[0];
445  scsi_out.status = (scsi_out.status == SCSIGood) ? SCSIGood :
447 
448  } break;
449 
450  case SCSIWrite16: {
451  uint8_t* tempptr = reinterpret_cast<uint8_t*>(&SCSI_msg[4]);
452 
454  uint32_t tmp = *reinterpret_cast<uint32_t*>(&tempptr[2]);
455  uint64_t write_offset = betoh(tmp);
456 
457  tmp = *reinterpret_cast<uint32_t*>(&tempptr[6]);
458  write_offset = (write_offset << 32) | betoh(tmp);
459 
460  tmp = *reinterpret_cast<uint32_t*>(&tempptr[10]);
461  uint32_t write_size = betoh(tmp);
462 
463  scsi_out.msgSize = write_size * blkSize;
464  scsi_out.offset = write_offset * blkSize;
465  scsi_out.expectMore = 0x01;
466 
467  if ((write_offset + write_size) > diskSize)
468  scsi_out.status = SCSIIllegalRequest;
469 
470  DPRINTF(UFSHostDevice, "Write16 offset: 0x%8x, for %d blocks\n",
471  write_offset, write_size);
472 
476  statusCheck(scsi_out.status, scsi_out.senseCode);
477  scsi_out.senseSize = scsi_out.senseCode[0];
478  scsi_out.status = (scsi_out.status == SCSIGood) ? SCSIGood :
480 
481  } break;
482 
483  case SCSIFormatUnit: {//not yet verified
484  scsi_out.msgSize = 0;
485  scsi_out.expectMore = 0x01;
486 
487  } break;
488 
489  case SCSISendDiagnostic: {//not yet verified
490  scsi_out.msgSize = 0;
491 
492  } break;
493 
494  case SCSISynchronizeCache: {
495  //do we have cache (we don't have cache at this moment)
496  //TODO: here will synchronization happen when cache is modelled
497  scsi_out.msgSize = 0;
498 
499  } break;
500 
501  //UFS SCSI additional command set for full functionality
502  case SCSIModeSelect10:
503  //TODO:
504  //scsi_out.expectMore = 0x01;//not supported due to modepage support
505  //code isn't dead, code suggest what is to be done when implemented
506  break;
507 
508  case SCSIModeSense6: case SCSIModeSense10: {
513  if ((SCSI_msg[4] & 0x3F0000) >> 16 == 0x0A) {//control page
514  scsi_out.message.dataMsg.resize((sizeof(controlPage) >> 2) + 2);
515  scsi_out.message.dataMsg[0] = 0x00000A00;//control page code
516  scsi_out.message.dataMsg[1] = 0x00000000;//See JEDEC220 ch8
517 
518  for (uint8_t count = 0; count < 3; count++)
519  scsi_out.message.dataMsg[2 + count] = controlPage[count];
520 
521  scsi_out.msgSize = 20;
522  DPRINTF(UFSHostDevice, "CONTROL page\n");
523 
524  } else if ((SCSI_msg[4] & 0x3F0000) >> 16 == 0x01) {//recovery page
525  scsi_out.message.dataMsg.resize((sizeof(recoveryPage) >> 2)
526  + 2);
527 
528  scsi_out.message.dataMsg[0] = 0x00000100;//recovery page code
529  scsi_out.message.dataMsg[1] = 0x00000000;//See JEDEC220 ch8
530 
531  for (uint8_t count = 0; count < 3; count++)
532  scsi_out.message.dataMsg[2 + count] = recoveryPage[count];
533 
534  scsi_out.msgSize = 20;
535  DPRINTF(UFSHostDevice, "RECOVERY page\n");
536 
537  } else if ((SCSI_msg[4] & 0x3F0000) >> 16 == 0x08) {//caching page
538 
539  scsi_out.message.dataMsg.resize((sizeof(cachingPage) >> 2) + 2);
540  scsi_out.message.dataMsg[0] = 0x00001200;//caching page code
541  scsi_out.message.dataMsg[1] = 0x00000000;//See JEDEC220 ch8
542 
543  for (uint8_t count = 0; count < 5; count++)
544  scsi_out.message.dataMsg[2 + count] = cachingPage[count];
545 
546  scsi_out.msgSize = 20;
547  DPRINTF(UFSHostDevice, "CACHE page\n");
548 
549  } else if ((SCSI_msg[4] & 0x3F0000) >> 16 == 0x3F) {//ALL the pages!
550 
551  scsi_out.message.dataMsg.resize(((sizeof(controlPage) +
552  sizeof(recoveryPage) +
553  sizeof(cachingPage)) >> 2)
554  + 2);
555  scsi_out.message.dataMsg[0] = 0x00003200;//all page code
556  scsi_out.message.dataMsg[1] = 0x00000000;//See JEDEC220 ch8
557 
558  for (uint8_t count = 0; count < 3; count++)
559  scsi_out.message.dataMsg[2 + count] = recoveryPage[count];
560 
561  for (uint8_t count = 0; count < 5; count++)
562  scsi_out.message.dataMsg[5 + count] = cachingPage[count];
563 
564  for (uint8_t count = 0; count < 3; count++)
565  scsi_out.message.dataMsg[10 + count] = controlPage[count];
566 
567  scsi_out.msgSize = 52;
568  DPRINTF(UFSHostDevice, "Return ALL the pages!!!\n");
569 
570  } else inform("Wrong mode page requested\n");
571 
572  scsi_out.message.dataCount = scsi_out.msgSize << 24;
573  } break;
574 
575  case SCSIRequestSense: {
576  scsi_out.msgSize = 0;
577 
578  } break;
579 
580  case SCSIUnmap:break;//not yet verified
581 
582  case SCSIWriteBuffer: {
583  scsi_out.expectMore = 0x01;
584 
585  uint8_t* tempptr = reinterpret_cast<uint8_t*>(&SCSI_msg[4]);
586 
588  uint32_t tmp = *reinterpret_cast<uint32_t*>(&tempptr[2]);
589  uint64_t write_offset = betoh(tmp) & 0xFFFFFF;
590 
591  tmp = *reinterpret_cast<uint32_t*>(&tempptr[5]);
592  uint32_t write_size = betoh(tmp) & 0xFFFFFF;
593 
594  scsi_out.msgSize = write_size;
595  scsi_out.offset = write_offset;
596 
597  } break;
598 
599  case SCSIReadBuffer: {
605  scsi_out.expectMore = 0x02;
606 
607  uint8_t* tempptr = reinterpret_cast<uint8_t*>(&SCSI_msg[4]);
608 
610  uint32_t tmp = *reinterpret_cast<uint32_t*>(&tempptr[2]);
611  uint64_t read_offset = betoh(tmp) & 0xFFFFFF;
612 
613  tmp = *reinterpret_cast<uint32_t*>(&tempptr[5]);
614  uint32_t read_size = betoh(tmp) & 0xFFFFFF;
615 
616  scsi_out.msgSize = read_size;
617  scsi_out.offset = read_offset;
618 
619  if ((read_offset + read_size) > capacityLower * blkSize)
620  scsi_out.status = SCSIIllegalRequest;
621 
622  DPRINTF(UFSHostDevice, "Read buffer location: 0x%8x\n",
623  read_offset);
624  DPRINTF(UFSHostDevice, "Number of bytes: 0x%8x\n", read_size);
625 
626  statusCheck(scsi_out.status, scsi_out.senseCode);
627  scsi_out.senseSize = scsi_out.senseCode[0];
628  scsi_out.status = (scsi_out.status == SCSIGood) ? SCSIGood :
630 
631  } break;
632 
633  case SCSIMaintenanceIn: {
640  DPRINTF(UFSHostDevice, "Ignoring Maintenance In command\n");
642  scsi_out.senseSize = scsi_out.senseCode[0];
643  scsi_out.status = (scsi_out.status == SCSIGood) ? SCSIGood :
645  scsi_out.msgSize = 0;
646  } break;
647 
648  default: {
650  scsi_out.senseSize = scsi_out.senseCode[0];
651  scsi_out.status = (scsi_out.status == SCSIGood) ? SCSIGood :
653  scsi_out.msgSize = 0;
654  inform("Unsupported scsi message type: %2x\n", SCSI_msg[4] & 0xFF);
655  inform("0x%8x\n", SCSI_msg[0]);
656  inform("0x%8x\n", SCSI_msg[1]);
657  inform("0x%8x\n", SCSI_msg[2]);
658  inform("0x%8x\n", SCSI_msg[3]);
659  inform("0x%8x\n", SCSI_msg[4]);
660  } break;
661  }
662 
663  return scsi_out;
664 }
665 
672 void
674  uint8_t* sensecodelist)
675 {
676  for (uint8_t count = 0; count < 19; count++)
677  sensecodelist[count] = 0;
678 
679  sensecodelist[0] = 18; //sense length
680  sensecodelist[1] = 0x70; //we send a valid frame
681  sensecodelist[3] = status & 0xF; //mask to be sure + sensecode
682  sensecodelist[8] = 0x1F; //data length
683 }
684 
689 void
691  uint32_t size)
692 {
694  for (int count = 0; count < (size / SectorSize); count++)
695  flashDisk->read(&(readaddr[SectorSize*count]), (offset /
696  SectorSize) + count);
697 }
698 
703 void
705  uint32_t size)
706 {
708  for (int count = 0; count < (size / SectorSize); count++)
709  flashDisk->write(&(writeaddr[SectorSize * count]),
710  (offset / SectorSize) + count);
711 }
712 
717 UFSHostDevice::UFSHostDevice(const UFSHostDeviceParams* p) :
718  DmaDevice(p),
719  pioAddr(p->pio_addr),
720  pioSize(0x0FFF),
721  pioDelay(p->pio_latency),
722  intNum(p->int_num),
723  gic(p->gic),
724  lunAvail(p->image.size()),
725  UFSSlots(p->ufs_slots - 1),
726  readPendingNum(0),
727  writePendingNum(0),
728  activeDoorbells(0),
729  pendingDoorbells(0),
730  countInt(0),
731  transferTrack(0),
732  taskCommandTrack(0),
733  idlePhaseStart(0),
734  SCSIResumeEvent([this]{ SCSIStart(); }, name()),
735  UTPEvent([this]{ finalUTP(); }, name())
736 {
737  DPRINTF(UFSHostDevice, "The hostcontroller hosts %d Logic units\n",
738  lunAvail);
739  UFSDevice.resize(lunAvail);
740 
742  &UFSHostDevice::LUNSignal>(this);
745 
746  for (int count = 0; count < lunAvail; count++) {
748  memReadCallback);
749  }
750 
751  if (UFSSlots > 31)
752  warn("UFSSlots = %d, this will results in %d command slots",
753  UFSSlots, (UFSSlots & 0x1F));
754 
755  if ((UFSSlots & 0x1F) == 0)
756  fatal("Number of UFS command slots should be between 1 and 32.");
757 
758  setValues();
759 }
760 
766 UFSHostDeviceParams::create()
767 {
768  return new UFSHostDevice(this);
769 }
770 
771 
772 void
774 {
776 
777  using namespace Stats;
778 
779  std::string UFSHost_name = name() + ".UFSDiskHost";
780 
781  // Register the stats
784  .name(UFSHost_name + ".currentSCSIQueue")
785  .desc("Most up to date length of the command queue")
786  .flags(none);
788  .name(UFSHost_name + ".currentReadSSDQueue")
789  .desc("Most up to date length of the read SSD queue")
790  .flags(none);
792  .name(UFSHost_name + ".currentWriteSSDQueue")
793  .desc("Most up to date length of the write SSD queue")
794  .flags(none);
795 
798  .name(UFSHost_name + ".totalReadSSD")
799  .desc("Number of bytes read from SSD")
800  .flags(none);
801 
803  .name(UFSHost_name + ".totalWrittenSSD")
804  .desc("Number of bytes written to SSD")
805  .flags(none);
806 
808  .name(UFSHost_name + ".totalReadDiskTransactions")
809  .desc("Number of transactions from disk")
810  .flags(none);
812  .name(UFSHost_name + ".totalWriteDiskTransactions")
813  .desc("Number of transactions to disk")
814  .flags(none);
816  .name(UFSHost_name + ".totalReadUFSTransactions")
817  .desc("Number of transactions from device")
818  .flags(none);
820  .name(UFSHost_name + ".totalWriteUFSTransactions")
821  .desc("Number of transactions to device")
822  .flags(none);
823 
826  .name(UFSHost_name + ".averageReadSSDBandwidth")
827  .desc("Average read bandwidth (bytes/s)")
828  .flags(nozero);
829 
831 
833  .name(UFSHost_name + ".averageWriteSSDBandwidth")
834  .desc("Average write bandwidth (bytes/s)")
835  .flags(nozero);
836 
838 
840  .name(UFSHost_name + ".averageSCSIQueueLength")
841  .desc("Average command queue length")
842  .flags(nozero);
844  .name(UFSHost_name + ".averageReadSSDQueueLength")
845  .desc("Average read queue length")
846  .flags(nozero);
848  .name(UFSHost_name + ".averageWriteSSDQueueLength")
849  .desc("Average write queue length")
850  .flags(nozero);
851 
854  .name(UFSHost_name + ".curDoorbell")
855  .desc("Most up to date number of doorbells used")
856  .flags(none);
857 
859 
861  .name(UFSHost_name + ".maxDoorbell")
862  .desc("Maximum number of doorbells utilized")
863  .flags(none);
865  .name(UFSHost_name + ".averageDoorbell")
866  .desc("Average number of Doorbells used")
867  .flags(nozero);
868 
871  .init(100)
872  .name(UFSHost_name + ".transactionLatency")
873  .desc("Histogram of transaction times")
874  .flags(pdf);
875 
877  .init(100)
878  .name(UFSHost_name + ".idlePeriods")
879  .desc("Histogram of idle times")
880  .flags(pdf);
881 
882 }
883 
888 {
894  UFSHCIMem.HCCAP = 0x06070000 | (UFSSlots & 0x1F);
895  UFSHCIMem.HCversion = 0x00010000; //version is 1.0
896  UFSHCIMem.HCHCDDID = 0xAA003C3C;// Arbitrary number
897  UFSHCIMem.HCHCPMID = 0x41524D48; //ARMH (not an official MIPI number)
898  UFSHCIMem.TRUTRLDBR = 0x00;
899  UFSHCIMem.TMUTMRLDBR = 0x00;
900  UFSHCIMem.CMDUICCMDR = 0x00;
901  // We can process CMD, TM, TR, device present
903  UFSHCIMem.TRUTRLBA = 0x00;
904  UFSHCIMem.TRUTRLBAU = 0x00;
905  UFSHCIMem.TMUTMRLBA = 0x00;
906  UFSHCIMem.TMUTMRLBAU = 0x00;
907 }
908 
915 {
916  AddrRangeList ranges;
917  ranges.push_back(RangeSize(pioAddr, pioSize));
918  return ranges;
919 }
920 
926 Tick
928 {
929  uint32_t data = 0;
930 
931  switch (pkt->getAddr() & 0xFF)
932  {
933 
935  data = UFSHCIMem.HCCAP;
936  break;
937 
938  case regUFSVersion:
939  data = UFSHCIMem.HCversion;
940  break;
941 
942  case regControllerDEVID:
943  data = UFSHCIMem.HCHCDDID;
944  break;
945 
946  case regControllerPRODID:
947  data = UFSHCIMem.HCHCPMID;
948  break;
949 
950  case regInterruptStatus:
953  //TODO: Revise and extend
954  clearInterrupt();
955  break;
956 
957  case regInterruptEnable:
959  break;
960 
961  case regControllerStatus:
963  break;
964 
965  case regControllerEnable:
967  break;
968 
970  data = UFSHCIMem.ORUECPA;
971  break;
972 
974  data = UFSHCIMem.ORUECDL;
975  break;
976 
978  data = UFSHCIMem.ORUECN;
979  break;
980 
982  data = UFSHCIMem.ORUECT;
983  break;
984 
985  case regUICErrorCodeDME:
986  data = UFSHCIMem.ORUECDME;
987  break;
988 
990  data = UFSHCIMem.ORUTRIACR;
991  break;
992 
994  data = UFSHCIMem.TRUTRLBA;
995  break;
996 
998  data = UFSHCIMem.TRUTRLBAU;
999  break;
1000 
1002  data = UFSHCIMem.TRUTRLDBR;
1003  break;
1004 
1006  data = UFSHCIMem.TRUTRLCLR;
1007  break;
1008 
1010  data = UFSHCIMem.TRUTRLRSR;
1011  break;
1012 
1014  data = UFSHCIMem.TMUTMRLBA;
1015  break;
1016 
1018  data = UFSHCIMem.TMUTMRLBAU;
1019  break;
1020 
1021  case regUTPTaskREQDoorbell:
1022  data = UFSHCIMem.TMUTMRLDBR;
1023  break;
1024 
1026  data = UFSHCIMem.TMUTMRLCLR;
1027  break;
1028 
1030  data = UFSHCIMem.TMUTMRLRSR;
1031  break;
1032 
1033  case regUICCommand:
1034  data = UFSHCIMem.CMDUICCMDR;
1035  break;
1036 
1037  case regUICCommandArg1:
1038  data = UFSHCIMem.CMDUCMDARG1;
1039  break;
1040 
1041  case regUICCommandArg2:
1042  data = UFSHCIMem.CMDUCMDARG2;
1043  break;
1044 
1045  case regUICCommandArg3:
1046  data = UFSHCIMem.CMDUCMDARG3;
1047  break;
1048 
1049  default:
1050  data = 0x00;
1051  break;
1052  }
1053 
1054  pkt->setLE<uint32_t>(data);
1055  pkt->makeResponse();
1056  return pioDelay;
1057 }
1058 
1064 Tick
1066 {
1067  uint32_t data = 0;
1068 
1069  switch (pkt->getSize()) {
1070 
1071  case 1:
1072  data = pkt->getLE<uint8_t>();
1073  break;
1074 
1075  case 2:
1076  data = pkt->getLE<uint16_t>();
1077  break;
1078 
1079  case 4:
1080  data = pkt->getLE<uint32_t>();
1081  break;
1082 
1083  default:
1084  panic("Undefined UFSHCD controller write size!\n");
1085  break;
1086  }
1087 
1088  switch (pkt->getAddr() & 0xFF)
1089  {
1090  case regControllerCapabilities://you shall not write to this
1091  break;
1092 
1093  case regUFSVersion://you shall not write to this
1094  break;
1095 
1096  case regControllerDEVID://you shall not write to this
1097  break;
1098 
1099  case regControllerPRODID://you shall not write to this
1100  break;
1101 
1102  case regInterruptStatus://you shall not write to this
1103  break;
1104 
1105  case regInterruptEnable:
1107  break;
1108 
1109  case regControllerStatus:
1111  break;
1112 
1113  case regControllerEnable:
1115  break;
1116 
1119  break;
1120 
1123  break;
1124 
1126  UFSHCIMem.ORUECN = data;
1127  break;
1128 
1130  UFSHCIMem.ORUECT = data;
1131  break;
1132 
1133  case regUICErrorCodeDME:
1135  break;
1136 
1139  break;
1140 
1143  if (((UFSHCIMem.TRUTRLBA | UFSHCIMem.TRUTRLBAU) != 0x00) &&
1144  ((UFSHCIMem.TMUTMRLBA | UFSHCIMem.TMUTMRLBAU)!= 0x00))
1146  break;
1147 
1149  UFSHCIMem.TRUTRLBAU = data;
1150  if (((UFSHCIMem.TRUTRLBA | UFSHCIMem.TRUTRLBAU) != 0x00) &&
1151  ((UFSHCIMem.TMUTMRLBA | UFSHCIMem.TMUTMRLBAU) != 0x00))
1153  break;
1154 
1156  if (!(UFSHCIMem.TRUTRLDBR) && data)
1159  requestHandler();
1160  break;
1161 
1164  break;
1165 
1168  break;
1169 
1172  if (((UFSHCIMem.TRUTRLBA | UFSHCIMem.TRUTRLBAU) != 0x00) &&
1173  ((UFSHCIMem.TMUTMRLBA | UFSHCIMem.TMUTMRLBAU) != 0x00))
1175  break;
1176 
1179  if (((UFSHCIMem.TRUTRLBA | UFSHCIMem.TRUTRLBAU) != 0x00) &&
1180  ((UFSHCIMem.TMUTMRLBA | UFSHCIMem.TMUTMRLBAU) != 0x00))
1182  break;
1183 
1184  case regUTPTaskREQDoorbell:
1186  requestHandler();
1187  break;
1188 
1191  break;
1192 
1195  break;
1196 
1197  case regUICCommand:
1199  requestHandler();
1200  break;
1201 
1202  case regUICCommandArg1:
1204  break;
1205 
1206  case regUICCommandArg2:
1208  break;
1209 
1210  case regUICCommandArg3:
1212  break;
1213 
1214  default:break;//nothing happens, you try to access a register that
1215  //does not exist
1216 
1217  }
1218 
1219  pkt->makeResponse();
1220  return pioDelay;
1221 }
1222 
1228 void
1230 {
1231  Addr address = 0x00;
1232  int mask = 0x01;
1233  int size;
1234  int count = 0;
1235  struct taskStart task_info;
1236  struct transferStart transferstart_info;
1237  transferstart_info.done = 0;
1238 
1244  while (((UFSHCIMem.CMDUICCMDR > 0x00) |
1245  ((UFSHCIMem.TMUTMRLDBR ^ taskCommandTrack) > 0x00) |
1246  ((UFSHCIMem.TRUTRLDBR ^ transferTrack) > 0x00)) ) {
1247 
1248  if (UFSHCIMem.CMDUICCMDR > 0x00) {
1253  commandHandler();
1256  UFSHCIMem.CMDUICCMDR = 0x00;
1257  return; //command, nothing more we can do
1258 
1259  } else if ((UFSHCIMem.TMUTMRLDBR ^ taskCommandTrack) > 0x00) {
1264  size = sizeof(UTPUPIUTaskReq);
1267  address = UFSHCIMem.TMUTMRLBAU;
1268  //<-64 bit
1269  address = (count * size) + (address << 32) +
1271  taskCommandTrack |= mask << count;
1272 
1273  inform("UFSmodel received a task from the system; this might"
1274  " lead to untested behaviour.\n");
1275 
1276  task_info.mask = mask << count;
1277  task_info.address = address;
1278  task_info.size = size;
1279  task_info.done = UFSHCIMem.TMUTMRLDBR;
1280  taskInfo.push_back(task_info);
1281  taskEventQueue.push_back(
1282  EventFunctionWrapper([this]{ taskStart(); }, name()));
1283  writeDevice(&taskEventQueue.back(), false, address, size,
1284  reinterpret_cast<uint8_t*>
1285  (&taskInfo.back().destination), 0, 0);
1286 
1287  } else if ((UFSHCIMem.TRUTRLDBR ^ transferTrack) > 0x00) {
1293  size = sizeof(UTPTransferReqDesc);
1296  address = UFSHCIMem.TRUTRLBAU;
1297  //<-64 bit
1298  address = (count * size) + (address << 32) + UFSHCIMem.TRUTRLBA;
1299 
1300  transferTrack |= mask << count;
1301  DPRINTF(UFSHostDevice, "Doorbell register: 0x%8x select #:"
1302  " 0x%8x completion info: 0x%8x\n", UFSHCIMem.TRUTRLDBR,
1303  count, transferstart_info.done);
1304 
1305  transferstart_info.done = UFSHCIMem.TRUTRLDBR;
1306 
1308  transactionStart[count] = curTick(); //note the start time
1309  ++activeDoorbells;
1313 
1319  transferstart_info.mask = mask << count;
1320  transferstart_info.address = address;
1321  transferstart_info.size = size;
1322  transferstart_info.done = UFSHCIMem.TRUTRLDBR;
1323  transferStartInfo.push_back(transferstart_info);
1324 
1326  transferStartInfo.back().destination = new struct
1328  DPRINTF(UFSHostDevice, "Initial transfer start: 0x%8x\n",
1329  transferstart_info.done);
1330  transferEventQueue.push_back(
1331  EventFunctionWrapper([this]{ transferStart(); }, name()));
1332 
1333  if (transferEventQueue.size() < 2) {
1334  writeDevice(&transferEventQueue.front(), false,
1335  address, size, reinterpret_cast<uint8_t*>
1336  (transferStartInfo.front().destination),0, 0);
1337  DPRINTF(UFSHostDevice, "Transfer scheduled\n");
1338  }
1339  }
1340  }
1341 }
1342 
1347 void
1349 {
1350  DPRINTF(UFSHostDevice, "Task start");
1351  taskHandler(&taskInfo.front().destination, taskInfo.front().mask,
1352  taskInfo.front().address, taskInfo.front().size);
1353  taskInfo.pop_front();
1354  taskEventQueue.pop_front();
1355 }
1356 
1361 void
1363 {
1364  DPRINTF(UFSHostDevice, "Enter transfer event\n");
1365  transferHandler(transferStartInfo.front().destination,
1366  transferStartInfo.front().mask,
1367  transferStartInfo.front().address,
1368  transferStartInfo.front().size,
1369  transferStartInfo.front().done);
1370 
1371  transferStartInfo.pop_front();
1372  DPRINTF(UFSHostDevice, "Transfer queue size at end of event: "
1373  "0x%8x\n", transferEventQueue.size());
1374 }
1375 
1381 void
1383 {
1384  if (UFSHCIMem.CMDUICCMDR == 0x16) {
1385  UFSHCIMem.ORHostControllerStatus |= 0x0F;//link startup
1386  }
1387 
1388 }
1389 
1395 void
1397  uint32_t req_pos, Addr finaladdress, uint32_t
1398  finalsize)
1399 {
1404  inform("taskHandler\n");
1405  inform("%8x\n", request_in->header.dWord0);
1406  inform("%8x\n", request_in->header.dWord1);
1407  inform("%8x\n", request_in->header.dWord2);
1408 
1409  request_in->header.dWord2 &= 0xffffff00;
1410 
1411  UFSHCIMem.TMUTMRLDBR &= ~(req_pos);
1412  taskCommandTrack &= ~(req_pos);
1414 
1415  readDevice(true, finaladdress, finalsize, reinterpret_cast<uint8_t*>
1416  (request_in), true, NULL);
1417 
1418 }
1419 
1431 void
1433  int req_pos, Addr finaladdress, uint32_t
1434  finalsize, uint32_t done)
1435 {
1436 
1437  Addr cmd_desc_addr = 0x00;
1438 
1439 
1440  //acknowledge handling of the message
1441  DPRINTF(UFSHostDevice, "SCSI message detected\n");
1442  request_in->header.dWord2 &= 0xffffff00;
1443  SCSIInfo.RequestIn = request_in;
1444  SCSIInfo.reqPos = req_pos;
1445  SCSIInfo.finalAddress = finaladdress;
1446  SCSIInfo.finalSize = finalsize;
1447  SCSIInfo.destination.resize(request_in->PRDTableOffset * 4
1448  + request_in->PRDTableLength * sizeof(UFSHCDSGEntry));
1449  SCSIInfo.done = done;
1450 
1451  assert(!SCSIResumeEvent.scheduled());
1455  cmd_desc_addr = request_in->commandDescBaseAddrHi;
1456  cmd_desc_addr = (cmd_desc_addr << 32) |
1457  (request_in->commandDescBaseAddrLo & 0xffffffff);
1458 
1459  writeDevice(&SCSIResumeEvent, false, cmd_desc_addr,
1460  SCSIInfo.destination.size(), &SCSIInfo.destination[0],0, 0);
1461 
1462  DPRINTF(UFSHostDevice, "SCSI scheduled\n");
1463 
1464  transferEventQueue.pop_front();
1465 }
1466 
1474 void
1476 {
1477  DPRINTF(UFSHostDevice, "SCSI message on hold until ready\n");
1478  uint32_t LUN = SCSIInfo.destination[2];
1479  UFSDevice[LUN]->SCSIInfoQueue.push_back(SCSIInfo);
1480 
1481  DPRINTF(UFSHostDevice, "SCSI queue %d has %d elements\n", LUN,
1482  UFSDevice[LUN]->SCSIInfoQueue.size());
1483 
1485  if (UFSDevice[LUN]->SCSIInfoQueue.size() < 2) //LUN is available
1486  SCSIResume(LUN);
1487 
1488  else if (UFSDevice[LUN]->SCSIInfoQueue.size() > 32)
1489  panic("SCSI queue is getting too big %d\n", UFSDevice[LUN]->
1490  SCSIInfoQueue.size());
1491 
1496  if (!transferEventQueue.empty()) {
1497 
1502  writeDevice(&transferEventQueue.front(), false,
1503  transferStartInfo.front().address,
1504  transferStartInfo.front().size, reinterpret_cast<uint8_t*>
1505  (transferStartInfo.front().destination), 0, 0);
1506 
1507  DPRINTF(UFSHostDevice, "Transfer scheduled");
1508  }
1509 }
1510 
1519 void
1521 {
1522  DPRINTF(UFSHostDevice, "SCSIresume\n");
1523  if (UFSDevice[lun_id]->SCSIInfoQueue.empty())
1524  panic("No SCSI message scheduled lun:%d Doorbell: 0x%8x", lun_id,
1526 
1528  struct UTPTransferReqDesc* request_in = UFSDevice[lun_id]->
1529  SCSIInfoQueue.front().RequestIn;
1530 
1531  uint32_t req_pos = UFSDevice[lun_id]->SCSIInfoQueue.front().reqPos;
1532 
1533  Addr finaladdress = UFSDevice[lun_id]->SCSIInfoQueue.front().
1534  finalAddress;
1535 
1536  uint32_t finalsize = UFSDevice[lun_id]->SCSIInfoQueue.front().finalSize;
1537 
1538  uint32_t* transfercommand = reinterpret_cast<uint32_t*>
1539  (&(UFSDevice[lun_id]->SCSIInfoQueue.front().destination[0]));
1540 
1541  DPRINTF(UFSHostDevice, "Task tag: 0x%8x\n", transfercommand[0]>>24);
1543  request_out_datain = UFSDevice[(transfercommand[0] & 0xFF0000) >> 16]->
1544  SCSICMDHandle(transfercommand);
1545 
1547 
1552  request_in->header.dWord0 = ((request_in->header.dWord0 >> 24) == 0x21)
1553  ? 0x36 : 0x21;
1554  UFSDevice[lun_id]->transferInfo.requestOut.header.dWord0 =
1555  request_in->header.dWord0 | (request_out_datain.LUN << 8)
1556  | (transfercommand[0] & 0xFF000000);
1558  UFSDevice[lun_id]->transferInfo.requestOut.header.dWord1 = 0x00000000 |
1559  (request_out_datain.status << 24);
1561  UFSDevice[lun_id]->transferInfo.requestOut.header.dWord2 = 0x00000000 |
1562  ((request_out_datain.senseSize + 2) << 24) | 0x05;
1564  UFSDevice[lun_id]->transferInfo.requestOut.senseDataLen =
1566 
1567  //data
1568  for (uint8_t count = 0; count<request_out_datain.senseSize; count++) {
1569  UFSDevice[lun_id]->transferInfo.requestOut.senseData[count] =
1571  }
1572 
1573  /*
1574  * At position defined by "request_in->PRDTableOffset" (counting 32 bit
1575  * words) in array "transfercommand" we have a scatter gather list, which
1576  * is usefull to us if we interpreted it as a UFSHCDSGEntry structure.
1577  */
1578  struct UFSHCDSGEntry* sglist = reinterpret_cast<UFSHCDSGEntry*>
1579  (&(transfercommand[(request_in->PRDTableOffset)]));
1580 
1581  uint32_t length = request_in->PRDTableLength;
1582  DPRINTF(UFSHostDevice, "# PRDT entries: %d\n", length);
1583 
1584  Addr response_addr = request_in->commandDescBaseAddrHi;
1585  response_addr = (response_addr << 32) |
1586  ((request_in->commandDescBaseAddrLo +
1587  (request_in->responseUPIULength << 2)) & 0xffffffff);
1588 
1590  UFSDevice[lun_id]->transferInfo.responseStartAddr = response_addr;
1591  UFSDevice[lun_id]->transferInfo.reqPos = req_pos;
1592  UFSDevice[lun_id]->transferInfo.size = finalsize;
1593  UFSDevice[lun_id]->transferInfo.address = finaladdress;
1594  UFSDevice[lun_id]->transferInfo.destination = reinterpret_cast<uint8_t*>
1595  (UFSDevice[lun_id]->SCSIInfoQueue.front().RequestIn);
1596  UFSDevice[lun_id]->transferInfo.finished = true;
1597  UFSDevice[lun_id]->transferInfo.lunID = request_out_datain.LUN;
1598 
1604  if (request_out_datain.expectMore == 0x01) {
1607  length, sglist);
1608 
1609  } else if (request_out_datain.expectMore == 0x02) {
1612  request_out_datain.offset, length, sglist);
1613 
1614  } else {
1616  uint32_t count = 0;
1617  uint32_t size_accum = 0;
1618  DPRINTF(UFSHostDevice, "Data DMA size: 0x%8x\n",
1620 
1622  while ((length > count) && size_accum
1623  < (request_out_datain.msgSize - 1) &&
1624  (request_out_datain.msgSize != 0x00)) {
1625  Addr SCSI_start = sglist[count].upperAddr;
1626  SCSI_start = (SCSI_start << 32) |
1627  (sglist[count].baseAddr & 0xFFFFFFFF);
1628  DPRINTF(UFSHostDevice, "Data DMA start: 0x%8x\n", SCSI_start);
1629  DPRINTF(UFSHostDevice, "Data DMA size: 0x%8x\n",
1630  (sglist[count].size + 1));
1638  uint32_t size_to_send = sglist[count].size + 1;
1639 
1640  if (request_out_datain.msgSize < (size_to_send + size_accum))
1641  size_to_send = request_out_datain.msgSize - size_accum;
1642 
1643  readDevice(false, SCSI_start, size_to_send,
1644  reinterpret_cast<uint8_t*>
1645  (&(request_out_datain.message.dataMsg[size_accum])),
1646  false, NULL);
1647 
1648  size_accum += size_to_send;
1649  DPRINTF(UFSHostDevice, "Total remaining: 0x%8x,accumulated so far"
1650  " : 0x%8x\n", (request_out_datain.msgSize - size_accum),
1651  size_accum);
1652 
1653  ++count;
1654  DPRINTF(UFSHostDevice, "Transfer #: %d\n", count);
1655  }
1656 
1658  transferDone(response_addr, req_pos, UFSDevice[lun_id]->
1659  transferInfo.requestOut, finalsize, finaladdress,
1660  reinterpret_cast<uint8_t*>(request_in), true, lun_id);
1661  }
1662 
1663  DPRINTF(UFSHostDevice, "SCSI resume done\n");
1664 }
1665 
1671 void
1673 {
1674  uint8_t this_lun = 0;
1675 
1676  //while we haven't found the right lun, keep searching
1677  while ((this_lun < lunAvail) && !UFSDevice[this_lun]->finishedCommand())
1678  ++this_lun;
1679 
1680  if (this_lun < lunAvail) {
1681  //Clear signal.
1682  UFSDevice[this_lun]->clearSignal();
1683  //found it; call transferDone
1684  transferDone(UFSDevice[this_lun]->transferInfo.responseStartAddr,
1685  UFSDevice[this_lun]->transferInfo.reqPos,
1686  UFSDevice[this_lun]->transferInfo.requestOut,
1687  UFSDevice[this_lun]->transferInfo.size,
1688  UFSDevice[this_lun]->transferInfo.address,
1689  UFSDevice[this_lun]->transferInfo.destination,
1690  UFSDevice[this_lun]->transferInfo.finished,
1691  UFSDevice[this_lun]->transferInfo.lunID);
1692  }
1693 
1694  else
1695  panic("no LUN finished in tick %d\n", curTick());
1696 }
1697 
1703 void
1704 UFSHostDevice::transferDone(Addr responseStartAddr, uint32_t req_pos,
1705  struct UTPUPIURSP request_out, uint32_t size,
1706  Addr address, uint8_t* destination,
1707  bool finished, uint32_t lun_id)
1708 {
1710  if (UFSDevice[lun_id]->SCSIInfoQueue.empty())
1711  panic("No SCSI message scheduled lun:%d Doorbell: 0x%8x", lun_id,
1713 
1714  DPRINTF(UFSHostDevice, "DMA start: 0x%8x; DMA size: 0x%8x\n",
1715  responseStartAddr, sizeof(request_out));
1716 
1717  struct transferStart lastinfo;
1718  lastinfo.mask = req_pos;
1719  lastinfo.done = finished;
1720  lastinfo.address = address;
1721  lastinfo.size = size;
1722  lastinfo.destination = reinterpret_cast<UTPTransferReqDesc*>
1723  (destination);
1724  lastinfo.lun_id = lun_id;
1725 
1726  transferEnd.push_back(lastinfo);
1727 
1728  DPRINTF(UFSHostDevice, "Transfer done start\n");
1729 
1730  readDevice(false, responseStartAddr, sizeof(request_out),
1731  reinterpret_cast<uint8_t*>
1732  (&(UFSDevice[lun_id]->transferInfo.requestOut)),
1733  true, &UTPEvent);
1734 }
1735 
1742 void
1744 {
1745  uint32_t lun_id = transferEnd.front().lun_id;
1746 
1747  UFSDevice[lun_id]->SCSIInfoQueue.pop_front();
1748  DPRINTF(UFSHostDevice, "SCSIInfoQueue size: %d, lun: %d\n",
1749  UFSDevice[lun_id]->SCSIInfoQueue.size(), lun_id);
1750 
1752  if (UFSHCIMem.TRUTRLDBR & transferEnd.front().mask) {
1753  uint8_t count = 0;
1754  while (!(transferEnd.front().mask & (0x1 << count)))
1755  ++count;
1757  transactionStart[count]);
1758  }
1759 
1761  readDevice(true, transferEnd.front().address,
1762  transferEnd.front().size, reinterpret_cast<uint8_t*>
1763  (transferEnd.front().destination), true, NULL);
1764 
1766  transferTrack &= ~(transferEnd.front().mask);
1767  --activeDoorbells;
1768  ++pendingDoorbells;
1769  garbage.push_back(transferEnd.front().destination);
1770  transferEnd.pop_front();
1771  DPRINTF(UFSHostDevice, "UTP handled\n");
1772 
1775 
1776  DPRINTF(UFSHostDevice, "activeDoorbells: %d, pendingDoorbells: %d,"
1777  " garbage: %d, TransferEvent: %d\n", activeDoorbells,
1778  pendingDoorbells, garbage.size(), transferEventQueue.size());
1779 
1781  if (!UFSDevice[lun_id]->SCSIInfoQueue.empty())
1782  SCSIResume(lun_id);
1783 }
1784 
1788 void
1790 {
1791  DPRINTF(UFSHostDevice, "Read done start\n");
1792  --readPendingNum;
1793 
1795  if (garbage.size() > 0) {
1796  delete garbage.front();
1797  garbage.pop_front();
1798  }
1799 
1801  if (!(UFSHCIMem.ORInterruptStatus & 0x01)) {
1804  }
1805 
1806 
1807  if (!readDoneEvent.empty()) {
1808  readDoneEvent.pop_front();
1809  }
1810 }
1811 
1816 void
1818 {
1820  countInt++;
1821 
1824  pendingDoorbells = 0;
1825  DPRINTF(UFSHostDevice, "Clear doorbell %X\n", UFSHCIMem.TRUTRLDBR);
1826 
1827  checkDrain();
1828 
1830  gic->sendInt(intNum);
1831  DPRINTF(UFSHostDevice, "Send interrupt @ transaction: 0x%8x!\n",
1832  countInt);
1833 }
1834 
1839 void
1841 {
1842  gic->clearInt(intNum);
1843  DPRINTF(UFSHostDevice, "Clear interrupt: 0x%8x!\n", countInt);
1844 
1845  checkDrain();
1846 
1847  if (!(UFSHCIMem.TRUTRLDBR)) {
1848  idlePhaseStart = curTick();
1849  }
1851 }
1852 
1877 void
1878 UFSHostDevice::writeDevice(Event* additional_action, bool toDisk, Addr
1879  start, int size, uint8_t* destination, uint64_t
1880  SCSIDiskOffset, uint32_t lun_id)
1881 {
1882  DPRINTF(UFSHostDevice, "Write transaction Start: 0x%8x; Size: %d\n",
1883  start, size);
1884 
1886  if (toDisk) {
1887  ++writePendingNum;
1888 
1889  while (!writeDoneEvent.empty() && (writeDoneEvent.front().when()
1890  < curTick()))
1891  writeDoneEvent.pop_front();
1892 
1893  writeDoneEvent.push_back(
1894  EventFunctionWrapper([this]{ writeDone(); },
1895  name()));
1896  assert(!writeDoneEvent.back().scheduled());
1897 
1899  struct transferInfo new_transfer;
1900  new_transfer.offset = SCSIDiskOffset;
1901  new_transfer.size = size;
1902  new_transfer.lunID = lun_id;
1903  new_transfer.filePointer = 0;
1904  SSDWriteinfo.push_back(new_transfer);
1905 
1907  SSDWriteinfo.back().buffer.resize(size);
1908 
1910  dmaPort.dmaAction(MemCmd::ReadReq, start, size,
1911  &writeDoneEvent.back(),
1912  &SSDWriteinfo.back().buffer[0], 0);
1913  //yes, a readreq at a write device function is correct.
1914  DPRINTF(UFSHostDevice, "Write to disk scheduled\n");
1915 
1916  } else {
1917  assert(!additional_action->scheduled());
1918  dmaPort.dmaAction(MemCmd::ReadReq, start, size,
1919  additional_action, destination, 0);
1920  DPRINTF(UFSHostDevice, "Write scheduled\n");
1921  }
1922 }
1923 
1929 void
1930 UFSHostDevice::manageWriteTransfer(uint8_t LUN, uint64_t offset, uint32_t
1931  sg_table_length, struct UFSHCDSGEntry*
1932  sglist)
1933 {
1934  struct writeToDiskBurst next_packet;
1935 
1936  next_packet.SCSIDiskOffset = offset;
1937 
1938  UFSDevice[LUN]->setTotalWrite(sg_table_length);
1939 
1944  for (uint32_t count = 0; count < sg_table_length; count++) {
1945  next_packet.start = sglist[count].upperAddr;
1946  next_packet.start = (next_packet.start << 32) |
1947  (sglist[count].baseAddr & 0xFFFFFFFF);
1948  next_packet.LUN = LUN;
1949  DPRINTF(UFSHostDevice, "Write data DMA start: 0x%8x\n",
1950  next_packet.start);
1951  DPRINTF(UFSHostDevice, "Write data DMA size: 0x%8x\n",
1952  (sglist[count].size + 1));
1953  assert(sglist[count].size > 0);
1954 
1955  if (count != 0)
1956  next_packet.SCSIDiskOffset = next_packet.SCSIDiskOffset +
1957  (sglist[count - 1].size + 1);
1958 
1959  next_packet.size = sglist[count].size + 1;
1960 
1962  if (dmaWriteInfo.empty())
1963  writeDevice(NULL, true, next_packet.start, next_packet.size,
1964  NULL, next_packet.SCSIDiskOffset, next_packet.LUN);
1965  else
1966  DPRINTF(UFSHostDevice, "Write not initiated queue: %d\n",
1967  dmaWriteInfo.size());
1968 
1969  dmaWriteInfo.push_back(next_packet);
1970  DPRINTF(UFSHostDevice, "Write Location: 0x%8x\n",
1971  next_packet.SCSIDiskOffset);
1972 
1973  DPRINTF(UFSHostDevice, "Write transfer #: 0x%8x\n", count + 1);
1974 
1976  stats.totalWrittenSSD += (sglist[count].size + 1);
1977  }
1978 
1981 }
1982 
1988 void
1990 {
1992  assert(dmaWriteInfo.size() > 0);
1993  dmaWriteInfo.pop_front();
1994  assert(SSDWriteinfo.size() > 0);
1995  uint32_t lun = SSDWriteinfo.front().lunID;
1996 
1998  DPRINTF(UFSHostDevice, "Write done entered, queue: %d\n",
1999  UFSDevice[lun]->SSDWriteDoneInfo.size());
2001  UFSDevice[lun]->writeFlash(&SSDWriteinfo.front().buffer[0],
2002  SSDWriteinfo.front().offset,
2003  SSDWriteinfo.front().size);
2004 
2010  UFSDevice[lun]->SSDWriteDoneInfo.push_back(SSDWriteinfo.front());
2011  SSDWriteinfo.pop_front();
2012 
2013  --writePendingNum;
2015  UFSDevice[lun]->SSDWriteStart();
2016 
2018  stats.currentWriteSSDQueue = UFSDevice[lun]->SSDWriteDoneInfo.size();
2019  stats.averageWriteSSDQueue = UFSDevice[lun]->SSDWriteDoneInfo.size();
2021 
2023  if (!dmaWriteInfo.empty())
2024  writeDevice(NULL, true, dmaWriteInfo.front().start,
2025  dmaWriteInfo.front().size, NULL,
2026  dmaWriteInfo.front().SCSIDiskOffset,
2027  dmaWriteInfo.front().LUN);
2028  DPRINTF(UFSHostDevice, "Write done end\n");
2029 }
2030 
2034 void
2036 {
2037  assert(SSDWriteDoneInfo.size() > 0);
2038  flashDevice->writeMemory(
2039  SSDWriteDoneInfo.front().offset,
2040  SSDWriteDoneInfo.front().size, memWriteCallback);
2041 
2042  SSDWriteDoneInfo.pop_front();
2043 
2044  DPRINTF(UFSHostDevice, "Write is started; left in queue: %d\n",
2045  SSDWriteDoneInfo.size());
2046 }
2047 
2048 
2053 void
2055 {
2056  DPRINTF(UFSHostDevice, "Write disk, aiming for %d messages, %d so far\n",
2057  totalWrite, amountOfWriteTransfers);
2058 
2059  //we have done one extra transfer
2060  ++amountOfWriteTransfers;
2061 
2063  assert(totalWrite >= amountOfWriteTransfers && totalWrite != 0);
2064 
2066  if (totalWrite == amountOfWriteTransfers) {
2067  DPRINTF(UFSHostDevice, "Write transactions finished\n");
2068  totalWrite = 0;
2069  amountOfWriteTransfers = 0;
2070 
2071  //Callback UFS Host
2072  setSignal();
2073  signalDone->process();
2074  }
2075 
2076 }
2077 
2083 void
2084 UFSHostDevice::readDevice(bool lastTransfer, Addr start, uint32_t size,
2085  uint8_t* destination, bool no_cache, Event*
2086  additional_action)
2087 {
2088  DPRINTF(UFSHostDevice, "Read start: 0x%8x; Size: %d, data[0]: 0x%8x\n",
2089  start, size, (reinterpret_cast<uint32_t *>(destination))[0]);
2090 
2092  if (lastTransfer) {
2093  ++readPendingNum;
2094  readDoneEvent.push_back(
2095  EventFunctionWrapper([this]{ readDone(); },
2096  name()));
2097  assert(!readDoneEvent.back().scheduled());
2098  dmaPort.dmaAction(MemCmd::WriteReq, start, size,
2099  &readDoneEvent.back(), destination, 0);
2100  //yes, a writereq at a read device function is correct.
2101 
2102  } else {
2103  if (additional_action != NULL)
2104  assert(!additional_action->scheduled());
2105 
2106  dmaPort.dmaAction(MemCmd::WriteReq, start, size,
2107  additional_action, destination, 0);
2108 
2109  }
2110 
2111 }
2112 
2118 void
2119 UFSHostDevice::manageReadTransfer(uint32_t size, uint32_t LUN, uint64_t
2120  offset, uint32_t sg_table_length,
2121  struct UFSHCDSGEntry* sglist)
2122 {
2123  uint32_t size_accum = 0;
2124 
2125  DPRINTF(UFSHostDevice, "Data READ size: %d\n", size);
2126 
2131  for (uint32_t count = 0; count < sg_table_length; count++) {
2132  struct transferInfo new_transfer;
2133  new_transfer.offset = sglist[count].upperAddr;
2134  new_transfer.offset = (new_transfer.offset << 32) |
2135  (sglist[count].baseAddr & 0xFFFFFFFF);
2136  new_transfer.filePointer = offset + size_accum;
2137  new_transfer.size = (sglist[count].size + 1);
2138  new_transfer.lunID = LUN;
2139 
2140  DPRINTF(UFSHostDevice, "Data READ start: 0x%8x; size: %d\n",
2141  new_transfer.offset, new_transfer.size);
2142 
2143  UFSDevice[LUN]->SSDReadInfo.push_back(new_transfer);
2144  UFSDevice[LUN]->SSDReadInfo.back().buffer.resize(sglist[count].size
2145  + 1);
2146 
2152  UFSDevice[LUN]->readFlash(&UFSDevice[LUN]->
2153  SSDReadInfo.back().buffer[0],
2154  offset + size_accum,
2155  sglist[count].size + 1);
2156 
2157  size_accum += (sglist[count].size + 1);
2158 
2159  DPRINTF(UFSHostDevice, "Transfer %d; Remaining: 0x%8x, Accumulated:"
2160  " 0x%8x\n", (count + 1), (size-size_accum), size_accum);
2161 
2163  stats.totalReadSSD += (sglist[count].size + 1);
2164  stats.currentReadSSDQueue = UFSDevice[LUN]->SSDReadInfo.size();
2165  stats.averageReadSSDQueue = UFSDevice[LUN]->SSDReadInfo.size();
2166  }
2167 
2168  UFSDevice[LUN]->SSDReadStart(sg_table_length);
2169 
2172 
2173 }
2174 
2175 
2176 
2183 void
2185 {
2186  totalRead = total_read;
2187  for (uint32_t number_handled = 0; number_handled < SSDReadInfo.size();
2188  number_handled++) {
2193  flashDevice->readMemory(SSDReadInfo.front().filePointer,
2194  SSDReadInfo.front().size, memReadCallback);
2195  }
2196 
2197 }
2198 
2199 
2204 void
2206 {
2207  DPRINTF(UFSHostDevice, "SSD read done at lun %d, Aiming for %d messages,"
2208  " %d so far\n", lunID, totalRead, amountOfReadTransfers);
2209 
2210  if (totalRead == amountOfReadTransfers) {
2211  totalRead = 0;
2212  amountOfReadTransfers = 0;
2213 
2215  setSignal();
2216  signalDone->process();
2217  }
2218 
2219 }
2220 
2225 void
2227 {
2228  ++amountOfReadTransfers;
2229 
2233  setReadSignal();
2234  deviceReadCallback->process();
2235 
2236  //Are we done yet?
2237  SSDReadDone();
2238 }
2239 
2245 void
2247 {
2248  DPRINTF(UFSHostDevice, "Read Callback\n");
2249  uint8_t this_lun = 0;
2250 
2251  //while we haven't found the right lun, keep searching
2252  while ((this_lun < lunAvail) && !UFSDevice[this_lun]->finishedRead())
2253  ++this_lun;
2254 
2255  DPRINTF(UFSHostDevice, "Found LUN %d messages pending for clean: %d\n",
2256  this_lun, SSDReadPending.size());
2257 
2258  if (this_lun < lunAvail) {
2259  //Clear signal.
2260  UFSDevice[this_lun]->clearReadSignal();
2261  SSDReadPending.push_back(UFSDevice[this_lun]->SSDReadInfo.front());
2262  UFSDevice[this_lun]->SSDReadInfo.pop_front();
2263  readGarbageEventQueue.push_back(
2264  EventFunctionWrapper([this]{ readGarbage(); }, name()));
2265 
2266  //make sure the queue is popped a the end of the dma transaction
2267  readDevice(false, SSDReadPending.front().offset,
2268  SSDReadPending.front().size,
2269  &SSDReadPending.front().buffer[0], false,
2270  &readGarbageEventQueue.back());
2271 
2274  }
2275  else
2276  panic("no read finished in tick %d\n", curTick());
2277 }
2278 
2283 void
2285 {
2286  DPRINTF(UFSHostDevice, "Clean read data, %d\n", SSDReadPending.size());
2287  SSDReadPending.pop_front();
2288  readGarbageEventQueue.pop_front();
2289 }
2290 
2295 void
2297 {
2299 
2300  const uint8_t* temp_HCI_mem = reinterpret_cast<const uint8_t*>(&UFSHCIMem);
2301  SERIALIZE_ARRAY(temp_HCI_mem, sizeof(HCIMem));
2302 
2303  uint32_t lun_avail = lunAvail;
2304  SERIALIZE_SCALAR(lun_avail);
2305 }
2306 
2307 
2312 void
2314 {
2316  uint8_t* temp_HCI_mem = reinterpret_cast<uint8_t*>(&UFSHCIMem);
2317  UNSERIALIZE_ARRAY(temp_HCI_mem, sizeof(HCIMem));
2318 
2319  uint32_t lun_avail;
2320  UNSERIALIZE_SCALAR(lun_avail);
2321  assert(lunAvail == lun_avail);
2322 }
2323 
2324 
2329 DrainState
2331 {
2332  if (UFSHCIMem.TRUTRLDBR) {
2333  DPRINTF(UFSHostDevice, "UFSDevice is draining...\n");
2334  return DrainState::Draining;
2335  } else {
2336  DPRINTF(UFSHostDevice, "UFSDevice drained\n");
2337  return DrainState::Drained;
2338  }
2339 }
2340 
2345 void
2347 {
2349  return;
2350 
2351  if (UFSHCIMem.TRUTRLDBR) {
2352  DPRINTF(UFSHostDevice, "UFSDevice is still draining; with %d active"
2353  " doorbells\n", activeDoorbells);
2354  } else {
2355  DPRINTF(UFSHostDevice, "UFSDevice is done draining\n");
2356  signalDrainDone();
2357  }
2358 }
count
Definition: misc.hh:703
Stats::Scalar currentSCSIQueue
Queue lengths.
Definition: ufs_device.hh:499
#define panic(...)
This implements a cprintf based panic() function.
Definition: logging.hh:163
const uint32_t lunAvail
Definition: ufs_device.hh:1003
std::deque< struct transferInfo > SSDReadPending
Information from the Disk, waiting to be pushed to the DMA.
Definition: ufs_device.hh:1113
#define DPRINTF(x,...)
Definition: trace.hh:225
AddrRange RangeSize(Addr start, Addr size)
Definition: addr_range.hh:580
const FlagsType pdf
Print the percent of the total that this entry represents.
Definition: info.hh:51
struct SCSIResumeInfo SCSIInfo
SCSI resume info information structure for SCSI resume.
Definition: ufs_device.hh:1077
void generateInterrupt()
set interrupt and sort out the doorbell register.
Definition: ufs_device.cc:1817
void SSDReadStart(uint32_t total_read)
Start the transactions to (and from) the disk The host will queue all the transactions.
Definition: ufs_device.cc:2184
std::deque< EventFunctionWrapper > writeDoneEvent
Definition: ufs_device.hh:1135
Callback * transferDoneCallback
Callbacks for the logic units.
Definition: ufs_device.hh:1142
int findLsbSet(uint64_t val)
Returns the bit position of the LSB that is set in the input.
Definition: bitfield.hh:220
std::deque< struct taskStart > taskInfo
When a task/transfer is started it needs information about the task/transfer it is about to perform...
Definition: ufs_device.hh:1097
#define fatal(...)
This implements a cprintf based fatal() function.
Definition: logging.hh:171
Generic callback class.
Definition: callback.hh:39
virtual void clearInt(uint32_t num)=0
Clear an interrupt from a device that is connected to the GIC.
AddrRangeList getAddrRanges() const override
Address range functions.
Definition: ufs_device.cc:914
void unserialize(CheckpointIn &cp) override
Unserialize; needed to restore from checkpoints.
Definition: ufs_device.cc:2313
void setValues()
Initialization function.
Definition: ufs_device.cc:887
std::deque< struct transferStart > transferStartInfo
Definition: ufs_device.hh:1098
Tick transactionStart[32]
Helper for latency stats These variables keep track of the latency for every doorbell.
Definition: ufs_device.hh:1054
uint32_t ORInterruptStatus
Operation and runtime registers.
Definition: ufs_device.hh:199
void finalUTP()
final UTP, sends the last acknowledge data structure to the system; prepares the clean up functions...
Definition: ufs_device.cc:1743
Stats::Scalar totalReadSSD
Amount of data read/written.
Definition: ufs_device.hh:504
EventFunctionWrapper UTPEvent
Wait for the moment where we can send the last frame.
Definition: ufs_device.hh:1160
void transferStart()
Transfer Start function.
Definition: ufs_device.cc:1362
std::deque< EventFunctionWrapper > readGarbageEventQueue
Event after a read to clean up the UTP data structures.
Definition: ufs_device.hh:1165
std::deque< EventFunctionWrapper > transferEventQueue
Definition: ufs_device.hh:1173
struct UTPUPIURSP - Response UPIU structure header: UPIU header DW-0 to DW-2 residualTransferCount: R...
Definition: ufs_device.hh:269
void manageWriteTransfer(uint8_t LUN, uint64_t offset, uint32_t sg_table_length, struct UFSHCDSGEntry *sglist)
Disk transfer management functions these set up the queues, and initiated them, leading to the data t...
Definition: ufs_device.cc:1930
void writeFlash(uint8_t *writeaddr, uint64_t offset, uint32_t size)
Write flash.
Definition: ufs_device.cc:704
struct UTPTransferReqDesc - UTRD structure header: UTRD header DW-0 to DW-3 commandDescBaseAddrLo: UC...
Definition: ufs_device.hh:340
void SSDWriteDone()
SSD Write Done; This is the callback function for the memory model.
Definition: ufs_device.cc:2054
UFS command flow state machine digraph CommandFlow{ node [fontsize=10]; IDLE -> transferHandler [ lab...
Definition: ufs_device.hh:168
UFSHostDevice(const UFSHostDeviceParams *p)
Constructor for the UFS Host device.
Definition: ufs_device.cc:717
DiskImage * flashDisk
The objects this model links to.
Definition: ufs_device.hh:676
struct UTPUPIUHeader header
Definition: ufs_device.hh:286
uint32_t ORInterruptEnable
Definition: ufs_device.hh:200
Bitfield< 23, 0 > offset
Definition: types.hh:152
Histogram & init(size_type size)
Set the parameters of this histogram.
Definition: statistics.hh:2641
void regStats() override
register statistics
Definition: ufs_device.cc:773
Stats::Scalar totalWriteDiskTransactions
Definition: ufs_device.hh:507
uint8_t pendingDoorbells
Definition: ufs_device.hh:1027
Definition: cprintf.cc:40
struct UTPUPIUTaskReq - Task request UPIU structure header - UPIU header structure DW0 to DW-2 inputP...
Definition: ufs_device.hh:285
Tick write(PacketPtr pkt) override
UFSHCD write function.
Definition: ufs_device.cc:1065
struct UFSHostDeviceStats stats
RequestHandler stats.
Definition: ufs_device.hh:1123
std::deque< EventFunctionWrapper > taskEventQueue
Multiple tasks transfers can be scheduled at once for the device, the only thing we know for sure abo...
Definition: ufs_device.hh:1172
std::deque< struct transferStart > transferEnd
To finish the transaction one needs information about the original message.
Definition: ufs_device.hh:1086
struct UFSHostDevice::UTPTransferReqDesc::RequestDescHeader header
uint32_t ORHostControllerEnable
Definition: ufs_device.hh:202
Derived & flags(Flags _flags)
Set the flags and marks this stat to print at the end of simulation.
Definition: statistics.hh:333
Tick read(PacketPtr pkt) override
register access functions
Definition: ufs_device.cc:927
void readFlash(uint8_t *readaddr, uint64_t offset, uint32_t size)
Disk access functions.
Definition: ufs_device.cc:690
Stats::Formula simSeconds
Definition: stat_control.cc:61
DrainState
Object drain/handover states.
Definition: drain.hh:71
void statusCheck(uint8_t status, uint8_t *sensecodelist)
Status of SCSI.
Definition: ufs_device.cc:673
This is a base class for UFS devices The UFS interface consists out of one host controller which conn...
void setLE(T v)
Set the value in the data pointer to v as little endian.
int readPendingNum
Track number of DMA transactions in progress.
Definition: ufs_device.hh:1014
const Addr pioAddr
Host controller information.
Definition: ufs_device.hh:998
void readCallback()
Read callback Call back function for the logic units to indicate the completion of a read action...
Definition: ufs_device.cc:2246
std::deque< struct UTPTransferReqDesc * > garbage
garbage queue, ensure clearing of the allocated memory
Definition: ufs_device.hh:1118
DrainState drainState() const
Return the current drain state of an object.
Definition: drain.hh:308
void unserialize(CheckpointIn &cp) override
Unserialize an object.
Bitfield< 5, 0 > status
unsigned getSize() const
Definition: packet.hh:730
void commandHandler()
Command handler function.
Definition: ufs_device.cc:1382
#define UNSERIALIZE_SCALAR(scalar)
Definition: serialize.hh:770
#define inform(...)
Definition: logging.hh:209
const int intNum
Definition: ufs_device.hh:1001
Draining buffers pending serialization/handover.
Tick curTick()
The current simulated tick.
Definition: core.hh:44
uint32_t HCCAP
Specify the host capabilities.
Definition: ufs_device.hh:191
uint8_t activeDoorbells
Statistics helper variables Active doorbells indicates how many doorbells are in teh process of being...
Definition: ufs_device.hh:1026
Callback * memReadCallback
Callbacks between Device and Memory.
Definition: ufs_device.hh:731
Stats::Average averageSCSIQueue
Average Queue lengths.
Definition: ufs_device.hh:516
SCSI reply structure.
Definition: ufs_device.hh:373
uint32_t ORHostControllerStatus
Definition: ufs_device.hh:201
virtual void sendInt(uint32_t num)=0
Post an interrupt from a device that is connected to the GIC.
static const unsigned int UPIUHeaderDataIndWord0
Definition: ufs_device.hh:738
uint64_t Tick
Tick count type.
Definition: types.hh:61
void clearInterrupt()
Interrupt control functions.
Definition: ufs_device.cc:1840
void SSDReadDone()
SSD Read done; Determines if the final callback of the transaction should be made at the end of a rea...
Definition: ufs_device.cc:2205
void serialize(CheckpointOut &cp) const override
Serialize an object.
std::vector< uint8_t > destination
Definition: ufs_device.hh:479
void taskStart()
Task Start function.
Definition: ufs_device.cc:1348
virtual std::streampos write(const uint8_t *data, std::streampos offset)=0
void readDevice(bool lastTransfer, Addr SCSIStart, uint32_t SCSISize, uint8_t *SCSIDestination, bool no_cache, Event *additional_action)
Dma transaction function: read device.
Definition: ufs_device.cc:2084
#define SectorSize
Definition: disk_image.hh:44
void serialize(CheckpointOut &cp) const override
Serialize; needed to make checkpoints.
Definition: ufs_device.cc:2296
virtual void initializeMemory(uint64_t disk_size, uint32_t sector_size)=0
Initialize Memory.
Addr getAddr() const
Definition: packet.hh:720
HCIMem UFSHCIMem
Host controller memory.
Definition: ufs_device.hh:1009
BaseGic * gic
Definition: ufs_device.hh:1002
const uint32_t capacityUpper
Definition: ufs_device.hh:693
Stats::Formula curDoorbell
Number of doorbells rung.
Definition: ufs_device.hh:521
void LUNSignal()
LU callback function to indicate that the action has completed.
Definition: ufs_device.cc:1672
DrainState drain() override
Drain; needed to enable checkpoints.
Definition: ufs_device.cc:2330
const FlagsType none
Nothing extra to print.
Definition: info.hh:43
std::deque< struct writeToDiskBurst > dmaWriteInfo
Information to get a DMA transaction.
Definition: ufs_device.hh:1103
static const unsigned int cachingPage[5]
Definition: ufs_device.hh:750
uint64_t Addr
Address type This will probably be moved somewhere else in the near future.
Definition: types.hh:140
uint32_t TMUTMRLBA
Task control registers.
Definition: ufs_device.hh:227
static const unsigned int UPIUHeaderDataIndWord2
Definition: ufs_device.hh:740
static const unsigned int UPIUHeaderDataIndWord1
Definition: ufs_device.hh:739
void transferHandler(struct UTPTransferReqDesc *request_in, int req_pos, Addr finaladdress, uint32_t finalsize, uint32_t done)
Transfer handler function.
Definition: ufs_device.cc:1432
void SCSIStart()
Transfer SCSI function.
Definition: ufs_device.cc:1475
UFSSCSIDevice(const UFSHostDeviceParams *p, uint32_t lun_id, Callback *transfer_cb, Callback *read_cb)
Constructor and destructor.
Definition: ufs_device.cc:75
void readGarbage()
Read garbage A read from disk data structure can vary in size and is therefor allocated on creation...
Definition: ufs_device.cc:2284
A Packet is used to encapsulate a transfer between two objects in the memory system (e...
Definition: packet.hh:249
uint32_t CMDUICCMDR
Command registers.
Definition: ufs_device.hh:236
void readCallback()
Functions to indicate that the action to the SSD has completed.
Definition: ufs_device.cc:2226
#define SERIALIZE_ARRAY(member, size)
Definition: serialize.hh:805
Callback * signalDone
Callbacks between Host and Device.
Definition: ufs_device.hh:725
EventFunctionWrapper SCSIResumeEvent
The events that control the functionality.
Definition: ufs_device.hh:1155
static const unsigned int UTPTaskREQCOMPL
Definition: ufs_device.hh:1179
RequestPtr dmaAction(Packet::Command cmd, Addr addr, int size, Event *event, uint8_t *data, Tick delay, Request::Flags flag=0)
Definition: dma_device.cc:197
std::deque< EventFunctionWrapper > readDoneEvent
Transfer flow events Basically these events form two queues, one from memory to UFS device (DMA) and ...
Definition: ufs_device.hh:1134
void manageReadTransfer(uint32_t size, uint32_t LUN, uint64_t offset, uint32_t sg_table_length, struct UFSHCDSGEntry *sglist)
Manage read transfer.
Definition: ufs_device.cc:2119
struct UTPTransferReqDesc * destination
Definition: ufs_device.hh:451
#define SERIALIZE_SCALAR(scalar)
Definition: serialize.hh:763
struct UTPUPIUHeader header
Definition: ufs_device.hh:324
bool scheduled() const
Determine if the current event is scheduled.
Definition: eventq.hh:459
Derived & name(const std::string &name)
Set the name and marks this stat to print at the end of simulation.
Definition: statistics.hh:276
static const unsigned int recoveryPage[3]
Definition: ufs_device.hh:749
void makeResponse()
Take a request packet and modify it in place to be suitable for returning as a response to that reque...
Definition: packet.hh:931
virtual const std::string name() const
Definition: sim_object.hh:129
#define UNSERIALIZE_ARRAY(member, size)
Definition: serialize.hh:813
Task start information.
Definition: ufs_device.hh:462
device layer: This is your Logic unit This layer implements the SCSI functionality of the UFS Device ...
Definition: ufs_device.hh:535
struct UTPTransferReqDesc * RequestIn
Definition: ufs_device.hh:475
void readDone()
Read done Started at the end of a transaction after the last read action.
Definition: ufs_device.cc:1789
const Addr pioSize
Definition: ufs_device.hh:999
std::ostream CheckpointOut
Definition: serialize.hh:63
static const unsigned int UTPTransferREQCOMPL
Bits of interest within UFS data packages.
Definition: ufs_device.hh:1178
Stats::Formula averageReadSSDBW
Average bandwidth for reads and writes.
Definition: ufs_device.hh:512
Host Controller Interface This is a set of registers that allow the driver to control the transaction...
Definition: ufs_device.hh:187
void writeDone()
Write done After a DMA write with data intended for the disk, this function is called.
Definition: ufs_device.cc:1989
Different events, and scenarios require different types of information.
Definition: ufs_device.hh:424
Definition: eventq.hh:245
const Tick pioDelay
Definition: ufs_device.hh:1000
virtual std::streampos read(uint8_t *data, std::streampos offset) const =0
struct UPIUMessage message
Definition: ufs_device.hh:381
uint32_t TRUTRLBA
Transfer control registers.
Definition: ufs_device.hh:218
struct SCSIReply request_out_datain
SCSI reply structure, used for direct answering.
Definition: ufs_device.hh:1069
uint32_t taskCommandTrack
Definition: ufs_device.hh:1047
void signalDrainDone() const
Signal that an object is drained.
Definition: drain.hh:289
std::deque< struct transferInfo > SSDWriteinfo
Information from DMA transaction to disk.
Definition: ufs_device.hh:1108
const uint32_t capacityLower
Definition: ufs_device.hh:692
Stats::Histogram transactionLatency
Histogram of latencies.
Definition: ufs_device.hh:526
const uint32_t blkSize
Logic unit dimensions.
Definition: ufs_device.hh:689
T getLE() const
Get the data in the packet byte swapped from little endian to host endian.
Transfer start information.
Definition: ufs_device.hh:450
T betoh(T value)
Definition: byteswap.hh:143
struct UFSHCDSGEntry - UFSHCI PRD Entry baseAddr: Lower 32bit physical address DW-0 upperAddr: Upper ...
Definition: ufs_device.hh:300
static const unsigned int controlPage[3]
These pages are SCSI specific.
Definition: ufs_device.hh:748
Callback * memReadCallback
Definition: ufs_device.hh:1143
Bitfield< 3, 0 > mask
Definition: types.hh:62
Derived & desc(const std::string &_desc)
Set the description and marks this stat to print at the end of simulation.
Definition: statistics.hh:309
void taskHandler(struct UTPUPIUTaskReq *request_in, uint32_t req_pos, Addr finaladdress, uint32_t finalsize)
Task handler function.
Definition: ufs_device.cc:1396
uint8_t length
Definition: inet.hh:329
uint32_t transferTrack
Track the transfer This is allows the driver to "group" certain transfers together by using a tag in ...
Definition: ufs_device.hh:1046
#define warn(...)
Definition: logging.hh:208
std::vector< uint32_t > dataMsg
Definition: ufs_device.hh:327
Disk transfer burst information.
Definition: ufs_device.hh:487
DmaPort dmaPort
Definition: dma_device.hh:168
virtual void regStats()
Callback to set stat parameters.
Definition: group.cc:64
Helper template class to turn a simple class member function into a callback.
Definition: callback.hh:62
std::vector< UFSSCSIDevice * > UFSDevice
logic units connected to the UFS Host device Note again that the "device" as such is represented by o...
Definition: ufs_device.hh:1062
struct LUNInfo lunInfo
Logic unit info; needed for SCSI Info messages and LU identification.
Definition: ufs_device.hh:699
void SSDWriteStart()
SSD write start.
Definition: ufs_device.cc:2035
uint32_t countInt
interrupt verification This keeps track of the number of interrupts generated.
Definition: ufs_device.hh:1036
struct SCSIReply SCSICMDHandle(uint32_t *SCSI_msg)
SCSI command handle function; determines what the command is and returns a reply structure that allow...
Definition: ufs_device.cc:159
const FlagsType nozero
Don&#39;t print if this is zero.
Definition: info.hh:57
Bitfield< 0 > p
Running normally.
const uint8_t UFSSlots
Definition: ufs_device.hh:1004
void requestHandler()
Handler functions.
Definition: ufs_device.cc:1229
const char data[]
void checkDrain()
Checkdrain; needed to enable checkpoints.
Definition: ufs_device.cc:2346
Counter value() const
Return the current value of this stat as its base type.
Definition: statistics.hh:700
void SCSIResume(uint32_t lun_id)
Starts the scsi handling function in the apropriate Logic unit, prepares the right data transfer sche...
Definition: ufs_device.cc:1520
Abstract superclass for simulation objects.
Definition: sim_object.hh:93
void transferDone(Addr responseStartAddr, uint32_t req_pos, struct UTPUPIURSP request_out, uint32_t size, Addr address, uint8_t *destination, bool finished, uint32_t lun_id)
transfer done, the beginning of the final stage of the transfer.
Definition: ufs_device.cc:1704
static const unsigned int UICCommandCOMPL
Definition: ufs_device.hh:1180
void sample(const U &v, int n=1)
Add a value to the distribtion n times.
Definition: statistics.hh:1896
static const unsigned int UICCommandReady
Definition: ufs_device.hh:1181
void writeDevice(Event *additional_action, bool toDisk, Addr start, int size, uint8_t *destination, uint64_t SCSIDiskOffset, uint32_t lun_id)
DMA transfer functions These allow the host to push/pull the data to the memory The provided event in...
Definition: ufs_device.cc:1878

Generated on Fri Jul 3 2020 15:53:01 for gem5 by doxygen 1.8.13