首頁  >  文章  >  後端開發  >  php模組開發相關的知識介紹

php模組開發相關的知識介紹

伊谢尔伦
伊谢尔伦原創
2017-06-22 13:55:391820瀏覽

PHP的程式碼架構

PHP所有的部分都處在一個被稱為TSRM的層中, TSRM層是負責線程安全管理的. 最底下的SAPI是對外提供服務的介面, 例如命令列的sapi為cli, php-fpm則是fastcgi的sapi, apache的模組方式也是一種sapi. 

PHP核心和Zend 引擎. PHP核心負責請求管理/網絡和檔案運算, Zend核心則負責編譯和執行/記憶體和資源的分配.

在所有這些之上, 是擴充層, PHP中多數對外介面都是透過擴充層來提供的, 例如, standard, string等語言基礎也被以擴展形式提供.

擴展(以後稱為模組)載入到PHP中的方式有​​兩種: 靜態編譯, 動態連結.

靜態編譯需要重新產生php的configure腳本, 這裡不再贅述. 動態連結方式是將模組編譯為一個.so檔, 然後動態的載入到php中.

載入.so檔的方式有兩種,一種是將其寫到php.ini檔案中, 例如: extension=apc.so, 另外一種就是在程式碼中使用dl('xxx.so').

dl($library)

函數的作用就是把一個模組載入進來, 使其內部提供的能力可用.

dl()函數的源代碼在PHP源代碼根目錄(簡寫為PHP_SRC_HOME)下, PHP_SRC_HOME /ext/standard/dl.c, 處理關鍵流程如下:

PHP_FUNCTION(dl)

PHPAPI PHP_FUNCTION(dl)  
{  
    //...  
    php_dl(filename, MODULE_TEMPORARY, return_value, 0 TSRMLS_CC);  
    //...   
}

php_dl

PHPAPI void php_dl(char *file, int type, zval *return_value, int start_now TSRMLS_DC)  
{  
    if (php_load_extension(file, type, start_now TSRMLS_CC) == FAILURE) {  
       //...  
}

php_load_extension

PHPAPI int php_load_extension(char *filename, int type, int start_now TSRMLS_DC) {  
    //文件名解析相关  
      
    //加载动态链接库  
    handle = DL_LOAD(libpath);  
      
    //加载错误处理  
      
    //获取模块的get_module函数(重点, 模块初始入口)  
    get_module = (zend_module_entry *(*)(void)) DL_FETCH_SYMBOL(handle, "get_module");  
      
    //get_module函数获取错误处理  
      
    //那个get_module()得到struct zend_module_entry  
    module_entry = get_module();  
    //...  
      
    //注册模块(重点, 函数在这里被注册)  
    if ((module_entry = zend_register_module_ex(module_entry TSRMLS_CC)) == NULL) {  
        //错误处理  
    }  
      
    //模块启动(重点, PHP_MINIT_FUNCTION)  
    if ((type == MODULE_TEMPORARY || start_now) && zend_startup_module_ex(module_entry TSRMLS_CC) == FAILURE) {  
        //错误处理  
    }  
      
    //模块请求启动(重点, PHP_RINIT_FUNCTION)  
    if ((type == MODULE_TEMPORARY || start_now) && module_entry->request_startup_func) {  
        //错误处理  
    }  
    return SUCCESS;  
}

流程中的重點問題

get_module函數

##get_module = (zend_module_entry *(*)(void)) DL_FETCH_SYMBOL(handle, "get_module");

這句話程式碼經過巨集擴展之後如下:

get_module = (zend_module_entry *(*)(void)) dlsym(handle, "_get_module");

google一下dlsym()函數是乾什麼的, 我們很容易理解這一句程式碼, 這是從剛才載入的動態連結函式庫中取得了一個

函數指標, 也就是我們在開發模組的時候定義的get_module函式.

經過巨集展開為(暫不考慮針對GNU的attribute和針對C++的extern “C”):

zend_module_entry *get_module(void) { return &sample_module_entry; }

透過把dl()函數的載入過程和模組開發時的定義連結起來, 我們可以看到, 模組被載入的時候, 我們自訂的zend_module_entry從這裡被傳遞出去.

模組的註冊

module_entry = zend_register_module_ex(module_entry TSRMLS_CC )

上面的代碼是從函數php_load_extension中摘出的, 我們繼續深入zend_register_module_ex()找到我們關注的函數註冊:

if (module->functions && zend_register_functions(NULL, module ->functions, NULL, module->type TSRMLS_CC)==FAILURE) {

繼續深入到zend_register_functions函數中:

ZEND_API int zend_register_functions(zend_class_entry *scope, const zend_function_entry *functions, HashTable *function_table, int type TSRMLS_DC) /* {{{ */ {  
    //...  
      
    //重点 如果没有函数符号表, 取全局函数符号表  
    if (!target_function_table) {  
        target_function_table = CG(function_table);  
    }  
      
    //...  
      
    //重点 循环zend_function_entry[]  
    while (ptr->fname) {  
        //向函数符号表增加函数  
        if (zend_hash_add(target_function_table, lowercase_name, fname_len+1, &function, sizeof(zend_function), (void**)?_function) == FAILURE) {  
            //错误处理  
        }  
      
        //...  
      
        //准备遍历zend_function_entry[]下一个元素  
        ptr++;  
      
        //...  
    }  
    //...  
      
    return SUCCESS;  
}

在獲取函數符號表的時候, 使用了CG巨集:

target_function_table = CG(function_table);

我們分兩種情況解開這個巨集:

//非线程安全  
compiler_globals.function_table  
      
//线程安全  
(((zend_compiler_globals *) (*((void ***) tsrm_ls))[ compiler_globals_id - 1])-> function_table)

最終, 它們所取得的都是一個全域

結構struct zend_compiler_globals中的function_table元素, 該元素是一個HashTable.

下面的循環就很好理解了, 把模組開發時zend_function_entry中的函數增加到HashTable中就歷到HashTable中了.

模組啟動/模組請求啟動

這兩個部分是很容易理解的, 分別對應的是模組開發中的PHP_MINIT_FUNCTION()和PHP_RINIT_FUNCTION()

以上是php模組開發相關的知識介紹的詳細內容。更多資訊請關注PHP中文網其他相關文章!

陳述:
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn