1f503b4c4SSepherosa Ziehau #include <sys/types.h>
2f503b4c4SSepherosa Ziehau #include <sys/socket.h>
3f503b4c4SSepherosa Ziehau
4f503b4c4SSepherosa Ziehau #include <arpa/inet.h>
5*1926f587SSepherosa Ziehau #include <net/if.h>
6f503b4c4SSepherosa Ziehau #include <netinet/in.h>
7f503b4c4SSepherosa Ziehau
8f503b4c4SSepherosa Ziehau #include <err.h>
9f503b4c4SSepherosa Ziehau #include <stdint.h>
10f503b4c4SSepherosa Ziehau #include <stdio.h>
11f503b4c4SSepherosa Ziehau #include <stdlib.h>
12f503b4c4SSepherosa Ziehau #include <string.h>
13f503b4c4SSepherosa Ziehau #include <unistd.h>
14f503b4c4SSepherosa Ziehau
15f503b4c4SSepherosa Ziehau static void
usage(const char * cmd)16f503b4c4SSepherosa Ziehau usage(const char *cmd)
17f503b4c4SSepherosa Ziehau {
18*1926f587SSepherosa Ziehau fprintf(stderr, "%s -p port -d dst (-i addr [-m] | -I iface) "
19*1926f587SSepherosa Ziehau "[-P bind_port]\n", cmd);
20f503b4c4SSepherosa Ziehau exit(1);
21f503b4c4SSepherosa Ziehau }
22f503b4c4SSepherosa Ziehau
23f503b4c4SSepherosa Ziehau int
main(int argc,char * argv[])24f503b4c4SSepherosa Ziehau main(int argc, char *argv[])
25f503b4c4SSepherosa Ziehau {
26f503b4c4SSepherosa Ziehau struct sockaddr_in dst, local_in;
27*1926f587SSepherosa Ziehau struct in_addr iface, mcast_if;
28*1926f587SSepherosa Ziehau int s, opt, n, loop = 0, use_mreq = 0, iface_idx;
29f503b4c4SSepherosa Ziehau uint8_t buf[18];
30*1926f587SSepherosa Ziehau socklen_t mcast_if_len;
31*1926f587SSepherosa Ziehau char mcast_if_str[INET_ADDRSTRLEN];
32f503b4c4SSepherosa Ziehau
33f503b4c4SSepherosa Ziehau memset(&dst, 0, sizeof(dst));
34f503b4c4SSepherosa Ziehau dst.sin_family = AF_INET;
35f503b4c4SSepherosa Ziehau
36f503b4c4SSepherosa Ziehau memset(&local_in, 0, sizeof(local_in));
37f503b4c4SSepherosa Ziehau local_in.sin_family = AF_INET;
38f503b4c4SSepherosa Ziehau
39f503b4c4SSepherosa Ziehau memset(&iface, 0, sizeof(iface));
40*1926f587SSepherosa Ziehau iface_idx = -1;
41f503b4c4SSepherosa Ziehau
42*1926f587SSepherosa Ziehau while ((opt = getopt(argc, argv, "I:P:d:i:mp:")) != -1) {
43f503b4c4SSepherosa Ziehau switch (opt) {
44*1926f587SSepherosa Ziehau case 'I':
45*1926f587SSepherosa Ziehau iface_idx = if_nametoindex(optarg);
46*1926f587SSepherosa Ziehau break;
47*1926f587SSepherosa Ziehau
48f503b4c4SSepherosa Ziehau case 'P':
49f503b4c4SSepherosa Ziehau local_in.sin_port = strtol(optarg, NULL, 10);
50f503b4c4SSepherosa Ziehau local_in.sin_port = htons(local_in.sin_port);
51f503b4c4SSepherosa Ziehau break;
52f503b4c4SSepherosa Ziehau
53f503b4c4SSepherosa Ziehau case 'd':
54f503b4c4SSepherosa Ziehau if (inet_pton(AF_INET, optarg, &dst.sin_addr) <= 0)
55f503b4c4SSepherosa Ziehau usage(argv[0]);
56f503b4c4SSepherosa Ziehau break;
57f503b4c4SSepherosa Ziehau
58f503b4c4SSepherosa Ziehau case 'i':
59f503b4c4SSepherosa Ziehau if (inet_pton(AF_INET, optarg, &iface) <= 0)
60f503b4c4SSepherosa Ziehau usage(argv[0]);
61f503b4c4SSepherosa Ziehau break;
62f503b4c4SSepherosa Ziehau
63*1926f587SSepherosa Ziehau case 'm':
64*1926f587SSepherosa Ziehau use_mreq = 1;
65*1926f587SSepherosa Ziehau break;
66*1926f587SSepherosa Ziehau
67f503b4c4SSepherosa Ziehau case 'p':
68f503b4c4SSepherosa Ziehau dst.sin_port = strtol(optarg, NULL, 10);
69f503b4c4SSepherosa Ziehau dst.sin_port = htons(dst.sin_port);
70f503b4c4SSepherosa Ziehau break;
71f503b4c4SSepherosa Ziehau
72f503b4c4SSepherosa Ziehau default:
73f503b4c4SSepherosa Ziehau usage(argv[0]);
74f503b4c4SSepherosa Ziehau }
75f503b4c4SSepherosa Ziehau }
76f503b4c4SSepherosa Ziehau
77*1926f587SSepherosa Ziehau if ((iface.s_addr == INADDR_ANY && iface_idx < 0) ||
78f503b4c4SSepherosa Ziehau dst.sin_addr.s_addr == INADDR_ANY || dst.sin_port == 0)
79f503b4c4SSepherosa Ziehau usage(argv[0]);
80f503b4c4SSepherosa Ziehau
81f503b4c4SSepherosa Ziehau s = socket(AF_INET, SOCK_DGRAM, 0);
82f503b4c4SSepherosa Ziehau if (s < 0)
83f503b4c4SSepherosa Ziehau err(2, "socket failed");
84f503b4c4SSepherosa Ziehau
85*1926f587SSepherosa Ziehau if (iface_idx >= 0) {
86*1926f587SSepherosa Ziehau struct ip_mreqn mreqn;
87*1926f587SSepherosa Ziehau
88*1926f587SSepherosa Ziehau fprintf(stderr, "ip_mreqn mcast_if ifindex %d\n", iface_idx);
89*1926f587SSepherosa Ziehau memset(&mreqn, 0, sizeof(mreqn));
90*1926f587SSepherosa Ziehau mreqn.imr_address = iface;
91*1926f587SSepherosa Ziehau mreqn.imr_ifindex = iface_idx;
92*1926f587SSepherosa Ziehau if (setsockopt(s, IPPROTO_IP, IP_MULTICAST_IF,
93*1926f587SSepherosa Ziehau &mreqn, sizeof(mreqn)) < 0)
94*1926f587SSepherosa Ziehau err(2, "setsockopt IP_MULTICAST_IF ip_mreqn failed");
95*1926f587SSepherosa Ziehau } else if (use_mreq) {
96*1926f587SSepherosa Ziehau struct ip_mreq mreq;
97*1926f587SSepherosa Ziehau
98*1926f587SSepherosa Ziehau fprintf(stderr, "ip_mreq mcast_if\n");
99*1926f587SSepherosa Ziehau memset(&mreq, 0, sizeof(mreq));
100*1926f587SSepherosa Ziehau mreq.imr_interface = iface;
101*1926f587SSepherosa Ziehau if (setsockopt(s, IPPROTO_IP, IP_MULTICAST_IF,
102*1926f587SSepherosa Ziehau &mreq, sizeof(mreq)) < 0)
103*1926f587SSepherosa Ziehau err(2, "setsockopt IP_MULTICAST_IF ip_mreq failed");
104*1926f587SSepherosa Ziehau } else {
105f503b4c4SSepherosa Ziehau if (setsockopt(s, IPPROTO_IP, IP_MULTICAST_IF,
106f503b4c4SSepherosa Ziehau &iface, sizeof(iface)) < 0)
107*1926f587SSepherosa Ziehau err(2, "setsockopt IP_MULTICAST_IF inaddr failed");
108*1926f587SSepherosa Ziehau }
109*1926f587SSepherosa Ziehau
110*1926f587SSepherosa Ziehau mcast_if_len = sizeof(mcast_if);
111*1926f587SSepherosa Ziehau if (getsockopt(s, IPPROTO_IP, IP_MULTICAST_IF,
112*1926f587SSepherosa Ziehau &mcast_if, &mcast_if_len) < 0)
113*1926f587SSepherosa Ziehau err(2, "getsockopt IP_MULTICAST_IF failed");
114*1926f587SSepherosa Ziehau fprintf(stderr, "ifindex %s\n", inet_ntop(AF_INET, &mcast_if,
115*1926f587SSepherosa Ziehau mcast_if_str, sizeof(mcast_if_str)));
116f503b4c4SSepherosa Ziehau
117f503b4c4SSepherosa Ziehau if (setsockopt(s, IPPROTO_IP, IP_MULTICAST_LOOP,
118f503b4c4SSepherosa Ziehau &loop, sizeof(loop)) < 0)
119f503b4c4SSepherosa Ziehau err(2, "setsockopt IP_MULTICAST_LOOP failed");
120f503b4c4SSepherosa Ziehau
121f503b4c4SSepherosa Ziehau if (local_in.sin_port != 0) {
122f503b4c4SSepherosa Ziehau local_in.sin_addr = iface;
123f503b4c4SSepherosa Ziehau if (bind(s, (const struct sockaddr *)&local_in,
124f503b4c4SSepherosa Ziehau sizeof(local_in)) < 0)
125f503b4c4SSepherosa Ziehau err(2, "bind failed");
126f503b4c4SSepherosa Ziehau }
127f503b4c4SSepherosa Ziehau
128f503b4c4SSepherosa Ziehau n = sendto(s, buf, sizeof(buf), 0,
129f503b4c4SSepherosa Ziehau (const struct sockaddr *)&dst, sizeof(dst));
130f503b4c4SSepherosa Ziehau if (n < 0)
131f503b4c4SSepherosa Ziehau err(2, "sendto failed");
132f503b4c4SSepherosa Ziehau else if (n < (int)sizeof(buf))
133f503b4c4SSepherosa Ziehau errx(2, "sent truncated data %d", n);
134f503b4c4SSepherosa Ziehau
135f503b4c4SSepherosa Ziehau exit(0);
136f503b4c4SSepherosa Ziehau }
137