windows下用UDP 广播在特定网卡上发包

背景

https://www.jianshu.com/p/a3692e86cb1e

总结

通过路由表来达到在指定网卡上发包的效果。

解决方案

  1. 更改路由表
    使用python的pysnmp库更方便
    这里给出的是windows自带的dos命令
route change 255.255.255.255 mask 255.255.255.255 192.168.1.101 metric 230 if 22

其中 192.168.1.101 是你本机的IP地址
metric 230 是你决定的metric值,比其他的小就行
if 22 是无线网络接口编号,也是通过route print命令看出来的


  1. 得到无线网卡信息之后,在特定网络广播(192.168.1.255)而不使用255.255.255.255
    main.c
#include<stdio.h>
#include<WinSock2.h>

#include "GetNICMsg.h"
void raise_message(char *msg);
char * changeIPaddrtoBroadcastIPaddr(char *ipaddr, char *netmask);

#pragma comment(lib,"ws2_32.lib")
#pragma warning(disable:4996)
#define RECV_PORT 2019
#define SEND_PORT 2019
#define MAXBUFLEN 1024
#define DONTPRINTIT 0
#define PRINTIT 1

#define FIRST_USING_WIRELESS_NIC 0

int main()
{
    WSADATA wsa;
    WSAStartup(0x201, &wsa);

    SOCKET sock_host;
    sock_host = socket(AF_INET, SOCK_DGRAM, 0);
    BOOL bBoardcast = TRUE;
    if (SOCKET_ERROR == setsockopt(sock_host, SOL_SOCKET, SO_BROADCAST, (char*)&bBoardcast, sizeof(bBoardcast)))
    {
        printf("setsockopt failed with error code: %d/n", WSAGetLastError());
        return -1;
    }

    char *ipaddr = NULL, *netmask = NULL,*ip_broadcast;
    
    if (getAllAdapterInfo(&ipaddr, &netmask, MIB_IF_TYPE_WiFi, DONTPRINTIT, FIRST_USING_WIRELESS_NIC)!= 0)
    {
        raise_message("找不到一个可用的无线网卡\n");
        closesocket(sock_host);
        return -1;
    }

        // 获得无线网卡的广播地址
    ip_broadcast = changeIPaddrtoBroadcastIPaddr(ipaddr, netmask);


    sockaddr_in addr_local, addr_remote;
    addr_remote.sin_addr.S_un.S_addr = inet_addr(ip_broadcast);
    addr_remote.sin_port = htons(SEND_PORT);
    addr_remote.sin_family = AF_INET;


    int retval, headlen = sizeof(addr_remote);
    char buf[MAXBUFLEN] = "hello???";
    
    
    
    int ID_list_temp[5] = { htons(1), htons(2), htons(4), htons(8), htons(16) };
    while (1)
    {
        for (int i = 0; i < 5; ++i)
        {
            retval = sendto(sock_host, (char *)(ID_list_temp + i), sizeof(int), 0, (sockaddr *)&addr_remote, headlen);
            if (retval < 0)
            {
                retval = WSAGetLastError();
                return -1;
            }
        }


    }

    return 0;

}

void raise_message(char *msg)
{
    printf(msg);
}

char * changeIPaddrtoBroadcastIPaddr(char *ipaddr,char *netmask)
{
    unsigned long int ipaddr_int = inet_addr(ipaddr);
    unsigned long int netmask_int = inet_addr(netmask);
    unsigned long int netmask_low = inet_addr("255.255.255.255");
    netmask_low = netmask_low^netmask_int;
    ipaddr_int = ipaddr_int | netmask_low;
    char *ipbroadcast;
    in_addr ipaddr_in;
    ipaddr_in.S_un.S_addr = ipaddr_int;
    ipbroadcast = inet_ntoa(ipaddr_in);
    return ipbroadcast;

}

GetNICMsg.cpp

#include "GetNICMsg.h"

