/* Test of buffer that accumulates a string by piecewise concatenation. Copyright (C) 2021-2025 Free Software Foundation, Inc. This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2, or (at your option) any later version. This program 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 General Public License for more details. You should have received a copy of the GNU General Public License along with this program; if not, see . */ /* Written by Bruno Haible , 2021. */ #include #include "string-buffer-reversed.h" #include #include #include #include "macros.h" static int my_prependf (struct string_buffer_reversed *buffer, const char *formatstring, ...) { va_list args; va_start (args, formatstring); int ret = sbr_prependvf (buffer, formatstring, args); va_end (args); return ret; } char invalid_format_string_1[] = "%&"; char invalid_format_string_2[] = "%^"; int main () { /* Test accumulation. */ { struct string_buffer_reversed buffer; sbr_init (&buffer); sbr_prepend1 (&buffer, '\377'); sbr_prepend1 (&buffer, 'x'); char *s = sbr_dupfree_c (&buffer); ASSERT (s != NULL && strcmp (s, "x\377") == 0); free (s); } { struct string_buffer_reversed buffer; sbr_init (&buffer); sbr_prepend1 (&buffer, '\377'); sbr_prepend1 (&buffer, 'x'); { string_desc_t sd = sbr_contents (&buffer); ASSERT (sd_length (sd) == 2); ASSERT (sd_char_at (sd, 0) == 'x'); ASSERT (sd_char_at (sd, 1) == '\377'); } sbr_prepend1 (&buffer, '\0'); sbr_prepend1 (&buffer, 'z'); { string_desc_t sd = sbr_contents (&buffer); ASSERT (sd_length (sd) == 4); ASSERT (sd_char_at (sd, 0) == 'z'); ASSERT (sd_char_at (sd, 1) == '\0'); ASSERT (sd_char_at (sd, 2) == 'x'); ASSERT (sd_char_at (sd, 3) == '\377'); } char *s = sbr_dupfree_c (&buffer); ASSERT (s != NULL && memcmp (s, "z\0x\377\0", 5) == 0); free (s); } /* Test simple string concatenation. */ { struct string_buffer_reversed buffer; sbr_init (&buffer); char *s = sbr_dupfree_c (&buffer); ASSERT (s != NULL && strcmp (s, "") == 0); free (s); } { struct string_buffer_reversed buffer; sbr_init (&buffer); sbr_prepend_c (&buffer, "efg"); sbr_prepend_c (&buffer, ""); sbr_prepend_c (&buffer, "abcd"); char *s = sbr_dupfree_c (&buffer); ASSERT (s != NULL && strcmp (s, "abcdefg") == 0); free (s); } { struct string_buffer_reversed buffer; sbr_init (&buffer); sbr_prepend_c (&buffer, "hij"); sbr_prepend_desc (&buffer, sd_new_addr (5, "de\0fg")); sbr_prepend_c (&buffer, "abc"); char *s = sbr_dupfree_c (&buffer); ASSERT (s != NULL && memcmp (s, "abcde\0fghij", 12) == 0); free (s); } /* Test printf-like formatting. */ { struct string_buffer_reversed buffer; sbr_init (&buffer); sbr_prepend_c (&buffer, ">"); sbr_prependf (&buffer, "%x", 3735928559U); sbr_prepend_c (&buffer, "<"); char *s = sbr_dupfree_c (&buffer); ASSERT (s != NULL && strcmp (s, "") == 0); free (s); } /* Test vprintf-like formatting. */ { struct string_buffer_reversed buffer; sbr_init (&buffer); sbr_prepend_c (&buffer, ">"); my_prependf (&buffer, "%x", 3735928559U); sbr_prepend_c (&buffer, "<"); char *s = sbr_dupfree_c (&buffer); ASSERT (s != NULL && strcmp (s, "") == 0); free (s); } /* Test printf-like formatting failure. On all systems except AIX, trying to convert the wide-character 0x76543210 to a multibyte string (in the "C" locale) fails. On all systems, invalid format directives make the vsnzprintf() call fail. */ { struct string_buffer_reversed buffer; int ret; sbr_init (&buffer); sbr_prepend_c (&buffer, ">"); ret = sbr_prependf (&buffer, "%lc", (wint_t) 0x76543210); #if !(defined _AIX || (defined _WIN32 && !defined __CYGWIN__)) ASSERT (ret < 0); ASSERT (errno == EILSEQ); #endif sbr_prepend_c (&buffer, "|"); ret = sbr_prependf (&buffer, invalid_format_string_1, 1); ASSERT (ret < 0); ASSERT (errno == EINVAL); sbr_prepend_c (&buffer, "|"); ret = sbr_prependf (&buffer, invalid_format_string_2, 2); ASSERT (ret < 0); ASSERT (errno == EINVAL); sbr_prepend_c (&buffer, "<"); char *s = sbr_dupfree_c (&buffer); ASSERT (s == NULL); } return test_exit_status; }