12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340 |
- /*
- +----------------------------------------------------------------------+
- | PHP Version 7 |
- +----------------------------------------------------------------------+
- | Copyright (c) 1997-2018 The PHP Group |
- +----------------------------------------------------------------------+
- | This source file is subject to version 3.01 of the PHP license, |
- | that is bundled with this package in the file LICENSE, and is |
- | available through the world-wide-web at the following url: |
- | http://www.php.net/license/3_01.txt |
- | If you did not receive a copy of the PHP license and are unable to |
- | obtain it through the world-wide-web, please send a note to |
- | license@php.net so we can mail you a copy immediately. |
- +----------------------------------------------------------------------+
- | Author: Andrei Zmievski <andrei@php.net> |
- +----------------------------------------------------------------------+
- */
- #ifdef HAVE_CONFIG_H
- #include "config.h"
- #endif
- #include "php.h"
- #if HAVE_WDDX
- #include "ext/xml/expat_compat.h"
- #include "php_wddx.h"
- #include "php_wddx_api.h"
- #define PHP_XML_INTERNAL
- #include "ext/xml/php_xml.h"
- #include "ext/standard/php_incomplete_class.h"
- #include "ext/standard/base64.h"
- #include "ext/standard/info.h"
- #include "zend_smart_str.h"
- #include "ext/standard/html.h"
- #include "ext/standard/php_string.h"
- #include "ext/date/php_date.h"
- #include "zend_globals.h"
- #define WDDX_BUF_LEN 256
- #define PHP_CLASS_NAME_VAR "php_class_name"
- #define EL_ARRAY "array"
- #define EL_BINARY "binary"
- #define EL_BOOLEAN "boolean"
- #define EL_CHAR "char"
- #define EL_CHAR_CODE "code"
- #define EL_NULL "null"
- #define EL_NUMBER "number"
- #define EL_PACKET "wddxPacket"
- #define EL_STRING "string"
- #define EL_STRUCT "struct"
- #define EL_VALUE "value"
- #define EL_VAR "var"
- #define EL_NAME "name"
- #define EL_VERSION "version"
- #define EL_RECORDSET "recordset"
- #define EL_FIELD "field"
- #define EL_DATETIME "dateTime"
- #define php_wddx_deserialize(a,b) \
- php_wddx_deserialize_ex(Z_STRVAL_P(a), Z_STRLEN_P(a), (b))
- #define SET_STACK_VARNAME \
- if (stack->varname) { \
- ent.varname = estrdup(stack->varname); \
- efree(stack->varname); \
- stack->varname = NULL; \
- } else \
- ent.varname = NULL; \
- static int le_wddx;
- typedef struct {
- zval data;
- enum {
- ST_ARRAY,
- ST_BOOLEAN,
- ST_NULL,
- ST_NUMBER,
- ST_STRING,
- ST_BINARY,
- ST_STRUCT,
- ST_RECORDSET,
- ST_FIELD,
- ST_DATETIME
- } type;
- char *varname;
- } st_entry;
- typedef struct {
- int top, max;
- char *varname;
- zend_bool done;
- void **elements;
- } wddx_stack;
- static void php_wddx_process_data(void *user_data, const XML_Char *s, int len);
- /* {{{ arginfo */
- ZEND_BEGIN_ARG_INFO_EX(arginfo_wddx_serialize_value, 0, 0, 1)
- ZEND_ARG_INFO(0, var)
- ZEND_ARG_INFO(0, comment)
- ZEND_END_ARG_INFO()
- ZEND_BEGIN_ARG_INFO_EX(arginfo_wddx_serialize_vars, 0, 0, 1)
- ZEND_ARG_VARIADIC_INFO(0, var_names)
- ZEND_END_ARG_INFO()
- ZEND_BEGIN_ARG_INFO_EX(arginfo_wddx_serialize_start, 0, 0, 0)
- ZEND_ARG_INFO(0, comment)
- ZEND_END_ARG_INFO()
- ZEND_BEGIN_ARG_INFO_EX(arginfo_wddx_packet_end, 0, 0, 1)
- ZEND_ARG_INFO(0, packet_id)
- ZEND_END_ARG_INFO()
- ZEND_BEGIN_ARG_INFO_EX(arginfo_wddx_add_vars, 0, 0, 2)
- ZEND_ARG_INFO(0, packet_id)
- ZEND_ARG_VARIADIC_INFO(0, var_names)
- ZEND_END_ARG_INFO()
- ZEND_BEGIN_ARG_INFO_EX(arginfo_wddx_deserialize, 0, 0, 1)
- ZEND_ARG_INFO(0, packet)
- ZEND_END_ARG_INFO()
- /* }}} */
- /* {{{ wddx_functions[]
- */
- static const zend_function_entry wddx_functions[] = {
- PHP_FE(wddx_serialize_value, arginfo_wddx_serialize_value)
- PHP_FE(wddx_serialize_vars, arginfo_wddx_serialize_vars)
- PHP_FE(wddx_packet_start, arginfo_wddx_serialize_start)
- PHP_FE(wddx_packet_end, arginfo_wddx_packet_end)
- PHP_FE(wddx_add_vars, arginfo_wddx_add_vars)
- PHP_FE(wddx_deserialize, arginfo_wddx_deserialize)
- PHP_FE_END
- };
- /* }}} */
- PHP_MINIT_FUNCTION(wddx);
- PHP_MINFO_FUNCTION(wddx);
- /* {{{ dynamically loadable module stuff */
- #ifdef COMPILE_DL_WDDX
- ZEND_GET_MODULE(wddx)
- #endif /* COMPILE_DL_WDDX */
- /* }}} */
- /* {{{ wddx_module_entry
- */
- zend_module_entry wddx_module_entry = {
- STANDARD_MODULE_HEADER,
- "wddx",
- wddx_functions,
- PHP_MINIT(wddx),
- NULL,
- NULL,
- NULL,
- PHP_MINFO(wddx),
- PHP_WDDX_VERSION,
- STANDARD_MODULE_PROPERTIES
- };
- /* }}} */
- /* {{{ wddx_stack_init
- */
- static int wddx_stack_init(wddx_stack *stack)
- {
- stack->top = 0;
- stack->elements = (void **) safe_emalloc(sizeof(void **), STACK_BLOCK_SIZE, 0);
- stack->max = STACK_BLOCK_SIZE;
- stack->varname = NULL;
- stack->done = 0;
- return SUCCESS;
- }
- /* }}} */
- /* {{{ wddx_stack_push
- */
- static int wddx_stack_push(wddx_stack *stack, void *element, int size)
- {
- if (stack->top >= stack->max) { /* we need to allocate more memory */
- stack->elements = (void **) erealloc(stack->elements,
- (sizeof(void **) * (stack->max += STACK_BLOCK_SIZE)));
- }
- stack->elements[stack->top] = (void *) emalloc(size);
- memcpy(stack->elements[stack->top], element, size);
- return stack->top++;
- }
- /* }}} */
- /* {{{ wddx_stack_top
- */
- static int wddx_stack_top(wddx_stack *stack, void **element)
- {
- if (stack->top > 0) {
- *element = stack->elements[stack->top - 1];
- return SUCCESS;
- } else {
- *element = NULL;
- return FAILURE;
- }
- }
- /* }}} */
- /* {{{ wddx_stack_is_empty
- */
- static int wddx_stack_is_empty(wddx_stack *stack)
- {
- if (stack->top == 0) {
- return 1;
- } else {
- return 0;
- }
- }
- /* }}} */
- /* {{{ wddx_stack_destroy
- */
- static int wddx_stack_destroy(wddx_stack *stack)
- {
- register int i;
- if (stack->elements) {
- for (i = 0; i < stack->top; i++) {
- if (Z_TYPE(((st_entry *)stack->elements[i])->data) != IS_UNDEF
- && ((st_entry *)stack->elements[i])->type != ST_FIELD) {
- zval_ptr_dtor(&((st_entry *)stack->elements[i])->data);
- }
- if (((st_entry *)stack->elements[i])->varname) {
- efree(((st_entry *)stack->elements[i])->varname);
- }
- efree(stack->elements[i]);
- }
- efree(stack->elements);
- }
- if (stack->varname) {
- efree(stack->varname);
- }
- return SUCCESS;
- }
- /* }}} */
- /* {{{ release_wddx_packet_rsrc
- */
- static void release_wddx_packet_rsrc(zend_resource *rsrc)
- {
- smart_str *str = (smart_str *)rsrc->ptr;
- smart_str_free(str);
- efree(str);
- }
- /* }}} */
- #include "ext/session/php_session.h"
- #if HAVE_PHP_SESSION && !defined(COMPILE_DL_SESSION)
- /* {{{ PS_SERIALIZER_ENCODE_FUNC
- */
- PS_SERIALIZER_ENCODE_FUNC(wddx)
- {
- wddx_packet *packet;
- zend_string *str;
- PS_ENCODE_VARS;
- packet = php_wddx_constructor();
- php_wddx_packet_start(packet, NULL, 0);
- php_wddx_add_chunk_static(packet, WDDX_STRUCT_S);
- PS_ENCODE_LOOP(
- php_wddx_serialize_var(packet, struc, key);
- );
- php_wddx_add_chunk_static(packet, WDDX_STRUCT_E);
- php_wddx_packet_end(packet);
- smart_str_0(packet);
- str = zend_string_copy(packet->s);
- php_wddx_destructor(packet);
- return str;
- }
- /* }}} */
- /* {{{ PS_SERIALIZER_DECODE_FUNC
- */
- PS_SERIALIZER_DECODE_FUNC(wddx)
- {
- zval retval;
- zval *ent;
- zend_string *key;
- zend_ulong idx;
- int ret;
- if (vallen == 0) {
- return SUCCESS;
- }
- ZVAL_UNDEF(&retval);
- if ((ret = php_wddx_deserialize_ex(val, vallen, &retval)) == SUCCESS) {
- if (Z_TYPE(retval) != IS_ARRAY) {
- zval_ptr_dtor_nogc(&retval);
- return FAILURE;
- }
- ZEND_HASH_FOREACH_KEY_VAL(Z_ARRVAL(retval), idx, key, ent) {
- if (key == NULL) {
- key = zend_long_to_str(idx);
- } else {
- zend_string_addref(key);
- }
- if (php_set_session_var(key, ent, NULL)) {
- Z_TRY_ADDREF_P(ent);
- }
- PS_ADD_VAR(key);
- zend_string_release_ex(key, 0);
- } ZEND_HASH_FOREACH_END();
- }
- zval_ptr_dtor(&retval);
- return ret;
- }
- /* }}} */
- #endif
- /* {{{ PHP_MINIT_FUNCTION
- */
- PHP_MINIT_FUNCTION(wddx)
- {
- le_wddx = zend_register_list_destructors_ex(release_wddx_packet_rsrc, NULL, "wddx", module_number);
- #if HAVE_PHP_SESSION && !defined(COMPILE_DL_SESSION)
- php_session_register_serializer("wddx",
- PS_SERIALIZER_ENCODE_NAME(wddx),
- PS_SERIALIZER_DECODE_NAME(wddx));
- #endif
- return SUCCESS;
- }
- /* }}} */
- /* {{{ PHP_MINFO_FUNCTION
- */
- PHP_MINFO_FUNCTION(wddx)
- {
- php_info_print_table_start();
- #if HAVE_PHP_SESSION && !defined(COMPILE_DL_SESSION)
- php_info_print_table_header(2, "WDDX Support", "enabled" );
- php_info_print_table_row(2, "WDDX Session Serializer", "enabled" );
- #else
- php_info_print_table_row(2, "WDDX Support", "enabled" );
- #endif
- php_info_print_table_end();
- }
- /* }}} */
- /* {{{ php_wddx_packet_start
- */
- void php_wddx_packet_start(wddx_packet *packet, char *comment, size_t comment_len)
- {
- php_wddx_add_chunk_static(packet, WDDX_PACKET_S);
- if (comment) {
- zend_string *escaped = php_escape_html_entities(
- comment, comment_len, 0, ENT_QUOTES, NULL);
- php_wddx_add_chunk_static(packet, WDDX_HEADER_S);
- php_wddx_add_chunk_static(packet, WDDX_COMMENT_S);
- php_wddx_add_chunk_ex(packet, ZSTR_VAL(escaped), ZSTR_LEN(escaped));
- php_wddx_add_chunk_static(packet, WDDX_COMMENT_E);
- php_wddx_add_chunk_static(packet, WDDX_HEADER_E);
- zend_string_release_ex(escaped, 0);
- } else {
- php_wddx_add_chunk_static(packet, WDDX_HEADER);
- }
- php_wddx_add_chunk_static(packet, WDDX_DATA_S);
- }
- /* }}} */
- /* {{{ php_wddx_packet_end
- */
- void php_wddx_packet_end(wddx_packet *packet)
- {
- php_wddx_add_chunk_static(packet, WDDX_DATA_E);
- php_wddx_add_chunk_static(packet, WDDX_PACKET_E);
- }
- /* }}} */
- #define FLUSH_BUF() \
- if (l > 0) { \
- php_wddx_add_chunk_ex(packet, buf, l); \
- l = 0; \
- }
- /* {{{ php_wddx_serialize_string
- */
- static void php_wddx_serialize_string(wddx_packet *packet, zval *var)
- {
- php_wddx_add_chunk_static(packet, WDDX_STRING_S);
- if (Z_STRLEN_P(var) > 0) {
- zend_string *buf = php_escape_html_entities(
- (unsigned char *) Z_STRVAL_P(var), Z_STRLEN_P(var), 0, ENT_QUOTES, NULL);
- php_wddx_add_chunk_ex(packet, ZSTR_VAL(buf), ZSTR_LEN(buf));
- zend_string_release_ex(buf, 0);
- }
- php_wddx_add_chunk_static(packet, WDDX_STRING_E);
- }
- /* }}} */
- /* {{{ php_wddx_serialize_number
- */
- static void php_wddx_serialize_number(wddx_packet *packet, zval *var)
- {
- char tmp_buf[WDDX_BUF_LEN], *dec_point;
- zend_string *str = zval_get_string_func(var);
- snprintf(tmp_buf, sizeof(tmp_buf), WDDX_NUMBER, ZSTR_VAL(str));
- zend_string_release_ex(str, 0);
- dec_point = strchr(tmp_buf, ',');
- if (dec_point) {
- *dec_point = '.';
- }
- php_wddx_add_chunk(packet, tmp_buf);
- }
- /* }}} */
- /* {{{ php_wddx_serialize_boolean
- */
- static void php_wddx_serialize_boolean(wddx_packet *packet, zval *var)
- {
- php_wddx_add_chunk(packet, Z_TYPE_P(var) == IS_TRUE ? WDDX_BOOLEAN_TRUE : WDDX_BOOLEAN_FALSE);
- }
- /* }}} */
- /* {{{ php_wddx_serialize_unset
- */
- static void php_wddx_serialize_unset(wddx_packet *packet)
- {
- php_wddx_add_chunk_static(packet, WDDX_NULL);
- }
- /* }}} */
- /* {{{ php_wddx_serialize_object
- */
- static void php_wddx_serialize_object(wddx_packet *packet, zval *obj)
- {
- /* OBJECTS_FIXME */
- zval *ent, fname, *varname;
- zval retval;
- zend_string *key;
- zend_ulong idx;
- char tmp_buf[WDDX_BUF_LEN];
- HashTable *objhash, *sleephash;
- zend_class_entry *ce;
- PHP_CLASS_ATTRIBUTES;
- PHP_SET_CLASS_ATTRIBUTES(obj);
- ce = Z_OBJCE_P(obj);
- if (!ce || ce->serialize || ce->unserialize) {
- php_error_docref(NULL, E_WARNING, "Class %s can not be serialized", ZSTR_VAL(class_name));
- PHP_CLEANUP_CLASS_ATTRIBUTES();
- return;
- }
- ZVAL_STRING(&fname, "__sleep");
- /*
- * We try to call __sleep() method on object. It's supposed to return an
- * array of property names to be serialized.
- */
- if (call_user_function(CG(function_table), obj, &fname, &retval, 0, 0) == SUCCESS) {
- if (!Z_ISUNDEF(retval) && (sleephash = HASH_OF(&retval))) {
- php_wddx_add_chunk_static(packet, WDDX_STRUCT_S);
- snprintf(tmp_buf, WDDX_BUF_LEN, WDDX_VAR_S, PHP_CLASS_NAME_VAR);
- php_wddx_add_chunk(packet, tmp_buf);
- php_wddx_add_chunk_static(packet, WDDX_STRING_S);
- php_wddx_add_chunk_ex(packet, ZSTR_VAL(class_name), ZSTR_LEN(class_name));
- php_wddx_add_chunk_static(packet, WDDX_STRING_E);
- php_wddx_add_chunk_static(packet, WDDX_VAR_E);
- objhash = Z_OBJPROP_P(obj);
- ZEND_HASH_FOREACH_VAL(sleephash, varname) {
- if (Z_TYPE_P(varname) != IS_STRING) {
- php_error_docref(NULL, E_NOTICE, "__sleep should return an array only containing the names of instance-variables to serialize.");
- continue;
- }
- if ((ent = zend_hash_find(objhash, Z_STR_P(varname))) != NULL) {
- php_wddx_serialize_var(packet, ent, Z_STR_P(varname));
- }
- } ZEND_HASH_FOREACH_END();
- php_wddx_add_chunk_static(packet, WDDX_STRUCT_E);
- }
- } else {
- php_wddx_add_chunk_static(packet, WDDX_STRUCT_S);
- snprintf(tmp_buf, WDDX_BUF_LEN, WDDX_VAR_S, PHP_CLASS_NAME_VAR);
- php_wddx_add_chunk(packet, tmp_buf);
- php_wddx_add_chunk_static(packet, WDDX_STRING_S);
- php_wddx_add_chunk_ex(packet, ZSTR_VAL(class_name), ZSTR_LEN(class_name));
- php_wddx_add_chunk_static(packet, WDDX_STRING_E);
- php_wddx_add_chunk_static(packet, WDDX_VAR_E);
- objhash = Z_OBJPROP_P(obj);
- ZEND_HASH_FOREACH_KEY_VAL(objhash, idx, key, ent) {
- if (ent == obj) {
- continue;
- }
- if (key) {
- const char *class_name, *prop_name;
- size_t prop_name_len;
- zend_string *tmp;
- zend_unmangle_property_name_ex(key, &class_name, &prop_name, &prop_name_len);
- tmp = zend_string_init(prop_name, prop_name_len, 0);
- php_wddx_serialize_var(packet, ent, tmp);
- zend_string_release_ex(tmp, 0);
- } else {
- key = zend_long_to_str(idx);
- php_wddx_serialize_var(packet, ent, key);
- zend_string_release_ex(key, 0);
- }
- } ZEND_HASH_FOREACH_END();
- php_wddx_add_chunk_static(packet, WDDX_STRUCT_E);
- }
- PHP_CLEANUP_CLASS_ATTRIBUTES();
- zval_ptr_dtor(&fname);
- zval_ptr_dtor(&retval);
- }
- /* }}} */
- /* {{{ php_wddx_serialize_array
- */
- static void php_wddx_serialize_array(wddx_packet *packet, zval *arr)
- {
- zval *ent;
- zend_string *key;
- int is_struct = 0;
- zend_ulong idx;
- HashTable *target_hash;
- char tmp_buf[WDDX_BUF_LEN];
- zend_ulong ind = 0;
- target_hash = Z_ARRVAL_P(arr);
- ZEND_HASH_FOREACH_KEY(target_hash, idx, key) {
- if (key) {
- is_struct = 1;
- break;
- }
- if (idx != ind) {
- is_struct = 1;
- break;
- }
- ind++;
- } ZEND_HASH_FOREACH_END();
- if (is_struct) {
- php_wddx_add_chunk_static(packet, WDDX_STRUCT_S);
- } else {
- snprintf(tmp_buf, sizeof(tmp_buf), WDDX_ARRAY_S, zend_hash_num_elements(target_hash));
- php_wddx_add_chunk(packet, tmp_buf);
- }
- ZEND_HASH_FOREACH_KEY_VAL(target_hash, idx, key, ent) {
- if (ent == arr) {
- continue;
- }
- if (is_struct) {
- if (key) {
- php_wddx_serialize_var(packet, ent, key);
- } else {
- key = zend_long_to_str(idx);
- php_wddx_serialize_var(packet, ent, key);
- zend_string_release_ex(key, 0);
- }
- } else {
- php_wddx_serialize_var(packet, ent, NULL);
- }
- } ZEND_HASH_FOREACH_END();
- if (is_struct) {
- php_wddx_add_chunk_static(packet, WDDX_STRUCT_E);
- } else {
- php_wddx_add_chunk_static(packet, WDDX_ARRAY_E);
- }
- }
- /* }}} */
- /* {{{ php_wddx_serialize_var
- */
- void php_wddx_serialize_var(wddx_packet *packet, zval *var, zend_string *name)
- {
- HashTable *ht;
- if (name) {
- char *tmp_buf;
- zend_string *name_esc = php_escape_html_entities((unsigned char *) ZSTR_VAL(name), ZSTR_LEN(name), 0, ENT_QUOTES, NULL);
- tmp_buf = emalloc(ZSTR_LEN(name_esc) + sizeof(WDDX_VAR_S));
- snprintf(tmp_buf, ZSTR_LEN(name_esc) + sizeof(WDDX_VAR_S), WDDX_VAR_S, ZSTR_VAL(name_esc));
- php_wddx_add_chunk(packet, tmp_buf);
- efree(tmp_buf);
- zend_string_release_ex(name_esc, 0);
- }
- if (Z_TYPE_P(var) == IS_INDIRECT) {
- var = Z_INDIRECT_P(var);
- }
- ZVAL_DEREF(var);
- switch (Z_TYPE_P(var)) {
- case IS_STRING:
- php_wddx_serialize_string(packet, var);
- break;
- case IS_LONG:
- case IS_DOUBLE:
- php_wddx_serialize_number(packet, var);
- break;
- case IS_TRUE:
- case IS_FALSE:
- php_wddx_serialize_boolean(packet, var);
- break;
- case IS_NULL:
- php_wddx_serialize_unset(packet);
- break;
- case IS_ARRAY:
- ht = Z_ARRVAL_P(var);
- if (Z_REFCOUNTED_P(var)) {
- if (GC_IS_RECURSIVE(ht)) {
- zend_throw_error(NULL, "WDDX doesn't support circular references");
- return;
- }
- GC_PROTECT_RECURSION(ht);
- }
- php_wddx_serialize_array(packet, var);
- if (Z_REFCOUNTED_P(var)) {
- GC_UNPROTECT_RECURSION(ht);
- }
- break;
- case IS_OBJECT:
- ht = Z_OBJPROP_P(var);
- if (GC_IS_RECURSIVE(ht)) {
- zend_throw_error(NULL, "WDDX doesn't support circular references");
- return;
- }
- GC_PROTECT_RECURSION(ht);
- php_wddx_serialize_object(packet, var);
- GC_UNPROTECT_RECURSION(ht);
- break;
- }
- if (name) {
- php_wddx_add_chunk_static(packet, WDDX_VAR_E);
- }
- }
- /* }}} */
- /* {{{ php_wddx_add_var
- */
- static void php_wddx_add_var(wddx_packet *packet, zval *name_var)
- {
- zval *val;
- HashTable *target_hash;
- if (Z_TYPE_P(name_var) == IS_STRING) {
- zend_array *symbol_table = zend_rebuild_symbol_table();
- if ((val = zend_hash_find(symbol_table, Z_STR_P(name_var))) != NULL) {
- if (Z_TYPE_P(val) == IS_INDIRECT) {
- val = Z_INDIRECT_P(val);
- }
- php_wddx_serialize_var(packet, val, Z_STR_P(name_var));
- }
- } else if (Z_TYPE_P(name_var) == IS_ARRAY || Z_TYPE_P(name_var) == IS_OBJECT) {
- int is_array = Z_TYPE_P(name_var) == IS_ARRAY;
- target_hash = HASH_OF(name_var);
- if (!Z_REFCOUNTED_P(name_var)) {
- ZEND_HASH_FOREACH_VAL(target_hash, val) {
- php_wddx_add_var(packet, val);
- } ZEND_HASH_FOREACH_END();
- } else {
- if (is_array) {
- if (GC_IS_RECURSIVE(target_hash)) {
- php_error_docref(NULL, E_WARNING, "recursion detected");
- return;
- }
- GC_PROTECT_RECURSION(target_hash);
- }
- ZEND_HASH_FOREACH_VAL(target_hash, val) {
- ZVAL_DEREF(val);
- php_wddx_add_var(packet, val);
- } ZEND_HASH_FOREACH_END();
- if (is_array) {
- GC_UNPROTECT_RECURSION(target_hash);
- }
- }
- }
- }
- /* }}} */
- /* {{{ php_wddx_push_element
- */
- static void php_wddx_push_element(void *user_data, const XML_Char *name, const XML_Char **atts)
- {
- st_entry ent;
- wddx_stack *stack = (wddx_stack *)user_data;
- if (!strcmp((char *)name, EL_PACKET)) {
- int i;
- if (atts) for (i=0; atts[i]; i++) {
- if (!strcmp((char *)atts[i], EL_VERSION)) {
- /* nothing for now */
- }
- }
- } else if (!strcmp((char *)name, EL_STRING)) {
- ent.type = ST_STRING;
- SET_STACK_VARNAME;
- ZVAL_STR(&ent.data, ZSTR_EMPTY_ALLOC());
- wddx_stack_push((wddx_stack *)stack, &ent, sizeof(st_entry));
- } else if (!strcmp((char *)name, EL_BINARY)) {
- ent.type = ST_BINARY;
- SET_STACK_VARNAME;
- ZVAL_STR(&ent.data, ZSTR_EMPTY_ALLOC());
- wddx_stack_push((wddx_stack *)stack, &ent, sizeof(st_entry));
- } else if (!strcmp((char *)name, EL_CHAR)) {
- int i;
- if (atts) for (i = 0; atts[i]; i++) {
- if (!strcmp((char *)atts[i], EL_CHAR_CODE) && atts[i+1] && atts[i+1][0]) {
- char tmp_buf[2];
- snprintf(tmp_buf, sizeof(tmp_buf), "%c", (char)strtol((char *)atts[i+1], NULL, 16));
- php_wddx_process_data(user_data, (XML_Char *) tmp_buf, strlen(tmp_buf));
- break;
- }
- }
- } else if (!strcmp((char *)name, EL_NUMBER)) {
- ent.type = ST_NUMBER;
- SET_STACK_VARNAME;
- ZVAL_LONG(&ent.data, 0);
- wddx_stack_push((wddx_stack *)stack, &ent, sizeof(st_entry));
- } else if (!strcmp((char *)name, EL_BOOLEAN)) {
- int i;
- ent.type = ST_BOOLEAN;
- SET_STACK_VARNAME;
- if (atts) for (i = 0; atts[i]; i++) {
- if (!strcmp((char *)atts[i], EL_VALUE) && atts[i+1] && atts[i+1][0]) {
- ZVAL_TRUE(&ent.data);
- wddx_stack_push((wddx_stack *)stack, &ent, sizeof(st_entry));
- php_wddx_process_data(user_data, atts[i+1], strlen((char *)atts[i+1]));
- break;
- }
- } else {
- ZVAL_FALSE(&ent.data);
- wddx_stack_push((wddx_stack *)stack, &ent, sizeof(st_entry));
- }
- } else if (!strcmp((char *)name, EL_NULL)) {
- ent.type = ST_NULL;
- SET_STACK_VARNAME;
- ZVAL_NULL(&ent.data);
- wddx_stack_push((wddx_stack *)stack, &ent, sizeof(st_entry));
- } else if (!strcmp((char *)name, EL_ARRAY)) {
- ent.type = ST_ARRAY;
- SET_STACK_VARNAME;
- array_init(&ent.data);
- wddx_stack_push((wddx_stack *)stack, &ent, sizeof(st_entry));
- } else if (!strcmp((char *)name, EL_STRUCT)) {
- ent.type = ST_STRUCT;
- SET_STACK_VARNAME;
- array_init(&ent.data);
- wddx_stack_push((wddx_stack *)stack, &ent, sizeof(st_entry));
- } else if (!strcmp((char *)name, EL_VAR)) {
- int i;
- if (atts) for (i = 0; atts[i]; i++) {
- if (!strcmp((char *)atts[i], EL_NAME) && atts[i+1] && atts[i+1][0]) {
- if (stack->varname) efree(stack->varname);
- stack->varname = estrdup((char *)atts[i+1]);
- break;
- }
- }
- } else if (!strcmp((char *)name, EL_RECORDSET)) {
- int i;
- ent.type = ST_RECORDSET;
- SET_STACK_VARNAME;
- array_init(&ent.data);
- if (atts) for (i = 0; atts[i]; i++) {
- if (!strcmp((char *)atts[i], "fieldNames") && atts[i+1] && atts[i+1][0]) {
- zval tmp;
- char *key;
- const char *p1, *p2, *endp;
- i++;
- endp = (char *)atts[i] + strlen((char *)atts[i]);
- p1 = (char *)atts[i];
- while ((p2 = php_memnstr(p1, ",", sizeof(",")-1, endp)) != NULL) {
- key = estrndup(p1, p2 - p1);
- array_init(&tmp);
- add_assoc_zval_ex(&ent.data, key, p2 - p1, &tmp);
- p1 = p2 + sizeof(",")-1;
- efree(key);
- }
- if (p1 <= endp) {
- array_init(&tmp);
- add_assoc_zval_ex(&ent.data, p1, endp - p1, &tmp);
- }
- break;
- }
- }
- wddx_stack_push((wddx_stack *)stack, &ent, sizeof(st_entry));
- } else if (!strcmp((char *)name, EL_FIELD)) {
- int i;
- st_entry ent;
- ent.type = ST_FIELD;
- ent.varname = NULL;
- ZVAL_UNDEF(&ent.data);
- if (atts) for (i = 0; atts[i]; i++) {
- if (!strcmp((char *)atts[i], EL_NAME) && atts[i+1] && atts[i+1][0]) {
- st_entry *recordset;
- zval *field;
- if (wddx_stack_top(stack, (void**)&recordset) == SUCCESS &&
- recordset->type == ST_RECORDSET &&
- (field = zend_hash_str_find(Z_ARRVAL(recordset->data), (char*)atts[i+1], strlen((char *)atts[i+1]))) != NULL) {
- ZVAL_COPY_VALUE(&ent.data, field);
- }
- break;
- }
- }
- wddx_stack_push((wddx_stack *)stack, &ent, sizeof(st_entry));
- } else if (!strcmp((char *)name, EL_DATETIME)) {
- ent.type = ST_DATETIME;
- SET_STACK_VARNAME;
- ZVAL_LONG(&ent.data, 0);
- wddx_stack_push((wddx_stack *)stack, &ent, sizeof(st_entry));
- }
- }
- /* }}} */
- /* {{{ php_wddx_pop_element
- */
- static void php_wddx_pop_element(void *user_data, const XML_Char *name)
- {
- st_entry *ent1, *ent2;
- wddx_stack *stack = (wddx_stack *)user_data;
- HashTable *target_hash;
- zend_class_entry *pce;
- zval obj;
- /* OBJECTS_FIXME */
- if (stack->top == 0) {
- return;
- }
- if (!strcmp((char *)name, EL_STRING) || !strcmp((char *)name, EL_NUMBER) ||
- !strcmp((char *)name, EL_BOOLEAN) || !strcmp((char *)name, EL_NULL) ||
- !strcmp((char *)name, EL_ARRAY) || !strcmp((char *)name, EL_STRUCT) ||
- !strcmp((char *)name, EL_RECORDSET) || !strcmp((char *)name, EL_BINARY) ||
- !strcmp((char *)name, EL_DATETIME)) {
- wddx_stack_top(stack, (void**)&ent1);
- if (Z_TYPE(ent1->data) == IS_UNDEF) {
- if (stack->top > 1) {
- stack->top--;
- efree(ent1);
- } else {
- stack->done = 1;
- }
- return;
- }
- if (!strcmp((char *)name, EL_BINARY)) {
- zend_string *new_str = NULL;
- if (ZSTR_EMPTY_ALLOC() != Z_STR(ent1->data)) {
- new_str = php_base64_decode(
- (unsigned char *)Z_STRVAL(ent1->data), Z_STRLEN(ent1->data));
- }
- zval_ptr_dtor(&ent1->data);
- if (new_str) {
- ZVAL_STR(&ent1->data, new_str);
- } else {
- ZVAL_EMPTY_STRING(&ent1->data);
- }
- }
- /* Call __wakeup() method on the object. */
- if (Z_TYPE(ent1->data) == IS_OBJECT) {
- zval fname, retval;
- ZVAL_STRING(&fname, "__wakeup");
- call_user_function(NULL, &ent1->data, &fname, &retval, 0, 0);
- zval_ptr_dtor(&fname);
- zval_ptr_dtor(&retval);
- }
- if (stack->top > 1) {
- stack->top--;
- wddx_stack_top(stack, (void**)&ent2);
- /* if non-existent field */
- if (Z_ISUNDEF(ent2->data)) {
- zval_ptr_dtor(&ent1->data);
- efree(ent1);
- return;
- }
- if (Z_TYPE(ent2->data) == IS_ARRAY || Z_TYPE(ent2->data) == IS_OBJECT) {
- target_hash = HASH_OF(&ent2->data);
- if (ent1->varname) {
- if (!strcmp(ent1->varname, PHP_CLASS_NAME_VAR) &&
- Z_TYPE(ent1->data) == IS_STRING && Z_STRLEN(ent1->data) &&
- ent2->type == ST_STRUCT && Z_TYPE(ent2->data) == IS_ARRAY) {
- zend_bool incomplete_class = 0;
- zend_str_tolower(Z_STRVAL(ent1->data), Z_STRLEN(ent1->data));
- zend_string_forget_hash_val(Z_STR(ent1->data));
- if ((pce = zend_hash_find_ptr(EG(class_table), Z_STR(ent1->data))) == NULL) {
- incomplete_class = 1;
- pce = PHP_IC_ENTRY;
- }
- if (pce != PHP_IC_ENTRY && (pce->serialize || pce->unserialize)) {
- zval_ptr_dtor(&ent2->data);
- ZVAL_UNDEF(&ent2->data);
- php_error_docref(NULL, E_WARNING, "Class %s can not be unserialized", Z_STRVAL(ent1->data));
- } else {
- /* Initialize target object */
- if (object_init_ex(&obj, pce) != SUCCESS || EG(exception)) {
- zval_ptr_dtor(&ent2->data);
- ZVAL_UNDEF(&ent2->data);
- php_error_docref(NULL, E_WARNING, "Class %s can not be instantiated", Z_STRVAL(ent1->data));
- } else {
- /* Merge current hashtable with object's default properties */
- zend_hash_merge(Z_OBJPROP(obj),
- Z_ARRVAL(ent2->data),
- zval_add_ref, 0);
- if (incomplete_class) {
- php_store_class_name(&obj, Z_STRVAL(ent1->data), Z_STRLEN(ent1->data));
- }
- /* Clean up old array entry */
- zval_ptr_dtor(&ent2->data);
- /* Set stack entry to point to the newly created object */
- ZVAL_COPY_VALUE(&ent2->data, &obj);
- }
- }
- /* Clean up class name var entry */
- zval_ptr_dtor(&ent1->data);
- } else if (Z_TYPE(ent2->data) == IS_OBJECT) {
- zend_update_property(Z_OBJCE(ent2->data), &ent2->data, ent1->varname, strlen(ent1->varname), &ent1->data);
- Z_TRY_DELREF(ent1->data);
- } else {
- zend_symtable_str_update(target_hash, ent1->varname, strlen(ent1->varname), &ent1->data);
- }
- efree(ent1->varname);
- } else {
- zend_hash_next_index_insert(target_hash, &ent1->data);
- }
- }
- efree(ent1);
- } else {
- stack->done = 1;
- }
- } else if (!strcmp((char *)name, EL_VAR) && stack->varname) {
- efree(stack->varname);
- stack->varname = NULL;
- } else if (!strcmp((char *)name, EL_FIELD)) {
- st_entry *ent;
- wddx_stack_top(stack, (void **)&ent);
- efree(ent);
- stack->top--;
- }
- }
- /* }}} */
- /* {{{ php_wddx_process_data
- */
- static void php_wddx_process_data(void *user_data, const XML_Char *s, int len)
- {
- st_entry *ent;
- wddx_stack *stack = (wddx_stack *)user_data;
- if (!wddx_stack_is_empty(stack) && !stack->done) {
- wddx_stack_top(stack, (void**)&ent);
- switch (ent->type) {
- case ST_BINARY:
- case ST_STRING:
- if (Z_STRLEN(ent->data) == 0) {
- zval_ptr_dtor(&ent->data);
- ZVAL_STRINGL(&ent->data, (char *)s, len);
- } else {
- Z_STR(ent->data) = zend_string_extend(Z_STR(ent->data), Z_STRLEN(ent->data) + len, 0);
- memcpy(Z_STRVAL(ent->data) + Z_STRLEN(ent->data) - len, (char *)s, len);
- Z_STRVAL(ent->data)[Z_STRLEN(ent->data)] = '\0';
- }
- break;
- case ST_NUMBER:
- ZVAL_STRINGL(&ent->data, (char *)s, len);
- convert_scalar_to_number(&ent->data);
- break;
- case ST_BOOLEAN:
- if (!strcmp((char *)s, "true")) {
- ZVAL_TRUE(&ent->data);
- } else if (!strcmp((char *)s, "false")) {
- ZVAL_FALSE(&ent->data);
- } else {
- zval_ptr_dtor(&ent->data);
- if (ent->varname) {
- efree(ent->varname);
- ent->varname = NULL;
- }
- ZVAL_UNDEF(&ent->data);
- }
- break;
- case ST_DATETIME: {
- zend_string *str;
- if (Z_TYPE(ent->data) == IS_STRING) {
- str = zend_string_safe_alloc(Z_STRLEN(ent->data), 1, len, 0);
- memcpy(ZSTR_VAL(str), Z_STRVAL(ent->data), Z_STRLEN(ent->data));
- memcpy(ZSTR_VAL(str) + Z_STRLEN(ent->data), s, len);
- ZSTR_VAL(str)[ZSTR_LEN(str)] = '\0';
- zval_ptr_dtor_str(&ent->data);
- } else {
- str = zend_string_init((char *)s, len, 0);
- }
- ZVAL_LONG(&ent->data, php_parse_date(ZSTR_VAL(str), NULL));
- /* date out of range < 1969 or > 2038 */
- if (Z_LVAL(ent->data) == -1) {
- ZVAL_STR_COPY(&ent->data, str);
- }
- zend_string_release_ex(str, 0);
- }
- break;
- default:
- break;
- }
- }
- }
- /* }}} */
- /* {{{ php_wddx_deserialize_ex
- */
- int php_wddx_deserialize_ex(const char *value, size_t vallen, zval *return_value)
- {
- wddx_stack stack;
- XML_Parser parser;
- st_entry *ent;
- int retval;
- wddx_stack_init(&stack);
- parser = XML_ParserCreate((XML_Char *) "UTF-8");
- XML_SetUserData(parser, &stack);
- XML_SetElementHandler(parser, php_wddx_push_element, php_wddx_pop_element);
- XML_SetCharacterDataHandler(parser, php_wddx_process_data);
- /* XXX value should be parsed in the loop to exhaust size_t */
- XML_Parse(parser, (const XML_Char *) value, (int)vallen, 1);
- XML_ParserFree(parser);
- if (stack.top == 1) {
- wddx_stack_top(&stack, (void**)&ent);
- if (Z_ISUNDEF(ent->data)) {
- retval = FAILURE;
- } else {
- ZVAL_COPY(return_value, &ent->data);
- retval = SUCCESS;
- }
- } else {
- retval = FAILURE;
- }
- wddx_stack_destroy(&stack);
- return retval;
- }
- /* }}} */
- /* {{{ proto string wddx_serialize_value(mixed var [, string comment])
- Creates a new packet and serializes the given value */
- PHP_FUNCTION(wddx_serialize_value)
- {
- zval *var;
- char *comment = NULL;
- size_t comment_len = 0;
- wddx_packet *packet;
- if (zend_parse_parameters(ZEND_NUM_ARGS(), "z|s", &var, &comment, &comment_len) == FAILURE) {
- return;
- }
- packet = php_wddx_constructor();
- php_wddx_packet_start(packet, comment, comment_len);
- php_wddx_serialize_var(packet, var, NULL);
- php_wddx_packet_end(packet);
- smart_str_0(packet);
- RETVAL_STR_COPY(packet->s);
- php_wddx_destructor(packet);
- }
- /* }}} */
- /* {{{ proto string wddx_serialize_vars(mixed var_name [, mixed ...])
- Creates a new packet and serializes given variables into a struct */
- PHP_FUNCTION(wddx_serialize_vars)
- {
- int num_args, i;
- wddx_packet *packet;
- zval *args = NULL;
- if (zend_parse_parameters(ZEND_NUM_ARGS(), "+", &args, &num_args) == FAILURE) {
- return;
- }
- packet = php_wddx_constructor();
- php_wddx_packet_start(packet, NULL, 0);
- php_wddx_add_chunk_static(packet, WDDX_STRUCT_S);
- for (i=0; i<num_args; i++) {
- zval *arg;
- if (!Z_ISREF(args[i])) {
- arg = &args[i];
- } else {
- arg = Z_REFVAL(args[i]);
- }
- if (Z_TYPE_P(arg) != IS_ARRAY && Z_TYPE_P(arg) != IS_OBJECT) {
- convert_to_string_ex(arg);
- }
- php_wddx_add_var(packet, arg);
- }
- php_wddx_add_chunk_static(packet, WDDX_STRUCT_E);
- php_wddx_packet_end(packet);
- smart_str_0(packet);
- RETVAL_STR_COPY(packet->s);
- php_wddx_destructor(packet);
- }
- /* }}} */
- /* {{{ php_wddx_constructor
- */
- wddx_packet *php_wddx_constructor(void)
- {
- smart_str *packet;
- packet = ecalloc(1, sizeof(smart_str));
- return packet;
- }
- /* }}} */
- /* {{{ php_wddx_destructor
- */
- void php_wddx_destructor(wddx_packet *packet)
- {
- smart_str_free(packet);
- efree(packet);
- }
- /* }}} */
- /* {{{ proto resource wddx_packet_start([string comment])
- Starts a WDDX packet with optional comment and returns the packet id */
- PHP_FUNCTION(wddx_packet_start)
- {
- char *comment = NULL;
- size_t comment_len = 0;
- wddx_packet *packet;
- comment = NULL;
- if (zend_parse_parameters(ZEND_NUM_ARGS(), "|s", &comment, &comment_len) == FAILURE) {
- return;
- }
- packet = php_wddx_constructor();
- php_wddx_packet_start(packet, comment, comment_len);
- php_wddx_add_chunk_static(packet, WDDX_STRUCT_S);
- RETURN_RES(zend_register_resource(packet, le_wddx));
- }
- /* }}} */
- /* {{{ proto string wddx_packet_end(resource packet_id)
- Ends specified WDDX packet and returns the string containing the packet */
- PHP_FUNCTION(wddx_packet_end)
- {
- zval *packet_id;
- wddx_packet *packet = NULL;
- if (zend_parse_parameters(ZEND_NUM_ARGS(), "r", &packet_id) == FAILURE) {
- return;
- }
- if ((packet = (wddx_packet *)zend_fetch_resource(Z_RES_P(packet_id), "WDDX packet ID", le_wddx)) == NULL) {
- RETURN_FALSE;
- }
- php_wddx_add_chunk_static(packet, WDDX_STRUCT_E);
- php_wddx_packet_end(packet);
- smart_str_0(packet);
- RETVAL_STR_COPY(packet->s);
- zend_list_close(Z_RES_P(packet_id));
- }
- /* }}} */
- /* {{{ proto bool wddx_add_vars(resource packet_id, mixed var_names [, mixed ...])
- Serializes given variables and adds them to packet given by packet_id */
- PHP_FUNCTION(wddx_add_vars)
- {
- int num_args, i;
- zval *args = NULL;
- zval *packet_id;
- wddx_packet *packet = NULL;
- if (zend_parse_parameters(ZEND_NUM_ARGS(), "r+", &packet_id, &args, &num_args) == FAILURE) {
- return;
- }
- if ((packet = (wddx_packet *)zend_fetch_resource(Z_RES_P(packet_id), "WDDX packet ID", le_wddx)) == NULL) {
- RETURN_FALSE;
- }
- for (i=0; i<num_args; i++) {
- zval *arg;
- if (!Z_ISREF(args[i])) {
- arg = &args[i];
- } else {
- arg = Z_REFVAL(args[i]);
- }
- if (Z_TYPE_P(arg) != IS_ARRAY && Z_TYPE_P(arg) != IS_OBJECT) {
- convert_to_string_ex(arg);
- }
- php_wddx_add_var(packet, arg);
- }
- RETURN_TRUE;
- }
- /* }}} */
- /* {{{ proto mixed wddx_deserialize(mixed packet)
- Deserializes given packet and returns a PHP value */
- PHP_FUNCTION(wddx_deserialize)
- {
- zval *packet;
- php_stream *stream = NULL;
- zend_string *payload = NULL;
- if (zend_parse_parameters(ZEND_NUM_ARGS(), "z", &packet) == FAILURE) {
- return;
- }
- if (Z_TYPE_P(packet) == IS_STRING) {
- payload = Z_STR_P(packet);
- } else if (Z_TYPE_P(packet) == IS_RESOURCE) {
- php_stream_from_zval(stream, packet);
- if (stream) {
- payload = php_stream_copy_to_mem(stream, PHP_STREAM_COPY_ALL, 0);
- }
- } else {
- php_error_docref(NULL, E_WARNING, "Expecting parameter 1 to be a string or a stream");
- return;
- }
- if (payload == NULL) {
- return;
- }
- php_wddx_deserialize_ex(ZSTR_VAL(payload), ZSTR_LEN(payload), return_value);
- if (stream) {
- efree(payload);
- }
- }
- /* }}} */
- #endif /* HAVE_LIBEXPAT */
- /*
- * Local variables:
- * tab-width: 4
- * c-basic-offset: 4
- * End:
- * vim600: sw=4 ts=4 fdm=marker
- * vim<600: sw=4 ts=4
- */
|