ONVIF协议网络摄像机(IPC)客户端程序开发使用ONVIF框架代码(C++)生成静态库04-->Windows
Posted Mango酱
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了ONVIF协议网络摄像机(IPC)客户端程序开发使用ONVIF框架代码(C++)生成静态库04-->Windows相关的知识,希望对你有一定的参考价值。
一 前提
先说一下,为什么不像前面Linux生成动态库那样去开发Onvif。因为Onvif的源码是没有那些__declspec(dlleXPort)指令,所以当你导出dll时,你会发现没有xxx.lib文件产生,在windows下你就没办法隐式调用dll了,除非你显示调用dll,即在程序中使用LoadLibrary()一个一个将要用到的函数加载进来,那么就会非常麻烦。
所以我们在windows下选择生成静态库去开发是比较好的选择。
注:生成dll有两种方法,一种是使用__declspec(dlleXPort)指令;另一种是使用DEF文件,可能这种方法方便,但是本人还没尝试过,有兴趣的可以试着这个思路去实现。
这里还是要提一点,为啥onvif开发要生成库比较好,因为使用gsoap生成的onvif源码是非常大的,如果不生成库,那么你每次更改自己的代码,即使onvif的源码没改变,每次编译都要4-8分钟,所以是非常浪费时间的,所以需要生成库进行开发。除非你不使用gsoap,而是自己实现onvif的客户端,那么就不需要关心是否生成库。
好了,我们确定了在windows要使用静态库开发,那么就开始撸。
二 编写cmake文件
为什么要使用cmake文件进行编译onvif源码生成静态库呢,因为cmake跨平台,方便Linux和Windows,在Linux下它会直接生成.a文件,在Windows下,它会先生成VS的项目,然后再由用户打开该VS项目进行生成静态库。
- 1)首先创建一个CMake_Onvif根目录,用于存放相关内容。
- 2)然后将上一篇我们使用gsoap生成的onvif源码 拷贝到这里,即ONVIFAPI里面的内容。
- 3)因为onvif依赖openssl库,所以我们需要下载一个openssl。
注意,openssl的版本与onvif的版本可能不一定兼容,我的openssl版本是"OpenSSL 1.1.1h 22 Sep 2020",onvif版本是gsoap_2.8.109.zip,这两个是没问题的。
后面当我使用新版本的gsoap_2.8.122.zip,配合"OpenSSL 1.1.1h 22 Sep 2020",发现会报X509参数错误(反正一大堆错误),看了一下是本应传回调函数的,但是实际上传了一个void*类型的成员参数,所以需要注意版本问题,建议如果你想用最新版的onvif,那么应该去openssl官网下载最新版本的openssl。这样能减少错误。
查看openssl的方法:
-
1)若有执行文件,则执行:openssl version。
-
2)若找不到执行文件,则去头文件的opensslv.h中找:
-
4)先创建一个build目录。待会用来存放执行cmake文件产生的临时文件。
-
5)创建一个名字为CMakeLists.txt的文件。添加以下内容:
重点讲一下下面比较重要的内容。
set(BUILD_USE_64BITS on)是用于开启64位静态库的编译;
“/bigobj”选项是因为onvif源码很大,不加的话会导致编译报错。cmake不加这个参数,可以直接在VS的:”属性->C/C++/命令行”加上也行。
cmake_minimum_required(VERSION 2.6)
project(ONVIF)
#打印
message(STATUS "binary dir: " $PROJECT_BINARY_DIR)
message(STATUS "source dir: " $PROJECT_SOURCE_DIR)
if (CMAKE_HOST_WIN32)
set(WINDOWS 1)
elseif (CMAKE_HOST_APPLE)
set(MACOS 1)
elseif (CMAKE_HOST_UNIX)
set(LINUX 1)
endif ()
if(WINDOWS)
message( "WINDOWS Platform......" )
set(BUILD_USE_64BITS on)
add_definitions(-DWITH_OPENSSL -DWITH_NONAMESPACES)
set(CMAKE_CXX_FLAGS "$CMAKE_CXX_FLAGS /bigobj")
#在Windows下,这样添加CMake参数,会被添加到C/C++的+选项的命令行当中,导致添加动态库错误
#set(CMAKE_CXX_FLAGS_DEBUG "$ENVCXXFLAGS -O0 -g -ggdb -w -Wall -std=c++11 -fPIC -lssl -lcrypto")
elseif(LINUX)
message( "LINUX Platform......" )
set(CMAKE_CXX_FLAGS_DEBUG "$ENVCXXFLAGS -O0 -g -ggdb -w -Wall -std=c++11 -fPIC -lpthread -lssl -lcrypto -DWITH_OPENSSL -DWITH_NONAMESPACES ")
else()
message( "MACOS Platform......" )
endif()
# 查找当前目录下的所有源文件,并将名称保存到 DIR_SRCS 变量
aux_source_directory(./ONVIFAPI/ DIR_SRCS)
set(CMAKE_BUILD_TYPE Debug)
#OPENSSL,尽量不要写当前路径,否则可以会报找不到动态库。
include_directories($PROJECT_SOURCE_DIR/openssl/include)
link_directories($PROJECT_SOURCE_DIR/openssl/lib)
# 生成链接库
#add_library(Onvif_Client SHARED $DIR_SRCS)
add_library(Onvif_Client STATIC $DIR_SRCS)
三 执行cmake文件
首先因为cmake是依赖VS这个软件去编译的,所以我们需要提前下载VS,我的是VS2015。然后我们可以查看当前cmake版本支持的编译器,执行:
cmake -G
图中看到,支持VS2017及以下的版本,因为我的是VS2015,所以执行命令:
cd build
# ..代表CMakeLists所在目录
cmake -G "Visual Studio 14 2015 Win64" ..
注意,若不加-G选项,cmake很可恶的,因为即使你在CMakeLists文件中加上set(BUILD_USE_64BITS on)设置了64位,但是它还是给你生成32bit的VS程序,即使你在VS将Win32平台改成x64,它还是不行,它会报错:fatal error LNK1112: 模块计算机类型“x64”与目标计算机类型“X86”冲突。
所以在执行cmake时,必须加上-G选项,表示我们要生成的是Win64位的VS项目。
四 运行生成的VS项目
通过上面后,在build目录下会生成一个VS项目,我们打开后缀为sln的VS项目。
然后右击ALL_BUILD,选择配置管理器,把生成的那一列全部勾上,不然待会会有个小报错,虽然不影响静态库的生成,但是总是不好看嘛。
然后执行程序即可。
我们会在Debug(你选择Release的话就是Release目录)目录下看的对应的静态库生成。
五 使用静态库
既然静态库生成了,那么肯定就是使用它呗。测试代码,以获取IPC的能力集为例子:
- 1)新建Win32控制台项目,我的名字是TestOnvif,然后空项目即可。
- 2)添加main.cpp文件。
- 3)把相关库引入进来,用到的库就是上面编译生成的静态库,以及openssl库。那么将onvif静态库、openssl动态库的include、lib路径引入到VS项目中即可。openssl的dll我没放到生成的执行程序也能运行,那么这里就不管它了。
main.cpp的内容:
#include <iostream>
#include "soapH.h"
#include "soapStub.h"
#include "wsseapi.h"
#include "smdevp.h"
#include "wsaapi.h"
#include "mecevp.h"
int main()
printf("nihao\\n");
struct soap *soap = NULL; // soap环境变量
if (NULL == (soap = soap_new()))
printf("nihao111\\n");
return -1;
printf("nihao222\\n");
soap_set_namespaces(soap, namespaces); // 设置soap的namespaces
soap->recv_timeout = 3; // 设置超时(超过指定时间没有数据就退出)
soap->send_timeout = 3;
soap->connect_timeout = 3;
#if defined(__linux__) || defined(__linux) // 参考https://www.genivia.com/dev.html#client-c的修改:
soap->socket_flags = MSG_NOSIGNAL; // To prevent connection reset errors
#endif
soap_set_mode(soap, SOAP_C_UTFSTRING); // 设置为UTF-8编码,否则叠加中文OSD会乱码
int result;
struct _tds__GetCapabilities req;
struct _tds__GetCapabilitiesResponse rep;
result = soap_call___tds__GetCapabilities(soap, "http://192.168.1.186/onvif/device_service", NULL, &req, rep);// PRE_AUTH
if (SOAP_OK != result)
const char **s = NULL;
return -1;
if (rep.Capabilities->Media != NULL)
printf("-------------------Media-------------------\\n");
printf("XAddr:%s\\n", rep.Capabilities->Media->XAddr.c_str());
printf("-------------------streaming-------------------\\n");
printf("RTPMulticast:%s\\n", (rep.Capabilities->Media->StreamingCapabilities->RTPMulticast) ? "Y" : "N");
printf("RTP_TCP:%s\\n", (rep.Capabilities->Media->StreamingCapabilities->RTP_USCORETCP) ? "Y" : "N");
printf("RTP_RTSP_TCP:%s\\n", (rep.Capabilities->Media->StreamingCapabilities->RTP_USCORERTSP_USCORETCP) ? "Y" : "N");
if (rep.Capabilities->PTZ != NULL)
printf("-------------------PTZ-------------------\\n");
printf("XAddr:%s\\n", rep.Capabilities->PTZ->XAddr.c_str());
Sleep(3);
return 0;
然后看到,相关能力集是能拿到的,且程序编译就10秒左右,完全没问题,可以达到开发的目的,所以windows下使用静态库开发onvif是没问题的。
Linux下onvif客户端关于ipc摄像头的搜索
设备搜索:要访问一个IPC摄像头,或者说要调用IPC摄像头提供的WEB服务接口,就要先知道其IP地址,这就是设备发现的过程,或者叫设备搜索的过程。IPC摄像头用的是239.255.255.250(端口3702),所以设备搜索的原理是,只要在设备上服务器监听239.255.255.250的3702端口。ONVIF规范并没有自己定义服务设备发现框架,而是复用了已经很成熟的WS-Discovery标准,根据.wsdl的文件,用gsoap产生框架代码,调用其产生的函数接口去实现设备的搜索。
1、gsoap框架代码:https://blog.csdn.net/weixin_42432281/article/details/84818575
2、上一部如果完成,就直接略过,将安装的gsoap-2.8\\gsoap目录下的两个文件:stdsoap2.c、stdsoap2.h拷贝到你工作目录下
3、注释stdsoap2.c如下代码:不注释的话会在编译运行的时候产生log日志,最后会发现磁盘已满的现象。
/*
#ifdef SOAP_DEBUG
#ifdef TANDEM_NONSTOP
soap_set_test_logfile(soap, "TESTLOG");
soap_set_sent_logfile(soap, "SENTLOG");
soap_set_recv_logfile(soap, "RECVLOG");
#else
soap_set_test_logfile(soap, "TEST.log");
soap_set_sent_logfile(soap, "SENT.log");
soap_set_recv_logfile(soap, "RECV.log");
#endif
#endif
*/
和修改
if (/*s == r || *r || */n < -128 || n > 127)
4、将安装的gsoap2.8目录下的import目录,拷贝到生成.c、.h的工作的文件夹里,cp gsoap-2.8/gsoap/import ./ ,REAMOD.txt是我写的记录文档,不必在意,其他的文件都拷贝到这个目录下
5、设备搜索的代码:我是直接copy别人的代码,做了一下修改(https://blog.csdn.net/saloon_yuan/article/details/27524875)
#include <stdio.h> #include <stdlib.h> #include <string.h> #include "soapH.h" #include "stdsoap2.h" #include "soapStub.h" #include "wsdd.nsmap" //命名空间 static struct soap* ONVIF_Initsoap(struct SOAP_ENV__Header *header, const char *was_To, const char *was_Action, int timeout) { struct soap *soap = NULL; // soap环境变量 unsigned char macaddr[6]; char _HwId[1024]; unsigned int Flagrand; soap = soap_new(); if(soap == NULL) { printf("[%d]soap = NULL\\n", __LINE__); return NULL; } soap_set_namespaces(soap, namespaces); // 设置soap的namespaces,即设置命名空间 // 设置超时(超过指定时间没有数据就退出) if(timeout > 0) { soap->recv_timeout = timeout; soap->send_timeout = timeout; soap->connect_timeout = timeout; } else { //Maximum waittime : 20s soap->recv_timeout = 20; soap->send_timeout = 20; soap->connect_timeout = 20; } soap_default_SOAP_ENV__Header(soap, header); //Create SessionID randomly,生成uuid(windows下叫guid,linux下叫uuid),格式为urn:uuid:8-4-4-4-12,由系统随机产生 srand((int)time(0)); Flagrand = rand()%9000 + 8888; macaddr[0] = 0x1; macaddr[1] = 0x2; macaddr[2] = 0x3; macaddr[3] = 0x4; macaddr[4] = 0x5; macaddr[5] = 0x6; sprintf(_HwId, "urn:uuid:%ud68a-1dd2-11b2-a105-%02X%02X%02X%02X%02X%02X", Flagrand, macaddr[0], macaddr[1], macaddr[2],macaddr[3],macaddr[4],macaddr[5]); header->wsa__MessageID = (char *)malloc(100); memset(header->wsa__MessageID, 0, 100); strncpy(header->wsa__MessageID, _HwId, strlen(_HwId)); //wsa__MessageID存放的是uuid if(was_Action != NULL) { header->wsa__Action = (char*)malloc(1024); memset(header->wsa__Action, \'\\0\', 1024); strncpy(header->wsa__Action, was_Action, 1024); // } if(was_To != NULL) { header->wsa__To = (char *)malloc(1024); memset(header->wsa__To, \'\\0\', 1024); strncpy(header->wsa__To, was_To, 1024);//"urn:schemas-xmlsoap-org:ws:2005:04:discovery"; } soap->header = header; return soap; } //释放函数 void ONVIF_soap_delete(struct soap *soap) { soap_destroy(soap); // remove deserialized class instances (C++ only) soap_end(soap); // Clean up deserialized data (except class instances) and temporary data soap_free(soap); // Reset and deallocate the context created with soap_new or soap_copy } int ONVIF_ClientDiscovery() { int FoundDevNo = 0; int retval = SOAP_OK; wsdd__ProbeType req; // 用于发送Probe消息 struct __wsdd__ProbeMatches resp; // 用于接收Probe应答 wsdd__ScopesType sScope; struct SOAP_ENV__Header header; struct soap* soap; const char *was_To = "urn:schemas-xmlsoap-org:ws:2005:04:discovery"; const char *was_Action = "http://schemas.xmlsoap.org/ws/2005/04/discovery/Probe"; //IP Adress and PortNo, broadCast const char *soap_endpoint = "soap.udp://239.255.255.250:3702/"; //设备上服务器监听239.255.255.250的3702端口 //Create new soap object with info soap = ONVIF_Initsoap(&header, was_To, was_Action, 10); soap_default_SOAP_ENV__Header(soap, &header); soap->header = &header; soap_default_wsdd__ScopesType(soap, &sScope); // 设置寻找设备的范围 sScope.__item = NULL; soap_default_wsdd__ProbeType(soap, &req); // 设置寻找设备的类型 req.Scopes = &sScope; req.Types = NULL; //"dn:NetworkVideoTransmitter"; //sent the message broadcast and wait retval = soap_send___wsdd__Probe(soap, soap_endpoint, NULL, &req); // 向组播地址广播Probe消息 while(retval == SOAP_OK) { printf("**************1**************\\n"); retval = soap_recv___wsdd__ProbeMatches(soap, &resp); if(retval == SOAP_OK) { if(soap->error) { printf("[%d]:recv soap error :%d, %s, %s\\n", __LINE__, soap->error, *soap_faultcode(soap), *soap_faultstring(soap)); retval = soap->error; } else //we find a device { FoundDevNo++; if(resp.wsdd__ProbeMatches->ProbeMatch != NULL && resp.wsdd__ProbeMatches->ProbeMatch->XAddrs != NULL) { printf("***** No %d Devices Information *****\\n", FoundDevNo); printf("Device Service Address : %s\\r\\n", resp.wsdd__ProbeMatches->ProbeMatch->XAddrs); printf("Device EP Address : %s\\r\\n", resp.wsdd__ProbeMatches->ProbeMatch->wsa__EndpointReference.Address); printf("Device Type : %s\\r\\n", resp.wsdd__ProbeMatches->ProbeMatch->Types); printf("Device Metadata Version: %d\\r\\n", resp.wsdd__ProbeMatches->ProbeMatch->MetadataVersion); printf("[%d]*********************************\\n", __LINE__); } } } else if(soap->error) { if(FoundDevNo == 0) { printf("No Device found!\\n"); retval = soap->error; } else { printf("Search end! Find %d Device! \\n", FoundDevNo); retval = 0; } break; } } //释放函数 ONVIF_soap_delete(soap); return retval; } int main(int argc, char *argv[]) { if(ONVIF_ClientDiscovery() != 0) { printf("discover failed! \\n"); return -1; } return 0; }
6、在编译时如果出现:对‘namespaces’未定义的引用,那是你在程序中没有加 #include "wsdd.nsmap" ,这个头文件,加上即可。
7、编译,生成可执行文件test:gcc -o test search_test.c stdsoap2.c soapC.c soapClient.c -I import/
8、运行test: ./test
设备搜索已完成!
设备搜索的主要目的是获取他服务器的地址:http://172.168.0.216/onvif/device_service,为下一步获取能力做准备。
以上是关于ONVIF协议网络摄像机(IPC)客户端程序开发使用ONVIF框架代码(C++)生成静态库04-->Windows的主要内容,如果未能解决你的问题,请参考以下文章