ubuntu16.04上编译android的可执行文件并调用本地so库

前言:

  找了蛮多资料的,发现目前实现的编译方式大致就两种,一种是直接使用android源码中的编译工具链,另一种就是使用独立的交叉编译工具链,第二种我还在实现中,配置步骤挺多的

,第一种实现方式挺方便的,不想折腾的就用第一种就好了。

 

1.先下载并配置ndk环境(根据自己机器的来选择要下载哪个版本,我这里用的是ubuntu)

下载地址:https://developer.android.google.cn/ndk/downloads/index.html

下载完成后,解压,unzip 下载好的zip包,结构如下图

 

这里的ndk-build是编译的主角,接着准备下一步

2.准备好要编译的源码文件

#include <unistd.h>
#include <stdio.h>
#include <dlfcn.h>

typedef void (*pfnfoo)(void);

int main()
{
    void *handler = dlopen("/data/local/tmp/libTargetLibrary.so", RTLD_NOW);
    if(NULL == handler)
    {
        printf("[-] dlopen fail");
        return -1;
    }
    
    pfnfoo foo = (pfnfoo)dlsym(handler, "foo");
    
    foo();
    return 0;
}

上面是加载本地so库的源码,流程也很简单,dlopen函数是打开加载so文件的,dlsym函数是找到我们要调用的jni函数,返回一个函数指针,其实安卓加载so文件核心也是这两步,只是帮我们封装好了,

这里建议看看android记载so文件的源码分析:https://www.jb51.net/article/136838.htm, 讲的挺不错的,很多时候看源码才能了解问题的本质是什么,源码看完,发现最后加载so文件的代码其实和

上面是差不多的,dlopen打开so文件,dlsym找到jni函数来进行调用。

3. 把ndk的环境变量配置好,类似windows的环境变量配置

  因为我们最好是在工作目录下编译,如果命令行无法直接使用ndk-build命令会很麻烦,为了省事,无论在哪个目录都能使用ndk,所以配环境变量是必需的,这里vim打开/home/你的用户名

/.bashrc文件,将export PATH="$PATH:/home/你的用户名/android-ndk-r10(前面下载并解压的文件名)”,source .bashrc文件的路径,就配置完成了

4. 新建一个编译工作目录,将源码文件放入其中,并新建一个mk文件

在文件中写如下:

LOCAL_PATH := $(call my-dir)
include $(CLEAR_VARS)
LOCAL_MODULE := HelloAndroid(编译后的可执行文件名字)
LOCAL_SRC_FILES :=HelloAndroid.c (源码文件名)
LOCAL_ARM_MODE :=arm
LOCAL_CFLAGS+= -pie -fPIE
LOCAL_LDFLAGS+= -pie -fPIE
include $(BUILD_EXECUTABLE)

5. 打开命令行,指定下采用我们编写的mk文件

 

 然后发现目录中出现了两个新目录

 

 6.libs是不是就是我们在android开发或者逆向中,非常常见的目录了,不过这次不是so文件,而是android上的可执行文件,因为我

mk文件上配置是要可执行文件的,所以编译so文件的话,你懂的。libs下面根据你的手机机器架构进行选择,然后push到手机上

给个777的权限,然后把本地要调用的so库放到源码中指定的目录下,方便它查找。 

 

参考链接:

https://blog.csdn.net/u010651541/article/details/50177867

https://www.cnblogs.com/xieyajie/p/4727706.html 

posted @ 2021-01-17 00:00  YenKoc  阅读(718)  评论(0编辑  收藏  举报