Annotation of embedaddon/php/Zend/zend_operators.h, revision 1.1
1.1 ! misho 1: /*
! 2: +----------------------------------------------------------------------+
! 3: | Zend Engine |
! 4: +----------------------------------------------------------------------+
! 5: | Copyright (c) 1998-2012 Zend Technologies Ltd. (http://www.zend.com) |
! 6: +----------------------------------------------------------------------+
! 7: | This source file is subject to version 2.00 of the Zend license, |
! 8: | that is bundled with this package in the file LICENSE, and is |
! 9: | available through the world-wide-web at the following url: |
! 10: | http://www.zend.com/license/2_00.txt. |
! 11: | If you did not receive a copy of the Zend license and are unable to |
! 12: | obtain it through the world-wide-web, please send a note to |
! 13: | license@zend.com so we can mail you a copy immediately. |
! 14: +----------------------------------------------------------------------+
! 15: | Authors: Andi Gutmans <andi@zend.com> |
! 16: | Zeev Suraski <zeev@zend.com> |
! 17: +----------------------------------------------------------------------+
! 18: */
! 19:
! 20: /* $Id: zend_operators.h 321634 2012-01-01 13:15:04Z felipe $ */
! 21:
! 22: #ifndef ZEND_OPERATORS_H
! 23: #define ZEND_OPERATORS_H
! 24:
! 25: #include <errno.h>
! 26: #include <math.h>
! 27: #include <assert.h>
! 28:
! 29: #ifdef HAVE_IEEEFP_H
! 30: #include <ieeefp.h>
! 31: #endif
! 32:
! 33: #include "zend_strtod.h"
! 34:
! 35: #if 0&&HAVE_BCMATH
! 36: #include "ext/bcmath/libbcmath/src/bcmath.h"
! 37: #endif
! 38:
! 39: BEGIN_EXTERN_C()
! 40: ZEND_API int add_function(zval *result, zval *op1, zval *op2 TSRMLS_DC);
! 41: ZEND_API int sub_function(zval *result, zval *op1, zval *op2 TSRMLS_DC);
! 42: ZEND_API int mul_function(zval *result, zval *op1, zval *op2 TSRMLS_DC);
! 43: ZEND_API int div_function(zval *result, zval *op1, zval *op2 TSRMLS_DC);
! 44: ZEND_API int mod_function(zval *result, zval *op1, zval *op2 TSRMLS_DC);
! 45: ZEND_API int boolean_xor_function(zval *result, zval *op1, zval *op2 TSRMLS_DC);
! 46: ZEND_API int boolean_not_function(zval *result, zval *op1 TSRMLS_DC);
! 47: ZEND_API int bitwise_not_function(zval *result, zval *op1 TSRMLS_DC);
! 48: ZEND_API int bitwise_or_function(zval *result, zval *op1, zval *op2 TSRMLS_DC);
! 49: ZEND_API int bitwise_and_function(zval *result, zval *op1, zval *op2 TSRMLS_DC);
! 50: ZEND_API int bitwise_xor_function(zval *result, zval *op1, zval *op2 TSRMLS_DC);
! 51: ZEND_API int shift_left_function(zval *result, zval *op1, zval *op2 TSRMLS_DC);
! 52: ZEND_API int shift_right_function(zval *result, zval *op1, zval *op2 TSRMLS_DC);
! 53: ZEND_API int concat_function(zval *result, zval *op1, zval *op2 TSRMLS_DC);
! 54:
! 55: ZEND_API int is_equal_function(zval *result, zval *op1, zval *op2 TSRMLS_DC);
! 56: ZEND_API int is_identical_function(zval *result, zval *op1, zval *op2 TSRMLS_DC);
! 57: ZEND_API int is_not_identical_function(zval *result, zval *op1, zval *op2 TSRMLS_DC);
! 58: ZEND_API int is_not_equal_function(zval *result, zval *op1, zval *op2 TSRMLS_DC);
! 59: ZEND_API int is_smaller_function(zval *result, zval *op1, zval *op2 TSRMLS_DC);
! 60: ZEND_API int is_smaller_or_equal_function(zval *result, zval *op1, zval *op2 TSRMLS_DC);
! 61:
! 62: ZEND_API zend_bool instanceof_function_ex(const zend_class_entry *instance_ce, const zend_class_entry *ce, zend_bool interfaces_only TSRMLS_DC);
! 63: ZEND_API zend_bool instanceof_function(const zend_class_entry *instance_ce, const zend_class_entry *ce TSRMLS_DC);
! 64: END_EXTERN_C()
! 65:
! 66: #if ZEND_DVAL_TO_LVAL_CAST_OK
! 67: # define zend_dval_to_lval(d) ((long) (d))
! 68: #elif SIZEOF_LONG == 4 && defined(HAVE_ZEND_LONG64)
! 69: static zend_always_inline long zend_dval_to_lval(double d)
! 70: {
! 71: if (d > LONG_MAX || d < LONG_MIN) {
! 72: return (long)(unsigned long)(zend_long64) d;
! 73: }
! 74: return (long) d;
! 75: }
! 76: #else
! 77: static zend_always_inline long zend_dval_to_lval(double d)
! 78: {
! 79: if (d > LONG_MAX) {
! 80: return (long)(unsigned long) d;
! 81: }
! 82: return (long) d;
! 83: }
! 84: #endif
! 85: /* }}} */
! 86:
! 87: #define ZEND_IS_DIGIT(c) ((c) >= '0' && (c) <= '9')
! 88: #define ZEND_IS_XDIGIT(c) (((c) >= 'A' && (c) <= 'F') || ((c) >= 'a' && (c) <= 'f'))
! 89:
! 90: /**
! 91: * Checks whether the string "str" with length "length" is numeric. The value
! 92: * of allow_errors determines whether it's required to be entirely numeric, or
! 93: * just its prefix. Leading whitespace is allowed.
! 94: *
! 95: * The function returns 0 if the string did not contain a valid number; IS_LONG
! 96: * if it contained a number that fits within the range of a long; or IS_DOUBLE
! 97: * if the number was out of long range or contained a decimal point/exponent.
! 98: * The number's value is returned into the respective pointer, *lval or *dval,
! 99: * if that pointer is not NULL.
! 100: */
! 101:
! 102: static inline zend_uchar is_numeric_string(const char *str, int length, long *lval, double *dval, int allow_errors)
! 103: {
! 104: const char *ptr;
! 105: int base = 10, digits = 0, dp_or_e = 0;
! 106: double local_dval;
! 107: zend_uchar type;
! 108:
! 109: if (!length) {
! 110: return 0;
! 111: }
! 112:
! 113: /* Skip any whitespace
! 114: * This is much faster than the isspace() function */
! 115: while (*str == ' ' || *str == '\t' || *str == '\n' || *str == '\r' || *str == '\v' || *str == '\f') {
! 116: str++;
! 117: length--;
! 118: }
! 119: ptr = str;
! 120:
! 121: if (*ptr == '-' || *ptr == '+') {
! 122: ptr++;
! 123: }
! 124:
! 125: if (ZEND_IS_DIGIT(*ptr)) {
! 126: /* Handle hex numbers
! 127: * str is used instead of ptr to disallow signs and keep old behavior */
! 128: if (length > 2 && *str == '0' && (str[1] == 'x' || str[1] == 'X')) {
! 129: base = 16;
! 130: ptr += 2;
! 131: }
! 132:
! 133: /* Skip any leading 0s */
! 134: while (*ptr == '0') {
! 135: ptr++;
! 136: }
! 137:
! 138: /* Count the number of digits. If a decimal point/exponent is found,
! 139: * it's a double. Otherwise, if there's a dval or no need to check for
! 140: * a full match, stop when there are too many digits for a long */
! 141: for (type = IS_LONG; !(digits >= MAX_LENGTH_OF_LONG && (dval || allow_errors == 1)); digits++, ptr++) {
! 142: check_digits:
! 143: if (ZEND_IS_DIGIT(*ptr) || (base == 16 && ZEND_IS_XDIGIT(*ptr))) {
! 144: continue;
! 145: } else if (base == 10) {
! 146: if (*ptr == '.' && dp_or_e < 1) {
! 147: goto process_double;
! 148: } else if ((*ptr == 'e' || *ptr == 'E') && dp_or_e < 2) {
! 149: const char *e = ptr + 1;
! 150:
! 151: if (*e == '-' || *e == '+') {
! 152: ptr = e++;
! 153: }
! 154: if (ZEND_IS_DIGIT(*e)) {
! 155: goto process_double;
! 156: }
! 157: }
! 158: }
! 159:
! 160: break;
! 161: }
! 162:
! 163: if (base == 10) {
! 164: if (digits >= MAX_LENGTH_OF_LONG) {
! 165: dp_or_e = -1;
! 166: goto process_double;
! 167: }
! 168: } else if (!(digits < SIZEOF_LONG * 2 || (digits == SIZEOF_LONG * 2 && ptr[-digits] <= '7'))) {
! 169: if (dval) {
! 170: local_dval = zend_hex_strtod(str, (char **)&ptr);
! 171: }
! 172: type = IS_DOUBLE;
! 173: }
! 174: } else if (*ptr == '.' && ZEND_IS_DIGIT(ptr[1])) {
! 175: process_double:
! 176: type = IS_DOUBLE;
! 177:
! 178: /* If there's a dval, do the conversion; else continue checking
! 179: * the digits if we need to check for a full match */
! 180: if (dval) {
! 181: local_dval = zend_strtod(str, (char **)&ptr);
! 182: } else if (allow_errors != 1 && dp_or_e != -1) {
! 183: dp_or_e = (*ptr++ == '.') ? 1 : 2;
! 184: goto check_digits;
! 185: }
! 186: } else {
! 187: return 0;
! 188: }
! 189:
! 190: if (ptr != str + length) {
! 191: if (!allow_errors) {
! 192: return 0;
! 193: }
! 194: if (allow_errors == -1) {
! 195: zend_error(E_NOTICE, "A non well formed numeric value encountered");
! 196: }
! 197: }
! 198:
! 199: if (type == IS_LONG) {
! 200: if (digits == MAX_LENGTH_OF_LONG - 1) {
! 201: int cmp = strcmp(&ptr[-digits], long_min_digits);
! 202:
! 203: if (!(cmp < 0 || (cmp == 0 && *str == '-'))) {
! 204: if (dval) {
! 205: *dval = zend_strtod(str, NULL);
! 206: }
! 207:
! 208: return IS_DOUBLE;
! 209: }
! 210: }
! 211:
! 212: if (lval) {
! 213: *lval = strtol(str, NULL, base);
! 214: }
! 215:
! 216: return IS_LONG;
! 217: } else {
! 218: if (dval) {
! 219: *dval = local_dval;
! 220: }
! 221:
! 222: return IS_DOUBLE;
! 223: }
! 224: }
! 225:
! 226: static inline char *
! 227: zend_memnstr(char *haystack, char *needle, int needle_len, char *end)
! 228: {
! 229: char *p = haystack;
! 230: char ne = needle[needle_len-1];
! 231:
! 232: if (needle_len == 1) {
! 233: return (char *)memchr(p, *needle, (end-p));
! 234: }
! 235:
! 236: if (needle_len > end-haystack) {
! 237: return NULL;
! 238: }
! 239:
! 240: end -= needle_len;
! 241:
! 242: while (p <= end) {
! 243: if ((p = (char *)memchr(p, *needle, (end-p+1))) && ne == p[needle_len-1]) {
! 244: if (!memcmp(needle, p, needle_len-1)) {
! 245: return p;
! 246: }
! 247: }
! 248:
! 249: if (p == NULL) {
! 250: return NULL;
! 251: }
! 252:
! 253: p++;
! 254: }
! 255:
! 256: return NULL;
! 257: }
! 258:
! 259: static inline void *zend_memrchr(const void *s, int c, size_t n)
! 260: {
! 261: register unsigned char *e;
! 262:
! 263: if (n <= 0) {
! 264: return NULL;
! 265: }
! 266:
! 267: for (e = (unsigned char *)s + n - 1; e >= (unsigned char *)s; e--) {
! 268: if (*e == (unsigned char)c) {
! 269: return (void *)e;
! 270: }
! 271: }
! 272:
! 273: return NULL;
! 274: }
! 275:
! 276: BEGIN_EXTERN_C()
! 277: ZEND_API int increment_function(zval *op1);
! 278: ZEND_API int decrement_function(zval *op2);
! 279:
! 280: ZEND_API void convert_scalar_to_number(zval *op TSRMLS_DC);
! 281: ZEND_API void _convert_to_string(zval *op ZEND_FILE_LINE_DC);
! 282: ZEND_API void convert_to_long(zval *op);
! 283: ZEND_API void convert_to_double(zval *op);
! 284: ZEND_API void convert_to_long_base(zval *op, int base);
! 285: ZEND_API void convert_to_null(zval *op);
! 286: ZEND_API void convert_to_boolean(zval *op);
! 287: ZEND_API void convert_to_array(zval *op);
! 288: ZEND_API void convert_to_object(zval *op);
! 289: ZEND_API void multi_convert_to_long_ex(int argc, ...);
! 290: ZEND_API void multi_convert_to_double_ex(int argc, ...);
! 291: ZEND_API void multi_convert_to_string_ex(int argc, ...);
! 292: ZEND_API int add_char_to_string(zval *result, const zval *op1, const zval *op2);
! 293: ZEND_API int add_string_to_string(zval *result, const zval *op1, const zval *op2);
! 294: #define convert_to_string(op) if ((op)->type != IS_STRING) { _convert_to_string((op) ZEND_FILE_LINE_CC); }
! 295:
! 296: ZEND_API double zend_string_to_double(const char *number, zend_uint length);
! 297:
! 298: ZEND_API int zval_is_true(zval *op);
! 299: ZEND_API int compare_function(zval *result, zval *op1, zval *op2 TSRMLS_DC);
! 300: ZEND_API int numeric_compare_function(zval *result, zval *op1, zval *op2 TSRMLS_DC);
! 301: ZEND_API int string_compare_function(zval *result, zval *op1, zval *op2 TSRMLS_DC);
! 302: #if HAVE_STRCOLL
! 303: ZEND_API int string_locale_compare_function(zval *result, zval *op1, zval *op2 TSRMLS_DC);
! 304: #endif
! 305:
! 306: ZEND_API void zend_str_tolower(char *str, unsigned int length);
! 307: ZEND_API char *zend_str_tolower_copy(char *dest, const char *source, unsigned int length);
! 308: ZEND_API char *zend_str_tolower_dup(const char *source, unsigned int length);
! 309:
! 310: ZEND_API int zend_binary_zval_strcmp(zval *s1, zval *s2);
! 311: ZEND_API int zend_binary_zval_strncmp(zval *s1, zval *s2, zval *s3);
! 312: ZEND_API int zend_binary_zval_strcasecmp(zval *s1, zval *s2);
! 313: ZEND_API int zend_binary_zval_strncasecmp(zval *s1, zval *s2, zval *s3);
! 314: ZEND_API int zend_binary_strcmp(const char *s1, uint len1, const char *s2, uint len2);
! 315: ZEND_API int zend_binary_strncmp(const char *s1, uint len1, const char *s2, uint len2, uint length);
! 316: ZEND_API int zend_binary_strcasecmp(const char *s1, uint len1, const char *s2, uint len2);
! 317: ZEND_API int zend_binary_strncasecmp(const char *s1, uint len1, const char *s2, uint len2, uint length);
! 318:
! 319: ZEND_API void zendi_smart_strcmp(zval *result, zval *s1, zval *s2);
! 320: ZEND_API void zend_compare_symbol_tables(zval *result, HashTable *ht1, HashTable *ht2 TSRMLS_DC);
! 321: ZEND_API void zend_compare_arrays(zval *result, zval *a1, zval *a2 TSRMLS_DC);
! 322: ZEND_API void zend_compare_objects(zval *result, zval *o1, zval *o2 TSRMLS_DC);
! 323:
! 324: ZEND_API int zend_atoi(const char *str, int str_len);
! 325: ZEND_API long zend_atol(const char *str, int str_len);
! 326:
! 327: ZEND_API void zend_locale_sprintf_double(zval *op ZEND_FILE_LINE_DC);
! 328: END_EXTERN_C()
! 329:
! 330: #define convert_to_ex_master(ppzv, lower_type, upper_type) \
! 331: if (Z_TYPE_PP(ppzv)!=IS_##upper_type) { \
! 332: SEPARATE_ZVAL_IF_NOT_REF(ppzv); \
! 333: convert_to_##lower_type(*ppzv); \
! 334: }
! 335:
! 336: #define convert_to_explicit_type(pzv, type) \
! 337: do { \
! 338: switch (type) { \
! 339: case IS_NULL: \
! 340: convert_to_null(pzv); \
! 341: break; \
! 342: case IS_LONG: \
! 343: convert_to_long(pzv); \
! 344: break; \
! 345: case IS_DOUBLE: \
! 346: convert_to_double(pzv); \
! 347: break; \
! 348: case IS_BOOL: \
! 349: convert_to_boolean(pzv); \
! 350: break; \
! 351: case IS_ARRAY: \
! 352: convert_to_array(pzv); \
! 353: break; \
! 354: case IS_OBJECT: \
! 355: convert_to_object(pzv); \
! 356: break; \
! 357: case IS_STRING: \
! 358: convert_to_string(pzv); \
! 359: break; \
! 360: default: \
! 361: assert(0); \
! 362: break; \
! 363: } \
! 364: } while (0);
! 365:
! 366: #define convert_to_explicit_type_ex(ppzv, str_type) \
! 367: if (Z_TYPE_PP(ppzv) != str_type) { \
! 368: SEPARATE_ZVAL_IF_NOT_REF(ppzv); \
! 369: convert_to_explicit_type(*ppzv, str_type); \
! 370: }
! 371:
! 372: #define convert_to_boolean_ex(ppzv) convert_to_ex_master(ppzv, boolean, BOOL)
! 373: #define convert_to_long_ex(ppzv) convert_to_ex_master(ppzv, long, LONG)
! 374: #define convert_to_double_ex(ppzv) convert_to_ex_master(ppzv, double, DOUBLE)
! 375: #define convert_to_string_ex(ppzv) convert_to_ex_master(ppzv, string, STRING)
! 376: #define convert_to_array_ex(ppzv) convert_to_ex_master(ppzv, array, ARRAY)
! 377: #define convert_to_object_ex(ppzv) convert_to_ex_master(ppzv, object, OBJECT)
! 378: #define convert_to_null_ex(ppzv) convert_to_ex_master(ppzv, null, NULL)
! 379:
! 380: #define convert_scalar_to_number_ex(ppzv) \
! 381: if (Z_TYPE_PP(ppzv)!=IS_LONG && Z_TYPE_PP(ppzv)!=IS_DOUBLE) { \
! 382: if (!Z_ISREF_PP(ppzv)) { \
! 383: SEPARATE_ZVAL(ppzv); \
! 384: } \
! 385: convert_scalar_to_number(*ppzv TSRMLS_CC); \
! 386: }
! 387:
! 388:
! 389: #define Z_LVAL(zval) (zval).value.lval
! 390: #define Z_BVAL(zval) ((zend_bool)(zval).value.lval)
! 391: #define Z_DVAL(zval) (zval).value.dval
! 392: #define Z_STRVAL(zval) (zval).value.str.val
! 393: #define Z_STRLEN(zval) (zval).value.str.len
! 394: #define Z_ARRVAL(zval) (zval).value.ht
! 395: #define Z_OBJVAL(zval) (zval).value.obj
! 396: #define Z_OBJ_HANDLE(zval) Z_OBJVAL(zval).handle
! 397: #define Z_OBJ_HT(zval) Z_OBJVAL(zval).handlers
! 398: #define Z_OBJCE(zval) zend_get_class_entry(&(zval) TSRMLS_CC)
! 399: #define Z_OBJPROP(zval) Z_OBJ_HT((zval))->get_properties(&(zval) TSRMLS_CC)
! 400: #define Z_OBJ_HANDLER(zval, hf) Z_OBJ_HT((zval))->hf
! 401: #define Z_RESVAL(zval) (zval).value.lval
! 402: #define Z_OBJDEBUG(zval,is_tmp) (Z_OBJ_HANDLER((zval),get_debug_info)?Z_OBJ_HANDLER((zval),get_debug_info)(&(zval),&is_tmp TSRMLS_CC):(is_tmp=0,Z_OBJ_HANDLER((zval),get_properties)?Z_OBJPROP(zval):NULL))
! 403:
! 404: #define Z_LVAL_P(zval_p) Z_LVAL(*zval_p)
! 405: #define Z_BVAL_P(zval_p) Z_BVAL(*zval_p)
! 406: #define Z_DVAL_P(zval_p) Z_DVAL(*zval_p)
! 407: #define Z_STRVAL_P(zval_p) Z_STRVAL(*zval_p)
! 408: #define Z_STRLEN_P(zval_p) Z_STRLEN(*zval_p)
! 409: #define Z_ARRVAL_P(zval_p) Z_ARRVAL(*zval_p)
! 410: #define Z_OBJPROP_P(zval_p) Z_OBJPROP(*zval_p)
! 411: #define Z_OBJCE_P(zval_p) Z_OBJCE(*zval_p)
! 412: #define Z_RESVAL_P(zval_p) Z_RESVAL(*zval_p)
! 413: #define Z_OBJVAL_P(zval_p) Z_OBJVAL(*zval_p)
! 414: #define Z_OBJ_HANDLE_P(zval_p) Z_OBJ_HANDLE(*zval_p)
! 415: #define Z_OBJ_HT_P(zval_p) Z_OBJ_HT(*zval_p)
! 416: #define Z_OBJ_HANDLER_P(zval_p, h) Z_OBJ_HANDLER(*zval_p, h)
! 417: #define Z_OBJDEBUG_P(zval_p,is_tmp) Z_OBJDEBUG(*zval_p,is_tmp)
! 418:
! 419: #define Z_LVAL_PP(zval_pp) Z_LVAL(**zval_pp)
! 420: #define Z_BVAL_PP(zval_pp) Z_BVAL(**zval_pp)
! 421: #define Z_DVAL_PP(zval_pp) Z_DVAL(**zval_pp)
! 422: #define Z_STRVAL_PP(zval_pp) Z_STRVAL(**zval_pp)
! 423: #define Z_STRLEN_PP(zval_pp) Z_STRLEN(**zval_pp)
! 424: #define Z_ARRVAL_PP(zval_pp) Z_ARRVAL(**zval_pp)
! 425: #define Z_OBJPROP_PP(zval_pp) Z_OBJPROP(**zval_pp)
! 426: #define Z_OBJCE_PP(zval_pp) Z_OBJCE(**zval_pp)
! 427: #define Z_RESVAL_PP(zval_pp) Z_RESVAL(**zval_pp)
! 428: #define Z_OBJVAL_PP(zval_pp) Z_OBJVAL(**zval_pp)
! 429: #define Z_OBJ_HANDLE_PP(zval_p) Z_OBJ_HANDLE(**zval_p)
! 430: #define Z_OBJ_HT_PP(zval_p) Z_OBJ_HT(**zval_p)
! 431: #define Z_OBJ_HANDLER_PP(zval_p, h) Z_OBJ_HANDLER(**zval_p, h)
! 432: #define Z_OBJDEBUG_PP(zval_pp,is_tmp) Z_OBJDEBUG(**zval_pp,is_tmp)
! 433:
! 434: #define Z_TYPE(zval) (zval).type
! 435: #define Z_TYPE_P(zval_p) Z_TYPE(*zval_p)
! 436: #define Z_TYPE_PP(zval_pp) Z_TYPE(**zval_pp)
! 437:
! 438: #if HAVE_SETLOCALE && defined(ZEND_WIN32) && !defined(ZTS) && defined(_MSC_VER) && (_MSC_VER >= 1400)
! 439: /* This is performance improvement of tolower() on Windows and VC2005
! 440: * Gives 10-18% on bench.php
! 441: */
! 442: #define ZEND_USE_TOLOWER_L 1
! 443: #endif
! 444:
! 445: #ifdef ZEND_USE_TOLOWER_L
! 446: ZEND_API void zend_update_current_locale(void);
! 447: #else
! 448: #define zend_update_current_locale()
! 449: #endif
! 450:
! 451: #endif
! 452:
! 453: /*
! 454: * Local variables:
! 455: * tab-width: 4
! 456: * c-basic-offset: 4
! 457: * indent-tabs-mode: t
! 458: * End:
! 459: */
FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>