You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
kamailio/test/udp_test_proxy.c

256 lines
5.9 KiB

/*
* $Id$
*
*
* Copyright (C) 2001-2003 FhG Fokus
*
* This file is part of ser, a free SIP server.
*
* ser 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
*
* For a license to use the ser software under conditions
* other than those described here, or to purchase support for this
* software, please contact iptel.org by e-mail at the following addresses:
* info@iptel.org
*
* ser 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 <stdlib.h>
#include <stdio.h>
#include <errno.h>
#include <string.h>
#include <ctype.h>
#include <netdb.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
static char *id="$Id$";
static char *version="udp_test_proxy 0.1";
static char* help_msg="\
Usage: udp_test_proxy -l address -s port -d address -p port [-n no] [-v]\n\
Options:\n\
-l address listen address\n\
-s port listen(source) port\n\
-d address destination address\n\
-p port destination port\n\
-n no number of processes\n\
-2 use a different socket for sending\n\
-v increase verbosity level\n\
-V version number\n\
-h this help message\n\
";
#define BUF_SIZE 65535
static char buf[BUF_SIZE];
int main(int argc, char** argv)
{
int sock;
int s_sock;
pid_t pid;
struct sockaddr_in addr;
struct sockaddr_in to;
int r, n, len;
char c;
struct hostent* he;
int verbose;
int sport, dport;
char *dst;
char *src;
char* tmp;
int use_diff_ssock;
/* init */
verbose=0;
dst=0;
sport=dport=0;
src=dst=0;
n=0;
use_diff_ssock=0;
opterr=0;
while ((c=getopt(argc,argv, "l:p:d:s:n:2vhV"))!=-1){
switch(c){
case '2':
use_diff_ssock=1;
break;
case 'v':
verbose++;
break;
case 'd':
dst=optarg;
break;
case 'l':
src=optarg;
break;
case 'p':
dport=strtol(optarg, &tmp, 10);
if ((tmp==0)||(*tmp)){
fprintf(stderr, "bad port number: -p %s\n", optarg);
goto error;
}
break;
case 's':
sport=strtol(optarg, &tmp, 10);
if ((tmp==0)||(*tmp)){
fprintf(stderr, "bad port number: -s %s\n", optarg);
goto error;
}
break;
case 'n':
n=strtol(optarg, &tmp, 10);
if ((tmp==0)||(*tmp)){
fprintf(stderr, "bad process number: -n %s\n", optarg);
goto error;
}
break;
case 'V':
printf("version: %s\n", version);
printf("%s\n",id);
exit(0);
break;
case 'h':
printf("version: %s\n", version);
printf("%s", help_msg);
exit(0);
break;
case '?':
if (isprint(optopt))
fprintf(stderr, "Unknown option `-%c<>\n", optopt);
else
fprintf(stderr, "Unknown character `\\x%x<>\n", optopt);
goto error;
case ':':
fprintf(stderr, "Option `-%c<> requires an argument.\n",
optopt);
goto error;
break;
default:
abort();
}
}
/* check if all the required params are present */
if (dst==0){
fprintf(stderr, "Missing destination (-d ...)\n");
exit(-1);
}
if (src==0){
fprintf(stderr, "Missing listen address (-l ...)\n");
exit(-1);
}
if(sport==0){
fprintf(stderr, "Missing source port number (-s port)\n");
exit(-1);
}else if(sport<0){
fprintf(stderr, "Invalid source port number (-s %d)\n", sport);
exit(-1);
}
if(dport==0){
fprintf(stderr, "Missing destination port number (-p port)\n");
exit(-1);
}else if(dport<0){
fprintf(stderr, "Invalid destination port number (-p %d)\n", dport);
exit(-1);
}
if(n<0){
fprintf(stderr, "Invalid process no (-n %d)\n", n);
exit(-1);
}
/* resolve destination */
he=gethostbyname(dst);
if (he==0){
fprintf(stderr, "ERROR: could not resolve %s\n", dst);
goto error;
}
/* set to*/
to.sin_family=he->h_addrtype;
to.sin_port=htons(dport);
memcpy(&to.sin_addr.s_addr, he->h_addr_list[0], he->h_length);
/* resolve source/listen */
he=gethostbyname(src);
if (he==0){
fprintf(stderr, "ERROR: could not resolve %s\n", dst);
goto error;
}
/* open socket*/
addr.sin_family=he->h_addrtype;
addr.sin_port=htons(sport);
memcpy(&addr.sin_addr.s_addr, he->h_addr_list[0], he->h_length);
s_sock=-1;
sock = socket(he->h_addrtype, SOCK_DGRAM, 0);
if (use_diff_ssock && (sock!=-1))
s_sock = socket(he->h_addrtype, SOCK_DGRAM, 0);
else
s_sock=sock;
if ((sock==-1)||(s_sock==-1)){
fprintf(stderr, "ERROR: socket: %s\n", strerror(errno));
goto error;
}
if (bind(sock, (struct sockaddr*) &addr, sizeof(struct sockaddr_in))==-1){
fprintf(stderr, "ERROR: bind: %s\n", strerror(errno));
goto error;
}
if (use_diff_ssock){
if (connect(s_sock, (struct sockaddr*) &to,
sizeof(struct sockaddr_in))==-1){
fprintf(stderr, "ERROR: connect: %s\n", strerror(errno));
goto error;
}
}
for(r=1; r<n; r++){
if ((pid=fork())==-1){
fprintf(stderr, "ERROR: fork: %s\n", strerror(errno));
goto error;
}
if (pid==0) break; /* child, skip */
}
if (verbose>3) printf("process starting\n");
for(;;){
len=read(sock, buf, BUF_SIZE);
if (len==-1){
fprintf(stderr, "ERROR: read: %s\n", strerror(errno));
continue;
}
if (verbose>2) putchar('r');
/* send it back*/
if (use_diff_ssock)
len=send(s_sock, buf, len, 0);
else
len=sendto(s_sock, buf, len, 0,
(struct sockaddr*) &to, sizeof(struct sockaddr_in));
if (len==-1){
fprintf(stderr, "ERROR: send: %s\n", strerror(errno));
continue;
}
if (verbose>1) putchar('.');
}
error:
exit(-1);
}