近日公司要求支持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 <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