libssh2移植

近日公司要求支持SFTP,而curl组件默认并不支持,需要移植libssh2。
首先下载libssh2源码,从github上下载了最新的源码。

git clone [email protected]: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 
#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

你可能感兴趣的:(Cmake,C/C++,android,android,cmake,linux,ndk)