blob: 047a380c06e1a5fb533922f8fe32085ebb36f838 [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 TestHeapBuilder(const TestHeapBuilder &);
13 TestHeapBuilder &operator=(const TestHeapBuilder &);
Austin Schuhe89fa2d2019-08-14 20:24:23 -070014
Austin Schuh272c6132020-11-14 16:37:52 -080015 public:
Austin Schuhe89fa2d2019-08-14 20:24:23 -070016 TestHeapBuilder()
Austin Schuh272c6132020-11-14 16:37:52 -080017 : flatbuffers::FlatBufferBuilder(2048, new OwnedAllocator(), true) {}
Austin Schuhe89fa2d2019-08-14 20:24:23 -070018
Austin Schuhe89fa2d2019-08-14 20:24:23 -070019 TestHeapBuilder(TestHeapBuilder &&other)
Austin Schuh272c6132020-11-14 16:37:52 -080020 : FlatBufferBuilder(std::move(other)) {}
Austin Schuhe89fa2d2019-08-14 20:24:23 -070021
22 TestHeapBuilder &operator=(TestHeapBuilder &&other) {
23 FlatBufferBuilder::operator=(std::move(other));
24 return *this;
25 }
Austin Schuhe89fa2d2019-08-14 20:24:23 -070026};
27
28// This class simulates flatbuffers::grpc::detail::SliceAllocatorMember
29struct AllocatorMember {
30 flatbuffers::DefaultAllocator member_allocator_;
31};
32
33struct GrpcLikeMessageBuilder : private AllocatorMember,
34 public flatbuffers::FlatBufferBuilder {
Austin Schuh272c6132020-11-14 16:37:52 -080035 private:
Austin Schuhe89fa2d2019-08-14 20:24:23 -070036 GrpcLikeMessageBuilder(const GrpcLikeMessageBuilder &);
37 GrpcLikeMessageBuilder &operator=(const GrpcLikeMessageBuilder &);
38
Austin Schuh272c6132020-11-14 16:37:52 -080039 public:
Austin Schuhe89fa2d2019-08-14 20:24:23 -070040 GrpcLikeMessageBuilder()
Austin Schuh272c6132020-11-14 16:37:52 -080041 : flatbuffers::FlatBufferBuilder(1024, &member_allocator_, false) {}
Austin Schuhe89fa2d2019-08-14 20:24:23 -070042
43 GrpcLikeMessageBuilder(GrpcLikeMessageBuilder &&other)
Austin Schuh272c6132020-11-14 16:37:52 -080044 : FlatBufferBuilder(1024, &member_allocator_, false) {
Austin Schuhe89fa2d2019-08-14 20:24:23 -070045 // Default construct and swap idiom.
46 Swap(other);
47 }
48
Austin Schuhe89fa2d2019-08-14 20:24:23 -070049 GrpcLikeMessageBuilder &operator=(GrpcLikeMessageBuilder &&other) {
50 // Construct temporary and swap idiom
51 GrpcLikeMessageBuilder temp(std::move(other));
52 Swap(temp);
53 return *this;
54 }
Austin Schuhe89fa2d2019-08-14 20:24:23 -070055
56 void Swap(GrpcLikeMessageBuilder &other) {
57 // No need to swap member_allocator_ because it's stateless.
58 FlatBufferBuilder::Swap(other);
Austin Schuh272c6132020-11-14 16:37:52 -080059 // After swapping the FlatBufferBuilder, we swap back the allocator, which
60 // restores the original allocator back in place. This is necessary because
61 // MessageBuilder's allocator is its own member (SliceAllocatorMember). The
62 // allocator passed to FlatBufferBuilder::vector_downward must point to this
63 // member.
Austin Schuhe89fa2d2019-08-14 20:24:23 -070064 buf_.swap_allocator(other.buf_);
65 }
66};
67
Austin Schuh272c6132020-11-14 16:37:52 -080068flatbuffers::Offset<Monster> populate1(
69 flatbuffers::FlatBufferBuilder &builder) {
70 auto name_offset = builder.CreateString(m1_name());
71 return CreateMonster(builder, nullptr, 0, 0, name_offset, 0, m1_color());
Austin Schuhe89fa2d2019-08-14 20:24:23 -070072}
73
Austin Schuh272c6132020-11-14 16:37:52 -080074flatbuffers::Offset<Monster> populate2(
75 flatbuffers::FlatBufferBuilder &builder) {
76 auto name_offset = builder.CreateString(m2_name());
77 return CreateMonster(builder, nullptr, 0, 0, name_offset, 0, m2_color());
Austin Schuhe89fa2d2019-08-14 20:24:23 -070078}
79
Austin Schuh272c6132020-11-14 16:37:52 -080080uint8_t *release_raw_base(flatbuffers::FlatBufferBuilder &fbb, size_t &size,
81 size_t &offset) {
Austin Schuhe89fa2d2019-08-14 20:24:23 -070082 return fbb.ReleaseRaw(size, offset);
83}
84
85void free_raw(flatbuffers::grpc::MessageBuilder &, uint8_t *) {
Austin Schuh272c6132020-11-14 16:37:52 -080086 // release_raw_base calls FlatBufferBuilder::ReleaseRaw on the argument
87 // MessageBuilder. It's semantically wrong as MessageBuilder has its own
88 // ReleaseRaw member function that takes three arguments. In such cases
89 // though, ~MessageBuilder() invokes ~SliceAllocator() that takes care of
90 // deleting memory as it calls grpc_slice_unref. Obviously, this behavior is
91 // very surprising as the pointer returned by FlatBufferBuilder::ReleaseRaw is
92 // not valid as soon as MessageBuilder goes out of scope. This problem does
93 // not occur with FlatBufferBuilder.
Austin Schuhe89fa2d2019-08-14 20:24:23 -070094}
95
96void free_raw(flatbuffers::FlatBufferBuilder &, uint8_t *buf) {
97 flatbuffers::DefaultAllocator().deallocate(buf, 0);
98}
99
Austin Schuh272c6132020-11-14 16:37:52 -0800100bool verify(const flatbuffers::DetachedBuffer &buf,
101 const std::string &expected_name, Color color) {
Austin Schuhe89fa2d2019-08-14 20:24:23 -0700102 const Monster *monster = flatbuffers::GetRoot<Monster>(buf.data());
Austin Schuh272c6132020-11-14 16:37:52 -0800103 return (monster->name()->str() == expected_name) &&
104 (monster->color() == color);
Austin Schuhe89fa2d2019-08-14 20:24:23 -0700105}
106
Austin Schuh272c6132020-11-14 16:37:52 -0800107bool verify(const uint8_t *buf, size_t offset, const std::string &expected_name,
108 Color color) {
109 const Monster *monster = flatbuffers::GetRoot<Monster>(buf + offset);
110 return (monster->name()->str() == expected_name) &&
111 (monster->color() == color);
Austin Schuhe89fa2d2019-08-14 20:24:23 -0700112}
113
Austin Schuh272c6132020-11-14 16:37:52 -0800114bool release_n_verify(flatbuffers::FlatBufferBuilder &fbb,
115 const std::string &expected_name, Color color) {
Austin Schuhe89fa2d2019-08-14 20:24:23 -0700116 flatbuffers::DetachedBuffer buf = fbb.Release();
117 return verify(buf, expected_name, color);
118}
119
Austin Schuh2dd86a92022-09-14 21:19:23 -0700120// forward-declared in test.cpp
121void FlatBufferBuilderTest();
122
Austin Schuhe89fa2d2019-08-14 20:24:23 -0700123void FlatBufferBuilderTest() {
124 using flatbuffers::FlatBufferBuilder;
125
126 BuilderTests<FlatBufferBuilder>::all_tests();
127 BuilderTests<TestHeapBuilder>::all_tests();
128 BuilderTests<GrpcLikeMessageBuilder>::all_tests();
129
130 BuilderReuseTestSelector tests[4] = {
Austin Schuh272c6132020-11-14 16:37:52 -0800131 REUSABLE_AFTER_RELEASE, REUSABLE_AFTER_RELEASE_RAW,
Austin Schuhe89fa2d2019-08-14 20:24:23 -0700132 REUSABLE_AFTER_RELEASE_AND_MOVE_ASSIGN,
133 REUSABLE_AFTER_RELEASE_RAW_AND_MOVE_ASSIGN
134 };
135
Austin Schuh272c6132020-11-14 16:37:52 -0800136 BuilderReuseTests<FlatBufferBuilder, FlatBufferBuilder>::run_tests(
137 TestSelector(tests, tests + 4));
138 BuilderReuseTests<TestHeapBuilder, TestHeapBuilder>::run_tests(
139 TestSelector(tests, tests + 4));
140 BuilderReuseTests<GrpcLikeMessageBuilder, GrpcLikeMessageBuilder>::run_tests(
141 TestSelector(tests, tests + 4));
Austin Schuhe89fa2d2019-08-14 20:24:23 -0700142}
Austin Schuh272c6132020-11-14 16:37:52 -0800143
Austin Schuh2dd86a92022-09-14 21:19:23 -0700144// forward-declared in test_builder.h
145void CheckTestGeneratedIsValid(const MyGame::Example::Color&);
146
Austin Schuh272c6132020-11-14 16:37:52 -0800147// Link-time check using pointer type.
James Kuszmaul8e62b022022-03-22 09:33:25 -0700148void CheckTestGeneratedIsValid(const MyGame::Example::Color &) {}