onvif开发之设备发现功能的实现--转

忙了一个多月,onvif总算告一段落了。这几个星期忙着其他的项目,也没有好好整理一下onvif的东西。接下来得好好整理一下自己的项目思路和项目经验,同时将自己的一些心得写出来,希望对人有所帮助。

        相信大多数兄弟和我一样,onvif开发,最开始做的就是发现功能。这两天登录onvif的官网看才发现,onvif版本在八月份有更新,已经更新到V2.4了,于是下载最新的版本来进行。代码的生成可以详见我的前一篇文章。V2.4版本新增了一个wsdl文件,现在用于生成源码的文件一共有18个。为了保证全功能,最好一次性生成包含所有功能的源码。然后根据最新生成的源码来实现onvif客户端和服务端的发现功能。

 

1. 创建onvif_test目录。以下这些源码由最新的gsoap(2.8.16)和最新的onvif的wsdl文件(2.4)生成。(截止2013.09.16)

onvif.h

soapClientLib.c

soapServerLib.c

soapC.c

soapClient.c

soapH.h

soapServer.c

soapStub.h

 

2.以下文件来自gsoap_2.8.16\gsoap-2.8\gsoap

stdsoap2.c

stdsoap2.h

 

3.以下文件来自gsoap_2.8.16\gsoap-2.8\gsoap\custom

duration.c

 

4.生成的soapClientLib.c和soapServerLib.c无实际作用,可直接删除。

 

5.增加自定义文件:

onvif_server.c   onvif服务端实现代码

onvif_client.c   onvif客户端实现代码

onvif_server_interface.c   onvif服务端接口实现

onvif_function.c  onvif实现函数,公用

onvif_function.h  onvif实现函数,公用

 

onvif_server.c为服务端的实现代码,主要是定义main函数,服务端主要是监听,并处理和应答消息。

Main函数定义如下:

[cpp] view plain copy

 

  1. int main(int argc,char ** argv)  
  2. {  
  3.     printf("[%s][%d][%s][%s] start \n", __FILE__, __LINE__, __TIME__, __func__);  
  4.   
  5.     int count = 0;  
  6.     struct soap ServerSoap;  
  7.     struct ip_mreq mcast;  
  8.       
  9.     soap_init1(&ServerSoap, SOAP_IO_UDP | SOAP_XML_IGNORENS);  
  10.     soap_set_namespaces(&ServerSoap,  namespaces);  
  11.       
  12.     printf("[%s][%d][%s][%s] ServerSoap.version = %d \n", __FILE__, __LINE__, __TIME__, __func__, ServerSoap.version);  
  13.       
  14.     if(!soap_valid_socket(soap_bind(&ServerSoap, NULL, ONVIF_LISTEN_PORT, 10)))  
  15.     {  
  16.         soap_print_fault(&ServerSoap, stderr);  
  17.         exit(1);  
  18.     }  
  19.       
  20.     mcast.imr_multiaddr.s_addr = inet_addr("239.255.255.250");  
  21.     mcast.imr_interface.s_addr = htonl(INADDR_ANY);  
  22.   
  23.     if(setsockopt(ServerSoap.master, IPPROTO_IP, IP_ADD_MEMBERSHIP, (char*)&mcast, sizeof(mcast)) < 0)  
  24.     {  
  25.             printf("setsockopt error! error code = %d,err string = %s\n",errno,strerror(errno));  
  26.         return 0;  
  27.     }  
  28.   
  29.     for(;;)  
  30.     {  
  31.         if(soap_serve(&ServerSoap))  
  32.         {  
  33.             soap_print_fault(&ServerSoap, stderr);  
  34.         }  
  35.   
  36.         soap_destroy(&ServerSoap);  
  37.         soap_end(&ServerSoap);  
  38.   
  39.         //客户端的IP地址  
  40.         printf("RECEIVE count %d, connection from IP = %lu.%lu.%lu.%lu socket = %d \r\n", count, ((ServerSoap.ip)>>24)&0xFF, ((ServerSoap.ip)>>16)&0xFF, ((ServerSoap.ip)>>8)&0xFF,(ServerSoap.ip)&0xFF, (ServerSoap.socket));  
  41.         count++;  
  42.     }  
  43.   
  44.     //分离运行时的环境  
  45.     soap_done(&ServerSoap);  
  46.   
  47.     return 0;  
  48. }  

