/* * This file is part of the Sofia-SIP package * * Copyright (C) 2005 Nokia Corporation. * * Contact: Pekka Pessi * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public License * as published by the Free Software Foundation; either version 2.1 of * the License, or (at your option) any later version. * * This library 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 * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA * 02110-1301 USA * */ /**@ingroup su_root_ex * * @file torture_su_root.c * * Test su_root_register functionality. * * @author Pekka Pessi * * Copyright (c) 2002 Nokia Research Center. All rights reserved. * * @date Created: Wed Jun 12 15:18:11 2002 ppessi */ #include "config.h" char const *name = "su_root_test"; #include #include #include #define TSTFLAGS rt->rt_flags #include typedef struct root_test_s root_test_t; typedef struct test_ep_s test_ep_t; #define SU_ROOT_MAGIC_T root_test_t #define SU_WAKEUP_ARG_T test_ep_t #include #include typedef struct test_ep_s { int i; int s; su_wait_t wait[1]; int registered; socklen_t addrlen; su_sockaddr_t addr[1]; } test_ep_at[1]; struct root_test_s { su_home_t rt_home[1]; int rt_flags; su_root_t *rt_root; short rt_family; int rt_status; int rt_received; int rt_wakeup; su_clone_r rt_clone; unsigned rt_fail_init:1; unsigned rt_fail_deinit:1; unsigned rt_success_init:1; unsigned rt_success_deinit:1; test_ep_at rt_ep[5]; }; /** Test root initialization */ int init_test(root_test_t *rt) { su_sockaddr_t su[1] = {{ 0 }}; int i; BEGIN(); su_init(); su->su_family = rt->rt_family; TEST_1(rt->rt_root = su_root_create(rt)); for (i = 0; i < 5; i++) { test_ep_t *ep = rt->rt_ep[i]; ep->i = i; ep->addrlen = su_sockaddr_size(su); TEST_1((ep->s = su_socket(su->su_family, SOCK_DGRAM, 0)) != -1); TEST_1(bind(ep->s, &su->su_sa, ep->addrlen) != -1); TEST_1(su_wait_create(ep->wait, ep->s, SU_WAIT_IN|SU_WAIT_ERR) != -1); TEST_1(getsockname(ep->s, &ep->addr->su_sa, &ep->addrlen) != -1); if (SU_HAS_INADDR_ANY(ep->addr)) { inet_pton(su->su_family, su->su_family == AF_INET ? "127.0.0.1" : "::1", SU_ADDR(ep->addr)); } } END(); } static int deinit_test(root_test_t *rt) { BEGIN(); TEST_VOID(su_root_destroy(rt->rt_root)); rt->rt_root = NULL; TEST_VOID(su_root_destroy(NULL)); su_deinit(); END(); } int wakeup(root_test_t *rt, su_wait_t *w, test_ep_t *ep) { char buffer[64]; int n, error; su_wait_events(w, ep->s); n = recv(ep->s, buffer, sizeof(buffer), 0); error = su_errno(); if (n < 0) fprintf(stderr, "%s: %s\n", "recv", su_strerror(error)); TEST_1(n > 0); rt->rt_received = ep->i; return 0; } static int wakeup0(root_test_t *rt, su_wait_t *w, test_ep_t *ep) { rt->rt_wakeup = 0; return wakeup(rt, w, ep); } static int wakeup1(root_test_t *rt, su_wait_t *w, test_ep_t *ep) { rt->rt_wakeup = 1; return wakeup(rt, w, ep); } static int wakeup2(root_test_t *rt, su_wait_t *w, test_ep_t *ep) { rt->rt_wakeup = 2; return wakeup(rt, w, ep); } static int wakeup3(root_test_t *rt, su_wait_t *w, test_ep_t *ep) { rt->rt_wakeup = 3; return wakeup(rt, w, ep); } static int wakeup4(root_test_t *rt, su_wait_t *w, test_ep_t *ep) { rt->rt_wakeup = 4; return wakeup(rt, w, ep); } static su_wakeup_f wakeups[5] = { wakeup0, wakeup1, wakeup2, wakeup3, wakeup4 }; static void test_run(root_test_t *rt) { rt->rt_received = -1; while (rt->rt_received == -1) { su_root_step(rt->rt_root, 200); } } static int register_test(root_test_t *rt) { int i; int s; char *msg = "foo"; BEGIN(); TEST_1((s = su_socket(rt->rt_family, SOCK_DGRAM, 0)) != -1); for (i = 0; i < 5; i++) { rt->rt_ep[i]->registered = su_root_register(rt->rt_root, rt->rt_ep[i]->wait, wakeups[i], rt->rt_ep[i], 0); TEST(rt->rt_ep[i]->registered, i + 1 + SU_HAVE_PTHREADS); } for (i = 0; i < 5; i++) { test_ep_t *ep = rt->rt_ep[i]; TEST_SIZE(su_sendto(s, msg, sizeof(msg), 0, ep->addr, ep->addrlen), sizeof(msg)); test_run(rt); TEST(rt->rt_received, i); TEST(rt->rt_wakeup, i); } for (i = 0; i < 5; i++) { TEST(su_root_unregister(rt->rt_root, rt->rt_ep[i]->wait, wakeups[i], rt->rt_ep[i]), rt->rt_ep[i]->registered); } for (i = 0; i < 5; i++) { rt->rt_ep[i]->registered = su_root_register(rt->rt_root, rt->rt_ep[i]->wait, wakeups[i], rt->rt_ep[i], 1); TEST_1(rt->rt_ep[i]->registered > 0); } for (i = 0; i < 5; i++) { test_ep_t *ep = rt->rt_ep[i]; TEST_SIZE(su_sendto(s, msg, sizeof(msg), 0, ep->addr, ep->addrlen), sizeof(msg)); test_run(rt); TEST(rt->rt_received, i); TEST(rt->rt_wakeup, i); } for (i = 0; i < 5; i++) { TEST(su_root_deregister(rt->rt_root, rt->rt_ep[i]->registered), rt->rt_ep[i]->registered); } for (i = 0; i < 5; i++) { test_ep_t *ep = rt->rt_ep[i]; TEST_1(su_wait_create(ep->wait, ep->s, SU_WAIT_IN|SU_WAIT_ERR) != -1); ep->registered = su_root_register(rt->rt_root, ep->wait, wakeups[i], ep, 1); TEST_1(ep->registered > 0); } for (i = 0; i < 5; i++) { test_ep_t *ep = rt->rt_ep[i]; TEST_SIZE(su_sendto(s, msg, sizeof(msg), 0, ep->addr, ep->addrlen), sizeof(msg)); test_run(rt); TEST(rt->rt_received, i); TEST(rt->rt_wakeup, i); } for (i = 0; i < 5; i++) { TEST(su_root_unregister(rt->rt_root, rt->rt_ep[i]->wait, wakeups[i], rt->rt_ep[i]), rt->rt_ep[i]->registered); } END(); } int fail_init(su_root_t *root, root_test_t *rt) { rt->rt_fail_init = 1; return -1; } void fail_deinit(su_root_t *root, root_test_t *rt) { rt->rt_fail_deinit = 1; } int success_init(su_root_t *root, root_test_t *rt) { rt->rt_success_init = 1; return 0; } void success_deinit(su_root_t *root, root_test_t *rt) { rt->rt_success_deinit = 1; } static int clone_test(root_test_t rt[1]) { BEGIN(); rt->rt_fail_init = 0; rt->rt_fail_deinit = 0; rt->rt_success_init = 0; rt->rt_success_deinit = 0; TEST(su_clone_start(rt->rt_root, rt->rt_clone, rt, fail_init, fail_deinit), SU_FAILURE); TEST_1(rt->rt_fail_init); TEST_1(rt->rt_fail_deinit); TEST(su_clone_start(rt->rt_root, rt->rt_clone, rt, success_init, success_deinit), SU_SUCCESS); TEST_1(rt->rt_success_init); TEST_1(!rt->rt_success_deinit); TEST_VOID(su_clone_wait(rt->rt_root, rt->rt_clone)); TEST_1(rt->rt_success_deinit); END(); } void usage(void) { fprintf(stderr, "usage: %s [-v]\n", name); } int main(int argc, char *argv[]) { root_test_t rt[1] = {{{ SU_HOME_INIT(rt) }}}; int retval = 0; int i; rt->rt_family = AF_INET; for (i = 1; argv[i]; i++) { if (strcmp(argv[i], "-v") == 0) rt->rt_flags |= tst_verbatim; #if SU_HAVE_IN6 else if (strcmp(argv[i], "-6") == 0) rt->rt_family = AF_INET6; #endif else usage(); } retval |= init_test(rt); retval |= register_test(rt); retval |= clone_test(rt); su_root_threading(rt->rt_root, 0); retval |= clone_test(rt); retval |= deinit_test(rt); return retval; }