亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 開發 > PHP > 正文

php中動態修改ini配置

2024-05-04 23:26:04
字體:
來源:轉載
供稿:網友
上一篇文章我們闡述php.ini配置原理,本文我們來講動態修改php配置,需要的朋友可以參考下
 
 

1,運行時改變配置
在前一篇中曾經談到,ini_set函數可以在php執行的過程中,動態修改php的部分配置。注意,僅僅是部分,并非所有的配置都可以動態修改。關于ini配置的可修改性,參見:http://php.net/manual/zh/configuration.changes.modes.php

我們直接進入ini_set的實現,函數雖然有點長,但是邏輯很清晰:

 

復制代碼代碼如下:

PHP_FUNCTION(ini_set)
{
    char *varname, *new_value;
    int varname_len, new_value_len;
    char *old_value;

 

    if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "ss", &varname, &varname_len, &new_value, &new_value_len) == FAILURE) {
        return;
    }

    // 去EG(ini_directives)中獲取配置的值
    old_value = zend_ini_string(varname, varname_len + 1, 0);

    /* copy to return here, because alter might free it! */
    if (old_value) {
        RETVAL_STRING(old_value, 1);
    } else {
        RETVAL_FALSE;
    }

    // 如果開啟了安全模式,那么如下這些ini配置可能涉及文件操作,需要要輔助檢查uid
#define _CHECK_PATH(var, var_len, ini) php_ini_check_path(var, var_len, ini, sizeof(ini))
    /* safe_mode & basedir check */
    if (PG(safe_mode) || PG(open_basedir)) {
        if (_CHECK_PATH(varname, varname_len, "error_log") ||
            _CHECK_PATH(varname, varname_len, "java.class.path") ||
            _CHECK_PATH(varname, varname_len, "java.home") ||
            _CHECK_PATH(varname, varname_len, "mail.log") ||
            _CHECK_PATH(varname, varname_len, "java.library.path") ||
            _CHECK_PATH(varname, varname_len, "vpopmail.directory")) {
            if (PG(safe_mode) && (!php_checkuid(new_value, NULL, CHECKUID_CHECK_FILE_AND_DIR))) {
                zval_dtor(return_value);
                RETURN_FALSE;
            }
            if (php_check_open_basedir(new_value TSRMLS_CC)) {
                zval_dtor(return_value);
                RETURN_FALSE;
            }
        }
    }

    // 在安全模式下,如下這些ini受到保護,不會被動態修改
    if (PG(safe_mode)) {
        if (!strncmp("max_execution_time", varname, sizeof("max_execution_time")) ||
            !strncmp("memory_limit", varname, sizeof("memory_limit")) ||
            !strncmp("child_terminate", varname, sizeof("child_terminate"))
        ) {
            zval_dtor(return_value);
            RETURN_FALSE;
        }
    }

    // 調用zend_alter_ini_entry_ex去動態修改ini配置
    if (zend_alter_ini_entry_ex(varname, varname_len + 1, new_value, new_value_len, PHP_INI_USER, PHP_INI_STAGE_RUNTIME, 0 TSRMLS_CC) == FAILURE) {
        zval_dtor(return_value);
        RETURN_FALSE;
    }
}

 

可以看到,除了一些必要的驗證工作,主要就是調用zend_alter_ini_entry_ex。

我們繼續跟進到zend_alter_ini_entry_ex函數中:

 

復制代碼代碼如下:

