root/usr.sbin/dhcpd/dhcpd.h
/*      $OpenBSD: dhcpd.h,v 1.73 2025/06/10 06:29:53 dlg Exp $ */

/*
 * Copyright (c) 1995, 1996, 1997, 1998, 1999
 * The Internet Software Consortium.    All rights reserved.
 *
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions
 * are met:
 *
 * 1. Redistributions of source code must retain the above copyright
 *    notice, this list of conditions and the following disclaimer.
 * 2. Redistributions in binary form must reproduce the above copyright
 *    notice, this list of conditions and the following disclaimer in the
 *    documentation and/or other materials provided with the distribution.
 * 3. Neither the name of The Internet Software Consortium nor the names
 *    of its contributors may be used to endorse or promote products derived
 *    from this software without specific prior written permission.
 *
 * THIS SOFTWARE IS PROVIDED BY THE INTERNET SOFTWARE CONSORTIUM AND
 * CONTRIBUTORS ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES,
 * INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
 * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
 * DISCLAIMED.  IN NO EVENT SHALL THE INTERNET SOFTWARE CONSORTIUM OR
 * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
 * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
 * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF
 * USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
 * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
 * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT
 * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
 * SUCH DAMAGE.
 *
 * This software has been written for the Internet Software Consortium
 * by Ted Lemon <mellon@fugue.com> in cooperation with Vixie
 * Enterprises.  To learn more about the Internet Software Consortium,
 * see ``http://www.vix.com/isc''.  To learn more about Vixie
 * Enterprises, see ``http://www.vix.com''.
 */

#define ifr_netmask ifr_addr

#define HAVE_SA_LEN
#define HAVE_MKSTEMP

#define DB_TIMEFMT      "%w %Y/%m/%d %T UTC"
#define OLD_DB_TIMEFMT  "%w %Y/%m/%d %T"

#define SERVER_PORT     67
#define CLIENT_PORT     68

struct iaddr {
        int len;
        unsigned char iabuf[16];
};

#define DEFAULT_HASH_SIZE       97

struct hash_bucket {
        struct hash_bucket *next;
        unsigned char *name;
        int len;
        unsigned char *value;
};

struct hash_table {
        int hash_count;
        struct hash_bucket *buckets[DEFAULT_HASH_SIZE];
};

struct option_data {
        int len;
        u_int8_t *data;
};

/* A dhcp packet and the pointers to its option values. */
struct packet {
        struct dhcp_packet *raw;
        int packet_length;
        int packet_type;
        int options_valid;
        int client_port;
        struct iaddr client_addr;
        struct interface_info *interface;       /* Interface on which packet
                                                   was received. */
        struct hardware *haddr;         /* Physical link address
                                           of local sender (maybe gateway). */
        struct shared_network *shared_network;
        struct option_data options[256];
        int got_requested_address;      /* True if client sent the
                                           dhcp-requested-address option. */
};

struct hardware {
        u_int8_t htype;
        u_int8_t hlen;
        u_int8_t haddr[16];
};

/* A dhcp lease declaration structure. */
struct lease {
        struct lease *next;
        struct lease *prev;
        struct lease *n_uid, *n_hw;
        struct lease *waitq_next;

        struct iaddr ip_addr;
        time_t starts, ends, timestamp;
        unsigned char *uid;
        int uid_len;
        int uid_max;
        unsigned char uid_buf[32];
        char *hostname;
        char *client_hostname;
        uint8_t *client_identifier;
        struct host_decl *host;
        struct subnet *subnet;
        struct shared_network *shared_network;
        struct hardware hardware_addr;

        int client_identifier_len;
        int flags;
#define STATIC_LEASE            1
#define BOOTP_LEASE             2
#define DYNAMIC_BOOTP_OK        4
#define PERSISTENT_FLAGS        (DYNAMIC_BOOTP_OK)
#define EPHEMERAL_FLAGS         (BOOTP_LEASE)
#define MS_NULL_TERMINATION     8
#define ABANDONED_LEASE         16
#define INFORM_NOLEASE          32

        struct lease_state *state;
        u_int8_t releasing;
};

struct lease_state {
        struct interface_info *ip;

        time_t offered_expiry;

        struct tree_cache *options[256];
        u_int32_t expiry, renewal, rebind;
        char filename[DHCP_FILE_LEN];
        char *server_name;

