热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

PHP内核-Apache2的SAPI

我们知道定义SAPI之前,首先要定义sapi_module_struct这个结构,相看源码:/soft/php-5.2.9/sapi/apache2handler/sapi_apache2.c,可以看到定义该结构,我直接复制过来:[cpp]staticsapi_module_structapach...

我们知道定义SAPI之前,首先要定义sapi_module_struct这个结构,相看源码:/soft/php-5.2.9/sapi/apache2handler/sapi_apache2.c,可以看到定义该结构,我直接复制过来:

[cpp] 
static sapi_module_struct apache2_sapi_module = { 
        "apache2handler", 
        "Apache 2.0 Handler", 
 
        php_apache2_startup,                            /* startup */ 
        php_module_shutdown_wrapper,                    /* shutdown */ 
 
        NULL,                                           /* activate */ 
        NULL,                                           /* deactivate */ 
 
        php_apache_sapi_ub_write,                       /* unbuffered write */ 
        php_apache_sapi_flush,                          /* flush */ 
        php_apache_sapi_get_stat,                       /* get uid */ 
        php_apache_sapi_getenv,                         /* getenv */ 
 
        php_error,                                      /* error handler */ 
 
        php_apache_sapi_header_handler,                 /* header handler */ 
        php_apache_sapi_send_headers,                   /* send headers handler */ 
        NULL,                                           /* send header handler */ 
 
        php_apache_sapi_read_post,                      /* read POST data */ 
        php_apache_sapi_read_COOKIEs,                   /* read COOKIEs */ 
 
        php_apache_sapi_register_variables, 
        php_apache_sapi_log_message,                    /* Log message */ 
        php_apache_sapi_get_request_time,               /* Request Time */ 
 
        STANDARD_SAPI_MODULE_PROPERTIES 
}; 
1,php_apache2_startup:当通过apache调用PHP时,这个函数会被调用。该函数定义如下,主要是对PHP进行初始化。
[cpp]
static int php_apache2_startup(sapi_module_struct *sapi_module) 

        if (php_module_startup(sapi_module, &php_apache_module, 1)==FAILURE) { 
                return FAILURE; 
        } 
        return SUCCESS; 

2,php_module_shutdown_wrapper :PHP的关闭函数。
3,PHP会在每个request的时候,处理一些初始化,资源分配的事务。这部分就是activate字段要定义的。

4,与activate的函数,就是deactiveate,它会提供一个handler, 用来处理收尾工作。

5,php_apache_sapi_ub_write:提供一个向Response数据写的接口。

[cpp]
static int 
php_apache_sapi_ub_write(const char *str, uint str_length TSRMLS_DC) 

        request_rec *r; 
        php_struct *ctx; 
 
        ctx = SG(server_context); 
        r = ctx->r; 
 
        if (ap_rwrite(str, str_length, r) <0) { 
                php_handle_aborted_connection(); 
        } 
 
        return str_length; /* we always consume all the data passed to us. */ 


6,php_apache_sapi_flush:提供给zend刷新缓存的句柄。

[cpp]
static void 
php_apache_sapi_flush(void *server_context) 

        php_struct *ctx; 
        request_rec *r; 
        TSRMLS_FETCH(); 
 
        ctx = server_context; 
 
        /* If we haven&#39;t registered a server_context yet,
         * then don&#39;t bother flushing. */ 
        if (!server_context) { 
                return; 
        } 
 
        r = ctx->r; 
 
        sapi_send_headers(TSRMLS_C); 
 
        r->status = SG(sapi_headers).http_response_code; 
        SG(headers_sent) = 1; 
 
        if (ap_rflush(r) <0 || r->connection->aborted) { 
                php_handle_aborted_connection(); 
        } 

7,php_apache_sapi_get_stat:这部分用来让Zend可以验证一个要执行脚本文件的state,从而判断文件是否据有执行权限等等。
[cpp]
static struct stat* 
php_apache_sapi_get_stat(TSRMLS_D) 

        php_struct *ctx = SG(server_context); 
 
        ctx->finfo.st_uid = ctx->r->finfo.user; 
        ctx->finfo.st_gid = ctx->r->finfo.group; 
        ctx->finfo.st_dev = ctx->r->finfo.device; 
        ctx->finfo.st_ino = ctx->r->finfo.inode; 
#if defined(NETWARE) && defined(CLIB_STAT_PATCH) 
        ctx->finfo.st_atime.tv_sec = apr_time_sec(ctx->r->finfo.atime); 
        ctx->finfo.st_mtime.tv_sec = apr_time_sec(ctx->r->finfo.mtime); 
        ctx->finfo.st_ctime.tv_sec = apr_time_sec(ctx->r->finfo.ctime); 
