aboutsummaryrefslogtreecommitdiff
path: root/flatcc/test/flatc_compat
diff options
context:
space:
mode:
Diffstat (limited to 'flatcc/test/flatc_compat')
-rw-r--r--flatcc/test/flatc_compat/.gitattributes2
-rw-r--r--flatcc/test/flatc_compat/CMakeLists.txt21
-rw-r--r--flatcc/test/flatc_compat/README.md10
-rw-r--r--flatcc/test/flatc_compat/flatc_compat.c226
-rwxr-xr-xflatcc/test/flatc_compat/flatc_compat.sh20
-rw-r--r--flatcc/test/flatc_compat/monsterdata_test.golden48
-rwxr-xr-xflatcc/test/flatc_compat/monsterdata_test.json51
-rw-r--r--flatcc/test/flatc_compat/monsterdata_test.monbin0 -> 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
new file mode 100644
index 0000000..08646d4
--- /dev/null
+++ b/flatcc/test/flatc_compat/monsterdata_test.mon
Binary files differ