linux网络编程十九:用SIGURG检测外带数据是否到达

本文介绍了一种在Linux环境中利用SIGURG信号通知应用程序有关外带数据到达的方法。通过具体的C语言代码示例,展示了如何设置信号处理器、配置socket选项以及接收和处理外带数据。
在linux环境下,内核通知应用程序外带数据到达主要有两种方法:
1. I/O复用技术

2. 使用SIGURG信号


这里使用SIGURG来实现:

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <errno.h>
#include <assert.h>
#include <fcntl.h>
#include <signal.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>

#define BUFFER_SIZE 1024

static int connfd;

void sig_urg(int sig);	//SIGURG信号处理
void addsig(int sig, void (*sig_handler)(int));

int main(int argc, char **argv)
{
	if (argc != 2) {
		fprintf(stderr, "Usge: %s port\n", argv[0]);
		return 1;
	}
	
	int port = atoi(argv[1]);
	int ret = 0;
	int error;
	
	struct sockaddr_in address;
	bzero(&address, sizeof(address));
	address.sin_family = AF_INET;
	address.sin_port = htons(port);
	address.sin_addr.s_addr = htonl(INADDR_ANY);
	
	int sockfd = socket(PF_INET, SOCK_STREAM, 0);
	if (sockfd == -1)
		return 1;
		
	printf("server start...\n");
	
	if ( (bind(sockfd, (struct sockaddr*)&address, sizeof(address)) == -1) ||
		(listen(sockfd, 5) == -1) ) {
		error = errno;
		while ( (close(sockfd) == -1) && errno == EINTR);
		errno = error;
		return 1;
	}
	
	struct sockaddr_in client;
	socklen_t client_addrlength = sizeof(client);
	
	connfd = accept(sockfd, (struct sockaddr*)&client, &client_addrlength);
	if (connfd < 0) {
		fprintf(stderr, "errno: %s\n", strerror(errno));
	}
	else {
		addsig(SIGURG, sig_urg);
		//使用SIGURG信号之前,必须设置socket的宿主进程或进程组
		fcntl(connfd, F_SETOWN, getpid());
		
		char buffer[BUFFER_SIZE];
		while (1) {
			memset(buffer, '\0', BUFFER_SIZE);
			ret = recv(connfd, buffer, BUFFER_SIZE-1, 0);
			if (ret <= 0)
				break;
				
			printf("got %d bytes of normal data '%s'\n", ret, buffer);
		}
		
		close(connfd);
	}
	
	close(sockfd);
	
	
	return 0;
}

void sig_urg(int sig)
{
	int save_errno = errno;
	char buffer[BUFFER_SIZE];
	memset(buffer, '\0', BUFFER_SIZE);
	
	//接收外带数据
	int ret = recv(connfd, buffer, BUFFER_SIZE-1, MSG_OOB);
	printf("got %d bytes of oob data '%s'\n", ret, buffer);
	
	errno = save_errno;
}

void addsig(int sig, void (*sig_handler)(int))
{
	struct sigaction sa;
	memset(&sa, '\0', sizeof(sa));
	
	sa.sa_handler = sig_handler;
	sa.sa_flags |= SA_RESTART;
	
	sigfillset(&sa.sa_mask);
	assert(sigaction(sig, &sa, NULL) != -1);
	
}


参考:《linux高性能服务器编程》






评论 1
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值