gem5  v21.0.0.0
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages
episode.cc
Go to the documentation of this file.
1 /*
2  * Copyright (c) 2017-2021 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 
35 
36 #include <fstream>
37 #include <unordered_set>
38 
41 
42 Episode::Episode(ProtocolTester* _tester, TesterThread* _thread, int num_loads,
43  int num_stores)
44  : tester(_tester),
45  thread(_thread),
46  numLoads(num_loads),
47  numStores(num_stores),
48  nextActionIdx(0)
49 {
50  assert(tester && thread);
51 
54  assert(numLanes > 0);
55 
57  assert(addrManager);
58 
60  // generate a sequence of actions
61  initActions();
62  isActive = true;
63 
64  DPRINTFN("Episode %d\n", episodeId);
65 }
66 
68 {
69  for (Episode::Action* action : actions) {
70  assert(action);
71  delete action;
72  }
73 }
74 
75 const Episode::Action*
77 {
78  if (nextActionIdx < actions.size())
79  return actions[nextActionIdx];
80  else
81  return nullptr;
82 }
83 
84 void
86 {
87  assert(nextActionIdx < actions.size());
88  nextActionIdx++;
89 }
90 
91 void
93 {
94  // first, push Atomic & then Acquire action
97 
98  // second, push a number of LD/ST actions
99  int num_loads = numLoads;
100  int num_stores = numStores;
101  while ((num_loads + num_stores) > 0) {
102  switch (random() % 2) {
103  case 0: // Load
104  if (num_loads > 0) {
105  actions.push_back(new Action(Action::Type::LOAD,
106  numLanes));
107  num_loads--;
108  }
109  break;
110  case 1: // Store
111  if (num_stores > 0) {
112  actions.push_back(new Action(Action::Type::STORE,
113  numLanes));
114  num_stores--;
115  }
116  break;
117  default:
118  assert(false);
119  }
120  }
121 
122  // last, push an Release & then Atomic action
123  actions.push_back(new Action(Action::Type::RELEASE, numLanes));
124  actions.push_back(new Action(Action::Type::ATOMIC, numLanes));
125 
126  // for each lane, pick a list of locations
127  Location normal_loc;
128 
129  for (int lane = 0; lane < numLanes; ++lane) {
131 
132  // first, we select atomic loc for this lane
133  // atomic loc for this lane should not have been picked yet
135  // pick randomly an atomic location
137  assert(atomicLocs[lane] >= 0);
138 
139  // go through each action in this lane and set its location
140  for (Action* action : actions) {
141  assert(action);
142 
143  switch (action->getType()) {
145  action->setLocation(lane, atomicLocs[lane]);
146  break;
147  case Action::Type::LOAD:
148  // pick randomly a normal location
149  normal_loc = addrManager->
150  getLoadLoc(atomicLocs[lane]);
151  assert(normal_loc >= AddressManager::INVALID_LOCATION);
152 
153  if (normal_loc != AddressManager::INVALID_LOCATION) {
154  // check DRF
155  if (!tester->checkDRF(atomicLocs[lane],
156  normal_loc, false) ||
157  !this->checkDRF(atomicLocs[lane], normal_loc,
158  false, lane)) {
159  panic("TestTh %d - Data race detected. STOPPED!\n",
161  }
162  }
163 
164  action->setLocation(lane, normal_loc);
165  break;
166  case Action::Type::STORE:
167  // pick randomly a normal location
168  normal_loc = addrManager->
169  getStoreLoc(atomicLocs[lane]);
170  assert(normal_loc >= AddressManager::INVALID_LOCATION);
171 
172  if (normal_loc != AddressManager::INVALID_LOCATION) {
173  // check DRF
174  if (!tester->checkDRF(atomicLocs[lane],
175  normal_loc, true) ||
176  !this->checkDRF(atomicLocs[lane], normal_loc,
177  true, lane)) {
178  panic("TestTh %d - Data race detected. STOPPED!\n",
180  }
181  }
182 
183  action->setLocation(lane, normal_loc);
184  break;
187  // no op
188  break;
189  default:
190  panic("Invalid action type\n");
191  }
192  }
193 
195  }
196 }
197 
198 void
200 {
201  // release all locations this episode has picked and used
202  Location atomic_loc, normal_loc;
203  for (int lane = 0; lane < numLanes; ++lane) {
206 
207  std::unordered_set<Location> unique_loc_set;
208 
209  for (Action* action : actions) {
210  assert(action);
211 
212  if (action->isAtomicAction()) {
213  if (atomic_loc == AddressManager::INVALID_LOCATION) {
214  atomic_loc = action->getLocation(lane);
215  } else {
216  // both atomic ops in the same lane must be
217  // at the same location
218  assert(atomic_loc == action->getLocation(lane));
219  }
220  } else if (!action->isMemFenceAction()) {
221  assert(atomic_loc >= 0);
222  normal_loc = action->getLocation(lane);
223 
224  if (normal_loc >= 0)
225  unique_loc_set.insert(normal_loc);
226  }
227  }
228 
229  // each unique loc can be released only once
230  for (Location loc : unique_loc_set)
231  addrManager->releaseLocation(atomic_loc, loc);
232  }
233 
234  // this episode is no longer active
235  isActive = false;
236 }
237 
238 bool
239 Episode::checkDRF(Location atomic_loc, Location loc, bool isStore,
240  int max_lane) const
241 {
242  assert(atomic_loc != AddressManager::INVALID_LOCATION);
243  assert(loc != AddressManager::INVALID_LOCATION);
244  assert(max_lane <= numLanes);
245 
246  for (int lane = 0; lane < max_lane; ++lane) {
247  if (atomic_loc == atomicLocs[lane]) {
248  for (const Action* action : actions) {
249  if (!action->isAtomicAction() &&
250  !action->isMemFenceAction()) {
251  if (isStore && loc == action->getLocation(lane)) {
252  warn("ST at location %d races against thread %d\n",
253  loc, thread->getTesterThreadId());
254  return false;
255  } else if (!isStore &&
256  action->getType() == Action::Type::STORE &&
257  loc == action->getLocation(lane)) {
258  warn("LD at location %d races against thread %d\n",
259  loc, thread->getTesterThreadId());
260  return false;
261  }
262  }
263  }
264  }
265  }
266 
267  return true;
268 }
269 
270 // -------------------- Action class ----------------------------
272  : type(t),
273  numLanes(num_lanes)
274 {
275  assert(numLanes > 0);
276  locations.resize(numLanes);
278 }
279 
280 void
282 {
283  assert(lane >= 0 && lane < numLanes);
284  locations[lane] = loc;
285 }
286 
289 {
290  assert(lane >= 0 && lane < numLanes);
291  return locations[lane];
292 }
293 
294 bool
296 {
297  return (type == Type::ATOMIC);
298 }
299 
300 bool
302 {
303  return (type == Type::ACQUIRE || type == Type::RELEASE);
304 }
305 
306 const std::string
308 {
309  if (type == Type::ACQUIRE)
310  return "ACQUIRE";
311  else if (type == Type::RELEASE)
312  return "RELEASE";
313  else if (type == Type::ATOMIC)
314  return "ATOMIC";
315  else if (type == Type::LOAD)
316  return "LOAD";
317  else if (type == Type::STORE)
318  return "STORE";
319  else
320  panic("Invalid action type\n");
321 }
Episode::Action::getLocation
Location getLocation(int lane) const
Definition: episode.cc:288
Episode::Action::isAtomicAction
bool isAtomicAction() const
Definition: episode.cc:295
Episode::Action::Type
Type
Definition: episode.hh:52
Episode::actions
ActionList actions
Definition: episode.hh:107
episode.hh
warn
#define warn(...)
Definition: logging.hh:239
Episode::thread
TesterThread * thread
Definition: episode.hh:100
AddressManager::Location
int32_t Location
Definition: address_manager.hh:126
Episode::~Episode
~Episode()
Definition: episode.cc:67
ProtocolTester::checkDRF
bool checkDRF(Location atomic_loc, Location loc, bool isStore) const
Definition: protocol_tester.cc:296
TesterThread
Definition: tester_thread.hh:48
Episode::Action::Type::LOAD
@ LOAD
Episode::initActions
void initActions()
Definition: episode.cc:92
Episode::atomicLocs
AtomicLocationList atomicLocs
Definition: episode.hh:110
Episode::Location
AddressManager::Location Location
Definition: episode.hh:47
ProtocolTester::getNextEpisodeID
int getNextEpisodeID()
Definition: protocol_tester.hh:134
Episode::Action::numLanes
int numLanes
Definition: episode.hh:72
Episode::numStores
int numStores
Definition: episode.hh:116
Episode::Action
Definition: episode.hh:50
Episode::tester
ProtocolTester * tester
Definition: episode.hh:99
Episode::checkDRF
bool checkDRF(Location atomic_loc, Location loc, bool isStore, int max_lane) const
Definition: episode.cc:239
ProtocolTester::getAddressManager
AddressManager * getAddressManager() const
Definition: protocol_tester.hh:127
tester_thread.hh
Episode::Action::Type::RELEASE
@ RELEASE
Episode::addrManager
AddressManager * addrManager
Definition: episode.hh:101
AddressManager::finishLocSelection
void finishLocSelection(Location atomic_loc)
Definition: address_manager.cc:116
Episode::Action::Type::ACQUIRE
@ ACQUIRE
Episode::numLanes
int numLanes
Definition: episode.hh:120
AddressManager::releaseLocation
void releaseLocation(Location atomic_loc, Location loc)
Definition: address_manager.cc:123
Episode::episodeId
int episodeId
Definition: episode.hh:104
ProtocolTester
Definition: protocol_tester.hh:68
Episode::popAction
void popAction()
Definition: episode.cc:85
Episode::Action::printType
const std::string printType() const
Definition: episode.cc:307
Episode::Action::locations
LocationList locations
Definition: episode.hh:74
Episode::completeEpisode
void completeEpisode()
Definition: episode.cc:199
Episode::Action::setLocation
void setLocation(int lane, Location loc)
Definition: episode.cc:281
Episode::isActive
bool isActive
Definition: episode.hh:113
TesterThread::getTesterThreadId
int getTesterThreadId() const
Definition: tester_thread.hh:74
Episode::Action::Type::STORE
@ STORE
ArmISA::t
Bitfield< 5 > t
Definition: miscregs_types.hh:67
Episode::numLoads
int numLoads
Definition: episode.hh:115
protocol_tester.hh
AddressManager::getAtomicLoc
Location getAtomicLoc()
Definition: address_manager.cc:94
Episode::Episode
Episode(ProtocolTester *tester, TesterThread *thread, int num_loads, int num_stores)
Definition: episode.cc:42
TesterThread::getNumLanes
int getNumLanes() const
Definition: tester_thread.hh:75
X86ISA::type
type
Definition: misc.hh:727
MipsISA::random
random
Definition: pra_constants.hh:50
DPRINTFN
#define DPRINTFN(...)
Definition: trace.hh:241
Episode::peekCurAction
const Action * peekCurAction() const
Definition: episode.cc:76
AddressManager::INVALID_LOCATION
static const int INVALID_LOCATION
Definition: address_manager.hh:151
Episode::Action::isMemFenceAction
bool isMemFenceAction() const
Definition: episode.cc:301
Episode::Action::Action
Action(Type t, int num_lanes)
Definition: episode.cc:271
Episode::nextActionIdx
int nextActionIdx
Definition: episode.hh:118
panic
#define panic(...)
This implements a cprintf based panic() function.
Definition: logging.hh:171
Episode::Action::Type::ATOMIC
@ ATOMIC

Generated on Tue Mar 23 2021 19:41:25 for gem5 by doxygen 1.8.17