mirror of
https://gitlab.gnome.org/GNOME/glib.git
synced 2025-02-09 20:35:49 +01:00
941 lines
25 KiB
C
941 lines
25 KiB
C
/* Unit tests for gprintf
|
|
* Copyright (C) 1995-1997 Peter Mattis, Spencer Kimball and Josh MacDonald
|
|
*
|
|
* This work is provided "as is"; redistribution and modification
|
|
* in whole or in part, in any medium, physical or electronic is
|
|
* permitted without restriction.
|
|
*
|
|
* This work 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.
|
|
*
|
|
* In no event shall the authors or contributors be liable for any
|
|
* direct, indirect, incidental, special, exemplary, or consequential
|
|
* damages (including, but not limited to, procurement of substitute
|
|
* goods or services; loss of use, data, or profits; or business
|
|
* interruption) however caused and on any theory of liability, whether
|
|
* in contract, strict liability, or tort (including negligence or
|
|
* otherwise) arising in any way out of the use of this software, even
|
|
* if advised of the possibility of such damage.
|
|
*/
|
|
|
|
#include <stdlib.h>
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
#include "glib.h"
|
|
#include "gstdio.h"
|
|
|
|
static void
|
|
test_retval_and_trunc (void)
|
|
{
|
|
gchar buf[128];
|
|
gint res;
|
|
|
|
res = g_snprintf (buf, 0, "abc");
|
|
g_assert_cmpint (res, ==, 3);
|
|
|
|
res = g_snprintf (NULL, 0, "abc");
|
|
g_assert_cmpint (res, ==, 3);
|
|
|
|
res = g_snprintf (buf, 5, "abc");
|
|
g_assert_cmpint (res, ==, 3);
|
|
|
|
res = g_snprintf (buf, 1, "abc");
|
|
g_assert_cmpint (res, ==, 3);
|
|
g_assert (buf[0] == '\0');
|
|
g_assert_cmpstr (buf, ==, "");
|
|
|
|
res = g_snprintf (buf, 2, "abc");
|
|
g_assert_cmpint (res, ==, 3);
|
|
g_assert (buf[1] == '\0');
|
|
g_assert_cmpstr (buf, ==, "a");
|
|
|
|
res = g_snprintf (buf, 3, "abc");
|
|
g_assert_cmpint (res, ==, 3);
|
|
g_assert (buf[2] == '\0');
|
|
g_assert_cmpstr (buf, ==, "ab");
|
|
|
|
res = g_snprintf (buf, 4, "abc");
|
|
g_assert_cmpint (res, ==, 3);
|
|
g_assert (buf[3] == '\0');
|
|
g_assert_cmpstr (buf, ==, "abc");
|
|
|
|
res = g_snprintf (buf, 5, "abc");
|
|
g_assert_cmpint (res, ==, 3);
|
|
g_assert (buf[3] == '\0');
|
|
g_assert_cmpstr (buf, ==, "abc");
|
|
}
|
|
|
|
static void
|
|
test_d (void)
|
|
{
|
|
gchar buf[128];
|
|
gint res;
|
|
|
|
/* %d basic formatting */
|
|
|
|
res = g_snprintf (buf, 128, "%d", 5);
|
|
g_assert_cmpint (res, ==, 1);
|
|
g_assert_cmpstr (buf, ==, "5");
|
|
|
|
res = g_snprintf (buf, 128, "%d", 0);
|
|
g_assert_cmpint (res, ==, 1);
|
|
g_assert_cmpstr (buf, ==, "0");
|
|
|
|
res = g_snprintf (buf, 128, "%.0d", 0);
|
|
g_assert_cmpint (res, ==, 0);
|
|
g_assert_cmpstr (buf, ==, "");
|
|
|
|
res = g_snprintf (buf, 128, "%.0d", 1);
|
|
g_assert_cmpint (res, ==, 1);
|
|
g_assert_cmpstr (buf, ==, "1");
|
|
|
|
res = g_snprintf (buf, 128, "%.d", 2);
|
|
g_assert_cmpint (res, ==, 1);
|
|
g_assert_cmpstr (buf, ==, "2");
|
|
|
|
res = g_snprintf (buf, 128, "%d", -1);
|
|
g_assert_cmpint (res, ==, 2);
|
|
g_assert_cmpstr (buf, ==, "-1");
|
|
|
|
res = g_snprintf (buf, 128, "%.3d", 5);
|
|
g_assert_cmpint (res, ==, 3);
|
|
g_assert_cmpstr (buf, ==, "005");
|
|
|
|
res = g_snprintf (buf, 128, "%.3d", -5);
|
|
g_assert_cmpint (res, ==, 4);
|
|
g_assert_cmpstr (buf, ==, "-005");
|
|
|
|
res = g_snprintf (buf, 128, "%5.3d", 5);
|
|
g_assert_cmpint (res, ==, 5);
|
|
g_assert_cmpstr (buf, ==, " 005");
|
|
|
|
res = g_snprintf (buf, 128, "%-5.3d", -5);
|
|
g_assert_cmpint (res, ==, 5);
|
|
g_assert_cmpstr (buf, ==, "-005 ");
|
|
|
|
/* %d, length modifiers */
|
|
|
|
res = g_snprintf (buf, 128, "%" G_GINT16_FORMAT, (gint16)-5);
|
|
g_assert_cmpint (res, ==, 2);
|
|
g_assert_cmpstr (buf, ==, "-5");
|
|
|
|
res = g_snprintf (buf, 128, "%" G_GUINT16_FORMAT, (guint16)5);
|
|
g_assert_cmpint (res, ==, 1);
|
|
g_assert_cmpstr (buf, ==, "5");
|
|
|
|
res = g_snprintf (buf, 128, "%" G_GINT32_FORMAT, (gint32)-5);
|
|
g_assert_cmpint (res, ==, 2);
|
|
g_assert_cmpstr (buf, ==, "-5");
|
|
|
|
res = g_snprintf (buf, 128, "%" G_GUINT32_FORMAT, (guint32)5);
|
|
g_assert_cmpint (res, ==, 1);
|
|
g_assert_cmpstr (buf, ==, "5");
|
|
|
|
res = g_snprintf (buf, 128, "%" G_GINT64_FORMAT, (gint64)-5);
|
|
g_assert_cmpint (res, ==, 2);
|
|
g_assert_cmpstr (buf, ==, "-5");
|
|
|
|
res = g_snprintf (buf, 128, "%" G_GUINT64_FORMAT, (guint64)5);
|
|
g_assert_cmpint (res, ==, 1);
|
|
g_assert_cmpstr (buf, ==, "5");
|
|
|
|
res = g_snprintf (buf, 128, "%" G_GSSIZE_FORMAT, (gssize)-5);
|
|
g_assert_cmpint (res, ==, 2);
|
|
g_assert_cmpstr (buf, ==, "-5");
|
|
|
|
res = g_snprintf (buf, 128, "%" G_GSIZE_FORMAT, (gsize)5);
|
|
g_assert_cmpint (res, ==, 1);
|
|
g_assert_cmpstr (buf, ==, "5");
|
|
|
|
/* %d, flags */
|
|
|
|
res = g_snprintf (buf, 128, "%-d", 5);
|
|
g_assert_cmpint (res, ==, 1);
|
|
g_assert_cmpstr (buf, ==, "5");
|
|
|
|
res = g_snprintf (buf, 128, "%-+d", 5);
|
|
g_assert_cmpint (res, ==, 2);
|
|
g_assert_cmpstr (buf, ==, "+5");
|
|
|
|
res = g_snprintf (buf, 128, "%+-d", 5);
|
|
g_assert_cmpint (res, ==, 2);
|
|
g_assert_cmpstr (buf, ==, "+5");
|
|
|
|
res = g_snprintf (buf, 128, "%+d", -5);
|
|
g_assert_cmpint (res, ==, 2);
|
|
g_assert_cmpstr (buf, ==, "-5");
|
|
|
|
res = g_snprintf (buf, 128, "% d", 5);
|
|
g_assert_cmpint (res, ==, 2);
|
|
g_assert_cmpstr (buf, ==, " 5");
|
|
|
|
res = g_snprintf (buf, 128, "% .0d", 0);
|
|
g_assert_cmpint (res, ==, 1);
|
|
g_assert_cmpstr (buf, ==, " ");
|
|
|
|
res = g_snprintf (buf, 128, "%03d", 5);
|
|
g_assert_cmpint (res, ==, 3);
|
|
g_assert_cmpstr (buf, ==, "005");
|
|
|
|
res = g_snprintf (buf, 128, "%03d", -5);
|
|
g_assert_cmpint (res, ==, 3);
|
|
g_assert_cmpstr (buf, ==, "-05");
|
|
}
|
|
|
|
/* gcc emits warnings for the following formats, since the C spec
|
|
* says some of the flags must be ignored. (The " " in "% +d" and
|
|
* the "0" in "%-03d".) But we need to test that our printf gets
|
|
* those rules right. So we fool gcc into not warning.
|
|
*
|
|
* These have to be in a separate function in order to use #pragma.
|
|
*/
|
|
#pragma GCC diagnostic push
|
|
#pragma GCC diagnostic ignored "-Wformat-nonliteral"
|
|
static void
|
|
test_d_invalid (void)
|
|
{
|
|
const gchar *fmt;
|
|
gchar buf[128];
|
|
gint res;
|
|
|
|
fmt = "% +d";
|
|
res = g_snprintf (buf, 128, fmt, 5);
|
|
g_assert_cmpint (res, ==, 2);
|
|
g_assert_cmpstr (buf, ==, "+5");
|
|
|
|
fmt = "%-03d";
|
|
res = g_snprintf (buf, 128, fmt, -5);
|
|
g_assert_cmpint (res, ==, 3);
|
|
g_assert_cmpstr (buf, ==, "-5 ");
|
|
}
|
|
#pragma GCC diagnostic pop
|
|
|
|
static void
|
|
test_o (void)
|
|
{
|
|
gchar buf[128];
|
|
gint res;
|
|
|
|
/* %o basic formatting */
|
|
|
|
res = g_snprintf (buf, 128, "%o", 5);
|
|
g_assert_cmpint (res, ==, 1);
|
|
g_assert_cmpstr (buf, ==, "5");
|
|
|
|
res = g_snprintf (buf, 128, "%o", 8);
|
|
g_assert_cmpint (res, ==, 2);
|
|
g_assert_cmpstr (buf, ==, "10");
|
|
|
|
res = g_snprintf (buf, 128, "%o", 0);
|
|
g_assert_cmpint (res, ==, 1);
|
|
g_assert_cmpstr (buf, ==, "0");
|
|
|
|
res = g_snprintf (buf, 128, "%.0o", 0);
|
|
g_assert_cmpint (res, ==, 0);
|
|
g_assert_cmpstr (buf, ==, "");
|
|
|
|
res = g_snprintf (buf, 128, "%.0o", 1);
|
|
g_assert_cmpint (res, ==, 1);
|
|
g_assert_cmpstr (buf, ==, "1");
|
|
|
|
res = g_snprintf (buf, 128, "%.3o", 5);
|
|
g_assert_cmpint (res, ==, 3);
|
|
g_assert_cmpstr (buf, ==, "005");
|
|
|
|
res = g_snprintf (buf, 128, "%.3o", 8);
|
|
g_assert_cmpint (res, ==, 3);
|
|
g_assert_cmpstr (buf, ==, "010");
|
|
|
|
res = g_snprintf (buf, 128, "%5.3o", 5);
|
|
g_assert_cmpint (res, ==, 5);
|
|
g_assert_cmpstr (buf, ==, " 005");
|
|
}
|
|
|
|
static void
|
|
test_u (void)
|
|
{
|
|
gchar buf[128];
|
|
gint res;
|
|
|
|
/* %u, basic formatting */
|
|
|
|
res = g_snprintf (buf, 128, "%u", 5);
|
|
g_assert_cmpint (res, ==, 1);
|
|
g_assert_cmpstr (buf, ==, "5");
|
|
|
|
res = g_snprintf (buf, 128, "%u", 0);
|
|
g_assert_cmpint (res, ==, 1);
|
|
g_assert_cmpstr (buf, ==, "0");
|
|
|
|
res = g_snprintf (buf, 128, "%.0u", 0);
|
|
g_assert_cmpint (res, ==, 0);
|
|
g_assert_cmpstr (buf, ==, "");
|
|
|
|
res = g_snprintf (buf, 128, "%.0u", 1);
|
|
g_assert_cmpint (res, ==, 1);
|
|
g_assert_cmpstr (buf, ==, "1");
|
|
|
|
res = g_snprintf (buf, 128, "%.3u", 5);
|
|
g_assert_cmpint (res, ==, 3);
|
|
g_assert_cmpstr (buf, ==, "005");
|
|
|
|
res = g_snprintf (buf, 128, "%5.3u", 5);
|
|
g_assert_cmpint (res, ==, 5);
|
|
g_assert_cmpstr (buf, ==, " 005");
|
|
}
|
|
|
|
static void
|
|
test_x (void)
|
|
{
|
|
gchar buf[128];
|
|
gint res;
|
|
|
|
/* %x, basic formatting */
|
|
|
|
res = g_snprintf (buf, 128, "%x", 5);
|
|
g_assert_cmpint (res, ==, 1);
|
|
g_assert_cmpstr (buf, ==, "5");
|
|
|
|
res = g_snprintf (buf, 128, "%x", 31);
|
|
g_assert_cmpint (res, ==, 2);
|
|
g_assert_cmpstr (buf, ==, "1f");
|
|
|
|
res = g_snprintf (buf, 128, "%x", 0);
|
|
g_assert_cmpint (res, ==, 1);
|
|
g_assert_cmpstr (buf, ==, "0");
|
|
|
|
res = g_snprintf (buf, 128, "%.0x", 0);
|
|
g_assert_cmpint (res, ==, 0);
|
|
g_assert_cmpstr (buf, ==, "");
|
|
|
|
res = g_snprintf (buf, 128, "%.0x", 1);
|
|
g_assert_cmpint (res, ==, 1);
|
|
g_assert_cmpstr (buf, ==, "1");
|
|
|
|
res = g_snprintf (buf, 128, "%.3x", 5);
|
|
g_assert_cmpint (res, ==, 3);
|
|
g_assert_cmpstr (buf, ==, "005");
|
|
|
|
res = g_snprintf (buf, 128, "%.3x", 31);
|
|
g_assert_cmpint (res, ==, 3);
|
|
g_assert_cmpstr (buf, ==, "01f");
|
|
|
|
res = g_snprintf (buf, 128, "%5.3x", 5);
|
|
g_assert_cmpint (res, ==, 5);
|
|
g_assert_cmpstr (buf, ==, " 005");
|
|
|
|
/* %x, flags */
|
|
|
|
res = g_snprintf (buf, 128, "%-x", 5);
|
|
g_assert_cmpint (res, ==, 1);
|
|
g_assert_cmpstr (buf, ==, "5");
|
|
|
|
res = g_snprintf (buf, 128, "%03x", 5);
|
|
g_assert_cmpint (res, ==, 3);
|
|
g_assert_cmpstr (buf, ==, "005");
|
|
|
|
res = g_snprintf (buf, 128, "%#x", 31);
|
|
g_assert_cmpint (res, ==, 4);
|
|
g_assert_cmpstr (buf, ==, "0x1f");
|
|
|
|
res = g_snprintf (buf, 128, "%#x", 0);
|
|
g_assert_cmpint (res, ==, 1);
|
|
g_assert_cmpstr (buf, ==, "0");
|
|
}
|
|
|
|
static void
|
|
test_X (void)
|
|
{
|
|
gchar buf[128];
|
|
gint res;
|
|
|
|
/* %X, basic formatting */
|
|
|
|
res = g_snprintf (buf, 128, "%X", 5);
|
|
g_assert_cmpint (res, ==, 1);
|
|
g_assert_cmpstr (buf, ==, "5");
|
|
|
|
res = g_snprintf (buf, 128, "%X", 31);
|
|
g_assert_cmpint (res, ==, 2);
|
|
g_assert_cmpstr (buf, ==, "1F");
|
|
|
|
res = g_snprintf (buf, 128, "%X", 0);
|
|
g_assert_cmpint (res, ==, 1);
|
|
g_assert_cmpstr (buf, ==, "0");
|
|
|
|
res = g_snprintf (buf, 128, "%.0X", 0);
|
|
g_assert_cmpint (res, ==, 0);
|
|
g_assert_cmpstr (buf, ==, "");
|
|
|
|
res = g_snprintf (buf, 128, "%.0X", 1);
|
|
g_assert_cmpint (res, ==, 1);
|
|
g_assert_cmpstr (buf, ==, "1");
|
|
|
|
res = g_snprintf (buf, 128, "%.3X", 5);
|
|
g_assert_cmpint (res, ==, 3);
|
|
g_assert_cmpstr (buf, ==, "005");
|
|
|
|
res = g_snprintf (buf, 128, "%.3X", 31);
|
|
g_assert_cmpint (res, ==, 3);
|
|
g_assert_cmpstr (buf, ==, "01F");
|
|
|
|
res = g_snprintf (buf, 128, "%5.3X", 5);
|
|
g_assert_cmpint (res, ==, 5);
|
|
g_assert_cmpstr (buf, ==, " 005");
|
|
|
|
/* %X, flags */
|
|
|
|
res = g_snprintf (buf, 128, "%-X", 5);
|
|
g_assert_cmpint (res, ==, 1);
|
|
g_assert_cmpstr (buf, ==, "5");
|
|
|
|
res = g_snprintf (buf, 128, "%03X", 5);
|
|
g_assert_cmpint (res, ==, 3);
|
|
g_assert_cmpstr (buf, ==, "005");
|
|
|
|
res = g_snprintf (buf, 128, "%#X", 31);
|
|
g_assert_cmpint (res, ==, 4);
|
|
g_assert_cmpstr (buf, ==, "0X1F");
|
|
|
|
res = g_snprintf (buf, 128, "%#X", 0);
|
|
g_assert_cmpint (res, ==, 1);
|
|
g_assert_cmpstr (buf, ==, "0");
|
|
}
|
|
|
|
static void
|
|
test_f (void)
|
|
{
|
|
gchar buf[128];
|
|
gint res;
|
|
|
|
/* %f, basic formattting */
|
|
|
|
res = g_snprintf (buf, 128, "%f", G_PI);
|
|
g_assert_cmpint (res, ==, 8);
|
|
g_assert (0 == strncmp (buf, "3.14159", 7));
|
|
|
|
res = g_snprintf (buf, 128, "%.8f", G_PI);
|
|
g_assert_cmpint (res, ==, 10);
|
|
g_assert (0 == strncmp (buf, "3.1415926", 9));
|
|
|
|
res = g_snprintf (buf, 128, "%.0f", G_PI);
|
|
g_assert_cmpint (res, ==, 1);
|
|
g_assert_cmpstr (buf, ==, "3");
|
|
|
|
res = g_snprintf (buf, 128, "%1.f", G_PI);
|
|
g_assert_cmpint (res, ==, 1);
|
|
g_assert_cmpstr (buf, ==, "3");
|
|
|
|
res = g_snprintf (buf, 128, "%3.f", G_PI);
|
|
g_assert_cmpint (res, ==, 3);
|
|
g_assert_cmpstr (buf, ==, " 3");
|
|
|
|
/* %f, flags */
|
|
|
|
res = g_snprintf (buf, 128, "%+f", G_PI);
|
|
g_assert_cmpint (res, ==, 9);
|
|
g_assert (0 == strncmp (buf, "+3.14159", 8));
|
|
|
|
res = g_snprintf (buf, 128, "% f", G_PI);
|
|
g_assert_cmpint (res, ==, 9);
|
|
g_assert (0 == strncmp (buf, " 3.14159", 8));
|
|
|
|
res = g_snprintf (buf, 128, "%#.0f", G_PI);
|
|
g_assert_cmpint (res, ==, 2);
|
|
g_assert_cmpstr (buf, ==, "3.");
|
|
|
|
res = g_snprintf (buf, 128, "%05.2f", G_PI);
|
|
g_assert_cmpint (res, ==, 5);
|
|
g_assert_cmpstr (buf, ==, "03.14");
|
|
}
|
|
|
|
static gboolean
|
|
same_value (const gchar *actual,
|
|
const gchar *expected)
|
|
{
|
|
gdouble actual_value, expected_value;
|
|
|
|
actual_value = g_ascii_strtod (actual, NULL);
|
|
expected_value = g_ascii_strtod (expected, NULL);
|
|
|
|
return actual_value == expected_value;
|
|
}
|
|
|
|
static void
|
|
test_e (void)
|
|
{
|
|
gchar buf[128];
|
|
gint res;
|
|
|
|
/* %e, basic formatting */
|
|
/* for %e we can't expect to reproduce exact strings and lengths, since SUS
|
|
* only guarantees that the exponent shall always contain at least two
|
|
* digits. On Windows, it seems to be at least three digits long.
|
|
* Therefore, we compare the results of parsing the expected result and the
|
|
* actual result.
|
|
*/
|
|
|
|
res = g_snprintf (buf, 128, "%e", G_PI);
|
|
g_assert_cmpint (res, >=, 12);
|
|
g_assert (same_value (buf, "3.141593e+00"));
|
|
|
|
res = g_snprintf (buf, 128, "%.8e", G_PI);
|
|
g_assert_cmpint (res, >=, 14);
|
|
g_assert (same_value (buf, "3.14159265e+00"));
|
|
|
|
res = g_snprintf (buf, 128, "%.0e", G_PI);
|
|
g_assert_cmpint (res, >=, 5);
|
|
g_assert (same_value (buf, "3e+00"));
|
|
|
|
res = g_snprintf (buf, 128, "%.1e", 0.0);
|
|
g_assert_cmpint (res, >=, 7);
|
|
g_assert (same_value (buf, "0.0e+00"));
|
|
|
|
res = g_snprintf (buf, 128, "%.1e", 0.00001);
|
|
g_assert_cmpint (res, >=, 7);
|
|
g_assert (same_value (buf, "1.0e-05"));
|
|
|
|
res = g_snprintf (buf, 128, "%.1e", 10000.0);
|
|
g_assert_cmpint (res, >=, 7);
|
|
g_assert (same_value (buf, "1.0e+04"));
|
|
|
|
/* %e, flags */
|
|
|
|
res = g_snprintf (buf, 128, "%+e", G_PI);
|
|
g_assert_cmpint (res, >=, 13);
|
|
g_assert (same_value (buf, "+3.141593e+00"));
|
|
|
|
res = g_snprintf (buf, 128, "% e", G_PI);
|
|
g_assert_cmpint (res, >=, 13);
|
|
g_assert (same_value (buf, " 3.141593e+00"));
|
|
|
|
res = g_snprintf (buf, 128, "%#.0e", G_PI);
|
|
g_assert_cmpint (res, >=, 6);
|
|
g_assert (same_value (buf, "3.e+00"));
|
|
|
|
res = g_snprintf (buf, 128, "%09.2e", G_PI);
|
|
g_assert_cmpint (res, >=, 9);
|
|
g_assert (same_value (buf, "03.14e+00"));
|
|
}
|
|
|
|
static void
|
|
test_c (void)
|
|
{
|
|
gchar buf[128];
|
|
gint res;
|
|
|
|
res = g_snprintf (buf, 128, "%c", 'a');
|
|
g_assert_cmpint (res, ==, 1);
|
|
g_assert_cmpstr (buf, ==, "a");
|
|
}
|
|
|
|
static void
|
|
test_s (void)
|
|
{
|
|
gchar buf[128];
|
|
gint res;
|
|
|
|
res = g_snprintf (buf, 128, "%.2s", "abc");
|
|
g_assert_cmpint (res, ==, 2);
|
|
g_assert_cmpstr (buf, ==, "ab");
|
|
|
|
res = g_snprintf (buf, 128, "%.6s", "abc");
|
|
g_assert_cmpint (res, ==, 3);
|
|
g_assert_cmpstr (buf, ==, "abc");
|
|
|
|
res = g_snprintf (buf, 128, "%5s", "abc");
|
|
g_assert_cmpint (res, ==, 5);
|
|
g_assert_cmpstr (buf, ==, " abc");
|
|
|
|
res = g_snprintf (buf, 128, "%-5s", "abc");
|
|
g_assert_cmpint (res, ==, 5);
|
|
g_assert_cmpstr (buf, ==, "abc ");
|
|
|
|
res = g_snprintf (buf, 128, "%5.2s", "abc");
|
|
g_assert_cmpint (res, ==, 5);
|
|
g_assert_cmpstr (buf, ==, " ab");
|
|
|
|
res = g_snprintf (buf, 128, "%*s", 5, "abc");
|
|
g_assert_cmpint (res, ==, 5);
|
|
g_assert_cmpstr (buf, ==, " abc");
|
|
|
|
res = g_snprintf (buf, 128, "%*s", -5, "abc");
|
|
g_assert_cmpint (res, ==, 5);
|
|
g_assert_cmpstr (buf, ==, "abc ");
|
|
|
|
res = g_snprintf (buf, 128, "%*.*s", 5, 2, "abc");
|
|
g_assert_cmpint (res, ==, 5);
|
|
g_assert_cmpstr (buf, ==, " ab");
|
|
}
|
|
|
|
static void
|
|
test_n (void)
|
|
{
|
|
gchar buf[128];
|
|
gint res;
|
|
gint i;
|
|
glong l;
|
|
|
|
res = g_snprintf (buf, 128, "abc%n", &i);
|
|
g_assert_cmpint (res, ==, 3);
|
|
g_assert_cmpstr (buf, ==, "abc");
|
|
g_assert_cmpint (i, ==, 3);
|
|
|
|
res = g_snprintf (buf, 128, "abc%ln", &l);
|
|
g_assert_cmpint (res, ==, 3);
|
|
g_assert_cmpstr (buf, ==, "abc");
|
|
g_assert_cmpint (l, ==, 3);
|
|
}
|
|
|
|
static void
|
|
test_percent (void)
|
|
{
|
|
gchar buf[128];
|
|
gint res;
|
|
|
|
res = g_snprintf (buf, 128, "%%");
|
|
g_assert_cmpint (res, ==, 1);
|
|
g_assert_cmpstr (buf, ==, "%");
|
|
}
|
|
|
|
static void
|
|
test_positional_params (void)
|
|
{
|
|
gchar buf[128];
|
|
gint res;
|
|
|
|
res = g_snprintf (buf, 128, "%2$c %1$c", 'b', 'a');
|
|
g_assert_cmpint (res, ==, 3);
|
|
g_assert_cmpstr (buf, ==, "a b");
|
|
|
|
res = g_snprintf (buf, 128, "%1$*2$.*3$s", "abc", 5, 2);
|
|
g_assert_cmpint (res, ==, 5);
|
|
g_assert_cmpstr (buf, ==, " ab");
|
|
|
|
res = g_snprintf (buf, 128, "%1$s%1$s", "abc");
|
|
g_assert_cmpint (res, ==, 6);
|
|
g_assert_cmpstr (buf, ==, "abcabc");
|
|
}
|
|
|
|
static void
|
|
test_positional_params2 (void)
|
|
{
|
|
if (g_test_subprocess ())
|
|
{
|
|
gint res;
|
|
|
|
res = g_printf ("%2$c %1$c\n", 'b', 'a');
|
|
g_assert_cmpint (res, ==, 4);
|
|
|
|
res = g_printf ("%1$*2$.*3$s\n", "abc", 5, 2);
|
|
g_assert_cmpint (res, ==, 6);
|
|
|
|
res = g_printf ("%1$s%1$s\n", "abc");
|
|
g_assert_cmpint (res, ==, 7);
|
|
return;
|
|
}
|
|
g_test_trap_subprocess (NULL, 0, 0);
|
|
g_test_trap_assert_passed ();
|
|
#ifndef G_OS_WIN32
|
|
g_test_trap_assert_stdout ("a b\n ab\nabcabc\n");
|
|
#else
|
|
g_test_trap_assert_stdout ("a b\r\n ab\r\nabcabc\r\n");
|
|
#endif
|
|
}
|
|
|
|
static void
|
|
test_positional_params3 (void)
|
|
{
|
|
gchar buf[128];
|
|
gint res;
|
|
|
|
res = g_sprintf (buf, "%2$c %1$c", 'b', 'a');
|
|
g_assert_cmpint (res, ==, 3);
|
|
g_assert_cmpstr (buf, ==, "a b");
|
|
|
|
res = g_sprintf (buf, "%1$*2$.*3$s", "abc", 5, 2);
|
|
g_assert_cmpint (res, ==, 5);
|
|
g_assert_cmpstr (buf, ==, " ab");
|
|
|
|
res = g_sprintf (buf, "%1$s%1$s", "abc");
|
|
g_assert_cmpint (res, ==, 6);
|
|
g_assert_cmpstr (buf, ==, "abcabc");
|
|
}
|
|
|
|
static void
|
|
test_percent2 (void)
|
|
{
|
|
if (g_test_subprocess ())
|
|
{
|
|
gint res;
|
|
|
|
res = g_printf ("%%");
|
|
g_assert_cmpint (res, ==, 1);
|
|
return;
|
|
}
|
|
g_test_trap_subprocess (NULL, 0, 0);
|
|
g_test_trap_assert_passed ();
|
|
g_test_trap_assert_stdout ("*%*");
|
|
}
|
|
|
|
static void
|
|
test_64bit (void)
|
|
{
|
|
gchar buf[128];
|
|
gint res;
|
|
|
|
res = g_snprintf (buf, 128, "%" G_GINT64_FORMAT, (gint64)123456);
|
|
g_assert_cmpint (res, ==, 6);
|
|
g_assert_cmpstr (buf, ==, "123456");
|
|
|
|
res = g_snprintf (buf, 128, "%" G_GINT64_FORMAT, (gint64)-123456);
|
|
g_assert_cmpint (res, ==, 7);
|
|
g_assert_cmpstr (buf, ==, "-123456");
|
|
|
|
res = g_snprintf (buf, 128, "%" G_GUINT64_FORMAT, (guint64)123456);
|
|
g_assert_cmpint (res, ==, 6);
|
|
g_assert_cmpstr (buf, ==, "123456");
|
|
|
|
res = g_snprintf (buf, 128, "%" G_GINT64_MODIFIER "o", (gint64)123456);
|
|
g_assert_cmpint (res, ==, 6);
|
|
g_assert_cmpstr (buf, ==, "361100");
|
|
|
|
res = g_snprintf (buf, 128, "%#" G_GINT64_MODIFIER "o", (gint64)123456);
|
|
g_assert_cmpint (res, ==, 7);
|
|
g_assert_cmpstr (buf, ==, "0361100");
|
|
|
|
res = g_snprintf (buf, 128, "%" G_GINT64_MODIFIER "x", (gint64)123456);
|
|
g_assert_cmpint (res, ==, 5);
|
|
g_assert_cmpstr (buf, ==, "1e240");
|
|
|
|
res = g_snprintf (buf, 128, "%#" G_GINT64_MODIFIER "x", (gint64)123456);
|
|
g_assert_cmpint (res, ==, 7);
|
|
g_assert_cmpstr (buf, ==, "0x1e240");
|
|
|
|
res = g_snprintf (buf, 128, "%" G_GINT64_MODIFIER "X", (gint64)123456);
|
|
g_assert_cmpint (res, ==, 5);
|
|
g_assert_cmpstr (buf, ==, "1E240");
|
|
|
|
#ifdef G_OS_WIN32
|
|
/* On Win32, test that the "ll" modifier also works, for backward
|
|
* compatibility. One really should use the G_GINT64_MODIFIER (which
|
|
* on Win32 is the "I64" that the (msvcrt) C library's printf uses),
|
|
* but "ll" used to work with the "trio" g_printf implementation in
|
|
* GLib 2.2, so it's best if it continues to work.
|
|
*/
|
|
|
|
/* However, gcc doesn't know about this, so we need to disable printf
|
|
* format warnings...
|
|
*/
|
|
#if __GNUC__ > 4 || (__GNUC__ == 4 && __GNUC_MINOR__ >= 6)
|
|
_Pragma ("GCC diagnostic push")
|
|
_Pragma ("GCC diagnostic ignored \"-Wformat\"")
|
|
_Pragma ("GCC diagnostic ignored \"-Wformat-extra-args\"")
|
|
#endif
|
|
|
|
res = g_snprintf (buf, 128, "%" "lli", (gint64)123456);
|
|
g_assert_cmpint (res, ==, 6);
|
|
g_assert_cmpstr (buf, ==, "123456");
|
|
|
|
res = g_snprintf (buf, 128, "%" "lli", (gint64)-123456);
|
|
g_assert_cmpint (res, ==, 7);
|
|
g_assert_cmpstr (buf, ==, "-123456");
|
|
|
|
res = g_snprintf (buf, 128, "%" "llu", (guint64)123456);
|
|
g_assert_cmpint (res, ==, 6);
|
|
g_assert_cmpstr (buf, ==, "123456");
|
|
|
|
res = g_snprintf (buf, 128, "%" "ll" "o", (gint64)123456);
|
|
g_assert_cmpint (res, ==, 6);
|
|
g_assert_cmpstr (buf, ==, "361100");
|
|
|
|
res = g_snprintf (buf, 128, "%#" "ll" "o", (gint64)123456);
|
|
g_assert_cmpint (res, ==, 7);
|
|
g_assert_cmpstr (buf, ==, "0361100");
|
|
|
|
res = g_snprintf (buf, 128, "%" "ll" "x", (gint64)123456);
|
|
g_assert_cmpint (res, ==, 5);
|
|
g_assert_cmpstr (buf, ==, "1e240");
|
|
|
|
res = g_snprintf (buf, 128, "%#" "ll" "x", (gint64)123456);
|
|
g_assert_cmpint (res, ==, 7);
|
|
g_assert_cmpstr (buf, ==, "0x1e240");
|
|
|
|
res = g_snprintf (buf, 128, "%" "ll" "X", (gint64)123456);
|
|
g_assert_cmpint (res, ==, 5);
|
|
g_assert_cmpstr (buf, ==, "1E240");
|
|
|
|
#if __GNUC__ > 4 || (__GNUC__ == 4 && __GNUC_MINOR__ >= 6)
|
|
_Pragma ("GCC diagnostic pop")
|
|
#endif
|
|
|
|
#endif
|
|
}
|
|
|
|
static void
|
|
test_64bit2_base (void)
|
|
{
|
|
gint res;
|
|
|
|
res = g_printf ("%" G_GINT64_FORMAT "\n", (gint64)123456);
|
|
g_assert_cmpint (res, ==, 7);
|
|
|
|
res = g_printf ("%" G_GINT64_FORMAT "\n", (gint64)-123456);
|
|
g_assert_cmpint (res, ==, 8);
|
|
|
|
res = g_printf ("%" G_GUINT64_FORMAT "\n", (guint64)123456);
|
|
g_assert_cmpint (res, ==, 7);
|
|
|
|
res = g_printf ("%" G_GINT64_MODIFIER "o\n", (gint64)123456);
|
|
g_assert_cmpint (res, ==, 7);
|
|
|
|
res = g_printf ("%#" G_GINT64_MODIFIER "o\n", (gint64)123456);
|
|
g_assert_cmpint (res, ==, 8);
|
|
|
|
res = g_printf ("%" G_GINT64_MODIFIER "x\n", (gint64)123456);
|
|
g_assert_cmpint (res, ==, 6);
|
|
|
|
res = g_printf ("%#" G_GINT64_MODIFIER "x\n", (gint64)123456);
|
|
g_assert_cmpint (res, ==, 8);
|
|
|
|
res = g_printf ("%" G_GINT64_MODIFIER "X\n", (gint64)123456);
|
|
g_assert_cmpint (res, ==, 6);
|
|
}
|
|
|
|
#ifdef G_OS_WIN32
|
|
static void
|
|
test_64bit2_win32 (void)
|
|
{
|
|
gint res;
|
|
|
|
/* On Win32, test that the "ll" modifier also works, for backward
|
|
* compatibility. One really should use the G_GINT64_MODIFIER (which
|
|
* on Win32 is the "I64" that the (msvcrt) C library's printf uses),
|
|
* but "ll" used to work with the "trio" g_printf implementation in
|
|
* GLib 2.2, so it's best if it continues to work.
|
|
*/
|
|
|
|
/* However, gcc doesn't know about this, so we need to disable printf
|
|
* format warnings...
|
|
*/
|
|
#if __GNUC__ > 4 || (__GNUC__ == 4 && __GNUC_MINOR__ >= 6)
|
|
_Pragma ("GCC diagnostic push")
|
|
_Pragma ("GCC diagnostic ignored \"-Wformat\"")
|
|
_Pragma ("GCC diagnostic ignored \"-Wformat-extra-args\"")
|
|
#endif
|
|
|
|
res = g_printf ("%" "lli\n", (gint64)123456);
|
|
g_assert_cmpint (res, ==, 7);
|
|
|
|
res = g_printf ("%" "lli\n", (gint64)-123456);
|
|
g_assert_cmpint (res, ==, 8);
|
|
|
|
res = g_printf ("%" "llu\n", (guint64)123456);
|
|
g_assert_cmpint (res, ==, 7);
|
|
|
|
res = g_printf ("%" "ll" "o\n", (gint64)123456);
|
|
g_assert_cmpint (res, ==, 7);
|
|
|
|
res = g_printf ("%#" "ll" "o\n", (gint64)123456);
|
|
g_assert_cmpint (res, ==, 8);
|
|
|
|
res = g_printf ("%" "ll" "x\n", (gint64)123456);
|
|
g_assert_cmpint (res, ==, 6);
|
|
|
|
res = g_printf ("%#" "ll" "x\n", (gint64)123456);
|
|
g_assert_cmpint (res, ==, 8);
|
|
|
|
res = g_printf ("%" "ll" "X\n", (gint64)123456);
|
|
g_assert_cmpint (res, ==, 6);
|
|
|
|
#if __GNUC__ > 4 || (__GNUC__ == 4 && __GNUC_MINOR__ >= 6)
|
|
_Pragma ("GCC diagnostic pop")
|
|
#endif
|
|
}
|
|
#endif
|
|
|
|
static void
|
|
test_64bit2 (void)
|
|
{
|
|
#ifndef G_OS_WIN32
|
|
g_test_trap_subprocess ("/printf/test-64bit/subprocess/base", 0, 0);
|
|
g_test_trap_assert_passed ();
|
|
g_test_trap_assert_stdout ("123456\n-123456\n123456\n"
|
|
"361100\n0361100\n1e240\n"
|
|
"0x1e240\n1E240\n");
|
|
|
|
#else
|
|
g_test_trap_subprocess ("/printf/test-64bit/subprocess/base", 0, 0);
|
|
g_test_trap_assert_passed ();
|
|
g_test_trap_assert_stdout ("123456\r\n-123456\r\n123456\r\n"
|
|
"361100\r\n0361100\r\n1e240\r\n"
|
|
"0x1e240\r\n1E240\r\n");
|
|
|
|
g_test_trap_subprocess ("/printf/test-64bit/subprocess/win32", 0, 0);
|
|
g_test_trap_assert_passed ();
|
|
g_test_trap_assert_stdout ("123456\r\n-123456\r\n123456\r\n"
|
|
"361100\r\n0361100\r\n1e240\r\n"
|
|
"0x1e240\r\n1E240\r\n");
|
|
#endif
|
|
}
|
|
|
|
G_GNUC_PRINTF(1, 2)
|
|
static gsize
|
|
upper_bound (const gchar *format, ...)
|
|
{
|
|
va_list args;
|
|
gsize res;
|
|
|
|
va_start (args, format);
|
|
res = g_printf_string_upper_bound (format, args);
|
|
va_end (args);
|
|
|
|
return res;
|
|
}
|
|
|
|
static void
|
|
test_upper_bound (void)
|
|
{
|
|
gsize res;
|
|
|
|
res = upper_bound ("bla %s %d: %g\n", "bla", 123, 0.123);
|
|
g_assert_cmpint (res, ==, 20);
|
|
}
|
|
|
|
int
|
|
main (int argc,
|
|
char *argv[])
|
|
{
|
|
g_test_init (&argc, &argv, NULL);
|
|
|
|
g_test_add_func ("/snprintf/retval-and-trunc", test_retval_and_trunc);
|
|
g_test_add_func ("/snprintf/%d", test_d);
|
|
g_test_add_func ("/snprintf/%d-invalid", test_d_invalid);
|
|
g_test_add_func ("/snprintf/%o", test_o);
|
|
g_test_add_func ("/snprintf/%u", test_u);
|
|
g_test_add_func ("/snprintf/%x", test_x);
|
|
g_test_add_func ("/snprintf/%X", test_X);
|
|
g_test_add_func ("/snprintf/%f", test_f);
|
|
g_test_add_func ("/snprintf/%e", test_e);
|
|
g_test_add_func ("/snprintf/%c", test_c);
|
|
g_test_add_func ("/snprintf/%s", test_s);
|
|
g_test_add_func ("/snprintf/%n", test_n);
|
|
g_test_add_func ("/snprintf/test-percent", test_percent);
|
|
g_test_add_func ("/snprintf/test-positional-params", test_positional_params);
|
|
g_test_add_func ("/snprintf/test-64bit", test_64bit);
|
|
|
|
g_test_add_func ("/printf/test-percent", test_percent2);
|
|
g_test_add_func ("/printf/test-positional-params", test_positional_params2);
|
|
g_test_add_func ("/printf/test-64bit", test_64bit2);
|
|
g_test_add_func ("/printf/test-64bit/subprocess/base", test_64bit2_base);
|
|
#ifdef G_OS_WIN32
|
|
g_test_add_func ("/printf/test-64bit/subprocess/win32", test_64bit2_win32);
|
|
#endif
|
|
|
|
g_test_add_func ("/sprintf/test-positional-params", test_positional_params3);
|
|
g_test_add_func ("/sprintf/upper-bound", test_upper_bound);
|
|
|
|
return g_test_run();
|
|
}
|