ZEND_API int zend_alter_ini_entry_ex(char *name, uint name_length, char *new_value, uint new_value_length, int modify_type, int stage, int force_change TSRMLS_DC) /* {{{ */
{
    zend_ini_entry *ini_entry;
    char *duplicate;
    zend_bool modifiable;
    zend_bool modified;

 

    // 找出EG(ini_directives)中對應的ini_entry
    if (zend_hash_find(EG(ini_directives), name, name_length, (void **) &ini_entry) == FAILURE) {
        return FAILURE;
    }

    // 是否被修改以及可修改性
    modifiable = ini_entry->modifiable;
    modified = ini_entry->modified;

    if (stage == ZEND_INI_STAGE_ACTIVATE && modify_type == ZEND_INI_SYSTEM) {
        ini_entry->modifiable = ZEND_INI_SYSTEM;
    }

    // 是否強制修改
    if (!force_change) {
        if (!(ini_entry->modifiable & modify_type)) {
            return FAILURE;
        }
    }

    // EG(modified_ini_directives)用于存放被修改過的ini_entry
    // 主要用做恢復
    if (!EG(modified_ini_directives)) {
        ALLOC_HASHTABLE(EG(modified_ini_directives));
        zend_hash_init(EG(modified_ini_directives), 8, NULL, NULL, 0);
    }
    
    // 將ini_entry中的值,值的長度,可修改范圍,保留到orig_xxx中去
    // 以便在請求結束的時候,可以對ini_entry做恢復
    if (!modified) {
        ini_entry->orig_value = ini_entry->value;
        ini_entry->orig_value_length = ini_entry->value_length;
        ini_entry->orig_modifiable = modifiable;
        ini_entry->modified = 1;
        zend_hash_add(EG(modified_ini_directives), name, name_length, &ini_entry, sizeof(zend_ini_entry*), NULL);
    }

    duplicate = estrndup(new_value, new_value_length);

    // 調用modify來更新XXX_G中對應的ini配置
    if (!ini_entry->on_modify || ini_entry->on_modify(ini_entry, duplicate, new_value_length, ini_entry->mh_arg1, ini_entry->mh_arg2, ini_entry->mh_arg3, stage TSRMLS_CC) == SUCCESS) {
        // 同上面,如果多次修改,則需要釋放前一次修改的值
        if (modified && ini_entry->orig_value != ini_entry->value) {
            efree(ini_entry->value);
        }
        ini_entry->value = duplicate;
        ini_entry->value_length = new_value_length;
    } else {
        efree(duplicate);
        return FAILURE;
    }

    return SUCCESS;
}

 

有3處邏輯需要我們仔細體會:

1)ini_entry中的modified字段用來表示該配置是否被動態修改過。一旦該ini配置發生修改,modified就會被置為1。上述代碼中有一段很關鍵:

 

復制代碼代碼如下:

// 如果多次調用ini_set,則orig_value等始終保持最原始的值
if (!modified) {
    ini_entry->orig_value = ini_entry->value;
    ini_entry->orig_value_length = ini_entry->value_length;
    ini_entry->orig_modifiable = modifiable;
    ini_entry->modified = 1;
    zend_hash_add(EG(modified_ini_directives), name, name_length, &ini_entry, sizeof(zend_ini_entry*), NULL);
}

 

這段代碼表示,不管我們先后在php代碼中調用幾次ini_set,只有第一次ini_set時才會進入這段邏輯,設置好orig_value。從第二次調用ini_set開始,便不會再次執行這段分支,因為此時的modified已經被置為1了。因此,ini_entry->orig_value始終保存的是第一次修改之前的配置值(即最原始的配置)。

2)為了能使ini_set修改的配置立即生效,需要on_modify回調函數。

如前一篇文中所述,調用on_modify是為了能夠更新模塊的全局變量。再次回憶下,首先,模塊全局變量中的配置已經不是字符串類型了,該用bool用bool、該用int用int。其次,每一個ini_entry中都存儲了該模塊全局變量的地址以及對應的偏移量,使得on_modify可以很迅速的進行內存修改。此外不要忘記,on_modify調用完了之后,仍需進一步更新ini_entry->value,這樣EG(ini_directives)中的配置值就是最新的了。

3)這里出現了一張新的hash表,EG(modified_ini_directives)。

EG(modified_ini_directives)只用于存放被動態修改過的ini配置,如果一個ini配置被動態修改過,那么它既存在于EG(ini_directives)中,又存在于EG(modified_ini_directives)中。既然每一個ini_entry都有modified字段做標記,那豈不是可以遍歷EG(ini_directives)來獲得所有被修改過的配置呢?

答案是肯定的。個人覺得,這里的EG(modified_ini_directives)主要還是為了提升性能,醬直接遍歷EG(modified_ini_directives)就足夠了。此外,把EG(modified_ini_directives)的初始化推遲到zend_alter_ini_entry_ex中,也可以看出php在細節上的性能優化點。

2,恢復配置
ini_set的作用時間和php.ini文件的作用時間是不一樣的,一旦請求執行結束,則ini_set會失效。此外,當我們代碼中調用了ini_restore函數,則之前通過ini_set設置的配置也會失效。