onvif_server_interface.c   此文件用来定义所有服务端需要填充的接口。这里我们填充__wsdd__Probe接口即可,其他赞不支持的接口可以用宏统一处理。

__wsdd__Probe的填充如下:

[cpp] view plain copy

 

  1. SOAP_FMAC5 int SOAP_FMAC6  __wsdd__Probe(struct soap* soap, struct wsdd__ProbeType *wsdd__Probe)  
  2. {  
  3.   
  4.     #define MACH_ADDR_LENGTH 6  
  5.     #define INFO_LENGTH 512  
  6.     #define LARGE_INFO_LENGTH 1024  
  7.     #define SMALL_INFO_LENGTH 512  
  8.       
  9.     printf("[%d] __wsdd__Probe start !\n", __LINE__);  
  10.       
  11.     unsigned char macaddr[6] = {0};  
  12.     char _IPAddr[INFO_LENGTH] = {0};  
  13.     char _HwId[1024] = {0};  
  14.       
  15.     wsdd__ProbeMatchesType ProbeMatches;  
  16.     ProbeMatches.ProbeMatch = (struct wsdd__ProbeMatchType *)soap_malloc(soap, sizeof(struct wsdd__ProbeMatchType));  
  17.     ProbeMatches.ProbeMatch->XAddrs = (char *)soap_malloc(soap, sizeof(char) * INFO_LENGTH);  
  18.     ProbeMatches.ProbeMatch->Types = (char *)soap_malloc(soap, sizeof(char) * INFO_LENGTH);  
  19.     ProbeMatches.ProbeMatch->Scopes = (struct wsdd__ScopesType*)soap_malloc(soap,sizeof(struct wsdd__ScopesType));  
  20.     ProbeMatches.ProbeMatch->wsa__EndpointReference.ReferenceProperties = (struct wsa__ReferencePropertiesType*)soap_malloc(soap,sizeof(struct wsa__ReferencePropertiesType));  
  21.     ProbeMatches.ProbeMatch->wsa__EndpointReference.ReferenceParameters = (struct wsa__ReferenceParametersType*)soap_malloc(soap,sizeof(struct wsa__ReferenceParametersType));  
  22.     ProbeMatches.ProbeMatch->wsa__EndpointReference.ServiceName = (struct wsa__ServiceNameType*)soap_malloc(soap,sizeof(struct wsa__ServiceNameType));  
  23.     ProbeMatches.ProbeMatch->wsa__EndpointReference.PortType = (char **)soap_malloc(soap, sizeof(char *) * SMALL_INFO_LENGTH);  
  24.     ProbeMatches.ProbeMatch->wsa__EndpointReference.__any = (char **)soap_malloc(soap, sizeof(char*) * SMALL_INFO_LENGTH);  
  25.     ProbeMatches.ProbeMatch->wsa__EndpointReference.__anyAttribute = (char *)soap_malloc(soap, sizeof(char) * SMALL_INFO_LENGTH);  
  26.     ProbeMatches.ProbeMatch->wsa__EndpointReference.Address = (char *)soap_malloc(soap, sizeof(char) * INFO_LENGTH);  
  27.   
  28.     netGetMac("eth4", macaddr); //eth0  根据实际情况填充  
  29.     macaddr[0]=0x01;macaddr[1]=0x01;macaddr[2]=0x01;macaddr[3]=0x01;macaddr[4]=0x01;macaddr[5]=0x01;  
  30.     sprintf(_HwId,"urn:uuid:2419d68a-2dd2-21b2-a205-%02X%02X%02X%02X%02X%02X",macaddr[0], macaddr[1], macaddr[2], macaddr[3], macaddr[4], macaddr[5]);  
  31.   
  32.   
  33.     unsigned int localIp = 0;  
  34.     netGetIp("eth4", &localIp); //eth0 根据实际情况填充  
  35.     sprintf(_IPAddr, "http://%s/onvif/device_service", inet_ntoa(*((struct in_addr *)&localIp)));  
  36.     printf("[%d] _IPAddr ==== %s\n", __LINE__, _IPAddr);  
  37.       
  38.     ProbeMatches.__sizeProbeMatch = 1;  
  39.     ProbeMatches.ProbeMatch->Scopes->__item =(char *)soap_malloc(soap, 1024);  
  40.     memset(ProbeMatches.ProbeMatch->Scopes->__item,0,sizeof(ProbeMatches.ProbeMatch->Scopes->__item));    
  41.   
  42.     //Scopes MUST BE  
  43.     strcat(ProbeMatches.ProbeMatch->Scopes->__item, "onvif://www.onvif.org/type/NetworkVideoTransmitter");  
  44.   
  45.     ProbeMatches.ProbeMatch->Scopes->MatchBy = NULL;  
  46.     strcpy(ProbeMatches.ProbeMatch->XAddrs, _IPAddr);  
  47.     strcpy(ProbeMatches.ProbeMatch->Types, wsdd__Probe->Types);  
  48.     printf("wsdd__Probe->Types=%s\n",wsdd__Probe->Types);  
  49.     ProbeMatches.ProbeMatch->MetadataVersion = 1;  
  50.       
  51.     //ws-discovery规定 为可选项  
  52.     ProbeMatches.ProbeMatch->wsa__EndpointReference.ReferenceProperties->__size = 0;  
  53.     ProbeMatches.ProbeMatch->wsa__EndpointReference.ReferenceProperties->__any = NULL;  
  54.     ProbeMatches.ProbeMatch->wsa__EndpointReference.ReferenceParameters->__size = 0;  
  55.     ProbeMatches.ProbeMatch->wsa__EndpointReference.ReferenceParameters->__any = NULL;  
  56.       
  57.     ProbeMatches.ProbeMatch->wsa__EndpointReference.PortType[0] = (char *)soap_malloc(soap, sizeof(char) * SMALL_INFO_LENGTH);  
  58.     //ws-discovery规定 为可选项  
  59.     strcpy(ProbeMatches.ProbeMatch->wsa__EndpointReference.PortType[0], "ttl");  
  60.     ProbeMatches.ProbeMatch->wsa__EndpointReference.ServiceName->__item = NULL;  
  61.     ProbeMatches.ProbeMatch->wsa__EndpointReference.ServiceName->PortName = NULL;  
  62.     ProbeMatches.ProbeMatch->wsa__EndpointReference.ServiceName->__anyAttribute = NULL;  
  63.     ProbeMatches.ProbeMatch->wsa__EndpointReference.__any[0] = (char *)soap_malloc(soap, sizeof(char) * SMALL_INFO_LENGTH);  
  64.     strcpy(ProbeMatches.ProbeMatch->wsa__EndpointReference.__any[0], "Any");  
  65.     strcpy(ProbeMatches.ProbeMatch->wsa__EndpointReference.__anyAttribute, "Attribute");  
  66.     ProbeMatches.ProbeMatch->wsa__EndpointReference.__size = 0;  
  67.     strcpy(ProbeMatches.ProbeMatch->wsa__EndpointReference.Address, _HwId);  
  68.       
  69.     soap->header->wsa__To = "http://schemas.xmlsoap.org/ws/2004/08/addressing/role/anonymous";  
  70.     soap->header->wsa__Action = "http://schemas.xmlsoap.org/ws/2005/04/discovery/ProbeMatches";  
  71.     soap->header->wsa__RelatesTo = (struct wsa__Relationship*)soap_malloc(soap, sizeof(struct wsa__Relationship));  
  72.     soap->header->wsa__RelatesTo->__item = soap->header->wsa__MessageID;  
  73.     soap->header->wsa__RelatesTo->RelationshipType = NULL;  
  74.     soap->header->wsa__RelatesTo->__anyAttribute = NULL;  
  75.   
  76.     soap->header->wsa__MessageID =(char *)soap_malloc(soap, sizeof(char) * INFO_LENGTH);  
  77.     strcpy(soap->header->wsa__MessageID,_HwId+4);  
  78.   
  79.         if (SOAP_OK == soap_send___wsdd__ProbeMatches(soap, "http://", NULL, &ProbeMatches))  
  80.         {  
  81.         printf("send ProbeMatches success !\n");  
  82.         return SOAP_OK;  
  83.     }  
  84.   
  85.     printf("[%d] soap error: %d, %s, %s\n", __LINE__, soap->error, *soap_faultcode(soap), *soap_faultstring(soap));    
  86.       
  87.     return soap->error;;  
  88.   
  89. }  

