網絡子係統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;
}
最後更新:2017-04-03 12:55:16