mirror of https://gitee.com/openkylin/glib2.0.git
230 lines
6.4 KiB
C
230 lines
6.4 KiB
C
/* GLib testing framework examples and tests
|
|
*
|
|
* Copyright © 2001 Hidetoshi Tajima
|
|
* Copyright © 2001 Ron Steinke
|
|
* Copyright © 2001 Owen Taylor
|
|
* Copyright © 2002 Manish Singh
|
|
* Copyright © 2011 Sjoerd Simons
|
|
* Copyright © 2012 Simon McVittie
|
|
* Copyright © 2013 Stef Walter
|
|
* Copyright © 2005, 2006, 2008, 2012, 2013 Matthias Clasen
|
|
* Copyright © 2020 Endless Mobile, Inc.
|
|
*
|
|
* SPDX-License-Identifier: LGPL-2.1-or-later
|
|
*
|
|
* 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.1 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, see <http://www.gnu.org/licenses/>.
|
|
*
|
|
* Author: Philip Withnall <withnall@endlessm.com>
|
|
*/
|
|
|
|
#include <glib.h>
|
|
#include <glib/gstdio.h>
|
|
|
|
static void
|
|
test_small_writes (void)
|
|
{
|
|
GIOChannel *io;
|
|
GIOStatus status = G_IO_STATUS_ERROR;
|
|
guint bytes_remaining;
|
|
gchar tmp;
|
|
GError *local_error = NULL;
|
|
|
|
io = g_io_channel_new_file ("iochannel-test-outfile", "w", &local_error);
|
|
g_assert_no_error (local_error);
|
|
|
|
g_io_channel_set_encoding (io, NULL, NULL);
|
|
g_io_channel_set_buffer_size (io, 1022);
|
|
|
|
bytes_remaining = 2 * g_io_channel_get_buffer_size (io);
|
|
tmp = 0;
|
|
|
|
while (bytes_remaining)
|
|
{
|
|
status = g_io_channel_write_chars (io, &tmp, 1, NULL, NULL);
|
|
if (status == G_IO_STATUS_ERROR)
|
|
break;
|
|
if (status == G_IO_STATUS_NORMAL)
|
|
bytes_remaining--;
|
|
}
|
|
|
|
g_assert_cmpint (status, ==, G_IO_STATUS_NORMAL);
|
|
|
|
g_io_channel_unref (io);
|
|
g_remove ("iochannel-test-outfile");
|
|
}
|
|
|
|
static void
|
|
test_read_write (void)
|
|
{
|
|
GIOChannel *gio_r, *gio_w ;
|
|
GError *local_error = NULL;
|
|
GString *buffer;
|
|
char *filename;
|
|
gint rlength = 0;
|
|
glong wlength = 0;
|
|
gsize length_out;
|
|
const gchar *encoding = "EUC-JP";
|
|
GIOStatus status;
|
|
const gsize buffer_size_bytes = 1024;
|
|
|
|
filename = g_test_build_filename (G_TEST_DIST, "iochannel-test-infile", NULL);
|
|
|
|
setbuf (stdout, NULL); /* For debugging */
|
|
|
|
gio_r = g_io_channel_new_file (filename, "r", &local_error);
|
|
g_assert_no_error (local_error);
|
|
|
|
gio_w = g_io_channel_new_file ("iochannel-test-outfile", "w", &local_error);
|
|
g_assert_no_error (local_error);
|
|
|
|
g_io_channel_set_encoding (gio_r, encoding, &local_error);
|
|
g_assert_no_error (local_error);
|
|
|
|
g_io_channel_set_buffer_size (gio_r, buffer_size_bytes);
|
|
|
|
status = g_io_channel_set_flags (gio_r, G_IO_FLAG_NONBLOCK, &local_error);
|
|
if (status == G_IO_STATUS_ERROR)
|
|
{
|
|
#ifdef G_OS_WIN32
|
|
g_test_message ("FIXME: not implemented on win32");
|
|
#else
|
|
/* Errors should not happen */
|
|
g_assert_no_error (local_error);
|
|
#endif
|
|
g_clear_error (&local_error);
|
|
}
|
|
buffer = g_string_sized_new (buffer_size_bytes);
|
|
|
|
while (TRUE)
|
|
{
|
|
do
|
|
status = g_io_channel_read_line_string (gio_r, buffer, NULL, &local_error);
|
|
while (status == G_IO_STATUS_AGAIN);
|
|
if (status != G_IO_STATUS_NORMAL)
|
|
break;
|
|
|
|
rlength += buffer->len;
|
|
|
|
do
|
|
status = g_io_channel_write_chars (gio_w, buffer->str, buffer->len,
|
|
&length_out, &local_error);
|
|
while (status == G_IO_STATUS_AGAIN);
|
|
if (status != G_IO_STATUS_NORMAL)
|
|
break;
|
|
|
|
wlength += length_out;
|
|
|
|
/* Ensure the whole line was written */
|
|
g_assert_cmpuint (length_out, ==, buffer->len);
|
|
|
|
g_test_message ("%s", buffer->str);
|
|
g_string_truncate (buffer, 0);
|
|
}
|
|
|
|
switch (status)
|
|
{
|
|
case G_IO_STATUS_EOF:
|
|
break;
|
|
case G_IO_STATUS_ERROR:
|
|
/* Errors should not happen */
|
|
g_assert_no_error (local_error);
|
|
g_clear_error (&local_error);
|
|
break;
|
|
default:
|
|
g_assert_not_reached ();
|
|
break;
|
|
}
|
|
|
|
do
|
|
status = g_io_channel_flush (gio_w, &local_error);
|
|
while (status == G_IO_STATUS_AGAIN);
|
|
|
|
if (status == G_IO_STATUS_ERROR)
|
|
{
|
|
/* Errors should not happen */
|
|
g_assert_no_error (local_error);
|
|
g_clear_error (&local_error);
|
|
}
|
|
|
|
g_test_message ("read %d bytes, wrote %ld bytes", rlength, wlength);
|
|
|
|
g_io_channel_unref (gio_r);
|
|
g_io_channel_unref (gio_w);
|
|
|
|
test_small_writes ();
|
|
|
|
g_free (filename);
|
|
g_string_free (buffer, TRUE);
|
|
}
|
|
|
|
static void
|
|
test_read_line_embedded_nuls (void)
|
|
{
|
|
const guint8 test_data[] = { 'H', 'i', '!', '\0', 'y', 'o', 'u', '\n', ':', ')', '\n' };
|
|
gint fd;
|
|
gchar *filename = NULL;
|
|
GIOChannel *channel = NULL;
|
|
GError *local_error = NULL;
|
|
gchar *line = NULL;
|
|
gsize line_length, terminator_pos;
|
|
GIOStatus status;
|
|
|
|
g_test_summary ("Test that reading a line containing embedded nuls works "
|
|
"when using non-standard line terminators.");
|
|
|
|
/* Write out a temporary file. */
|
|
fd = g_file_open_tmp ("glib-test-io-channel-XXXXXX", &filename, &local_error);
|
|
g_assert_no_error (local_error);
|
|
g_close (fd, NULL);
|
|
fd = -1;
|
|
|
|
g_file_set_contents (filename, (const gchar *) test_data, sizeof (test_data), &local_error);
|
|
g_assert_no_error (local_error);
|
|
|
|
/* Create the channel. */
|
|
channel = g_io_channel_new_file (filename, "r", &local_error);
|
|
g_assert_no_error (local_error);
|
|
|
|
/* Only break on newline characters, not nuls.
|
|
* Use length -1 here to exercise glib#2323; the case where length > 0
|
|
* is covered in glib/tests/protocol.c. */
|
|
g_io_channel_set_line_term (channel, "\n", -1);
|
|
g_io_channel_set_encoding (channel, NULL, &local_error);
|
|
g_assert_no_error (local_error);
|
|
|
|
status = g_io_channel_read_line (channel, &line, &line_length,
|
|
&terminator_pos, &local_error);
|
|
g_assert_no_error (local_error);
|
|
g_assert_cmpint (status, ==, G_IO_STATUS_NORMAL);
|
|
g_assert_cmpuint (line_length, ==, 8);
|
|
g_assert_cmpuint (terminator_pos, ==, 7);
|
|
g_assert_cmpmem (line, line_length, test_data, 8);
|
|
|
|
g_free (line);
|
|
g_io_channel_unref (channel);
|
|
g_free (filename);
|
|
}
|
|
|
|
int
|
|
main (int argc,
|
|
char *argv[])
|
|
{
|
|
g_test_init (&argc, &argv, NULL);
|
|
|
|
g_test_add_func ("/io-channel/read-write", test_read_write);
|
|
g_test_add_func ("/io-channel/read-line/embedded-nuls", test_read_line_embedded_nuls);
|
|
|
|
return g_test_run ();
|
|
}
|