gem5  v20.0.0.0
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages
coroutine.hh
Go to the documentation of this file.
1 /*
2  * Copyright (c) 2018 ARM Limited
3  * All rights reserved
4  *
5  * The license below extends only to copyright in the software and shall
6  * not be construed as granting a license to any other intellectual
7  * property including but not limited to intellectual property relating
8  * to a hardware implementation of the functionality of the software
9  * licensed hereunder. You may use the software subject to the license
10  * terms below provided that you ensure that this notice is replicated
11  * unmodified and in its entirety in all distributions of the software,
12  * modified or unmodified, in source code or in binary form.
13  *
14  * Redistribution and use in source and binary forms, with or without
15  * modification, are permitted provided that the following conditions are
16  * met: redistributions of source code must retain the above copyright
17  * notice, this list of conditions and the following disclaimer;
18  * redistributions in binary form must reproduce the above copyright
19  * notice, this list of conditions and the following disclaimer in the
20  * documentation and/or other materials provided with the distribution;
21  * neither the name of the copyright holders nor the names of its
22  * contributors may be used to endorse or promote products derived from
23  * this software without specific prior written permission.
24  *
25  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
26  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
27  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
28  * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
29  * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
30  * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
31  * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
32  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
33  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
34  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
35  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
36  */
37 
38 #ifndef __BASE_COROUTINE_HH__
39 #define __BASE_COROUTINE_HH__
40 
41 #include <functional>
42 #include <stack>
43 
44 #include "base/fiber.hh"
45 
46 namespace m5
47 {
48 
61 template <typename Arg, typename Ret>
62 class Coroutine : public Fiber
63 {
64 
65  // This empty struct type is meant to replace coroutine channels
66  // in case the channel should be void (Coroutine template parameters
67  // are void. (See following ArgChannel, RetChannel typedef)
68  struct Empty {};
69  using ArgChannel = typename std::conditional<
70  std::is_same<Arg, void>::value, Empty, std::stack<Arg>>::type;
71 
72  using RetChannel = typename std::conditional<
73  std::is_same<Ret, void>::value, Empty, std::stack<Ret>>::type;
74 
75  public:
83  class CallerType
84  {
85  friend class Coroutine;
86  protected:
87  CallerType(Coroutine& _coro) : coro(_coro), callerFiber(nullptr) {}
88 
89  public:
97  template <typename T = Ret>
98  CallerType&
99  operator()(typename std::enable_if<
100  !std::is_same<T, void>::value, T>::type param)
101  {
102  retChannel.push(param);
103  callerFiber->run();
104  return *this;
105  }
106 
113  template <typename T = Ret>
114  typename std::enable_if<std::is_same<T, void>::value,
115  CallerType>::type&
117  {
118  callerFiber->run();
119  return *this;
120  }
121 
132  template <typename T = Arg>
133  typename std::enable_if<!std::is_same<T, void>::value, T>::type
134  get()
135  {
136  auto& args_channel = coro.argsChannel;
137  while (args_channel.empty()) {
138  callerFiber->run();
139  }
140 
141  auto ret = args_channel.top();
142  args_channel.pop();
143  return ret;
144  }
145 
146  private:
150  };
151 
152  Coroutine() = delete;
153  Coroutine(const Coroutine& rhs) = delete;
154  Coroutine& operator=(const Coroutine& rhs) = delete;
155 
171  Coroutine(std::function<void(CallerType&)> f, bool run_coroutine = true)
172  : Fiber(), task(f), caller(*this)
173  {
174  // When desired, run the Coroutine after it is created
175  if (run_coroutine)
176  this->call();
177  }
178 
179  virtual ~Coroutine() {}
180 
181  public:
191  template <typename T = Arg>
192  Coroutine&
193  operator()(typename std::enable_if<
194  !std::is_same<T, void>::value, T>::type param)
195  {
196  argsChannel.push(param);
197  this->call();
198  return *this;
199  }
200 
207  template <typename T = Arg>
208  typename std::enable_if<std::is_same<T, void>::value, Coroutine>::type&
210  {
211  this->call();
212  return *this;
213  }
214 
225  template <typename T = Ret>
226  typename std::enable_if<!std::is_same<T, void>::value, T>::type
227  get()
228  {
229  auto& ret_channel = caller.retChannel;
230  while (ret_channel.empty()) {
231  this->call();
232  }
233 
234  auto ret = ret_channel.top();
235  ret_channel.pop();
236  return ret;
237  }
238 
240  operator bool() const { return !this->finished(); }
241 
242  private:
249  void main() override { this->task(caller); }
250 
251  void
253  {
255  run();
256  }
257 
258  private:
261 
263  std::function<void(CallerType&)> task;
264 
267 };
268 
269 } //namespace m5
270 
271 #endif // __BASE_COROUTINE_HH__
void run()
Start executing the fiber represented by this object.
Definition: fiber.cc:163
bool finished() const
Returns whether the "main" function of this fiber has finished.
Definition: fiber.hh:81
void main() override
Overriding base (Fiber) main.
Definition: coroutine.hh:249
CallerType caller
Coroutine caller.
Definition: coroutine.hh:266
std::enable_if< std::is_same< T, void >::value, Coroutine >::type & operator()()
operator() is the way we can jump inside the coroutine.
Definition: coroutine.hh:209
Coroutine(std::function< void(CallerType &)> f, bool run_coroutine=true)
Coroutine constructor.
Definition: coroutine.hh:171
CallerType: A reference to an object of this class will be passed to the coroutine task...
Definition: coroutine.hh:83
Bitfield< 6 > f
Coroutine & operator()(typename std::enable_if< !std::is_same< T, void >::value, T >::type param)
Coroutine interface.
Definition: coroutine.hh:193
uint8_t type
Definition: inet.hh:328
CallerType & operator()(typename std::enable_if< !std::is_same< T, void >::value, T >::type param)
operator() is the way we can jump outside the coroutine and return a value to the caller...
Definition: coroutine.hh:99
This template defines a Coroutine wrapper type with a Boost-like interface.
Definition: coroutine.hh:62
std::enable_if< std::is_same< T, void >::value, CallerType >::type & operator()()
operator() is the way we can jump outside the coroutine
Definition: coroutine.hh:116
Coroutine & operator=(const Coroutine &rhs)=delete
CallerType(Coroutine &_coro)
Definition: coroutine.hh:87
virtual ~Coroutine()
Definition: coroutine.hh:179
std::function< void(CallerType &)> task
Coroutine task.
Definition: coroutine.hh:263
This class represents a fiber, which is a light weight sort of thread which is cooperatively schedule...
Definition: fiber.hh:62
Coroutine()=delete
ArgChannel argsChannel
Arguments for the coroutine.
Definition: coroutine.hh:260
Definition: compiler.hh:90
static Fiber * currentFiber()
Get a pointer to the current running Fiber.
Definition: fiber.cc:181
typename std::conditional< std::is_same< Arg, void >::value, Empty, std::stack< Arg > >::type ArgChannel
Definition: coroutine.hh:70
typename std::conditional< std::is_same< Ret, void >::value, Empty, std::stack< Ret > >::type RetChannel
Definition: coroutine.hh:73

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