X-Git-Url: https://git.draconx.ca/gitweb/upkg.git/blobdiff_plain/76b8a4bf05701afcd1be69ff4490a7c6be431dc4..HEAD:/src/uobject/uobject.c
diff --git a/src/uobject/uobject.c b/src/uobject/uobject.c
index df875b2..cf07735 100644
--- a/src/uobject/uobject.c
+++ b/src/uobject/uobject.c
@@ -1,26 +1,27 @@
/*
- * upkg: tool for manipulating Unreal Tournament packages.
- * Copyright © 2009-2011 Nick Bowler
+ * upkg: tool for manipulating Unreal Tournament packages.
+ * Copyright © 2009-2012, 2015, 2020, 2022 Nick Bowler
*
- * This program is free software: you can redistribute it and/or modify
- * it under the terms of the GNU General Public License as published by
- * the Free Software Foundation, either version 3 of the License, or
- * (at your option) any later version.
+ * This program is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
*
- * This program is distributed in the hope that it will be useful,
- * but WITHOUT ANY WARRANTY; without even the implied warranty of
- * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- * GNU General Public License for more details.
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
*
- * You should have received a copy of the GNU General Public License
- * along with this program. If not, see .
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see .
*/
+#include
#include
#include
#include
+#include
#include
-#include
#include
#include
#include
@@ -34,11 +35,14 @@
#define U_OBJECT_GET_PRIV(o) \
G_TYPE_INSTANCE_GET_PRIVATE(o, U_TYPE_OBJECT, struct u_object_priv)
+#ifndef SIZE_MAX
+# define SIZE_MAX ((size_t)-1)
+#endif
struct prop_head {
const char *prop_name, *struct_name;
unsigned long size, array_idx;
- bool tag_msb;
+ gboolean tag_msb;
enum {
PROPERTY_END,
@@ -190,6 +194,108 @@ decode_prop_header(struct upkg *upkg, struct prop_head *head,
return len;
}
+/*
+ * TODO: Make this use the uobject_module stuff so packages are not loaded
+ * more than once.
+ */
+static GTypeModule *
+open_import_package(UObject *uo, const struct upkg_import *i)
+{
+ GTypeModule *pkg;
+
+ assert(i->parent == NULL);
+
+ if (strcmp(i->class_package, "Core") != 0
+ || strcmp(i->class_name, "Package") != 0) {
+ u_err(uo, "import root must be Core.Package");
+ return NULL;
+ }
+
+ pkg = u_pkg_open(i->name);
+ if (!pkg || !g_type_module_use(pkg)) {
+ u_err(uo, "failed to open package: %s", i->name);
+ return NULL;
+ }
+
+ if (!U_PKG(pkg)->pkg) {
+ g_type_module_unuse(pkg);
+ u_err(uo, "failed to open package: %s", pkg->name);
+ return NULL;
+ }
+
+ return pkg;
+}
+
+/* Find the export index of a package based on the import chain from another
+ * package. Returns the (non-negative) offset on success, or a negative value
+ * on failure. */
+static long resolve_import(struct upkg *pkg, const struct upkg_import *import)
+{
+ long current, index;
+
+ if (!import->parent)
+ return -1;
+
+ current = resolve_import(pkg, import->parent);
+ if (current != -1 && current >= 0)
+ return -42;
+
+ index = upkg_export_find(pkg, current, import->name);
+ if (index < 0)
+ return -42;
+
+ return index;
+}
+
+static GObject *get_import_object(UObject *uo, unsigned long index)
+{
+ const struct upkg_import *import;
+ GObject *obj = NULL;
+ GTypeModule *pkg;
+ long obj_index;
+
+ import = upkg_get_import(uo->pkg_file->pkg, index);
+ if (!import) {
+ u_err(uo, "invalid package import: %ld", index);
+ return NULL;
+ }
+
+ /* Get the package name at the top of the hierarchy. */
+ for (const struct upkg_import *i = import; i; i = i->parent) {
+ if (i->parent == NULL) {
+ pkg = open_import_package(uo, i);
+ if (!pkg)
+ return NULL;
+ }
+ }
+
+ obj_index = resolve_import(U_PKG(pkg)->pkg, import);
+ if (obj_index < 0) {
+ u_err(uo, "could not find import in package: %s", pkg->name);
+ goto out;
+ }
+
+ obj = u_object_new_from_package(pkg, obj_index);
+out:
+ g_type_module_unuse(pkg);
+ return obj;
+}
+
+GObject *u_object_get_by_link(GObject *go, long index)
+{
+ g_return_val_if_fail(IS_U_OBJECT(go), NULL);
+ UObject *uo = U_OBJECT(go);
+
+ if (index == 0)
+ return NULL;
+
+ if (index < 0) {
+ return get_import_object(uo, -(index+1));
+ }
+
+ return u_object_new_from_package(uo->pkg, index-1);
+}
+
static int decode_object_property(UObject *uo, GValue *val, unsigned long len)
{
struct u_object_priv *priv = U_OBJECT_GET_PRIV(uo);
@@ -198,15 +304,10 @@ static int decode_object_property(UObject *uo, GValue *val, unsigned long len)
int rc;
rc = upkg_decode_index(&index, priv->buf+len, priv->nbuf-len);
- if (rc == 0 || index == 0)
+ if (rc == 0)
return -1;
- if (index < 0) {
- fprintf(stderr, "Imports not supported yet.\n");
- } else {
- obj = u_object_new_from_package(uo->pkg, index-1);
- }
-
+ obj = u_object_get_by_link(G_OBJECT(uo), index);
g_value_init(val, U_TYPE_OBJECT);
g_value_take_object(val, obj);
return 0;
@@ -227,20 +328,22 @@ static unsigned long deserialize_property(UObject *uo, struct prop_head *head)
case PROPERTY_END:
break;
case PROPERTY_BYTE:
- if (priv->nbuf-len < 1)
+ if (head->size != 1 || priv->nbuf-len < head->size)
return 0;
g_value_init(&val, G_TYPE_UCHAR);
g_value_set_uchar(&val, priv->buf[len]);
g_object_set_property(G_OBJECT(uo), head->prop_name, &val);
break;
case PROPERTY_INTEGER:
- if (priv->nbuf-len < 4)
+ if (head->size != 4 || priv->nbuf-len < head->size)
return 0;
g_value_init(&val, G_TYPE_ULONG);
g_value_set_ulong(&val, unpack_32_le(priv->buf+len));
g_object_set_property(G_OBJECT(uo), head->prop_name, &val);
break;
case PROPERTY_BOOLEAN:
+ if (head->size != 0)
+ return 0;
g_value_init(&val, G_TYPE_BOOLEAN);
g_value_set_boolean(&val, head->tag_msb);
g_object_set_property(G_OBJECT(uo), head->prop_name, &val);
@@ -250,10 +353,23 @@ static unsigned long deserialize_property(UObject *uo, struct prop_head *head)
if (rc != 0)
return 0;
g_object_set_property(G_OBJECT(uo), head->prop_name, &val);
+
+ /*
+ * g_object_set_property increments refcount,
+ * release our reference.
+ */
+ g_object_unref(g_value_get_object(&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, u_unpack_binary32_le(priv->buf+len));
+ g_object_set_property(G_OBJECT(uo), head->prop_name, &val);
break;
default:
- fprintf(stderr, "Unhandled property type %x\n",
- (unsigned)head->type);
+ u_warn(uo, "%s: unsupported property type %u",
+ head->prop_name, (unsigned)head->type);
}
len += head->size;
@@ -314,17 +430,17 @@ static int deserialize(UObject *uo)
/*
* Get the full hierarchical object name for an export, used for diagnostics.
- * Currently, this does not include the package name (which is actually
- * required for this to be a "full" name) as it's not yet passed to the
- * deserialize method.
+ * The package name is passed in pname.
*
* Returns a buffer allocated by malloc on success, or NULL on failure.
*/
-static char *get_obj_fullname(const struct upkg_export *e)
+static char *get_obj_fullname(const char *pname, const struct upkg_export *e)
{
- size_t total_len = 0, len;
+ size_t total_len = strlen(pname) + 1, len;
char *fullname;
+ assert(e != NULL);
+
for (const struct upkg_export *c = e; c; c = c->parent) {
len = strlen(c->name) + 1;
if (total_len > SIZE_MAX - len)
@@ -345,6 +461,10 @@ static char *get_obj_fullname(const struct upkg_export *e)
fullname[total_len + len] = c == e ? '\0' : '.';
}
+ assert(total_len == strlen(pname)+1);
+ memcpy(fullname, pname, total_len-1);
+ fullname[total_len-1] = '.';
+
return fullname;
}
@@ -371,7 +491,7 @@ int u_object_deserialize(GObject *obj, GTypeModule *pkg, unsigned long idx)
e = upkg_get_export(upkg, idx);
if (!e)
return -1;
- uo->pkg_name = get_obj_fullname(e);
+ uo->pkg_name = get_obj_fullname(pkg->name, e);
if (!uo->pkg_name)
return -1;
@@ -389,13 +509,16 @@ GObject *u_object_new_from_package(GTypeModule *pkg, unsigned long idx)
g_return_val_if_fail(IS_U_PKG(pkg), NULL);
const struct upkg_export *export;
- const char *class, *package;
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) {
@@ -475,18 +598,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);