gem5  v21.0.1.0
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages
Throttle.cc
Go to the documentation of this file.
1 /*
2  * Copyright (c) 1999-2008 Mark D. Hill and David A. Wood
3  * All rights reserved.
4  *
5  * Redistribution and use in source and binary forms, with or without
6  * modification, are permitted provided that the following conditions are
7  * met: redistributions of source code must retain the above copyright
8  * notice, this list of conditions and the following disclaimer;
9  * redistributions in binary form must reproduce the above copyright
10  * notice, this list of conditions and the following disclaimer in the
11  * documentation and/or other materials provided with the distribution;
12  * neither the name of the copyright holders nor the names of its
13  * contributors may be used to endorse or promote products derived from
14  * this software without specific prior written permission.
15  *
16  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
17  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
18  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
19  * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
20  * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
21  * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
22  * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
23  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
24  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
25  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
26  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
27  */
28 
30 
31 #include <cassert>
32 
33 #include "base/cast.hh"
34 #include "base/cprintf.hh"
35 #include "debug/RubyNetwork.hh"
41 
42 const int MESSAGE_SIZE_MULTIPLIER = 1000;
43 //const int BROADCAST_SCALING = 4; // Have a 16p system act like a 64p systems
44 const int BROADCAST_SCALING = 1;
45 const int PRIORITY_SWITCH_LIMIT = 128;
46 
47 static int network_message_to_size(Message* net_msg_ptr);
48 
49 Throttle::Throttle(int sID, RubySystem *rs, NodeID node, Cycles link_latency,
50  int link_bandwidth_multiplier, int endpoint_bandwidth,
51  Switch *em)
52  : Consumer(em),
53  m_switch_id(sID), m_switch(em), m_node(node),
54  m_ruby_system(rs),
55  throttleStats(em, node)
56 {
57  m_vnets = 0;
58 
59  assert(link_bandwidth_multiplier > 0);
60  m_link_bandwidth_multiplier = link_bandwidth_multiplier;
61 
62  m_link_latency = link_latency;
63  m_endpoint_bandwidth = endpoint_bandwidth;
64 
67 }
68 
69 void
71  const std::vector<MessageBuffer*>& out_vec)
72 {
73  assert(in_vec.size() == out_vec.size());
74 
75  for (int vnet = 0; vnet < in_vec.size(); ++vnet) {
76  MessageBuffer *in_ptr = in_vec[vnet];
77  MessageBuffer *out_ptr = out_vec[vnet];
78 
79  m_vnets++;
80  m_units_remaining.push_back(0);
81  m_in.push_back(in_ptr);
82  m_out.push_back(out_ptr);
83 
84  // Set consumer and description
85  in_ptr->setConsumer(this);
86  std::string desc = "[Queue to Throttle " +
88  }
89 }
90 
91 void
92 Throttle::operateVnet(int vnet, int &bw_remaining, bool &schedule_wakeup,
93  MessageBuffer *in, MessageBuffer *out)
94 {
95  if (out == nullptr || in == nullptr) {
96  return;
97  }
98 
99  assert(m_units_remaining[vnet] >= 0);
100  Tick current_time = m_switch->clockEdge();
101 
102  while (bw_remaining > 0 && (in->isReady(current_time) ||
103  m_units_remaining[vnet] > 0) &&
104  out->areNSlotsAvailable(1, current_time)) {
105  // See if we are done transferring the previous message on
106  // this virtual network
107  if (m_units_remaining[vnet] == 0 && in->isReady(current_time)) {
108  // Find the size of the message we are moving
109  MsgPtr msg_ptr = in->peekMsgPtr();
110  Message *net_msg_ptr = msg_ptr.get();
111  m_units_remaining[vnet] +=
112  network_message_to_size(net_msg_ptr);
113 
114  DPRINTF(RubyNetwork, "throttle: %d my bw %d bw spent "
115  "enqueueing net msg %d time: %lld.\n",
118 
119  // Move the message
120  in->dequeue(current_time);
121  out->enqueue(msg_ptr, current_time,
123 
124  // Count the message
125  (*(throttleStats.
126  m_msg_counts[net_msg_ptr->getMessageSize()]))[vnet]++;
127  DPRINTF(RubyNetwork, "%s\n", *out);
128  }
129 
130  // Calculate the amount of bandwidth we spent on this message
131  int diff = m_units_remaining[vnet] - bw_remaining;
132  m_units_remaining[vnet] = std::max(0, diff);
133  bw_remaining = std::max(0, -diff);
134  }
135 
136  if (bw_remaining > 0 && (in->isReady(current_time) ||
137  m_units_remaining[vnet] > 0) &&
138  !out->areNSlotsAvailable(1, current_time)) {
139  DPRINTF(RubyNetwork, "vnet: %d", vnet);
140 
141  // schedule me to wakeup again because I'm waiting for my
142  // output queue to become available
143  schedule_wakeup = true;
144  }
145 }
146 
147 void
149 {
150  // Limits the number of message sent to a limited number of bytes/cycle.
151  assert(getLinkBandwidth() > 0);
152  int bw_remaining = getLinkBandwidth();
153 
155  bool schedule_wakeup = false;
156 
157  // variable for deciding the direction in which to iterate
158  bool iteration_direction = false;
159 
160 
161  // invert priorities to avoid starvation seen in the component network
164  iteration_direction = true;
165  }
166 
167  if (iteration_direction) {
168  for (int vnet = 0; vnet < m_vnets; ++vnet) {
169  operateVnet(vnet, bw_remaining, schedule_wakeup,
170  m_in[vnet], m_out[vnet]);
171  }
172  } else {
173  for (int vnet = m_vnets-1; vnet >= 0; --vnet) {
174  operateVnet(vnet, bw_remaining, schedule_wakeup,
175  m_in[vnet], m_out[vnet]);
176  }
177  }
178 
179  // We should only wake up when we use the bandwidth
180  // This is only mostly true
181  // assert(bw_remaining != getLinkBandwidth());
182 
183  // Record that we used some or all of the link bandwidth this cycle
184  double ratio = 1.0 - (double(bw_remaining) / double(getLinkBandwidth()));
185 
186  // If ratio = 0, we used no bandwidth, if ratio = 1, we used all
187  m_link_utilization_proxy += ratio;
188 
189  if (bw_remaining > 0 && !schedule_wakeup) {
190  // We have extra bandwidth and our output buffer was
191  // available, so we must not have anything else to do until
192  // another message arrives.
193  DPRINTF(RubyNetwork, "%s not scheduled again\n", *this);
194  } else {
195  DPRINTF(RubyNetwork, "%s scheduled again\n", *this);
196 
197  // We are out of bandwidth for this cycle, so wakeup next
198  // cycle and continue
199  scheduleEvent(Cycles(1));
200  }
201 }
202 
203 void
205 {
206  for (MessageSizeType type = MessageSizeType_FIRST;
207  type < MessageSizeType_NUM; ++type) {
208  throttleStats.m_msg_counts[(unsigned int)type] =
210  csprintf("msg_count.%s", MessageSizeType_to_string(type)).c_str());
211  throttleStats.m_msg_counts[(unsigned int)type]
213  .flags(Stats::nozero)
214  ;
215 
216  throttleStats.m_msg_bytes[(unsigned int) type] =
218  csprintf("msg_bytes.%s", MessageSizeType_to_string(type)).c_str());
219  throttleStats.m_msg_bytes[(unsigned int) type]
220  ->flags(Stats::nozero)
221  ;
222 
223  *(throttleStats.m_msg_bytes[(unsigned int) type]) =
226  }
227 }
228 
229 void
231 {
233 }
234 
235 void
237 {
238  double time_delta = double(m_ruby_system->curCycle() -
240 
242  100.0 * m_link_utilization_proxy / time_delta;
243 }
244 
245 void
246 Throttle::print(std::ostream& out) const
247 {
248  ccprintf(out, "[%i bw: %i]", m_node, getLinkBandwidth());
249 }
250 
251 int
253 {
254  assert(net_msg_ptr != NULL);
255 
256  int size = Network::MessageSizeType_to_int(net_msg_ptr->getMessageSize());
257  size *= MESSAGE_SIZE_MULTIPLIER;
258 
259  // Artificially increase the size of broadcast messages
260  if (BROADCAST_SCALING > 1 && net_msg_ptr->getDestination().isBroadcast())
261  size *= BROADCAST_SCALING;
262 
263  return size;
264 }
265 
268  : Stats::Group(parent, csprintf("throttle%02i", nodeID).c_str()),
269  m_link_utilization(this, "link_utilization")
270 {
271 
272 }
Throttle::m_link_latency
Cycles m_link_latency
Definition: Throttle.hh:103
MESSAGE_SIZE_MULTIPLIER
const int MESSAGE_SIZE_MULTIPLIER
Definition: Throttle.cc:42
Throttle::m_out
std::vector< MessageBuffer * > m_out
Definition: Throttle.hh:94
Throttle::ThrottleStats::m_msg_bytes
Stats::Formula * m_msg_bytes[MessageSizeType_NUM]
Definition: Throttle.hh:118
MessageBuffer::dequeue
Tick dequeue(Tick current_time, bool decrement_messages=true)
Updates the delay cycles of the message at the head of the queue, removes it from the queue and retur...
Definition: MessageBuffer.cc:273
Throttle::m_endpoint_bandwidth
int m_endpoint_bandwidth
Definition: Throttle.hh:105
Message::getDestination
virtual const NetDest & getDestination() const
Definition: Message.hh:108
Throttle::wakeup
void wakeup()
Definition: Throttle.cc:148
Throttle::collateStats
void collateStats()
Definition: Throttle.cc:236
Throttle::m_switch_id
const int m_switch_id
Definition: Throttle.hh:98
Throttle::m_vnets
unsigned int m_vnets
Definition: Throttle.hh:95
Throttle::m_link_utilization_proxy
double m_link_utilization_proxy
Definition: Throttle.hh:108
sc_dt::to_string
const std::string to_string(sc_enc enc)
Definition: sc_fxdefs.cc:91
Throttle::m_link_bandwidth_multiplier
int m_link_bandwidth_multiplier
Definition: Throttle.hh:102
Tick
uint64_t Tick
Tick count type.
Definition: types.hh:59
Throttle::clearStats
void clearStats()
Definition: Throttle.cc:230
cast.hh
network_message_to_size
static int network_message_to_size(Message *net_msg_ptr)
Definition: Throttle.cc:252
std::vector< MessageBuffer * >
Throttle::Throttle
Throttle(int sID, RubySystem *rs, NodeID node, Cycles link_latency, int link_bandwidth_multiplier, int endpoint_bandwidth, Switch *em)
Definition: Throttle.cc:49
Throttle::print
void print(std::ostream &out) const
Definition: Throttle.cc:246
Message::getMessageSize
virtual const MessageSizeType & getMessageSize() const
Definition: Message.hh:72
MessageBuffer::setConsumer
void setConsumer(Consumer *consumer)
Definition: MessageBuffer.hh:96
Switch
Definition: Switch.hh:71
MessageBuffer::enqueue
void enqueue(MsgPtr message, Tick curTime, Tick delta)
Definition: MessageBuffer.cc:191
Stats::Vector
A vector of scalar stats.
Definition: statistics.hh:2007
Throttle.hh
Throttle::m_node
NodeID m_node
Definition: Throttle.hh:100
MessageBuffer::peekMsgPtr
const MsgPtr & peekMsgPtr() const
Definition: MessageBuffer.hh:115
Throttle::ThrottleStats::ThrottleStats
ThrottleStats(Stats::Group *parent, const NodeID &nodeID)
Definition: Throttle.cc:267
Clocked::cyclesToTicks
Tick cyclesToTicks(Cycles c) const
Definition: clocked_object.hh:224
PRIORITY_SWITCH_LIMIT
const int PRIORITY_SWITCH_LIMIT
Definition: Throttle.cc:45
DPRINTF
#define DPRINTF(x,...)
Definition: trace.hh:237
Throttle::m_switch
Switch * m_switch
Definition: Throttle.hh:99
RubySystem
Definition: RubySystem.hh:52
Throttle::m_ruby_system
RubySystem * m_ruby_system
Definition: Throttle.hh:106
Throttle::m_wakeups_wo_switch
int m_wakeups_wo_switch
Definition: Throttle.hh:104
MessageBuffer::isReady
bool isReady(Tick current_time) const
Definition: MessageBuffer.cc:482
RubySystem::getStartCycle
Cycles getStartCycle()
Definition: RubySystem.hh:68
Consumer
Definition: Consumer.hh:55
Clocked::curCycle
Cycles curCycle() const
Determine the current cycle, corresponding to a tick aligned to a clock edge.
Definition: clocked_object.hh:192
Clocked::clockEdge
Tick clockEdge(Cycles cycles=Cycles(0)) const
Determine the tick when a cycle begins, by default the current one, but the argument also enables the...
Definition: clocked_object.hh:174
X86ISA::em
Bitfield< 2 > em
Definition: misc.hh:602
cprintf.hh
Network::MessageSizeType_to_int
static uint32_t MessageSizeType_to_int(MessageSizeType size_type)
Definition: Network.cc:158
BROADCAST_SCALING
const int BROADCAST_SCALING
Definition: Throttle.cc:44
RubySystem.hh
Network::getNumberOfVirtualNetworks
static uint32_t getNumberOfVirtualNetworks()
Definition: Network.hh:84
MsgPtr
std::shared_ptr< Message > MsgPtr
Definition: Message.hh:53
MessageBuffer::areNSlotsAvailable
bool areNSlotsAvailable(unsigned int n, Tick curTime)
Definition: MessageBuffer.cc:121
Stats::nozero
const FlagsType nozero
Don't print if this is zero.
Definition: info.hh:58
MessageBuffer.hh
Network.hh
Switch.hh
Throttle::ThrottleStats::m_msg_counts
Stats::Vector * m_msg_counts[MessageSizeType_NUM]
Definition: Throttle.hh:117
Throttle::throttleStats
Throttle::ThrottleStats throttleStats
Throttle::regStats
void regStats()
Definition: Throttle.cc:204
Stats::Formula
A formula for statistics that is calculated when printed.
Definition: statistics.hh:2538
Throttle::m_in
std::vector< MessageBuffer * > m_in
Definition: Throttle.hh:93
Stats::Group
Statistics container.
Definition: group.hh:87
ccprintf
void ccprintf(cp::Print &print)
Definition: cprintf.hh:127
ArmISA::rs
Bitfield< 9, 8 > rs
Definition: miscregs_types.hh:372
Cycles
Cycles is a wrapper class for representing cycle counts, i.e.
Definition: types.hh:79
Throttle::addLinks
void addLinks(const std::vector< MessageBuffer * > &in_vec, const std::vector< MessageBuffer * > &out_vec)
Definition: Throttle.cc:70
Message
Definition: Message.hh:56
NodeID
unsigned int NodeID
Definition: TypeDefines.hh:34
X86ISA::type
type
Definition: misc.hh:727
Throttle::getLinkBandwidth
int getLinkBandwidth() const
Definition: Throttle.hh:73
Stats
Definition: statistics.cc:53
Throttle::operateVnet
void operateVnet(int vnet, int &bw_remainin, bool &schedule_wakeup, MessageBuffer *in, MessageBuffer *out)
Definition: Throttle.cc:92
NetDest::isBroadcast
bool isBroadcast() const
Definition: NetDest.cc:168
MessageBuffer
Definition: MessageBuffer.hh:68
Throttle::m_units_remaining
std::vector< int > m_units_remaining
Definition: Throttle.hh:96
Message.hh
csprintf
std::string csprintf(const char *format, const Args &...args)
Definition: cprintf.hh:158
Throttle::init
void init(NodeID node, Cycles link_latency, int link_bandwidth_multiplier, int endpoint_bandwidth)
Consumer::scheduleEvent
void scheduleEvent(Cycles timeDelta)
Definition: Consumer.cc:50
Stats::constant
Temp constant(T val)
Definition: statistics.hh:2864
Throttle::ThrottleStats::m_link_utilization
Stats::Scalar m_link_utilization
Definition: Throttle.hh:116

Generated on Tue Jun 22 2021 15:28:30 for gem5 by doxygen 1.8.17