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 <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

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章