千家信息网

SylixOS移植ICE库总结

发表于:2025-01-24 作者:千家信息网编辑
千家信息网最后更新 2025年01月24日,1 ICE 简介ICE(Internet Communications Engine)是一种面向对象的中间件平台,主要用于网络通讯。它为面向对象的"客户端-服务器"模型的应用提供了一组很好的工具和AP
千家信息网最后更新 2025年01月24日SylixOS移植ICE库总结

1 ICE 简介

ICE(Internet Communications Engine)是一种面向对象的中间件平台,主要用于网络通讯。它为面向对象的"客户端-服务器"模型的应用提供了一组很好的工具和API接口。目前在全世界被应用于很多项目之中。ICE中间件号称标准统一,开源,跨平台,跨语言,分布式,安全,服务透明,负载均衡,面向对象,性能优越,防火期穿透,通讯屏蔽。因此相比Corba,DCOM,SOAP,J2EE等的中间件技术,自然是集众多优点于一身,而却没有他们的缺点。


2 资源获取

在官网下载第三方中间件资源,ZeroC ICE的官方下载网站为:https://zeroc.com/downloads/ice#source。本文中使用的版本是V3.7,下载后解压文件,文件目录如图 2.1所示。

图2.1 Ice源码解压后的文件

cpp/、csharp/、java/、js/、objective-c/、php/、python/等目录包含了对应语言的Ice库实现代码,config/目录中存放控制编译方式的相关配置文件,slice/中存放了slice相关工具的代码实现。


3 Linux 平台编译

ICE支持多种语言,并且默认情况下会编译所有语言支持。这里只需要移植C++相关代码,需要修改config/Make.rules中的supported-languages属性,删除不需要的语言支持只保留C++支持,如图 3.1所示。

图 3.1 supported-languages属性修改

修改配置文件后直接在根目录make即可进入编译过程。为了方便后期的移植工作,这里使用V=s选项打印编译详细过程并保存到Make.log文件中,命令:# make V=s > Make.log。编译结束后查看Make.log内容确认编译成功。


4 移植工作

本次移植仅移植C++支持部分的ICE主库即libIce.so,因此只需要使用到cpp目录下的代码文件。由于是C++代码,在创建工程时需要选中C++工程选项。ICE主库代码编译需要依赖libbz2,也需要提前准备。


4.1 工程创建

在Real-Evo IDE中创建libIce的动态库工程。创建时选中C++工程选项并打开专家模式,如图 4.1所示。

图 4.1 创建工程选项

删除src目录下的libIce.c文件,导入源码包中的cpp/目录下与主库相关的代码文件。这里需要注意的是很多代码文件是编译过程中使用slice工具创建的,因此需要在linux环境下编译成功后才能导出所有的文件。其中包括目录cpp/include/generated/Ice/、cpp/include/Ice/、cpp/include/IceUtil/、cpp/src/Ice/、cpp/src/IceUtil/。最终的目录结构如图 4.2所示。

图 4.2 Ice代码目录结构


4.2 配置libIce.mk

分析Make.log中编译libIce.so的过程,罗列出所有需要编译的cpp文件,并逐一加入LOCAL_SRCS中,如图 4.3所示。

图 4.3 添加cpp文件

添加相关的头文件搜索目录,其中包括libbz2的工程路径,如图 4.4所示。

图 4.4 头文件搜索路径

添加宏配置,如图 4.5所示。

图 4.5 添加宏配置

添加编译选项,如图 4.6所示。

图 4.6 添加编译选项

添加依赖库,包括对libbz2库的依赖,如图 4.7所示。

图 4.7 添加依赖库

打开C++编译扩展,如图 4.8所示。

图 4.8 打开C++编译扩展


4.3 代码修改

默认编译会出现若干编译错误,需对部分代码进行修改,具体修改内容如下:

修改src/ice-master/cpp/src/Ice/Cond.cpp,使用CLOCK_REALTIME模式或是时间信息,如图 4.9所示。

图 4.9 修改获取时间方式

修改src/ice-master/cpp/include/IceUtil/Config.h,增加DSP大小端配置,如图 4.10所示。

图 4.10 DSP大小端配置

修改src/ice-master/cpp/include/Ice/Service.h,DSP中interrupt符号冲突,更改为interrupt2,如图 4.11所示。

图 4.11 interrupt符号冲突

修改src/ice-master/cpp/include/IceUtil/Config.h,消除DSP中GNUC版本错误,如图 4.12所示。

图 4.12消除GNUC版本错误

修改src/ice-master/cpp/src/Ice/DynamicLibrary.cpp,DSP中删除动态库相关操作,如图 4.13所示。

