Ioctl возвращает 22 - недопустимый аргумент при компиляции с помощью компилятора clang для freebsd

Проблема: у меня проблема с образцом программы, чтобы получить IP-адрес, который работает нормально, когда я компилирую его с FreeBSD, встроенным в gcc. Но когда я использую кросс-компилятор clang версии 3.7.1, я получаю ошибку недопустимого аргумента при вызове ioctl. Вот пример программы -

    #include <stdio.h>
#include <unistd.h>
#include <string.h> /* for strncpy */

#include <sys/types.h>
#include <sys/socket.h>
#include <sys/ioctl.h>
#include <netinet/in.h>
#include <net/if.h>
#include <arpa/inet.h>
#include <errno.h>

int
main()
{
 int fd;
 struct ifreq ifr;

 fd = socket(AF_INET, SOCK_DGRAM, 0);

 /* I want to get an IPv4 IP address */
 ifr.ifr_addr.sa_family = AF_INET;

 /* I want IP address attached to "eth0"/em0 */
 strncpy(ifr.ifr_name, "em0", IFNAMSIZ-1);

 ioctl(fd, SIOCGIFADDR, &ifr);
printf("\n ioctl retrun : %d",errno);

 close(fd);

 /* display result */
 printf("\n%s\n", inet_ntoa(((struct sockaddr_in *)&ifr.ifr_addr)->sin_addr));

 return 0;
}

Вы никогда не должны проверять значение errno, если предыдущая библиотечная функция не вызвала возврат ошибки (-1 в случае ioctl) и задокументирована как установка errno при ошибке (что делает ioctl). Библиотечные функции не устанавливают errno в 0, поэтому ненулевое значение errno может остаться от предыдущего вызова.

rici 26.10.2018 05:53

Кроме того, strncpy не решает для вас никаких проблем во время этого звонка. Вы также можете использовать strcpy. Однако это не твоя проблема. Вы говорите, что используете clang как «кросс-компилятор». Что такое хостовая ОС и на какой платформе вы запускаете скомпилированный код? (На самом деле, у меня такие же сомнения по поводу вашего использования GCC. Не должно иметь значения, какой компилятор вы использовали для сборки FreeBSD, но, возможно, вы имели в виду не это.) Проблема, вероятно, связана с точными заголовочными файлами, которые вы используете с использованием...

rici 26.10.2018 06:09

В частности, если вы используете файлы заголовков для платформы A и запускаете скомпилированный код на платформе B, то вполне вероятно, что вызовы ioctl завершатся ошибкой, потому что разные системы имеют радикально разные номера селекторов IOCTL. Эти числа определены как макросы в файлах системных заголовков, поэтому важно использовать правильные заголовки для платформы, на которой будет запускаться программа.

rici 26.10.2018 06:22
2
3
389
0

Другие вопросы по теме