Die hier gelesene PHP-Version ist PHP-7.1.0 RC3, und die Plattform zum Lesen des Codes ist Linux
Das erste, was Sie tun müssen, ist, den Eingang zu PHP zu finden. PHP hat viele Modi, Apache, PHP -fpm, CLI-Modus. Wenn ich starten möchte, kann ich nur mit dem einfachsten CLI-Modell beginnen.
Dann muss ich zunächst herausfinden, wie
php -r 'echo 12;'
dieser Befehl ausgeführt wird.
Suchen Sie zunächst nach dem Haupteingang, denn wir sehen uns das Kommandozeilen-PHP-Programm an. Daher befindet sich dieser Eintrag in sapi/cli/php_cli.c.
Zuerst eine Reihe von Variablen definieren
01 int c; 02 zend_file_handle file_handle; 03 int behavior = PHP_MODE_STANDARD; 04 char *reflection_what = NULL; 05 volatile int request_started = 0; 06 volatile int exit_status = 0; 07 char *php_optarg = NULL, *orig_optarg = NULL; 08 int php_optind = 1, orig_optind = 1; 09 char *exec_direct=NULL, *exec_run=NULL, *exec_begin=NULL, *exec_end=NULL; 10 char *arg_free=NULL, **arg_excp=&arg_free; 11 char *script_file=NULL, *translated_path = NULL; 12 int interactive=0; 13 int lineno = 0; 14 const char *param_error=NULL; 15 int hide_argv = 0;
Dann dieses
sapi_module_struct *sapi_module = &cli_sapi_module;
Dies ist eine sapi_module_struct-Struktur, die wichtigste Datenstruktur in Sapi. Seine Definition befindet sich in main/SAPI.h.
Das Folgende ist der Code mit hinzugefügten Kommentaren:
01 struct _sapi_module_struct { // SAPI模块结构 02 char *name; // 应用层名称,比如cli,cgi等 03 char *pretty_name; // 应用层更易读的名字,比如cli对应的就是Command Line Interface 04 05 int (*startup)(struct _sapi_module_struct *sapi_module); // 当一个应用要调用php的时候,这个模块启动的时候会调用的函数 06 int (*shutdown)(struct _sapi_module_struct *sapi_module); // 当一个应用要调用php的时候,这个模块结束的时候会调用的函数 07 08 int (*activate)(void); // 在处理每个request的时候,激活需要调用的函数 09 int (*deactivate)(void); // 在处理完每个request的时候,收尾时候要调用的函数 10 11 size_t (*ub_write)(const char *str, size_t str_length); // 这个函数告诉php如何输出数据 12 void (*flush)(void *server_context); // 提供给php的刷新缓存的函数指针 13 zend_stat_t *(*get_stat)(void); // 用来判断要执行文件的权限,来判断是否有执行权限 14 char *(*getenv)(char *name, size_t name_len); // 获取环境变量的方法 15 16 void (*sapi_error)(int type, const char *error_msg, ...) ZEND_ATTRIBUTE_FORMAT(printf, 2, 3); // 错误处理方法 17 18 int (*header_handler)(sapi_header_struct *sapi_header, sapi_header_op_enum op, sapi_headers_struct *sapi_headers); // 这个函数会在我们调用header()的时候被调用 19 int (*send_headers)(sapi_headers_struct *sapi_headers); // 发送所有的header 20 void (*send_header)(sapi_header_struct *sapi_header, void *server_context); // 单独发送某一个header 21 22 size_t (*read_post)(char *buffer, size_t count_bytes); // 如何获取HTTP POST中的数据 23 char *(*read_cookies)(void); // 如何获取cookie中的数据 24 25 void (*register_server_variables)(zval *track_vars_array); // 这个函数可以给$_SERVER中获取变量 26 void (*log_message)(char *message, int syslog_type_int); // 输出错误信息函数 27 double (*get_request_time)(void); // 获取请求时间的函数 28 void (*terminate_process)(void); // TODO: 调用exit的时候调用的方法 29 30 char *php_ini_path_override; // PHP的ini文件被复写了所复写的地址 31 32 void (*default_post_reader)(void); // 这里和前面的read_post有个差别,read_post负责如何获取POST数据,而这里的函数负责如何解析POST数据 33 void (*treat_data)(int arg, char *str, zval *destArray); // 对数据进行处理,比如进行安全过滤等。 default_post_reader/tread_data/input_filter是三个能对输入进行过滤和处理的函数 34 char *executable_location; // 执行的地理位置 35 36 int php_ini_ignore; // 是否不使用任何ini配置文件,比如php -n 就将这个位置设置为1 37 int php_ini_ignore_cwd; // 不在当前路径寻找php.ini 38 39 int (*get_fd)(int *fd); // 获取执行文件的fd 40 41 int (*force_http_10)(void); // 强制使用http1.0 42 43 int (*get_target_uid)(uid_t *); // 获取执行程序的uid 44 int (*get_target_gid)(gid_t *); // 获取执行程序的gid 45 46 unsigned int (*input_filter)(int arg, char *var, char **val, size_t val_len, size_t *new_val_len); // 对输入进行过滤。比如将输入参数填充到自动全局变量$_GET, $_POST, $_COOKIE中 47 48 void (*ini_defaults)(HashTable *configuration_hash); // 默认的ini配置 49 int phpinfo_as_text; // 是否打印phpinfo信息 50 51 char *ini_entries; // 有没有附带的ini配置,比如使用php -d date.timezone=America/Adak,可以在命令行中设置时区 52 const zend_function_entry *additional_functions; // 每个SAPI模块特有的一些函数注册,比如cli的cli_get_process_title 53 unsigned int (*input_filter_init)(void); // TODO: 54 };
Es gibt ein paar Punkte zum Zusammenfassen:
Der CLI-Modus erfordert kein Senden von Headern, daher sind die drei Funktionen, die der Header-Verarbeitung
1 sapi_cli_header_handler 2 sapi_cli_send_headers 3 sapi_cli_send_header
entsprechen, tatsächlich leere Implementierungen.
Dasselbe gilt für Cookies
sapi_cli_read_cookies
Wir werden einige andere definierte Funktionen analysieren, wenn wir auf sie stoßen.
main
Zurück zur Hauptfunktion. Basierend auf der obigen Struktur verstehen wir
1 argv = save_ps_args(argc, argv); //这里获取一次当前执行进程的参数,环境变量等。为的是对特定平台,修正下argv变量以供后续使用。 2 3 cli_sapi_module.additional_functions = additional_functions; // cli模式特有的函数
Signal
1 #ifdef HAVE_SIGNAL_H 2 #if defined(SIGPIPE) && defined(SIG_IGN) 3 // 忽略SIGPIPE是为了如果php是socket的客户端,那么当服务端关闭的话,会返回一个PIPE的信号,为的是当前的程序不会因为这个而结束 4 signal(SIGPIPE, SIG_IGN); 5 #endif 6 #endif
Das Obige ist PHP Der Inhalt der Kernel-Analyse (1). Weitere verwandte Inhalte finden Sie auf der chinesischen PHP-Website (m.sbmmt.com)!