Невозможно изменить IP-адрес с помощью ioctl SIOCSIFADDR

#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;
}