/* * Copyright (c) 2021 Advanced Micro Devices, Inc. * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions are met: * * 1. Redistributions of source code must retain the above copyright notice, * this list of conditions and the following disclaimer. * * 2. Redistributions in binary form must reproduce the above copyright notice, * this list of conditions and the following disclaimer in the documentation * and/or other materials provided with the distribution. * * 3. Neither the name of the copyright holder nor the names of its * contributors may be used to endorse or promote products derived from this * software without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE * POSSIBILITY OF SUCH DAMAGE. * */ #include "dev/amdgpu/pm4_packet_processor.hh" #include "debug/PM4PacketProcessor.hh" #include "dev/amdgpu/amdgpu_device.hh" #include "dev/amdgpu/hwreg_defines.hh" #include "dev/amdgpu/interrupt_handler.hh" #include "dev/amdgpu/pm4_mmio.hh" #include "dev/amdgpu/sdma_engine.hh" #include "dev/hsa/hw_scheduler.hh" #include "enums/GfxVersion.hh" #include "gpu-compute/gpu_command_processor.hh" #include "gpu-compute/shader.hh" #include "mem/packet.hh" #include "mem/packet_access.hh" namespace gem5 { PM4PacketProcessor::PM4PacketProcessor(const PM4PacketProcessorParams &p) : DmaVirtDevice(p) { memset(&kiq, 0, sizeof(QueueDesc)); memset(&pq, 0, sizeof(QueueDesc)); } /** * AMDGPUDevice will perform DMA operations on VAs, and because * page faults are not currently supported for Vega 10, we * must be able to find the pages mapped for the process. */ TranslationGenPtr PM4PacketProcessor::translate(Addr vaddr, Addr size) { if (gpuDevice->getVM().inAGP(vaddr)) { // Use AGP translation gen return TranslationGenPtr( new AMDGPUVM::AGPTranslationGen(&gpuDevice->getVM(), vaddr, size)); } // Assume GART otherwise as this is the only other translation aperture // available to the PM4 packet processor. return TranslationGenPtr( new AMDGPUVM::GARTTranslationGen(&gpuDevice->getVM(), vaddr, size)); } AddrRangeList PM4PacketProcessor::getAddrRanges() const { AddrRangeList ranges; return ranges; } void PM4PacketProcessor::setGPUDevice(AMDGPUDevice *gpu_device) { gpuDevice = gpu_device; } Addr PM4PacketProcessor::getGARTAddr(Addr addr) const { if (!gpuDevice->getVM().inAGP(addr)) { Addr low_bits = bits(addr, 11, 0); addr = (((addr >> 12) << 3) << 12) | low_bits; } return addr; } PM4Queue * PM4PacketProcessor::getQueue(Addr offset, bool gfx) { auto result = queuesMap.find(offset); if (result == queuesMap.end()) { if (gfx) mapPq(offset); else mapKiq(offset); return queuesMap[offset]; } return result->second; } void PM4PacketProcessor::mapKiq(Addr offset) { DPRINTF(PM4PacketProcessor, "Mapping KIQ\n"); newQueue((QueueDesc *)&kiq, offset); } void PM4PacketProcessor::mapPq(Addr offset) { DPRINTF(PM4PacketProcessor, "Mapping PQ\n"); newQueue((QueueDesc *)&pq, offset); } void PM4PacketProcessor::newQueue(QueueDesc *mqd, Addr offset, PM4MapQueues *pkt, int id) { if (id == -1) id = queues.size(); /* 256 bytes aligned address */ mqd->base <<= 8; PM4Queue *q = new PM4Queue(id, mqd, offset, pkt); queuesMap[offset] = q; queues[id] = q; /* we are assumming only compute queues can be map from MQDs */ QueueType qt; qt = mqd->aql ? QueueType::ComputeAQL : QueueType::Compute; gpuDevice->setDoorbellType(offset, qt); DPRINTF(PM4PacketProcessor, "New PM4 queue %d, base: %p offset: %p\n", id, q->base(), q->offset()); } void PM4PacketProcessor::process(PM4Queue *q, Addr wptrOffset) { q->wptr(wptrOffset * sizeof(uint32_t)); if (!q->processing()) { q->processing(true); decodeNext(q); } } void PM4PacketProcessor::decodeNext(PM4Queue *q) { DPRINTF(PM4PacketProcessor, "PM4 decode queue %d rptr %p, wptr %p\n", q->id(), q->rptr(), q->wptr()); if (q->rptr() < q->wptr()) { /* Additional braces here are needed due to a clang compilation bug falsely throwing a "suggest braces around initialization of subject" error. More info on this bug is available here: https://stackoverflow.com/questions/31555584 */ PM4Header h{{{0, 0, 0, 0, 0, 0}}}; auto cb = new DmaVirtCallback( [ = ] (PM4Header header) { decodeHeader(q, header); }, h); dmaReadVirt(getGARTAddr(q->rptr()), sizeof(uint32_t), cb, &cb->dmaBuffer); } else { q->processing(false); if (q->ib()) { q->ib(false); decodeNext(q); } } } void PM4PacketProcessor::decodeHeader(PM4Queue *q, PM4Header header) { DPRINTF(PM4PacketProcessor, "PM4 packet %p\n", header.opcode); q->incRptr(sizeof(PM4Header)); DmaVirtCallback *cb = nullptr; void *dmaBuffer = nullptr; switch(header.opcode) { case IT_NOP: { DPRINTF(PM4PacketProcessor, "PM4 nop, count %p\n", header.count); DPRINTF(PM4PacketProcessor, "rptr %p wptr %p\n", q->rptr(), q->wptr()); if (header.count != 0x3fff) { q->incRptr((header.count + 1) * sizeof(uint32_t)); } decodeNext(q); } break; case IT_WRITE_DATA: { dmaBuffer = new PM4WriteData(); cb = new DmaVirtCallback( [ = ] (const uint64_t &) { writeData(q, (PM4WriteData *)dmaBuffer); }); dmaReadVirt(getGARTAddr(q->rptr()), sizeof(PM4WriteData), cb, dmaBuffer); } break; case IT_MAP_QUEUES: { dmaBuffer = new PM4MapQueues(); cb = new DmaVirtCallback( [ = ] (const uint64_t &) { mapQueues(q, (PM4MapQueues *)dmaBuffer); }); dmaReadVirt(getGARTAddr(q->rptr()), sizeof(PM4MapQueues), cb, dmaBuffer); } break; case IT_RELEASE_MEM: { dmaBuffer = new PM4ReleaseMem(); cb = new DmaVirtCallback( [ = ] (const uint64_t &) { releaseMem(q, (PM4ReleaseMem *)dmaBuffer); }); dmaReadVirt(getGARTAddr(q->rptr()), sizeof(PM4ReleaseMem), cb, dmaBuffer); } break; case IT_INDIRECT_BUFFER: { dmaBuffer = new PM4IndirectBuf(); cb = new DmaVirtCallback( [ = ] (const uint64_t &) { indirectBuffer(q, (PM4IndirectBuf *)dmaBuffer); }); dmaReadVirt(getGARTAddr(q->rptr()), sizeof(PM4IndirectBuf), cb, dmaBuffer); } break; case IT_SWITCH_BUFFER: { dmaBuffer = new PM4SwitchBuf(); cb = new DmaVirtCallback( [ = ] (const uint64_t &) { switchBuffer(q, (PM4SwitchBuf *)dmaBuffer); }); dmaReadVirt(getGARTAddr(q->rptr()), sizeof(PM4SwitchBuf), cb, dmaBuffer); } break; case IT_SET_UCONFIG_REG: { dmaBuffer = new PM4SetUconfigReg(); cb = new DmaVirtCallback( [ = ] (const uint64_t &) { setUconfigReg(q, (PM4SetUconfigReg *)dmaBuffer); }); dmaReadVirt(getGARTAddr(q->rptr()), sizeof(PM4SetUconfigReg), cb, dmaBuffer); } break; case IT_WAIT_REG_MEM: { dmaBuffer = new PM4WaitRegMem(); cb = new DmaVirtCallback( [ = ] (const uint64_t &) { waitRegMem(q, (PM4WaitRegMem *)dmaBuffer); }); dmaReadVirt(getGARTAddr(q->rptr()), sizeof(PM4WaitRegMem), cb, dmaBuffer); } break; case IT_MAP_PROCESS: { dmaBuffer = new PM4MapProcess(); cb = new DmaVirtCallback( [ = ] (const uint64_t &) { mapProcess(q, (PM4MapProcess *)dmaBuffer); }); dmaReadVirt(getGARTAddr(q->rptr()), sizeof(PM4MapProcess), cb, dmaBuffer); } break; case IT_UNMAP_QUEUES: { dmaBuffer = new PM4UnmapQueues(); cb = new DmaVirtCallback( [ = ] (const uint64_t &) { unmapQueues(q, (PM4UnmapQueues *)dmaBuffer); }); dmaReadVirt(getGARTAddr(q->rptr()), sizeof(PM4UnmapQueues), cb, dmaBuffer); } break; case IT_RUN_LIST: { dmaBuffer = new PM4RunList(); cb = new DmaVirtCallback( [ = ] (const uint64_t &) { runList(q, (PM4RunList *)dmaBuffer); }); dmaReadVirt(getGARTAddr(q->rptr()), sizeof(PM4RunList), cb, dmaBuffer); } break; case IT_QUERY_STATUS: { dmaBuffer = new PM4QueryStatus(); cb = new DmaVirtCallback( [ = ] (const uint64_t &) { queryStatus(q, (PM4QueryStatus *)dmaBuffer); }); dmaReadVirt(getGARTAddr(q->rptr()), sizeof(PM4QueryStatus), cb, dmaBuffer); } break; case IT_INVALIDATE_TLBS: { DPRINTF(PM4PacketProcessor, "Functionaly invalidating all TLBs\n"); gpuDevice->getVM().invalidateTLBs(); q->incRptr((header.count + 1) * sizeof(uint32_t)); decodeNext(q); } break; default: { warn("PM4 packet opcode 0x%x not supported.\n", header.opcode); DPRINTF(PM4PacketProcessor, "PM4 packet opcode 0x%x not supported.\n", header.opcode); q->incRptr((header.count + 1) * sizeof(uint32_t)); decodeNext(q); } break; } } void PM4PacketProcessor::writeData(PM4Queue *q, PM4WriteData *pkt) { q->incRptr(sizeof(PM4WriteData)); Addr addr = getGARTAddr(pkt->destAddr); DPRINTF(PM4PacketProcessor, "PM4 write addr: %p data: %p.\n", addr, pkt->data); auto cb = new DmaVirtCallback( [ = ](const uint32_t &) { writeDataDone(q, pkt, addr); }); //TODO: the specs indicate that pkt->data holds the number of dword that //need to be written. dmaWriteVirt(addr, sizeof(uint32_t), cb, &pkt->data); if (!pkt->writeConfirm) decodeNext(q); } void PM4PacketProcessor::writeDataDone(PM4Queue *q, PM4WriteData *pkt, Addr addr) { DPRINTF(PM4PacketProcessor, "PM4 write completed to %p, %p.\n", addr, pkt->data); if (pkt->writeConfirm) decodeNext(q); delete pkt; } void PM4PacketProcessor::mapQueues(PM4Queue *q, PM4MapQueues *pkt) { q->incRptr(sizeof(PM4MapQueues)); DPRINTF(PM4PacketProcessor, "MAPQueues queueSel: %d, vmid: %d, me: %d, " "pipe: %d, queueSlot: %d, queueType: %d, allocFormat: %d, " "engineSel: %d, numQueues: %d, checkDisable: %d, doorbellOffset:" " %d, mqdAddr: %lx, wptrAddr: %lx\n", pkt->queueSel, pkt->vmid, pkt->me, pkt->pipe, pkt->queueSlot, pkt->queueType, pkt->allocFormat, pkt->engineSel, pkt->numQueues, pkt->checkDisable, pkt->doorbellOffset, pkt->mqdAddr, pkt->wptrAddr); // Partially reading the mqd with an offset of 96 dwords if (pkt->engineSel == 0 || pkt->engineSel == 1 || pkt->engineSel == 4) { Addr addr = getGARTAddr(pkt->mqdAddr + 96 * sizeof(uint32_t)); DPRINTF(PM4PacketProcessor, "Mapping mqd from %p %p (vmid %d - last vmid %d).\n", addr, pkt->mqdAddr, pkt->vmid, gpuDevice->lastVMID()); gpuDevice->mapDoorbellToVMID(pkt->doorbellOffset, gpuDevice->lastVMID()); QueueDesc *mqd = new QueueDesc(); memset(mqd, 0, sizeof(QueueDesc)); auto cb = new DmaVirtCallback( [ = ] (const uint32_t &) { processMQD(pkt, q, addr, mqd, gpuDevice->lastVMID()); }); dmaReadVirt(addr, sizeof(QueueDesc), cb, mqd); } else if (pkt->engineSel == 2 || pkt->engineSel == 3) { SDMAQueueDesc *sdmaMQD = new SDMAQueueDesc(); memset(sdmaMQD, 0, sizeof(SDMAQueueDesc)); // For SDMA we read the full MQD, so there is no offset calculation. Addr addr = getGARTAddr(pkt->mqdAddr); auto cb = new DmaVirtCallback( [ = ] (const uint32_t &) { processSDMAMQD(pkt, q, addr, sdmaMQD, gpuDevice->lastVMID()); }); dmaReadVirt(addr, sizeof(SDMAQueueDesc), cb, sdmaMQD); } else { panic("Unknown engine for MQD: %d\n", pkt->engineSel); } decodeNext(q); } void PM4PacketProcessor::processMQD(PM4MapQueues *pkt, PM4Queue *q, Addr addr, QueueDesc *mqd, uint16_t vmid) { DPRINTF(PM4PacketProcessor, "MQDbase: %lx, active: %d, vmid: %d, base: " "%lx, rptr: %x aqlPtr: %lx\n", mqd->mqdBase, mqd->hqd_active, mqd->hqd_vmid, mqd->base, mqd->rptr, mqd->aqlRptr); Addr offset = mqd->doorbell & 0x1ffffffc; newQueue(mqd, offset, pkt); PM4Queue *new_q = queuesMap[offset]; gpuDevice->insertQId(vmid, new_q->id()); if (mqd->aql) { // The queue size is encoded in the cp_hqd_pq_control field in the // kernel driver in the 6 lowest bits as log2(queue_size / 4) - 1 // number of dwords. // // https://github.com/RadeonOpenCompute/ROCK-Kernel-Driver/blob/ // roc-4.3.x/drivers/gpu/drm/amd/amdgpu/gfx_v9_0.c#L3561 // // Queue size is then 2^(cp_hqd_pq_control[5:0] + 1) dword. Multiply // by 4 to get the number of bytes as HSAPP expects. int mqd_size = (1 << ((mqd->hqd_pq_control & 0x3f) + 1)) * 4; auto &hsa_pp = gpuDevice->CP()->hsaPacketProc(); hsa_pp.setDeviceQueueDesc(mqd->aqlRptr, mqd->base, new_q->id(), mqd_size, 8, GfxVersion::gfx900, offset, mqd->mqdReadIndex); } DPRINTF(PM4PacketProcessor, "PM4 mqd read completed, base %p, mqd %p, " "hqdAQL %d.\n", mqd->base, mqd->mqdBase, mqd->aql); } void PM4PacketProcessor::processSDMAMQD(PM4MapQueues *pkt, PM4Queue *q, Addr addr, SDMAQueueDesc *mqd, uint16_t vmid) { DPRINTF(PM4PacketProcessor, "SDMAMQD: rb base: %#lx rptr: %#x/%#x wptr: " "%#x/%#x ib: %#x/%#x size: %d ctrl: %#x\n", mqd->rb_base, mqd->sdmax_rlcx_rb_rptr, mqd->sdmax_rlcx_rb_rptr_hi, mqd->sdmax_rlcx_rb_wptr, mqd->sdmax_rlcx_rb_wptr_hi, mqd->sdmax_rlcx_ib_base_lo, mqd->sdmax_rlcx_ib_base_hi, mqd->sdmax_rlcx_ib_size, mqd->sdmax_rlcx_rb_cntl); // Engine 2 points to SDMA0 while engine 3 points to SDMA1 assert(pkt->engineSel == 2 || pkt->engineSel == 3); SDMAEngine *sdma_eng = gpuDevice->getSDMAById(pkt->engineSel - 2); // Register RLC queue with SDMA sdma_eng->registerRLCQueue(pkt->doorbellOffset << 2, mqd->rb_base << 8); // Register doorbell with GPU device gpuDevice->setSDMAEngine(pkt->doorbellOffset << 2, sdma_eng); gpuDevice->setDoorbellType(pkt->doorbellOffset << 2, RLC); } void PM4PacketProcessor::releaseMem(PM4Queue *q, PM4ReleaseMem *pkt) { q->incRptr(sizeof(PM4ReleaseMem)); Addr addr = getGARTAddr(pkt->addr); DPRINTF(PM4PacketProcessor, "PM4 release_mem event %d eventIdx %d intSel " "%d destSel %d dataSel %d, address %p data %p, intCtx %p\n", pkt->event, pkt->eventIdx, pkt->intSelect, pkt->destSelect, pkt->dataSelect, addr, pkt->dataLo, pkt->intCtxId); DPRINTF(PM4PacketProcessor, "PM4 release_mem destSel 0 bypasses caches to MC.\n"); if (pkt->dataSelect == 1) { auto cb = new DmaVirtCallback( [ = ](const uint32_t &) { releaseMemDone(q, pkt, addr); }, pkt->dataLo); dmaWriteVirt(addr, sizeof(uint32_t), cb, &cb->dmaBuffer); } else { panic("Unimplemented PM4ReleaseMem.dataSelect"); } } void PM4PacketProcessor::releaseMemDone(PM4Queue *q, PM4ReleaseMem *pkt, Addr addr) { DPRINTF(PM4PacketProcessor, "PM4 release_mem wrote %d to %p\n", pkt->dataLo, addr); if (pkt->intSelect == 2) { DPRINTF(PM4PacketProcessor, "PM4 interrupt, ctx: %d, me: %d, pipe: " "%d, queueSlot:%d\n", pkt->intCtxId, q->me(), q->pipe(), q->queue()); // Rearranging the queue field of PM4MapQueues as the interrupt RingId // format specified in PM4ReleaseMem pkt. uint32_t ringId = (q->me() << 6) | (q->pipe() << 4) | q->queue(); gpuDevice->getIH()->prepareInterruptCookie(pkt->intCtxId, ringId, SOC15_IH_CLIENTID_GRBM_CP, CP_EOP); gpuDevice->getIH()->submitInterruptCookie(); } delete pkt; decodeNext(q); } void PM4PacketProcessor::updateReadIndex(Addr offset, uint64_t rd_idx) { assert(queuesMap.count(offset)); queuesMap[offset]->getMQD()->mqdReadIndex = rd_idx; } void PM4PacketProcessor::unmapQueues(PM4Queue *q, PM4UnmapQueues *pkt) { q->incRptr(sizeof(PM4UnmapQueues)); DPRINTF(PM4PacketProcessor, "PM4 unmap_queues queueSel: %d numQueues: %d " "pasid: %p doorbellOffset0 %p \n", pkt->queueSel, pkt->numQueues, pkt->pasid, pkt->doorbellOffset0); switch (pkt->queueSel) { case 0: switch (pkt->numQueues) { case 1: gpuDevice->deallocateVmid( gpuDevice->getVMID(pkt->doorbellOffset0)); gpuDevice->deallocateVmid( gpuDevice->getVMID(pkt->doorbellOffset1)); gpuDevice->deallocateVmid( gpuDevice->getVMID(pkt->doorbellOffset2)); gpuDevice->deallocateVmid( gpuDevice->getVMID(pkt->doorbellOffset3)); break; case 2: gpuDevice->deallocateVmid( gpuDevice->getVMID(pkt->doorbellOffset1)); gpuDevice->deallocateVmid( gpuDevice->getVMID(pkt->doorbellOffset2)); gpuDevice->deallocateVmid( gpuDevice->getVMID(pkt->doorbellOffset3)); break; case 3: gpuDevice->deallocateVmid( gpuDevice->getVMID(pkt->doorbellOffset2)); gpuDevice->deallocateVmid( gpuDevice->getVMID(pkt->doorbellOffset3)); break; case 4: gpuDevice->deallocateVmid( gpuDevice->getVMID(pkt->doorbellOffset3)); break; default: panic("Unrecognized number of queues %d\n", pkt->numQueues); } break; case 1: gpuDevice->deallocatePasid(pkt->pasid); break; case 2: break; case 3: { auto &hsa_pp = gpuDevice->CP()->hsaPacketProc(); for (auto iter : gpuDevice->getUsedVMIDs()) { for (auto id : iter.second) { assert(queues.count(id)); // Do not unmap KMD queues if (queues[id]->privileged()) { continue; } QueueDesc *mqd = queues[id]->getMQD(); DPRINTF(PM4PacketProcessor, "Unmapping queue %d with read " "index %ld\n", id, mqd->mqdReadIndex); // Partially writing the mqd with an offset of 96 dwords Addr addr = getGARTAddr(queues[id]->mqdBase() + 96 * sizeof(uint32_t)); Addr mqd_base = queues[id]->mqdBase(); auto cb = new DmaVirtCallback( [ = ] (const uint32_t &) { doneMQDWrite(mqd_base, addr); }); mqd->base >>= 8; dmaWriteVirt(addr, sizeof(QueueDesc), cb, mqd); queues.erase(id); hsa_pp.unsetDeviceQueueDesc(id, 8); } } gpuDevice->deallocateAllQueues(); } break; default: panic("Unrecognized options\n"); break; } delete pkt; decodeNext(q); } void PM4PacketProcessor::doneMQDWrite(Addr mqdAddr, Addr addr) { DPRINTF(PM4PacketProcessor, "PM4 unmap_queues MQD %p wrote to addr %p\n", mqdAddr, addr); } void PM4PacketProcessor::mapProcess(PM4Queue *q, PM4MapProcess *pkt) { q->incRptr(sizeof(PM4MapProcess)); uint16_t vmid = gpuDevice->allocateVMID(pkt->pasid); DPRINTF(PM4PacketProcessor, "PM4 map_process pasid: %p vmid: %d quantum: " "%d pt: %p signal: %p\n", pkt->pasid, vmid, pkt->processQuantum, pkt->ptBase, pkt->completionSignal); gpuDevice->getVM().setPageTableBase(vmid, pkt->ptBase); gpuDevice->CP()->shader()->setHwReg(HW_REG_SH_MEM_BASES, pkt->shMemBases); // Setup the apertures that gem5 uses. These values are bits [63:48]. Addr lds_base = (Addr)bits(pkt->shMemBases, 31, 16) << 48; Addr scratch_base = (Addr)bits(pkt->shMemBases, 15, 0) << 48; // There does not seem to be any register for the limit, but the driver // assumes scratch and LDS have a 4GB aperture, so use that. gpuDevice->CP()->shader()->setLdsApe(lds_base, lds_base + 0xFFFFFFFF); gpuDevice->CP()->shader()->setScratchApe(scratch_base, scratch_base + 0xFFFFFFFF); delete pkt; decodeNext(q); } void PM4PacketProcessor::runList(PM4Queue *q, PM4RunList *pkt) { DPRINTF(PM4PacketProcessor, "PM4 run_list base: %p size: %d\n", pkt->ibBase, pkt->ibSize); q->incRptr(sizeof(PM4RunList)); q->ib(true); q->ibBase(pkt->ibBase); q->rptr(0); q->wptr(pkt->ibSize * sizeof(uint32_t)); delete pkt; decodeNext(q); } void PM4PacketProcessor::indirectBuffer(PM4Queue *q, PM4IndirectBuf *pkt) { DPRINTF(PM4PacketProcessor, "PM4 indirect buffer, base: %p.\n", pkt->ibBase); q->incRptr(sizeof(PM4IndirectBuf)); q->ib(true); q->ibBase(pkt->ibBase); q->wptr(pkt->ibSize * sizeof(uint32_t)); decodeNext(q); } void PM4PacketProcessor::switchBuffer(PM4Queue *q, PM4SwitchBuf *pkt) { q->incRptr(sizeof(PM4SwitchBuf)); q->ib(true); DPRINTF(PM4PacketProcessor, "PM4 switching buffer, rptr: %p.\n", q->wptr()); decodeNext(q); } void PM4PacketProcessor::setUconfigReg(PM4Queue *q, PM4SetUconfigReg *pkt) { q->incRptr(sizeof(PM4SetUconfigReg)); // SET_UCONFIG_REG_START and pkt->offset are dword addresses uint32_t reg_addr = (PACKET3_SET_UCONFIG_REG_START + pkt->offset) * 4; gpuDevice->setRegVal(reg_addr, pkt->data); decodeNext(q); } void PM4PacketProcessor::waitRegMem(PM4Queue *q, PM4WaitRegMem *pkt) { q->incRptr(sizeof(PM4WaitRegMem)); DPRINTF(PM4PacketProcessor, "PM4 WAIT_REG_MEM\nfunc: %d memSpace: %d op: " "%d\n", pkt->function, pkt->memSpace, pkt->operation); DPRINTF(PM4PacketProcessor, " AddrLo/Reg1: %lx\n", pkt->memAddrLo); DPRINTF(PM4PacketProcessor, " AddrHi/Reg2: %lx\n", pkt->memAddrHi); DPRINTF(PM4PacketProcessor, " Reference: %lx\n", pkt->reference); DPRINTF(PM4PacketProcessor, " Mask: %lx\n", pkt->mask); DPRINTF(PM4PacketProcessor, " Poll Interval: %lx\n", pkt->pollInterval); decodeNext(q); } void PM4PacketProcessor::queryStatus(PM4Queue *q, PM4QueryStatus *pkt) { q->incRptr(sizeof(PM4QueryStatus)); DPRINTF(PM4PacketProcessor, "PM4 query status contextId: %d, interruptSel:" " %d command: %d, pasid: %d, doorbellOffset: %d, engineSel: %d " "addr: %lx, data: %lx\n", pkt->contextId, pkt->interruptSel, pkt->command, pkt->pasid, pkt->doorbellOffset, pkt->engineSel, pkt->addr, pkt->data); if (pkt->interruptSel == 0 && pkt->command == 2) { // Write data value to fence address Addr addr = getGARTAddr(pkt->addr); DPRINTF(PM4PacketProcessor, "Using GART addr %lx\n", addr); auto cb = new DmaVirtCallback( [ = ] (const uint64_t &) { queryStatusDone(q, pkt); }, pkt->data); dmaWriteVirt(addr, sizeof(uint64_t), cb, &cb->dmaBuffer); } else { // No other combinations used in amdkfd v9 panic("query_status with interruptSel %d command %d not supported", pkt->interruptSel, pkt->command); } } void PM4PacketProcessor::queryStatusDone(PM4Queue *q, PM4QueryStatus *pkt) { DPRINTF(PM4PacketProcessor, "PM4 query status complete\n"); delete pkt; decodeNext(q); } void PM4PacketProcessor::writeMMIO(PacketPtr pkt, Addr mmio_offset) { switch (mmio_offset) { /* Hardware queue descriptor (HQD) registers */ case mmCP_HQD_VMID: setHqdVmid(pkt->getLE()); break; case mmCP_HQD_ACTIVE: setHqdActive(pkt->getLE()); break; case mmCP_HQD_PQ_BASE: setHqdPqBase(pkt->getLE()); break; case mmCP_HQD_PQ_BASE_HI: setHqdPqBaseHi(pkt->getLE()); break; case mmCP_HQD_PQ_DOORBELL_CONTROL: setHqdPqDoorbellCtrl(pkt->getLE()); gpuDevice->setDoorbellType(getKiqDoorbellOffset(), Compute); break; case mmCP_HQD_PQ_RPTR: setHqdPqPtr(pkt->getLE()); break; case mmCP_HQD_PQ_WPTR_LO: setHqdPqWptrLo(pkt->getLE()); break; case mmCP_HQD_PQ_WPTR_HI: setHqdPqWptrHi(pkt->getLE()); break; case mmCP_HQD_PQ_RPTR_REPORT_ADDR: setHqdPqRptrReportAddr(pkt->getLE()); break; case mmCP_HQD_PQ_RPTR_REPORT_ADDR_HI: setHqdPqRptrReportAddrHi(pkt->getLE()); break; case mmCP_HQD_PQ_WPTR_POLL_ADDR: setHqdPqWptrPollAddr(pkt->getLE()); break; case mmCP_HQD_PQ_WPTR_POLL_ADDR_HI: setHqdPqWptrPollAddrHi(pkt->getLE()); break; case mmCP_HQD_IB_CONTROL: setHqdIbCtrl(pkt->getLE()); break; /* Ring buffer registers */ case mmCP_RB_VMID: setRbVmid(pkt->getLE()); break; case mmCP_RB0_CNTL: setRbCntl(pkt->getLE()); break; case mmCP_RB0_WPTR: setRbWptrLo(pkt->getLE()); break; case mmCP_RB0_WPTR_HI: setRbWptrHi(pkt->getLE()); break; case mmCP_RB0_RPTR_ADDR: setRbRptrAddrLo(pkt->getLE()); break; case mmCP_RB0_RPTR_ADDR_HI: setRbRptrAddrHi(pkt->getLE()); break; case mmCP_RB_WPTR_POLL_ADDR_LO: setRbWptrPollAddrLo(pkt->getLE()); break; case mmCP_RB_WPTR_POLL_ADDR_HI: setRbWptrPollAddrHi(pkt->getLE()); break; case mmCP_RB0_BASE: setRbBaseLo(pkt->getLE()); break; case mmCP_RB0_BASE_HI: setRbBaseHi(pkt->getLE()); break; case mmCP_RB_DOORBELL_CONTROL: setRbDoorbellCntrl(pkt->getLE()); gpuDevice->setDoorbellType(getPqDoorbellOffset(), Gfx); break; case mmCP_RB_DOORBELL_RANGE_LOWER: setRbDoorbellRangeLo(pkt->getLE()); break; case mmCP_RB_DOORBELL_RANGE_UPPER: setRbDoorbellRangeHi(pkt->getLE()); break; default: break; } } void PM4PacketProcessor::setHqdVmid(uint32_t data) { kiq.hqd_vmid = data; } void PM4PacketProcessor::setHqdActive(uint32_t data) { kiq.hqd_active = data; } void PM4PacketProcessor::setHqdPqBase(uint32_t data) { kiq.hqd_pq_base_lo = data; } void PM4PacketProcessor::setHqdPqBaseHi(uint32_t data) { kiq.hqd_pq_base_hi = data; } void PM4PacketProcessor::setHqdPqDoorbellCtrl(uint32_t data) { kiq.hqd_pq_doorbell_control = data; } void PM4PacketProcessor::setHqdPqPtr(uint32_t data) { kiq.rptr = data; } void PM4PacketProcessor::setHqdPqWptrLo(uint32_t data) { /* Write pointer communicated through doorbell value. */ } void PM4PacketProcessor::setHqdPqWptrHi(uint32_t data) { /* Write pointer communicated through doorbell value. */ } void PM4PacketProcessor::setHqdPqRptrReportAddr(uint32_t data) { kiq.hqd_pq_rptr_report_addr_lo = data; } void PM4PacketProcessor::setHqdPqRptrReportAddrHi(uint32_t data) { kiq.hqd_pq_rptr_report_addr_hi = data; } void PM4PacketProcessor::setHqdPqWptrPollAddr(uint32_t data) { kiq.hqd_pq_wptr_poll_addr_lo = data; } void PM4PacketProcessor::setHqdPqWptrPollAddrHi(uint32_t data) { kiq.hqd_pq_wptr_poll_addr_hi = data; } void PM4PacketProcessor::setHqdIbCtrl(uint32_t data) { kiq.hqd_ib_control = data; } void PM4PacketProcessor::setRbVmid(uint32_t data) { pq.hqd_vmid = data; } void PM4PacketProcessor::setRbCntl(uint32_t data) { pq.hqd_pq_control = data; } void PM4PacketProcessor::setRbWptrLo(uint32_t data) { pq.queueWptrLo = data; } void PM4PacketProcessor::setRbWptrHi(uint32_t data) { pq.queueWptrHi = data; } void PM4PacketProcessor::setRbRptrAddrLo(uint32_t data) { pq.queueRptrAddrLo = data; } void PM4PacketProcessor::setRbRptrAddrHi(uint32_t data) { pq.queueRptrAddrHi = data; } void PM4PacketProcessor::setRbWptrPollAddrLo(uint32_t data) { pq.hqd_pq_wptr_poll_addr_lo = data; } void PM4PacketProcessor::setRbWptrPollAddrHi(uint32_t data) { pq.hqd_pq_wptr_poll_addr_hi = data; } void PM4PacketProcessor::setRbBaseLo(uint32_t data) { pq.hqd_pq_base_lo = data; } void PM4PacketProcessor::setRbBaseHi(uint32_t data) { pq.hqd_pq_base_hi = data; } void PM4PacketProcessor::setRbDoorbellCntrl(uint32_t data) { pq.hqd_pq_doorbell_control = data; pq.doorbellOffset = data & 0x1ffffffc; } void PM4PacketProcessor::setRbDoorbellRangeLo(uint32_t data) { pq.doorbellRangeLo = data; } void PM4PacketProcessor::setRbDoorbellRangeHi(uint32_t data) { pq.doorbellRangeHi = data; } void PM4PacketProcessor::serialize(CheckpointOut &cp) const { // Serialize the DmaVirtDevice base class DmaVirtDevice::serialize(cp); int num_queues = queues.size(); Addr id[num_queues]; Addr mqd_base[num_queues]; Addr base[num_queues]; Addr rptr[num_queues]; Addr wptr[num_queues]; Addr ib_base[num_queues]; Addr ib_rptr[num_queues]; Addr ib_wptr[num_queues]; Addr offset[num_queues]; bool processing[num_queues]; bool ib[num_queues]; int i = 0; for (auto iter : queues) { PM4Queue *q = iter.second; id[i] = q->id(); mqd_base[i] = q->mqdBase(); bool cur_state = q->ib(); q->ib(false); base[i] = q->base() >> 8; rptr[i] = q->getRptr(); wptr[i] = q->getWptr(); q->ib(true); ib_base[i] = q->ibBase(); ib_rptr[i] = q->getRptr(); ib_wptr[i] = q->getWptr(); q->ib(cur_state); offset[i] = q->offset(); processing[i] = q->processing(); ib[i] = q->ib(); i++; } SERIALIZE_SCALAR(num_queues); SERIALIZE_ARRAY(id, num_queues); SERIALIZE_ARRAY(mqd_base, num_queues); SERIALIZE_ARRAY(base, num_queues); SERIALIZE_ARRAY(rptr, num_queues); SERIALIZE_ARRAY(wptr, num_queues); SERIALIZE_ARRAY(ib_base, num_queues); SERIALIZE_ARRAY(ib_rptr, num_queues); SERIALIZE_ARRAY(ib_wptr, num_queues); SERIALIZE_ARRAY(offset, num_queues); SERIALIZE_ARRAY(processing, num_queues); SERIALIZE_ARRAY(ib, num_queues); } void PM4PacketProcessor::unserialize(CheckpointIn &cp) { // Serialize the DmaVirtDevice base class DmaVirtDevice::unserialize(cp); int num_queues = 0; UNSERIALIZE_SCALAR(num_queues); Addr id[num_queues]; Addr mqd_base[num_queues]; Addr base[num_queues]; Addr rptr[num_queues]; Addr wptr[num_queues]; Addr ib_base[num_queues]; Addr ib_rptr[num_queues]; Addr ib_wptr[num_queues]; Addr offset[num_queues]; bool processing[num_queues]; bool ib[num_queues]; UNSERIALIZE_ARRAY(id, num_queues); UNSERIALIZE_ARRAY(mqd_base, num_queues); UNSERIALIZE_ARRAY(base, num_queues); UNSERIALIZE_ARRAY(rptr, num_queues); UNSERIALIZE_ARRAY(wptr, num_queues); UNSERIALIZE_ARRAY(ib_base, num_queues); UNSERIALIZE_ARRAY(ib_rptr, num_queues); UNSERIALIZE_ARRAY(ib_wptr, num_queues); UNSERIALIZE_ARRAY(offset, num_queues); UNSERIALIZE_ARRAY(processing, num_queues); UNSERIALIZE_ARRAY(ib, num_queues); for (int i = 0; i < num_queues; i++) { QueueDesc *mqd = new QueueDesc(); memset(mqd, 0, sizeof(QueueDesc)); mqd->mqdBase = mqd_base[i] >> 8; mqd->base = base[i]; mqd->rptr = rptr[i]; mqd->ibBase = ib_base[i]; mqd->ibRptr = ib_rptr[i]; newQueue(mqd, offset[i], nullptr, id[i]); queues[id[i]]->ib(false); queues[id[i]]->wptr(wptr[i]); queues[id[i]]->ib(true); queues[id[i]]->wptr(ib_wptr[i]); queues[id[i]]->offset(offset[i]); queues[id[i]]->processing(processing[i]); queues[id[i]]->ib(ib[i]); DPRINTF(PM4PacketProcessor, "PM4 queue %d, rptr: %p wptr: %p\n", queues[id[i]]->id(), queues[id[i]]->rptr(), queues[id[i]]->wptr()); } } } // namespace gem5