Austin Schuh | 09d7ffa | 2019-10-03 23:43:34 -0700 | [diff] [blame] | 1 | #ifndef AOS_FLATBUFFER_MERGE_H_ |
| 2 | #define AOS_FLATBUFFER_MERGE_H_ |
| 3 | |
| 4 | #include <cstddef> |
| 5 | #include <string> |
| 6 | |
| 7 | #include "flatbuffers/flatbuffers.h" |
| 8 | |
Philipp Schrader | 790cb54 | 2023-07-05 21:06:52 -0700 | [diff] [blame] | 9 | #include "aos/flatbuffers.h" |
| 10 | |
Austin Schuh | 09d7ffa | 2019-10-03 23:43:34 -0700 | [diff] [blame] | 11 | namespace aos { |
| 12 | |
Austin Schuh | e93d864 | 2019-10-13 15:27:07 -0700 | [diff] [blame] | 13 | // Merges 2 flat buffers with the provided type table into the builder. Returns |
| 14 | // the offset to the flatbuffers. |
| 15 | // One or both of t1 and t2 must be non-null. If one is null, this method |
Austin Schuh | 48d10d6 | 2022-10-16 22:19:23 -0700 | [diff] [blame] | 16 | // copies instead of merging. |
Austin Schuh | e93d864 | 2019-10-13 15:27:07 -0700 | [diff] [blame] | 17 | flatbuffers::Offset<flatbuffers::Table> MergeFlatBuffers( |
| 18 | const flatbuffers::TypeTable *typetable, const flatbuffers::Table *t1, |
| 19 | const flatbuffers::Table *t2, flatbuffers::FlatBufferBuilder *fbb); |
Austin Schuh | 09d7ffa | 2019-10-03 23:43:34 -0700 | [diff] [blame] | 20 | |
| 21 | template <class T> |
Austin Schuh | e93d864 | 2019-10-13 15:27:07 -0700 | [diff] [blame] | 22 | inline flatbuffers::Offset<T> MergeFlatBuffers( |
Austin Schuh | add6eb3 | 2020-11-09 21:24:26 -0800 | [diff] [blame] | 23 | const flatbuffers::Table *t1, const flatbuffers::Table *t2, |
| 24 | flatbuffers::FlatBufferBuilder *fbb) { |
Austin Schuh | e93d864 | 2019-10-13 15:27:07 -0700 | [diff] [blame] | 25 | return MergeFlatBuffers(T::MiniReflectTypeTable(), t1, t2, fbb).o; |
| 26 | } |
| 27 | |
| 28 | template <class T> |
Austin Schuh | 40485ed | 2019-10-26 21:51:44 -0700 | [diff] [blame] | 29 | inline aos::FlatbufferDetachedBuffer<T> MergeFlatBuffers(const T *fb1, |
| 30 | const T *fb2) { |
James Kuszmaul | f3a3be2 | 2020-01-04 12:12:00 -0800 | [diff] [blame] | 31 | flatbuffers::FlatBufferBuilder fbb; |
Austin Schuh | d7b15da | 2020-02-17 15:06:11 -0800 | [diff] [blame] | 32 | fbb.ForceDefaults(true); |
Austin Schuh | 12417a0 | 2021-02-07 22:44:24 -0800 | [diff] [blame] | 33 | fbb.DedupVtables(false); |
James Kuszmaul | f3a3be2 | 2020-01-04 12:12:00 -0800 | [diff] [blame] | 34 | fbb.Finish(MergeFlatBuffers<T>( |
| 35 | reinterpret_cast<const flatbuffers::Table *>(fb1), |
| 36 | reinterpret_cast<const flatbuffers::Table *>(fb2), &fbb)); |
| 37 | return aos::FlatbufferDetachedBuffer<T>(fbb.Release()); |
Austin Schuh | e93d864 | 2019-10-13 15:27:07 -0700 | [diff] [blame] | 38 | } |
| 39 | |
| 40 | template <class T> |
Austin Schuh | e695821 | 2020-10-19 11:48:14 -0700 | [diff] [blame] | 41 | inline flatbuffers::Offset<T> MergeFlatBuffers( |
| 42 | const T *fb1, const T *fb2, flatbuffers::FlatBufferBuilder *fbb) { |
| 43 | return MergeFlatBuffers<T>(reinterpret_cast<const flatbuffers::Table *>(fb1), |
| 44 | reinterpret_cast<const flatbuffers::Table *>(fb2), |
| 45 | fbb); |
| 46 | } |
| 47 | |
Austin Schuh | add6eb3 | 2020-11-09 21:24:26 -0800 | [diff] [blame] | 48 | template <class T> |
| 49 | inline aos::FlatbufferDetachedBuffer<T> MergeFlatBuffers( |
| 50 | const aos::Flatbuffer<T> &fb1, const aos::Flatbuffer<T> &fb2) { |
| 51 | return aos::FlatbufferDetachedBuffer<T>( |
| 52 | MergeFlatBuffers<T>(&fb1.message(), &fb2.message())); |
| 53 | } |
| 54 | |
Austin Schuh | 4b5c22a | 2020-11-30 22:58:43 -0800 | [diff] [blame] | 55 | template <class T> |
| 56 | inline flatbuffers::Offset<T> MergeFlatBuffers( |
| 57 | const aos::Flatbuffer<T> &fb1, const aos::Flatbuffer<T> &fb2, |
| 58 | flatbuffers::FlatBufferBuilder *fbb) { |
| 59 | return MergeFlatBuffers<T>( |
| 60 | reinterpret_cast<const flatbuffers::Table *>(&fb1.message()), |
| 61 | reinterpret_cast<const flatbuffers::Table *>(&fb2.message()), fbb); |
| 62 | } |
| 63 | |
Austin Schuh | a4fc60f | 2020-11-01 23:06:47 -0800 | [diff] [blame] | 64 | // Copies a flatbuffer by walking the tree and copying all the pieces. This |
| 65 | // converts DAGs to trees. |
Austin Schuh | e695821 | 2020-10-19 11:48:14 -0700 | [diff] [blame] | 66 | template <class T> |
Austin Schuh | a4fc60f | 2020-11-01 23:06:47 -0800 | [diff] [blame] | 67 | inline flatbuffers::Offset<T> RecursiveCopyFlatBuffer( |
Austin Schuh | e93d864 | 2019-10-13 15:27:07 -0700 | [diff] [blame] | 68 | const T *t1, flatbuffers::FlatBufferBuilder *fbb) { |
| 69 | return MergeFlatBuffers<T>(reinterpret_cast<const flatbuffers::Table *>(t1), |
| 70 | nullptr, fbb); |
| 71 | } |
| 72 | |
Austin Schuh | a4fc60f | 2020-11-01 23:06:47 -0800 | [diff] [blame] | 73 | // Copies a flatbuffer by finding the extents of the memory using the typetable |
| 74 | // and copying the containing memory. This doesn't allocate memory, and |
| 75 | // preserves DAGs. |
| 76 | flatbuffers::Offset<flatbuffers::Table> CopyFlatBuffer( |
| 77 | const flatbuffers::Table *t1, const flatbuffers::TypeTable *typetable, |
| 78 | flatbuffers::FlatBufferBuilder *fbb); |
| 79 | |
| 80 | template <class T> |
| 81 | inline flatbuffers::Offset<T> CopyFlatBuffer( |
| 82 | const T *t1, flatbuffers::FlatBufferBuilder *fbb) { |
| 83 | return flatbuffers::Offset<T>( |
| 84 | CopyFlatBuffer(reinterpret_cast<const flatbuffers::Table *>(t1), |
| 85 | T::MiniReflectTypeTable(), fbb) |
| 86 | .o); |
| 87 | } |
| 88 | |
| 89 | template <class T> |
| 90 | inline flatbuffers::Offset<T> CopyFlatBuffer( |
| 91 | const Flatbuffer<T> &t1, flatbuffers::FlatBufferBuilder *fbb) { |
| 92 | return flatbuffers::Offset<T>( |
| 93 | CopyFlatBuffer( |
| 94 | reinterpret_cast<const flatbuffers::Table *>(&t1.message()), |
| 95 | T::MiniReflectTypeTable(), fbb) |
| 96 | .o); |
| 97 | } |
| 98 | |
Austin Schuh | add6eb3 | 2020-11-09 21:24:26 -0800 | [diff] [blame] | 99 | namespace flatbuffer_merge_internal { |
| 100 | |
| 101 | inline flatbuffers::uoffset_t DoBlindCopyFlatBuffer( |
| 102 | const void *message, absl::Span<const uint8_t> span, |
| 103 | flatbuffers::FlatBufferBuilder *fbb) { |
Austin Schuh | a4fc60f | 2020-11-01 23:06:47 -0800 | [diff] [blame] | 104 | // Enforce 8 byte alignment so anything inside the flatbuffer can be read. |
| 105 | fbb->Align(sizeof(flatbuffers::largest_scalar_t)); |
| 106 | |
| 107 | // We don't know how much of the start of the flatbuffer is padding. The |
| 108 | // safest thing to do from an alignment point of view (without looking inside) |
| 109 | // is to copy the initial offset and leave it as dead space. |
Austin Schuh | add6eb3 | 2020-11-09 21:24:26 -0800 | [diff] [blame] | 110 | fbb->PushBytes(span.data(), span.size()); |
| 111 | // Then, compute the offset from the back by computing the distance from the |
| 112 | // front to the start of the message. |
Austin Schuh | a4fc60f | 2020-11-01 23:06:47 -0800 | [diff] [blame] | 113 | return fbb->GetSize() - |
Austin Schuh | add6eb3 | 2020-11-09 21:24:26 -0800 | [diff] [blame] | 114 | static_cast<flatbuffers::uoffset_t>( |
| 115 | reinterpret_cast<const uint8_t *>(message) - span.data()); |
| 116 | } |
| 117 | |
| 118 | } // namespace flatbuffer_merge_internal |
| 119 | |
| 120 | // Copies a flatbuffer by copying all the data without looking inside and |
| 121 | // pointing inside it. |
| 122 | template <class T> |
| 123 | inline flatbuffers::Offset<T> BlindCopyFlatBuffer( |
| 124 | const NonSizePrefixedFlatbuffer<T> &t, |
| 125 | flatbuffers::FlatBufferBuilder *fbb) { |
| 126 | return flatbuffer_merge_internal::DoBlindCopyFlatBuffer(&t.message(), |
| 127 | t.span(), fbb); |
| 128 | } |
| 129 | |
| 130 | // Copies a flatbuffer by copying all the data without looking inside and |
| 131 | // pointing inside it. |
| 132 | template <class T> |
| 133 | inline flatbuffers::Offset<T> BlindCopyFlatBuffer( |
| 134 | const SizePrefixedFlatbuffer<T> &t, flatbuffers::FlatBufferBuilder *fbb) { |
| 135 | return flatbuffer_merge_internal::DoBlindCopyFlatBuffer(&t.message(), |
| 136 | t.span(), fbb); |
Austin Schuh | a4fc60f | 2020-11-01 23:06:47 -0800 | [diff] [blame] | 137 | } |
| 138 | |
Austin Schuh | e93d864 | 2019-10-13 15:27:07 -0700 | [diff] [blame] | 139 | template <class T> |
Austin Schuh | cbe9d5a | 2020-11-01 23:25:23 -0800 | [diff] [blame] | 140 | inline flatbuffers::Offset<flatbuffers::Vector<flatbuffers::Offset<T>>> |
Austin Schuh | 5c255aa | 2020-11-05 18:32:46 -0800 | [diff] [blame] | 141 | RecursiveCopyVectorTable(const flatbuffers::Vector<flatbuffers::Offset<T>> *t1, |
| 142 | flatbuffers::FlatBufferBuilder *fbb) { |
Austin Schuh | cbe9d5a | 2020-11-01 23:25:23 -0800 | [diff] [blame] | 143 | if (t1 == nullptr) { |
| 144 | return 0; |
| 145 | } |
| 146 | std::vector<flatbuffers::Offset<T>> v; |
| 147 | for (const T *t : *t1) { |
Austin Schuh | 5c255aa | 2020-11-05 18:32:46 -0800 | [diff] [blame] | 148 | v.emplace_back(RecursiveCopyFlatBuffer(t, fbb)); |
Austin Schuh | cbe9d5a | 2020-11-01 23:25:23 -0800 | [diff] [blame] | 149 | } |
| 150 | return fbb->CreateVector(v); |
| 151 | } |
| 152 | |
| 153 | inline flatbuffers::Offset< |
| 154 | flatbuffers::Vector<flatbuffers::Offset<flatbuffers::String>>> |
| 155 | CopyVectorSharedString( |
| 156 | const flatbuffers::Vector<flatbuffers::Offset<flatbuffers::String>> *t1, |
| 157 | flatbuffers::FlatBufferBuilder *fbb) { |
| 158 | if (t1 == nullptr) { |
| 159 | return 0; |
| 160 | } |
| 161 | std::vector<flatbuffers::Offset<flatbuffers::String>> v; |
| 162 | for (const flatbuffers::String *t : *t1) { |
| 163 | v.emplace_back(fbb->CreateSharedString(t)); |
| 164 | } |
| 165 | return fbb->CreateVector(v); |
| 166 | } |
| 167 | |
| 168 | template <class T> |
Austin Schuh | 40485ed | 2019-10-26 21:51:44 -0700 | [diff] [blame] | 169 | inline FlatbufferDetachedBuffer<T> CopyFlatBuffer(const T *t) { |
Austin Schuh | e93d864 | 2019-10-13 15:27:07 -0700 | [diff] [blame] | 170 | flatbuffers::FlatBufferBuilder fbb; |
Austin Schuh | d7b15da | 2020-02-17 15:06:11 -0800 | [diff] [blame] | 171 | fbb.ForceDefaults(true); |
Austin Schuh | 12417a0 | 2021-02-07 22:44:24 -0800 | [diff] [blame] | 172 | fbb.DedupVtables(false); |
Austin Schuh | e93d864 | 2019-10-13 15:27:07 -0700 | [diff] [blame] | 173 | fbb.Finish(CopyFlatBuffer<T>(t, &fbb)); |
Austin Schuh | 40485ed | 2019-10-26 21:51:44 -0700 | [diff] [blame] | 174 | return FlatbufferDetachedBuffer<T>(fbb.Release()); |
Austin Schuh | e93d864 | 2019-10-13 15:27:07 -0700 | [diff] [blame] | 175 | } |
| 176 | |
Austin Schuh | a4fc60f | 2020-11-01 23:06:47 -0800 | [diff] [blame] | 177 | template <class T> |
| 178 | inline FlatbufferDetachedBuffer<T> RecursiveCopyFlatBuffer(const T *t) { |
| 179 | flatbuffers::FlatBufferBuilder fbb; |
| 180 | fbb.ForceDefaults(true); |
Austin Schuh | 12417a0 | 2021-02-07 22:44:24 -0800 | [diff] [blame] | 181 | fbb.DedupVtables(false); |
Austin Schuh | a4fc60f | 2020-11-01 23:06:47 -0800 | [diff] [blame] | 182 | fbb.Finish(RecursiveCopyFlatBuffer<T>(t, &fbb)); |
| 183 | return FlatbufferDetachedBuffer<T>(fbb.Release()); |
| 184 | } |
| 185 | |
Austin Schuh | 30d7db9 | 2020-01-26 16:45:47 -0800 | [diff] [blame] | 186 | // Compares 2 flatbuffers. Returns true if they match, false otherwise. |
| 187 | bool CompareFlatBuffer(const flatbuffers::TypeTable *typetable, |
| 188 | const flatbuffers::Table *t1, |
| 189 | const flatbuffers::Table *t2); |
| 190 | |
| 191 | template <class T> |
| 192 | inline bool CompareFlatBuffer(const T *t1, const T *t2) { |
| 193 | return CompareFlatBuffer(T::MiniReflectTypeTable(), |
| 194 | reinterpret_cast<const flatbuffers::Table *>(t1), |
| 195 | reinterpret_cast<const flatbuffers::Table *>(t2)); |
| 196 | } |
| 197 | |
Austin Schuh | 97789fc | 2020-08-01 14:42:45 -0700 | [diff] [blame] | 198 | template <class T> |
Austin Schuh | add6eb3 | 2020-11-09 21:24:26 -0800 | [diff] [blame] | 199 | inline bool CompareFlatBuffer(const aos::NonSizePrefixedFlatbuffer<T> &t1, |
| 200 | const aos::NonSizePrefixedFlatbuffer<T> &t2) { |
| 201 | return t1.span() == t2.span(); |
| 202 | } |
| 203 | |
| 204 | template <class T> |
| 205 | inline bool CompareFlatBuffer(const aos::SizePrefixedFlatbuffer<T> &t1, |
| 206 | const aos::SizePrefixedFlatbuffer<T> &t2) { |
Austin Schuh | 97789fc | 2020-08-01 14:42:45 -0700 | [diff] [blame] | 207 | return t1.span() == t2.span(); |
| 208 | } |
| 209 | |
Austin Schuh | 09d7ffa | 2019-10-03 23:43:34 -0700 | [diff] [blame] | 210 | } // namespace aos |
| 211 | |
| 212 | #endif // AOS_FLATBUFFER_MERGE_H_ |