默认系统是禁止发送广播包的,需要设置SOL_SOCKET选项来打开广播包发送。这主要是为了防止一些程序并不是设计用来发送广播消息,因为用户的错误输入,而发送了广播消息。比如一个UDP 程序接受一个目标IP地址做为命令行参数,但用户却把这个地址写成了一个广播地址,这时候这个选项的作用就体现出来了,与其让这个UDP程序检验用户输入的是不是个广播地址不如让内核来检测,如果是一个广播地址,但SO_BROADCAST 选项却没有被设定,EACCES错误就会被返回。
下面的例子用于循环发送广播包:
#include <stdlib.h>
#include <stdio.h>
#include <errno.h>
#include <string.h>
#include <unistd.h>
#include <netdb.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <sys/types.h>
#include <arpa/inet.h>
#define SEND_SIZE (10)
int
main(int argc, char **argv)
{
int i, skfd, ret;
const int opt = 1;
struct sockaddr_in addr;
char buf[SEND_SIZE] = {0};
//创建数据报式套接字skfd
if((skfd = socket(AF_INET,SOCK_DGRAM,0)) < 0) {
perror("socket");
exit(EXIT_FAILURE);
}
/*
* 打开广播选项
*/
ret = setsockopt(skfd, SOL_SOCKET, SO_BROADCAST, (char*)&opt, sizeof(opt));
if (ret == -1) {
perror("setsockopt");
return -1;
}
for(i=0; i<SEND_SIZE; i++) {
buf[i] = i + '0';
}
addr.sin_family = AF_INET;
addr.sin_addr.s_addr = htonl(INADDR_BROADCAST);
addr.sin_port=htons(8888);
while (1) {
ret = sendto(skfd, buf, SEND_SIZE, 0, (struct sockaddr*)&addr, sizeof(struct sockaddr_in));
if(ret != SEND_SIZE) {
perror("sendto");
} else {
printf("send success!\n");
}
sleep(1);
}
close(skfd);
return 0;
}