carp: add carp_group_demote_adj()
[dragonfly.git] / sbin / ifconfig / ifcarp.c
1 /*
2  * Copyright (c) 2002 Michael Shalayeff. All rights reserved.
3  * Copyright (c) 2003 Ryan McBride. All rights reserved.
4  *
5  * Redistribution and use in source and binary forms, with or without
6  * modification, are permitted provided that the following conditions
7  * are met:
8  * 1. Redistributions of source code must retain the above copyright
9  *    notice, this list of conditions and the following disclaimer.
10  * 2. Redistributions in binary form must reproduce the above copyright
11  *    notice, this list of conditions and the following disclaimer in the
12  *    documentation and/or other materials provided with the distribution.
13  *
14  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
15  * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
16  * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
17  * IN NO EVENT SHALL THE AUTHOR OR HIS RELATIVES BE LIABLE FOR ANY DIRECT,
18  * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
19  * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
20  * SERVICES; LOSS OF MIND, USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
21  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
22  * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING
23  * IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF
24  * THE POSSIBILITY OF SUCH DAMAGE.
25  */
26 /*
27  * $FreeBSD: src/sbin/ifconfig/ifcarp.c,v 1.2 2005/02/22 14:07:47 glebius Exp $
28  * $OpenBSD: ifconfig.c,v 1.82 2003/10/19 05:43:35 mcbride Exp $
29  * $DragonFly: src/sbin/ifconfig/ifcarp.c,v 1.1 2007/08/16 20:03:55 dillon Exp $
30  */
31
32 #include <sys/param.h>
33 #include <sys/ioctl.h>
34 #include <sys/socket.h>
35 #include <sys/sockio.h>
36
37 #include <stdlib.h>
38 #include <unistd.h>
39
40 #include <net/ethernet.h>
41 #include <net/if.h>
42 #include <netinet/in.h>
43 #include <netinet/ip_carp.h>
44 #include <net/route.h>
45
46 #include <arpa/inet.h>
47
48 #include <ctype.h>
49 #include <stdio.h>
50 #include <string.h>
51 #include <stdlib.h>
52 #include <unistd.h>
53 #include <err.h>
54 #include <errno.h>
55
56 #include "ifconfig.h"
57
58 static const char *carp_states[] = { CARP_STATES };
59
60 void carp_status(int s);
61 void setcarp_advbase(const char *,int, int, const struct afswtch *rafp);
62 void setcarp_advskew(const char *, int, int, const struct afswtch *rafp);
63 void setcarp_passwd(const char *, int, int, const struct afswtch *rafp);
64 void setcarp_vhid(const char *, int, int, const struct afswtch *rafp);
65
66 void
67 carp_status(int s)
68 {
69         const char *state;
70         struct carpreq carpr;
71         struct ifdrv ifd;
72         char devname[IFNAMSIZ];
73
74         memset((char *)&carpr, 0, sizeof(struct carpreq));
75         ifr.ifr_data = (caddr_t)&carpr;
76
77         if (ioctl(s, SIOCGVH, (caddr_t)&ifr) == -1)
78                 return;
79
80         if (carpr.carpr_vhid > 0) {
81                 if (carpr.carpr_state > CARP_MAXSTATE)
82                         state = "<UNKNOWN>";
83                 else
84                         state = carp_states[carpr.carpr_state];
85
86                 printf("\tcarp: %s vhid %d advbase %d advskew %d\n",
87                     state, carpr.carpr_vhid, carpr.carpr_advbase,
88                     carpr.carpr_advskew);
89         }
90
91         memset(&ifd, 0, sizeof(ifd));
92         strncpy(ifd.ifd_name, ifr.ifr_name, sizeof(ifd.ifd_name));
93         ifd.ifd_cmd = CARPGDEVNAME;
94         ifd.ifd_len = sizeof(devname);
95         ifd.ifd_data = devname;
96         if (ioctl(s, SIOCGDRVSPEC, &ifd) < 0)
97                 strlcpy(devname, "none", sizeof(devname));
98         if (devname[0] != '\0')
99                 printf("\tcarpdev: %s\n", devname);
100 }
101
102 void
103 setcarp_passwd(const char *val, int d, int s, const struct afswtch *afp)
104 {
105         struct carpreq carpr;
106
107         memset((char *)&carpr, 0, sizeof(struct carpreq));
108         ifr.ifr_data = (caddr_t)&carpr;
109
110         if (ioctl(s, SIOCGVH, (caddr_t)&ifr) == -1)
111                 err(1, "SIOCGVH");
112
113         /* XXX Should hash the password into the key here, perhaps? */
114         strlcpy(carpr.carpr_key, val, CARP_KEY_LEN);
115
116         if (ioctl(s, SIOCSVH, (caddr_t)&ifr) == -1)
117                 err(1, "SIOCSVH");
118 }
119
120 void
121 setcarp_vhid(const char *val, int d, int s, const struct afswtch *afp)
122 {
123         int vhid;
124         struct carpreq carpr;
125
126         vhid = atoi(val);
127
128         if (vhid <= 0)
129                 errx(1, "vhid must be greater than 0");
130
131         memset((char *)&carpr, 0, sizeof(struct carpreq));
132         ifr.ifr_data = (caddr_t)&carpr;
133
134         if (ioctl(s, SIOCGVH, (caddr_t)&ifr) == -1)
135                 err(1, "SIOCGVH");
136
137         carpr.carpr_vhid = vhid;
138
139         if (ioctl(s, SIOCSVH, (caddr_t)&ifr) == -1)
140                 err(1, "SIOCSVH");
141 }
142
143 void
144 setcarp_advskew(const char *val, int d, int s, const struct afswtch *afp)
145 {
146         int advskew;
147         struct carpreq carpr;
148
149         advskew = atoi(val);
150
151         memset((char *)&carpr, 0, sizeof(struct carpreq));
152         ifr.ifr_data = (caddr_t)&carpr;
153
154         if (ioctl(s, SIOCGVH, (caddr_t)&ifr) == -1)
155                 err(1, "SIOCGVH");
156
157         carpr.carpr_advskew = advskew;
158
159         if (ioctl(s, SIOCSVH, (caddr_t)&ifr) == -1)
160                 err(1, "SIOCSVH");
161 }
162
163 void
164 setcarp_advbase(const char *val, int d, int s, const struct afswtch *afp)
165 {
166         int advbase;
167         struct carpreq carpr;
168
169         advbase = atoi(val);
170
171         memset((char *)&carpr, 0, sizeof(struct carpreq));
172         ifr.ifr_data = (caddr_t)&carpr;
173
174         if (ioctl(s, SIOCGVH, (caddr_t)&ifr) == -1)
175                 err(1, "SIOCGVH");
176
177         carpr.carpr_advbase = advbase;
178
179         if (ioctl(s, SIOCSVH, (caddr_t)&ifr) == -1)
180                 err(1, "SIOCSVH");
181 }
182
183 static void
184 getcarp_vhaddr(const char *val, int d, int s, const struct afswtch *afp)
185 {
186 #define VHADDR_PFMT     "%-15s %-15s %s\n"
187
188         struct ifdrv ifd;
189         struct ifcarpvhaddr *carpa;
190         int count, i;
191
192         memset(&ifd, 0, sizeof(ifd));
193         strncpy(ifd.ifd_name, ifr.ifr_name, sizeof(ifd.ifd_name));
194         ifd.ifd_cmd = CARPGVHADDR;
195         if (ioctl(s, SIOCGDRVSPEC, &ifd) < 0)
196                 return;
197         if (ifd.ifd_len != 0) {
198                 carpa = malloc(ifd.ifd_len);
199                 if (carpa == NULL)
200                         return;
201
202                 ifd.ifd_cmd = CARPGVHADDR;
203                 ifd.ifd_data = carpa;
204                 if (ioctl(s, SIOCGDRVSPEC, &ifd) < 0) {
205                         free(carpa);
206                         return;
207                 }
208         } else {
209                 carpa = NULL;
210         }
211         count = ifd.ifd_len / sizeof(*carpa);
212         if (count != 0)
213                 printf(VHADDR_PFMT, "virtual addr", "backing addr", "flags");
214         for (i = 0; i < count; ++i) {
215                 char flags[16];
216                 char baddr[INET_ADDRSTRLEN];
217                 int a = 0;
218
219                 memset(flags, 0, sizeof(flags));
220                 flags[a] = '*';
221                 if (carpa[i].carpa_flags & CARP_VHAF_OWNER)
222                         flags[a++] = 'O';
223
224                 memset(baddr, 0, sizeof(baddr));
225                 baddr[0] = '*';
226                 if (carpa[i].carpa_baddr.sin_addr.s_addr != INADDR_ANY) {
227                         inet_ntop(AF_INET, &carpa[i].carpa_baddr.sin_addr,
228                                   baddr, sizeof(baddr));
229                 }
230
231                 printf(VHADDR_PFMT, inet_ntoa(carpa[i].carpa_addr.sin_addr),
232                        baddr, flags);
233         }
234         if (carpa != NULL)
235                 free(carpa);
236
237 #undef VHADDR_PFMT
238 }
239
240 static struct cmd carp_cmds[] = {
241         DEF_CMD_ARG("advbase",  setcarp_advbase),
242         DEF_CMD_ARG("advskew",  setcarp_advskew),
243         DEF_CMD_ARG("pass",     setcarp_passwd),
244         DEF_CMD_ARG("vhid",     setcarp_vhid),
245         DEF_CMD("vhaddr", 1,    getcarp_vhaddr)
246 };
247 static struct afswtch af_carp = {
248         .af_name        = "af_carp",
249         .af_af          = AF_UNSPEC,
250         .af_other_status = carp_status,
251 };
252
253 static __constructor void
254 carp_ctor(void)
255 {
256 #define N(a)    (sizeof(a) / sizeof(a[0]))
257         int i;
258
259         for (i = 0; i < N(carp_cmds);  i++)
260                 cmd_register(&carp_cmds[i]);
261         af_register(&af_carp);
262 #undef N
263 }