12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262 |
- #ifdef HAVE_CONFIG_H
- # include "config.h"
- #endif
- #include "php.h"
- #include "zend_exceptions.h"
- #include "php_spl.h"
- #include "spl_functions.h"
- #include "spl_engine.h"
- #include "spl_iterators.h"
- #include "spl_heap.h"
- #include "spl_exceptions.h"
- #define PTR_HEAP_BLOCK_SIZE 64
- #define SPL_HEAP_CORRUPTED 0x00000001
- #define SPL_PQUEUE_EXTR_MASK 0x00000003
- #define SPL_PQUEUE_EXTR_BOTH 0x00000003
- #define SPL_PQUEUE_EXTR_DATA 0x00000001
- #define SPL_PQUEUE_EXTR_PRIORITY 0x00000002
- zend_object_handlers spl_handler_SplHeap;
- zend_object_handlers spl_handler_SplPriorityQueue;
- PHPAPI zend_class_entry *spl_ce_SplHeap;
- PHPAPI zend_class_entry *spl_ce_SplMaxHeap;
- PHPAPI zend_class_entry *spl_ce_SplMinHeap;
- PHPAPI zend_class_entry *spl_ce_SplPriorityQueue;
- typedef void *spl_ptr_heap_element;
- typedef void (*spl_ptr_heap_dtor_func)(spl_ptr_heap_element TSRMLS_DC);
- typedef void (*spl_ptr_heap_ctor_func)(spl_ptr_heap_element TSRMLS_DC);
- typedef int (*spl_ptr_heap_cmp_func)(spl_ptr_heap_element, spl_ptr_heap_element, void* TSRMLS_DC);
- typedef struct _spl_ptr_heap {
- spl_ptr_heap_element *elements;
- spl_ptr_heap_ctor_func ctor;
- spl_ptr_heap_dtor_func dtor;
- spl_ptr_heap_cmp_func cmp;
- int count;
- int max_size;
- int flags;
- } spl_ptr_heap;
- typedef struct _spl_heap_object spl_heap_object;
- typedef struct _spl_heap_it spl_heap_it;
- struct _spl_heap_object {
- zend_object std;
- spl_ptr_heap *heap;
- zval *retval;
- int flags;
- zend_class_entry *ce_get_iterator;
- zend_function *fptr_cmp;
- zend_function *fptr_count;
- HashTable *debug_info;
- };
- struct _spl_heap_it {
- zend_user_iterator intern;
- int flags;
- spl_heap_object *object;
- };
- static void spl_ptr_heap_zval_dtor(spl_ptr_heap_element elem TSRMLS_DC) {
- if (elem) {
- zval_ptr_dtor((zval **)&elem);
- }
- }
- static void spl_ptr_heap_zval_ctor(spl_ptr_heap_element elem TSRMLS_DC) {
- Z_ADDREF_P((zval *)elem);
- }
- static int spl_ptr_heap_cmp_cb_helper(zval *object, spl_heap_object *heap_object, zval *a, zval *b, long *result TSRMLS_DC) {
- zval *result_p = NULL;
- zend_call_method_with_2_params(&object, heap_object->std.ce, &heap_object->fptr_cmp, "compare", &result_p, a, b);
- if (EG(exception)) {
- return FAILURE;
- }
- convert_to_long(result_p);
- *result = Z_LVAL_P(result_p);
- zval_ptr_dtor(&result_p);
- return SUCCESS;
- }
- static zval **spl_pqueue_extract_helper(zval **value, int flags)
- {
- if ((flags & SPL_PQUEUE_EXTR_BOTH) == SPL_PQUEUE_EXTR_BOTH) {
- return value;
- } else if ((flags & SPL_PQUEUE_EXTR_BOTH) > 0) {
- if ((flags & SPL_PQUEUE_EXTR_DATA) == SPL_PQUEUE_EXTR_DATA) {
- zval **data;
- if (zend_hash_find(Z_ARRVAL_PP(value), "data", sizeof("data"), (void **) &data) == SUCCESS) {
- return data;
- }
- } else {
- zval **priority;
- if (zend_hash_find(Z_ARRVAL_PP(value), "priority", sizeof("priority"), (void **) &priority) == SUCCESS) {
- return priority;
- }
- }
- }
- return NULL;
- }
- static int spl_ptr_heap_zval_max_cmp(spl_ptr_heap_element a, spl_ptr_heap_element b, void* object TSRMLS_DC) {
- zval result;
- if (EG(exception)) {
- return 0;
- }
- if (object) {
- spl_heap_object *heap_object = (spl_heap_object*)zend_object_store_get_object((zval *)object TSRMLS_CC);
- if (heap_object->fptr_cmp) {
- long lval = 0;
- if (spl_ptr_heap_cmp_cb_helper((zval *)object, heap_object, (zval *)a, (zval *)b, &lval TSRMLS_CC) == FAILURE) {
-
- return 0;
- }
- return lval;
- }
- }
- INIT_ZVAL(result);
- compare_function(&result, (zval *)a, (zval *)b TSRMLS_CC);
- return Z_LVAL(result);
- }
- static int spl_ptr_heap_zval_min_cmp(spl_ptr_heap_element a, spl_ptr_heap_element b, void* object TSRMLS_DC) {
- zval result;
- if (EG(exception)) {
- return 0;
- }
- if (object) {
- spl_heap_object *heap_object = (spl_heap_object*)zend_object_store_get_object((zval *)object TSRMLS_CC);
- if (heap_object->fptr_cmp) {
- long lval = 0;
- if (spl_ptr_heap_cmp_cb_helper((zval *)object, heap_object, (zval *)a, (zval *)b, &lval TSRMLS_CC) == FAILURE) {
-
- return 0;
- }
- return lval;
- }
- }
- INIT_ZVAL(result);
- compare_function(&result, (zval *)b, (zval *)a TSRMLS_CC);
- return Z_LVAL(result);
- }
- static int spl_ptr_pqueue_zval_cmp(spl_ptr_heap_element a, spl_ptr_heap_element b, void* object TSRMLS_DC) {
- zval result;
- zval **a_priority_pp = spl_pqueue_extract_helper((zval **)&a, SPL_PQUEUE_EXTR_PRIORITY);
- zval **b_priority_pp = spl_pqueue_extract_helper((zval **)&b, SPL_PQUEUE_EXTR_PRIORITY);
- if ((!a_priority_pp) || (!b_priority_pp)) {
- zend_error(E_RECOVERABLE_ERROR, "Unable to extract from the PriorityQueue node");
- return 0;
- }
- if (EG(exception)) {
- return 0;
- }
- if (object) {
- spl_heap_object *heap_object = (spl_heap_object*)zend_object_store_get_object(object TSRMLS_CC);
- if (heap_object->fptr_cmp) {
- long lval = 0;
- if (spl_ptr_heap_cmp_cb_helper((zval *)object, heap_object, *a_priority_pp, *b_priority_pp, &lval TSRMLS_CC) == FAILURE) {
-
- return 0;
- }
- return lval;
- }
- }
- INIT_ZVAL(result);
- compare_function(&result, *a_priority_pp, *b_priority_pp TSRMLS_CC);
- return Z_LVAL(result);
- }
- static spl_ptr_heap *spl_ptr_heap_init(spl_ptr_heap_cmp_func cmp, spl_ptr_heap_ctor_func ctor, spl_ptr_heap_dtor_func dtor)
- {
- spl_ptr_heap *heap = emalloc(sizeof(spl_ptr_heap));
- heap->dtor = dtor;
- heap->ctor = ctor;
- heap->cmp = cmp;
- heap->elements = safe_emalloc(sizeof(spl_ptr_heap_element), PTR_HEAP_BLOCK_SIZE, 0);
- heap->max_size = PTR_HEAP_BLOCK_SIZE;
- heap->count = 0;
- heap->flags = 0;
- return heap;
- }
- static void spl_ptr_heap_insert(spl_ptr_heap *heap, spl_ptr_heap_element elem, void *cmp_userdata TSRMLS_DC) {
- int i;
- if (heap->count+1 > heap->max_size) {
-
- heap->elements = (void **) safe_erealloc(heap->elements, sizeof(spl_ptr_heap_element), (heap->max_size), (sizeof(spl_ptr_heap_element) * (heap->max_size)));
- heap->max_size *= 2;
- }
- heap->ctor(elem TSRMLS_CC);
-
- for(i = heap->count; i > 0 && heap->cmp(heap->elements[(i-1)/2], elem, cmp_userdata TSRMLS_CC) < 0; i = (i-1)/2) {
- heap->elements[i] = heap->elements[(i-1)/2];
- }
- heap->count++;
- if (EG(exception)) {
-
- heap->flags |= SPL_HEAP_CORRUPTED;
- }
- heap->elements[i] = elem;
- }
- static spl_ptr_heap_element spl_ptr_heap_top(spl_ptr_heap *heap) {
- if (heap->count == 0) {
- return NULL;
- }
- return heap->elements[0];
- }
- static spl_ptr_heap_element spl_ptr_heap_delete_top(spl_ptr_heap *heap, void *cmp_userdata TSRMLS_DC) {
- int i, j;
- const int limit = (heap->count-1)/2;
- spl_ptr_heap_element top;
- spl_ptr_heap_element bottom;
- if (heap->count == 0) {
- return NULL;
- }
- top = heap->elements[0];
- bottom = heap->elements[--heap->count];
- for( i = 0; i < limit; i = j)
- {
-
- j = i*2+1;
- if(j != heap->count && heap->cmp(heap->elements[j+1], heap->elements[j], cmp_userdata TSRMLS_CC) > 0) {
- j++;
- }
-
- if(heap->cmp(bottom, heap->elements[j], cmp_userdata TSRMLS_CC) < 0) {
- heap->elements[i] = heap->elements[j];
- } else {
- break;
- }
- }
- if (EG(exception)) {
-
- heap->flags |= SPL_HEAP_CORRUPTED;
- }
- heap->elements[i] = bottom;
- heap->dtor(top TSRMLS_CC);
- return top;
- }
- static spl_ptr_heap *spl_ptr_heap_clone(spl_ptr_heap *from TSRMLS_DC) {
- int i;
- spl_ptr_heap *heap = emalloc(sizeof(spl_ptr_heap));
- heap->dtor = from->dtor;
- heap->ctor = from->ctor;
- heap->cmp = from->cmp;
- heap->max_size = from->max_size;
- heap->count = from->count;
- heap->flags = from->flags;
- heap->elements = safe_emalloc(sizeof(spl_ptr_heap_element),from->max_size,0);
- memcpy(heap->elements, from->elements, sizeof(spl_ptr_heap_element)*from->max_size);
- for (i=0; i < heap->count; ++i) {
- heap->ctor(heap->elements[i] TSRMLS_CC);
- }
- return heap;
- }
- static void spl_ptr_heap_destroy(spl_ptr_heap *heap TSRMLS_DC) {
- int i;
- for (i=0; i < heap->count; ++i) {
- heap->dtor(heap->elements[i] TSRMLS_CC);
- }
- efree(heap->elements);
- efree(heap);
- }
- static int spl_ptr_heap_count(spl_ptr_heap *heap) {
- return heap->count;
- }
- zend_object_iterator *spl_heap_get_iterator(zend_class_entry *ce, zval *object, int by_ref TSRMLS_DC);
- static void spl_heap_object_free_storage(void *object TSRMLS_DC)
- {
- int i;
- spl_heap_object *intern = (spl_heap_object *)object;
- zend_object_std_dtor(&intern->std TSRMLS_CC);
- for (i = 0; i < intern->heap->count; ++i) {
- if (intern->heap->elements[i]) {
- zval_ptr_dtor((zval **)&intern->heap->elements[i]);
- }
- }
- spl_ptr_heap_destroy(intern->heap TSRMLS_CC);
- zval_ptr_dtor(&intern->retval);
- if (intern->debug_info != NULL) {
- zend_hash_destroy(intern->debug_info);
- efree(intern->debug_info);
- }
- efree(object);
- }
- static zend_object_value spl_heap_object_new_ex(zend_class_entry *class_type, spl_heap_object **obj, zval *orig, int clone_orig TSRMLS_DC)
- {
- zend_object_value retval;
- spl_heap_object *intern;
- zend_class_entry *parent = class_type;
- int inherited = 0;
- intern = ecalloc(1, sizeof(spl_heap_object));
- *obj = intern;
- ALLOC_INIT_ZVAL(intern->retval);
- zend_object_std_init(&intern->std, class_type TSRMLS_CC);
- object_properties_init(&intern->std, class_type);
- intern->flags = 0;
- intern->fptr_cmp = NULL;
- intern->debug_info = NULL;
- if (orig) {
- spl_heap_object *other = (spl_heap_object*)zend_object_store_get_object(orig TSRMLS_CC);
- intern->ce_get_iterator = other->ce_get_iterator;
- if (clone_orig) {
- int i;
- intern->heap = spl_ptr_heap_clone(other->heap TSRMLS_CC);
- for (i = 0; i < intern->heap->count; ++i) {
- if (intern->heap->elements[i]) {
- Z_ADDREF_P((zval *)intern->heap->elements[i]);
- }
- }
- } else {
- intern->heap = other->heap;
- }
- intern->flags = other->flags;
- } else {
- intern->heap = spl_ptr_heap_init(spl_ptr_heap_zval_max_cmp, spl_ptr_heap_zval_ctor, spl_ptr_heap_zval_dtor);
- }
- retval.handlers = &spl_handler_SplHeap;
- while (parent) {
- if (parent == spl_ce_SplPriorityQueue) {
- intern->heap->cmp = spl_ptr_pqueue_zval_cmp;
- intern->flags = SPL_PQUEUE_EXTR_DATA;
- retval.handlers = &spl_handler_SplPriorityQueue;
- break;
- }
- if (parent == spl_ce_SplMinHeap) {
- intern->heap->cmp = spl_ptr_heap_zval_min_cmp;
- break;
- }
- if (parent == spl_ce_SplMaxHeap) {
- intern->heap->cmp = spl_ptr_heap_zval_max_cmp;
- break;
- }
- if (parent == spl_ce_SplHeap) {
- break;
- }
- parent = parent->parent;
- inherited = 1;
- }
- retval.handle = zend_objects_store_put(intern, (zend_objects_store_dtor_t)zend_objects_destroy_object, spl_heap_object_free_storage, NULL TSRMLS_CC);
- if (!parent) {
- php_error_docref(NULL TSRMLS_CC, E_COMPILE_ERROR, "Internal compiler error, Class is not child of SplHeap");
- }
- if (inherited) {
- zend_hash_find(&class_type->function_table, "compare", sizeof("compare"), (void **) &intern->fptr_cmp);
- if (intern->fptr_cmp->common.scope == parent) {
- intern->fptr_cmp = NULL;
- }
- zend_hash_find(&class_type->function_table, "count", sizeof("count"), (void **) &intern->fptr_count);
- if (intern->fptr_count->common.scope == parent) {
- intern->fptr_count = NULL;
- }
- }
- return retval;
- }
- static zend_object_value spl_heap_object_new(zend_class_entry *class_type TSRMLS_DC)
- {
- spl_heap_object *tmp;
- return spl_heap_object_new_ex(class_type, &tmp, NULL, 0 TSRMLS_CC);
- }
- static zend_object_value spl_heap_object_clone(zval *zobject TSRMLS_DC)
- {
- zend_object_value new_obj_val;
- zend_object *old_object;
- zend_object *new_object;
- zend_object_handle handle = Z_OBJ_HANDLE_P(zobject);
- spl_heap_object *intern;
- old_object = zend_objects_get_address(zobject TSRMLS_CC);
- new_obj_val = spl_heap_object_new_ex(old_object->ce, &intern, zobject, 1 TSRMLS_CC);
- new_object = &intern->std;
- zend_objects_clone_members(new_object, new_obj_val, old_object, handle TSRMLS_CC);
- return new_obj_val;
- }
- static int spl_heap_object_count_elements(zval *object, long *count TSRMLS_DC)
- {
- spl_heap_object *intern = (spl_heap_object*)zend_object_store_get_object(object TSRMLS_CC);
- if (intern->fptr_count) {
- zval *rv;
- zend_call_method_with_0_params(&object, intern->std.ce, &intern->fptr_count, "count", &rv);
- if (rv) {
- zval_ptr_dtor(&intern->retval);
- MAKE_STD_ZVAL(intern->retval);
- ZVAL_ZVAL(intern->retval, rv, 1, 1);
- convert_to_long(intern->retval);
- *count = (long) Z_LVAL_P(intern->retval);
- return SUCCESS;
- }
- *count = 0;
- return FAILURE;
- }
-
- *count = spl_ptr_heap_count(intern->heap);
- return SUCCESS;
- }
- static HashTable* spl_heap_object_get_debug_info_helper(zend_class_entry *ce, zval *obj, int *is_temp TSRMLS_DC) {
- spl_heap_object *intern = (spl_heap_object*)zend_object_store_get_object(obj TSRMLS_CC);
- zval *tmp, zrv, *heap_array;
- char *pnstr;
- int pnlen;
- int i;
- *is_temp = 0;
- if (!intern->std.properties) {
- rebuild_object_properties(&intern->std);
- }
- if (intern->debug_info == NULL) {
- ALLOC_HASHTABLE(intern->debug_info);
- ZEND_INIT_SYMTABLE_EX(intern->debug_info, zend_hash_num_elements(intern->std.properties) + 1, 0);
- }
- if (intern->debug_info->nApplyCount == 0) {
- INIT_PZVAL(&zrv);
- Z_ARRVAL(zrv) = intern->debug_info;
- zend_hash_copy(intern->debug_info, intern->std.properties, (copy_ctor_func_t) zval_add_ref, (void *) &tmp, sizeof(zval *));
- pnstr = spl_gen_private_prop_name(ce, "flags", sizeof("flags")-1, &pnlen TSRMLS_CC);
- add_assoc_long_ex(&zrv, pnstr, pnlen+1, intern->flags);
- efree(pnstr);
- pnstr = spl_gen_private_prop_name(ce, "isCorrupted", sizeof("isCorrupted")-1, &pnlen TSRMLS_CC);
- add_assoc_bool_ex(&zrv, pnstr, pnlen+1, intern->heap->flags&SPL_HEAP_CORRUPTED);
- efree(pnstr);
- ALLOC_INIT_ZVAL(heap_array);
- array_init(heap_array);
- for (i = 0; i < intern->heap->count; ++i) {
- add_index_zval(heap_array, i, (zval *)intern->heap->elements[i]);
- Z_ADDREF_P(intern->heap->elements[i]);
- }
- pnstr = spl_gen_private_prop_name(ce, "heap", sizeof("heap")-1, &pnlen TSRMLS_CC);
- add_assoc_zval_ex(&zrv, pnstr, pnlen+1, heap_array);
- efree(pnstr);
- }
- return intern->debug_info;
- }
- static HashTable* spl_heap_object_get_debug_info(zval *obj, int *is_temp TSRMLS_DC)
- {
- return spl_heap_object_get_debug_info_helper(spl_ce_SplHeap, obj, is_temp TSRMLS_CC);
- }
- static HashTable* spl_pqueue_object_get_debug_info(zval *obj, int *is_temp TSRMLS_DC)
- {
- return spl_heap_object_get_debug_info_helper(spl_ce_SplPriorityQueue, obj, is_temp TSRMLS_CC);
- }
- SPL_METHOD(SplHeap, count)
- {
- long count;
- spl_heap_object *intern = (spl_heap_object*)zend_object_store_get_object(getThis() TSRMLS_CC);
- if (zend_parse_parameters_none() == FAILURE) {
- return;
- }
- count = spl_ptr_heap_count(intern->heap);
- RETURN_LONG(count);
- }
- SPL_METHOD(SplHeap, isEmpty)
- {
- spl_heap_object *intern = (spl_heap_object*)zend_object_store_get_object(getThis() TSRMLS_CC);
- if (zend_parse_parameters_none() == FAILURE) {
- return;
- }
- RETURN_BOOL(spl_ptr_heap_count(intern->heap)==0);
- }
- SPL_METHOD(SplHeap, insert)
- {
- zval *value;
- spl_heap_object *intern;
- if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "z", &value) == FAILURE) {
- return;
- }
- intern = (spl_heap_object*)zend_object_store_get_object(getThis() TSRMLS_CC);
- if (intern->heap->flags & SPL_HEAP_CORRUPTED) {
- zend_throw_exception(spl_ce_RuntimeException, "Heap is corrupted, heap properties are no longer ensured.", 0 TSRMLS_CC);
- return;
- }
- SEPARATE_ARG_IF_REF(value);
- spl_ptr_heap_insert(intern->heap, value, getThis() TSRMLS_CC);
- RETURN_TRUE;
- }
- SPL_METHOD(SplHeap, extract)
- {
- zval *value;
- spl_heap_object *intern;
- if (zend_parse_parameters_none() == FAILURE) {
- return;
- }
- intern = (spl_heap_object*)zend_object_store_get_object(getThis() TSRMLS_CC);
- if (intern->heap->flags & SPL_HEAP_CORRUPTED) {
- zend_throw_exception(spl_ce_RuntimeException, "Heap is corrupted, heap properties are no longer ensured.", 0 TSRMLS_CC);
- return;
- }
- value = (zval *)spl_ptr_heap_delete_top(intern->heap, getThis() TSRMLS_CC);
- if (!value) {
- zend_throw_exception(spl_ce_RuntimeException, "Can't extract from an empty heap", 0 TSRMLS_CC);
- return;
- }
- RETURN_ZVAL(value, 1, 1);
- }
- SPL_METHOD(SplPriorityQueue, insert)
- {
- zval *data, *priority, *elem;
- spl_heap_object *intern;
- if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "zz", &data, &priority) == FAILURE) {
- return;
- }
- intern = (spl_heap_object*)zend_object_store_get_object(getThis() TSRMLS_CC);
- if (intern->heap->flags & SPL_HEAP_CORRUPTED) {
- zend_throw_exception(spl_ce_RuntimeException, "Heap is corrupted, heap properties are no longer ensured.", 0 TSRMLS_CC);
- return;
- }
- SEPARATE_ARG_IF_REF(data);
- SEPARATE_ARG_IF_REF(priority);
- ALLOC_INIT_ZVAL(elem);
- array_init(elem);
- add_assoc_zval_ex(elem, "data", sizeof("data"), data);
- add_assoc_zval_ex(elem, "priority", sizeof("priority"), priority);
- spl_ptr_heap_insert(intern->heap, elem, getThis() TSRMLS_CC);
- RETURN_TRUE;
- }
- SPL_METHOD(SplPriorityQueue, extract)
- {
- zval *value, *value_out, **value_out_pp;
- spl_heap_object *intern;
- if (zend_parse_parameters_none() == FAILURE) {
- return;
- }
- intern = (spl_heap_object*)zend_object_store_get_object(getThis() TSRMLS_CC);
- if (intern->heap->flags & SPL_HEAP_CORRUPTED) {
- zend_throw_exception(spl_ce_RuntimeException, "Heap is corrupted, heap properties are no longer ensured.", 0 TSRMLS_CC);
- return;
- }
- value = (zval *)spl_ptr_heap_delete_top(intern->heap, getThis() TSRMLS_CC);
- if (!value) {
- zend_throw_exception(spl_ce_RuntimeException, "Can't extract from an empty heap", 0 TSRMLS_CC);
- return;
- }
- value_out_pp = spl_pqueue_extract_helper(&value, intern->flags);
- if (!value_out_pp) {
- zend_error(E_RECOVERABLE_ERROR, "Unable to extract from the PriorityQueue node");
- zval_ptr_dtor(&value);
- return;
- }
- value_out = *value_out_pp;
- Z_ADDREF_P(value_out);
- zval_ptr_dtor(&value);
- RETURN_ZVAL(value_out, 1, 1);
- }
- SPL_METHOD(SplPriorityQueue, top)
- {
- zval *value, **value_out;
- spl_heap_object *intern;
- if (zend_parse_parameters_none() == FAILURE) {
- return;
- }
- intern = (spl_heap_object*)zend_object_store_get_object(getThis() TSRMLS_CC);
- if (intern->heap->flags & SPL_HEAP_CORRUPTED) {
- zend_throw_exception(spl_ce_RuntimeException, "Heap is corrupted, heap properties are no longer ensured.", 0 TSRMLS_CC);
- return;
- }
- value = (zval *)spl_ptr_heap_top(intern->heap);
- if (!value) {
- zend_throw_exception(spl_ce_RuntimeException, "Can't peek at an empty heap", 0 TSRMLS_CC);
- return;
- }
- value_out = spl_pqueue_extract_helper(&value, intern->flags);
- if (!value_out) {
- zend_error(E_RECOVERABLE_ERROR, "Unable to extract from the PriorityQueue node");
- return;
- }
- RETURN_ZVAL(*value_out, 1, 0);
- }
- SPL_METHOD(SplPriorityQueue, setExtractFlags)
- {
- long value;
- spl_heap_object *intern;
- if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "l", &value) == FAILURE) {
- return;
- }
- intern = (spl_heap_object*)zend_object_store_get_object(getThis() TSRMLS_CC);
- intern->flags = value & SPL_PQUEUE_EXTR_MASK;
- RETURN_LONG(intern->flags);
- }
- SPL_METHOD(SplHeap, recoverFromCorruption)
- {
- spl_heap_object *intern;
- if (zend_parse_parameters_none() == FAILURE) {
- return;
- }
- intern = (spl_heap_object*)zend_object_store_get_object(getThis() TSRMLS_CC);
- intern->heap->flags = intern->heap->flags & ~SPL_HEAP_CORRUPTED;
- RETURN_TRUE;
- }
- SPL_METHOD(SplPriorityQueue, compare)
- {
- zval *a, *b;
- if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "zz", &a, &b) == FAILURE) {
- return;
- }
- RETURN_LONG(spl_ptr_heap_zval_max_cmp(a, b, NULL TSRMLS_CC));
- }
- SPL_METHOD(SplHeap, top)
- {
- zval *value;
- spl_heap_object *intern;
- if (zend_parse_parameters_none() == FAILURE) {
- return;
- }
- intern = (spl_heap_object*)zend_object_store_get_object(getThis() TSRMLS_CC);
- if (intern->heap->flags & SPL_HEAP_CORRUPTED) {
- zend_throw_exception(spl_ce_RuntimeException, "Heap is corrupted, heap properties are no longer ensured.", 0 TSRMLS_CC);
- return;
- }
- value = (zval *)spl_ptr_heap_top(intern->heap);
- if (!value) {
- zend_throw_exception(spl_ce_RuntimeException, "Can't peek at an empty heap", 0 TSRMLS_CC);
- return;
- }
- RETURN_ZVAL(value, 1, 0);
- }
- SPL_METHOD(SplMinHeap, compare)
- {
- zval *a, *b;
- if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "zz", &a, &b) == FAILURE) {
- return;
- }
- RETURN_LONG(spl_ptr_heap_zval_min_cmp(a, b, NULL TSRMLS_CC));
- }
- SPL_METHOD(SplMaxHeap, compare)
- {
- zval *a, *b;
- if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "zz", &a, &b) == FAILURE) {
- return;
- }
- RETURN_LONG(spl_ptr_heap_zval_max_cmp(a, b, NULL TSRMLS_CC));
- }
- static void spl_heap_it_dtor(zend_object_iterator *iter TSRMLS_DC)
- {
- spl_heap_it *iterator = (spl_heap_it *)iter;
- zend_user_it_invalidate_current(iter TSRMLS_CC);
- zval_ptr_dtor((zval**)&iterator->intern.it.data);
- efree(iterator);
- }
- static void spl_heap_it_rewind(zend_object_iterator *iter TSRMLS_DC)
- {
-
- }
- static int spl_heap_it_valid(zend_object_iterator *iter TSRMLS_DC)
- {
- spl_heap_it *iterator = (spl_heap_it *)iter;
- return (iterator->object->heap->count != 0 ? SUCCESS : FAILURE);
- }
- static void spl_heap_it_get_current_data(zend_object_iterator *iter, zval ***data TSRMLS_DC)
- {
- spl_heap_it *iterator = (spl_heap_it *)iter;
- zval **element = (zval **)&iterator->object->heap->elements[0];
- if (iterator->object->heap->flags & SPL_HEAP_CORRUPTED) {
- zend_throw_exception(spl_ce_RuntimeException, "Heap is corrupted, heap properties are no longer ensured.", 0 TSRMLS_CC);
- return;
- }
- if (iterator->object->heap->count == 0 || !*element) {
- *data = NULL;
- } else {
- *data = element;
- }
- }
- static void spl_pqueue_it_get_current_data(zend_object_iterator *iter, zval ***data TSRMLS_DC)
- {
- spl_heap_it *iterator = (spl_heap_it *)iter;
- zval **element = (zval **)&iterator->object->heap->elements[0];
- if (iterator->object->heap->flags & SPL_HEAP_CORRUPTED) {
- zend_throw_exception(spl_ce_RuntimeException, "Heap is corrupted, heap properties are no longer ensured.", 0 TSRMLS_CC);
- return;
- }
- if (iterator->object->heap->count == 0 || !*element) {
- *data = NULL;
- } else {
- *data = spl_pqueue_extract_helper(element, iterator->object->flags);
- if (!*data) {
- zend_error(E_RECOVERABLE_ERROR, "Unable to extract from the PriorityQueue node");
- }
- }
- }
- static void spl_heap_it_get_current_key(zend_object_iterator *iter, zval *key TSRMLS_DC)
- {
- spl_heap_it *iterator = (spl_heap_it *)iter;
- ZVAL_LONG(key, iterator->object->heap->count - 1);
- }
- static void spl_heap_it_move_forward(zend_object_iterator *iter TSRMLS_DC)
- {
- zval *object = (zval*)((zend_user_iterator *)iter)->it.data;
- spl_heap_it *iterator = (spl_heap_it *)iter;
- spl_ptr_heap_element elem;
- if (iterator->object->heap->flags & SPL_HEAP_CORRUPTED) {
- zend_throw_exception(spl_ce_RuntimeException, "Heap is corrupted, heap properties are no longer ensured.", 0 TSRMLS_CC);
- return;
- }
- elem = spl_ptr_heap_delete_top(iterator->object->heap, object TSRMLS_CC);
- if (elem != NULL) {
- zval_ptr_dtor((zval **)&elem);
- }
- zend_user_it_invalidate_current(iter TSRMLS_CC);
- }
- SPL_METHOD(SplHeap, key)
- {
- spl_heap_object *intern = (spl_heap_object*)zend_object_store_get_object(getThis() TSRMLS_CC);
-
- if (zend_parse_parameters_none() == FAILURE) {
- return;
- }
-
- RETURN_LONG(intern->heap->count - 1);
- }
- SPL_METHOD(SplHeap, next)
- {
- spl_heap_object *intern = (spl_heap_object*)zend_object_store_get_object(getThis() TSRMLS_CC);
- spl_ptr_heap_element elem = spl_ptr_heap_delete_top(intern->heap, getThis() TSRMLS_CC);
-
- if (zend_parse_parameters_none() == FAILURE) {
- return;
- }
- if (elem != NULL) {
- zval_ptr_dtor((zval **)&elem);
- }
- }
- SPL_METHOD(SplHeap, valid)
- {
- spl_heap_object *intern = (spl_heap_object*)zend_object_store_get_object(getThis() TSRMLS_CC);
-
- if (zend_parse_parameters_none() == FAILURE) {
- return;
- }
- RETURN_BOOL(intern->heap->count != 0);
- }
- SPL_METHOD(SplHeap, rewind)
- {
- if (zend_parse_parameters_none() == FAILURE) {
- return;
- }
-
- }
- SPL_METHOD(SplHeap, current)
- {
- spl_heap_object *intern = (spl_heap_object*)zend_object_store_get_object(getThis() TSRMLS_CC);
- zval *element = (zval *)intern->heap->elements[0];
-
- if (zend_parse_parameters_none() == FAILURE) {
- return;
- }
- if (!intern->heap->count || !element) {
- RETURN_NULL();
- } else {
- RETURN_ZVAL(element, 1, 0);
- }
- }
- SPL_METHOD(SplPriorityQueue, current)
- {
- spl_heap_object *intern = (spl_heap_object*)zend_object_store_get_object(getThis() TSRMLS_CC);
- zval **element = (zval **)&intern->heap->elements[0];
-
- if (zend_parse_parameters_none() == FAILURE) {
- return;
- }
- if (!intern->heap->count || !*element) {
- RETURN_NULL();
- } else {
- zval **data = spl_pqueue_extract_helper(element, intern->flags);
- if (!data) {
- zend_error(E_RECOVERABLE_ERROR, "Unable to extract from the PriorityQueue node");
- RETURN_NULL();
- }
- RETURN_ZVAL(*data, 1, 0);
- }
- }
- zend_object_iterator_funcs spl_heap_it_funcs = {
- spl_heap_it_dtor,
- spl_heap_it_valid,
- spl_heap_it_get_current_data,
- spl_heap_it_get_current_key,
- spl_heap_it_move_forward,
- spl_heap_it_rewind
- };
- zend_object_iterator_funcs spl_pqueue_it_funcs = {
- spl_heap_it_dtor,
- spl_heap_it_valid,
- spl_pqueue_it_get_current_data,
- spl_heap_it_get_current_key,
- spl_heap_it_move_forward,
- spl_heap_it_rewind
- };
- zend_object_iterator *spl_heap_get_iterator(zend_class_entry *ce, zval *object, int by_ref TSRMLS_DC)
- {
- spl_heap_it *iterator;
- spl_heap_object *heap_object = (spl_heap_object*)zend_object_store_get_object(object TSRMLS_CC);
- if (by_ref) {
- zend_throw_exception(spl_ce_RuntimeException, "An iterator cannot be used with foreach by reference", 0 TSRMLS_CC);
- return NULL;
- }
- Z_ADDREF_P(object);
- iterator = emalloc(sizeof(spl_heap_it));
- iterator->intern.it.data = (void*)object;
- iterator->intern.it.funcs = &spl_heap_it_funcs;
- iterator->intern.ce = ce;
- iterator->intern.value = NULL;
- iterator->flags = heap_object->flags;
- iterator->object = heap_object;
- return (zend_object_iterator*)iterator;
- }
- zend_object_iterator *spl_pqueue_get_iterator(zend_class_entry *ce, zval *object, int by_ref TSRMLS_DC)
- {
- spl_heap_it *iterator;
- spl_heap_object *heap_object = (spl_heap_object*)zend_object_store_get_object(object TSRMLS_CC);
- if (by_ref) {
- zend_throw_exception(spl_ce_RuntimeException, "An iterator cannot be used with foreach by reference", 0 TSRMLS_CC);
- return NULL;
- }
- Z_ADDREF_P(object);
- iterator = emalloc(sizeof(spl_heap_it));
- iterator->intern.it.data = (void*)object;
- iterator->intern.it.funcs = &spl_pqueue_it_funcs;
- iterator->intern.ce = ce;
- iterator->intern.value = NULL;
- iterator->flags = heap_object->flags;
- iterator->object = heap_object;
- return (zend_object_iterator*)iterator;
- }
- ZEND_BEGIN_ARG_INFO(arginfo_heap_insert, 0)
- ZEND_ARG_INFO(0, value)
- ZEND_END_ARG_INFO()
- ZEND_BEGIN_ARG_INFO(arginfo_heap_compare, 0)
- ZEND_ARG_INFO(0, a)
- ZEND_ARG_INFO(0, b)
- ZEND_END_ARG_INFO()
- ZEND_BEGIN_ARG_INFO(arginfo_pqueue_insert, 0)
- ZEND_ARG_INFO(0, value)
- ZEND_ARG_INFO(0, priority)
- ZEND_END_ARG_INFO()
- ZEND_BEGIN_ARG_INFO(arginfo_pqueue_setflags, 0)
- ZEND_ARG_INFO(0, flags)
- ZEND_END_ARG_INFO()
- ZEND_BEGIN_ARG_INFO(arginfo_splheap_void, 0)
- ZEND_END_ARG_INFO()
- static const zend_function_entry spl_funcs_SplMinHeap[] = {
- SPL_ME(SplMinHeap, compare, arginfo_heap_compare, ZEND_ACC_PROTECTED)
- {NULL, NULL, NULL}
- };
- static const zend_function_entry spl_funcs_SplMaxHeap[] = {
- SPL_ME(SplMaxHeap, compare, arginfo_heap_compare, ZEND_ACC_PROTECTED)
- {NULL, NULL, NULL}
- };
- static const zend_function_entry spl_funcs_SplPriorityQueue[] = {
- SPL_ME(SplPriorityQueue, compare, arginfo_heap_compare, ZEND_ACC_PUBLIC)
- SPL_ME(SplPriorityQueue, insert, arginfo_pqueue_insert, ZEND_ACC_PUBLIC)
- SPL_ME(SplPriorityQueue, setExtractFlags, arginfo_pqueue_setflags, ZEND_ACC_PUBLIC)
- SPL_ME(SplPriorityQueue, top, arginfo_splheap_void, ZEND_ACC_PUBLIC)
- SPL_ME(SplPriorityQueue, extract, arginfo_splheap_void, ZEND_ACC_PUBLIC)
- SPL_ME(SplHeap, count, arginfo_splheap_void, ZEND_ACC_PUBLIC)
- SPL_ME(SplHeap, isEmpty, arginfo_splheap_void, ZEND_ACC_PUBLIC)
- SPL_ME(SplHeap, rewind, arginfo_splheap_void, ZEND_ACC_PUBLIC)
- SPL_ME(SplPriorityQueue, current, arginfo_splheap_void, ZEND_ACC_PUBLIC)
- SPL_ME(SplHeap, key, arginfo_splheap_void, ZEND_ACC_PUBLIC)
- SPL_ME(SplHeap, next, arginfo_splheap_void, ZEND_ACC_PUBLIC)
- SPL_ME(SplHeap, valid, arginfo_splheap_void, ZEND_ACC_PUBLIC)
- SPL_ME(SplHeap, recoverFromCorruption, arginfo_splheap_void, ZEND_ACC_PUBLIC)
- {NULL, NULL, NULL}
- };
- static const zend_function_entry spl_funcs_SplHeap[] = {
- SPL_ME(SplHeap, extract, arginfo_splheap_void, ZEND_ACC_PUBLIC)
- SPL_ME(SplHeap, insert, arginfo_heap_insert, ZEND_ACC_PUBLIC)
- SPL_ME(SplHeap, top, arginfo_splheap_void, ZEND_ACC_PUBLIC)
- SPL_ME(SplHeap, count, arginfo_splheap_void, ZEND_ACC_PUBLIC)
- SPL_ME(SplHeap, isEmpty, arginfo_splheap_void, ZEND_ACC_PUBLIC)
- SPL_ME(SplHeap, rewind, arginfo_splheap_void, ZEND_ACC_PUBLIC)
- SPL_ME(SplHeap, current, arginfo_splheap_void, ZEND_ACC_PUBLIC)
- SPL_ME(SplHeap, key, arginfo_splheap_void, ZEND_ACC_PUBLIC)
- SPL_ME(SplHeap, next, arginfo_splheap_void, ZEND_ACC_PUBLIC)
- SPL_ME(SplHeap, valid, arginfo_splheap_void, ZEND_ACC_PUBLIC)
- SPL_ME(SplHeap, recoverFromCorruption, arginfo_splheap_void, ZEND_ACC_PUBLIC)
- ZEND_FENTRY(compare, NULL, NULL, ZEND_ACC_PROTECTED|ZEND_ACC_ABSTRACT)
- {NULL, NULL, NULL}
- };
- PHP_MINIT_FUNCTION(spl_heap)
- {
- REGISTER_SPL_STD_CLASS_EX(SplHeap, spl_heap_object_new, spl_funcs_SplHeap);
- memcpy(&spl_handler_SplHeap, zend_get_std_object_handlers(), sizeof(zend_object_handlers));
- spl_handler_SplHeap.clone_obj = spl_heap_object_clone;
- spl_handler_SplHeap.count_elements = spl_heap_object_count_elements;
- spl_handler_SplHeap.get_debug_info = spl_heap_object_get_debug_info;
- REGISTER_SPL_IMPLEMENTS(SplHeap, Iterator);
- REGISTER_SPL_IMPLEMENTS(SplHeap, Countable);
- spl_ce_SplHeap->get_iterator = spl_heap_get_iterator;
- REGISTER_SPL_SUB_CLASS_EX(SplMinHeap, SplHeap, spl_heap_object_new, spl_funcs_SplMinHeap);
- REGISTER_SPL_SUB_CLASS_EX(SplMaxHeap, SplHeap, spl_heap_object_new, spl_funcs_SplMaxHeap);
- spl_ce_SplMaxHeap->get_iterator = spl_heap_get_iterator;
- spl_ce_SplMinHeap->get_iterator = spl_heap_get_iterator;
- REGISTER_SPL_STD_CLASS_EX(SplPriorityQueue, spl_heap_object_new, spl_funcs_SplPriorityQueue);
- memcpy(&spl_handler_SplPriorityQueue, zend_get_std_object_handlers(), sizeof(zend_object_handlers));
- spl_handler_SplPriorityQueue.clone_obj = spl_heap_object_clone;
- spl_handler_SplPriorityQueue.count_elements = spl_heap_object_count_elements;
- spl_handler_SplPriorityQueue.get_debug_info = spl_pqueue_object_get_debug_info;
- REGISTER_SPL_IMPLEMENTS(SplPriorityQueue, Iterator);
- REGISTER_SPL_IMPLEMENTS(SplPriorityQueue, Countable);
- spl_ce_SplPriorityQueue->get_iterator = spl_pqueue_get_iterator;
- REGISTER_SPL_CLASS_CONST_LONG(SplPriorityQueue, "EXTR_BOTH", SPL_PQUEUE_EXTR_BOTH);
- REGISTER_SPL_CLASS_CONST_LONG(SplPriorityQueue, "EXTR_PRIORITY", SPL_PQUEUE_EXTR_PRIORITY);
- REGISTER_SPL_CLASS_CONST_LONG(SplPriorityQueue, "EXTR_DATA", SPL_PQUEUE_EXTR_DATA);
- return SUCCESS;
- }
|