Improve coverage of stream tests
[platform/upstream/glib.git] / gio / tests / data-output-stream.c
1 /* GLib testing framework examples and tests
2  * Copyright (C) 2008 Red Hat, Inc.
3  * Authors: Tomas Bzatek <tbzatek@redhat.com>
4  *
5  * This work is provided "as is"; redistribution and modification
6  * in whole or in part, in any medium, physical or electronic is
7  * permitted without restriction.
8  *
9  * This work is distributed in the hope that it will be useful,
10  * but WITHOUT ANY WARRANTY; without even the implied warranty of
11  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
12  *
13  * In no event shall the authors or contributors be liable for any
14  * direct, indirect, incidental, special, exemplary, or consequential
15  * damages (including, but not limited to, procurement of substitute
16  * goods or services; loss of use, data, or profits; or business
17  * interruption) however caused and on any theory of liability, whether
18  * in contract, strict liability, or tort (including negligence or
19  * otherwise) arising in any way out of the use of this software, even
20  * if advised of the possibility of such damage.
21  */
22
23 #include <glib/glib.h>
24 #include <gio/gio.h>
25 #include <stdlib.h>
26 #include <string.h>
27
28 #define MAX_LINES               0xFFF   
29 #define MAX_LINES_BUFF          0xFFFFFF
30 #define MAX_BYTES_BINARY        0x100   
31
32 static void
33 test_basic (void)
34 {
35   GOutputStream *stream;
36   GOutputStream *base_stream;
37   gpointer data;
38   gint val;
39
40   data = g_malloc0 (MAX_LINES_BUFF);
41   
42   /* initialize objects */
43   base_stream = g_memory_output_stream_new (data, MAX_LINES_BUFF, NULL, NULL);
44   stream = G_OUTPUT_STREAM (g_data_output_stream_new (base_stream));
45
46   g_object_get (stream, "byte-order", &val, NULL);
47   g_assert_cmpint (val, ==, G_DATA_STREAM_BYTE_ORDER_BIG_ENDIAN);
48   g_object_set (stream, "byte-order", G_DATA_STREAM_BYTE_ORDER_LITTLE_ENDIAN, NULL);
49   g_assert_cmpint (g_data_output_stream_get_byte_order (G_DATA_OUTPUT_STREAM (stream)), ==, G_DATA_STREAM_BYTE_ORDER_LITTLE_ENDIAN);
50
51   g_object_unref (stream);
52   g_object_unref (base_stream);
53 }
54
55 static void
56 test_read_lines (GDataStreamNewlineType newline_type)
57 {
58   GOutputStream *stream;
59   GOutputStream *base_stream;
60   GError *error = NULL;
61   gpointer data;
62   char *lines;
63   int size;
64   int i;
65
66 #define TEST_STRING     "some_text"
67   
68   const char* endl[4] = {"\n", "\r", "\r\n", "\n"};
69   
70   
71   data = g_malloc0 (MAX_LINES_BUFF);
72   lines = g_malloc0 ((strlen (TEST_STRING) + strlen (endl[newline_type])) * MAX_LINES + 1);
73   
74   /* initialize objects */
75   base_stream = g_memory_output_stream_new (data, MAX_LINES_BUFF, NULL, NULL);
76   stream = G_OUTPUT_STREAM (g_data_output_stream_new (base_stream));
77
78   
79   /*  fill data */
80   for (i = 0; i < MAX_LINES; i++)
81     {
82       gboolean res;
83       char *s = g_strconcat (TEST_STRING, endl[newline_type], NULL);
84       res = g_data_output_stream_put_string (G_DATA_OUTPUT_STREAM (stream), s, NULL, &error);
85       g_stpcpy ((char*)(lines + i*strlen(s)), s);
86       g_assert_no_error (error);
87       g_assert (res == TRUE);
88     }
89
90   /*  Byte order testing */
91   g_data_output_stream_set_byte_order (G_DATA_OUTPUT_STREAM (stream), G_DATA_STREAM_BYTE_ORDER_BIG_ENDIAN);
92   g_assert_cmpint (g_data_output_stream_get_byte_order (G_DATA_OUTPUT_STREAM (stream)), ==, G_DATA_STREAM_BYTE_ORDER_BIG_ENDIAN);
93   g_data_output_stream_set_byte_order (G_DATA_OUTPUT_STREAM (stream), G_DATA_STREAM_BYTE_ORDER_LITTLE_ENDIAN);
94   g_assert_cmpint (g_data_output_stream_get_byte_order (G_DATA_OUTPUT_STREAM (stream)), ==, G_DATA_STREAM_BYTE_ORDER_LITTLE_ENDIAN);
95   
96   /*  compare data */
97   size = strlen (data);
98   g_assert_cmpint (size, <, MAX_LINES_BUFF);
99   g_assert_cmpstr ((char*)data, ==, lines);
100   
101   g_object_unref (base_stream);
102   g_object_unref (stream);
103   g_free (data);
104   g_free (lines);
105 }
106
107 static void
108 test_read_lines_LF (void)
109 {
110   test_read_lines (G_DATA_STREAM_NEWLINE_TYPE_LF);
111 }
112
113 static void
114 test_read_lines_CR (void)
115 {
116   test_read_lines (G_DATA_STREAM_NEWLINE_TYPE_CR);
117 }
118
119 static void
120 test_read_lines_CR_LF (void)
121 {
122   test_read_lines (G_DATA_STREAM_NEWLINE_TYPE_CR_LF);
123 }
124
125 enum TestDataType {
126   TEST_DATA_BYTE = 0,
127   TEST_DATA_INT16,
128   TEST_DATA_UINT16,
129   TEST_DATA_INT32,
130   TEST_DATA_UINT32,
131   TEST_DATA_INT64,
132   TEST_DATA_UINT64
133 };
134
135 static void
136 test_data_array (guchar *buffer, gsize len,
137                  enum TestDataType data_type, GDataStreamByteOrder byte_order)
138 {
139   GOutputStream *stream;
140   GOutputStream *base_stream;
141   guchar *stream_data;
142   
143   GError *error = NULL;
144   guint pos;
145   GDataStreamByteOrder native;
146   gboolean swap;
147   gboolean res;
148   
149   /*  create objects */
150   stream_data = g_malloc0 (len);
151   base_stream = g_memory_output_stream_new (stream_data, len, NULL, NULL);
152   stream = G_OUTPUT_STREAM (g_data_output_stream_new (base_stream));
153   g_data_output_stream_set_byte_order (G_DATA_OUTPUT_STREAM (stream), byte_order);
154   
155   /*  Set flag to swap bytes if needed */
156   native = (G_BYTE_ORDER == G_BIG_ENDIAN) ? G_DATA_STREAM_BYTE_ORDER_BIG_ENDIAN : G_DATA_STREAM_BYTE_ORDER_LITTLE_ENDIAN;
157   swap = (byte_order != G_DATA_STREAM_BYTE_ORDER_HOST_ENDIAN) && (byte_order != native);
158
159   /* set len to length of buffer cast to actual type */
160   switch (data_type)
161     {
162     case TEST_DATA_BYTE:
163       break;
164     case TEST_DATA_INT16:
165     case TEST_DATA_UINT16:
166       g_assert_cmpint (len % 2, ==, 0);
167     case TEST_DATA_INT32:
168     case TEST_DATA_UINT32:
169       g_assert_cmpint (len % 4, ==, 0);
170     case TEST_DATA_INT64:
171     case TEST_DATA_UINT64:
172       g_assert_cmpint (len % 8, ==, 0);
173       len /= 8;
174       break;
175     default:
176       g_assert_not_reached ();
177       break;
178     }
179
180   /*  Write data to the file */
181   for (pos = 0; pos < len; pos++)
182     {
183       switch (data_type)
184         {
185         case TEST_DATA_BYTE:
186           res = g_data_output_stream_put_byte (G_DATA_OUTPUT_STREAM (stream), buffer[pos], NULL, &error);
187           break;
188         case TEST_DATA_INT16:
189           res = g_data_output_stream_put_int16 (G_DATA_OUTPUT_STREAM (stream), ((gint16 *) buffer)[pos], NULL, &error);
190           break;
191         case TEST_DATA_UINT16:
192           res = g_data_output_stream_put_uint16 (G_DATA_OUTPUT_STREAM (stream), ((guint16 *) buffer)[pos], NULL, &error);
193           break;
194         case TEST_DATA_INT32:
195           res = g_data_output_stream_put_int32 (G_DATA_OUTPUT_STREAM (stream), ((gint32 *) buffer)[pos], NULL, &error);
196           break;
197         case TEST_DATA_UINT32:
198           res = g_data_output_stream_put_uint32 (G_DATA_OUTPUT_STREAM (stream), ((guint32 *) buffer)[pos], NULL, &error);
199           break;
200         case TEST_DATA_INT64:
201           res = g_data_output_stream_put_int64 (G_DATA_OUTPUT_STREAM (stream), ((gint64 *) buffer)[pos], NULL, &error);
202           break;
203         case TEST_DATA_UINT64:
204           res = g_data_output_stream_put_uint64 (G_DATA_OUTPUT_STREAM (stream), ((guint64 *) buffer)[pos], NULL, &error);
205           break;
206         default:
207           g_assert_not_reached ();
208           break;
209         }
210       g_assert_no_error (error);
211       g_assert_cmpint (res, ==, TRUE);
212     }
213   
214   /*  Compare data back */
215   for (pos = 0; pos < len; pos++)
216     {
217       switch (data_type)
218         {
219         case TEST_DATA_BYTE:
220           /* swapping unnecessary */
221           g_assert_cmpint (buffer[pos], ==, stream_data[pos]);
222           break;
223         case TEST_DATA_UINT16:
224           if (swap)
225             g_assert_cmpint (GUINT16_SWAP_LE_BE (((guint16 *) buffer)[pos]), ==, ((guint16 *) stream_data)[pos]);
226           else
227             g_assert_cmpint (((guint16 *) buffer)[pos], ==, ((guint16 *) stream_data)[pos]);
228           break;
229         case TEST_DATA_INT16:
230           if (swap)
231             g_assert_cmpint ((gint16) GUINT16_SWAP_LE_BE (((gint16 *) buffer)[pos]), ==, ((gint16 *) stream_data)[pos]);
232           else
233             g_assert_cmpint (((gint16 *) buffer)[pos], ==, ((gint16 *) stream_data)[pos]);
234           break;
235         case TEST_DATA_UINT32:
236           if (swap)
237             g_assert_cmpint (GUINT32_SWAP_LE_BE (((guint32 *) buffer)[pos]), ==, ((guint32 *) stream_data)[pos]);
238           else
239             g_assert_cmpint (((guint32 *) buffer)[pos], ==, ((guint32 *) stream_data)[pos]);
240           break;
241         case TEST_DATA_INT32:
242           if (swap)
243             g_assert_cmpint ((gint32) GUINT32_SWAP_LE_BE (((gint32 *) buffer)[pos]), ==, ((gint32 *) stream_data)[pos]);
244           else
245             g_assert_cmpint (((gint32 *) buffer)[pos], ==, ((gint32 *) stream_data)[pos]);
246           break;
247         case TEST_DATA_UINT64:
248           if (swap)
249             g_assert_cmpint (GUINT64_SWAP_LE_BE (((guint64 *) buffer)[pos]), ==, ((guint64 *) stream_data)[pos]);
250           else
251             g_assert_cmpint (((guint64 *) buffer)[pos], ==, ((guint64 *) stream_data)[pos]);
252           break;
253         case TEST_DATA_INT64:
254           if (swap)
255             g_assert_cmpint ((gint64) GUINT64_SWAP_LE_BE (((gint64 *) buffer)[pos]), ==, ((gint64 *) stream_data)[pos]);
256           else
257             g_assert_cmpint (((gint64 *) buffer)[pos], ==, ((gint64 *) stream_data)[pos]);
258           break;
259         default:
260             g_assert_not_reached ();
261           break;
262         }
263     }
264   
265   g_object_unref (base_stream);
266   g_object_unref (stream);
267   g_free (stream_data);
268 }
269
270 static void
271 test_read_int (void)
272 {
273   GRand *randomizer;
274   gpointer buffer;
275   int i;
276   
277   randomizer = g_rand_new ();
278   buffer = g_malloc0(MAX_BYTES_BINARY);
279   
280   /*  Fill in some random data */
281   for (i = 0; i < MAX_BYTES_BINARY; i++)
282     {
283       guchar x = 0;
284       while (! x)  x = (guchar)g_rand_int (randomizer);
285       *(guchar*)((guchar*)buffer + sizeof (guchar) * i) = x; 
286     }
287
288   for (i = 0; i < 3; i++)
289     {
290       int j;
291       for (j = 0; j <= TEST_DATA_UINT64; j++)
292         test_data_array (buffer, MAX_BYTES_BINARY, j, i);
293     }
294   
295   g_rand_free (randomizer);
296   g_free (buffer);
297 }
298
299 int
300 main (int   argc,
301       char *argv[])
302 {
303   g_type_init ();
304   g_test_init (&argc, &argv, NULL);
305
306   g_test_add_func ("/data-output-stream/basic", test_basic);
307   g_test_add_func ("/data-output-stream/write-lines-LF", test_read_lines_LF);
308   g_test_add_func ("/data-output-stream/write-lines-CR", test_read_lines_CR);
309   g_test_add_func ("/data-output-stream/write-lines-CR-LF", test_read_lines_CR_LF);
310   g_test_add_func ("/data-output-stream/write-int", test_read_int);
311
312   return g_test_run();
313 }