WAIT_MEM_REG, REG_RMW (untested), COND_WRITE

This commit is contained in:
Ben Vanik 2013-10-19 00:46:35 -07:00
parent a63db2b7fd
commit 1616d7ba22
1 changed files with 138 additions and 12 deletions

View File

@ -289,27 +289,152 @@ uint32_t RingBufferWorker::ExecutePacket(PacketArgs& args) {
case PM4_WAIT_REG_MEM: case PM4_WAIT_REG_MEM:
// wait until a register or memory location is a specific value // wait until a register or memory location is a specific value
XELOGGPU("[%.8X] Packet(%.8X): PM4_WAIT_REG_MEM", {
packet_ptr, packet); XELOGGPU("[%.8X] Packet(%.8X): PM4_WAIT_REG_MEM",
LOG_DATA(count); packet_ptr, packet);
ADVANCE_PTR(count); LOG_DATA(count);
uint32_t wait_info = READ_AND_ADVANCE_PTR();
uint32_t poll_reg_addr = READ_AND_ADVANCE_PTR();
uint32_t ref = READ_AND_ADVANCE_PTR();
uint32_t mask = READ_AND_ADVANCE_PTR();
uint32_t wait = READ_AND_ADVANCE_PTR();
bool matched = false;
do {
uint32_t value;
if (wait_info & 0x10) {
// Memory.
value = XEGETUINT32BE(p + TRANSLATE_ADDR(poll_reg_addr));
} else {
// Register.
XEASSERT(poll_reg_addr < kXEGpuRegisterCount);
value = regs->values[poll_reg_addr].u32;
}
switch (wait_info & 0x7) {
case 0x0: // Always.
matched = true;
break;
case 0x1: // Less than reference.
matched = (value & mask) < ref;
break;
case 0x2: // Less than or equal to reference.
matched = (value & mask) <= ref;
break;
case 0x3: // Equal to reference.
matched = (value & mask) == ref;
break;
case 0x4: // Not equal to reference.
matched = (value & mask) != ref;
break;
case 0x5: // Greater than or equal to reference.
matched = (value & mask) >= ref;
break;
case 0x6: // Greater than reference.
matched = (value & mask) > ref;
break;
default:
XELOGE("Unsupported wait comparison type!");
XEASSERTALWAYS();
break;
}
if (!matched) {
// Wait.
SwitchToThread();
}
} while (!matched);
}
break; break;
case PM4_REG_RMW: case PM4_REG_RMW:
// register read/modify/write // register read/modify/write
// ? (used during shader upload and edram setup) // ? (used during shader upload and edram setup)
XELOGGPU("[%.8X] Packet(%.8X): PM4_REG_RMW", {
packet_ptr, packet); XELOGGPU("[%.8X] Packet(%.8X): PM4_REG_RMW",
LOG_DATA(count); packet_ptr, packet);
ADVANCE_PTR(count); LOG_DATA(count);
uint32_t rmw_info = READ_AND_ADVANCE_PTR();
uint32_t and_mask = READ_AND_ADVANCE_PTR();
uint32_t or_mask = READ_AND_ADVANCE_PTR();
uint32_t value = regs->values[rmw_info & 0x1FFF].u32;
if ((rmw_info >> 30) & 0x1) {
// | reg
value |= regs->values[or_mask & 0x1FFF].u32;
} else {
// | imm
value |= or_mask;
}
if ((rmw_info >> 31) & 0x1) {
// & reg
value &= regs->values[and_mask & 0x1FFF].u32;
} else {
// & imm
value &= and_mask;
}
regs->values[rmw_info & 0x1FFF].u32 = value;
}
break; break;
case PM4_COND_WRITE: case PM4_COND_WRITE:
// conditional write to memory or register // conditional write to memory or register
XELOGGPU("[%.8X] Packet(%.8X): PM4_COND_WRITE", {
packet_ptr, packet); XELOGGPU("[%.8X] Packet(%.8X): PM4_COND_WRITE",
LOG_DATA(count); packet_ptr, packet);
ADVANCE_PTR(count); LOG_DATA(count);
uint32_t wait_info = READ_AND_ADVANCE_PTR();
uint32_t poll_reg_addr = READ_AND_ADVANCE_PTR();
uint32_t ref = READ_AND_ADVANCE_PTR();
uint32_t mask = READ_AND_ADVANCE_PTR();
uint32_t wait = READ_AND_ADVANCE_PTR();
uint32_t write_reg_addr = READ_AND_ADVANCE_PTR();
uint32_t write_data = READ_AND_ADVANCE_PTR();
uint32_t value;
if (wait_info & 0x10) {
// Memory.
value = XEGETUINT32BE(p + TRANSLATE_ADDR(poll_reg_addr));
} else {
// Register.
XEASSERT(poll_reg_addr < kXEGpuRegisterCount);
value = regs->values[poll_reg_addr].u32;
}
bool matched = false;
switch (wait_info & 0x7) {
case 0x0: // Always.
matched = true;
break;
case 0x1: // Less than reference.
matched = (value & mask) < ref;
break;
case 0x2: // Less than or equal to reference.
matched = (value & mask) <= ref;
break;
case 0x3: // Equal to reference.
matched = (value & mask) == ref;
break;
case 0x4: // Not equal to reference.
matched = (value & mask) != ref;
break;
case 0x5: // Greater than or equal to reference.
matched = (value & mask) >= ref;
break;
case 0x6: // Greater than reference.
matched = (value & mask) > ref;
break;
default:
XELOGE("Unsupported wait comparison type!");
XEASSERTALWAYS();
break;
}
if (matched) {
// Write.
if (wait_info & 0x100) {
// Memory.
XESETUINT32BE(p + TRANSLATE_ADDR(write_reg_addr), write_data);
} else {
// Register.
XEASSERT(write_reg_addr < kXEGpuRegisterCount);
regs->values[write_reg_addr].u32 = write_data;
}
}
}
break; break;
case PM4_EVENT_WRITE: case PM4_EVENT_WRITE:
@ -326,6 +451,7 @@ uint32_t RingBufferWorker::ExecutePacket(PacketArgs& args) {
packet_ptr, packet); packet_ptr, packet);
LOG_DATA(count); LOG_DATA(count);
uint32_t d0 = READ_AND_ADVANCE_PTR(); // 3? uint32_t d0 = READ_AND_ADVANCE_PTR(); // 3?
XEASSERT(d0 == 0x3);
uint32_t d1 = READ_AND_ADVANCE_PTR(); // ptr uint32_t d1 = READ_AND_ADVANCE_PTR(); // ptr
uint32_t d2 = READ_AND_ADVANCE_PTR(); // value? uint32_t d2 = READ_AND_ADVANCE_PTR(); // value?
if (!(d1 & 0xC0000000)) { if (!(d1 & 0xC0000000)) {