图 4.13 DSP中删除动态库相关操作

修改src/ice-master/cpp/src/Ice/Instance.cpp,增加头文件grp.h,如图 4.14所示。

图 4.14增加头文件grp.h

修改src/ice-master/cpp/src/Ice/MetricsAdminI.cpp,DSP中增加部分缺失接口,如图 4.15图 4.16图 4.17所示。

图 4.15 DSP中增加部分缺失接口

图 4.16 DSP中增加部分缺失接口

图 4.17 DSP中增加部分缺失接口

src/ice-master/cpp/src/Ice/Network.cpp,DSP中增加缺失的部分接口,如图 4.18图 4.19图 4.20图 4.21图 4.22所示。

图 4.18 DSP中增加缺失的部分接口

图4.19 DSP中增加缺失的部分接口

图 4.20 DSP中增加缺失的部分接口

图 4.21 DSP中增加缺失的部分接口

图 4.22 DSP中增加缺失的部分接口

src/ice-master/cpp/src/Ice/Service.cpp,删除fork调用相关代码,如图 4.23所示。

图 4.23删除fork调用相关代码

src/ice-master/cpp/src/Ice/Thread.cpp,纠正pthread_join返回值差异,如图 4.24所示。

图 4.24纠正pthread_join返回值差异

src/ice-master/cpp/src/IceUtil/RecMutex.cpp,纠正pthread_mutex_trylock返回值差异,如图 4.25所示。

图 4.25纠正pthread_mutex_trylock返回值差异

src/ice-master/cpp/src/IceUtil/UtilException.cpp,DSP中使用strdup代替接口__cxa_demangle,如图 4.26所示。

图 4.26 DSP中使用strdup代替接口__cxa_demangle


5 ICE 基本使用

ICE应用架构是服务器客户端模式,包括服务端和客户端两部分代码。客户端和服务器代码由slice工具所生成的接口代码联系到一起。下面使用ICE运行一个简单的helloworld测试例程。

5.1 生成接口代码

编写接口代码demo.ice,如程序清单 5.1所示。

程序清单 5.1 demo.ice

module demo{    interface printer    {        void printerstr(string msg);    };};

在linux中使用ICE源码中编译生成的slice工具生成接口代码:

# cpp/bin/slice2cpp demo.ice

可在当前目录中生成接口代码demo.cpp、demo.h。


5.2 创建测试工程

在RealEvo-IDE中创建应用工程,导入代码demo.cpp、demo.h,并创建代码文件server.cpp、client.cpp,内容如程序清单 5.2所示。

程序清单 5.2 server.cpp和client.cpp

client.cpp#include #include using namespace demo;using namespace std;intmain(int argc, char * argv[]){    int status = 0;    Ice::CommunicatorPtr ic;    try {        ic = Ice::initialize(argc, argv);        Ice::ObjectPrx base = ic->stringToProxy(                "SimplePrinter:default -p 10000");        printerPrx printer = printerPrx::checkedCast(base);        if (!printer)            throw "Invalid proxy";        printer->printerstr("Hello World!");    } catch (const Ice::Exception & ex) {        cerr << ex << endl;        status = 1;    } catch (const char * msg) {        cerr << msg << endl;        status = 1;    }    if (ic)        ic->destroy();    return status;}server.cpp#include #include using namespace demo;using namespace std;class PrinterI : public printer {    public:        virtual void printerstr(const string & s,                const Ice::Current &);};voidPrinterI::printerstr(const string & s, const Ice::Current &){    cout << s << endl;}intmain(int argc, char* argv[]){    int status = 0;    Ice::CommunicatorPtr ic;    try {        ic = Ice::initialize(argc, argv);        Ice::ObjectAdapterPtr adapter            = ic->createObjectAdapterWithEndpoints(                    "SimplePrinterAdapter", "default -p 10000");        Ice::ObjectPtr object = new PrinterI;        adapter->add(object,                ic->stringToIdentity("SimplePrinter"));        adapter->activate();        ic->waitForShutdown();    } catch (const Ice::Exception & e) {        cerr << e << endl;        status = 1;    } catch (const char * msg) {        cerr << msg << endl;        status = 1;    }    if (ic)        ic->destroy();    return status;}

应用程序同样使用专家模式,需创建server.mk和client.mk两份Makefile文件。分别加入对应的代码,server.mk中编译demo.cpp、server.cpp,client.mk中编译demo.cpp、client.cpp。并依赖ICE库。编译生成两份可执行文件,server和client。在SylixOS设备上先后执行server和client,若在执行client时server出现"Hello World"信息表示client成功调用server接口,运行结果正常。


0