近日公司要求支持SFTP,而curl组件默认并不支持,需要移植libssh2。
首先下载libssh2源码,从github上下载了最新的源码。
git clone git@github.com:libssh2/libssh2.git
决定按github的方法,直接编译一下
mkdir build
cd build
cmake ..
make
cd -
编译成功,于是开始配置cmake,通过报错信息整理出下面命令,指定openssl的include文件夹位置,libcrypto和libssl库,我这里用的是静态库,使用动态库换成so。
cd build
rm -rf * //删除编译过的所有内容
cmake .. -DCMAKE_TOOLCHAIN_FILE=$ANDROID_NDK/build/cmake/android.toolchain.cmake \
-DANDROID_NDK=$ANDROID_NDK \
-DANDROID_ABI=armeabi-v7a \
-DOPENSSL_CRYPTO_LIBRARY=/home/admin/code/work/libcurl/libs/armeabi-v7a/libcrypto.a \
-DOPENSSL_SSL_LIBRARY=/home/admin/code/work/libcurl/libs/armeabi-v7a/libssl.a \
-DOPENSSL_INCLUDE_DIR=/home/admin/code/work/libcurl/include
make
额!编译报错了。
/home/admin/code/work/libssh2/src/openssl.c:658: error: undefined reference to 'ENGINE_load_builtin_engines'
/home/admin/code/work/libssh2/src/openssl.c:659: error: undefined reference to 'ENGINE_register_all_complete'
clang: error: linker command failed with exit code 1 (use -v to see invocation)
make[2]: *** [example/example-tcpip-forward] Error 1
make[1]: *** [example/CMakeFiles/example-tcpip-forward.dir/all] Error 2
make: *** [all] Error 2
检查openssl发现这个版本属于OPENSSL_NO_ENGINE,于是在CMakeLists.txt里增加下面命令,编译成功了。
ADD_DEFINITIONS("-DOPENSSL_NO_ENGINE=1")
然后android需要32位和64位的库,使用shell脚本让其能自动编译出32位和64位的库。
#!/bin/bash
ROOT_PATH=`pwd`
# openssl实际地址
OPENSSL_PATH=$ROOT_PATH/openssl
mkdir -p build
make_android() {
cd build
rm -rf *
cmake .. -DCMAKE_TOOLCHAIN_FILE=$ANDROID_NDK/build/cmake/android.toolchain.cmake \
-DANDROID_NDK=$ANDROID_NDK \
-DANDROID_ABI=$1 \
-DOPENSSL_CRYPTO_LIBRARY=$OPENSSL_PATH/libs/$1/libcrypto.a \
-DOPENSSL_SSL_LIBRARY=$OPENSSL_PATH/libs/$1/libssl.a \
-DOPENSSL_INCLUDE_DIR=$OPENSSL_PATH/include
make
cd -
}
make_android armeabi
make_android armeabi-v7a
make_android arm64-v8a
编译,还报错!!!
什么鬼,一看编译arm64-v8a时出错了,sys/time.h头文件没有导入,再次检查ndk在arm64-v8a时找不到__uint128_t变量导致,网上找了好像资料也未解决此问题,有的说要降ndk版本,很麻烦就放弃了。查看报错的源文件,发现在添加sys/time.h头文件是有宏判断的。果断将宏去掉试试,如下:
#if 1//def HAVE_SYS_TIME_H
# include <sys/time.h>
#endif
OK,大功告成!!
完成后又思考了一下,如果不只一个文件需要改头文件导入是不是让自己很被动,有没有更好办法呢?OPENSSL_NO_ENGINE在用x86编译并不需要打开,只要android才要打开,要怎么兼容呢?
另外cmake生成的库,执行文件很不好找,是不是可以归纳到一个文件夹下呢?带着这些疑问又修改了CMakeLists.txt和build.sh。最终如下:
CMakeLists.txt
if (DEFINED ANDROID_ABI)
ADD_DEFINITIONS("-DOPENSSL_NO_ENGINE=1")
if (${ANDROID_ABI} STREQUAL "arm64-v8a")
ADD_DEFINITIONS("-DHAVE_SYS_TIME_H")
endif()
endif()
build.sh
#!/bin/bash
ROOT_PATH=`pwd`
mkdir -p build
mkdir -p install
make_android() {
cd build
OPENSSL_PATH=$ROOT_PATH/openssl
rm -rf *
cmake .. -DCMAKE_TOOLCHAIN_FILE=$ANDROID_NDK/build/cmake/android.toolchain.cmake \
-DANDROID_NDK=$ANDROID_NDK \
-DANDROID_ABI=$1 \
-DCMAKE_BUILD_TYPE=Release \
-DOPENSSL_CRYPTO_LIBRARY=$OPENSSL_PATH/libs/$1/libcrypto.a \
-DOPENSSL_SSL_LIBRARY=$OPENSSL_PATH/libs/$1/libssl.a \
-DOPENSSL_INCLUDE_DIR=$OPENSSL_PATH/include \
-DCMAKE_LIBRARY_OUTPUT_DIRECTORY=$ROOT_PATH/install/lib/$1 \
-DCMAKE_ARCHIVE_OUTPUT_DIRECTORY=$ROOT_PATH/install/lib/$1 \
-DCMAKE_RUNTIME_OUTPUT_DIRECTORY=$ROOT_PATH/install/bin/$1
make
cd -
}
make_linux(){
cd build
rm -rf *
cmake .. -DCMAKE_LIBRARY_OUTPUT_DIRECTORY=$ROOT_PATH/install/lib \
-DCMAKE_ARCHIVE_OUTPUT_DIRECTORY=$ROOT_PATH/install/lib \
-DCMAKE_RUNTIME_OUTPUT_DIRECTORY=$ROOT_PATH/install/bin
make
cd -
}
make_android armeabi
make_android armeabi-v7a
make_android arm64-v8a
make_linux