]> git.draconx.ca Git - upkg.git/commitdiff
uobject: Add support for float properties.
authorNick Bowler <nbowler@draconx.ca>
Sun, 20 May 2012 03:05:34 +0000 (23:05 -0400)
committerNick Bowler <nbowler@draconx.ca>
Sun, 20 May 2012 03:05:34 +0000 (23:05 -0400)
Since it's not too hard to add these, do so.  Add the DrawScale property
to Engine.Texture to demonstrate it.

m4/.gitignore
m4/gnulib-cache.m4
src/engine/texture.gob
src/uobject/uobject.c

index f6d1341fedda3e999db5f0335c1c4b79f0ab710c..d191decb58176926f892bce6e32eab5554d712b3 100644 (file)
@@ -1,19 +1,33 @@
 00gnulib.m4
 argz.m4
+copysignf.m4
+exponentd.m4
+exponentf.m4
+exponentl.m4
 extensions.m4
+float_h.m4
+fpieee.m4
 getopt.m4
 gnulib-common.m4
 gnulib-comp.m4
 gnulib-tool.m4
 include_next.m4
+isnand.m4
+isnanf.m4
+isnanl.m4
+ldexp.m4
+ldexpf.m4
 libtool.m4
 ltdl.m4
 ltoptions.m4
 ltsugar.m4
 ltversion.m4
 lt~obsolete.m4
+math_h.m4
+mathfunc.m4
 nocrash.m4
 off_t.m4
+signbit.m4
 ssize_t.m4
 stddef_h.m4
 strcase.m4
index 3b9685ccf6cec26296137afefa6bb31b95cfe546..da9334c623f32e7e047bd6eaa9e57e6053d83b7a 100644 (file)
 
 
 # Specification in the form of a command-line invocation:
-#   gnulib-tool --import --dir=. --lib=libgnu --source-base=lib --m4-base=m4 --doc-base=doc --tests-base=tests --aux-dir=. --makefile-name=gnulib.mk.in --conditional-dependencies --libtool --macro-prefix=gl --no-vc-files getopt-gnu strcase
+#   gnulib-tool --import --dir=. --lib=libgnu --source-base=lib --m4-base=m4 --doc-base=doc --tests-base=tests --aux-dir=. --makefile-name=gnulib.mk.in --conditional-dependencies --libtool --macro-prefix=gl --no-vc-files copysignf getopt-gnu ldexpf strcase
 
 # Specification in the form of a few gnulib-tool.m4 macro invocations:
 gl_LOCAL_DIR([])
 gl_MODULES([
+  copysignf
   getopt-gnu
+  ldexpf
   strcase
 ])
 gl_AVOID([])
index ea5e2665a869ecd80c8bad718834fe1722160d41..49af5381a37b43c19d9babf16ce11e5ccea77be6 100644 (file)
@@ -117,6 +117,8 @@ class Engine:Texture from U:Object (dynamic)
        private unsigned VClamp;
        private unsigned VBits;
 
+       private float DrawScale;
+
        private Engine:Palette *Palette;
        private Engine:Texture *DetailTexture;
 
@@ -270,6 +272,12 @@ class Engine:Texture from U:Object (dynamic)
                , link
                );
 
+       property FLOAT DrawScale
+               ( nick = "Draw Scale"
+               , blurb = "Relative size to parent surface"
+               , link
+               );
+
        property OBJECT Palette
                ( nick = "Palette"
                , blurb = "Reference to the texture's palette."
index a0826902d5a029998b807da9486a2ef112b02538..627abe68e32ac3dc2405a1887fd3ad85178e4d10 100644 (file)
@@ -19,6 +19,7 @@
 #include <stdio.h>
 #include <stdlib.h>
 #include <string.h>
+#include <math.h>
 #include <stdarg.h>
 #include <stdbool.h>
 #include <inttypes.h>
@@ -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);