glib/gio/tests/unix-fd.c
Philip Withnall 07c4b6c68e glib: Pass O_CLOEXEC rather than FD_CLOEXEC to g_unix_open_pipe()
See the previous commit.

Signed-off-by: Philip Withnall <philip@tecnocode.co.uk>
2023-08-16 14:31:40 +01:00

305 lines
7.6 KiB
C
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

#include <gio/gio.h>
#include <gio/gnetworking.h>
#include <gio/gunixfdmessage.h>
#include <gio/gunixsocketaddress.h>
#ifdef G_OS_UNIX
#include <glib-unix.h>
#include <unistd.h>
#endif
#include <glib/gstdio.h>
#include <fcntl.h>
#include <string.h>
#ifdef G_OS_WIN32
#include <io.h>
#endif
/* ensures that no FDs are left open at the end */
static void
check_fd_list (const gint *fd_list)
{
gint i;
for (i = 0; i < 40; i++)
{
int my_fd;
my_fd = dup (0);
g_assert_cmpint (fd_list[i], ==, my_fd);
}
for (i = 0; i < 40; i++)
g_close (fd_list[i], NULL);
}
static void
create_fd_list (gint *fd_list)
{
gint i;
for (i = 0; i < 40; i++)
{
fd_list[i] = dup (0);
g_assert_cmpint (fd_list[i], >, 0);
}
for (i = 0; i < 40; i++)
g_close (fd_list[i], NULL);
}
static void
test_fd_list (void)
{
GError *err = NULL;
GUnixFDList *list;
const gint *peek;
gint *stolen;
gint fd_list[40];
gint sv[3];
gint s;
create_fd_list (fd_list);
sv[2] = -1;
#ifdef G_OS_WIN32
s = _pipe (sv, 4096, _O_NOINHERIT | _O_BINARY);
g_assert_cmpint (s, ==, 0);
#else
g_unix_open_pipe (sv, O_CLOEXEC, &err);
g_assert_no_error (err);
#endif
list = g_unix_fd_list_new_from_array (sv, -1);
peek = g_unix_fd_list_peek_fds (list, &s);
g_assert_cmpint (s, ==, 2);
g_assert_cmpint (peek[0], ==, sv[0]);
g_assert_cmpint (peek[1], ==, sv[1]);
s = g_unix_fd_list_get (list, 0, &err);
g_assert_no_error (err);
g_close (s, &err);
g_assert_no_error (err);
s = g_unix_fd_list_get (list, 1, &err);
g_assert_no_error (err);
g_close (s, &err);
g_assert_no_error (err);
s = g_unix_fd_list_append (list, sv[0], &err);
g_assert_no_error (err);
g_assert_cmpint (s, >=, 0);
stolen = g_unix_fd_list_steal_fds (list, &s);
g_assert_cmpint (s, ==, 3);
g_assert_cmpint (stolen[0], ==, sv[0]);
g_assert_cmpint (stolen[1], ==, sv[1]);
g_assert_cmpint (stolen[2], >=, 0);
g_close (stolen[0], &err);
g_assert_no_error (err);
g_close (stolen[1], &err);
g_assert_no_error (err);
g_close (stolen[2], &err);
g_assert_no_error (err);
g_free (stolen);
g_object_unref (list);
check_fd_list (fd_list);
}
static void
test_scm (void)
{
#ifndef G_OS_WIN32
GError *err = NULL;
GUnixFDMessage *message;
GUnixFDMessage **mv;
GUnixFDList *list, *l2;
GSocket *sockets[2];
GSocketAddress *addr;
const gint *peek;
char buffer[1024];
gint fd_list[40];
GOutputVector ov;
GInputVector iv;
gint *stolen;
gint sv[3];
gint flags;
gint nm;
gint s, i;
gchar *path;
GByteArray *array;
gboolean abstract;
GUnixSocketAddressType type;
create_fd_list (fd_list);
s = socketpair (PF_UNIX, SOCK_STREAM, 0, sv);
g_assert_cmpint (s, ==, 0);
sv[2] = -1;
list = g_unix_fd_list_new_from_array (sv, -1);
message = G_UNIX_FD_MESSAGE (g_unix_fd_message_new_with_fd_list (list));
g_object_unref (list);
g_assert (g_unix_fd_message_get_fd_list (message) == list);
g_object_get (message, "fd-list", &l2, NULL);
g_assert (l2 == list);
g_assert_cmpint (g_unix_fd_list_get_length (list), ==, 2);
peek = g_unix_fd_list_peek_fds (list, &s);
g_assert_cmpint (s, ==, 2);
stolen = g_unix_fd_message_steal_fds (message, &s);
g_assert_cmpint (s, ==, 2);
g_assert (stolen == peek);
g_assert_cmpint (stolen[0], ==, sv[0]);
g_assert_cmpint (stolen[1], ==, sv[1]);
g_assert_cmpint (stolen[2], ==, sv[2]);
g_free (stolen);
g_unix_fd_message_append_fd (message, sv[0], &err);
g_assert_no_error (err);
g_close (sv[0], &err);
g_assert_no_error (err);
g_unix_fd_message_append_fd (message, sv[1], &err);
g_assert_no_error (err);
g_close (sv[1], &err);
g_assert_no_error (err);
for (i = 0; i < 3; i++)
{
s = g_unix_fd_list_get (list, 0, &err);
g_assert_no_error (err);
g_close (s, &err);
g_assert_no_error (err);
s = g_unix_fd_list_get (list, 1, &err);
g_assert_no_error (err);
g_close (s, &err);
g_assert_no_error (err);
}
g_object_unref (message);
g_object_unref (list);
message = G_UNIX_FD_MESSAGE (g_unix_fd_message_new ());
list = g_unix_fd_message_get_fd_list (message);
s = pipe (sv);
g_assert_cmpint (s, ==, 0);
s = g_unix_fd_list_append (list, sv[0], &err);
g_assert_no_error (err);
g_assert_cmpint (s, >=, 0);
s = g_unix_fd_list_append (list, sv[1], &err);
g_assert_no_error (err);
g_assert_cmpint (s, >=, 0);
g_close (sv[0], &err);
g_assert_no_error (err);
g_close (sv[1], &err);
g_assert_no_error (err);
s = g_unix_fd_list_get (list, 0, &err);
g_assert_no_error (err);
g_close (s, &err);
g_assert_no_error (err);
s = g_unix_fd_list_get (list, 1, &err);
g_assert_no_error (err);
g_close (s, &err);
g_assert_no_error (err);
s = socketpair (PF_UNIX, SOCK_STREAM, 0, sv);
g_assert_cmpint (s, ==, 0);
sockets[0] = g_socket_new_from_fd (sv[0], &err);
g_assert_no_error (err);
g_assert (G_IS_SOCKET (sockets[0]));
sockets[1] = g_socket_new_from_fd (sv[1], &err);
g_assert_no_error (err);
g_assert (G_IS_SOCKET (sockets[1]));
addr = g_socket_get_local_address (sockets[0], &err);
g_assert_no_error (err);
g_assert (G_IS_UNIX_SOCKET_ADDRESS (addr));
g_assert_cmpint (g_unix_socket_address_get_address_type (G_UNIX_SOCKET_ADDRESS (addr)), ==, G_UNIX_SOCKET_ADDRESS_ANONYMOUS);
g_assert_cmpint (g_unix_socket_address_get_path_len (G_UNIX_SOCKET_ADDRESS (addr)), ==, 0);
G_GNUC_BEGIN_IGNORE_DEPRECATIONS
g_assert (!g_unix_socket_address_get_is_abstract (G_UNIX_SOCKET_ADDRESS (addr)));
G_GNUC_END_IGNORE_DEPRECATIONS
g_object_get (addr,
"path", &path,
"path-as-array", &array,
"abstract", &abstract,
"address-type", &type,
NULL);
g_assert_cmpstr (path, ==, "");
g_assert_cmpint (array->len, ==, 0);
g_assert (!abstract);
g_assert (type == G_UNIX_SOCKET_ADDRESS_ANONYMOUS);
g_free (path);
g_byte_array_free (array, TRUE);
g_object_unref (addr);
buffer[0] = 0xff;
ov.buffer = buffer;
ov.size = 1;
s = g_socket_send_message (sockets[0], NULL, &ov, 1,
(GSocketControlMessage **) &message,
1, 0, NULL, &err);
g_assert_no_error (err);
g_assert_cmpint (s, ==, 1);
g_object_unref (message);
message = NULL;
flags = 0;
iv.buffer = buffer;
iv.size = 1;
s = g_socket_receive_message (sockets[1], NULL, &iv, 1,
(GSocketControlMessage ***) &mv,
&nm, &flags, NULL, &err);
g_assert_no_error (err);
g_assert_cmpint (s, ==, 1);
g_object_unref (sockets[0]);
g_object_unref (sockets[1]);
g_assert_cmpint (nm, ==, 1);
message = mv[0];
g_free (mv);
g_assert (G_IS_UNIX_FD_MESSAGE (message));
list = g_object_ref (g_unix_fd_message_get_fd_list (message));
g_object_unref (message);
peek = g_unix_fd_list_peek_fds (list, &s);
g_assert_cmpint (s, ==, 2);
sv[0] = g_unix_fd_list_get (list, 1, &err);
g_assert_no_error (err);
strcpy (buffer, "failure to say failure to say 'i love gnome-panel!'.");
s = write (sv[0], buffer, strlen (buffer) + 1);
g_assert_cmpint (s, ==, strlen (buffer) + 1);
g_close (sv[0], NULL);
memset (buffer, 0xff, sizeof buffer);
s = read (peek[0], buffer, sizeof buffer);
g_assert_cmpint (s, ==, 53);
g_assert_cmpstr (buffer, ==,
"failure to say failure to say 'i love gnome-panel!'.");
g_object_unref (list);
check_fd_list (fd_list);
#else
g_test_skip ("FD SCM support doesnt exist on Windows");
#endif
}
int
main (int argc, char **argv)
{
g_test_init (&argc, &argv, NULL);
g_test_add_func ("/unix-fd/fd-list", test_fd_list);
g_test_add_func ("/unix-fd/scm", test_scm);
return g_test_run();
}