網絡子係統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