COMMS
Template library intended to help with implementation of communication protocols.
Loading...
Searching...
No Matches
AssignHelper.h
1//
2// Copyright 2020 - 2025 (C). Alex Robenko. All rights reserved.
3//
4// This Source Code Form is subject to the terms of the Mozilla Public
5// License, v. 2.0. If a copy of the MPL was not distributed with this
6// file, You can obtain one at http://mozilla.org/MPL/2.0/.
7
8#pragma once
9
10#include <type_traits>
11#include <iterator>
12
13#include "comms/util/detect.h"
15#include "comms/Assert.h"
16#include "comms/details/tag.h"
17
18namespace comms
19{
20
21namespace util
22{
23
24namespace details
25{
26
27template <typename...>
28class AssignHelper
29{
30public:
31 template <typename T, typename TIter>
32 static void assign(T& obj, TIter from, TIter to)
33 {
34 using ObjType = typename std::decay<decltype(obj)>::type;
35 static_assert(!std::is_same<Tag<ObjType>, UnknownTag<> >::value, "Assignment to provided type is not supported");
36 assignInternal(obj, from, to, Tag<ObjType>());
37 }
38
39private:
40 template <typename... TParams>
41 using UseAssignTag = comms::details::tag::Tag1<>;
42
43 template <typename... TParams>
44 using UsePtrSizeConstructorTag = comms::details::tag::Tag2<>;
45
46 template <typename... TParams>
47 using StdSpanTag = comms::details::tag::Tag3<>;
48
49 template <typename... TParams>
50 using UnknownTag = comms::details::tag::Tag4<>;
51
52 template <typename T>
53 using ConstructorTag =
54 typename comms::util::LazyShallowConditional<
55 comms::util::detect::hasPtrSizeConstructor<T>()
56 >::template Type<
57 UsePtrSizeConstructorTag,
58 UnknownTag
59 >;
60
61 template <typename T>
62 using SpanConstructorTag =
63 typename comms::util::LazyShallowConditional<
64 comms::util::detect::details::IsStdSpan<T>::Value
65 >::template Type<
66 StdSpanTag,
67 ConstructorTag,
68 T
69 >;
70
71 template <typename T>
72 using Tag =
73 typename comms::util::LazyShallowConditional<
74 comms::util::detect::hasAssignFunc<T>()
75 >::template Type<
76 UseAssignTag,
77 SpanConstructorTag,
78 T
79 >;
80
81 template <typename T, typename TIter, typename... TParams>
82 static void assignInternal(T& obj, TIter from, TIter to, UseAssignTag<TParams...>)
83 {
84 obj.assign(from, to);
85 }
86
87 template <typename T, typename TIter, typename... TParams>
88 static void assignInternal(T& obj, TIter from, TIter to, UsePtrSizeConstructorTag<TParams...>)
89 {
90 using IterType = typename std::decay<TIter>::type;
91 using IterTag = typename std::iterator_traits<IterType>::iterator_category;
92 static_assert(std::is_base_of<std::random_access_iterator_tag, IterTag>::value,
93 "Only random access iterator is supported for provided type assignments");
94
95 auto diff = std::distance(from, to);
96 if (diff < 0) {
97 static constexpr bool Invalid_iterators_used_for_assignment = false;
98 static_cast<void>(Invalid_iterators_used_for_assignment);
99 COMMS_ASSERT(Invalid_iterators_used_for_assignment);
100 return;
101 }
102
103 using ObjType = typename std::decay<decltype(obj)>::type;
104 obj = ObjType(&(*from), static_cast<std::size_t>(diff));
105 }
106
107 template <typename T, typename TIter, typename... TParams>
108 static void assignInternal(T& obj, TIter from, TIter to, StdSpanTag<TParams...>)
109 {
110 using ObjType = typename std::decay<decltype(obj)>::type;
111 using ConstPointerType = typename ObjType::const_pointer;
112 using PointerType = typename ObjType::pointer;
113 auto fromPtr = const_cast<PointerType>(reinterpret_cast<ConstPointerType>(&(*from)));
114 auto toPtr = const_cast<PointerType>(reinterpret_cast<ConstPointerType>(&(*to)));
115 assignInternal(obj, fromPtr, toPtr, UsePtrSizeConstructorTag<TParams...>());
116 }
117};
118
119} // namespace details
120
121} // namespace util
122
123} // namespace comms
This file contains classes required for generic custom assertion functionality.
#define COMMS_ASSERT(expr)
Generic assert macro.
Definition Assert.h:170
void assign(T &obj, TIter from, TIter to)
Assigns a new value to provided object.
Definition assign.h:39
Main namespace for all classes / functions of COMMS library.
Replacement to some types from standard type_traits.
Various compile-time detection functions of whether specific member functions and/or types exist.