/** * Copyright (C) 2012-2013 Steven Barth * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License v2 as published by * the Free Software Foundation. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * */ #pragma once #include #include #include #include #include #include #include #include #include // RFC 6106 defines this router advertisement option #define ND_OPT_ROUTE_INFO 24 #define ND_OPT_RECURSIVE_DNS 25 #define ND_OPT_DNS_SEARCH 31 #define INFINITE_VALID(x) ((x) == 0) #define _unused __attribute__((unused)) #define _packed __attribute__((packed)) #define ALL_IPV6_NODES "ff02::1" #define ALL_IPV6_ROUTERS "ff02::2" #define IN6_IS_ADDR_ULA(a) (((a)->s6_addr32[0] & htonl(0xfe000000)) == htonl(0xfc000000)) struct interface; struct nl_sock; extern struct list_head leases; struct odhcpd_event { struct uloop_fd uloop; void (*handle_dgram)(void *addr, void *data, size_t len, struct interface *iface, void *dest_addr); void (*handle_error)(struct odhcpd_event *e, int error); void (*recv_msgs)(struct odhcpd_event *e); }; union if_addr { struct in_addr in; struct in6_addr in6; }; struct netevent_handler_info { struct interface *iface; union { struct { union if_addr dst; uint8_t dst_len; union if_addr gateway; } rt; struct { union if_addr dst; uint16_t state; uint8_t flags; } neigh; struct { struct odhcpd_ipaddr *addrs; size_t len; } addrs_old; union if_addr addr; }; }; enum netevents { NETEV_IFINDEX_CHANGE, NETEV_ADDR_ADD, NETEV_ADDR_DEL, NETEV_ADDRLIST_CHANGE, NETEV_ADDR6_ADD, NETEV_ADDR6_DEL, NETEV_ADDR6LIST_CHANGE, NETEV_ROUTE6_ADD, NETEV_ROUTE6_DEL, NETEV_NEIGH6_ADD, NETEV_NEIGH6_DEL, }; struct netevent_handler { struct list_head head; void (*cb) (unsigned long event, struct netevent_handler_info *info); }; struct odhcpd_ipaddr { union if_addr addr; uint8_t prefix; uint32_t preferred; uint32_t valid; /* ipv6 only */ uint8_t dprefix; /* ipv4 only */ struct in_addr broadcast; }; enum odhcpd_mode { MODE_DISABLED, MODE_SERVER, MODE_RELAY, MODE_HYBRID }; enum odhcpd_assignment_flags { OAF_TENTATIVE = (1 << 0), OAF_BOUND = (1 << 1), OAF_STATIC = (1 << 2), OAF_BROKEN_HOSTNAME = (1 << 3), }; struct config { bool legacy; bool main_dhcpv4; char *dhcp_cb; char *dhcp_statefile; int log_level; } config; struct lease { struct list_head head; struct in_addr ipaddr; uint32_t hostid; struct ether_addr mac; uint16_t duid_len; uint8_t *duid; uint32_t dhcpv4_leasetime; char hostname[]; }; struct interface { struct avl_node avl; int ifindex; char *ifname; const char *name; // IPv6 runtime data struct odhcpd_ipaddr *addr6; size_t addr6_len; // RA runtime data struct uloop_timeout timer_rs; // DHCPv6 runtime data struct odhcpd_event dhcpv6_event; struct list_head ia_assignments; // NDP runtime data struct odhcpd_event ndp_event; // IPv4 runtime data struct odhcpd_ipaddr *addr4; size_t addr4_len; // DHCPv4 runtime data struct odhcpd_event dhcpv4_event; struct list_head dhcpv4_assignments; struct list_head dhcpv4_fr_ips; // Managed PD char dhcpv6_pd_manager[128]; struct in6_addr dhcpv6_pd_cer; // Services enum odhcpd_mode ra; enum odhcpd_mode dhcpv6; enum odhcpd_mode ndp; enum odhcpd_mode dhcpv4; // Config bool inuse; bool external; bool master; bool ignore; bool always_rewrite_dns; bool ra_not_onlink; bool ra_advrouter; bool ra_useleasetime; bool ra_dns; bool no_dynamic_dhcp; uint8_t pio_filter_length; struct in6_addr pio_filter_addr; // RA int learn_routes; int default_router; int ra_managed; int route_preference; int ra_maxinterval; int ra_mininterval; int ra_lifetime; uint32_t ra_reachabletime; uint32_t ra_retranstime; uint32_t ra_hoplimit; int ra_mtu; // DHCPv4 struct in_addr dhcpv4_start; struct in_addr dhcpv4_end; struct in_addr dhcpv4_start_ip; struct in_addr dhcpv4_end_ip; struct in_addr dhcpv4_local; struct in_addr dhcpv4_bcast; struct in_addr dhcpv4_mask; struct in_addr *dhcpv4_router; size_t dhcpv4_router_cnt; struct in_addr *dhcpv4_dns; size_t dhcpv4_dns_cnt; uint32_t dhcpv4_leasetime; bool dhcpv4_forcereconf; // DNS struct in6_addr *dns; size_t dns_cnt; uint8_t *search; size_t search_len; // DHCPV6 void *dhcpv6_raw; size_t dhcpv6_raw_len; bool dhcpv6_assignall; bool dhcpv6_pd; bool dhcpv6_na; char *upstream; size_t upstream_len; char *filter_class; }; extern struct avl_tree interfaces; #define RA_MANAGED_NO_MFLAG 0 #define RA_MANAGED_MFLAG 1 #define RA_MANAGED_NO_AFLAG 2 // Exported main functions int odhcpd_register(struct odhcpd_event *event); int odhcpd_deregister(struct odhcpd_event *event); void odhcpd_process(struct odhcpd_event *event); ssize_t odhcpd_send(int socket, struct sockaddr_in6 *dest, struct iovec *iov, size_t iov_len, const struct interface *iface); int odhcpd_get_interface_dns_addr(const struct interface *iface, struct in6_addr *addr); struct interface* odhcpd_get_interface_by_name(const char *name); int odhcpd_get_interface_config(const char *ifname, const char *what); int odhcpd_get_mac(const struct interface *iface, uint8_t mac[6]); struct interface* odhcpd_get_interface_by_index(int ifindex); struct interface* odhcpd_get_master_interface(void); int odhcpd_urandom(void *data, size_t len); void odhcpd_run(void); time_t odhcpd_time(void); ssize_t odhcpd_unhexlify(uint8_t *dst, size_t len, const char *src); void odhcpd_hexlify(char *dst, const uint8_t *src, size_t len); const char *odhcpd_print_mac(const uint8_t *mac, const size_t len); int odhcpd_bmemcmp(const void *av, const void *bv, size_t bits); void odhcpd_bmemcpy(void *av, const void *bv, size_t bits); int odhcpd_netmask2bitlen(bool v6, void *mask); bool odhcpd_bitlen2netmask(bool v6, unsigned int bits, void *mask); bool odhcpd_valid_hostname(const char *name); int config_parse_interface(void *data, size_t len, const char *iname, bool overwrite); #ifdef WITH_UBUS int ubus_init(void); const char* ubus_get_ifname(const char *name); void ubus_apply_network(void); bool ubus_has_prefix(const char *name, const char *ifname); void ubus_bcast_dhcp_event(const char *type, const uint8_t *mac, const size_t mac_len, const struct in_addr *addr, const char *name, const char *interface); #endif int netlink_add_netevent_handler(struct netevent_handler *hdlr); ssize_t netlink_get_interface_addrs(const int ifindex, bool v6, struct odhcpd_ipaddr **addrs); int netlink_setup_route(const struct in6_addr *addr, const int prefixlen, const int ifindex, const struct in6_addr *gw, const uint32_t metric, const bool add); int netlink_setup_proxy_neigh(const struct in6_addr *addr, const int ifindex, const bool add); int netlink_setup_addr(struct odhcpd_ipaddr *addr, const int ifindex, const bool v6, const bool add); void netlink_dump_neigh_table(const bool proxy); void netlink_dump_addr_table(const bool v6); // Exported module initializers int netlink_init(void); int router_init(void); int dhcpv6_init(void); int ndp_init(void); #ifdef DHCPV4_SUPPORT int dhcpv4_init(void); int dhcpv4_setup_interface(struct interface *iface, bool enable); #endif int router_setup_interface(struct interface *iface, bool enable); int dhcpv6_setup_interface(struct interface *iface, bool enable); int ndp_setup_interface(struct interface *iface, bool enable); void odhcpd_reload(void);