mcast_leave.c 2.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110
  1. #include "unp.h"
  2. int
  3. mcast_leave(int sockfd, const SA *grp, socklen_t grplen)
  4. {
  5. #ifdef MCAST_JOIN_GROUP
  6. struct group_req req;
  7. req.gr_interface = 0;
  8. if (grplen > sizeof(req.gr_group)) {
  9. errno = EINVAL;
  10. return -1;
  11. }
  12. memcpy(&req.gr_group, grp, grplen);
  13. return (setsockopt(sockfd, family_to_level(grp->sa_family),
  14. MCAST_LEAVE_GROUP, &req, sizeof(req)));
  15. #else
  16. switch (grp->sa_family) {
  17. case AF_INET: {
  18. struct ip_mreq mreq;
  19. memcpy(&mreq.imr_multiaddr,
  20. &((const struct sockaddr_in *) grp)->sin_addr,
  21. sizeof(struct in_addr));
  22. mreq.imr_interface.s_addr = htonl(INADDR_ANY);
  23. return(setsockopt(sockfd, IPPROTO_IP, IP_DROP_MEMBERSHIP,
  24. &mreq, sizeof(mreq)));
  25. }
  26. #ifdef IPV6
  27. #ifndef IPV6_LEAVE_GROUP /* APIv0 compatibility */
  28. #define IPV6_LEAVE_GROUP IPV6_DROP_MEMBERSHIP
  29. #endif
  30. case AF_INET6: {
  31. struct ipv6_mreq mreq6;
  32. memcpy(&mreq6.ipv6mr_multiaddr,
  33. &((const struct sockaddr_in6 *) grp)->sin6_addr,
  34. sizeof(struct in6_addr));
  35. mreq6.ipv6mr_interface = 0;
  36. return(setsockopt(sockfd, IPPROTO_IPV6, IPV6_LEAVE_GROUP,
  37. &mreq6, sizeof(mreq6)));
  38. }
  39. #endif
  40. default:
  41. errno = EAFNOSUPPORT;
  42. return(-1);
  43. }
  44. #endif
  45. }
  46. void
  47. Mcast_leave(int sockfd, const SA *grp, socklen_t grplen)
  48. {
  49. if (mcast_leave(sockfd, grp, grplen) < 0)
  50. err_sys("mcast_leave error");
  51. }
  52. int
  53. mcast_leave_source_group(int sockfd, const SA *src, socklen_t srclen,
  54. const SA *grp, socklen_t grplen)
  55. {
  56. #ifdef MCAST_LEAVE_SOURCE_GROUP
  57. struct group_source_req req;
  58. req.gsr_interface = 0;
  59. if (grplen > sizeof(req.gsr_group) || srclen > sizeof(req.gsr_source)) {
  60. errno = EINVAL;
  61. return -1;
  62. }
  63. memcpy(&req.gsr_group, grp, grplen);
  64. memcpy(&req.gsr_source, src, srclen);
  65. return (setsockopt(sockfd, family_to_level(grp->sa_family),
  66. MCAST_LEAVE_SOURCE_GROUP, &req, sizeof(req)));
  67. #else
  68. switch (grp->sa_family) {
  69. #ifdef IP_DROP_SOURCE_MEMBERSHIP
  70. case AF_INET: {
  71. struct ip_mreq_source mreq;
  72. memcpy(&mreq.imr_multiaddr,
  73. &((struct sockaddr_in *) grp)->sin_addr,
  74. sizeof(struct in_addr));
  75. memcpy(&mreq.imr_sourceaddr,
  76. &((struct sockaddr_in *) src)->sin_addr,
  77. sizeof(struct in_addr));
  78. mreq.imr_interface.s_addr = htonl(INADDR_ANY);
  79. return(setsockopt(sockfd, IPPROTO_IP, IP_DROP_SOURCE_MEMBERSHIP,
  80. &mreq, sizeof(mreq)));
  81. }
  82. #endif
  83. #ifdef IPV6
  84. case AF_INET6: /* IPv6 source-specific API is MCAST_LEAVE_SOURCE_GROUP */
  85. #endif
  86. default:
  87. errno = EAFNOSUPPORT;
  88. return(-1);
  89. }
  90. #endif
  91. }
  92. void
  93. Mcast_leave_source_group(int sockfd, const SA *src, socklen_t srclen,
  94. const SA *grp, socklen_t grplen)
  95. {
  96. if (mcast_leave_source_group(sockfd, src, srclen, grp, grplen) < 0)
  97. err_sys("mcast_leave_source_group error");
  98. }