void GetMsgfromcur(char** ipaddr, char **netmask,PIP_ADAPTER_INFO cur, bool printit)
{
    IP_ADDR_STRING *pIpAddrString = &(cur->IpAddressList);
    if (printit)
    {
        
        cout << "IP:" << pIpAddrString->IpAddress.String << endl;
        cout << "子网掩码:" << pIpAddrString->IpMask.String << endl;
        cout << "Context:" << pIpAddrString->Context << endl;

        char hex[16] = { '0', '1', '2', '3', '4', '5', '6', '7', '8', '9', 'A', 'B', 'C', 'D', 'E', 'F' };

        // mac 地址一般6个字节 
        // mac 二进制转16进制字符串
        char macStr[18] = { 0 };//12+5+1
        int k = 0;
        for (int j = 0; j < cur->AddressLength; j++){

            macStr[k++] = hex[(cur->Address[j] & 0xf0) >> 4];
            macStr[k++] = hex[cur->Address[j] & 0x0f];
            macStr[k++] = '-';
        }
        macStr[k - 1] = 0;

        cout << "MAC:" << macStr << endl; // mac地址 16进制字符串表示 
        cout << "--------------------------------------------------" << endl;

    }
    *ipaddr = (char *)malloc(sizeof(IP_ADDRESS_STRING));
    strcpy(*ipaddr, pIpAddrString->IpAddress.String);
    *netmask = (char *)malloc(sizeof(IP_ADDRESS_STRING));
    strcpy(*netmask, pIpAddrString->IpMask.String);
    

}

int getAllAdapterInfo(char** ipaddr, char **netmask, int NICtype, bool printit, int sequence)
{
    if (sequence == 0)
        sequence = 1;
    int CurrentSequence = 1;

    //char *ipaddr = (char *)malloc(sizeof(IP_ADDRESS_STRING));


    PIP_ADAPTER_INFO pIpAdapterInfo = new IP_ADAPTER_INFO[ADAPTERNUM];// 20个网卡空间 足够了 
    unsigned long stSize = sizeof(IP_ADAPTER_INFO) * ADAPTERNUM;
    // 获取所有网卡信息,参数二为输入输出参数 
    int nRel = GetAdaptersInfo(pIpAdapterInfo, &stSize);
    // 空间不足
    if (ERROR_BUFFER_OVERFLOW == nRel) {
        // 释放空间
        if (pIpAdapterInfo != NULL)
            delete[] pIpAdapterInfo;
        return -1;
    }

    PIP_ADAPTER_INFO cur = pIpAdapterInfo;
    // 多个网卡 通过链表形式链接起来的 
    while (cur){
        
        if (cur->Type == NICtype)
        {
            switch (cur->Type) {
            case MIB_IF_TYPE_OTHER:
                break;
            case MIB_IF_TYPE_ETHERNET:
                GetMsgfromcur(ipaddr, netmask,cur, printit);
                if (strcmp(UeslessIPaddr, *ipaddr) != 0)
                {
                    if (CurrentSequence == sequence)
                        return 0;
                    else
                        ++CurrentSequence;
                }

                break;
            case MIB_IF_TYPE_TOKENRING:
                break;
            case MIB_IF_TYPE_FDDI:
                break;
            case MIB_IF_TYPE_PPP:
                break;
            case MIB_IF_TYPE_LOOPBACK:
                break;
            case MIB_IF_TYPE_SLIP:
                break;
            case MIB_IF_TYPE_WiFi:
                GetMsgfromcur(ipaddr, netmask, cur, printit);
                
                if (strcmp(UeslessIPaddr, *ipaddr) != 0)
                {
                    if (CurrentSequence == sequence)
                        return 0;
                    else
                        ++CurrentSequence;
                }

                break;
            default://无线网卡,Unknown type

                break;
            }
        }


        cur = cur->Next;
        
    }

    // 释放空间
    if (pIpAdapterInfo != NULL)
        delete[] pIpAdapterInfo;
}

