GMSSL2.x编译鸿蒙静态库和动态库及使用

一、编译环境准备

1.1 开发工具

DevEco-Studio下载

1.2 SDK下载

下载编译第三方库的SDK有两种方式,第一种方式从官方渠道根据电脑系统选择对应的SDK版本,第二种方式通过DevEco-Studio下载SDK。本文只介绍通过DevEco-Studio下载SDK的方式。

  • 安装SDK到本地
  • 根据SDK安装位置获取SDK
  • 将SDK复制到GMSSL的同级目录并更名为ohos-sdk

二、编译鸿蒙架构

第三方库编译鸿蒙架构有两种方式。一种是使用C/C++原生构建工具configure、makefile编译,一种是使用lycium框架快速交叉编译。本文只详细介绍通过configure、makefile编译的方式。

2.1 原生工具构建

  • 编写编译脚本

    #!/bin/bash
    
    # Define paths and environment variables
    # 获取脚本当前所在路径
    CURRENT_PATH="$( cd "$( dirname "${BASH_SOURCE[0]}" )" && pwd )"
    # 获取GmSSL源码路径
    GMSSL_DIR=$CURRENT_PATH/GmSSL2.5.4
    # 取脚本执行的第一个参数
    OHOS_TARGET_ABI=$1
    # 获取鸿蒙SDK路径
    export OHOS_SDK=$CURRENT_PATH/ohos-sdk
    
    # 编译64位架构
    if [ "$OHOS_TARGET_ABI" == "arm64-v8a" ]
    then
        export AS=${OHOS_SDK}/native/llvm/bin/llvm-as
        export CC=${OHOS_SDK}/native/llvm/bin/aarch64-unknown-linux-ohos-clang
        export CXX=${OHOS_SDK}/native/llvm/bin/aarch64-unknown-linux-ohos-clang++
        export LD=${OHOS_SDK}/native/llvm/bin/ld.lld
        export STRIP=${OHOS_SDK}/native/llvm/bin/llvm-strip
        export RANLIB=${OHOS_SDK}/native/llvm/bin/llvm-ranlib
        export OBJDUMP=${OHOS_SDK}/native/llvm/bin/llvm-objdump
        export OBJCOPY=${OHOS_SDK}/native/llvm/bin/llvm-objcopy
        export NM=${OHOS_SDK}/native/llvm/bin/llvm-nm
        export AR=${OHOS_SDK}/native/llvm/bin/llvm-ar
        export CFLAGS="-DOHOS_NDK -fPIC -D__MUSL__=1"
        export CXXFLAGS="-DOHOS_NDK -fPIC -D__MUSL__=1"
        
        outdir=arm64-v8a
        cd ${GMSSL_DIR}
        ./Configure linux-aarch64
        
    elif [ "$OHOS_TARGET_ABI" == "armeabi-v7a" ]
    then
    #编译32位架构
        export AS=${OHOS_SDK}/native/llvm/bin/llvm-as
        export CC=${OHOS_SDK}/native/llvm/bin/armv7-unknown-linux-ohos-clang
        export CXX=${OHOS_SDK}/native/llvm/bin/armv7-unknown-linux-ohos-clang++
        export LD=${OHOS_SDK}/native/llvm/bin/ld.lld
        export STRIP=${OHOS_SDK}/native/llvm/bin/llvm-strip
        export RANLIB=${OHOS_SDK}/native/llvm/bin/llvm-ranlib
        export OBJDUMP=${OHOS_SDK}/native/llvm/bin/llvm-objdump
        export OBJCOPY=${OHOS_SDK}/native/llvm/bin/llvm-objcopy
        export NM=${OHOS_SDK}/native/llvm/bin/llvm-nm
        export AR=${OHOS_SDK}/native/llvm/bin/llvm-ar
        export CFLAGS="-DOHOS_NDK -fPIC -D__MUSL__=1"
        export CXXFLAGS="-DOHOS_NDK -fPIC -D__MUSL__=1"
    
        outdir=armeabi-v7a
        cd ${GMSSL_DIR}
        ./Configure linux-generic32
        
    elif [ "$OHOS_TARGET_ABI" == "x86_64" ]
    then
    #编译32位架构
        export AS=${OHOS_SDK}/native/llvm/bin/llvm-as
        export CC=${OHOS_SDK}/native/llvm/bin/x86_64-unknown-linux-ohos-clang
        export CXX=${OHOS_SDK}/native/llvm/bin/x86_64-unknown-linux-ohos-clang++
        export LD=${OHOS_SDK}/native/llvm/bin/ld.lld
        export STRIP=${OHOS_SDK}/native/llvm/bin/llvm-strip
        export RANLIB=${OHOS_SDK}/native/llvm/bin/llvm-ranlib
        export OBJDUMP=${OHOS_SDK}/native/llvm/bin/llvm-objdump
        export OBJCOPY=${OHOS_SDK}/native/llvm/bin/llvm-objcopy
        export NM=${OHOS_SDK}/native/llvm/bin/llvm-nm
        export AR=${OHOS_SDK}/native/llvm/bin/llvm-ar
        export CFLAGS="-DOHOS_NDK -fPIC -D__MUSL__=1"
        export CXXFLAGS="-DOHOS_NDK -fPIC -D__MUSL__=1"
    
        outdir=x86_64
        # Navigate to OpenSSL directory
        ./Configure linux-x86_64
        
    else
        echo "Unsupported target ABI: $ANDROID_TARGET_ABI"
        exit 1
    fi
    
    make
    
    # Copy the outputs
    OUTPUT_INCLUDE=$CURRENT_PATH/ohos_libs/include
    OUTPUT_LIB=$CURRENT_PATH/ohos_libs/lib/${outdir}
    mkdir -p $OUTPUT_INCLUDE
    mkdir -p $OUTPUT_LIB
    cp -RL include/openssl $OUTPUT_INCLUDE
    cp libcrypto.so.1.1 $OUTPUT_LIB
    cp libcrypto.a $OUTPUT_LIB
    cp libssl.so.1.1 $OUTPUT_LIB
    cp libssl.a $OUTPUT_LIB
    

    将编写好的脚本文件命名为ohos.sh。文件之间的目录结构如下:

  • 执行脚本

    cd到HarmonyOS-SDK目录下,依次执行如下命令,分别编译64位、32位、x86_64的鸿蒙架构。

    ohos.sh  arm64-v8a
    
    ohos.sh  armeabi-v7a
    
    ohos.sh  x86_64
    
  • 编译问题

    解决方案:删除GMSSL2.5.4源码中对getcontext、makecontext、setcontext等函数的引用。

