17.8. Операции с кэшем ARP
17.8. Операции с кэшем ARP
Операции с кэшем ARP также осуществляются с помощью функции ioctl. В этих запросах используется структура arpreq, показанная в листинге 17.9 и определяемая в заголовочном файле <net/if_arp.h>.
Листинг 17.9. Структура arpreq, используемая с вызовами ioctl для кэша ARP
struct arpreq {
struct sockaddr arp_pa; /* адрес протокола */
struct sockaddr arp_ha; /* аппаратный адрес */
int arp_flags; /* флаги */
};
#define ATF_INUSE 0x01 /* запись, которую нужно использовать */
#define ATF_COM 0x02 /* завершенная запись */
#define ATF_PERM 0x04 /* постоянная запись */
#define ATF_PUBL 0x08 /* опубликованная запись (отсылается другим узлам) */
Третий аргумент функции ioctl должен указывать на одну из этих структур. Поддерживаются следующие три вызова:
? SIOCSARP. Добавляет новую запись в кэш ARP или изменяет существующую запись. arp_pa — это структура адреса сокета Интернета, содержащая IP-адрес, a arp_ha — это общая структура адреса сокета с элементом ss_family, равным AF_UNSPEC, и элементом sa_data, содержащим аппаратный адрес (например, 6-байтовый адрес Ethernet). Два флага ATF_PERM и ATF_PUBL могут быть заданы приложением. Два других флага, ATF_INUSE и ATF_COM, устанавливаются ядром.
? SIOCDARP. Удаляет запись из кэша ARP. Вызывающий процесс задает интернет-адрес удаляемой записи.
? SIOCGARP. Получает запись из кэша ARP. Вызывающий процесс задает интернет-адрес, и соответствующий адрес Ethernet возвращается вместе с флагами.
Добавлять или удалять записи может только привилегированный пользователь. Эти три вызова обычно делает программа arp.
ПРИМЕЧАНИЕ
Запросы функции ioctl, связанные с ARP, не поддерживаются в некоторых более новых системах, использующих для описанных операций ARP маршрутизирующие сокеты.
Обратите внимание, что невозможно с помощью функции ioctl перечислить все записи кэша ARP. Большинство версий команды arp при использовании флага -a (перечисление всех записей кэша ARP) считывают память ядра (/dev/kmem), чтобы получить текущее содержимое кэша ARP. Мы увидим более простой (и предпочтительный) способ, основанный на применении функции sysctl, описанной в разделе 18.4.
Данный текст является ознакомительным фрагментом.