3

我正在尝试使用gethostbyname()带有交叉编译器的函数来编译程序arm-none-linux-gnueabi,但是当我在 android 上运行我的二进制文件时它不起作用。

我的代码在下面:

  /* gethostbyname-example.c */

  #include <stdio.h>
  #include <unistd.h>
  #include <stdlib.h>
  #include <string.h>
  #include <errno.h>
  #include <sys/socket.h>
  #include <netinet/in.h>
  #include <arpa/inet.h>
  #include <netdb.h>

  extern int h_errno;

  int main(int argc,char **argv) {
     int x, x2;
     struct hostent *hp;

     for ( x=1; x<argc; ++x ) {
        hp = gethostbyname(argv[x]);
        if ( !hp ) {
           fprintf(stderr,
                   "%s: host '%s'\n",
                   hstrerror(h_errno),
                   argv[x]);
           continue;
        }

        printf("Host %s : \n" ,argv[x]);
        printf(" Officially:\t%s\n", hp->h_name);
        fputs(" Aliases:\t",stdout);
        for ( x2=0; hp->h_aliases[x2]; ++x2 ) {
           if ( x2 ) {
              fputs(", ",stdout);
             }
        fputs(hp->h_aliases[x2],stdout);
        }     
        fputc('\n',stdout);
        printf(" Type:\t\t%s\n",
               hp->h_addrtype == AF_INET
               ? "AF_INET" : "AF_INET6");
        if ( hp->h_addrtype == AF_INET ) {
           for ( x2=0; hp->h_addr_list[x2]; ++x2 ) {
              printf(" Address:\t%s\n",
                     inet_ntoa( *(struct in_addr *)
                      hp->h_addr_list[x2]));
           }
        }
     putchar('\n');
     }
     return 0;
  }

arm-none-linux-gnueabi-gcc用操作系统ubuntu 12.04和交叉编译器在我的笔记本电脑上编译Sourcery Codebench

$ arm-none-linux-gnueabi-gcc gethostbyname-example.c --static -o gethostbyname-example
/tmp/ccE0xjBG.o: In function `main':
lookup.c:(.text+0x38): warning: Using 'gethostbyname' in statically linked applications requires at runtime the shared libraries from the glibc version used for linking
$ ls
gethostbyname-example.c gethostbyname-example
$ file gethostbyname-example
gethostbyname-example: ELF 32-bit LSB executable, ARM, version 1 (SYSV), statically linked, for GNU/Linux 2.6.16, not stripped

然后我gethostbyname-example在我的 chroot 手臂上测试二进制文件,我将二进制文件复制粘贴到chroot文件夹,然后我登录到chroot,然后我执行,顺便说一句,我从这里https://wiki.ubuntu.com/ARM/BuildEABIChroot创建 chroot ,如果我在 chroot 上运行它,得到如下结果:

# ./gethostbyname-example www.google.com
Host google.com : 
 Officially:    www.google.com
 Aliases:   
 Type:      AF_INET
 Address:   74.125.135.139
 Address:   74.125.135.100
 Address:   74.125.135.101
 Address:   74.125.135.102
 Address:   74.125.135.113
 Address:   74.125.135.138

然后,我还将使用 adb push 的二进制 gethostbyname-example 上传到我的 android 设备,如果我运行二进制 gethostbyname-example,则会收到如下错误:

# ./gethostbyname-example www.google.com
# Unknown server error: host 'www.google.com'
# ping www.google.com
# PING www.google.com (74.125.135.99) 56(84) bytes of data.
# 64 bytes from ni-in-f99.1e100.net (74.125.135.99): icmp_seq=1 ttl=49 time=798 ms
# 64 bytes from ni-in-f99.1e100.net (74.125.135.99): icmp_seq=2 ttl=49 time=1039 ms

在我的 android 设备中,我在 /etc/ 和 /system/etc/ 中有一个文件主机,带有谷歌 DNS

8.8.8.8 
8.8.4.4

大概是什么原因导致这个失败,

谢谢你。

4

2 回答 2

4

简短说明

glibc 中的 DNS 查找需要 /lib/libnss_dns.so.2 ,这在 Android 上不可用。

另一个区别是 Android 存储 DNS 设置 /system/etc/resolv.conf,使用仿生 C 库构建的原生 Android 应用程序将在那里查找要查询的 DNS 服务器。使用 glibc 构建的应用程序将/etc/resolv.conf在 Android 上不存在。

更长的解释

您已经使用 glibc 构建了二进制文件,而 Android 使用了仿生 C 库。

在静态链接到 libc 时,这在大多数情况下应该没问题,因为我不知道 android 内核中有足够严重的变化来破坏简单的用户空间应用程序。不过,您应该使用 Android NDK 来构建 android 本机应用程序,

然而,当静态链接到 glibc 时,会出现一些问题。glibc 中的某些功能,例如用于查找用户信息、主机名和其他通常在 /etc/nsswitch.conf 中配置的内容的功能调用其他共享库来实际完成工作。无论您是否静态链接到 glibc 本身,都会发生这种情况。这些文件通常是 glibc 系统上的 /lib/libnss_* 文件。

这些共享库中的大多数都是 glibc 的一部分,并且将安装在 glibc 系统上。但它们在 Android 上不可用。当这些共享库可用时,依赖于使用这些辅助共享库的函数的应用程序无法正常工作 - gethostbyname() 就是其中之一,对于正常的 DNS 查询,它需要 /lib/libnss_dns.so.2

于 2013-02-27T18:09:53.087 回答
0

有关解释,请参阅没有回复,解决方案请按照下列步骤操作:

  • 从这里下载 android NDK:https ://developer.android.com/ndk/downloads/index.html (我使用的是 linux 64bit)
  • 解压android-ndk-r13b-linux-x86_64.zip
  • 进入刚刚解压的目录
  • 启动 ./build/tools/make_standalone_toolchain.py 以查看可能的选项
  • 例如 ./build/tools/make_standalone_toolchain.py --arch arm --api 9
  • 该工具将为您构建 arm-linux-androideabi.tar.bz2
  • 用 tar xjvf arm-linux-androideabi.tar.bz2 解压
  • 它将创建 ./arm-linux-androideabi 目录
  • 现在您可以使用 ./arm-linux-androideabi/bin/arm-linux-androideabi-gcc yourprog.c -o yourprog -pie 进行编译
  • Android >= 5 需要 -pie
  • 您生成的可执行文件是本机的,解析器将按预期工作
于 2017-01-20T02:20:16.787 回答