一个简单广播数据包代码

在IPV4中报文的传输分为三种方式:单播,广播,多播。

以下是一个简单的Demo


/*
*filename: recv.c 
*compile&link: gcc -o recv recv.c
*/

#include 
< stdio.h >
#include 
< sys / types.h >
#include 
< sys / socket.h >
#include 
< netinet / in .h >


int  main( int  argc ,  char *  argv[] )
{
    
int fd;
    
int nread;
    
struct sockaddr_in addr;
    
char buf[128];

    fd 
= socket(PF_INET,SOCK_DGRAM,0);
    
if ( fd == -1 )
    
{
        perror(
"socket:");
        
return -1;
    }

        
    memset(
&addr,0,sizeof(struct sockaddr_in) );
    addr.sin_addr.s_addr 
= INADDR_ANY;
    addr.sin_port 
= htons(5000);
    addr.sin_family 
= AF_INET;

    bind(fd,(
struct sockaddr*)&addr,sizeof(struct sockaddr_in));

    
while ( 1 )
    
{
        memset(buf,
0,128);
        nread 
= recvfrom(fd,buf,128,0,NULL,NULL);
        
if ( nread == -1 )
        
{
            perror(
"recvfrom:");
            
break;
        }


        printf(
"received: %s",buf);
    }

    
    close(fd);
    
return 0;
}

 

 

 


/*
* filename : send.c
* compile&link : gcc -o send send.c
*/


#include 
< stdio.h >
#include 
< sys / types.h >
#include 
< sys / socket.h >
#include 
< netinet / in .h >   // 定义了INADDE_ANY,INADDR_BROADCAST等宏

int  enable_broadcast( int  fd, int  benable)
{
    
int nresult = setsockopt(fd,SOL_SOCKET,SO_BROADCAST,(const
void*)&benable,(socklen_t)sizeof(int));

    
if ( nresult == -1 )
    
{
        perror(
"setsockopt:");
        
return -1;
    }

    
return 1;
}


int  main(  int  argc ,  char *  argv[] )
{
    
int fd;
    
struct sockaddr_in addr;
    
char* msg = "hello ";

    fd 
= socket(PF_INET,SOCK_DGRAM,0);
    
if ( fd == -1 )
    
{
        perror(
"socket:");
        
return -1;
    }


    enable_broadcast(fd,
1);
    
    memset(
&addr,0,sizeof(struct sockaddr_in));
    addr.sin_addr.s_addr 
= INADDR_BROADCAST;
    addr.sin_family 
= AF_INET;
    addr.sin_port 
= htons(5000);
    
    
if ( -1 == sendto(fd,msg,strlen(msg),0,(struct sockaddr*)&addr,sizeof(struct sockaddr_in)) )
    
{
        perror(
"sendto:");
        
return -1;
    }

    
return 0;
}

 

运行:

#  ./recv&

#./send

 

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
要在Linux中使用C语言ping一个广播地址,你可以使用原始套接字(raw socket)来发送ICMP回显请求(ping)消息。下面是一个简单的示例代码: ```c #include <stdio.h> #include <stdlib.h> #include <string.h> #include <unistd.h> #include <sys/socket.h> #include <netinet/ip.h> #include <netinet/ip_icmp.h> #include <arpa/inet.h> #define PACKET_SIZE 64 #define PING_TIMEOUT 1 unsigned short checksum(void *b, int len) { unsigned short *buf = b; unsigned int sum = 0; unsigned short result; for (sum = 0; len > 1; len -= 2) sum += *buf++; if (len == 1) sum += *(unsigned char *) buf; sum = (sum >> 16) + (sum & 0xffff); sum += (sum >> 16); result = ~sum; return result; } int main() { int sockfd, ttl = 64; struct sockaddr_in addr; struct iphdr *ip; struct icmphdr *icmp; char packet[PACKET_SIZE]; // 创建原始套接字 sockfd = socket(AF_INET, SOCK_RAW, IPPROTO_ICMP); if (sockfd < 0) { perror("socket"); exit(1); } // 设置IP头部参数 memset(&addr, 0, sizeof(struct sockaddr_in)); addr.sin_family = AF_INET; addr.sin_addr.s_addr = inet_addr("255.255.255.255"); // 填充ICMP数据包 memset(packet, 0, sizeof(packet)); icmp = (struct icmphdr *) packet; icmp->type = ICMP_ECHO; icmp->code = 0; icmp->un.echo.sequence = rand(); icmp->un.echo.id = getpid(); icmp->checksum = 0; icmp->checksum = checksum(packet, sizeof(struct icmphdr)); // 设置IP头部参数 ip = (struct iphdr *) packet; ip->ihl = 5; ip->version = 4; ip->ttl = ttl; ip->protocol = IPPROTO_ICMP; ip->saddr = INADDR_ANY; ip->daddr = addr.sin_addr.s_addr; ip->tot_len = sizeof(struct iphdr) + sizeof(struct icmphdr); // 设置套接字选项,允许发送广播消息 int broadcastPermission = 1; if (setsockopt(sockfd, SOL_SOCKET, SO_BROADCAST, (void *) &broadcastPermission, sizeof(broadcastPermission)) < 0) { perror("setsockopt"); exit(1); } // 发送ICMP数据包 if (sendto(sockfd, packet, sizeof(packet), 0, (struct sockaddr *) &addr, sizeof(struct sockaddr_in)) < 0) { perror("sendto"); exit(1); } // 等待接收ICMP回复 struct timeval timeout; timeout.tv_sec = PING_TIMEOUT; timeout.tv_usec = 0; fd_set readfds; FD_ZERO(&readfds); FD_SET(sockfd, &readfds); int ready = select(sockfd + 1, &readfds, NULL, NULL, &timeout); if (ready == -1) { perror("select"); exit(1); } else if (ready == 0) { printf("Ping timeout\n"); exit(1); } else { printf("Ping received\n"); } close(sockfd); return 0; } ``` 注意:需要以root或具有特权的用户身份运行此代码,因为原始套接字需要特权访问。 这段代码使用原始套接字创建一个ICMP回显请求(ping)消息,并发送到广播地址(255.255.255.255)。然后,它等待接收ICMP回复。如果成功接收到回复,它将打印"Ping received",否则将打印"Ping timeout"。 请确保在使用此代码之前已经安装了libcap-dev库,并在编译时链接libcap库。例如,使用以下命令编译代码: ``` gcc ping_broadcast.c -o ping_broadcast -lcap ``` 请记住,使用原始套接字需要小心,因为它可以用于进行网络攻击。在实际应用中,请确保遵循适当的安全措施和法律法规。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值