blob: 3a12d79d2b7842cd9661a4669de322dcbebc3b21 [file] [log] [blame]
Austin Schuhe89fa2d2019-08-14 20:24:23 -07001#include "test_builder.h"
2
Austin Schuh272c6132020-11-14 16:37:52 -08003#include "flatbuffers/stl_emulation.h"
4#include "monster_test_generated.h"
Austin Schuhe89fa2d2019-08-14 20:24:23 -07005
Austin Schuh272c6132020-11-14 16:37:52 -08006using namespace MyGame::Example;
Austin Schuhe89fa2d2019-08-14 20:24:23 -07007
8struct OwnedAllocator : public flatbuffers::DefaultAllocator {};
9
10class TestHeapBuilder : public flatbuffers::FlatBufferBuilder {
Austin Schuh272c6132020-11-14 16:37:52 -080011 private:
Austin Schuhe89fa2d2019-08-14 20:24:23 -070012 // clang-format off
13 #if !defined(FLATBUFFERS_CPP98_STL)
14 TestHeapBuilder(const TestHeapBuilder &);
15 TestHeapBuilder &operator=(const TestHeapBuilder &);
16 #endif // !defined(FLATBUFFERS_CPP98_STL)
17 // clang-format on
18
Austin Schuh272c6132020-11-14 16:37:52 -080019 public:
Austin Schuhe89fa2d2019-08-14 20:24:23 -070020 TestHeapBuilder()
Austin Schuh272c6132020-11-14 16:37:52 -080021 : flatbuffers::FlatBufferBuilder(2048, new OwnedAllocator(), true) {}
Austin Schuhe89fa2d2019-08-14 20:24:23 -070022
23 // clang-format off
24 #if !defined(FLATBUFFERS_CPP98_STL)
25 // clang-format on
26 TestHeapBuilder(TestHeapBuilder &&other)
Austin Schuh272c6132020-11-14 16:37:52 -080027 : FlatBufferBuilder(std::move(other)) {}
Austin Schuhe89fa2d2019-08-14 20:24:23 -070028
29 TestHeapBuilder &operator=(TestHeapBuilder &&other) {
30 FlatBufferBuilder::operator=(std::move(other));
31 return *this;
32 }
33 // clang-format off
34 #endif // !defined(FLATBUFFERS_CPP98_STL)
35 // clang-format on
36};
37
38// This class simulates flatbuffers::grpc::detail::SliceAllocatorMember
39struct AllocatorMember {
40 flatbuffers::DefaultAllocator member_allocator_;
41};
42
43struct GrpcLikeMessageBuilder : private AllocatorMember,
44 public flatbuffers::FlatBufferBuilder {
Austin Schuh272c6132020-11-14 16:37:52 -080045 private:
Austin Schuhe89fa2d2019-08-14 20:24:23 -070046 GrpcLikeMessageBuilder(const GrpcLikeMessageBuilder &);
47 GrpcLikeMessageBuilder &operator=(const GrpcLikeMessageBuilder &);
48
Austin Schuh272c6132020-11-14 16:37:52 -080049 public:
Austin Schuhe89fa2d2019-08-14 20:24:23 -070050 GrpcLikeMessageBuilder()
Austin Schuh272c6132020-11-14 16:37:52 -080051 : flatbuffers::FlatBufferBuilder(1024, &member_allocator_, false) {}
Austin Schuhe89fa2d2019-08-14 20:24:23 -070052
53 GrpcLikeMessageBuilder(GrpcLikeMessageBuilder &&other)
Austin Schuh272c6132020-11-14 16:37:52 -080054 : FlatBufferBuilder(1024, &member_allocator_, false) {
Austin Schuhe89fa2d2019-08-14 20:24:23 -070055 // Default construct and swap idiom.
56 Swap(other);
57 }
58
59 // clang-format off
60 #if !defined(FLATBUFFERS_CPP98_STL)
61 // clang-format on
62 GrpcLikeMessageBuilder &operator=(GrpcLikeMessageBuilder &&other) {
63 // Construct temporary and swap idiom
64 GrpcLikeMessageBuilder temp(std::move(other));
65 Swap(temp);
66 return *this;
67 }
68 // clang-format off
69 #endif // !defined(FLATBUFFERS_CPP98_STL)
70 // clang-format on
71
72 void Swap(GrpcLikeMessageBuilder &other) {
73 // No need to swap member_allocator_ because it's stateless.
74 FlatBufferBuilder::Swap(other);
Austin Schuh272c6132020-11-14 16:37:52 -080075 // After swapping the FlatBufferBuilder, we swap back the allocator, which
76 // restores the original allocator back in place. This is necessary because
77 // MessageBuilder's allocator is its own member (SliceAllocatorMember). The
78 // allocator passed to FlatBufferBuilder::vector_downward must point to this
79 // member.
Austin Schuhe89fa2d2019-08-14 20:24:23 -070080 buf_.swap_allocator(other.buf_);
81 }
82};
83
Austin Schuh272c6132020-11-14 16:37:52 -080084flatbuffers::Offset<Monster> populate1(
85 flatbuffers::FlatBufferBuilder &builder) {
86 auto name_offset = builder.CreateString(m1_name());
87 return CreateMonster(builder, nullptr, 0, 0, name_offset, 0, m1_color());
Austin Schuhe89fa2d2019-08-14 20:24:23 -070088}
89
Austin Schuh272c6132020-11-14 16:37:52 -080090flatbuffers::Offset<Monster> populate2(
91 flatbuffers::FlatBufferBuilder &builder) {
92 auto name_offset = builder.CreateString(m2_name());
93 return CreateMonster(builder, nullptr, 0, 0, name_offset, 0, m2_color());
Austin Schuhe89fa2d2019-08-14 20:24:23 -070094}
95
Austin Schuh272c6132020-11-14 16:37:52 -080096uint8_t *release_raw_base(flatbuffers::FlatBufferBuilder &fbb, size_t &size,
97 size_t &offset) {
Austin Schuhe89fa2d2019-08-14 20:24:23 -070098 return fbb.ReleaseRaw(size, offset);
99}
100
101void free_raw(flatbuffers::grpc::MessageBuilder &, uint8_t *) {
Austin Schuh272c6132020-11-14 16:37:52 -0800102 // release_raw_base calls FlatBufferBuilder::ReleaseRaw on the argument
103 // MessageBuilder. It's semantically wrong as MessageBuilder has its own
104 // ReleaseRaw member function that takes three arguments. In such cases
105 // though, ~MessageBuilder() invokes ~SliceAllocator() that takes care of
106 // deleting memory as it calls grpc_slice_unref. Obviously, this behavior is
107 // very surprising as the pointer returned by FlatBufferBuilder::ReleaseRaw is
108 // not valid as soon as MessageBuilder goes out of scope. This problem does
109 // not occur with FlatBufferBuilder.
Austin Schuhe89fa2d2019-08-14 20:24:23 -0700110}
111
112void free_raw(flatbuffers::FlatBufferBuilder &, uint8_t *buf) {
113 flatbuffers::DefaultAllocator().deallocate(buf, 0);
114}
115
Austin Schuh272c6132020-11-14 16:37:52 -0800116bool verify(const flatbuffers::DetachedBuffer &buf,
117 const std::string &expected_name, Color color) {
Austin Schuhe89fa2d2019-08-14 20:24:23 -0700118 const Monster *monster = flatbuffers::GetRoot<Monster>(buf.data());
Austin Schuh272c6132020-11-14 16:37:52 -0800119 return (monster->name()->str() == expected_name) &&
120 (monster->color() == color);
Austin Schuhe89fa2d2019-08-14 20:24:23 -0700121}
122
Austin Schuh272c6132020-11-14 16:37:52 -0800123bool verify(const uint8_t *buf, size_t offset, const std::string &expected_name,
124 Color color) {
125 const Monster *monster = flatbuffers::GetRoot<Monster>(buf + offset);
126 return (monster->name()->str() == expected_name) &&
127 (monster->color() == color);
Austin Schuhe89fa2d2019-08-14 20:24:23 -0700128}
129
Austin Schuh272c6132020-11-14 16:37:52 -0800130bool release_n_verify(flatbuffers::FlatBufferBuilder &fbb,
131 const std::string &expected_name, Color color) {
Austin Schuhe89fa2d2019-08-14 20:24:23 -0700132 flatbuffers::DetachedBuffer buf = fbb.Release();
133 return verify(buf, expected_name, color);
134}
135
136void FlatBufferBuilderTest() {
137 using flatbuffers::FlatBufferBuilder;
138
139 BuilderTests<FlatBufferBuilder>::all_tests();
140 BuilderTests<TestHeapBuilder>::all_tests();
141 BuilderTests<GrpcLikeMessageBuilder>::all_tests();
142
143 BuilderReuseTestSelector tests[4] = {
Austin Schuh272c6132020-11-14 16:37:52 -0800144 REUSABLE_AFTER_RELEASE, REUSABLE_AFTER_RELEASE_RAW,
Austin Schuhe89fa2d2019-08-14 20:24:23 -0700145 REUSABLE_AFTER_RELEASE_AND_MOVE_ASSIGN,
146 REUSABLE_AFTER_RELEASE_RAW_AND_MOVE_ASSIGN
147 };
148
Austin Schuh272c6132020-11-14 16:37:52 -0800149 BuilderReuseTests<FlatBufferBuilder, FlatBufferBuilder>::run_tests(
150 TestSelector(tests, tests + 4));
151 BuilderReuseTests<TestHeapBuilder, TestHeapBuilder>::run_tests(
152 TestSelector(tests, tests + 4));
153 BuilderReuseTests<GrpcLikeMessageBuilder, GrpcLikeMessageBuilder>::run_tests(
154 TestSelector(tests, tests + 4));
Austin Schuhe89fa2d2019-08-14 20:24:23 -0700155}
Austin Schuh272c6132020-11-14 16:37:52 -0800156
157// Link-time check using pointer type.
158void CheckTestGeneratedIsValid(const MyGame::Example::Color &) {}