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