onvif_client.c   onvif客户端实现代码,主要是定义客户端的main函数:

Main定义如下:

[cpp] view plain copy

 

  1. int main()    
  2. {    
  3.     printf("[%s][%d][%s][%s] start \n", __FILE__, __LINE__, __TIME__, __func__);  
  4.   
  5.     int result = 0;    
  6.     wsdd__ProbeType req;  
  7.     struct __wsdd__ProbeMatches resp;  
  8.     wsdd__ScopesType sScope;  
  9.     struct SOAP_ENV__Header header;    
  10.       
  11.     struct soap *soap;    
  12.     soap = soap_new();    
  13.     if(NULL == soap )    
  14.     {    
  15.         printf("sopa new error\r\n");    
  16.         return -1;    
  17.     }    
  18.   
  19.     soap->recv_timeout = 10;    
  20.     soap_set_namespaces(soap, namespaces);    
  21.     soap_default_SOAP_ENV__Header(soap, &header);    
  22.   
  23.     uuid_t uuid;  
  24.     char guid_string[100];  
  25.     uuid_generate(uuid);  
  26.     uuid_unparse(uuid, guid_string);  
  27.   
  28.     header.wsa__MessageID = guid_string;   
  29.     header.wsa__To = "urn:schemas-xmlsoap-org:ws:2005:04:discovery";    
  30.     header.wsa__Action = "http://schemas.xmlsoap.org/ws/2005/04/discovery/Probe";    
  31.     soap->header = &header;    
  32.       
  33.     soap_default_wsdd__ScopesType(soap, &sScope);    
  34.     sScope.__item = "";    
  35.     soap_default_wsdd__ProbeType(soap, &req);    
  36.     req.Scopes = &sScope;    
  37.     req.Types = ""; //"dn:NetworkVideoTransmitter";    
  38.   
  39.     int i = 0;        
  40.        result = soap_send___wsdd__Probe(soap, MULTICAST_ADDRESS, NULL, &req);    
  41.        while(result == SOAP_OK)    
  42.        {    
  43.         result = soap_recv___wsdd__ProbeMatches(soap, &resp);    
  44.         if(result == SOAP_OK)    
  45.         {    
  46.             if(soap->error)    
  47.             {    
  48.                 printf("soap error 1: %d, %s, %s\n", soap->error, *soap_faultcode(soap), *soap_faultstring(soap));    
  49.                 result = soap->error;    
  50.             }    
  51.             else    
  52.             {    
  53.                 printf("guog *********************************************\r\n");    
  54.                 if(soap->header->wsa__MessageID)    
  55.                 {    
  56.                     printf("MessageID   : %s\r\n", soap->header->wsa__MessageID);    
  57.                 }    
  58.                 if(soap->header->wsa__RelatesTo && soap->header->wsa__RelatesTo->__item)    
  59.                 {    
  60.                     printf("RelatesTo   : %s\r\n", soap->header->wsa__RelatesTo->__item);    
  61.                 }    
  62.                 if(soap->header->wsa__To)    
  63.                 {    
  64.                     printf("To          : %s\r\n", soap->header->wsa__To);    
  65.                 }    
  66.                 if(soap->header->wsa__Action)    
  67.                 {    
  68.                     printf("Action      : %s\r\n", soap->header->wsa__Action);    
  69.                 }    
  70.   
  71.                 for(i = 0; i < resp.wsdd__ProbeMatches->__sizeProbeMatch; i++)    
  72.                 {    
  73.                     printf("__sizeProbeMatch        : %d\r\n", resp.wsdd__ProbeMatches->__sizeProbeMatch);    
  74.                     printf("wsa__EndpointReference       : %p\r\n", resp.wsdd__ProbeMatches->ProbeMatch->wsa__EndpointReference);    
  75.                     printf("Target EP Address       : %s\r\n", resp.wsdd__ProbeMatches->ProbeMatch->wsa__EndpointReference.Address);    
  76.                     printf("Target Type             : %s\r\n", resp.wsdd__ProbeMatches->ProbeMatch->Types);    
  77.                     printf("Target Service Address  : %s\r\n", resp.wsdd__ProbeMatches->ProbeMatch->XAddrs);    
  78.                     printf("Target Metadata Version : %d\r\n", resp.wsdd__ProbeMatches->ProbeMatch->MetadataVersion);    
  79.                     if(resp.wsdd__ProbeMatches->ProbeMatch->Scopes)    
  80.                     {    
  81.                         printf("Target Scopes Address   : %s\r\n", resp.wsdd__ProbeMatches->ProbeMatch->Scopes->__item);    
  82.                     }    
  83.                 }  
  84.             }    
  85.         }    
  86.         else if (soap->error)    
  87.         {    
  88.             printf("[%d] soap error 2: %d, %s, %s\n", __LINE__, soap->error, *soap_faultcode(soap), *soap_faultstring(soap));    
  89.             result = soap->error;    
  90.         }    
  91.        }    
  92.   
  93.     soap_destroy(soap);   
  94.     soap_end(soap);   
  95.     soap_free(soap);  
  96.   
  97.     printf("[%d] guog discover over !\n", __LINE__);  
  98.       
  99.     return result;    
  100. }    

