/*
* Copyright (c) Crackerjack Project., 2007-2008, Hitachi, Ltd
* Copyright (c) 2017 Petr Vorel <pvorel@suse.cz>
*
* Authors:
* Takahiro Yasui <takahiro.yasui.mp@hitachi.com>,
* Yumiko Sugita <yumiko.sugita.yf@hitachi.com>,
* Satoshi Fujiwara <sa-fuji@sdl.hitachi.co.jp>
*
* 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 would 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, see <http://www.gnu.org/licenses/>.
*/
#include <errno.h>
#include <limits.h>
static int fd, fd_root, fd_nonblock, fd_maxint = INT_MAX - 1, fd_invalid = -1;
static struct timespec ts;
#include "mq_timed.h"
static struct test_case tcase[] = {
{
.fd = &fd,
.len = 0,
.ret = 0,
.err = 0,
},
{
.fd = &fd,
.len = 1,
.ret = 0,
.err = 0,
},
{
.fd = &fd,
.len = MAX_MSGSIZE,
.ret = 0,
.err = 0,
},
{
.fd = &fd,
.len = 1,
.prio = MQ_PRIO_MAX - 1,
.ret = 0,
.err = 0,
},
{
.fd = &fd,
.len = MAX_MSGSIZE + 1,
.ret = -1,
.err = EMSGSIZE,
},
{
.fd = &fd_invalid,
.len = 0,
.ret = -1,
.err = EBADF,
},
{
.fd = &fd_maxint,
.len = 0,
.ret = -1,
.err = EBADF,
},
{
.fd = &fd_root,
.len = 0,
.ret = -1,
.err = EBADF,
},
{
.fd = &fd_nonblock,
.len = 16,
.ret = -1,
.err = EAGAIN,
},
{
.fd = &fd,
.len = 1,
.prio = MQ_PRIO_MAX,
.ret = -1,
.err = EINVAL,
},
{
.fd = &fd,
.len = 16,
.rq = &(struct timespec) {.tv_sec = -1, .tv_nsec = 0},
.send = 1,
.ret = -1,
.err = EINVAL,
},
{
.fd = &fd,
.len = 16,
.rq = &(struct timespec) {.tv_sec = 0, .tv_nsec = -1},
.send = 1,
.ret = -1,
.err = EINVAL,
},
{
.fd = &fd,
.len = 16,
.rq = &(struct timespec) {.tv_sec = 0, .tv_nsec = 1000000000},
.send = 1,
.ret = -1,
.err = EINVAL,
},
{
.fd = &fd,
.len = 16,
.rq = &ts,
.send = 1,
.timeout = 1,
.ret = -1,
.err = ETIMEDOUT,
},
{
.fd = &fd,
.len = 16,
.send = 1,
.signal = 1,
.rq = &ts,
.ret = -1,
.err = EINTR,
},
};
static void do_test(unsigned int i)
{
const struct test_case *tc = &tcase[i];
unsigned int j;
unsigned int prio;
size_t len = MAX_MSGSIZE;
char rmsg[len];
pid_t pid = -1;
if (tc->signal)
pid = set_sig(tc->rq);
if (tc->timeout)
set_timeout(tc->rq);
if (tc->send) {
for (j = 0; j < MSG_LENGTH; j++)
send_msg(*tc->fd, tc->len, tc->prio);
}
TEST(mq_timedsend(*tc->fd, smsg, tc->len, tc->prio, tc->rq));
if (pid > 0)
kill_pid(pid);
if (TEST_RETURN < 0) {
if (tc->err != TEST_ERRNO)
tst_res(TFAIL | TTERRNO,
"mq_timedsend failed unexpectedly, expected %s",
tst_strerrno(tc->err));
else
tst_res(TPASS | TTERRNO, "mq_timedreceive failed expectedly");
if (*tc->fd == fd)
cleanup_queue(fd);
return;
}
TEST(mq_timedreceive(*tc->fd, rmsg, len, &prio, tc->rq));
if (*tc->fd == fd)
cleanup_queue(fd);
if (TEST_RETURN < 0) {
if (tc->err != TEST_ERRNO) {
tst_res(TFAIL | TTERRNO,
"mq_timedreceive failed unexpectedly, expected %s",
tst_strerrno(tc->err));
return;
}
if (tc->ret >= 0) {
tst_res(TFAIL | TTERRNO, "mq_timedreceive returned %ld, expected %d",
TEST_RETURN, tc->ret);
return;
}
}
if (tc->len != TEST_RETURN) {
tst_res(TFAIL, "mq_timedreceive wrong length %ld, expected %d",
TEST_RETURN, tc->len);
return;
}
if (tc->prio != prio) {
tst_res(TFAIL, "mq_timedreceive wrong prio %d, expected %d",
prio, tc->prio);
return;
}
for (j = 0; j < tc->len; j++) {
if (rmsg[j] != smsg[j]) {
tst_res(TFAIL,
"mq_timedreceive wrong data %d in %u, expected %d",
rmsg[j], i, smsg[j]);
return;
}
}
tst_res(TPASS, "mq_timedreceive returned %ld, priority %u, length: %zu",
TEST_RETURN, prio, len);
}
static struct tst_test test = {
.tcnt = ARRAY_SIZE(tcase),
.test = do_test,
.setup = setup_common,
.cleanup = cleanup_common,
.forks_child = 1,
};