void getAllAdapterInfo()
{

    PIP_ADAPTER_INFO pIpAdapterInfo = new IP_ADAPTER_INFO[ADAPTERNUM];// 20个网卡空间 足够了 
    unsigned long stSize = sizeof(IP_ADAPTER_INFO) * ADAPTERNUM;
    // 获取所有网卡信息,参数二为输入输出参数 
    int nRel = GetAdaptersInfo(pIpAdapterInfo, &stSize);
    // 空间不足
    if (ERROR_BUFFER_OVERFLOW == nRel) {
        // 释放空间
        if (pIpAdapterInfo != NULL)
            delete[] pIpAdapterInfo;
        return;
    }

    PIP_ADAPTER_INFO cur = pIpAdapterInfo;
    // 多个网卡 通过链表形式链接起来的 
    while (cur){
        cout << "网卡描述:" << cur->Description << endl;
        switch (cur->Type) {
        case MIB_IF_TYPE_OTHER:
            break;
        case MIB_IF_TYPE_ETHERNET:
        {
            IP_ADDR_STRING *pIpAddrString = &(cur->IpAddressList);
            cout << "IP:" << pIpAddrString->IpAddress.String << endl;
            cout << "子网掩码:" << pIpAddrString->IpMask.String << endl;
            cout << "Context:" << pIpAddrString->Context << endl;
        }
            break;
        case MIB_IF_TYPE_TOKENRING:
            break;
        case MIB_IF_TYPE_FDDI:
            break;
        case MIB_IF_TYPE_PPP:
            break;
        case MIB_IF_TYPE_LOOPBACK:
            break;
        case MIB_IF_TYPE_SLIP:
            break;

        default://无线网卡,Unknown type
        {
            IP_ADDR_STRING *pIpAddrString = &(cur->IpAddressList);
            cout << "IP:" << pIpAddrString->IpAddress.String << endl;
            cout << "子网掩码:" << pIpAddrString->IpMask.String << endl;

        }
            break;
        }
        char hex[16] = { '0', '1', '2', '3', '4', '5', '6', '7', '8', '9', 'A', 'B', 'C', 'D', 'E', 'F' };

        // mac 地址一般6个字节 
        // mac 二进制转16进制字符串
        char macStr[18] = { 0 };//12+5+1
        int k = 0;
        for (int j = 0; j < cur->AddressLength; j++){

            macStr[k++] = hex[(cur->Address[j] & 0xf0) >> 4];
            macStr[k++] = hex[cur->Address[j] & 0x0f];
            macStr[k++] = '-';
        }
        macStr[k - 1] = 0;

        cout << "MAC:" << macStr << endl; // mac地址 16进制字符串表示 
        cur = cur->Next;
        cout << "--------------------------------------------------" << endl;
    }

    // 释放空间
    if (pIpAdapterInfo != NULL)
        delete[] pIpAdapterInfo;
}

GetNICMsg.h

#ifndef __GETNICMSG__
#define __GETNICMSG__

#include <iostream>
#include <vector>
#include <WinSock2.h>
#include <Iphlpapi.h>
#pragma comment(lib,"Iphlpapi.lib") 
#pragma comment(lib,"ws2_32.lib")
#pragma warning(disable:4996)
using namespace std;



#define ADAPTERNUM 20
#define MIB_IF_TYPE_WiFi 71
#define UeslessIPaddr "0.0.0.0"
int getAllAdapterInfo(char** ipaddr, char **netmask, int NICtype, bool printit, int sequence);
void getAllAdapterInfo();
void GetMsgfromcur(char** ipaddr, char **netmask, PIP_ADAPTER_INFO cur, bool printit);
#endif
  1. 通过setsockopt函数确定在哪个网卡上发送
    在windows下不可行,但是在Linux下可以通过setsockopt(..., SOL_SOCKET, SO_BINDTODEVICE, "ethX")完成。
    参考https://stackoverflow.com/questions/683624/udp-broadcast-on-all-interfaces
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,029评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,238评论 3 388
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,576评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,214评论 1 287
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,324评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,392评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,416评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,196评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,631评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,919评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,090评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,767评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,410评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,090评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,328评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,952评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,979评论 2 351

推荐阅读更多精彩内容