1
0
mirror of git://jb55.com/damus synced 2024-09-16 02:03:45 +00:00
damus/nostrdb/flatcc/flatcc_identifier.h
2023-08-25 19:05:34 -07:00

149 lines
4.7 KiB
C

#ifndef FLATCC_IDENTIFIER_H
#define FLATCC_IDENTIFIER_H
#ifdef __cplusplus
extern "C" {
#endif
#ifndef FLATCC_FLATBUFFERS_H
#error "include via flatcc/flatcc_flatbuffers.h"
#endif
#ifndef UINT8_MAX
#include <stdint.h>
#endif
/*
* FlatBuffers identifiers are normally specified by "file_identifer" in
* the schema, but a standard hash of the fully qualified type name can
* also be used. This file implements such a mapping, but the generated
* headers also contain the necessary information for known types.
*/
/*
* Returns the type hash of a given name in native endian format.
* Generated code already provides these, but if a name was changed
* in the schema it may be relevant to recompute the hash manually.
*
* The wire-format of this value should always be little endian.
*
* Note: this must be the fully qualified name, e.g. in the namespace
* "MyGame.Example":
*
* flatbuffers_type_hash_from_name("MyGame.Example.Monster");
*
* or, in the global namespace just:
*
* flatbuffers_type_hash_from_name("MyTable");
*
* This assumes 32 bit hash type. For other sizes, other FNV-1a
* constants would be required.
*
* Note that we reserve hash value 0 for missing or ignored value.
*/
static inline flatbuffers_thash_t flatbuffers_type_hash_from_name(const char *name)
{
uint32_t hash = UINT32_C(2166136261);
while (*name) {
hash ^= (unsigned char)*name;
hash = hash * UINT32_C(16777619);
++name;
}
if (hash == 0) {
hash = UINT32_C(2166136261);
}
return hash;
}
/*
* Type hash encoded as little endian file identifier string.
* Note: if type hash is 0, the identifier should be null which
* we cannot return in this interface.
*/
static inline void flatbuffers_identifier_from_type_hash(flatbuffers_thash_t type_hash, flatbuffers_fid_t out_identifier)
{
out_identifier[0] = (char)(type_hash & 0xff);
type_hash >>= 8;
out_identifier[1] = (char)(type_hash & 0xff);
type_hash >>= 8;
out_identifier[2] = (char)(type_hash & 0xff);
type_hash >>= 8;
out_identifier[3] = (char)(type_hash & 0xff);
}
/* Native integer encoding of file identifier. */
static inline flatbuffers_thash_t flatbuffers_type_hash_from_identifier(const flatbuffers_fid_t identifier)
{
uint8_t *p = (uint8_t *)identifier;
return identifier ?
(uint32_t)p[0] + (((uint32_t)p[1]) << 8) + (((uint32_t)p[2]) << 16) + (((uint32_t)p[3]) << 24) : 0;
}
/*
* Convert a null terminated string identifier like "MONS" or "X" into a
* native type hash identifier, usually for comparison. This will not
* work with type hash strings because they can contain null bytes.
*/
static inline flatbuffers_thash_t flatbuffers_type_hash_from_string(const char *identifier)
{
flatbuffers_thash_t h = 0;
const uint8_t *p = (const uint8_t *)identifier;
if (!p[0]) return h;
h += ((flatbuffers_thash_t)p[0]);
if (!p[1]) return h;
h += ((flatbuffers_thash_t)p[1]) << 8;
if (!p[2]) return h;
h += ((flatbuffers_thash_t)p[2]) << 16;
/* No need to test for termination here. */
h += ((flatbuffers_thash_t)p[3]) << 24;
return h;
}
/*
* Computes the little endian wire format of the type hash. It can be
* used as a file identifer argument to various flatcc buffer calls.
*
* `flatbuffers_fid_t` is just `char [4]` for the default flatbuffers
* type system defined in `flatcc/flatcc_types.h`.
*/
static inline void flatbuffers_identifier_from_name(const char *name, flatbuffers_fid_t out_identifier)
{
flatbuffers_identifier_from_type_hash(flatbuffers_type_hash_from_name(name), out_identifier);
}
/*
* This is a collision free hash (a permutation) of the type hash to
* provide better distribution for use in hash tables. It is likely not
* necessary in praxis, and for uniqueness of identifiers it provides no
* advantage over just using the FNV-1a type hash, except when truncating
* the identifier to less than 32-bits.
*
* Note: the output should not be used in transmission. It provides no
* additional information and just complicates matters. Furthermore, the
* unmodified type hash has the benefit that it can seed a child namespace.
*/
static inline uint32_t flatbuffers_disperse_type_hash(flatbuffers_thash_t type_hash)
{
/* http://stackoverflow.com/a/12996028 */
uint32_t x = type_hash;
x = ((x >> 16) ^ x) * UINT32_C(0x45d9f3b);
x = ((x >> 16) ^ x) * UINT32_C(0x45d9f3b);
x = ((x >> 16) ^ x);
return x;
}
/* We have hardcoded assumptions about identifier size. */
static_assert(sizeof(flatbuffers_fid_t) == 4, "unexpected file identifier size");
static_assert(sizeof(flatbuffers_thash_t) == 4, "unexpected type hash size");
#ifdef __cplusplus
}
#endif
#endif /* FLATCC_IDENTIFIER_H */