119a6dd33SSiva Chandra Reddy //===-- Unittests for the fopencookie function ----------------------------===//
219a6dd33SSiva Chandra Reddy //
319a6dd33SSiva Chandra Reddy // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
419a6dd33SSiva Chandra Reddy // See https://llvm.org/LICENSE.txt for license information.
519a6dd33SSiva Chandra Reddy // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
619a6dd33SSiva Chandra Reddy //
719a6dd33SSiva Chandra Reddy //===----------------------------------------------------------------------===//
819a6dd33SSiva Chandra Reddy 
9*9db0037bSSiva Chandra Reddy #include "src/stdio/clearerr.h"
1019a6dd33SSiva Chandra Reddy #include "src/stdio/fclose.h"
11*9db0037bSSiva Chandra Reddy #include "src/stdio/feof.h"
12*9db0037bSSiva Chandra Reddy #include "src/stdio/ferror.h"
1319a6dd33SSiva Chandra Reddy #include "src/stdio/fflush.h"
1419a6dd33SSiva Chandra Reddy #include "src/stdio/fopencookie.h"
1519a6dd33SSiva Chandra Reddy #include "src/stdio/fread.h"
1619a6dd33SSiva Chandra Reddy #include "src/stdio/fseek.h"
1719a6dd33SSiva Chandra Reddy #include "src/stdio/fwrite.h"
1819a6dd33SSiva Chandra Reddy #include "utils/UnitTest/MemoryMatcher.h"
1919a6dd33SSiva Chandra Reddy #include "utils/UnitTest/Test.h"
2019a6dd33SSiva Chandra Reddy 
2119a6dd33SSiva Chandra Reddy #include <errno.h>
2219a6dd33SSiva Chandra Reddy #include <stdio.h>
2319a6dd33SSiva Chandra Reddy #include <stdlib.h>
2419a6dd33SSiva Chandra Reddy 
2519a6dd33SSiva Chandra Reddy using MemoryView = __llvm_libc::memory::testing::MemoryView;
2619a6dd33SSiva Chandra Reddy 
2719a6dd33SSiva Chandra Reddy struct StringStream {
2819a6dd33SSiva Chandra Reddy   char *buf;
2919a6dd33SSiva Chandra Reddy   size_t bufsize; // Size of buf
3019a6dd33SSiva Chandra Reddy   size_t endpos;  // 1 more than current fill size
3119a6dd33SSiva Chandra Reddy   size_t offset;  // Current read/write location
3219a6dd33SSiva Chandra Reddy };
3319a6dd33SSiva Chandra Reddy 
write_ss(void * cookie,const char * buf,size_t size)3419a6dd33SSiva Chandra Reddy ssize_t write_ss(void *cookie, const char *buf, size_t size) {
3519a6dd33SSiva Chandra Reddy   auto *ss = reinterpret_cast<StringStream *>(cookie);
3619a6dd33SSiva Chandra Reddy   if (ss->offset + size > ss->bufsize)
3719a6dd33SSiva Chandra Reddy     ss->buf =
3819a6dd33SSiva Chandra Reddy         reinterpret_cast<char *>(realloc(ss->buf, (ss->offset + size) * 2));
3919a6dd33SSiva Chandra Reddy   for (size_t i = 0; i < size; ++i, ss->offset += 1)
4019a6dd33SSiva Chandra Reddy     ss->buf[ss->offset] = buf[i];
4119a6dd33SSiva Chandra Reddy   if (ss->offset > ss->endpos)
4219a6dd33SSiva Chandra Reddy     ss->endpos = ss->offset;
4319a6dd33SSiva Chandra Reddy   return size;
4419a6dd33SSiva Chandra Reddy }
4519a6dd33SSiva Chandra Reddy 
read_ss(void * cookie,char * buf,size_t size)4619a6dd33SSiva Chandra Reddy ssize_t read_ss(void *cookie, char *buf, size_t size) {
4719a6dd33SSiva Chandra Reddy   auto *ss = reinterpret_cast<StringStream *>(cookie);
4819a6dd33SSiva Chandra Reddy   ssize_t copysize = size;
4919a6dd33SSiva Chandra Reddy   if (ss->offset + size > ss->endpos) {
5019a6dd33SSiva Chandra Reddy     // You cannot copy more than what you have available.
5119a6dd33SSiva Chandra Reddy     copysize = ss->endpos - ss->offset;
5219a6dd33SSiva Chandra Reddy     if (copysize < 0)
5319a6dd33SSiva Chandra Reddy       copysize = 0; // A seek could have moved offset past the endpos
5419a6dd33SSiva Chandra Reddy   }
5519a6dd33SSiva Chandra Reddy   for (size_t i = 0; i < size_t(copysize); ++i, ++ss->offset)
5619a6dd33SSiva Chandra Reddy     buf[i] = ss->buf[ss->offset];
5719a6dd33SSiva Chandra Reddy   return copysize;
5819a6dd33SSiva Chandra Reddy }
5919a6dd33SSiva Chandra Reddy 
seek_ss(void * cookie,off64_t * offset,int whence)6019a6dd33SSiva Chandra Reddy int seek_ss(void *cookie, off64_t *offset, int whence) {
6119a6dd33SSiva Chandra Reddy   auto *ss = reinterpret_cast<StringStream *>(cookie);
6219a6dd33SSiva Chandra Reddy   off64_t new_offset;
6319a6dd33SSiva Chandra Reddy   if (whence == SEEK_SET) {
6419a6dd33SSiva Chandra Reddy     new_offset = *offset;
6519a6dd33SSiva Chandra Reddy   } else if (whence == SEEK_CUR) {
6619a6dd33SSiva Chandra Reddy     new_offset = *offset + ss->offset;
6719a6dd33SSiva Chandra Reddy   } else if (whence == SEEK_END) {
6819a6dd33SSiva Chandra Reddy     new_offset = *offset + ss->endpos;
6919a6dd33SSiva Chandra Reddy   } else {
7019a6dd33SSiva Chandra Reddy     errno = EINVAL;
7119a6dd33SSiva Chandra Reddy     return -1;
7219a6dd33SSiva Chandra Reddy   }
7319a6dd33SSiva Chandra Reddy   if (new_offset < 0 || size_t(new_offset) > ss->bufsize)
7419a6dd33SSiva Chandra Reddy     return -1;
7519a6dd33SSiva Chandra Reddy   ss->offset = new_offset;
7619a6dd33SSiva Chandra Reddy   *offset = new_offset;
7719a6dd33SSiva Chandra Reddy   return 0;
7819a6dd33SSiva Chandra Reddy }
7919a6dd33SSiva Chandra Reddy 
close_ss(void * cookie)8019a6dd33SSiva Chandra Reddy int close_ss(void *cookie) {
8119a6dd33SSiva Chandra Reddy   auto *ss = reinterpret_cast<StringStream *>(cookie);
8219a6dd33SSiva Chandra Reddy   free(ss->buf);
8319a6dd33SSiva Chandra Reddy   ss->buf = nullptr;
8419a6dd33SSiva Chandra Reddy   ss->bufsize = ss->endpos = ss->offset = 0;
8519a6dd33SSiva Chandra Reddy   return 0;
8619a6dd33SSiva Chandra Reddy }
8719a6dd33SSiva Chandra Reddy 
8819a6dd33SSiva Chandra Reddy constexpr cookie_io_functions_t STRING_STREAM_FUNCS = {&read_ss, &write_ss,
8919a6dd33SSiva Chandra Reddy                                                        &seek_ss, &close_ss};
9019a6dd33SSiva Chandra Reddy 
TEST(LlvmLibcFOpenCookie,ReadOnlyCookieTest)9119a6dd33SSiva Chandra Reddy TEST(LlvmLibcFOpenCookie, ReadOnlyCookieTest) {
9219a6dd33SSiva Chandra Reddy   constexpr char CONTENT[] = "Hello,readonly!";
9319a6dd33SSiva Chandra Reddy   auto *ss = reinterpret_cast<StringStream *>(malloc(sizeof(StringStream)));
9419a6dd33SSiva Chandra Reddy   ss->buf = reinterpret_cast<char *>(malloc(sizeof(CONTENT)));
9519a6dd33SSiva Chandra Reddy   ss->bufsize = sizeof(CONTENT);
9619a6dd33SSiva Chandra Reddy   ss->offset = 0;
9719a6dd33SSiva Chandra Reddy   ss->endpos = ss->bufsize;
9819a6dd33SSiva Chandra Reddy   for (size_t i = 0; i < sizeof(CONTENT); ++i)
9919a6dd33SSiva Chandra Reddy     ss->buf[i] = CONTENT[i];
10019a6dd33SSiva Chandra Reddy 
10119a6dd33SSiva Chandra Reddy   ::FILE *f = __llvm_libc::fopencookie(ss, "r", STRING_STREAM_FUNCS);
10219a6dd33SSiva Chandra Reddy   ASSERT_TRUE(f != nullptr);
10319a6dd33SSiva Chandra Reddy   char read_data[sizeof(CONTENT)];
10419a6dd33SSiva Chandra Reddy   ASSERT_EQ(sizeof(CONTENT),
10519a6dd33SSiva Chandra Reddy             __llvm_libc::fread(read_data, 1, sizeof(CONTENT), f));
10619a6dd33SSiva Chandra Reddy   ASSERT_STREQ(read_data, CONTENT);
10719a6dd33SSiva Chandra Reddy 
108*9db0037bSSiva Chandra Reddy   // Reading another time should trigger eof.
109*9db0037bSSiva Chandra Reddy   ASSERT_NE(sizeof(CONTENT),
110*9db0037bSSiva Chandra Reddy             __llvm_libc::fread(read_data, 1, sizeof(CONTENT), f));
111*9db0037bSSiva Chandra Reddy   ASSERT_NE(__llvm_libc::feof(f), 0);
112*9db0037bSSiva Chandra Reddy 
11319a6dd33SSiva Chandra Reddy   ASSERT_EQ(0, __llvm_libc::fseek(f, 0, SEEK_SET));
11419a6dd33SSiva Chandra Reddy   // Should be an error to write.
11519a6dd33SSiva Chandra Reddy   ASSERT_EQ(size_t(0), __llvm_libc::fwrite(CONTENT, 1, sizeof(CONTENT), f));
116*9db0037bSSiva Chandra Reddy   ASSERT_NE(__llvm_libc::ferror(f), 0);
117*9db0037bSSiva Chandra Reddy   ASSERT_NE(errno, 0);
11819a6dd33SSiva Chandra Reddy   errno = 0;
11919a6dd33SSiva Chandra Reddy 
120*9db0037bSSiva Chandra Reddy   __llvm_libc::clearerr(f);
121*9db0037bSSiva Chandra Reddy   ASSERT_EQ(__llvm_libc::ferror(f), 0);
122*9db0037bSSiva Chandra Reddy 
12319a6dd33SSiva Chandra Reddy   ASSERT_EQ(0, __llvm_libc::fclose(f));
12419a6dd33SSiva Chandra Reddy   free(ss);
12519a6dd33SSiva Chandra Reddy }
12619a6dd33SSiva Chandra Reddy 
TEST(LlvmLibcFOpenCookie,WriteOnlyCookieTest)12719a6dd33SSiva Chandra Reddy TEST(LlvmLibcFOpenCookie, WriteOnlyCookieTest) {
12819a6dd33SSiva Chandra Reddy   size_t INIT_BUFSIZE = 32;
12919a6dd33SSiva Chandra Reddy   auto *ss = reinterpret_cast<StringStream *>(malloc(sizeof(StringStream)));
13019a6dd33SSiva Chandra Reddy   ss->buf = reinterpret_cast<char *>(malloc(INIT_BUFSIZE));
13119a6dd33SSiva Chandra Reddy   ss->bufsize = INIT_BUFSIZE;
13219a6dd33SSiva Chandra Reddy   ss->offset = 0;
13319a6dd33SSiva Chandra Reddy   ss->endpos = 0;
13419a6dd33SSiva Chandra Reddy 
13519a6dd33SSiva Chandra Reddy   ::FILE *f = __llvm_libc::fopencookie(ss, "w", STRING_STREAM_FUNCS);
13619a6dd33SSiva Chandra Reddy   ASSERT_TRUE(f != nullptr);
13719a6dd33SSiva Chandra Reddy 
13819a6dd33SSiva Chandra Reddy   constexpr char WRITE_DATA[] = "Hello,writeonly!";
13919a6dd33SSiva Chandra Reddy   ASSERT_EQ(sizeof(WRITE_DATA),
14019a6dd33SSiva Chandra Reddy             __llvm_libc::fwrite(WRITE_DATA, 1, sizeof(WRITE_DATA), f));
14119a6dd33SSiva Chandra Reddy   // Flushing will ensure the data to be written to the string stream.
14219a6dd33SSiva Chandra Reddy   ASSERT_EQ(0, __llvm_libc::fflush(f));
14319a6dd33SSiva Chandra Reddy   ASSERT_STREQ(WRITE_DATA, ss->buf);
14419a6dd33SSiva Chandra Reddy 
14519a6dd33SSiva Chandra Reddy   ASSERT_EQ(0, __llvm_libc::fseek(f, 0, SEEK_SET));
14619a6dd33SSiva Chandra Reddy   char read_data[sizeof(WRITE_DATA)];
14719a6dd33SSiva Chandra Reddy   // Should be an error to read.
14819a6dd33SSiva Chandra Reddy   ASSERT_EQ(size_t(0), __llvm_libc::fread(read_data, 1, sizeof(WRITE_DATA), f));
149*9db0037bSSiva Chandra Reddy   ASSERT_NE(__llvm_libc::ferror(f), 0);
15019a6dd33SSiva Chandra Reddy   ASSERT_EQ(errno, EBADF);
15119a6dd33SSiva Chandra Reddy   errno = 0;
15219a6dd33SSiva Chandra Reddy 
153*9db0037bSSiva Chandra Reddy   __llvm_libc::clearerr(f);
154*9db0037bSSiva Chandra Reddy   ASSERT_EQ(__llvm_libc::ferror(f), 0);
155*9db0037bSSiva Chandra Reddy 
15619a6dd33SSiva Chandra Reddy   ASSERT_EQ(0, __llvm_libc::fclose(f));
15719a6dd33SSiva Chandra Reddy   free(ss);
15819a6dd33SSiva Chandra Reddy }
15919a6dd33SSiva Chandra Reddy 
TEST(LlvmLibcFOpenCookie,AppendOnlyCookieTest)16019a6dd33SSiva Chandra Reddy TEST(LlvmLibcFOpenCookie, AppendOnlyCookieTest) {
16119a6dd33SSiva Chandra Reddy   constexpr char INITIAL_CONTENT[] = "1234567890987654321";
16219a6dd33SSiva Chandra Reddy   constexpr char WRITE_DATA[] = "append";
16319a6dd33SSiva Chandra Reddy   auto *ss = reinterpret_cast<StringStream *>(malloc(sizeof(StringStream)));
16419a6dd33SSiva Chandra Reddy   ss->buf = reinterpret_cast<char *>(malloc(sizeof(INITIAL_CONTENT)));
16519a6dd33SSiva Chandra Reddy   ss->bufsize = sizeof(INITIAL_CONTENT);
16619a6dd33SSiva Chandra Reddy   ss->offset = ss->bufsize; // We want to open the file in append mode.
16719a6dd33SSiva Chandra Reddy   ss->endpos = ss->bufsize;
16819a6dd33SSiva Chandra Reddy   for (size_t i = 0; i < sizeof(INITIAL_CONTENT); ++i)
16919a6dd33SSiva Chandra Reddy     ss->buf[i] = INITIAL_CONTENT[i];
17019a6dd33SSiva Chandra Reddy 
17119a6dd33SSiva Chandra Reddy   ::FILE *f = __llvm_libc::fopencookie(ss, "a", STRING_STREAM_FUNCS);
17219a6dd33SSiva Chandra Reddy   ASSERT_TRUE(f != nullptr);
17319a6dd33SSiva Chandra Reddy 
17419a6dd33SSiva Chandra Reddy   constexpr size_t READ_SIZE = 5;
17519a6dd33SSiva Chandra Reddy   char read_data[READ_SIZE];
17619a6dd33SSiva Chandra Reddy   // This is not a readable file.
17719a6dd33SSiva Chandra Reddy   ASSERT_EQ(__llvm_libc::fread(read_data, 1, READ_SIZE, f), size_t(0));
178*9db0037bSSiva Chandra Reddy   ASSERT_NE(__llvm_libc::ferror(f), 0);
17919a6dd33SSiva Chandra Reddy   EXPECT_NE(errno, 0);
18019a6dd33SSiva Chandra Reddy   errno = 0;
18119a6dd33SSiva Chandra Reddy 
182*9db0037bSSiva Chandra Reddy   __llvm_libc::clearerr(f);
183*9db0037bSSiva Chandra Reddy   ASSERT_EQ(__llvm_libc::ferror(f), 0);
184*9db0037bSSiva Chandra Reddy 
18519a6dd33SSiva Chandra Reddy   ASSERT_EQ(__llvm_libc::fwrite(WRITE_DATA, 1, sizeof(WRITE_DATA), f),
18619a6dd33SSiva Chandra Reddy             sizeof(WRITE_DATA));
18719a6dd33SSiva Chandra Reddy   EXPECT_EQ(__llvm_libc::fflush(f), 0);
18819a6dd33SSiva Chandra Reddy   EXPECT_EQ(ss->endpos, sizeof(WRITE_DATA) + sizeof(INITIAL_CONTENT));
18919a6dd33SSiva Chandra Reddy 
19019a6dd33SSiva Chandra Reddy   ASSERT_EQ(__llvm_libc::fclose(f), 0);
19119a6dd33SSiva Chandra Reddy   free(ss);
19219a6dd33SSiva Chandra Reddy }
19319a6dd33SSiva Chandra Reddy 
TEST(LlvmLibcFOpenCookie,ReadUpdateCookieTest)19419a6dd33SSiva Chandra Reddy TEST(LlvmLibcFOpenCookie, ReadUpdateCookieTest) {
19519a6dd33SSiva Chandra Reddy   const char INITIAL_CONTENT[] = "1234567890987654321";
19619a6dd33SSiva Chandra Reddy   auto *ss = reinterpret_cast<StringStream *>(malloc(sizeof(StringStream)));
19719a6dd33SSiva Chandra Reddy   ss->buf = reinterpret_cast<char *>(malloc(sizeof(INITIAL_CONTENT)));
19819a6dd33SSiva Chandra Reddy   ss->bufsize = sizeof(INITIAL_CONTENT);
19919a6dd33SSiva Chandra Reddy   ss->offset = 0;
20019a6dd33SSiva Chandra Reddy   ss->endpos = ss->bufsize;
20119a6dd33SSiva Chandra Reddy   for (size_t i = 0; i < sizeof(INITIAL_CONTENT); ++i)
20219a6dd33SSiva Chandra Reddy     ss->buf[i] = INITIAL_CONTENT[i];
20319a6dd33SSiva Chandra Reddy 
20419a6dd33SSiva Chandra Reddy   ::FILE *f = __llvm_libc::fopencookie(ss, "r+", STRING_STREAM_FUNCS);
20519a6dd33SSiva Chandra Reddy   ASSERT_TRUE(f != nullptr);
20619a6dd33SSiva Chandra Reddy 
20719a6dd33SSiva Chandra Reddy   constexpr size_t READ_SIZE = sizeof(INITIAL_CONTENT) / 2;
20819a6dd33SSiva Chandra Reddy   char read_data[READ_SIZE];
20919a6dd33SSiva Chandra Reddy   ASSERT_EQ(READ_SIZE, __llvm_libc::fread(read_data, 1, READ_SIZE, f));
21019a6dd33SSiva Chandra Reddy 
21119a6dd33SSiva Chandra Reddy   MemoryView src1(INITIAL_CONTENT, READ_SIZE), dst1(read_data, READ_SIZE);
21219a6dd33SSiva Chandra Reddy   EXPECT_MEM_EQ(src1, dst1);
21319a6dd33SSiva Chandra Reddy 
21419a6dd33SSiva Chandra Reddy   ASSERT_EQ(__llvm_libc::fseek(f, 0, SEEK_SET), 0);
21519a6dd33SSiva Chandra Reddy   constexpr char WRITE_DATA[] = "hello, file";
21619a6dd33SSiva Chandra Reddy   ASSERT_EQ(sizeof(WRITE_DATA),
21719a6dd33SSiva Chandra Reddy             __llvm_libc::fwrite(WRITE_DATA, 1, sizeof(WRITE_DATA), f));
21819a6dd33SSiva Chandra Reddy   ASSERT_EQ(__llvm_libc::fflush(f), 0);
21919a6dd33SSiva Chandra Reddy   EXPECT_STREQ(ss->buf, WRITE_DATA);
22019a6dd33SSiva Chandra Reddy 
22119a6dd33SSiva Chandra Reddy   ASSERT_EQ(__llvm_libc::fclose(f), 0);
22219a6dd33SSiva Chandra Reddy   free(ss);
22319a6dd33SSiva Chandra Reddy }
22419a6dd33SSiva Chandra Reddy 
TEST(LlvmLibcFOpenCookie,WriteUpdateCookieTest)22519a6dd33SSiva Chandra Reddy TEST(LlvmLibcFOpenCookie, WriteUpdateCookieTest) {
22619a6dd33SSiva Chandra Reddy   constexpr char WRITE_DATA[] = "hello, file";
22719a6dd33SSiva Chandra Reddy   auto *ss = reinterpret_cast<StringStream *>(malloc(sizeof(StringStream)));
22819a6dd33SSiva Chandra Reddy   ss->buf = reinterpret_cast<char *>(malloc(sizeof(WRITE_DATA)));
22919a6dd33SSiva Chandra Reddy   ss->bufsize = sizeof(WRITE_DATA);
23019a6dd33SSiva Chandra Reddy   ss->offset = 0;
23119a6dd33SSiva Chandra Reddy   ss->endpos = 0;
23219a6dd33SSiva Chandra Reddy 
23319a6dd33SSiva Chandra Reddy   ::FILE *f = __llvm_libc::fopencookie(ss, "w+", STRING_STREAM_FUNCS);
23419a6dd33SSiva Chandra Reddy   ASSERT_TRUE(f != nullptr);
23519a6dd33SSiva Chandra Reddy 
23619a6dd33SSiva Chandra Reddy   ASSERT_EQ(sizeof(WRITE_DATA),
23719a6dd33SSiva Chandra Reddy             __llvm_libc::fwrite(WRITE_DATA, 1, sizeof(WRITE_DATA), f));
23819a6dd33SSiva Chandra Reddy 
23919a6dd33SSiva Chandra Reddy   ASSERT_EQ(__llvm_libc::fseek(f, 0, SEEK_SET), 0);
24019a6dd33SSiva Chandra Reddy 
24119a6dd33SSiva Chandra Reddy   char read_data[sizeof(WRITE_DATA)];
24219a6dd33SSiva Chandra Reddy   ASSERT_EQ(__llvm_libc::fread(read_data, 1, sizeof(read_data), f),
24319a6dd33SSiva Chandra Reddy             sizeof(read_data));
24419a6dd33SSiva Chandra Reddy   EXPECT_STREQ(read_data, WRITE_DATA);
24519a6dd33SSiva Chandra Reddy 
24619a6dd33SSiva Chandra Reddy   ASSERT_EQ(__llvm_libc::fclose(f), 0);
24719a6dd33SSiva Chandra Reddy   free(ss);
24819a6dd33SSiva Chandra Reddy }
249