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

📄 ipe.c

📁 udp-cast!可以用multicasting的方式把檔案傳送到一個群組
💻 C
字号:
#include <stdio.h>#include <netinet/in.h>#include <stdlib.h>#include <pthread.h>#include <string.h>#include <net/if.h>#include <unistd.h>#include <arpa/inet.h>#include "../udpcast/rateGovernor.h"/** * Example of a rate governor. * Rate governors are dynamically loadable modules that allow to control the * speed at which udpcast transfers data. * * In order to do this, udpcast calls the governor's wait function before * transmitting a packet. The governor may then wait some time until it * returns, thusly controling the transmission speed. * * This example rate governor acts upon the traffic control packets that a * "Logic Innovations IP Encapsulator 3000" sends *//** * One route. We assume here that all fields are in network byte order */struct route {  in_addr_t ip_address;  in_addr_t ip_mask;  uint32_t buffer_length; /* how many bytes IPE buffer can hold */  uint32_t buffer_level;  /* current occupation level of IPE buffer */};/** * Entire traffic control packet. We assume here that all fields are in * network byte order */struct ipe_packet {  int total_routes;  int total_length;  struct route route[4096];};struct ipe {  struct sockaddr_in recv; /* address to which the traffic control packets			    * are sent */  char *interface; /* name of network interface where traffic control		      packets will be received */  int fd; /* file descriptor on which the traffic control packets are	     received */  int maxFillLevel; /* percent of maximum buffer fill level */  int havePacket; /* have we at least received one packet? */  pthread_t thread; /* thread receiving the multicasted announcements */  pthread_mutex_t mutex; /* mutex protecting the traffic control packets */  pthread_cond_t cond; /* this condition will be signal when the traffic			  control packet has been updated */  struct ipe_packet packet[2]; /* array of two traffic control packets. One				* is active, whereas the other is ready to				* receive new data. After having received and				* validated a new packet, the receiving thread				* will switch the roles of both slots */  int activePacket; /* which one of the 2 packets is currently active? */  struct route *lastMatchedRoute; /* pointer to last matched route				     (performance optimization) */};/** * answers whether the given route matches the IP */static int routeMatches(struct route *route, in_addr_t ip){  return (route->ip_address & route->ip_mask) == (ip & route->ip_mask);}/** * Find route to ip in packet. Returns NULL if none found */static struct route *findRouteInPacket(struct ipe_packet *packet, in_addr_t ip){  int i=0;  for(i=0; i< packet->total_routes; i++)    if(routeMatches(&packet->route[i], ip))      return &packet->route[i];  return NULL;}/** * Find route to given IP address in packet. First check whether cached route * matches, and if not, search through last received packet */static struct route *findRoute(struct ipe *ipe, in_addr_t ip){  if(ipe->lastMatchedRoute && routeMatches(ipe->lastMatchedRoute, ip))    return ipe->lastMatchedRoute;  ipe->lastMatchedRoute = findRouteInPacket(&ipe->packet[ipe->activePacket],ip);  return ipe->lastMatchedRoute;}/** * Thread listening for traffic control packets */static void *run(void *p){  struct ipe *me = (struct ipe *) p;  while(1) {    int n;    struct ipe_packet *packet = &me->packet[1-me->activePacket];    n = recv(me->fd, packet, sizeof(*packet), 0);    if(n < 0 ) {      perror("receive traffic control");      continue;    }    /* Quick sanity check of packet...*/    if(n < 8) {      fprintf(stderr, "Incomplete traffic control header\n");      continue;    }    if(n < ntohl(packet->total_routes) * 16) {      fprintf(stderr, "Incomplete traffic control data\n");      continue;    }    /* packet is ok, activate it */    pthread_mutex_lock(&me->mutex);    me->activePacket = 1 - me->activePacket;    me->havePacket = 1;    me->lastMatchedRoute = NULL;    pthread_cond_signal(&me->cond);    pthread_mutex_unlock(&me->mutex);   }}/** * Allocate and initialize new instance private data structure for this * rate governor */static void *ipe_initialize(void){  struct ipe *me;  me = calloc(sizeof(struct ipe),1);  me->maxFillLevel=80;    pthread_mutex_init(&me->mutex, NULL);  pthread_cond_init(&me->cond, NULL);  return me;}/** * Set property. All properties are passed as strings, and should be * converted to the appropriate type by this rate governor * * In this example, the following properties are supported *  - ip	the multicast ip address which receives the traffic control *		packets *  - port	the UDP port which receives the traffic control packets *  - if	the network interface (default eth0) which receives the *		traffic control packets *  - maxFillLevel stop transmitting packets if buffers on IPE are fuller than *		this level (expressed in percent of available buffer size). *		Default is 80 */static void ipe_setProp(void *p, const char *key, const char *value){  struct ipe *me = (struct ipe *) p;  if(!strcmp("ip", key)) {    inet_aton(value, &me->recv.sin_addr);  } else if(!strcmp("port", key)) {    char *eptr;    me->recv.sin_port = htons(strtoul(value,&eptr, 0));    if(*eptr)      fprintf(stderr, "Bad port %s\n", value);  } else if(!strcmp("if", key)) {    me->interface = strdup(value);  } else if(!strcmp("maxFillLevel", key)) {    char *eptr;    me->maxFillLevel = strtoul(value, &eptr, 0);    if(*eptr)      fprintf(stderr, "Bad port %s\n", value);  } else {    fprintf(stderr, "Unknown parameter %s=%s\n", key, value);  }}/** * This method will be called after all properties have been set. * In this method, the rate governor knows its configuration, and may start * with *  - opening sockets and other communication channels *  - communicate with remote servers *  - start background threads */static void ipe_endConfig(void *data){  struct ipe *me = (struct ipe *) data;  char *interface;  int sock;  struct ip_mreqn mreq; /* used for subscribing to multicast */  int r; /* generic return value */  sock = socket(PF_INET, SOCK_DGRAM, 0);  if(socket < 0) {    perror("socket");    return;  }  if(bind(sock, (struct sockaddr*) &me->recv, sizeof(struct sockaddr_in)) < 0) {    perror("bind");    return;  }  if(me->interface)    interface = me->interface;  else    interface = "eth0";  mreq.imr_ifindex = if_nametoindex(interface);  mreq.imr_address.s_addr = 0;  mreq.imr_multiaddr = me->recv.sin_addr;  r = setsockopt(sock, SOL_IP, IP_ADD_MEMBERSHIP, (char*)&mreq, sizeof(mreq));  if(r < 0) {    perror("add membership error");    /* consider this error non fatal, so that it still works with unicast     * addresses */  }  me->fd = sock;  pthread_create(&me->thread, NULL, run, me);}/** * This method is called by udpcast just before it transmits a packet. * The rate governor may uses this to wait until the ougoing channel is * ready to receive more data * Parameters: *  p	  the rate governor private data *  fd    file descriptor to which data is going to be sent *  ip    ip address to which data is going to be sent *  bytes bytes number of bytes which will be sent */static void ipe_wait(void *p, int fd, in_addr_t ip, long bytes){  struct ipe *me = (struct ipe *) p;  struct route *route;  int level=0;  int length=0;  pthread_mutex_lock(&me->mutex);  while(1) {    if(me->havePacket) {      route = findRoute(me, ip);      if(route == NULL)	/* no route found => no ratelimit... */	break;      level = ntohl(route->buffer_level);      length = ntohl(route->buffer_length);      if(bytes < length * me->maxFillLevel / 100  - level)	/* enough space left for new packet */	break;    }    pthread_cond_wait(&me->cond, &me->mutex);  }  if(route) {    /* Account packet that will be sent */    level += bytes;    route->buffer_level = htonl(level);  }  pthread_mutex_unlock(&me->mutex);}/** * Finalize is called just before program exits. This can be used to *  - signal remote servers that this sender is going away *  - close filedescriptors/communications channels *  - stop background threads *  - deallocate data structures */static void ipe_shutdown(void *p) {  struct ipe *me = (struct ipe *) p;  void *v;  pthread_cancel(me->thread);  pthread_join(me->thread, &v);  close(me->fd);  if(me->interface)    free(me->interface);}/** * Table of methods, to be picket up by udpcast */struct rateGovernor_t governor = {  ipe_initialize,  ipe_setProp,  ipe_endConfig,  ipe_wait,  ipe_shutdown,};

⌨️ 快捷键说明

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