xref: /freebsd/lib/libc/tests/stdio/open_wmemstream_test.c (revision 559a218c9b257775fb249b67945fe4a05b7a6b9f)
11ee02192SEnji Cooper /*-
21ee02192SEnji Cooper  * Copyright (c) 2013 Hudson River Trading LLC
31ee02192SEnji Cooper  * Written by: John H. Baldwin <jhb@FreeBSD.org>
41ee02192SEnji Cooper  * All rights reserved.
51ee02192SEnji Cooper  *
61ee02192SEnji Cooper  * Redistribution and use in source and binary forms, with or without
71ee02192SEnji Cooper  * modification, are permitted provided that the following conditions
81ee02192SEnji Cooper  * are met:
91ee02192SEnji Cooper  * 1. Redistributions of source code must retain the above copyright
101ee02192SEnji Cooper  *    notice, this list of conditions and the following disclaimer.
111ee02192SEnji Cooper  * 2. Redistributions in binary form must reproduce the above copyright
121ee02192SEnji Cooper  *    notice, this list of conditions and the following disclaimer in the
131ee02192SEnji Cooper  *    documentation and/or other materials provided with the distribution.
141ee02192SEnji Cooper  *
151ee02192SEnji Cooper  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
161ee02192SEnji Cooper  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
171ee02192SEnji Cooper  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
181ee02192SEnji Cooper  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
191ee02192SEnji Cooper  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
201ee02192SEnji Cooper  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
211ee02192SEnji Cooper  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
221ee02192SEnji Cooper  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
231ee02192SEnji Cooper  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
241ee02192SEnji Cooper  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
251ee02192SEnji Cooper  * SUCH DAMAGE.
261ee02192SEnji Cooper  */
271ee02192SEnji Cooper 
281ee02192SEnji Cooper #include <err.h>
291ee02192SEnji Cooper #include <errno.h>
301ee02192SEnji Cooper #include <limits.h>
311ee02192SEnji Cooper #include <stdint.h>
321ee02192SEnji Cooper #include <stdio.h>
331ee02192SEnji Cooper #include <stdlib.h>
341ee02192SEnji Cooper #include <string.h>
351ee02192SEnji Cooper #include <wchar.h>
361ee02192SEnji Cooper 
371ee02192SEnji Cooper #include <atf-c.h>
381ee02192SEnji Cooper 
391ee02192SEnji Cooper static wchar_t *buf;
401ee02192SEnji Cooper static size_t len;
411ee02192SEnji Cooper 
421ee02192SEnji Cooper static void
assert_stream(const wchar_t * contents)431ee02192SEnji Cooper assert_stream(const wchar_t *contents)
441ee02192SEnji Cooper {
451ee02192SEnji Cooper 	if (wcslen(contents) != len)
461ee02192SEnji Cooper 		printf("bad length %zd for \"%ls\"\n", len, contents);
471ee02192SEnji Cooper 	else if (wcsncmp(buf, contents, wcslen(contents)) != 0)
481ee02192SEnji Cooper 		printf("bad buffer \"%ls\" for \"%ls\"\n", buf, contents);
491ee02192SEnji Cooper }
501ee02192SEnji Cooper 
511ee02192SEnji Cooper ATF_TC_WITHOUT_HEAD(open_group_test);
ATF_TC_BODY(open_group_test,tc)521ee02192SEnji Cooper ATF_TC_BODY(open_group_test, tc)
531ee02192SEnji Cooper {
541ee02192SEnji Cooper 	FILE *fp;
551ee02192SEnji Cooper 	off_t eob;
561ee02192SEnji Cooper 
571ee02192SEnji Cooper 	fp = open_wmemstream(&buf, &len);
581ee02192SEnji Cooper 	ATF_REQUIRE_MSG(fp != NULL, "open_wmemstream failed");
591ee02192SEnji Cooper 
601ee02192SEnji Cooper 	fwprintf(fp, L"hello my world");
611ee02192SEnji Cooper 	fflush(fp);
621ee02192SEnji Cooper 	assert_stream(L"hello my world");
631ee02192SEnji Cooper 	eob = ftello(fp);
641ee02192SEnji Cooper 	rewind(fp);
651ee02192SEnji Cooper 	fwprintf(fp, L"good-bye");
661ee02192SEnji Cooper 	fseeko(fp, eob, SEEK_SET);
671ee02192SEnji Cooper 	fclose(fp);
681ee02192SEnji Cooper 	assert_stream(L"good-bye world");
691ee02192SEnji Cooper 	free(buf);
701ee02192SEnji Cooper }
711ee02192SEnji Cooper 
721ee02192SEnji Cooper ATF_TC_WITHOUT_HEAD(simple_tests);
ATF_TC_BODY(simple_tests,tc)731ee02192SEnji Cooper ATF_TC_BODY(simple_tests, tc)
741ee02192SEnji Cooper {
751ee02192SEnji Cooper 	static const wchar_t zerobuf[] =
761ee02192SEnji Cooper 	    { L'f', L'o', L'o', 0, 0, 0, 0, L'b', L'a', L'r', 0 };
771ee02192SEnji Cooper 	wchar_t c;
781ee02192SEnji Cooper 	FILE *fp;
791ee02192SEnji Cooper 
801ee02192SEnji Cooper 	fp = open_wmemstream(&buf, NULL);
811ee02192SEnji Cooper 	ATF_REQUIRE_MSG(fp == NULL, "open_wmemstream did not fail");
821ee02192SEnji Cooper 	ATF_REQUIRE_MSG(errno == EINVAL,
831ee02192SEnji Cooper 	    "open_wmemstream didn't fail with EINVAL");
841ee02192SEnji Cooper 	fp = open_wmemstream(NULL, &len);
851ee02192SEnji Cooper 	ATF_REQUIRE_MSG(fp == NULL, "open_wmemstream did not fail");
861ee02192SEnji Cooper 	ATF_REQUIRE_MSG(errno == EINVAL,
871ee02192SEnji Cooper 	    "open_wmemstream didn't fail with EINVAL");
881ee02192SEnji Cooper 	fp = open_wmemstream(&buf, &len);
891ee02192SEnji Cooper 	ATF_REQUIRE_MSG(fp != NULL, "open_memstream failed; errno=%d", errno);
901ee02192SEnji Cooper 	fflush(fp);
911ee02192SEnji Cooper 	assert_stream(L"");
921ee02192SEnji Cooper 	if (fwide(fp, 0) <= 0)
931ee02192SEnji Cooper 		printf("stream is not wide-oriented\n");
941ee02192SEnji Cooper 
951ee02192SEnji Cooper 	fwprintf(fp, L"fo");
961ee02192SEnji Cooper 	fflush(fp);
971ee02192SEnji Cooper 	assert_stream(L"fo");
981ee02192SEnji Cooper 	fputwc(L'o', fp);
991ee02192SEnji Cooper 	fflush(fp);
1001ee02192SEnji Cooper 	assert_stream(L"foo");
1011ee02192SEnji Cooper 	rewind(fp);
1021ee02192SEnji Cooper 	fflush(fp);
1031ee02192SEnji Cooper 	assert_stream(L"");
1041ee02192SEnji Cooper 	fseek(fp, 0, SEEK_END);
1051ee02192SEnji Cooper 	fflush(fp);
1061ee02192SEnji Cooper 	assert_stream(L"foo");
1071ee02192SEnji Cooper 
1081ee02192SEnji Cooper 	/*
1091ee02192SEnji Cooper 	 * Test seeking out past the current end.  Should zero-fill the
1101ee02192SEnji Cooper 	 * intermediate area.
1111ee02192SEnji Cooper 	 */
1121ee02192SEnji Cooper 	fseek(fp, 4, SEEK_END);
1131ee02192SEnji Cooper 	fwprintf(fp, L"bar");
1141ee02192SEnji Cooper 	fflush(fp);
1151ee02192SEnji Cooper 
1161ee02192SEnji Cooper 	/*
1171ee02192SEnji Cooper 	 * Can't use assert_stream() here since this should contain
1181ee02192SEnji Cooper 	 * embedded null characters.
1191ee02192SEnji Cooper 	 */
1201ee02192SEnji Cooper 	if (len != 10)
1211ee02192SEnji Cooper 		printf("bad length %zd for zero-fill test\n", len);
1221ee02192SEnji Cooper 	else if (memcmp(buf, zerobuf, sizeof(zerobuf)) != 0)
1231ee02192SEnji Cooper 		printf("bad buffer for zero-fill test\n");
1241ee02192SEnji Cooper 
1251ee02192SEnji Cooper 	fseek(fp, 3, SEEK_SET);
1261ee02192SEnji Cooper 	fwprintf(fp, L" in ");
1271ee02192SEnji Cooper 	fflush(fp);
1281ee02192SEnji Cooper 	assert_stream(L"foo in ");
1291ee02192SEnji Cooper 	fseek(fp, 0, SEEK_END);
1301ee02192SEnji Cooper 	fflush(fp);
1311ee02192SEnji Cooper 	assert_stream(L"foo in bar");
1321ee02192SEnji Cooper 
1331ee02192SEnji Cooper 	rewind(fp);
1341ee02192SEnji Cooper 	if (fread(&c, sizeof(c), 1, fp) != 0)
1351ee02192SEnji Cooper 		printf("fread did not fail\n");
1361ee02192SEnji Cooper 	else if (!ferror(fp))
1371ee02192SEnji Cooper 		printf("error indicator not set after fread\n");
1381ee02192SEnji Cooper 	else
1391ee02192SEnji Cooper 		clearerr(fp);
1401ee02192SEnji Cooper 
1411ee02192SEnji Cooper 	fseek(fp, 4, SEEK_SET);
1421ee02192SEnji Cooper 	fwprintf(fp, L"bar baz");
1431ee02192SEnji Cooper 	fclose(fp);
1441ee02192SEnji Cooper 	assert_stream(L"foo bar baz");
1451ee02192SEnji Cooper 	free(buf);
1461ee02192SEnji Cooper }
1471ee02192SEnji Cooper 
1481ee02192SEnji Cooper ATF_TC_WITHOUT_HEAD(seek_tests);
ATF_TC_BODY(seek_tests,tc)1491ee02192SEnji Cooper ATF_TC_BODY(seek_tests, tc)
1501ee02192SEnji Cooper {
1511ee02192SEnji Cooper 	FILE *fp;
1521ee02192SEnji Cooper 
1531ee02192SEnji Cooper 	fp = open_wmemstream(&buf, &len);
1541ee02192SEnji Cooper 	ATF_REQUIRE_MSG(fp != NULL, "open_wmemstream failed; errno=%d", errno);
1551ee02192SEnji Cooper 
1561ee02192SEnji Cooper #define SEEK_FAIL(offset, whence, error) do {			\
1571ee02192SEnji Cooper 	errno = 0;						\
1581ee02192SEnji Cooper 	ATF_REQUIRE_MSG(fseeko(fp, (offset), (whence)) != 0,	\
159*9cd70b19SEnji Cooper 	    "fseeko(%s, %s) did not fail, set pos to %jd",	\
1601ee02192SEnji Cooper 	    __STRING(offset), __STRING(whence),			\
1611ee02192SEnji Cooper 	    (intmax_t)ftello(fp));				\
1621ee02192SEnji Cooper 	ATF_REQUIRE_MSG(errno == (error),			\
163*9cd70b19SEnji Cooper 	    "fseeko(%s, %s) failed with %d rather than %s",	\
1641ee02192SEnji Cooper 	    __STRING(offset), __STRING(whence),	errno,		\
1651ee02192SEnji Cooper 	    __STRING(error));					\
1661ee02192SEnji Cooper } while (0)
1671ee02192SEnji Cooper 
1681ee02192SEnji Cooper #define SEEK_OK(offset, whence, result) do {			\
1691ee02192SEnji Cooper 	ATF_REQUIRE_MSG(fseeko(fp, (offset), (whence)) == 0,	\
1701ee02192SEnji Cooper 	    "fseeko(%s, %s) failed: %s",			\
1711ee02192SEnji Cooper 	    __STRING(offset), __STRING(whence), strerror(errno)); \
1721ee02192SEnji Cooper 	ATF_REQUIRE_MSG(ftello(fp) == (result),			\
173*9cd70b19SEnji Cooper 	    "fseeko(%s, %s) seeked to %jd rather than %s",	\
1741ee02192SEnji Cooper 	    __STRING(offset), __STRING(whence),			\
1751ee02192SEnji Cooper 	    (intmax_t)ftello(fp), __STRING(result));		\
1761ee02192SEnji Cooper } while (0)
1771ee02192SEnji Cooper 
1781ee02192SEnji Cooper 	SEEK_FAIL(-1, SEEK_SET, EINVAL);
1791ee02192SEnji Cooper 	SEEK_FAIL(-1, SEEK_CUR, EINVAL);
1801ee02192SEnji Cooper 	SEEK_FAIL(-1, SEEK_END, EINVAL);
1811ee02192SEnji Cooper 	fwprintf(fp, L"foo");
1821ee02192SEnji Cooper 	SEEK_OK(-1, SEEK_CUR, 2);
1831ee02192SEnji Cooper 	SEEK_OK(0, SEEK_SET, 0);
1841ee02192SEnji Cooper 	SEEK_OK(-1, SEEK_END, 2);
1851ee02192SEnji Cooper 	SEEK_OK(OFF_MAX - 1, SEEK_SET, OFF_MAX - 1);
1861ee02192SEnji Cooper 	SEEK_FAIL(2, SEEK_CUR, EOVERFLOW);
1871ee02192SEnji Cooper 	fclose(fp);
1881ee02192SEnji Cooper }
1891ee02192SEnji Cooper 
ATF_TP_ADD_TCS(tp)1901ee02192SEnji Cooper ATF_TP_ADD_TCS(tp)
1911ee02192SEnji Cooper {
1921ee02192SEnji Cooper 
1931ee02192SEnji Cooper 	ATF_TP_ADD_TC(tp, open_group_test);
1941ee02192SEnji Cooper 	ATF_TP_ADD_TC(tp, simple_tests);
1951ee02192SEnji Cooper 	ATF_TP_ADD_TC(tp, seek_tests);
1961ee02192SEnji Cooper 
1971ee02192SEnji Cooper 	return (atf_no_error());
1981ee02192SEnji Cooper }
199