        struct iaddr from;

        int max_message_size;
        u_int8_t *prl;
        int prl_len;
        int got_requested_address;      /* True if client sent the
                                           dhcp-requested-address option. */
        int got_server_identifier;      /* True if client sent the
                                           dhcp-server-identifier option. */
        struct shared_network *shared_network;  /* Shared network of interface
                                                   on which request arrived. */

        u_int32_t xid;
        u_int16_t secs;
        u_int16_t bootp_flags;
        struct in_addr ciaddr;
        struct in_addr giaddr;
        u_int8_t hops;
        u_int8_t offer;
        struct hardware haddr;
};

#define ROOT_GROUP      0
#define HOST_DECL       1
#define SHARED_NET_DECL 2
#define SUBNET_DECL     3
#define CLASS_DECL      4
#define GROUP_DECL      5

/* Group of declarations that share common parameters. */
struct group {
        struct group *next;

        struct subnet *subnet;
        struct shared_network *shared_network;

        time_t default_lease_time;
        time_t max_lease_time;
        time_t bootp_lease_cutoff;
        time_t bootp_lease_length;

        char *filename;
        char *server_name;
        struct iaddr next_server;

        int boot_unknown_clients;
        int dynamic_bootp;
        int allow_bootp;
        int allow_booting;
        int get_lease_hostnames;
        int use_host_decl_names;
        int use_lease_addr_for_default_route;
        int authoritative;
        int always_reply_rfc1048;
        int echo_client_id;

        struct tree_cache *options[256];
};

/* A dhcp host declaration structure. */
struct host_decl {
        struct host_decl *n_ipaddr;
        char *name;
        struct hardware interface;
        struct tree_cache *fixed_addr;
        struct group *group;
};

struct shared_network {
        struct shared_network *next;
        char *name;
        struct subnet *subnets;
        struct interface_info *interface;
        struct lease *leases;
        struct lease *insertion_point;
        struct lease *last_lease;

        struct group *group;
};

struct subnet {
        struct subnet *next_subnet;
        struct subnet *next_sibling;
        struct shared_network *shared_network;
        struct interface_info *interface;
        struct iaddr interface_address;
        struct iaddr net;
        struct iaddr netmask;

        struct group *group;
};

struct class {
        char *name;

        struct group *group;
};

/* privsep message. fixed length for easy parsing */
struct pf_cmd {
        struct in_addr ip;
        u_int32_t type;
};

/* Information about each network interface. */

struct interface_info {
        struct interface_info *next;    /* Next interface in list... */
        struct shared_network *shared_network;
                                /* Networks connected to this interface. */
        struct hardware hw_address;     /* Its physical address. */
        struct in_addr primary_address; /* Primary interface address. */
        char name[IFNAMSIZ];            /* Its name... */
        int rfdesc;                     /* Its read file descriptor. */
        int wfdesc;                     /* Its write file descriptor, if
                                           different. */
        unsigned char *rbuf;            /* Read buffer, if required. */
        size_t rbuf_max;                /* Size of read buffer. */
        size_t rbuf_offset;             /* Current offset into buffer. */
        size_t rbuf_len;                /* Length of data in buffer. */

        struct ifreq *ifp;              /* Pointer to ifreq struct. */

        int noifmedia;
        int errors;
        int dead;
        u_int16_t       index;
        int is_udpsock;
        ssize_t (*send_packet)(struct interface_info *, struct dhcp_packet *,
            size_t, struct in_addr, struct sockaddr_in *, struct hardware *);
};

struct dhcpd_timeout {
        struct dhcpd_timeout *next;
        time_t when;
        void (*func)(void *);
        void *what;
};

struct protocol {
        struct protocol *next;
        int fd;
        void (*handler)(struct protocol *);
        void *local;
        int pfd; /* slot used in the pollfd array */
};

#define _PATH_DHCPD_CONF        "/etc/dhcpd.conf"
#define _PATH_DHCPD_DB          "/var/db/dhcpd.leases"
#define _PATH_DEV_PF            "/dev/pf"
#define DHCPD_LOG_FACILITY      LOG_DAEMON

#define MAX_TIME 0x7fffffff
#define MIN_TIME 0

/* External definitions... */