三、使用库文件

鸿蒙工程可以使用.a的静态库和.so的动态库,两种类型的库引用一种即可。

3.1 创建NAPI工程

NAPI是OpenHarmony系统中的原生模块扩展开发框架,提供JavaScript与C/C++模块之间相互调用的交互能力。

  • 添加二进制文件到工程

如果该三方库二进制文件为so文件,还需要将so文件拷贝到工程目录的entry/libs/${OHOS_ARCH}/目录下,如下图:

3.2 配置链接

添加二进制文件后需要在cpp目录的CMakeLists.txt文件中添加对应target_link_libraries才能被工程引用。

  • 配置静态库链接

    target_link_libraries(product PRIVATE ${CMAKE_CURRENT_SOURCE_DIR}/thirdparty/gmssl/${OHOS_ARCH}/lib/libcrypto.a)
    target_link_libraries(product PRIVATE ${CMAKE_CURRENT_SOURCE_DIR}/thirdparty/gmssl/${OHOS_ARCH}/lib/libssl.a)
    
  • 配置动态库链接

    target_link_libraries(product PRIVATE ${CMAKE_CURRENT_SOURCE_DIR}/thirdparty/gmssl/${OHOS_ARCH}/lib/libcrypto.so)
    target_link_libraries(product PRIVATE ${CMAKE_CURRENT_SOURCE_DIR}/thirdparty/gmssl/${OHOS_ARCH}/lib/libssl.so)
    
  • 配置头文件

    target_include_directories(product PRIVATE ${CMAKE_CURRENT_SOURCE_DIR}/thirdparty/gmssl/${OHOS_ARCH}/include)
    

四、编写NAPI接口

