mem: Enforce insertion order on the cache response path
This patch enforces insertion order transmission of packets on the response path in the cache. Note that the logic to enforce order is already present in the packet queue, this patch simply turns it on for queues in the response path. Without this patch, there are corner cases where a request-response is faster than a response-response forwarded through the cache. This violation of queuing order causes problems in the snoop filter leaving it with inaccurate information. This causes assert failures in the snoop filter later on. A follow on patch relaxes the order enforcement in the packet queue to limit the performance impact.
This commit is contained in:
parent
6b70afd0d4
commit
52c8ae5187
3 changed files with 6 additions and 6 deletions
6
src/mem/cache/cache.cc
vendored
6
src/mem/cache/cache.cc
vendored
|
@ -666,7 +666,7 @@ Cache::recvTimingReq(PacketPtr pkt)
|
||||||
// lat, neglecting responseLatency, modelling hit latency
|
// lat, neglecting responseLatency, modelling hit latency
|
||||||
// just as lookupLatency or or the value of lat overriden
|
// just as lookupLatency or or the value of lat overriden
|
||||||
// by access(), that calls accessBlock() function.
|
// by access(), that calls accessBlock() function.
|
||||||
cpuSidePort->schedTimingResp(pkt, request_time);
|
cpuSidePort->schedTimingResp(pkt, request_time, true);
|
||||||
} else {
|
} else {
|
||||||
// queue the packet for deletion, as the sending cache is
|
// queue the packet for deletion, as the sending cache is
|
||||||
// still relying on it; if the block is found in access(),
|
// still relying on it; if the block is found in access(),
|
||||||
|
@ -723,7 +723,7 @@ Cache::recvTimingReq(PacketPtr pkt)
|
||||||
std::memset(pkt->getPtr<uint8_t>(), 0xFF, pkt->getSize());
|
std::memset(pkt->getPtr<uint8_t>(), 0xFF, pkt->getSize());
|
||||||
// request_time is used here, taking into account lat and the delay
|
// request_time is used here, taking into account lat and the delay
|
||||||
// charged if the packet comes from the xbar.
|
// charged if the packet comes from the xbar.
|
||||||
cpuSidePort->schedTimingResp(pkt, request_time);
|
cpuSidePort->schedTimingResp(pkt, request_time, true);
|
||||||
|
|
||||||
// If an outstanding request is in progress (we found an
|
// If an outstanding request is in progress (we found an
|
||||||
// MSHR) this is set to null
|
// MSHR) this is set to null
|
||||||
|
@ -1330,7 +1330,7 @@ Cache::recvTimingResp(PacketPtr pkt)
|
||||||
}
|
}
|
||||||
// Reset the bus additional time as it is now accounted for
|
// Reset the bus additional time as it is now accounted for
|
||||||
tgt_pkt->headerDelay = tgt_pkt->payloadDelay = 0;
|
tgt_pkt->headerDelay = tgt_pkt->payloadDelay = 0;
|
||||||
cpuSidePort->schedTimingResp(tgt_pkt, completion_time);
|
cpuSidePort->schedTimingResp(tgt_pkt, completion_time, true);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case MSHR::Target::FromPrefetcher:
|
case MSHR::Target::FromPrefetcher:
|
||||||
|
|
|
@ -860,7 +860,7 @@ DRAMCtrl::accessAndRespond(PacketPtr pkt, Tick static_latency)
|
||||||
|
|
||||||
// queue the packet in the response queue to be sent out after
|
// queue the packet in the response queue to be sent out after
|
||||||
// the static latency has passed
|
// the static latency has passed
|
||||||
port.schedTimingResp(pkt, response_time);
|
port.schedTimingResp(pkt, response_time, true);
|
||||||
} else {
|
} else {
|
||||||
// @todo the packet is going to be deleted, and the DRAMPacket
|
// @todo the packet is going to be deleted, and the DRAMPacket
|
||||||
// is still having a pointer to it
|
// is still having a pointer to it
|
||||||
|
|
|
@ -88,8 +88,8 @@ class QueuedSlavePort : public SlavePort
|
||||||
* @param pkt Packet to send
|
* @param pkt Packet to send
|
||||||
* @param when Absolute time (in ticks) to send packet
|
* @param when Absolute time (in ticks) to send packet
|
||||||
*/
|
*/
|
||||||
void schedTimingResp(PacketPtr pkt, Tick when)
|
void schedTimingResp(PacketPtr pkt, Tick when, bool force_order = false)
|
||||||
{ respQueue.schedSendTiming(pkt, when); }
|
{ respQueue.schedSendTiming(pkt, when, force_order); }
|
||||||
|
|
||||||
/** Check the list of buffered packets against the supplied
|
/** Check the list of buffered packets against the supplied
|
||||||
* functional request. */
|
* functional request. */
|
||||||
|
|
Loading…
Reference in a new issue