diff --git a/src/hb-null.hh b/src/hb-null.hh index 51f653d2a3650a28674c1934a4180f5a9f9fd1e3..92c7a351922950c33e931db914e36fdbfbb2ab3e 100644 --- a/src/hb-null.hh +++ b/src/hb-null.hh @@ -38,13 +38,26 @@ #define HB_NULL_POOL_SIZE 1024 +/* Use SFINAE to sniff whether T has min_size; in which case return T::null_size, + * otherwise return sizeof(T). */ +template +struct _hb_null_size +{ enum { value = sizeof (T) }; }; +template +struct _hb_null_size +{ enum { value = T::null_size }; }; + +template +struct hb_null_size +{ enum { value = _hb_null_size::value }; }; + extern HB_INTERNAL hb_vector_size_impl_t const _hb_NullPool[(HB_NULL_POOL_SIZE + sizeof (hb_vector_size_impl_t) - 1) / sizeof (hb_vector_size_impl_t)]; /* Generic nul-content Null objects. */ template static inline Type const & Null (void) { - static_assert (sizeof (Type) <= HB_NULL_POOL_SIZE, "Increase HB_NULL_POOL_SIZE."); + static_assert (hb_null_size::value <= HB_NULL_POOL_SIZE, "Increase HB_NULL_POOL_SIZE."); return *reinterpret_cast (_hb_NullPool); } #define Null(Type) Null::value>::value>() @@ -85,7 +98,7 @@ extern HB_INTERNAL /* CRAP pool: Common Region for Access Protection. */ template static inline Type& Crap (void) { - static_assert (sizeof (Type) <= HB_NULL_POOL_SIZE, "Increase HB_NULL_POOL_SIZE."); + static_assert (hb_null_size::value <= HB_NULL_POOL_SIZE, "Increase HB_NULL_POOL_SIZE."); Type *obj = reinterpret_cast (_hb_CrapPool); memcpy (obj, &Null(Type), sizeof (*obj)); return *obj;