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