Update to dhcpcd-9.4.1 with the following changes:
[dragonfly.git] / contrib / dhcpcd / src / if.h
1 /* SPDX-License-Identifier: BSD-2-Clause */
2 /*
3  * dhcpcd - DHCP client daemon
4  * Copyright (c) 2006-2021 Roy Marples <roy@marples.name>
5  * All rights reserved
6
7  * Redistribution and use in source and binary forms, with or without
8  * modification, are permitted provided that the following conditions
9  * are met:
10  * 1. Redistributions of source code must retain the above copyright
11  *    notice, this list of conditions and the following disclaimer.
12  * 2. Redistributions in binary form must reproduce the above copyright
13  *    notice, this list of conditions and the following disclaimer in the
14  *    documentation and/or other materials provided with the distribution.
15  *
16  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
17  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
18  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
19  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
20  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
21  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
22  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
23  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
24  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
25  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
26  * SUCH DAMAGE.
27  */
28
29 #ifndef INTERFACE_H
30 #define INTERFACE_H
31
32 #include <net/if.h>
33 #include <net/route.h>          /* for RTM_ADD et all */
34 #include <netinet/in.h>
35 #ifdef BSD
36 #include <netinet/in_var.h>     /* for IN_IFF_TENTATIVE et all */
37 #endif
38
39 #include <ifaddrs.h>
40
41 /* If the interface does not support carrier status (ie PPP),
42  * dhcpcd can poll it for the relevant flags periodically */
43 #define IF_POLL_UP      100     /* milliseconds */
44
45 /*
46  * Systems which handle 1 address per alias.
47  * Currenly this is just Solaris.
48  * While Linux can do aliased addresses, it is only useful for their
49  * legacy ifconfig(8) tool which cannot display >1 IPv4 address
50  * (it can display many IPv6 addresses which makes the limitation odd).
51  * Linux has ip(8) which is a more feature rich tool, without the above
52  * restriction.
53  */
54 #ifndef ALIAS_ADDR
55 #  ifdef __sun
56 #    define ALIAS_ADDR
57 #  endif
58 #endif
59
60 #include "config.h"
61
62 /* POSIX defines ioctl request as an int, which Solaris and musl use.
63  * Everyone else use an unsigned long, which happens to be the bigger one
64  * so we use that in our on wire API. */
65 #ifdef IOCTL_REQUEST_TYPE
66 typedef IOCTL_REQUEST_TYPE      ioctl_request_t;
67 #else
68 typedef unsigned long           ioctl_request_t;
69 #endif
70
71 #include "dhcpcd.h"
72 #include "ipv4.h"
73 #include "ipv6.h"
74 #include "route.h"
75
76 #define EUI64_ADDR_LEN                  8
77 #define INFINIBAND_ADDR_LEN             20
78
79 /* Linux 2.4 doesn't define this */
80 #ifndef ARPHRD_IEEE1394
81 #  define ARPHRD_IEEE1394               24
82 #endif
83
84 /* The BSD's don't define this yet */
85 #ifndef ARPHRD_INFINIBAND
86 #  define ARPHRD_INFINIBAND             32
87 #endif
88
89 /* Maximum frame length.
90  * Support jumbo frames and some extra. */
91 #define FRAMEHDRLEN_MAX                 14      /* only ethernet support */
92 #define FRAMELEN_MAX                    (FRAMEHDRLEN_MAX + 9216)
93
94 #define UDPLEN_MAX                      64 * 1024
95
96 /* Work out if we have a private address or not
97  * 10/8
98  * 172.16/12
99  * 192.168/16
100  */
101 #ifndef IN_PRIVATE
102 # define IN_PRIVATE(addr) (((addr & IN_CLASSA_NET) == 0x0a000000) ||          \
103             ((addr & 0xfff00000)    == 0xac100000) ||                         \
104             ((addr & IN_CLASSB_NET) == 0xc0a80000))
105 #endif
106
107 #ifndef CLLADDR
108 #ifdef AF_LINK
109 #  define CLLADDR(sdl) (const void *)((sdl)->sdl_data + (sdl)->sdl_nlen)
110 #endif
111 #endif
112
113 #ifdef __sun
114 /* Solaris stupidly defines this for compat with BSD
115  * but then ignores it. */
116 #undef RTF_CLONING
117
118 /* This interface is busted on DilOS at least.
119  * It used to work, but lukily Solaris can fall back to
120  * IP_PKTINFO. */
121 #undef IP_RECVIF
122
123 /* Solaris getifaddrs is very un-suitable for dhcpcd.
124  * See if-sun.c for details why. */
125 struct ifaddrs;
126 int if_getifaddrs(struct ifaddrs **);
127 #define getifaddrs      if_getifaddrs
128 int if_getsubnet(struct dhcpcd_ctx *, const char *, int, void *, size_t);
129 #endif
130
131 int if_ioctl(struct dhcpcd_ctx *, ioctl_request_t, void *, size_t);
132 #ifdef HAVE_PLEDGE
133 #define pioctl(ctx, req, data, len) if_ioctl((ctx), (req), (data), (len))
134 #else
135 #define pioctl(ctx, req, data, len) ioctl((ctx)->pf_inet_fd, (req),(data),(len))
136 #endif
137 int if_getflags(struct interface *);
138 int if_setflag(struct interface *, short, short);
139 #define if_up(ifp) if_setflag((ifp), (IFF_UP | IFF_RUNNING), 0)
140 #define if_down(ifp) if_setflag((ifp), 0, IFF_UP);
141 bool if_is_link_up(const struct interface *);
142 bool if_valid_hwaddr(const uint8_t *, size_t);
143 struct if_head *if_discover(struct dhcpcd_ctx *, struct ifaddrs **,
144     int, char * const *);
145 void if_markaddrsstale(struct if_head *);
146 void if_learnaddrs(struct dhcpcd_ctx *, struct if_head *, struct ifaddrs **);
147 void if_deletestaleaddrs(struct if_head *);
148 struct interface *if_find(struct if_head *, const char *);
149 struct interface *if_findindex(struct if_head *, unsigned int);
150 struct interface *if_loopback(struct dhcpcd_ctx *);
151 void if_free(struct interface *);
152 int if_domtu(const struct interface *, short int);
153 #define if_getmtu(ifp) if_domtu((ifp), 0)
154 #define if_setmtu(ifp, mtu) if_domtu((ifp), (mtu))
155 int if_carrier(struct interface *, const void *);
156 bool if_roaming(struct interface *);
157
158 #ifdef ALIAS_ADDR
159 int if_makealias(char *, size_t, const char *, int);
160 #endif
161
162 int if_mtu_os(const struct interface *);
163
164 /*
165  * Helper to decode an interface name of bge0:1 to
166  * devname = bge0, drvname = bge0, ppa = 0, lun = 1.
167  * If ppa or lun are invalid they are set to -1.
168  */
169 struct if_spec {
170         char ifname[IF_NAMESIZE];
171         char devname[IF_NAMESIZE];
172         char drvname[IF_NAMESIZE];
173         int ppa;
174         int vlid;
175         int lun;
176 };
177 int if_nametospec(const char *, struct if_spec *);
178
179 /* The below functions are provided by if-KERNEL.c */
180 int os_init(void);
181 int if_conf(struct interface *);
182 int if_init(struct interface *);
183 int if_getssid(struct interface *);
184 int if_ignoregroup(int, const char *);
185 bool if_ignore(struct dhcpcd_ctx *, const char *);
186 int if_vimaster(struct dhcpcd_ctx *ctx, const char *);
187 unsigned short if_vlanid(const struct interface *);
188 char * if_getnetworknamespace(char *, size_t);
189 int if_opensockets(struct dhcpcd_ctx *);
190 int if_opensockets_os(struct dhcpcd_ctx *);
191 void if_closesockets(struct dhcpcd_ctx *);
192 void if_closesockets_os(struct dhcpcd_ctx *);
193 int if_handlelink(struct dhcpcd_ctx *);
194 int if_randomisemac(struct interface *);
195 int if_setmac(struct interface *ifp, void *, uint8_t);
196
197 /* dhcpcd uses the same routing flags as BSD.
198  * If the platform doesn't use these flags,
199  * map them in the platform interace file. */
200 #ifndef RTM_ADD
201 #define RTM_ADD         0x1     /* Add Route */
202 #define RTM_DELETE      0x2     /* Delete Route */
203 #define RTM_CHANGE      0x3     /* Change Metrics or flags */
204 #define RTM_GET         0x4     /* Report Metrics */
205 #endif
206
207 /* Define SOCK_CLOEXEC and SOCK_NONBLOCK for systems that lack it.
208  * xsocket() in if.c will map them to fctnl FD_CLOEXEC and O_NONBLOCK. */
209 #ifdef SOCK_CLOEXEC
210 # define HAVE_SOCK_CLOEXEC
211 #else
212 # define SOCK_CLOEXEC   0x10000000
213 #endif
214 #ifdef SOCK_NONBLOCK
215 # define HAVE_SOCK_NONBLOCK
216 #else
217 # define SOCK_NONBLOCK  0x20000000
218 #endif
219 #ifndef SOCK_CXNB
220 #define SOCK_CXNB       SOCK_CLOEXEC | SOCK_NONBLOCK
221 #endif
222 int xsocket(int, int, int);
223 int xsocketpair(int, int, int, int[2]);
224
225 int if_route(unsigned char, const struct rt *rt);
226 int if_initrt(struct dhcpcd_ctx *, rb_tree_t *, int);
227
228 int if_missfilter(struct interface *, struct sockaddr *);
229 int if_missfilter_apply(struct dhcpcd_ctx *);
230
231 #ifdef INET
232 int if_address(unsigned char, const struct ipv4_addr *);
233 int if_addrflags(const struct interface *, const struct in_addr *,
234     const char *);
235
236 #endif
237
238 #ifdef INET6
239 void if_disable_rtadv(void);
240 void if_setup_inet6(const struct interface *);
241 int ip6_forwarding(const char *ifname);
242
243 struct ra;
244 struct ipv6_addr;
245
246 int if_applyra(const struct ra *);
247 int if_address6(unsigned char, const struct ipv6_addr *);
248 int if_addrflags6(const struct interface *, const struct in6_addr *,
249     const char *);
250 int if_getlifetime6(struct ipv6_addr *);
251
252 #else
253 #define if_checkipv6(a, b, c) (-1)
254 #endif
255
256 int if_machinearch(char *, size_t);
257 struct interface *if_findifpfromcmsg(struct dhcpcd_ctx *,
258     struct msghdr *, int *);
259
260 #ifdef __linux__
261 int if_linksocket(struct sockaddr_nl *, int, int);
262 int if_getnetlink(struct dhcpcd_ctx *, struct iovec *, int, int,
263     int (*)(struct dhcpcd_ctx *, void *, struct nlmsghdr *), void *);
264 #endif
265 #endif