每一個php請求執行完畢之后,會觸發php_request_shutdown,它和php_request_startup是兩個相對應過程。如果php是掛接在apache/nginx下,則每處理完一個http請求,就會調用php_request_shutdown;如果php以CLI模式來運行,則腳本執行完畢之后,也會調用php_request_shutdown。

在php_request_shutdown中,我們可以看到針對ini的恢復處理:

 

復制代碼代碼如下:

/* 7. Shutdown scanner/executor/compiler and restore ini entries */
zend_deactivate(TSRMLS_C);

 

進入zend_deactivate,可以進一步看到調用了zend_ini_deactivate函數,由zend_ini_deactivate來負責將php的配置進行恢復。

 

復制代碼代碼如下:

zend_try {
    zend_ini_deactivate(TSRMLS_C);
} zend_end_try();

 

具體來看看zend_ini_deactivate的實現:

 

復制代碼代碼如下:

ZEND_API int zend_ini_deactivate(TSRMLS_D) /* {{{ */
{
    if (EG(modified_ini_directives)) {
        // 遍歷EG(modified_ini_directives)中這張表
        // 對每一個ini_entry調用zend_restore_ini_entry_wrapper
        zend_hash_apply(EG(modified_ini_directives), (apply_func_t) zend_restore_ini_entry_wrapper TSRMLS_CC);
        
        // 回收操作
        zend_hash_destroy(EG(modified_ini_directives));
        FREE_HASHTABLE(EG(modified_ini_directives));
        EG(modified_ini_directives) = NULL;
    }
    return SUCCESS;
}

 

從zend_hash_apply來看,真正恢復ini的任務最終落地到了zend_restore_ini_entry_wrapper回調函數。

 

復制代碼代碼如下:

static int zend_restore_ini_entry_wrapper(zend_ini_entry **ini_entry TSRMLS_DC)
{
    // zend_restore_ini_entry_wrapper就是zend_restore_ini_entry_cb的封裝
    zend_restore_ini_entry_cb(*ini_entry, ZEND_INI_STAGE_DEACTIVATE TSRMLS_CC);
    return 1;
}

 

static int zend_restore_ini_entry_cb(zend_ini_entry *ini_entry, int stage TSRMLS_DC)
{
    int result = FAILURE;

    // 只看修改過的ini項
    if (ini_entry->modified) {
        if (ini_entry->on_modify) {
            // 使用orig_value,對XXX_G內的相關字段進行重新設置
            zend_try {
                result = ini_entry->on_modify(ini_entry, ini_entry->orig_value, ini_entry->orig_value_length, ini_entry->mh_arg1, ini_entry->mh_arg2, ini_entry->mh_arg3, stage TSRMLS_CC);
            } zend_end_try();
        }
        if (stage == ZEND_INI_STAGE_RUNTIME && result == FAILURE) {
            /* runtime failure is OK */
            return 1;
        }
        if (ini_entry->value != ini_entry->orig_value) {
            efree(ini_entry->value);
        }
        
        // ini_entry本身恢復到最原始的值
        ini_entry->value = ini_entry->orig_value;
        ini_entry->value_length = ini_entry->orig_value_length;
        ini_entry->modifiable = ini_entry->orig_modifiable;
        ini_entry->modified = 0;
        ini_entry->orig_value = NULL;
        ini_entry->orig_value_length = 0;
        ini_entry->orig_modifiable = 0;
    }
    return 0;
}

 

邏輯都蠻清晰的,相信讀者可以看明白??偨Y一下關于ini配置的恢復流程:

 

復制代碼代碼如下:

php_request_shutdown--->zend_deactivate--->zend_ini_deactivate--->zend_restore_ini_entry_wrapper--->zend_restore_ini_entry_cb

 

3,配置的銷毀
在sapi生命周期結束的時候,比如apache關閉,cli程序執行完畢等等。一旦進入到這個階段,之前所說的configuration_hash,EG(ini_directives)等都需要被銷毀,其用到的內存空間需要被釋放。

1,php會依次結束所有的模塊,在每個模塊的PHP_MSHUTDOWN_FUNCTION中調用UNREGISTER_INI_ENTRIES。UNREGISTER_INI_ENTRIES和REGISTER_INI_ENTRIES對應,但是UNREGISTER_INI_ENTRIES并不負責模塊全局空間的釋放,XXX_globals這塊內存放在靜態數據區上,無需人為回收。

