• Winsock协议目录


    SPI提供3种协议:

      1 分层协议

        在基础协议的上层,依靠底层基础协议实现更高级的通信服务。

      2 基础协议

        能够独立安全地,和远程断电实现数据通信的协议,相对于分层协议而言的

      3 协议连

        将一系列的基础协议,分层协议按特定的顺序连接在一起的链状结构。

    协议特性:

      winsock用WSAPROTOCOL_INFO结构描述特定协议的完整信息。称为一个winsock目录入口。

      两个重要标识:

        ProviderId:服务开发商提供的全局唯一标识

        dwCatalogEntryId为每个WSAPROTOCOL_INFO结构安排的唯一标识,称为目录入口ID。

    使用winsockapi函数枚举协议

      使用WSAEnumProtocols仅能枚举基础协议和协议链,不能枚举分层协议

    
    

    #include "..\common\initsock.h"
    #include <windows.h>
    #include <stdio.h>


    LPWSAPROTOCOL_INFO GetProvider(LPINT lpnTotalProtocols) { DWORD dwSize
    = 0; LPWSAPROTOCOL_INFO pProtoInfo = NULL; if(::WSAEnumProtocols(NULL,pProtoInfo,&dwSize) == SOCKET_ERROR) { if(::WSAGetLastError() != WSAENOBUFS) return NULL; } pProtoInfo = (LPWSAPROTOCOL_INFO)::GlobalAlloc(GPTR,dwSize); *lpnTotalProtocols = ::WSAEnumProtocols(NULL,pProtoInfo,&dwSize); return pProtoInfo; } void FreeProvider(LPWSAPROTOCOL_INFO pProtoInfo) { ::GlobalFree(pProtoInfo); } CInitSock theSock; void main() { int nTotalProtocols; LPWSAPROTOCOL_INFO pProtoInfo = GetProvider(&nTotalProtocols); if(pProtoInfo != NULL) { for(int i=0;i<nTotalProtocols;i++) { printf("Protocol:%s\n",pProtoInfo[i].szProtocol); printf("CatalogEntryId:%d ChainLen:%d\n\n", pProtoInfo[i].dwCatalogEntryId,pProtoInfo[i].ProtocolChain.ChainLen); } FreeProvider(pProtoInfo); } }

    执行结果:

    使用winsock SPI函数枚举协议

      winsock SPI提供的枚举协议的函数是WSCEnumProtocols,能够枚举各种协议,包括分层协议,基础协议,协议链。

      与WSAEnumProtocols相比,只多了一个 lpError的参数

    下面是WSCEnumProtocols代码:

    ///////////////////////////////////////////////////////////
    // WSCEnumProtocols.cpp文件
    
    
    #include <Ws2spi.h>        //  SPI函数定义在Ws2spi.h文件中
    #include <windows.h>
    #include <stdio.h>
    #pragma comment(lib, "WS2_32")    // 链接到WS2_32.lib
    
    LPWSAPROTOCOL_INFOW GetProvider(LPINT lpnTotalProtocols)
    {
        int nError;
        DWORD dwSize = 0;
        LPWSAPROTOCOL_INFOW pProtoInfo = NULL;
        
        //  取得需要的缓冲区长度
        if(::WSCEnumProtocols(NULL, pProtoInfo, &dwSize, &nError) == SOCKET_ERROR)
        {
            if(nError != WSAENOBUFS)
                return NULL;
        }
        // 申请缓冲区,再次调用WSCEnumProtocols函数
        pProtoInfo = (LPWSAPROTOCOL_INFOW)::GlobalAlloc(GPTR, dwSize);
        *lpnTotalProtocols = ::WSCEnumProtocols(NULL, pProtoInfo, &dwSize, &nError);
        return pProtoInfo;
    }
    
    void FreeProvider(LPWSAPROTOCOL_INFOW pProtoInfo)
    {
        ::GlobalFree(pProtoInfo);
    }
    
    void main()
    {
        LPWSAPROTOCOL_INFOW pProtoInfo;
        int nProtocols;
        pProtoInfo = GetProvider(&nProtocols);
    
        for(int i=0; i<nProtocols; i++)
        {
            
            printf(" Protocol: %ws \n", pProtoInfo[i].szProtocol);
            printf(" CatalogEntryId: %d        ChainLen: %d \n\n", 
                pProtoInfo[i].dwCatalogEntryId, pProtoInfo[i].ProtocolChain.ChainLen);
            
        }
    }

    执行结果:

  • 相关阅读:
    【转】使用外部看门狗请三思!
    【转】一招解决MCU启动异常
    【转】电源芯片选型,容易忽略的“QC”
    【转】为什么 MQTT 是最适合物联网的网络协议
    【转】中国芯酸往事
    函数输入参数类型为枚举时,则限定函数参数的接收范围,非枚举值输入则错误
    函数形参为指针与非指针的区别
    【转】函数中的形参问题(指针形参、引用形参、二重指针作为形参)
    【转】Example of using the --info linker option
    【转】STM32
  • 原文地址:https://www.cnblogs.com/xing901022/p/2736099.html
Copyright © 2020-2023  润新知