/* parse.c */
extern int warnings_occurred;
int     parse_warn(char *, ...) __attribute__ ((__format__ (__printf__, 1,
            2)));

/* options.c */
void     parse_options(struct packet *);
void     parse_option_buffer(struct packet *, unsigned char *, int);
int      cons_options(struct packet *, struct dhcp_packet *, int,
            struct tree_cache **, int, int, int, u_int8_t *, int);
void     do_packet(struct interface_info *, struct dhcp_packet *, int,
            unsigned int, struct iaddr, struct hardware *);

/* dhcpd.c */
extern time_t           cur_time;
extern struct group     root_group;

extern u_int16_t        server_port;
extern u_int16_t        client_port;

extern char             *path_dhcpd_conf;
extern char             *path_dhcpd_db;

int     main(int, char *[]);
void    lease_pinged(struct iaddr, u_int8_t *, int);
void    lease_ping_timeout(void *);
void    periodic_scan(void *);

/* conflex.c */
extern int       lexline, lexchar;
extern char     *token_line, *tlname;
extern int       eol_token;

void    new_parse(char *);
int     next_token(char **, FILE *);
int     peek_token(char **, FILE *);

/* confpars.c */
int      readconf(void);
void     read_leases(void);
int      parse_statement(FILE *, struct group *, int, struct host_decl *, int);
void     parse_allow_deny(FILE *, struct group *, int);
void     skip_to_semi(FILE *);
int      parse_boolean(FILE *);
int      parse_semi(FILE *);
int      parse_lbrace(FILE *);
void     parse_host_declaration(FILE *, struct group *);
char    *parse_host_name(FILE *);
void     parse_class_declaration(FILE *, struct group *, int);
void     parse_lease_time(FILE *, time_t *);
void     parse_shared_net_declaration(FILE *, struct group *);
void     parse_subnet_declaration(FILE *, struct shared_network *);
void     parse_group_declaration(FILE *, struct group *);
void     parse_hardware_param(FILE *, struct hardware *);
char    *parse_string(FILE *);

struct tree             *parse_ip_addr_or_hostname(FILE *, int);
struct tree_cache       *parse_fixed_addr_param(FILE *);
void                     parse_option_param(FILE *, struct group *);
struct lease            *parse_lease_declaration(FILE *);
void                     parse_address_range(FILE *, struct subnet *);
time_t                   parse_date(FILE *);
unsigned char           *parse_numeric_aggregate(FILE *, unsigned char *,
                            int *, int, int, int);
void                     convert_num(unsigned char *, char *, int, int);
struct tree             *parse_domain_and_comp(FILE *);

/* tree.c */
pair                     cons(caddr_t, pair);
struct tree_cache       *tree_cache(struct tree *);
struct tree             *tree_const(unsigned char *, int);
struct tree             *tree_concat(struct tree *, struct tree *);
struct tree             *tree_limit(struct tree *, int);
int                      tree_evaluate(struct tree_cache *);

/* dhcp.c */
extern int      outstanding_pings;

void dhcp(struct packet *, int);
void dhcpdiscover(struct packet *);
void dhcprequest(struct packet *);
void dhcprelease(struct packet *);
void dhcpdecline(struct packet *);
void dhcpinform(struct packet *);
void nak_lease(struct packet *, struct iaddr *cip);
void ack_lease(struct packet *, struct lease *, unsigned int, time_t);
void dhcp_reply(struct lease *);
struct lease *find_lease(struct packet *, struct shared_network *, int *);
struct lease *mockup_lease(struct packet *, struct shared_network *,
    struct host_decl *);

/* bootp.c */
void bootp(struct packet *);

/* memory.c */
void enter_host(struct host_decl *);
struct host_decl *find_hosts_by_haddr(int, unsigned char *, int);
struct host_decl *find_hosts_by_uid(unsigned char *, int);
struct subnet *find_host_for_network(struct host_decl **, struct iaddr *,
    struct shared_network *);
void new_address_range(struct iaddr, struct iaddr, struct subnet *, int);
extern struct subnet *find_grouped_subnet(struct shared_network *,
    struct iaddr);
