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