mirror of
https://gitlab.gnome.org/GNOME/glib.git
synced 2025-09-07 00:18:44 +02:00
Add an mmap() wrapper called GMappedFile. (#148218, David Schleef, Behdad
2005-06-24 Matthias Clasen <mclasen@redhat.com> Add an mmap() wrapper called GMappedFile. (#148218, David Schleef, Behdad Esfahbod) * glib/gmappedfile.[hc]: New files. * configure.in: Check for mmap. * glib/Makefile.am: Add new files. * glib/glib.symbols: Add new functions. * glib/glib.h: Include gmappedfile.h * tests/mapping-test.c: Tests for GMappedFile. * tests/Makefile.am: Add new file.
This commit is contained in:
committed by
Matthias Clasen
parent
132e20bcb6
commit
748c230205
@@ -77,6 +77,7 @@ test_programs = \
|
||||
keyfile-test \
|
||||
list-test \
|
||||
mainloop-test \
|
||||
mapping-test \
|
||||
markup-escape-test \
|
||||
module-test \
|
||||
node-test \
|
||||
@@ -133,6 +134,7 @@ keyfile_test_LDADD = $(progs_ldadd)
|
||||
list_test_LDADD = $(progs_ldadd)
|
||||
mainloop_test_LDADD = $(thread_ldadd)
|
||||
markup_test_LDADD = $(progs_ldadd)
|
||||
mapping_test_LDADD = $(progs_ldadd)
|
||||
markup_escape_test_LDADD = $(progs_ldadd)
|
||||
module_test_LDADD = $(module_ldadd) $(module_test_exp)
|
||||
module_test_LDFLAGS = $(G_MODULE_LDFLAGS)
|
||||
|
233
tests/mapping-test.c
Normal file
233
tests/mapping-test.c
Normal file
@@ -0,0 +1,233 @@
|
||||
/* GLIB - Library of useful routines for C programming
|
||||
* Copyright (C) 2005 Matthias Clasen
|
||||
*
|
||||
* This library is free software; you can redistribute it and/or
|
||||
* modify it under the terms of the GNU Lesser General Public
|
||||
* License as published by the Free Software Foundation; either
|
||||
* version 2 of the License, or (at your option) any later version.
|
||||
*
|
||||
* This library 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
|
||||
* Lesser General Public License for more details.
|
||||
*
|
||||
* You should have received a copy of the GNU Lesser General Public
|
||||
* License along with this library; if not, write to the
|
||||
* Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
||||
* Boston, MA 02111-1307, USA.
|
||||
*/
|
||||
|
||||
#include <stdlib.h>
|
||||
#include <string.h>
|
||||
#include <unistd.h>
|
||||
#include <sys/types.h>
|
||||
#include <signal.h>
|
||||
|
||||
#include "glib.h"
|
||||
|
||||
static gchar *dir, *filename, *displayname, *childname;
|
||||
|
||||
static gboolean stop = FALSE;
|
||||
|
||||
static void
|
||||
handle_usr1 (int signum)
|
||||
{
|
||||
stop = TRUE;
|
||||
}
|
||||
|
||||
static gboolean
|
||||
check_stop (gpointer data)
|
||||
{
|
||||
GMainLoop *loop = data;
|
||||
|
||||
if (stop)
|
||||
g_main_loop_quit (loop);
|
||||
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
static void
|
||||
write_or_die (const gchar *filename,
|
||||
const gchar *contents,
|
||||
gssize length)
|
||||
{
|
||||
GError *error = NULL;
|
||||
gchar *displayname;
|
||||
|
||||
if (!g_file_set_contents (filename, contents, length, &error))
|
||||
{
|
||||
displayname = g_filename_display_name (childname);
|
||||
g_print ("failed to write '%s': %s\n",
|
||||
displayname, error->message);
|
||||
exit (1);
|
||||
}
|
||||
}
|
||||
|
||||
static GMappedFile *
|
||||
map_or_die (const gchar *filename,
|
||||
gboolean writable)
|
||||
{
|
||||
GError *error = NULL;
|
||||
GMappedFile *map;
|
||||
gchar *displayname;
|
||||
|
||||
map = g_mapped_file_new (filename, writable, &error);
|
||||
if (!map)
|
||||
{
|
||||
displayname = g_filename_display_name (childname);
|
||||
g_print ("failed to map '%s' non-writable, shared: %s\n",
|
||||
displayname, error->message);
|
||||
exit (1);
|
||||
}
|
||||
|
||||
return map;
|
||||
}
|
||||
|
||||
static int
|
||||
child_main (int argc, char *argv[])
|
||||
{
|
||||
GMappedFile *map;
|
||||
GMainLoop *loop;
|
||||
|
||||
map = map_or_die (filename, FALSE);
|
||||
|
||||
loop = g_main_loop_new (NULL, FALSE);
|
||||
|
||||
signal (SIGUSR1, handle_usr1);
|
||||
g_idle_add (check_stop, loop);
|
||||
g_main_loop_run (loop);
|
||||
|
||||
write_or_die (childname,
|
||||
g_mapped_file_get_contents (map),
|
||||
g_mapped_file_get_length (map));
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void
|
||||
test_mapping (void)
|
||||
{
|
||||
GMappedFile *map;
|
||||
|
||||
write_or_die (filename, "ABC", -1);
|
||||
|
||||
map = map_or_die (filename, FALSE);
|
||||
g_assert (g_mapped_file_get_length (map) == 3);
|
||||
g_mapped_file_free (map);
|
||||
|
||||
map = map_or_die (filename, TRUE);
|
||||
g_assert (g_mapped_file_get_length (map) == 3);
|
||||
g_mapped_file_free (map);
|
||||
}
|
||||
|
||||
static void
|
||||
test_private (void)
|
||||
{
|
||||
GError *error = NULL;
|
||||
GMappedFile *map;
|
||||
gchar *buffer;
|
||||
gsize len;
|
||||
|
||||
write_or_die (filename, "ABC", -1);
|
||||
map = map_or_die (filename, TRUE);
|
||||
|
||||
buffer = (gchar *)g_mapped_file_get_contents (map);
|
||||
buffer[0] = '1';
|
||||
buffer[1] = '2';
|
||||
buffer[2] = '3';
|
||||
g_mapped_file_free (map);
|
||||
|
||||
if (!g_file_get_contents (filename, &buffer, &len, &error))
|
||||
{
|
||||
g_print ("failed to read '%s': %s\n",
|
||||
displayname, error->message);
|
||||
exit (1);
|
||||
|
||||
}
|
||||
g_assert (len == 3);
|
||||
g_assert (strcmp (buffer, "ABC") == 0);
|
||||
g_free (buffer);
|
||||
|
||||
}
|
||||
|
||||
static void
|
||||
test_child_private (gchar *argv0)
|
||||
{
|
||||
GError *error = NULL;
|
||||
GMappedFile *map;
|
||||
gchar *buffer;
|
||||
gsize len;
|
||||
gchar *child_argv[3];
|
||||
GPid child_pid;
|
||||
|
||||
write_or_die (filename, "ABC", -1);
|
||||
map = map_or_die (filename, TRUE);
|
||||
|
||||
child_argv[0] = argv0;
|
||||
child_argv[1] = "mapchild";
|
||||
child_argv[2] = NULL;
|
||||
if (!g_spawn_async (dir, child_argv, NULL,
|
||||
0, NULL, NULL, &child_pid, &error))
|
||||
{
|
||||
g_print ("failed to spawn child: %s\n",
|
||||
error->message);
|
||||
exit (1);
|
||||
}
|
||||
|
||||
/* give the child some time to set up its mapping */
|
||||
sleep (2);
|
||||
|
||||
buffer = (gchar *)g_mapped_file_get_contents (map);
|
||||
buffer[0] = '1';
|
||||
buffer[1] = '2';
|
||||
buffer[2] = '3';
|
||||
g_mapped_file_free (map);
|
||||
|
||||
kill (child_pid, SIGUSR1);
|
||||
|
||||
/* give the child some time to write the file */
|
||||
sleep (2);
|
||||
|
||||
if (!g_file_get_contents (childname, &buffer, &len, &error))
|
||||
{
|
||||
gchar *name;
|
||||
|
||||
name = g_filename_display_name (childname);
|
||||
g_print ("failed to read '%s': %s\n", name, error->message);
|
||||
exit (1);
|
||||
}
|
||||
g_assert (len == 3);
|
||||
g_assert (strcmp (buffer, "ABC") == 0);
|
||||
g_free (buffer);
|
||||
}
|
||||
|
||||
static int
|
||||
parent_main (int argc,
|
||||
char *argv[])
|
||||
{
|
||||
/* test mapping with various flag combinations */
|
||||
test_mapping ();
|
||||
|
||||
/* test private modification */
|
||||
test_private ();
|
||||
|
||||
/* test multiple clients, non-shared */
|
||||
test_child_private (argv[0]);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
int
|
||||
main (int argc,
|
||||
char *argv[])
|
||||
{
|
||||
dir = g_get_current_dir ();
|
||||
filename = g_build_filename (dir, "maptest", NULL);
|
||||
displayname = g_filename_display_name (filename);
|
||||
childname = g_build_filename (dir, "mapchild", NULL);
|
||||
|
||||
if (argc > 1)
|
||||
return child_main (argc, argv);
|
||||
else
|
||||
return parent_main (argc, argv);
|
||||
}
|
Reference in New Issue
Block a user