File: members.h

package info (click to toggle)
libnop 0.0~git20200728.45dfe0f-5
  • links: PTS, VCS
  • area: main
  • in suites: trixie
  • size: 1,452 kB
  • sloc: cpp: 13,946; ansic: 3,537; makefile: 100; python: 73
file content (125 lines) | stat: -rw-r--r-- 4,017 bytes parent folder | download | duplicates (4)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
/*
 * Copyright 2017 The Native Object Protocols Authors
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *      http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

#ifndef LIBNOP_INCLUDE_NOP_BASE_MEMBERS_H_
#define LIBNOP_INCLUDE_NOP_BASE_MEMBERS_H_

#include <nop/base/encoding.h>
#include <nop/base/logical_buffer.h>
#include <nop/base/utility.h>
#include <nop/types/detail/member_pointer.h>

namespace nop {

//
// struct/class T encoding format:
//
// +-----+---------+-----//----+
// | STC | INT64:N | N MEMBERS |
// +-----+---------+-----//----+
//
// Members must be valid encodings of their member type.
//

template <typename T>
struct Encoding<T, EnableIfHasMemberList<T>> : EncodingIO<T> {
  static constexpr EncodingByte Prefix(const T& /*value*/) {
    return EncodingByte::Structure;
  }

  static constexpr std::size_t Size(const T& value) {
    return BaseEncodingSize(Prefix(value)) + Encoding<SizeType>::Size(Count) +
           Size(value, Index<Count>{});
  }

  static constexpr bool Match(EncodingByte prefix) {
    return prefix == EncodingByte::Structure;
  }

  template <typename Writer>
  static constexpr Status<void> WritePayload(EncodingByte /*prefix*/,
                                             const T& value, Writer* writer) {
    auto status = Encoding<SizeType>::Write(Count, writer);
    if (!status)
      return status;
    else
      return WriteMembers(value, writer, Index<Count>{});
  }

  template <typename Reader>
  static constexpr Status<void> ReadPayload(EncodingByte /*prefix*/, T* value,
                                            Reader* reader) {
    SizeType size = 0;
    auto status = Encoding<SizeType>::Read(&size, reader);
    if (!status)
      return status;
    else if (size != Count)
      return ErrorStatus::InvalidMemberCount;
    else
      return ReadMembers(value, reader, Index<Count>{});
  }

 private:
  enum : std::size_t { Count = MemberListTraits<T>::MemberList::Count };

  using MemberList = typename MemberListTraits<T>::MemberList;

  template <std::size_t Index>
  using PointerAt = typename MemberList::template At<Index>;

  static constexpr std::size_t Size(const T& /*value*/, Index<0>) { return 0; }

  template <std::size_t index>
  static constexpr std::size_t Size(const T& value, Index<index>) {
    return Size(value, Index<index - 1>{}) + PointerAt<index - 1>::Size(value);
  }

  template <typename Writer>
  static constexpr Status<void> WriteMembers(const T& /*value*/,
                                             Writer* /*writer*/, Index<0>) {
    return {};
  }

  template <std::size_t index, typename Writer>
  static constexpr Status<void> WriteMembers(const T& value, Writer* writer,
                                             Index<index>) {
    auto status = WriteMembers(value, writer, Index<index - 1>{});
    if (!status)
      return status;
    else
      return PointerAt<index - 1>::Write(value, writer, MemberList{});
  }

  template <typename Reader>
  static constexpr Status<void> ReadMembers(T* /*value*/, Reader* /*reader*/,
                                            Index<0>) {
    return {};
  }

  template <std::size_t index, typename Reader>
  static constexpr Status<void> ReadMembers(T* value, Reader* reader,
                                            Index<index>) {
    auto status = ReadMembers(value, reader, Index<index - 1>{});
    if (!status)
      return status;
    else
      return PointerAt<index - 1>::Read(value, reader, MemberList{});
  }
};

}  // namespace nop

#endif  // LIBNOP_INCLUDE_NOP_BASE_MEMBERS_H_