| /* Licensed to the Apache Software Foundation (ASF) under one or more |
| * contributor license agreements. See the NOTICE file distributed with |
| * this work for additional information regarding copyright ownership. |
| * The ASF licenses this file to You under the Apache License, Version 2.0 |
| * (the "License"); you may not use this file except in compliance with |
| * the License. You may obtain a copy of the License at |
| * |
| * http://www.apache.org/licenses/LICENSE-2.0 |
| * |
| * Unless required by applicable law or agreed to in writing, software |
| * distributed under the License is distributed on an "AS IS" BASIS, |
| * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| * See the License for the specific language governing permissions and |
| * limitations under the License. |
| */ |
| |
| #include "apr_file_io.h" |
| #include "apr_file_info.h" |
| #include "apr_network_io.h" |
| #include "apr_errno.h" |
| #include "apr_general.h" |
| #include "apr_poll.h" |
| #include "apr_lib.h" |
| #include "apr_strings.h" |
| #include "apr_thread_proc.h" |
| #include "testutil.h" |
| |
| #define DIRNAME "data" |
| #define FILENAME DIRNAME "/file_datafile.txt" |
| #define TESTSTR "This is the file data file." |
| |
| #define TESTREAD_BLKSIZE 1024 |
| #define APR_BUFFERSIZE 4096 /* This should match APR's buffer size. */ |
| |
| |
| |
| static void test_open_noreadwrite(abts_case *tc, void *data) |
| { |
| apr_status_t rv; |
| apr_file_t *thefile = NULL; |
| |
| rv = apr_file_open(&thefile, FILENAME, |
| APR_FOPEN_CREATE | APR_FOPEN_EXCL, |
| APR_FPROT_UREAD | APR_FPROT_UWRITE | APR_FPROT_GREAD, p); |
| ABTS_TRUE(tc, rv != APR_SUCCESS); |
| ABTS_INT_EQUAL(tc, 1, APR_STATUS_IS_EACCES(rv)); |
| ABTS_PTR_EQUAL(tc, NULL, thefile); |
| } |
| |
| static void test_open_excl(abts_case *tc, void *data) |
| { |
| apr_status_t rv; |
| apr_file_t *thefile = NULL; |
| |
| rv = apr_file_open(&thefile, FILENAME, |
| APR_FOPEN_CREATE | APR_FOPEN_EXCL | APR_FOPEN_WRITE, |
| APR_FPROT_UREAD | APR_FPROT_UWRITE | APR_FPROT_GREAD, p); |
| ABTS_TRUE(tc, rv != APR_SUCCESS); |
| ABTS_INT_EQUAL(tc, 1, APR_STATUS_IS_EEXIST(rv)); |
| ABTS_PTR_EQUAL(tc, NULL, thefile); |
| } |
| |
| static void test_open_read(abts_case *tc, void *data) |
| { |
| apr_status_t rv; |
| apr_file_t *filetest = NULL; |
| |
| rv = apr_file_open(&filetest, FILENAME, |
| APR_FOPEN_READ, |
| APR_FPROT_UREAD | APR_FPROT_UWRITE | APR_FPROT_GREAD, p); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| ABTS_PTR_NOTNULL(tc, filetest); |
| apr_file_close(filetest); |
| } |
| |
| static void link_existing(abts_case *tc, void *data) |
| { |
| apr_status_t rv; |
| |
| rv = apr_file_link("data/file_datafile.txt", "data/file_datafile2.txt"); |
| apr_file_remove("data/file_datafile2.txt", p); |
| ABTS_ASSERT(tc, "Couldn't create hardlink to file", rv == APR_SUCCESS); |
| } |
| |
| static void link_nonexisting(abts_case *tc, void *data) |
| { |
| apr_status_t rv; |
| |
| rv = apr_file_link("data/does_not_exist.txt", "data/fake.txt"); |
| ABTS_ASSERT(tc, "", rv != APR_SUCCESS); |
| } |
| |
| static void test_read(abts_case *tc, void *data) |
| { |
| apr_status_t rv; |
| apr_size_t nbytes = 256; |
| char *str = apr_pcalloc(p, nbytes + 1); |
| apr_file_t *filetest = NULL; |
| |
| rv = apr_file_open(&filetest, FILENAME, |
| APR_FOPEN_READ, |
| APR_FPROT_UREAD | APR_FPROT_UWRITE | APR_FPROT_GREAD, p); |
| |
| APR_ASSERT_SUCCESS(tc, "Opening test file " FILENAME, rv); |
| rv = apr_file_read(filetest, str, &nbytes); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| ABTS_SIZE_EQUAL(tc, strlen(TESTSTR), nbytes); |
| ABTS_STR_EQUAL(tc, TESTSTR, str); |
| |
| apr_file_close(filetest); |
| } |
| |
| static void test_readzero(abts_case *tc, void *data) |
| { |
| apr_status_t rv; |
| apr_size_t nbytes = 0; |
| char *str = NULL; |
| apr_file_t *filetest; |
| |
| rv = apr_file_open(&filetest, FILENAME, APR_FOPEN_READ, APR_FPROT_OS_DEFAULT, p); |
| APR_ASSERT_SUCCESS(tc, "Opening test file " FILENAME, rv); |
| |
| rv = apr_file_read(filetest, str, &nbytes); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| ABTS_SIZE_EQUAL(tc, 0, nbytes); |
| |
| apr_file_close(filetest); |
| } |
| |
| static void test_filename(abts_case *tc, void *data) |
| { |
| const char *str; |
| apr_status_t rv; |
| apr_file_t *filetest = NULL; |
| |
| rv = apr_file_open(&filetest, FILENAME, |
| APR_FOPEN_READ, |
| APR_FPROT_UREAD | APR_FPROT_UWRITE | APR_FPROT_GREAD, p); |
| APR_ASSERT_SUCCESS(tc, "Opening test file " FILENAME, rv); |
| |
| rv = apr_file_name_get(&str, filetest); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| ABTS_STR_EQUAL(tc, FILENAME, str); |
| |
| apr_file_close(filetest); |
| } |
| |
| static void test_fileclose(abts_case *tc, void *data) |
| { |
| char str; |
| apr_status_t rv; |
| apr_size_t one = 1; |
| apr_file_t *filetest = NULL; |
| |
| rv = apr_file_open(&filetest, FILENAME, |
| APR_FOPEN_READ, |
| APR_FPROT_UREAD | APR_FPROT_UWRITE | APR_FPROT_GREAD, p); |
| APR_ASSERT_SUCCESS(tc, "Opening test file " FILENAME, rv); |
| |
| rv = apr_file_close(filetest); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| /* We just closed the file, so this should fail */ |
| rv = apr_file_read(filetest, &str, &one); |
| ABTS_INT_EQUAL(tc, 1, APR_STATUS_IS_EBADF(rv)); |
| } |
| |
| static void test_file_remove(abts_case *tc, void *data) |
| { |
| apr_status_t rv; |
| apr_file_t *filetest = NULL; |
| |
| rv = apr_file_remove(FILENAME, p); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| |
| rv = apr_file_open(&filetest, FILENAME, APR_FOPEN_READ, |
| APR_FPROT_UREAD | APR_FPROT_UWRITE | APR_FPROT_GREAD, p); |
| ABTS_INT_EQUAL(tc, 1, APR_STATUS_IS_ENOENT(rv)); |
| } |
| |
| static void test_open_write(abts_case *tc, void *data) |
| { |
| apr_status_t rv; |
| apr_file_t *filetest = NULL; |
| |
| filetest = NULL; |
| rv = apr_file_open(&filetest, FILENAME, |
| APR_FOPEN_WRITE, |
| APR_FPROT_UREAD | APR_FPROT_UWRITE | APR_FPROT_GREAD, p); |
| ABTS_INT_EQUAL(tc, 1, APR_STATUS_IS_ENOENT(rv)); |
| ABTS_PTR_EQUAL(tc, NULL, filetest); |
| } |
| |
| static void test_open_writecreate(abts_case *tc, void *data) |
| { |
| apr_status_t rv; |
| apr_file_t *filetest = NULL; |
| |
| filetest = NULL; |
| rv = apr_file_open(&filetest, FILENAME, |
| APR_FOPEN_WRITE | APR_FOPEN_CREATE, |
| APR_FPROT_UREAD | APR_FPROT_UWRITE | APR_FPROT_GREAD, p); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| |
| apr_file_close(filetest); |
| } |
| |
| static void test_write(abts_case *tc, void *data) |
| { |
| apr_status_t rv; |
| apr_size_t bytes = strlen(TESTSTR); |
| apr_file_t *filetest = NULL; |
| |
| rv = apr_file_open(&filetest, FILENAME, |
| APR_FOPEN_WRITE | APR_FOPEN_CREATE, |
| APR_FPROT_UREAD | APR_FPROT_UWRITE | APR_FPROT_GREAD, p); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| |
| rv = apr_file_write(filetest, TESTSTR, &bytes); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| |
| apr_file_close(filetest); |
| } |
| |
| static void test_open_readwrite(abts_case *tc, void *data) |
| { |
| apr_status_t rv; |
| apr_file_t *filetest = NULL; |
| |
| filetest = NULL; |
| rv = apr_file_open(&filetest, FILENAME, |
| APR_FOPEN_READ | APR_FOPEN_WRITE, |
| APR_FPROT_UREAD | APR_FPROT_UWRITE | APR_FPROT_GREAD, p); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| ABTS_PTR_NOTNULL(tc, filetest); |
| |
| apr_file_close(filetest); |
| } |
| |
| static void test_seek(abts_case *tc, void *data) |
| { |
| apr_status_t rv; |
| apr_off_t offset = 5; |
| apr_size_t nbytes = 256; |
| char *str = apr_pcalloc(p, nbytes + 1); |
| apr_file_t *filetest = NULL; |
| |
| rv = apr_file_open(&filetest, FILENAME, |
| APR_FOPEN_READ, |
| APR_FPROT_UREAD | APR_FPROT_UWRITE | APR_FPROT_GREAD, p); |
| APR_ASSERT_SUCCESS(tc, "Open test file " FILENAME, rv); |
| |
| rv = apr_file_read(filetest, str, &nbytes); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| ABTS_SIZE_EQUAL(tc, strlen(TESTSTR), nbytes); |
| ABTS_STR_EQUAL(tc, TESTSTR, str); |
| |
| memset(str, 0, nbytes + 1); |
| |
| rv = apr_file_seek(filetest, SEEK_SET, &offset); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| |
| rv = apr_file_read(filetest, str, &nbytes); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| ABTS_SIZE_EQUAL(tc, strlen(TESTSTR) - 5, nbytes); |
| ABTS_STR_EQUAL(tc, &TESTSTR[5], str); |
| |
| apr_file_close(filetest); |
| |
| /* Test for regression of sign error bug with SEEK_END and |
| buffered files. */ |
| rv = apr_file_open(&filetest, FILENAME, |
| APR_FOPEN_READ | APR_FOPEN_BUFFERED, |
| APR_FPROT_UREAD | APR_FPROT_UWRITE | APR_FPROT_GREAD, p); |
| APR_ASSERT_SUCCESS(tc, "Open test file " FILENAME, rv); |
| |
| offset = -5; |
| rv = apr_file_seek(filetest, SEEK_END, &offset); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| ABTS_SIZE_EQUAL(tc, strlen(TESTSTR) - 5, nbytes); |
| |
| memset(str, 0, nbytes + 1); |
| nbytes = 256; |
| rv = apr_file_read(filetest, str, &nbytes); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| ABTS_SIZE_EQUAL(tc, 5, nbytes); |
| ABTS_STR_EQUAL(tc, &TESTSTR[strlen(TESTSTR) - 5], str); |
| |
| apr_file_close(filetest); |
| } |
| |
| static void test_userdata_set(abts_case *tc, void *data) |
| { |
| apr_status_t rv; |
| apr_file_t *filetest = NULL; |
| |
| rv = apr_file_open(&filetest, FILENAME, |
| APR_FOPEN_WRITE, |
| APR_FPROT_UREAD | APR_FPROT_UWRITE | APR_FPROT_GREAD, p); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| |
| rv = apr_file_data_set(filetest, "This is a test", |
| "test", apr_pool_cleanup_null); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| apr_file_close(filetest); |
| } |
| |
| static void test_userdata_get(abts_case *tc, void *data) |
| { |
| apr_status_t rv; |
| void *udata; |
| char *teststr; |
| apr_file_t *filetest = NULL; |
| |
| rv = apr_file_open(&filetest, FILENAME, |
| APR_FOPEN_WRITE, |
| APR_FPROT_UREAD | APR_FPROT_UWRITE | APR_FPROT_GREAD, p); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| |
| rv = apr_file_data_set(filetest, "This is a test", |
| "test", apr_pool_cleanup_null); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| |
| rv = apr_file_data_get(&udata, "test", filetest); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| teststr = udata; |
| ABTS_STR_EQUAL(tc, "This is a test", teststr); |
| |
| apr_file_close(filetest); |
| } |
| |
| static void test_userdata_getnokey(abts_case *tc, void *data) |
| { |
| apr_status_t rv; |
| void *teststr; |
| apr_file_t *filetest = NULL; |
| |
| rv = apr_file_open(&filetest, FILENAME, |
| APR_FOPEN_WRITE, |
| APR_FPROT_UREAD | APR_FPROT_UWRITE | APR_FPROT_GREAD, p); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| |
| rv = apr_file_data_get(&teststr, "nokey", filetest); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| ABTS_PTR_EQUAL(tc, NULL, teststr); |
| apr_file_close(filetest); |
| } |
| |
| static void test_buffer_set_get(abts_case *tc, void *data) |
| { |
| apr_status_t rv; |
| apr_size_t bufsize; |
| apr_file_t *filetest = NULL; |
| char * buffer; |
| |
| rv = apr_file_open(&filetest, FILENAME, |
| APR_FOPEN_WRITE | APR_FOPEN_BUFFERED, |
| APR_FPROT_UREAD | APR_FPROT_UWRITE | APR_FPROT_GREAD, p); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| |
| bufsize = apr_file_buffer_size_get(filetest); |
| ABTS_SIZE_EQUAL(tc, APR_BUFFERSIZE, bufsize); |
| |
| buffer = apr_pcalloc(p, 10240); |
| rv = apr_file_buffer_set(filetest, buffer, 10240); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| |
| bufsize = apr_file_buffer_size_get(filetest); |
| ABTS_SIZE_EQUAL(tc, 10240, bufsize); |
| |
| rv = apr_file_buffer_set(filetest, buffer, 12); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| |
| bufsize = apr_file_buffer_size_get(filetest); |
| ABTS_SIZE_EQUAL(tc, 12, bufsize); |
| |
| apr_file_close(filetest); |
| } |
| static void test_getc(abts_case *tc, void *data) |
| { |
| apr_file_t *f = NULL; |
| apr_status_t rv; |
| char ch; |
| |
| rv = apr_file_open(&f, FILENAME, APR_FOPEN_READ, 0, p); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| |
| apr_file_getc(&ch, f); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| ABTS_INT_EQUAL(tc, (int)TESTSTR[0], (int)ch); |
| apr_file_close(f); |
| } |
| |
| static void test_ungetc(abts_case *tc, void *data) |
| { |
| apr_file_t *f = NULL; |
| apr_status_t rv; |
| char ch; |
| |
| rv = apr_file_open(&f, FILENAME, APR_FOPEN_READ, 0, p); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| |
| apr_file_getc(&ch, f); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| ABTS_INT_EQUAL(tc, (int)TESTSTR[0], (int)ch); |
| |
| apr_file_ungetc('X', f); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| |
| apr_file_getc(&ch, f); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| ABTS_INT_EQUAL(tc, 'X', (int)ch); |
| |
| apr_file_close(f); |
| } |
| |
| static void test_gets(abts_case *tc, void *data) |
| { |
| apr_file_t *f = NULL; |
| apr_status_t rv; |
| char *str = apr_palloc(p, 256); |
| |
| rv = apr_file_open(&f, FILENAME, APR_FOPEN_READ, 0, p); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| |
| rv = apr_file_gets(str, 256, f); |
| /* Only one line in the test file, so APR will encounter EOF on the first |
| * call to gets, but we should get APR_SUCCESS on this call and |
| * APR_EOF on the next. |
| */ |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| ABTS_STR_EQUAL(tc, TESTSTR, str); |
| rv = apr_file_gets(str, 256, f); |
| ABTS_INT_EQUAL(tc, APR_EOF, rv); |
| ABTS_STR_EQUAL(tc, "", str); |
| /* Calling gets after EOF should return EOF. */ |
| rv = apr_file_gets(str, 256, f); |
| ABTS_INT_EQUAL(tc, APR_EOF, rv); |
| ABTS_STR_EQUAL(tc, "", str); |
| apr_file_close(f); |
| } |
| |
| static void test_gets_buffered(abts_case *tc, void *data) |
| { |
| apr_file_t *f = NULL; |
| apr_status_t rv; |
| char *str = apr_palloc(p, 256); |
| |
| /* This will deadlock gets before the r524355 fix. */ |
| rv = apr_file_open(&f, FILENAME, APR_FOPEN_READ|APR_FOPEN_BUFFERED|APR_FOPEN_XTHREAD, 0, p); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| |
| rv = apr_file_gets(str, 256, f); |
| /* Only one line in the test file, so APR will encounter EOF on the first |
| * call to gets, but we should get APR_SUCCESS on this call and |
| * APR_EOF on the next. |
| */ |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| ABTS_STR_EQUAL(tc, TESTSTR, str); |
| rv = apr_file_gets(str, 256, f); |
| ABTS_INT_EQUAL(tc, APR_EOF, rv); |
| ABTS_STR_EQUAL(tc, "", str); |
| /* Calling gets after EOF should return EOF. */ |
| rv = apr_file_gets(str, 256, f); |
| ABTS_INT_EQUAL(tc, APR_EOF, rv); |
| ABTS_STR_EQUAL(tc, "", str); |
| apr_file_close(f); |
| } |
| |
| static void test_gets_empty(abts_case *tc, void *data) |
| { |
| apr_status_t rv; |
| apr_file_t *f; |
| const char *fname = "data/testgets_empty.dat"; |
| char buf[256]; |
| |
| apr_file_remove(fname, p); |
| |
| rv = apr_file_open(&f, fname, APR_FOPEN_CREATE | APR_FOPEN_WRITE, |
| APR_FPROT_OS_DEFAULT, p); |
| APR_ASSERT_SUCCESS(tc, "open test file", rv); |
| apr_file_close(f); |
| |
| rv = apr_file_open(&f, fname, APR_FOPEN_READ, APR_FPROT_OS_DEFAULT, p); |
| APR_ASSERT_SUCCESS(tc, "re-open test file", rv); |
| |
| rv = apr_file_gets(buf, sizeof(buf), f); |
| ABTS_INT_EQUAL(tc, APR_EOF, rv); |
| ABTS_STR_EQUAL(tc, "", buf); |
| /* Calling gets after EOF should return EOF. */ |
| rv = apr_file_gets(buf, sizeof(buf), f); |
| ABTS_INT_EQUAL(tc, APR_EOF, rv); |
| ABTS_STR_EQUAL(tc, "", buf); |
| apr_file_close(f); |
| } |
| |
| static void test_gets_multiline(abts_case *tc, void *data) |
| { |
| apr_status_t rv; |
| apr_file_t *f; |
| const char *fname = "data/testgets_multiline.dat"; |
| char buf[256]; |
| |
| apr_file_remove(fname, p); |
| |
| rv = apr_file_open(&f, fname, APR_FOPEN_CREATE | APR_FOPEN_WRITE, |
| APR_FPROT_OS_DEFAULT, p); |
| APR_ASSERT_SUCCESS(tc, "open test file", rv); |
| rv = apr_file_puts("a\nb\n", f); |
| APR_ASSERT_SUCCESS(tc, "write test data", rv); |
| apr_file_close(f); |
| |
| rv = apr_file_open(&f, fname, APR_FOPEN_READ, APR_FPROT_OS_DEFAULT, p); |
| APR_ASSERT_SUCCESS(tc, "re-open test file", rv); |
| |
| memset(buf, 0, sizeof(buf)); |
| rv = apr_file_gets(buf, sizeof(buf), f); |
| APR_ASSERT_SUCCESS(tc, "read first line", rv); |
| ABTS_STR_EQUAL(tc, "a\n", buf); |
| |
| memset(buf, 0, sizeof(buf)); |
| rv = apr_file_gets(buf, sizeof(buf), f); |
| APR_ASSERT_SUCCESS(tc, "read second line", rv); |
| ABTS_STR_EQUAL(tc, "b\n", buf); |
| |
| rv = apr_file_gets(buf, sizeof(buf), f); |
| ABTS_INT_EQUAL(tc, APR_EOF, rv); |
| ABTS_STR_EQUAL(tc, "", buf); |
| /* Calling gets after EOF should return EOF. */ |
| rv = apr_file_gets(buf, sizeof(buf), f); |
| ABTS_INT_EQUAL(tc, APR_EOF, rv); |
| ABTS_STR_EQUAL(tc, "", buf); |
| apr_file_close(f); |
| } |
| |
| static void test_gets_small_buf(abts_case *tc, void *data) |
| { |
| apr_status_t rv; |
| apr_file_t *f; |
| const char *fname = "data/testgets_small_buf.dat"; |
| char buf[2]; |
| |
| apr_file_remove(fname, p); |
| |
| rv = apr_file_open(&f, fname, APR_FOPEN_CREATE | APR_FOPEN_WRITE, |
| APR_FPROT_OS_DEFAULT, p); |
| APR_ASSERT_SUCCESS(tc, "open test file", rv); |
| rv = apr_file_puts("ab\n", f); |
| apr_file_close(f); |
| |
| rv = apr_file_open(&f, fname, APR_FOPEN_READ, APR_FPROT_OS_DEFAULT, p); |
| APR_ASSERT_SUCCESS(tc, "re-open test file", rv); |
| /* Buffer is too small to hold the full line, test that gets properly |
| * returns the line content character by character. |
| */ |
| memset(buf, 0, sizeof(buf)); |
| rv = apr_file_gets(buf, sizeof(buf), f); |
| APR_ASSERT_SUCCESS(tc, "read first chunk", rv); |
| ABTS_STR_EQUAL(tc, "a", buf); |
| |
| memset(buf, 0, sizeof(buf)); |
| rv = apr_file_gets(buf, sizeof(buf), f); |
| APR_ASSERT_SUCCESS(tc, "read second chunk", rv); |
| ABTS_STR_EQUAL(tc, "b", buf); |
| |
| memset(buf, 0, sizeof(buf)); |
| rv = apr_file_gets(buf, sizeof(buf), f); |
| APR_ASSERT_SUCCESS(tc, "read third chunk", rv); |
| ABTS_STR_EQUAL(tc, "\n", buf); |
| |
| rv = apr_file_gets(buf, sizeof(buf), f); |
| ABTS_INT_EQUAL(tc, APR_EOF, rv); |
| ABTS_STR_EQUAL(tc, "", buf); |
| /* Calling gets after EOF should return EOF. */ |
| rv = apr_file_gets(buf, sizeof(buf), f); |
| ABTS_INT_EQUAL(tc, APR_EOF, rv); |
| ABTS_STR_EQUAL(tc, "", buf); |
| apr_file_close(f); |
| } |
| |
| static void test_gets_ungetc(abts_case *tc, void *data) |
| { |
| apr_status_t rv; |
| apr_file_t *f; |
| const char *fname = "data/testgets_ungetc.dat"; |
| char buf[256]; |
| |
| apr_file_remove(fname, p); |
| |
| rv = apr_file_open(&f, fname, APR_FOPEN_CREATE | APR_FOPEN_WRITE, |
| APR_FPROT_OS_DEFAULT, p); |
| APR_ASSERT_SUCCESS(tc, "open test file", rv); |
| rv = apr_file_puts("a\n", f); |
| APR_ASSERT_SUCCESS(tc, "write test data", rv); |
| apr_file_close(f); |
| |
| rv = apr_file_open(&f, fname, APR_FOPEN_READ, APR_FPROT_OS_DEFAULT, p); |
| APR_ASSERT_SUCCESS(tc, "re-open test file", rv); |
| |
| rv = apr_file_ungetc('b', f); |
| APR_ASSERT_SUCCESS(tc, "call ungetc", rv); |
| memset(buf, 0, sizeof(buf)); |
| rv = apr_file_gets(buf, sizeof(buf), f); |
| APR_ASSERT_SUCCESS(tc, "read line", rv); |
| ABTS_STR_EQUAL(tc, "ba\n", buf); |
| |
| rv = apr_file_ungetc('\n', f); |
| APR_ASSERT_SUCCESS(tc, "call ungetc with EOL", rv); |
| memset(buf, 0, sizeof(buf)); |
| rv = apr_file_gets(buf, sizeof(buf), f); |
| APR_ASSERT_SUCCESS(tc, "read line", rv); |
| ABTS_STR_EQUAL(tc, "\n", buf); |
| |
| rv = apr_file_gets(buf, sizeof(buf), f); |
| ABTS_INT_EQUAL(tc, APR_EOF, rv); |
| ABTS_STR_EQUAL(tc, "", buf); |
| /* Calling gets after EOF should return EOF. */ |
| rv = apr_file_gets(buf, sizeof(buf), f); |
| ABTS_INT_EQUAL(tc, APR_EOF, rv); |
| ABTS_STR_EQUAL(tc, "", buf); |
| apr_file_close(f); |
| } |
| |
| static void test_gets_buffered_big(abts_case *tc, void *data) |
| { |
| apr_status_t rv; |
| apr_file_t *f; |
| const char *fname = "data/testgets_buffered_big.dat"; |
| char hugestr[APR_BUFFERSIZE + 2]; |
| char buf[APR_BUFFERSIZE + 2]; |
| |
| apr_file_remove(fname, p); |
| |
| rv = apr_file_open(&f, fname, APR_FOPEN_CREATE | APR_FOPEN_WRITE, |
| APR_FPROT_OS_DEFAULT, p); |
| APR_ASSERT_SUCCESS(tc, "open test file", rv); |
| /* Test an edge case with a buffered file and the line that exceeds |
| * the default buffer size by 1 (the line itself fits into the buffer, |
| * but the line + EOL does not). |
| */ |
| memset(hugestr, 'a', sizeof(hugestr)); |
| hugestr[sizeof(hugestr) - 2] = '\n'; |
| hugestr[sizeof(hugestr) - 1] = '\0'; |
| rv = apr_file_puts(hugestr, f); |
| APR_ASSERT_SUCCESS(tc, "write first line", rv); |
| rv = apr_file_puts("b\n", f); |
| APR_ASSERT_SUCCESS(tc, "write second line", rv); |
| apr_file_close(f); |
| |
| rv = apr_file_open(&f, fname, APR_FOPEN_READ | APR_FOPEN_BUFFERED, |
| APR_FPROT_OS_DEFAULT, p); |
| APR_ASSERT_SUCCESS(tc, "re-open test file", rv); |
| |
| memset(buf, 0, sizeof(buf)); |
| rv = apr_file_gets(buf, sizeof(buf), f); |
| APR_ASSERT_SUCCESS(tc, "read first line", rv); |
| ABTS_STR_EQUAL(tc, hugestr, buf); |
| |
| memset(buf, 0, sizeof(buf)); |
| rv = apr_file_gets(buf, sizeof(buf), f); |
| APR_ASSERT_SUCCESS(tc, "read second line", rv); |
| ABTS_STR_EQUAL(tc, "b\n", buf); |
| |
| rv = apr_file_gets(buf, sizeof(buf), f); |
| ABTS_INT_EQUAL(tc, APR_EOF, rv); |
| ABTS_STR_EQUAL(tc, "", buf); |
| /* Calling gets after EOF should return EOF. */ |
| rv = apr_file_gets(buf, sizeof(buf), f); |
| ABTS_INT_EQUAL(tc, APR_EOF, rv); |
| ABTS_STR_EQUAL(tc, "", buf); |
| apr_file_close(f); |
| } |
| |
| static void test_bigread(abts_case *tc, void *data) |
| { |
| apr_file_t *f = NULL; |
| apr_status_t rv; |
| char buf[APR_BUFFERSIZE * 2]; |
| apr_size_t nbytes; |
| |
| /* Create a test file with known content. |
| */ |
| rv = apr_file_open(&f, "data/created_file", |
| APR_FOPEN_CREATE | APR_FOPEN_WRITE | APR_FOPEN_TRUNCATE, |
| APR_FPROT_UREAD | APR_FPROT_UWRITE, p); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| |
| nbytes = APR_BUFFERSIZE; |
| memset(buf, 0xFE, nbytes); |
| |
| rv = apr_file_write(f, buf, &nbytes); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| ABTS_SIZE_EQUAL(tc, APR_BUFFERSIZE, nbytes); |
| |
| rv = apr_file_close(f); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| |
| f = NULL; |
| rv = apr_file_open(&f, "data/created_file", APR_FOPEN_READ, 0, p); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| |
| nbytes = sizeof buf; |
| rv = apr_file_read(f, buf, &nbytes); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| ABTS_SIZE_EQUAL(tc, APR_BUFFERSIZE, nbytes); |
| |
| rv = apr_file_close(f); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| |
| rv = apr_file_remove("data/created_file", p); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| } |
| |
| /* This is a horrible name for this function. We are testing APR, not how |
| * Apache uses APR. And, this function tests _way_ too much stuff. |
| */ |
| static void test_mod_neg(abts_case *tc, void *data) |
| { |
| apr_status_t rv; |
| apr_file_t *f; |
| const char *s; |
| int i; |
| apr_size_t nbytes; |
| char buf[8192]; |
| apr_off_t cur; |
| const char *fname = "data/modneg.dat"; |
| |
| rv = apr_file_open(&f, fname, |
| APR_FOPEN_CREATE | APR_FOPEN_WRITE, |
| APR_FPROT_UREAD | APR_FPROT_UWRITE, p); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| |
| s = "body56789\n"; |
| nbytes = strlen(s); |
| rv = apr_file_write(f, s, &nbytes); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| ABTS_SIZE_EQUAL(tc, strlen(s), nbytes); |
| |
| for (i = 0; i < 7980; i++) { |
| s = "0"; |
| nbytes = strlen(s); |
| rv = apr_file_write(f, s, &nbytes); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| ABTS_SIZE_EQUAL(tc, strlen(s), nbytes); |
| } |
| |
| s = "end456789\n"; |
| nbytes = strlen(s); |
| rv = apr_file_write(f, s, &nbytes); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| ABTS_SIZE_EQUAL(tc, strlen(s), nbytes); |
| |
| for (i = 0; i < 10000; i++) { |
| s = "1"; |
| nbytes = strlen(s); |
| rv = apr_file_write(f, s, &nbytes); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| ABTS_SIZE_EQUAL(tc, strlen(s), nbytes); |
| } |
| |
| rv = apr_file_close(f); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| |
| rv = apr_file_open(&f, fname, APR_FOPEN_READ, 0, p); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| |
| rv = apr_file_gets(buf, 11, f); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| ABTS_STR_EQUAL(tc, "body56789\n", buf); |
| |
| cur = 0; |
| rv = apr_file_seek(f, APR_CUR, &cur); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| ABTS_ASSERT(tc, "File Pointer Mismatch, expected 10", cur == 10); |
| |
| nbytes = sizeof(buf); |
| rv = apr_file_read(f, buf, &nbytes); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| ABTS_SIZE_EQUAL(tc, nbytes, sizeof(buf)); |
| |
| cur = -((apr_off_t)nbytes - 7980); |
| rv = apr_file_seek(f, APR_CUR, &cur); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| ABTS_ASSERT(tc, "File Pointer Mismatch, expected 7990", cur == 7990); |
| |
| rv = apr_file_gets(buf, 11, f); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| ABTS_STR_EQUAL(tc, "end456789\n", buf); |
| |
| rv = apr_file_close(f); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| |
| rv = apr_file_remove(fname, p); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| } |
| |
| /* Test that the contents of file FNAME are equal to data EXPECT of |
| * length EXPECTLEN. */ |
| static void file_contents_equal(abts_case *tc, |
| const char *fname, |
| const void *expect, |
| apr_size_t expectlen) |
| { |
| void *actual = apr_palloc(p, expectlen); |
| apr_file_t *f; |
| |
| APR_ASSERT_SUCCESS(tc, "open file", |
| apr_file_open(&f, fname, APR_FOPEN_READ|APR_FOPEN_BUFFERED, |
| 0, p)); |
| APR_ASSERT_SUCCESS(tc, "read from file", |
| apr_file_read_full(f, actual, expectlen, NULL)); |
| |
| ABTS_ASSERT(tc, "matched expected file contents", |
| memcmp(expect, actual, expectlen) == 0); |
| |
| APR_ASSERT_SUCCESS(tc, "close file", apr_file_close(f)); |
| } |
| |
| #define LINE1 "this is a line of text\n" |
| #define LINE2 "this is a second line of text\n" |
| |
| static void test_puts(abts_case *tc, void *data) |
| { |
| apr_file_t *f; |
| const char *fname = "data/testputs.dat"; |
| |
| APR_ASSERT_SUCCESS(tc, "open file for writing", |
| apr_file_open(&f, fname, |
| APR_FOPEN_WRITE|APR_FOPEN_CREATE|APR_FOPEN_TRUNCATE, |
| APR_FPROT_OS_DEFAULT, p)); |
| |
| APR_ASSERT_SUCCESS(tc, "write line to file", |
| apr_file_puts(LINE1, f)); |
| APR_ASSERT_SUCCESS(tc, "write second line to file", |
| apr_file_puts(LINE2, f)); |
| |
| APR_ASSERT_SUCCESS(tc, "close for writing", |
| apr_file_close(f)); |
| |
| file_contents_equal(tc, fname, LINE1 LINE2, strlen(LINE1 LINE2)); |
| } |
| |
| static void test_writev(abts_case *tc, void *data) |
| { |
| apr_file_t *f; |
| apr_size_t nbytes; |
| struct iovec vec[5]; |
| const char *fname = "data/testwritev.dat"; |
| |
| APR_ASSERT_SUCCESS(tc, "open file for writing", |
| apr_file_open(&f, fname, |
| APR_FOPEN_WRITE|APR_FOPEN_CREATE|APR_FOPEN_TRUNCATE, |
| APR_FPROT_OS_DEFAULT, p)); |
| |
| vec[0].iov_base = LINE1; |
| vec[0].iov_len = strlen(LINE1); |
| |
| APR_ASSERT_SUCCESS(tc, "writev of size 1 to file", |
| apr_file_writev(f, vec, 1, &nbytes)); |
| |
| file_contents_equal(tc, fname, LINE1, strlen(LINE1)); |
| |
| vec[0].iov_base = LINE1; |
| vec[0].iov_len = strlen(LINE1); |
| vec[1].iov_base = LINE2; |
| vec[1].iov_len = strlen(LINE2); |
| vec[2].iov_base = LINE1; |
| vec[2].iov_len = strlen(LINE1); |
| vec[3].iov_base = LINE1; |
| vec[3].iov_len = strlen(LINE1); |
| vec[4].iov_base = LINE2; |
| vec[4].iov_len = strlen(LINE2); |
| |
| APR_ASSERT_SUCCESS(tc, "writev of size 5 to file", |
| apr_file_writev(f, vec, 5, &nbytes)); |
| |
| APR_ASSERT_SUCCESS(tc, "close for writing", |
| apr_file_close(f)); |
| |
| file_contents_equal(tc, fname, LINE1 LINE1 LINE2 LINE1 LINE1 LINE2, |
| strlen(LINE1)*4 + strlen(LINE2)*2); |
| |
| } |
| |
| static void test_writev_full(abts_case *tc, void *data) |
| { |
| apr_file_t *f; |
| apr_size_t nbytes; |
| struct iovec vec[5]; |
| const char *fname = "data/testwritev_full.dat"; |
| |
| APR_ASSERT_SUCCESS(tc, "open file for writing", |
| apr_file_open(&f, fname, |
| APR_FOPEN_WRITE|APR_FOPEN_CREATE|APR_FOPEN_TRUNCATE, |
| APR_FPROT_OS_DEFAULT, p)); |
| |
| vec[0].iov_base = LINE1; |
| vec[0].iov_len = strlen(LINE1); |
| vec[1].iov_base = LINE2; |
| vec[1].iov_len = strlen(LINE2); |
| vec[2].iov_base = LINE1; |
| vec[2].iov_len = strlen(LINE1); |
| vec[3].iov_base = LINE1; |
| vec[3].iov_len = strlen(LINE1); |
| vec[4].iov_base = LINE2; |
| vec[4].iov_len = strlen(LINE2); |
| |
| APR_ASSERT_SUCCESS(tc, "writev_full of size 5 to file", |
| apr_file_writev_full(f, vec, 5, &nbytes)); |
| |
| ABTS_SIZE_EQUAL(tc, strlen(LINE1)*3 + strlen(LINE2)*2, nbytes); |
| |
| APR_ASSERT_SUCCESS(tc, "close for writing", |
| apr_file_close(f)); |
| |
| file_contents_equal(tc, fname, LINE1 LINE2 LINE1 LINE1 LINE2, |
| strlen(LINE1)*3 + strlen(LINE2)*2); |
| |
| } |
| |
| static void test_writev_buffered(abts_case *tc, void *data) |
| { |
| apr_file_t *f; |
| apr_size_t nbytes; |
| struct iovec vec[2]; |
| const char *fname = "data/testwritev_buffered.dat"; |
| |
| APR_ASSERT_SUCCESS(tc, "open file for writing", |
| apr_file_open(&f, fname, |
| APR_FOPEN_WRITE | APR_FOPEN_CREATE | APR_FOPEN_TRUNCATE | |
| APR_FOPEN_BUFFERED, APR_FPROT_OS_DEFAULT, p)); |
| |
| nbytes = strlen(TESTSTR); |
| APR_ASSERT_SUCCESS(tc, "buffered write", |
| apr_file_write(f, TESTSTR, &nbytes)); |
| |
| vec[0].iov_base = LINE1; |
| vec[0].iov_len = strlen(LINE1); |
| vec[1].iov_base = LINE2; |
| vec[1].iov_len = strlen(LINE2); |
| |
| APR_ASSERT_SUCCESS(tc, "writev of size 2 to file", |
| apr_file_writev(f, vec, 2, &nbytes)); |
| |
| APR_ASSERT_SUCCESS(tc, "close for writing", |
| apr_file_close(f)); |
| |
| file_contents_equal(tc, fname, TESTSTR LINE1 LINE2, |
| strlen(TESTSTR) + strlen(LINE1) + strlen(LINE2)); |
| } |
| |
| static void test_writev_buffered_seek(abts_case *tc, void *data) |
| { |
| apr_file_t *f; |
| apr_status_t rv; |
| apr_off_t off = 0; |
| struct iovec vec[3]; |
| apr_size_t nbytes = strlen(TESTSTR); |
| char *str = apr_pcalloc(p, nbytes+1); |
| const char *fname = "data/testwritev_buffered.dat"; |
| |
| APR_ASSERT_SUCCESS(tc, "open file for writing", |
| apr_file_open(&f, fname, |
| APR_FOPEN_WRITE | APR_FOPEN_READ | APR_FOPEN_BUFFERED, |
| APR_FPROT_OS_DEFAULT, p)); |
| |
| rv = apr_file_read(f, str, &nbytes); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| ABTS_STR_EQUAL(tc, TESTSTR, str); |
| APR_ASSERT_SUCCESS(tc, "buffered seek", apr_file_seek(f, APR_SET, &off)); |
| |
| vec[0].iov_base = LINE1; |
| vec[0].iov_len = strlen(LINE1); |
| vec[1].iov_base = LINE2; |
| vec[1].iov_len = strlen(LINE2); |
| vec[2].iov_base = TESTSTR; |
| vec[2].iov_len = strlen(TESTSTR); |
| |
| APR_ASSERT_SUCCESS(tc, "writev of size 2 to file", |
| apr_file_writev(f, vec, 3, &nbytes)); |
| |
| APR_ASSERT_SUCCESS(tc, "close for writing", |
| apr_file_close(f)); |
| |
| file_contents_equal(tc, fname, LINE1 LINE2 TESTSTR, |
| strlen(LINE1) + strlen(LINE2) + strlen(TESTSTR)); |
| |
| APR_ASSERT_SUCCESS(tc, "remove file", apr_file_remove(fname, p)); |
| } |
| |
| static void test_truncate(abts_case *tc, void *data) |
| { |
| apr_status_t rv; |
| apr_file_t *f; |
| const char *fname = "data/testtruncate.dat"; |
| const char *s; |
| apr_size_t nbytes; |
| apr_finfo_t finfo; |
| |
| apr_file_remove(fname, p); |
| |
| rv = apr_file_open(&f, fname, |
| APR_FOPEN_CREATE | APR_FOPEN_WRITE, |
| APR_FPROT_UREAD | APR_FPROT_UWRITE, p); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| |
| s = "some data"; |
| nbytes = strlen(s); |
| rv = apr_file_write(f, s, &nbytes); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| ABTS_SIZE_EQUAL(tc, strlen(s), nbytes); |
| |
| rv = apr_file_close(f); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| |
| rv = apr_file_open(&f, fname, |
| APR_FOPEN_TRUNCATE | APR_FOPEN_WRITE, |
| APR_FPROT_UREAD | APR_FPROT_UWRITE, p); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| |
| rv = apr_file_close(f); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| |
| rv = apr_stat(&finfo, fname, APR_FINFO_SIZE, p); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| ABTS_ASSERT(tc, "File size mismatch, expected 0 (empty)", finfo.size == 0); |
| |
| rv = apr_file_remove(fname, p); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| } |
| |
| static void test_file_trunc(abts_case *tc, void *data) |
| { |
| apr_status_t rv; |
| apr_file_t *f; |
| const char *fname = "data/testtruncate.dat"; |
| const char *s; |
| apr_size_t nbytes; |
| apr_finfo_t finfo; |
| char c; |
| |
| apr_file_remove(fname, p); |
| |
| rv = apr_file_open(&f, fname, |
| APR_FOPEN_CREATE | APR_FOPEN_READ | |
| APR_FOPEN_WRITE, |
| APR_FPROT_UREAD | APR_FPROT_UWRITE, p); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| |
| s = "some data"; |
| nbytes = strlen(s); |
| rv = apr_file_write(f, s, &nbytes); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| ABTS_SIZE_EQUAL(tc, strlen(s), nbytes); |
| rv = apr_file_trunc(f, 4); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| /* Test apr_file_info_get(). */ |
| rv = apr_file_info_get(&finfo, APR_FINFO_SIZE, f); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| ABTS_INT_EQUAL(tc, 4, (int)finfo.size); |
| /* EOF is not reported until the next read. */ |
| rv = apr_file_eof(f); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| rv = apr_file_getc(&c, f); |
| ABTS_INT_EQUAL(tc, APR_EOF, rv); |
| rv = apr_file_eof(f); |
| ABTS_INT_EQUAL(tc, APR_EOF, rv); |
| |
| rv = apr_file_close(f); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| rv = apr_stat(&finfo, fname, APR_FINFO_SIZE, p); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| ABTS_INT_EQUAL(tc, 4, (int)finfo.size); |
| |
| rv = apr_file_remove(fname, p); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| } |
| |
| static void test_file_trunc_buffered_write(abts_case *tc, void *data) |
| { |
| apr_status_t rv; |
| apr_file_t *f; |
| const char *fname = "data/testtruncate_buffered_write.dat"; |
| const char *s; |
| apr_size_t nbytes; |
| apr_finfo_t finfo; |
| char c; |
| |
| apr_file_remove(fname, p); |
| |
| rv = apr_file_open(&f, fname, |
| APR_FOPEN_CREATE | APR_FOPEN_READ | |
| APR_FOPEN_WRITE | APR_FOPEN_BUFFERED, |
| APR_FPROT_UREAD | APR_FPROT_UWRITE, p); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| |
| s = "some data"; |
| nbytes = strlen(s); |
| rv = apr_file_write(f, s, &nbytes); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| ABTS_SIZE_EQUAL(tc, strlen(s), nbytes); |
| rv = apr_file_trunc(f, 4); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| /* Test apr_file_info_get(). */ |
| rv = apr_file_info_get(&finfo, APR_FINFO_SIZE, f); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| ABTS_INT_EQUAL(tc, 4, (int)finfo.size); |
| /* EOF is not reported until the next read. */ |
| rv = apr_file_eof(f); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| rv = apr_file_getc(&c, f); |
| ABTS_INT_EQUAL(tc, APR_EOF, rv); |
| rv = apr_file_eof(f); |
| ABTS_INT_EQUAL(tc, APR_EOF, rv); |
| |
| rv = apr_file_close(f); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| rv = apr_stat(&finfo, fname, APR_FINFO_SIZE, p); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| ABTS_INT_EQUAL(tc, 4, (int)finfo.size); |
| |
| rv = apr_file_remove(fname, p); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| } |
| |
| static void test_file_trunc_buffered_write2(abts_case *tc, void *data) |
| { |
| apr_status_t rv; |
| apr_file_t *f; |
| const char *fname = "data/testtruncate_buffered_write2.dat"; |
| apr_finfo_t finfo; |
| char c; |
| |
| apr_file_remove(fname, p); |
| |
| rv = apr_file_open(&f, fname, |
| APR_FOPEN_CREATE | APR_FOPEN_READ | |
| APR_FOPEN_WRITE | APR_FOPEN_BUFFERED, |
| APR_FPROT_OS_DEFAULT, p); |
| APR_ASSERT_SUCCESS(tc, "open test file", rv); |
| |
| rv = apr_file_puts("abc", f); |
| APR_ASSERT_SUCCESS(tc, "write first string", rv); |
| rv = apr_file_flush(f); |
| APR_ASSERT_SUCCESS(tc, "flush", rv); |
| rv = apr_file_puts("def", f); |
| APR_ASSERT_SUCCESS(tc, "write second string", rv); |
| /* Truncate behind the write buffer. */ |
| rv = apr_file_trunc(f, 2); |
| APR_ASSERT_SUCCESS(tc, "truncate the file", rv); |
| /* Test apr_file_info_get(). */ |
| rv = apr_file_info_get(&finfo, APR_FINFO_SIZE, f); |
| APR_ASSERT_SUCCESS(tc, "get file info", rv); |
| ABTS_INT_EQUAL(tc, 2, (int)finfo.size); |
| /* EOF is not reported until the next read. */ |
| rv = apr_file_eof(f); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| rv = apr_file_getc(&c, f); |
| ABTS_INT_EQUAL(tc, APR_EOF, rv); |
| rv = apr_file_eof(f); |
| ABTS_INT_EQUAL(tc, APR_EOF, rv); |
| |
| apr_file_close(f); |
| |
| rv = apr_stat(&finfo, fname, APR_FINFO_SIZE, p); |
| APR_ASSERT_SUCCESS(tc, "stat file", rv); |
| ABTS_INT_EQUAL(tc, 2, (int)finfo.size); |
| |
| apr_file_remove(fname, p); |
| } |
| |
| static void test_file_trunc_buffered_read(abts_case *tc, void *data) |
| { |
| apr_status_t rv; |
| apr_file_t *f; |
| const char *fname = "data/testtruncate_buffered_read.dat"; |
| apr_finfo_t finfo; |
| char c; |
| |
| apr_file_remove(fname, p); |
| |
| rv = apr_file_open(&f, fname, |
| APR_FOPEN_CREATE | APR_FOPEN_READ | |
| APR_FOPEN_WRITE, APR_FPROT_OS_DEFAULT, p); |
| APR_ASSERT_SUCCESS(tc, "open test file", rv); |
| |
| rv = apr_file_puts("abc", f); |
| APR_ASSERT_SUCCESS(tc, "write test data", rv); |
| apr_file_close(f); |
| |
| rv = apr_file_open(&f, fname, |
| APR_FOPEN_READ | APR_FOPEN_WRITE | |
| APR_FOPEN_BUFFERED, APR_FPROT_OS_DEFAULT, p); |
| APR_ASSERT_SUCCESS(tc, "re-open test file", rv); |
| |
| /* Read to fill in the buffer. */ |
| rv = apr_file_getc(&c, f); |
| APR_ASSERT_SUCCESS(tc, "read char", rv); |
| /* Truncate the file. */ |
| rv = apr_file_trunc(f, 1); |
| APR_ASSERT_SUCCESS(tc, "truncate the file", rv); |
| /* Test apr_file_info_get(). */ |
| rv = apr_file_info_get(&finfo, APR_FINFO_SIZE, f); |
| APR_ASSERT_SUCCESS(tc, "get file info", rv); |
| ABTS_INT_EQUAL(tc, 1, (int)finfo.size); |
| /* EOF is not reported until the next read. */ |
| rv = apr_file_eof(f); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| rv = apr_file_getc(&c, f); |
| ABTS_INT_EQUAL(tc, APR_EOF, rv); |
| rv = apr_file_eof(f); |
| ABTS_INT_EQUAL(tc, APR_EOF, rv); |
| |
| apr_file_close(f); |
| |
| rv = apr_stat(&finfo, fname, APR_FINFO_SIZE, p); |
| APR_ASSERT_SUCCESS(tc, "stat file", rv); |
| ABTS_INT_EQUAL(tc, 1, (int)finfo.size); |
| |
| apr_file_remove(fname, p); |
| } |
| |
| static void test_bigfprintf(abts_case *tc, void *data) |
| { |
| apr_file_t *f; |
| const char *fname = "data/testbigfprintf.dat"; |
| char *to_write; |
| int i; |
| |
| apr_file_remove(fname, p); |
| |
| APR_ASSERT_SUCCESS(tc, "open test file", |
| apr_file_open(&f, fname, |
| APR_FOPEN_CREATE|APR_FOPEN_WRITE, |
| APR_FPROT_UREAD|APR_FPROT_UWRITE, p)); |
| |
| |
| to_write = malloc(HUGE_STRING_LEN + 3); |
| |
| for (i = 0; i < HUGE_STRING_LEN + 1; ++i) |
| to_write[i] = 'A' + i%26; |
| |
| strcpy(to_write + HUGE_STRING_LEN, "42"); |
| |
| i = apr_file_printf(f, "%s", to_write); |
| ABTS_INT_EQUAL(tc, HUGE_STRING_LEN + 2, i); |
| |
| apr_file_close(f); |
| |
| file_contents_equal(tc, fname, to_write, HUGE_STRING_LEN + 2); |
| |
| free(to_write); |
| } |
| |
| static void test_fail_write_flush(abts_case *tc, void *data) |
| { |
| apr_file_t *f; |
| const char *fname = "data/testflush.dat"; |
| apr_status_t rv; |
| char buf[APR_BUFFERSIZE]; |
| int n; |
| |
| apr_file_remove(fname, p); |
| |
| APR_ASSERT_SUCCESS(tc, "open test file", |
| apr_file_open(&f, fname, |
| APR_FOPEN_CREATE|APR_FOPEN_READ|APR_FOPEN_BUFFERED, |
| APR_FPROT_UREAD|APR_FPROT_UWRITE, p)); |
| |
| memset(buf, 'A', sizeof buf); |
| |
| /* Try three writes. One of these should fail when it exceeds the |
| * internal buffer and actually tries to write to the file, which |
| * was opened read-only and hence should be unwritable. */ |
| for (n = 0, rv = APR_SUCCESS; n < 4 && rv == APR_SUCCESS; n++) { |
| apr_size_t bytes = sizeof buf; |
| rv = apr_file_write(f, buf, &bytes); |
| } |
| |
| ABTS_ASSERT(tc, "failed to write to read-only buffered fd", |
| rv != APR_SUCCESS); |
| |
| apr_file_close(f); |
| } |
| |
| static void test_fail_read_flush(abts_case *tc, void *data) |
| { |
| apr_file_t *f; |
| const char *fname = "data/testflush.dat"; |
| apr_status_t rv; |
| char buf[2]; |
| |
| apr_file_remove(fname, p); |
| |
| APR_ASSERT_SUCCESS(tc, "open test file", |
| apr_file_open(&f, fname, |
| APR_FOPEN_CREATE|APR_FOPEN_READ|APR_FOPEN_BUFFERED, |
| APR_FPROT_UREAD|APR_FPROT_UWRITE, p)); |
| |
| /* this write should be buffered. */ |
| APR_ASSERT_SUCCESS(tc, "buffered write should succeed", |
| apr_file_puts("hello", f)); |
| |
| /* Now, trying a read should fail since the write must be flushed, |
| * and should fail with something other than EOF since the file is |
| * opened read-only. */ |
| rv = apr_file_read_full(f, buf, 2, NULL); |
| |
| ABTS_ASSERT(tc, "read should flush buffered write and fail", |
| rv != APR_SUCCESS && rv != APR_EOF); |
| |
| /* Likewise for gets */ |
| rv = apr_file_gets(buf, 2, f); |
| |
| ABTS_ASSERT(tc, "gets should flush buffered write and fail", |
| rv != APR_SUCCESS && rv != APR_EOF); |
| |
| /* Likewise for seek. */ |
| { |
| apr_off_t offset = 0; |
| |
| rv = apr_file_seek(f, APR_SET, &offset); |
| } |
| |
| ABTS_ASSERT(tc, "seek should flush buffered write and fail", |
| rv != APR_SUCCESS && rv != APR_EOF); |
| |
| apr_file_close(f); |
| } |
| |
| static void test_xthread(abts_case *tc, void *data) |
| { |
| apr_file_t *f; |
| const char *fname = "data/testxthread.dat"; |
| apr_status_t rv; |
| apr_int32_t flags = APR_FOPEN_CREATE|APR_FOPEN_READ|APR_FOPEN_WRITE|APR_FOPEN_APPEND|APR_FOPEN_XTHREAD; |
| char buf[128] = { 0 }; |
| |
| /* Test for bug 38438, opening file with append + xthread and seeking to |
| the end of the file resulted in writes going to the beginning not the |
| end. */ |
| |
| apr_file_remove(fname, p); |
| |
| APR_ASSERT_SUCCESS(tc, "open test file", |
| apr_file_open(&f, fname, flags, |
| APR_FPROT_UREAD|APR_FPROT_UWRITE, p)); |
| |
| APR_ASSERT_SUCCESS(tc, "write should succeed", |
| apr_file_puts("hello", f)); |
| |
| apr_file_close(f); |
| |
| APR_ASSERT_SUCCESS(tc, "open test file", |
| apr_file_open(&f, fname, flags, |
| APR_FPROT_UREAD|APR_FPROT_UWRITE, p)); |
| |
| /* Seek to the end. */ |
| { |
| apr_off_t offset = 0; |
| |
| rv = apr_file_seek(f, APR_END, &offset); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| } |
| |
| APR_ASSERT_SUCCESS(tc, "more writes should succeed", |
| apr_file_puts("world", f)); |
| |
| /* Back to the beginning. */ |
| { |
| apr_off_t offset = 0; |
| |
| rv = apr_file_seek(f, APR_SET, &offset); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| } |
| |
| apr_file_read_full(f, buf, sizeof(buf), NULL); |
| |
| ABTS_STR_EQUAL(tc, "helloworld", buf); |
| |
| apr_file_close(f); |
| } |
| |
| static void test_append(abts_case *tc, void *data) |
| { |
| apr_file_t *f1; |
| apr_file_t *f2; |
| const char *fname = "data/testappend.dat"; |
| apr_int32_t flags = APR_FOPEN_CREATE | APR_FOPEN_READ | APR_FOPEN_WRITE | APR_FOPEN_APPEND; |
| char buf[128]; |
| apr_off_t offset; |
| |
| apr_file_remove(fname, p); |
| |
| /* Open test file with APR_FOPEN_APPEND, but without APR_FOPEN_XTHREAD. */ |
| APR_ASSERT_SUCCESS(tc, "open test file", |
| apr_file_open(&f1, fname, flags, APR_FPROT_OS_DEFAULT, p)); |
| |
| /* Open test file with APR_FOPEN_APPEND and APR_FOPEN_XTHREAD. */ |
| APR_ASSERT_SUCCESS(tc, "open test file", |
| apr_file_open(&f2, fname, flags | APR_FOPEN_XTHREAD, APR_FPROT_OS_DEFAULT, p)); |
| |
| APR_ASSERT_SUCCESS(tc, "write should succeed", |
| apr_file_puts("w1", f1)); |
| offset = 0; |
| APR_ASSERT_SUCCESS(tc, "seek should succeed", |
| apr_file_seek(f1, APR_CUR, &offset)); |
| ABTS_INT_EQUAL(tc, 2, (int) offset); |
| |
| APR_ASSERT_SUCCESS(tc, "write should succeed", |
| apr_file_puts("w2", f2)); |
| offset = 0; |
| APR_ASSERT_SUCCESS(tc, "seek should succeed", |
| apr_file_seek(f2, APR_CUR, &offset)); |
| ABTS_INT_EQUAL(tc, 4, (int) offset); |
| |
| APR_ASSERT_SUCCESS(tc, "write should succeed", |
| apr_file_puts("w3", f1)); |
| offset = 0; |
| APR_ASSERT_SUCCESS(tc, "seek should succeed", |
| apr_file_seek(f1, APR_CUR, &offset)); |
| ABTS_INT_EQUAL(tc, 6, (int) offset); |
| |
| APR_ASSERT_SUCCESS(tc, "write should succeed", |
| apr_file_puts("w4", f2)); |
| |
| offset = 0; |
| APR_ASSERT_SUCCESS(tc, "seek should succeed", |
| apr_file_seek(f2, APR_CUR, &offset)); |
| ABTS_INT_EQUAL(tc, 8, (int) offset); |
| |
| /* Check file content file using F1. */ |
| offset = 0; |
| APR_ASSERT_SUCCESS(tc, "seek should succeed", |
| apr_file_seek(f1, APR_SET, &offset)); |
| memset(buf, 0, sizeof(buf)); |
| apr_file_read_full(f1, buf, sizeof(buf), NULL); |
| ABTS_STR_EQUAL(tc, "w1w2w3w4", buf); |
| |
| /* Check file content file using F2. */ |
| offset = 0; |
| APR_ASSERT_SUCCESS(tc, "seek should succeed", |
| apr_file_seek(f2, APR_SET, &offset)); |
| memset(buf, 0, sizeof(buf)); |
| apr_file_read_full(f2, buf, sizeof(buf), NULL); |
| ABTS_STR_EQUAL(tc, "w1w2w3w4", buf); |
| |
| apr_file_close(f1); |
| apr_file_close(f2); |
| } |
| |
| static void test_large_write_buffered(abts_case *tc, void *data) |
| { |
| apr_status_t rv; |
| apr_file_t *f; |
| const char *fname = "data/testlarge_write_buffered.dat"; |
| apr_size_t len; |
| apr_size_t bytes_written; |
| apr_size_t bytes_read; |
| char *buf; |
| char *buf2; |
| |
| apr_file_remove(fname, p); |
| |
| rv = apr_file_open(&f, fname, |
| APR_FOPEN_CREATE | APR_FOPEN_WRITE | APR_FOPEN_BUFFERED, |
| APR_FPROT_OS_DEFAULT, p); |
| APR_ASSERT_SUCCESS(tc, "open test file for writing", rv); |
| |
| /* Test a single large write. */ |
| len = 80000; |
| buf = apr_palloc(p, len); |
| memset(buf, 'a', len); |
| rv = apr_file_write_full(f, buf, len, &bytes_written); |
| APR_ASSERT_SUCCESS(tc, "write to file", rv); |
| ABTS_INT_EQUAL(tc, (int)len, (int)bytes_written); |
| apr_file_close(f); |
| |
| rv = apr_file_open(&f, fname, APR_FOPEN_READ, |
| APR_FPROT_OS_DEFAULT, p); |
| APR_ASSERT_SUCCESS(tc, "open test file for reading", rv); |
| |
| buf2 = apr_palloc(p, len + 1); |
| rv = apr_file_read_full(f, buf2, len + 1, &bytes_read); |
| ABTS_INT_EQUAL(tc, APR_EOF, rv); |
| ABTS_INT_EQUAL(tc, (int)len, (int)bytes_read); |
| ABTS_TRUE(tc, memcmp(buf, buf2, len) == 0); |
| apr_file_close(f); |
| |
| apr_file_remove(fname, p); |
| } |
| |
| static void test_two_large_writes_buffered(abts_case *tc, void *data) |
| { |
| apr_status_t rv; |
| apr_file_t *f; |
| const char *fname = "data/testtwo_large_writes_buffered.dat"; |
| apr_size_t len; |
| apr_size_t bytes_written; |
| apr_size_t bytes_read; |
| char *buf; |
| char *buf2; |
| |
| apr_file_remove(fname, p); |
| |
| rv = apr_file_open(&f, fname, |
| APR_FOPEN_CREATE | APR_FOPEN_WRITE | APR_FOPEN_BUFFERED, |
| APR_FPROT_OS_DEFAULT, p); |
| APR_ASSERT_SUCCESS(tc, "open test file for writing", rv); |
| |
| /* Test two consecutive large writes. */ |
| len = 80000; |
| buf = apr_palloc(p, len); |
| memset(buf, 'a', len); |
| |
| rv = apr_file_write_full(f, buf, len / 2, &bytes_written); |
| APR_ASSERT_SUCCESS(tc, "write to file", rv); |
| ABTS_INT_EQUAL(tc, (int)(len / 2), (int)bytes_written); |
| |
| rv = apr_file_write_full(f, buf, len / 2, &bytes_written); |
| APR_ASSERT_SUCCESS(tc, "write to file", rv); |
| ABTS_INT_EQUAL(tc, (int)(len / 2), (int)bytes_written); |
| |
| apr_file_close(f); |
| |
| rv = apr_file_open(&f, fname, APR_FOPEN_READ, |
| APR_FPROT_OS_DEFAULT, p); |
| APR_ASSERT_SUCCESS(tc, "open test file for reading", rv); |
| |
| buf2 = apr_palloc(p, len + 1); |
| rv = apr_file_read_full(f, buf2, len + 1, &bytes_read); |
| ABTS_INT_EQUAL(tc, APR_EOF, rv); |
| ABTS_INT_EQUAL(tc, (int) len, (int)bytes_read); |
| ABTS_TRUE(tc, memcmp(buf, buf2, len) == 0); |
| apr_file_close(f); |
| |
| apr_file_remove(fname, p); |
| } |
| |
| static void test_small_and_large_writes_buffered(abts_case *tc, void *data) |
| { |
| apr_status_t rv; |
| apr_file_t *f; |
| const char *fname = "data/testtwo_large_writes_buffered.dat"; |
| apr_size_t len; |
| apr_size_t bytes_written; |
| apr_size_t bytes_read; |
| char *buf; |
| char *buf2; |
| |
| apr_file_remove(fname, p); |
| |
| rv = apr_file_open(&f, fname, |
| APR_FOPEN_CREATE | APR_FOPEN_WRITE | APR_FOPEN_BUFFERED, |
| APR_FPROT_OS_DEFAULT, p); |
| APR_ASSERT_SUCCESS(tc, "open test file for writing", rv); |
| |
| /* Test small write followed by a large write. */ |
| len = 80000; |
| buf = apr_palloc(p, len); |
| memset(buf, 'a', len); |
| |
| rv = apr_file_write_full(f, buf, 5, &bytes_written); |
| APR_ASSERT_SUCCESS(tc, "write to file", rv); |
| ABTS_INT_EQUAL(tc, 5, (int)bytes_written); |
| |
| rv = apr_file_write_full(f, buf, len - 5, &bytes_written); |
| APR_ASSERT_SUCCESS(tc, "write to file", rv); |
| ABTS_INT_EQUAL(tc, (int)(len - 5), (int)bytes_written); |
| |
| apr_file_close(f); |
| |
| rv = apr_file_open(&f, fname, APR_FOPEN_READ, |
| APR_FPROT_OS_DEFAULT, p); |
| APR_ASSERT_SUCCESS(tc, "open test file for reading", rv); |
| |
| buf2 = apr_palloc(p, len + 1); |
| rv = apr_file_read_full(f, buf2, len + 1, &bytes_read); |
| ABTS_INT_EQUAL(tc, APR_EOF, rv); |
| ABTS_INT_EQUAL(tc, (int) len, (int)bytes_read); |
| ABTS_TRUE(tc, memcmp(buf, buf2, len) == 0); |
| apr_file_close(f); |
| |
| apr_file_remove(fname, p); |
| } |
| |
| static void test_write_buffered_spanning_over_bufsize(abts_case *tc, void *data) |
| { |
| apr_status_t rv; |
| apr_file_t *f; |
| const char *fname = "data/testwrite_buffered_spanning_over_bufsize.dat"; |
| apr_size_t len; |
| apr_size_t bytes_written; |
| apr_size_t bytes_read; |
| char *buf; |
| char *buf2; |
| |
| apr_file_remove(fname, p); |
| |
| rv = apr_file_open(&f, fname, |
| APR_FOPEN_CREATE | APR_FOPEN_WRITE | APR_FOPEN_BUFFERED, |
| APR_FPROT_OS_DEFAULT, p); |
| APR_ASSERT_SUCCESS(tc, "open test file for writing", rv); |
| |
| /* Test three writes than span over the default buffer size. */ |
| len = APR_BUFFERSIZE + 1; |
| buf = apr_palloc(p, len); |
| memset(buf, 'a', len); |
| |
| rv = apr_file_write_full(f, buf, APR_BUFFERSIZE - 1, &bytes_written); |
| APR_ASSERT_SUCCESS(tc, "write to file", rv); |
| ABTS_INT_EQUAL(tc, APR_BUFFERSIZE - 1, (int)bytes_written); |
| |
| rv = apr_file_write_full(f, buf, 2, &bytes_written); |
| APR_ASSERT_SUCCESS(tc, "write to file", rv); |
| ABTS_INT_EQUAL(tc, 2, (int)bytes_written); |
| |
| apr_file_close(f); |
| |
| rv = apr_file_open(&f, fname, APR_FOPEN_READ, |
| APR_FPROT_OS_DEFAULT, p); |
| APR_ASSERT_SUCCESS(tc, "open test file for reading", rv); |
| |
| buf2 = apr_palloc(p, len + 1); |
| rv = apr_file_read_full(f, buf2, len + 1, &bytes_read); |
| ABTS_INT_EQUAL(tc, APR_EOF, rv); |
| ABTS_INT_EQUAL(tc, (int)len, (int)bytes_read); |
| ABTS_TRUE(tc, memcmp(buf, buf2, len) == 0); |
| apr_file_close(f); |
| |
| apr_file_remove(fname, p); |
| } |
| |
| #if APR_HAS_THREADS |
| typedef struct thread_file_append_ctx_t { |
| apr_pool_t *pool; |
| const char *fname; |
| apr_size_t chunksize; |
| char val; |
| int num_writes; |
| char *errmsg; |
| } thread_file_append_ctx_t; |
| |
| static void * APR_THREAD_FUNC thread_file_append_func(apr_thread_t *thd, void *data) |
| { |
| thread_file_append_ctx_t *ctx = data; |
| apr_status_t rv; |
| apr_file_t *f; |
| int i; |
| char *writebuf; |
| char *readbuf; |
| |
| rv = apr_file_open(&f, ctx->fname, |
| APR_FOPEN_READ | APR_FOPEN_WRITE | APR_FOPEN_APPEND, |
| APR_FPROT_OS_DEFAULT, ctx->pool); |
| if (rv) { |
| apr_thread_exit(thd, rv); |
| return NULL; |
| } |
| |
| writebuf = apr_palloc(ctx->pool, ctx->chunksize); |
| memset(writebuf, ctx->val, ctx->chunksize); |
| readbuf = apr_palloc(ctx->pool, ctx->chunksize); |
| |
| for (i = 0; i < ctx->num_writes; i++) { |
| apr_size_t bytes_written; |
| apr_size_t bytes_read; |
| apr_off_t offset; |
| |
| rv = apr_file_write_full(f, writebuf, ctx->chunksize, &bytes_written); |
| if (rv) { |
| apr_thread_exit(thd, rv); |
| return NULL; |
| } |
| /* After writing the data, seek back from the current offset and |
| * verify what we just wrote. */ |
| offset = -((apr_off_t)ctx->chunksize); |
| rv = apr_file_seek(f, APR_CUR, &offset); |
| if (rv) { |
| apr_thread_exit(thd, rv); |
| return NULL; |
| } |
| rv = apr_file_read_full(f, readbuf, ctx->chunksize, &bytes_read); |
| if (rv) { |
| apr_thread_exit(thd, rv); |
| return NULL; |
| } |
| if (memcmp(readbuf, writebuf, ctx->chunksize) != 0) { |
| ctx->errmsg = apr_psprintf( |
| ctx->pool, |
| "Unexpected data at file offset %" APR_OFF_T_FMT, |
| offset); |
| apr_thread_exit(thd, APR_SUCCESS); |
| return NULL; |
| } |
| } |
| |
| apr_file_close(f); |
| apr_thread_exit(thd, APR_SUCCESS); |
| |
| return NULL; |
| } |
| #endif /* APR_HAS_THREADS */ |
| |
| static void test_atomic_append(abts_case *tc, void *data) |
| { |
| #if APR_HAS_THREADS |
| apr_status_t rv; |
| apr_status_t thread_rv; |
| apr_file_t *f; |
| const char *fname = "data/testatomic_append.dat"; |
| unsigned int seed; |
| thread_file_append_ctx_t ctx1 = {0}; |
| thread_file_append_ctx_t ctx2 = {0}; |
| apr_thread_t *t1; |
| apr_thread_t *t2; |
| |
| apr_file_remove(fname, p); |
| |
| rv = apr_file_open(&f, fname, APR_FOPEN_WRITE | APR_FOPEN_CREATE, |
| APR_FPROT_OS_DEFAULT, p); |
| APR_ASSERT_SUCCESS(tc, "create file", rv); |
| apr_file_close(f); |
| |
| seed = (unsigned int)apr_time_now(); |
| abts_log_message("Random seed for test_atomic_append() is %u", seed); |
| srand(seed); |
| |
| /* Create two threads appending data to the same file. */ |
| apr_pool_create(&ctx1.pool, p); |
| ctx1.fname = fname; |
| ctx1.chunksize = 1 + rand() % 8192; |
| ctx1.val = 'A'; |
| ctx1.num_writes = 1000; |
| rv = apr_thread_create(&t1, NULL, thread_file_append_func, &ctx1, p); |
| APR_ASSERT_SUCCESS(tc, "create thread", rv); |
| |
| apr_pool_create(&ctx2.pool, p); |
| ctx2.fname = fname; |
| ctx2.chunksize = 1 + rand() % 8192; |
| ctx2.val = 'B'; |
| ctx2.num_writes = 1000; |
| rv = apr_thread_create(&t2, NULL, thread_file_append_func, &ctx2, p); |
| APR_ASSERT_SUCCESS(tc, "create thread", rv); |
| |
| rv = apr_thread_join(&thread_rv, t1); |
| APR_ASSERT_SUCCESS(tc, "join thread", rv); |
| APR_ASSERT_SUCCESS(tc, "no thread errors", thread_rv); |
| if (ctx1.errmsg) { |
| ABTS_FAIL(tc, ctx1.errmsg); |
| } |
| rv = apr_thread_join(&thread_rv, t2); |
| APR_ASSERT_SUCCESS(tc, "join thread", rv); |
| APR_ASSERT_SUCCESS(tc, "no thread errors", thread_rv); |
| if (ctx2.errmsg) { |
| ABTS_FAIL(tc, ctx2.errmsg); |
| } |
| |
| apr_file_remove(fname, p); |
| #else |
| ABTS_SKIP(tc, data, "This test requires APR thread support."); |
| #endif /* APR_HAS_THREADS */ |
| } |
| |
| static void test_append_locked(abts_case *tc, void *data) |
| { |
| apr_status_t rv; |
| apr_file_t *f; |
| const char *fname = "data/testappend_locked.dat"; |
| apr_size_t bytes_written; |
| apr_size_t bytes_read; |
| char buf[64] = {0}; |
| |
| apr_file_remove(fname, p); |
| |
| rv = apr_file_open(&f, fname, |
| APR_FOPEN_WRITE | APR_FOPEN_CREATE | APR_FOPEN_APPEND, |
| APR_FPROT_OS_DEFAULT, p); |
| APR_ASSERT_SUCCESS(tc, "create file", rv); |
| |
| rv = apr_file_lock(f, APR_FLOCK_EXCLUSIVE); |
| APR_ASSERT_SUCCESS(tc, "lock file", rv); |
| |
| /* PR50058: Appending to a locked file should not deadlock. */ |
| rv = apr_file_write_full(f, "abc", 3, &bytes_written); |
| APR_ASSERT_SUCCESS(tc, "write to file", rv); |
| |
| apr_file_unlock(f); |
| apr_file_close(f); |
| |
| rv = apr_file_open(&f, fname, APR_FOPEN_READ, APR_FPROT_OS_DEFAULT, p); |
| APR_ASSERT_SUCCESS(tc, "open file", rv); |
| |
| rv = apr_file_read_full(f, buf, sizeof(buf), &bytes_read); |
| ABTS_INT_EQUAL(tc, APR_EOF, rv); |
| ABTS_INT_EQUAL(tc, 3, (int)bytes_read); |
| ABTS_STR_EQUAL(tc, "abc", buf); |
| |
| apr_file_close(f); |
| apr_file_remove(fname, p); |
| } |
| |
| static void test_append_read(abts_case *tc, void *data) |
| { |
| apr_status_t rv; |
| apr_file_t *f; |
| const char *fname = "data/testappend_read.dat"; |
| apr_off_t offset; |
| char buf[64]; |
| |
| apr_file_remove(fname, p); |
| |
| /* Create file with contents. */ |
| rv = apr_file_open(&f, fname, APR_FOPEN_WRITE | APR_FOPEN_CREATE, |
| APR_FPROT_OS_DEFAULT, p); |
| APR_ASSERT_SUCCESS(tc, "create file", rv); |
| |
| rv = apr_file_puts("abc", f); |
| APR_ASSERT_SUCCESS(tc, "write to file", rv); |
| apr_file_close(f); |
| |
| /* Re-open it with APR_FOPEN_APPEND. */ |
| rv = apr_file_open(&f, fname, |
| APR_FOPEN_READ | APR_FOPEN_WRITE | APR_FOPEN_APPEND, |
| APR_FPROT_OS_DEFAULT, p); |
| APR_ASSERT_SUCCESS(tc, "open file", rv); |
| |
| /* Test the initial file offset. Even though we used APR_FOPEN_APPEND, |
| * the offset should be kept in the beginning of the file until the |
| * first append. (Previously, the Windows implementation performed |
| * an erroneous seek to the file's end right after opening it.) |
| */ |
| offset = 0; |
| rv = apr_file_seek(f, APR_CUR, &offset); |
| APR_ASSERT_SUCCESS(tc, "get file offset", rv); |
| ABTS_INT_EQUAL(tc, 0, (int)offset); |
| |
| /* Test reading from the file. */ |
| rv = apr_file_gets(buf, sizeof(buf), f); |
| APR_ASSERT_SUCCESS(tc, "read from file", rv); |
| ABTS_STR_EQUAL(tc, "abc", buf); |
| |
| /* Test the file offset after reading. */ |
| offset = 0; |
| rv = apr_file_seek(f, APR_CUR, &offset); |
| APR_ASSERT_SUCCESS(tc, "get file offset", rv); |
| ABTS_INT_EQUAL(tc, 3, (int)offset); |
| |
| apr_file_close(f); |
| apr_file_remove(fname, p); |
| } |
| |
| static void test_empty_read_buffered(abts_case *tc, void *data) |
| { |
| apr_status_t rv; |
| apr_file_t *f; |
| const char *fname = "data/testempty_read_buffered.dat"; |
| apr_size_t len; |
| apr_size_t bytes_read; |
| char buf[64]; |
| |
| apr_file_remove(fname, p); |
| |
| rv = apr_file_open(&f, fname, APR_FOPEN_CREATE | APR_FOPEN_WRITE, |
| APR_FPROT_OS_DEFAULT, p); |
| APR_ASSERT_SUCCESS(tc, "create empty test file", rv); |
| apr_file_close(f); |
| |
| rv = apr_file_open(&f, fname, APR_FOPEN_READ | APR_FOPEN_BUFFERED, |
| APR_FPROT_OS_DEFAULT, p); |
| APR_ASSERT_SUCCESS(tc, "open test file for reading", rv); |
| |
| /* Test an empty read. */ |
| len = 1; |
| rv = apr_file_read_full(f, buf, len, &bytes_read); |
| ABTS_INT_EQUAL(tc, APR_EOF, rv); |
| ABTS_INT_EQUAL(tc, 0, (int)bytes_read); |
| rv = apr_file_eof(f); |
| ABTS_INT_EQUAL(tc, APR_EOF, rv); |
| apr_file_close(f); |
| |
| apr_file_remove(fname, p); |
| } |
| |
| static void test_large_read_buffered(abts_case *tc, void *data) |
| { |
| apr_status_t rv; |
| apr_file_t *f; |
| const char *fname = "data/testlarge_read_buffered.dat"; |
| apr_size_t len; |
| apr_size_t bytes_written; |
| apr_size_t bytes_read; |
| char *buf; |
| char *buf2; |
| |
| apr_file_remove(fname, p); |
| |
| rv = apr_file_open(&f, fname, APR_FOPEN_CREATE | APR_FOPEN_WRITE, |
| APR_FPROT_OS_DEFAULT, p); |
| APR_ASSERT_SUCCESS(tc, "open test file for writing", rv); |
| len = 80000; |
| buf = apr_palloc(p, len); |
| memset(buf, 'a', len); |
| rv = apr_file_write_full(f, buf, len, &bytes_written); |
| APR_ASSERT_SUCCESS(tc, "write to file", rv); |
| apr_file_close(f); |
| |
| rv = apr_file_open(&f, fname, APR_FOPEN_READ | APR_FOPEN_BUFFERED, |
| APR_FPROT_OS_DEFAULT, p); |
| APR_ASSERT_SUCCESS(tc, "open test file for reading", rv); |
| |
| /* Test a single large read. */ |
| buf2 = apr_palloc(p, len); |
| rv = apr_file_read_full(f, buf2, len, &bytes_read); |
| APR_ASSERT_SUCCESS(tc, "read from file", rv); |
| ABTS_INT_EQUAL(tc, (int)len, (int)bytes_read); |
| ABTS_TRUE(tc, memcmp(buf, buf2, bytes_read) == 0); |
| rv = apr_file_eof(f); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| |
| /* Test that we receive an EOF. */ |
| len = 1; |
| rv = apr_file_read_full(f, buf2, len, &bytes_read); |
| ABTS_INT_EQUAL(tc, APR_EOF, rv); |
| ABTS_INT_EQUAL(tc, 0, (int)bytes_read); |
| rv = apr_file_eof(f); |
| ABTS_INT_EQUAL(tc, APR_EOF, rv); |
| apr_file_close(f); |
| |
| apr_file_remove(fname, p); |
| } |
| |
| static void test_two_large_reads_buffered(abts_case *tc, void *data) |
| { |
| apr_status_t rv; |
| apr_file_t *f; |
| const char *fname = "data/testtwo_large_reads_buffered.dat"; |
| apr_size_t len; |
| apr_size_t bytes_written; |
| apr_size_t bytes_read; |
| char *buf; |
| char *buf2; |
| |
| apr_file_remove(fname, p); |
| |
| rv = apr_file_open(&f, fname, APR_FOPEN_CREATE | APR_FOPEN_WRITE, |
| APR_FPROT_OS_DEFAULT, p); |
| APR_ASSERT_SUCCESS(tc, "open test file for writing", rv); |
| len = 80000; |
| buf = apr_palloc(p, len); |
| memset(buf, 'a', len); |
| rv = apr_file_write_full(f, buf, len, &bytes_written); |
| APR_ASSERT_SUCCESS(tc, "write to file", rv); |
| apr_file_close(f); |
| |
| rv = apr_file_open(&f, fname, APR_FOPEN_READ | APR_FOPEN_BUFFERED, |
| APR_FPROT_OS_DEFAULT, p); |
| APR_ASSERT_SUCCESS(tc, "open test file for reading", rv); |
| |
| /* Test two consecutive large reads. */ |
| buf2 = apr_palloc(p, len); |
| memset(buf2, 0, len); |
| rv = apr_file_read_full(f, buf2, len / 2, &bytes_read); |
| APR_ASSERT_SUCCESS(tc, "read from file", rv); |
| ABTS_INT_EQUAL(tc, (int)(len / 2), (int)bytes_read); |
| ABTS_TRUE(tc, memcmp(buf, buf2, bytes_read) == 0); |
| rv = apr_file_eof(f); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| |
| memset(buf2, 0, len); |
| rv = apr_file_read_full(f, buf2, len / 2, &bytes_read); |
| APR_ASSERT_SUCCESS(tc, "read from file", rv); |
| ABTS_INT_EQUAL(tc, (int)(len / 2), (int)bytes_read); |
| ABTS_TRUE(tc, memcmp(buf + len / 2, buf2, bytes_read) == 0); |
| rv = apr_file_eof(f); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| |
| /* Test that we receive an EOF. */ |
| len = 1; |
| rv = apr_file_read_full(f, buf2, len, &bytes_read); |
| ABTS_INT_EQUAL(tc, APR_EOF, rv); |
| ABTS_INT_EQUAL(tc, 0, (int)bytes_read); |
| rv = apr_file_eof(f); |
| ABTS_INT_EQUAL(tc, APR_EOF, rv); |
| apr_file_close(f); |
| |
| apr_file_remove(fname, p); |
| } |
| |
| static void test_small_and_large_reads_buffered(abts_case *tc, void *data) |
| { |
| apr_status_t rv; |
| apr_file_t *f; |
| const char *fname = "data/testtwo_large_reads_buffered.dat"; |
| apr_size_t len; |
| apr_size_t bytes_written; |
| apr_size_t bytes_read; |
| char *buf; |
| char *buf2; |
| |
| apr_file_remove(fname, p); |
| |
| rv = apr_file_open(&f, fname, APR_FOPEN_CREATE | APR_FOPEN_WRITE, |
| APR_FPROT_OS_DEFAULT, p); |
| APR_ASSERT_SUCCESS(tc, "open test file for writing", rv); |
| len = 80000; |
| buf = apr_palloc(p, len); |
| memset(buf, 'a', len); |
| rv = apr_file_write_full(f, buf, len, &bytes_written); |
| APR_ASSERT_SUCCESS(tc, "write to file", rv); |
| apr_file_close(f); |
| |
| rv = apr_file_open(&f, fname, APR_FOPEN_READ | APR_FOPEN_BUFFERED, |
| APR_FPROT_OS_DEFAULT, p); |
| APR_ASSERT_SUCCESS(tc, "open test file for reading", rv); |
| |
| /* Test small read followed by a large read. */ |
| buf2 = apr_palloc(p, len); |
| memset(buf2, 0, len); |
| rv = apr_file_read_full(f, buf2, 5, &bytes_read); |
| APR_ASSERT_SUCCESS(tc, "read from file", rv); |
| ABTS_INT_EQUAL(tc, 5, (int)bytes_read); |
| ABTS_TRUE(tc, memcmp(buf, buf2, bytes_read) == 0); |
| rv = apr_file_eof(f); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| |
| memset(buf2, 0, len); |
| rv = apr_file_read_full(f, buf2, len - 5, &bytes_read); |
| APR_ASSERT_SUCCESS(tc, "read from file", rv); |
| ABTS_INT_EQUAL(tc, (int)(len - 5), (int)bytes_read); |
| ABTS_TRUE(tc, memcmp(buf + 5, buf2, bytes_read) == 0); |
| rv = apr_file_eof(f); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| |
| /* Test that we receive an EOF. */ |
| len = 1; |
| rv = apr_file_read_full(f, buf2, len, &bytes_read); |
| ABTS_INT_EQUAL(tc, APR_EOF, rv); |
| ABTS_INT_EQUAL(tc, 0, (int)bytes_read); |
| rv = apr_file_eof(f); |
| ABTS_INT_EQUAL(tc, APR_EOF, rv); |
| apr_file_close(f); |
| |
| apr_file_remove(fname, p); |
| } |
| |
| static void test_read_buffered_spanning_over_bufsize(abts_case *tc, void *data) |
| { |
| apr_status_t rv; |
| apr_file_t *f; |
| const char *fname = "data/testread_buffered_spanning_over_bufsize.dat"; |
| apr_size_t len; |
| apr_size_t bytes_written; |
| apr_size_t bytes_read; |
| char *buf; |
| char *buf2; |
| |
| apr_file_remove(fname, p); |
| |
| rv = apr_file_open(&f, fname, APR_FOPEN_CREATE | APR_FOPEN_WRITE, |
| APR_FPROT_OS_DEFAULT, p); |
| APR_ASSERT_SUCCESS(tc, "open test file for writing", rv); |
| len = APR_BUFFERSIZE + 1; |
| buf = apr_palloc(p, len); |
| memset(buf, 'a', len); |
| rv = apr_file_write_full(f, buf, len, &bytes_written); |
| APR_ASSERT_SUCCESS(tc, "write to file", rv); |
| apr_file_close(f); |
| |
| rv = apr_file_open(&f, fname, APR_FOPEN_READ | APR_FOPEN_BUFFERED, |
| APR_FPROT_OS_DEFAULT, p); |
| APR_ASSERT_SUCCESS(tc, "open test file for reading", rv); |
| |
| /* Test reads than span over the default buffer size. */ |
| buf2 = apr_palloc(p, len); |
| memset(buf2, 0, len); |
| rv = apr_file_read_full(f, buf2, APR_BUFFERSIZE - 1, &bytes_read); |
| APR_ASSERT_SUCCESS(tc, "read from file", rv); |
| ABTS_INT_EQUAL(tc, APR_BUFFERSIZE - 1, (int)bytes_read); |
| ABTS_TRUE(tc, memcmp(buf, buf2, bytes_read) == 0); |
| rv = apr_file_eof(f); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| |
| memset(buf2, 0, len); |
| rv = apr_file_read_full(f, buf2, 2, &bytes_read); |
| APR_ASSERT_SUCCESS(tc, "read from file", rv); |
| ABTS_INT_EQUAL(tc, 2, (int)bytes_read); |
| ABTS_TRUE(tc, memcmp(buf, buf2, bytes_read) == 0); |
| rv = apr_file_eof(f); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| |
| /* Test that we receive an EOF. */ |
| len = 1; |
| rv = apr_file_read_full(f, buf2, len, &bytes_read); |
| ABTS_INT_EQUAL(tc, APR_EOF, rv); |
| ABTS_INT_EQUAL(tc, 0, (int)bytes_read); |
| rv = apr_file_eof(f); |
| ABTS_INT_EQUAL(tc, APR_EOF, rv); |
| apr_file_close(f); |
| |
| apr_file_remove(fname, p); |
| } |
| |
| static void test_single_byte_reads_buffered(abts_case *tc, void *data) |
| { |
| apr_status_t rv; |
| apr_file_t *f; |
| const char *fname = "data/testsingle_byte_reads_buffered.dat"; |
| apr_size_t len; |
| apr_size_t bytes_written; |
| apr_size_t bytes_read; |
| char *buf; |
| apr_size_t total; |
| |
| apr_file_remove(fname, p); |
| |
| rv = apr_file_open(&f, fname, APR_FOPEN_CREATE | APR_FOPEN_WRITE, |
| APR_FPROT_OS_DEFAULT, p); |
| APR_ASSERT_SUCCESS(tc, "open test file for writing", rv); |
| len = 40000; |
| buf = apr_palloc(p, len); |
| memset(buf, 'a', len); |
| rv = apr_file_write_full(f, buf, len, &bytes_written); |
| APR_ASSERT_SUCCESS(tc, "write to file", rv); |
| apr_file_close(f); |
| |
| rv = apr_file_open(&f, fname, APR_FOPEN_READ | APR_FOPEN_BUFFERED, |
| APR_FPROT_OS_DEFAULT, p); |
| APR_ASSERT_SUCCESS(tc, "open test file for reading", rv); |
| |
| total = 0; |
| while (1) { |
| memset(buf, 0, len); |
| rv = apr_file_read_full(f, buf, 1, &bytes_read); |
| if (rv == APR_EOF) { |
| break; |
| } |
| APR_ASSERT_SUCCESS(tc, "read from file", rv); |
| ABTS_INT_EQUAL(tc, 1, (int)bytes_read); |
| ABTS_INT_EQUAL(tc, 'a', buf[0]); |
| total += bytes_read; |
| } |
| ABTS_INT_EQUAL(tc, (int)len, (int)total); |
| |
| rv = apr_file_eof(f); |
| ABTS_INT_EQUAL(tc, APR_EOF, rv); |
| apr_file_close(f); |
| |
| apr_file_remove(fname, p); |
| } |
| |
| static void test_read_buffered_seek(abts_case *tc, void *data) |
| { |
| apr_status_t rv; |
| apr_file_t *f; |
| const char *fname = "data/testtest_read_buffered_seek.dat"; |
| apr_size_t len; |
| apr_size_t bytes_written; |
| apr_size_t bytes_read; |
| char buf[64]; |
| apr_off_t off; |
| |
| apr_file_remove(fname, p); |
| |
| rv = apr_file_open(&f, fname, APR_FOPEN_CREATE | APR_FOPEN_WRITE, |
| APR_FPROT_OS_DEFAULT, p); |
| APR_ASSERT_SUCCESS(tc, "open test file for writing", rv); |
| rv = apr_file_write_full(f, "abcdef", 6, &bytes_written); |
| APR_ASSERT_SUCCESS(tc, "write to file", rv); |
| apr_file_close(f); |
| |
| rv = apr_file_open(&f, fname, APR_FOPEN_READ | APR_FOPEN_BUFFERED, |
| APR_FPROT_OS_DEFAULT, p); |
| APR_ASSERT_SUCCESS(tc, "open test file for reading", rv); |
| |
| /* Read one byte. */ |
| memset(buf, 0, sizeof(buf)); |
| rv = apr_file_read_full(f, buf, 1, &bytes_read); |
| APR_ASSERT_SUCCESS(tc, "read from file", rv); |
| ABTS_INT_EQUAL(tc, 1, (int)bytes_read); |
| ABTS_INT_EQUAL(tc, 'a', buf[0]); |
| |
| /* Seek into the middle of the file. */ |
| off = 3; |
| rv = apr_file_seek(f, APR_SET, &off); |
| APR_ASSERT_SUCCESS(tc, "change file read offset", rv); |
| ABTS_INT_EQUAL(tc, 3, (int)off); |
| |
| /* Read three bytes. */ |
| memset(buf, 0, sizeof(buf)); |
| rv = apr_file_read_full(f, buf, 3, &bytes_read); |
| APR_ASSERT_SUCCESS(tc, "read from file", rv); |
| ABTS_INT_EQUAL(tc, 3, (int)bytes_read); |
| ABTS_TRUE(tc, memcmp(buf, "def", bytes_read) == 0); |
| |
| rv = apr_file_eof(f); |
| ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); |
| |
| /* Test that we receive an EOF. */ |
| len = 1; |
| rv = apr_file_read_full(f, buf, len, &bytes_read); |
| ABTS_INT_EQUAL(tc, APR_EOF, rv); |
| ABTS_INT_EQUAL(tc, 0, (int)bytes_read); |
| rv = apr_file_eof(f); |
| ABTS_INT_EQUAL(tc, APR_EOF, rv); |
| apr_file_close(f); |
| |
| apr_file_remove(fname, p); |
| } |
| |
| static void test_datasync_on_file(abts_case *tc, void *data) |
| { |
| apr_status_t rv; |
| apr_file_t *f; |
| const char *fname = DIRNAME "/testtest_datasync.dat"; |
| apr_size_t bytes_written; |
| |
| apr_file_remove(fname, p); |
| |
| rv = apr_file_open(&f, fname, APR_FOPEN_CREATE | APR_FOPEN_WRITE, |
| APR_FPROT_OS_DEFAULT, p); |
| APR_ASSERT_SUCCESS(tc, "open test file for writing", rv); |
| rv = apr_file_write_full(f, "abcdef", 6, &bytes_written); |
| APR_ASSERT_SUCCESS(tc, "write to file", rv); |
| rv = apr_file_datasync(f); |
| APR_ASSERT_SUCCESS(tc, "sync file contents", rv); |
| apr_file_close(f); |
| |
| apr_file_remove(fname, p); |
| } |
| |
| static void test_datasync_on_stream(abts_case *tc, void *data) |
| { |
| apr_status_t rv; |
| apr_file_t *f; |
| apr_size_t bytes_written; |
| |
| rv = apr_file_open_stdout(&f, p); |
| APR_ASSERT_SUCCESS(tc, "open stdout", rv); |
| rv = apr_file_write_full(f, "abcdef\b\b\b\b\b\b\b", 12, &bytes_written); |
| APR_ASSERT_SUCCESS(tc, "write to stdout", rv); |
| rv = apr_file_datasync(f); |
| if (rv != APR_SUCCESS) { |
| #if defined(__APPLE__) |
| ABTS_INT_EQUAL(tc, ENOTSUP, rv); |
| #else |
| ABTS_INT_EQUAL(tc, APR_EINVAL, rv); |
| #endif |
| } |
| } |
| |
| abts_suite *testfile(abts_suite *suite) |
| { |
| suite = ADD_SUITE(suite) |
| |
| abts_run_test(suite, test_open_noreadwrite, NULL); |
| abts_run_test(suite, test_open_excl, NULL); |
| abts_run_test(suite, test_open_read, NULL); |
| abts_run_test(suite, test_open_readwrite, NULL); |
| abts_run_test(suite, link_existing, NULL); |
| abts_run_test(suite, link_nonexisting, NULL); |
| abts_run_test(suite, test_read, NULL); |
| abts_run_test(suite, test_readzero, NULL); |
| abts_run_test(suite, test_seek, NULL); |
| abts_run_test(suite, test_filename, NULL); |
| abts_run_test(suite, test_fileclose, NULL); |
| abts_run_test(suite, test_file_remove, NULL); |
| abts_run_test(suite, test_open_write, NULL); |
| abts_run_test(suite, test_open_writecreate, NULL); |
| abts_run_test(suite, test_write, NULL); |
| abts_run_test(suite, test_userdata_set, NULL); |
| abts_run_test(suite, test_userdata_get, NULL); |
| abts_run_test(suite, test_userdata_getnokey, NULL); |
| abts_run_test(suite, test_getc, NULL); |
| abts_run_test(suite, test_ungetc, NULL); |
| abts_run_test(suite, test_gets, NULL); |
| abts_run_test(suite, test_gets_buffered, NULL); |
| abts_run_test(suite, test_gets_empty, NULL); |
| abts_run_test(suite, test_gets_multiline, NULL); |
| abts_run_test(suite, test_gets_small_buf, NULL); |
| abts_run_test(suite, test_gets_ungetc, NULL); |
| abts_run_test(suite, test_gets_buffered_big, NULL); |
| abts_run_test(suite, test_puts, NULL); |
| abts_run_test(suite, test_writev, NULL); |
| abts_run_test(suite, test_writev_full, NULL); |
| abts_run_test(suite, test_writev_buffered, NULL); |
| abts_run_test(suite, test_writev_buffered_seek, NULL); |
| abts_run_test(suite, test_bigread, NULL); |
| abts_run_test(suite, test_mod_neg, NULL); |
| abts_run_test(suite, test_truncate, NULL); |
| abts_run_test(suite, test_file_trunc, NULL); |
| abts_run_test(suite, test_file_trunc_buffered_write, NULL); |
| abts_run_test(suite, test_file_trunc_buffered_write2, NULL); |
| abts_run_test(suite, test_file_trunc_buffered_read, NULL); |
| abts_run_test(suite, test_bigfprintf, NULL); |
| abts_run_test(suite, test_fail_write_flush, NULL); |
| abts_run_test(suite, test_fail_read_flush, NULL); |
| abts_run_test(suite, test_buffer_set_get, NULL); |
| abts_run_test(suite, test_xthread, NULL); |
| abts_run_test(suite, test_append, NULL); |
| abts_run_test(suite, test_large_write_buffered, NULL); |
| abts_run_test(suite, test_two_large_writes_buffered, NULL); |
| abts_run_test(suite, test_small_and_large_writes_buffered, NULL); |
| abts_run_test(suite, test_write_buffered_spanning_over_bufsize, NULL); |
| abts_run_test(suite, test_atomic_append, NULL); |
| abts_run_test(suite, test_append_locked, NULL); |
| abts_run_test(suite, test_append_read, NULL); |
| abts_run_test(suite, test_empty_read_buffered, NULL); |
| abts_run_test(suite, test_large_read_buffered, NULL); |
| abts_run_test(suite, test_two_large_reads_buffered, NULL); |
| abts_run_test(suite, test_small_and_large_reads_buffered, NULL); |
| abts_run_test(suite, test_read_buffered_spanning_over_bufsize, NULL); |
| abts_run_test(suite, test_single_byte_reads_buffered, NULL); |
| abts_run_test(suite, test_read_buffered_seek, NULL); |
| abts_run_test(suite, test_datasync_on_file, NULL); |
| abts_run_test(suite, test_datasync_on_stream, NULL); |
| |
| return suite; |
| } |
| |