gem5  v20.0.0.0
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages
token_port.cc
Go to the documentation of this file.
1 /*
2  * Copyright (c) 2016-2020 Advanced Micro Devices, Inc.
3  * All rights reserved.
4  *
5  * For use for simulation and test purposes only
6  *
7  * Redistribution and use in source and binary forms, with or without
8  * modification, are permitted provided that the following conditions are met:
9  *
10  * 1. Redistributions of source code must retain the above copyright notice,
11  * this list of conditions and the following disclaimer.
12  *
13  * 2. Redistributions in binary form must reproduce the above copyright notice,
14  * this list of conditions and the following disclaimer in the documentation
15  * and/or other materials provided with the distribution.
16  *
17  * 3. Neither the name of the copyright holder nor the names of its
18  * contributors may be used to endorse or promote products derived from this
19  * software without specific prior written permission.
20  *
21  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
22  * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
23  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
24  * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
25  * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
26  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
27  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
28  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
29  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
30  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
31  * POSSIBILITY OF SUCH DAMAGE.
32  *
33  * Authors: Matthew Poremba
34  */
35 
36 
37 #include "mem/token_port.hh"
38 
39 #include "base/trace.hh"
40 #include "debug/TokenPort.hh"
41 
42 void
44 {
45  MasterPort::bind(peer);
46 }
47 
48 void
50 {
51  panic_if(!tokenManager, "TokenManager not set for %s.\n", name());
52 
53  tokenManager->recvTokens(num_tokens);
54 }
55 
56 bool
58 {
59  panic_if(!tokenManager, "TokenManager not set for %s.\n", name());
60 
61  return tokenManager->haveTokens(num_tokens);
62 }
63 
64 void
66 {
67  panic_if(!tokenManager, "TokenManager not set for %s.\n", name());
68 
69  tokenManager->acquireTokens(num_tokens);
70 }
71 
72 void
74 {
75  tokenManager = _tokenManager;
76 }
77 
78 void
80 {
81  fatal_if(!tokenMasterPort, "Tried sendTokens to non-token master!\n");
82 
83  // Send tokens to a master
84  tokenMasterPort->recvTokens(num_tokens);
85 }
86 
87 void
89 {
90  // TokenSlavePort is allowed to bind to either TokenMasterPort or a
91  // MasterPort as fallback. If the type is a MasterPort, tokenMasterPort
92  // is set to nullptr to indicate tokens should not be exchanged.
93  auto *token_master_port = dynamic_cast<TokenMasterPort*>(&peer);
94  auto *master_port = dynamic_cast<MasterPort*>(&peer);
95  if (!token_master_port && !master_port) {
96  fatal("Attempt to bind port %s to unsupported slave port %s.",
97  name(), peer.name());
98  } else if (token_master_port) {
99  // slave port keeps track of the master port
100  tokenMasterPort = token_master_port;
101 
102  // master port also keeps track of slave port
103  tokenMasterPort->bind(*this);
104  } else if (master_port) {
105  tokenMasterPort = nullptr;
106  }
107 }
108 
109 void
111 {
113  tokenMasterPort = nullptr;
114 }
115 
116 void
118 {
119  // fallback to QueuedSlavePort-like impl for now
120  panic_if(respQueue.empty(),
121  "Attempted to retry a response when no retry was queued!\n");
122 
123  PacketPtr pkt = respQueue.front();
124  bool success = SlavePort::sendTimingResp(pkt);
125 
126  if (success) {
127  respQueue.pop_front();
128  }
129 }
130 
131 bool
133 {
134  bool success = SlavePort::sendTimingResp(pkt);
135 
136  if (!success) {
137  respQueue.push_back(pkt);
138  }
139 
140  return success;
141 }
142 
144 {
145  availableTokens = init_tokens;
146  maxTokens = init_tokens;
147 }
148 
149 int
151 {
152  return maxTokens;
153 }
154 
155 void
157 {
158  availableTokens += num_tokens;
159 
160  DPRINTF(TokenPort, "Received %d tokens, have %d\n",
161  num_tokens, availableTokens);
162 
163  panic_if(availableTokens > maxTokens,
164  "More tokens available than the maximum after recvTokens!\n");
165 }
166 
167 bool
169 {
170  return (availableTokens >= num_tokens);
171 }
172 
173 void
175 {
176  panic_if(!haveTokens(num_tokens),
177  "Attempted to acquire more tokens than are available!\n");
178 
179  availableTokens -= num_tokens;
180 
181  DPRINTF(TokenPort, "Acquired %d tokens, have %d\n",
182  num_tokens, availableTokens);
183 }
A MasterPort is a specialisation of a BaseMasterPort, which implements the default protocol for the t...
Definition: port.hh:71
#define DPRINTF(x,...)
Definition: trace.hh:225
Ports are used to interface objects to each other.
Definition: port.hh:56
int getMaxTokenCount() const
Return the maximum possible tokens.
Definition: token_port.cc:150
#define fatal(...)
This implements a cprintf based fatal() function.
Definition: logging.hh:171
void slaveUnbind()
Called by the master port to unbind.
Definition: port.cc:121
TokenManager * tokenManager
Definition: token_port.hh:47
TokenManager(int init_tokens)
Definition: token_port.cc:143
void bind(Port &peer) override
Bind this master port to slave port.
Definition: token_port.cc:43
void recvRespRetry() override
Called by the peer if sendTimingResp was called on this protocol (causing recvTimingResp to be called...
Definition: token_port.cc:117
void recvTokens(int num_tokens)
Increment the number of available tokens by num_tokens.
Definition: token_port.cc:156
void setTokenManager(TokenManager *_tokenManager)
Specify a token manger, which will handle tracking of tokens for a TokenMasterPort/SlaveMasterPort pa...
Definition: token_port.cc:73
bool sendTimingResp(PacketPtr pkt)
Attempt to send a timing response to the master port by calling its corresponding receive function...
Definition: port.hh:353
void bind(Port &peer) override
Bind this slave port to a master port.
Definition: token_port.cc:88
#define fatal_if(cond,...)
Conditional fatal macro that checks the supplied condition and only causes a fatal error if the condi...
Definition: logging.hh:199
void bind(Port &peer) override
Bind this master port to a slave port.
Definition: port.cc:63
void recvTokens(int num_tokens)
Receive tokens returned by the slave port.
Definition: token_port.cc:49
void acquireTokens(int num_tokens)
Decrement the number of available tokens by num_tokens.
Definition: token_port.cc:174
A Packet is used to encapsulate a transfer between two objects in the memory system (e...
Definition: packet.hh:249
void sendTokens(int num_tokens)
Return num_tokens tokens back to the master port.
Definition: token_port.cc:79
bool haveTokens(int num_tokens)
Query is num_tokens tokens are available.
Definition: token_port.cc:168
const std::string name() const
Return port name (for DPRINTF).
Definition: port.hh:102
bool sendTimingResp(PacketPtr pkt)
Definition: token_port.cc:132
void acquireTokens(int num_tokens)
Acquire tokens by decrementing the number of available tokens across the port.
Definition: token_port.cc:65
#define panic_if(cond,...)
Conditional panic macro that checks the supplied condition and only panics if the condition is true a...
Definition: logging.hh:181
void unbind() override
Unbind this slave port and associated master port.
Definition: token_port.cc:110
bool haveTokens(int num_tokens)
Query if there are at least num_tokens tokens available to acquire.
Definition: token_port.cc:57

Generated on Thu May 28 2020 16:21:35 for gem5 by doxygen 1.8.13