#c #sockets #networking #ioctl
#c #сокеты #сеть #ioctl
Вопрос:
Я пытаюсь написать программу на языке Си для изменения IP-адреса и маски подсети сетевого интерфейса. Однако вызов ioctl
с помощью команды SIOCSIFADDR
всегда возвращается EINVAL
. Мой программный код выглядит следующим образом.
/* inet_config.c */
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <fcntl.h>
#include <errno.h>
#include <sys/types.h>
#include <sys/ioctl.h>
#include <sys/socket.h>
#include <netdb.h>
#include <net/if.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#define _INTERFACE_NAME argv[1]
#define _INET_ADDRESS argv[2]
#define _SUBNET_MASK argv[3]
int main(int argc, char ** argv) {
int sockfd, inet_addr_config_result, subnet_mask_config_result, ioctl_resu<
struct ifreq ifr;
struct sockaddr_in *inet_addr, *subnet_mask;
/* Prepare the struct ifreq */
bzero(ifr.ifr_name, IFNAMSIZ);
bzero((char *)amp;(ifr.ifr_addr), sizeof(struct sockaddr));
bzero((char *)amp;(ifr.ifr_netmask), sizeof(struct sockaddr));
strcpy(ifr.ifr_name, _INTERFACE_NAME);
ifr.ifr_addr.sa_family = AF_INET;
inet_addr = (struct sockaddr_in *)amp;(ifr.ifr_addr);
inet_addr_config_result = inet_pton(AF_INET, _INET_ADDRESS, amp;(inet_addr->sin_addr));
ifr.ifr_netmask.sa_family = AF_INET;
subnet_mask = (struct sockaddr_in *)amp;(ifr.ifr_netmask);
subnet_mask_config_result = inet_pton(AF_INET, _SUBNET_MASK, amp;(subnet_mask->sin_addr));
/* Error handling */
if(inet_addr_config_result == 0)
{
fprintf(stderr, "%s: inet_pton: Invalid IPv4 address.n", argv[0]);
exit(EXIT_FAILURE);
}
if(inet_addr_config_result < 0)
{
fprintf(stderr, "%s: inet_pton: ", argv[0]);
perror("");
exit(EXIT_FAILURE);
}
if(subnet_mask_config_result == 0)
{
fprintf(stderr, "%s: inet_pton: Invalid IPv4 subnet mask.n", argv[0]);
exit(EXIT_FAILURE);
}
if(subnet_mask_config_result < 0)
{
fprintf(stderr, "%s: inet_pton: ", argv[0]);
perror("");
exit(EXIT_FAILURE);
}
/* Open socket for ioctl calls */
sockfd = socket(AF_INET, SOCK_DGRAM, 0);
if(sockfd < 0)
{
fprintf(stderr, "%s: socket: ", argv[0]);
perror("");
exit(EXIT_FAILURE);
}
/* Call ioctl to configure network devices */
ioctl_result = ioctl(sockfd, SIOCSIFADDR, amp;ifr); // Set IP address
if(ioctl_result < 0)
{
fprintf(stderr, "%s: ioctl SIOCSIFADDR: ", argv[0]);
perror("");
exit(EXIT_FAILURE);
}
ioctl_result = ioctl(sockfd, SIOCSIFNETMASK, amp;ifr); // Set subnet mask
if(ioctl_result < 0)
{
fprintf(stderr, "%s: ioctl SIOCSIFNETMASK: ", argv[0]);
perror("");
exit(EXIT_FAILURE);
}
ioctl_result = ioctl(sockfd, SIOCGIFFLAGS, amp;ifr);
if(ioctl_result < 0)
{
fprintf(stderr, "%s: ioctl SIOCGIFFLAGS: ", argv[0]);
perror("");
exit(EXIT_FAILURE);
}
ifr.ifr_flags |= (IFF_UP | IFF_RUNNING);
ioctl(sockfd, SIOCSIFFLAGS, amp;ifr);
if(ioctl_result < 0)
{
fprintf(stderr, "%s: ioctl SIOCSIFFLAGS: ", argv[0]);
perror("");
exit(EXIT_FAILURE);
}
printf("Network device configuredn");
return 0;
}
При вводе команды
./inet_config wlp2s0 192.168.2.0 255.255.255.0
Я получаю следующее сообщение об ошибке:
./inet_config: ioctl SIOCSIFADDR: Invalid argument
( inet_config
это имя двоичного файла, скомпилированного из приведенного выше кода, и wlp2s0
это мой интерфейс WiFi.)
Комментарии:
1. Попробуйте вызвать свою программу с помощью strace:
strace ./inet_config wlp2s2 ...
Ответ №1:
Ваша проблема связана с ifr
переменной: она состоит из a union
, so ifr_addr
и ifr_netmask
указывает на одну и ту же зону памяти:
struct ifreq {
char ifr_name[IFNAMSIZ]; /* Interface name */
union { // <-- all your problem comes from this union
struct sockaddr ifr_addr;
struct sockaddr ifr_dstaddr;
struct sockaddr ifr_broadaddr;
struct sockaddr ifr_netmask;
struct sockaddr ifr_hwaddr;
short ifr_flags;
int ifr_ifindex;
int ifr_metric;
int ifr_mtu;
struct ifmap ifr_map;
char ifr_slave[IFNAMSIZ];
char ifr_newname[IFNAMSIZ];
char *ifr_data;
};
};
Ваш код может работать с некоторыми исправлениями: адрес и маска хранятся в двух sockaddr_in
структурах, которые копируются ifr
непосредственно перед его использованием.
/* inet_config.c */
[...]
int main(int argc, char ** argv) {
int sockfd, inet_addr_config_result, subnet_mask_config_result, ioctl_resu<
struct ifreq ifr;
/// note: no pointer here
struct sockaddr_in inet_addr, subnet_mask;
/* Prepare the struct ifreq */
bzero(ifr.ifr_name, IFNAMSIZ);
strcpy(ifr.ifr_name, _INTERFACE_NAME);
/// note: prepare the two struct sockaddr_in
inet_addr.sin_family = AF_INET;
inet_addr_config_result = inet_pton(AF_INET, _INET_ADDRESS, amp;(inet_addr.sin_addr));
subnet_mask.sin_family = AF_INET;
subnet_mask_config_result = inet_pton(AF_INET, _SUBNET_MASK, amp;(subnet_mask.sin_addr));
/* Error handling */
[...]
/* Open socket for ioctl calls */
sockfd = socket(AF_INET, SOCK_DGRAM, 0);
if(sockfd < 0)
{
fprintf(stderr, "%s: socket: ", argv[0]);
perror("");
exit(EXIT_FAILURE);
}
/* Call ioctl to configure network devices */
/// put addr in ifr structure
memcpy(amp;(ifr.ifr_addr), amp;inet_addr, sizeof (struct sockaddr));
ioctl_result = ioctl(sockfd, SIOCSIFADDR, amp;ifr); // Set IP address
if(ioctl_result < 0)
{
fprintf(stderr, "%s: ioctl SIOCSIFADDR: ", argv[0]);
perror("");
exit(EXIT_FAILURE);
}
/// put mask in ifr structure
memcpy(amp;(ifr.ifr_addr), amp;subnet_mask, sizeof (struct sockaddr));
ioctl_result = ioctl(sockfd, SIOCSIFNETMASK, amp;ifr); // Set subnet mask
if(ioctl_result < 0)
{
fprintf(stderr, "%s: ioctl SIOCSIFNETMASK: ", argv[0]);
perror("");
exit(EXIT_FAILURE);
}
[....]
printf("Network device configuredn");
return 0;
}