Pregunta

Este programa típico de "ruta agregar IP gw predeterminada" me da un error:

SIOCADDRT falló:Ioctl inapropiado para el dispositivo

Hice ifconfig para verificar el nombre de la interfaz.¿Hay alguna manera de enumerar todos los ioctl apropiados para mi dispositivo?

int main(char** args) {
        int sockfd;
        struct rtentry route;
        struct sockaddr_in *addr;
        int err = 0;

        // create the socket
        if(sockfd = socket(AF_INET, SOCK_DGRAM, 0)<0){
                perror("socket");
                exit(1);
        }

        memset(&route, 0, sizeof(route));
        addr = (struct sockaddr_in*) &route.rt_gateway;
        addr->sin_family = AF_INET;
        addr->sin_addr.s_addr = inet_addr("192.168.10.1");
        addr = (struct sockaddr_in*) &route.rt_dst;
        addr->sin_family = AF_INET;
        addr->sin_addr.s_addr = INADDR_ANY;
        addr = (struct sockaddr_in*) &route.rt_genmask;
        addr->sin_family = AF_INET;
        addr->sin_addr.s_addr = INADDR_ANY;

        route.rt_dev = "eth0";

        route.rt_flags = RTF_UP | RTF_GATEWAY;
        route.rt_metric = 0;
        if ((err = ioctl(sockfd, SIOCADDRT, &route)) != 0) {
                perror("SIOCADDRT failed");
                exit(1);
        }
}

Nuevo error después de cambiar el código a:

 addr = (struct sockaddr_in*) &route.rt_gateway;
        addr->sin_family = AF_INET;
        addr->sin_addr.s_addr = inet_addr("192.168.10.1");
        addr = (struct sockaddr_in*) &route.rt_dst;
        addr->sin_family = AF_INET;
        addr->sin_addr.s_addr = inet_addr("192.168.10.3");
        addr = (struct sockaddr_in*) &route.rt_genmask;
        addr->sin_family = AF_INET;
        addr->sin_addr.s_addr = inet_addr("255.255.255.0");

salida del gdb:

37      if ((err = ioctl(sockfd, SIOCADDRT, &route)) != 0) {
(gdb) p route
$1 = {rt_pad1 = 0, rt_dst = {sa_family = 2, sa_data = "\000\000\300\250\n\003\000\000\000\000\000\000\000"}, 
  rt_gateway = {sa_family = 2, sa_data = "\000\000\300\250\n\001\000\000\000\000\000\000\000"}, 
  rt_genmask = {sa_family = 2, sa_data = "\000\000\377\377\377\000\000\000\000\000\000\000\000"}, 
  rt_flags = 3, rt_pad2 = 0, rt_pad3 = 0, rt_tos = 0 '\000', rt_class = 0 '\000', rt_pad4 = {0, 0, 0}, 
  rt_metric = 0, rt_dev = 0x4008af "em1", rt_mtu = 0, rt_window = 0, rt_irtt = 0}
(gdb) n
38          perror("SIOCADDRT failed");
(gdb) 
SIOCADDRT failed: Invalid argument
¿Fue útil?

Solución

Supongo que el error está aquí:

    // create the socket
    if(sockfd = socket(AF_INET, SOCK_DGRAM, 0)<0)

La expresion socket(AF_INET, SOCK_DGRAM, 0)<0 Se evalúa antes de la asignación.Esto sucede porque los operadores de asignación tienen menor prioridad que las llamadas a funciones y las comparaciones.Puedes leer más sobre esto en este artículo de Wikipedia:

Entonces lo que realmente tienes es:

    // create the socket
    if(sockfd = (socket(AF_INET, SOCK_DGRAM, 0)<0))

Eso se evalúa como:

    // create the socket
    if(sockfd = 0)

Entonces estás tratando de realizar un ioctl() en el descriptor de archivo no válido.

Envuelva la asignación entre paréntesis o saque la expresión de la condición, así:

    // create the socket
    int sockfd = socket(AF_INET, SOCK_DGRAM, 0);
    if(sockfd < 0)

Por cierto, GCC o Clang deberían detectarlo si pasas -Wall opción.

Licenciado bajo: CC-BY-SA con atribución
No afiliado a StackOverflow
scroll top