网络子系统77_套接字接收

//	accept系统调用
//	步骤:
//		1.由内核公共部分创建一个新套接字描述符,并分配其对应的文件描述符
//		2.新套接字使用accept之上的套接字相同的套接字类型和操作
//		3.交由具体协议完成accept
//		4.如果accept调用者要求返回peer地址,通过新套接字获取地址,并复制到用户地址空间
//		5.将新套接字的文件描述安装到调用者的进程控制块,返回新套接字的用户空间文件描述符
1.1 SYSCALL_DEFINE4(accept4, int, fd, struct sockaddr __user *, upeer_sockaddr,
		int __user *, upeer_addrlen, int, flags)
{
	struct socket *sock, *newsock;
	struct file *newfile;
	int err, len, newfd, fput_needed;
	struct sockaddr_storage address;

	//查找套接字描述符
	sock = sockfd_lookup_light(fd, &err, &fput_needed);
	if (!sock)
		goto out;

	//分配新套接字描述符
	newsock = sock_alloc();

	//新套接字与accept之上的套接字保持相同的套接字类型和操作
	newsock->type = sock->type;
	newsock->ops = sock->ops;

	//分配用户空间文件描述符
	newfd = get_unused_fd_flags(flags);
	//为新套接字创建文件描述符
	newfile = sock_alloc_file(newsock, flags, sock->sk->sk_prot_creator->name);
	//交由具体协议处理accept
	err = sock->ops->accept(sock, newsock, sock->file->f_flags);
	if (err < 0)
		goto out_fd;
	//获取peer地址
	if (upeer_sockaddr) {
		if (newsock->ops->getname(newsock, (struct sockaddr *)&address,
					  &len, 2) < 0) {
			err = -ECONNABORTED;
			goto out_fd;
		}
		//拷贝到用户地址空间
		err = move_addr_to_user(&address,
					len, upeer_sockaddr, upeer_addrlen);
		if (err < 0)
			goto out_fd;
	}
	//为新套接字安装文件描述符
	fd_install(newfd, newfile);
	//返回新套接字的文件描述符
	err = newfd;

out_put:
	fput_light(sock->file, fput_needed);
out:
	return err;
out_fd:
	fput(newfile);
	put_unused_fd(newfd);
	goto out_put;
}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值