#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;
};
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;
struct hardware *haddr;
struct shared_network *shared_network;
struct option_data options[256];
int got_requested_address;
};
struct hardware {
u_int8_t htype;
u_int8_t hlen;
u_int8_t haddr[16];
};
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;
int got_server_identifier;
struct shared_network *shared_network;
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
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];
};
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;
};
struct pf_cmd {
struct in_addr ip;
u_int32_t type;
};
struct interface_info {
struct interface_info *next;
struct shared_network *shared_network;
struct hardware hw_address;
struct in_addr primary_address;
char name[IFNAMSIZ];
int rfdesc;
int wfdesc;
unsigned char *rbuf;
size_t rbuf_max;
size_t rbuf_offset;
size_t rbuf_len;
struct ifreq *ifp;
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;
};
#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
extern int warnings_occurred;
int parse_warn(char *, ...) __attribute__ ((__format__ (__printf__, 1,
2)));
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 *);
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 *);
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 *);
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 *);
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 *);
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 *);
void bootp(struct packet *);
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);
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 *);
char *print_hw_addr(int, int, unsigned char *);
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 *);
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 *);
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);
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);
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);
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);
int write_lease(struct lease *);
int commit_leases(void);
void db_startup(void);
void new_lease_file(void);
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);
void icmp_startup(int, void (*)(struct iaddr, u_int8_t *, int));
int icmp_echorequest(struct iaddr *);
void icmp_echoreply(struct protocol *);
__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 *);
void udpsock_startup(struct in_addr);