/*
*
* Copyright (c) International Business Machines Corp., 2001
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See
* the GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
*/
/*
* NAME
* readv02.c
*
* DESCRIPTION
* Testcase to check the error conditions of the readv(2) system call.
*
* CALLS
* readv()
*
* ALGORITHM
* Create a IO vector, and attempt to readv() various components of it.
*
* USAGE
* readv02
*
* HISTORY
* 07/2001 Ported by Wayne Boyer
*
* RESTRICTIONS
* None
*/
#include <sys/types.h>
#include <sys/uio.h>
#include <fcntl.h>
#include <sys/mman.h>
#include <memory.h>
#include <errno.h>
#include "test.h"
#include "safe_macros.h"
#define K_1 1024
#define M_1 K_1 * K_1
#define G_1 M_1 * K_1
#define NBUFS 4
#define CHUNK 64
#define MAX_IOVEC 16
#define DATA_FILE "readv_data_file"
char buf1[K_1], buf2[K_1], buf3[K_1];
struct iovec rd_iovec[MAX_IOVEC] = {
/* iov_base *//* iov_len */
/* Test case #1 */
{buf2, -1},
{(buf2 + CHUNK), CHUNK},
{(buf2 + CHUNK * 2), CHUNK},
/* Test case #2 */
{(buf2 + CHUNK * 3), G_1},
{(buf2 + CHUNK * 4), G_1},
{(buf2 + CHUNK * 5), G_1},
/* Test case #3 */
{(caddr_t) - 1, CHUNK},
{(buf2 + CHUNK * 6), CHUNK},
{(buf2 + CHUNK * 8), CHUNK},
/* Test case #4 */
{(buf2 + CHUNK * 9), CHUNK}
};
char f_name[K_1];
int fd[4];
char *buf_list[NBUFS];
char *TCID = "readv02";
int TST_TOTAL = 1;
char *bad_addr = 0;
int init_buffs(char **);
int fill_mem(char *, int, int);
long l_seek(int, long, int);
char *getenv();
void setup();
void cleanup();
int main(int ac, char **av)
{
int lc;
tst_parse_opts(ac, av, NULL, NULL);
setup();
/* The following loop checks looping state if -i option given */
for (lc = 0; TEST_LOOPING(lc); lc++) {
/* reset tst_count in case we are looping */
tst_count = 0;
//test1:
if (readv(fd[0], rd_iovec, 1) < 0) {
if (errno != EINVAL) {
tst_resm(TFAIL, "readv() set an illegal errno:"
" expected: EINVAL, got %d", errno);
} else {
tst_resm(TPASS, "got EINVAL");
}
} else {
tst_resm(TFAIL, "Error: readv returned a positive "
"value");
}
//test2:
l_seek(fd[0], CHUNK * 6, 0);
if (readv(fd[0], (rd_iovec + 6), 3) < 0) {
if (errno != EFAULT) {
tst_resm(TFAIL, "expected errno = EFAULT, "
"got %d", errno);
} else {
tst_resm(TPASS, "got EFAULT");
}
if (memcmp((buf_list[0] + CHUNK * 6),
(buf_list[1] + CHUNK * 6), CHUNK * 3) != 0) {
tst_resm(TFAIL, "Error: readv() partially "
"overlaid buf[2]");
}
} else {
tst_resm(TFAIL, "Error: readv returned a positive "
"value");
}
//test3:
if (readv(fd[1], (rd_iovec + 9), 1) < 0) {
if (errno != EBADF) {
tst_resm(TFAIL, "expected errno = EBADF, "
"got %d", errno);
} else {
tst_resm(TPASS, "got EBADF");
}
} else {
tst_resm(TFAIL, "Error: readv returned a positive "
"value");
}
//test4:
l_seek(fd[0], CHUNK * 10, 0);
if (readv(fd[0], (rd_iovec + 10), -1) < 0) {
if (errno != EINVAL) {
tst_resm(TFAIL, "expected errno = EINVAL, "
"got %d", errno);
} else {
tst_resm(TPASS, "got EINVAL");
}
} else {
tst_resm(TFAIL, "Error: readv returned a positive "
"value");
}
}
close(fd[0]);
close(fd[1]);
cleanup();
tst_exit();
}
/*
* setup() - performs all ONE TIME setup for this test.
*/
void setup(void)
{
int nbytes;
tst_sig(NOFORK, DEF_HANDLER, cleanup);
TEST_PAUSE;
/* make a temporary directory and cd to it */
tst_tmpdir();
buf_list[0] = buf1;
buf_list[1] = buf2;
buf_list[2] = buf3;
buf_list[3] = NULL;
init_buffs(buf_list);
sprintf(f_name, "%s.%d", DATA_FILE, getpid());
if ((fd[0] = open(f_name, O_WRONLY | O_CREAT, 0666)) < 0) {
tst_brkm(TBROK, cleanup, "open failed: fname = %s, "
"errno = %d", f_name, errno);
} else {
if ((nbytes = write(fd[0], buf_list[2], K_1)) != K_1) {
tst_brkm(TBROK, cleanup, "write failed: nbytes "
"= %d " "errno = %d", nbytes, errno);
}
}
SAFE_CLOSE(cleanup, fd[0]);
if ((fd[0] = open(f_name, O_RDONLY, 0666)) < 0) {
tst_brkm(TBROK, cleanup, "open failed: fname = %s, "
"errno = %d", f_name, errno);
}
fd[1] = -1; /* Invalid file descriptor */
bad_addr = mmap(0, 1, PROT_NONE,
MAP_PRIVATE_EXCEPT_UCLINUX | MAP_ANONYMOUS, 0, 0);
if (bad_addr == MAP_FAILED) {
tst_brkm(TBROK, cleanup, "mmap failed");
}
rd_iovec[6].iov_base = bad_addr;
}
/*
* cleanup() - performs all ONE TIME cleanup for this test at
* completion or premature exit.
*/
void cleanup(void)
{
SAFE_UNLINK(NULL, f_name);
tst_rmdir();
}
int init_buffs(char *pbufs[])
{
int i;
for (i = 0; pbufs[i] != NULL; i++) {
switch (i) {
case 0:
/*FALLTHROUGH*/ case 1:
fill_mem(pbufs[i], 0, 1);
break;
case 2:
fill_mem(pbufs[i], 1, 0);
break;
default:
tst_brkm(TBROK, cleanup, "Error in init_buffs()");
}
}
return 0;
}
int fill_mem(char *c_ptr, int c1, int c2)
{
int count;
for (count = 1; count <= K_1 / CHUNK; count++) {
if (count & 0x01) { /* if odd */
memset(c_ptr, c1, CHUNK);
} else { /* if even */
memset(c_ptr, c2, CHUNK);
}
}
return 0;
}
long l_seek(int fdesc, long offset, int whence)
{
SAFE_LSEEK(cleanup, fdesc, offset, whence);
return 0;
}