Merge branch 'master' into new-llc
Conflicts: src/main/scala/coherence.scala src/main/scala/memserdes.scala src/main/scala/tilelink.scala
This commit is contained in:
commit
a519a43f23
@ -99,6 +99,7 @@ class MetadataArray[T <: Metadata](makeRstVal: () => T) extends CacheModule {
|
|||||||
|
|
||||||
abstract trait L2HellaCacheParameters extends CacheParameters
|
abstract trait L2HellaCacheParameters extends CacheParameters
|
||||||
with CoherenceAgentParameters
|
with CoherenceAgentParameters
|
||||||
|
with TileLinkParameters
|
||||||
|
|
||||||
abstract class L2HellaCacheBundle extends Bundle with L2HellaCacheParameters
|
abstract class L2HellaCacheBundle extends Bundle with L2HellaCacheParameters
|
||||||
abstract class L2HellaCacheModule extends Module with L2HellaCacheParameters
|
abstract class L2HellaCacheModule extends Module with L2HellaCacheParameters
|
||||||
@ -176,16 +177,16 @@ class L2MetadataArray extends L2HellaCacheModule {
|
|||||||
|
|
||||||
class L2DataReadReq extends L2HellaCacheBundle with HasL2Id {
|
class L2DataReadReq extends L2HellaCacheBundle with HasL2Id {
|
||||||
val way_en = Bits(width = nWays)
|
val way_en = Bits(width = nWays)
|
||||||
val addr = Bits(width = params(TLAddrBits))
|
val addr = Bits(width = tlAddrBits)
|
||||||
}
|
}
|
||||||
|
|
||||||
class L2DataWriteReq extends L2DataReadReq {
|
class L2DataWriteReq extends L2DataReadReq {
|
||||||
val wmask = Bits(width = params(TLWriteMaskBits))
|
val wmask = Bits(width = tlWriteMaskBits)
|
||||||
val data = Bits(width = params(TLDataBits))
|
val data = Bits(width = tlDataBits)
|
||||||
}
|
}
|
||||||
|
|
||||||
class L2DataResp extends Bundle with HasL2Id {
|
class L2DataResp extends Bundle with HasL2Id with TileLinkParameters {
|
||||||
val data = Bits(width = params(TLDataBits))
|
val data = Bits(width = tlDataBits)
|
||||||
}
|
}
|
||||||
|
|
||||||
class L2DataArray extends L2HellaCacheModule {
|
class L2DataArray extends L2HellaCacheModule {
|
||||||
@ -295,7 +296,7 @@ class TSHRFile(bankId: Int, innerId: String, outerId: String) extends L2HellaCac
|
|||||||
val any_release_conflict = trackerList.tail.map(_.io.has_release_conflict).reduce(_||_)
|
val any_release_conflict = trackerList.tail.map(_.io.has_release_conflict).reduce(_||_)
|
||||||
val block_releases = Bool(false)
|
val block_releases = Bool(false)
|
||||||
val conflict_idx = Vec(trackerList.map(_.io.has_release_conflict)).lastIndexWhere{b: Bool => b}
|
val conflict_idx = Vec(trackerList.map(_.io.has_release_conflict)).lastIndexWhere{b: Bool => b}
|
||||||
val release_idx = Mux(voluntary, Mux(any_release_conflict, conflict_idx, UInt(0)), release.bits.payload.master_xact_id)
|
val release_idx = Mux(voluntary, Mux(any_release_conflict, conflict_idx, UInt(0)), conflict_idx)
|
||||||
for( i <- 0 until trackerList.size ) {
|
for( i <- 0 until trackerList.size ) {
|
||||||
val t = trackerList(i).io.inner
|
val t = trackerList(i).io.inner
|
||||||
t.release.bits := release.bits
|
t.release.bits := release.bits
|
||||||
@ -380,7 +381,8 @@ class L2VoluntaryReleaseTracker(trackerId: Int, bankId: Int, innerId: String, ou
|
|||||||
|
|
||||||
io.inner.grant.bits.header.src := UInt(bankId)
|
io.inner.grant.bits.header.src := UInt(bankId)
|
||||||
io.inner.grant.bits.header.dst := init_client_id
|
io.inner.grant.bits.header.dst := init_client_id
|
||||||
io.inner.grant.bits.payload := Grant(co.getGrantType(xact, xact_internal.meta.coh),
|
io.inner.grant.bits.payload := Grant(Bool(false),
|
||||||
|
co.getGrantTypeOnVoluntaryWriteback(xact_internal.meta.coh),
|
||||||
xact.client_xact_id,
|
xact.client_xact_id,
|
||||||
UInt(trackerId))
|
UInt(trackerId))
|
||||||
|
|
||||||
@ -436,7 +438,7 @@ class L2VoluntaryReleaseTracker(trackerId: Int, bankId: Int, innerId: String, ou
|
|||||||
is(s_grant) {
|
is(s_grant) {
|
||||||
io.inner.grant.valid := Bool(true)
|
io.inner.grant.valid := Bool(true)
|
||||||
when(io.inner.grant.ready) {
|
when(io.inner.grant.ready) {
|
||||||
state := Mux(co.requiresAckForGrant(c_gnt.payload.g_type),
|
state := Mux(co.requiresAckForGrant(c_gnt.payload),
|
||||||
s_busy, s_idle)
|
s_busy, s_idle)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -463,11 +465,11 @@ class L2AcquireTracker(trackerId: Int, bankId: Int, innerId: String, outerId: St
|
|||||||
val pending_probes = Reg(init = co.dir().flush)
|
val pending_probes = Reg(init = co.dir().flush)
|
||||||
val curr_p_id = co.dir().next(pending_probes)
|
val curr_p_id = co.dir().next(pending_probes)
|
||||||
|
|
||||||
val is_uncached = co.messageIsUncached(xact)
|
val is_uncached = xact.uncached
|
||||||
val tag_match = xact_internal.tag_match
|
val tag_match = xact_internal.tag_match
|
||||||
val needs_writeback = !tag_match && co.needsWriteback(xact_internal.meta.coh)
|
val needs_writeback = !tag_match && co.needsWriteback(xact_internal.meta.coh)
|
||||||
val is_hit = tag_match && co.isHit(xact, xact_internal.meta.coh)
|
val is_hit = tag_match && co.isHit(xact, xact_internal.meta.coh)
|
||||||
val needs_probes = co.requiresProbes(xact.a_type, xact_internal.meta.coh)
|
val needs_probes = co.requiresProbes(xact, xact_internal.meta.coh)
|
||||||
//TODO: does allocate
|
//TODO: does allocate
|
||||||
|
|
||||||
io.has_acquire_conflict := co.isCoherenceConflict(xact.addr, c_acq.payload.addr) &&
|
io.has_acquire_conflict := co.isCoherenceConflict(xact.addr, c_acq.payload.addr) &&
|
||||||
@ -485,28 +487,19 @@ class L2AcquireTracker(trackerId: Int, bankId: Int, innerId: String, outerId: St
|
|||||||
c_gnt.header.dst)
|
c_gnt.header.dst)
|
||||||
|
|
||||||
val addr_wb = Cat(xact_internal.meta.tag, xact.addr(untagBits-1,blockOffBits))
|
val addr_wb = Cat(xact_internal.meta.tag, xact.addr(untagBits-1,blockOffBits))
|
||||||
val outer_write_acq = Bundle(Acquire(co.getUncachedWriteAcquireType,
|
val outer_write_acq = Bundle(UncachedWrite(xact.addr, UInt(trackerId), xact.data),
|
||||||
xact.addr,
|
{ case TLId => outerId })
|
||||||
UInt(trackerId),
|
val outer_write_wb = Bundle(UncachedWrite(addr_wb, UInt(trackerId), wb_buffer),
|
||||||
xact.data), { case TLId => outerId })
|
{ case TLId => outerId })
|
||||||
val outer_write_wb = Bundle(Acquire(co.getUncachedWriteAcquireType,
|
val outer_read = Bundle(UncachedRead( xact.addr, UInt(trackerId)), { case TLId => outerId })
|
||||||
addr_wb,
|
|
||||||
UInt(trackerId),
|
|
||||||
wb_buffer), { case TLId => outerId })
|
|
||||||
val outer_read = Bundle(Acquire(co.getUncachedReadAcquireType,
|
|
||||||
xact.addr,
|
|
||||||
UInt(trackerId)), { case TLId => outerId })
|
|
||||||
io.outer.acquire.valid := Bool(false)
|
io.outer.acquire.valid := Bool(false)
|
||||||
io.outer.acquire.bits.payload := outer_read //default
|
io.outer.acquire.bits.payload := outer_read //default
|
||||||
io.outer.acquire.bits.header.src := UInt(bankId)
|
io.outer.acquire.bits.header.src := UInt(bankId)
|
||||||
io.outer.grant.ready := Bool(true) //grant.data -> xact.data
|
io.outer.grant.ready := Bool(true) //grant.data -> xact.data
|
||||||
|
|
||||||
val cprb_for_cacq = Probe(co.getProbeType(xact, xact_internal.meta.coh),
|
val cprb_for_cacq = Probe(co.getProbeType(xact, xact_internal.meta.coh), xact.addr)
|
||||||
xact.addr,
|
val cprb_for_mwb = Probe(co.getProbeTypeOnVoluntaryWriteback, addr_wb)
|
||||||
UInt(trackerId))
|
|
||||||
val cprb_for_mwb = Probe(co.getProbeTypeOnVoluntaryWriteback,
|
|
||||||
addr_wb,
|
|
||||||
UInt(trackerId))
|
|
||||||
//TODO inner_probe_mprb
|
//TODO inner_probe_mprb
|
||||||
io.inner.probe.valid := Bool(false)
|
io.inner.probe.valid := Bool(false)
|
||||||
io.inner.probe.bits.header.src := UInt(bankId)
|
io.inner.probe.bits.header.src := UInt(bankId)
|
||||||
@ -515,11 +508,11 @@ class L2AcquireTracker(trackerId: Int, bankId: Int, innerId: String, outerId: St
|
|||||||
cprb_for_mwb,
|
cprb_for_mwb,
|
||||||
cprb_for_cacq)
|
cprb_for_cacq)
|
||||||
|
|
||||||
val cgnt_for_cacq = Grant(co.getGrantType(xact, xact_internal.meta.coh),
|
val cgnt_for_cacq = Grant(xact.uncached, co.getGrantType(xact, xact_internal.meta.coh),
|
||||||
xact.client_xact_id,
|
xact.client_xact_id,
|
||||||
UInt(trackerId),
|
UInt(trackerId),
|
||||||
xact.data)
|
xact.data)
|
||||||
val cgnt_for_cwb = Grant(crel_wb_g_type, UInt(0), UInt(trackerId), UInt(0))
|
val cgnt_for_cwb = Grant(Bool(false), crel_wb_g_type, UInt(0), UInt(trackerId), UInt(0))
|
||||||
io.inner.grant.valid := Bool(false)
|
io.inner.grant.valid := Bool(false)
|
||||||
io.inner.grant.bits.header.src := UInt(bankId)
|
io.inner.grant.bits.header.src := UInt(bankId)
|
||||||
io.inner.grant.bits.header.dst := Mux(crel_was_voluntary, crel_wb_src, init_client_id)
|
io.inner.grant.bits.header.dst := Mux(crel_was_voluntary, crel_wb_src, init_client_id)
|
||||||
@ -569,12 +562,12 @@ class L2AcquireTracker(trackerId: Int, bankId: Int, innerId: String, outerId: St
|
|||||||
val _tag_match = io.meta_resp.bits.tag_match
|
val _tag_match = io.meta_resp.bits.tag_match
|
||||||
val _needs_writeback = !_tag_match && co.needsWriteback(coh)
|
val _needs_writeback = !_tag_match && co.needsWriteback(coh)
|
||||||
val _is_hit = _tag_match && co.isHit(xact, coh)
|
val _is_hit = _tag_match && co.isHit(xact, coh)
|
||||||
val _needs_probes = co.requiresProbes(xact.a_type, coh)
|
val _needs_probes = co.requiresProbes(xact, coh)
|
||||||
xact_internal := io.meta_resp.bits
|
xact_internal := io.meta_resp.bits
|
||||||
when(_needs_probes) {
|
when(_needs_probes) {
|
||||||
val mask_incoherent = co.dir().full(coh.sharers) & ~io.tile_incoherent
|
val mask_incoherent = co.dir().full(coh.sharers) & ~io.tile_incoherent
|
||||||
val mask_self = mask_incoherent &
|
val mask_self = mask_incoherent &
|
||||||
~(!(co.requiresSelfProbe(xact.a_type) || _needs_writeback) << init_client_id)
|
~(!(co.requiresSelfProbe(xact) || _needs_writeback) << init_client_id)
|
||||||
pending_probes := mask_self
|
pending_probes := mask_self
|
||||||
release_count := co.dir().count(mask_self)
|
release_count := co.dir().count(mask_self)
|
||||||
crel_had_data := Bool(false)
|
crel_had_data := Bool(false)
|
||||||
@ -610,7 +603,7 @@ class L2AcquireTracker(trackerId: Int, bankId: Int, innerId: String, outerId: St
|
|||||||
when(co.isVoluntary(c_rel.payload)) {
|
when(co.isVoluntary(c_rel.payload)) {
|
||||||
crel_was_voluntary := Bool(true)
|
crel_was_voluntary := Bool(true)
|
||||||
crel_wb_src := c_rel.header.src
|
crel_wb_src := c_rel.header.src
|
||||||
crel_wb_g_type := co.getGrantType(c_rel.payload, xact_internal.meta.coh)
|
crel_wb_g_type := co.getGrantTypeOnVoluntaryWriteback(xact_internal.meta.coh)
|
||||||
}
|
}
|
||||||
when(!co.isVoluntary(c_rel.payload)) {
|
when(!co.isVoluntary(c_rel.payload)) {
|
||||||
release_count := release_count - UInt(1)
|
release_count := release_count - UInt(1)
|
||||||
@ -692,7 +685,7 @@ class L2AcquireTracker(trackerId: Int, bankId: Int, innerId: String, outerId: St
|
|||||||
is(s_grant) {
|
is(s_grant) {
|
||||||
io.inner.grant.valid := Bool(true)
|
io.inner.grant.valid := Bool(true)
|
||||||
when(io.inner.grant.ready) {
|
when(io.inner.grant.ready) {
|
||||||
state := Mux(co.requiresAckForGrant(c_gnt.payload.g_type),
|
state := Mux(co.requiresAckForGrant(c_gnt.payload),
|
||||||
s_busy, s_idle)
|
s_busy, s_idle)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -61,15 +61,15 @@ abstract class DirectoryRepresentation(val width: Int) {
|
|||||||
def full(s: UInt): UInt
|
def full(s: UInt): UInt
|
||||||
}
|
}
|
||||||
|
|
||||||
class NullRepresentation extends DirectoryRepresentation(1) {
|
class NullRepresentation(nClients: Int) extends DirectoryRepresentation(1) {
|
||||||
def pop(prev: UInt, id: UInt) = UInt(0)
|
def pop(prev: UInt, id: UInt) = UInt(0)
|
||||||
def push(prev: UInt, id: UInt) = UInt(0)
|
def push(prev: UInt, id: UInt) = UInt(0)
|
||||||
def flush = UInt(0)
|
def flush = UInt(0)
|
||||||
def none(s: UInt) = Bool(false)
|
def none(s: UInt) = Bool(false)
|
||||||
def one(s: UInt) = Bool(false)
|
def one(s: UInt) = Bool(false)
|
||||||
def count(s: UInt) = UInt(0)
|
def count(s: UInt) = UInt(nClients)
|
||||||
def next(s: UInt) = UInt(0)
|
def next(s: UInt) = UInt(0)
|
||||||
def full(s: UInt) = UInt(0)
|
def full(s: UInt) = SInt(-1, width = nClients).toUInt
|
||||||
}
|
}
|
||||||
|
|
||||||
class FullRepresentation(nClients: Int) extends DirectoryRepresentation(nClients) {
|
class FullRepresentation(nClients: Int) extends DirectoryRepresentation(nClients) {
|
||||||
@ -124,58 +124,49 @@ abstract class CoherencePolicy(val dir: () => DirectoryRepresentation) {
|
|||||||
def getReleaseTypeOnVoluntaryWriteback(): UInt
|
def getReleaseTypeOnVoluntaryWriteback(): UInt
|
||||||
def getReleaseTypeOnProbe(p: Probe, m: ClientMetadata): UInt
|
def getReleaseTypeOnProbe(p: Probe, m: ClientMetadata): UInt
|
||||||
def getGrantType(a: Acquire, m: MasterMetadata): UInt
|
def getGrantType(a: Acquire, m: MasterMetadata): UInt
|
||||||
def getGrantType(r: Release, m: MasterMetadata): UInt
|
def getGrantTypeOnVoluntaryWriteback(m: MasterMetadata): UInt
|
||||||
|
|
||||||
def messageHasData (rel: SourcedMessage): Bool
|
def messageHasData(rel: SourcedMessage): Bool
|
||||||
def messageUpdatesDataArray (reply: Grant): Bool
|
def messageUpdatesDataArray(g: Grant): Bool
|
||||||
def messageIsUncached(acq: Acquire): Bool
|
|
||||||
def isVoluntary(rel: Release): Bool
|
def isVoluntary(rel: Release): Bool
|
||||||
def isVoluntary(gnt: Grant): Bool
|
def isVoluntary(gnt: Grant): Bool
|
||||||
|
def requiresOuterRead(acq: Acquire, m: MasterMetadata): Bool
|
||||||
def requiresOuterRead(a_type: UInt): Bool
|
def requiresOuterWrite(acq: Acquire, m: MasterMetadata): Bool
|
||||||
def requiresOuterWrite(a_type: UInt): Bool
|
def requiresSelfProbe(a: Acquire): Bool
|
||||||
def requiresSelfProbe(a_type: UInt): Bool
|
def requiresProbes(a: Acquire, m: MasterMetadata): Bool
|
||||||
def requiresProbes(a_type: UInt, m: MasterMetadata): Bool
|
def requiresAckForGrant(g: Grant): Bool
|
||||||
def requiresAckForGrant(g_type: UInt): Bool
|
def requiresAckForRelease(r: Release): Bool
|
||||||
def requiresAckForRelease(r_type: UInt): Bool
|
|
||||||
def pendingVoluntaryReleaseIsSufficient(r_type: UInt, p_type: UInt): Bool
|
def pendingVoluntaryReleaseIsSufficient(r_type: UInt, p_type: UInt): Bool
|
||||||
|
|
||||||
def isCoherenceConflict(addr1: UInt, addr2: UInt): Bool
|
def isCoherenceConflict(addr1: UInt, addr2: UInt): Bool
|
||||||
|
|
||||||
|
def getGrantTypeForUncached(a: Acquire, m: MasterMetadata): UInt = {
|
||||||
|
MuxLookup(a.a_type, Grant.uncachedRead, Array(
|
||||||
|
Acquire.uncachedRead -> Grant.uncachedRead,
|
||||||
|
Acquire.uncachedWrite -> Grant.uncachedWrite,
|
||||||
|
Acquire.uncachedAtomic -> Grant.uncachedAtomic
|
||||||
|
))
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
trait UncachedTransactions {
|
class MICoherence(dir: () => DirectoryRepresentation) extends CoherencePolicy(dir) {
|
||||||
def getUncachedReadAcquireType: UInt
|
|
||||||
def getUncachedWriteAcquireType: UInt
|
|
||||||
def getUncachedReadWordAcquireType: UInt
|
|
||||||
def getUncachedWriteWordAcquireType: UInt
|
|
||||||
def getUncachedAtomicAcquireType: UInt
|
|
||||||
def isUncachedReadTransaction(acq: Acquire): Bool
|
|
||||||
def getUncachedReadGrantType: UInt
|
|
||||||
}
|
|
||||||
|
|
||||||
abstract class CoherencePolicyWithUncached(dir: () => DirectoryRepresentation) extends CoherencePolicy(dir)
|
|
||||||
with UncachedTransactions
|
|
||||||
|
|
||||||
class MICoherence(dir: () => DirectoryRepresentation) extends CoherencePolicyWithUncached(dir) {
|
|
||||||
def nClientStates = 2
|
def nClientStates = 2
|
||||||
def nMasterStates = 2
|
def nMasterStates = 2
|
||||||
def nAcquireTypes = 6
|
def nAcquireTypes = 1
|
||||||
def nProbeTypes = 2
|
def nProbeTypes = 2
|
||||||
def nReleaseTypes = 5
|
def nReleaseTypes = 5
|
||||||
def nGrantTypes = 7
|
def nGrantTypes = 2
|
||||||
|
|
||||||
val clientInvalid :: clientValid :: Nil = Enum(UInt(), nClientStates)
|
val clientInvalid :: clientValid :: Nil = Enum(UInt(), nClientStates)
|
||||||
val masterInvalid :: masterValid :: Nil = Enum(UInt(), nMasterStates)
|
val masterInvalid :: masterValid :: Nil = Enum(UInt(), nMasterStates)
|
||||||
|
|
||||||
val acquireReadExclusive :: acquireReadUncached :: acquireWriteUncached :: acquireReadWordUncached :: acquireWriteWordUncached :: acquireAtomicUncached :: Nil = Enum(UInt(), nAcquireTypes)
|
val acquireReadExclusive :: Nil = Enum(UInt(), nAcquireTypes)
|
||||||
val probeInvalidate :: probeCopy :: Nil = Enum(UInt(), nProbeTypes)
|
val probeInvalidate :: probeCopy :: Nil = Enum(UInt(), nProbeTypes)
|
||||||
val releaseVoluntaryInvalidateData :: releaseInvalidateData :: releaseCopyData :: releaseInvalidateAck :: releaseCopyAck :: Nil = Enum(UInt(), nReleaseTypes)
|
val releaseVoluntaryInvalidateData :: releaseInvalidateData :: releaseCopyData :: releaseInvalidateAck :: releaseCopyAck :: Nil = Enum(UInt(), nReleaseTypes)
|
||||||
val grantVoluntaryAck :: grantReadExclusive :: grantReadUncached :: grantWriteUncached :: grantReadWordUncached :: grantWriteWordUncached :: grantAtomicUncached :: Nil = Enum(UInt(), nGrantTypes)
|
val grantVoluntaryAck :: grantReadExclusive :: Nil = Enum(UInt(), nGrantTypes)
|
||||||
|
|
||||||
val uncachedAcquireTypeVec = Vec(acquireReadUncached, acquireWriteUncached, acquireReadWordUncached, acquireWriteWordUncached, acquireAtomicUncached)
|
|
||||||
val hasDataAcquireTypeVec = Vec(acquireWriteUncached, acquireWriteWordUncached, acquireAtomicUncached)
|
|
||||||
val hasDataReleaseTypeVec = Vec(releaseVoluntaryInvalidateData, releaseInvalidateData, releaseCopyData)
|
val hasDataReleaseTypeVec = Vec(releaseVoluntaryInvalidateData, releaseInvalidateData, releaseCopyData)
|
||||||
val hasDataGrantTypeVec = Vec(grantReadExclusive, grantReadUncached, grantReadWordUncached, grantAtomicUncached)
|
val hasDataGrantTypeVec = Vec(grantReadExclusive)
|
||||||
|
|
||||||
def isHit (cmd: UInt, m: ClientMetadata): Bool = isValid(m)
|
def isHit (cmd: UInt, m: ClientMetadata): Bool = isValid(m)
|
||||||
def isValid (m: ClientMetadata): Bool = m.state != clientInvalid
|
def isValid (m: ClientMetadata): Bool = m.state != clientInvalid
|
||||||
@ -201,15 +192,8 @@ class MICoherence(dir: () => DirectoryRepresentation) extends CoherencePolicyWit
|
|||||||
M_CLN -> clientValid
|
M_CLN -> clientValid
|
||||||
)))(this)
|
)))(this)
|
||||||
def clientMetadataOnFlush = clientMetadataOnCacheControl(M_INV)
|
def clientMetadataOnFlush = clientMetadataOnCacheControl(M_INV)
|
||||||
def clientMetadataOnGrant(incoming: Grant, outstanding: Acquire) = ClientMetadata(
|
def clientMetadataOnGrant(incoming: Grant, outstanding: Acquire) =
|
||||||
MuxLookup(incoming.g_type, clientInvalid, Array(
|
ClientMetadata(Mux(incoming.uncached, clientInvalid, clientValid))(this)
|
||||||
grantReadExclusive -> clientValid,
|
|
||||||
grantReadUncached -> clientInvalid,
|
|
||||||
grantWriteUncached -> clientInvalid,
|
|
||||||
grantReadWordUncached -> clientInvalid,
|
|
||||||
grantWriteWordUncached -> clientInvalid,
|
|
||||||
grantAtomicUncached -> clientInvalid
|
|
||||||
)))(this)
|
|
||||||
def clientMetadataOnProbe(incoming: Probe, m: ClientMetadata) = ClientMetadata(
|
def clientMetadataOnProbe(incoming: Probe, m: ClientMetadata) = ClientMetadata(
|
||||||
MuxLookup(incoming.p_type, m.state, Array(
|
MuxLookup(incoming.p_type, m.state, Array(
|
||||||
probeInvalidate -> clientInvalid,
|
probeInvalidate -> clientInvalid,
|
||||||
@ -232,16 +216,8 @@ class MICoherence(dir: () => DirectoryRepresentation) extends CoherencePolicyWit
|
|||||||
))
|
))
|
||||||
}
|
}
|
||||||
|
|
||||||
def getUncachedReadAcquireType = acquireReadUncached
|
|
||||||
def getUncachedWriteAcquireType = acquireWriteUncached
|
|
||||||
def getUncachedReadWordAcquireType = acquireReadWordUncached
|
|
||||||
def getUncachedWriteWordAcquireType = acquireWriteWordUncached
|
|
||||||
def getUncachedAtomicAcquireType = acquireAtomicUncached
|
|
||||||
def isUncachedReadTransaction(acq: Acquire) = acq.a_type === acquireReadUncached
|
|
||||||
def getUncachedReadGrantType = grantReadUncached
|
|
||||||
|
|
||||||
def isVoluntary(rel: Release) = rel.r_type === releaseVoluntaryInvalidateData
|
def isVoluntary(rel: Release) = rel.r_type === releaseVoluntaryInvalidateData
|
||||||
def isVoluntary(gnt: Grant) = gnt.g_type === grantVoluntaryAck
|
def isVoluntary(gnt: Grant) = !gnt.uncached && gnt.g_type === grantVoluntaryAck
|
||||||
|
|
||||||
def getAcquireTypeOnPrimaryMiss(cmd: UInt, m: ClientMetadata): UInt = acquireReadExclusive
|
def getAcquireTypeOnPrimaryMiss(cmd: UInt, m: ClientMetadata): UInt = acquireReadExclusive
|
||||||
def getAcquireTypeOnSecondaryMiss(cmd: UInt, m: ClientMetadata, outstanding: Acquire): UInt = acquireReadExclusive
|
def getAcquireTypeOnSecondaryMiss(cmd: UInt, m: ClientMetadata, outstanding: Acquire): UInt = acquireReadExclusive
|
||||||
@ -260,80 +236,63 @@ class MICoherence(dir: () => DirectoryRepresentation) extends CoherencePolicyWit
|
|||||||
}
|
}
|
||||||
|
|
||||||
def messageHasData(msg: SourcedMessage) = msg match {
|
def messageHasData(msg: SourcedMessage) = msg match {
|
||||||
case acq: Acquire => hasDataAcquireTypeVec.contains(acq.a_type)
|
case acq: Acquire => Mux(acq.uncached, Acquire.hasData(acq.a_type), Bool(false))
|
||||||
case grant: Grant => hasDataGrantTypeVec.contains(grant.g_type)
|
case gnt: Grant => Mux(gnt.uncached, Grant.hasData(gnt.g_type), hasDataGrantTypeVec.contains(gnt.g_type))
|
||||||
case rel: Release => hasDataReleaseTypeVec.contains(rel.r_type)
|
case rel: Release => hasDataReleaseTypeVec.contains(rel.r_type)
|
||||||
case _ => Bool(false)
|
case _ => Bool(false)
|
||||||
}
|
}
|
||||||
def messageUpdatesDataArray (reply: Grant): Bool = {
|
def messageUpdatesDataArray(g: Grant): Bool = {
|
||||||
(reply.g_type === grantReadExclusive)
|
Mux(g.uncached, Bool(false),
|
||||||
|
(g.g_type === grantReadExclusive))
|
||||||
}
|
}
|
||||||
def messageIsUncached(acq: Acquire): Bool = uncachedAcquireTypeVec.contains(acq.a_type)
|
|
||||||
|
|
||||||
def isCoherenceConflict(addr1: UInt, addr2: UInt): Bool = (addr1 === addr2)
|
def isCoherenceConflict(addr1: UInt, addr2: UInt): Bool = (addr1 === addr2)
|
||||||
|
|
||||||
def getGrantType(a: Acquire, m: MasterMetadata): UInt = {
|
def getGrantType(a: Acquire, m: MasterMetadata): UInt =
|
||||||
MuxLookup(a.a_type, grantReadUncached, Array(
|
Mux(a.uncached, getGrantTypeForUncached(a, m), grantReadExclusive)
|
||||||
acquireReadExclusive -> grantReadExclusive,
|
|
||||||
acquireReadUncached -> grantReadUncached,
|
|
||||||
acquireWriteUncached -> grantWriteUncached,
|
|
||||||
acquireReadWordUncached -> grantReadWordUncached,
|
|
||||||
acquireWriteWordUncached -> grantWriteWordUncached,
|
|
||||||
acquireAtomicUncached -> grantAtomicUncached
|
|
||||||
))
|
|
||||||
}
|
|
||||||
|
|
||||||
def getGrantType(r: Release, m: MasterMetadata): UInt = {
|
def getGrantTypeOnVoluntaryWriteback(m: MasterMetadata): UInt = grantVoluntaryAck
|
||||||
MuxLookup(r.r_type, grantReadUncached, Array(
|
|
||||||
releaseVoluntaryInvalidateData -> grantVoluntaryAck
|
|
||||||
))
|
|
||||||
}
|
|
||||||
|
|
||||||
def getProbeType(a: Acquire, m: MasterMetadata): UInt = {
|
def getProbeType(a: Acquire, m: MasterMetadata): UInt = {
|
||||||
MuxLookup(a.a_type, probeCopy, Array(
|
Mux(a.uncached,
|
||||||
acquireReadExclusive -> probeInvalidate,
|
MuxLookup(a.a_type, probeCopy, Array(
|
||||||
acquireReadUncached -> probeCopy,
|
Acquire.uncachedRead -> probeCopy,
|
||||||
acquireWriteUncached -> probeInvalidate,
|
Acquire.uncachedWrite -> probeInvalidate,
|
||||||
acquireReadWordUncached -> probeCopy,
|
Acquire.uncachedAtomic -> probeInvalidate
|
||||||
acquireWriteWordUncached -> probeInvalidate,
|
)), probeInvalidate)
|
||||||
acquireAtomicUncached -> probeInvalidate
|
|
||||||
))
|
|
||||||
}
|
}
|
||||||
def getProbeTypeOnVoluntaryWriteback: UInt = probeInvalidate
|
|
||||||
|
|
||||||
def requiresOuterRead(a_type: UInt) = {
|
def getProbeTypeOnVoluntaryWriteback: UInt = probeInvalidate
|
||||||
(a_type != acquireWriteUncached)
|
def requiresOuterRead(acq: Acquire, m: MasterMetadata) =
|
||||||
}
|
Mux(acq.uncached, Acquire.requiresOuterRead(acq.a_type), Bool(true))
|
||||||
def requiresOuterWrite(a_type: UInt) = {
|
def requiresOuterWrite(acq: Acquire, m: MasterMetadata) =
|
||||||
(a_type === acquireWriteUncached)
|
Mux(acq.uncached, Acquire.requiresOuterWrite(acq.a_type), Bool(false))
|
||||||
}
|
|
||||||
def requiresAckForGrant(g_type: UInt) = g_type != grantVoluntaryAck
|
def requiresAckForGrant(g: Grant) = g.uncached || g.g_type != grantVoluntaryAck
|
||||||
def requiresAckForRelease(r_type: UInt) = Bool(false)
|
def requiresAckForRelease(r: Release) = Bool(false)
|
||||||
def requiresSelfProbe(a_type: UInt) = a_type === acquireReadUncached
|
def requiresSelfProbe(a: Acquire) = a.uncached && a.a_type === Acquire.uncachedRead
|
||||||
def requiresProbes(a_type: UInt, m: MasterMetadata) = !dir().none(m.sharers)
|
def requiresProbes(a: Acquire, m: MasterMetadata) = !dir().none(m.sharers)
|
||||||
def pendingVoluntaryReleaseIsSufficient(r_type: UInt, p_type: UInt): Bool = (r_type === releaseVoluntaryInvalidateData)
|
def pendingVoluntaryReleaseIsSufficient(r_type: UInt, p_type: UInt): Bool = (r_type === releaseVoluntaryInvalidateData)
|
||||||
}
|
}
|
||||||
|
|
||||||
class MEICoherence(dir: () => DirectoryRepresentation) extends CoherencePolicyWithUncached(dir) {
|
class MEICoherence(dir: () => DirectoryRepresentation) extends CoherencePolicy(dir) {
|
||||||
def nClientStates = 3
|
def nClientStates = 3
|
||||||
def nMasterStates = 2
|
def nMasterStates = 2
|
||||||
def nAcquireTypes = 7
|
def nAcquireTypes = 2
|
||||||
def nProbeTypes = 3
|
def nProbeTypes = 3
|
||||||
def nReleaseTypes = 7
|
def nReleaseTypes = 7
|
||||||
def nGrantTypes = 8
|
def nGrantTypes = 2
|
||||||
|
|
||||||
val clientInvalid :: clientExclusiveClean :: clientExclusiveDirty :: Nil = Enum(UInt(), nClientStates)
|
val clientInvalid :: clientExclusiveClean :: clientExclusiveDirty :: Nil = Enum(UInt(), nClientStates)
|
||||||
val masterInvalid :: masterValid :: Nil = Enum(UInt(), nMasterStates)
|
val masterInvalid :: masterValid :: Nil = Enum(UInt(), nMasterStates)
|
||||||
|
|
||||||
val acquireReadExclusiveClean :: acquireReadExclusiveDirty :: acquireReadUncached :: acquireWriteUncached :: acquireReadWordUncached :: acquireWriteWordUncached :: acquireAtomicUncached :: Nil = Enum(UInt(), nAcquireTypes)
|
val acquireReadExclusiveClean :: acquireReadExclusiveDirty :: Nil = Enum(UInt(), nAcquireTypes)
|
||||||
val probeInvalidate :: probeDowngrade :: probeCopy :: Nil = Enum(UInt(), nProbeTypes)
|
val probeInvalidate :: probeDowngrade :: probeCopy :: Nil = Enum(UInt(), nProbeTypes)
|
||||||
val releaseVoluntaryInvalidateData :: releaseInvalidateData :: releaseDowngradeData :: releaseCopyData :: releaseInvalidateAck :: releaseDowngradeAck :: releaseCopyAck :: Nil = Enum(UInt(), nReleaseTypes)
|
val releaseVoluntaryInvalidateData :: releaseInvalidateData :: releaseDowngradeData :: releaseCopyData :: releaseInvalidateAck :: releaseDowngradeAck :: releaseCopyAck :: Nil = Enum(UInt(), nReleaseTypes)
|
||||||
val grantVoluntaryAck :: grantReadExclusive :: grantReadUncached :: grantWriteUncached :: grantReadExclusiveAck :: grantReadWordUncached :: grantWriteWordUncached :: grantAtomicUncached :: Nil = Enum(UInt(), nGrantTypes)
|
val grantVoluntaryAck :: grantReadExclusive :: Nil = Enum(UInt(), nGrantTypes)
|
||||||
|
|
||||||
val uncachedAcquireTypeVec = Vec(acquireReadUncached, acquireWriteUncached, acquireReadWordUncached, acquireWriteWordUncached, acquireAtomicUncached)
|
|
||||||
val hasDataAcquireTypeVec = Vec(acquireWriteUncached, acquireWriteWordUncached, acquireAtomicUncached)
|
|
||||||
val hasDataReleaseTypeVec = Vec(releaseVoluntaryInvalidateData, releaseInvalidateData, releaseDowngradeData, releaseCopyData)
|
val hasDataReleaseTypeVec = Vec(releaseVoluntaryInvalidateData, releaseInvalidateData, releaseDowngradeData, releaseCopyData)
|
||||||
val hasDataGrantTypeVec = Vec(grantReadExclusive, grantReadUncached, grantReadWordUncached, grantAtomicUncached)
|
val hasDataGrantTypeVec = Vec(grantReadExclusive)
|
||||||
|
|
||||||
def isHit (cmd: UInt, m: ClientMetadata) = isValid(m)
|
def isHit (cmd: UInt, m: ClientMetadata) = isValid(m)
|
||||||
def isValid (m: ClientMetadata) = m.state != clientInvalid
|
def isValid (m: ClientMetadata) = m.state != clientInvalid
|
||||||
@ -341,7 +300,7 @@ class MEICoherence(dir: () => DirectoryRepresentation) extends CoherencePolicyWi
|
|||||||
def isValid (m: MasterMetadata) = m.state != masterInvalid
|
def isValid (m: MasterMetadata) = m.state != masterInvalid
|
||||||
|
|
||||||
def needsTransactionOnSecondaryMiss(cmd: UInt, outstanding: Acquire): Bool = {
|
def needsTransactionOnSecondaryMiss(cmd: UInt, outstanding: Acquire): Bool = {
|
||||||
(isRead(cmd) && messageIsUncached(outstanding)) ||
|
(isRead(cmd) && outstanding.uncached) ||
|
||||||
(isWriteIntent(cmd) && (outstanding.a_type != acquireReadExclusiveDirty))
|
(isWriteIntent(cmd) && (outstanding.a_type != acquireReadExclusiveDirty))
|
||||||
}
|
}
|
||||||
def needsTransactionOnCacheControl(cmd: UInt, m: ClientMetadata): Bool = {
|
def needsTransactionOnCacheControl(cmd: UInt, m: ClientMetadata): Bool = {
|
||||||
@ -365,16 +324,9 @@ class MEICoherence(dir: () => DirectoryRepresentation) extends CoherencePolicyWi
|
|||||||
)))(this)
|
)))(this)
|
||||||
def clientMetadataOnFlush() = clientMetadataOnCacheControl(M_INV)
|
def clientMetadataOnFlush() = clientMetadataOnCacheControl(M_INV)
|
||||||
def clientMetadataOnGrant(incoming: Grant, outstanding: Acquire) = ClientMetadata(
|
def clientMetadataOnGrant(incoming: Grant, outstanding: Acquire) = ClientMetadata(
|
||||||
MuxLookup(incoming.g_type, clientInvalid, Array(
|
Mux(incoming.uncached, clientInvalid,
|
||||||
grantReadExclusive -> Mux(outstanding.a_type === acquireReadExclusiveDirty,
|
Mux(outstanding.a_type === acquireReadExclusiveDirty, clientExclusiveDirty,
|
||||||
clientExclusiveDirty, clientExclusiveClean),
|
clientExclusiveClean)))(this)
|
||||||
grantReadExclusiveAck -> clientExclusiveDirty,
|
|
||||||
grantReadUncached -> clientInvalid,
|
|
||||||
grantWriteUncached -> clientInvalid,
|
|
||||||
grantReadWordUncached -> clientInvalid,
|
|
||||||
grantWriteWordUncached -> clientInvalid,
|
|
||||||
grantAtomicUncached -> clientInvalid
|
|
||||||
)))(this)
|
|
||||||
def clientMetadataOnProbe(incoming: Probe, m: ClientMetadata) = ClientMetadata(
|
def clientMetadataOnProbe(incoming: Probe, m: ClientMetadata) = ClientMetadata(
|
||||||
MuxLookup(incoming.p_type, m.state, Array(
|
MuxLookup(incoming.p_type, m.state, Array(
|
||||||
probeInvalidate -> clientInvalid,
|
probeInvalidate -> clientInvalid,
|
||||||
@ -394,21 +346,12 @@ class MEICoherence(dir: () => DirectoryRepresentation) extends CoherencePolicyWi
|
|||||||
val cached = MasterMetadata(masterValid, dir().push(m.sharers, dst))(this)
|
val cached = MasterMetadata(masterValid, dir().push(m.sharers, dst))(this)
|
||||||
val uncached = MasterMetadata(masterValid, m.sharers)(this)
|
val uncached = MasterMetadata(masterValid, m.sharers)(this)
|
||||||
MuxBundle(uncached, Array(
|
MuxBundle(uncached, Array(
|
||||||
g.is(grantReadExclusive) -> cached,
|
g.is(grantReadExclusive) -> cached
|
||||||
g.is(grantReadExclusiveAck) -> cached
|
|
||||||
))
|
))
|
||||||
}
|
}
|
||||||
|
|
||||||
def getUncachedReadAcquireType = acquireReadUncached
|
|
||||||
def getUncachedWriteAcquireType = acquireWriteUncached
|
|
||||||
def getUncachedReadWordAcquireType = acquireReadWordUncached
|
|
||||||
def getUncachedWriteWordAcquireType = acquireWriteWordUncached
|
|
||||||
def getUncachedAtomicAcquireType = acquireAtomicUncached
|
|
||||||
def isUncachedReadTransaction(acq: Acquire) = acq.a_type === acquireReadUncached
|
|
||||||
def getUncachedReadGrantType = grantReadUncached
|
|
||||||
|
|
||||||
def isVoluntary(rel: Release) = rel.r_type === releaseVoluntaryInvalidateData
|
def isVoluntary(rel: Release) = rel.r_type === releaseVoluntaryInvalidateData
|
||||||
def isVoluntary(gnt: Grant) = gnt.g_type === grantVoluntaryAck
|
def isVoluntary(gnt: Grant) = !gnt.uncached && gnt.g_type === grantVoluntaryAck
|
||||||
|
|
||||||
def getAcquireTypeOnPrimaryMiss(cmd: UInt, m: ClientMetadata): UInt = {
|
def getAcquireTypeOnPrimaryMiss(cmd: UInt, m: ClientMetadata): UInt = {
|
||||||
Mux(isWriteIntent(cmd), acquireReadExclusiveDirty, acquireReadExclusiveClean)
|
Mux(isWriteIntent(cmd), acquireReadExclusiveDirty, acquireReadExclusiveClean)
|
||||||
@ -433,84 +376,65 @@ class MEICoherence(dir: () => DirectoryRepresentation) extends CoherencePolicyWi
|
|||||||
}
|
}
|
||||||
|
|
||||||
def messageHasData(msg: SourcedMessage) = msg match {
|
def messageHasData(msg: SourcedMessage) = msg match {
|
||||||
case acq: Acquire => hasDataAcquireTypeVec.contains(acq.a_type)
|
case acq: Acquire => Mux(acq.uncached, Acquire.hasData(acq.a_type), Bool(false))
|
||||||
case grant: Grant => hasDataGrantTypeVec.contains(grant.g_type)
|
case gnt: Grant => Mux(gnt.uncached, Grant.hasData(gnt.g_type), hasDataGrantTypeVec.contains(gnt.g_type))
|
||||||
case rel: Release => hasDataReleaseTypeVec.contains(rel.r_type)
|
case rel: Release => hasDataReleaseTypeVec.contains(rel.r_type)
|
||||||
case _ => Bool(false)
|
case _ => Bool(false)
|
||||||
}
|
}
|
||||||
def messageUpdatesDataArray (reply: Grant): Bool = {
|
def messageUpdatesDataArray(g: Grant): Bool = {
|
||||||
(reply.g_type === grantReadExclusive)
|
Mux(g.uncached, Bool(false),
|
||||||
|
(g.g_type === grantReadExclusive))
|
||||||
}
|
}
|
||||||
def messageIsUncached(acq: Acquire): Bool = uncachedAcquireTypeVec.contains(acq.a_type)
|
|
||||||
|
|
||||||
def isCoherenceConflict(addr1: UInt, addr2: UInt): Bool = (addr1 === addr2)
|
def isCoherenceConflict(addr1: UInt, addr2: UInt): Bool = (addr1 === addr2)
|
||||||
|
|
||||||
def getGrantType(a: Acquire, m: MasterMetadata): UInt = {
|
def getGrantType(a: Acquire, m: MasterMetadata): UInt = {
|
||||||
MuxLookup(a.a_type, grantReadUncached, Array(
|
Mux(a.uncached, getGrantTypeForUncached(a, m), grantReadExclusive)
|
||||||
acquireReadExclusiveClean -> grantReadExclusive,
|
|
||||||
acquireReadExclusiveDirty -> grantReadExclusive,
|
|
||||||
acquireReadUncached -> grantReadUncached,
|
|
||||||
acquireWriteUncached -> grantWriteUncached,
|
|
||||||
acquireReadWordUncached -> grantReadWordUncached,
|
|
||||||
acquireWriteWordUncached -> grantWriteWordUncached,
|
|
||||||
acquireAtomicUncached -> grantAtomicUncached
|
|
||||||
))
|
|
||||||
}
|
}
|
||||||
def getGrantType(r: Release, m: MasterMetadata): UInt = {
|
def getGrantTypeOnVoluntaryWriteback(m: MasterMetadata): UInt = grantVoluntaryAck
|
||||||
MuxLookup(r.r_type, grantReadUncached, Array(
|
|
||||||
releaseVoluntaryInvalidateData -> grantVoluntaryAck
|
|
||||||
))
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
def getProbeType(a: Acquire, m: MasterMetadata): UInt = {
|
def getProbeType(a: Acquire, m: MasterMetadata): UInt = {
|
||||||
MuxLookup(a.a_type, probeCopy, Array(
|
Mux(a.uncached,
|
||||||
acquireReadExclusiveClean -> probeInvalidate,
|
MuxLookup(a.a_type, probeCopy, Array(
|
||||||
acquireReadExclusiveDirty -> probeInvalidate,
|
Acquire.uncachedRead -> probeCopy,
|
||||||
acquireReadUncached -> probeCopy,
|
Acquire.uncachedWrite -> probeInvalidate,
|
||||||
acquireWriteUncached -> probeInvalidate,
|
Acquire.uncachedAtomic -> probeInvalidate
|
||||||
acquireReadWordUncached -> probeCopy,
|
)), probeInvalidate)
|
||||||
acquireWriteWordUncached -> probeInvalidate,
|
|
||||||
acquireAtomicUncached -> probeInvalidate
|
|
||||||
))
|
|
||||||
}
|
}
|
||||||
def getProbeTypeOnVoluntaryWriteback: UInt = probeInvalidate
|
def getProbeTypeOnVoluntaryWriteback: UInt = probeInvalidate
|
||||||
|
|
||||||
def requiresOuterRead(a_type: UInt) = {
|
def requiresOuterRead(acq: Acquire, m: MasterMetadata) =
|
||||||
(a_type != acquireWriteUncached)
|
Mux(acq.uncached, Acquire.requiresOuterRead(acq.a_type), Bool(true))
|
||||||
}
|
def requiresOuterWrite(acq: Acquire, m: MasterMetadata) =
|
||||||
def requiresOuterWrite(a_type: UInt) = {
|
Mux(acq.uncached, Acquire.requiresOuterWrite(acq.a_type), Bool(false))
|
||||||
(a_type === acquireWriteUncached)
|
|
||||||
}
|
def requiresAckForGrant(g: Grant) = g.uncached || g.g_type != grantVoluntaryAck
|
||||||
def requiresAckForGrant(g_type: UInt) = g_type != grantVoluntaryAck
|
def requiresAckForRelease(r: Release) = Bool(false)
|
||||||
def requiresAckForRelease(r_type: UInt) = Bool(false)
|
def requiresSelfProbe(a: Acquire) = a.uncached && a.a_type === Acquire.uncachedRead
|
||||||
def requiresSelfProbe(a_type: UInt) = a_type === acquireReadUncached
|
def requiresProbes(a: Acquire, m: MasterMetadata) = !dir().none(m.sharers)
|
||||||
def requiresProbes(a_type: UInt, m: MasterMetadata) = !dir().none(m.sharers)
|
|
||||||
|
|
||||||
def pendingVoluntaryReleaseIsSufficient(r_type: UInt, p_type: UInt): Bool = (r_type === releaseVoluntaryInvalidateData)
|
def pendingVoluntaryReleaseIsSufficient(r_type: UInt, p_type: UInt): Bool = (r_type === releaseVoluntaryInvalidateData)
|
||||||
}
|
}
|
||||||
|
|
||||||
class MSICoherence(dir: () => DirectoryRepresentation) extends CoherencePolicyWithUncached(dir) {
|
class MSICoherence(dir: () => DirectoryRepresentation) extends CoherencePolicy(dir) {
|
||||||
def nClientStates = 3
|
def nClientStates = 3
|
||||||
def nMasterStates = 2
|
def nMasterStates = 2
|
||||||
def nAcquireTypes = 7
|
def nAcquireTypes = 2
|
||||||
def nProbeTypes = 3
|
def nProbeTypes = 3
|
||||||
def nReleaseTypes = 7
|
def nReleaseTypes = 7
|
||||||
def nGrantTypes = 9
|
def nGrantTypes = 3
|
||||||
|
|
||||||
val clientInvalid :: clientShared :: clientExclusiveDirty :: Nil = Enum(UInt(), nClientStates)
|
val clientInvalid :: clientShared :: clientExclusiveDirty :: Nil = Enum(UInt(), nClientStates)
|
||||||
//val masterInvalid :: masterShared :: masterExclusive :: Nil = Enum(UInt(), nMasterStates)
|
//val masterInvalid :: masterShared :: masterExclusive :: Nil = Enum(UInt(), nMasterStates)
|
||||||
val masterInvalid :: masterValid :: Nil = Enum(UInt(), nMasterStates)
|
val masterInvalid :: masterValid :: Nil = Enum(UInt(), nMasterStates)
|
||||||
|
|
||||||
val acquireReadShared :: acquireReadExclusive :: acquireReadUncached :: acquireWriteUncached :: acquireReadWordUncached :: acquireWriteWordUncached :: acquireAtomicUncached :: Nil = Enum(UInt(), nAcquireTypes)
|
val acquireReadShared :: acquireReadExclusive :: Nil = Enum(UInt(), nAcquireTypes)
|
||||||
val probeInvalidate :: probeDowngrade :: probeCopy :: Nil = Enum(UInt(), nProbeTypes)
|
val probeInvalidate :: probeDowngrade :: probeCopy :: Nil = Enum(UInt(), nProbeTypes)
|
||||||
val releaseVoluntaryInvalidateData :: releaseInvalidateData :: releaseDowngradeData :: releaseCopyData :: releaseInvalidateAck :: releaseDowngradeAck :: releaseCopyAck :: Nil = Enum(UInt(), nReleaseTypes)
|
val releaseVoluntaryInvalidateData :: releaseInvalidateData :: releaseDowngradeData :: releaseCopyData :: releaseInvalidateAck :: releaseDowngradeAck :: releaseCopyAck :: Nil = Enum(UInt(), nReleaseTypes)
|
||||||
val grantVoluntaryAck :: grantReadShared :: grantReadExclusive :: grantReadUncached :: grantWriteUncached :: grantReadExclusiveAck :: grantReadWordUncached :: grantWriteWordUncached :: grantAtomicUncached :: Nil = Enum(UInt(), nGrantTypes)
|
val grantVoluntaryAck :: grantReadShared :: grantReadExclusive :: Nil = Enum(UInt(), nGrantTypes)
|
||||||
|
|
||||||
val uncachedAcquireTypeVec = Vec(acquireReadUncached, acquireWriteUncached, acquireReadWordUncached, acquireWriteWordUncached, acquireAtomicUncached)
|
|
||||||
val hasDataAcquireTypeVec = Vec(acquireWriteUncached, acquireWriteWordUncached, acquireAtomicUncached)
|
|
||||||
val hasDataReleaseTypeVec = Vec(releaseVoluntaryInvalidateData, releaseInvalidateData, releaseDowngradeData, releaseCopyData)
|
val hasDataReleaseTypeVec = Vec(releaseVoluntaryInvalidateData, releaseInvalidateData, releaseDowngradeData, releaseCopyData)
|
||||||
val hasDataGrantTypeVec = Vec(grantReadShared, grantReadExclusive, grantReadUncached, grantReadWordUncached, grantAtomicUncached)
|
val hasDataGrantTypeVec = Vec(grantReadShared, grantReadExclusive)
|
||||||
|
|
||||||
def isHit (cmd: UInt, m: ClientMetadata): Bool = {
|
def isHit (cmd: UInt, m: ClientMetadata): Bool = {
|
||||||
Mux(isWriteIntent(cmd), (m.state === clientExclusiveDirty),
|
Mux(isWriteIntent(cmd), (m.state === clientExclusiveDirty),
|
||||||
@ -523,7 +447,7 @@ class MSICoherence(dir: () => DirectoryRepresentation) extends CoherencePolicyWi
|
|||||||
def isValid (m: MasterMetadata) = m.state != masterInvalid
|
def isValid (m: MasterMetadata) = m.state != masterInvalid
|
||||||
|
|
||||||
def needsTransactionOnSecondaryMiss(cmd: UInt, outstanding: Acquire): Bool = {
|
def needsTransactionOnSecondaryMiss(cmd: UInt, outstanding: Acquire): Bool = {
|
||||||
(isRead(cmd) && messageIsUncached(outstanding)) ||
|
(isRead(cmd) && outstanding.uncached) ||
|
||||||
(isWriteIntent(cmd) && (outstanding.a_type != acquireReadExclusive))
|
(isWriteIntent(cmd) && (outstanding.a_type != acquireReadExclusive))
|
||||||
}
|
}
|
||||||
def needsTransactionOnCacheControl(cmd: UInt, m: ClientMetadata): Bool = {
|
def needsTransactionOnCacheControl(cmd: UInt, m: ClientMetadata): Bool = {
|
||||||
@ -546,16 +470,9 @@ class MSICoherence(dir: () => DirectoryRepresentation) extends CoherencePolicyWi
|
|||||||
)))(this)
|
)))(this)
|
||||||
def clientMetadataOnFlush() = clientMetadataOnCacheControl(M_INV)
|
def clientMetadataOnFlush() = clientMetadataOnCacheControl(M_INV)
|
||||||
def clientMetadataOnGrant(incoming: Grant, outstanding: Acquire) = ClientMetadata(
|
def clientMetadataOnGrant(incoming: Grant, outstanding: Acquire) = ClientMetadata(
|
||||||
MuxLookup(incoming.g_type, clientInvalid, Array(
|
Mux(incoming.uncached, clientInvalid,
|
||||||
grantReadShared -> clientShared,
|
Mux(incoming.g_type === grantReadShared, clientShared,
|
||||||
grantReadExclusive -> clientExclusiveDirty,
|
clientExclusiveDirty)))(this)
|
||||||
grantReadExclusiveAck -> clientExclusiveDirty,
|
|
||||||
grantReadUncached -> clientInvalid,
|
|
||||||
grantWriteUncached -> clientInvalid,
|
|
||||||
grantReadWordUncached -> clientInvalid,
|
|
||||||
grantWriteWordUncached -> clientInvalid,
|
|
||||||
grantAtomicUncached -> clientInvalid
|
|
||||||
)))(this)
|
|
||||||
def clientMetadataOnProbe(incoming: Probe, m: ClientMetadata) = ClientMetadata(
|
def clientMetadataOnProbe(incoming: Probe, m: ClientMetadata) = ClientMetadata(
|
||||||
MuxLookup(incoming.p_type, m.state, Array(
|
MuxLookup(incoming.p_type, m.state, Array(
|
||||||
probeInvalidate -> clientInvalid,
|
probeInvalidate -> clientInvalid,
|
||||||
@ -576,21 +493,12 @@ class MSICoherence(dir: () => DirectoryRepresentation) extends CoherencePolicyWi
|
|||||||
val uncached = MasterMetadata(masterValid, m.sharers)(this)
|
val uncached = MasterMetadata(masterValid, m.sharers)(this)
|
||||||
MuxBundle(uncached, Array(
|
MuxBundle(uncached, Array(
|
||||||
g.is(grantReadShared) -> cached,
|
g.is(grantReadShared) -> cached,
|
||||||
g.is(grantReadExclusive) -> cached,
|
g.is(grantReadExclusive) -> cached
|
||||||
g.is(grantReadExclusiveAck) -> cached
|
|
||||||
))
|
))
|
||||||
}
|
}
|
||||||
|
|
||||||
def getUncachedReadAcquireType = acquireReadUncached
|
|
||||||
def getUncachedWriteAcquireType = acquireWriteUncached
|
|
||||||
def getUncachedReadWordAcquireType = acquireReadWordUncached
|
|
||||||
def getUncachedWriteWordAcquireType = acquireWriteWordUncached
|
|
||||||
def getUncachedAtomicAcquireType = acquireAtomicUncached
|
|
||||||
def isUncachedReadTransaction(acq: Acquire) = acq.a_type === acquireReadUncached
|
|
||||||
def getUncachedReadGrantType = grantReadUncached
|
|
||||||
|
|
||||||
def isVoluntary(rel: Release) = rel.r_type === releaseVoluntaryInvalidateData
|
def isVoluntary(rel: Release) = rel.r_type === releaseVoluntaryInvalidateData
|
||||||
def isVoluntary(gnt: Grant) = gnt.g_type === grantVoluntaryAck
|
def isVoluntary(gnt: Grant) = !gnt.uncached && gnt.g_type === grantVoluntaryAck
|
||||||
|
|
||||||
def getAcquireTypeOnPrimaryMiss(cmd: UInt, m: ClientMetadata): UInt = {
|
def getAcquireTypeOnPrimaryMiss(cmd: UInt, m: ClientMetadata): UInt = {
|
||||||
Mux(isWriteIntent(cmd), acquireReadExclusive, acquireReadShared)
|
Mux(isWriteIntent(cmd), acquireReadExclusive, acquireReadShared)
|
||||||
@ -615,80 +523,72 @@ class MSICoherence(dir: () => DirectoryRepresentation) extends CoherencePolicyWi
|
|||||||
}
|
}
|
||||||
|
|
||||||
def messageHasData(msg: SourcedMessage) = msg match {
|
def messageHasData(msg: SourcedMessage) = msg match {
|
||||||
case acq: Acquire => hasDataAcquireTypeVec.contains(acq.a_type)
|
case acq: Acquire => Mux(acq.uncached, Acquire.hasData(acq.a_type), Bool(false))
|
||||||
case grant: Grant => hasDataGrantTypeVec.contains(grant.g_type)
|
case gnt: Grant => Mux(gnt.uncached, Grant.hasData(gnt.g_type), hasDataGrantTypeVec.contains(gnt.g_type))
|
||||||
case rel: Release => hasDataReleaseTypeVec.contains(rel.r_type)
|
case rel: Release => hasDataReleaseTypeVec.contains(rel.r_type)
|
||||||
case _ => Bool(false)
|
case _ => Bool(false)
|
||||||
}
|
}
|
||||||
def messageUpdatesDataArray (reply: Grant): Bool = {
|
def messageUpdatesDataArray(g: Grant): Bool = {
|
||||||
(reply.g_type === grantReadShared || reply.g_type === grantReadExclusive)
|
Mux(g.uncached, Bool(false),
|
||||||
|
(g.g_type === grantReadShared || g.g_type === grantReadExclusive))
|
||||||
}
|
}
|
||||||
def messageIsUncached(acq: Acquire): Bool = uncachedAcquireTypeVec.contains(acq.a_type)
|
|
||||||
|
|
||||||
def isCoherenceConflict(addr1: UInt, addr2: UInt): Bool = (addr1 === addr2)
|
def isCoherenceConflict(addr1: UInt, addr2: UInt): Bool = (addr1 === addr2)
|
||||||
|
|
||||||
def getGrantType(a: Acquire, m: MasterMetadata): UInt = {
|
def getGrantType(a: Acquire, m: MasterMetadata): UInt = {
|
||||||
MuxLookup(a.a_type, grantReadUncached, Array(
|
Mux(a.uncached, getGrantTypeForUncached(a, m),
|
||||||
acquireReadShared -> Mux(!dir().none(m.sharers), grantReadShared, grantReadExclusive),
|
Mux(a.a_type === acquireReadShared,
|
||||||
acquireReadExclusive -> grantReadExclusive,
|
Mux(!dir().none(m.sharers), grantReadShared, grantReadExclusive),
|
||||||
acquireReadUncached -> grantReadUncached,
|
grantReadExclusive))
|
||||||
acquireWriteUncached -> grantWriteUncached,
|
|
||||||
acquireReadWordUncached -> grantReadWordUncached,
|
|
||||||
acquireWriteWordUncached -> grantWriteWordUncached,
|
|
||||||
acquireAtomicUncached -> grantAtomicUncached
|
|
||||||
))
|
|
||||||
}
|
|
||||||
def getGrantType(r: Release, m: MasterMetadata): UInt = {
|
|
||||||
MuxLookup(r.r_type, grantReadUncached, Array(
|
|
||||||
releaseVoluntaryInvalidateData -> grantVoluntaryAck
|
|
||||||
))
|
|
||||||
}
|
}
|
||||||
|
def getGrantTypeOnVoluntaryWriteback(m: MasterMetadata): UInt = grantVoluntaryAck
|
||||||
|
|
||||||
def getProbeType(a: Acquire, m: MasterMetadata): UInt = {
|
def getProbeType(a: Acquire, m: MasterMetadata): UInt = {
|
||||||
MuxLookup(a.a_type, probeCopy, Array(
|
Mux(a.uncached,
|
||||||
acquireReadShared -> probeDowngrade,
|
MuxLookup(a.a_type, probeCopy, Array(
|
||||||
acquireReadExclusive -> probeInvalidate,
|
Acquire.uncachedRead -> probeCopy,
|
||||||
acquireReadUncached -> probeCopy,
|
Acquire.uncachedWrite -> probeInvalidate,
|
||||||
acquireWriteUncached -> probeInvalidate
|
Acquire.uncachedAtomic -> probeInvalidate
|
||||||
))
|
)),
|
||||||
|
MuxLookup(a.a_type, probeInvalidate, Array(
|
||||||
|
acquireReadShared -> probeDowngrade,
|
||||||
|
acquireReadExclusive -> probeInvalidate
|
||||||
|
)))
|
||||||
}
|
}
|
||||||
def getProbeTypeOnVoluntaryWriteback: UInt = probeInvalidate
|
def getProbeTypeOnVoluntaryWriteback: UInt = probeInvalidate
|
||||||
|
|
||||||
def requiresOuterRead(a_type: UInt) = {
|
def requiresOuterRead(acq: Acquire, m: MasterMetadata) =
|
||||||
(a_type != acquireWriteUncached)
|
Mux(acq.uncached, Acquire.requiresOuterRead(acq.a_type), Bool(true))
|
||||||
}
|
def requiresOuterWrite(acq: Acquire, m: MasterMetadata) =
|
||||||
def requiresOuterWrite(a_type: UInt) = {
|
Mux(acq.uncached, Acquire.requiresOuterWrite(acq.a_type), Bool(false))
|
||||||
(a_type === acquireWriteUncached)
|
|
||||||
}
|
def requiresAckForGrant(g: Grant) = g.uncached || g.g_type != grantVoluntaryAck
|
||||||
def requiresAckForGrant(g_type: UInt) = g_type != grantVoluntaryAck
|
def requiresAckForRelease(r: Release) = Bool(false)
|
||||||
def requiresAckForRelease(r_type: UInt) = Bool(false)
|
def requiresSelfProbe(a: Acquire) = a.uncached && a.a_type === Acquire.uncachedRead
|
||||||
def requiresSelfProbe(a_type: UInt) = a_type === acquireReadUncached
|
def requiresProbes(a: Acquire, m: MasterMetadata) = !dir().none(m.sharers)
|
||||||
def requiresProbes(a_type: UInt, m: MasterMetadata) = !dir().none(m.sharers)
|
|
||||||
|
|
||||||
def pendingVoluntaryReleaseIsSufficient(r_type: UInt, p_type: UInt): Bool = (r_type === releaseVoluntaryInvalidateData)
|
def pendingVoluntaryReleaseIsSufficient(r_type: UInt, p_type: UInt): Bool = (r_type === releaseVoluntaryInvalidateData)
|
||||||
}
|
}
|
||||||
|
|
||||||
class MESICoherence(dir: () => DirectoryRepresentation) extends CoherencePolicyWithUncached(dir) {
|
class MESICoherence(dir: () => DirectoryRepresentation) extends CoherencePolicy(dir) {
|
||||||
def nClientStates = 4
|
def nClientStates = 4
|
||||||
def nMasterStates = 2
|
def nMasterStates = 2
|
||||||
def nAcquireTypes = 7
|
def nAcquireTypes = 2
|
||||||
def nProbeTypes = 3
|
def nProbeTypes = 3
|
||||||
def nReleaseTypes = 7
|
def nReleaseTypes = 7
|
||||||
def nGrantTypes = 9
|
def nGrantTypes = 4
|
||||||
|
|
||||||
val clientInvalid :: clientShared :: clientExclusiveClean :: clientExclusiveDirty :: Nil = Enum(UInt(), nClientStates)
|
val clientInvalid :: clientShared :: clientExclusiveClean :: clientExclusiveDirty :: Nil = Enum(UInt(), nClientStates)
|
||||||
//val masterInvalid :: masterShared :: masterExclusive :: Nil = Enum(UInt(), nMasterStates)
|
//val masterInvalid :: masterShared :: masterExclusive :: Nil = Enum(UInt(), nMasterStates)
|
||||||
val masterInvalid :: masterValid :: Nil = Enum(UInt(), nMasterStates)
|
val masterInvalid :: masterValid :: Nil = Enum(UInt(), nMasterStates)
|
||||||
|
|
||||||
val acquireReadShared :: acquireReadExclusive :: acquireReadUncached :: acquireWriteUncached :: acquireReadWordUncached :: acquireWriteWordUncached :: acquireAtomicUncached :: Nil = Enum(UInt(), nAcquireTypes)
|
val acquireReadShared :: acquireReadExclusive :: Nil = Enum(UInt(), nAcquireTypes)
|
||||||
val probeInvalidate :: probeDowngrade :: probeCopy :: Nil = Enum(UInt(), nProbeTypes)
|
val probeInvalidate :: probeDowngrade :: probeCopy :: Nil = Enum(UInt(), nProbeTypes)
|
||||||
val releaseVoluntaryInvalidateData :: releaseInvalidateData :: releaseDowngradeData :: releaseCopyData :: releaseInvalidateAck :: releaseDowngradeAck :: releaseCopyAck :: Nil = Enum(UInt(), nReleaseTypes)
|
val releaseVoluntaryInvalidateData :: releaseInvalidateData :: releaseDowngradeData :: releaseCopyData :: releaseInvalidateAck :: releaseDowngradeAck :: releaseCopyAck :: Nil = Enum(UInt(), nReleaseTypes)
|
||||||
val grantVoluntaryAck :: grantReadShared :: grantReadExclusive :: grantReadUncached :: grantWriteUncached :: grantReadExclusiveAck :: grantReadWordUncached :: grantWriteWordUncached :: grantAtomicUncached :: Nil = Enum(UInt(), nGrantTypes)
|
val grantVoluntaryAck :: grantReadShared :: grantReadExclusive :: grantReadExclusiveAck :: Nil = Enum(UInt(), nGrantTypes)
|
||||||
|
|
||||||
val uncachedAcquireTypeVec = Vec(acquireReadUncached, acquireWriteUncached, acquireReadWordUncached, acquireWriteWordUncached, acquireAtomicUncached)
|
|
||||||
val hasDataAcquireTypeVec = Vec(acquireWriteUncached, acquireWriteWordUncached, acquireAtomicUncached)
|
|
||||||
val hasDataReleaseTypeVec = Vec(releaseVoluntaryInvalidateData, releaseInvalidateData, releaseDowngradeData, releaseCopyData)
|
val hasDataReleaseTypeVec = Vec(releaseVoluntaryInvalidateData, releaseInvalidateData, releaseDowngradeData, releaseCopyData)
|
||||||
val hasDataGrantTypeVec = Vec(grantReadShared, grantReadExclusive, grantReadUncached, grantReadWordUncached, grantAtomicUncached)
|
val hasDataGrantTypeVec = Vec(grantReadShared, grantReadExclusive)
|
||||||
|
|
||||||
def isHit (cmd: UInt, m: ClientMetadata): Bool = {
|
def isHit (cmd: UInt, m: ClientMetadata): Bool = {
|
||||||
Mux(isWriteIntent(cmd), (m.state === clientExclusiveClean || m.state === clientExclusiveDirty),
|
Mux(isWriteIntent(cmd), (m.state === clientExclusiveClean || m.state === clientExclusiveDirty),
|
||||||
@ -701,7 +601,7 @@ class MESICoherence(dir: () => DirectoryRepresentation) extends CoherencePolicyW
|
|||||||
def isValid (m: MasterMetadata) = m.state != masterInvalid
|
def isValid (m: MasterMetadata) = m.state != masterInvalid
|
||||||
|
|
||||||
def needsTransactionOnSecondaryMiss(cmd: UInt, outstanding: Acquire): Bool = {
|
def needsTransactionOnSecondaryMiss(cmd: UInt, outstanding: Acquire): Bool = {
|
||||||
(isRead(cmd) && messageIsUncached(outstanding)) ||
|
(isRead(cmd) && outstanding.uncached) ||
|
||||||
(isWriteIntent(cmd) && (outstanding.a_type != acquireReadExclusive))
|
(isWriteIntent(cmd) && (outstanding.a_type != acquireReadExclusive))
|
||||||
}
|
}
|
||||||
def needsTransactionOnCacheControl(cmd: UInt, m: ClientMetadata): Bool = {
|
def needsTransactionOnCacheControl(cmd: UInt, m: ClientMetadata): Bool = {
|
||||||
@ -725,16 +625,13 @@ class MESICoherence(dir: () => DirectoryRepresentation) extends CoherencePolicyW
|
|||||||
)))(this)
|
)))(this)
|
||||||
def clientMetadataOnFlush = clientMetadataOnCacheControl(M_INV)
|
def clientMetadataOnFlush = clientMetadataOnCacheControl(M_INV)
|
||||||
def clientMetadataOnGrant(incoming: Grant, outstanding: Acquire) = ClientMetadata(
|
def clientMetadataOnGrant(incoming: Grant, outstanding: Acquire) = ClientMetadata(
|
||||||
MuxLookup(incoming.g_type, clientInvalid, Array(
|
Mux(incoming.uncached, clientInvalid,
|
||||||
grantReadShared -> clientShared,
|
MuxLookup(incoming.g_type, clientInvalid, Array(
|
||||||
grantReadExclusive -> Mux(outstanding.a_type === acquireReadExclusive, clientExclusiveDirty, clientExclusiveClean),
|
grantReadShared -> clientShared,
|
||||||
grantReadExclusiveAck -> clientExclusiveDirty,
|
grantReadExclusive -> Mux(outstanding.a_type === acquireReadExclusive,
|
||||||
grantReadUncached -> clientInvalid,
|
clientExclusiveDirty, clientExclusiveClean),
|
||||||
grantWriteUncached -> clientInvalid,
|
grantReadExclusiveAck -> clientExclusiveDirty
|
||||||
grantReadWordUncached -> clientInvalid,
|
))))(this)
|
||||||
grantWriteWordUncached -> clientInvalid,
|
|
||||||
grantAtomicUncached -> clientInvalid
|
|
||||||
)))(this)
|
|
||||||
def clientMetadataOnProbe(incoming: Probe, m: ClientMetadata) = ClientMetadata(
|
def clientMetadataOnProbe(incoming: Probe, m: ClientMetadata) = ClientMetadata(
|
||||||
MuxLookup(incoming.p_type, m.state, Array(
|
MuxLookup(incoming.p_type, m.state, Array(
|
||||||
probeInvalidate -> clientInvalid,
|
probeInvalidate -> clientInvalid,
|
||||||
@ -760,16 +657,8 @@ class MESICoherence(dir: () => DirectoryRepresentation) extends CoherencePolicyW
|
|||||||
))
|
))
|
||||||
}
|
}
|
||||||
|
|
||||||
def getUncachedReadAcquireType = acquireReadUncached
|
|
||||||
def getUncachedWriteAcquireType = acquireWriteUncached
|
|
||||||
def getUncachedReadWordAcquireType = acquireReadWordUncached
|
|
||||||
def getUncachedWriteWordAcquireType = acquireWriteWordUncached
|
|
||||||
def getUncachedAtomicAcquireType = acquireAtomicUncached
|
|
||||||
def isUncachedReadTransaction(acq: Acquire) = acq.a_type === acquireReadUncached
|
|
||||||
def getUncachedReadGrantType = grantReadUncached
|
|
||||||
|
|
||||||
def isVoluntary(rel: Release) = rel.r_type === releaseVoluntaryInvalidateData
|
def isVoluntary(rel: Release) = rel.r_type === releaseVoluntaryInvalidateData
|
||||||
def isVoluntary(gnt: Grant) = gnt.g_type === grantVoluntaryAck
|
def isVoluntary(gnt: Grant) = !gnt.uncached && gnt.g_type === grantVoluntaryAck
|
||||||
|
|
||||||
def getAcquireTypeOnPrimaryMiss(cmd: UInt, m: ClientMetadata): UInt = {
|
def getAcquireTypeOnPrimaryMiss(cmd: UInt, m: ClientMetadata): UInt = {
|
||||||
Mux(isWriteIntent(cmd), acquireReadExclusive, acquireReadShared)
|
Mux(isWriteIntent(cmd), acquireReadExclusive, acquireReadShared)
|
||||||
@ -794,84 +683,71 @@ class MESICoherence(dir: () => DirectoryRepresentation) extends CoherencePolicyW
|
|||||||
}
|
}
|
||||||
|
|
||||||
def messageHasData(msg: SourcedMessage) = msg match {
|
def messageHasData(msg: SourcedMessage) = msg match {
|
||||||
case acq: Acquire => hasDataAcquireTypeVec.contains(acq.a_type)
|
case acq: Acquire => Mux(acq.uncached, Acquire.hasData(acq.a_type), Bool(false))
|
||||||
case grant: Grant => hasDataGrantTypeVec.contains(grant.g_type)
|
case gnt: Grant => Mux(gnt.uncached, Grant.hasData(gnt.g_type), hasDataGrantTypeVec.contains(gnt.g_type))
|
||||||
case rel: Release => hasDataReleaseTypeVec.contains(rel.r_type)
|
case rel: Release => hasDataReleaseTypeVec.contains(rel.r_type)
|
||||||
case _ => Bool(false)
|
case _ => Bool(false)
|
||||||
}
|
}
|
||||||
def messageUpdatesDataArray (reply: Grant): Bool = {
|
def messageUpdatesDataArray(g: Grant): Bool = {
|
||||||
(reply.g_type === grantReadShared || reply.g_type === grantReadExclusive)
|
Mux(g.uncached, Bool(false),
|
||||||
|
(g.g_type === grantReadShared || g.g_type === grantReadExclusive))
|
||||||
}
|
}
|
||||||
def messageIsUncached(acq: Acquire): Bool = uncachedAcquireTypeVec.contains(acq.a_type)
|
|
||||||
|
|
||||||
def isCoherenceConflict(addr1: UInt, addr2: UInt): Bool = (addr1 === addr2)
|
def isCoherenceConflict(addr1: UInt, addr2: UInt): Bool = (addr1 === addr2)
|
||||||
|
|
||||||
def getGrantType(a: Acquire, m: MasterMetadata): UInt = {
|
def getGrantType(a: Acquire, m: MasterMetadata): UInt = {
|
||||||
MuxLookup(a.a_type, grantReadUncached, Array(
|
Mux(a.uncached, getGrantTypeForUncached(a, m),
|
||||||
acquireReadShared -> Mux(!dir().none(m.sharers), grantReadShared, grantReadExclusive),
|
Mux(a.a_type === acquireReadShared,
|
||||||
acquireReadExclusive -> grantReadExclusive,
|
Mux(!dir().none(m.sharers), grantReadShared, grantReadExclusive),
|
||||||
acquireReadUncached -> grantReadUncached,
|
grantReadExclusive))
|
||||||
acquireWriteUncached -> grantWriteUncached,
|
|
||||||
acquireReadWordUncached -> grantReadWordUncached,
|
|
||||||
acquireWriteWordUncached -> grantWriteWordUncached,
|
|
||||||
acquireAtomicUncached -> grantAtomicUncached
|
|
||||||
))
|
|
||||||
}
|
}
|
||||||
def getGrantType(r: Release, m: MasterMetadata): UInt = {
|
def getGrantTypeOnVoluntaryWriteback(m: MasterMetadata): UInt = grantVoluntaryAck
|
||||||
MuxLookup(r.r_type, grantReadUncached, Array(
|
|
||||||
releaseVoluntaryInvalidateData -> grantVoluntaryAck
|
|
||||||
))
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
def getProbeType(a: Acquire, m: MasterMetadata): UInt = {
|
def getProbeType(a: Acquire, m: MasterMetadata): UInt = {
|
||||||
MuxLookup(a.a_type, probeCopy, Array(
|
Mux(a.uncached,
|
||||||
acquireReadShared -> probeDowngrade,
|
MuxLookup(a.a_type, probeCopy, Array(
|
||||||
acquireReadExclusive -> probeInvalidate,
|
Acquire.uncachedRead -> probeCopy,
|
||||||
acquireReadUncached -> probeCopy,
|
Acquire.uncachedWrite -> probeInvalidate,
|
||||||
acquireWriteUncached -> probeInvalidate,
|
Acquire.uncachedAtomic -> probeInvalidate
|
||||||
acquireReadWordUncached -> probeCopy,
|
)),
|
||||||
acquireWriteWordUncached -> probeInvalidate,
|
MuxLookup(a.a_type, probeCopy, Array(
|
||||||
acquireAtomicUncached -> probeInvalidate
|
acquireReadShared -> probeDowngrade,
|
||||||
))
|
acquireReadExclusive -> probeInvalidate
|
||||||
|
)))
|
||||||
}
|
}
|
||||||
def getProbeTypeOnVoluntaryWriteback: UInt = probeInvalidate
|
def getProbeTypeOnVoluntaryWriteback: UInt = probeInvalidate
|
||||||
|
|
||||||
def requiresOuterRead(a_type: UInt) = {
|
def requiresOuterRead(acq: Acquire, m: MasterMetadata) =
|
||||||
(a_type != acquireWriteUncached)
|
Mux(acq.uncached, Acquire.requiresOuterRead(acq.a_type), Bool(true))
|
||||||
}
|
def requiresOuterWrite(acq: Acquire, m: MasterMetadata) =
|
||||||
def requiresOuterWrite(a_type: UInt) = {
|
Mux(acq.uncached, Acquire.requiresOuterWrite(acq.a_type), Bool(false))
|
||||||
(a_type === acquireWriteUncached)
|
|
||||||
}
|
|
||||||
|
|
||||||
def requiresAckForGrant(g_type: UInt) = g_type != grantVoluntaryAck
|
def requiresAckForGrant(g: Grant) = g.uncached || g.g_type != grantVoluntaryAck
|
||||||
def requiresAckForRelease(r_type: UInt) = Bool(false)
|
def requiresAckForRelease(r: Release) = Bool(false)
|
||||||
def requiresSelfProbe(a_type: UInt) = a_type === acquireReadUncached
|
def requiresSelfProbe(a: Acquire) = a.uncached && a.a_type === Acquire.uncachedRead
|
||||||
def requiresProbes(a_type: UInt, m: MasterMetadata) = !dir().none(m.sharers)
|
def requiresProbes(a: Acquire, m: MasterMetadata) = !dir().none(m.sharers)
|
||||||
|
|
||||||
def pendingVoluntaryReleaseIsSufficient(r_type: UInt, p_type: UInt): Bool = (r_type === releaseVoluntaryInvalidateData)
|
def pendingVoluntaryReleaseIsSufficient(r_type: UInt, p_type: UInt): Bool = (r_type === releaseVoluntaryInvalidateData)
|
||||||
}
|
}
|
||||||
|
|
||||||
class MigratoryCoherence(dir: () => DirectoryRepresentation) extends CoherencePolicyWithUncached(dir) {
|
class MigratoryCoherence(dir: () => DirectoryRepresentation) extends CoherencePolicy(dir) {
|
||||||
def nClientStates = 7
|
def nClientStates = 7
|
||||||
def nMasterStates = 2
|
def nMasterStates = 2
|
||||||
def nAcquireTypes = 8
|
def nAcquireTypes = 3
|
||||||
def nProbeTypes = 4
|
def nProbeTypes = 4
|
||||||
def nReleaseTypes = 11
|
def nReleaseTypes = 11
|
||||||
def nGrantTypes = 9
|
def nGrantTypes = 5
|
||||||
|
|
||||||
val clientInvalid :: clientShared :: clientExclusiveClean :: clientExclusiveDirty :: clientSharedByTwo :: clientMigratoryClean :: clientMigratoryDirty :: Nil = Enum(UInt(), nClientStates)
|
val clientInvalid :: clientShared :: clientExclusiveClean :: clientExclusiveDirty :: clientSharedByTwo :: clientMigratoryClean :: clientMigratoryDirty :: Nil = Enum(UInt(), nClientStates)
|
||||||
//val masterInvalid :: masterShared :: masterExclusive :: Nil = Enum(UInt(), nMasterStates)
|
//val masterInvalid :: masterShared :: masterExclusive :: Nil = Enum(UInt(), nMasterStates)
|
||||||
val masterInvalid :: masterValid :: Nil = Enum(UInt(), nMasterStates)
|
val masterInvalid :: masterValid :: Nil = Enum(UInt(), nMasterStates)
|
||||||
|
|
||||||
val acquireReadShared :: acquireReadExclusive :: acquireReadUncached :: acquireWriteUncached :: acquireReadWordUncached :: acquireWriteWordUncached :: acquireAtomicUncached :: acquireInvalidateOthers :: Nil = Enum(UInt(), nAcquireTypes)
|
val acquireReadShared :: acquireReadExclusive :: acquireInvalidateOthers :: Nil = Enum(UInt(), nAcquireTypes)
|
||||||
val probeInvalidate :: probeDowngrade :: probeCopy :: probeInvalidateOthers :: Nil = Enum(UInt(), nProbeTypes)
|
val probeInvalidate :: probeDowngrade :: probeCopy :: probeInvalidateOthers :: Nil = Enum(UInt(), nProbeTypes)
|
||||||
val releaseVoluntaryInvalidateData :: releaseInvalidateData :: releaseDowngradeData :: releaseCopyData :: releaseInvalidateAck :: releaseDowngradeAck :: releaseCopyAck :: releaseDowngradeDataMigratory :: releaseDowngradeAckHasCopy :: releaseInvalidateDataMigratory :: releaseInvalidateAckMigratory :: Nil = Enum(UInt(), nReleaseTypes)
|
val releaseVoluntaryInvalidateData :: releaseInvalidateData :: releaseDowngradeData :: releaseCopyData :: releaseInvalidateAck :: releaseDowngradeAck :: releaseCopyAck :: releaseDowngradeDataMigratory :: releaseDowngradeAckHasCopy :: releaseInvalidateDataMigratory :: releaseInvalidateAckMigratory :: Nil = Enum(UInt(), nReleaseTypes)
|
||||||
val grantVoluntaryAck :: grantReadShared :: grantReadExclusive :: grantReadUncached :: grantWriteUncached :: grantReadExclusiveAck :: grantReadWordUncached :: grantWriteWordUncached :: grantAtomicUncached :: grantReadMigratory :: Nil = Enum(UInt(), nGrantTypes)
|
val grantVoluntaryAck :: grantReadShared :: grantReadExclusive :: grantReadExclusiveAck :: grantReadMigratory :: Nil = Enum(UInt(), nGrantTypes)
|
||||||
|
|
||||||
val uncachedAcquireTypeVec = Vec(acquireReadUncached, acquireWriteUncached, acquireReadWordUncached, acquireWriteWordUncached, acquireAtomicUncached)
|
val hasDataGrantTypeVec = Vec(grantReadShared, grantReadExclusive, grantReadMigratory)
|
||||||
val hasDataAcquireTypeVec = Vec(acquireWriteUncached, acquireWriteWordUncached, acquireAtomicUncached)
|
|
||||||
val hasDataGrantTypeVec = Vec(grantReadShared, grantReadExclusive, grantReadUncached, grantReadMigratory, grantReadWordUncached, grantAtomicUncached)
|
|
||||||
val hasDataReleaseTypeVec = Vec(releaseVoluntaryInvalidateData, releaseInvalidateData, releaseDowngradeData, releaseCopyData, releaseInvalidateDataMigratory, releaseDowngradeDataMigratory)
|
val hasDataReleaseTypeVec = Vec(releaseVoluntaryInvalidateData, releaseInvalidateData, releaseDowngradeData, releaseCopyData, releaseInvalidateDataMigratory, releaseDowngradeDataMigratory)
|
||||||
|
|
||||||
def isHit (cmd: UInt, m: ClientMetadata): Bool = {
|
def isHit (cmd: UInt, m: ClientMetadata): Bool = {
|
||||||
@ -884,7 +760,7 @@ class MigratoryCoherence(dir: () => DirectoryRepresentation) extends CoherencePo
|
|||||||
def isValid (m: MasterMetadata) = m.state != masterInvalid
|
def isValid (m: MasterMetadata) = m.state != masterInvalid
|
||||||
|
|
||||||
def needsTransactionOnSecondaryMiss(cmd: UInt, outstanding: Acquire): Bool = {
|
def needsTransactionOnSecondaryMiss(cmd: UInt, outstanding: Acquire): Bool = {
|
||||||
(isRead(cmd) && messageIsUncached(outstanding)) ||
|
(isRead(cmd) && outstanding.uncached) ||
|
||||||
(isWriteIntent(cmd) && (outstanding.a_type != acquireReadExclusive && outstanding.a_type != acquireInvalidateOthers))
|
(isWriteIntent(cmd) && (outstanding.a_type != acquireReadExclusive && outstanding.a_type != acquireInvalidateOthers))
|
||||||
}
|
}
|
||||||
def needsTransactionOnCacheControl(cmd: UInt, m: ClientMetadata): Bool = {
|
def needsTransactionOnCacheControl(cmd: UInt, m: ClientMetadata): Bool = {
|
||||||
@ -909,22 +785,18 @@ class MigratoryCoherence(dir: () => DirectoryRepresentation) extends CoherencePo
|
|||||||
)))(this)
|
)))(this)
|
||||||
def clientMetadataOnFlush = clientMetadataOnCacheControl(M_INV)
|
def clientMetadataOnFlush = clientMetadataOnCacheControl(M_INV)
|
||||||
def clientMetadataOnGrant(incoming: Grant, outstanding: Acquire) = ClientMetadata(
|
def clientMetadataOnGrant(incoming: Grant, outstanding: Acquire) = ClientMetadata(
|
||||||
MuxLookup(incoming.g_type, clientInvalid, Array(
|
Mux(incoming.uncached, clientInvalid,
|
||||||
grantReadShared -> clientShared,
|
MuxLookup(incoming.g_type, clientInvalid, Array(
|
||||||
grantReadExclusive -> MuxLookup(outstanding.a_type, clientExclusiveDirty, Array(
|
grantReadShared -> clientShared,
|
||||||
acquireReadExclusive -> clientExclusiveDirty,
|
grantReadExclusive -> MuxLookup(outstanding.a_type, clientExclusiveDirty, Array(
|
||||||
acquireReadShared -> clientExclusiveClean)),
|
acquireReadExclusive -> clientExclusiveDirty,
|
||||||
grantReadExclusiveAck -> clientExclusiveDirty,
|
acquireReadShared -> clientExclusiveClean)),
|
||||||
grantReadUncached -> clientInvalid,
|
grantReadExclusiveAck -> clientExclusiveDirty,
|
||||||
grantWriteUncached -> clientInvalid,
|
grantReadMigratory -> MuxLookup(outstanding.a_type, clientMigratoryDirty, Array(
|
||||||
grantReadWordUncached -> clientInvalid,
|
acquireInvalidateOthers -> clientMigratoryDirty,
|
||||||
grantWriteWordUncached -> clientInvalid,
|
acquireReadExclusive -> clientMigratoryDirty,
|
||||||
grantAtomicUncached -> clientInvalid,
|
acquireReadShared -> clientMigratoryClean))
|
||||||
grantReadMigratory -> MuxLookup(outstanding.a_type, clientMigratoryDirty, Array(
|
))))(this)
|
||||||
acquireInvalidateOthers -> clientMigratoryDirty,
|
|
||||||
acquireReadExclusive -> clientMigratoryDirty,
|
|
||||||
acquireReadShared -> clientMigratoryClean))
|
|
||||||
)))(this)
|
|
||||||
def clientMetadataOnProbe(incoming: Probe, m: ClientMetadata) = ClientMetadata(
|
def clientMetadataOnProbe(incoming: Probe, m: ClientMetadata) = ClientMetadata(
|
||||||
MuxLookup(incoming.p_type, m.state, Array(
|
MuxLookup(incoming.p_type, m.state, Array(
|
||||||
probeInvalidate -> clientInvalid,
|
probeInvalidate -> clientInvalid,
|
||||||
@ -960,16 +832,8 @@ class MigratoryCoherence(dir: () => DirectoryRepresentation) extends CoherencePo
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
def getUncachedReadAcquireType = acquireReadUncached
|
|
||||||
def getUncachedWriteAcquireType = acquireWriteUncached
|
|
||||||
def getUncachedReadWordAcquireType = acquireReadWordUncached
|
|
||||||
def getUncachedWriteWordAcquireType = acquireWriteWordUncached
|
|
||||||
def getUncachedAtomicAcquireType = acquireAtomicUncached
|
|
||||||
def isUncachedReadTransaction(acq: Acquire) = acq.a_type === acquireReadUncached
|
|
||||||
def getUncachedReadGrantType = grantReadUncached
|
|
||||||
|
|
||||||
def isVoluntary(rel: Release) = rel.r_type === releaseVoluntaryInvalidateData
|
def isVoluntary(rel: Release) = rel.r_type === releaseVoluntaryInvalidateData
|
||||||
def isVoluntary(gnt: Grant) = gnt.g_type === grantVoluntaryAck
|
def isVoluntary(gnt: Grant) = !gnt.uncached && gnt.g_type === grantVoluntaryAck
|
||||||
|
|
||||||
def getAcquireTypeOnPrimaryMiss(cmd: UInt, m: ClientMetadata): UInt = {
|
def getAcquireTypeOnPrimaryMiss(cmd: UInt, m: ClientMetadata): UInt = {
|
||||||
Mux(isWriteIntent(cmd), Mux(m.state === clientInvalid, acquireReadExclusive, acquireInvalidateOthers), acquireReadShared)
|
Mux(isWriteIntent(cmd), Mux(m.state === clientInvalid, acquireReadExclusive, acquireInvalidateOthers), acquireReadShared)
|
||||||
@ -996,60 +860,52 @@ class MigratoryCoherence(dir: () => DirectoryRepresentation) extends CoherencePo
|
|||||||
}
|
}
|
||||||
|
|
||||||
def messageHasData(msg: SourcedMessage) = msg match {
|
def messageHasData(msg: SourcedMessage) = msg match {
|
||||||
case acq: Acquire => hasDataAcquireTypeVec.contains(acq.a_type)
|
case acq: Acquire => Mux(acq.uncached, Acquire.hasData(acq.a_type), Bool(false))
|
||||||
case grant: Grant => hasDataGrantTypeVec.contains(grant.g_type)
|
case gnt: Grant => Mux(gnt.uncached, Grant.hasData(gnt.g_type), hasDataGrantTypeVec.contains(gnt.g_type))
|
||||||
case rel: Release => hasDataReleaseTypeVec.contains(rel.r_type)
|
case rel: Release => hasDataReleaseTypeVec.contains(rel.r_type)
|
||||||
case _ => Bool(false)
|
case _ => Bool(false)
|
||||||
}
|
}
|
||||||
def messageUpdatesDataArray (reply: Grant): Bool = Vec(grantReadShared, grantReadExclusive, grantReadMigratory).contains(reply.g_type)
|
def messageUpdatesDataArray(g: Grant): Bool = {
|
||||||
def messageIsUncached(acq: Acquire): Bool = uncachedAcquireTypeVec.contains(acq.a_type)
|
Mux(g.uncached, Bool(false),
|
||||||
|
Vec(grantReadShared, grantReadExclusive, grantReadMigratory).contains(g.g_type))
|
||||||
|
}
|
||||||
|
|
||||||
def isCoherenceConflict(addr1: UInt, addr2: UInt): Bool = (addr1 === addr2)
|
def isCoherenceConflict(addr1: UInt, addr2: UInt): Bool = (addr1 === addr2)
|
||||||
|
|
||||||
def getGrantType(a: Acquire, m: MasterMetadata): UInt = {
|
def getGrantType(a: Acquire, m: MasterMetadata): UInt = {
|
||||||
MuxLookup(a.a_type, grantReadUncached, Array(
|
Mux(a.uncached, getGrantTypeForUncached(a, m),
|
||||||
acquireReadShared -> Mux(!dir().none(m.sharers), grantReadShared, grantReadExclusive), //TODO: what is count? Depend on release.p_type???
|
MuxLookup(a.a_type, grantReadShared, Array(
|
||||||
acquireReadExclusive -> grantReadExclusive,
|
acquireReadShared -> Mux(!dir().none(m.sharers), grantReadShared, grantReadExclusive),
|
||||||
acquireReadUncached -> grantReadUncached,
|
acquireReadExclusive -> grantReadExclusive,
|
||||||
acquireWriteUncached -> grantWriteUncached,
|
acquireInvalidateOthers -> grantReadExclusiveAck //TODO: add this to MESI for broadcast?
|
||||||
acquireReadWordUncached -> grantReadWordUncached,
|
)))
|
||||||
acquireWriteWordUncached -> grantWriteWordUncached,
|
|
||||||
acquireAtomicUncached -> grantAtomicUncached,
|
|
||||||
acquireInvalidateOthers -> grantReadExclusiveAck //TODO: add this to MESI?
|
|
||||||
))
|
|
||||||
}
|
}
|
||||||
def getGrantType(r: Release, m: MasterMetadata): UInt = {
|
def getGrantTypeOnVoluntaryWriteback(m: MasterMetadata): UInt = grantVoluntaryAck
|
||||||
MuxLookup(r.r_type, grantReadUncached, Array(
|
|
||||||
releaseVoluntaryInvalidateData -> grantVoluntaryAck
|
|
||||||
))
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
def getProbeType(a: Acquire, m: MasterMetadata): UInt = {
|
def getProbeType(a: Acquire, m: MasterMetadata): UInt = {
|
||||||
MuxLookup(a.a_type, probeCopy, Array(
|
Mux(a.uncached,
|
||||||
acquireReadShared -> probeDowngrade,
|
MuxLookup(a.a_type, probeCopy, Array(
|
||||||
acquireReadExclusive -> probeInvalidate,
|
Acquire.uncachedRead -> probeCopy,
|
||||||
acquireReadUncached -> probeCopy,
|
Acquire.uncachedWrite -> probeInvalidate,
|
||||||
acquireWriteUncached -> probeInvalidate,
|
Acquire.uncachedAtomic -> probeInvalidate
|
||||||
acquireReadWordUncached -> probeCopy,
|
)),
|
||||||
acquireWriteWordUncached -> probeInvalidate,
|
MuxLookup(a.a_type, probeCopy, Array(
|
||||||
acquireAtomicUncached -> probeInvalidate,
|
acquireReadShared -> probeDowngrade,
|
||||||
acquireInvalidateOthers -> probeInvalidateOthers
|
acquireReadExclusive -> probeInvalidate,
|
||||||
))
|
acquireInvalidateOthers -> probeInvalidateOthers
|
||||||
|
)))
|
||||||
}
|
}
|
||||||
def getProbeTypeOnVoluntaryWriteback: UInt = probeInvalidate
|
def getProbeTypeOnVoluntaryWriteback: UInt = probeInvalidate
|
||||||
|
|
||||||
def requiresOuterRead(a_type: UInt) = {
|
def requiresOuterRead(acq: Acquire, m: MasterMetadata) =
|
||||||
(a_type != acquireWriteUncached && a_type != acquireInvalidateOthers)
|
Mux(acq.uncached, Acquire.requiresOuterRead(acq.a_type), acq.a_type != acquireInvalidateOthers)
|
||||||
}
|
def requiresOuterWrite(acq: Acquire, m: MasterMetadata) =
|
||||||
def requiresOuterWrite(a_type: UInt) = {
|
Mux(acq.uncached, Acquire.requiresOuterWrite(acq.a_type), Bool(false))
|
||||||
(a_type === acquireWriteUncached || a_type === acquireWriteWordUncached || a_type === acquireAtomicUncached)
|
|
||||||
}
|
|
||||||
|
|
||||||
def requiresAckForGrant(g_type: UInt) = g_type != grantVoluntaryAck
|
def requiresAckForGrant(g: Grant) = g.uncached || g.g_type != grantVoluntaryAck
|
||||||
def requiresAckForRelease(r_type: UInt) = Bool(false)
|
def requiresAckForRelease(r: Release) = Bool(false)
|
||||||
def requiresSelfProbe(a_type: UInt) = a_type === acquireReadUncached
|
def requiresSelfProbe(a: Acquire) = a.uncached && a.a_type === Acquire.uncachedRead
|
||||||
def requiresProbes(a_type: UInt, m: MasterMetadata) = !dir().none(m.sharers)
|
def requiresProbes(a: Acquire, m: MasterMetadata) = !dir().none(m.sharers)
|
||||||
|
|
||||||
def pendingVoluntaryReleaseIsSufficient(r_type: UInt, p_type: UInt): Bool = (r_type === releaseVoluntaryInvalidateData)
|
def pendingVoluntaryReleaseIsSufficient(r_type: UInt, p_type: UInt): Bool = (r_type === releaseVoluntaryInvalidateData)
|
||||||
}
|
}
|
||||||
|
@ -16,7 +16,9 @@ trait MemoryOpConstants {
|
|||||||
val MT_BU = Bits("b100")
|
val MT_BU = Bits("b100")
|
||||||
val MT_HU = Bits("b101")
|
val MT_HU = Bits("b101")
|
||||||
val MT_WU = Bits("b110")
|
val MT_WU = Bits("b110")
|
||||||
|
val MT_CB = Bits("b111") // cache block
|
||||||
|
|
||||||
|
val NUM_XA_OPS = 9
|
||||||
val M_SZ = 5
|
val M_SZ = 5
|
||||||
val M_X = Bits("b?????");
|
val M_X = Bits("b?????");
|
||||||
val M_XRD = Bits("b00000"); // int load
|
val M_XRD = Bits("b00000"); // int load
|
||||||
|
@ -135,7 +135,7 @@ class HTIF(pcr_RESET: Int) extends Module with HTIFParameters {
|
|||||||
mem_acked := Bool(true)
|
mem_acked := Bool(true)
|
||||||
mem_gxid := io.mem.grant.bits.payload.master_xact_id
|
mem_gxid := io.mem.grant.bits.payload.master_xact_id
|
||||||
mem_gsrc := io.mem.grant.bits.header.src
|
mem_gsrc := io.mem.grant.bits.header.src
|
||||||
mem_needs_ack := co.requiresAckForGrant(io.mem.grant.bits.payload.g_type)
|
mem_needs_ack := co.requiresAckForGrant(io.mem.grant.bits.payload)
|
||||||
}
|
}
|
||||||
io.mem.grant.ready := Bool(true)
|
io.mem.grant.ready := Bool(true)
|
||||||
|
|
||||||
@ -193,8 +193,8 @@ class HTIF(pcr_RESET: Int) extends Module with HTIFParameters {
|
|||||||
acq_q.io.enq.valid := state === state_mem_rreq || state === state_mem_wreq
|
acq_q.io.enq.valid := state === state_mem_rreq || state === state_mem_wreq
|
||||||
val init_addr = addr.toUInt >> UInt(offsetBits-3)
|
val init_addr = addr.toUInt >> UInt(offsetBits-3)
|
||||||
acq_q.io.enq.bits := Mux(cmd === cmd_writemem,
|
acq_q.io.enq.bits := Mux(cmd === cmd_writemem,
|
||||||
Acquire(co.getUncachedWriteAcquireType, init_addr, UInt(0)),
|
UncachedWrite(init_addr, UInt(0)),
|
||||||
Acquire(co.getUncachedReadAcquireType, init_addr, UInt(0)))
|
UncachedRead(init_addr))
|
||||||
io.mem.acquire.valid := acq_q.io.deq.valid
|
io.mem.acquire.valid := acq_q.io.deq.valid
|
||||||
acq_q.io.deq.ready := io.mem.acquire.ready
|
acq_q.io.deq.ready := io.mem.acquire.ready
|
||||||
io.mem.acquire.bits.payload := acq_q.io.deq.bits
|
io.mem.acquire.bits.payload := acq_q.io.deq.bits
|
||||||
|
@ -268,7 +268,7 @@ class MemIOUncachedTileLinkIOConverter(qDepth: Int) extends Module {
|
|||||||
// Aggregate incoming MemIO responses into TL Grants
|
// Aggregate incoming MemIO responses into TL Grants
|
||||||
io.mem.resp.ready := !active_in || cnt_in < UInt(cnt_max)
|
io.mem.resp.ready := !active_in || cnt_in < UInt(cnt_max)
|
||||||
io.uncached.grant.valid := active_in && (cnt_in === UInt(cnt_max))
|
io.uncached.grant.valid := active_in && (cnt_in === UInt(cnt_max))
|
||||||
io.uncached.grant.bits.payload := Grant(co.getUncachedReadGrantType, tag_in, UInt(0), buf_in)
|
io.uncached.grant.bits.payload := Grant(Bool(true), Grant.uncachedRead, tag_in, UInt(0), buf_in)
|
||||||
when(!active_in && io.mem.resp.valid) {
|
when(!active_in && io.mem.resp.valid) {
|
||||||
active_in := Bool(true)
|
active_in := Bool(true)
|
||||||
cnt_in := UInt(1)
|
cnt_in := UInt(1)
|
||||||
|
@ -2,65 +2,90 @@
|
|||||||
|
|
||||||
package uncore
|
package uncore
|
||||||
import Chisel._
|
import Chisel._
|
||||||
|
import scala.math.max
|
||||||
|
|
||||||
case object TLId extends Field[String]
|
case object TLId extends Field[String]
|
||||||
case object TLCoherence extends Field[CoherencePolicyWithUncached]
|
case object TLCoherence extends Field[CoherencePolicy]
|
||||||
case object TLAddrBits extends Field[Int]
|
case object TLAddrBits extends Field[Int]
|
||||||
case object TLMasterXactIdBits extends Field[Int]
|
case object TLMasterXactIdBits extends Field[Int]
|
||||||
case object TLClientXactIdBits extends Field[Int]
|
case object TLClientXactIdBits extends Field[Int]
|
||||||
case object TLDataBits extends Field[Int]
|
case object TLDataBits extends Field[Int]
|
||||||
case object TLWriteMaskBits extends Field[Int]
|
|
||||||
case object TLWordAddrBits extends Field[Int]
|
|
||||||
case object TLAtomicOpBits extends Field[Int]
|
|
||||||
|
|
||||||
trait HasPhysicalAddress extends Bundle {
|
abstract trait TileLinkParameters extends UsesParameters {
|
||||||
val addr = UInt(width = params(TLAddrBits))
|
val tlAddrBits = params(TLAddrBits)
|
||||||
|
val tlClientXactIdBits = params(TLClientXactIdBits)
|
||||||
|
val tlMasterXactIdBits = params(TLMasterXactIdBits)
|
||||||
|
val tlDataBits = params(TLDataBits)
|
||||||
|
val tlWriteMaskBits = if(tlDataBits/8 < 1) 1 else tlDataBits
|
||||||
|
val tlSubblockAddrBits = log2Up(tlWriteMaskBits)
|
||||||
|
val tlAtomicOpcodeBits = log2Up(NUM_XA_OPS)
|
||||||
|
val tlUncachedOperandSizeBits = MT_SZ
|
||||||
|
val tlSubblockUnionBits = max(tlWriteMaskBits,
|
||||||
|
(tlSubblockAddrBits +
|
||||||
|
tlUncachedOperandSizeBits +
|
||||||
|
tlAtomicOpcodeBits))
|
||||||
}
|
}
|
||||||
|
|
||||||
trait HasClientTransactionId extends Bundle {
|
class TLBundle extends Bundle with TileLinkParameters
|
||||||
val client_xact_id = Bits(width = params(TLClientXactIdBits))
|
|
||||||
|
trait HasPhysicalAddress extends TLBundle {
|
||||||
|
val addr = UInt(width = tlAddrBits)
|
||||||
}
|
}
|
||||||
|
|
||||||
trait HasMasterTransactionId extends Bundle {
|
trait HasClientTransactionId extends TLBundle {
|
||||||
val master_xact_id = Bits(width = params(TLMasterXactIdBits))
|
val client_xact_id = Bits(width = tlClientXactIdBits)
|
||||||
}
|
}
|
||||||
|
|
||||||
trait HasTileLinkData extends Bundle {
|
trait HasMasterTransactionId extends TLBundle {
|
||||||
val data = Bits(width = params(TLDataBits))
|
val master_xact_id = Bits(width = tlMasterXactIdBits)
|
||||||
}
|
}
|
||||||
|
|
||||||
trait SourcedMessage extends Bundle
|
trait HasTileLinkData extends TLBundle {
|
||||||
|
val data = UInt(width = tlDataBits)
|
||||||
|
}
|
||||||
|
|
||||||
|
trait SourcedMessage extends TLBundle
|
||||||
trait ClientSourcedMessage extends SourcedMessage
|
trait ClientSourcedMessage extends SourcedMessage
|
||||||
trait MasterSourcedMessage extends SourcedMessage
|
trait MasterSourcedMessage extends SourcedMessage
|
||||||
|
|
||||||
object Acquire
|
class Acquire extends ClientSourcedMessage
|
||||||
{
|
with HasPhysicalAddress
|
||||||
def apply(a_type: Bits, addr: UInt, client_xact_id: UInt): Acquire = {
|
with HasClientTransactionId
|
||||||
|
with HasTileLinkData {
|
||||||
|
val uncached = Bool()
|
||||||
|
val a_type = UInt(width = max(log2Up(Acquire.nUncachedAcquireTypes), params(TLCoherence).acquireTypeWidth))
|
||||||
|
val subblock = Bits(width = tlSubblockUnionBits)
|
||||||
|
val sbAddrOff = tlSubblockAddrBits + tlUncachedOperandSizeBits
|
||||||
|
val opSzOff = tlUncachedOperandSizeBits + sbAddrOff
|
||||||
|
def operand_sz(dummy: Int = 0) = subblock(tlUncachedOperandSizeBits-1, 0)
|
||||||
|
def subblock_addr(dummy: Int = 0) = subblock(sbAddrOff-1, tlUncachedOperandSizeBits)
|
||||||
|
def atomic_op(dummy: Int = 0) = subblock(opSzOff-1, sbAddrOff)
|
||||||
|
def write_mask(dummy: Int = 0) = subblock(tlWriteMaskBits-1, 0)
|
||||||
|
def is(t: UInt) = a_type === t
|
||||||
|
}
|
||||||
|
|
||||||
|
object Acquire {
|
||||||
|
val nUncachedAcquireTypes = 3
|
||||||
|
//val uncachedRead :: uncachedWrite :: uncachedAtomic :: Nil = Enum(UInt(), nUncachedAcquireTypes)
|
||||||
|
def uncachedRead = UInt(0)
|
||||||
|
def uncachedWrite = UInt(1)
|
||||||
|
def uncachedAtomic = UInt(2)
|
||||||
|
def hasData(a_type: UInt) = Vec(uncachedWrite, uncachedAtomic).contains(a_type)
|
||||||
|
def requiresOuterRead(a_type: UInt) = a_type != uncachedWrite
|
||||||
|
def requiresOuterWrite(a_type: UInt) = a_type === uncachedWrite
|
||||||
|
|
||||||
|
def apply(a_type: Bits, addr: UInt, client_xact_id: UInt, data: UInt): Acquire = {
|
||||||
val acq = new Acquire
|
val acq = new Acquire
|
||||||
|
acq.uncached := Bool(false)
|
||||||
acq.a_type := a_type
|
acq.a_type := a_type
|
||||||
acq.addr := addr
|
acq.addr := addr
|
||||||
acq.client_xact_id := client_xact_id
|
acq.client_xact_id := client_xact_id
|
||||||
acq.data := Bits(0)
|
|
||||||
acq.write_mask := Bits(0)
|
|
||||||
acq.subword_addr := Bits(0)
|
|
||||||
acq.atomic_opcode := Bits(0)
|
|
||||||
acq
|
|
||||||
}
|
|
||||||
def apply(a_type: Bits, addr: UInt, client_xact_id: UInt, data: UInt): Acquire = {
|
|
||||||
val acq = apply(a_type, addr, client_xact_id)
|
|
||||||
acq.data := data
|
acq.data := data
|
||||||
|
acq.subblock := UInt(0)
|
||||||
acq
|
acq
|
||||||
}
|
}
|
||||||
def apply(a_type: UInt, addr: UInt, client_xact_id: UInt, write_mask: Bits, data: UInt): Acquire = {
|
def apply(a_type: Bits, addr: UInt, client_xact_id: UInt): Acquire = {
|
||||||
val acq = apply(a_type, addr, client_xact_id, data)
|
apply(a_type, addr, client_xact_id, UInt(0))
|
||||||
acq.write_mask := write_mask
|
|
||||||
acq
|
|
||||||
}
|
|
||||||
def apply(a_type: UInt, addr: UInt, client_xact_id: UInt, subword_addr: UInt, atomic_opcode: UInt, data: UInt): Acquire = {
|
|
||||||
val acq = apply(a_type, addr, client_xact_id, data)
|
|
||||||
acq.subword_addr := subword_addr
|
|
||||||
acq.atomic_opcode := atomic_opcode
|
|
||||||
acq
|
|
||||||
}
|
}
|
||||||
def apply(a: Acquire): Acquire = {
|
def apply(a: Acquire): Acquire = {
|
||||||
val acq = new Acquire
|
val acq = new Acquire
|
||||||
@ -69,94 +94,117 @@ object Acquire
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
class Acquire extends ClientSourcedMessage
|
object UncachedRead {
|
||||||
with HasPhysicalAddress
|
def apply(addr: UInt, client_xact_id: UInt, subblock_addr: UInt, operand_sz: UInt): Acquire = {
|
||||||
with HasClientTransactionId
|
val acq = Acquire(Acquire.uncachedRead, addr, client_xact_id)
|
||||||
with HasTileLinkData {
|
acq.uncached := Bool(true)
|
||||||
val a_type = UInt(width = params(TLCoherence).acquireTypeWidth)
|
acq.subblock := Cat(subblock_addr, operand_sz)
|
||||||
val write_mask = Bits(width = params(TLWriteMaskBits))
|
acq
|
||||||
val subword_addr = Bits(width = params(TLWordAddrBits))
|
}
|
||||||
val atomic_opcode = Bits(width = params(TLAtomicOpBits))
|
def apply(addr: UInt, client_xact_id: UInt): Acquire = {
|
||||||
def is(t: UInt) = a_type === t
|
apply(addr, client_xact_id, UInt(0), MT_CB)
|
||||||
|
}
|
||||||
|
def apply(addr: UInt): Acquire = {
|
||||||
|
apply(addr, UInt(0), UInt(0), MT_CB)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
object Probe
|
object UncachedWrite {
|
||||||
{
|
def apply(addr: UInt, client_xact_id: UInt, write_mask: Bits, data: UInt): Acquire = {
|
||||||
def apply(p_type: UInt, addr: UInt, master_xact_id: UInt) = {
|
val acq = Acquire(Acquire.uncachedWrite, addr, client_xact_id, data)
|
||||||
|
acq.uncached := Bool(true)
|
||||||
|
acq.subblock := write_mask
|
||||||
|
acq
|
||||||
|
}
|
||||||
|
def apply(addr: UInt, client_xact_id: UInt, data: UInt): Acquire = {
|
||||||
|
apply(addr, client_xact_id, SInt(-1), data)
|
||||||
|
}
|
||||||
|
def apply(addr: UInt, data: UInt): Acquire = {
|
||||||
|
apply(addr, UInt(0), data)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
object UncachedAtomic {
|
||||||
|
def apply(addr: UInt, client_xact_id: UInt, atomic_opcode: UInt,
|
||||||
|
subblock_addr: UInt, operand_sz: UInt, data: UInt): Acquire = {
|
||||||
|
val acq = Acquire(Acquire.uncachedAtomic, addr, client_xact_id, data)
|
||||||
|
acq.uncached := Bool(true)
|
||||||
|
acq.subblock := Cat(atomic_opcode, subblock_addr, operand_sz)
|
||||||
|
acq
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
object Probe {
|
||||||
|
def apply(p_type: UInt, addr: UInt) = {
|
||||||
val prb = new Probe
|
val prb = new Probe
|
||||||
prb.p_type := p_type
|
prb.p_type := p_type
|
||||||
prb.addr := addr
|
prb.addr := addr
|
||||||
prb.master_xact_id := master_xact_id
|
|
||||||
prb
|
prb
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
class Probe extends MasterSourcedMessage
|
class Probe extends MasterSourcedMessage
|
||||||
with HasPhysicalAddress
|
with HasPhysicalAddress {
|
||||||
with HasMasterTransactionId {
|
|
||||||
val p_type = UInt(width = params(TLCoherence).probeTypeWidth)
|
val p_type = UInt(width = params(TLCoherence).probeTypeWidth)
|
||||||
def is(t: UInt) = p_type === t
|
def is(t: UInt) = p_type === t
|
||||||
}
|
}
|
||||||
|
|
||||||
object Release
|
object Release {
|
||||||
{
|
def apply(r_type: UInt, addr: UInt, client_xact_id: UInt, data: UInt): Release = {
|
||||||
def apply(r_type: UInt, addr: UInt, data: UInt): Release = {
|
|
||||||
val rel = new Release
|
|
||||||
rel.r_type := r_type
|
|
||||||
rel.addr := addr
|
|
||||||
rel.data := data
|
|
||||||
rel
|
|
||||||
}
|
|
||||||
def apply(r_type: UInt, addr: UInt, client_xact_id: UInt, master_xact_id: UInt): Release = {
|
|
||||||
val rel = new Release
|
val rel = new Release
|
||||||
rel.r_type := r_type
|
rel.r_type := r_type
|
||||||
rel.addr := addr
|
rel.addr := addr
|
||||||
rel.client_xact_id := client_xact_id
|
rel.client_xact_id := client_xact_id
|
||||||
rel.master_xact_id := master_xact_id
|
|
||||||
rel.data := UInt(0)
|
|
||||||
rel
|
|
||||||
}
|
|
||||||
def apply(r_type: UInt, addr: UInt, client_xact_id: UInt, master_xact_id: UInt, data: UInt): Release = {
|
|
||||||
val rel = apply(r_type, addr, client_xact_id, master_xact_id)
|
|
||||||
rel.data := data
|
rel.data := data
|
||||||
rel
|
rel
|
||||||
}
|
}
|
||||||
|
def apply(r_type: UInt, addr: UInt, client_xact_id: UInt): Release = {
|
||||||
|
apply(r_type, addr, client_xact_id, UInt(0))
|
||||||
|
}
|
||||||
|
def apply(r_type: UInt, addr: UInt): Release = {
|
||||||
|
apply(r_type, addr, UInt(0), UInt(0))
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
class Release extends ClientSourcedMessage
|
class Release extends ClientSourcedMessage
|
||||||
with HasPhysicalAddress
|
with HasPhysicalAddress
|
||||||
with HasClientTransactionId
|
with HasClientTransactionId
|
||||||
with HasMasterTransactionId
|
|
||||||
with HasTileLinkData {
|
with HasTileLinkData {
|
||||||
val r_type = UInt(width = params(TLCoherence).releaseTypeWidth)
|
val r_type = UInt(width = params(TLCoherence).releaseTypeWidth)
|
||||||
def is(t: UInt) = r_type === t
|
def is(t: UInt) = r_type === t
|
||||||
}
|
}
|
||||||
|
|
||||||
object Grant
|
|
||||||
{
|
|
||||||
def apply(g_type: UInt, client_xact_id: UInt, master_xact_id: UInt): Grant = {
|
|
||||||
val gnt = new Grant
|
|
||||||
gnt.g_type := g_type
|
|
||||||
gnt.client_xact_id := client_xact_id
|
|
||||||
gnt.master_xact_id := master_xact_id
|
|
||||||
gnt.data := UInt(0)
|
|
||||||
gnt
|
|
||||||
}
|
|
||||||
def apply(g_type: UInt, client_xact_id: UInt, master_xact_id: UInt, data: UInt): Grant = {
|
|
||||||
val gnt = apply(g_type, client_xact_id, master_xact_id)
|
|
||||||
gnt.data := data
|
|
||||||
gnt
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
class Grant extends MasterSourcedMessage
|
class Grant extends MasterSourcedMessage
|
||||||
with HasTileLinkData
|
with HasTileLinkData
|
||||||
with HasClientTransactionId
|
with HasClientTransactionId
|
||||||
with HasMasterTransactionId {
|
with HasMasterTransactionId {
|
||||||
val g_type = UInt(width = params(TLCoherence).grantTypeWidth)
|
val uncached = Bool()
|
||||||
|
val g_type = UInt(width = max(log2Up(Grant.nUncachedGrantTypes), params(TLCoherence).grantTypeWidth))
|
||||||
def is(t: UInt) = g_type === t
|
def is(t: UInt) = g_type === t
|
||||||
}
|
}
|
||||||
|
|
||||||
|
object Grant {
|
||||||
|
val nUncachedGrantTypes = 3
|
||||||
|
//val uncachedRead :: uncachedWrite :: uncachedAtomic :: Nil = Enum(UInt(), nUncachedGrantTypes)
|
||||||
|
def uncachedRead = UInt(0)
|
||||||
|
def uncachedWrite = UInt(1)
|
||||||
|
def uncachedAtomic = UInt(2)
|
||||||
|
def hasData(g_type: UInt) = Vec(uncachedRead, uncachedAtomic).contains(g_type)
|
||||||
|
|
||||||
|
def apply(uncached: Bool, g_type: UInt, client_xact_id: UInt, master_xact_id: UInt, data: UInt): Grant = {
|
||||||
|
val gnt = new Grant
|
||||||
|
gnt.uncached := uncached
|
||||||
|
gnt.g_type := g_type
|
||||||
|
gnt.client_xact_id := client_xact_id
|
||||||
|
gnt.master_xact_id := master_xact_id
|
||||||
|
gnt.data := data
|
||||||
|
gnt
|
||||||
|
}
|
||||||
|
def apply(uncached: Bool, g_type: UInt, client_xact_id: UInt, master_xact_id: UInt): Grant = {
|
||||||
|
apply(uncached, g_type, client_xact_id, master_xact_id, UInt(0))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
class Finish extends ClientSourcedMessage with HasMasterTransactionId
|
class Finish extends ClientSourcedMessage with HasMasterTransactionId
|
||||||
|
|
||||||
|
|
||||||
|
@ -5,14 +5,17 @@ import Chisel._
|
|||||||
|
|
||||||
case object NReleaseTransactors extends Field[Int]
|
case object NReleaseTransactors extends Field[Int]
|
||||||
case object NAcquireTransactors extends Field[Int]
|
case object NAcquireTransactors extends Field[Int]
|
||||||
|
case object L2StoreDataQueueDepth extends Field[Int]
|
||||||
case object NClients extends Field[Int]
|
case object NClients extends Field[Int]
|
||||||
|
|
||||||
abstract trait CoherenceAgentParameters extends UsesParameters {
|
abstract trait CoherenceAgentParameters extends UsesParameters {
|
||||||
val co = params(TLCoherence)
|
val co = params(TLCoherence)
|
||||||
val nReleaseTransactors = params(NReleaseTransactors)
|
val nReleaseTransactors = 1
|
||||||
val nAcquireTransactors = params(NAcquireTransactors)
|
val nAcquireTransactors = params(NAcquireTransactors)
|
||||||
val nTransactors = nReleaseTransactors + nAcquireTransactors
|
val nTransactors = nReleaseTransactors + nAcquireTransactors
|
||||||
val nClients = params(NClients)
|
val nClients = params(NClients)
|
||||||
|
val sdqDepth = params(L2StoreDataQueueDepth)
|
||||||
|
val sdqIdBits = math.max(log2Up(nReleaseTransactors) + 1, log2Up(params(L2StoreDataQueueDepth))) + 1
|
||||||
}
|
}
|
||||||
|
|
||||||
abstract class CoherenceAgent(innerId: String, outerId: String) extends Module
|
abstract class CoherenceAgent(innerId: String, outerId: String) extends Module
|
||||||
@ -27,11 +30,19 @@ abstract class CoherenceAgent(innerId: String, outerId: String) extends Module
|
|||||||
class L2CoherenceAgent(bankId: Int, innerId: String, outerId: String) extends
|
class L2CoherenceAgent(bankId: Int, innerId: String, outerId: String) extends
|
||||||
CoherenceAgent(innerId, outerId) {
|
CoherenceAgent(innerId, outerId) {
|
||||||
|
|
||||||
|
// Queue to store impending UncachedWrite data
|
||||||
|
val sdq_val = Reg(init=Bits(0, sdqDepth))
|
||||||
|
val sdq_alloc_id = PriorityEncoder(~sdq_val(sdqDepth-1,0))
|
||||||
|
val sdq_rdy = !sdq_val.andR
|
||||||
|
val sdq_enq = io.inner.acquire.valid && io.inner.acquire.ready && co.messageHasData(io.inner.acquire.bits.payload)
|
||||||
|
val sdq = Vec.fill(sdqDepth){Reg(io.inner.acquire.bits.payload.data)}
|
||||||
|
when (sdq_enq) { sdq(sdq_alloc_id) := io.inner.acquire.bits.payload.data }
|
||||||
|
|
||||||
// Create SHRs for outstanding transactions
|
// Create SHRs for outstanding transactions
|
||||||
val trackerList = (0 until nReleaseTransactors).map(id =>
|
val trackerList = (0 until nReleaseTransactors).map(id =>
|
||||||
Module(new VoluntaryReleaseTracker(id, bankId, innerId, outerId))) ++
|
Module(new VoluntaryReleaseTracker(id, bankId, innerId, outerId), {case TLDataBits => sdqIdBits})) ++
|
||||||
(nReleaseTransactors until nTransactors).map(id =>
|
(nReleaseTransactors until nTransactors).map(id =>
|
||||||
Module(new AcquireTracker(id, bankId, innerId, outerId)))
|
Module(new AcquireTracker(id, bankId, innerId, outerId), {case TLDataBits => sdqIdBits}))
|
||||||
|
|
||||||
// Propagate incoherence flags
|
// Propagate incoherence flags
|
||||||
trackerList.map(_.io.tile_incoherent := io.incoherent.toBits)
|
trackerList.map(_.io.tile_incoherent := io.incoherent.toBits)
|
||||||
@ -46,10 +57,11 @@ class L2CoherenceAgent(bankId: Int, innerId: String, outerId: String) extends
|
|||||||
val t = trackerList(i).io.inner
|
val t = trackerList(i).io.inner
|
||||||
alloc_arb.io.in(i).valid := t.acquire.ready
|
alloc_arb.io.in(i).valid := t.acquire.ready
|
||||||
t.acquire.bits := acquire.bits
|
t.acquire.bits := acquire.bits
|
||||||
|
t.acquire.bits.payload.data := Cat(sdq_alloc_id, UInt(1))
|
||||||
t.acquire.valid := alloc_arb.io.in(i).ready
|
t.acquire.valid := alloc_arb.io.in(i).ready
|
||||||
}
|
}
|
||||||
acquire.ready := trackerList.map(_.io.inner.acquire.ready).reduce(_||_) && !block_acquires
|
acquire.ready := trackerList.map(_.io.inner.acquire.ready).reduce(_||_) && sdq_rdy && !block_acquires
|
||||||
alloc_arb.io.out.ready := acquire.valid && !block_acquires
|
alloc_arb.io.out.ready := acquire.valid && sdq_rdy && !block_acquires
|
||||||
|
|
||||||
// Handle probe request generation
|
// Handle probe request generation
|
||||||
val probe_arb = Module(new Arbiter(new LogicalNetworkIO(new Probe), trackerList.size))
|
val probe_arb = Module(new Arbiter(new LogicalNetworkIO(new Probe), trackerList.size))
|
||||||
@ -62,17 +74,24 @@ class L2CoherenceAgent(bankId: Int, innerId: String, outerId: String) extends
|
|||||||
val any_release_conflict = trackerList.tail.map(_.io.has_release_conflict).reduce(_||_)
|
val any_release_conflict = trackerList.tail.map(_.io.has_release_conflict).reduce(_||_)
|
||||||
val block_releases = Bool(false)
|
val block_releases = Bool(false)
|
||||||
val conflict_idx = Vec(trackerList.map(_.io.has_release_conflict)).lastIndexWhere{b: Bool => b}
|
val conflict_idx = Vec(trackerList.map(_.io.has_release_conflict)).lastIndexWhere{b: Bool => b}
|
||||||
//val release_idx = Mux(voluntary, Mux(any_release_conflict, conflict_idx, UInt(0)), release.bits.payload.master_xact_id) // TODO: Add merging logic to allow allocated AcquireTracker to handle conflicts, send all necessary grants, use first sufficient response
|
val release_idx = Mux(voluntary, UInt(0), conflict_idx)
|
||||||
val release_idx = Mux(voluntary, UInt(0), release.bits.payload.master_xact_id)
|
// TODO: Add merging logic to allow allocated AcquireTracker to handle conflicts, send all necessary grants, use first sufficient response
|
||||||
for( i <- 0 until trackerList.size ) {
|
for( i <- 0 until trackerList.size ) {
|
||||||
val t = trackerList(i).io.inner
|
val t = trackerList(i).io.inner
|
||||||
t.release.bits := release.bits
|
t.release.bits := release.bits
|
||||||
|
t.release.bits.payload.data := (if (i < nReleaseTransactors) Cat(UInt(i), UInt(2)) else UInt(0))
|
||||||
t.release.valid := release.valid && (release_idx === UInt(i)) && !block_releases
|
t.release.valid := release.valid && (release_idx === UInt(i)) && !block_releases
|
||||||
}
|
}
|
||||||
release.ready := Vec(trackerList.map(_.io.inner.release.ready)).read(release_idx) && !block_releases
|
release.ready := Vec(trackerList.map(_.io.inner.release.ready)).read(release_idx) && !block_releases
|
||||||
|
|
||||||
|
val vwbdq = Vec.fill(nReleaseTransactors){ Reg(release.bits.payload.data) }
|
||||||
|
when(voluntary && release.fire()) {
|
||||||
|
vwbdq(release_idx) := release.bits.payload.data
|
||||||
|
}
|
||||||
|
|
||||||
// Reply to initial requestor
|
// Reply to initial requestor
|
||||||
val grant_arb = Module(new Arbiter(new LogicalNetworkIO(new Grant), trackerList.size))
|
val grant_arb = Module(new Arbiter(new LogicalNetworkIO(new Grant), trackerList.size))
|
||||||
|
io.inner.grant.bits.payload.data := io.outer.grant.bits.payload.data
|
||||||
io.inner.grant <> grant_arb.io.out
|
io.inner.grant <> grant_arb.io.out
|
||||||
grant_arb.io.in zip trackerList map { case (arb, t) => arb <> t.io.inner.grant }
|
grant_arb.io.in zip trackerList map { case (arb, t) => arb <> t.io.inner.grant }
|
||||||
|
|
||||||
@ -84,9 +103,22 @@ class L2CoherenceAgent(bankId: Int, innerId: String, outerId: String) extends
|
|||||||
|
|
||||||
// Create an arbiter for the one memory port
|
// Create an arbiter for the one memory port
|
||||||
val outer_arb = Module(new UncachedTileLinkIOArbiterThatPassesId(trackerList.size),
|
val outer_arb = Module(new UncachedTileLinkIOArbiterThatPassesId(trackerList.size),
|
||||||
{case TLId => outerId})
|
{ case TLId => outerId; case TLDataBits => sdqIdBits })
|
||||||
outer_arb.io.in zip trackerList map { case(arb, t) => arb <> t.io.outer }
|
outer_arb.io.in zip trackerList map { case(arb, t) => arb <> t.io.outer }
|
||||||
|
val is_in_sdq = outer_arb.io.out.acquire.bits.payload.data(0)
|
||||||
|
val is_in_vwbdq = outer_arb.io.out.acquire.bits.payload.data(1)
|
||||||
|
val free_sdq_id = outer_arb.io.out.acquire.bits.payload.data >> UInt(1)
|
||||||
|
val free_vwbdq_id = outer_arb.io.out.acquire.bits.payload.data >> UInt(2)
|
||||||
|
val free_sdq = io.outer.acquire.fire() && co.messageHasData(io.outer.acquire.bits.payload) && is_in_sdq
|
||||||
|
io.outer.acquire.bits.payload.data := Mux(is_in_sdq, sdq(free_sdq_id),
|
||||||
|
Mux(is_in_vwbdq, vwbdq(free_vwbdq_id), release.bits.payload.data))
|
||||||
io.outer <> outer_arb.io.out
|
io.outer <> outer_arb.io.out
|
||||||
|
|
||||||
|
// Update SDQ valid bits
|
||||||
|
when (io.outer.acquire.valid || sdq_enq) {
|
||||||
|
sdq_val := sdq_val & ~(UIntToOH(free_sdq_id) & Fill(sdqDepth, free_sdq)) |
|
||||||
|
PriorityEncoderOH(~sdq_val(sdqDepth-1,0)) & Fill(sdqDepth, sdq_enq)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -112,17 +144,16 @@ class VoluntaryReleaseTracker(trackerId: Int, bankId: Int, innerId: String, oute
|
|||||||
val state = Reg(init=s_idle)
|
val state = Reg(init=s_idle)
|
||||||
val xact = Reg{ new Release }
|
val xact = Reg{ new Release }
|
||||||
val init_client_id = Reg(init=UInt(0, width = log2Up(nClients)))
|
val init_client_id = Reg(init=UInt(0, width = log2Up(nClients)))
|
||||||
val incoming_rel = io.inner.release.bits
|
|
||||||
|
|
||||||
io.has_acquire_conflict := Bool(false)
|
io.has_acquire_conflict := Bool(false)
|
||||||
io.has_release_conflict := co.isCoherenceConflict(xact.addr, incoming_rel.payload.addr) &&
|
io.has_release_conflict := co.isCoherenceConflict(xact.addr, c_rel.payload.addr) &&
|
||||||
(state != s_idle)
|
(state != s_idle)
|
||||||
|
|
||||||
io.outer.grant.ready := Bool(false)
|
io.outer.grant.ready := Bool(false)
|
||||||
io.outer.acquire.valid := Bool(false)
|
io.outer.acquire.valid := Bool(false)
|
||||||
io.outer.acquire.bits.header.src := UInt(bankId)
|
io.outer.acquire.bits.header.src := UInt(bankId)
|
||||||
//io.outer.acquire.bits.header.dst TODO
|
//io.outer.acquire.bits.header.dst TODO
|
||||||
io.outer.acquire.bits.payload := Bundle(Acquire(co.getUncachedWriteAcquireType,
|
io.outer.acquire.bits.payload := Bundle(UncachedWrite(
|
||||||
xact.addr,
|
xact.addr,
|
||||||
UInt(trackerId),
|
UInt(trackerId),
|
||||||
xact.data),
|
xact.data),
|
||||||
@ -133,17 +164,18 @@ class VoluntaryReleaseTracker(trackerId: Int, bankId: Int, innerId: String, oute
|
|||||||
io.inner.grant.valid := Bool(false)
|
io.inner.grant.valid := Bool(false)
|
||||||
io.inner.grant.bits.header.src := UInt(bankId)
|
io.inner.grant.bits.header.src := UInt(bankId)
|
||||||
io.inner.grant.bits.header.dst := init_client_id
|
io.inner.grant.bits.header.dst := init_client_id
|
||||||
io.inner.grant.bits.payload := Grant(co.getGrantType(xact, co.masterMetadataOnFlush),
|
io.inner.grant.bits.payload := Grant(Bool(false),
|
||||||
|
co.getGrantTypeOnVoluntaryWriteback(co.masterMetadataOnFlush),
|
||||||
xact.client_xact_id,
|
xact.client_xact_id,
|
||||||
UInt(trackerId))
|
UInt(trackerId))
|
||||||
|
|
||||||
switch (state) {
|
switch (state) {
|
||||||
is(s_idle) {
|
is(s_idle) {
|
||||||
io.inner.release.ready := Bool(true)
|
|
||||||
when( io.inner.release.valid ) {
|
when( io.inner.release.valid ) {
|
||||||
xact := incoming_rel.payload
|
io.inner.release.ready := Bool(true)
|
||||||
init_client_id := incoming_rel.header.src
|
xact := c_rel.payload
|
||||||
state := Mux(co.messageHasData(incoming_rel.payload), s_mem, s_ack)
|
init_client_id := c_rel.header.src
|
||||||
|
state := Mux(co.messageHasData(c_rel.payload), s_mem, s_ack)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
is(s_mem) {
|
is(s_mem) {
|
||||||
@ -169,15 +201,13 @@ class AcquireTracker(trackerId: Int, bankId: Int, innerId: String, outerId: Stri
|
|||||||
val curr_p_id = PriorityEncoder(probe_flags)
|
val curr_p_id = PriorityEncoder(probe_flags)
|
||||||
|
|
||||||
val pending_outer_write = co.messageHasData(xact)
|
val pending_outer_write = co.messageHasData(xact)
|
||||||
val pending_outer_read = co.requiresOuterRead(xact.a_type)
|
val pending_outer_read = co.requiresOuterRead(xact, co.masterMetadataOnFlush)
|
||||||
val outer_write_acq = Bundle(Acquire(co.getUncachedWriteAcquireType,
|
val outer_write_acq = Bundle(UncachedWrite(xact.addr, UInt(trackerId), xact.data),
|
||||||
xact.addr, UInt(trackerId), xact.data),
|
{ case TLId => outerId })
|
||||||
{ case TLId => outerId })
|
val outer_write_rel = Bundle(UncachedWrite(xact.addr, UInt(trackerId), UInt(0)), // Special SQDId
|
||||||
val outer_write_rel = Bundle(Acquire(co.getUncachedWriteAcquireType,
|
{ case TLId => outerId })
|
||||||
xact.addr, UInt(trackerId), c_rel.payload.data),
|
val outer_read = Bundle(UncachedRead(xact.addr, UInt(trackerId)),
|
||||||
{ case TLId => outerId })
|
{ case TLId => outerId })
|
||||||
val outer_read = Bundle(Acquire(co.getUncachedReadAcquireType, xact.addr, UInt(trackerId)),
|
|
||||||
{ case TLId => outerId })
|
|
||||||
|
|
||||||
val probe_initial_flags = Bits(width = nClients)
|
val probe_initial_flags = Bits(width = nClients)
|
||||||
probe_initial_flags := Bits(0)
|
probe_initial_flags := Bits(0)
|
||||||
@ -200,18 +230,16 @@ class AcquireTracker(trackerId: Int, bankId: Int, innerId: String, outerId: Stri
|
|||||||
io.inner.probe.valid := Bool(false)
|
io.inner.probe.valid := Bool(false)
|
||||||
io.inner.probe.bits.header.src := UInt(bankId)
|
io.inner.probe.bits.header.src := UInt(bankId)
|
||||||
io.inner.probe.bits.header.dst := curr_p_id
|
io.inner.probe.bits.header.dst := curr_p_id
|
||||||
io.inner.probe.bits.payload := Probe(co.getProbeType(xact, co.masterMetadataOnFlush),
|
io.inner.probe.bits.payload := Probe(co.getProbeType(xact, co.masterMetadataOnFlush), xact.addr)
|
||||||
xact.addr,
|
|
||||||
UInt(trackerId))
|
|
||||||
|
|
||||||
val grant_type = co.getGrantType(xact, co.masterMetadataOnFlush)
|
|
||||||
io.inner.grant.valid := Bool(false)
|
io.inner.grant.valid := Bool(false)
|
||||||
io.inner.grant.bits.header.src := UInt(bankId)
|
io.inner.grant.bits.header.src := UInt(bankId)
|
||||||
io.inner.grant.bits.header.dst := init_client_id
|
io.inner.grant.bits.header.dst := init_client_id
|
||||||
io.inner.grant.bits.payload := Grant(grant_type,
|
io.inner.grant.bits.payload := Grant(xact.uncached,
|
||||||
|
co.getGrantType(xact, co.masterMetadataOnFlush),
|
||||||
xact.client_xact_id,
|
xact.client_xact_id,
|
||||||
UInt(trackerId),
|
UInt(trackerId),
|
||||||
m_gnt.payload.data)
|
UInt(0)) // Data bypassed in parent
|
||||||
|
|
||||||
io.inner.acquire.ready := Bool(false)
|
io.inner.acquire.ready := Bool(false)
|
||||||
io.inner.release.ready := Bool(false)
|
io.inner.release.ready := Bool(false)
|
||||||
@ -220,7 +248,7 @@ class AcquireTracker(trackerId: Int, bankId: Int, innerId: String, outerId: Stri
|
|||||||
is(s_idle) {
|
is(s_idle) {
|
||||||
io.inner.acquire.ready := Bool(true)
|
io.inner.acquire.ready := Bool(true)
|
||||||
val needs_outer_write = co.messageHasData(c_acq.payload)
|
val needs_outer_write = co.messageHasData(c_acq.payload)
|
||||||
val needs_outer_read = co.requiresOuterRead(c_acq.payload.a_type)
|
val needs_outer_read = co.requiresOuterRead(c_acq.payload, co.masterMetadataOnFlush)
|
||||||
when( io.inner.acquire.valid ) {
|
when( io.inner.acquire.valid ) {
|
||||||
xact := c_acq.payload
|
xact := c_acq.payload
|
||||||
init_client_id := c_acq.header.src
|
init_client_id := c_acq.header.src
|
||||||
@ -268,7 +296,7 @@ class AcquireTracker(trackerId: Int, bankId: Int, innerId: String, outerId: Stri
|
|||||||
io.outer.acquire.valid := Bool(true)
|
io.outer.acquire.valid := Bool(true)
|
||||||
io.outer.acquire.bits.payload := outer_read
|
io.outer.acquire.bits.payload := outer_read
|
||||||
when(io.outer.acquire.ready) {
|
when(io.outer.acquire.ready) {
|
||||||
state := Mux(co.requiresAckForGrant(grant_type), s_busy, s_idle)
|
state := Mux(co.requiresAckForGrant(io.inner.grant.bits.payload), s_busy, s_idle)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
is(s_mem_write) {
|
is(s_mem_write) {
|
||||||
@ -281,7 +309,7 @@ class AcquireTracker(trackerId: Int, bankId: Int, innerId: String, outerId: Stri
|
|||||||
is(s_make_grant) {
|
is(s_make_grant) {
|
||||||
io.inner.grant.valid := Bool(true)
|
io.inner.grant.valid := Bool(true)
|
||||||
when(io.inner.grant.ready) {
|
when(io.inner.grant.ready) {
|
||||||
state := Mux(co.requiresAckForGrant(grant_type), s_busy, s_idle)
|
state := Mux(co.requiresAckForGrant(io.inner.grant.bits.payload), s_busy, s_idle)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
is(s_busy) { // Nothing left to do but wait for transaction to complete
|
is(s_busy) { // Nothing left to do but wait for transaction to complete
|
||||||
|
Loading…
Reference in New Issue
Block a user