6.将wsdd.nsmap改为nsmap.h,并删除其余的*.nsmap(都一样)

 

7.编写makefile文件。注意在makefile中打开开关DEBUG,以便跟踪日志。

 

8.tcpdump为gcc环境的抓包工具,调试的时候用。

 

make编译通过,运行,客户端发现功能ok;

    但是服务端的发现功能却不行;别急,这是由于SOAP的版本问题;soap的版本是根据命名空间来自动确定的;在soap结构体的version字段表示soap版本;

以下命名空间表示SOAP1.1版本:

{"SOAP-ENV", "http://schemas.xmlsoap.org/soap/envelope/", "http://www.w3.org/*/soap-envelope", NULL},

{"SOAP-ENC", "http://schemas.xmlsoap.org/soap/encoding/", "http://www.w3.org/*/soap-encoding", NULL}, //1.1

 

以下命名空间表示SOAP1.2版本:

{"SOAP-ENV", "http://www.w3.org/2003/05/soap-envelope", "http://schemas.xmlsoap.org/soap/envelope/", NULL},

{"SOAP-ENC", "http://www.w3.org/2003/05/soap-encoding", "http://schemas.xmlsoap.org/soap/encoding/", NULL},  //1.2

    注意确定自己当前的命名空间。我们用SOAP1.2版本才能被测试工具发现(ONVIF Device Test Tool version 13.06)。