UNREGISTER_INI_ENTRIES主要做的事情,是將某個模塊的ini_entry配置從EG(ini_directives)表中刪除。刪除之后,ini_entry本身的空間會被回收,但是ini_entry->value不一定會被回收。

當所有模塊的PHP_MSHUTDOWN_FUNCTION都調用UNREGISTER_INI_ENTRIES一遍之后,EG(ini_directives)中只剩下了Core模塊的ini配置。此時,就需要手動調用UNREGISTER_INI_ENTRIES,來完成對Core模塊配置的刪除工作。

 

復制代碼代碼如下:

void php_module_shutdown(TSRMLS_D)
{
    ...
    
    // zend_shutdown會依次關閉除了Core之外的所有php模塊
    // 關閉時會調用各個模塊的PHP_MSHUTDOWN_FUNCTION
    zend_shutdown(TSRMLS_C);
    
    ...

 

    // 至此,EG(ini_directives)中只剩下了Core模塊的配置
    // 這里手動清理一下
    UNREGISTER_INI_ENTRIES();
    
    // 回收configuration_hash
    php_shutdown_config();

    // 回收EG(ini_directives)
    zend_ini_shutdown(TSRMLS_C);

    ...
}

 

當手動調用UNREGISTER_INI_ENTRIES完成之后,EG(ini_directives)已經不包含任何的元素,理論上講,此時的EG(ini_directives)是一張空的hash表。

2,configuration_hash的回收發生在EG(ini_directives)之后,上面貼出的代碼中有關于php_shutdown_config的函數調用。php_shutdown_config主要負責回收configuration_hash。

 

復制代碼代碼如下:

int php_shutdown_config(void)
{
    // 回收configuration_hash
    zend_hash_destroy(&configuration_hash);
    
    ...
    
    return SUCCESS;
}

 

注意zend_hash_destroy并不會釋放configuration_hash本身的空間,同XXX_G訪問的模塊全局空間一樣,configuration_hash也是一個全局變量,無需手動回收。

3,當php_shutdown_config完成時,只剩下EG(ini_directives)的自身空間還沒被釋放。因此最后一步調用zend_ini_shutdown。zend_ini_shutdown用于釋放EG(ini_directives)。在前文已經提到,此時的EG(ini_directives)理論上是一張空的hash表,因此該HashTable本身所占用的空間需要被釋放。

 

復制代碼代碼如下:

ZEND_API int zend_ini_shutdown(TSRMLS_D)
{
    // EG(ini_directives)是動態分配出的空間,需要回收
    zend_hash_destroy(EG(ini_directives));
    free(EG(ini_directives));
    return SUCCESS;
}

 

4,總結


