微信公众号:郑尔多斯
关注可了解更多的Nginx
知识。任何问题或建议,请公众号留言;
关注公众号,有趣有内涵的文章第一时间送达!
上一篇文章我们详细的讲解了auto/init
文件,该文件主要是初始化一些文件目录,便于后面的编译过程。
configure
执行auto/init
之后就会执行auto/sources
文件,所以本文分析一下auto/sources
文件,这个文件虽然内容很多,但是结构非常简单,全部是初始化操作,为后面的Makefile
文件生成各种依赖。比如编译Core
模块用到的依赖,编译pcre
模块用到的依赖等等。
该文件主要初始化了一下几部分内容:
下面我们分析一下各个模块的内容。
首先定义了一个CORE_MODULES
变量,这个变量表示Nginx
的核心模块,从定义中可以看出来,它包括了ngx_core_module
, ngx_errlog_module
,以及ngx_conf_module
这三个模块。
CORE_MODULES="ngx_core_module ngx_errlog_module ngx_conf_module"
定义了CORE_INCS
表示核心模块的头文件所在目录。
CORE_INCS="src/core"
定义了一个CORE_DEPS
(depandencies:
依赖)变量表示核心模块的依赖文件。编译核心模块的时候要使用到这些依赖文件。
CORE_DEPS="src/core/nginx.h \
src/core/ngx_config.h \
src/core/ngx_core.h \
src/core/ngx_log.h \
src/core/ngx_palloc.h \
src/core/ngx_array.h \
src/core/ngx_list.h \
src/core/ngx_hash.h \
src/core/ngx_buf.h \
src/core/ngx_queue.h \
src/core/ngx_string.h \
src/core/ngx_parse.h \
src/core/ngx_inet.h \
src/core/ngx_file.h \
src/core/ngx_crc.h \
src/core/ngx_crc32.h \
src/core/ngx_murmurhash.h \
src/core/ngx_md5.h \
src/core/ngx_sha1.h \
src/core/ngx_rbtree.h \
src/core/ngx_radix_tree.h \
src/core/ngx_slab.h \
src/core/ngx_times.h \
src/core/ngx_shmtx.h \
src/core/ngx_connection.h \
src/core/ngx_cycle.h \
src/core/ngx_conf_file.h \
src/core/ngx_resolver.h \
src/core/ngx_open_file_cache.h \
src/core/ngx_crypt.h"
定义了CORE_SRCS
变量表示核心模块的源文件路径,这些源文件就是实现核心模块的代码。
CORE_SRCS="src/core/nginx.c \
src/core/ngx_log.c \
src/core/ngx_palloc.c \
src/core/ngx_array.c \
src/core/ngx_list.c \
src/core/ngx_hash.c \
src/core/ngx_buf.c \
src/core/ngx_queue.c \
src/core/ngx_output_chain.c \
src/core/ngx_string.c \
src/core/ngx_parse.c \
src/core/ngx_inet.c \
src/core/ngx_file.c \
src/core/ngx_crc32.c \
src/core/ngx_murmurhash.c \
src/core/ngx_md5.c \
src/core/ngx_rbtree.c \
src/core/ngx_radix_tree.c \
src/core/ngx_slab.c \
src/core/ngx_times.c \
src/core/ngx_shmtx.c \
src/core/ngx_connection.c \
src/core/ngx_cycle.c \
src/core/ngx_spinlock.c \
src/core/ngx_cpuinfo.c \
src/core/ngx_conf_file.c \
src/core/ngx_resolver.c \
src/core/ngx_open_file_cache.c \
src/core/ngx_crypt.c"
正则表达式(PCRE
)模块是Nginx
实现rewrite
模块的关键,在auto/sources
中定义了依赖文件和源文件两个变量来编译正则表达式。
REGEX_DEPS=src/core/ngx_regex.h
REGEX_SRCS=src/core/ngx_regex.c
这个模块是ssl
加密模块,用来实现HTTPS
加密。定义了三个变量来编译该模块,语法同CORE
模块。
OPENSSL_MODULE=ngx_openssl_module
OPENSSL_DEPS=src/event/ngx_event_openssl.h
OPENSSL_SRCS=src/event/ngx_event_openssl.c
事件模块是nginx
非常重要的模块,我们都知道nginx
的高并发就是因为使用了优秀的I/O
模块。nginx
为了可移植性,定义了一个事件框架的模型,在不同的平台上面可以使用不同的事件处理机制,通常在Linux
环境下,我们都是用epoll
事件机制。
同样的,auto/sources
定义了几个变量分别表示事件模块的模块名,头文件,依赖文件和源码文件,可以参考CORE_MODULE
的分析。
EVENT_MODULES="ngx_events_module ngx_event_core_module"
EVENT_INCS="src/event src/event/modules"
EVENT_DEPS="src/event/ngx_event.h \
src/event/ngx_event_timer.h \
src/event/ngx_event_posted.h \
src/event/ngx_event_busy_lock.h \
src/event/ngx_event_connect.h \
src/event/ngx_event_pipe.h"
EVENT_SRCS="src/event/ngx_event.c \
src/event/ngx_event_timer.c \
src/event/ngx_event_posted.c \
src/event/ngx_event_busy_lock.c \
src/event/ngx_event_accept.c \
src/event/ngx_event_connect.c \
src/event/ngx_event_pipe.c"
在上面的第4
步中,我们提到了为了在多个平台上运行,Nginx
实现了一套事件机制,这样用户就可以根据当前的平台选择不同的事件处理方法了。比如在Mac
上面可以使用kqueue
,在linux
上面可以使用select
,epoll
等。Nginx
自带了很多种事件处理模块的实现,我们可以根据自己的需求选用不用的模块。
每种模块的定义方式都是相同的。
#select 模块
SELECT_MODULE=ngx_select_module
SELECT_SRCS=src/event/modules/ngx_select_module.c
WIN32_SELECT_SRCS=src/event/modules/ngx_win32_select_module.c
### poll模块
POLL_MODULE=ngx_poll_module
POLL_SRCS=src/event/modules/ngx_poll_module.c
# kqueue模块
KQUEUE_MODULE=ngx_kqueue_module
KQUEUE_SRCS=src/event/modules/ngx_kqueue_module.c
### devpoll模块
DEVPOLL_MODULE=ngx_devpoll_module
DEVPOLL_SRCS=src/event/modules/ngx_devpoll_module.c
EVENTPORT_MODULE=ngx_eventport_module
EVENTPORT_SRCS=src/event/modules/ngx_eventport_module.c
EPOLL_MODULE=ngx_epoll_module
EPOLL_SRCS=src/event/modules/ngx_epoll_module.c
RTSIG_MODULE=ngx_rtsig_module
RTSIG_SRCS=src/event/modules/ngx_rtsig_module.c
# windows平台使用iocp模块
IOCP_MODULE=ngx_iocp_module
IOCP_SRCS=src/event/modules/ngx_iocp_module.c
### aio模块,这是linux平台的一种文件异步处理机制
AIO_MODULE=ngx_aio_module
AIO_SRCS="src/event/modules/ngx_aio_module.c \
src/os/unix/ngx_aio_read.c \
src/os/unix/ngx_aio_write.c \
src/os/unix/ngx_aio_read_chain.c \
src/os/unix/ngx_aio_write_chain.c"
FILE_AIO_SRCS="src/os/unix/ngx_file_aio_read.c"
LINUX_AIO_SRCS="src/os/unix/ngx_linux_aio_read.c"
对于不同的平台,nginx
都实现了一种平台特定代码,用于最底层的数据传输等。我们只看unix/linux
相关的部分,其余的平台也是一样的。其实也没啥要说的哦,就是定义了一些依赖文件,头文件,源文件的目录。
UNIX_INCS="$CORE_INCS $EVENT_INCS src/os/unix"
UNIX_DEPS="$CORE_DEPS $EVENT_DEPS \
src/os/unix/ngx_time.h \
src/os/unix/ngx_errno.h \
src/os/unix/ngx_alloc.h \
src/os/unix/ngx_files.h \
src/os/unix/ngx_channel.h \
src/os/unix/ngx_shmem.h \
src/os/unix/ngx_process.h \
src/os/unix/ngx_setproctitle.h \
src/os/unix/ngx_atomic.h \
src/os/unix/ngx_gcc_atomic_x86.h \
src/os/unix/ngx_thread.h \
src/os/unix/ngx_socket.h \
src/os/unix/ngx_os.h \
src/os/unix/ngx_user.h \
src/os/unix/ngx_process_cycle.h"
# add to UNIX_DEPS
# src/os/unix/ngx_gcc_atomic_amd64.h \
# src/os/unix/ngx_gcc_atomic_sparc64.h \
# src/os/unix/ngx_gcc_atomic_ppc.h \
# src/os/unix/ngx_sunpro_atomic_sparc64.h \
# src/os/unix/ngx_sunpro_x86.il \
# src/os/unix/ngx_sunpro_amd64.il \
# src/os/unix/ngx_sunpro_sparc64.il \
UNIX_SRCS="$CORE_SRCS $EVENT_SRCS \
src/os/unix/ngx_time.c \
src/os/unix/ngx_errno.c \
src/os/unix/ngx_alloc.c \
src/os/unix/ngx_files.c \
src/os/unix/ngx_socket.c \
src/os/unix/ngx_recv.c \
src/os/unix/ngx_readv_chain.c \
src/os/unix/ngx_udp_recv.c \
src/os/unix/ngx_send.c \
src/os/unix/ngx_writev_chain.c \
src/os/unix/ngx_channel.c \
src/os/unix/ngx_shmem.c \
src/os/unix/ngx_process.c \
src/os/unix/ngx_daemon.c \
src/os/unix/ngx_setproctitle.c \
src/os/unix/ngx_posix_init.c \
src/os/unix/ngx_user.c \
src/os/unix/ngx_process_cycle.c"
linux
相关的内容如下:
LINUX_DEPS="src/os/unix/ngx_linux_config.h src/os/unix/ngx_linux.h"
LINUX_SRCS=src/os/unix/ngx_linux_init.c
LINUX_SENDFILE_SRCS=src/os/unix/ngx_linux_sendfile_chain.c
下面的重头戏来了,开始定义HTTP
相关的内容了。
定义了一些基础的模块,是必须要编译到nginx
中的内容,包括ngx_http_module
,ngx_http_core_module
,ngx_http_log_module
,ngx_http_upstream_module
,如下:
# the http modules that have their logging formats
# must be after ngx_http_log_module
HTTP_MODULES="ngx_http_module \
ngx_http_core_module \
ngx_http_log_module \
ngx_http_upstream_module"
以及HTTP
模块的头文件,依赖文件和源代码文件路径,如下:
HTTP_INCS="src/http src/http/modules"
HTTP_DEPS="src/http/ngx_http.h \
src/http/ngx_http_request.h \
src/http/ngx_http_config.h \
src/http/ngx_http_core_module.h \
src/http/ngx_http_cache.h \
src/http/ngx_http_variables.h \
src/http/ngx_http_script.h \
src/http/ngx_http_upstream.h \
src/http/ngx_http_upstream_round_robin.h \
src/http/ngx_http_busy_lock.h"
HTTP_SRCS="src/http/ngx_http.c \
src/http/ngx_http_core_module.c \
src/http/ngx_http_special_response.c \
src/http/ngx_http_request.c \
src/http/ngx_http_parse.c \
src/http/ngx_http_header_filter_module.c \
src/http/ngx_http_write_filter_module.c \
src/http/ngx_http_copy_filter_module.c \
src/http/modules/ngx_http_log_module.c \
src/http/ngx_http_request_body.c \
src/http/ngx_http_variables.c \
src/http/ngx_http_script.c \
src/http/ngx_http_upstream.c \
src/http/ngx_http_upstream_round_robin.c \
src/http/ngx_http_parse_time.c \
src/http/modules/ngx_http_static_module.c \
src/http/modules/ngx_http_index_module.c \
src/http/modules/ngx_http_chunked_filter_module.c \
src/http/modules/ngx_http_range_filter_module.c \
src/http/modules/ngx_http_headers_filter_module.c \
src/http/modules/ngx_http_not_modified_filter_module.c"
剩下的是一些自定义模块,我们在configure
的时候可以选择是否编译,如果不编译,那么在生成的Makefile
中就不会生成相应的代码,我们随便找一个分析一下:
HTTP_REALIP_MODULE=ngx_http_realip_module
HTTP_REALIP_SRCS=src/http/modules/ngx_http_realip_module.c
上面的模块是获取客户端真实ip
的模块,如果我们在configure
的时候选择编译该模块,那么生成Makefile
的时候就会将相应的源码和依赖包含进去。
这部分忽略,因为我从来没用过nginx
当邮件服务器。
但是它的格式和其他模块一模一样,可以参考上面的东东自行分析该部分。
Google Perftool
它是由google
开发的用来分析C/C++
程序性能的一套工具,这里的性能分析主要包括内存和CPU
两个方面,内存分析使用google-perftool
所提供的tcmalloc
,CPU
分析使用它所提供的profiler
。
这里简单了解一下就行了,和nginx
的实现无关。
NGX_GOOGLE_PERFTOOLS_MODULE=ngx_google_perftools_module
NGX_GOOGLE_PERFTOOLS_SRCS=src/misc/ngx_google_perftools_module.c
这个就是一个测试模块,测试是否支持C++
内嵌C
语法,没什么作用(至少我没发现有啥作用)。
一大篇文章,详细的分析了auto/sources
模块的功能,希望对你有所帮助。请继续关注后续文章。