2006-01-31 18:12:49 +01:00
|
|
|
/*
|
2015-02-11 16:23:47 +01:00
|
|
|
* Copyright (c) 2012-2015 ARM Limited
|
MEM: Remove the Broadcast destination from the packet
This patch simplifies the packet by removing the broadcast flag and
instead more firmly relying on (and enforcing) the semantics of
transactions in the classic memory system, i.e. request packets are
routed from a master to a slave based on the address, and when they
are created they have neither a valid source, nor destination. On
their way to the slave, the request packet is updated with a source
field for all modules that multiplex packets from multiple master
(e.g. a bus). When a request packet is turned into a response packet
(at the final slave), it moves the potentially populated source field
to the destination field, and the response packet is routed through
any multiplexing components back to the master based on the
destination field.
Modules that connect multiplexing components, such as caches and
bridges store any existing source and destination field in the sender
state as a stack (just as before).
The packet constructor is simplified in that there is no longer a need
to pass the Packet::Broadcast as the destination (this was always the
case for the classic memory system). In the case of Ruby, rather than
using the parameter to the constructor we now rely on setDest, as
there is already another three-argument constructor in the packet
class.
In many places where the packet information was printed as part of
DPRINTFs, request packets would be printed with a numeric "dest" that
would always be -1 (Broadcast) and that field is now removed from the
printing.
2012-04-14 11:45:55 +02:00
|
|
|
* All rights reserved
|
|
|
|
*
|
|
|
|
* The license below extends only to copyright in the software and shall
|
|
|
|
* not be construed as granting a license to any other intellectual
|
|
|
|
* property including but not limited to intellectual property relating
|
|
|
|
* to a hardware implementation of the functionality of the software
|
|
|
|
* licensed hereunder. You may use the software subject to the license
|
|
|
|
* terms below provided that you ensure that this notice is replicated
|
|
|
|
* unmodified and in its entirety in all distributions of the software,
|
|
|
|
* modified or unmodified, in source code or in binary form.
|
|
|
|
*
|
2006-05-02 00:53:28 +02:00
|
|
|
* Copyright (c) 2006 The Regents of The University of Michigan
|
2010-08-25 23:08:27 +02:00
|
|
|
* Copyright (c) 2010 Advanced Micro Devices, Inc.
|
2006-01-31 18:12:49 +01:00
|
|
|
* All rights reserved.
|
|
|
|
*
|
|
|
|
* Redistribution and use in source and binary forms, with or without
|
|
|
|
* modification, are permitted provided that the following conditions are
|
|
|
|
* met: redistributions of source code must retain the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer;
|
|
|
|
* 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;
|
|
|
|
* neither the name of the copyright holders 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
|
|
|
|
* OWNER 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.
|
2006-06-01 01:26:56 +02:00
|
|
|
*
|
|
|
|
* Authors: Ron Dreslinski
|
|
|
|
* Steve Reinhardt
|
|
|
|
* Ali Saidi
|
2012-05-30 11:29:42 +02:00
|
|
|
* Andreas Hansson
|
2006-01-31 18:12:49 +01:00
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @file
|
2006-05-31 04:30:42 +02:00
|
|
|
* Declaration of the Packet class.
|
2006-01-31 18:12:49 +01:00
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef __MEM_PACKET_HH__
|
|
|
|
#define __MEM_PACKET_HH__
|
|
|
|
|
2011-04-15 19:44:06 +02:00
|
|
|
#include <bitset>
|
2006-10-20 08:38:45 +02:00
|
|
|
#include <cassert>
|
|
|
|
#include <list>
|
|
|
|
|
2008-11-10 20:51:17 +01:00
|
|
|
#include "base/cast.hh"
|
2007-02-07 07:31:15 +01:00
|
|
|
#include "base/compiler.hh"
|
2008-11-10 20:51:17 +01:00
|
|
|
#include "base/flags.hh"
|
2007-01-27 21:38:04 +01:00
|
|
|
#include "base/misc.hh"
|
2008-01-02 21:20:15 +01:00
|
|
|
#include "base/printable.hh"
|
2009-05-17 23:34:50 +02:00
|
|
|
#include "base/types.hh"
|
2009-05-17 23:34:52 +02:00
|
|
|
#include "mem/request.hh"
|
2007-03-06 20:13:43 +01:00
|
|
|
#include "sim/core.hh"
|
2006-02-15 20:21:09 +01:00
|
|
|
|
2012-01-31 18:05:52 +01:00
|
|
|
class Packet;
|
2006-10-20 09:10:12 +02:00
|
|
|
typedef Packet *PacketPtr;
|
2006-02-15 20:21:09 +01:00
|
|
|
typedef uint8_t* PacketDataPtr;
|
2006-06-28 23:28:33 +02:00
|
|
|
typedef std::list<PacketPtr> PacketList;
|
|
|
|
|
2007-02-07 19:53:37 +01:00
|
|
|
class MemCmd
|
|
|
|
{
|
2008-11-10 20:51:17 +01:00
|
|
|
friend class Packet;
|
2007-02-07 19:53:37 +01:00
|
|
|
|
2008-11-10 20:51:17 +01:00
|
|
|
public:
|
|
|
|
/**
|
|
|
|
* List of all commands associated with a packet.
|
|
|
|
*/
|
2007-02-07 19:53:37 +01:00
|
|
|
enum Command
|
|
|
|
{
|
|
|
|
InvalidCmd,
|
|
|
|
ReadReq,
|
|
|
|
ReadResp,
|
2008-01-03 00:22:38 +01:00
|
|
|
ReadRespWithInvalidate,
|
2007-05-22 15:29:48 +02:00
|
|
|
WriteReq,
|
2007-02-07 19:53:37 +01:00
|
|
|
WriteResp,
|
|
|
|
Writeback,
|
2015-07-03 16:14:37 +02:00
|
|
|
CleanEvict,
|
2007-02-07 19:53:37 +01:00
|
|
|
SoftPFReq,
|
|
|
|
HardPFReq,
|
|
|
|
SoftPFResp,
|
|
|
|
HardPFResp,
|
2015-07-03 16:14:41 +02:00
|
|
|
WriteLineReq,
|
2007-02-07 19:53:37 +01:00
|
|
|
UpgradeReq,
|
2010-06-17 00:25:57 +02:00
|
|
|
SCUpgradeReq, // Special "weak" upgrade for StoreCond
|
2007-06-21 20:59:17 +02:00
|
|
|
UpgradeResp,
|
2010-06-17 00:25:57 +02:00
|
|
|
SCUpgradeFailReq, // Failed SCUpgradeReq in MSHR (never sent)
|
|
|
|
UpgradeFailResp, // Valid for SCUpgradeReq only
|
2007-02-07 19:53:37 +01:00
|
|
|
ReadExReq,
|
|
|
|
ReadExResp,
|
mem: Add ReadCleanReq and ReadSharedReq packets
This patch adds two new read requests packets:
ReadCleanReq - For a cache to explicitly request clean data. The
response is thus exclusive or shared, but not owned or modified. The
read-only caches (see previous patch) use this request type to ensure
they do not get dirty data.
ReadSharedReq - We add this to distinguish cache read requests from
those issued by other masters, such as devices and CPUs. Thus, devices
use ReadReq, and caches use ReadCleanReq, ReadExReq, or
ReadSharedReq. For the latter, the response can be any state, shared,
exclusive, owned or even modified.
Both ReadCleanReq and ReadSharedReq re-use the normal ReadResp. The
two transactions are aligned with the emerging cache-coherent TLM
standard and the AMBA nomenclature.
With this change, the normal ReadReq should never be used by a cache,
and is reserved for the actual (non-caching) masters in the system. We
thus have a way of identifying if a request came from a cache or
not. The introduction of ReadSharedReq thus removes the need for the
current isTopLevel hack, and also allows us to stop relying on
checking the packet size to determine if the source is a cache or
not. This is fixed in follow-on patches.
2015-07-03 16:14:40 +02:00
|
|
|
ReadCleanReq,
|
|
|
|
ReadSharedReq,
|
2007-06-18 02:27:53 +02:00
|
|
|
LoadLockedReq,
|
|
|
|
StoreCondReq,
|
2010-09-09 20:40:19 +02:00
|
|
|
StoreCondFailReq, // Failed StoreCondReq in MSHR (never sent)
|
2007-06-18 02:27:53 +02:00
|
|
|
StoreCondResp,
|
2007-02-12 19:06:30 +01:00
|
|
|
SwapReq,
|
|
|
|
SwapResp,
|
2008-10-12 21:08:51 +02:00
|
|
|
MessageReq,
|
|
|
|
MessageResp,
|
2007-06-30 19:16:18 +02:00
|
|
|
// Error responses
|
|
|
|
// @TODO these should be classified as responses rather than
|
|
|
|
// requests; coding them as requests initially for backwards
|
|
|
|
// compatibility
|
|
|
|
InvalidDestError, // packet dest field invalid
|
|
|
|
BadAddressError, // memory address invalid
|
2011-07-01 02:49:26 +02:00
|
|
|
FunctionalReadError, // unable to fulfill functional read
|
|
|
|
FunctionalWriteError, // unable to fulfill functional write
|
2008-01-02 21:20:15 +01:00
|
|
|
// Fake simulator-only commands
|
|
|
|
PrintReq, // Print state matching address
|
2011-03-28 17:49:45 +02:00
|
|
|
FlushReq, //request for a cache flush
|
2015-07-03 16:14:41 +02:00
|
|
|
InvalidateReq, // request for address to be invalidated
|
|
|
|
InvalidateResp,
|
2007-02-07 19:53:37 +01:00
|
|
|
NUM_MEM_CMDS
|
|
|
|
};
|
|
|
|
|
|
|
|
private:
|
2008-11-10 20:51:17 +01:00
|
|
|
/**
|
|
|
|
* List of command attributes.
|
|
|
|
*/
|
2007-02-07 19:53:37 +01:00
|
|
|
enum Attribute
|
|
|
|
{
|
2007-06-18 02:27:53 +02:00
|
|
|
IsRead, //!< Data flows from responder to requester
|
|
|
|
IsWrite, //!< Data flows from requester to responder
|
2010-06-17 00:25:57 +02:00
|
|
|
IsUpgrade,
|
2007-02-07 19:53:37 +01:00
|
|
|
IsInvalidate,
|
2007-06-18 02:27:53 +02:00
|
|
|
NeedsExclusive, //!< Requires exclusive copy to complete in-cache
|
|
|
|
IsRequest, //!< Issued by requester
|
|
|
|
IsResponse, //!< Issue by responder
|
|
|
|
NeedsResponse, //!< Requester needs response from target
|
2007-02-07 19:53:37 +01:00
|
|
|
IsSWPrefetch,
|
|
|
|
IsHWPrefetch,
|
2009-04-19 13:25:01 +02:00
|
|
|
IsLlsc, //!< Alpha/MIPS LL or SC access
|
2007-06-18 02:27:53 +02:00
|
|
|
HasData, //!< There is an associated payload
|
2007-06-30 19:16:18 +02:00
|
|
|
IsError, //!< Error response
|
2008-01-02 21:20:15 +01:00
|
|
|
IsPrint, //!< Print state matching address (for debugging)
|
2011-03-28 17:49:45 +02:00
|
|
|
IsFlush, //!< Flush the address from caches
|
2007-02-07 19:53:37 +01:00
|
|
|
NUM_COMMAND_ATTRIBUTES
|
|
|
|
};
|
|
|
|
|
2008-11-10 20:51:17 +01:00
|
|
|
/**
|
|
|
|
* Structure that defines attributes and other data associated
|
|
|
|
* with a Command.
|
|
|
|
*/
|
|
|
|
struct CommandInfo
|
|
|
|
{
|
|
|
|
/// Set of attribute flags.
|
2007-02-07 19:53:37 +01:00
|
|
|
const std::bitset<NUM_COMMAND_ATTRIBUTES> attributes;
|
2008-11-10 20:51:17 +01:00
|
|
|
/// Corresponding response for requests; InvalidCmd if no
|
|
|
|
/// response is applicable.
|
2007-02-07 19:53:37 +01:00
|
|
|
const Command response;
|
2008-11-10 20:51:17 +01:00
|
|
|
/// String representation (for printing)
|
2007-02-07 19:53:37 +01:00
|
|
|
const std::string str;
|
|
|
|
};
|
|
|
|
|
2008-11-10 20:51:17 +01:00
|
|
|
/// Array to map Command enum to associated info.
|
2007-02-07 19:53:37 +01:00
|
|
|
static const CommandInfo commandInfo[];
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
|
|
|
Command cmd;
|
|
|
|
|
2008-11-10 20:51:17 +01:00
|
|
|
bool
|
|
|
|
testCmdAttrib(MemCmd::Attribute attrib) const
|
|
|
|
{
|
2007-02-07 19:53:37 +01:00
|
|
|
return commandInfo[cmd].attributes[attrib] != 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
2014-12-02 12:08:19 +01:00
|
|
|
bool isRead() const { return testCmdAttrib(IsRead); }
|
|
|
|
bool isWrite() const { return testCmdAttrib(IsWrite); }
|
|
|
|
bool isUpgrade() const { return testCmdAttrib(IsUpgrade); }
|
|
|
|
bool isRequest() const { return testCmdAttrib(IsRequest); }
|
|
|
|
bool isResponse() const { return testCmdAttrib(IsResponse); }
|
|
|
|
bool needsExclusive() const { return testCmdAttrib(NeedsExclusive); }
|
|
|
|
bool needsResponse() const { return testCmdAttrib(NeedsResponse); }
|
|
|
|
bool isInvalidate() const { return testCmdAttrib(IsInvalidate); }
|
2014-12-02 12:07:52 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Check if this particular packet type carries payload data. Note
|
|
|
|
* that this does not reflect if the data pointer of the packet is
|
|
|
|
* valid or not.
|
|
|
|
*/
|
2007-02-07 19:53:37 +01:00
|
|
|
bool hasData() const { return testCmdAttrib(HasData); }
|
2009-04-20 06:44:15 +02:00
|
|
|
bool isLLSC() const { return testCmdAttrib(IsLlsc); }
|
2014-05-13 19:20:49 +02:00
|
|
|
bool isSWPrefetch() const { return testCmdAttrib(IsSWPrefetch); }
|
|
|
|
bool isHWPrefetch() const { return testCmdAttrib(IsHWPrefetch); }
|
|
|
|
bool isPrefetch() const { return testCmdAttrib(IsSWPrefetch) ||
|
|
|
|
testCmdAttrib(IsHWPrefetch); }
|
2007-06-30 19:16:18 +02:00
|
|
|
bool isError() const { return testCmdAttrib(IsError); }
|
2008-01-02 21:20:15 +01:00
|
|
|
bool isPrint() const { return testCmdAttrib(IsPrint); }
|
2011-03-28 17:49:45 +02:00
|
|
|
bool isFlush() const { return testCmdAttrib(IsFlush); }
|
2007-02-07 19:53:37 +01:00
|
|
|
|
2008-11-10 20:51:17 +01:00
|
|
|
const Command
|
|
|
|
responseCommand() const
|
|
|
|
{
|
2007-02-07 19:53:37 +01:00
|
|
|
return commandInfo[cmd].response;
|
|
|
|
}
|
|
|
|
|
2008-11-10 20:51:17 +01:00
|
|
|
/// Return the string to a cmd given by idx.
|
|
|
|
const std::string &toString() const { return commandInfo[cmd].str; }
|
2007-02-07 19:53:37 +01:00
|
|
|
int toInt() const { return (int)cmd; }
|
|
|
|
|
2008-11-10 20:51:17 +01:00
|
|
|
MemCmd(Command _cmd) : cmd(_cmd) { }
|
|
|
|
MemCmd(int _cmd) : cmd((Command)_cmd) { }
|
|
|
|
MemCmd() : cmd(InvalidCmd) { }
|
2007-02-07 19:53:37 +01:00
|
|
|
|
2008-11-10 20:51:17 +01:00
|
|
|
bool operator==(MemCmd c2) const { return (cmd == c2.cmd); }
|
|
|
|
bool operator!=(MemCmd c2) const { return (cmd != c2.cmd); }
|
2007-02-07 19:53:37 +01:00
|
|
|
};
|
|
|
|
|
2006-01-31 18:12:49 +01:00
|
|
|
/**
|
2006-05-31 04:30:42 +02:00
|
|
|
* A Packet is used to encapsulate a transfer between two objects in
|
|
|
|
* the memory system (e.g., the L1 and L2 cache). (In contrast, a
|
|
|
|
* single Request travels all the way from the requester to the
|
|
|
|
* ultimate destination and back, possibly being conveyed by several
|
|
|
|
* different Packets along the way.)
|
2006-01-31 18:12:49 +01:00
|
|
|
*/
|
2012-06-05 07:23:08 +02:00
|
|
|
class Packet : public Printable
|
2006-01-31 18:12:49 +01:00
|
|
|
{
|
2006-06-29 22:07:19 +02:00
|
|
|
public:
|
2008-11-10 20:51:17 +01:00
|
|
|
typedef uint32_t FlagsType;
|
|
|
|
typedef ::Flags<FlagsType> Flags;
|
|
|
|
|
|
|
|
private:
|
|
|
|
static const FlagsType PUBLIC_FLAGS = 0x00000000;
|
|
|
|
static const FlagsType PRIVATE_FLAGS = 0x00007F0F;
|
|
|
|
static const FlagsType COPY_FLAGS = 0x0000000F;
|
|
|
|
|
|
|
|
static const FlagsType SHARED = 0x00000001;
|
|
|
|
// Special control flags
|
|
|
|
/// Special timing-mode atomic snoop for multi-level coherence.
|
|
|
|
static const FlagsType EXPRESS_SNOOP = 0x00000002;
|
|
|
|
/// Does supplier have exclusive copy?
|
|
|
|
/// Useful for multi-level coherence.
|
|
|
|
static const FlagsType SUPPLY_EXCLUSIVE = 0x00000004;
|
|
|
|
// Snoop response flags
|
|
|
|
static const FlagsType MEM_INHIBIT = 0x00000008;
|
|
|
|
/// Are the 'addr' and 'size' fields valid?
|
|
|
|
static const FlagsType VALID_ADDR = 0x00000100;
|
|
|
|
static const FlagsType VALID_SIZE = 0x00000200;
|
|
|
|
/// Is the data pointer set to a value that shouldn't be freed
|
|
|
|
/// when the packet is destroyed?
|
|
|
|
static const FlagsType STATIC_DATA = 0x00001000;
|
|
|
|
/// The data pointer points to a value that should be freed when
|
2014-12-02 12:07:43 +01:00
|
|
|
/// the packet is destroyed. The pointer is assumed to be pointing
|
|
|
|
/// to an array, and delete [] is consequently called
|
2008-11-10 20:51:17 +01:00
|
|
|
static const FlagsType DYNAMIC_DATA = 0x00002000;
|
2011-07-01 02:49:26 +02:00
|
|
|
/// suppress the error if this packet encounters a functional
|
|
|
|
/// access failure.
|
|
|
|
static const FlagsType SUPPRESS_FUNC_ERROR = 0x00008000;
|
2015-03-27 09:55:54 +01:00
|
|
|
// Signal block present to squash prefetch and cache evict packets
|
|
|
|
// through express snoop flag
|
|
|
|
static const FlagsType BLOCK_CACHED = 0x00010000;
|
2008-11-10 20:51:17 +01:00
|
|
|
|
|
|
|
Flags flags;
|
2007-02-07 19:53:37 +01:00
|
|
|
|
2008-11-10 20:51:17 +01:00
|
|
|
public:
|
2007-02-07 19:53:37 +01:00
|
|
|
typedef MemCmd::Command Command;
|
|
|
|
|
2008-11-10 20:51:17 +01:00
|
|
|
/// The command field of the packet.
|
2007-06-30 19:16:18 +02:00
|
|
|
MemCmd cmd;
|
|
|
|
|
2008-11-10 20:51:17 +01:00
|
|
|
/// A pointer to the original request.
|
2014-12-02 12:07:50 +01:00
|
|
|
const RequestPtr req;
|
2007-06-30 19:16:18 +02:00
|
|
|
|
2006-04-25 01:31:50 +02:00
|
|
|
private:
|
2008-11-10 20:51:17 +01:00
|
|
|
/**
|
|
|
|
* A pointer to the data being transfered. It can be differnt
|
|
|
|
* sizes at each level of the heirarchy so it belongs in the
|
|
|
|
* packet, not request. This may or may not be populated when a
|
|
|
|
* responder recieves the packet. If not populated it memory should
|
|
|
|
* be allocated.
|
2006-04-25 01:31:50 +02:00
|
|
|
*/
|
|
|
|
PacketDataPtr data;
|
|
|
|
|
2008-11-10 20:51:17 +01:00
|
|
|
/// The address of the request. This address could be virtual or
|
|
|
|
/// physical, depending on the system configuration.
|
2006-01-31 18:12:49 +01:00
|
|
|
Addr addr;
|
|
|
|
|
2014-01-24 22:29:30 +01:00
|
|
|
/// True if the request targets the secure memory space.
|
|
|
|
bool _isSecure;
|
|
|
|
|
2008-11-10 20:51:17 +01:00
|
|
|
/// The size of the request or transfer.
|
2009-06-05 08:21:12 +02:00
|
|
|
unsigned size;
|
2006-05-26 20:17:33 +02:00
|
|
|
|
2008-11-10 20:51:17 +01:00
|
|
|
/**
|
|
|
|
* The original value of the command field. Only valid when the
|
2007-06-30 19:16:18 +02:00
|
|
|
* current command field is an error condition; in that case, the
|
|
|
|
* previous contents of the command field are copied here. This
|
|
|
|
* field is *not* set on non-error responses.
|
|
|
|
*/
|
|
|
|
MemCmd origCmd;
|
|
|
|
|
2012-01-10 01:10:05 +01:00
|
|
|
/**
|
2015-03-02 10:00:52 +01:00
|
|
|
* Track the bytes found that satisfy a functional read.
|
2012-01-10 01:10:05 +01:00
|
|
|
*/
|
2015-03-02 10:00:52 +01:00
|
|
|
std::vector<bool> bytesValid;
|
2012-01-10 01:10:05 +01:00
|
|
|
|
2006-05-26 20:17:33 +02:00
|
|
|
public:
|
2006-06-28 20:35:00 +02:00
|
|
|
|
2013-02-19 11:56:06 +01:00
|
|
|
/**
|
2015-02-11 16:23:47 +01:00
|
|
|
* The extra delay from seeing the packet until the header is
|
2014-09-20 23:18:32 +02:00
|
|
|
* transmitted. This delay is used to communicate the crossbar
|
|
|
|
* forwarding latency to the neighbouring object (e.g. a cache)
|
|
|
|
* that actually makes the packet wait. As the delay is relative,
|
|
|
|
* a 32-bit unsigned should be sufficient.
|
2013-02-19 11:56:06 +01:00
|
|
|
*/
|
2015-02-11 16:23:47 +01:00
|
|
|
uint32_t headerDelay;
|
2006-10-10 05:24:21 +02:00
|
|
|
|
2013-02-19 11:56:06 +01:00
|
|
|
/**
|
2015-02-11 16:23:47 +01:00
|
|
|
* The extra pipelining delay from seeing the packet until the end of
|
|
|
|
* payload is transmitted by the component that provided it (if
|
|
|
|
* any). This includes the header delay. Similar to the header
|
|
|
|
* delay, this is used to make up for the fact that the
|
2014-09-20 23:18:32 +02:00
|
|
|
* crossbar does not make the packet wait. As the delay is
|
|
|
|
* relative, a 32-bit unsigned should be sufficient.
|
2013-02-19 11:56:06 +01:00
|
|
|
*/
|
2015-02-11 16:23:47 +01:00
|
|
|
uint32_t payloadDelay;
|
2006-10-10 05:24:21 +02:00
|
|
|
|
2008-11-10 20:51:17 +01:00
|
|
|
/**
|
|
|
|
* A virtual base opaque structure used to hold state associated
|
2013-02-19 11:56:05 +01:00
|
|
|
* with the packet (e.g., an MSHR), specific to a MemObject that
|
|
|
|
* sees the packet. A pointer to this state is returned in the
|
|
|
|
* packet's response so that the MemObject in question can quickly
|
|
|
|
* look up the state needed to process it. A specific subclass
|
|
|
|
* would be derived from this to carry state specific to a
|
|
|
|
* particular sending device.
|
|
|
|
*
|
|
|
|
* As multiple MemObjects may add their SenderState throughout the
|
|
|
|
* memory system, the SenderStates create a stack, where a
|
|
|
|
* MemObject can add a new Senderstate, as long as the
|
|
|
|
* predecessing SenderState is restored when the response comes
|
|
|
|
* back. For this reason, the predecessor should always be
|
|
|
|
* populated with the current SenderState of a packet before
|
|
|
|
* modifying the senderState field in the request packet.
|
2008-11-10 20:51:17 +01:00
|
|
|
*/
|
|
|
|
struct SenderState
|
|
|
|
{
|
2013-02-19 11:56:05 +01:00
|
|
|
SenderState* predecessor;
|
|
|
|
SenderState() : predecessor(NULL) {}
|
2006-05-26 20:17:33 +02:00
|
|
|
virtual ~SenderState() {}
|
|
|
|
};
|
2006-01-31 18:12:49 +01:00
|
|
|
|
2008-01-02 22:46:22 +01:00
|
|
|
/**
|
|
|
|
* Object used to maintain state of a PrintReq. The senderState
|
|
|
|
* field of a PrintReq should always be of this type.
|
|
|
|
*/
|
2012-06-05 07:23:08 +02:00
|
|
|
class PrintReqState : public SenderState
|
2008-11-10 20:51:17 +01:00
|
|
|
{
|
|
|
|
private:
|
|
|
|
/**
|
|
|
|
* An entry in the label stack.
|
|
|
|
*/
|
|
|
|
struct LabelStackEntry
|
|
|
|
{
|
2008-01-02 21:20:15 +01:00
|
|
|
const std::string label;
|
|
|
|
std::string *prefix;
|
|
|
|
bool labelPrinted;
|
2008-11-10 20:51:17 +01:00
|
|
|
LabelStackEntry(const std::string &_label, std::string *_prefix);
|
2008-01-02 21:20:15 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
typedef std::list<LabelStackEntry> LabelStack;
|
|
|
|
LabelStack labelStack;
|
|
|
|
|
|
|
|
std::string *curPrefixPtr;
|
|
|
|
|
|
|
|
public:
|
|
|
|
std::ostream &os;
|
|
|
|
const int verbosity;
|
|
|
|
|
|
|
|
PrintReqState(std::ostream &os, int verbosity = 0);
|
|
|
|
~PrintReqState();
|
|
|
|
|
2008-11-10 20:51:17 +01:00
|
|
|
/**
|
|
|
|
* Returns the current line prefix.
|
|
|
|
*/
|
2008-01-02 21:20:15 +01:00
|
|
|
const std::string &curPrefix() { return *curPrefixPtr; }
|
2008-01-02 22:46:22 +01:00
|
|
|
|
2008-11-10 20:51:17 +01:00
|
|
|
/**
|
|
|
|
* Push a label onto the label stack, and prepend the given
|
2008-01-02 22:46:22 +01:00
|
|
|
* prefix string onto the current prefix. Labels will only be
|
2008-11-10 20:51:17 +01:00
|
|
|
* printed if an object within the label's scope is printed.
|
|
|
|
*/
|
2008-01-02 21:20:15 +01:00
|
|
|
void pushLabel(const std::string &lbl,
|
|
|
|
const std::string &prefix = " ");
|
2008-11-10 20:51:17 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Pop a label off the label stack.
|
|
|
|
*/
|
2008-01-02 21:20:15 +01:00
|
|
|
void popLabel();
|
2008-11-10 20:51:17 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Print all of the pending unprinted labels on the
|
2008-01-02 22:46:22 +01:00
|
|
|
* stack. Called by printObj(), so normally not called by
|
2008-11-10 20:51:17 +01:00
|
|
|
* users unless bypassing printObj().
|
|
|
|
*/
|
2008-01-02 21:20:15 +01:00
|
|
|
void printLabels();
|
2008-11-10 20:51:17 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Print a Printable object to os, because it matched the
|
|
|
|
* address on a PrintReq.
|
|
|
|
*/
|
2008-01-02 21:20:15 +01:00
|
|
|
void printObj(Printable *obj);
|
|
|
|
};
|
|
|
|
|
2008-11-10 20:51:17 +01:00
|
|
|
/**
|
|
|
|
* This packet's sender state. Devices should use dynamic_cast<>
|
|
|
|
* to cast to the state appropriate to the sender. The intent of
|
|
|
|
* this variable is to allow a device to attach extra information
|
2013-02-19 11:56:05 +01:00
|
|
|
* to a request. A response packet must return the sender state
|
2008-11-10 20:51:17 +01:00
|
|
|
* that was attached to the original request (even if a new packet
|
|
|
|
* is created).
|
|
|
|
*/
|
2006-05-31 04:30:42 +02:00
|
|
|
SenderState *senderState;
|
2006-01-31 18:12:49 +01:00
|
|
|
|
2013-02-19 11:56:05 +01:00
|
|
|
/**
|
|
|
|
* Push a new sender state to the packet and make the current
|
|
|
|
* sender state the predecessor of the new one. This should be
|
|
|
|
* prefered over direct manipulation of the senderState member
|
|
|
|
* variable.
|
|
|
|
*
|
|
|
|
* @param sender_state SenderState to push at the top of the stack
|
|
|
|
*/
|
|
|
|
void pushSenderState(SenderState *sender_state);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Pop the top of the state stack and return a pointer to it. This
|
|
|
|
* assumes the current sender state is not NULL. This should be
|
|
|
|
* preferred over direct manipulation of the senderState member
|
|
|
|
* variable.
|
|
|
|
*
|
|
|
|
* @return The current top of the stack
|
|
|
|
*/
|
|
|
|
SenderState *popSenderState();
|
|
|
|
|
2013-02-19 11:56:06 +01:00
|
|
|
/**
|
|
|
|
* Go through the sender state stack and return the first instance
|
|
|
|
* that is of type T (as determined by a dynamic_cast). If there
|
|
|
|
* is no sender state of type T, NULL is returned.
|
|
|
|
*
|
|
|
|
* @return The topmost state of type T
|
|
|
|
*/
|
|
|
|
template <typename T>
|
|
|
|
T * findNextSenderState() const
|
|
|
|
{
|
|
|
|
T *t = NULL;
|
|
|
|
SenderState* sender_state = senderState;
|
|
|
|
while (t == NULL && sender_state != NULL) {
|
|
|
|
t = dynamic_cast<T*>(sender_state);
|
|
|
|
sender_state = sender_state->predecessor;
|
|
|
|
}
|
|
|
|
return t;
|
|
|
|
}
|
|
|
|
|
2008-11-10 20:51:17 +01:00
|
|
|
/// Return the string name of the cmd field (for debugging and
|
|
|
|
/// tracing).
|
2007-02-07 19:53:37 +01:00
|
|
|
const std::string &cmdString() const { return cmd.toString(); }
|
2006-06-28 20:35:00 +02:00
|
|
|
|
2008-11-10 20:51:17 +01:00
|
|
|
/// Return the index of this command.
|
2007-02-07 19:53:37 +01:00
|
|
|
inline int cmdToIndex() const { return cmd.toInt(); }
|
2006-06-28 20:35:00 +02:00
|
|
|
|
2014-12-02 12:08:19 +01:00
|
|
|
bool isRead() const { return cmd.isRead(); }
|
|
|
|
bool isWrite() const { return cmd.isWrite(); }
|
|
|
|
bool isUpgrade() const { return cmd.isUpgrade(); }
|
|
|
|
bool isRequest() const { return cmd.isRequest(); }
|
|
|
|
bool isResponse() const { return cmd.isResponse(); }
|
|
|
|
bool needsExclusive() const { return cmd.needsExclusive(); }
|
|
|
|
bool needsResponse() const { return cmd.needsResponse(); }
|
|
|
|
bool isInvalidate() const { return cmd.isInvalidate(); }
|
|
|
|
bool hasData() const { return cmd.hasData(); }
|
|
|
|
bool isLLSC() const { return cmd.isLLSC(); }
|
|
|
|
bool isError() const { return cmd.isError(); }
|
|
|
|
bool isPrint() const { return cmd.isPrint(); }
|
|
|
|
bool isFlush() const { return cmd.isFlush(); }
|
2007-06-30 19:16:18 +02:00
|
|
|
|
|
|
|
// Snoop flags
|
2014-12-02 12:07:46 +01:00
|
|
|
void assertMemInhibit()
|
|
|
|
{
|
|
|
|
assert(isRequest());
|
|
|
|
assert(!flags.isSet(MEM_INHIBIT));
|
|
|
|
flags.set(MEM_INHIBIT);
|
|
|
|
}
|
2013-10-31 19:41:13 +01:00
|
|
|
bool memInhibitAsserted() const { return flags.isSet(MEM_INHIBIT); }
|
|
|
|
void assertShared() { flags.set(SHARED); }
|
|
|
|
bool sharedAsserted() const { return flags.isSet(SHARED); }
|
2007-06-30 19:16:18 +02:00
|
|
|
|
2007-07-16 05:11:06 +02:00
|
|
|
// Special control flags
|
2013-10-31 19:41:13 +01:00
|
|
|
void setExpressSnoop() { flags.set(EXPRESS_SNOOP); }
|
|
|
|
bool isExpressSnoop() const { return flags.isSet(EXPRESS_SNOOP); }
|
|
|
|
void setSupplyExclusive() { flags.set(SUPPLY_EXCLUSIVE); }
|
|
|
|
void clearSupplyExclusive() { flags.clear(SUPPLY_EXCLUSIVE); }
|
|
|
|
bool isSupplyExclusive() const { return flags.isSet(SUPPLY_EXCLUSIVE); }
|
|
|
|
void setSuppressFuncError() { flags.set(SUPPRESS_FUNC_ERROR); }
|
|
|
|
bool suppressFuncError() const { return flags.isSet(SUPPRESS_FUNC_ERROR); }
|
2015-03-27 09:55:54 +01:00
|
|
|
void setBlockCached() { flags.set(BLOCK_CACHED); }
|
|
|
|
bool isBlockCached() const { return flags.isSet(BLOCK_CACHED); }
|
2015-07-03 16:14:37 +02:00
|
|
|
void clearBlockCached() { flags.clear(BLOCK_CACHED); }
|
2007-07-16 05:11:06 +02:00
|
|
|
|
2007-06-30 19:16:18 +02:00
|
|
|
// Network error conditions... encapsulate them as methods since
|
|
|
|
// their encoding keeps changing (from result field to command
|
|
|
|
// field, etc.)
|
2008-11-10 20:51:17 +01:00
|
|
|
void
|
|
|
|
setBadAddress()
|
|
|
|
{
|
|
|
|
assert(isResponse());
|
|
|
|
cmd = MemCmd::BadAddressError;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool hadBadAddress() const { return cmd == MemCmd::BadAddressError; }
|
2007-08-27 06:45:40 +02:00
|
|
|
void copyError(Packet *pkt) { assert(pkt->isError()); cmd = pkt->cmd; }
|
2006-06-30 16:25:25 +02:00
|
|
|
|
2008-12-06 23:18:18 +01:00
|
|
|
Addr getAddr() const { assert(flags.isSet(VALID_ADDR)); return addr; }
|
2012-09-25 18:49:40 +02:00
|
|
|
/**
|
|
|
|
* Update the address of this packet mid-transaction. This is used
|
|
|
|
* by the address mapper to change an already set address to a new
|
|
|
|
* one based on the system configuration. It is intended to remap
|
|
|
|
* an existing address, so it asserts that the current address is
|
|
|
|
* valid.
|
|
|
|
*/
|
|
|
|
void setAddr(Addr _addr) { assert(flags.isSet(VALID_ADDR)); addr = _addr; }
|
|
|
|
|
2009-06-05 08:21:12 +02:00
|
|
|
unsigned getSize() const { assert(flags.isSet(VALID_SIZE)); return size; }
|
2008-11-10 20:51:17 +01:00
|
|
|
Addr getOffset(int blkSize) const { return getAddr() & (Addr)(blkSize - 1); }
|
|
|
|
|
2014-01-24 22:29:30 +01:00
|
|
|
bool isSecure() const
|
|
|
|
{
|
|
|
|
assert(flags.isSet(VALID_ADDR));
|
|
|
|
return _isSecure;
|
|
|
|
}
|
|
|
|
|
2010-08-20 20:46:12 +02:00
|
|
|
/**
|
|
|
|
* It has been determined that the SC packet should successfully update
|
|
|
|
* memory. Therefore, convert this SC packet to a normal write.
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
convertScToWrite()
|
|
|
|
{
|
|
|
|
assert(isLLSC());
|
|
|
|
assert(isWrite());
|
|
|
|
cmd = MemCmd::WriteReq;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* When ruby is in use, Ruby will monitor the cache line and thus M5
|
|
|
|
* phys memory should treat LL ops as normal reads.
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
convertLlToRead()
|
|
|
|
{
|
|
|
|
assert(isLLSC());
|
|
|
|
assert(isRead());
|
|
|
|
cmd = MemCmd::ReadReq;
|
|
|
|
}
|
|
|
|
|
2008-11-10 20:51:17 +01:00
|
|
|
/**
|
|
|
|
* Constructor. Note that a Request object must be constructed
|
|
|
|
* first, but the Requests's physical address and size fields need
|
2012-05-30 11:29:07 +02:00
|
|
|
* not be valid. The command must be supplied.
|
2008-11-10 20:51:17 +01:00
|
|
|
*/
|
2014-12-02 12:07:50 +01:00
|
|
|
Packet(const RequestPtr _req, MemCmd _cmd)
|
2014-09-09 10:36:31 +02:00
|
|
|
: cmd(_cmd), req(_req), data(nullptr), addr(0), _isSecure(false),
|
2015-03-02 10:00:52 +01:00
|
|
|
size(0), headerDelay(0), payloadDelay(0),
|
2013-02-19 11:56:06 +01:00
|
|
|
senderState(NULL)
|
2006-06-29 22:07:19 +02:00
|
|
|
{
|
2009-04-21 03:40:00 +02:00
|
|
|
if (req->hasPaddr()) {
|
|
|
|
addr = req->getPaddr();
|
|
|
|
flags.set(VALID_ADDR);
|
2014-01-24 22:29:30 +01:00
|
|
|
_isSecure = req->isSecure();
|
2009-04-21 03:40:00 +02:00
|
|
|
}
|
|
|
|
if (req->hasSize()) {
|
|
|
|
size = req->getSize();
|
|
|
|
flags.set(VALID_SIZE);
|
|
|
|
}
|
2006-06-29 22:07:19 +02:00
|
|
|
}
|
|
|
|
|
2008-11-10 20:51:17 +01:00
|
|
|
/**
|
|
|
|
* Alternate constructor if you are trying to create a packet with
|
|
|
|
* a request that is for a whole block, not the address from the
|
|
|
|
* req. this allows for overriding the size/addr of the req.
|
|
|
|
*/
|
2014-12-02 12:07:50 +01:00
|
|
|
Packet(const RequestPtr _req, MemCmd _cmd, int _blkSize)
|
2014-09-09 10:36:31 +02:00
|
|
|
: cmd(_cmd), req(_req), data(nullptr), addr(0), _isSecure(false),
|
2015-02-11 16:23:47 +01:00
|
|
|
headerDelay(0), payloadDelay(0),
|
2013-02-19 11:56:06 +01:00
|
|
|
senderState(NULL)
|
2006-05-26 20:17:33 +02:00
|
|
|
{
|
2009-04-21 03:40:00 +02:00
|
|
|
if (req->hasPaddr()) {
|
|
|
|
addr = req->getPaddr() & ~(_blkSize - 1);
|
|
|
|
flags.set(VALID_ADDR);
|
2014-01-24 22:29:30 +01:00
|
|
|
_isSecure = req->isSecure();
|
2009-04-21 03:40:00 +02:00
|
|
|
}
|
|
|
|
size = _blkSize;
|
|
|
|
flags.set(VALID_SIZE);
|
2007-06-18 02:27:53 +02:00
|
|
|
}
|
|
|
|
|
2008-11-10 20:51:17 +01:00
|
|
|
/**
|
|
|
|
* Alternate constructor for copying a packet. Copy all fields
|
2007-07-14 22:14:53 +02:00
|
|
|
* *except* if the original packet's data was dynamic, don't copy
|
|
|
|
* that, as we can't guarantee that the new packet's lifetime is
|
|
|
|
* less than that of the original packet. In this case the new
|
2008-11-10 20:51:17 +01:00
|
|
|
* packet should allocate its own data.
|
|
|
|
*/
|
2014-12-02 12:07:54 +01:00
|
|
|
Packet(PacketPtr pkt, bool clear_flags, bool alloc_data)
|
2008-11-10 20:51:17 +01:00
|
|
|
: cmd(pkt->cmd), req(pkt->req),
|
2014-12-02 12:07:54 +01:00
|
|
|
data(nullptr),
|
2014-01-24 22:29:30 +01:00
|
|
|
addr(pkt->addr), _isSecure(pkt->_isSecure), size(pkt->size),
|
2015-03-02 10:00:52 +01:00
|
|
|
bytesValid(pkt->bytesValid),
|
2015-02-11 16:23:47 +01:00
|
|
|
headerDelay(pkt->headerDelay),
|
|
|
|
payloadDelay(pkt->payloadDelay),
|
2013-02-19 11:56:06 +01:00
|
|
|
senderState(pkt->senderState)
|
2007-06-18 02:27:53 +02:00
|
|
|
{
|
2014-12-02 12:07:54 +01:00
|
|
|
if (!clear_flags)
|
2008-11-10 20:51:17 +01:00
|
|
|
flags.set(pkt->flags & COPY_FLAGS);
|
|
|
|
|
2012-05-30 11:29:42 +02:00
|
|
|
flags.set(pkt->flags & (VALID_ADDR|VALID_SIZE));
|
2014-12-02 12:07:41 +01:00
|
|
|
|
2014-12-02 12:07:54 +01:00
|
|
|
// should we allocate space for data, or not, the express
|
|
|
|
// snoops do not need to carry any data as they only serve to
|
|
|
|
// co-ordinate state changes
|
|
|
|
if (alloc_data) {
|
|
|
|
// even if asked to allocate data, if the original packet
|
|
|
|
// holds static data, then the sender will not be doing
|
|
|
|
// any memcpy on receiving the response, thus we simply
|
|
|
|
// carry the pointer forward
|
|
|
|
if (pkt->flags.isSet(STATIC_DATA)) {
|
|
|
|
data = pkt->data;
|
|
|
|
flags.set(STATIC_DATA);
|
|
|
|
} else {
|
|
|
|
allocate();
|
|
|
|
}
|
|
|
|
}
|
2014-05-13 19:20:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2015-02-11 19:48:50 +01:00
|
|
|
* Generate the appropriate read MemCmd based on the Request flags.
|
2014-05-13 19:20:48 +02:00
|
|
|
*/
|
2015-02-11 19:48:50 +01:00
|
|
|
static MemCmd
|
|
|
|
makeReadCmd(const RequestPtr req)
|
2014-05-13 19:20:48 +02:00
|
|
|
{
|
2015-02-11 19:48:50 +01:00
|
|
|
if (req->isLLSC())
|
|
|
|
return MemCmd::LoadLockedReq;
|
|
|
|
else if (req->isPrefetch())
|
|
|
|
return MemCmd::SoftPFReq;
|
|
|
|
else
|
|
|
|
return MemCmd::ReadReq;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Generate the appropriate write MemCmd based on the Request flags.
|
|
|
|
*/
|
|
|
|
static MemCmd
|
|
|
|
makeWriteCmd(const RequestPtr req)
|
|
|
|
{
|
|
|
|
if (req->isLLSC())
|
|
|
|
return MemCmd::StoreCondReq;
|
|
|
|
else if (req->isSwap())
|
|
|
|
return MemCmd::SwapReq;
|
|
|
|
else
|
|
|
|
return MemCmd::WriteReq;
|
2014-05-13 19:20:48 +02:00
|
|
|
}
|
2012-01-10 01:10:05 +01:00
|
|
|
|
2014-05-13 19:20:48 +02:00
|
|
|
/**
|
|
|
|
* Constructor-like methods that return Packets based on Request objects.
|
2015-02-11 19:48:50 +01:00
|
|
|
* Fine-tune the MemCmd type if it's not a vanilla read or write.
|
2014-05-13 19:20:48 +02:00
|
|
|
*/
|
|
|
|
static PacketPtr
|
2014-12-02 12:07:50 +01:00
|
|
|
createRead(const RequestPtr req)
|
2014-05-13 19:20:48 +02:00
|
|
|
{
|
2015-02-11 19:48:50 +01:00
|
|
|
return new Packet(req, makeReadCmd(req));
|
2014-05-13 19:20:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static PacketPtr
|
2014-12-02 12:07:50 +01:00
|
|
|
createWrite(const RequestPtr req)
|
2014-05-13 19:20:48 +02:00
|
|
|
{
|
2015-02-11 19:48:50 +01:00
|
|
|
return new Packet(req, makeWriteCmd(req));
|
2006-05-26 20:17:33 +02:00
|
|
|
}
|
2006-04-12 23:46:25 +02:00
|
|
|
|
2008-11-10 20:51:17 +01:00
|
|
|
/**
|
|
|
|
* clean up packet variables
|
|
|
|
*/
|
2006-04-25 01:31:50 +02:00
|
|
|
~Packet()
|
2008-03-24 06:08:02 +01:00
|
|
|
{
|
2015-06-09 15:21:18 +02:00
|
|
|
// Delete the request object if this is a request packet which
|
|
|
|
// does not need a response, because the requester will not get
|
|
|
|
// a chance. If the request packet needs a response then the
|
|
|
|
// request will be deleted on receipt of the response
|
|
|
|
// packet. We also make sure to never delete the request for
|
|
|
|
// express snoops, even for cases when responses are not
|
|
|
|
// needed (CleanEvict and Writeback), since the snoop packet
|
|
|
|
// re-uses the same request.
|
|
|
|
if (req && isRequest() && !needsResponse() &&
|
|
|
|
!isExpressSnoop()) {
|
2008-03-24 06:08:02 +01:00
|
|
|
delete req;
|
2015-06-09 15:21:18 +02:00
|
|
|
}
|
2008-11-10 20:51:17 +01:00
|
|
|
deleteData();
|
2008-03-24 06:08:02 +01:00
|
|
|
}
|
2006-04-25 01:31:50 +02:00
|
|
|
|
2007-06-18 02:27:53 +02:00
|
|
|
/**
|
|
|
|
* Take a request packet and modify it in place to be suitable for
|
2015-01-22 11:01:31 +01:00
|
|
|
* returning as a response to that request.
|
2007-06-18 02:27:53 +02:00
|
|
|
*/
|
2008-11-10 20:51:17 +01:00
|
|
|
void
|
|
|
|
makeResponse()
|
2007-06-18 02:27:53 +02:00
|
|
|
{
|
2006-05-31 04:30:42 +02:00
|
|
|
assert(needsResponse());
|
2006-07-06 22:52:05 +02:00
|
|
|
assert(isRequest());
|
2007-08-27 06:45:40 +02:00
|
|
|
origCmd = cmd;
|
2007-02-07 19:53:37 +01:00
|
|
|
cmd = cmd.responseCommand();
|
2008-11-14 13:55:30 +01:00
|
|
|
|
2010-06-17 00:25:57 +02:00
|
|
|
// responses are never express, even if the snoop that
|
|
|
|
// triggered them was
|
|
|
|
flags.clear(EXPRESS_SNOOP);
|
2007-06-18 02:27:53 +02:00
|
|
|
}
|
|
|
|
|
2008-11-10 20:51:17 +01:00
|
|
|
void
|
|
|
|
makeAtomicResponse()
|
2007-06-18 02:27:53 +02:00
|
|
|
{
|
2007-06-30 19:16:18 +02:00
|
|
|
makeResponse();
|
2006-05-26 20:17:33 +02:00
|
|
|
}
|
|
|
|
|
2008-11-10 20:51:17 +01:00
|
|
|
void
|
|
|
|
makeTimingResponse()
|
2006-10-20 08:38:45 +02:00
|
|
|
{
|
2007-06-30 19:16:18 +02:00
|
|
|
makeResponse();
|
2006-10-06 05:28:03 +02:00
|
|
|
}
|
|
|
|
|
2011-07-01 02:49:26 +02:00
|
|
|
void
|
|
|
|
setFunctionalResponseStatus(bool success)
|
|
|
|
{
|
|
|
|
if (!success) {
|
|
|
|
if (isWrite()) {
|
|
|
|
cmd = MemCmd::FunctionalWriteError;
|
|
|
|
} else {
|
|
|
|
cmd = MemCmd::FunctionalReadError;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-03-13 02:31:08 +01:00
|
|
|
void
|
|
|
|
setSize(unsigned size)
|
|
|
|
{
|
|
|
|
assert(!flags.isSet(VALID_SIZE));
|
|
|
|
|
|
|
|
this->size = size;
|
|
|
|
flags.set(VALID_SIZE);
|
|
|
|
}
|
|
|
|
|
2006-06-09 01:43:50 +02:00
|
|
|
|
2006-10-20 08:38:45 +02:00
|
|
|
/**
|
|
|
|
* Set the data pointer to the following value that should not be
|
2014-12-02 12:07:54 +01:00
|
|
|
* freed. Static data allows us to do a single memcpy even if
|
|
|
|
* multiple packets are required to get from source to destination
|
|
|
|
* and back. In essence the pointer is set calling dataStatic on
|
|
|
|
* the original packet, and whenever this packet is copied and
|
|
|
|
* forwarded the same pointer is passed on. When a packet
|
|
|
|
* eventually reaches the destination holding the data, it is
|
|
|
|
* copied once into the location originally set. On the way back
|
|
|
|
* to the source, no copies are necessary.
|
2006-10-20 08:38:45 +02:00
|
|
|
*/
|
2006-04-25 01:31:50 +02:00
|
|
|
template <typename T>
|
2006-10-20 08:38:45 +02:00
|
|
|
void
|
|
|
|
dataStatic(T *p)
|
|
|
|
{
|
2014-12-02 12:07:43 +01:00
|
|
|
assert(flags.noneSet(STATIC_DATA|DYNAMIC_DATA));
|
2006-10-20 08:38:45 +02:00
|
|
|
data = (PacketDataPtr)p;
|
2008-11-10 20:51:17 +01:00
|
|
|
flags.set(STATIC_DATA);
|
2006-10-20 08:38:45 +02:00
|
|
|
}
|
2006-04-25 01:31:50 +02:00
|
|
|
|
2014-12-02 12:07:38 +01:00
|
|
|
/**
|
|
|
|
* Set the data pointer to the following value that should not be
|
|
|
|
* freed. This version of the function allows the pointer passed
|
|
|
|
* to us to be const. To avoid issues down the line we cast the
|
|
|
|
* constness away, the alternative would be to keep both a const
|
|
|
|
* and non-const data pointer and cleverly choose between
|
|
|
|
* them. Note that this is only allowed for static data.
|
|
|
|
*/
|
|
|
|
template <typename T>
|
|
|
|
void
|
|
|
|
dataStaticConst(const T *p)
|
|
|
|
{
|
2014-12-02 12:07:43 +01:00
|
|
|
assert(flags.noneSet(STATIC_DATA|DYNAMIC_DATA));
|
2014-12-02 12:07:38 +01:00
|
|
|
data = const_cast<PacketDataPtr>(p);
|
|
|
|
flags.set(STATIC_DATA);
|
|
|
|
}
|
|
|
|
|
2006-10-20 08:38:45 +02:00
|
|
|
/**
|
|
|
|
* Set the data pointer to a value that should have delete []
|
2014-12-02 12:07:54 +01:00
|
|
|
* called on it. Dynamic data is local to this packet, and as the
|
|
|
|
* packet travels from source to destination, forwarded packets
|
|
|
|
* will allocate their own data. When a packet reaches the final
|
|
|
|
* destination it will populate the dynamic data of that specific
|
|
|
|
* packet, and on the way back towards the source, memcpy will be
|
|
|
|
* invoked in every step where a new packet was created e.g. in
|
|
|
|
* the caches. Ultimately when the response reaches the source a
|
|
|
|
* final memcpy is needed to extract the data from the packet
|
|
|
|
* before it is deallocated.
|
2006-04-25 01:31:50 +02:00
|
|
|
*/
|
|
|
|
template <typename T>
|
2006-10-20 08:38:45 +02:00
|
|
|
void
|
|
|
|
dataDynamic(T *p)
|
|
|
|
{
|
2014-12-02 12:07:43 +01:00
|
|
|
assert(flags.noneSet(STATIC_DATA|DYNAMIC_DATA));
|
2006-10-20 08:38:45 +02:00
|
|
|
data = (PacketDataPtr)p;
|
2008-11-10 20:51:17 +01:00
|
|
|
flags.set(DYNAMIC_DATA);
|
2006-10-20 08:38:45 +02:00
|
|
|
}
|
2006-04-25 01:31:50 +02:00
|
|
|
|
2008-11-10 20:51:17 +01:00
|
|
|
/**
|
|
|
|
* get a pointer to the data ptr.
|
|
|
|
*/
|
2006-04-25 01:31:50 +02:00
|
|
|
template <typename T>
|
2006-10-20 08:38:45 +02:00
|
|
|
T*
|
2014-12-02 12:07:34 +01:00
|
|
|
getPtr()
|
2006-10-20 08:38:45 +02:00
|
|
|
{
|
2014-12-02 12:07:34 +01:00
|
|
|
assert(flags.isSet(STATIC_DATA|DYNAMIC_DATA));
|
2006-10-20 08:38:45 +02:00
|
|
|
return (T*)data;
|
|
|
|
}
|
2006-04-25 01:31:50 +02:00
|
|
|
|
2014-12-02 12:07:36 +01:00
|
|
|
template <typename T>
|
|
|
|
const T*
|
|
|
|
getConstPtr() const
|
|
|
|
{
|
|
|
|
assert(flags.isSet(STATIC_DATA|DYNAMIC_DATA));
|
|
|
|
return (const T*)data;
|
|
|
|
}
|
|
|
|
|
2008-11-10 20:51:17 +01:00
|
|
|
/**
|
|
|
|
* return the value of what is pointed to in the packet.
|
|
|
|
*/
|
2006-04-25 01:31:50 +02:00
|
|
|
template <typename T>
|
2014-12-02 12:07:36 +01:00
|
|
|
T get() const;
|
2006-04-25 01:31:50 +02:00
|
|
|
|
2008-11-10 20:51:17 +01:00
|
|
|
/**
|
|
|
|
* set the value in the data pointer to v.
|
|
|
|
*/
|
2006-04-25 01:31:50 +02:00
|
|
|
template <typename T>
|
2006-05-02 00:53:28 +02:00
|
|
|
void set(T v);
|
2006-04-25 01:31:50 +02:00
|
|
|
|
2007-06-18 02:27:53 +02:00
|
|
|
/**
|
|
|
|
* Copy data into the packet from the provided pointer.
|
|
|
|
*/
|
2008-11-10 20:51:17 +01:00
|
|
|
void
|
2014-12-02 12:07:36 +01:00
|
|
|
setData(const uint8_t *p)
|
2007-06-18 02:27:53 +02:00
|
|
|
{
|
2014-12-02 12:07:54 +01:00
|
|
|
// we should never be copying data onto itself, which means we
|
|
|
|
// must idenfity packets with static data, as they carry the
|
|
|
|
// same pointer from source to destination and back
|
|
|
|
assert(p != getPtr<uint8_t>() || flags.isSet(STATIC_DATA));
|
|
|
|
|
2010-09-30 16:35:19 +02:00
|
|
|
if (p != getPtr<uint8_t>())
|
2014-12-02 12:07:54 +01:00
|
|
|
// for packet with allocated dynamic data, we copy data from
|
|
|
|
// one to the other, e.g. a forwarded response to a response
|
2010-09-30 16:35:19 +02:00
|
|
|
std::memcpy(getPtr<uint8_t>(), p, getSize());
|
2007-06-18 02:27:53 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Copy data into the packet from the provided block pointer,
|
|
|
|
* which is aligned to the given block size.
|
|
|
|
*/
|
2008-11-10 20:51:17 +01:00
|
|
|
void
|
2014-12-02 12:07:36 +01:00
|
|
|
setDataFromBlock(const uint8_t *blk_data, int blkSize)
|
2007-06-18 02:27:53 +02:00
|
|
|
{
|
|
|
|
setData(blk_data + getOffset(blkSize));
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Copy data from the packet to the provided block pointer, which
|
|
|
|
* is aligned to the given block size.
|
|
|
|
*/
|
2008-11-10 20:51:17 +01:00
|
|
|
void
|
2014-12-02 12:07:36 +01:00
|
|
|
writeData(uint8_t *p) const
|
2007-06-18 02:27:53 +02:00
|
|
|
{
|
2014-12-02 12:07:36 +01:00
|
|
|
std::memcpy(p, getConstPtr<uint8_t>(), getSize());
|
2007-06-18 02:27:53 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Copy data from the packet to the memory at the provided pointer.
|
|
|
|
*/
|
2008-11-10 20:51:17 +01:00
|
|
|
void
|
2014-12-02 12:07:36 +01:00
|
|
|
writeDataToBlock(uint8_t *blk_data, int blkSize) const
|
2007-06-18 02:27:53 +02:00
|
|
|
{
|
|
|
|
writeData(blk_data + getOffset(blkSize));
|
|
|
|
}
|
|
|
|
|
2006-10-20 08:38:45 +02:00
|
|
|
/**
|
|
|
|
* delete the data pointed to in the data pointer. Ok to call to
|
|
|
|
* matter how data was allocted.
|
|
|
|
*/
|
2008-11-10 20:51:17 +01:00
|
|
|
void
|
|
|
|
deleteData()
|
|
|
|
{
|
2014-12-02 12:07:43 +01:00
|
|
|
if (flags.isSet(DYNAMIC_DATA))
|
2008-11-10 20:51:17 +01:00
|
|
|
delete [] data;
|
|
|
|
|
2014-12-02 12:07:43 +01:00
|
|
|
flags.clear(STATIC_DATA|DYNAMIC_DATA);
|
2008-11-10 20:51:17 +01:00
|
|
|
data = NULL;
|
|
|
|
}
|
2006-04-25 01:31:50 +02:00
|
|
|
|
2014-12-02 12:07:41 +01:00
|
|
|
/** Allocate memory for the packet. */
|
2008-11-10 20:51:17 +01:00
|
|
|
void
|
|
|
|
allocate()
|
|
|
|
{
|
2008-12-06 23:18:18 +01:00
|
|
|
assert(flags.noneSet(STATIC_DATA|DYNAMIC_DATA));
|
2014-12-02 12:07:43 +01:00
|
|
|
flags.set(DYNAMIC_DATA);
|
2008-11-14 13:55:30 +01:00
|
|
|
data = new uint8_t[getSize()];
|
2008-11-10 20:51:17 +01:00
|
|
|
}
|
|
|
|
|
2007-06-18 02:27:53 +02:00
|
|
|
/**
|
|
|
|
* Check a functional request against a memory value stored in
|
2014-12-02 12:07:52 +01:00
|
|
|
* another packet (i.e. an in-transit request or
|
|
|
|
* response). Returns true if the current packet is a read, and
|
|
|
|
* the other packet provides the data, which is then copied to the
|
|
|
|
* current packet. If the current packet is a write, and the other
|
|
|
|
* packet intersects this one, then we update the data
|
|
|
|
* accordingly.
|
2007-06-18 02:27:53 +02:00
|
|
|
*/
|
2008-11-10 20:51:17 +01:00
|
|
|
bool
|
2014-12-02 12:07:52 +01:00
|
|
|
checkFunctional(PacketPtr other)
|
2008-11-10 20:51:17 +01:00
|
|
|
{
|
2014-12-02 12:07:52 +01:00
|
|
|
// all packets that are carrying a payload should have a valid
|
|
|
|
// data pointer
|
2014-01-24 22:29:30 +01:00
|
|
|
return checkFunctional(other, other->getAddr(), other->isSecure(),
|
2014-12-02 12:07:52 +01:00
|
|
|
other->getSize(),
|
|
|
|
other->hasData() ?
|
|
|
|
other->getPtr<uint8_t>() : NULL);
|
2007-06-18 02:27:53 +02:00
|
|
|
}
|
2006-01-31 18:12:49 +01:00
|
|
|
|
2015-07-03 16:14:37 +02:00
|
|
|
/**
|
|
|
|
* Is this request notification of a clean or dirty eviction from the cache.
|
|
|
|
**/
|
|
|
|
bool
|
|
|
|
evictingBlock() const
|
|
|
|
{
|
|
|
|
return (cmd == MemCmd::Writeback ||
|
|
|
|
cmd == MemCmd::CleanEvict);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Does the request need to check for cached copies of the same block
|
|
|
|
* in the memory hierarchy above.
|
|
|
|
**/
|
|
|
|
bool
|
|
|
|
mustCheckAbove() const
|
|
|
|
{
|
|
|
|
return (cmd == MemCmd::HardPFReq ||
|
|
|
|
evictingBlock());
|
|
|
|
}
|
|
|
|
|
2014-12-02 12:07:52 +01:00
|
|
|
/**
|
|
|
|
* Check a functional request against a memory value represented
|
|
|
|
* by a base/size pair and an associated data array. If the
|
|
|
|
* current packet is a read, it may be satisfied by the memory
|
|
|
|
* value. If the current packet is a write, it may update the
|
|
|
|
* memory value.
|
|
|
|
*/
|
|
|
|
bool
|
|
|
|
checkFunctional(Printable *obj, Addr base, bool is_secure, int size,
|
|
|
|
uint8_t *_data);
|
|
|
|
|
2008-01-02 22:46:22 +01:00
|
|
|
/**
|
|
|
|
* Push label for PrintReq (safe to call unconditionally).
|
|
|
|
*/
|
2008-11-10 20:51:17 +01:00
|
|
|
void
|
|
|
|
pushLabel(const std::string &lbl)
|
|
|
|
{
|
|
|
|
if (isPrint())
|
|
|
|
safe_cast<PrintReqState*>(senderState)->pushLabel(lbl);
|
2008-01-02 21:20:15 +01:00
|
|
|
}
|
|
|
|
|
2008-01-02 22:46:22 +01:00
|
|
|
/**
|
|
|
|
* Pop label for PrintReq (safe to call unconditionally).
|
|
|
|
*/
|
2008-11-10 20:51:17 +01:00
|
|
|
void
|
|
|
|
popLabel()
|
|
|
|
{
|
|
|
|
if (isPrint())
|
|
|
|
safe_cast<PrintReqState*>(senderState)->popLabel();
|
2008-01-02 21:20:15 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void print(std::ostream &o, int verbosity = 0,
|
|
|
|
const std::string &prefix = "") const;
|
2013-04-22 19:20:33 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* A no-args wrapper of print(std::ostream...)
|
|
|
|
* meant to be invoked from DPRINTFs
|
|
|
|
* avoiding string overheads in fast mode
|
|
|
|
* @return string with the request's type and start<->end addresses
|
|
|
|
*/
|
|
|
|
std::string print() const;
|
2008-01-02 21:20:15 +01:00
|
|
|
};
|
2006-10-12 20:15:09 +02:00
|
|
|
|
2006-01-31 18:12:49 +01:00
|
|
|
#endif //__MEM_PACKET_HH
|