X-Git-Url: https://git.draconx.ca/gitweb/upkg.git/blobdiff_plain/e404a2b5e81aa989c541023f08702e254236f98a..d78fbc3a4f4bdef475f85b0c1ebeaddd74cce141:/src/uobject/uobject.c diff --git a/src/uobject/uobject.c b/src/uobject/uobject.c index a082690..9c5bfa1 100644 --- a/src/uobject/uobject.c +++ b/src/uobject/uobject.c @@ -19,6 +19,7 @@ #include #include #include +#include #include #include #include @@ -299,6 +300,41 @@ static int decode_object_property(UObject *uo, GValue *val, unsigned long len) return 0; } +/* + * Deserialize an IEEE 754 binary32 value in "little endian" (for whatever + * that term is worth in this context). That is, when interpreted as a little + * endian 32-bit unsigned integer: bit 31 is the sign, bits 30-23 are the + * (biased) exponent, and bits 22-0 are the encoded part of the significand. + * + * The implementation is designed to be agnostic of the platform's actual + * float type, but the conversion may be lossy if "float" is not itself a + * binary32 format. NaN payloads are not preserved. + */ +static float unpack_binary32_le(const unsigned char *buf) +{ + unsigned long raw; + long significand; + int exponent; + float result; + + raw = unpack_32_le(buf); + exponent = (raw & 0x7f800000) >> 23; + significand = (raw & 0x007fffff) >> 0; + + switch (exponent) { + case 255: + result = significand ? NAN : INFINITY; + break; + default: + significand |= 0x00800000; + /* fall through */ + case 0: + result = ldexpf(significand, exponent-126-24); + } + + return copysignf(result, raw & 0x80000000 ? -1 : 1); +} + static unsigned long deserialize_property(UObject *uo, struct prop_head *head) { struct u_object_priv *priv = U_OBJECT_GET_PRIV(uo); @@ -340,6 +376,13 @@ static unsigned long deserialize_property(UObject *uo, struct prop_head *head) return 0; g_object_set_property(G_OBJECT(uo), head->prop_name, &val); break; + case PROPERTY_FLOAT: + if (head->size != 4 || priv->nbuf-len < head->size) + return 0; + g_value_init(&val, G_TYPE_FLOAT); + g_value_set_float(&val, unpack_binary32_le(priv->buf+len)); + g_object_set_property(G_OBJECT(uo), head->prop_name, &val); + break; default: u_warn(uo, "%s: unsupported property type %u", head->prop_name, (unsigned)head->type); @@ -481,13 +524,17 @@ GObject *u_object_new_from_package(GTypeModule *pkg, unsigned long idx) { g_return_val_if_fail(IS_U_PKG(pkg), NULL); - const char *class, *package; + const struct upkg_export *export; GObject *obj = NULL; GType type; - class = upkg_export_class(U_PKG(pkg)->pkg, idx, &package); + export = upkg_get_export(U_PKG(pkg)->pkg, idx); + if (!export) { + u_err(pkg, "invalid package export: %lu", idx); + return NULL; + } - type = u_object_module_get_class(package, class); + type = u_object_module_get_class(pkg, export->class); if (type) { obj = g_object_new(type, NULL); if (u_object_deserialize(obj, pkg, idx) != 0) { @@ -567,18 +614,20 @@ static char *prepend_fmt(const char *prefix, const char *fmt) /* Logging helpers that automatically prepend the UObject class information. */ void u_vlog_full(GObject *o, GLogLevelFlags level, const char *fmt, va_list ap) { - g_return_if_fail(IS_U_OBJECT(o)); - UObject *uo = U_OBJECT(o); - char *new_fmt = NULL; - - if (uo->pkg_name) { - new_fmt = prepend_fmt(uo->pkg_name, fmt); - if (!new_fmt) { - g_log(G_OBJECT_TYPE_NAME(o), level, "%s", - uo->pkg_file->name); - } else { + char *new_fmt = NULL, *obj_prefix = NULL; + + if (IS_U_OBJECT(o)) { + obj_prefix = U_OBJECT(o)->pkg_name; + } else if (G_IS_TYPE_MODULE(o)) { + obj_prefix = G_TYPE_MODULE(o)->name; + } + + if (obj_prefix) { + new_fmt = prepend_fmt(obj_prefix, fmt); + if (!new_fmt) + g_log(G_OBJECT_TYPE_NAME(o), level, "%s", obj_prefix); + else fmt = new_fmt; - } } g_logv(G_OBJECT_TYPE_NAME(o), level, fmt, ap);