gem5  v20.0.0.0
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages
RoutingUnit.cc
Go to the documentation of this file.
1 /*
2  * Copyright (c) 2008 Princeton University
3  * Copyright (c) 2016 Georgia Institute of Technology
4  * All rights reserved.
5  *
6  * Redistribution and use in source and binary forms, with or without
7  * modification, are permitted provided that the following conditions are
8  * met: redistributions of source code must retain the above copyright
9  * notice, this list of conditions and the following disclaimer;
10  * redistributions in binary form must reproduce the above copyright
11  * notice, this list of conditions and the following disclaimer in the
12  * documentation and/or other materials provided with the distribution;
13  * neither the name of the copyright holders nor the names of its
14  * contributors may be used to endorse or promote products derived from
15  * this software without specific prior written permission.
16  *
17  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
18  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
19  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
20  * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
21  * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
22  * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
23  * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
24  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
25  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
26  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
27  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
28  */
29 
30 
32 
33 #include "base/cast.hh"
34 #include "base/logging.hh"
38 
40 {
41  m_router = router;
42  m_routing_table.clear();
43  m_weight_table.clear();
44 }
45 
46 void
47 RoutingUnit::addRoute(const NetDest& routing_table_entry)
48 {
49  m_routing_table.push_back(routing_table_entry);
50 }
51 
52 void
53 RoutingUnit::addWeight(int link_weight)
54 {
55  m_weight_table.push_back(link_weight);
56 }
57 
58 /*
59  * This is the default routing algorithm in garnet.
60  * The routing table is populated during topology creation.
61  * Routes can be biased via weight assignments in the topology file.
62  * Correct weight assignments are critical to provide deadlock avoidance.
63  */
64 
65 int
66 RoutingUnit::lookupRoutingTable(int vnet, NetDest msg_destination)
67 {
68  // First find all possible output link candidates
69  // For ordered vnet, just choose the first
70  // (to make sure different packets don't choose different routes)
71  // For unordered vnet, randomly choose any of the links
72  // To have a strict ordering between links, they should be given
73  // different weights in the topology file
74 
75  int output_link = -1;
76  int min_weight = INFINITE_;
77  std::vector<int> output_link_candidates;
78  int num_candidates = 0;
79 
80  // Identify the minimum weight among the candidate output links
81  for (int link = 0; link < m_routing_table.size(); link++) {
82  if (msg_destination.intersectionIsNotEmpty(m_routing_table[link])) {
83 
84  if (m_weight_table[link] <= min_weight)
85  min_weight = m_weight_table[link];
86  }
87  }
88 
89  // Collect all candidate output links with this minimum weight
90  for (int link = 0; link < m_routing_table.size(); link++) {
91  if (msg_destination.intersectionIsNotEmpty(m_routing_table[link])) {
92 
93  if (m_weight_table[link] == min_weight) {
94 
95  num_candidates++;
96  output_link_candidates.push_back(link);
97  }
98  }
99  }
100 
101  if (output_link_candidates.size() == 0) {
102  fatal("Fatal Error:: No Route exists from this Router.");
103  exit(0);
104  }
105 
106  // Randomly select any candidate output link
107  int candidate = 0;
108  if (!(m_router->get_net_ptr())->isVNetOrdered(vnet))
109  candidate = rand() % num_candidates;
110 
111  output_link = output_link_candidates.at(candidate);
112  return output_link;
113 }
114 
115 
116 void
117 RoutingUnit::addInDirection(PortDirection inport_dirn, int inport_idx)
118 {
119  m_inports_dirn2idx[inport_dirn] = inport_idx;
120  m_inports_idx2dirn[inport_idx] = inport_dirn;
121 }
122 
123 void
124 RoutingUnit::addOutDirection(PortDirection outport_dirn, int outport_idx)
125 {
126  m_outports_dirn2idx[outport_dirn] = outport_idx;
127  m_outports_idx2dirn[outport_idx] = outport_dirn;
128 }
129 
130 // outportCompute() is called by the InputUnit
131 // It calls the routing table by default.
132 // A template for adaptive topology-specific routing algorithm
133 // implementations using port directions rather than a static routing
134 // table is provided here.
135 
136 int
138  PortDirection inport_dirn)
139 {
140  int outport = -1;
141 
142  if (route.dest_router == m_router->get_id()) {
143 
144  // Multiple NIs may be connected to this router,
145  // all with output port direction = "Local"
146  // Get exact outport id from table
147  outport = lookupRoutingTable(route.vnet, route.net_dest);
148  return outport;
149  }
150 
151  // Routing Algorithm set in GarnetNetwork.py
152  // Can be over-ridden from command line using --routing-algorithm = 1
153  RoutingAlgorithm routing_algorithm =
155 
156  switch (routing_algorithm) {
157  case TABLE_: outport =
158  lookupRoutingTable(route.vnet, route.net_dest); break;
159  case XY_: outport =
160  outportComputeXY(route, inport, inport_dirn); break;
161  // any custom algorithm
162  case CUSTOM_: outport =
163  outportComputeCustom(route, inport, inport_dirn); break;
164  default: outport =
165  lookupRoutingTable(route.vnet, route.net_dest); break;
166  }
167 
168  assert(outport != -1);
169  return outport;
170 }
171 
172 // XY routing implemented using port directions
173 // Only for reference purpose in a Mesh
174 // By default Garnet uses the routing table
175 int
177  int inport,
178  PortDirection inport_dirn)
179 {
180  PortDirection outport_dirn = "Unknown";
181 
182  int M5_VAR_USED num_rows = m_router->get_net_ptr()->getNumRows();
183  int num_cols = m_router->get_net_ptr()->getNumCols();
184  assert(num_rows > 0 && num_cols > 0);
185 
186  int my_id = m_router->get_id();
187  int my_x = my_id % num_cols;
188  int my_y = my_id / num_cols;
189 
190  int dest_id = route.dest_router;
191  int dest_x = dest_id % num_cols;
192  int dest_y = dest_id / num_cols;
193 
194  int x_hops = abs(dest_x - my_x);
195  int y_hops = abs(dest_y - my_y);
196 
197  bool x_dirn = (dest_x >= my_x);
198  bool y_dirn = (dest_y >= my_y);
199 
200  // already checked that in outportCompute() function
201  assert(!(x_hops == 0 && y_hops == 0));
202 
203  if (x_hops > 0) {
204  if (x_dirn) {
205  assert(inport_dirn == "Local" || inport_dirn == "West");
206  outport_dirn = "East";
207  } else {
208  assert(inport_dirn == "Local" || inport_dirn == "East");
209  outport_dirn = "West";
210  }
211  } else if (y_hops > 0) {
212  if (y_dirn) {
213  // "Local" or "South" or "West" or "East"
214  assert(inport_dirn != "North");
215  outport_dirn = "North";
216  } else {
217  // "Local" or "North" or "West" or "East"
218  assert(inport_dirn != "South");
219  outport_dirn = "South";
220  }
221  } else {
222  // x_hops == 0 and y_hops == 0
223  // this is not possible
224  // already checked that in outportCompute() function
225  panic("x_hops == y_hops == 0");
226  }
227 
228  return m_outports_dirn2idx[outport_dirn];
229 }
230 
231 // Template for implementing custom routing algorithm
232 // using port directions. (Example adaptive)
233 int
235  int inport,
236  PortDirection inport_dirn)
237 {
238  panic("%s placeholder executed", __FUNCTION__);
239 }
#define panic(...)
This implements a cprintf based panic() function.
Definition: logging.hh:163
std::map< int, PortDirection > m_inports_idx2dirn
Definition: RoutingUnit.hh:81
#define fatal(...)
This implements a cprintf based fatal() function.
Definition: logging.hh:171
Bitfield< 3 > exit
Definition: misc.hh:848
int outportComputeXY(RouteInfo route, int inport, PortDirection inport_dirn)
Definition: RoutingUnit.cc:176
void addWeight(int link_weight)
Definition: RoutingUnit.cc:53
int get_id()
Definition: Router.hh:80
int getNumRows() const
int lookupRoutingTable(int vnet, NetDest net_dest)
Definition: RoutingUnit.cc:66
std::string PortDirection
Definition: Topology.hh:55
#define INFINITE_
Definition: CommonTypes.hh:60
std::map< int, PortDirection > m_outports_idx2dirn
Definition: RoutingUnit.hh:82
GarnetNetwork * get_net_ptr()
Definition: Router.hh:87
void addRoute(const NetDest &routing_table_entry)
Definition: RoutingUnit.cc:47
int dest_router
Definition: CommonTypes.hh:56
std::map< PortDirection, int > m_inports_dirn2idx
Definition: RoutingUnit.hh:80
int outportComputeCustom(RouteInfo route, int inport, PortDirection inport_dirn)
Definition: RoutingUnit.cc:234
void addInDirection(PortDirection inport_dirn, int inport)
Definition: RoutingUnit.cc:117
Definition: Router.hh:56
NetDest net_dest
Definition: CommonTypes.hh:50
int outportCompute(RouteInfo route, int inport, PortDirection inport_dirn)
Definition: RoutingUnit.cc:137
void addOutDirection(PortDirection outport_dirn, int outport)
Definition: RoutingUnit.cc:124
bool intersectionIsNotEmpty(const NetDest &other_netDest) const
Definition: NetDest.cc:216
Router * m_router
Definition: RoutingUnit.hh:73
std::vector< int > m_weight_table
Definition: RoutingUnit.hh:77
std::vector< NetDest > m_routing_table
Definition: RoutingUnit.hh:76
std::map< PortDirection, int > m_outports_dirn2idx
Definition: RoutingUnit.hh:83
RoutingUnit(Router *router)
Definition: RoutingUnit.cc:39
int getRoutingAlgorithm() const
RoutingAlgorithm
Definition: CommonTypes.hh:43

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