/* $Id$ */ /* * Copyright (C) 2008-2011 Teluu Inc. (http://www.teluu.com) * Copyright (C) 2003-2008 Benny Prijono * * 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ #include "test.h" #include #include #define THIS_FILE "transport_tcp_test.c" /* * TCP transport test. */ #if PJ_HAS_TCP static pj_status_t multi_listener_test(pjsip_tpfactory *factory[], unsigned num_factory, pjsip_transport *tp[], unsigned *num_tp) { pj_status_t status; unsigned i = 0; pj_str_t s; pjsip_transport *tcp; pjsip_tpfactory *tpfactory = NULL; pj_sockaddr_in rem_addr; pjsip_tpselector tp_sel; unsigned ntp = 0; for (;iaddr_name.host, (pj_uint16_t)tpfactory->addr_name.port); if (status != PJ_SUCCESS) { app_perror(" Error: possibly invalid TCP address name", status); return -11; } /* Acquire transport without selector. */ status = pjsip_endpt_acquire_transport(endpt, PJSIP_TRANSPORT_TCP, &rem_addr, sizeof(rem_addr), NULL, &tcp); if (status != PJ_SUCCESS || tcp == NULL) { app_perror(" Error: unable to acquire TCP transport", status); return -12; } tp[ntp++] = tcp; /* After pjsip_endpt_acquire_transport, TCP transport must have * reference counter 1. */ if (pj_atomic_get(tcp->ref_cnt) != 1) return -13; /* Acquire with the same remote address, should return the same tp. */ status = pjsip_endpt_acquire_transport(endpt, PJSIP_TRANSPORT_TCP, &rem_addr, sizeof(rem_addr), NULL, &tcp); if (status != PJ_SUCCESS || tcp == NULL) { app_perror(" Error: unable to acquire TCP transport", status); return -14; } /* Should return existing transport. */ if (tp[ntp-1] != tcp) { return -15; } /* Using the same TCP transport, it must have reference counter 2. */ if (pj_atomic_get(tcp->ref_cnt) != 2) return -16; /* Decrease the reference. */ pjsip_transport_dec_ref(tcp); /* Test basic transport attributes */ status = generic_transport_test(tcp); if (status != PJ_SUCCESS) return status; /* Check again that reference counter is 1. */ if (pj_atomic_get(tcp->ref_cnt) != 1) return -17; /* Acquire transport test with selector. */ pj_bzero(&tp_sel, sizeof(tp_sel)); tp_sel.type = PJSIP_TPSELECTOR_LISTENER; tp_sel.u.listener = factory[num_factory/2]; pj_sockaddr_in_init(&rem_addr, pj_cstr(&s, "1.1.1.1"), 80); status = pjsip_endpt_acquire_transport(endpt, PJSIP_TRANSPORT_TCP, &rem_addr, sizeof(rem_addr), &tp_sel, &tcp); if (status != PJ_SUCCESS) { app_perror(" Error: unable to acquire TCP transport", status); return -18; } /* The transport should have the same factory set on the selector. */ if (tcp->factory != factory[num_factory/2]) return -19; /* The transport should be newly created. */ for (i = 0; i < ntp; ++i) { if (tp[i] == tcp) { break; } } if (i != ntp) return -20; tp[ntp++] = tcp; for (i = 0; iref_cnt) != 1) return -21; } *num_tp = ntp; return PJ_SUCCESS; } int transport_tcp_test(void) { enum { SEND_RECV_LOOP = 8 }; enum { NUM_LISTENER = 4 }; enum { NUM_TP = 8 }; pjsip_tpfactory *tpfactory[NUM_LISTENER]; pjsip_transport *tcp[NUM_TP]; pj_sockaddr_in rem_addr; pj_status_t status; char url[PJSIP_MAX_URL_SIZE]; char addr[PJ_INET_ADDRSTRLEN]; int rtt[SEND_RECV_LOOP], min_rtt; int pkt_lost; unsigned i; unsigned num_listener = NUM_LISTENER; unsigned num_tp = NUM_TP; status = multi_listener_test(tpfactory, num_listener, tcp, &num_tp); if (status != PJ_SUCCESS) return status; /* Get the last listener address */ status = pj_sockaddr_in_init(&rem_addr, &tpfactory[0]->addr_name.host, (pj_uint16_t)tpfactory[0]->addr_name.port); pj_ansi_sprintf(url, "sip:alice@%s:%d;transport=tcp", pj_inet_ntop2(pj_AF_INET(), &rem_addr.sin_addr, addr, sizeof(addr)), pj_ntohs(rem_addr.sin_port)); /* Load test */ if (transport_load_test(url) != 0) return -60; /* Basic transport's send/receive loopback test. */ for (i=0; iref_cnt) != 1) return -80; } for (i = 0; i < num_tp; ++i) { /* Destroy this transport. */ pjsip_transport_dec_ref(tcp[i]); /* Force destroy this transport. */ status = pjsip_transport_destroy(tcp[i]); if (status != PJ_SUCCESS) return -90; } for (i = 0; i < num_listener; ++i) { /* Unregister factory */ status = pjsip_tpmgr_unregister_tpfactory(pjsip_endpt_get_tpmgr(endpt), tpfactory[i]); if (status != PJ_SUCCESS) return -95; } /* Flush events. */ PJ_LOG(3,(THIS_FILE, " Flushing events, 1 second...")); flush_events(1000); /* Done */ return 0; } #else /* PJ_HAS_TCP */ int transport_tcp_test(void) { return 0; } #endif /* PJ_HAS_TCP */