udp.c
上传用户:hepax88
上传日期:2007-01-03
资源大小:1101k
文件大小:6k
源码类别:

TCP/IP协议栈

开发平台:

Visual C++

  1. /* Internet User Data Protocol (UDP)
  2.  * Copyright 1991 Phil Karn, KA9Q
  3.  */
  4. #include "global.h"
  5. #include "mbuf.h"
  6. #include "netuser.h"
  7. #include "iface.h"
  8. #include "udp.h"
  9. #include "ip.h"
  10. #include "internet.h"
  11. #include "icmp.h"
  12. static struct udp_cb *lookup_udp(struct socket *socket);
  13. struct mib_entry Udp_mib[] = {
  14. "", 0,
  15. "udpInDatagrams", 0,
  16. "udpNoPorts", 0,
  17. "udpInErrors", 0,
  18. "udpOutDatagrams", 0,
  19. };
  20. /* UDP control structures list */
  21. struct udp_cb *Udps;
  22. /* Create a UDP control block for lsocket, so that we can queue
  23.  * incoming datagrams.
  24.  */
  25. struct udp_cb *
  26. open_udp(lsocket,r_upcall)
  27. struct socket *lsocket;
  28. void (*r_upcall)();
  29. {
  30. register struct udp_cb *up;
  31. if((up = lookup_udp(lsocket)) != NULL){
  32. /* Already exists */
  33. Net_error = CON_EXISTS;
  34. return NULL;
  35. }
  36. up = (struct udp_cb *)callocw(1,sizeof (struct udp_cb));
  37. up->socket.address = lsocket->address;
  38. up->socket.port = lsocket->port;
  39. up->r_upcall = r_upcall;
  40. up->next = Udps;
  41. Udps = up;
  42. return up;
  43. }
  44. /* Send a UDP datagram */
  45. int
  46. send_udp(
  47. struct socket *lsocket, /* Source socket */
  48. struct socket *fsocket, /* Destination socket */
  49. char tos, /* Type-of-service for IP */
  50. char ttl, /* Time-to-live for IP */
  51. struct mbuf **bpp, /* Data field, if any */
  52. uint16 length, /* Length of data field */
  53. uint16 id, /* Optional ID field for IP */
  54. char df /* Don't Fragment flag for IP */
  55. ){
  56. struct pseudo_header ph;
  57. struct udp udp;
  58. int32 laddr;
  59. if(bpp == NULL)
  60. return -1;
  61. if(length != 0 && *bpp != NULL)
  62. trim_mbuf(bpp,length);
  63. else
  64. length = len_p(*bpp);
  65. length += UDPHDR;
  66. laddr = lsocket->address;
  67. if(laddr == INADDR_ANY)
  68. laddr = locaddr(fsocket->address);
  69. udp.source = lsocket->port;
  70. udp.dest = fsocket->port;
  71. udp.length = length;
  72. /* Create IP pseudo-header, compute checksum and send it */
  73. ph.length = length;
  74. ph.source = laddr;
  75. ph.dest = fsocket->address;
  76. ph.protocol = UDP_PTCL;
  77. htonudp(&udp,bpp,&ph);
  78. udpOutDatagrams++;
  79. ip_send(laddr,fsocket->address,UDP_PTCL,tos,ttl,bpp,length,id,df);
  80. return (int)length;
  81. }
  82. /* Accept a waiting datagram, if available. Returns length of datagram */
  83. int
  84. recv_udp(up,fsocket,bp)
  85. register struct udp_cb *up;
  86. struct socket *fsocket; /* Place to stash incoming socket */
  87. struct mbuf **bp; /* Place to stash data packet */
  88. {
  89. struct socket sp;
  90. struct mbuf *buf;
  91. uint16 length;
  92. if(up == NULL){
  93. Net_error = NO_CONN;
  94. return -1;
  95. }
  96. if(up->rcvcnt == 0){
  97. Net_error = WOULDBLK;
  98. return -1;
  99. }
  100. buf = dequeue(&up->rcvq);
  101. up->rcvcnt--;
  102. /* Strip socket header */
  103. pullup(&buf,&sp,sizeof(struct socket));
  104. /* Fill in the user's foreign socket structure, if given */
  105. if(fsocket != NULL){
  106. fsocket->address = sp.address;
  107. fsocket->port = sp.port;
  108. }
  109. /* Hand data to user */
  110. length = len_p(buf);
  111. if(bp != NULL)
  112. *bp = buf;
  113. else
  114. free_p(&buf);
  115. return (int)length;
  116. }
  117. /* Delete a UDP control block */
  118. int
  119. del_udp(conn)
  120. struct udp_cb *conn;
  121. {
  122. struct mbuf *bp;
  123. register struct udp_cb *up;
  124. struct udp_cb *udplast = NULL;
  125. for(up = Udps;up != NULL;udplast = up,up = up->next){
  126. if(up == conn)
  127. break;
  128. }
  129. if(up == NULL){
  130. /* Either conn was NULL or not found on list */
  131. Net_error = INVALID;
  132. return -1;
  133. }
  134. /* Get rid of any pending packets */
  135. while(up->rcvcnt != 0){
  136. bp = up->rcvq;
  137. up->rcvq = up->rcvq->anext;
  138. free_p(&bp);
  139. up->rcvcnt--;
  140. }
  141. /* Remove from list */
  142. if(udplast != NULL)
  143. udplast->next = up->next;
  144. else
  145. Udps = up->next; /* was first on list */
  146. free(up);
  147. return 0;
  148. }
  149. /* Process an incoming UDP datagram */
  150. void
  151. udp_input(
  152. struct iface *iface, /* Input interface */
  153. struct ip *ip, /* IP header */
  154. struct mbuf **bpp, /* UDP header and data */
  155. int rxbroadcast, /* The only protocol that accepts 'em */
  156. int32 said
  157. ){
  158. struct pseudo_header ph;
  159. struct udp udp;
  160. struct udp_cb *up;
  161. struct socket lsocket;
  162. struct socket fsocket;
  163. uint16 length;
  164. if(bpp == NULL || *bpp == NULL)
  165. return;
  166. /* Create pseudo-header and verify checksum */
  167. ph.source = ip->source;
  168. ph.dest = ip->dest;
  169. ph.protocol = ip->protocol;
  170. length = ip->length - IPLEN - ip->optlen;
  171. ph.length = length;
  172. /* Peek at header checksum before we extract the header. This
  173.  * allows us to bypass cksum() if the checksum field was not
  174.  * set by the sender.
  175.  */
  176. udp.checksum = udpcksum(*bpp);
  177. if(udp.checksum != 0 && cksum(&ph,*bpp,length) != 0){
  178. /* Checksum non-zero, and wrong */
  179. udpInErrors++;
  180. free_p(bpp);
  181. return;
  182. }
  183. /* Extract UDP header in host order */
  184. if(ntohudp(&udp,bpp) != 0){
  185. /* Truncated header */
  186. udpInErrors++;
  187. free_p(bpp);
  188. return;
  189. }
  190. /* If this was a broadcast packet, pretend it was sent to us */
  191. if(rxbroadcast){
  192. lsocket.address = iface->addr;
  193. } else
  194. lsocket.address = ip->dest;
  195. lsocket.port = udp.dest;
  196. /* See if there's somebody around to read it */
  197. if((up = lookup_udp(&lsocket)) == NULL){
  198. /* Nope, return an ICMP message */
  199. if(!rxbroadcast){
  200. htonudp(&udp,bpp,&ph);
  201. icmp_output(ip,*bpp,ICMP_DEST_UNREACH,ICMP_PORT_UNREACH,NULL);
  202. }
  203. udpNoPorts++;
  204. free_p(bpp);
  205. return;
  206. }
  207. /* Prepend the foreign socket info */
  208. fsocket.address = ip->source;
  209. fsocket.port = udp.source;
  210. pushdown(bpp,&fsocket,sizeof(fsocket));
  211. /* Queue it */
  212. enqueue(&up->rcvq,bpp);
  213. up->rcvcnt++;
  214. udpInDatagrams++;
  215. if(up->r_upcall)
  216. (*up->r_upcall)(iface,up,up->rcvcnt);
  217. }
  218. /* Look up UDP socket. 
  219.  * Return control block pointer or NULL if nonexistant
  220.  * As side effect, move control block to top of list to speed future
  221.  * searches.
  222.  */
  223. static struct udp_cb *
  224. lookup_udp(socket)
  225. struct socket *socket;
  226. {
  227. register struct udp_cb *up;
  228. struct udp_cb *uplast = NULL;
  229. for(up = Udps;up != NULL;uplast = up,up = up->next){
  230. if(socket->port == up->socket.port
  231.  && (socket->address == up->socket.address
  232.  || up->socket.address == INADDR_ANY)){
  233. if(uplast != NULL){
  234. /* Move to top of list */
  235. uplast->next = up->next;
  236. up->next = Udps;
  237. Udps = up;
  238. }
  239. return up;
  240. }
  241. }
  242. return NULL;
  243. }
  244. /* Attempt to reclaim unused space in UDP receive queues */
  245. void
  246. udp_garbage(red)
  247. int red;
  248. {
  249. register struct udp_cb *udp;
  250. for(udp = Udps;udp != NULL; udp = udp->next){
  251. mbuf_crunch(&udp->rcvq);
  252. }
  253. }