2022-02-06 20:05:29 -05:00
|
|
|
// Copyright 2010-2018 Nicholas J. Kain <njkain at gmail dot com>
|
|
|
|
// SPDX-License-Identifier: MIT
|
2011-07-04 20:07:16 -04:00
|
|
|
#ifndef ARP_H_
|
|
|
|
#define ARP_H_
|
2010-11-15 20:06:50 -05:00
|
|
|
|
2014-04-14 15:06:31 -04:00
|
|
|
#include <stdbool.h>
|
2010-12-01 23:33:25 -05:00
|
|
|
#include <stdint.h>
|
|
|
|
#include <net/if_arp.h>
|
2014-03-12 17:05:43 -04:00
|
|
|
#include "ndhc.h"
|
2011-07-02 03:51:44 -04:00
|
|
|
#include "dhcp.h"
|
2010-12-24 09:32:58 -05:00
|
|
|
|
2010-12-01 23:33:25 -05:00
|
|
|
struct arpMsg {
|
2011-07-02 06:31:57 -04:00
|
|
|
// Ethernet header
|
|
|
|
uint8_t h_dest[6]; // 00 destination ether addr
|
|
|
|
uint8_t h_source[6]; // 06 source ether addr
|
|
|
|
uint16_t h_proto; // 0c packet type ID field
|
2010-12-01 23:33:25 -05:00
|
|
|
|
2011-07-02 06:31:57 -04:00
|
|
|
// ARP packet
|
|
|
|
uint16_t htype; // 0e hardware type (must be ARPHRD_ETHER)
|
|
|
|
uint16_t ptype; // 10 protocol type (must be ETH_P_IP)
|
|
|
|
uint8_t hlen; // 12 hardware address length (must be 6)
|
|
|
|
uint8_t plen; // 13 protocol address length (must be 4)
|
|
|
|
uint16_t operation; // 14 ARP opcode
|
|
|
|
uint8_t smac[6]; // 16 sender's hardware address
|
|
|
|
uint8_t sip4[4]; // 1c sender's IP address
|
|
|
|
uint8_t dmac[6]; // 20 target's hardware address
|
|
|
|
uint8_t dip4[4]; // 26 target's IP address
|
|
|
|
uint8_t pad[18]; // 2a pad for min. ethernet payload (60 bytes)
|
2010-12-01 23:33:25 -05:00
|
|
|
};
|
|
|
|
|
2022-08-10 11:32:30 -04:00
|
|
|
extern unsigned arp_probe_wait;
|
|
|
|
extern unsigned arp_probe_num;
|
|
|
|
extern unsigned arp_probe_min;
|
|
|
|
extern unsigned arp_probe_max;
|
2011-07-05 13:03:55 -04:00
|
|
|
|
2015-02-15 06:38:03 -05:00
|
|
|
typedef enum {
|
|
|
|
AS_NONE = 0, // Nothing to react to wrt ARP
|
2017-02-24 06:27:15 -05:00
|
|
|
AS_ANNOUNCE, // Announcing our MAC/IP mapping to ethernet peers.
|
2015-02-15 06:38:03 -05:00
|
|
|
AS_COLLISION_CHECK, // Checking to see if another host has our IP before
|
|
|
|
// accepting a new lease.
|
2017-02-24 07:39:14 -05:00
|
|
|
AS_QUERY_GW_SEND, // Sending arp_ping to query the gateway MAC.
|
2015-02-15 06:38:03 -05:00
|
|
|
AS_GW_CHECK, // Seeing if the default GW still exists on the local
|
|
|
|
// segment after the hardware link was lost.
|
|
|
|
AS_GW_QUERY, // Finding the default GW MAC address.
|
|
|
|
AS_DEFENSE, // Defending our IP address (RFC5227)
|
|
|
|
AS_MAX,
|
|
|
|
} arp_state_t;
|
|
|
|
|
|
|
|
typedef enum {
|
|
|
|
ASEND_COLLISION_CHECK,
|
|
|
|
ASEND_GW_PING,
|
|
|
|
ASEND_ANNOUNCE,
|
|
|
|
ASEND_MAX,
|
|
|
|
} arp_send_t;
|
|
|
|
|
|
|
|
struct arp_stats {
|
|
|
|
long long ts;
|
2022-08-10 11:32:30 -04:00
|
|
|
unsigned count;
|
2015-02-15 06:38:03 -05:00
|
|
|
};
|
|
|
|
|
|
|
|
struct arp_data {
|
|
|
|
struct dhcpmsg dhcp_packet; // Used only for AS_COLLISION_CHECK
|
|
|
|
struct arpMsg reply;
|
|
|
|
struct arp_stats send_stats[ASEND_MAX];
|
|
|
|
long long wake_ts[AS_MAX];
|
|
|
|
long long last_conflict_ts; // TS of the last conflicting ARP seen.
|
|
|
|
long long arp_check_start_ts; // TS of when we started the
|
|
|
|
// AS_COLLISION_CHECK state.
|
|
|
|
unsigned int total_conflicts; // Total number of address conflicts on
|
|
|
|
// the interface. Never decreases.
|
2022-08-10 11:32:30 -04:00
|
|
|
unsigned gw_check_initpings; // Initial count of ASEND_GW_PING when
|
2015-02-15 06:38:03 -05:00
|
|
|
// AS_GW_CHECK was entered.
|
|
|
|
uint16_t probe_wait_time; // Time to wait for a COLLISION_CHECK reply
|
|
|
|
// (in ms?).
|
|
|
|
bool using_bpf:1; // Is a BPF installed on the ARP socket?
|
|
|
|
bool relentless_def:1; // Don't give up defense no matter what.
|
|
|
|
bool router_replied:1;
|
|
|
|
bool server_replied:1;
|
|
|
|
};
|
|
|
|
|
2022-01-11 22:35:19 -05:00
|
|
|
void arp_reset_state(struct client_state_t *cs);
|
2015-02-18 05:31:13 -05:00
|
|
|
|
2022-01-11 22:35:19 -05:00
|
|
|
bool arp_packet_get(struct client_state_t *cs);
|
2015-02-15 06:38:03 -05:00
|
|
|
|
2014-04-14 15:06:31 -04:00
|
|
|
void set_arp_relentless_def(bool v);
|
2022-01-11 22:35:19 -05:00
|
|
|
int arp_check(struct client_state_t *cs,
|
|
|
|
struct dhcpmsg *packet);
|
|
|
|
int arp_gw_check(struct client_state_t *cs);
|
|
|
|
int arp_set_defense_mode(struct client_state_t *cs);
|
|
|
|
int arp_gw_failed(struct client_state_t *cs);
|
|
|
|
|
|
|
|
int arp_do_collision_check(struct client_state_t *cs);
|
|
|
|
int arp_collision_timeout(struct client_state_t *cs, long long nowts);
|
|
|
|
|
|
|
|
int arp_query_gateway(struct client_state_t *cs);
|
|
|
|
int arp_query_gateway_timeout(struct client_state_t *cs, long long nowts);
|
|
|
|
|
|
|
|
int arp_announce(struct client_state_t *cs);
|
|
|
|
int arp_announce_timeout(struct client_state_t *cs, long long nowts);
|
|
|
|
|
|
|
|
int arp_do_defense(struct client_state_t *cs);
|
|
|
|
int arp_defense_timeout(struct client_state_t *cs, long long nowts);
|
|
|
|
int arp_do_gw_query(struct client_state_t *cs);
|
|
|
|
int arp_gw_query_timeout(struct client_state_t *cs, long long nowts);
|
|
|
|
int arp_do_gw_check(struct client_state_t *cs);
|
|
|
|
int arp_gw_check_timeout(struct client_state_t *cs, long long nowts);
|
2015-02-18 05:31:13 -05:00
|
|
|
|
|
|
|
// No action needs to be taken.
|
|
|
|
#define ARPR_OK 0
|
|
|
|
// There was no conflict with another host.
|
|
|
|
#define ARPR_FREE 1
|
|
|
|
// Another host already has our assigned address.
|
|
|
|
#define ARPR_CONFLICT -1
|
|
|
|
// The operation couldn't complete because of an error such as rfkill.
|
|
|
|
#define ARPR_FAIL -2
|
|
|
|
|
|
|
|
|
2014-03-17 22:10:58 -04:00
|
|
|
long long arp_get_wake_ts(void);
|
2010-11-15 20:06:50 -05:00
|
|
|
|
2022-02-06 20:05:29 -05:00
|
|
|
#endif
|
|
|
|
|