不清楚可以查看soap_set_namespaces接口和soap_set_local_namespaces接口;

于是将nsmap.h中的命名空间前两行改为:

{"SOAP-ENV", "http://www.w3.org/2003/05/soap-envelope", "http://schemas.xmlsoap.org/soap/envelope/", NULL},

{"SOAP-ENC", "http://www.w3.org/2003/05/soap-encoding", "http://schemas.xmlsoap.org/soap/encoding/", NULL},  //1.2

再编译运行,发现功能ok。

 

 

最后,将所用的工具和源码提供给大家,大家根据需要下载。

onvif测试工具V13.06:http://download.csdn.net/detail/u011597695/6288593

gsoap 2.8.16版本: http://download.csdn.net/detail/u011597695/6288615

onvif v2.4版本的wsdl文件:http://download.csdn.net/detail/u011597695/6288627

onvif v2.4版本的wsdl文件(适于离线生成源码):http://download.csdn.net/detail/u011597695/6288647

onvif源代码V2.4:http://download.csdn.net/detail/u011597695/6288663

 

from:http://blog.csdn.net/love_xjhu/article/details/11821037

时间: 2024-10-09 00:58:53

onvif开发之设备发现功能的实现--转的相关文章

ONVIF客户端搜索设备获取rtsp地址开发笔记(精华篇)

原文  http://blog.csdn.net/gubenpeiyuan/article/details/25618177   概要: 目前ONVIF协议家族设备已占据数字监控行业半壁江山以上,亲,作为开发者的你还在犹豫是否了解下吗?本文介绍了ONVIF客户端从设备搜索,鉴权,能力获取,媒体信息获取,URI地址获取的整套流程.文章只讲述了比较重要或其他博文没有讲述的开发点,详细可以参考文末参考文章.最后,能获得rtsp地址之后,然后去做其他功能比如录像,ptz这些就非常得心应手了.本文出自CS

Onvif开发之服务端成功对接Rtsp视频流篇

