14.6.4. Accessor Macros
A large set of
macros makes it easy to access
fields of a zval. For example:
zval foo;
char *string;
/* initialize foo and string */
Z_STRVAL(foo) = string;
The Z_STRVAL( ) macro accesses the string field of
a zval. There are accessor macros for every data
type that can be stored in a zval. Because you
often have pointers to zvals, and sometimes even
pointers to pointers to zvals, each macro comes in
three flavors, as shown in Table 14-1.
Table 14-1. zval accessor macros
Long
|
Boolean
|
Double
|
String value
|
String length
|
Z_LVAL( )
|
Z_BVAL( )
|
Z_DVAL( )
|
Z_STRVAL( )
|
Z_STRLEN( )
|
Z_LVAL_P( )
|
Z_BVAL_P( )
|
Z_DVAL_P( )
|
Z_STRVAL_P( )
|
Z_STRLEN_P( )
|
Z_LVAL_PP( )
|
Z_BVAL_PP( )
|
Z_DVAL_PP( )
|
Z_STRVAL_PP( )
|
Z_STRLEN_PP( )
|
HashTable
|
Object
|
Object properties
|
Object class entry
|
Resource value
|
Z_ARRVAL( )
|
Z_OBJ( )
|
Z_OBJPROP( )
|
Z_OBJCE( )
|
Z_RESVAL( )
|
Z_ARRVAL_P( )
|
Z_OBJ_P( )
|
Z_OBJPROP_P( )
|
Z_OBJCE_P( )
|
Z_RESVAL_P( )
|
Z_ARRVAL_PP( )
|
Z_OBJ_PP( )
|
Z_OBJPROP_PP( )
|
Z_OBJCE_PP( )
|
Z_RESVAL_PP( )
|
There are macros to identify the active type of a
zval (or zval
*, or zval
**). They are Z_TYPE( ),
Z_TYPE_P( ), and Z_TYPE_PP( ).
The possible return values are:
-
IS_LONG
-
IS_BOOL
-
IS_DOUBLE
-
IS_STRING
-
IS_ARRAY
-
IS_OBJECT
-
IS_RESOURCE
-
IS_NULL
The following code shows the rot13( ) function
rewritten using low-level functions:
PHP_FUNCTION(rot13)
{
zval **arg;
char *ch, cap;
int i;
if (ZEND_NUM_ARGS( ) != 1 || zend_get_parameters_ex(1, &arg) == FAILURE) {
WRONG_PARAM_COUNT;
}
SEPARATE_ZVAL(arg);
convert_to_string_ex(arg);
for(i=0, ch=Z_STRVAL_PP(arg); i<Z_STRLEN_PP(arg); i++, ch++) {
cap = *ch & 32;
*ch &= ~cap;
*ch = ((*ch>='A') && (*ch<='Z') ? ((*ch-'A'+13) % 26+'A') : *ch) | cap;
}
RETURN_STRINGL(Z_STRVAL_PP(arg), Z_STRLEN_PP(arg), 1);
}
Rather than using the handy zend_parse_parameters(
) function, we fetch the zval directly
using zend_get_parameters_ex(
). We then create a separate copy so
that we can modify this copy without changing the passed container
directly. Then we return it. Note that this is not an improvement on
our function, merely a rewrite to show how you might use the various
accessor macros.
Here's an even lower-level approach that skips the
SEPARATE_ZVAL( ) approach and goes right to a
zval_copy_ctor(
):
PHP_FUNCTION(rot13)
{
zval **arg;
char *ch, cap;
int i;
if (ZEND_NUM_ARGS( ) != 1 || zend_get_parameters_ex(1, &arg) == FAILURE) {
WRONG_PARAM_COUNT;
}
*return_value = **arg;
zval_copy_ctor(return_value);
convert_to_string(return_value);
for(i=0, ch=return_value->value.str.val;
i<return_value->value.str.len; i++, ch++) {
cap = *ch & 32;
*ch &= ~cap;
*ch = ((*ch>='A') && (*ch<='Z') ? ((*ch-'A'+13) % 26 + 'A') : *ch) | cap;
}
}
The value returned from a PHP function is returned in a special
zval container called
return_value, which is automatically allocated. In
the example, we assign return_value to the passed
arg container, call zval_copy_ctor(
) to make a copy, and ensure that we convert the data to a
string.
We also skipped the
zval
dereferencing convenience macros
Z_STRVAL_PP( ) and Z_STRLEN_PP(
) and instead dereferenced the
return_value zval container
manually. Going this low-level is not recommended, however, as
changes in the underlying data structures could break your
extension.