摘要:重點分析其中宏,返回表的,后面用來作為索引查詢的依據(jù)。中資源的解析比中解析簡單快捷很多,得益于其結構的改變。先從逆向通過其在中索引層層關聯(lián),找到該類資源的釋放回調函數(shù),然后對該資源執(zhí)行釋放回調函數(shù)。
PHP7 使用資源包裹第三方擴展原理分析 注冊資源類型源碼PHP 擴展開發(fā)的文章,我均已更新至《TIPI》
本篇承接上篇 PHP7 使用資源包裹第三方擴展的實現(xiàn)
[c] ZEND_API int zend_register_list_destructors_ex(rsrc_dtor_func_t ld, rsrc_dtor_func_t pld, const char *type_name, int module_number) { zend_rsrc_list_dtors_entry *lde; zval zv; lde = malloc(sizeof(zend_rsrc_list_dtors_entry)); lde->list_dtor_ex = ld; lde->plist_dtor_ex = pld; lde->module_number = module_number; lde->resource_id = list_destructors.nNextFreeElement; lde->type_name = type_name; ZVAL_PTR(&zv, lde); if (zend_hash_next_index_insert(&list_destructors, &zv) == NULL) { return FAILURE; } return list_destructors.nNextFreeElement-1; }
其中
[c] ZVAL_PTR(&zv, lde);
等價于
[c] zv.value.ptr = (lde); zv.u1.type_info = IS_PTR;
list_destructors 是一個全局靜態(tài) HashTable,資源類型注冊時,將一個 zval 結構體變量 zv 存放入 list_destructors 的 arData 中,而 zv 的 value.ptr 卻指向了 zend_rsrc_list_dtors_entry *lde ,lde中包含的該種資源釋放函數(shù)指針、持久資源的釋放函數(shù)指針,資源類型名稱,該資源在 hashtable 中的索引依據(jù) (resource_id)等。
而這里的 resource_id 則是該函數(shù)的返回值,所以后面我們在解析該類型變量時,都需要將 resource_id 帶上。
整個的注冊步驟可以總結為下圖:
[c] ZEND_API zend_resource* zend_register_resource(void *rsrc_pointer, int rsrc_type) { zval *zv; zv = zend_list_insert(rsrc_pointer, rsrc_type); return Z_RES_P(zv); }
該函數(shù)的功能則是將 zend_list_insert 返回的 zval 中的資源指針返回。Z_RES_P 宏在 Zend/zend_types.h 中定義。
重點分析 zend_list_insert
[c] ZEND_API zval *zend_list_insert(void *ptr, int type) { int index; zval zv; index = zend_hash_next_free_element(&EG(regular_list)); if (index == 0) { index = 1; } ZVAL_NEW_RES(&zv, index, ptr, type); return zend_hash_index_add_new(&EG(regular_list), index, &zv); }
其中zend_hash_next_free_element宏,返回&EG(regular_list)表的nNextFreeElement,后面用來作為索引查詢的依據(jù)。
而ZVAL_NEW_RES宏是 PHP7 新增的一套東西,把一個資源裝載到zval里去,因為PHP7 中Bucket只能存zval了。
[c] #define ZVAL_NEW_RES(z, h, p, t) do { zend_resource *_res = (zend_resource *) emalloc(sizeof(zend_resource)); zval *__z; GC_REFCOUNT(_res) = 1; GC_TYPE_INFO(_res) = IS_RESOURCE; _res->handle = (h); _res->type = (t); _res->ptr = (p); __z = (z); Z_RES_P(__z) = _res; Z_TYPE_INFO_P(__z) = IS_RESOURCE_EX; } while (0)
代碼比較清晰,首先根據(jù)h,p,t新建了一個資源,然后一起存入了z這個zval的結構體。(最后兩個宏前面剛剛討論過了)
最后就是zend_hash_index_add_new宏了,追蹤代碼發(fā)現(xiàn)其最后等價于調用的是
[c] _zend_hash_index_add_or_update_i(&EG(regular_list), index, &zv, HASH_ADD | HASH_ADD_NEW ZEND_FILE_LINE_RELAY_CC)
關于PHP7 HashTable的具體操作,這里暫不做細致的分析,后期更新前面的數(shù)據(jù)結構的章節(jié)。注冊的整個邏輯如下圖:
解析資源源碼分析[c] ZEND_API void *zend_fetch_resource(zend_resource *res, const char *resource_type_name, int resource_type) { if (resource_type == res->type) { return res->ptr; } if (resource_type_name) { const char *space; const char *class_name = get_active_class_name(&space); zend_error(E_WARNING, "%s%s%s(): supplied resource is not a valid %s resource", class_name, space, get_active_function_name(), resource_type_name); } return NULL; }
在上面的例子中我們是這樣解析的
[c] (FILE *)zend_fetch_resource(Z_RES_P(filehandle), TIPI_FILE_TYPE, le_tipi_file)
首先通過Z_RES_P宏,獲取filehandle這個zval變量中的zend_resource。然后zend_fetch_resource中只是對比了zend_resource的type與我們預想的資源類型是否一致,然后返回了zend_resource的*ptr,最后轉換成FILE *指針。
PHP7 中資源的解析比 PHP5中解析簡單快捷很多,得益于其 zval 結構的改變。
原來PHP5中則需要通過EG(regular_list)查找,如下圖所示:
而現(xiàn)在 PHP7的解析則直接從zval里解析出zend_resource,如下圖所示:
刪除資源源碼分析[c] ZEND_API int zend_list_close(zend_resource *res) { if (GC_REFCOUNT(res) <= 0) { return zend_list_free(res); } else if (res->type >= 0) { zend_resource_dtor(res); } return SUCCESS; }
與PHP5 不同的地方,這里不是每次都進來將其引用計數(shù)減一操作,而是直接調用zend_resource_dtor函數(shù)。
[c] static void zend_resource_dtor(zend_resource *res) { zend_rsrc_list_dtors_entry *ld; zend_resource r = *res; res->type = -1; res->ptr = NULL; ld = zend_hash_index_find_ptr(&list_destructors, r.type); if (ld) { if (ld->list_dtor_ex) { ld->list_dtor_ex(&r); } } else { zend_error(E_WARNING, "Unknown list entry type (%d)", r.type); } }
如果引用計數(shù)已經(jīng)等于0或者小于0了,那么才從EG(regular_list)中刪除
[c] ZEND_API int zend_list_free(zend_resource *res) { if (GC_REFCOUNT(res) <= 0) { return zend_hash_index_del(&EG(regular_list), res->handle); } else { return SUCCESS; } }
原理圖還是引用上面的注冊資源類型、并注冊資源的圖。
先從zend_resource逆向通過其type在list_destructors中索引層層關聯(lián),找到該類資源的釋放回調函數(shù),然后對該資源執(zhí)行釋放回調函數(shù)。
而后面的從EG(regular_list)中刪除,則是通過res->handler做為索引的依據(jù)。
文章版權歸作者所有,未經(jīng)允許請勿轉載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉載請注明本文地址:http://m.specialneedsforspecialkids.com/yun/23257.html
摘要:釋放用于在不同請求中始終存在的永久資源的函數(shù)。這是新增的數(shù)據(jù)結構,在則是。這使得引擎做一些例如資源類型,注冊變量等的一次初始化。也就是說該函數(shù)的返回值為指針。會將返回值的設為,設為。資源的刪除資源刪除傳入需要被刪除的資源即可。 PHP 擴展開發(fā)的文章,我均已更新至《TIPI》 在閱讀下面的內容之前,我們假定你已經(jīng)對 PHP 7 基本的數(shù)據(jù)結構 都有大致的了解了,這是下面內容閱讀的前提。...
摘要:也就是說該函數(shù)的返回值為指針。會將返回值的設為,設為。而這里的則是該函數(shù)的返回值,所以后面我們在解析該類型變量時,都需要將帶上。 在閱讀下面的內容之前,我們假定你已經(jīng)對 PHP 7 基本的數(shù)據(jù)結構都有大致的了解了,這是下面內容閱讀的前提。 我們分為兩大塊: 首先實現(xiàn)一個自定義的文件打開、讀取、寫入、關閉的文件操作擴展; 然后分析各個操作背后的實現(xiàn)原理,其中某些部分的實現(xiàn)我會和 PHP ...
摘要:業(yè)務和架構不分家,架構是建立在對業(yè)務的理解之上的。主鍵最好保持順序遞增,隨機主鍵會導致聚簇索引樹頻繁分裂,隨機增多,數(shù)據(jù)離散,性能下降。沒有索引的更新,可能會導致全表數(shù)據(jù)都被鎖住。 本博客并非全部原創(chuàng),其實是一個知識的歸納和匯總,里面我引用了很多網(wǎng)上、書上的內容。也給出了相關的鏈接。 本文涉及的知識點比較多,大家可以根據(jù)關鍵字去搜索相關的內容和購買相應的書籍進行系統(tǒng)的學習。不對的地方...
摘要:如果現(xiàn)有子進程中的線程總數(shù)不能滿足負載,控制進程將派生新的子進程。為解決線程的并發(fā)問題,引入了線程安全資源管理器。的全拼,用來存放各個線程的鏈表。 PHP 進階之路 - 零基礎構建自己的服務治理框架(上) PHP 進階之路 - 零基礎構建自己的服務治理框架(下) PHP 進階之路 - 億級 pv 網(wǎng)站架構的技術細節(jié)與套路 PHP 進階之路 - 億級 pv 網(wǎng)站架構實戰(zhàn)之性能壓榨 注...
閱讀 3825·2021-10-12 10:11
閱讀 3644·2021-09-13 10:27
閱讀 2552·2019-08-30 15:53
閱讀 1978·2019-08-29 18:33
閱讀 2198·2019-08-29 14:03
閱讀 1002·2019-08-29 13:27
閱讀 3324·2019-08-28 18:07
閱讀 784·2019-08-26 13:23