1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663 |
- #include <stdio.h>
- #include <signal.h>
- #include "zend.h"
- #include "zend_compile.h"
- #include "zend_execute.h"
- #include "zend_API.h"
- #include "zend_stack.h"
- #include "zend_constants.h"
- #include "zend_extensions.h"
- #include "zend_exceptions.h"
- #include "zend_closures.h"
- #include "zend_generators.h"
- #include "zend_vm.h"
- #include "zend_float.h"
- #ifdef HAVE_SYS_TIME_H
- #include <sys/time.h>
- #endif
- #ifdef HAVE_UNISTD_H
- #include <unistd.h>
- #endif
- ZEND_API void (*zend_execute_ex)(zend_execute_data *execute_data);
- ZEND_API void (*zend_execute_internal)(zend_execute_data *execute_data, zval *return_value);
- ZEND_API const zend_fcall_info empty_fcall_info = { 0, {{0}, {{0}}, {0}}, NULL, NULL, NULL, 0, 0 };
- ZEND_API const zend_fcall_info_cache empty_fcall_info_cache = { NULL, NULL, NULL, NULL };
- #ifdef ZEND_WIN32
- ZEND_TLS HANDLE tq_timer = NULL;
- #endif
- #if 0&&ZEND_DEBUG
- static void (*original_sigsegv_handler)(int);
- static void zend_handle_sigsegv(int dummy)
- {
- fflush(stdout);
- fflush(stderr);
- if (original_sigsegv_handler == zend_handle_sigsegv) {
- signal(SIGSEGV, original_sigsegv_handler);
- } else {
- signal(SIGSEGV, SIG_DFL);
- }
- {
- fprintf(stderr, "SIGSEGV caught on opcode %d on opline %d of %s() at %s:%d\n\n",
- active_opline->opcode,
- active_opline-EG(active_op_array)->opcodes,
- get_active_function_name(),
- zend_get_executed_filename(),
- zend_get_executed_lineno());
- #ifdef ZEND_WIN32
- fflush(stderr);
- #endif
- }
- if (original_sigsegv_handler!=zend_handle_sigsegv) {
- original_sigsegv_handler(dummy);
- }
- }
- #endif
- static void zend_extension_activator(zend_extension *extension)
- {
- if (extension->activate) {
- extension->activate();
- }
- }
- static void zend_extension_deactivator(zend_extension *extension)
- {
- if (extension->deactivate) {
- extension->deactivate();
- }
- }
- static int clean_non_persistent_constant_full(zval *zv)
- {
- zend_constant *c = Z_PTR_P(zv);
- return (ZEND_CONSTANT_FLAGS(c) & CONST_PERSISTENT) ? ZEND_HASH_APPLY_KEEP : ZEND_HASH_APPLY_REMOVE;
- }
- static int clean_non_persistent_function_full(zval *zv)
- {
- zend_function *function = Z_PTR_P(zv);
- return (function->type == ZEND_INTERNAL_FUNCTION) ? ZEND_HASH_APPLY_KEEP : ZEND_HASH_APPLY_REMOVE;
- }
- static int clean_non_persistent_class_full(zval *zv)
- {
- zend_class_entry *ce = Z_PTR_P(zv);
- return (ce->type == ZEND_INTERNAL_CLASS) ? ZEND_HASH_APPLY_KEEP : ZEND_HASH_APPLY_REMOVE;
- }
- void init_executor(void)
- {
- zend_init_fpu();
- ZVAL_NULL(&EG(uninitialized_zval));
- ZVAL_ERROR(&EG(error_zval));
- #if 0&&ZEND_DEBUG
- original_sigsegv_handler = signal(SIGSEGV, zend_handle_sigsegv);
- #endif
- EG(symtable_cache_ptr) = EG(symtable_cache) - 1;
- EG(symtable_cache_limit) = EG(symtable_cache) + SYMTABLE_CACHE_SIZE - 1;
- EG(no_extensions) = 0;
- EG(function_table) = CG(function_table);
- EG(class_table) = CG(class_table);
- EG(in_autoload) = NULL;
- EG(autoload_func) = NULL;
- EG(error_handling) = EH_NORMAL;
- EG(flags) = EG_FLAGS_INITIAL;
- zend_vm_stack_init();
- zend_hash_init(&EG(symbol_table), 64, NULL, ZVAL_PTR_DTOR, 0);
- zend_llist_apply(&zend_extensions, (llist_apply_func_t) zend_extension_activator);
- zend_hash_init(&EG(included_files), 8, NULL, NULL, 0);
- EG(ticks_count) = 0;
- ZVAL_UNDEF(&EG(user_error_handler));
- ZVAL_UNDEF(&EG(user_exception_handler));
- EG(current_execute_data) = NULL;
- zend_stack_init(&EG(user_error_handlers_error_reporting), sizeof(int));
- zend_stack_init(&EG(user_error_handlers), sizeof(zval));
- zend_stack_init(&EG(user_exception_handlers), sizeof(zval));
- zend_objects_store_init(&EG(objects_store), 1024);
- EG(full_tables_cleanup) = 0;
- EG(vm_interrupt) = 0;
- EG(timed_out) = 0;
- EG(exception) = NULL;
- EG(prev_exception) = NULL;
- EG(fake_scope) = NULL;
- EG(ht_iterators_count) = sizeof(EG(ht_iterators_slots)) / sizeof(HashTableIterator);
- EG(ht_iterators_used) = 0;
- EG(ht_iterators) = EG(ht_iterators_slots);
- memset(EG(ht_iterators), 0, sizeof(EG(ht_iterators_slots)));
- EG(each_deprecation_thrown) = 0;
- EG(persistent_constants_count) = EG(zend_constants)->nNumUsed;
- EG(persistent_functions_count) = EG(function_table)->nNumUsed;
- EG(persistent_classes_count) = EG(class_table)->nNumUsed;
- EG(active) = 1;
- }
- static int zval_call_destructor(zval *zv)
- {
- if (Z_TYPE_P(zv) == IS_INDIRECT) {
- zv = Z_INDIRECT_P(zv);
- }
- if (Z_TYPE_P(zv) == IS_OBJECT && Z_REFCOUNT_P(zv) == 1) {
- return ZEND_HASH_APPLY_REMOVE;
- } else {
- return ZEND_HASH_APPLY_KEEP;
- }
- }
- static void zend_unclean_zval_ptr_dtor(zval *zv)
- {
- if (Z_TYPE_P(zv) == IS_INDIRECT) {
- zv = Z_INDIRECT_P(zv);
- }
- i_zval_ptr_dtor(zv ZEND_FILE_LINE_CC);
- }
- static void zend_throw_or_error(int fetch_type, zend_class_entry *exception_ce, const char *format, ...)
- {
- va_list va;
- char *message = NULL;
- va_start(va, format);
- zend_vspprintf(&message, 0, format, va);
- if (fetch_type & ZEND_FETCH_CLASS_EXCEPTION) {
- zend_throw_error(exception_ce, "%s", message);
- } else {
- zend_error(E_ERROR, "%s", message);
- }
- efree(message);
- va_end(va);
- }
- void shutdown_destructors(void)
- {
- if (CG(unclean_shutdown)) {
- EG(symbol_table).pDestructor = zend_unclean_zval_ptr_dtor;
- }
- zend_try {
- uint32_t symbols;
- do {
- symbols = zend_hash_num_elements(&EG(symbol_table));
- zend_hash_reverse_apply(&EG(symbol_table), (apply_func_t) zval_call_destructor);
- } while (symbols != zend_hash_num_elements(&EG(symbol_table)));
- zend_objects_store_call_destructors(&EG(objects_store));
- } zend_catch {
-
- zend_objects_store_mark_destructed(&EG(objects_store));
- } zend_end_try();
- }
- void shutdown_executor(void)
- {
- zend_string *key;
- zval *zv;
- #if ZEND_DEBUG
- zend_bool fast_shutdown = 0;
- #else
- zend_bool fast_shutdown = is_zend_mm() && !EG(full_tables_cleanup);
- #endif
- zend_try {
- zend_llist_destroy(&CG(open_files));
- } zend_end_try();
- zend_try {
- zend_close_rsrc_list(&EG(regular_list));
- } zend_end_try();
- zend_objects_store_free_object_storage(&EG(objects_store), fast_shutdown);
-
-
- EG(active) = 0;
- zend_try {
- zend_llist_apply(&zend_extensions, (llist_apply_func_t) zend_extension_deactivator);
- } zend_end_try();
- if (fast_shutdown) {
-
- zend_hash_discard(EG(zend_constants), EG(persistent_constants_count));
- zend_hash_discard(EG(function_table), EG(persistent_functions_count));
- zend_hash_discard(EG(class_table), EG(persistent_classes_count));
- zend_cleanup_internal_classes();
- } else {
- zend_hash_graceful_reverse_destroy(&EG(symbol_table));
- #if ZEND_DEBUG
- if (gc_enabled() && !CG(unclean_shutdown)) {
- gc_collect_cycles();
- }
- #endif
-
- if (Z_TYPE(EG(user_error_handler)) != IS_UNDEF) {
- zval_ptr_dtor(&EG(user_error_handler));
- ZVAL_UNDEF(&EG(user_error_handler));
- }
- if (Z_TYPE(EG(user_exception_handler)) != IS_UNDEF) {
- zval_ptr_dtor(&EG(user_exception_handler));
- ZVAL_UNDEF(&EG(user_exception_handler));
- }
- zend_stack_clean(&EG(user_error_handlers_error_reporting), NULL, 1);
- zend_stack_clean(&EG(user_error_handlers), (void (*)(void *))ZVAL_PTR_DTOR, 1);
- zend_stack_clean(&EG(user_exception_handlers), (void (*)(void *))ZVAL_PTR_DTOR, 1);
- zend_vm_stack_destroy();
- if (EG(full_tables_cleanup)) {
- zend_hash_reverse_apply(EG(zend_constants), clean_non_persistent_constant_full);
- zend_hash_reverse_apply(EG(function_table), clean_non_persistent_function_full);
- zend_hash_reverse_apply(EG(class_table), clean_non_persistent_class_full);
- } else {
- ZEND_HASH_REVERSE_FOREACH_STR_KEY_VAL(EG(zend_constants), key, zv) {
- zend_constant *c = Z_PTR_P(zv);
- if (ZEND_CONSTANT_FLAGS(c) & CONST_PERSISTENT) {
- break;
- }
- zval_ptr_dtor_nogc(&c->value);
- if (c->name) {
- zend_string_release_ex(c->name, 0);
- }
- efree(c);
- zend_string_release_ex(key, 0);
- } ZEND_HASH_FOREACH_END_DEL();
- ZEND_HASH_REVERSE_FOREACH_STR_KEY_VAL(EG(function_table), key, zv) {
- zend_function *func = Z_PTR_P(zv);
- if (func->type == ZEND_INTERNAL_FUNCTION) {
- break;
- }
- destroy_op_array(&func->op_array);
- zend_string_release_ex(key, 0);
- } ZEND_HASH_FOREACH_END_DEL();
- ZEND_HASH_REVERSE_FOREACH_STR_KEY_VAL(EG(class_table), key, zv) {
- zend_class_entry *ce = Z_PTR_P(zv);
- if (ce->type == ZEND_INTERNAL_CLASS) {
- break;
- }
- destroy_zend_class(zv);
- zend_string_release_ex(key, 0);
- } ZEND_HASH_FOREACH_END_DEL();
- }
- zend_cleanup_internal_classes();
- while (EG(symtable_cache_ptr)>=EG(symtable_cache)) {
- zend_hash_destroy(*EG(symtable_cache_ptr));
- FREE_HASHTABLE(*EG(symtable_cache_ptr));
- EG(symtable_cache_ptr)--;
- }
- zend_hash_destroy(&EG(included_files));
- zend_stack_destroy(&EG(user_error_handlers_error_reporting));
- zend_stack_destroy(&EG(user_error_handlers));
- zend_stack_destroy(&EG(user_exception_handlers));
- zend_objects_store_destroy(&EG(objects_store));
- if (EG(in_autoload)) {
- zend_hash_destroy(EG(in_autoload));
- FREE_HASHTABLE(EG(in_autoload));
- }
- if (EG(ht_iterators) != EG(ht_iterators_slots)) {
- efree(EG(ht_iterators));
- }
- }
- #if ZEND_DEBUG
- if (EG(ht_iterators_used) && !CG(unclean_shutdown)) {
- zend_error(E_WARNING, "Leaked %" PRIu32 " hashtable iterators", EG(ht_iterators_used));
- }
- #endif
- EG(ht_iterators_used) = 0;
- zend_shutdown_fpu();
- }
- ZEND_API const char *get_active_class_name(const char **space)
- {
- zend_function *func;
- if (!zend_is_executing()) {
- if (space) {
- *space = "";
- }
- return "";
- }
- func = EG(current_execute_data)->func;
- switch (func->type) {
- case ZEND_USER_FUNCTION:
- case ZEND_INTERNAL_FUNCTION:
- {
- zend_class_entry *ce = func->common.scope;
- if (space) {
- *space = ce ? "::" : "";
- }
- return ce ? ZSTR_VAL(ce->name) : "";
- }
- default:
- if (space) {
- *space = "";
- }
- return "";
- }
- }
- ZEND_API const char *get_active_function_name(void)
- {
- zend_function *func;
- if (!zend_is_executing()) {
- return NULL;
- }
- func = EG(current_execute_data)->func;
- switch (func->type) {
- case ZEND_USER_FUNCTION: {
- zend_string *function_name = func->common.function_name;
- if (function_name) {
- return ZSTR_VAL(function_name);
- } else {
- return "main";
- }
- }
- break;
- case ZEND_INTERNAL_FUNCTION:
- return ZSTR_VAL(func->common.function_name);
- break;
- default:
- return NULL;
- }
- }
- ZEND_API const char *zend_get_executed_filename(void)
- {
- zend_execute_data *ex = EG(current_execute_data);
- while (ex && (!ex->func || !ZEND_USER_CODE(ex->func->type))) {
- ex = ex->prev_execute_data;
- }
- if (ex) {
- return ZSTR_VAL(ex->func->op_array.filename);
- } else {
- return "[no active file]";
- }
- }
- ZEND_API zend_string *zend_get_executed_filename_ex(void)
- {
- zend_execute_data *ex = EG(current_execute_data);
- while (ex && (!ex->func || !ZEND_USER_CODE(ex->func->type))) {
- ex = ex->prev_execute_data;
- }
- if (ex) {
- return ex->func->op_array.filename;
- } else {
- return NULL;
- }
- }
- ZEND_API uint32_t zend_get_executed_lineno(void)
- {
- zend_execute_data *ex = EG(current_execute_data);
- while (ex && (!ex->func || !ZEND_USER_CODE(ex->func->type))) {
- ex = ex->prev_execute_data;
- }
- if (ex) {
- if (EG(exception) && ex->opline->opcode == ZEND_HANDLE_EXCEPTION &&
- ex->opline->lineno == 0 && EG(opline_before_exception)) {
- return EG(opline_before_exception)->lineno;
- }
- return ex->opline->lineno;
- } else {
- return 0;
- }
- }
- ZEND_API zend_class_entry *zend_get_executed_scope(void)
- {
- zend_execute_data *ex = EG(current_execute_data);
- while (1) {
- if (!ex) {
- return NULL;
- } else if (ex->func && (ZEND_USER_CODE(ex->func->type) || ex->func->common.scope)) {
- return ex->func->common.scope;
- }
- ex = ex->prev_execute_data;
- }
- }
- ZEND_API zend_bool zend_is_executing(void)
- {
- return EG(current_execute_data) != 0;
- }
- ZEND_API int zend_use_undefined_constant(zend_string *name, zend_ast_attr attr, zval *result)
- {
- char *colon;
- if (UNEXPECTED(EG(exception))) {
- return FAILURE;
- } else if ((colon = (char*)zend_memrchr(ZSTR_VAL(name), ':', ZSTR_LEN(name)))) {
- zend_throw_error(NULL, "Undefined class constant '%s'", ZSTR_VAL(name));
- return FAILURE;
- } else if ((attr & IS_CONSTANT_UNQUALIFIED) == 0) {
- zend_throw_error(NULL, "Undefined constant '%s'", ZSTR_VAL(name));
- return FAILURE;
- } else {
- char *actual = ZSTR_VAL(name);
- size_t actual_len = ZSTR_LEN(name);
- char *slash = (char *) zend_memrchr(actual, '\\', actual_len);
- if (slash) {
- actual = slash + 1;
- actual_len -= (actual - ZSTR_VAL(name));
- }
- zend_error(E_WARNING, "Use of undefined constant %s - assumed '%s' (this will throw an Error in a future version of PHP)", actual, actual);
- if (EG(exception)) {
- return FAILURE;
- } else {
- zend_string *result_str = zend_string_init(actual, actual_len, 0);
- zval_ptr_dtor_nogc(result);
- ZVAL_NEW_STR(result, result_str);
- }
- }
- return SUCCESS;
- }
- ZEND_API int zval_update_constant_ex(zval *p, zend_class_entry *scope)
- {
- if (Z_TYPE_P(p) == IS_CONSTANT_AST) {
- zend_ast *ast = Z_ASTVAL_P(p);
- if (ast->kind == ZEND_AST_CONSTANT) {
- zend_string *name = zend_ast_get_constant_name(ast);
- zval *zv = zend_get_constant_ex(name, scope, ast->attr);
- if (UNEXPECTED(zv == NULL)) {
- return zend_use_undefined_constant(name, ast->attr, p);
- }
- zval_ptr_dtor_nogc(p);
- ZVAL_COPY_OR_DUP(p, zv);
- } else {
- zval tmp;
- if (UNEXPECTED(zend_ast_evaluate(&tmp, ast, scope) != SUCCESS)) {
- return FAILURE;
- }
- zval_ptr_dtor_nogc(p);
- ZVAL_COPY_VALUE(p, &tmp);
- }
- }
- return SUCCESS;
- }
- ZEND_API int zval_update_constant(zval *pp)
- {
- return zval_update_constant_ex(pp, EG(current_execute_data) ? zend_get_executed_scope() : CG(active_class_entry));
- }
- int _call_user_function_ex(zval *object, zval *function_name, zval *retval_ptr, uint32_t param_count, zval params[], int no_separation)
- {
- zend_fcall_info fci;
- fci.size = sizeof(fci);
- fci.object = object ? Z_OBJ_P(object) : NULL;
- ZVAL_COPY_VALUE(&fci.function_name, function_name);
- fci.retval = retval_ptr;
- fci.param_count = param_count;
- fci.params = params;
- fci.no_separation = (zend_bool) no_separation;
- return zend_call_function(&fci, NULL);
- }
- int zend_call_function(zend_fcall_info *fci, zend_fcall_info_cache *fci_cache)
- {
- uint32_t i;
- zend_execute_data *call, dummy_execute_data;
- zend_fcall_info_cache fci_cache_local;
- zend_function *func;
- ZVAL_UNDEF(fci->retval);
- if (!EG(active)) {
- return FAILURE;
- }
- if (EG(exception)) {
- return FAILURE;
- }
- ZEND_ASSERT(fci->size == sizeof(zend_fcall_info));
-
- if (!EG(current_execute_data)) {
-
- memset(&dummy_execute_data, 0, sizeof(zend_execute_data));
- EG(current_execute_data) = &dummy_execute_data;
- } else if (EG(current_execute_data)->func &&
- ZEND_USER_CODE(EG(current_execute_data)->func->common.type) &&
- EG(current_execute_data)->opline->opcode != ZEND_DO_FCALL &&
- EG(current_execute_data)->opline->opcode != ZEND_DO_ICALL &&
- EG(current_execute_data)->opline->opcode != ZEND_DO_UCALL &&
- EG(current_execute_data)->opline->opcode != ZEND_DO_FCALL_BY_NAME) {
-
- dummy_execute_data = *EG(current_execute_data);
- dummy_execute_data.prev_execute_data = EG(current_execute_data);
- dummy_execute_data.call = NULL;
- dummy_execute_data.opline = NULL;
- dummy_execute_data.func = NULL;
- EG(current_execute_data) = &dummy_execute_data;
- }
- if (!fci_cache || !fci_cache->function_handler) {
- char *error = NULL;
- if (!fci_cache) {
- fci_cache = &fci_cache_local;
- }
- if (!zend_is_callable_ex(&fci->function_name, fci->object, IS_CALLABLE_CHECK_SILENT, NULL, fci_cache, &error)) {
- if (error) {
- zend_string *callable_name
- = zend_get_callable_name_ex(&fci->function_name, fci->object);
- zend_error(E_WARNING, "Invalid callback %s, %s", ZSTR_VAL(callable_name), error);
- efree(error);
- zend_string_release_ex(callable_name, 0);
- }
- if (EG(current_execute_data) == &dummy_execute_data) {
- EG(current_execute_data) = dummy_execute_data.prev_execute_data;
- }
- return FAILURE;
- } else if (error) {
-
- if (error[0] >= 'a' && error[0] <= 'z') {
- error[0] += ('A' - 'a');
- }
- zend_error(E_DEPRECATED, "%s", error);
- efree(error);
- if (UNEXPECTED(EG(exception))) {
- if (EG(current_execute_data) == &dummy_execute_data) {
- EG(current_execute_data) = dummy_execute_data.prev_execute_data;
- }
- return FAILURE;
- }
- }
- }
- func = fci_cache->function_handler;
- fci->object = (func->common.fn_flags & ZEND_ACC_STATIC) ?
- NULL : fci_cache->object;
- call = zend_vm_stack_push_call_frame(ZEND_CALL_TOP_FUNCTION | ZEND_CALL_DYNAMIC,
- func, fci->param_count, fci_cache->called_scope, fci->object);
- if (UNEXPECTED(func->common.fn_flags & ZEND_ACC_DEPRECATED)) {
- zend_error(E_DEPRECATED, "Function %s%s%s() is deprecated",
- func->common.scope ? ZSTR_VAL(func->common.scope->name) : "",
- func->common.scope ? "::" : "",
- ZSTR_VAL(func->common.function_name));
- if (UNEXPECTED(EG(exception))) {
- zend_vm_stack_free_call_frame(call);
- if (EG(current_execute_data) == &dummy_execute_data) {
- EG(current_execute_data) = dummy_execute_data.prev_execute_data;
- }
- return FAILURE;
- }
- }
- for (i=0; i<fci->param_count; i++) {
- zval *param;
- zval *arg = &fci->params[i];
- if (ARG_SHOULD_BE_SENT_BY_REF(func, i + 1)) {
- if (UNEXPECTED(!Z_ISREF_P(arg))) {
- if (!fci->no_separation) {
-
- ZVAL_NEW_REF(arg, arg);
- } else if (!ARG_MAY_BE_SENT_BY_REF(func, i + 1)) {
-
- zend_error(E_WARNING,
- "Parameter %d to %s%s%s() expected to be a reference, value given", i+1,
- func->common.scope ? ZSTR_VAL(func->common.scope->name) : "",
- func->common.scope ? "::" : "",
- ZSTR_VAL(func->common.function_name));
- if (UNEXPECTED(EG(exception))) {
- ZEND_CALL_NUM_ARGS(call) = i;
- zend_vm_stack_free_args(call);
- zend_vm_stack_free_call_frame(call);
- if (EG(current_execute_data) == &dummy_execute_data) {
- EG(current_execute_data) = dummy_execute_data.prev_execute_data;
- }
- return FAILURE;
- }
- }
- }
- } else {
- if (Z_ISREF_P(arg) &&
- !(func->common.fn_flags & ZEND_ACC_CALL_VIA_TRAMPOLINE)) {
-
- arg = Z_REFVAL_P(arg);
- }
- }
- param = ZEND_CALL_ARG(call, i+1);
- ZVAL_COPY(param, arg);
- }
- if (UNEXPECTED(func->op_array.fn_flags & ZEND_ACC_CLOSURE)) {
- uint32_t call_info;
- GC_ADDREF(ZEND_CLOSURE_OBJECT(func));
- call_info = ZEND_CALL_CLOSURE;
- if (func->common.fn_flags & ZEND_ACC_FAKE_CLOSURE) {
- call_info |= ZEND_CALL_FAKE_CLOSURE;
- }
- ZEND_ADD_CALL_FLAG(call, call_info);
- }
- if (func->type == ZEND_USER_FUNCTION) {
- int call_via_handler = (func->common.fn_flags & ZEND_ACC_CALL_VIA_TRAMPOLINE) != 0;
- const zend_op *current_opline_before_exception = EG(opline_before_exception);
- zend_init_func_execute_data(call, &func->op_array, fci->retval);
- zend_execute_ex(call);
- EG(opline_before_exception) = current_opline_before_exception;
- if (call_via_handler) {
-
- fci_cache->function_handler = NULL;
- }
- } else if (func->type == ZEND_INTERNAL_FUNCTION) {
- int call_via_handler = (func->common.fn_flags & ZEND_ACC_CALL_VIA_TRAMPOLINE) != 0;
- ZVAL_NULL(fci->retval);
- call->prev_execute_data = EG(current_execute_data);
- call->return_value = NULL;
- EG(current_execute_data) = call;
- if (EXPECTED(zend_execute_internal == NULL)) {
-
- func->internal_function.handler(call, fci->retval);
- } else {
- zend_execute_internal(call, fci->retval);
- }
- EG(current_execute_data) = call->prev_execute_data;
- zend_vm_stack_free_args(call);
- if (EG(exception)) {
- zval_ptr_dtor(fci->retval);
- ZVAL_UNDEF(fci->retval);
- }
- if (call_via_handler) {
-
- fci_cache->function_handler = NULL;
- }
- } else {
- ZVAL_NULL(fci->retval);
-
- if (fci->object) {
- call->prev_execute_data = EG(current_execute_data);
- EG(current_execute_data) = call;
- fci->object->handlers->call_method(func->common.function_name, fci->object, call, fci->retval);
- EG(current_execute_data) = call->prev_execute_data;
- } else {
- zend_throw_error(NULL, "Cannot call overloaded function for non-object");
- }
- zend_vm_stack_free_args(call);
- if (func->type == ZEND_OVERLOADED_FUNCTION_TEMPORARY) {
- zend_string_release_ex(func->common.function_name, 0);
- }
- efree(func);
- if (EG(exception)) {
- zval_ptr_dtor(fci->retval);
- ZVAL_UNDEF(fci->retval);
- }
- }
- zend_vm_stack_free_call_frame(call);
- if (EG(current_execute_data) == &dummy_execute_data) {
- EG(current_execute_data) = dummy_execute_data.prev_execute_data;
- }
- if (UNEXPECTED(EG(exception))) {
- if (UNEXPECTED(!EG(current_execute_data))) {
- zend_throw_exception_internal(NULL);
- } else if (EG(current_execute_data)->func &&
- ZEND_USER_CODE(EG(current_execute_data)->func->common.type)) {
- zend_rethrow_exception(EG(current_execute_data));
- }
- }
- return SUCCESS;
- }
- ZEND_API zend_class_entry *zend_lookup_class_ex(zend_string *name, const zval *key, int use_autoload)
- {
- zend_class_entry *ce = NULL;
- zval args[1], *zv;
- zval local_retval;
- zend_string *lc_name;
- zend_fcall_info fcall_info;
- zend_fcall_info_cache fcall_cache;
- zend_class_entry *orig_fake_scope;
- if (key) {
- lc_name = Z_STR_P(key);
- } else {
- if (name == NULL || !ZSTR_LEN(name)) {
- return NULL;
- }
- if (ZSTR_VAL(name)[0] == '\\') {
- lc_name = zend_string_alloc(ZSTR_LEN(name) - 1, 0);
- zend_str_tolower_copy(ZSTR_VAL(lc_name), ZSTR_VAL(name) + 1, ZSTR_LEN(name) - 1);
- } else {
- lc_name = zend_string_tolower(name);
- }
- }
- zv = zend_hash_find(EG(class_table), lc_name);
- if (zv) {
- if (!key) {
- zend_string_release_ex(lc_name, 0);
- }
- return (zend_class_entry*)Z_PTR_P(zv);
- }
-
- if (!use_autoload || zend_is_compiling()) {
- if (!key) {
- zend_string_release_ex(lc_name, 0);
- }
- return NULL;
- }
- if (!EG(autoload_func)) {
- zend_function *func = zend_fetch_function(ZSTR_KNOWN(ZEND_STR_MAGIC_AUTOLOAD));
- if (func) {
- EG(autoload_func) = func;
- } else {
- if (!key) {
- zend_string_release_ex(lc_name, 0);
- }
- return NULL;
- }
- }
-
- if (!key && strspn(ZSTR_VAL(name), "0123456789_abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ\200\201\202\203\204\205\206\207\210\211\212\213\214\215\216\217\220\221\222\223\224\225\226\227\230\231\232\233\234\235\236\237\240\241\242\243\244\245\246\247\250\251\252\253\254\255\256\257\260\261\262\263\264\265\266\267\270\271\272\273\274\275\276\277\300\301\302\303\304\305\306\307\310\311\312\313\314\315\316\317\320\321\322\323\324\325\326\327\330\331\332\333\334\335\336\337\340\341\342\343\344\345\346\347\350\351\352\353\354\355\356\357\360\361\362\363\364\365\366\367\370\371\372\373\374\375\376\377\\") != ZSTR_LEN(name)) {
- zend_string_release_ex(lc_name, 0);
- return NULL;
- }
- if (EG(in_autoload) == NULL) {
- ALLOC_HASHTABLE(EG(in_autoload));
- zend_hash_init(EG(in_autoload), 8, NULL, NULL, 0);
- }
- if (zend_hash_add_empty_element(EG(in_autoload), lc_name) == NULL) {
- if (!key) {
- zend_string_release_ex(lc_name, 0);
- }
- return NULL;
- }
- ZVAL_UNDEF(&local_retval);
- if (ZSTR_VAL(name)[0] == '\\') {
- ZVAL_STRINGL(&args[0], ZSTR_VAL(name) + 1, ZSTR_LEN(name) - 1);
- } else {
- ZVAL_STR_COPY(&args[0], name);
- }
- fcall_info.size = sizeof(fcall_info);
- ZVAL_STR_COPY(&fcall_info.function_name, EG(autoload_func)->common.function_name);
- fcall_info.retval = &local_retval;
- fcall_info.param_count = 1;
- fcall_info.params = args;
- fcall_info.object = NULL;
- fcall_info.no_separation = 1;
- fcall_cache.function_handler = EG(autoload_func);
- fcall_cache.called_scope = NULL;
- fcall_cache.object = NULL;
- orig_fake_scope = EG(fake_scope);
- EG(fake_scope) = NULL;
- zend_exception_save();
- if ((zend_call_function(&fcall_info, &fcall_cache) == SUCCESS) && !EG(exception)) {
- ce = zend_hash_find_ptr(EG(class_table), lc_name);
- }
- zend_exception_restore();
- EG(fake_scope) = orig_fake_scope;
- zval_ptr_dtor(&args[0]);
- zval_ptr_dtor_str(&fcall_info.function_name);
- zend_hash_del(EG(in_autoload), lc_name);
- zval_ptr_dtor(&local_retval);
- if (!key) {
- zend_string_release_ex(lc_name, 0);
- }
- return ce;
- }
- ZEND_API zend_class_entry *zend_lookup_class(zend_string *name)
- {
- return zend_lookup_class_ex(name, NULL, 1);
- }
- ZEND_API zend_class_entry *zend_get_called_scope(zend_execute_data *ex)
- {
- while (ex) {
- if (Z_TYPE(ex->This) == IS_OBJECT) {
- return Z_OBJCE(ex->This);
- } else if (Z_CE(ex->This)) {
- return Z_CE(ex->This);
- } else if (ex->func) {
- if (ex->func->type != ZEND_INTERNAL_FUNCTION || ex->func->common.scope) {
- return NULL;
- }
- }
- ex = ex->prev_execute_data;
- }
- return NULL;
- }
- ZEND_API zend_object *zend_get_this_object(zend_execute_data *ex)
- {
- while (ex) {
- if (Z_TYPE(ex->This) == IS_OBJECT) {
- return Z_OBJ(ex->This);
- } else if (ex->func) {
- if (ex->func->type != ZEND_INTERNAL_FUNCTION || ex->func->common.scope) {
- return NULL;
- }
- }
- ex = ex->prev_execute_data;
- }
- return NULL;
- }
- ZEND_API int zend_eval_stringl(char *str, size_t str_len, zval *retval_ptr, char *string_name)
- {
- zval pv;
- zend_op_array *new_op_array;
- uint32_t original_compiler_options;
- int retval;
- if (retval_ptr) {
- ZVAL_NEW_STR(&pv, zend_string_alloc(str_len + sizeof("return ;")-1, 0));
- memcpy(Z_STRVAL(pv), "return ", sizeof("return ") - 1);
- memcpy(Z_STRVAL(pv) + sizeof("return ") - 1, str, str_len);
- Z_STRVAL(pv)[Z_STRLEN(pv) - 1] = ';';
- Z_STRVAL(pv)[Z_STRLEN(pv)] = '\0';
- } else {
- ZVAL_STRINGL(&pv, str, str_len);
- }
-
- original_compiler_options = CG(compiler_options);
- CG(compiler_options) = ZEND_COMPILE_DEFAULT_FOR_EVAL;
- new_op_array = zend_compile_string(&pv, string_name);
- CG(compiler_options) = original_compiler_options;
- if (new_op_array) {
- zval local_retval;
- EG(no_extensions)=1;
- new_op_array->scope = zend_get_executed_scope();
- zend_try {
- ZVAL_UNDEF(&local_retval);
- zend_execute(new_op_array, &local_retval);
- } zend_catch {
- destroy_op_array(new_op_array);
- efree_size(new_op_array, sizeof(zend_op_array));
- zend_bailout();
- } zend_end_try();
- if (Z_TYPE(local_retval) != IS_UNDEF) {
- if (retval_ptr) {
- ZVAL_COPY_VALUE(retval_ptr, &local_retval);
- } else {
- zval_ptr_dtor(&local_retval);
- }
- } else {
- if (retval_ptr) {
- ZVAL_NULL(retval_ptr);
- }
- }
- EG(no_extensions)=0;
- destroy_op_array(new_op_array);
- efree_size(new_op_array, sizeof(zend_op_array));
- retval = SUCCESS;
- } else {
- retval = FAILURE;
- }
- zval_ptr_dtor_str(&pv);
- return retval;
- }
- ZEND_API int zend_eval_string(char *str, zval *retval_ptr, char *string_name)
- {
- return zend_eval_stringl(str, strlen(str), retval_ptr, string_name);
- }
- ZEND_API int zend_eval_stringl_ex(char *str, size_t str_len, zval *retval_ptr, char *string_name, int handle_exceptions)
- {
- int result;
- result = zend_eval_stringl(str, str_len, retval_ptr, string_name);
- if (handle_exceptions && EG(exception)) {
- zend_exception_error(EG(exception), E_ERROR);
- result = FAILURE;
- }
- return result;
- }
- ZEND_API int zend_eval_string_ex(char *str, zval *retval_ptr, char *string_name, int handle_exceptions)
- {
- return zend_eval_stringl_ex(str, strlen(str), retval_ptr, string_name, handle_exceptions);
- }
- static void zend_set_timeout_ex(zend_long seconds, int reset_signals);
- ZEND_API ZEND_NORETURN void zend_timeout(int dummy)
- {
- #if defined(PHP_WIN32)
- # ifndef ZTS
-
- if (EG(hard_timeout) > 0) {
- EG(timed_out) = 0;
- zend_set_timeout_ex(EG(hard_timeout), 1);
-
- EG(hard_timeout) = 0;
- }
- # endif
- #else
- EG(timed_out) = 0;
- zend_set_timeout_ex(0, 1);
- #endif
- zend_error_noreturn(E_ERROR, "Maximum execution time of " ZEND_LONG_FMT " second%s exceeded", EG(timeout_seconds), EG(timeout_seconds) == 1 ? "" : "s");
- }
- #ifndef ZEND_WIN32
- static void zend_timeout_handler(int dummy)
- {
- #ifndef ZTS
- if (EG(timed_out)) {
-
- const char *error_filename = NULL;
- uint32_t error_lineno = 0;
- char log_buffer[2048];
- int output_len = 0;
- if (zend_is_compiling()) {
- error_filename = ZSTR_VAL(zend_get_compiled_filename());
- error_lineno = zend_get_compiled_lineno();
- } else if (zend_is_executing()) {
- error_filename = zend_get_executed_filename();
- if (error_filename[0] == '[') {
- error_filename = NULL;
- error_lineno = 0;
- } else {
- error_lineno = zend_get_executed_lineno();
- }
- }
- if (!error_filename) {
- error_filename = "Unknown";
- }
- output_len = snprintf(log_buffer, sizeof(log_buffer), "\nFatal error: Maximum execution time of " ZEND_LONG_FMT "+" ZEND_LONG_FMT " seconds exceeded (terminated) in %s on line %d\n", EG(timeout_seconds), EG(hard_timeout), error_filename, error_lineno);
- if (output_len > 0) {
- write(2, log_buffer, MIN(output_len, sizeof(log_buffer)));
- }
- _exit(124);
- }
- #endif
- if (zend_on_timeout) {
- #ifdef ZEND_SIGNALS
-
- SIGG(running) = 0;
- #endif
- zend_on_timeout(EG(timeout_seconds));
- }
- EG(timed_out) = 1;
- EG(vm_interrupt) = 1;
- #ifndef ZTS
- if (EG(hard_timeout) > 0) {
-
- zend_set_timeout_ex(EG(hard_timeout), 1);
- }
- #endif
- }
- #endif
- #ifdef ZEND_WIN32
- VOID CALLBACK tq_timer_cb(PVOID arg, BOOLEAN timed_out)
- {
- zend_executor_globals *eg;
-
- if (!timed_out) {
- return;
- }
- eg = (zend_executor_globals *)arg;
- eg->timed_out = 1;
- eg->vm_interrupt = 1;
- }
- #endif
- #ifndef SIGPROF
- #define SIGPROF 27
- #endif
- static void zend_set_timeout_ex(zend_long seconds, int reset_signals)
- {
- #ifdef ZEND_WIN32
- zend_executor_globals *eg;
- if(!seconds) {
- return;
- }
-
- if (NULL != tq_timer) {
- if (!DeleteTimerQueueTimer(NULL, tq_timer, NULL)) {
- tq_timer = NULL;
- zend_error_noreturn(E_ERROR, "Could not delete queued timer");
- return;
- }
- tq_timer = NULL;
- }
-
- eg = ZEND_MODULE_GLOBALS_BULK(executor);
- if (!CreateTimerQueueTimer(&tq_timer, NULL, (WAITORTIMERCALLBACK)tq_timer_cb, (VOID*)eg, seconds*1000, 0, WT_EXECUTEONLYONCE)) {
- tq_timer = NULL;
- zend_error_noreturn(E_ERROR, "Could not queue new timer");
- return;
- }
- #else
- # ifdef HAVE_SETITIMER
- {
- struct itimerval t_r;
- int signo;
- if(seconds) {
- t_r.it_value.tv_sec = seconds;
- t_r.it_value.tv_usec = t_r.it_interval.tv_sec = t_r.it_interval.tv_usec = 0;
- # ifdef __CYGWIN__
- setitimer(ITIMER_REAL, &t_r, NULL);
- }
- signo = SIGALRM;
- # else
- setitimer(ITIMER_PROF, &t_r, NULL);
- }
- signo = SIGPROF;
- # endif
- if (reset_signals) {
- # ifdef ZEND_SIGNALS
- zend_signal(signo, zend_timeout_handler);
- # else
- sigset_t sigset;
- # ifdef HAVE_SIGACTION
- struct sigaction act;
- act.sa_handler = zend_timeout_handler;
- sigemptyset(&act.sa_mask);
- act.sa_flags = SA_RESETHAND | SA_NODEFER;
- sigaction(signo, &act, NULL);
- # else
- signal(signo, zend_timeout_handler);
- # endif
- sigemptyset(&sigset);
- sigaddset(&sigset, signo);
- sigprocmask(SIG_UNBLOCK, &sigset, NULL);
- # endif
- }
- }
- # endif
- #endif
- }
- void zend_set_timeout(zend_long seconds, int reset_signals)
- {
- EG(timeout_seconds) = seconds;
- zend_set_timeout_ex(seconds, reset_signals);
- EG(timed_out) = 0;
- }
- void zend_unset_timeout(void)
- {
- #ifdef ZEND_WIN32
- if (NULL != tq_timer) {
- if (!DeleteTimerQueueTimer(NULL, tq_timer, NULL)) {
- EG(timed_out) = 0;
- tq_timer = NULL;
- zend_error_noreturn(E_ERROR, "Could not delete queued timer");
- return;
- }
- tq_timer = NULL;
- }
- EG(timed_out) = 0;
- #else
- # ifdef HAVE_SETITIMER
- if (EG(timeout_seconds)) {
- struct itimerval no_timeout;
- no_timeout.it_value.tv_sec = no_timeout.it_value.tv_usec = no_timeout.it_interval.tv_sec = no_timeout.it_interval.tv_usec = 0;
- #ifdef __CYGWIN__
- setitimer(ITIMER_REAL, &no_timeout, NULL);
- #else
- setitimer(ITIMER_PROF, &no_timeout, NULL);
- #endif
- }
- # endif
- EG(timed_out) = 0;
- #endif
- }
- zend_class_entry *zend_fetch_class(zend_string *class_name, int fetch_type)
- {
- zend_class_entry *ce, *scope;
- int fetch_sub_type = fetch_type & ZEND_FETCH_CLASS_MASK;
- check_fetch_type:
- switch (fetch_sub_type) {
- case ZEND_FETCH_CLASS_SELF:
- scope = zend_get_executed_scope();
- if (UNEXPECTED(!scope)) {
- zend_throw_or_error(fetch_type, NULL, "Cannot access self:: when no class scope is active");
- }
- return scope;
- case ZEND_FETCH_CLASS_PARENT:
- scope = zend_get_executed_scope();
- if (UNEXPECTED(!scope)) {
- zend_throw_or_error(fetch_type, NULL, "Cannot access parent:: when no class scope is active");
- return NULL;
- }
- if (UNEXPECTED(!scope->parent)) {
- zend_throw_or_error(fetch_type, NULL, "Cannot access parent:: when current class scope has no parent");
- }
- return scope->parent;
- case ZEND_FETCH_CLASS_STATIC:
- ce = zend_get_called_scope(EG(current_execute_data));
- if (UNEXPECTED(!ce)) {
- zend_throw_or_error(fetch_type, NULL, "Cannot access static:: when no class scope is active");
- return NULL;
- }
- return ce;
- case ZEND_FETCH_CLASS_AUTO: {
- fetch_sub_type = zend_get_class_fetch_type(class_name);
- if (UNEXPECTED(fetch_sub_type != ZEND_FETCH_CLASS_DEFAULT)) {
- goto check_fetch_type;
- }
- }
- break;
- }
- if (fetch_type & ZEND_FETCH_CLASS_NO_AUTOLOAD) {
- return zend_lookup_class_ex(class_name, NULL, 0);
- } else if ((ce = zend_lookup_class_ex(class_name, NULL, 1)) == NULL) {
- if (!(fetch_type & ZEND_FETCH_CLASS_SILENT) && !EG(exception)) {
- if (fetch_sub_type == ZEND_FETCH_CLASS_INTERFACE) {
- zend_throw_or_error(fetch_type, NULL, "Interface '%s' not found", ZSTR_VAL(class_name));
- } else if (fetch_sub_type == ZEND_FETCH_CLASS_TRAIT) {
- zend_throw_or_error(fetch_type, NULL, "Trait '%s' not found", ZSTR_VAL(class_name));
- } else {
- zend_throw_or_error(fetch_type, NULL, "Class '%s' not found", ZSTR_VAL(class_name));
- }
- }
- return NULL;
- }
- return ce;
- }
- zend_class_entry *zend_fetch_class_by_name(zend_string *class_name, const zval *key, int fetch_type)
- {
- zend_class_entry *ce;
- if (fetch_type & ZEND_FETCH_CLASS_NO_AUTOLOAD) {
- return zend_lookup_class_ex(class_name, key, 0);
- } else if ((ce = zend_lookup_class_ex(class_name, key, 1)) == NULL) {
- if ((fetch_type & ZEND_FETCH_CLASS_SILENT) == 0 && !EG(exception)) {
- if ((fetch_type & ZEND_FETCH_CLASS_MASK) == ZEND_FETCH_CLASS_INTERFACE) {
- zend_throw_or_error(fetch_type, NULL, "Interface '%s' not found", ZSTR_VAL(class_name));
- } else if ((fetch_type & ZEND_FETCH_CLASS_MASK) == ZEND_FETCH_CLASS_TRAIT) {
- zend_throw_or_error(fetch_type, NULL, "Trait '%s' not found", ZSTR_VAL(class_name));
- } else {
- zend_throw_or_error(fetch_type, NULL, "Class '%s' not found", ZSTR_VAL(class_name));
- }
- }
- return NULL;
- }
- return ce;
- }
- #define MAX_ABSTRACT_INFO_CNT 3
- #define MAX_ABSTRACT_INFO_FMT "%s%s%s%s"
- #define DISPLAY_ABSTRACT_FN(idx) \
- ai.afn[idx] ? ZEND_FN_SCOPE_NAME(ai.afn[idx]) : "", \
- ai.afn[idx] ? "::" : "", \
- ai.afn[idx] ? ZSTR_VAL(ai.afn[idx]->common.function_name) : "", \
- ai.afn[idx] && ai.afn[idx + 1] ? ", " : (ai.afn[idx] && ai.cnt > MAX_ABSTRACT_INFO_CNT ? ", ..." : "")
- typedef struct _zend_abstract_info {
- zend_function *afn[MAX_ABSTRACT_INFO_CNT + 1];
- int cnt;
- int ctor;
- } zend_abstract_info;
- static void zend_verify_abstract_class_function(zend_function *fn, zend_abstract_info *ai)
- {
- if (fn->common.fn_flags & ZEND_ACC_ABSTRACT) {
- if (ai->cnt < MAX_ABSTRACT_INFO_CNT) {
- ai->afn[ai->cnt] = fn;
- }
- if (fn->common.fn_flags & ZEND_ACC_CTOR) {
- if (!ai->ctor) {
- ai->cnt++;
- ai->ctor = 1;
- } else {
- ai->afn[ai->cnt] = NULL;
- }
- } else {
- ai->cnt++;
- }
- }
- }
- void zend_verify_abstract_class(zend_class_entry *ce)
- {
- zend_function *func;
- zend_abstract_info ai;
- if ((ce->ce_flags & ZEND_ACC_IMPLICIT_ABSTRACT_CLASS) && !(ce->ce_flags & (ZEND_ACC_TRAIT | ZEND_ACC_EXPLICIT_ABSTRACT_CLASS))) {
- memset(&ai, 0, sizeof(ai));
- ZEND_HASH_FOREACH_PTR(&ce->function_table, func) {
- zend_verify_abstract_class_function(func, &ai);
- } ZEND_HASH_FOREACH_END();
- if (ai.cnt) {
- zend_error_noreturn(E_ERROR, "Class %s contains %d abstract method%s and must therefore be declared abstract or implement the remaining methods (" MAX_ABSTRACT_INFO_FMT MAX_ABSTRACT_INFO_FMT MAX_ABSTRACT_INFO_FMT ")",
- ZSTR_VAL(ce->name), ai.cnt,
- ai.cnt > 1 ? "s" : "",
- DISPLAY_ABSTRACT_FN(0),
- DISPLAY_ABSTRACT_FN(1),
- DISPLAY_ABSTRACT_FN(2)
- );
- }
- }
- }
- ZEND_API int zend_delete_global_variable(zend_string *name)
- {
- return zend_hash_del_ind(&EG(symbol_table), name);
- }
- ZEND_API zend_array *zend_rebuild_symbol_table(void)
- {
- zend_execute_data *ex;
- zend_array *symbol_table;
-
- ex = EG(current_execute_data);
- while (ex && (!ex->func || !ZEND_USER_CODE(ex->func->common.type))) {
- ex = ex->prev_execute_data;
- }
- if (!ex) {
- return NULL;
- }
- if (ZEND_CALL_INFO(ex) & ZEND_CALL_HAS_SYMBOL_TABLE) {
- return ex->symbol_table;
- }
- ZEND_ADD_CALL_FLAG(ex, ZEND_CALL_HAS_SYMBOL_TABLE);
- if (EG(symtable_cache_ptr) >= EG(symtable_cache)) {
-
- symbol_table = ex->symbol_table = *(EG(symtable_cache_ptr)--);
- if (!ex->func->op_array.last_var) {
- return symbol_table;
- }
- zend_hash_extend(symbol_table, ex->func->op_array.last_var, 0);
- } else {
- symbol_table = ex->symbol_table = zend_new_array(ex->func->op_array.last_var);
- if (!ex->func->op_array.last_var) {
- return symbol_table;
- }
- zend_hash_real_init_mixed(symbol_table);
-
- }
- if (EXPECTED(ex->func->op_array.last_var)) {
- zend_string **str = ex->func->op_array.vars;
- zend_string **end = str + ex->func->op_array.last_var;
- zval *var = ZEND_CALL_VAR_NUM(ex, 0);
- do {
- _zend_hash_append_ind(symbol_table, *str, var);
- str++;
- var++;
- } while (str != end);
- }
- return symbol_table;
- }
- ZEND_API void zend_attach_symbol_table(zend_execute_data *execute_data)
- {
- zend_op_array *op_array = &execute_data->func->op_array;
- HashTable *ht = execute_data->symbol_table;
-
- if (EXPECTED(op_array->last_var)) {
- zend_string **str = op_array->vars;
- zend_string **end = str + op_array->last_var;
- zval *var = EX_VAR_NUM(0);
- do {
- zval *zv = zend_hash_find_ex(ht, *str, 1);
- if (zv) {
- if (Z_TYPE_P(zv) == IS_INDIRECT) {
- zval *val = Z_INDIRECT_P(zv);
- ZVAL_COPY_VALUE(var, val);
- } else {
- ZVAL_COPY_VALUE(var, zv);
- }
- } else {
- ZVAL_UNDEF(var);
- zv = zend_hash_add_new(ht, *str, var);
- }
- ZVAL_INDIRECT(zv, var);
- str++;
- var++;
- } while (str != end);
- }
- }
- ZEND_API void zend_detach_symbol_table(zend_execute_data *execute_data)
- {
- zend_op_array *op_array = &execute_data->func->op_array;
- HashTable *ht = execute_data->symbol_table;
-
- if (EXPECTED(op_array->last_var)) {
- zend_string **str = op_array->vars;
- zend_string **end = str + op_array->last_var;
- zval *var = EX_VAR_NUM(0);
- do {
- if (Z_TYPE_P(var) == IS_UNDEF) {
- zend_hash_del(ht, *str);
- } else {
- zend_hash_update(ht, *str, var);
- ZVAL_UNDEF(var);
- }
- str++;
- var++;
- } while (str != end);
- }
- }
- ZEND_API int zend_set_local_var(zend_string *name, zval *value, int force)
- {
- zend_execute_data *execute_data = EG(current_execute_data);
- while (execute_data && (!execute_data->func || !ZEND_USER_CODE(execute_data->func->common.type))) {
- execute_data = execute_data->prev_execute_data;
- }
- if (execute_data) {
- if (!(EX_CALL_INFO() & ZEND_CALL_HAS_SYMBOL_TABLE)) {
- zend_ulong h = zend_string_hash_val(name);
- zend_op_array *op_array = &execute_data->func->op_array;
- if (EXPECTED(op_array->last_var)) {
- zend_string **str = op_array->vars;
- zend_string **end = str + op_array->last_var;
- do {
- if (ZSTR_H(*str) == h &&
- zend_string_equal_content(*str, name)) {
- zval *var = EX_VAR_NUM(str - op_array->vars);
- ZVAL_COPY_VALUE(var, value);
- return SUCCESS;
- }
- str++;
- } while (str != end);
- }
- if (force) {
- zend_array *symbol_table = zend_rebuild_symbol_table();
- if (symbol_table) {
- zend_hash_update(symbol_table, name, value);
- return SUCCESS;
- }
- }
- } else {
- zend_hash_update_ind(execute_data->symbol_table, name, value);
- return SUCCESS;
- }
- }
- return FAILURE;
- }
- ZEND_API int zend_set_local_var_str(const char *name, size_t len, zval *value, int force)
- {
- zend_execute_data *execute_data = EG(current_execute_data);
- while (execute_data && (!execute_data->func || !ZEND_USER_CODE(execute_data->func->common.type))) {
- execute_data = execute_data->prev_execute_data;
- }
- if (execute_data) {
- if (!(EX_CALL_INFO() & ZEND_CALL_HAS_SYMBOL_TABLE)) {
- zend_ulong h = zend_hash_func(name, len);
- zend_op_array *op_array = &execute_data->func->op_array;
- if (EXPECTED(op_array->last_var)) {
- zend_string **str = op_array->vars;
- zend_string **end = str + op_array->last_var;
- do {
- if (ZSTR_H(*str) == h &&
- ZSTR_LEN(*str) == len &&
- memcmp(ZSTR_VAL(*str), name, len) == 0) {
- zval *var = EX_VAR_NUM(str - op_array->vars);
- zval_ptr_dtor(var);
- ZVAL_COPY_VALUE(var, value);
- return SUCCESS;
- }
- str++;
- } while (str != end);
- }
- if (force) {
- zend_array *symbol_table = zend_rebuild_symbol_table();
- if (symbol_table) {
- zend_hash_str_update(symbol_table, name, len, value);
- return SUCCESS;
- }
- }
- } else {
- zend_hash_str_update_ind(execute_data->symbol_table, name, len, value);
- return SUCCESS;
- }
- }
- return FAILURE;
- }
- ZEND_API int zend_forbid_dynamic_call(const char *func_name)
- {
- zend_execute_data *ex = EG(current_execute_data);
- ZEND_ASSERT(ex != NULL && ex->func != NULL);
- if (ZEND_CALL_INFO(ex) & ZEND_CALL_DYNAMIC) {
- zend_error(E_WARNING, "Cannot call %s dynamically", func_name);
- return FAILURE;
- }
- return SUCCESS;
- }
|