配置完三方库的链接和头文件路径后,可以根据各自的业务逻辑调用三方库对应的接口完成NAPI接口的编写,NAPI接口开发可以参照以下文档学习。

  1. 使用Node-API实现跨语言交互开发流程

  2. C/C++三方库使用

4.1 NAPI编写示例

NAPI接口编写的基本思路是将从JavaScript层传入的参数转成C/C++数据类型,然后调用第三方库接口执行业务逻辑,最后将执行的结果转成JavaScript数据类型返回给JavaScript层。

Xnip2024-07-22_15-43-46.jpg
  • 编写接口

    napi_init.cpp文件中编写NAPI接口。

第一步:获取JS层传入的参数转成C/C++数据类型并将转换的结果作为参数传入第三方库函数中。

static napi_value gm_post(napi_env env, napi_callback_info info){
    size_t argc = 4;
    napi_value args[4];
    // 获取JS参数
    napi_get_cb_info(env, info, &argc, args , nullptr, nullptr);
    char url[512] = {0},parameter[MAX_BUF] = {0}, filePath[1024] = {0};
    size_t url_len,parameter_len,filePath_len;
    double timeOut;
    // 获取请求地址 napi(JS)数据类型转c/c++数据类型
    napi_get_value_string_utf8(env, args[0], url, sizeof(url), &url_len);
    // 获取请求参数 napi(JS)数据类型转c/c++数据类型
    napi_get_value_string_utf8(env, args[1], parameter, sizeof(parameter), &parameter_len);
    // 获取超时时间
    napi_get_value_double(env, args[2], &timeOut);
    // 获取证书路径
    napi_get_value_string_utf8(env, args[3], filePath, sizeof(filePath), &filePath_len);
    // 调用gmssl库的post请求方法
    resp_t *resp = infosecPost(url, parameter, timeOut, NULL, filePath);
    // 格式转换后返回响应数据
    return transformRespToNapi(env, resp);
}

第二步:将第三方库返回的结果转成JavaScript数据类型

static napi_value transformRespToNapi(napi_env env,resp_t *resp){
    napi_value res_obj;
    // 1.创建对象
    napi_status status = napi_create_object(env, &res_obj);
    if (status != napi_ok) {
        napi_throw_error(env,"-1","创建响应数据对象异常");
        return nullptr;
    }
    // 2.创建data属性值
    napi_value res_data;
    status = napi_create_string_utf8(env, resp->data, NAPI_AUTO_LENGTH, &res_data);
    if (status != napi_ok) {
        napi_throw_error(env,"-2","创建属性值异常");
        return nullptr;
    }
    status = napi_set_named_property(env, res_obj, "data", res_data);
    if (status != napi_ok) {
        napi_throw_error(env,"-3","设置对象的属性异常");
        return nullptr;
    }
    // 3.创建code属性
    napi_value res_code;
    status = napi_create_double(env, resp->code, &res_code);
     if (status != napi_ok) {
        napi_throw_error(env,"-2","创建属性值异常");
        return nullptr;
    }
    status = napi_set_named_property(env, res_obj, "code", res_code);
    if (status != napi_ok) {
        napi_throw_error(env,"-3","设置对象的属性异常");
        return nullptr;
    }
    resp_free(resp);
    return res_obj;
}

第三步:ArkTS接口与C/C++接口绑定和映射

EXTERN_C_START
static napi_value Init(napi_env env, napi_value exports)
{
    napi_property_descriptor desc[] = {
        { "send_gm_post", nullptr, gm_post, nullptr, nullptr, nullptr, napi_default, nullptr }
    };
    napi_define_properties(env, exports, sizeof(desc) / sizeof(desc[0]), desc);
    return exports;
}
EXTERN_C_END
说明:send_gm_post是ArkTS的函数名,gm_post是C/C++函数名。
  • 声明ArkTS侧的方法

Index.d.ts文件中声明ArkTS侧的方法。

export const send_gm_post: (url: string,parameters:string,timeOut:number,caPath:string) => object;
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 217,406评论 6 503
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,732评论 3 393
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 163,711评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,380评论 1 293
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,432评论 6 392
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,301评论 1 301
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,145评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,008评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,443评论 1 314
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,649评论 3 334
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,795评论 1 347
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,501评论 5 345
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,119评论 3 328
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,731评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,865评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,899评论 2 370
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,724评论 2 354