[retro-gtk/wip/aplazas/c-port: 12/25] Port CoreDescriptor to C
- From: Adrien Plazas <aplazas src gnome org>
- To: commits-list gnome org
- Cc:
- Subject: [retro-gtk/wip/aplazas/c-port: 12/25] Port CoreDescriptor to C
- Date: Tue, 10 Oct 2017 16:39:50 +0000 (UTC)
commit f84817d9b7703dcc3eca929be2537b918bd7a5d1
Author: Adrien Plazas <kekun plazas laposte net>
Date: Thu Sep 7 12:10:03 2017 +0200
Port CoreDescriptor to C
retro-gtk/Makefile.am | 5 +-
retro-gtk/retro-core-descriptor-error.vala | 8 -
retro-gtk/retro-core-descriptor.c | 1010 ++++++++++++++++++++++++++++
retro-gtk/retro-core-descriptor.h | 75 ++
retro-gtk/retro-core-descriptor.vala | 178 -----
retro-gtk/retro-gtk.h | 1 +
retro-gtk/retro-module-iterator.h | 4 +-
7 files changed, 1090 insertions(+), 191 deletions(-)
---
diff --git a/retro-gtk/Makefile.am b/retro-gtk/Makefile.am
index c31c60d..bd0edee 100644
--- a/retro-gtk/Makefile.am
+++ b/retro-gtk/Makefile.am
@@ -82,8 +82,7 @@ libretro_gtk_la_SOURCES = \
retro-analog-id.c \
retro-analog-index.c \
retro-cairo-display.c \
- retro-core-descriptor.vala \
- retro-core-descriptor-error.vala \
+ retro-core-descriptor.c \
retro-core-view.c \
retro-core-view-input-device.c \
retro-device-type.c \
@@ -115,6 +114,8 @@ video/retro-video-converter.c: retro-gtk-internal.h
retro-core.c: retro-gtk-internal.h
+# retro-core-descriptor.c: retro-gtk-internal.h
+
retro-core-view.c: retro-gtk-internal.h
retro-core-view-input-device.c: retro-gtk-internal.h
diff --git a/retro-gtk/retro-core-descriptor.c b/retro-gtk/retro-core-descriptor.c
new file mode 100644
index 0000000..b7381ae
--- /dev/null
+++ b/retro-gtk/retro-core-descriptor.c
@@ -0,0 +1,1010 @@
+// This file is part of retro-gtk. License: GPL-3.0+.
+
+#include "retro-core-descriptor.h"
+
+struct _RetroCoreDescriptor
+{
+ GObject parent_instance;
+ gchar *filename;
+ GKeyFile *key_file;
+};
+
+G_DEFINE_TYPE (RetroCoreDescriptor, retro_core_descriptor, G_TYPE_OBJECT)
+
+#define RETRO_CORE_DESCRIPTOR_ERROR (retro_core_descriptor_error_quark ())
+
+enum {
+ RETRO_CORE_DESCRIPTOR_ERROR_REQUIRED_GROUP_NOT_FOUND,
+ RETRO_CORE_DESCRIPTOR_ERROR_REQUIRED_KEY_NOT_FOUND,
+ RETRO_CORE_DESCRIPTOR_ERROR_FIRMWARE_NOT_FOUND
+};
+
+#define RETRO_CORE_DESCRIPTOR_LIBRETRO_GROUP "Libretro"
+#define RETRO_CORE_DESCRIPTOR_PLATFORM_GROUP_PREFIX "Platform:"
+#define RETRO_CORE_DESCRIPTOR_FIRMWARE_GROUP_PREFIX "Firmware:"
+
+#define RETRO_CORE_DESCRIPTOR_TYPE_KEY "Type"
+#define RETRO_CORE_DESCRIPTOR_NAME_KEY "Name"
+#define RETRO_CORE_DESCRIPTOR_ICON_KEY "Icon"
+#define RETRO_CORE_DESCRIPTOR_MODULE_KEY "Module"
+#define RETRO_CORE_DESCRIPTOR_LIBRETRO_VERSION_KEY "LibretroVersion"
+
+#define RETRO_CORE_DESCRIPTOR_PLATFORM_MIME_TYPE_KEY "MimeType"
+#define RETRO_CORE_DESCRIPTOR_PLATFORM_FIRMWARES_KEY "Firmwares"
+
+#define RETRO_CORE_DESCRIPTOR_FIRMWARE_PATH_KEY "Path"
+#define RETRO_CORE_DESCRIPTOR_FIRMWARE_MD5_KEY "MD5"
+#define RETRO_CORE_DESCRIPTOR_FIRMWARE_SHA512_KEY "SHA-512"
+#define RETRO_CORE_DESCRIPTOR_FIRMWARE_MANDATORY_KEY "Mandatory"
+
+#define RETRO_CORE_DESCRIPTOR_TYPE_GAME "Game"
+#define RETRO_CORE_DESCRIPTOR_TYPE_EMULATOR "Emulator"
+
+/* Private */
+
+static GQuark
+retro_core_descriptor_error_quark (void)
+{
+ return g_quark_from_static_string ("retro-core-descriptor-error-quark");
+}
+
+static void
+retro_core_descriptor_finalize (GObject *object)
+{
+ RetroCoreDescriptor *self = RETRO_CORE_DESCRIPTOR (object);
+
+ g_free (self->filename);
+ g_key_file_unref (self->key_file);
+
+ G_OBJECT_CLASS (retro_core_descriptor_parent_class)->finalize (object);
+}
+
+static void
+retro_core_descriptor_class_init (RetroCoreDescriptorClass *klass)
+{
+ GObjectClass *object_class = G_OBJECT_CLASS (klass);
+
+ object_class->finalize = retro_core_descriptor_finalize;
+}
+
+static void
+retro_core_descriptor_init (RetroCoreDescriptor *self)
+{
+}
+
+static gboolean
+has_group_prefixed (RetroCoreDescriptor *self,
+ const gchar *group_prefix,
+ const gchar *group_suffix)
+{
+ gchar *group;
+ gboolean result;
+
+ g_return_val_if_fail (RETRO_IS_CORE_DESCRIPTOR (self), FALSE);
+ g_return_val_if_fail (group_prefix != NULL, FALSE);
+ g_return_val_if_fail (group_suffix != NULL, FALSE);
+
+ group = g_strconcat (group_prefix, group_suffix, NULL);
+ result = g_key_file_has_group (self->key_file, group);
+ g_free (group);
+
+ return result;
+}
+
+static gboolean
+has_key_prefixed (RetroCoreDescriptor *self,
+ const gchar *group_prefix,
+ const gchar *group_suffix,
+ const gchar *key,
+ GError **error)
+{
+ gchar *group;
+ gboolean result;
+ GError *tmp_error = NULL;
+
+ g_return_val_if_fail (RETRO_IS_CORE_DESCRIPTOR (self), FALSE);
+ g_return_val_if_fail (group_prefix != NULL, FALSE);
+ g_return_val_if_fail (group_suffix != NULL, FALSE);
+ g_return_val_if_fail (key != NULL, FALSE);
+
+ group = g_strconcat (group_prefix, group_suffix, NULL);
+ result = g_key_file_has_key (self->key_file,
+ group,
+ key,
+ &tmp_error);
+ if (G_UNLIKELY (tmp_error != NULL)) {
+ g_free (group);
+ g_propagate_error (error, tmp_error);
+
+ return FALSE;
+ }
+
+ g_free (group);
+
+ return result;
+}
+
+static gchar *
+get_string_prefixed (RetroCoreDescriptor *self,
+ const gchar *group_prefix,
+ const gchar *group_suffix,
+ const gchar *key,
+ GError **error)
+{
+ gchar *group;
+ gchar *result;
+ GError *tmp_error = NULL;
+
+ g_return_val_if_fail (RETRO_IS_CORE_DESCRIPTOR (self), NULL);
+ g_return_val_if_fail (group_prefix != NULL, NULL);
+ g_return_val_if_fail (group_suffix != NULL, NULL);
+ g_return_val_if_fail (key != NULL, NULL);
+
+ group = g_strconcat (group_prefix, group_suffix, NULL);
+
+ result = g_key_file_get_string (self->key_file,
+ group,
+ key,
+ &tmp_error);
+ if (G_UNLIKELY (tmp_error != NULL)) {
+ g_free (group);
+ g_propagate_error (error, tmp_error);
+
+ return NULL;
+ }
+
+ g_free (group);
+
+ return result;
+}
+
+static gchar **
+get_string_list_prefixed (RetroCoreDescriptor *self,
+ const gchar *group_prefix,
+ const gchar *group_suffix,
+ const gchar *key,
+ gsize *length,
+ GError **error)
+{
+ gchar *group;
+ gchar **result;
+ GError *tmp_error = NULL;
+
+ g_return_val_if_fail (RETRO_IS_CORE_DESCRIPTOR (self), NULL);
+ g_return_val_if_fail (group_prefix != NULL, NULL);
+ g_return_val_if_fail (group_suffix != NULL, NULL);
+ g_return_val_if_fail (key != NULL, NULL);
+ g_return_val_if_fail (length != NULL, NULL);
+
+ group = g_strconcat (group_prefix, group_suffix, NULL);
+ result = g_key_file_get_string_list (self->key_file,
+ group,
+ key,
+ length,
+ &tmp_error);
+ if (G_UNLIKELY (tmp_error != NULL)) {
+ g_free (group);
+ g_propagate_error (error, tmp_error);
+
+ return NULL;
+ }
+
+ g_free (group);
+
+ return result;
+}
+
+static void
+retro_core_descriptor_check_has_required_key (RetroCoreDescriptor *self,
+ const gchar *group,
+ const gchar *key,
+ GError **error)
+{
+ gboolean has_key;
+ GError *tmp_error = NULL;
+
+ g_return_if_fail (RETRO_IS_CORE_DESCRIPTOR (self));
+ g_return_if_fail (group != NULL);
+ g_return_if_fail (key != NULL);
+
+ has_key = g_key_file_has_key (self->key_file,
+ RETRO_CORE_DESCRIPTOR_LIBRETRO_GROUP,
+ RETRO_CORE_DESCRIPTOR_TYPE_KEY,
+ &tmp_error);
+ if (G_UNLIKELY (tmp_error != NULL)) {
+ g_propagate_error (error, tmp_error);
+
+ return;
+ }
+
+ if (!has_key)
+ g_set_error (error,
+ RETRO_CORE_DESCRIPTOR_ERROR,
+ RETRO_CORE_DESCRIPTOR_ERROR_REQUIRED_KEY_NOT_FOUND,
+ "%s isn't a valid Libretro core descriptor: "
+ "required key %s not found in group [%s].",
+ self->filename,
+ key,
+ group);
+}
+
+// FIXME static
+void
+retro_core_descriptor_check_libretro_group (RetroCoreDescriptor *self,
+ GError **error)
+{
+ GError *tmp_error = NULL;
+
+ g_return_if_fail (RETRO_IS_CORE_DESCRIPTOR (self));
+
+ retro_core_descriptor_check_has_required_key (self,
+ RETRO_CORE_DESCRIPTOR_LIBRETRO_GROUP,
+ RETRO_CORE_DESCRIPTOR_TYPE_KEY,
+ &tmp_error);
+ if (G_UNLIKELY (tmp_error != NULL)) {
+ g_propagate_error (error, tmp_error);
+
+ return;
+ }
+
+ retro_core_descriptor_check_has_required_key (self,
+ RETRO_CORE_DESCRIPTOR_LIBRETRO_GROUP,
+ RETRO_CORE_DESCRIPTOR_NAME_KEY,
+ &tmp_error);
+ if (G_UNLIKELY (tmp_error != NULL)) {
+ g_propagate_error (error, tmp_error);
+
+ return;
+ }
+
+ retro_core_descriptor_check_has_required_key (self,
+ RETRO_CORE_DESCRIPTOR_LIBRETRO_GROUP,
+ RETRO_CORE_DESCRIPTOR_MODULE_KEY,
+ &tmp_error);
+ if (G_UNLIKELY (tmp_error != NULL)) {
+ g_propagate_error (error, tmp_error);
+
+ return;
+ }
+
+ retro_core_descriptor_check_has_required_key (self,
+ RETRO_CORE_DESCRIPTOR_LIBRETRO_GROUP,
+ RETRO_CORE_DESCRIPTOR_LIBRETRO_VERSION_KEY,
+ &tmp_error);
+ if (G_UNLIKELY (tmp_error != NULL)) {
+ g_propagate_error (error, tmp_error);
+
+ return;
+ }
+}
+
+// FIXME static
+void
+retro_core_descriptor_check_platform_group (RetroCoreDescriptor *self,
+ const gchar *group,
+ GError **error)
+{
+ gboolean has_key;
+ gchar **firmwares;
+ gchar **firmware_p;
+ gchar *firmware_group;
+ GError *tmp_error = NULL;
+
+ g_return_if_fail (RETRO_IS_CORE_DESCRIPTOR (self));
+ g_return_if_fail (group != NULL);
+
+ retro_core_descriptor_check_has_required_key (self,
+ group,
+ RETRO_CORE_DESCRIPTOR_PLATFORM_MIME_TYPE_KEY,
+ &tmp_error);
+
+ has_key = g_key_file_has_key (self->key_file,
+ group,
+ RETRO_CORE_DESCRIPTOR_PLATFORM_FIRMWARES_KEY,
+ &tmp_error);
+ if (G_UNLIKELY (tmp_error != NULL)) {
+ g_propagate_error (error, tmp_error);
+
+ return;
+ }
+
+ if (!has_key)
+ return;
+
+ firmwares = g_key_file_get_string_list (self->key_file,
+ group,
+ RETRO_CORE_DESCRIPTOR_PLATFORM_FIRMWARES_KEY,
+ NULL,
+ &tmp_error);
+ if (G_UNLIKELY (tmp_error != NULL)) {
+ g_propagate_error (error, tmp_error);
+
+ return;
+ }
+
+ for (firmware_p = firmwares; firmware_p != NULL; firmware_p++) {
+ firmware_group = g_strconcat (RETRO_CORE_DESCRIPTOR_FIRMWARE_GROUP_PREFIX, *firmware_p, NULL);
+ if (!g_key_file_has_group (self->key_file, firmware_group)) {
+ g_set_error (error,
+ RETRO_CORE_DESCRIPTOR_ERROR,
+ RETRO_CORE_DESCRIPTOR_ERROR_FIRMWARE_NOT_FOUND,
+ "%s isn't a valid Libretro core descriptor:"
+ "[%s] mentioned in [%s] not found.",
+ self->filename,
+ firmware_group,
+ group);
+
+ g_free (firmware_group);
+ g_strfreev (firmwares);
+
+ return;
+ }
+
+ g_free (firmware_group);
+ }
+
+ g_strfreev (firmwares);
+}
+
+static void
+retro_core_descriptor_check_firmware_group (RetroCoreDescriptor *self,
+ const gchar *group,
+ GError **error)
+{
+ GError *tmp_error = NULL;
+
+ g_return_if_fail (RETRO_IS_CORE_DESCRIPTOR (self));
+ g_return_if_fail (group != NULL);
+
+ retro_core_descriptor_check_has_required_key (self,
+ group,
+ RETRO_CORE_DESCRIPTOR_FIRMWARE_PATH_KEY,
+ &tmp_error);
+ if (G_UNLIKELY (tmp_error != NULL)) {
+ g_propagate_error (error, tmp_error);
+
+ return;
+ }
+
+ retro_core_descriptor_check_has_required_key (self,
+ group,
+ RETRO_CORE_DESCRIPTOR_FIRMWARE_MANDATORY_KEY,
+ &tmp_error);
+ if (G_UNLIKELY (tmp_error != NULL)) {
+ g_propagate_error (error, tmp_error);
+
+ return;
+ }
+}
+
+/* Public */
+
+/**
+ * retro_core_descriptor_has_icon:
+ * @self: a #RetroCoreDescriptor
+ * @error: return location for a #GError, or %NULL
+ *
+ * Gets whether the core has an icon.
+ *
+ * Returns: whether the core has an icon
+ */
+gboolean
+retro_core_descriptor_has_icon (RetroCoreDescriptor *self,
+ GError **error)
+{
+ gboolean has_icon;
+ GError *tmp_error = NULL;
+
+ g_return_val_if_fail (RETRO_IS_CORE_DESCRIPTOR (self), FALSE);
+
+ has_icon = g_key_file_has_key (self->key_file, RETRO_CORE_DESCRIPTOR_LIBRETRO_GROUP,
RETRO_CORE_DESCRIPTOR_ICON_KEY, &tmp_error);
+ if (G_UNLIKELY (tmp_error != NULL)) {
+ g_propagate_error (error, tmp_error);
+
+ return FALSE;
+ }
+
+ return has_icon;
+}
+
+/**
+ * retro_core_descriptor_get_uri
+ * @self: a #RetroCoreDescriptor
+ *
+ * Gets the URI of the file of @self.
+ *
+ * Returns: the URI of the file of @self, free it with g_free()
+ */
+gchar *
+retro_core_descriptor_get_uri (RetroCoreDescriptor *self)
+{
+ gchar *result;
+ GFile *file;
+
+ g_return_val_if_fail (RETRO_IS_CORE_DESCRIPTOR (self), NULL);
+
+ file = g_file_new_for_path (self->filename);
+ result = g_file_get_uri (file);
+ g_object_unref (file);
+
+ return result;
+}
+
+/**
+ * retro_core_descriptor_get_id
+ * @self: a #RetroCoreDescriptor
+ *
+ * Gets the ID of @self.
+ *
+ * Returns: the ID of @self, free it with g_free()
+ */
+gchar *
+retro_core_descriptor_get_id (RetroCoreDescriptor *self)
+{
+ g_return_val_if_fail (RETRO_IS_CORE_DESCRIPTOR (self), NULL);
+
+ return g_path_get_basename (self->filename);
+}
+
+/**
+ * retro_core_descriptor_get_is_game
+ * @self: a #RetroCoreDescriptor
+ * @error: return location for a #GError, or %NULL
+ *
+ * Gets whether the core is a game, and hence can't load games.
+ *
+ * Returns: whether the core is a game
+ */
+gboolean
+retro_core_descriptor_get_is_game (RetroCoreDescriptor *self,
+ GError **error)
+{
+ gchar *type;
+ gboolean is_game;
+ GError *tmp_error = NULL;
+
+ g_return_val_if_fail (RETRO_IS_CORE_DESCRIPTOR (self), FALSE);
+
+ type = g_key_file_get_string (self->key_file,
+ RETRO_CORE_DESCRIPTOR_LIBRETRO_GROUP,
+ RETRO_CORE_DESCRIPTOR_TYPE_KEY,
+ &tmp_error);
+ if (G_UNLIKELY (tmp_error != NULL)) {
+ g_propagate_error (error, tmp_error);
+
+ return FALSE;
+ }
+
+ is_game = g_strcmp0 (type, RETRO_CORE_DESCRIPTOR_TYPE_GAME) == 0;
+ g_free (type);
+
+ return is_game;
+}
+
+/**
+ * retro_core_descriptor_get_is_emulator
+ * @self: a #RetroCoreDescriptor
+ * @error: return location for a #GError, or %NULL
+ *
+ * Gets whether the core is an emulator, and hence need games to be loaded.
+ *
+ * Returns: whether the core is an emulator
+ */
+gboolean
+retro_core_descriptor_get_is_emulator (RetroCoreDescriptor *self,
+ GError **error)
+{
+ gchar *type;
+ gboolean is_game;
+ GError *tmp_error = NULL;
+
+ g_return_val_if_fail (RETRO_IS_CORE_DESCRIPTOR (self), FALSE);
+
+ type = g_key_file_get_string (self->key_file,
+ RETRO_CORE_DESCRIPTOR_LIBRETRO_GROUP,
+ RETRO_CORE_DESCRIPTOR_TYPE_KEY,
+ &tmp_error);
+ if (G_UNLIKELY (tmp_error != NULL)) {
+ g_propagate_error (error, tmp_error);
+
+ return FALSE;
+ }
+
+ is_game = g_strcmp0 (type, RETRO_CORE_DESCRIPTOR_TYPE_EMULATOR) == 0;
+ g_free (type);
+
+ return is_game;
+}
+
+/**
+ * retro_core_descriptor_get_name
+ * @self: a #RetroCoreDescriptor
+ * @error: return location for a #GError, or %NULL
+ *
+ * Gets the name, or %NULL if it doesn't exist.
+ *
+ * Returns: (nullable) (transfer full): a string or %NULL, free it with g_free()
+ */
+gchar *
+retro_core_descriptor_get_name (RetroCoreDescriptor *self,
+ GError **error)
+{
+ return g_key_file_get_string (self->key_file,
+ RETRO_CORE_DESCRIPTOR_LIBRETRO_GROUP,
+ RETRO_CORE_DESCRIPTOR_NAME_KEY,
+ error);
+}
+
+/**
+ * retro_core_descriptor_get_icon
+ * @self: a #RetroCoreDescriptor
+ * @error: return location for a #GError, or %NULL
+ *
+ * Gets the icon, or %NULL if it doesn't exist.
+ *
+ * Returns: (nullable) (transfer full): a #GIcon or %NULL
+ */
+GIcon *
+retro_core_descriptor_get_icon (RetroCoreDescriptor *self,
+ GError **error)
+{
+ gchar *icon_name;
+ GIcon *icon;
+ GError *tmp_error = NULL;
+
+ g_return_val_if_fail (RETRO_IS_CORE_DESCRIPTOR (self), NULL);
+
+ icon_name = g_key_file_get_string (self->key_file,
+ RETRO_CORE_DESCRIPTOR_LIBRETRO_GROUP,
+ RETRO_CORE_DESCRIPTOR_ICON_KEY,
+ &tmp_error);
+ if (G_UNLIKELY (tmp_error != NULL)) {
+ g_propagate_error (error, tmp_error);
+
+ return NULL;
+ }
+
+ icon = G_ICON (g_themed_icon_new (icon_name));
+ g_free (icon_name);
+
+ return icon;
+}
+
+/**
+ * retro_core_descriptor_get_module
+ * @self: a #RetroCoreDescriptor
+ * @error: return location for a #GError, or %NULL
+ *
+ * Gets the module file name, or %NULL if it doesn't exist.
+ *
+ * Returns: (nullable) (transfer full): a string or %NULL, free it with g_free()
+ */
+char *
+retro_core_descriptor_get_module (RetroCoreDescriptor *self,
+ GError **error)
+{
+ g_return_val_if_fail (RETRO_IS_CORE_DESCRIPTOR (self), FALSE);
+
+ return g_key_file_get_string (self->key_file,
+ RETRO_CORE_DESCRIPTOR_LIBRETRO_GROUP,
+ RETRO_CORE_DESCRIPTOR_MODULE_KEY,
+ error);
+}
+
+/**
+ * retro_core_descriptor_get_module_file
+ * @self: a #RetroCoreDescriptor
+ * @error: return location for a #GError, or %NULL
+ *
+ * Gets the module file, or %NULL if it doesn't exist.
+ *
+ * Returns: (nullable) (transfer full): a #GFile or %NULL
+ */
+GFile *
+retro_core_descriptor_get_module_file (RetroCoreDescriptor *self,
+ GError **error)
+{
+ GFile *file;
+ GFile *dir;
+ gchar *module;
+ GFile *module_file;
+ GError *tmp_error = NULL;
+
+ g_return_val_if_fail (RETRO_IS_CORE_DESCRIPTOR (self), NULL);
+
+ file = g_file_new_for_path (self->filename);
+ dir = g_file_get_parent (file);
+ g_object_unref (file);
+ if (dir == NULL)
+ return NULL;
+
+ module = retro_core_descriptor_get_module (self, &tmp_error);
+ if (G_UNLIKELY (tmp_error != NULL)) {
+ g_propagate_error (error, tmp_error);
+ g_object_unref (dir);
+
+ return NULL;
+ }
+
+ module_file = g_file_get_child (dir, module);
+ if (!g_file_query_exists (module_file, NULL)) {
+ g_object_unref (dir);
+ g_free (module);
+ g_object_unref (module_file);
+
+ return NULL;
+ }
+
+ g_object_unref (dir);
+ g_free (module);
+
+ return module_file;
+}
+
+/**
+ * retro_core_descriptor_has_platform:
+ * @self: a #RetroCoreDescriptor
+ * @platform: a platform name
+ *
+ * Gets whether the core descriptor declares the given platform.
+ *
+ * Returns: whether the core descriptor declares the given platform
+ */
+gboolean
+retro_core_descriptor_has_platform (RetroCoreDescriptor *self,
+ const gchar *platform)
+{
+ return has_group_prefixed (self,
+ RETRO_CORE_DESCRIPTOR_PLATFORM_GROUP_PREFIX,
+ platform);
+}
+
+/**
+ * retro_core_descriptor_has_firmwares:
+ * @self: a #RetroCoreDescriptor
+ * @platform: a platform name
+ * @error: return location for a #GError, or %NULL
+ *
+ * Gets whether the platform has associated firmwares.
+ *
+ * Returns: whether the platform has associated firmwares
+ */
+gboolean
+retro_core_descriptor_has_firmwares (RetroCoreDescriptor *self,
+ const gchar *platform,
+ GError **error)
+{
+ return has_key_prefixed (self,
+ RETRO_CORE_DESCRIPTOR_PLATFORM_GROUP_PREFIX,
+ platform,
+ RETRO_CORE_DESCRIPTOR_PLATFORM_FIRMWARES_KEY,
+ error);
+}
+
+/**
+ * retro_core_descriptor_has_firmware_md5:
+ * @self: a #RetroCoreDescriptor
+ * @firmware: a firmware name
+ * @error: return location for a #GError, or %NULL
+ *
+ * Gets whether the firmware declares its MD5 fingerprint.
+ *
+ * Returns: whether the firmware declares its MD5 fingerprint
+ */
+gboolean
+retro_core_descriptor_has_firmware_md5 (RetroCoreDescriptor *self,
+ const gchar *firmware,
+ GError **error)
+{
+ return has_key_prefixed (self,
+ RETRO_CORE_DESCRIPTOR_FIRMWARE_GROUP_PREFIX,
+ firmware,
+ RETRO_CORE_DESCRIPTOR_FIRMWARE_MD5_KEY,
+ error);
+}
+
+/**
+ * retro_core_descriptor_has_firmware_sha512:
+ * @self: a #RetroCoreDescriptor
+ * @firmware: a firmware name
+ * @error: return location for a #GError, or %NULL
+ *
+ * Gets whether the firmware declares its SHA512 fingerprint.
+ *
+ * Returns: whether the firmware declares its SHA512 fingerprint
+ */
+gboolean
+retro_core_descriptor_has_firmware_sha512 (RetroCoreDescriptor *self,
+ const gchar *firmware,
+ GError **error)
+{
+ return has_key_prefixed (self,
+ RETRO_CORE_DESCRIPTOR_FIRMWARE_GROUP_PREFIX,
+ firmware,
+ RETRO_CORE_DESCRIPTOR_FIRMWARE_SHA512_KEY,
+ error);
+}
+
+/**
+ * retro_core_descriptor_get_mime_type
+ * @self: a #RetroCoreDescriptor
+ * @platform: a platform name
+ * @length: (out) (optional): return location for the number of returned
+ * strings, or %NULL
+ * @error: return location for a #GError, or %NULL
+ *
+ * Gets the list of MIME types accepted used by the core for this platform.
+ *
+ * Returns: (array zero-terminated=1 length=length) (element-type utf8)
+ * (transfer full): a %NULL-terminated string array or %NULL, the array should
+ * be freed with g_strfreev()
+ */
+gchar **
+retro_core_descriptor_get_mime_type (RetroCoreDescriptor *self,
+ const gchar *platform,
+ gsize *length,
+ GError **error)
+{
+ return get_string_list_prefixed (self,
+ RETRO_CORE_DESCRIPTOR_PLATFORM_GROUP_PREFIX,
+ platform,
+ RETRO_CORE_DESCRIPTOR_PLATFORM_MIME_TYPE_KEY,
+ length,
+ error);
+}
+
+/**
+ * retro_core_descriptor_get_firmwares
+ * @self: a #RetroCoreDescriptor
+ * @platform: a platform name
+ * @length: (out) (optional): return location for the number of returned
+ * strings, or %NULL
+ * @error: return location for a #GError, or %NULL
+ *
+ * Gets the list of firmwares used by the core for this platform.
+ *
+ * Returns: (array zero-terminated=1 length=length) (element-type utf8)
+ * (transfer full): a %NULL-terminated string array or %NULL, the array should
+ * be freed with g_strfreev()
+ */
+gchar **
+retro_core_descriptor_get_firmwares (RetroCoreDescriptor *self,
+ const gchar *platform,
+ gsize *length,
+ GError **error)
+{
+ return get_string_list_prefixed (self,
+ RETRO_CORE_DESCRIPTOR_PLATFORM_GROUP_PREFIX,
+ platform,
+ RETRO_CORE_DESCRIPTOR_PLATFORM_FIRMWARES_KEY,
+ length,
+ error);
+}
+
+/**
+ * retro_core_descriptor_get_firmware_path
+ * @self: a #RetroCoreDescriptor
+ * @firmware: a firmware name
+ * @error: return location for a #GError, or %NULL
+ *
+ * Gets the demanded path to the firmware file, or %NULL.
+ *
+ * Returns: (nullable) (transfer full): a string or %NULL, free it with g_free()
+ */
+gchar *
+retro_core_descriptor_get_firmware_path (RetroCoreDescriptor *self,
+ const gchar *firmware,
+ GError **error)
+{
+ return get_string_prefixed (self,
+ RETRO_CORE_DESCRIPTOR_FIRMWARE_GROUP_PREFIX,
+ firmware,
+ RETRO_CORE_DESCRIPTOR_FIRMWARE_PATH_KEY,
+ error);
+}
+
+/**
+ * retro_core_descriptor_get_firmware_md5
+ * @self: a #RetroCoreDescriptor
+ * @firmware: a firmware name
+ * @error: return location for a #GError, or %NULL
+ *
+ * Gets the MD5 fingerprint of the firmware file, or %NULL.
+ *
+ * Returns: (nullable) (transfer full): a string or %NULL, free it with g_free()
+ */
+gchar *
+retro_core_descriptor_get_firmware_md5 (RetroCoreDescriptor *self,
+ const gchar *firmware,
+ GError **error)
+{
+ return get_string_prefixed (self,
+ RETRO_CORE_DESCRIPTOR_FIRMWARE_GROUP_PREFIX,
+ firmware,
+ RETRO_CORE_DESCRIPTOR_FIRMWARE_MD5_KEY,
+ error);
+}
+
+/**
+ * retro_core_descriptor_get_firmware_sha512
+ * @self: a #RetroCoreDescriptor
+ * @firmware: a firmware name
+ * @error: return location for a #GError, or %NULL
+ *
+ * Gets the SHA512 fingerprint of the firmware file, or %NULL.
+ *
+ * Returns: (nullable) (transfer full): a string or %NULL, free it with g_free()
+ */
+gchar *
+retro_core_descriptor_get_firmware_sha512 (RetroCoreDescriptor *self,
+ const gchar *firmware,
+ GError **error)
+{
+ return get_string_prefixed (self,
+ RETRO_CORE_DESCRIPTOR_FIRMWARE_GROUP_PREFIX,
+ firmware,
+ RETRO_CORE_DESCRIPTOR_FIRMWARE_SHA512_KEY,
+ error);
+}
+
+/**
+ * retro_core_descriptor_get_is_firmware_mandatory:
+ * @self: a #RetroCoreDescriptor
+ * @firmware: a firmware name
+ * @error: return location for a #GError, or %NULL
+ *
+ * Gets whether the firmware is mandatory for the core to function.
+ *
+ * Returns: whether the firmware is mandatory for the core to function
+ */
+gboolean
+retro_core_descriptor_get_is_firmware_mandatory (RetroCoreDescriptor *self,
+ const gchar *firmware,
+ GError **error)
+{
+ gchar *group;
+ gboolean result;
+ GError *tmp_error = NULL;
+
+ g_return_val_if_fail (RETRO_IS_CORE_DESCRIPTOR (self), FALSE);
+ g_return_val_if_fail (firmware != NULL, FALSE);
+
+ group = g_strconcat (RETRO_CORE_DESCRIPTOR_FIRMWARE_GROUP_PREFIX, firmware, NULL);
+ result = g_key_file_get_boolean (self->key_file,
+ group,
+ RETRO_CORE_DESCRIPTOR_FIRMWARE_MANDATORY_KEY,
+ &tmp_error);
+ if (G_UNLIKELY (tmp_error != NULL)) {
+ g_free (group);
+ g_propagate_error (error, tmp_error);
+
+ return FALSE;
+ }
+
+ g_free (group);
+
+ return result;
+}
+
+/**
+ * retro_core_descriptor_get_platform_supports_mime_types:
+ * @self: a #RetroCoreDescriptor
+ * @platform: a platform name
+ * @mime_types: (array zero-terminated=1) (element-type utf8): the MIME types
+ * @error: return location for a #GError, or %NULL
+ *
+ * Gets whether the platform supports all of the given MIME types.
+ *
+ * Returns: whether the platform supports all of the given MIME types
+ */
+gboolean
+retro_core_descriptor_get_platform_supports_mime_types (RetroCoreDescriptor *self,
+ const gchar *platform,
+ const gchar * const *mime_types,
+ GError **error)
+{
+ gchar **supported_mime_types;
+ gsize supported_mime_types_length;
+ GError *tmp_error = NULL;
+
+ g_return_val_if_fail (RETRO_IS_CORE_DESCRIPTOR (self), FALSE);
+ g_return_val_if_fail (platform != NULL, FALSE);
+ g_return_val_if_fail (mime_types != NULL, FALSE);
+
+ supported_mime_types =
+ retro_core_descriptor_get_mime_type (self,
+ platform,
+ &supported_mime_types_length,
+ &tmp_error);
+ if (G_UNLIKELY (tmp_error != NULL)) {
+ g_propagate_error (error, tmp_error);
+
+ return FALSE;
+ }
+
+ for (; *mime_types != NULL; mime_types++)
+ if (!g_strv_contains ((const char * const *) supported_mime_types,
+ *mime_types)) {
+ g_strfreev (supported_mime_types);
+
+ return FALSE;
+ }
+
+ g_strfreev (supported_mime_types);
+
+ return TRUE;
+}
+
+/**
+ * retro_core_descriptor_new:
+ * @filename: the file name of the core descriptor
+ * @error: return location for a #GError, or %NULL
+ *
+ * Creates a new #RetroCoreDescriptor.
+ *
+ * Returns: (transfer full): a new #RetroCoreDescriptor
+ */
+RetroCoreDescriptor *
+retro_core_descriptor_new (const gchar *filename,
+ GError **error)
+{
+ RetroCoreDescriptor *self;
+ gchar ** groups;
+ gsize groups_length;
+ gsize i;
+ GError *tmp_error = NULL;
+
+ g_return_val_if_fail (filename != NULL, NULL);
+
+ self = g_object_new (RETRO_TYPE_CORE_DESCRIPTOR, NULL);
+ self->filename = g_strdup (filename);
+ self->key_file = g_key_file_new ();
+ g_key_file_load_from_file (self->key_file,
+ filename,
+ G_KEY_FILE_NONE,
+ &tmp_error);
+ if (G_UNLIKELY (tmp_error != NULL)) {
+ g_object_unref (self);
+ g_propagate_error (error, tmp_error);
+
+ return NULL;
+ }
+
+ retro_core_descriptor_check_libretro_group (self, &tmp_error);
+ if (G_UNLIKELY (tmp_error != NULL)) {
+ g_object_unref (self);
+ g_propagate_error (error, tmp_error);
+
+ return NULL;
+ }
+
+ groups = g_key_file_get_groups (self->key_file, &groups_length);
+ for (i = 0; i < groups_length; i++) {
+ if (g_str_has_prefix (groups[i],
+ RETRO_CORE_DESCRIPTOR_PLATFORM_GROUP_PREFIX)) {
+ retro_core_descriptor_check_platform_group (self, groups[i], &tmp_error);
+ if (G_UNLIKELY (tmp_error != NULL)) {
+ g_strfreev (groups);
+ g_object_unref (self);
+ g_propagate_error (error, tmp_error);
+
+ return NULL;
+ }
+ }
+ else if (g_str_has_prefix (groups[i],
+ RETRO_CORE_DESCRIPTOR_FIRMWARE_GROUP_PREFIX)) {
+ retro_core_descriptor_check_firmware_group (self, groups[i], &tmp_error);
+ if (G_UNLIKELY (tmp_error != NULL)) {
+ g_strfreev (groups);
+ g_object_unref (self);
+ g_propagate_error (error, tmp_error);
+
+ return NULL;
+ }
+ }
+ }
+
+ g_strfreev (groups);
+
+ return self;
+}
diff --git a/retro-gtk/retro-core-descriptor.h b/retro-gtk/retro-core-descriptor.h
new file mode 100644
index 0000000..ae1663d
--- /dev/null
+++ b/retro-gtk/retro-core-descriptor.h
@@ -0,0 +1,75 @@
+// This file is part of retro-gtk. License: GPL-3.0+.
+
+#ifndef RETRO_CORE_DESCRIPTOR_H
+#define RETRO_CORE_DESCRIPTOR_H
+
+#if !defined(__RETRO_GTK_INSIDE__) && !defined(RETRO_GTK_COMPILATION)
+# error "Only <retro-gtk.h> can be included directly."
+#endif
+
+#include <gio/gio.h>
+#include <glib-object.h>
+
+G_BEGIN_DECLS
+
+#define RETRO_TYPE_CORE_DESCRIPTOR (retro_core_descriptor_get_type())
+
+G_DECLARE_FINAL_TYPE (RetroCoreDescriptor, retro_core_descriptor, RETRO, CORE_DESCRIPTOR, GObject)
+
+RetroCoreDescriptor *retro_core_descriptor_new (const gchar *filename,
+ GError **error);
+gboolean retro_core_descriptor_has_icon (RetroCoreDescriptor *self,
+ GError **error);
+gchar *retro_core_descriptor_get_uri (RetroCoreDescriptor *self);
+gchar *retro_core_descriptor_get_id (RetroCoreDescriptor *self);
+gboolean retro_core_descriptor_get_is_game (RetroCoreDescriptor *self,
+ GError **error);
+gboolean retro_core_descriptor_get_is_emulator (RetroCoreDescriptor *self,
+ GError **error);
+char *retro_core_descriptor_get_name (RetroCoreDescriptor *self,
+ GError **error);
+GIcon *retro_core_descriptor_get_icon (RetroCoreDescriptor *self,
+ GError **error);
+char *retro_core_descriptor_get_module (RetroCoreDescriptor *self,
+ GError **error);
+GFile *retro_core_descriptor_get_module_file (RetroCoreDescriptor *self,
+ GError **error);
+gboolean retro_core_descriptor_has_platform (RetroCoreDescriptor *self,
+ const gchar *platform);
+gboolean retro_core_descriptor_has_firmwares (RetroCoreDescriptor *self,
+ const gchar *platform,
+ GError **error);
+gboolean retro_core_descriptor_has_firmware_md5 (RetroCoreDescriptor *self,
+ const gchar *firmware,
+ GError **error);
+gboolean retro_core_descriptor_has_firmware_sha512 (RetroCoreDescriptor *self,
+ const gchar *firmware,
+ GError **error);
+gchar **retro_core_descriptor_get_mime_type (RetroCoreDescriptor *self,
+ const gchar *platform,
+ gsize *length,
+ GError **error);
+gchar **retro_core_descriptor_get_firmwares (RetroCoreDescriptor *self,
+ const gchar *platform,
+ gsize *length,
+ GError **error);
+gchar *retro_core_descriptor_get_firmware_path (RetroCoreDescriptor *self,
+ const gchar *firmware,
+ GError **error);
+gchar * retro_core_descriptor_get_firmware_md5 (RetroCoreDescriptor *self,
+ const gchar *firmware,
+ GError **error);
+gchar * retro_core_descriptor_get_firmware_sha512 (RetroCoreDescriptor *self,
+ const gchar *firmware,
+ GError **error);
+gboolean retro_core_descriptor_get_is_firmware_mandatory (RetroCoreDescriptor *self,
+ const gchar *firmware,
+ GError **error);
+gboolean retro_core_descriptor_get_platform_supports_mime_types (RetroCoreDescriptor *self,
+ const gchar *platform,
+ const gchar * const *mime_types,
+ GError **error);
+
+G_END_DECLS
+
+#endif /* RETRO_CORE_DESCRIPTOR_H */
diff --git a/retro-gtk/retro-gtk.h b/retro-gtk/retro-gtk.h
index 2bce4f5..3cdcf2c 100644
--- a/retro-gtk/retro-gtk.h
+++ b/retro-gtk/retro-gtk.h
@@ -11,6 +11,7 @@
#include "retro-analog-id.h"
#include "retro-analog-index.h"
+#include "retro-core-descriptor.h"
#include "retro-core-view.h"
#include "retro-device-type.h"
#include "retro-gtk-vala.h"
diff --git a/retro-gtk/retro-module-iterator.h b/retro-gtk/retro-module-iterator.h
index 300c04f..ad85356 100644
--- a/retro-gtk/retro-module-iterator.h
+++ b/retro-gtk/retro-module-iterator.h
@@ -8,12 +8,10 @@
#endif
#include <glib-object.h>
+#include "retro-core-descriptor.h"
G_BEGIN_DECLS
-// FIXME Remove as soon as possible.
-typedef struct _RetroCoreDescriptor RetroCoreDescriptor;
-
#define RETRO_TYPE_MODULE_ITERATOR (retro_module_iterator_get_type())
G_DECLARE_FINAL_TYPE (RetroModuleIterator, retro_module_iterator, RETRO, MODULE_ITERATOR, GObject)
[
Date Prev][
Date Next] [
Thread Prev][
Thread Next]
[
Thread Index]
[
Date Index]
[
Author Index]