mirror of
https://gitlab.gnome.org/GNOME/glib.git
synced 2025-01-16 09:16:15 +01:00
Merge branch 'wip/baedert/gparam' into 'master'
More small GKeyFile performance improvements See merge request GNOME/glib!1832
This commit is contained in:
commit
9a7ca661a3
@ -477,9 +477,7 @@ g_io_modules_scan_all_in_directory_with_scope (const char *dirname,
|
|||||||
|
|
||||||
filename = g_build_filename (dirname, "giomodule.cache", NULL);
|
filename = g_build_filename (dirname, "giomodule.cache", NULL);
|
||||||
|
|
||||||
cache = g_hash_table_new_full (g_str_hash, g_str_equal,
|
cache = NULL;
|
||||||
g_free, (GDestroyNotify)g_strfreev);
|
|
||||||
|
|
||||||
cache_time = 0;
|
cache_time = 0;
|
||||||
if (g_stat (filename, &statbuf) == 0 &&
|
if (g_stat (filename, &statbuf) == 0 &&
|
||||||
g_file_get_contents (filename, &data, NULL, NULL))
|
g_file_get_contents (filename, &data, NULL, NULL))
|
||||||
@ -523,6 +521,10 @@ g_io_modules_scan_all_in_directory_with_scope (const char *dirname,
|
|||||||
while (g_ascii_isspace (*colon))
|
while (g_ascii_isspace (*colon))
|
||||||
colon++;
|
colon++;
|
||||||
|
|
||||||
|
if (G_UNLIKELY (!cache))
|
||||||
|
cache = g_hash_table_new_full (g_str_hash, g_str_equal,
|
||||||
|
g_free, (GDestroyNotify)g_strfreev);
|
||||||
|
|
||||||
extension_points = g_strsplit (colon, ",", -1);
|
extension_points = g_strsplit (colon, ",", -1);
|
||||||
g_hash_table_insert (cache, file, extension_points);
|
g_hash_table_insert (cache, file, extension_points);
|
||||||
}
|
}
|
||||||
@ -536,13 +538,15 @@ g_io_modules_scan_all_in_directory_with_scope (const char *dirname,
|
|||||||
GIOExtensionPoint *extension_point;
|
GIOExtensionPoint *extension_point;
|
||||||
GIOModule *module;
|
GIOModule *module;
|
||||||
gchar *path;
|
gchar *path;
|
||||||
char **extension_points;
|
char **extension_points = NULL;
|
||||||
int i;
|
int i;
|
||||||
|
|
||||||
path = g_build_filename (dirname, name, NULL);
|
path = g_build_filename (dirname, name, NULL);
|
||||||
module = g_io_module_new (path);
|
module = g_io_module_new (path);
|
||||||
|
|
||||||
|
if (cache)
|
||||||
extension_points = g_hash_table_lookup (cache, name);
|
extension_points = g_hash_table_lookup (cache, name);
|
||||||
|
|
||||||
if (extension_points != NULL &&
|
if (extension_points != NULL &&
|
||||||
g_stat (path, &statbuf) == 0 &&
|
g_stat (path, &statbuf) == 0 &&
|
||||||
statbuf.st_ctime <= cache_time)
|
statbuf.st_ctime <= cache_time)
|
||||||
@ -577,6 +581,7 @@ g_io_modules_scan_all_in_directory_with_scope (const char *dirname,
|
|||||||
|
|
||||||
g_dir_close (dir);
|
g_dir_close (dir);
|
||||||
|
|
||||||
|
if (cache)
|
||||||
g_hash_table_destroy (cache);
|
g_hash_table_destroy (cache);
|
||||||
|
|
||||||
g_free (filename);
|
g_free (filename);
|
||||||
|
@ -929,6 +929,7 @@ g_resource_enumerate_children (GResource *resource,
|
|||||||
|
|
||||||
if (*path == 0)
|
if (*path == 0)
|
||||||
{
|
{
|
||||||
|
if (error)
|
||||||
g_set_error (error, G_RESOURCE_ERROR, G_RESOURCE_ERROR_NOT_FOUND,
|
g_set_error (error, G_RESOURCE_ERROR, G_RESOURCE_ERROR_NOT_FOUND,
|
||||||
_("The resource at “%s” does not exist"),
|
_("The resource at “%s” does not exist"),
|
||||||
path);
|
path);
|
||||||
@ -968,6 +969,7 @@ g_resource_enumerate_children (GResource *resource,
|
|||||||
|
|
||||||
if (children == NULL)
|
if (children == NULL)
|
||||||
{
|
{
|
||||||
|
if (error)
|
||||||
g_set_error (error, G_RESOURCE_ERROR, G_RESOURCE_ERROR_NOT_FOUND,
|
g_set_error (error, G_RESOURCE_ERROR, G_RESOURCE_ERROR_NOT_FOUND,
|
||||||
_("The resource at “%s” does not exist"),
|
_("The resource at “%s” does not exist"),
|
||||||
path);
|
path);
|
||||||
@ -1237,6 +1239,7 @@ g_resources_enumerate_children (const gchar *path,
|
|||||||
|
|
||||||
if (hash == NULL)
|
if (hash == NULL)
|
||||||
{
|
{
|
||||||
|
if (error)
|
||||||
g_set_error (error, G_RESOURCE_ERROR, G_RESOURCE_ERROR_NOT_FOUND,
|
g_set_error (error, G_RESOURCE_ERROR, G_RESOURCE_ERROR_NOT_FOUND,
|
||||||
_("The resource at “%s” does not exist"),
|
_("The resource at “%s” does not exist"),
|
||||||
path);
|
path);
|
||||||
|
@ -897,6 +897,8 @@ get_contents_posix (const gchar *filename,
|
|||||||
if (fd < 0)
|
if (fd < 0)
|
||||||
{
|
{
|
||||||
int saved_errno = errno;
|
int saved_errno = errno;
|
||||||
|
|
||||||
|
if (error)
|
||||||
set_file_error (error,
|
set_file_error (error,
|
||||||
filename,
|
filename,
|
||||||
_("Failed to open file “%s”: %s"),
|
_("Failed to open file “%s”: %s"),
|
||||||
@ -909,6 +911,7 @@ get_contents_posix (const gchar *filename,
|
|||||||
if (fstat (fd, &stat_buf) < 0)
|
if (fstat (fd, &stat_buf) < 0)
|
||||||
{
|
{
|
||||||
int saved_errno = errno;
|
int saved_errno = errno;
|
||||||
|
if (error)
|
||||||
set_file_error (error,
|
set_file_error (error,
|
||||||
filename,
|
filename,
|
||||||
_("Failed to get attributes of file “%s”: fstat() failed: %s"),
|
_("Failed to get attributes of file “%s”: fstat() failed: %s"),
|
||||||
@ -939,6 +942,7 @@ get_contents_posix (const gchar *filename,
|
|||||||
if (f == NULL)
|
if (f == NULL)
|
||||||
{
|
{
|
||||||
int saved_errno = errno;
|
int saved_errno = errno;
|
||||||
|
if (error)
|
||||||
set_file_error (error,
|
set_file_error (error,
|
||||||
filename,
|
filename,
|
||||||
_("Failed to open file “%s”: fdopen() failed: %s"),
|
_("Failed to open file “%s”: fdopen() failed: %s"),
|
||||||
@ -969,6 +973,7 @@ get_contents_win32 (const gchar *filename,
|
|||||||
if (f == NULL)
|
if (f == NULL)
|
||||||
{
|
{
|
||||||
int saved_errno = errno;
|
int saved_errno = errno;
|
||||||
|
if (error)
|
||||||
set_file_error (error,
|
set_file_error (error,
|
||||||
filename,
|
filename,
|
||||||
_("Failed to open file “%s”: %s"),
|
_("Failed to open file “%s”: %s"),
|
||||||
@ -1165,6 +1170,7 @@ write_to_file (const gchar *contents,
|
|||||||
if (saved_errno == EINTR)
|
if (saved_errno == EINTR)
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
|
if (err)
|
||||||
set_file_error (err,
|
set_file_error (err,
|
||||||
dest_file, _("Failed to write file “%s”: write() failed: %s"),
|
dest_file, _("Failed to write file “%s”: write() failed: %s"),
|
||||||
saved_errno);
|
saved_errno);
|
||||||
@ -1185,6 +1191,7 @@ write_to_file (const gchar *contents,
|
|||||||
if (do_fsync && g_fsync (fd) != 0)
|
if (do_fsync && g_fsync (fd) != 0)
|
||||||
{
|
{
|
||||||
int saved_errno = errno;
|
int saved_errno = errno;
|
||||||
|
if (err)
|
||||||
set_file_error (err,
|
set_file_error (err,
|
||||||
dest_file, _("Failed to write file “%s”: fsync() failed: %s"),
|
dest_file, _("Failed to write file “%s”: fsync() failed: %s"),
|
||||||
saved_errno);
|
saved_errno);
|
||||||
@ -1353,6 +1360,7 @@ g_file_set_contents_full (const gchar *filename,
|
|||||||
if (fd == -1)
|
if (fd == -1)
|
||||||
{
|
{
|
||||||
int saved_errno = errno;
|
int saved_errno = errno;
|
||||||
|
if (error)
|
||||||
set_file_error (error,
|
set_file_error (error,
|
||||||
tmp_filename, _("Failed to create file “%s”: %s"),
|
tmp_filename, _("Failed to create file “%s”: %s"),
|
||||||
saved_errno);
|
saved_errno);
|
||||||
@ -1396,6 +1404,7 @@ g_file_set_contents_full (const gchar *filename,
|
|||||||
if (g_unlink (filename) == -1)
|
if (g_unlink (filename) == -1)
|
||||||
{
|
{
|
||||||
int saved_errno = errno;
|
int saved_errno = errno;
|
||||||
|
if (error)
|
||||||
set_file_error (error,
|
set_file_error (error,
|
||||||
filename,
|
filename,
|
||||||
_("Existing file “%s” could not be removed: g_unlink() failed: %s"),
|
_("Existing file “%s” could not be removed: g_unlink() failed: %s"),
|
||||||
@ -1462,6 +1471,7 @@ consistent_out:
|
|||||||
mode, error);
|
mode, error);
|
||||||
#endif /* O_NOFOLLOW */
|
#endif /* O_NOFOLLOW */
|
||||||
|
|
||||||
|
if (error)
|
||||||
set_file_error (error,
|
set_file_error (error,
|
||||||
filename, _("Failed to open file “%s”: %s"),
|
filename, _("Failed to open file “%s”: %s"),
|
||||||
saved_errno);
|
saved_errno);
|
||||||
@ -1764,6 +1774,7 @@ g_get_tmp_name (const gchar *tmpl,
|
|||||||
if (retval == -1)
|
if (retval == -1)
|
||||||
{
|
{
|
||||||
int saved_errno = errno;
|
int saved_errno = errno;
|
||||||
|
if (error)
|
||||||
set_file_error (error,
|
set_file_error (error,
|
||||||
fulltemplate,
|
fulltemplate,
|
||||||
_("Failed to create file “%s”: %s"),
|
_("Failed to create file “%s”: %s"),
|
||||||
@ -2301,6 +2312,7 @@ g_file_read_link (const gchar *filename,
|
|||||||
if (read_size < 0)
|
if (read_size < 0)
|
||||||
{
|
{
|
||||||
int saved_errno = errno;
|
int saved_errno = errno;
|
||||||
|
if (error)
|
||||||
set_file_error (error,
|
set_file_error (error,
|
||||||
filename,
|
filename,
|
||||||
_("Failed to read the symbolic link “%s”: %s"),
|
_("Failed to read the symbolic link “%s”: %s"),
|
||||||
@ -2329,6 +2341,7 @@ g_file_read_link (const gchar *filename,
|
|||||||
if (read_size < 0)
|
if (read_size < 0)
|
||||||
{
|
{
|
||||||
int saved_errno = errno;
|
int saved_errno = errno;
|
||||||
|
if (error)
|
||||||
set_file_error (error,
|
set_file_error (error,
|
||||||
filename,
|
filename,
|
||||||
_("Failed to read the symbolic link “%s”: %s"),
|
_("Failed to read the symbolic link “%s”: %s"),
|
||||||
|
@ -631,9 +631,9 @@ g_key_file_init (GKeyFile *key_file)
|
|||||||
{
|
{
|
||||||
key_file->current_group = g_slice_new0 (GKeyFileGroup);
|
key_file->current_group = g_slice_new0 (GKeyFileGroup);
|
||||||
key_file->groups = g_list_prepend (NULL, key_file->current_group);
|
key_file->groups = g_list_prepend (NULL, key_file->current_group);
|
||||||
key_file->group_hash = g_hash_table_new (g_str_hash, g_str_equal);
|
key_file->group_hash = NULL;
|
||||||
key_file->start_group = NULL;
|
key_file->start_group = NULL;
|
||||||
key_file->parse_buffer = g_string_sized_new (128);
|
key_file->parse_buffer = NULL;
|
||||||
key_file->list_separator = ';';
|
key_file->list_separator = ';';
|
||||||
key_file->flags = 0;
|
key_file->flags = 0;
|
||||||
}
|
}
|
||||||
@ -1473,6 +1473,9 @@ g_key_file_parse_data (GKeyFile *key_file,
|
|||||||
|
|
||||||
parse_error = NULL;
|
parse_error = NULL;
|
||||||
|
|
||||||
|
if (!key_file->parse_buffer)
|
||||||
|
key_file->parse_buffer = g_string_sized_new (128);
|
||||||
|
|
||||||
i = 0;
|
i = 0;
|
||||||
while (i < length)
|
while (i < length)
|
||||||
{
|
{
|
||||||
@ -1529,6 +1532,9 @@ g_key_file_flush_parse_buffer (GKeyFile *key_file,
|
|||||||
|
|
||||||
g_return_if_fail (key_file != NULL);
|
g_return_if_fail (key_file != NULL);
|
||||||
|
|
||||||
|
if (!key_file->parse_buffer)
|
||||||
|
return;
|
||||||
|
|
||||||
file_error = NULL;
|
file_error = NULL;
|
||||||
|
|
||||||
if (key_file->parse_buffer->len > 0)
|
if (key_file->parse_buffer->len > 0)
|
||||||
@ -3831,6 +3837,9 @@ g_key_file_add_group (GKeyFile *key_file,
|
|||||||
if (key_file->start_group == NULL)
|
if (key_file->start_group == NULL)
|
||||||
key_file->start_group = group;
|
key_file->start_group = group;
|
||||||
|
|
||||||
|
if (!key_file->group_hash)
|
||||||
|
key_file->group_hash = g_hash_table_new (g_str_hash, g_str_equal);
|
||||||
|
|
||||||
g_hash_table_insert (key_file->group_hash, (gpointer)group->name, group);
|
g_hash_table_insert (key_file->group_hash, (gpointer)group->name, group);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -3883,7 +3892,10 @@ g_key_file_remove_group_node (GKeyFile *key_file,
|
|||||||
group = (GKeyFileGroup *) group_node->data;
|
group = (GKeyFileGroup *) group_node->data;
|
||||||
|
|
||||||
if (group->name)
|
if (group->name)
|
||||||
|
{
|
||||||
|
g_assert (key_file->group_hash);
|
||||||
g_hash_table_remove (key_file->group_hash, group->name);
|
g_hash_table_remove (key_file->group_hash, group->name);
|
||||||
|
}
|
||||||
|
|
||||||
/* If the current group gets deleted make the current group the last
|
/* If the current group gets deleted make the current group the last
|
||||||
* added group.
|
* added group.
|
||||||
@ -4089,6 +4101,9 @@ static GKeyFileGroup *
|
|||||||
g_key_file_lookup_group (GKeyFile *key_file,
|
g_key_file_lookup_group (GKeyFile *key_file,
|
||||||
const gchar *group_name)
|
const gchar *group_name)
|
||||||
{
|
{
|
||||||
|
if (!key_file->group_hash)
|
||||||
|
return NULL;
|
||||||
|
|
||||||
return (GKeyFileGroup *)g_hash_table_lookup (key_file->group_hash, group_name);
|
return (GKeyFileGroup *)g_hash_table_lookup (key_file->group_hash, group_name);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1203,20 +1203,12 @@ pspec_compare_id (gconstpointer a,
|
|||||||
return strcmp (pspec1->name, pspec2->name);
|
return strcmp (pspec1->name, pspec2->name);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline GSList*
|
static inline gboolean
|
||||||
pspec_list_remove_overridden_and_redirected (GSList *plist,
|
should_list_pspec (GParamSpec *pspec,
|
||||||
GHashTable *ht,
|
|
||||||
GType owner_type,
|
GType owner_type,
|
||||||
guint *n_p)
|
GHashTable *ht)
|
||||||
{
|
{
|
||||||
GSList *rlist = NULL;
|
|
||||||
|
|
||||||
while (plist)
|
|
||||||
{
|
|
||||||
GSList *tmp = plist->next;
|
|
||||||
GParamSpec *pspec = plist->data;
|
|
||||||
GParamSpec *found;
|
GParamSpec *found;
|
||||||
gboolean remove = FALSE;
|
|
||||||
|
|
||||||
/* Remove paramspecs that are redirected, and also paramspecs
|
/* Remove paramspecs that are redirected, and also paramspecs
|
||||||
* that have are overridden by non-redirected properties.
|
* that have are overridden by non-redirected properties.
|
||||||
@ -1224,31 +1216,17 @@ pspec_list_remove_overridden_and_redirected (GSList *plist,
|
|||||||
* best corresponds to what the application sees.
|
* best corresponds to what the application sees.
|
||||||
*/
|
*/
|
||||||
if (g_param_spec_get_redirect_target (pspec))
|
if (g_param_spec_get_redirect_target (pspec))
|
||||||
remove = TRUE;
|
return FALSE;
|
||||||
else
|
|
||||||
{
|
|
||||||
found = param_spec_ht_lookup (ht, pspec->name, owner_type, TRUE);
|
found = param_spec_ht_lookup (ht, pspec->name, owner_type, TRUE);
|
||||||
if (found != pspec)
|
if (found != pspec)
|
||||||
{
|
{
|
||||||
GParamSpec *redirect = g_param_spec_get_redirect_target (found);
|
GParamSpec *redirect = g_param_spec_get_redirect_target (found);
|
||||||
if (redirect != pspec)
|
if (redirect != pspec)
|
||||||
remove = TRUE;
|
return FALSE;
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (remove)
|
return TRUE;
|
||||||
{
|
|
||||||
g_slist_free_1 (plist);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
plist->next = rlist;
|
|
||||||
rlist = plist;
|
|
||||||
*n_p += 1;
|
|
||||||
}
|
|
||||||
plist = tmp;
|
|
||||||
}
|
|
||||||
return rlist;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
@ -1260,18 +1238,23 @@ pool_depth_list (gpointer key,
|
|||||||
gpointer *data = user_data;
|
gpointer *data = user_data;
|
||||||
GSList **slists = data[0];
|
GSList **slists = data[0];
|
||||||
GType owner_type = (GType) data[1];
|
GType owner_type = (GType) data[1];
|
||||||
|
GHashTable *ht = data[2];
|
||||||
|
int *count = data[3];
|
||||||
|
|
||||||
if (g_type_is_a (owner_type, pspec->owner_type))
|
if (g_type_is_a (owner_type, pspec->owner_type) &&
|
||||||
|
should_list_pspec (pspec, owner_type, ht))
|
||||||
{
|
{
|
||||||
if (G_TYPE_IS_INTERFACE (pspec->owner_type))
|
if (G_TYPE_IS_INTERFACE (pspec->owner_type))
|
||||||
{
|
{
|
||||||
slists[0] = g_slist_prepend (slists[0], pspec);
|
slists[0] = g_slist_prepend (slists[0], pspec);
|
||||||
|
*count = *count + 1;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
guint d = g_type_depth (pspec->owner_type);
|
guint d = g_type_depth (pspec->owner_type);
|
||||||
|
|
||||||
slists[d - 1] = g_slist_prepend (slists[d - 1], pspec);
|
slists[d - 1] = g_slist_prepend (slists[d - 1], pspec);
|
||||||
|
*count = *count + 1;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -1294,9 +1277,15 @@ pool_depth_list_for_interface (gpointer key,
|
|||||||
gpointer *data = user_data;
|
gpointer *data = user_data;
|
||||||
GSList **slists = data[0];
|
GSList **slists = data[0];
|
||||||
GType owner_type = (GType) data[1];
|
GType owner_type = (GType) data[1];
|
||||||
|
GHashTable *ht = data[2];
|
||||||
|
int *count = data[3];
|
||||||
|
|
||||||
if (pspec->owner_type == owner_type)
|
if (pspec->owner_type == owner_type &&
|
||||||
|
should_list_pspec (pspec, owner_type, ht))
|
||||||
|
{
|
||||||
slists[0] = g_slist_prepend (slists[0], pspec);
|
slists[0] = g_slist_prepend (slists[0], pspec);
|
||||||
|
*count = *count + 1;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -1319,19 +1308,21 @@ g_param_spec_pool_list (GParamSpecPool *pool,
|
|||||||
{
|
{
|
||||||
GParamSpec **pspecs, **p;
|
GParamSpec **pspecs, **p;
|
||||||
GSList **slists, *node;
|
GSList **slists, *node;
|
||||||
gpointer data[2];
|
gpointer data[4];
|
||||||
guint d, i;
|
guint d, i;
|
||||||
|
int n_pspecs = 0;
|
||||||
|
|
||||||
g_return_val_if_fail (pool != NULL, NULL);
|
g_return_val_if_fail (pool != NULL, NULL);
|
||||||
g_return_val_if_fail (owner_type > 0, NULL);
|
g_return_val_if_fail (owner_type > 0, NULL);
|
||||||
g_return_val_if_fail (n_pspecs_p != NULL, NULL);
|
g_return_val_if_fail (n_pspecs_p != NULL, NULL);
|
||||||
|
|
||||||
g_mutex_lock (&pool->mutex);
|
g_mutex_lock (&pool->mutex);
|
||||||
*n_pspecs_p = 0;
|
|
||||||
d = g_type_depth (owner_type);
|
d = g_type_depth (owner_type);
|
||||||
slists = g_new0 (GSList*, d);
|
slists = g_new0 (GSList*, d);
|
||||||
data[0] = slists;
|
data[0] = slists;
|
||||||
data[1] = (gpointer) owner_type;
|
data[1] = (gpointer) owner_type;
|
||||||
|
data[2] = pool->hash_table;
|
||||||
|
data[3] = &n_pspecs;
|
||||||
|
|
||||||
g_hash_table_foreach (pool->hash_table,
|
g_hash_table_foreach (pool->hash_table,
|
||||||
G_TYPE_IS_INTERFACE (owner_type) ?
|
G_TYPE_IS_INTERFACE (owner_type) ?
|
||||||
@ -1339,9 +1330,7 @@ g_param_spec_pool_list (GParamSpecPool *pool,
|
|||||||
pool_depth_list,
|
pool_depth_list,
|
||||||
&data);
|
&data);
|
||||||
|
|
||||||
for (i = 0; i < d; i++)
|
pspecs = g_new (GParamSpec*, n_pspecs + 1);
|
||||||
slists[i] = pspec_list_remove_overridden_and_redirected (slists[i], pool->hash_table, owner_type, n_pspecs_p);
|
|
||||||
pspecs = g_new (GParamSpec*, *n_pspecs_p + 1);
|
|
||||||
p = pspecs;
|
p = pspecs;
|
||||||
for (i = 0; i < d; i++)
|
for (i = 0; i < d; i++)
|
||||||
{
|
{
|
||||||
@ -1354,10 +1343,11 @@ g_param_spec_pool_list (GParamSpecPool *pool,
|
|||||||
g_free (slists);
|
g_free (slists);
|
||||||
g_mutex_unlock (&pool->mutex);
|
g_mutex_unlock (&pool->mutex);
|
||||||
|
|
||||||
|
*n_pspecs_p = n_pspecs;
|
||||||
|
|
||||||
return pspecs;
|
return pspecs;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/* --- auxiliary functions --- */
|
/* --- auxiliary functions --- */
|
||||||
typedef struct
|
typedef struct
|
||||||
{
|
{
|
||||||
|
Loading…
Reference in New Issue
Block a user