bind9運行主要是兩個線程和一個do while循環(huán),這三部分構(gòu)成了bind的核心代碼。
do {
result = isc_app_run();
if (result == ISC_R_RELOAD) {
ns_server_reloadwanted(ns_g_server);
} else if (result != ISC_R_SUCCESS) {
UNEXPECTED_ERROR(__FILE__, __LINE__,
"isc_app_run(): %s",
isc_result_totext(result));
/*
* Force exit.
*/
result = ISC_R_SUCCESS;
}
} while (result != ISC_R_SUCCESS);
這個do while循環(huán),主要是result = isc_app_run();這個函數(shù)實際上等待各種結(jié)束程序信號,如果需要重啟服務(wù),就調(diào)用ns_server_reloadwanted。
在setup函數(shù)里中的create_managers(void)里,有兩個函數(shù)要注意
isc_taskmgr_create、isc_socketmgr_create2這兩個函數(shù)分別起兩個線程,分別是run和watch。
下面是watch中的關(guān)鍵代碼就是
cc = epoll_wait(manager->epoll_fd, manager->events,
done = process_fds(manager, manager->events, cc)這兩句代碼一看就知道是處理epool的。它是在isc_socketmgr_create2中的
if (isc_thread_create(watcher, manager, &manager->watcher) !=ISC_R_SUCCESS)中起的線程;process_fds執(zhí)行到最后就是isc_task_send(ev->ev_sender, (isc_event_t **)&iev);這個函數(shù)執(zhí)行完就會轉(zhuǎn)到下面的run線程函數(shù)中。process_fd是具體的業(yè)務(wù)處理,process_ctlfd這個函數(shù)雖然不起眼但是很重要,在這個函數(shù)中有 wakeup_socket,里面最重要的就是result = watch_fd(manager, fd, msg);,在watch_fd中,關(guān)鍵代碼就是epoll_ctl(manager->epoll_fd, EPOLL_CTL_ADD, fd, &event),這里重新注冊時間,然后epool_wait再等待,而后再處理,如此反復(fù)。
result = isc_taskmgr_create(ns_g_mctx, ns_g_cpus, 0, &ns_g_taskmgr),#define isc_taskmgr_create isc__taskmgr_create,所以這里實際調(diào)用的是isc__taskmgr_create,這個函數(shù)里if (isc_thread_create(run, manager,&manager->threads[manager->workers])這里啟動run線程函數(shù),run主要執(zhí)行的代碼就是dispatch(manager),dispath主要執(zhí)行的是(event->ev_action)( (isc_task_t *)task,event);ev->ev_action其實就是個函數(shù)指針;從這里可以看出,bind的處理模式和windows的消息處理機制很相似。
bind的epoll采用的是ET模式,邊沿觸發(fā);只對新到的數(shù)據(jù)進行通知,而內(nèi)核緩沖區(qū)中如果是舊數(shù)據(jù)則不進行通知,每次epoll_wait后,處理完畢后再調(diào)用epoll_ctl;這里實際是個循環(huán)處理過程,epoll_wait等待,然后加入到events數(shù)組中,然后處理,在調(diào)用cpoll_ctrl重新設(shè)置事件類型,再等待,如此循環(huán)。在bind中,sokcet.c中的線程函數(shù)watcher中有epooll_wait函數(shù),在啟動這個線程函數(shù)前在setup_watcher(isc_mem_t *mctx, isc__socketmgr_t *manager) 這個函數(shù)中已經(jīng)調(diào)用了epoll_create。