前面篇介绍onvif服务端的发现功能,继续在之前的代码基础上完成一个RTSP流的工作,也就是客户端通过ONVIF协议来预览设备端在这个之前必须确定几个简单的条件1 设备端能被发现2 设备端支持RTSP协议,并且能够通过VLC进行正常的预览 通过onvif协议设备需要做的几个基本事情:1  __tds__GetCapabilities获取设备能力2  __trt__GetProfiles获取设备的配置信息3  获取前段设备的视频编码和视频源的一些基本需要的信息4 __trt__GetStreamU

Onvif开发之基础介绍篇

什么是Onvif协议,谁开启了Onvif时代?                                                                 ONVIF:原意为 开放型网络视频接口论坛,即 Open Network Video Interface Forum ,是安讯士.博世.索尼等三家公司在2008年共同成立的一个国际性 开放型 网络视频产品 标准网络接口 的开发论坛,后来由这个技术开发论坛共同制定的开放性行业标准,就用该论坛的大写字母命名,即ONVIF 网络

onvif开发总结

ONVIF开发经验总结                     ONVIF开发经验总结....................................................................................................... 1 一.  利用gsoap2.8.14生成Onvif相关源代码................................................................ 2 1.   

Onvif开发之客户端鉴权获取参数篇

前面一篇已经介绍了客户端如何发些设备,下面这篇简单介绍下在发现设备后,如何通过ONVIF协议来获取设备的相关参数 ONVIF中不管是客户端还是设备端,最先实现的接口都是关于能力的那个接口,在客户端实现的函数名也就是[soap_call___tds__GetServiceCapabilities]通过获取的接口才知道设备具有那些能力,能够进行那些操作,服务端最基本的也需要实现这接口,让客户端知道设备支持那些基本操作.但是当设备端作了加密处理的话,即使你实现了这些接口,也不能正常获取到参数的,所以需

开发IoT设备必须考虑的四种挑战

物联网是当今科技行业的热门词汇.但它也是一个不可否认的现实.物联网已经引发第四次工业革命,不管我们喜欢还是不喜欢,它将不可避免地成为我们生活的一部分.物联网技术已经逐渐应用于越来越多的行业,而且越来越多的企业都试图在未来的物联网世界占居一席之地. 现在的问题是,很多企业只顾深入研究物联网开发,而忽视评估或了解摆在他们面前的关键挑战.许多这些公司甚至根本没有一点IT行业和软件开发的背景,并且他们中的大多数都专注于提供互联网连接设备,从而让他们面临同样的竞争.即使是那些拥有软件和硬件设计经验者,往往

onvif开发实战2--总结框架搭建

完成框架搭建后,编写自己的主函数起onvif服务 编写makefile objs = onvif.o onvif_func.o duration.o soapC.o soapServer.o stdsoap2.o onvif:$(objs)    gcc -o onvif $(objs) .PHONY:cleanclean:    #-rm onvif    rm *[!C.o].o   发现提示好多函数没有定义,在头文件soapStub.h中定义的,直接把没有定义的函数声明拷贝到一个onvif

入侵防御设备的功能仍未得到全面应用

入侵防御设备的功能仍未得到全面应用涉及的几个原因包括可靠性.吞吐率.流量延迟和误报率.基于网络的入侵防御系统(IPS)是一种嵌入式(in-line)设备,目的在于检测及阻止多种多样的攻击:不过 新的研究显示,这种设备的使用仍然常常更像是被动监控流量的入侵检测系统.Infonetics研究公司对169名负责为所在公司管理IPS的安全专业人士进行了调查,旨在查明IPS过滤器用于阻止攻击的全部功能是不是真正得到了使用:如果没有真正得到使用,查明其中的原因.IPS厂商TippingPoint委托这家研究

入侵防御设备的功能仍未得到全面应用涉及的原因

入侵防御设备的功能仍未得到全面应用涉及的几个原因包括可靠性.吞吐率.流量延迟和误报率.基于网络的入侵防御系统(IPS)是一种嵌入式(in-line)设备,目的在于检测及阻止多种多样的攻击:不过新的研究显示,这种设备的使用仍然常常更像是被动监控流量的入侵检测系统.Infonetics研究公司对169名负责为所在公司管理IPS的安全专业人士进行了调查,旨在查明IPS过滤器用于阻止攻击的全部功能是不是真正得到了使用:如果没有真正得到使用,查明其中的原因.IPS厂商TippingPoint委托这家研究公