diff options
Diffstat (limited to 'flatcc/test/flatc_compat')
-rw-r--r-- | flatcc/test/flatc_compat/.gitattributes | 2 | ||||
-rw-r--r-- | flatcc/test/flatc_compat/CMakeLists.txt | 21 | ||||
-rw-r--r-- | flatcc/test/flatc_compat/README.md | 10 | ||||
-rw-r--r-- | flatcc/test/flatc_compat/flatc_compat.c | 226 | ||||
-rwxr-xr-x | flatcc/test/flatc_compat/flatc_compat.sh | 20 | ||||
-rw-r--r-- | flatcc/test/flatc_compat/monsterdata_test.golden | 48 | ||||
-rwxr-xr-x | flatcc/test/flatc_compat/monsterdata_test.json | 51 | ||||
-rw-r--r-- | flatcc/test/flatc_compat/monsterdata_test.mon | bin | 0 -> 336 bytes |
8 files changed, 378 insertions, 0 deletions
diff --git a/flatcc/test/flatc_compat/.gitattributes b/flatcc/test/flatc_compat/.gitattributes new file mode 100644 index 0000000..cf2b141 --- /dev/null +++ b/flatcc/test/flatc_compat/.gitattributes @@ -0,0 +1,2 @@ +# We do a binary comparison test, so we need it to be unchanged on Windows. +monsterdata_test.golden -text diff --git a/flatcc/test/flatc_compat/CMakeLists.txt b/flatcc/test/flatc_compat/CMakeLists.txt new file mode 100644 index 0000000..a472979 --- /dev/null +++ b/flatcc/test/flatc_compat/CMakeLists.txt @@ -0,0 +1,21 @@ +include(CTest) + +set(INC_DIR "${PROJECT_SOURCE_DIR}/include") +set(GEN_DIR "${CMAKE_CURRENT_BINARY_DIR}/generated") +set(FBS_DIR "${PROJECT_SOURCE_DIR}/test/monster_test") + +include_directories("${GEN_DIR}" "${INC_DIR}") + +add_custom_target(gen_flatc_compat ALL) +add_custom_command ( + TARGET gen_flatc_compat + COMMAND ${CMAKE_COMMAND} -E make_directory "${GEN_DIR}" + COMMAND ${CMAKE_COMMAND} -E copy "${CMAKE_CURRENT_SOURCE_DIR}/monsterdata_test.mon" ${CMAKE_CURRENT_BINARY_DIR} + COMMAND flatcc_cli -a -o "${GEN_DIR}" "${FBS_DIR}/monster_test.fbs" + DEPENDS flatcc_cli "${FBS_DIR}/monster_test.fbs" "${FBS_DIR}/include_test1.fbs" "${FBS_DIR}/include_test2.fbs" +) +add_executable(flatc_compat flatc_compat.c) +add_dependencies(flatc_compat gen_flatc_compat) +target_link_libraries(flatc_compat flatccrt) + +add_test(flatc_compat flatc_compat${CMAKE_EXECUTABLE_SUFFIX}) diff --git a/flatcc/test/flatc_compat/README.md b/flatcc/test/flatc_compat/README.md new file mode 100644 index 0000000..ffc0bf0 --- /dev/null +++ b/flatcc/test/flatc_compat/README.md @@ -0,0 +1,10 @@ +Basic sanity check to verify that a `flatcc` generated reader can read +binaries generated by Googles `flatc` compiler. + +`monsterdata_test.mon` is generated by Googles flatc compiler using the +`monsterdata_test.golden` json as input and `monster_test.fbs` as schema. + +`monsterdata_test.json` was also copied for completeness - it apparently +relates to undocumented hash attributed that converts json strings into +hashes where the golden file has the correct target type with respect to +the schema.. diff --git a/flatcc/test/flatc_compat/flatc_compat.c b/flatcc/test/flatc_compat/flatc_compat.c new file mode 100644 index 0000000..24aae19 --- /dev/null +++ b/flatcc/test/flatc_compat/flatc_compat.c @@ -0,0 +1,226 @@ +#include <stdlib.h> +#include <assert.h> + +#include "monster_test_reader.h" +#include "monster_test_verifier.h" +#include "flatcc/support/readfile.h" +#include "flatcc/support/hexdump.h" + +#define align_up(alignment, size) \ + (((size_t)(size) + (size_t)(alignment) - 1) & ~((size_t)(alignment) - 1)) + +const char *filename = "monsterdata_test.mon"; + +#undef ns +#define ns(x) MyGame_Example_ ## x + +int verify_monster(void *buffer) +{ + ns(Monster_table_t) monster, mini; + ns(Vec3_struct_t) vec; + ns(Test_struct_t) test; + ns(Test_vec_t) testvec; + ns(Any_union_type_t) mini_type; + flatbuffers_string_t name; + size_t offset; + flatbuffers_uint8_vec_t inv; + flatbuffers_string_vec_t aofs; + flatbuffers_string_t s; + size_t i; + + if (!(monster = ns(Monster_as_root(buffer)))) { + printf("Monster not available\n"); + return -1; + } + if (ns(Monster_hp(monster)) != 80) { + printf("Health points are not as expected\n"); + return -1; + } + if (!(vec = ns(Monster_pos(monster)))) { + printf("Position is absent\n"); + return -1; + } + offset = (size_t)((char *)vec - (char *)buffer); + if (offset & 15) { + printf("Force align of Vec3 struct not correct\n"); + return -1; + } + if (ns(Vec3_x(vec)) != 1) { + printf("Position failing on x coordinate\n"); + return -1; + } + if (ns(Vec3_y(vec)) != 2) { + printf("Position failing on y coordinate\n"); + return -1; + } + if (ns(Vec3_z(vec)) != 3) { + printf("Position failing on z coordinate\n"); + return -1; + } + if (ns(Vec3_test1(vec)) != 3) { + printf("Vec3 test1 is not 3\n"); + return -1; + } + if (ns(Vec3_test2(vec)) != ns(Color_Green)) { + printf("Vec3 test2 not Green\n"); + return -1; + } + test = ns(Vec3_test3(vec)); + if (ns(Test_a(test)) != 5 || ns(Test_b(test) != 6)) { + printf("test3 not valid in Vec3\n"); + return -1; + } + name = ns(Monster_name(monster)); + if (flatbuffers_string_len(name) != 9) { + printf("Name length is not correct\n"); + return -1; + } + if (!name || strcmp(name, "MyMonster")) { + printf("Name is not correct\n"); + return -1; + } + inv = ns(Monster_inventory(monster)); + if (flatbuffers_uint8_vec_len(inv) != 5) { + printf("Inventory has wrong length\n"); + return -1; + } + for (i = 0; i < 5; ++i) { + if (flatbuffers_uint8_vec_at(inv, i) != i) { + printf("Inventory item #%d is wrong\n", (int)i); + return -1; + } + } + if (!(aofs = ns(Monster_testarrayofstring(monster)))) { + printf("Array of string not present\n"); + return -1; + } + if (flatbuffers_string_vec_len(aofs) != 2) { + printf("Array of string has wrong vector length\n"); + return -1; + } + s = flatbuffers_string_vec_at(aofs, 0); + if (strcmp(s, "test1")) { + printf("First string array element is wrong\n"); + return -1; + } + s = flatbuffers_string_vec_at(aofs, 1); + if (strcmp(s, "test2")) { + printf("Second string array element is wrong\n"); + return -1; + } + mini_type = ns(Monster_test_type(monster)); + if (mini_type != ns(Any_Monster)) { + printf("Not any monster\n"); + return -1; + } + mini = ns(Monster_test(monster)); + if (!mini) { + printf("test monster not there\n"); + return -1; + } + if (strcmp(ns(Monster_name(mini)), "Fred")) { + printf("test monster isn't Fred\n"); + return -1; + } + testvec = ns(Monster_test4(monster)); + if (ns(Test_vec_len(testvec)) != 2) { + printf("Test struct vector has wrong length\n"); + return -1; + } + test = ns(Test_vec_at(testvec, 0)); + if (ns(Test_a(test) != 10)) { + printf("Testvec[0].a is wrong\n"); + return -1; + } + if (ns(Test_b(test) != 20)) { + printf("Testvec[0].b is wrong\n"); + return -1; + } + test = ns(Test_vec_at(testvec, 1)); + if (ns(Test_a(test) != 30)) { + printf("Testvec[1].a is wrong\n"); + return -1; + } + if (ns(Test_b(test) != 40)) { + printf("Testvec[1].b is wrong\n"); + return -1; + } + assert(ns(Monster_testhashs32_fnv1(monster)) == -579221183L); + assert(ns(Monster_testhashu32_fnv1(monster)) == 3715746113L); + assert(ns(Monster_testhashs64_fnv1(monster)) == 7930699090847568257LL); + assert(ns(Monster_testhashu64_fnv1(monster)) == 7930699090847568257LL); + assert(ns(Monster_testhashs32_fnv1a(monster)) == -1904106383L); + assert(ns(Monster_testhashu32_fnv1a(monster)) == 2390860913L); + assert(ns(Monster_testhashs64_fnv1a(monster)) == 4898026182817603057LL); + assert(ns(Monster_testhashu64_fnv1a(monster)) == 4898026182817603057LL); + return 0; +} + + +/* We take arguments so test can run without copying sources. */ +#define usage \ +"wrong number of arguments:\n" \ +"usage: <program> [<input-filename>]\n" + +int main(int argc, char *argv[]) +{ + int ret; + size_t size; + void *buffer, *raw_buffer; + + if (argc != 1 && argc != 2) { + fprintf(stderr, usage); + exit(1); + } + if (argc == 2) { + filename = argv[1]; + } + + raw_buffer = readfile(filename, 1024, &size); + buffer = aligned_alloc(256, align_up(256, size)); + memcpy(buffer, raw_buffer, size); + free(raw_buffer); + + if (!buffer) { + fprintf(stderr, "could not read binary test file: %s\n", filename); + return -1; + } + hexdump("monsterdata_test.mon", buffer, size, stderr); + /* + * Not automated, but verifying size - 3 fails as expected because the last + * object in the file is a string, and the zero termination fails. + * size - 1 and size - 2 verifies because the buffers contains + * padding. Note that `flatcc` does not pad at the end beyond whatever + * is stored (normally a vtable), but this is generated with `flatc + * v1.1`. + */ + if (flatcc_verify_ok != ns(Monster_verify_as_root_with_identifier(buffer, size, "MONS"))) { +#if FLATBUFFERS_PROTOCOL_IS_BE + fprintf(stderr, "flatc golden reference buffer was correctly rejected by flatcc verificiation\n" + "because flatc is little endian and flatcc has been compiled for big endian protocol format\n"); + ret = 0; + goto done; +#else + fprintf(stderr, "could not verify foreign monster file\n"); + ret = -1; + goto done; +#endif + } + +#if FLATBUFFERS_PROTOCOL_IS_BE + fprintf(stderr, "flatcc compiled with big endian protocol failed to reject reference little endian buffer\n"); + ret = -1; + goto done; +#else + if (flatcc_verify_ok != ns(Monster_verify_as_root(buffer, size))) { + fprintf(stderr, "could not verify foreign monster file with default identifier\n"); + ret = -1; + goto done; + } + ret = verify_monster(buffer); +#endif + +done: + aligned_free(buffer); + return ret; +} diff --git a/flatcc/test/flatc_compat/flatc_compat.sh b/flatcc/test/flatc_compat/flatc_compat.sh new file mode 100755 index 0000000..1b2dbb9 --- /dev/null +++ b/flatcc/test/flatc_compat/flatc_compat.sh @@ -0,0 +1,20 @@ +#!/usr/bin/env bash + +set -e +cd `dirname $0`/../.. +ROOT=`pwd` +TMP=build/tmp/test + +${ROOT}/scripts/build.sh + +mkdir -p ${TMP}/flatc_compat +rm -rf ${TMP}/flatc_compat/* +bin/flatcc -a -o ${TMP}/flatc_compat test/monster_test/monster_test.fbs + +cp test/flatc_compat/*.{json,mon,c} ${TMP}/flatc_compat/ +cd ${TMP}/flatc_compat +cc -g -I ${ROOT}/include flatc_compat.c \ + ${ROOT}/lib/libflatccrt.a -o flatc_compat_d +echo "Google FPL flatc compatibility test - reading flatc generated binary" +./flatc_compat_d + diff --git a/flatcc/test/flatc_compat/monsterdata_test.golden b/flatcc/test/flatc_compat/monsterdata_test.golden new file mode 100644 index 0000000..73afc42 --- /dev/null +++ b/flatcc/test/flatc_compat/monsterdata_test.golden @@ -0,0 +1,48 @@ +{ + pos: { + x: 1, + y: 2, + z: 3, + test1: 3, + test2: Green, + test3: { + a: 5, + b: 6 + } + }, + hp: 80, + name: "MyMonster", + inventory: [ + 0, + 1, + 2, + 3, + 4 + ], + test_type: Monster, + test: { + name: "Fred" + }, + test4: [ + { + a: 10, + b: 20 + }, + { + a: 30, + b: 40 + } + ], + testarrayofstring: [ + "test1", + "test2" + ], + testhashs32_fnv1: -579221183, + testhashu32_fnv1: 3715746113, + testhashs64_fnv1: 7930699090847568257, + testhashu64_fnv1: 7930699090847568257, + testhashs32_fnv1a: -1904106383, + testhashu32_fnv1a: 2390860913, + testhashs64_fnv1a: 4898026182817603057, + testhashu64_fnv1a: 4898026182817603057 +} diff --git a/flatcc/test/flatc_compat/monsterdata_test.json b/flatcc/test/flatc_compat/monsterdata_test.json new file mode 100755 index 0000000..a718efa --- /dev/null +++ b/flatcc/test/flatc_compat/monsterdata_test.json @@ -0,0 +1,51 @@ +{ + pos: { + x: 1, + y: 2, + z: 3, + test1: 3, + test2: Green, + test3: { + a: 5, + b: 6 + } + }, + hp: 80, + name: "MyMonster", + inventory: [ + 0, + 1, + 2, + 3, + 4 + ], + test_type: Monster, + test: { + name: "Fred" + }, + test4: [ + { + a: 10, + b: 20 + }, + { + a: 30, + b: 40 + } + ], + testarrayofstring: [ + "test1", + "test2" + ], + testarrayofbools:[ + true, false, true + ], + testhashs32_fnv1: "This string is being hashed!", + testhashu32_fnv1: "This string is being hashed!", + testhashs64_fnv1: "This string is being hashed!", + testhashu64_fnv1: "This string is being hashed!", + testhashs32_fnv1a: "This string is being hashed!", + testhashu32_fnv1a: "This string is being hashed!", + testhashs64_fnv1a: "This string is being hashed!", + testhashu64_fnv1a: "This string is being hashed!", +} diff --git a/flatcc/test/flatc_compat/monsterdata_test.mon b/flatcc/test/flatc_compat/monsterdata_test.mon Binary files differnew file mode 100644 index 0000000..08646d4 --- /dev/null +++ b/flatcc/test/flatc_compat/monsterdata_test.mon |