#else 
        ctx->finfo.st_atime = apr_time_sec(ctx->r->finfo.atime); 
        ctx->finfo.st_mtime = apr_time_sec(ctx->r->finfo.mtime); 
        ctx->finfo.st_ctime = apr_time_sec(ctx->r->finfo.ctime); 
#endif 
 
        ctx->finfo.st_size = ctx->r->finfo.size; 
        ctx->finfo.st_nlink = ctx->r->finfo.nlink; 
 
        return &ctx->finfo; 

8,php_apache_sapi_getenv:为Zend提供了一个根据name来查找环境变量的接口,当我们在脚本中调用getenv的时候,就会间接的调用这个句柄。
[cpp] 
static char * 
php_apache_sapi_getenv(char *name, size_t name_len TSRMLS_DC) 

        php_struct *ctx = SG(server_context); 
        const char *env_var; 
 
        env_var = apr_table_get(ctx->r->subprocess_env, name); 
 
        return (char *) env_var; 

9,php_error:错误处理函数,直接调用PHP错误处理函数。
10,php_apache_sapi_header_handler:在调用PHP的header()函数时,会调用这个函数。

[cpp]
static int 
php_apache_sapi_header_handler(sapi_header_struct *sapi_header,sapi_headers_struct *sapi_headers TSRMLS_DC) 

        php_struct *ctx; 
        char *val, *ptr; 
 
        ctx = SG(server_context); 
 
        val = strchr(sapi_header->header, &#39;:&#39;); 
 
        if (!val) { 
                sapi_free_header(sapi_header); 
                return 0; 
        } 
        ptr = val; 
 
        *val = &#39;\0&#39;; 
 
        do { 
                val++; 
        } while (*val == &#39; &#39;); 
        if (!strcasecmp(sapi_header->header, "content-type")) { 
                if (ctx->content_type) { 
                        efree(ctx->content_type); 
                } 
                ctx->content_type = estrdup(val); 
        } else if (sapi_header->replace) { 
                apr_table_set(ctx->r->headers_out, sapi_header->header, val); 
        } else { 
                apr_table_add(ctx->r->headers_out, sapi_header->header, val); 
        } 
        *ptr = &#39;:&#39;; 
 
        return SAPI_HEADER_ADD; 

11,php_apache_sapi_send_headers:当要真正发送header的时候,这个函数会被调用。
[cpp] 
static int 
php_apache_sapi_send_headers(sapi_headers_struct *sapi_headers TSRMLS_DC) 

        php_struct *ctx = SG(server_context); 
        const char *sline = SG(sapi_headers).http_status_line; 
 
        ctx->r->status = SG(sapi_headers).http_response_code; 
 
        /* httpd requires that r->status_line is set to the first digit of
         * the status-code: */ 
        if (sline && strlen(sline) > 12 && strncmp(sline, "HTTP/1.", 7) == 0 && sline[8] == &#39; &#39;) { 
                ctx->r->status_line = apr_pstrdup(ctx->r->pool, sline + 9); 
                ctx->r->proto_num = 1000 + (sline[7]-&#39;0&#39;); 
                if ((sline[7]-&#39;0&#39;) == 0) { 
                        apr_table_set(ctx->r->subprocess_env, "force-response-1.0", "true"); 
                } 
        } 
 
        /*      call ap_set_content_type only once, else each time we call it,
                configured output filters for that content type will be added */ 
        if (!ctx->content_type) { 
                ctx->content_type = sapi_get_default_content_type(TSRMLS_C); 
        } 
        ap_set_content_type(ctx->r, apr_pstrdup(ctx->r->pool, ctx->content_type)); 
        efree(ctx->content_type); 
        ctx->content_type = NULL; 
 
        return SAPI_HEADER_SENT_SUCCESSFULLY; 

12,在php_apache_sapi_send_headers指针下面有一个域,用来指明发送每一个单独的header时调用。
13,php_apache_sapi_read_post:表示如何读取POST数据。

[cpp]
static int  
php_apache_sapi_read_post(char *buf, uint count_bytes TSRMLS_DC) 

        apr_size_t len, tlen=0; 
        php_struct *ctx = SG(server_context); 
        request_rec *r; 
        apr_bucket_brigade *brigade; 
 
        r = ctx->r; 
        brigade = ctx->brigade; 
        len = count_bytes; 
 
        /*
         * This loop is needed because ap_get_brigade() can return us partial data
         * which would cause premature termination of request read. Therefor we
         * need to make sure that if data is available we fill the buffer completely.
         */ 
 
        while (ap_get_brigade(r->input_filters, brigade, AP_MODE_READBYTES, APR_BLOCK_READ, len) == APR_SUCCESS) { 
                apr_brigade_flatten(brigade, buf, &len); 
                apr_brigade_cleanup(brigade); 
                tlen += len; 
                if (tlen == count_bytes || !len) { 
                        break; 
                } 
                buf += len; 
                len = count_bytes - tlen; 
        } 
 
        return tlen; 


14,php_apache_sapi_read_COOKIE:如何读取COOKIE。

[cpp]
static char * 
php_apache_sapi_read_COOKIEs(TSRMLS_D) 

        php_struct *ctx = SG(server_context); 
        const char *http_COOKIE; 
 
        http_COOKIE = apr_table_get(ctx->r->headers_in, "COOKIE"); 
 
        /* The SAPI interface should use &#39;const char *&#39; */ 
        return (char *) http_COOKIE; 

15,php_apache_sapi_register_variables:提供接口,用于给$_SERVER[]数组提供变量。
[cpp]
static void 
php_apache_sapi_register_variables(zval *track_vars_array TSRMLS_DC) 

        php_struct *ctx = SG(server_context); 
        const apr_array_header_t *arr = apr_table_elts(ctx->r->subprocess_env); 
        char *key, *val; 
        int new_val_len; 
 
        APR_ARRAY_FOREACH_OPEN(arr, key, val) 
                if (!val) { 
                        val = ""; 
                } 
                if (sapi_module.input_filter(PARSE_SERVER, key, &val, strlen(val), &new_val_len TSRMLS_CC)) { 
                        php_register_variable_safe(key, val, new_val_len, track_vars_array TSRMLS_CC); 
                } 
        APR_ARRAY_FOREACH_CLOSE() 
 
        if (sapi_module.input_filter(PARSE_SERVER, "PHP_SELF", &ctx->r->uri, strlen(ctx->r->uri), &new_val_len TSRMLS_CC)) { 
                php_register_variable_safe("PHP_SELF", ctx->r->uri, new_val_len, track_vars_array TSRMLS_CC); 
        } 

16,php_apache_sapi_log_message:输出错误信息。
[cpp]
static void php_apache_sapi_log_message(char *msg) 

        php_struct *ctx; 
        TSRMLS_FETCH(); 
 
        ctx = SG(server_context); 
 
        if (ctx == NULL) { /* we haven&#39;t initialized our ctx yet, oh well */ 
                ap_log_error(APLOG_MARK, APLOG_ERR | APLOG_STARTUP, 0, NULL, "%s", msg); 
        } else { 
                ap_log_rerror(APLOG_MARK, APLOG_ERR, 0, ctx->r, "%s", msg); 
        } 


17,php_apache_sapi_get_request_time:获取请求时间。

[cpp] 
static time_t php_apache_sapi_get_request_time(TSRMLS_D) { 
        php_struct *ctx = SG(server_context); 
        return apr_time_sec(ctx->r->request_time); 

这就完成了apache的SAPI定义。之后,就是当用户用URL请求apache服务,这些函数指针就会在适当的时候,发挥作用了(被调用)。

 

 

 


推荐阅读
  • 如何使用 net.sf.extjwnl.data.Word 类及其代码示例详解 ... [详细]
  • Git基础操作指南:掌握必备技能
    掌握 Git 基础操作是每个开发者必备的技能。本文详细介绍了 Git 的基本命令和使用方法,包括初始化仓库、配置用户信息、添加文件、提交更改以及查看版本历史等关键步骤。通过这些操作,读者可以快速上手并高效管理代码版本。例如,使用 `git config --global user.name` 和 `git config --global user.email` 来设置全局用户名和邮箱,确保每次提交时都能正确标识提交者信息。 ... [详细]
  • 在本教程中,我们将详细介绍如何使用 ArcGIS API 3.x for JavaScript 绘制风向流动图。如果您对所涉及的 API 类不熟悉,建议参考 Esri 官方网站上的 ArcGIS API 3.x for JavaScript 文档,其中提供了详尽的类介绍和使用说明。此外,我们还将提供完整的源代码,帮助您更好地理解和实现这一功能。 ... [详细]
  • 解决基于XML配置的MyBatis在Spring整合中出现“无效绑定语句(未找到):com.music.dao.MusicDao.findAll”问题的方法
    在将Spring与MyBatis进行整合时,作者遇到了“无效绑定语句(未找到):com.music.dao.MusicDao.findAll”的问题。该问题主要出现在使用XML文件配置DAO层的情况下,而注解方式配置则未出现类似问题。作者详细分析了两个配置文件之间的差异,并最终找到了解决方案。本文将详细介绍问题的原因及解决方法,帮助读者避免类似问题的发生。 ... [详细]
  • 基于Node.js的高性能实时消息推送系统通过集成Socket.IO和Express框架,实现了高效的高并发消息转发功能。该系统能够支持大量用户同时在线,并确保消息的实时性和可靠性,适用于需要即时通信的应用场景。 ... [详细]
  • 在Linux环境下,本文详细探讨了Apache服务器中CGI技术的应用与实现。首先,通过使用yum包管理器安装了必要的软件,如PHP。安装完成后,对Apache服务器进行了配置,确保CGI功能正常运行。此外,还介绍了如何编写和调试CGI脚本,以及如何在实际环境中部署这些脚本以提供动态网页内容。实验结果表明,通过合理的配置和优化,Apache服务器能够高效地支持CGI应用程序,为用户提供丰富的交互体验。 ... [详细]
  • 本文深入探讨了原型模式在软件设计中的应用与实现。原型模式通过使用已有的实例作为原型来创建新对象,而不是直接通过类实例化。这种方式不仅简化了对象的创建过程,还提高了系统的灵活性和效率。具体来说,原型模式涉及一个支持克隆功能的接口或基类,子类通过实现该接口来提供具体的克隆方法,从而实现对象的快速复制。此外,文章还详细分析了原型模式的优缺点及其在实际项目中的应用场景,为开发者提供了实用的指导和建议。 ... [详细]
  • 本文深入解析了 Apache 配置文件 `httpd.conf` 和 `.htaccess` 的优化方法,探讨了如何通过合理配置提升服务器性能和安全性。文章详细介绍了这两个文件的关键参数及其作用,并提供了实际应用中的最佳实践,帮助读者更好地理解和运用 Apache 配置。 ... [详细]
  • 在《PHP应用性能优化实战指南:从理论到实践的全面解析》一文中,作者分享了一次实际的PHP应用优化经验。文章回顾了先前进行的一次优化项目,指出即使系统运行时间较长后出现的各种问题和性能瓶颈,通过采用一些通用的优化策略仍然能够有效解决。文中不仅详细阐述了优化的具体步骤和方法,还结合实例分析了优化前后的性能对比,为读者提供了宝贵的参考和借鉴。 ... [详细]
  • HBase客户端Table类中getRpcTimeout方法的应用与编程实例解析 ... [详细]
  • 本文介绍了如何通过掌握 IScroll 技巧来实现流畅的上拉加载和下拉刷新功能。首先,需要按正确的顺序引入相关文件:1. Zepto;2. iScroll.js;3. scroll-probe.js。此外,还提供了完整的代码示例,可在 GitHub 仓库中查看。通过这些步骤,开发者可以轻松实现高效、流畅的滚动效果,提升用户体验。 ... [详细]
  • 掌握PHP框架开发与应用的核心知识点:构建高效PHP框架所需的技术与能力综述
    掌握PHP框架开发与应用的核心知识点对于构建高效PHP框架至关重要。本文综述了开发PHP框架所需的关键技术和能力,包括但不限于对PHP语言的深入理解、设计模式的应用、数据库操作、安全性措施以及性能优化等方面。对于初学者而言,熟悉主流框架如Laravel、Symfony等的实际应用场景,有助于更好地理解和掌握自定义框架开发的精髓。 ... [详细]
  • 本文详细介绍了如何在Linux系统中搭建51单片机的开发与编程环境,重点讲解了使用Makefile进行项目管理的方法。首先,文章指导读者安装SDCC(Small Device C Compiler),这是一个专为小型设备设计的C语言编译器,适合用于51单片机的开发。随后,通过具体的实例演示了如何配置Makefile文件,以实现代码的自动化编译与链接过程,从而提高开发效率。此外,还提供了常见问题的解决方案及优化建议,帮助开发者快速上手并解决实际开发中可能遇到的技术难题。 ... [详细]
  • 抖音AI特效风靡网络,真人瞬间变身动漫角色,吴亦凡、PDD和戚薇纷纷沉迷其中
    近期,抖音推出的一款名为“变身漫画”的AI特效在社交媒体上迅速走红,吸引了大量用户尝试。不仅普通网友积极参与,连吴亦凡、PDD和戚薇等明星也纷纷加入,体验将真人瞬间转化为动漫角色的神奇效果。这一特效凭借其高度的趣味性和创新性,迅速成为网络热议的话题。 ... [详细]
  • 如何利用Git实现高效的多人协作开发(远程仓库应用实例)——Ares Zhao
    Git作为一种分布式版本控制系统,每位开发者都是本地仓库的管理者。然而,为了实现团队间的高效协作,需要将本地的开发成果推送至远程共享仓库,以便其他成员能够同步更新。本文将以GitHub为例,详细介绍如何通过设置和使用远程仓库,实现多人协作开发的最佳实践。 ... [详细]
author-avatar
hcl春丽
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有