| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263 |
- #include "gai_hdr.h"## 1 ##src/libgai/ga_port.c##
- /*## 2 ##src/libgai/ga_port.c##
- * Go through all the addrinfo structures, checking for a match of the## 3 ##src/libgai/ga_port.c##
- * socket type and filling in the socket type, and then the port number## 4 ##src/libgai/ga_port.c##
- * in the corresponding socket address structures.## 5 ##src/libgai/ga_port.c##
- *## 6 ##src/libgai/ga_port.c##
- * The AI_CLONE flag works as follows. Consider a multihomed host with## 7 ##src/libgai/ga_port.c##
- * two IP addresses and no socket type specified by the caller. After## 8 ##src/libgai/ga_port.c##
- * the "host" search there are two addrinfo structures, one per IP address.## 9 ##src/libgai/ga_port.c##
- * Assuming a service supported by both TCP and UDP (say the daytime## 10 ##src/libgai/ga_port.c##
- * service) we need to return *four* addrinfo structures:## 11 ##src/libgai/ga_port.c##
- * IP#1, SOCK_STREAM, TCP port,## 12 ##src/libgai/ga_port.c##
- * IP#1, SOCK_DGRAM, UDP port,## 13 ##src/libgai/ga_port.c##
- * IP#2, SOCK_STREAM, TCP port,## 14 ##src/libgai/ga_port.c##
- * IP#2, SOCK_DGRAM, UDP port.## 15 ##src/libgai/ga_port.c##
- * To do this, when the "host" loop creates an addrinfo structure, if the## 16 ##src/libgai/ga_port.c##
- * caller has not specified a socket type (hintsp->ai_socktype == 0), the## 17 ##src/libgai/ga_port.c##
- * AI_CLONE flag is set. When the following function finds an entry like## 18 ##src/libgai/ga_port.c##
- * this it is handled as follows: If the entry's ai_socktype is still 0,## 19 ##src/libgai/ga_port.c##
- * this is the first use of the structure, and the ai_socktype field is set.## 20 ##src/libgai/ga_port.c##
- * But, if the entry's ai_socktype is nonzero, then we clone a new addrinfo## 21 ##src/libgai/ga_port.c##
- * structure and set it's ai_socktype to the new value. Although we only## 22 ##src/libgai/ga_port.c##
- * need two socket types today (SOCK_STREAM and SOCK_DGRAM) this algorithm## 23 ##src/libgai/ga_port.c##
- * will handle any number. Also notice that Posix.1g requires all socket## 24 ##src/libgai/ga_port.c##
- * types to be nonzero.## 25 ##src/libgai/ga_port.c##
- */## 26 ##src/libgai/ga_port.c##
- /* include ga_port */
- int## 27 ##src/libgai/ga_port.c##
- ga_port(struct addrinfo *aihead, int port, int socktype)## 28 ##src/libgai/ga_port.c##
- /* port must be in network byte order */## 29 ##src/libgai/ga_port.c##
- {## 30 ##src/libgai/ga_port.c##
- int nfound = 0;## 31 ##src/libgai/ga_port.c##
- struct addrinfo *ai;## 32 ##src/libgai/ga_port.c##
- for (ai = aihead; ai != NULL; ai = ai->ai_next) {## 33 ##src/libgai/ga_port.c##
- if (ai->ai_flags & AI_CLONE) {## 34 ##src/libgai/ga_port.c##
- if (ai->ai_socktype != 0) {## 35 ##src/libgai/ga_port.c##
- if ((ai = ga_clone(ai)) == NULL)## 36 ##src/libgai/ga_port.c##
- return (-1); /* memory allocation error */## 37 ##src/libgai/ga_port.c##
- /* ai points to newly cloned entry, which is what we want */## 38 ##src/libgai/ga_port.c##
- }## 39 ##src/libgai/ga_port.c##
- } else if (ai->ai_socktype != socktype)## 40 ##src/libgai/ga_port.c##
- continue; /* ignore if mismatch on socket type */## 41 ##src/libgai/ga_port.c##
- ai->ai_socktype = socktype;## 42 ##src/libgai/ga_port.c##
- switch (ai->ai_family) {## 43 ##src/libgai/ga_port.c##
- case AF_INET:## 44 ##src/libgai/ga_port.c##
- ((struct sockaddr_in *) ai->ai_addr)->sin_port = port;## 45 ##src/libgai/ga_port.c##
- nfound++;## 46 ##src/libgai/ga_port.c##
- break;## 47 ##src/libgai/ga_port.c##
- case AF_INET6:## 48 ##src/libgai/ga_port.c##
- ((struct sockaddr_in6 *) ai->ai_addr)->sin6_port = port;## 49 ##src/libgai/ga_port.c##
- nfound++;## 50 ##src/libgai/ga_port.c##
- break;## 51 ##src/libgai/ga_port.c##
- }## 52 ##src/libgai/ga_port.c##
- }## 53 ##src/libgai/ga_port.c##
- return (nfound);## 54 ##src/libgai/ga_port.c##
- }## 55 ##src/libgai/ga_port.c##
- /* end ga_port */
|