mcast_join.c 7.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288
  1. /* include mcast_join1 */
  2. #include "unp.h"
  3. #include <net/if.h>
  4. int
  5. mcast_join(int sockfd, const SA *grp, socklen_t grplen,
  6. const char *ifname, u_int ifindex)
  7. {
  8. #ifdef MCAST_JOIN_GROUP
  9. struct group_req req;
  10. if (ifindex > 0) {
  11. req.gr_interface = ifindex;
  12. } else if (ifname != NULL) {
  13. if ( (req.gr_interface = if_nametoindex(ifname)) == 0) {
  14. errno = ENXIO; /* i/f name not found */
  15. return(-1);
  16. }
  17. } else
  18. req.gr_interface = 0;
  19. if (grplen > sizeof(req.gr_group)) {
  20. errno = EINVAL;
  21. return -1;
  22. }
  23. memcpy(&req.gr_group, grp, grplen);
  24. return (setsockopt(sockfd, family_to_level(grp->sa_family),
  25. MCAST_JOIN_GROUP, &req, sizeof(req)));
  26. #else
  27. /* end mcast_join1 */
  28. /* include mcast_join2 */
  29. switch (grp->sa_family) {
  30. case AF_INET: {
  31. struct ip_mreq mreq;
  32. struct ifreq ifreq;
  33. memcpy(&mreq.imr_multiaddr,
  34. &((const struct sockaddr_in *) grp)->sin_addr,
  35. sizeof(struct in_addr));
  36. if (ifindex > 0) {
  37. if (if_indextoname(ifindex, ifreq.ifr_name) == NULL) {
  38. errno = ENXIO; /* i/f index not found */
  39. return(-1);
  40. }
  41. goto doioctl;
  42. } else if (ifname != NULL) {
  43. strncpy(ifreq.ifr_name, ifname, IFNAMSIZ);
  44. doioctl:
  45. if (ioctl(sockfd, SIOCGIFADDR, &ifreq) < 0)
  46. return(-1);
  47. memcpy(&mreq.imr_interface,
  48. &((struct sockaddr_in *) &ifreq.ifr_addr)->sin_addr,
  49. sizeof(struct in_addr));
  50. } else
  51. mreq.imr_interface.s_addr = htonl(INADDR_ANY);
  52. return(setsockopt(sockfd, IPPROTO_IP, IP_ADD_MEMBERSHIP,
  53. &mreq, sizeof(mreq)));
  54. }
  55. /* end mcast_join2 */
  56. /* include mcast_join3 */
  57. #ifdef IPV6
  58. #ifndef IPV6_JOIN_GROUP /* APIv0 compatibility */
  59. #define IPV6_JOIN_GROUP IPV6_ADD_MEMBERSHIP
  60. #endif
  61. case AF_INET6: {
  62. struct ipv6_mreq mreq6;
  63. memcpy(&mreq6.ipv6mr_multiaddr,
  64. &((const struct sockaddr_in6 *) grp)->sin6_addr,
  65. sizeof(struct in6_addr));
  66. if (ifindex > 0) {
  67. mreq6.ipv6mr_interface = ifindex;
  68. } else if (ifname != NULL) {
  69. if ( (mreq6.ipv6mr_interface = if_nametoindex(ifname)) == 0) {
  70. errno = ENXIO; /* i/f name not found */
  71. return(-1);
  72. }
  73. } else
  74. mreq6.ipv6mr_interface = 0;
  75. return(setsockopt(sockfd, IPPROTO_IPV6, IPV6_JOIN_GROUP,
  76. &mreq6, sizeof(mreq6)));
  77. }
  78. #endif
  79. default:
  80. errno = EAFNOSUPPORT;
  81. return(-1);
  82. }
  83. #endif
  84. }
  85. /* end mcast_join3 */
  86. void
  87. Mcast_join(int sockfd, const SA *grp, socklen_t grplen,
  88. const char *ifname, u_int ifindex)
  89. {
  90. if (mcast_join(sockfd, grp, grplen, ifname, ifindex) < 0)
  91. err_sys("mcast_join error");
  92. }
  93. int
  94. mcast_join_source_group(int sockfd, const SA *src, socklen_t srclen,
  95. const SA *grp, socklen_t grplen,
  96. const char *ifname, u_int ifindex)
  97. {
  98. #ifdef MCAST_JOIN_SOURCE_GROUP
  99. struct group_source_req req;
  100. if (ifindex > 0) {
  101. req.gsr_interface = ifindex;
  102. } else if (ifname != NULL) {
  103. if ( (req.gsr_interface = if_nametoindex(ifname)) == 0) {
  104. errno = ENXIO; /* i/f name not found */
  105. return(-1);
  106. }
  107. } else
  108. req.gsr_interface = 0;
  109. if (grplen > sizeof(req.gsr_group) || srclen > sizeof(req.gsr_source)) {
  110. errno = EINVAL;
  111. return -1;
  112. }
  113. memcpy(&req.gsr_group, grp, grplen);
  114. memcpy(&req.gsr_source, src, srclen);
  115. return (setsockopt(sockfd, family_to_level(grp->sa_family),
  116. MCAST_JOIN_SOURCE_GROUP, &req, sizeof(req)));
  117. #else
  118. switch (grp->sa_family) {
  119. #ifdef IP_ADD_SOURCE_MEMBERSHIP
  120. case AF_INET: {
  121. struct ip_mreq_source mreq;
  122. struct ifreq ifreq;
  123. memcpy(&mreq.imr_multiaddr,
  124. &((struct sockaddr_in *) grp)->sin_addr,
  125. sizeof(struct in_addr));
  126. memcpy(&mreq.imr_sourceaddr,
  127. &((struct sockaddr_in *) src)->sin_addr,
  128. sizeof(struct in_addr));
  129. if (ifindex > 0) {
  130. if (if_indextoname(ifindex, ifreq.ifr_name) == NULL) {
  131. errno = ENXIO; /* i/f index not found */
  132. return(-1);
  133. }
  134. goto doioctl;
  135. } else if (ifname != NULL) {
  136. strncpy(ifreq.ifr_name, ifname, IFNAMSIZ);
  137. doioctl:
  138. if (ioctl(sockfd, SIOCGIFADDR, &ifreq) < 0)
  139. return(-1);
  140. memcpy(&mreq.imr_interface,
  141. &((struct sockaddr_in *) &ifreq.ifr_addr)->sin_addr,
  142. sizeof(struct in_addr));
  143. } else
  144. mreq.imr_interface.s_addr = htonl(INADDR_ANY);
  145. return(setsockopt(sockfd, IPPROTO_IP, IP_ADD_SOURCE_MEMBERSHIP,
  146. &mreq, sizeof(mreq)));
  147. }
  148. #endif
  149. #ifdef IPV6
  150. case AF_INET6: /* IPv6 source-specific API is MCAST_JOIN_SOURCE_GROUP */
  151. #endif
  152. default:
  153. errno = EAFNOSUPPORT;
  154. return(-1);
  155. }
  156. #endif
  157. }
  158. void
  159. Mcast_join_source_group(int sockfd, const SA *src, socklen_t srclen,
  160. const SA *grp, socklen_t grplen,
  161. const char *ifname, u_int ifindex)
  162. {
  163. if (mcast_join_source_group(sockfd, src, srclen, grp, grplen,
  164. ifname, ifindex) < 0)
  165. err_sys("mcast_join_source_group error");
  166. }
  167. int
  168. mcast_block_source(int sockfd, const SA *src, socklen_t srclen,
  169. const SA *grp, socklen_t grplen)
  170. {
  171. #ifdef MCAST_BLOCK_SOURCE
  172. struct group_source_req req;
  173. req.gsr_interface = 0;
  174. if (grplen > sizeof(req.gsr_group) || srclen > sizeof(req.gsr_source)) {
  175. errno = EINVAL;
  176. return -1;
  177. }
  178. memcpy(&req.gsr_group, grp, grplen);
  179. memcpy(&req.gsr_source, src, srclen);
  180. return (setsockopt(sockfd, family_to_level(grp->sa_family),
  181. MCAST_BLOCK_SOURCE, &req, sizeof(req)));
  182. #else
  183. switch (grp->sa_family) {
  184. #ifdef IP_BLOCK_SOURCE
  185. case AF_INET: {
  186. struct ip_mreq_source mreq;
  187. memcpy(&mreq.imr_multiaddr,
  188. &((struct sockaddr_in *) grp)->sin_addr,
  189. sizeof(struct in_addr));
  190. memcpy(&mreq.imr_sourceaddr,
  191. &((struct sockaddr_in *) src)->sin_addr,
  192. sizeof(struct in_addr));
  193. mreq.imr_interface.s_addr = htonl(INADDR_ANY);
  194. return(setsockopt(sockfd, IPPROTO_IP, IP_BLOCK_SOURCE,
  195. &mreq, sizeof(mreq)));
  196. }
  197. #endif
  198. #ifdef IPV6
  199. case AF_INET6: /* IPv6 source-specific API is MCAST_BLOCK_SOURCE */
  200. #endif
  201. default:
  202. errno = EAFNOSUPPORT;
  203. return(-1);
  204. }
  205. #endif
  206. }
  207. void
  208. Mcast_block_source(int sockfd, const SA *src, socklen_t srclen,
  209. const SA *grp, socklen_t grplen)
  210. {
  211. if (mcast_block_source(sockfd, src, srclen, grp, grplen) < 0)
  212. err_sys("mcast_block_source error");
  213. }
  214. int
  215. mcast_unblock_source(int sockfd, const SA *src, socklen_t srclen,
  216. const SA *grp, socklen_t grplen)
  217. {
  218. #ifdef MCAST_UNBLOCK_SOURCE
  219. struct group_source_req req;
  220. req.gsr_interface = 0;
  221. if (grplen > sizeof(req.gsr_group) || srclen > sizeof(req.gsr_source)) {
  222. errno = EINVAL;
  223. return -1;
  224. }
  225. memcpy(&req.gsr_group, grp, grplen);
  226. memcpy(&req.gsr_source, src, srclen);
  227. return (setsockopt(sockfd, family_to_level(grp->sa_family),
  228. MCAST_UNBLOCK_SOURCE, &req, sizeof(req)));
  229. #else
  230. switch (grp->sa_family) {
  231. #ifdef IP_UNBLOCK_SOURCE
  232. case AF_INET: {
  233. struct ip_mreq_source mreq;
  234. memcpy(&mreq.imr_multiaddr,
  235. &((struct sockaddr_in *) grp)->sin_addr,
  236. sizeof(struct in_addr));
  237. memcpy(&mreq.imr_sourceaddr,
  238. &((struct sockaddr_in *) src)->sin_addr,
  239. sizeof(struct in_addr));
  240. mreq.imr_interface.s_addr = htonl(INADDR_ANY);
  241. return(setsockopt(sockfd, IPPROTO_IP, IP_UNBLOCK_SOURCE,
  242. &mreq, sizeof(mreq)));
  243. }
  244. #endif
  245. #ifdef IPV6
  246. case AF_INET6: /* IPv6 source-specific API is MCAST_UNBLOCK_SOURCE */
  247. #endif
  248. default:
  249. errno = EAFNOSUPPORT;
  250. return(-1);
  251. }
  252. #endif
  253. }
  254. void
  255. Mcast_unblock_source(int sockfd, const SA *src, socklen_t srclen,
  256. const SA *grp, socklen_t grplen)
  257. {
  258. if (mcast_unblock_source(sockfd, src, srclen, grp, grplen) < 0)
  259. err_sys("mcast_unblock_source error");
  260. }