xref: /openbmc/phosphor-host-ipmid/include/ipmid/utility.hpp (revision fbc6c9d743d22682f3753841071b8460ad50b5d9)
1e7329c71SVernon Mauery /**
2e7329c71SVernon Mauery  * Copyright © 2018 Intel Corporation
3e7329c71SVernon Mauery  *
4e7329c71SVernon Mauery  * Licensed under the Apache License, Version 2.0 (the "License");
5e7329c71SVernon Mauery  * you may not use this file except in compliance with the License.
6e7329c71SVernon Mauery  * You may obtain a copy of the License at
7e7329c71SVernon Mauery  *
8e7329c71SVernon Mauery  *     http://www.apache.org/licenses/LICENSE-2.0
9e7329c71SVernon Mauery  *
10e7329c71SVernon Mauery  * Unless required by applicable law or agreed to in writing, software
11e7329c71SVernon Mauery  * distributed under the License is distributed on an "AS IS" BASIS,
12e7329c71SVernon Mauery  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13e7329c71SVernon Mauery  * See the License for the specific language governing permissions and
14e7329c71SVernon Mauery  * limitations under the License.
15e7329c71SVernon Mauery  */
16e7329c71SVernon Mauery #pragma once
17e7329c71SVernon Mauery #include <boost/asio/spawn.hpp>
18e7329c71SVernon Mauery #include <boost/callable_traits.hpp>
19*fbc6c9d7SPatrick Williams 
20e7329c71SVernon Mauery #include <cstdint>
21e7329c71SVernon Mauery #include <map>
22e7329c71SVernon Mauery #include <memory>
23e7329c71SVernon Mauery #include <string>
24e7329c71SVernon Mauery #include <tuple>
25e7329c71SVernon Mauery #include <type_traits>
26e7329c71SVernon Mauery #include <vector>
27e7329c71SVernon Mauery 
28e7329c71SVernon Mauery namespace ipmi
29e7329c71SVernon Mauery {
30e7329c71SVernon Mauery 
31ded4c74eSVernon Mauery // forward declare Context and Request for NonIpmiArgsCount
32e7329c71SVernon Mauery struct Context;
33ded4c74eSVernon Mauery namespace message
34ded4c74eSVernon Mauery {
35ded4c74eSVernon Mauery struct Request;
36ded4c74eSVernon Mauery }
37e7329c71SVernon Mauery 
38e7329c71SVernon Mauery namespace utility
39e7329c71SVernon Mauery {
40e7329c71SVernon Mauery 
41e7329c71SVernon Mauery /**
42e7329c71SVernon Mauery  * @brief a utility template to extract the args after N from a tuple
43e7329c71SVernon Mauery  *
44e7329c71SVernon Mauery  * Given a tuple of type <T1, ...TN, TN+1, ...>, provide type = tuple<TN+1,...>
45e7329c71SVernon Mauery  */
46e7329c71SVernon Mauery template <std::size_t N, typename FirstArg, typename... Rest>
47e7329c71SVernon Mauery struct StripFirstArgs;
48e7329c71SVernon Mauery 
49e7329c71SVernon Mauery template <std::size_t N, typename FirstArg, typename... Rest>
50*fbc6c9d7SPatrick Williams struct StripFirstArgs<N, std::tuple<FirstArg, Rest...>> :
51*fbc6c9d7SPatrick Williams     StripFirstArgs<N - 1, std::tuple<Rest...>>
52*fbc6c9d7SPatrick Williams {};
53e7329c71SVernon Mauery 
54e7329c71SVernon Mauery template <typename FirstArg, typename... Rest>
55e7329c71SVernon Mauery struct StripFirstArgs<0, std::tuple<FirstArg, Rest...>>
56e7329c71SVernon Mauery {
57e7329c71SVernon Mauery     using type = std::tuple<FirstArg, Rest...>;
58e7329c71SVernon Mauery };
59e7329c71SVernon Mauery template <std::size_t N>
60e7329c71SVernon Mauery struct StripFirstArgs<N, std::tuple<>>
61e7329c71SVernon Mauery {
62e7329c71SVernon Mauery     using type = std::tuple<>;
63e7329c71SVernon Mauery };
64e7329c71SVernon Mauery 
65e7329c71SVernon Mauery /**
66e7329c71SVernon Mauery  * @brief a utility template to extract the remaining args from a tuple
67e7329c71SVernon Mauery  *
68e7329c71SVernon Mauery  * Given a tuple of type <T1, T2,...>, provide type = tuple<T2,...>
69e7329c71SVernon Mauery  */
70e7329c71SVernon Mauery template <typename T>
71e7329c71SVernon Mauery using StripFirstArg = StripFirstArgs<1, T>;
72e7329c71SVernon Mauery 
73e7329c71SVernon Mauery /**
74e7329c71SVernon Mauery  * @brief a utility template to find the number of non-special arguments
75e7329c71SVernon Mauery  *
76e7329c71SVernon Mauery  * Given a tuple, count the args after the first special args
77e7329c71SVernon Mauery  */
78e7329c71SVernon Mauery template <typename FirstArg, typename... Rest>
79e7329c71SVernon Mauery struct NonIpmiArgsCount;
80e7329c71SVernon Mauery 
81e7329c71SVernon Mauery template <>
82e7329c71SVernon Mauery struct NonIpmiArgsCount<std::tuple<>>
83e7329c71SVernon Mauery {
84e7329c71SVernon Mauery     constexpr static std::size_t size()
85e7329c71SVernon Mauery     {
86e7329c71SVernon Mauery         return 0;
87e7329c71SVernon Mauery     }
88e7329c71SVernon Mauery };
89e7329c71SVernon Mauery template <typename FirstArg, typename... OtherArgs>
90e7329c71SVernon Mauery struct NonIpmiArgsCount<std::tuple<FirstArg, OtherArgs...>>
91e7329c71SVernon Mauery {
sizeipmi::utility::NonIpmiArgsCount92e7329c71SVernon Mauery     constexpr static std::size_t size()
93e7329c71SVernon Mauery     {
94ded4c74eSVernon Mauery         if constexpr (std::is_same<
95ded4c74eSVernon Mauery                           FirstArg,
96ded4c74eSVernon Mauery                           std::shared_ptr<ipmi::message::Request>>::value ||
97ded4c74eSVernon Mauery                       std::is_same<FirstArg,
98ded4c74eSVernon Mauery                                    std::shared_ptr<ipmi::Context>>::value ||
99e7329c71SVernon Mauery                       std::is_same<FirstArg, boost::asio::yield_context>::value)
100e7329c71SVernon Mauery         {
101e7329c71SVernon Mauery             return 1 + NonIpmiArgsCount<std::tuple<OtherArgs...>>::size();
102e7329c71SVernon Mauery         }
103e7329c71SVernon Mauery         else
104e7329c71SVernon Mauery         {
105e7329c71SVernon Mauery             return NonIpmiArgsCount<std::tuple<OtherArgs...>>::size();
106e7329c71SVernon Mauery         }
107e7329c71SVernon Mauery     }
108e7329c71SVernon Mauery };
109e7329c71SVernon Mauery 
110e7329c71SVernon Mauery /**
111e7329c71SVernon Mauery  * @brief a utility template to find the type of the first arg
112e7329c71SVernon Mauery  *
113e7329c71SVernon Mauery  * Given a tuple, provide the type of the first element
114e7329c71SVernon Mauery  */
115e7329c71SVernon Mauery template <typename T>
116e7329c71SVernon Mauery struct GetFirstArg
117e7329c71SVernon Mauery {
118e7329c71SVernon Mauery     using type = void;
119e7329c71SVernon Mauery };
120e7329c71SVernon Mauery 
121e7329c71SVernon Mauery template <typename FirstArg, typename... Rest>
122e7329c71SVernon Mauery struct GetFirstArg<std::tuple<FirstArg, Rest...>>
123e7329c71SVernon Mauery {
124e7329c71SVernon Mauery     using type = FirstArg;
125e7329c71SVernon Mauery };
126e7329c71SVernon Mauery 
127e7329c71SVernon Mauery /**
128e7329c71SVernon Mauery  * @brief a utility template to remove const and reference from types
129e7329c71SVernon Mauery  *
130e7329c71SVernon Mauery  * Given a tuple, provide the type of the first element
131e7329c71SVernon Mauery  */
132e7329c71SVernon Mauery template <typename... Args>
133e7329c71SVernon Mauery struct DecayTuple;
134e7329c71SVernon Mauery 
135e7329c71SVernon Mauery template <typename... Args>
136e7329c71SVernon Mauery struct DecayTuple<std::tuple<Args...>>
137e7329c71SVernon Mauery {
138e7329c71SVernon Mauery     using type = std::tuple<typename std::decay<Args>::type...>;
139e7329c71SVernon Mauery };
140e7329c71SVernon Mauery 
141e7329c71SVernon Mauery /** @brief Convert T[N] to T* if is_same<Tbase,T>
142e7329c71SVernon Mauery  *
143e7329c71SVernon Mauery  *  @tparam Tbase - The base type expected.
144e7329c71SVernon Mauery  *  @tparam T - The type to convert.
145e7329c71SVernon Mauery  */
146e7329c71SVernon Mauery template <typename Tbase, typename T>
147e7329c71SVernon Mauery using ArrayToPtr_t = typename std::conditional_t<
148e7329c71SVernon Mauery     std::is_array<T>::value,
149e7329c71SVernon Mauery     std::conditional_t<std::is_same<Tbase, std::remove_extent_t<T>>::value,
150e7329c71SVernon Mauery                        std::add_pointer_t<std::remove_extent_t<T>>, T>,
151e7329c71SVernon Mauery     T>;
152e7329c71SVernon Mauery 
153e7329c71SVernon Mauery /** @brief Downcast type submembers.
154e7329c71SVernon Mauery  *
155e7329c71SVernon Mauery  * This allows std::tuple and std::pair members to be downcast to their
156e7329c71SVernon Mauery  * non-const, nonref versions of themselves to limit duplication in template
157e7329c71SVernon Mauery  * specializations
158e7329c71SVernon Mauery  *
159e7329c71SVernon Mauery  *  1. Remove references.
160e7329c71SVernon Mauery  *  2. Remove 'const' and 'volatile'.
161e7329c71SVernon Mauery  *  3. Convert 'char[N]' to 'char*'.
162e7329c71SVernon Mauery  */
163e7329c71SVernon Mauery template <typename T>
164e7329c71SVernon Mauery struct DowncastMembers
165e7329c71SVernon Mauery {
166e7329c71SVernon Mauery     using type = T;
167e7329c71SVernon Mauery };
168e7329c71SVernon Mauery template <typename... Args>
169e7329c71SVernon Mauery struct DowncastMembers<std::pair<Args...>>
170e7329c71SVernon Mauery {
171e7329c71SVernon Mauery     using type = std::pair<utility::ArrayToPtr_t<
172e7329c71SVernon Mauery         char, std::remove_cv_t<std::remove_reference_t<Args>>>...>;
173e7329c71SVernon Mauery };
174e7329c71SVernon Mauery 
175e7329c71SVernon Mauery template <typename... Args>
176e7329c71SVernon Mauery struct DowncastMembers<std::tuple<Args...>>
177e7329c71SVernon Mauery {
178e7329c71SVernon Mauery     using type = std::tuple<utility::ArrayToPtr_t<
179e7329c71SVernon Mauery         char, std::remove_cv_t<std::remove_reference_t<Args>>>...>;
180e7329c71SVernon Mauery };
181e7329c71SVernon Mauery 
182e7329c71SVernon Mauery template <typename T>
183e7329c71SVernon Mauery using DowncastMembers_t = typename DowncastMembers<T>::type;
184e7329c71SVernon Mauery 
185e7329c71SVernon Mauery /** @brief Convert some C++ types to others for 'TypeId' conversion purposes.
186e7329c71SVernon Mauery  *
187e7329c71SVernon Mauery  *  Similar C++ types have the same dbus type-id, so 'downcast' those to limit
188e7329c71SVernon Mauery  *  duplication in TypeId template specializations.
189e7329c71SVernon Mauery  *
190e7329c71SVernon Mauery  *  1. Remove references.
191e7329c71SVernon Mauery  *  2. Remove 'const' and 'volatile'.
192e7329c71SVernon Mauery  *  3. Convert 'char[N]' to 'char*'.
193e7329c71SVernon Mauery  */
194e7329c71SVernon Mauery template <typename T>
195e7329c71SVernon Mauery struct TypeIdDowncast
196e7329c71SVernon Mauery {
197e7329c71SVernon Mauery     using type = utility::ArrayToPtr_t<
198e7329c71SVernon Mauery         char, DowncastMembers_t<std::remove_cv_t<std::remove_reference_t<T>>>>;
199e7329c71SVernon Mauery };
200e7329c71SVernon Mauery 
201e7329c71SVernon Mauery template <typename T>
202e7329c71SVernon Mauery using TypeIdDowncast_t = typename TypeIdDowncast<T>::type;
203e7329c71SVernon Mauery 
204e7329c71SVernon Mauery /** @brief Detect if a type is a tuple
205e7329c71SVernon Mauery  *
206e7329c71SVernon Mauery  */
207e7329c71SVernon Mauery template <typename>
208e7329c71SVernon Mauery struct is_tuple : std::false_type
209*fbc6c9d7SPatrick Williams {};
210e7329c71SVernon Mauery 
211e7329c71SVernon Mauery template <typename... T>
212e7329c71SVernon Mauery struct is_tuple<std::tuple<T...>> : std::true_type
213*fbc6c9d7SPatrick Williams {};
214e7329c71SVernon Mauery 
215a3dd7661SVernon Mauery /** @brief used for static_assert in a constexpr-if else statement
216a3dd7661SVernon Mauery  */
217a3dd7661SVernon Mauery template <typename T>
218a3dd7661SVernon Mauery struct dependent_false : std::false_type
219*fbc6c9d7SPatrick Williams {};
220a3dd7661SVernon Mauery 
221e7329c71SVernon Mauery } // namespace utility
222e7329c71SVernon Mauery 
223e7329c71SVernon Mauery } // namespace ipmi
224