11de7b4b8SPedro F. Giffuni /*- 21de7b4b8SPedro F. Giffuni * SPDX-License-Identifier: BSD-4-Clause 31de7b4b8SPedro F. Giffuni * 49906480aSPeter Wemm * Copyright (c) 1997 Peter Wemm. 59906480aSPeter Wemm * All rights reserved. 69906480aSPeter Wemm * 79906480aSPeter Wemm * Redistribution and use in source and binary forms, with or without 89906480aSPeter Wemm * modification, are permitted provided that the following conditions 99906480aSPeter Wemm * are met: 109906480aSPeter Wemm * 1. Redistributions of source code must retain the above copyright 119906480aSPeter Wemm * notice, this list of conditions and the following disclaimer. 129906480aSPeter Wemm * 2. Redistributions in binary form must reproduce the above copyright 139906480aSPeter Wemm * notice, this list of conditions and the following disclaimer in the 149906480aSPeter Wemm * documentation and/or other materials provided with the distribution. 159906480aSPeter Wemm * 3. All advertising materials mentioning features or use of this software 169906480aSPeter Wemm * must display the following acknowledgement: 179906480aSPeter Wemm * This product includes software developed for the FreeBSD Project 189906480aSPeter Wemm * by Peter Wemm. 199906480aSPeter Wemm * 4. The name of the author may not be used to endorse or promote products 209906480aSPeter Wemm * derived from this software without specific prior written permission. 219906480aSPeter Wemm * 229906480aSPeter Wemm * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR 239906480aSPeter Wemm * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES 249906480aSPeter Wemm * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. 259906480aSPeter Wemm * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, 269906480aSPeter Wemm * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, 279906480aSPeter Wemm * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; 289906480aSPeter Wemm * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED 299906480aSPeter Wemm * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, 309906480aSPeter Wemm * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 319906480aSPeter Wemm * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 329906480aSPeter Wemm * SUCH DAMAGE. 339906480aSPeter Wemm * 349906480aSPeter Wemm * so there! 359906480aSPeter Wemm * 367f3dea24SPeter Wemm * $FreeBSD$ 379906480aSPeter Wemm */ 389906480aSPeter Wemm 398b222425SRyan Moeller #pragma once 408b222425SRyan Moeller 418b222425SRyan Moeller #include <libifconfig.h> 42982cd5aeSAlexander V. Chernikov #include <stdbool.h> 438b222425SRyan Moeller 445faf8dcbSSam Leffler #define __constructor __attribute__((constructor)) 459906480aSPeter Wemm 465faf8dcbSSam Leffler struct afswtch; 475faf8dcbSSam Leffler struct cmd; 48*6e3a9d7fSAlexander V. Chernikov struct ifconfig_context; 495faf8dcbSSam Leffler 50*6e3a9d7fSAlexander V. Chernikov typedef void c_func(const struct ifconfig_context *ctx, const char *cmd, int arg); 51*6e3a9d7fSAlexander V. Chernikov typedef void c_func2(const struct ifconfig_context *ctx, const char *arg1, 52*6e3a9d7fSAlexander V. Chernikov const char *arg2); 53*6e3a9d7fSAlexander V. Chernikov typedef void c_func3(const struct ifconfig_context *ctx, const char *cmd, 54*6e3a9d7fSAlexander V. Chernikov const char *arg); 555faf8dcbSSam Leffler 565faf8dcbSSam Leffler struct cmd { 575faf8dcbSSam Leffler const char *c_name; 585faf8dcbSSam Leffler int c_parameter; 595faf8dcbSSam Leffler #define NEXTARG 0xffffff /* has following arg */ 605faf8dcbSSam Leffler #define NEXTARG2 0xfffffe /* has 2 following args */ 615faf8dcbSSam Leffler #define OPTARG 0xfffffd /* has optional following arg */ 6242d5cb09SKonstantin Belousov #define SPARAM 0xfffffc /* parameter is string c_sparameter */ 6342d5cb09SKonstantin Belousov const char *c_sparameter; 645faf8dcbSSam Leffler union { 655faf8dcbSSam Leffler c_func *c_func; 665faf8dcbSSam Leffler c_func2 *c_func2; 6742d5cb09SKonstantin Belousov c_func3 *c_func3; 68c508bf7aSStefan Farfeleder } c_u; 692fa02c5fSSam Leffler int c_iscloneop; 705faf8dcbSSam Leffler struct cmd *c_next; 715faf8dcbSSam Leffler }; 725faf8dcbSSam Leffler void cmd_register(struct cmd *); 735faf8dcbSSam Leffler 7458ac5e5bSSam Leffler typedef void callback_func(int s, void *); 7558ac5e5bSSam Leffler void callback_register(callback_func *, void *); 7658ac5e5bSSam Leffler 775faf8dcbSSam Leffler /* 78*6e3a9d7fSAlexander V. Chernikov * Macros for initializing command handlers. 795faf8dcbSSam Leffler */ 805faf8dcbSSam Leffler 817aecd12dSKonstantin Belousov #define DEF_CMD(name, param, func) { \ 827aecd12dSKonstantin Belousov .c_name = (name), \ 837aecd12dSKonstantin Belousov .c_parameter = (param), \ 847aecd12dSKonstantin Belousov .c_u = { .c_func = (func) }, \ 857aecd12dSKonstantin Belousov .c_iscloneop = 0, \ 8642d5cb09SKonstantin Belousov .c_next = NULL, \ 877aecd12dSKonstantin Belousov } 887aecd12dSKonstantin Belousov #define DEF_CMD_ARG(name, func) { \ 897aecd12dSKonstantin Belousov .c_name = (name), \ 907aecd12dSKonstantin Belousov .c_parameter = NEXTARG, \ 917aecd12dSKonstantin Belousov .c_u = { .c_func = (func) }, \ 927aecd12dSKonstantin Belousov .c_iscloneop = 0, \ 937aecd12dSKonstantin Belousov .c_next = NULL, \ 947aecd12dSKonstantin Belousov } 957aecd12dSKonstantin Belousov #define DEF_CMD_OPTARG(name, func) { \ 967aecd12dSKonstantin Belousov .c_name = (name), \ 977aecd12dSKonstantin Belousov .c_parameter = OPTARG, \ 987aecd12dSKonstantin Belousov .c_u = { .c_func = (func) }, \ 997aecd12dSKonstantin Belousov .c_iscloneop = 0, \ 1007aecd12dSKonstantin Belousov .c_next = NULL, \ 1017aecd12dSKonstantin Belousov } 1027aecd12dSKonstantin Belousov #define DEF_CMD_ARG2(name, func) { \ 1037aecd12dSKonstantin Belousov .c_name = (name), \ 1047aecd12dSKonstantin Belousov .c_parameter = NEXTARG2, \ 1057aecd12dSKonstantin Belousov .c_u = { .c_func2 = (func) }, \ 1067aecd12dSKonstantin Belousov .c_iscloneop = 0, \ 1077aecd12dSKonstantin Belousov .c_next = NULL, \ 1087aecd12dSKonstantin Belousov } 10942d5cb09SKonstantin Belousov #define DEF_CMD_SARG(name, sparam, func) { \ 11042d5cb09SKonstantin Belousov .c_name = (name), \ 11142d5cb09SKonstantin Belousov .c_parameter = SPARAM, \ 11242d5cb09SKonstantin Belousov .c_sparameter = (sparam), \ 11342d5cb09SKonstantin Belousov .c_u = { .c_func3 = (func) }, \ 11442d5cb09SKonstantin Belousov .c_iscloneop = 0, \ 11542d5cb09SKonstantin Belousov .c_next = NULL, \ 11642d5cb09SKonstantin Belousov } 1177aecd12dSKonstantin Belousov #define DEF_CLONE_CMD(name, param, func) { \ 1187aecd12dSKonstantin Belousov .c_name = (name), \ 1197aecd12dSKonstantin Belousov .c_parameter = (param), \ 1207aecd12dSKonstantin Belousov .c_u = { .c_func = (func) }, \ 1217aecd12dSKonstantin Belousov .c_iscloneop = 1, \ 1227aecd12dSKonstantin Belousov .c_next = NULL, \ 1237aecd12dSKonstantin Belousov } 1247aecd12dSKonstantin Belousov #define DEF_CLONE_CMD_ARG(name, func) { \ 1257aecd12dSKonstantin Belousov .c_name = (name), \ 1267aecd12dSKonstantin Belousov .c_parameter = NEXTARG, \ 1277aecd12dSKonstantin Belousov .c_u = { .c_func = (func) }, \ 1287aecd12dSKonstantin Belousov .c_iscloneop = 1, \ 1297aecd12dSKonstantin Belousov .c_next = NULL, \ 1307aecd12dSKonstantin Belousov } 1317aecd12dSKonstantin Belousov #define DEF_CLONE_CMD_ARG2(name, func) { \ 1327aecd12dSKonstantin Belousov .c_name = (name), \ 1337aecd12dSKonstantin Belousov .c_parameter = NEXTARG2, \ 1347aecd12dSKonstantin Belousov .c_u = { .c_func2 = (func) }, \ 1357aecd12dSKonstantin Belousov .c_iscloneop = 1, \ 1367aecd12dSKonstantin Belousov .c_next = NULL, \ 1377aecd12dSKonstantin Belousov } 1385faf8dcbSSam Leffler 139*6e3a9d7fSAlexander V. Chernikov struct snl_state; 140*6e3a9d7fSAlexander V. Chernikov struct ifconfig_args; 141*6e3a9d7fSAlexander V. Chernikov struct ifconfig_context { 142*6e3a9d7fSAlexander V. Chernikov struct ifconfig_args *args; 143*6e3a9d7fSAlexander V. Chernikov const struct afswtch *afp; 144*6e3a9d7fSAlexander V. Chernikov int io_s; /* fd to use for ioctl() */ 145*6e3a9d7fSAlexander V. Chernikov struct snl_state *io_ss; /* NETLINK_ROUTE socket */ 146*6e3a9d7fSAlexander V. Chernikov }; 147*6e3a9d7fSAlexander V. Chernikov typedef const struct ifconfig_context if_ctx; 148*6e3a9d7fSAlexander V. Chernikov 149*6e3a9d7fSAlexander V. Chernikov #define ioctl_ctx(ctx, _req, ...) ioctl((ctx)->io_s, _req, ## __VA_ARGS__) 150*6e3a9d7fSAlexander V. Chernikov 151cb8c905aSSam Leffler struct ifaddrs; 1525faf8dcbSSam Leffler struct addrinfo; 1535faf8dcbSSam Leffler 1545faf8dcbSSam Leffler enum { 15526056fa8SAlexander V. Chernikov RIDADDR, 15626056fa8SAlexander V. Chernikov ADDR, 15726056fa8SAlexander V. Chernikov MASK, 15826056fa8SAlexander V. Chernikov DSTADDR, 1595faf8dcbSSam Leffler }; 1605faf8dcbSSam Leffler 1614c91a5dfSAlexander V. Chernikov struct snl_parsed_addr; 1624c91a5dfSAlexander V. Chernikov struct snl_parsed_link; 1634c91a5dfSAlexander V. Chernikov typedef struct snl_parsed_link if_link_t; 1644c91a5dfSAlexander V. Chernikov typedef struct snl_parsed_addr if_addr_t; 1654c91a5dfSAlexander V. Chernikov 166a72b7890SAlexander V. Chernikov typedef void af_setvhid_f(int vhid); 167*6e3a9d7fSAlexander V. Chernikov typedef void af_status_nl_f(if_ctx *ctx, if_link_t *link, if_addr_t *ifa); 168*6e3a9d7fSAlexander V. Chernikov typedef void af_status_f(if_ctx *ctx, const struct ifaddrs *); 169*6e3a9d7fSAlexander V. Chernikov typedef void af_other_status_f(if_ctx *ctx); 170*6e3a9d7fSAlexander V. Chernikov typedef void af_postproc_f(if_ctx *ctx, int newaddr, int ifflags); 1714c91a5dfSAlexander V. Chernikov 1725faf8dcbSSam Leffler struct afswtch { 1735faf8dcbSSam Leffler const char *af_name; /* as given on cmd line, e.g. "inet" */ 1745faf8dcbSSam Leffler short af_af; /* AF_* */ 17590c4b74cSSam Leffler /* 17690c4b74cSSam Leffler * Status is handled one of two ways; if there is an 17790c4b74cSSam Leffler * address associated with the interface then the 17890c4b74cSSam Leffler * associated address family af_status method is invoked 17990c4b74cSSam Leffler * with the appropriate addressin info. Otherwise, if 18090c4b74cSSam Leffler * all possible info is to be displayed and af_other_status 18190c4b74cSSam Leffler * is defined then it is invoked after all address status 18290c4b74cSSam Leffler * is presented. 18390c4b74cSSam Leffler */ 1844c91a5dfSAlexander V. Chernikov #ifndef WITHOUT_NETLINK 185*6e3a9d7fSAlexander V. Chernikov af_status_nl_f *af_status; 1864c91a5dfSAlexander V. Chernikov #else 187*6e3a9d7fSAlexander V. Chernikov af_status_f *af_status; 1884c91a5dfSAlexander V. Chernikov #endif 189*6e3a9d7fSAlexander V. Chernikov af_other_status_f *af_other_status; 1905faf8dcbSSam Leffler void (*af_getaddr)(const char *, int); 1915faf8dcbSSam Leffler /* parse prefix method (IPv6) */ 1925faf8dcbSSam Leffler void (*af_getprefix)(const char *, int); 193*6e3a9d7fSAlexander V. Chernikov af_postproc_f *af_postproc; 194a72b7890SAlexander V. Chernikov af_setvhid_f *af_setvhid; /* Set CARP vhid for an address */ 1955faf8dcbSSam Leffler u_long af_difaddr; /* set dst if address ioctl */ 1965faf8dcbSSam Leffler u_long af_aifaddr; /* set if address ioctl */ 1975faf8dcbSSam Leffler void *af_ridreq; /* */ 1985faf8dcbSSam Leffler void *af_addreq; /* */ 1995faf8dcbSSam Leffler struct afswtch *af_next; 2005faf8dcbSSam Leffler 2015faf8dcbSSam Leffler /* XXX doesn't fit model */ 2025faf8dcbSSam Leffler void (*af_status_tunnel)(int); 2035faf8dcbSSam Leffler void (*af_settunnel)(int s, struct addrinfo *srcres, 2045faf8dcbSSam Leffler struct addrinfo *dstres); 2055faf8dcbSSam Leffler }; 2065faf8dcbSSam Leffler void af_register(struct afswtch *); 2075faf8dcbSSam Leffler 208982cd5aeSAlexander V. Chernikov struct ifconfig_args { 209982cd5aeSAlexander V. Chernikov bool all; /* Match everything */ 210982cd5aeSAlexander V. Chernikov bool downonly; /* Down-only items */ 211982cd5aeSAlexander V. Chernikov bool uponly; /* Up-only items */ 212982cd5aeSAlexander V. Chernikov bool namesonly; /* Output only names */ 213982cd5aeSAlexander V. Chernikov bool noload; /* Do not load relevant kernel modules */ 214982cd5aeSAlexander V. Chernikov bool supmedia; /* Supported media */ 215982cd5aeSAlexander V. Chernikov bool printkeys; /* Print security keys */ 216d1cd0344SAlexander V. Chernikov bool allfamilies; /* Print all families */ 217982cd5aeSAlexander V. Chernikov int verbose; /* verbosity level */ 218982cd5aeSAlexander V. Chernikov int argc; 219982cd5aeSAlexander V. Chernikov char **argv; 220a3cb80bcSAlexander V. Chernikov const char *ifname; /* Requested interface name */ 221982cd5aeSAlexander V. Chernikov const char *matchgroup; /* Group name to match */ 222982cd5aeSAlexander V. Chernikov const char *nogroup; /* Group name to exclude */ 223982cd5aeSAlexander V. Chernikov const struct afswtch *afp; /* AF we're operating on */ 2247e49aa86SYan Ka Chiu const char *jail_name; /* Jail name or jail id specified */ 225982cd5aeSAlexander V. Chernikov }; 226982cd5aeSAlexander V. Chernikov 2275faf8dcbSSam Leffler struct option { 2285faf8dcbSSam Leffler const char *opt; 2295faf8dcbSSam Leffler const char *opt_usage; 2305faf8dcbSSam Leffler void (*cb)(const char *arg); 2315faf8dcbSSam Leffler struct option *next; 2325faf8dcbSSam Leffler }; 2335faf8dcbSSam Leffler void opt_register(struct option *); 2345faf8dcbSSam Leffler 2358b222425SRyan Moeller extern ifconfig_handle_t *lifh; 2365faf8dcbSSam Leffler extern struct ifreq ifr; 237a4fa9864SBrooks Davis extern char name[IFNAMSIZ]; /* name of interface */ 2389906480aSPeter Wemm extern int allmedia; 239b7eb47e2SRobert Watson extern int printkeys; 2405faf8dcbSSam Leffler extern int newaddr; 2415faf8dcbSSam Leffler extern int verbose; 2426fe2e176SKristof Provost extern int printifname; 24347528c67SAndrew Gallatin extern int exit_code; 244*6e3a9d7fSAlexander V. Chernikov extern struct ifconfig_args global_args; 24544cd85d4SAlexander V. Chernikov extern char *f_inet, *f_inet6, *f_ether, *f_addr; 2469906480aSPeter Wemm 247*6e3a9d7fSAlexander V. Chernikov void setifcap(if_ctx *ctx, const char *, int value); 248*6e3a9d7fSAlexander V. Chernikov void setifcapnv(if_ctx *ctx, const char *vname, const char *arg); 24997ed1257SBill Paul 2505faf8dcbSSam Leffler void Perror(const char *cmd); 2515faf8dcbSSam Leffler void printb(const char *s, unsigned value, const char *bits); 252419d8080SPoul-Henning Kamp 253982cd5aeSAlexander V. Chernikov void ifmaybeload(struct ifconfig_args *args, const char *name); 2545faf8dcbSSam Leffler 255c7cffd65SAlexander V. Chernikov typedef int clone_match_func(const char *); 256db82353dSSam Leffler typedef void clone_callback_func(int, struct ifreq *); 257c7cffd65SAlexander V. Chernikov void clone_setdefcallback_prefix(const char *, clone_callback_func *); 258c7cffd65SAlexander V. Chernikov void clone_setdefcallback_filter(clone_match_func *, clone_callback_func *); 2598f3591a5SSam Leffler 260f732123eSAlexander V. Chernikov void sfp_status(int s, struct ifreq *ifr, int verbose); 261f732123eSAlexander V. Chernikov 262d1cd0344SAlexander V. Chernikov struct sockaddr_dl; 263d1cd0344SAlexander V. Chernikov bool match_ether(const struct sockaddr_dl *sdl); 2644c91a5dfSAlexander V. Chernikov bool match_if_flags(struct ifconfig_args *args, int if_flags); 26526056fa8SAlexander V. Chernikov int ifconfig(int argc, char *const *argv, int iscreate, const struct afswtch *uafp); 2664c91a5dfSAlexander V. Chernikov bool group_member(const char *ifname, const char *match, const char *nomatch); 2674c91a5dfSAlexander V. Chernikov void print_ifcap(struct ifconfig_args *args, int s); 2684c91a5dfSAlexander V. Chernikov void tunnel_status(int s); 2694c91a5dfSAlexander V. Chernikov struct afswtch *af_getbyfamily(int af); 270*6e3a9d7fSAlexander V. Chernikov void af_other_status(if_ctx *ctx); 2714c91a5dfSAlexander V. Chernikov void print_ifstatus(int s); 2724c91a5dfSAlexander V. Chernikov void print_metric(int s); 2734c91a5dfSAlexander V. Chernikov 2744c91a5dfSAlexander V. Chernikov /* Netlink-related functions */ 2754c91a5dfSAlexander V. Chernikov void list_interfaces_nl(struct ifconfig_args *args); 2764c91a5dfSAlexander V. Chernikov 2778f3591a5SSam Leffler /* 2788f3591a5SSam Leffler * XXX expose this so modules that neeed to know of any pending 2798f3591a5SSam Leffler * operations on ifmedia can avoid cmd line ordering confusion. 2808f3591a5SSam Leffler */ 2812803fa47SRyan Moeller struct ifmediareq *ifmedia_getstate(void); 28208b68b0eSGleb Smirnoff 28308b68b0eSGleb Smirnoff void print_vhid(const struct ifaddrs *, const char *); 28408b68b0eSGleb Smirnoff 28505952067SHans Petter Selasky void ioctl_ifcreate(int s, struct ifreq *); 28644cd85d4SAlexander V. Chernikov 28744cd85d4SAlexander V. Chernikov /* Helpers */ 28844cd85d4SAlexander V. Chernikov struct sockaddr_in; 28944cd85d4SAlexander V. Chernikov struct sockaddr_in6; 29044cd85d4SAlexander V. Chernikov struct sockaddr; 29144cd85d4SAlexander V. Chernikov 29244cd85d4SAlexander V. Chernikov static inline struct sockaddr_in6 * 29344cd85d4SAlexander V. Chernikov satosin6(struct sockaddr *sa) 29444cd85d4SAlexander V. Chernikov { 29544cd85d4SAlexander V. Chernikov return ((struct sockaddr_in6 *)(void *)sa); 29644cd85d4SAlexander V. Chernikov } 29744cd85d4SAlexander V. Chernikov 29844cd85d4SAlexander V. Chernikov static inline struct sockaddr_in * 29944cd85d4SAlexander V. Chernikov satosin(struct sockaddr *sa) 30044cd85d4SAlexander V. Chernikov { 30144cd85d4SAlexander V. Chernikov return ((struct sockaddr_in *)(void *)sa); 30244cd85d4SAlexander V. Chernikov } 303