gem5  v20.0.0.0
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages
smmu_v3.cc
Go to the documentation of this file.
1 /*
2  * Copyright (c) 2013, 2018-2019 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 
38 #include "dev/arm/smmu_v3.hh"
39 
40 #include <cstddef>
41 #include <cstring>
42 
43 #include "base/bitfield.hh"
44 #include "base/cast.hh"
45 #include "base/logging.hh"
46 #include "base/trace.hh"
47 #include "base/types.hh"
48 #include "debug/Checkpoint.hh"
49 #include "debug/SMMUv3.hh"
51 #include "mem/packet_access.hh"
52 #include "sim/system.hh"
53 
54 SMMUv3::SMMUv3(SMMUv3Params *params) :
55  ClockedObject(params),
56  system(*params->system),
57  masterId(params->system->getMasterId(this)),
58  masterPort(name() + ".master", *this),
59  masterTableWalkPort(name() + ".master_walker", *this),
60  controlPort(name() + ".control", *this, params->reg_map),
61  tlb(params->tlb_entries, params->tlb_assoc, params->tlb_policy),
62  configCache(params->cfg_entries, params->cfg_assoc, params->cfg_policy),
63  ipaCache(params->ipa_entries, params->ipa_assoc, params->ipa_policy),
64  walkCache({ { params->walk_S1L0, params->walk_S1L1,
65  params->walk_S1L2, params->walk_S1L3,
66  params->walk_S2L0, params->walk_S2L1,
67  params->walk_S2L2, params->walk_S2L3 } },
68  params->walk_assoc, params->walk_policy),
69  tlbEnable(params->tlb_enable),
70  configCacheEnable(params->cfg_enable),
71  ipaCacheEnable(params->ipa_enable),
72  walkCacheEnable(params->walk_enable),
73  tableWalkPortEnable(false),
74  walkCacheNonfinalEnable(params->wc_nonfinal_enable),
75  walkCacheS1Levels(params->wc_s1_levels),
76  walkCacheS2Levels(params->wc_s2_levels),
77  masterPortWidth(params->master_port_width),
78  tlbSem(params->tlb_slots),
79  ifcSmmuSem(1),
80  smmuIfcSem(1),
81  configSem(params->cfg_slots),
82  ipaSem(params->ipa_slots),
83  walkSem(params->walk_slots),
84  masterPortSem(1),
85  transSem(params->xlate_slots),
86  ptwSem(params->ptw_slots),
87  cycleSem(1),
88  tlbLat(params->tlb_lat),
89  ifcSmmuLat(params->ifc_smmu_lat),
90  smmuIfcLat(params->smmu_ifc_lat),
91  configLat(params->cfg_lat),
92  ipaLat(params->ipa_lat),
93  walkLat(params->walk_lat),
94  slaveInterfaces(params->slave_interfaces),
95  commandExecutor(name() + ".cmd_exec", *this),
96  regsMap(params->reg_map),
98 {
100  "Invalid register map size: %#x different than SMMU_REG_SIZE = %#x\n",
102 
103  // Init smmu registers to 0
104  memset(&regs, 0, sizeof(regs));
105 
106  // Setup RO ID registers
107  regs.idr0 = params->smmu_idr0;
108  regs.idr1 = params->smmu_idr1;
109  regs.idr2 = params->smmu_idr2;
110  regs.idr3 = params->smmu_idr3;
111  regs.idr4 = params->smmu_idr4;
112  regs.idr5 = params->smmu_idr5;
113  regs.iidr = params->smmu_iidr;
114  regs.aidr = params->smmu_aidr;
115 
116  // TODO: At the moment it possible to set the ID registers to hold
117  // any possible value. It would be nice to have a sanity check here
118  // at construction time in case some idx registers are programmed to
119  // store an unallowed values or if the are configuration conflicts.
120  warn("SMMUv3 IDx register values unchecked\n");
121 
122  for (auto ifc : slaveInterfaces)
123  ifc->setSMMU(this);
124 }
125 
126 bool
128 {
129  DPRINTF(SMMUv3, "[t] master resp addr=%#x size=%#x\n",
130  pkt->getAddr(), pkt->getSize());
131 
132  // @todo: We need to pay for this and not just zero it out
133  pkt->headerDelay = pkt->payloadDelay = 0;
134 
135  SMMUProcess *proc =
137 
138  runProcessTiming(proc, pkt);
139 
140  return true;
141 }
142 
143 void
145 {
146  assert(!packetsToRetry.empty());
147 
148  while (!packetsToRetry.empty()) {
149  SMMUAction a = packetsToRetry.front();
150 
152 
153  DPRINTF(SMMUv3, "[t] master retr addr=%#x size=%#x\n",
154  a.pkt->getAddr(), a.pkt->getSize());
155 
156  if (!masterPort.sendTimingReq(a.pkt))
157  break;
158 
159  packetsToRetry.pop();
160 
161  /*
162  * ACTION_SEND_REQ_FINAL means that we have just forwarded the packet
163  * on the master interface; this means that we no longer hold on to
164  * that transaction and therefore can accept a new one.
165  * If the slave port was stalled then unstall it (send retry).
166  */
167  if (a.type == ACTION_SEND_REQ_FINAL)
169  }
170 }
171 
172 bool
174 {
175  DPRINTF(SMMUv3, "[t] master HWTW resp addr=%#x size=%#x\n",
176  pkt->getAddr(), pkt->getSize());
177 
178  // @todo: We need to pay for this and not just zero it out
179  pkt->headerDelay = pkt->payloadDelay = 0;
180 
181  SMMUProcess *proc =
183 
184  runProcessTiming(proc, pkt);
185 
186  return true;
187 }
188 
189 void
191 {
192  assert(tableWalkPortEnable);
193  assert(!packetsTableWalkToRetry.empty());
194 
195  while (!packetsTableWalkToRetry.empty()) {
197 
198  assert(a.type==ACTION_SEND_REQ);
199 
200  DPRINTF(SMMUv3, "[t] master HWTW retr addr=%#x size=%#x\n",
201  a.pkt->getAddr(), a.pkt->getSize());
202 
204  break;
205 
207  }
208 }
209 
210 void
212 {
213  for (auto ifc : slaveInterfaces) {
214  ifc->scheduleDeviceRetry();
215  }
216 }
217 
220 {
221  if (system.isAtomicMode()) {
222  return runProcessAtomic(proc, pkt);
223  } else if (system.isTimingMode()) {
224  return runProcessTiming(proc, pkt);
225  } else {
226  panic("Not in timing or atomic mode!");
227  }
228 }
229 
232 {
233  SMMUAction action;
234  Tick delay = 0;
235  bool finished = false;
236 
237  do {
238  action = proc->run(pkt);
239 
240  switch (action.type) {
241  case ACTION_SEND_REQ:
242  // Send an MMU initiated request on the table walk port if it is
243  // enabled. Otherwise, fall through and handle same as the final
244  // ACTION_SEND_REQ_FINAL request.
245  if (tableWalkPortEnable) {
246  delay += masterTableWalkPort.sendAtomic(action.pkt);
247  pkt = action.pkt;
248  break;
249  }
252  delay += masterPort.sendAtomic(action.pkt);
253  pkt = action.pkt;
254  break;
255 
256  case ACTION_SEND_RESP:
258  case ACTION_SLEEP:
259  finished = true;
260  break;
261 
262  case ACTION_DELAY:
263  delay += action.delay;
264  break;
265 
266  case ACTION_TERMINATE:
267  panic("ACTION_TERMINATE in atomic mode\n");
268 
269  default:
270  panic("Unknown action\n");
271  }
272  } while (!finished);
273 
274  action.delay = delay;
275 
276  return action;
277 }
278 
281 {
282  SMMUAction action = proc->run(pkt);
283 
284  switch (action.type) {
285  case ACTION_SEND_REQ:
286  // Send an MMU initiated request on the table walk port if it is
287  // enabled. Otherwise, fall through and handle same as the final
288  // ACTION_SEND_REQ_FINAL request.
289  if (tableWalkPortEnable) {
290  action.pkt->pushSenderState(proc);
291 
292  DPRINTF(SMMUv3, "[t] master HWTW req addr=%#x size=%#x\n",
293  action.pkt->getAddr(), action.pkt->getSize());
294 
295  if (packetsTableWalkToRetry.empty()
298  } else {
299  DPRINTF(SMMUv3, "[t] master HWTW req needs retry,"
300  " qlen=%d\n", packetsTableWalkToRetry.size());
301  packetsTableWalkToRetry.push(action);
302  }
303 
304  break;
305  }
308  action.pkt->pushSenderState(proc);
309 
310  DPRINTF(SMMUv3, "[t] master req addr=%#x size=%#x\n",
311  action.pkt->getAddr(), action.pkt->getSize());
312 
313  if (packetsToRetry.empty() && masterPort.sendTimingReq(action.pkt)) {
315  } else {
316  DPRINTF(SMMUv3, "[t] master req needs retry, qlen=%d\n",
317  packetsToRetry.size());
318  packetsToRetry.push(action);
319  }
320 
321  break;
322 
323  case ACTION_SEND_RESP:
324  // @todo: We need to pay for this and not just zero it out
325  action.pkt->headerDelay = action.pkt->payloadDelay = 0;
326 
327  DPRINTF(SMMUv3, "[t] slave resp addr=%#x size=%#x\n",
328  action.pkt->getAddr(),
329  action.pkt->getSize());
330 
331  assert(action.ifc);
332  action.ifc->schedTimingResp(action.pkt);
333 
334  delete proc;
335  break;
336 
338  // @todo: We need to pay for this and not just zero it out
339  action.pkt->headerDelay = action.pkt->payloadDelay = 0;
340 
341  DPRINTF(SMMUv3, "[t] ATS slave resp addr=%#x size=%#x\n",
342  action.pkt->getAddr(), action.pkt->getSize());
343 
344  assert(action.ifc);
345  action.ifc->schedAtsTimingResp(action.pkt);
346 
347  delete proc;
348  break;
349 
350  case ACTION_DELAY:
351  case ACTION_SLEEP:
352  break;
353 
354  case ACTION_TERMINATE:
355  delete proc;
356  break;
357 
358  default:
359  panic("Unknown action\n");
360  }
361 
362  return action;
363 }
364 
365 void
367 {
368  DPRINTF(SMMUv3, "processCommands()\n");
369 
370  if (system.isAtomicMode()) {
372  (void) a;
373  } else if (system.isTimingMode()) {
374  if (!commandExecutor.isBusy())
376  } else {
377  panic("Not in timing or atomic mode!");
378  }
379 }
380 
381 void
383 {
384  switch (cmd.dw0.type) {
385  case CMD_PRF_CONFIG:
386  DPRINTF(SMMUv3, "CMD_PREFETCH_CONFIG - ignored\n");
387  break;
388 
389  case CMD_PRF_ADDR:
390  DPRINTF(SMMUv3, "CMD_PREFETCH_ADDR - ignored\n");
391  break;
392 
393  case CMD_CFGI_STE: {
394  DPRINTF(SMMUv3, "CMD_CFGI_STE sid=%#x\n", cmd.dw0.sid);
395  configCache.invalidateSID(cmd.dw0.sid);
396 
397  for (auto slave_interface : slaveInterfaces) {
398  slave_interface->microTLB->invalidateSID(cmd.dw0.sid);
399  slave_interface->mainTLB->invalidateSID(cmd.dw0.sid);
400  }
401  break;
402  }
403 
404  case CMD_CFGI_STE_RANGE: {
405  const auto range = cmd.dw1.range;
406  if (range == 31) {
407  // CMD_CFGI_ALL is an alias of CMD_CFGI_STE_RANGE with
408  // range = 31
409  DPRINTF(SMMUv3, "CMD_CFGI_ALL\n");
411 
412  for (auto slave_interface : slaveInterfaces) {
413  slave_interface->microTLB->invalidateAll();
414  slave_interface->mainTLB->invalidateAll();
415  }
416  } else {
417  DPRINTF(SMMUv3, "CMD_CFGI_STE_RANGE\n");
418  const auto start_sid = cmd.dw0.sid & ~((1 << (range + 1)) - 1);
419  const auto end_sid = start_sid + (1 << (range + 1)) - 1;
420  for (auto sid = start_sid; sid <= end_sid; sid++) {
422 
423  for (auto slave_interface : slaveInterfaces) {
424  slave_interface->microTLB->invalidateSID(sid);
425  slave_interface->mainTLB->invalidateSID(sid);
426  }
427  }
428  }
429  break;
430  }
431 
432  case CMD_CFGI_CD: {
433  DPRINTF(SMMUv3, "CMD_CFGI_CD sid=%#x ssid=%#x\n",
434  cmd.dw0.sid, cmd.dw0.ssid);
435  configCache.invalidateSSID(cmd.dw0.sid, cmd.dw0.ssid);
436 
437  for (auto slave_interface : slaveInterfaces) {
438  slave_interface->microTLB->invalidateSSID(
439  cmd.dw0.sid, cmd.dw0.ssid);
440  slave_interface->mainTLB->invalidateSSID(
441  cmd.dw0.sid, cmd.dw0.ssid);
442  }
443  break;
444  }
445 
446  case CMD_CFGI_CD_ALL: {
447  DPRINTF(SMMUv3, "CMD_CFGI_CD_ALL sid=%#x\n", cmd.dw0.sid);
448  configCache.invalidateSID(cmd.dw0.sid);
449 
450  for (auto slave_interface : slaveInterfaces) {
451  slave_interface->microTLB->invalidateSID(cmd.dw0.sid);
452  slave_interface->mainTLB->invalidateSID(cmd.dw0.sid);
453  }
454  break;
455  }
456 
457  case CMD_TLBI_NH_ALL: {
458  DPRINTF(SMMUv3, "CMD_TLBI_NH_ALL vmid=%#x\n", cmd.dw0.vmid);
459  for (auto slave_interface : slaveInterfaces) {
460  slave_interface->microTLB->invalidateVMID(cmd.dw0.vmid);
461  slave_interface->mainTLB->invalidateVMID(cmd.dw0.vmid);
462  }
463  tlb.invalidateVMID(cmd.dw0.vmid);
464  walkCache.invalidateVMID(cmd.dw0.vmid);
465  break;
466  }
467 
468  case CMD_TLBI_NH_ASID: {
469  DPRINTF(SMMUv3, "CMD_TLBI_NH_ASID asid=%#x vmid=%#x\n",
470  cmd.dw0.asid, cmd.dw0.vmid);
471  for (auto slave_interface : slaveInterfaces) {
472  slave_interface->microTLB->invalidateASID(
473  cmd.dw0.asid, cmd.dw0.vmid);
474  slave_interface->mainTLB->invalidateASID(
475  cmd.dw0.asid, cmd.dw0.vmid);
476  }
477  tlb.invalidateASID(cmd.dw0.asid, cmd.dw0.vmid);
478  walkCache.invalidateASID(cmd.dw0.asid, cmd.dw0.vmid);
479  break;
480  }
481 
482  case CMD_TLBI_NH_VAA: {
483  const Addr addr = cmd.addr();
484  DPRINTF(SMMUv3, "CMD_TLBI_NH_VAA va=%#08x vmid=%#x\n",
485  addr, cmd.dw0.vmid);
486  for (auto slave_interface : slaveInterfaces) {
487  slave_interface->microTLB->invalidateVAA(
488  addr, cmd.dw0.vmid);
489  slave_interface->mainTLB->invalidateVAA(
490  addr, cmd.dw0.vmid);
491  }
492  tlb.invalidateVAA(addr, cmd.dw0.vmid);
493  const bool leaf_only = cmd.dw1.leaf ? true : false;
494  walkCache.invalidateVAA(addr, cmd.dw0.vmid, leaf_only);
495  break;
496  }
497 
498  case CMD_TLBI_NH_VA: {
499  const Addr addr = cmd.addr();
500  DPRINTF(SMMUv3, "CMD_TLBI_NH_VA va=%#08x asid=%#x vmid=%#x\n",
501  addr, cmd.dw0.asid, cmd.dw0.vmid);
502  for (auto slave_interface : slaveInterfaces) {
503  slave_interface->microTLB->invalidateVA(
504  addr, cmd.dw0.asid, cmd.dw0.vmid);
505  slave_interface->mainTLB->invalidateVA(
506  addr, cmd.dw0.asid, cmd.dw0.vmid);
507  }
508  tlb.invalidateVA(addr, cmd.dw0.asid, cmd.dw0.vmid);
509  const bool leaf_only = cmd.dw1.leaf ? true : false;
510  walkCache.invalidateVA(addr, cmd.dw0.asid, cmd.dw0.vmid,
511  leaf_only);
512  break;
513  }
514 
515  case CMD_TLBI_S2_IPA: {
516  const Addr addr = cmd.addr();
517  DPRINTF(SMMUv3, "CMD_TLBI_S2_IPA ipa=%#08x vmid=%#x\n",
518  addr, cmd.dw0.vmid);
519  // This does not invalidate TLBs containing
520  // combined Stage1 + Stage2 translations, as per the spec.
521  ipaCache.invalidateIPA(addr, cmd.dw0.vmid);
522 
523  if (!cmd.dw1.leaf)
524  walkCache.invalidateVMID(cmd.dw0.vmid);
525  break;
526  }
527 
528  case CMD_TLBI_S12_VMALL: {
529  DPRINTF(SMMUv3, "CMD_TLBI_S12_VMALL vmid=%#x\n", cmd.dw0.vmid);
530  for (auto slave_interface : slaveInterfaces) {
531  slave_interface->microTLB->invalidateVMID(cmd.dw0.vmid);
532  slave_interface->mainTLB->invalidateVMID(cmd.dw0.vmid);
533  }
534  tlb.invalidateVMID(cmd.dw0.vmid);
535  ipaCache.invalidateVMID(cmd.dw0.vmid);
536  walkCache.invalidateVMID(cmd.dw0.vmid);
537  break;
538  }
539 
540  case CMD_TLBI_NSNH_ALL: {
541  DPRINTF(SMMUv3, "CMD_TLBI_NSNH_ALL\n");
542  for (auto slave_interface : slaveInterfaces) {
543  slave_interface->microTLB->invalidateAll();
544  slave_interface->mainTLB->invalidateAll();
545  }
546  tlb.invalidateAll();
549  break;
550  }
551 
552  case CMD_RESUME:
553  DPRINTF(SMMUv3, "CMD_RESUME\n");
554  panic("resume unimplemented");
555  break;
556 
557  default:
558  warn("Unimplemented command %#x\n", cmd.dw0.type);
559  break;
560  }
561 }
562 
563 const PageTableOps*
564 SMMUv3::getPageTableOps(uint8_t trans_granule)
565 {
566  static V8PageTableOps4k ptOps4k;
567  static V8PageTableOps16k ptOps16k;
568  static V8PageTableOps64k ptOps64k;
569 
570  switch (trans_granule) {
571  case TRANS_GRANULE_4K: return &ptOps4k;
572  case TRANS_GRANULE_16K: return &ptOps16k;
573  case TRANS_GRANULE_64K: return &ptOps64k;
574  default:
575  panic("Unknown translation granule size %d", trans_granule);
576  }
577 }
578 
579 Tick
581 {
582  DPRINTF(SMMUv3, "readControl: addr=%08x size=%d\n",
583  pkt->getAddr(), pkt->getSize());
584 
585  int offset = pkt->getAddr() - regsMap.start();
586  assert(offset >= 0 && offset < SMMU_REG_SIZE);
587 
588  if (inSecureBlock(offset)) {
589  warn("smmu: secure registers (0x%x) are not implemented\n",
590  offset);
591  }
592 
593  auto reg_ptr = regs.data + offset;
594 
595  switch (pkt->getSize()) {
596  case sizeof(uint32_t):
597  pkt->setLE<uint32_t>(*reinterpret_cast<uint32_t *>(reg_ptr));
598  break;
599  case sizeof(uint64_t):
600  pkt->setLE<uint64_t>(*reinterpret_cast<uint64_t *>(reg_ptr));
601  break;
602  default:
603  panic("smmu: unallowed access size: %d bytes\n", pkt->getSize());
604  break;
605  }
606 
607  pkt->makeAtomicResponse();
608 
609  return 0;
610 }
611 
612 Tick
614 {
615  int offset = pkt->getAddr() - regsMap.start();
616  assert(offset >= 0 && offset < SMMU_REG_SIZE);
617 
618  DPRINTF(SMMUv3, "writeControl: addr=%08x size=%d data=%16x\n",
619  pkt->getAddr(), pkt->getSize(),
620  pkt->getSize() == sizeof(uint64_t) ?
621  pkt->getLE<uint64_t>() : pkt->getLE<uint32_t>());
622 
623  switch (offset) {
624  case offsetof(SMMURegs, cr0):
625  assert(pkt->getSize() == sizeof(uint32_t));
626  regs.cr0 = regs.cr0ack = pkt->getLE<uint32_t>();
627  break;
628 
629  case offsetof(SMMURegs, cr1):
630  case offsetof(SMMURegs, cr2):
631  case offsetof(SMMURegs, strtab_base_cfg):
632  case offsetof(SMMURegs, eventq_cons):
633  case offsetof(SMMURegs, eventq_irq_cfg1):
634  case offsetof(SMMURegs, priq_cons):
635  assert(pkt->getSize() == sizeof(uint32_t));
636  *reinterpret_cast<uint32_t *>(regs.data + offset) =
637  pkt->getLE<uint32_t>();
638  break;
639 
640  case offsetof(SMMURegs, cmdq_cons):
641  assert(pkt->getSize() == sizeof(uint32_t));
642  if (regs.cr0 & CR0_CMDQEN_MASK) {
643  warn("CMDQ is enabled: ignoring write to CMDQ_CONS\n");
644  } else {
645  *reinterpret_cast<uint32_t *>(regs.data + offset) =
646  pkt->getLE<uint32_t>();
647  }
648  break;
649 
650  case offsetof(SMMURegs, cmdq_prod):
651  assert(pkt->getSize() == sizeof(uint32_t));
652  *reinterpret_cast<uint32_t *>(regs.data + offset) =
653  pkt->getLE<uint32_t>();
655  break;
656 
657  case offsetof(SMMURegs, strtab_base):
658  case offsetof(SMMURegs, eventq_irq_cfg0):
659  assert(pkt->getSize() == sizeof(uint64_t));
660  *reinterpret_cast<uint64_t *>(regs.data + offset) =
661  pkt->getLE<uint64_t>();
662  break;
663 
664  case offsetof(SMMURegs, cmdq_base):
665  assert(pkt->getSize() == sizeof(uint64_t));
666  if (regs.cr0 & CR0_CMDQEN_MASK) {
667  warn("CMDQ is enabled: ignoring write to CMDQ_BASE\n");
668  } else {
669  *reinterpret_cast<uint64_t *>(regs.data + offset) =
670  pkt->getLE<uint64_t>();
671  regs.cmdq_cons = 0;
672  regs.cmdq_prod = 0;
673  }
674  break;
675 
676  case offsetof(SMMURegs, eventq_base):
677  assert(pkt->getSize() == sizeof(uint64_t));
678  *reinterpret_cast<uint64_t *>(regs.data + offset) =
679  pkt->getLE<uint64_t>();
680  regs.eventq_cons = 0;
681  regs.eventq_prod = 0;
682  break;
683 
684  case offsetof(SMMURegs, priq_base):
685  assert(pkt->getSize() == sizeof(uint64_t));
686  *reinterpret_cast<uint64_t *>(regs.data + offset) =
687  pkt->getLE<uint64_t>();
688  regs.priq_cons = 0;
689  regs.priq_prod = 0;
690  break;
691 
692  default:
693  if (inSecureBlock(offset)) {
694  warn("smmu: secure registers (0x%x) are not implemented\n",
695  offset);
696  } else {
697  warn("smmu: write to read-only/undefined register at 0x%x\n",
698  offset);
699  }
700  }
701 
702  pkt->makeAtomicResponse();
703 
704  return 0;
705 }
706 
707 bool
708 SMMUv3::inSecureBlock(uint32_t offs) const
709 {
710  if (offs >= offsetof(SMMURegs, _secure_regs) && offs < SMMU_SECURE_SZ)
711  return true;
712  else
713  return false;
714 }
715 
716 void
718 {
719  // make sure both sides are connected and have the same block size
720  if (!masterPort.isConnected())
721  fatal("Master port is not connected.\n");
722 
723  // If the second master port is connected for the table walks, enable
724  // the mode to send table walks through this port instead
726  tableWalkPortEnable = true;
727 
728  // notify the master side of our address ranges
729  for (auto ifc : slaveInterfaces) {
730  ifc->sendRange();
731  }
732 
733  if (controlPort.isConnected())
735 }
736 
737 void
739 {
741 
742  using namespace Stats;
743 
744  for (size_t i = 0; i < slaveInterfaces.size(); i++) {
745  slaveInterfaces[i]->microTLB->regStats(
746  csprintf("%s.utlb%d", name(), i));
747  slaveInterfaces[i]->mainTLB->regStats(
748  csprintf("%s.maintlb%d", name(), i));
749  }
750 
751  tlb.regStats(name() + ".tlb");
752  configCache.regStats(name() + ".cfg");
753  ipaCache.regStats(name() + ".ipa");
754  walkCache.regStats(name() + ".walk");
755 
757  .name(name() + ".steL1Fetches")
758  .desc("STE L1 fetches")
759  .flags(pdf);
760 
761  steFetches
762  .name(name() + ".steFetches")
763  .desc("STE fetches")
764  .flags(pdf);
765 
767  .name(name() + ".cdL1Fetches")
768  .desc("CD L1 fetches")
769  .flags(pdf);
770 
771  cdFetches
772  .name(name() + ".cdFetches")
773  .desc("CD fetches")
774  .flags(pdf);
775 
777  .init(0, 2000000, 2000)
778  .name(name() + ".translationTimeDist")
779  .desc("Time to translate address")
780  .flags(pdf);
781 
783  .init(0, 2000000, 2000)
784  .name(name() + ".ptwTimeDist")
785  .desc("Time to walk page tables")
786  .flags(pdf);
787 }
788 
791 {
792  // Wait until the Command Executor is not busy
793  if (commandExecutor.isBusy()) {
794  return DrainState::Draining;
795  }
796  return DrainState::Drained;
797 }
798 
799 void
801 {
802  DPRINTF(Checkpoint, "Serializing SMMUv3\n");
803 
804  SERIALIZE_ARRAY(regs.data, sizeof(regs.data) / sizeof(regs.data[0]));
805 }
806 
807 void
809 {
810  DPRINTF(Checkpoint, "Unserializing SMMUv3\n");
811 
812  UNSERIALIZE_ARRAY(regs.data, sizeof(regs.data) / sizeof(regs.data[0]));
813 }
814 
815 Port&
816 SMMUv3::getPort(const std::string &name, PortID id)
817 {
818  if (name == "master") {
819  return masterPort;
820  } else if (name == "master_walker") {
821  return masterTableWalkPort;
822  } else if (name == "control") {
823  return controlPort;
824  } else {
825  return ClockedObject::getPort(name, id);
826  }
827 }
828 
829 SMMUv3*
830 SMMUv3Params::create()
831 {
832  return new SMMUv3(this);
833 }
SMMUSemaphore ifcSmmuSem
Definition: smmu_v3.hh:114
#define panic(...)
This implements a cprintf based panic() function.
Definition: logging.hh:163
const Cycles tlbLat
Definition: smmu_v3.hh:126
#define DPRINTF(x,...)
Definition: trace.hh:225
Bitfield< 63, 48 > asid
void unserialize(CheckpointIn &cp) override
Unserialize an object.
Definition: smmu_v3.cc:808
std::queue< SMMUAction > packetsTableWalkToRetry
Definition: smmu_v3.hh:151
const FlagsType pdf
Print the percent of the total that this entry represents.
Definition: info.hh:51
IPACache ipaCache
Definition: smmu_v3.hh:99
Ports are used to interface objects to each other.
Definition: port.hh:56
void invalidateVAA(Addr va, uint16_t vmid, const bool leaf_only)
bool inSecureBlock(uint32_t offs) const
Definition: smmu_v3.cc:708
void sendRangeChange() const
Called by the owner to send a range change.
Definition: port.hh:282
SMMUSemaphore tlbSem
Definition: smmu_v3.hh:113
SMMUAction runProcess(SMMUProcess *proc, PacketPtr pkt)
Definition: smmu_v3.cc:219
Stats::Scalar steFetches
Definition: smmu_v3.hh:135
Stats::Distribution translationTimeDist
Definition: smmu_v3.hh:138
#define fatal(...)
This implements a cprintf based fatal() function.
Definition: logging.hh:171
std::queue< SMMUAction > packetsToRetry
Definition: smmu_v3.hh:150
const std::string & name()
Definition: trace.cc:50
SMMUAction run(PacketPtr pkt)
Bitfield< 7 > i
void invalidateVMID(uint16_t vmid)
void invalidateAll()
SMMUSemaphore ptwSem
Definition: smmu_v3.hh:122
uint32_t idr3
SMMUSemaphore ipaSem
Definition: smmu_v3.hh:117
uint32_t iidr
virtual void init() override
init() is called after all C++ SimObjects have been created and all ports are connected.
Definition: smmu_v3.cc:717
void serialize(CheckpointOut &cp) const override
Serialize an object.
Definition: smmu_v3.cc:800
uint32_t aidr
void schedAtsTimingResp(PacketPtr pkt)
const Cycles smmuIfcLat
Definition: smmu_v3.hh:128
const Cycles ifcSmmuLat
Definition: smmu_v3.hh:127
Tick readControl(PacketPtr pkt)
Definition: smmu_v3.cc:580
Tick writeControl(PacketPtr pkt)
Definition: smmu_v3.cc:613
Bitfield< 8 > a
void invalidateVMID(uint16_t vmid)
uint64_t addr() const
uint32_t idr4
ip6_addr_t addr
Definition: inet.hh:330
void scheduleSlaveRetries()
Definition: smmu_v3.cc:211
const unsigned walkCacheS1Levels
Definition: smmu_v3.hh:109
void invalidateSSID(uint32_t sid, uint32_t ssid)
Bitfield< 23, 0 > offset
Definition: types.hh:152
const unsigned masterPortWidth
Definition: smmu_v3.hh:111
EventWrapper< SMMUv3, &SMMUv3::processCommands > processCommandsEvent
Definition: smmu_v3.hh:161
SMMUv3SlaveInterface * ifc
Definition: smmu_v3_proc.hh:70
bool sendTimingReq(PacketPtr pkt)
Attempt to send a timing request to the slave port by calling its corresponding receive function...
Definition: port.hh:441
bool isConnected() const
Is this port currently connected to a peer?
Definition: port.hh:124
Stats::Scalar steL1Fetches
Definition: smmu_v3.hh:134
Definition: cprintf.cc:40
void invalidateVA(Addr va, uint16_t asid, uint16_t vmid, const bool leaf_only)
void processCommand(const SMMUCommand &cmd)
Definition: smmu_v3.cc:382
Derived & flags(Flags _flags)
Set the flags and marks this stat to print at the end of simulation.
Definition: statistics.hh:333
const bool tlbEnable
Definition: smmu_v3.hh:102
Bitfield< 31, 12 > ssid
const Params * params() const
Stats::Scalar cdFetches
Definition: smmu_v3.hh:137
DrainState
Object drain/handover states.
Definition: drain.hh:71
virtual Port & getPort(const std::string &name, PortID id=InvalidPortID) override
Get a port with a given name and index.
Definition: smmu_v3.cc:816
std::vector< SMMUv3SlaveInterface * > slaveInterfaces
Definition: smmu_v3.hh:141
PacketPtr pkt
Definition: smmu_v3_proc.hh:69
void setLE(T v)
Set the value in the data pointer to v as little endian.
Stats::Scalar cdL1Fetches
Definition: smmu_v3.hh:136
void masterRecvReqRetry()
Definition: smmu_v3.cc:144
SMMUControlPort controlPort
Definition: smmu_v3.hh:95
void invalidateAll()
unsigned getSize() const
Definition: packet.hh:730
Bitfield< 63, 32 > sid
Draining buffers pending serialization/handover.
bool isAtomicMode() const
Is the system in atomic mode?
Definition: system.hh:131
virtual Port & getPort(const std::string &if_name, PortID idx=InvalidPortID)
Get a port with a given name and index.
Definition: sim_object.cc:123
std::string csprintf(const char *format, const Args &...args)
Definition: cprintf.hh:158
uint32_t priq_prod
void masterTableWalkRecvReqRetry()
Definition: smmu_v3.cc:190
uint32_t headerDelay
The extra delay from seeing the packet until the header is transmitted.
Definition: packet.hh:360
#define M5_FALLTHROUGH
Definition: compiler.hh:84
SMMUCommandExecProcess commandExecutor
Definition: smmu_v3.hh:143
void invalidateAll()
void makeAtomicResponse()
Definition: packet.hh:943
uint32_t eventq_prod
SMMUv3(SMMUv3Params *p)
Definition: smmu_v3.cc:54
SMMUSemaphore transSem
Definition: smmu_v3.hh:121
const Cycles ipaLat
Definition: smmu_v3.hh:130
uint32_t cr0ack
uint64_t Tick
Tick count type.
Definition: types.hh:61
Stats::Distribution ptwTimeDist
Definition: smmu_v3.hh:139
The ClockedObject class extends the SimObject with a clock and accessor functions to relate ticks to ...
uint32_t idr5
SMMUActionType type
Definition: smmu_v3_proc.hh:68
uint32_t cmdq_cons
SMMUSemaphore smmuIfcSem
Definition: smmu_v3.hh:115
uint32_t idr0
bool masterRecvTimingResp(PacketPtr pkt)
Definition: smmu_v3.cc:127
void invalidateIPA(Addr ipa, uint16_t vmid)
void invalidateSID(uint32_t sid)
const Cycles walkLat
Definition: smmu_v3.hh:131
Addr getAddr() const
Definition: packet.hh:720
#define fatal_if(cond,...)
Conditional fatal macro that checks the supplied condition and only causes a fatal error if the condi...
Definition: logging.hh:199
void invalidateASID(uint16_t asid, uint16_t vmid)
void schedule(Event &event, Tick when)
Definition: eventq.hh:934
Distribution & init(Counter min, Counter max, Counter bkt)
Set the parameters of this distribution.
Definition: statistics.hh:2606
Bitfield< 59, 56 > tlb
WalkCache walkCache
Definition: smmu_v3.hh:100
void schedTimingResp(PacketPtr pkt)
Tick nextCycle() const
Based on the clock of the object, determine the start tick of the first cycle that is at least one cy...
DrainState drain() override
Notify an object that it needs to drain its state.
Definition: smmu_v3.cc:790
Defines global host-dependent types: Counter, Tick, and (indirectly) {int,uint}{8,16,32,64}_t.
uint64_t Addr
Address type This will probably be moved somewhere else in the near future.
Definition: types.hh:140
SMMUSemaphore masterPortSem
Definition: smmu_v3.hh:119
uint32_t payloadDelay
The extra pipelining delay from seeing the packet until the end of payload is transmitted by the comp...
Definition: packet.hh:378
T safe_cast(U ptr)
Definition: cast.hh:59
A Packet is used to encapsulate a transfer between two objects in the memory system (e...
Definition: packet.hh:249
#define SERIALIZE_ARRAY(member, size)
Definition: serialize.hh:805
SMMUMasterPort masterPort
Definition: smmu_v3.hh:93
Bitfield< 15 > system
Definition: misc.hh:997
ARMArchTLB tlb
Definition: smmu_v3.hh:97
SMMUSemaphore configSem
Definition: smmu_v3.hh:116
This is an implementation of the SMMUv3 architecture.
Derived & name(const std::string &name)
Set the name and marks this stat to print at the end of simulation.
Definition: statistics.hh:276
uint32_t eventq_cons
virtual const std::string name() const
Definition: sim_object.hh:129
#define UNSERIALIZE_ARRAY(member, size)
Definition: serialize.hh:813
void invalidateVAA(Addr va, uint16_t vmid)
uint32_t cr0
bool masterTableWalkRecvTimingResp(PacketPtr pkt)
Definition: smmu_v3.cc:173
SMMUAction runProcessTiming(SMMUProcess *proc, PacketPtr pkt)
Definition: smmu_v3.cc:280
SMMURegs regs
Definition: smmu_v3.hh:146
std::ostream CheckpointOut
Definition: serialize.hh:63
SMMUSemaphore walkSem
Definition: smmu_v3.hh:118
uint8_t data[SMMU_REG_SIZE]
const System & system
Definition: smmu_v3.hh:90
uint32_t idr1
void invalidateAll()
const PageTableOps * getPageTableOps(uint8_t trans_granule)
Definition: smmu_v3.cc:564
const bool walkCacheNonfinalEnable
Definition: smmu_v3.hh:108
void invalidateVMID(uint16_t vmid)
SMMUSemaphore cycleSem
Definition: smmu_v3.hh:123
void processCommands()
Definition: smmu_v3.cc:366
const bool walkCacheEnable
Definition: smmu_v3.hh:105
T getLE() const
Get the data in the packet byte swapped from little endian to host endian.
void pushSenderState(SenderState *sender_state)
Push a new sender state to the packet and make the current sender state the predecessor of the new on...
Definition: packet.cc:316
Addr start() const
Get the start address of the range.
Definition: addr_range.hh:293
Addr size() const
Get the size of the address range.
Definition: addr_range.hh:280
SenderState * popSenderState()
Pop the top of the state stack and return a pointer to it.
Definition: packet.cc:324
Derived & desc(const std::string &_desc)
Set the description and marks this stat to print at the end of simulation.
Definition: statistics.hh:309
int16_t PortID
Port index/ID type, and a symbolic name for an invalid port id.
Definition: types.hh:235
uint32_t idr2
const bool ipaCacheEnable
Definition: smmu_v3.hh:104
Bitfield< 4, 0 > range
#define warn(...)
Definition: logging.hh:208
const unsigned walkCacheS2Levels
Definition: smmu_v3.hh:110
virtual void regStats()
Callback to set stat parameters.
Definition: group.cc:64
uint32_t priq_cons
ConfigCache configCache
Definition: smmu_v3.hh:98
bool isTimingMode() const
Is the system in timing mode?
Definition: system.hh:142
Bitfield< 47, 32 > vmid
void invalidateVA(Addr va, uint16_t asid, uint16_t vmid)
uint32_t cmdq_prod
Tick sendAtomic(PacketPtr pkt)
Send an atomic request packet, where the data is moved and the state is updated in zero time...
Definition: port.hh:423
void regStats(const std::string &name) override
Running normally.
const bool configCacheEnable
Definition: smmu_v3.hh:103
SMMUAction runProcessAtomic(SMMUProcess *proc, PacketPtr pkt)
Definition: smmu_v3.cc:231
const AddrRange regsMap
Definition: smmu_v3.hh:145
virtual void regStats(const std::string &name)
virtual void regStats() override
Callback to set stat parameters.
Definition: smmu_v3.cc:738
bool tableWalkPortEnable
Definition: smmu_v3.hh:106
const Cycles configLat
Definition: smmu_v3.hh:129
void invalidateASID(uint16_t asid, uint16_t vmid)
SMMUMasterTableWalkPort masterTableWalkPort
Definition: smmu_v3.hh:94

Generated on Thu May 28 2020 16:21:32 for gem5 by doxygen 1.8.13