發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
九九久久久久久久久激情| 国产精品欧美激情| 国产精品久久久久久av下载红粉| 国产激情999| 欧美大片免费观看在线观看网站推荐| 亚洲精品国产精品国自产观看浪潮| 国产精品久久91| 91av在线网站| 亚洲午夜精品久久久久久性色| 狠狠干狠狠久久| 在线观看欧美日韩国产| 国产裸体写真av一区二区| 一区二区三区国产在线观看| 亚洲成色777777女色窝| 欧美精品少妇videofree| 国内精品美女av在线播放| 国产精品影片在线观看| 88xx成人精品| 97人人做人人爱| 俺去啦;欧美日韩| 日韩欧美在线视频日韩欧美在线视频| 亚洲激情在线观看| 亚洲成色777777女色窝| 亚洲精品天天看| 91精品国产91久久久久久吃药| 性亚洲最疯狂xxxx高清| 精品久久久久久中文字幕大豆网| 日韩av在线资源| 欧美综合在线第二页| 国产成人鲁鲁免费视频a| 日韩国产精品一区| 国产精品视频免费在线观看| 国产精品扒开腿爽爽爽视频| 成人精品一区二区三区电影黑人| 国产精品亚洲一区二区三区| 欧洲日韩成人av| 精品国内自产拍在线观看| 日韩大片在线观看视频| 久久亚洲一区二区三区四区五区高| 福利二区91精品bt7086| 欧美视频在线观看免费| 91精品国产91久久久久久吃药| 国产91在线播放| 91精品国产网站| 久久久久久久久久久免费| 欧美精品18videos性欧美| 久久亚洲国产精品| 久久久久久久久电影| 久久久精品2019中文字幕神马| 久久精品国亚洲| 亚洲精品有码在线| 国产成人亚洲综合青青| 欧美二区乱c黑人| 伊人久久大香线蕉av一区二区| 俺去亚洲欧洲欧美日韩| 欧美极品少妇xxxxⅹ喷水| 国产午夜精品视频免费不卡69堂| 91天堂在线观看| 欧美www在线| 亚洲精品资源美女情侣酒店| 66m—66摸成人免费视频| 懂色av中文一区二区三区天美| 2019中文字幕在线免费观看| 中文字幕在线观看日韩| 欧美午夜精品久久久久久浪潮| 亚洲精品wwwww| 亚洲国产又黄又爽女人高潮的| 精品少妇一区二区30p| 国产精品永久免费在线| 亚洲天堂一区二区三区| 亚洲性无码av在线| 久久中文精品视频| 伊人久久久久久久久久久| 国产一区二区美女视频| 狠狠久久五月精品中文字幕| 正在播放欧美视频| 欧美www视频在线观看| 精品人伦一区二区三区蜜桃免费| 日韩美女视频中文字幕| 亚洲精品视频二区| 亚洲欧美国产一本综合首页| 国产精品18久久久久久麻辣| 亚洲欧洲日产国码av系列天堂| 欧美国产日韩在线| 久久精品国产电影| 国产精品精品久久久久久| 欧美与黑人午夜性猛交久久久| 欧美成人精品在线视频| 国产亚洲精品va在线观看| 国产成人精品在线| 韩国三级电影久久久久久| 亚洲国产成人精品一区二区| 97精品国产97久久久久久| 欧美日韩国产一中文字不卡| 久久九九国产精品怡红院| 黄色成人av网| 国产精品成人一区二区三区吃奶| 国产精品女人网站| 久久精品国产精品亚洲| 北条麻妃99精品青青久久| 欧美午夜性色大片在线观看| 国产日韩欧美电影在线观看| 亚洲成人免费在线视频| 91av国产在线| 黑人巨大精品欧美一区免费视频| 欧美日韩中文字幕在线视频| 精品国产区一区二区三区在线观看| 97超级碰碰碰久久久| 日韩一区在线视频| 国语自产在线不卡| 久久精品91久久久久久再现| 国产精品wwww| 亚洲一区二区久久久| 日韩av123| 91国内免费在线视频| 亚洲三级av在线| 91社区国产高清| 最新亚洲国产精品| 亚洲最大的免费| 亚洲一区二区黄| 日本乱人伦a精品| 欧美自拍视频在线观看| 全亚洲最色的网站在线观看| 欧美性xxxx极品hd满灌| 日本久久精品视频| 久久精品中文字幕电影| 欧美成人一二三| 成人黄色影片在线| 在线观看视频99| 欧美又大粗又爽又黄大片视频| 国产成人在线精品| 91精品久久久久久久久青青| 欧美在线观看www| 国产福利精品av综合导导航| 亚洲国产欧美一区二区三区同亚洲| 久久精品久久久久电影| 精品无人区太爽高潮在线播放| 亚洲天天在线日亚洲洲精| 国产不卡视频在线| 亚洲男人天天操| 亚洲女人天堂成人av在线| 日本久久久久久久久久久| 日韩有码片在线观看| 久久久久久国产精品三级玉女聊斋| 国产精品h在线观看| 欧洲成人免费aa| 日韩欧美成人网| 国产精品国产三级国产专播精品人| 国产第一区电影| 55夜色66夜色国产精品视频| 国产日韩欧美黄色| 日韩成人av网| 亚洲最大成人在线| 国产福利成人在线| 欧美—级a级欧美特级ar全黄| 久久综合色88| 欧美国产一区二区三区| 日韩欧美亚洲范冰冰与中字| 国模精品视频一区二区三区| 青草青草久热精品视频在线观看| 欧美日韩中文字幕在线| 欧美日韩一区二区三区在线免费观看| 亚洲色图15p| 国产丝袜一区视频在线观看|