事實上很多我們平時遇到的問題都和這個端口選擇過程相關(guān),如果能深度理解這個過程,將有助于我們對這些問題的深刻理解。還是讓我們借助一段簡單到只有兩句的代碼,從這個來講起!
int?main(){
?fd?=?socket(AF_INET,SOCK_STREAM,?0);
?connect(fd,?...);
?...
}
一、創(chuàng)建 socket
客戶端在發(fā)起連接的時候,需要事先創(chuàng)建一個 socket。在 c 語言中,就是調(diào)用 socket 函數(shù),例如socket(AF_INET,SOCK_STREAM, 0)
這句。socket 函數(shù)執(zhí)行完畢后,在用戶層視角我們是看到返回了一個文件描述符 fd。但在內(nèi)核中其實是一套內(nèi)核對象組合,大體結(jié)構(gòu)如下。從上圖我們看到,socket 在內(nèi)核里并不是一個內(nèi)核對象。而是包含 file、socket、sock 等多個相關(guān)內(nèi)核對象構(gòu)成,每個內(nèi)核對象還定義了 ops 操作函數(shù)集合。在后面的內(nèi)核源碼執(zhí)行過程中,我們需要時不時回頭來看這些內(nèi)核對象,這里先簡單了解一下就行。這些內(nèi)核對象都是在 socket 系統(tǒng)調(diào)用執(zhí)行過程中創(chuàng)建出來的。為了避免喧賓奪主,這里只列出入口代碼,詳細過程就不展開介紹了。
//file:?net/socket.c
SYSCALL_DEFINE3(socket,?int,?family,?int,?type,?int,?protocol)
{
?//創(chuàng)建?socket、sock?等內(nèi)核對象,并初始化
?sock_create(family,?type,?protocol,?