extern struct subnet *find_subnet(struct iaddr);
void enter_shared_network(struct shared_network *);
int subnet_inner_than(struct subnet *, struct subnet *, int);
void enter_subnet(struct subnet *);
void enter_lease(struct lease *);
int supersede_lease(struct lease *, struct lease *, int);
void release_lease(struct lease *);
void abandon_lease(struct lease *, char *);
struct lease *find_lease_by_uid(unsigned char *, int);
struct lease *find_lease_by_hw_addr(unsigned char *, int);
struct lease *find_lease_by_ip_addr(struct iaddr);
void uid_hash_add(struct lease *);
void uid_hash_delete(struct lease *);
void hw_hash_add(struct lease *);
void hw_hash_delete(struct lease *);
struct class *add_class(int, char *);
struct class *find_class(int, unsigned char *, int);
struct group *clone_group(struct group *, char *);
void write_leases(void);

/* alloc.c */
struct tree_cache *new_tree_cache(char *);
struct lease_state *new_lease_state(char *);
void free_lease_state(struct lease_state *, char *);
void free_tree_cache(struct tree_cache *);

/* print.c */
char *print_hw_addr(int, int, unsigned char *);

/* bpf.c */
int if_register_bpf(struct interface_info *);
void if_register_send(struct interface_info *);
void if_register_receive(struct interface_info *);
ssize_t receive_packet(struct interface_info *, unsigned char *, size_t,
    struct sockaddr_in *, struct hardware *);

/* dispatch.c */
extern struct interface_info *interfaces;
extern struct protocol *protocols;
extern struct dhcpd_timeout *timeouts;
void discover_interfaces(void);
void dispatch(void);
int locate_network(struct packet *);
void got_one(struct protocol *);
void add_timeout(time_t, void (*)(void *), void *);
void cancel_timeout(void (*)(void *), void *);
void add_protocol (char *, int, void (*)(struct protocol *), void *);
void remove_protocol(struct protocol *);

/* hash.c */
struct hash_table *new_hash(void);
void add_hash(struct hash_table *, unsigned char *, int, unsigned char *);
void delete_hash_entry(struct hash_table *, unsigned char *, int);
unsigned char *hash_lookup(struct hash_table *, unsigned char *, int);

/* tables.c */
extern struct option dhcp_options[256];
extern unsigned char dhcp_option_default_priority_list[256];
extern char *hardware_types[256];
extern struct hash_table universe_hash;
extern struct universe dhcp_universe;
void initialize_universes(void);

/* convert.c */
u_int32_t getULong(unsigned char *);
u_int16_t getUShort(unsigned char *);
void putULong(unsigned char *, u_int32_t);
void putLong(unsigned char *, int32_t);
void putUShort(unsigned char *, unsigned int);
void putShort(unsigned char *, int);

/* inet.c */
struct iaddr subnet_number(struct iaddr, struct iaddr);
struct iaddr ip_addr(struct iaddr, struct iaddr, u_int32_t);
u_int32_t host_addr(struct iaddr, struct iaddr);
int addr_eq(struct iaddr, struct iaddr);
char *piaddr(struct iaddr);

/* db.c */
int write_lease(struct lease *);
int commit_leases(void);
void db_startup(void);
void new_lease_file(void);

/* packet.c */
void assemble_hw_header(struct interface_info *, unsigned char *,
    int *, struct hardware *);
void assemble_udp_ip_header(struct interface_info *, unsigned char *,
    int *, u_int32_t, u_int32_t, unsigned int, unsigned char *, int);
ssize_t decode_hw_header(unsigned char *, u_int32_t, struct hardware *);
ssize_t decode_udp_ip_header(unsigned char *, u_int32_t, struct sockaddr_in *,
    u_int16_t);
u_int32_t       checksum(unsigned char *, u_int32_t, u_int32_t);
u_int32_t       wrapsum(u_int32_t);

/* icmp.c */
void icmp_startup(int, void (*)(struct iaddr, u_int8_t *, int));
int icmp_echorequest(struct iaddr *);
void icmp_echoreply(struct protocol *);

/* pfutils.c */
__dead void pftable_handler(void);
void pf_change_table(int, int, struct in_addr, char *);
void pf_kill_state(int, struct in_addr);
size_t atomicio(ssize_t (*)(int, void *, size_t), int, void *, size_t);
#define vwrite (ssize_t (*)(int, void *, size_t))write
void pfmsg(char, struct lease *);

/* udpsock.c */
void udpsock_startup(struct in_addr);