⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 su_source_test.c

📁 this is simple sip stack.
💻 C
字号:
/* * This file is part of the Sofia-SIP package * * Copyright (C) 2005-2006 Nokia Corporation. * * Contact: Pekka Pessi <pekka.pessi@nokia.com> * * 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 * @CFILE su_source_test.c * * @brief Test program for glib and su root event loop integration. * * @author Pekka Pessi <Pekka.Pessi@nokia.com> *  * @date Created: Thu Mar 18 19:40:51 1999 pessi */#if HAVE_CONFIG_H#include "config.h"#endif#include <stdlib.h>#include <string.h>#include <stdio.h>#include <signal.h>#include <assert.h>struct pinger;#define SU_ROOT_MAGIC_T struct pinger#define SU_INTERNAL_P   su_root_t *#define SU_MSG_ARG_T    su_sockaddr_t#include "sofia-sip/su.h"#include "sofia-sip/su_wait.h"#include "sofia-sip/su_log.h"#include "sofia-sip/su_source.h"struct pinger {  enum { PINGER = 1, PONGER = 2 } const sort;  char const *  name;  unsigned      running : 1;  unsigned      : 0;  su_root_t    *root;  su_socket_t   s;  su_timer_t   *t;  int           id;  int           rindex;  su_time_t     when;  su_sockaddr_t addr;  double        rtt_total;  int           rtt_n;};short opt_family = AF_INET;short opt_verbatim = 0;short opt_singlethread = 0;static su_socket_t udpsocket(void) {  su_socket_t s;  su_sockaddr_t su = { 0 };  socklen_t sulen = sizeof(su);  char nbuf[64];  su.su_family = opt_family;  su_getlocalip(&su);  s = su_socket(su.su_family, SOCK_DGRAM, 0);  if (s == INVALID_SOCKET) {    su_perror("udpsocket: socket");    exit(1);  }  if (bind(s, &su.su_sa, su_sockaddr_size(&su)) == SOCKET_ERROR) {    su_perror("udpsocket: bind");    exit(1);  }  if (getsockname(s, &su.su_sa, &sulen) == SOCKET_ERROR) {    su_perror("udpsocket: getsockname");    exit(1);   }  if (opt_verbatim)    printf("udpsocket: using address [%s]:%u\n",	   inet_ntop(su.su_family, SU_ADDR(&su), nbuf, sizeof(nbuf)),	   ntohs(su.su_sin.sin_port));  return s;}static char *snow(su_time_t now){  static char buf[24];  su_time_print(buf, sizeof(buf), &now);  return buf;}void do_ping(struct pinger *p, su_timer_t *t, void *p0){  char buf[1024];  assert(p == su_root_magic(su_timer_root(t)));  assert(p->sort == PINGER);  p->when = su_now();  snprintf(buf, sizeof(buf), "Ping %d at %s", p->id++, snow(p->when));  if (sendto(p->s, buf, strlen(buf), 0, 	     &p->addr.su_sa, su_sockaddr_size(&p->addr)) == -1) {    su_perror("do_ping: send");  }  if (opt_verbatim) {    puts(buf);    fflush(stdout);  }}intdo_rtt(struct pinger *p, su_wait_t *w, void *p0){  su_sockaddr_t su;  struct sockaddr * const susa = &su.su_sa;  socklen_t susize[] = { sizeof(su)};  char buf[1024];  char nbuf[1024];  int n;  su_time_t now = su_now();  double rtt;  assert(p0 == p);  assert(p->sort == PINGER);  rtt = su_time_diff(now, p->when);  p->rtt_total += rtt, p->rtt_n++;  su_wait_events(w, p->s);  n = recvfrom(p->s, buf, sizeof(buf) - 1, 0, susa, susize);  if (n < 0) {	  su_perror("do_rtt: recvfrom");	  return 0;  }  buf[n] = 0;  if (opt_verbatim)    printf("do_rtt: %d bytes from [%s]:%u: \"%s\", rtt = %lg ms\n",	   n, inet_ntop(su.su_family, SU_ADDR(&su), nbuf, sizeof(nbuf)),	   ntohs(su.su_sin.sin_port), buf, rtt / 1000);  do_ping(p, p->t, NULL);  return 0;}voiddo_pong(struct pinger *p, su_timer_t *t, void *p0){  char buf[1024];  assert(p == su_root_magic(su_timer_root(t)));  assert(p->sort == PONGER);  p->id = 0;  snprintf(buf, sizeof(buf), "Pong at %s", snow(su_now()));  if (sendto(p->s, buf, strlen(buf), 0, 	     &p->addr.su_sa, su_sockaddr_size(&p->addr)) == -1) {    su_perror("do_pong: send");  }  if (opt_verbatim) {    puts(buf);    fflush(stdout);  }}intdo_recv(struct pinger *p, su_wait_t *w, void *p0){  su_sockaddr_t su;  socklen_t susize[] = { sizeof(su)};  char buf[1024];  char nbuf[1024];  int n;  su_time_t now = su_now();  assert(p0 == p);  assert(p->sort == PONGER);  su_wait_events(w, p->s);  n = recvfrom(p->s, buf, sizeof(buf) - 1, 0, &su.su_sa, susize);  if (n < 0) {	  su_perror("do_recv: recvfrom");	  return 0;  }  buf[n] = 0;  if (opt_verbatim)    printf("do_recv: %d bytes from [%s]:%u: \"%s\" at %s\n",	   n, inet_ntop(su.su_family, SU_ADDR(&su), nbuf, sizeof(nbuf)),	   ntohs(su.su_sin.sin_port), buf, snow(now));  fflush(stdout);#if 0  if (p->id)    puts("do_recv: already a pending reply");  if (su_timer_set(p->t, do_pong, p) < 0) {    fprintf(stderr, "do_recv: su_timer_set() error\n");    return 0;  }  p->id = 1;#else  do_pong(p, p->t, NULL);#endif  return 0;}voiddo_exit(struct pinger *x, su_timer_t *t, void *x0){  if (opt_verbatim)    printf("do_exit at %s\n", snow(su_now()));  su_root_break(su_timer_root(t));}intdo_init(su_root_t *root, struct pinger *p){  su_wait_t w;  su_socket_t s;  long interval;  su_timer_t *t;  su_wakeup_f f;  int index, index0;  switch (p->sort) {  case PINGER: f = do_rtt;  interval = 200; break;  case PONGER: f = do_recv; interval = 40;  break;  default:    return SU_FAILURE;  }  /* Create a sockets,  */  s = udpsocket();  if (su_wait_create(&w, s, SU_WAIT_IN) == SOCKET_ERROR)    su_perror("su_wait_create"), exit(1);  p->s = s;  p->t = t = su_timer_create(su_root_task(root), interval);  if (t == NULL) {    su_perror("su_timer_create");    return SU_FAILURE;  }  index0 = su_root_register(root, &w, f, p, 0);  if (index0 == SOCKET_ERROR) {    su_perror("su_root_register");    return SU_FAILURE;  }  index = su_root_register(root, &w, f, p, 0);  if (index == SOCKET_ERROR) {    su_perror("su_root_register");    return SU_FAILURE;  }  su_root_deregister(root, index0);  p->rindex = index;  return 0;}voiddo_destroy(su_root_t *root, struct pinger *p){  if (opt_verbatim)    printf("do_destroy %s at %s\n", p->name, snow(su_now()));  su_root_deregister(root, p->rindex);  su_timer_destroy(p->t), p->t = NULL;  p->running = 0;}voidstart_ping(struct pinger *p, su_msg_r msg, su_sockaddr_t *arg){  if (!p->running)    return;  if (opt_verbatim)    printf("start_ping: %s\n", p->name);  p->addr = *arg;  p->id = 1;  su_timer_set_at(p->t, do_ping, p, su_now());}voidstart_pong(struct pinger *p, su_msg_r msg, su_sockaddr_t *arg){  su_msg_r reply;  if (!p->running)    return;  if (opt_verbatim)    printf("start_pong: %s\n", p->name);  p->addr = *arg;  if (su_msg_reply(reply, msg, start_ping, sizeof(p->addr)) == 0) {    socklen_t sinsize[1] = { sizeof(p->addr) };    if (getsockname(p->s, (struct sockaddr*)su_msg_data(reply), sinsize)	== SOCKET_ERROR)      su_perror("start_pong: getsockname()"), exit(1);    su_msg_send(reply);  }  else {    fprintf(stderr, "su_msg_create failed!\n");  }}voidinit_ping(struct pinger *p, su_msg_r msg, su_sockaddr_t *arg){  su_msg_r reply;  if (opt_verbatim)    printf("init_ping: %s\n", p->name);  if (su_msg_reply(reply, msg, start_pong, sizeof(p->addr)) == 0) {    socklen_t sinsize[1] = { sizeof(p->addr) };    if (getsockname(p->s, (struct sockaddr*)su_msg_data(reply), sinsize)	== SOCKET_ERROR)      su_perror("start_pong: getsockname()"), exit(1);    su_msg_send(reply);  }  else {    fprintf(stderr, "su_msg_reply failed!\n");  }}staticRETSIGTYPE term(int n){  exit(1);}voidtime_test(void){  su_time_t now = su_now(), then = now;  su_duration_t t1, t2;  su_duration_t us;  for (us = 0; us < 1000000; us += 300) {    then.tv_sec = now.tv_sec;    if ((then.tv_usec = now.tv_usec + us) >= 1000000)      then.tv_usec -= 1000000, then.tv_sec++;    t1 = su_duration(now, then);    t2 = su_duration(then, now);    assert(t1 == -t2);  }  if (opt_verbatim)    printf("time_test: passed\n");}char const name[] = "su_test";voidusage(int exitcode){  fprintf(stderr, "usage: %s [-6vs] [pid]\n", name);  exit(exitcode);}/* * test su_wait functionality: * * Create a ponger, waking up do_recv() when data arrives,  *                  then scheduling do_pong() by timer * * Create a pinger, executed from timer, scheduling do_ping(), *                  waking up do_rtt() when data arrives * * Create a timer, executing do_exit() after 10 seconds */int main(int argc, char *argv[]){  su_root_t *root;  su_clone_r ping = SU_CLONE_R_INIT, pong = SU_CLONE_R_INIT;  su_msg_r start_msg = SU_MSG_R_INIT;  su_timer_t *t;  unsigned long sleeppid = 0;  struct pinger     pinger = { PINGER, "ping", 1 },     ponger = { PONGER, "pong", 1 };  char *argv0 = argv[0];  while (argv[1]) {    if (strcmp(argv[1], "-v") == 0) {      opt_verbatim = 1;      argv++;    }#if SU_HAVE_IN6    else if (strcmp(argv[1], "-6") == 0) {      opt_family = AF_INET6;      argv++;    }#endif    else if (strcmp(argv[1], "-s") == 0) {      opt_singlethread = 1;      argv++;    }    else if (strlen(argv[1]) == strspn(argv[1], "0123456789")) {      sleeppid = strtoul(argv[1], NULL, 10);      argv++;    }    else {      usage(1);    }  }  signal(SIGTERM, term);  su_init(); atexit(su_deinit);  time_test();  root = su_root_source_create(NULL);   if (!root) perror("su_root_glib_create"), exit(1);    if (!g_source_attach(su_root_gsource(root), NULL))     perror("g_source_attach"), exit(1);    su_root_threading(root, 0 && !opt_singlethread);  if (su_clone_start(root, ping, &pinger, do_init, do_destroy) != 0)    perror("su_clone_start"), exit(1);  if (su_clone_start(root, pong, &ponger, do_init, do_destroy) != 0)    perror("su_clone_start"), exit(1);   /* Test timer, exiting after 200 milliseconds */  t = su_timer_create(su_root_task(root), 200L);  if (t == NULL)    su_perror("su_timer_create"), exit(1);  su_timer_set(t, (su_timer_f)do_exit, NULL);  su_msg_create(start_msg, su_clone_task(ping), su_clone_task(pong), 		init_ping, 0);  su_msg_send(start_msg);  su_root_run(root);  su_clone_wait(root, ping);  su_clone_wait(root, pong);  su_timer_destroy(t);  if (pinger.rtt_n) {    printf("%s executed %u pings in %g, mean rtt=%g sec\n", name, 	   pinger.rtt_n, pinger.rtt_total, pinger.rtt_total / pinger.rtt_n);  }  su_root_destroy(root);  if (opt_verbatim)    printf("%s exiting\n", argv0); #ifndef HAVE_WIN32   if (sleeppid)     kill(sleeppid, SIGTERM);#endif  return 0;}

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -