gem5  v21.0.1.0
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages
dispatch.hh
Go to the documentation of this file.
1 /*
2  * Copyright 2019 Google Inc.
3  *
4  * Redistribution and use in source and binary forms, with or without
5  * modification, are permitted provided that the following conditions are
6  * met: redistributions of source code must retain the above copyright
7  * notice, this list of conditions and the following disclaimer;
8  * redistributions in binary form must reproduce the above copyright
9  * notice, this list of conditions and the following disclaimer in the
10  * documentation and/or other materials provided with the distribution;
11  * neither the name of the copyright holders nor the names of its
12  * contributors may be used to endorse or promote products derived from
13  * this software without specific prior written permission.
14  *
15  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
16  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
17  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
18  * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
19  * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
20  * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
21  * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
22  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
23  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
24  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
25  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
26  */
27 
28 #ifndef __SIM_GUEST_ABI_DISPATCH_HH__
29 #define __SIM_GUEST_ABI_DISPATCH_HH__
30 
31 #include <functional>
32 #include <sstream>
33 #include <type_traits>
34 
36 #include "sim/guest_abi/layout.hh"
37 
38 class ThreadContext;
39 
40 namespace GuestABI
41 {
42 
43 /*
44  * These functions will likely be common among all ABIs and implement the
45  * mechanism of gathering arguments, calling the target function, and then
46  * storing the result. They might need to be overridden if, for instance,
47  * the location of arguments need to be determined in a different order.
48  * For example, there might be an ABI which gathers arguments starting
49  * from the last in the list instead of the first. This is unlikely but
50  * still possible to support by redefining these functions..
51  */
52 
53 // With no arguments to gather, call the target function and store the
54 // result.
55 template <typename ABI, bool store_ret, typename Ret>
56 static typename std::enable_if_t<!std::is_void<Ret>::value && store_ret, Ret>
57 callFrom(ThreadContext *tc, typename ABI::State &state,
58  std::function<Ret(ThreadContext *)> target)
59 {
60  Ret ret = target(tc);
61  storeResult<ABI, Ret>(tc, ret, state);
62  return ret;
63 }
64 
65 template <typename ABI, bool store_ret, typename Ret>
66 static typename std::enable_if_t<!std::is_void<Ret>::value && !store_ret, Ret>
67 callFrom(ThreadContext *tc, typename ABI::State &state,
68  std::function<Ret(ThreadContext *)> target)
69 {
70  return target(tc);
71 }
72 
73 // With no arguments to gather and nothing to return, call the target function.
74 template <typename ABI>
75 static void
76 callFrom(ThreadContext *tc, typename ABI::State &state,
77  std::function<void(ThreadContext *)> target)
78 {
79  target(tc);
80 }
81 
82 // Recursively gather arguments for target from tc until we get to the base
83 // case above.
84 template <typename ABI, bool store_ret, typename Ret,
85  typename NextArg, typename ...Args>
86 static typename std::enable_if_t<!std::is_void<Ret>::value, Ret>
87 callFrom(ThreadContext *tc, typename ABI::State &state,
88  std::function<Ret(ThreadContext *, NextArg, Args...)> target)
89 {
90  // Extract the next argument from the thread context.
91  NextArg next = getArgument<ABI, NextArg>(tc, state);
92 
93  // Build a partial function which adds the next argument to the call.
94  std::function<Ret(ThreadContext *, Args...)> partial =
95  [target,next](ThreadContext *_tc, Args... args) {
96  return target(_tc, next, args...);
97  };
98 
99  // Recursively handle any remaining arguments.
100  return callFrom<ABI, store_ret, Ret, Args...>(tc, state, partial);
101 }
102 
103 // Recursively gather arguments for target from tc until we get to the base
104 // case above. This version is for functions that don't return anything.
105 template <typename ABI, typename NextArg, typename ...Args>
106 static void
107 callFrom(ThreadContext *tc, typename ABI::State &state,
108  std::function<void(ThreadContext *, NextArg, Args...)> target)
109 {
110  // Extract the next argument from the thread context.
111  NextArg next = getArgument<ABI, NextArg>(tc, state);
112 
113  // Build a partial function which adds the next argument to the call.
114  std::function<void(ThreadContext *, Args...)> partial =
115  [target,next](ThreadContext *_tc, Args... args) {
116  target(_tc, next, args...);
117  };
118 
119  // Recursively handle any remaining arguments.
120  callFrom<ABI, Args...>(tc, state, partial);
121 }
122 
123 
124 
125 /*
126  * These functions are like the ones above, except they print the arguments
127  * a target function would be called with instead of actually calling it.
128  */
129 
130 // With no arguments to print, add the closing parenthesis and return.
131 template <typename ABI, typename Ret>
132 static void
133 dumpArgsFrom(int count, std::ostream &os, ThreadContext *tc,
134  typename ABI::State &state)
135 {
136  os << ")";
137 }
138 
139 // Recursively gather arguments for target from tc until we get to the base
140 // case above, and append those arguments to the string stream being
141 // constructed.
142 template <typename ABI, typename Ret, typename NextArg, typename ...Args>
143 static void
144 dumpArgsFrom(int count, std::ostream &os, ThreadContext *tc,
145  typename ABI::State &state)
146 {
147  // Either open the parenthesis or add a comma, depending on where we are
148  // in the argument list.
149  os << (count ? ", " : "(");
150 
151  // Extract the next argument from the thread context.
152  NextArg next = getArgument<ABI, NextArg>(tc, state);
153 
154  // Add this argument to the list.
155  os << next;
156 
157  // Recursively handle any remaining arguments.
158  dumpArgsFrom<ABI, Ret, Args...>(count + 1, os, tc, state);
159 }
160 
161 } // namespace GuestABI
162 
163 #endif // __SIM_GUEST_ABI_DISPATCH_HH__
X86ISA::os
Bitfield< 17 > os
Definition: misc.hh:803
definition.hh
X86ISA::count
count
Definition: misc.hh:703
ThreadContext
ThreadContext is the external interface to all thread state for anything outside of the CPU.
Definition: thread_context.hh:88
GuestABI
Definition: aapcs32.hh:66
GuestABI::callFrom
static std::enable_if_t<!std::is_void< Ret >::value &&store_ret, Ret > callFrom(ThreadContext *tc, typename ABI::State &state, std::function< Ret(ThreadContext *)> target)
Definition: dispatch.hh:57
GuestABI::dumpArgsFrom
static void dumpArgsFrom(int count, std::ostream &os, ThreadContext *tc, typename ABI::State &state)
Definition: dispatch.hh:133
layout.hh

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