1999-04-08 02:10:10 +08:00
/*
+ - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - +
| Zend Engine |
+ - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - +
2010-01-06 04:46:53 +08:00
| Copyright ( c ) 1998 - 2010 Zend Technologies Ltd . ( http : //www.zend.com) |
1999-04-08 02:10:10 +08:00
+ - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - +
2001-12-11 23:16:21 +08:00
| This source file is subject to version 2.00 of the Zend license , |
1999-07-16 22:58:16 +08:00
| that is bundled with this package in the file LICENSE , and is |
2003-06-11 04:04:29 +08:00
| available through the world - wide - web at the following url : |
2001-12-11 23:16:21 +08:00
| http : //www.zend.com/license/2_00.txt. |
1999-07-16 22:58:16 +08:00
| If you did not receive a copy of the Zend license and are unable to |
| obtain it through the world - wide - web , please send a note to |
| license @ zend . com so we can mail you a copy immediately . |
1999-04-08 02:10:10 +08:00
+ - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - +
| Authors : Andi Gutmans < andi @ zend . com > |
| Zeev Suraski < zeev @ zend . com > |
+ - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - +
*/
2003-02-01 09:49:15 +08:00
/* $Id$ */
1999-07-16 22:58:16 +08:00
2004-12-30 23:18:24 +08:00
# include <zend_language_parser.h>
1999-04-08 02:10:10 +08:00
# include "zend.h"
# include "zend_compile.h"
2003-04-11 01:02:31 +08:00
# include "zend_constants.h"
1999-04-08 02:10:10 +08:00
# include "zend_llist.h"
# include "zend_API.h"
2007-02-24 10:17:47 +08:00
# include "zend_exceptions.h"
2008-02-12 08:21:15 +08:00
# include "tsrm_virtual_cwd.h"
1999-04-08 02:10:10 +08:00
2008-06-29 16:21:35 +08:00
# ifdef ZEND_MULTIBYTE
# include "zend_multibyte.h"
# endif /* ZEND_MULTIBYTE */
2010-04-20 18:57:45 +08:00
# define CONSTANT_EX(op_array, op) \
( op_array ) - > literals [ op ] . constant
# define CONSTANT(op) \
CONSTANT_EX ( CG ( active_op_array ) , op )
# define SET_NODE(target, src) do { \
target # # _type = ( src ) - > op_type ; \
if ( ( src ) - > op_type = = IS_CONST ) { \
target . constant = zend_add_literal ( CG ( active_op_array ) , & ( src ) - > u . constant ) ; \
} else { \
target = ( src ) - > u . op ; \
} \
} while ( 0 )
# define GET_NODE(target, src) do { \
( target ) - > op_type = src # # _type ; \
if ( ( target ) - > op_type = = IS_CONST ) { \
( target ) - > u . constant = CONSTANT ( src . constant ) ; \
} else { \
( target ) - > u . op = src ; \
( target ) - > EA = 0 ; \
} \
} while ( 0 )
# define COPY_NODE(target, src) do { \
target # # _type = src # # _type ; \
target = src ; \
} while ( 0 )
2010-04-21 22:58:33 +08:00
# define CALCULATE_LITERAL_HASH(num) do { \
if ( IS_INTERNED ( Z_STRVAL ( CONSTANT ( num ) ) ) ) { \
Z_HASH_P ( & CONSTANT ( num ) ) = INTERNED_HASH ( Z_STRVAL ( CONSTANT ( num ) ) ) ; \
} else { \
Z_HASH_P ( & CONSTANT ( num ) ) = zend_hash_func ( Z_STRVAL ( CONSTANT ( num ) ) , Z_STRLEN ( CONSTANT ( num ) ) + 1 ) ; \
} \
} while ( 0 )
2010-04-20 18:57:45 +08:00
2001-07-28 18:51:54 +08:00
ZEND_API zend_op_array * ( * zend_compile_file ) ( zend_file_handle * file_handle , int type TSRMLS_DC ) ;
2006-06-13 20:57:48 +08:00
ZEND_API zend_op_array * ( * zend_compile_string ) ( zval * source_string , char * filename TSRMLS_DC ) ;
1999-04-08 02:10:10 +08:00
# ifndef ZTS
ZEND_API zend_compiler_globals compiler_globals ;
ZEND_API zend_executor_globals executor_globals ;
# endif
2009-07-27 22:11:53 +08:00
static void zend_duplicate_property_info ( zend_property_info * property_info ) /* { { { */
2003-08-24 03:37:39 +08:00
{
2010-04-20 19:05:54 +08:00
if ( ! IS_INTERNED ( property_info - > name ) ) {
property_info - > name = estrndup ( property_info - > name , property_info - > name_length ) ;
}
2005-04-25 01:32:05 +08:00
if ( property_info - > doc_comment ) {
property_info - > doc_comment = estrndup ( property_info - > doc_comment , property_info - > doc_comment_len ) ;
}
2003-08-24 03:37:39 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
2003-08-24 03:37:39 +08:00
2009-07-27 22:11:53 +08:00
static void zend_duplicate_property_info_internal ( zend_property_info * property_info ) /* { { { */
2003-08-24 03:37:39 +08:00
{
2010-04-20 19:05:54 +08:00
if ( ! IS_INTERNED ( property_info - > name ) ) {
property_info - > name = zend_strndup ( property_info - > name , property_info - > name_length ) ;
}
2003-08-24 03:37:39 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
2003-08-24 03:37:39 +08:00
2009-07-27 22:11:53 +08:00
static void zend_destroy_property_info ( zend_property_info * property_info ) /* { { { */
2003-08-24 03:37:39 +08:00
{
2010-04-20 19:05:54 +08:00
str_efree ( property_info - > name ) ;
2005-04-20 06:04:59 +08:00
if ( property_info - > doc_comment ) {
efree ( property_info - > doc_comment ) ;
}
2003-08-24 03:37:39 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
2003-08-24 03:37:39 +08:00
2009-07-27 22:11:53 +08:00
static void zend_destroy_property_info_internal ( zend_property_info * property_info ) /* { { { */
2003-08-24 03:37:39 +08:00
{
2010-04-20 19:05:54 +08:00
str_free ( property_info - > name ) ;
2003-08-24 03:37:39 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
2003-08-24 03:37:39 +08:00
2009-07-27 22:11:53 +08:00
static void build_runtime_defined_function_key ( zval * result , const char * name , int name_length TSRMLS_DC ) /* { { { */
1999-09-03 05:06:05 +08:00
{
2004-01-12 03:27:13 +08:00
char char_pos_buf [ 32 ] ;
uint char_pos_len ;
1999-09-03 05:06:05 +08:00
char * filename ;
2008-03-17 05:06:55 +08:00
char_pos_len = zend_sprintf ( char_pos_buf , " %p " , LANG_SCNG ( yy_text ) ) ;
2000-08-10 03:22:35 +08:00
if ( CG ( active_op_array ) - > filename ) {
filename = CG ( active_op_array ) - > filename ;
1999-09-03 05:06:05 +08:00
} else {
filename = " - " ;
}
2004-01-12 03:27:13 +08:00
/* NULL, name length, filename length, last accepting char position length */
result - > value . str . len = 1 + name_length + strlen ( filename ) + char_pos_len ;
2008-06-29 16:21:35 +08:00
# ifdef ZEND_MULTIBYTE
/* must be binary safe */
result - > value . str . val = ( char * ) safe_emalloc ( result - > value . str . len , 1 , 1 ) ;
result - > value . str . val [ 0 ] = ' \0 ' ;
sprintf ( result - > value . str . val + 1 , " %s%s%s " , name , filename , char_pos_buf ) ;
# else
2007-02-24 10:17:47 +08:00
zend_spprintf ( & result - > value . str . val , 0 , " %c%s%s%s " , ' \0 ' , name , filename , char_pos_buf ) ;
2008-06-29 16:21:35 +08:00
# endif /* ZEND_MULTIBYTE */
1999-09-03 05:06:05 +08:00
result - > type = IS_STRING ;
2007-10-07 13:22:07 +08:00
Z_SET_REFCOUNT_P ( result , 1 ) ;
1999-09-03 05:06:05 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-09-03 05:06:05 +08:00
2009-07-27 22:11:53 +08:00
int zend_auto_global_arm ( zend_auto_global * auto_global TSRMLS_DC ) /* { { { */
2003-03-02 18:04:53 +08:00
{
auto_global - > armed = ( auto_global - > auto_global_callback ? 1 : 0 ) ;
return 0 ;
}
2009-07-27 22:11:53 +08:00
/* }}} */
2003-03-02 18:04:53 +08:00
2009-07-27 22:11:53 +08:00
ZEND_API int zend_auto_global_disable_jit ( const char * varname , zend_uint varname_length TSRMLS_DC ) /* { { { */
2004-03-17 03:46:04 +08:00
{
zend_auto_global * auto_global ;
if ( zend_hash_find ( CG ( auto_globals ) , varname , varname_length + 1 , ( void * * ) & auto_global ) = = FAILURE ) {
return FAILURE ;
}
auto_global - > armed = 0 ;
return SUCCESS ;
}
2009-07-27 22:11:53 +08:00
/* }}} */
2004-03-17 03:46:04 +08:00
2009-07-27 22:11:53 +08:00
static void init_compiler_declarables ( TSRMLS_D ) /* { { { */
2000-01-25 03:00:30 +08:00
{
2006-05-10 07:53:23 +08:00
Z_TYPE ( CG ( declarables ) . ticks ) = IS_LONG ;
Z_LVAL ( CG ( declarables ) . ticks ) = 0 ;
2000-01-25 03:00:30 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
2000-01-25 03:00:30 +08:00
2009-07-27 22:11:53 +08:00
void zend_init_compiler_data_structures ( TSRMLS_D ) /* { { { */
1999-04-08 02:10:10 +08:00
{
zend_stack_init ( & CG ( bp_stack ) ) ;
zend_stack_init ( & CG ( function_call_stack ) ) ;
zend_stack_init ( & CG ( switch_cond_stack ) ) ;
1999-09-09 22:15:17 +08:00
zend_stack_init ( & CG ( foreach_copy_stack ) ) ;
1999-04-08 02:10:10 +08:00
zend_stack_init ( & CG ( object_stack ) ) ;
2000-01-25 03:00:30 +08:00
zend_stack_init ( & CG ( declare_stack ) ) ;
1999-04-08 02:10:10 +08:00
CG ( active_class_entry ) = NULL ;
zend_llist_init ( & CG ( list_llist ) , sizeof ( list_llist_element ) , NULL , 0 ) ;
zend_llist_init ( & CG ( dimension_llist ) , sizeof ( int ) , NULL , 0 ) ;
2001-07-16 23:48:31 +08:00
zend_stack_init ( & CG ( list_stack ) ) ;
2000-02-04 22:45:58 +08:00
CG ( in_compilation ) = 0 ;
2003-01-06 00:09:59 +08:00
CG ( start_lineno ) = 0 ;
2007-09-29 03:52:53 +08:00
CG ( current_namespace ) = NULL ;
2008-11-25 17:56:32 +08:00
CG ( in_namespace ) = 0 ;
CG ( has_bracketed_namespaces ) = 0 ;
2007-09-29 03:52:53 +08:00
CG ( current_import ) = NULL ;
2001-07-28 18:51:54 +08:00
init_compiler_declarables ( TSRMLS_C ) ;
2003-03-02 18:04:53 +08:00
zend_hash_apply ( CG ( auto_globals ) , ( apply_func_t ) zend_auto_global_arm TSRMLS_CC ) ;
2008-03-28 22:35:01 +08:00
zend_stack_init ( & CG ( labels_stack ) ) ;
CG ( labels ) = NULL ;
2008-06-29 16:21:35 +08:00
# ifdef ZEND_MULTIBYTE
CG ( script_encoding_list ) = NULL ;
CG ( script_encoding_list_size ) = 0 ;
CG ( internal_encoding ) = NULL ;
CG ( encoding_detector ) = NULL ;
CG ( encoding_converter ) = NULL ;
CG ( encoding_oddlen ) = NULL ;
2009-11-17 06:52:00 +08:00
CG ( encoding_declared ) = 0 ;
2008-06-29 16:21:35 +08:00
# endif /* ZEND_MULTIBYTE */
2008-03-17 05:06:55 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
2008-03-17 05:06:55 +08:00
2009-07-27 22:11:53 +08:00
ZEND_API void file_handle_dtor ( zend_file_handle * fh ) /* { { { */
2008-03-17 05:06:55 +08:00
{
TSRMLS_FETCH ( ) ;
2003-08-11 13:24:42 +08:00
2008-03-17 05:06:55 +08:00
zend_file_handle_dtor ( fh TSRMLS_CC ) ;
2001-05-07 03:30:31 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
2001-05-07 03:30:31 +08:00
2009-07-27 22:11:53 +08:00
void init_compiler ( TSRMLS_D ) /* { { { */
2001-05-07 03:30:31 +08:00
{
2006-05-02 23:49:26 +08:00
CG ( active_op_array ) = NULL ;
2001-07-28 18:51:54 +08:00
zend_init_compiler_data_structures ( TSRMLS_C ) ;
2001-07-27 18:10:39 +08:00
zend_init_rsrc_list ( TSRMLS_C ) ;
2001-05-07 03:30:31 +08:00
zend_hash_init ( & CG ( filenames_table ) , 5 , NULL , ( dtor_func_t ) free_estring , 0 ) ;
2008-03-17 05:06:55 +08:00
zend_llist_init ( & CG ( open_files ) , sizeof ( zend_file_handle ) , ( void ( * ) ( void * ) ) file_handle_dtor , 0 ) ;
2001-05-07 03:30:31 +08:00
CG ( unclean_shutdown ) = 0 ;
1999-04-08 02:10:10 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
void shutdown_compiler ( TSRMLS_D ) /* { { { */
1999-04-08 02:10:10 +08:00
{
zend_stack_destroy ( & CG ( bp_stack ) ) ;
zend_stack_destroy ( & CG ( function_call_stack ) ) ;
zend_stack_destroy ( & CG ( switch_cond_stack ) ) ;
1999-09-09 22:15:17 +08:00
zend_stack_destroy ( & CG ( foreach_copy_stack ) ) ;
1999-04-08 02:10:10 +08:00
zend_stack_destroy ( & CG ( object_stack ) ) ;
2000-01-25 03:00:30 +08:00
zend_stack_destroy ( & CG ( declare_stack ) ) ;
2001-07-16 23:48:31 +08:00
zend_stack_destroy ( & CG ( list_stack ) ) ;
2000-08-20 01:50:42 +08:00
zend_hash_destroy ( & CG ( filenames_table ) ) ;
1999-05-22 10:13:01 +08:00
zend_llist_destroy ( & CG ( open_files ) ) ;
2008-03-28 22:35:01 +08:00
zend_stack_destroy ( & CG ( labels_stack ) ) ;
2008-06-29 16:21:35 +08:00
# ifdef ZEND_MULTIBYTE
if ( CG ( script_encoding_list ) ) {
efree ( CG ( script_encoding_list ) ) ;
}
# endif /* ZEND_MULTIBYTE */
1999-04-08 02:10:10 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
ZEND_API char * zend_set_compiled_filename ( const char * new_compiled_filename TSRMLS_DC ) /* { { { */
1999-04-08 02:10:10 +08:00
{
2000-08-20 01:50:42 +08:00
char * * pp , * p ;
int length = strlen ( new_compiled_filename ) ;
1999-04-08 02:10:10 +08:00
2004-10-17 07:16:50 +08:00
if ( zend_hash_find ( & CG ( filenames_table ) , new_compiled_filename , length + 1 , ( void * * ) & pp ) = = SUCCESS ) {
2000-09-17 16:58:39 +08:00
CG ( compiled_filename ) = * pp ;
2000-08-20 01:50:42 +08:00
return * pp ;
}
p = estrndup ( new_compiled_filename , length ) ;
zend_hash_update ( & CG ( filenames_table ) , new_compiled_filename , length + 1 , & p , sizeof ( char * ) , ( void * * ) & pp ) ;
1999-04-08 02:10:10 +08:00
CG ( compiled_filename ) = p ;
1999-08-03 02:40:10 +08:00
return p ;
1999-04-08 02:10:10 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
ZEND_API void zend_restore_compiled_filename ( char * original_compiled_filename TSRMLS_DC ) /* { { { */
1999-04-08 02:10:10 +08:00
{
CG ( compiled_filename ) = original_compiled_filename ;
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2000-02-04 22:45:58 +08:00
2009-07-27 22:11:53 +08:00
ZEND_API char * zend_get_compiled_filename ( TSRMLS_D ) /* { { { */
2000-02-04 22:45:58 +08:00
{
return CG ( compiled_filename ) ;
}
2009-07-27 22:11:53 +08:00
/* }}} */
2000-02-04 22:45:58 +08:00
2009-07-27 22:11:53 +08:00
ZEND_API int zend_get_compiled_lineno ( TSRMLS_D ) /* { { { */
2000-02-04 22:45:58 +08:00
{
return CG ( zend_lineno ) ;
}
2009-07-27 22:11:53 +08:00
/* }}} */
2000-02-04 22:45:58 +08:00
2009-07-27 22:11:53 +08:00
ZEND_API zend_bool zend_is_compiling ( TSRMLS_D ) /* { { { */
1999-04-08 02:10:10 +08:00
{
2000-02-04 22:45:58 +08:00
return CG ( in_compilation ) ;
1999-04-08 02:10:10 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
static zend_uint get_temporary_variable ( zend_op_array * op_array ) /* { { { */
1999-04-08 02:10:10 +08:00
{
2009-09-03 22:33:11 +08:00
return ( op_array - > T ) + + * ZEND_MM_ALIGNED_SIZE ( sizeof ( temp_variable ) ) ;
1999-04-08 02:10:10 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2010-04-20 19:05:54 +08:00
static int lookup_cv ( zend_op_array * op_array , char * name , int name_len TSRMLS_DC ) /* { { { */
2004-10-05 03:54:35 +08:00
{
int i = 0 ;
ulong hash_value = zend_inline_hash_func ( name , name_len + 1 ) ;
while ( i < op_array - > last_var ) {
2010-04-20 19:05:54 +08:00
if ( op_array - > vars [ i ] . name = = name | |
( op_array - > vars [ i ] . hash_value = = hash_value & &
op_array - > vars [ i ] . name_len = = name_len & &
memcmp ( op_array - > vars [ i ] . name , name , name_len ) = = 0 ) ) {
str_efree ( name ) ;
return i ;
2004-10-05 03:54:35 +08:00
}
i + + ;
}
i = op_array - > last_var ;
op_array - > last_var + + ;
if ( op_array - > last_var > op_array - > size_var ) {
2006-05-10 07:53:23 +08:00
op_array - > size_var + = 16 ; /* FIXME */
2004-10-05 03:54:35 +08:00
op_array - > vars = erealloc ( op_array - > vars , op_array - > size_var * sizeof ( zend_compiled_variable ) ) ;
}
2010-04-20 19:05:54 +08:00
op_array - > vars [ i ] . name = CG ( new_interned_string ) ( name , name_len + 1 , 1 TSRMLS_CC ) ;
2004-10-05 03:54:35 +08:00
op_array - > vars [ i ] . name_len = name_len ;
op_array - > vars [ i ] . hash_value = hash_value ;
return i ;
}
2009-07-27 22:11:53 +08:00
/* }}} */
2004-10-05 03:54:35 +08:00
2010-04-20 18:57:45 +08:00
void zend_del_literal ( zend_op_array * op_array , int n ) /* { { { */
{
zval_dtor ( & CONSTANT_EX ( op_array , n ) ) ;
if ( n + 1 = = op_array - > last_literal ) {
op_array - > last_literal - - ;
} else {
Z_TYPE ( CONSTANT_EX ( op_array , n ) ) = IS_NULL ;
}
}
/* }}} */
1999-04-08 02:10:10 +08:00
2010-04-20 18:57:45 +08:00
int zend_add_literal ( zend_op_array * op_array , const zval * zv ) /* { { { */
{
int i = op_array - > last_literal ;
op_array - > last_literal + + ;
if ( i > = op_array - > size_literal ) {
op_array - > size_literal + = 16 ; /* FIXME */
op_array - > literals = ( zend_literal * ) erealloc ( op_array - > literals , op_array - > size_literal * sizeof ( zend_literal ) ) ;
}
2010-04-20 19:05:54 +08:00
if ( Z_TYPE_P ( zv ) = = IS_STRING | | Z_TYPE_P ( zv ) = = IS_CONSTANT ) {
zval * z = ( zval * ) zv ;
TSRMLS_FETCH ( ) ;
Z_STRVAL_P ( z ) =
CG ( new_interned_string ) ( Z_STRVAL_P ( zv ) , Z_STRLEN_P ( zv ) + 1 , 1 TSRMLS_CC ) ;
}
2010-04-20 18:57:45 +08:00
CONSTANT_EX ( op_array , i ) = * zv ;
Z_SET_REFCOUNT ( CONSTANT_EX ( op_array , i ) , 2 ) ;
Z_SET_ISREF ( CONSTANT_EX ( op_array , i ) ) ;
return i ;
}
/* }}} */
2010-04-21 23:08:10 +08:00
int zend_add_func_name_literal ( zend_op_array * op_array , const zval * zv TSRMLS_DC ) /* { { { */
2010-04-21 22:58:33 +08:00
{
int ret ;
char * lc_name ;
zval c ;
int lc_literal ;
if ( op_array - > last_literal > 0 & &
& op_array - > literals [ op_array - > last_literal - 1 ] . constant = = zv ) {
/* we already have function name as last literal (do nothing) */
ret = op_array - > last_literal - 1 ;
} else {
ret = zend_add_literal ( op_array , zv ) ;
}
lc_name = zend_str_tolower_dup ( Z_STRVAL_P ( zv ) , Z_STRLEN_P ( zv ) ) ;
ZVAL_STRINGL ( & c , lc_name , Z_STRLEN_P ( zv ) , 0 ) ;
lc_literal = zend_add_literal ( CG ( active_op_array ) , & c ) ;
CALCULATE_LITERAL_HASH ( lc_literal ) ;
return ret ;
}
/* }}} */
2010-04-27 20:09:13 +08:00
int zend_add_ns_func_name_literal ( zend_op_array * op_array , const zval * zv TSRMLS_DC ) /* { { { */
{
int ret ;
char * lc_name , * ns_separator ;
int lc_len ;
zval c ;
int lc_literal ;
if ( op_array - > last_literal > 0 & &
& op_array - > literals [ op_array - > last_literal - 1 ] . constant = = zv ) {
/* we already have function name as last literal (do nothing) */
ret = op_array - > last_literal - 1 ;
} else {
ret = zend_add_literal ( op_array , zv ) ;
}
lc_name = zend_str_tolower_dup ( Z_STRVAL_P ( zv ) , Z_STRLEN_P ( zv ) ) ;
ZVAL_STRINGL ( & c , lc_name , Z_STRLEN_P ( zv ) , 0 ) ;
lc_literal = zend_add_literal ( CG ( active_op_array ) , & c ) ;
CALCULATE_LITERAL_HASH ( lc_literal ) ;
2010-04-29 04:47:13 +08:00
ns_separator = ( char * ) zend_memrchr ( Z_STRVAL_P ( zv ) , ' \\ ' , Z_STRLEN_P ( zv ) ) + 1 ;
2010-04-27 20:09:13 +08:00
lc_len = Z_STRLEN_P ( zv ) - ( ns_separator - Z_STRVAL_P ( zv ) ) ;
lc_name = zend_str_tolower_dup ( ns_separator , lc_len ) ;
ZVAL_STRINGL ( & c , lc_name , lc_len , 0 ) ;
lc_literal = zend_add_literal ( CG ( active_op_array ) , & c ) ;
CALCULATE_LITERAL_HASH ( lc_literal ) ;
return ret ;
}
/* }}} */
2010-04-21 22:58:33 +08:00
2010-04-21 23:08:10 +08:00
int zend_add_class_name_literal ( zend_op_array * op_array , const zval * zv TSRMLS_DC ) /* { { { */
2010-04-21 22:58:33 +08:00
{
int ret ;
char * lc_name ;
int lc_len ;
zval c ;
int lc_literal ;
if ( op_array - > last_literal > 0 & &
& op_array - > literals [ op_array - > last_literal - 1 ] . constant = = zv ) {
/* we already have function name as last literal (do nothing) */
ret = op_array - > last_literal - 1 ;
} else {
ret = zend_add_literal ( op_array , zv ) ;
}
if ( Z_STRVAL_P ( zv ) [ 0 ] = = ' \\ ' ) {
lc_len = Z_STRLEN_P ( zv ) - 1 ;
lc_name = zend_str_tolower_dup ( Z_STRVAL_P ( zv ) + 1 , lc_len ) ;
} else {
lc_len = Z_STRLEN_P ( zv ) ;
lc_name = zend_str_tolower_dup ( Z_STRVAL_P ( zv ) , lc_len ) ;
}
ZVAL_STRINGL ( & c , lc_name , lc_len , 0 ) ;
lc_literal = zend_add_literal ( CG ( active_op_array ) , & c ) ;
CALCULATE_LITERAL_HASH ( lc_literal ) ;
return ret ;
}
/* }}} */
2010-04-22 23:03:17 +08:00
int zend_add_const_name_literal ( zend_op_array * op_array , const zval * zv , int unqualified TSRMLS_DC ) /* { { { */
{
int ret , tmp_literal ;
char * name , * tmp_name , * ns_separator ;
int name_len , ns_len ;
zval c ;
if ( op_array - > last_literal > 0 & &
& op_array - > literals [ op_array - > last_literal - 1 ] . constant = = zv ) {
/* we already have function name as last literal (do nothing) */
ret = op_array - > last_literal - 1 ;
} else {
ret = zend_add_literal ( op_array , zv ) ;
}
/* skip leading '\\' */
if ( Z_STRVAL_P ( zv ) [ 0 ] = = ' \\ ' ) {
name_len = Z_STRLEN_P ( zv ) - 1 ;
name = Z_STRVAL_P ( zv ) + 1 ;
} else {
name_len = Z_STRLEN_P ( zv ) ;
name = Z_STRVAL_P ( zv ) ;
}
ns_separator = zend_memrchr ( name , ' \\ ' , name_len ) ;
if ( ns_separator ) {
ns_len = ns_separator - name ;
} else {
ns_len = 0 ;
}
if ( ns_len ) {
/* lowercased namespace name & original constant name */
tmp_name = estrndup ( name , name_len ) ;
zend_str_tolower ( tmp_name , ns_len ) ;
ZVAL_STRINGL ( & c , tmp_name , name_len , 0 ) ;
tmp_literal = zend_add_literal ( CG ( active_op_array ) , & c ) ;
CALCULATE_LITERAL_HASH ( tmp_literal ) ;
/* lowercased namespace name & lowercased constant name */
tmp_name = zend_str_tolower_dup ( name , name_len ) ;
ZVAL_STRINGL ( & c , tmp_name , name_len , 0 ) ;
tmp_literal = zend_add_literal ( CG ( active_op_array ) , & c ) ;
CALCULATE_LITERAL_HASH ( tmp_literal ) ;
}
if ( ns_len ) {
if ( ! unqualified ) {
return ret ;
}
ns_len + + ;
name + = ns_len ;
name_len - = ns_len ;
}
/* original constant name */
tmp_name = estrndup ( name , name_len ) ;
ZVAL_STRINGL ( & c , tmp_name , name_len , 0 ) ;
tmp_literal = zend_add_literal ( CG ( active_op_array ) , & c ) ;
CALCULATE_LITERAL_HASH ( tmp_literal ) ;
/* lowercased constant name */
tmp_name = zend_str_tolower_dup ( name , name_len ) ;
ZVAL_STRINGL ( & c , tmp_name , name_len , 0 ) ;
tmp_literal = zend_add_literal ( CG ( active_op_array ) , & c ) ;
CALCULATE_LITERAL_HASH ( tmp_literal ) ;
return ret ;
}
/* }}} */
2010-04-20 18:57:45 +08:00
# define LITERAL_STRINGL(op, str, len, copy) do { \
zval _c ; \
ZVAL_STRINGL ( & _c , str , len , copy ) ; \
op . constant = zend_add_literal ( CG ( active_op_array ) , & _c ) ; \
} while ( 0 )
# define LITERAL_LONG(op, val) do { \
zval _c ; \
ZVAL_LONG ( & _c , val ) ; \
op . constant = zend_add_literal ( CG ( active_op_array ) , & _c ) ; \
} while ( 0 )
# define LITERAL_LONG_EX(op_array, op, val) do { \
zval _c ; \
ZVAL_LONG ( & _c , val ) ; \
op . constant = zend_add_literal ( op_array , & _c ) ; \
} while ( 0 )
# define LITERAL_NULL(op) do { \
zval _c ; \
INIT_ZVAL ( _c ) ; \
op . constant = zend_add_literal ( CG ( active_op_array ) , & _c ) ; \
} while ( 0 )
2010-04-21 22:58:33 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_binary_op ( zend_uchar op , znode * result , const znode * op1 , const znode * op2 TSRMLS_DC ) /* { { { */
1999-04-08 02:10:10 +08:00
{
2001-07-28 18:51:54 +08:00
zend_op * opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
1999-04-08 02:10:10 +08:00
opline - > opcode = op ;
2010-04-20 18:57:45 +08:00
opline - > result_type = IS_TMP_VAR ;
opline - > result . var = get_temporary_variable ( CG ( active_op_array ) ) ;
SET_NODE ( opline - > op1 , op1 ) ;
SET_NODE ( opline - > op2 , op2 ) ;
GET_NODE ( result , opline - > result ) ;
1999-04-08 02:10:10 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_unary_op ( zend_uchar op , znode * result , const znode * op1 TSRMLS_DC ) /* { { { */
1999-04-08 02:10:10 +08:00
{
2001-07-28 18:51:54 +08:00
zend_op * opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
1999-04-08 02:10:10 +08:00
opline - > opcode = op ;
2010-04-20 18:57:45 +08:00
opline - > result_type = IS_TMP_VAR ;
opline - > result . var = get_temporary_variable ( CG ( active_op_array ) ) ;
SET_NODE ( opline - > op1 , op1 ) ;
GET_NODE ( result , opline - > result ) ;
1999-04-08 02:10:10 +08:00
SET_UNUSED ( opline - > op2 ) ;
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2010-04-20 18:57:45 +08:00
# define MAKE_NOP(opline) { opline->opcode = ZEND_NOP; memset(&opline->result,0,sizeof(opline->result)); memset(&opline->op1,0,sizeof(opline->op1)); memset(&opline->op2,0,sizeof(opline->op2)); opline->result_type=opline->op1_type=opline->op2_type=IS_UNUSED; }
2002-03-10 21:42:37 +08:00
2003-01-27 23:13:01 +08:00
2009-07-27 22:11:53 +08:00
static void zend_do_op_data ( zend_op * data_op , const znode * value TSRMLS_DC ) /* { { { */
2003-01-27 23:13:01 +08:00
{
data_op - > opcode = ZEND_OP_DATA ;
2010-04-20 18:57:45 +08:00
SET_NODE ( data_op - > op1 , value ) ;
2003-01-27 23:13:01 +08:00
SET_UNUSED ( data_op - > op2 ) ;
2002-03-10 21:42:37 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_binary_assign_op ( zend_uchar op , znode * result , const znode * op1 , const znode * op2 TSRMLS_DC ) /* { { { */
1999-04-08 02:10:10 +08:00
{
2005-06-08 23:01:35 +08:00
int last_op_number = get_next_op_number ( CG ( active_op_array ) ) ;
2001-07-28 18:51:54 +08:00
zend_op * opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
2003-07-31 01:40:54 +08:00
2005-06-08 23:01:35 +08:00
if ( last_op_number > 0 ) {
zend_op * last_op = & CG ( active_op_array ) - > opcodes [ last_op_number - 1 ] ;
2006-05-10 07:53:23 +08:00
2005-06-08 23:01:35 +08:00
switch ( last_op - > opcode ) {
case ZEND_FETCH_OBJ_RW :
last_op - > opcode = op ;
last_op - > extended_value = ZEND_ASSIGN_OBJ ;
zend_do_op_data ( opline , op2 TSRMLS_CC ) ;
SET_UNUSED ( opline - > result ) ;
2010-04-20 18:57:45 +08:00
GET_NODE ( result , last_op - > result ) ;
2005-06-08 23:01:35 +08:00
return ;
case ZEND_FETCH_DIM_RW :
last_op - > opcode = op ;
last_op - > extended_value = ZEND_ASSIGN_DIM ;
zend_do_op_data ( opline , op2 TSRMLS_CC ) ;
2010-04-20 18:57:45 +08:00
opline - > op2 . var = get_temporary_variable ( CG ( active_op_array ) ) ;
opline - > op2_type = IS_VAR ;
2005-06-08 23:01:35 +08:00
SET_UNUSED ( opline - > result ) ;
2010-04-20 18:57:45 +08:00
GET_NODE ( result , last_op - > result ) ;
2005-06-08 23:01:35 +08:00
return ;
default :
break ;
}
2002-03-10 21:42:37 +08:00
}
2005-06-08 23:01:35 +08:00
opline - > opcode = op ;
2010-04-20 18:57:45 +08:00
SET_NODE ( opline - > op1 , op1 ) ;
SET_NODE ( opline - > op2 , op2 ) ;
opline - > result_type = IS_VAR ;
opline - > result . var = get_temporary_variable ( CG ( active_op_array ) ) ;
GET_NODE ( result , opline - > result ) ;
1999-04-08 02:10:10 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
void fetch_simple_variable_ex ( znode * result , znode * varname , int bp , zend_uchar op TSRMLS_DC ) /* { { { */
1999-04-08 02:10:10 +08:00
{
1999-09-20 23:44:30 +08:00
zend_op opline ;
zend_op * opline_ptr ;
1999-04-08 02:10:10 +08:00
zend_llist * fetch_list_ptr ;
2008-01-24 01:55:55 +08:00
if ( varname - > op_type = = IS_CONST ) {
if ( Z_TYPE ( varname - > u . constant ) ! = IS_STRING ) {
convert_to_string ( & varname - > u . constant ) ;
}
if ( ! zend_is_auto_global ( varname - > u . constant . value . str . val , varname - > u . constant . value . str . len TSRMLS_CC ) & &
! ( varname - > u . constant . value . str . len = = ( sizeof ( " this " ) - 1 ) & &
! memcmp ( varname - > u . constant . value . str . val , " this " , sizeof ( " this " ) ) ) & &
( CG ( active_op_array ) - > last = = 0 | |
CG ( active_op_array ) - > opcodes [ CG ( active_op_array ) - > last - 1 ] . opcode ! = ZEND_BEGIN_SILENCE ) ) {
result - > op_type = IS_CV ;
2010-04-20 19:05:54 +08:00
result - > u . op . var = lookup_cv ( CG ( active_op_array ) , varname - > u . constant . value . str . val , varname - > u . constant . value . str . len TSRMLS_CC ) ;
varname - > u . constant . value . str . val = CG ( active_op_array ) - > vars [ result - > u . op . var ] . name ;
2010-04-20 18:57:45 +08:00
result - > EA = 0 ;
2008-01-24 01:55:55 +08:00
return ;
}
2004-10-05 03:54:35 +08:00
}
1999-09-20 23:44:30 +08:00
if ( bp ) {
opline_ptr = & opline ;
2001-07-28 18:51:54 +08:00
init_op ( opline_ptr TSRMLS_CC ) ;
1999-09-20 23:44:30 +08:00
} else {
2001-07-28 18:51:54 +08:00
opline_ptr = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
1999-09-20 23:44:30 +08:00
}
opline_ptr - > opcode = op ;
2010-04-20 18:57:45 +08:00
opline_ptr - > result_type = IS_VAR ;
opline_ptr - > result . var = get_temporary_variable ( CG ( active_op_array ) ) ;
SET_NODE ( opline_ptr - > op1 , varname ) ;
GET_NODE ( result , opline_ptr - > result ) ;
1999-09-20 23:44:30 +08:00
SET_UNUSED ( opline_ptr - > op2 ) ;
2010-04-20 18:57:45 +08:00
opline_ptr - > extended_value = ZEND_FETCH_LOCAL ;
2001-08-09 01:18:16 +08:00
2010-04-20 18:57:45 +08:00
if ( varname - > op_type = = IS_CONST ) {
2010-04-21 22:58:33 +08:00
CALCULATE_LITERAL_HASH ( opline_ptr - > op1 . constant ) ;
2003-03-06 05:36:30 +08:00
if ( zend_is_auto_global ( varname - > u . constant . value . str . val , varname - > u . constant . value . str . len TSRMLS_CC ) ) {
2010-04-20 18:57:45 +08:00
opline_ptr - > extended_value = ZEND_FETCH_GLOBAL ;
2004-02-10 19:44:17 +08:00
}
2001-08-09 01:18:16 +08:00
}
1999-04-08 02:10:10 +08:00
if ( bp ) {
zend_stack_top ( & CG ( bp_stack ) , ( void * * ) & fetch_list_ptr ) ;
1999-09-20 23:44:30 +08:00
zend_llist_add_element ( fetch_list_ptr , opline_ptr ) ;
1999-04-08 02:10:10 +08:00
}
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
void fetch_simple_variable ( znode * result , znode * varname , int bp TSRMLS_DC ) /* { { { */
1999-09-20 22:45:36 +08:00
{
/* the default mode must be Write, since fetch_simple_variable() is used to define function arguments */
2001-07-28 18:51:54 +08:00
fetch_simple_variable_ex ( result , varname , bp , ZEND_FETCH_W TSRMLS_CC ) ;
1999-09-20 22:45:36 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-09-20 22:45:36 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_fetch_static_member ( znode * result , znode * class_name TSRMLS_DC ) /* { { { */
2001-11-25 16:49:09 +08:00
{
2007-09-29 03:52:53 +08:00
znode class_node ;
2001-11-25 16:49:09 +08:00
zend_llist * fetch_list_ptr ;
zend_llist_element * le ;
zend_op * opline_ptr ;
2004-10-05 03:54:35 +08:00
zend_op opline ;
2001-11-25 16:49:09 +08:00
2007-09-29 03:52:53 +08:00
zend_do_fetch_class ( & class_node , class_name TSRMLS_CC ) ;
2001-11-25 16:49:09 +08:00
zend_stack_top ( & CG ( bp_stack ) , ( void * * ) & fetch_list_ptr ) ;
2004-10-05 03:54:35 +08:00
if ( result - > op_type = = IS_CV ) {
init_op ( & opline TSRMLS_CC ) ;
opline . opcode = ZEND_FETCH_W ;
2010-04-20 18:57:45 +08:00
opline . result_type = IS_VAR ;
opline . result . var = get_temporary_variable ( CG ( active_op_array ) ) ;
opline . op1_type = IS_CONST ;
LITERAL_STRINGL ( opline . op1 , estrdup ( CG ( active_op_array ) - > vars [ result - > u . op . var ] . name ) , CG ( active_op_array ) - > vars [ result - > u . op . var ] . name_len , 0 ) ;
2010-04-21 22:58:33 +08:00
CALCULATE_LITERAL_HASH ( opline . op1 . constant ) ;
2004-10-05 03:54:35 +08:00
SET_UNUSED ( opline . op2 ) ;
2010-04-20 18:57:45 +08:00
SET_NODE ( opline . op2 , & class_node ) ;
GET_NODE ( result , opline . result ) ;
opline . extended_value | = ZEND_FETCH_STATIC_MEMBER ;
opline_ptr = & opline ;
2004-10-05 03:54:35 +08:00
zend_llist_add_element ( fetch_list_ptr , & opline ) ;
} else {
le = fetch_list_ptr - > head ;
2001-11-25 16:49:09 +08:00
2004-10-05 03:54:35 +08:00
opline_ptr = ( zend_op * ) le - > data ;
2010-04-20 18:57:45 +08:00
if ( opline_ptr - > opcode ! = ZEND_FETCH_W & & opline_ptr - > op1_type = = IS_CV ) {
2004-10-05 03:54:35 +08:00
init_op ( & opline TSRMLS_CC ) ;
opline . opcode = ZEND_FETCH_W ;
2010-04-20 18:57:45 +08:00
opline . result_type = IS_VAR ;
opline . result . var = get_temporary_variable ( CG ( active_op_array ) ) ;
opline . op1_type = IS_CONST ;
LITERAL_STRINGL ( opline . op1 , estrdup ( CG ( active_op_array ) - > vars [ opline_ptr - > op1 . var ] . name ) , CG ( active_op_array ) - > vars [ opline_ptr - > op1 . var ] . name_len , 0 ) ;
2010-04-21 22:58:33 +08:00
CALCULATE_LITERAL_HASH ( opline . op1 . constant ) ;
2004-10-05 03:54:35 +08:00
SET_UNUSED ( opline . op2 ) ;
2010-04-20 18:57:45 +08:00
SET_NODE ( opline . op2 , & class_node ) ;
opline . extended_value | = ZEND_FETCH_STATIC_MEMBER ;
COPY_NODE ( opline_ptr - > op1 , opline . result ) ;
2004-10-05 03:54:35 +08:00
zend_llist_prepend_element ( fetch_list_ptr , & opline ) ;
} else {
2010-04-20 18:57:45 +08:00
SET_NODE ( opline_ptr - > op2 , & class_node ) ;
opline_ptr - > extended_value | = ZEND_FETCH_STATIC_MEMBER ;
2004-10-05 03:54:35 +08:00
}
}
2001-11-25 16:49:09 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
void fetch_array_begin ( znode * result , znode * varname , znode * first_dim TSRMLS_DC ) /* { { { */
1999-04-08 02:10:10 +08:00
{
2001-07-28 18:51:54 +08:00
fetch_simple_variable ( result , varname , 1 TSRMLS_CC ) ;
1999-04-08 02:10:10 +08:00
2001-07-28 18:51:54 +08:00
fetch_array_dim ( result , result , first_dim TSRMLS_CC ) ;
1999-04-08 02:10:10 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
void fetch_array_dim ( znode * result , const znode * parent , const znode * dim TSRMLS_DC ) /* { { { */
1999-04-08 02:10:10 +08:00
{
1999-09-20 23:44:30 +08:00
zend_op opline ;
1999-04-08 02:10:10 +08:00
zend_llist * fetch_list_ptr ;
2001-07-28 18:51:54 +08:00
init_op ( & opline TSRMLS_CC ) ;
1999-09-20 23:44:30 +08:00
opline . opcode = ZEND_FETCH_DIM_W ; /* the backpatching routine assumes W */
2010-04-20 18:57:45 +08:00
opline . result_type = IS_VAR ;
opline . result . var = get_temporary_variable ( CG ( active_op_array ) ) ;
SET_NODE ( opline . op1 , parent ) ;
SET_NODE ( opline . op2 , dim ) ;
if ( opline . op2_type = = IS_CONST & & Z_TYPE ( CONSTANT ( opline . op2 . constant ) ) = = IS_STRING ) {
long index ;
int numeric = 0 ;
ZEND_HANDLE_NUMERIC_EX ( Z_STRVAL ( CONSTANT ( opline . op2 . constant ) ) , Z_STRLEN ( CONSTANT ( opline . op2 . constant ) ) + 1 , index , numeric = 1 ) ;
if ( numeric ) {
zval_dtor ( & CONSTANT ( opline . op2 . constant ) ) ;
ZVAL_LONG ( & CONSTANT ( opline . op2 . constant ) , index ) ;
} else {
2010-04-21 22:58:33 +08:00
CALCULATE_LITERAL_HASH ( opline . op2 . constant ) ;
2010-04-20 18:57:45 +08:00
}
}
GET_NODE ( result , opline . result ) ;
1999-04-08 02:10:10 +08:00
zend_stack_top ( & CG ( bp_stack ) , ( void * * ) & fetch_list_ptr ) ;
1999-09-20 23:44:30 +08:00
zend_llist_add_element ( fetch_list_ptr , & opline ) ;
1999-04-08 02:10:10 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
void fetch_string_offset ( znode * result , const znode * parent , const znode * offset TSRMLS_DC ) /* { { { */
2000-01-29 18:16:04 +08:00
{
2001-07-28 18:51:54 +08:00
fetch_array_dim ( result , parent , offset TSRMLS_CC ) ;
2000-01-29 18:16:04 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
2000-01-29 18:16:04 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_print ( znode * result , const znode * arg TSRMLS_DC ) /* { { { */
1999-04-08 02:10:10 +08:00
{
2001-07-28 18:51:54 +08:00
zend_op * opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
1999-04-08 02:10:10 +08:00
2010-04-20 18:57:45 +08:00
opline - > result_type = IS_TMP_VAR ;
opline - > result . var = get_temporary_variable ( CG ( active_op_array ) ) ;
1999-04-24 08:12:55 +08:00
opline - > opcode = ZEND_PRINT ;
2010-04-20 18:57:45 +08:00
SET_NODE ( opline - > op1 , arg ) ;
1999-04-08 02:10:10 +08:00
SET_UNUSED ( opline - > op2 ) ;
2010-04-20 18:57:45 +08:00
GET_NODE ( result , opline - > result ) ;
1999-04-08 02:10:10 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_echo ( const znode * arg TSRMLS_DC ) /* { { { */
1999-04-08 02:10:10 +08:00
{
2001-07-28 18:51:54 +08:00
zend_op * opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
1999-04-08 02:10:10 +08:00
1999-04-24 08:12:55 +08:00
opline - > opcode = ZEND_ECHO ;
2010-04-20 18:57:45 +08:00
SET_NODE ( opline - > op1 , arg ) ;
1999-04-08 02:10:10 +08:00
SET_UNUSED ( opline - > op2 ) ;
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_abstract_method ( const znode * function_name , znode * modifiers , const znode * body TSRMLS_DC ) /* { { { */
2002-11-21 02:00:23 +08:00
{
2003-03-05 19:14:44 +08:00
char * method_type ;
if ( CG ( active_class_entry ) - > ce_flags & ZEND_ACC_INTERFACE ) {
2006-05-12 05:07:39 +08:00
Z_LVAL ( modifiers - > u . constant ) | = ZEND_ACC_ABSTRACT ;
2003-03-05 19:14:44 +08:00
method_type = " Interface " ;
} else {
method_type = " Abstract " ;
}
2003-02-11 17:48:37 +08:00
if ( modifiers - > u . constant . value . lval & ZEND_ACC_ABSTRACT ) {
2005-06-10 01:12:54 +08:00
if ( modifiers - > u . constant . value . lval & ZEND_ACC_PRIVATE ) {
zend_error ( E_COMPILE_ERROR , " %s function %s::%s() cannot be declared private " , method_type , CG ( active_class_entry ) - > name , function_name - > u . constant . value . str . val ) ;
}
2006-05-12 05:07:39 +08:00
if ( Z_LVAL ( body - > u . constant ) = = ZEND_ACC_ABSTRACT ) {
2003-02-11 17:48:37 +08:00
zend_op * opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
2002-11-21 02:00:23 +08:00
2003-02-11 17:48:37 +08:00
opline - > opcode = ZEND_RAISE_ABSTRACT_ERROR ;
SET_UNUSED ( opline - > op1 ) ;
SET_UNUSED ( opline - > op2 ) ;
} else {
/* we had code in the function body */
2003-03-05 19:14:44 +08:00
zend_error ( E_COMPILE_ERROR , " %s function %s::%s() cannot contain body " , method_type , CG ( active_class_entry ) - > name , function_name - > u . constant . value . str . val ) ;
2003-02-11 17:48:37 +08:00
}
} else {
2003-03-05 19:14:44 +08:00
if ( body - > u . constant . value . lval = = ZEND_ACC_ABSTRACT ) {
zend_error ( E_COMPILE_ERROR , " Non-abstract method %s::%s() must contain body " , CG ( active_class_entry ) - > name , function_name - > u . constant . value . str . val ) ;
2003-02-11 17:48:37 +08:00
}
}
2002-11-21 02:00:23 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
2002-11-21 02:00:23 +08:00
2009-07-27 22:11:53 +08:00
static zend_bool opline_is_fetch_this ( const zend_op * opline TSRMLS_DC ) /* { { { */
2004-02-04 22:25:25 +08:00
{
2010-04-20 18:57:45 +08:00
if ( ( opline - > opcode = = ZEND_FETCH_W ) & & ( opline - > op1_type = = IS_CONST )
& & ( Z_TYPE ( CONSTANT ( opline - > op1 . constant ) ) = = IS_STRING )
& & ( Z_STRLEN ( CONSTANT ( opline - > op1 . constant ) ) = = ( sizeof ( " this " ) - 1 ) )
& & ! memcmp ( Z_STRVAL ( CONSTANT ( opline - > op1 . constant ) ) , " this " , sizeof ( " this " ) ) ) {
2004-02-04 22:25:25 +08:00
return 1 ;
} else {
return 0 ;
}
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2010-04-20 18:57:45 +08:00
void zend_do_assign ( znode * result , znode * variable , znode * value TSRMLS_DC ) /* { { { */
1999-04-08 02:10:10 +08:00
{
2007-12-28 21:22:00 +08:00
int last_op_number ;
zend_op * opline ;
if ( value - > op_type = = IS_CV ) {
zend_llist * fetch_list_ptr ;
zend_stack_top ( & CG ( bp_stack ) , ( void * * ) & fetch_list_ptr ) ;
if ( fetch_list_ptr & & fetch_list_ptr - > head ) {
opline = ( zend_op * ) fetch_list_ptr - > head - > data ;
if ( opline - > opcode = = ZEND_FETCH_DIM_W & &
2010-04-20 18:57:45 +08:00
opline - > op1_type = = IS_CV & &
opline - > op1 . var = = value - > u . op . var ) {
2007-12-28 21:22:00 +08:00
opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
opline - > opcode = ZEND_FETCH_R ;
2010-04-20 18:57:45 +08:00
opline - > result_type = IS_VAR ;
opline - > result . var = get_temporary_variable ( CG ( active_op_array ) ) ;
opline - > op1_type = IS_CONST ;
LITERAL_STRINGL ( opline - > op1 ,
CG ( active_op_array ) - > vars [ value - > u . op . var ] . name ,
CG ( active_op_array ) - > vars [ value - > u . op . var ] . name_len , 1 ) ;
2010-04-21 22:58:33 +08:00
CALCULATE_LITERAL_HASH ( opline - > op1 . constant ) ;
2007-12-28 21:22:00 +08:00
SET_UNUSED ( opline - > op2 ) ;
2010-04-20 18:57:45 +08:00
opline - > extended_value = ZEND_FETCH_LOCAL ;
GET_NODE ( value , opline - > result ) ;
2007-12-28 21:22:00 +08:00
}
}
}
2008-05-07 20:04:39 +08:00
zend_do_end_variable_parse ( variable , BP_VAR_W , 0 TSRMLS_CC ) ;
2007-12-28 21:22:00 +08:00
last_op_number = get_next_op_number ( CG ( active_op_array ) ) ;
opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
2004-02-04 21:56:41 +08:00
2008-05-07 20:04:39 +08:00
if ( variable - > op_type = = IS_CV ) {
2010-04-20 18:57:45 +08:00
if ( variable - > u . op . var = = CG ( active_op_array ) - > this_var ) {
2008-05-07 20:04:39 +08:00
zend_error ( E_COMPILE_ERROR , " Cannot re-assign $this " ) ;
}
} else if ( variable - > op_type = = IS_VAR ) {
2005-10-10 17:50:05 +08:00
int n = 0 ;
while ( last_op_number - n > 0 ) {
zend_op * last_op ;
2006-05-10 07:53:23 +08:00
last_op = & CG ( active_op_array ) - > opcodes [ last_op_number - n - 1 ] ;
2005-10-10 17:50:05 +08:00
2010-04-20 18:57:45 +08:00
if ( last_op - > result_type = = IS_VAR & &
last_op - > result . var = = variable - > u . op . var ) {
2005-10-10 17:50:05 +08:00
if ( last_op - > opcode = = ZEND_FETCH_OBJ_W ) {
if ( n > 0 ) {
2008-11-12 08:44:56 +08:00
int opline_no = ( opline - CG ( active_op_array ) - > opcodes ) / sizeof ( * opline ) ;
2005-10-10 17:50:05 +08:00
* opline = * last_op ;
MAKE_NOP ( last_op ) ;
2008-11-12 08:44:56 +08:00
/* last_op = opline; */
2005-10-10 17:50:05 +08:00
opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
2008-11-12 08:44:56 +08:00
/* get_next_op can realloc, we need to move last_op */
last_op = & CG ( active_op_array ) - > opcodes [ opline_no ] ;
2005-10-10 17:50:05 +08:00
}
last_op - > opcode = ZEND_ASSIGN_OBJ ;
zend_do_op_data ( opline , value TSRMLS_CC ) ;
SET_UNUSED ( opline - > result ) ;
2010-04-20 18:57:45 +08:00
GET_NODE ( result , last_op - > result ) ;
2005-10-10 17:50:05 +08:00
return ;
} else if ( last_op - > opcode = = ZEND_FETCH_DIM_W ) {
if ( n > 0 ) {
2008-11-12 08:44:56 +08:00
int opline_no = ( opline - CG ( active_op_array ) - > opcodes ) / sizeof ( * opline ) ;
2005-10-10 17:50:05 +08:00
* opline = * last_op ;
MAKE_NOP ( last_op ) ;
2008-11-12 08:44:56 +08:00
/* last_op = opline; */
/* TBFixed: this can realloc opcodes, leaving last_op pointing wrong */
2005-10-10 17:50:05 +08:00
opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
2008-11-12 08:44:56 +08:00
/* get_next_op can realloc, we need to move last_op */
last_op = & CG ( active_op_array ) - > opcodes [ opline_no ] ;
2005-10-10 17:50:05 +08:00
}
last_op - > opcode = ZEND_ASSIGN_DIM ;
zend_do_op_data ( opline , value TSRMLS_CC ) ;
2010-04-20 18:57:45 +08:00
opline - > op2 . var = get_temporary_variable ( CG ( active_op_array ) ) ;
opline - > op2_type = IS_VAR ;
2005-10-10 17:50:05 +08:00
SET_UNUSED ( opline - > result ) ;
2010-04-20 18:57:45 +08:00
GET_NODE ( result , last_op - > result ) ;
2005-10-10 17:50:05 +08:00
return ;
} else if ( opline_is_fetch_this ( last_op TSRMLS_CC ) ) {
zend_error ( E_COMPILE_ERROR , " Cannot re-assign $this " ) ;
} else {
break ;
}
2005-06-08 23:01:35 +08:00
}
2005-10-10 17:50:05 +08:00
n + + ;
2005-06-08 23:01:35 +08:00
}
2002-03-10 21:42:37 +08:00
}
2005-06-08 23:01:35 +08:00
opline - > opcode = ZEND_ASSIGN ;
2010-04-20 18:57:45 +08:00
SET_NODE ( opline - > op1 , variable ) ;
SET_NODE ( opline - > op2 , value ) ;
opline - > result_type = IS_VAR ;
opline - > result . var = get_temporary_variable ( CG ( active_op_array ) ) ;
GET_NODE ( result , opline - > result ) ;
1999-04-08 02:10:10 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
static inline zend_bool zend_is_function_or_method_call ( const znode * variable ) /* { { { */
2005-05-06 01:37:25 +08:00
{
2010-04-20 18:57:45 +08:00
zend_uint type = variable - > EA ;
2005-05-06 01:37:25 +08:00
return ( ( type & ZEND_PARSED_METHOD_CALL ) | | ( type = = ZEND_PARSED_FUNCTION_CALL ) ) ;
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_assign_ref ( znode * result , const znode * lvar , const znode * rvar TSRMLS_DC ) /* { { { */
1999-04-08 02:10:10 +08:00
{
2008-05-07 20:04:39 +08:00
zend_op * opline ;
2005-06-08 23:01:35 +08:00
2008-05-07 20:04:39 +08:00
if ( lvar - > op_type = = IS_CV ) {
2010-04-20 18:57:45 +08:00
if ( lvar - > u . op . var = = CG ( active_op_array ) - > this_var ) {
2008-05-07 20:04:39 +08:00
zend_error ( E_COMPILE_ERROR , " Cannot re-assign $this " ) ;
}
} else if ( lvar - > op_type = = IS_VAR ) {
int last_op_number = get_next_op_number ( CG ( active_op_array ) ) ;
if ( last_op_number > 0 ) {
opline = & CG ( active_op_array ) - > opcodes [ last_op_number - 1 ] ;
if ( opline_is_fetch_this ( opline TSRMLS_CC ) ) {
zend_error ( E_COMPILE_ERROR , " Cannot re-assign $this " ) ;
}
}
}
2005-06-08 23:01:35 +08:00
2008-05-07 20:04:39 +08:00
opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
2005-06-08 23:01:35 +08:00
opline - > opcode = ZEND_ASSIGN_REF ;
2005-05-06 01:37:25 +08:00
if ( zend_is_function_or_method_call ( rvar ) ) {
2006-05-10 07:53:23 +08:00
opline - > extended_value = ZEND_RETURNS_FUNCTION ;
2010-04-20 18:57:45 +08:00
} else if ( rvar - > EA & ZEND_PARSED_NEW ) {
2008-07-24 19:47:51 +08:00
opline - > extended_value = ZEND_RETURNS_NEW ;
2005-05-06 01:37:25 +08:00
} else {
opline - > extended_value = 0 ;
}
1999-04-08 02:10:10 +08:00
if ( result ) {
2010-04-20 18:57:45 +08:00
opline - > result_type = IS_VAR ;
opline - > result . var = get_temporary_variable ( CG ( active_op_array ) ) ;
GET_NODE ( result , opline - > result ) ;
1999-04-08 02:10:10 +08:00
} else {
2010-04-20 18:57:45 +08:00
opline - > result_type = IS_UNUSED | EXT_TYPE_UNUSED ;
1999-04-08 02:10:10 +08:00
}
2010-04-20 18:57:45 +08:00
SET_NODE ( opline - > op1 , lvar ) ;
SET_NODE ( opline - > op2 , rvar ) ;
1999-04-08 02:10:10 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
static inline void do_begin_loop ( TSRMLS_D ) /* { { { */
1999-04-08 02:10:10 +08:00
{
zend_brk_cont_element * brk_cont_element ;
int parent ;
parent = CG ( active_op_array ) - > current_brk_cont ;
CG ( active_op_array ) - > current_brk_cont = CG ( active_op_array ) - > last_brk_cont ;
brk_cont_element = get_next_brk_cont_element ( CG ( active_op_array ) ) ;
2006-09-20 05:36:54 +08:00
brk_cont_element - > start = get_next_op_number ( CG ( active_op_array ) ) ;
1999-04-08 02:10:10 +08:00
brk_cont_element - > parent = parent ;
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
static inline void do_end_loop ( int cont_addr , int has_loop_var TSRMLS_DC ) /* { { { */
1999-04-08 02:10:10 +08:00
{
2008-02-20 20:05:57 +08:00
if ( ! has_loop_var ) {
/* The start fileld is used to free temporary variables in case of exceptions.
* We won ' t try to free something of we don ' t have loop variable .
*/
CG ( active_op_array ) - > brk_cont_array [ CG ( active_op_array ) - > current_brk_cont ] . start = - 1 ;
}
1999-04-08 02:10:10 +08:00
CG ( active_op_array ) - > brk_cont_array [ CG ( active_op_array ) - > current_brk_cont ] . cont = cont_addr ;
CG ( active_op_array ) - > brk_cont_array [ CG ( active_op_array ) - > current_brk_cont ] . brk = get_next_op_number ( CG ( active_op_array ) ) ;
CG ( active_op_array ) - > current_brk_cont = CG ( active_op_array ) - > brk_cont_array [ CG ( active_op_array ) - > current_brk_cont ] . parent ;
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_while_cond ( const znode * expr , znode * close_bracket_token TSRMLS_DC ) /* { { { */
1999-04-08 02:10:10 +08:00
{
int while_cond_op_number = get_next_op_number ( CG ( active_op_array ) ) ;
2001-07-28 18:51:54 +08:00
zend_op * opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
1999-04-08 02:10:10 +08:00
opline - > opcode = ZEND_JMPZ ;
2010-04-20 18:57:45 +08:00
SET_NODE ( opline - > op1 , expr ) ;
close_bracket_token - > u . op . opline_num = while_cond_op_number ;
1999-04-08 02:10:10 +08:00
SET_UNUSED ( opline - > op2 ) ;
2001-07-28 18:51:54 +08:00
do_begin_loop ( TSRMLS_C ) ;
1999-04-08 02:10:10 +08:00
INC_BPC ( CG ( active_op_array ) ) ;
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_while_end ( const znode * while_token , const znode * close_bracket_token TSRMLS_DC ) /* { { { */
1999-04-08 02:10:10 +08:00
{
2001-07-28 18:51:54 +08:00
zend_op * opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
1999-04-08 02:10:10 +08:00
/* add unconditional jump */
opline - > opcode = ZEND_JMP ;
2010-04-20 18:57:45 +08:00
opline - > op1 . opline_num = while_token - > u . op . opline_num ;
1999-04-08 02:10:10 +08:00
SET_UNUSED ( opline - > op1 ) ;
SET_UNUSED ( opline - > op2 ) ;
2006-05-10 07:53:23 +08:00
1999-04-08 02:10:10 +08:00
/* update while's conditional jmp */
2010-04-20 18:57:45 +08:00
CG ( active_op_array ) - > opcodes [ close_bracket_token - > u . op . opline_num ] . op2 . opline_num = get_next_op_number ( CG ( active_op_array ) ) ;
1999-04-08 02:10:10 +08:00
2010-04-20 18:57:45 +08:00
do_end_loop ( while_token - > u . op . opline_num , 0 TSRMLS_CC ) ;
1999-04-08 02:10:10 +08:00
DEC_BPC ( CG ( active_op_array ) ) ;
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_for_cond ( const znode * expr , znode * second_semicolon_token TSRMLS_DC ) /* { { { */
1999-04-08 02:10:10 +08:00
{
int for_cond_op_number = get_next_op_number ( CG ( active_op_array ) ) ;
2001-07-28 18:51:54 +08:00
zend_op * opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
1999-04-08 02:10:10 +08:00
opline - > opcode = ZEND_JMPZNZ ;
2010-04-20 18:57:45 +08:00
SET_NODE ( opline - > op1 , expr ) ; /* the conditional expression */
second_semicolon_token - > u . op . opline_num = for_cond_op_number ;
1999-04-08 02:10:10 +08:00
SET_UNUSED ( opline - > op2 ) ;
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_for_before_statement ( const znode * cond_start , const znode * second_semicolon_token TSRMLS_DC ) /* { { { */
1999-04-08 02:10:10 +08:00
{
2001-07-28 18:51:54 +08:00
zend_op * opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
1999-04-08 02:10:10 +08:00
opline - > opcode = ZEND_JMP ;
2010-04-20 18:57:45 +08:00
opline - > op1 . opline_num = cond_start - > u . op . opline_num ;
CG ( active_op_array ) - > opcodes [ second_semicolon_token - > u . op . opline_num ] . extended_value = get_next_op_number ( CG ( active_op_array ) ) ;
1999-04-08 02:10:10 +08:00
SET_UNUSED ( opline - > op1 ) ;
SET_UNUSED ( opline - > op2 ) ;
2001-07-28 18:51:54 +08:00
do_begin_loop ( TSRMLS_C ) ;
1999-04-08 02:10:10 +08:00
INC_BPC ( CG ( active_op_array ) ) ;
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_for_end ( const znode * second_semicolon_token TSRMLS_DC ) /* { { { */
1999-04-08 02:10:10 +08:00
{
2001-07-28 18:51:54 +08:00
zend_op * opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
1999-04-08 02:10:10 +08:00
opline - > opcode = ZEND_JMP ;
2010-04-20 18:57:45 +08:00
opline - > op1 . opline_num = second_semicolon_token - > u . op . opline_num + 1 ;
CG ( active_op_array ) - > opcodes [ second_semicolon_token - > u . op . opline_num ] . op2 . opline_num = get_next_op_number ( CG ( active_op_array ) ) ;
1999-04-08 02:10:10 +08:00
SET_UNUSED ( opline - > op1 ) ;
SET_UNUSED ( opline - > op2 ) ;
2010-04-20 18:57:45 +08:00
do_end_loop ( second_semicolon_token - > u . op . opline_num + 1 , 0 TSRMLS_CC ) ;
1999-04-08 02:10:10 +08:00
DEC_BPC ( CG ( active_op_array ) ) ;
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_pre_incdec ( znode * result , const znode * op1 , zend_uchar op TSRMLS_DC ) /* { { { */
1999-04-08 02:10:10 +08:00
{
2005-06-08 23:01:35 +08:00
int last_op_number = get_next_op_number ( CG ( active_op_array ) ) ;
zend_op * opline ;
if ( last_op_number > 0 ) {
zend_op * last_op = & CG ( active_op_array ) - > opcodes [ last_op_number - 1 ] ;
if ( last_op - > opcode = = ZEND_FETCH_OBJ_RW ) {
last_op - > opcode = ( op = = ZEND_PRE_INC ) ? ZEND_PRE_INC_OBJ : ZEND_PRE_DEC_OBJ ;
2010-04-20 18:57:45 +08:00
last_op - > result_type = IS_VAR ;
last_op - > result . var = get_temporary_variable ( CG ( active_op_array ) ) ;
GET_NODE ( result , last_op - > result ) ;
2005-06-08 23:01:35 +08:00
return ;
}
2002-03-10 21:42:37 +08:00
}
2005-06-08 23:01:35 +08:00
opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
opline - > opcode = op ;
2010-04-20 18:57:45 +08:00
SET_NODE ( opline - > op1 , op1 ) ;
2005-06-08 23:01:35 +08:00
SET_UNUSED ( opline - > op2 ) ;
2010-04-20 18:57:45 +08:00
opline - > result_type = IS_VAR ;
opline - > result . var = get_temporary_variable ( CG ( active_op_array ) ) ;
GET_NODE ( result , opline - > result ) ;
1999-04-08 02:10:10 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_post_incdec ( znode * result , const znode * op1 , zend_uchar op TSRMLS_DC ) /* { { { */
1999-04-08 02:10:10 +08:00
{
2005-06-08 23:01:35 +08:00
int last_op_number = get_next_op_number ( CG ( active_op_array ) ) ;
zend_op * opline ;
2010-04-20 18:57:45 +08:00
if ( last_op_number > 0 ) {
2005-06-08 23:01:35 +08:00
zend_op * last_op = & CG ( active_op_array ) - > opcodes [ last_op_number - 1 ] ;
if ( last_op - > opcode = = ZEND_FETCH_OBJ_RW ) {
last_op - > opcode = ( op = = ZEND_POST_INC ) ? ZEND_POST_INC_OBJ : ZEND_POST_DEC_OBJ ;
2010-04-20 18:57:45 +08:00
last_op - > result_type = IS_TMP_VAR ;
last_op - > result . var = get_temporary_variable ( CG ( active_op_array ) ) ;
GET_NODE ( result , last_op - > result ) ;
2005-06-08 23:01:35 +08:00
return ;
}
2002-03-10 21:42:37 +08:00
}
2005-06-08 23:01:35 +08:00
opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
opline - > opcode = op ;
2010-04-20 18:57:45 +08:00
SET_NODE ( opline - > op1 , op1 ) ;
2005-06-08 23:01:35 +08:00
SET_UNUSED ( opline - > op2 ) ;
2010-04-20 18:57:45 +08:00
opline - > result_type = IS_TMP_VAR ;
opline - > result . var = get_temporary_variable ( CG ( active_op_array ) ) ;
GET_NODE ( result , opline - > result ) ;
1999-04-08 02:10:10 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_if_cond ( const znode * cond , znode * closing_bracket_token TSRMLS_DC ) /* { { { */
1999-04-08 02:10:10 +08:00
{
int if_cond_op_number = get_next_op_number ( CG ( active_op_array ) ) ;
2001-07-28 18:51:54 +08:00
zend_op * opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
1999-04-08 02:10:10 +08:00
opline - > opcode = ZEND_JMPZ ;
2010-04-20 18:57:45 +08:00
SET_NODE ( opline - > op1 , cond ) ;
closing_bracket_token - > u . op . opline_num = if_cond_op_number ;
1999-04-08 02:10:10 +08:00
SET_UNUSED ( opline - > op2 ) ;
INC_BPC ( CG ( active_op_array ) ) ;
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_if_after_statement ( const znode * closing_bracket_token , unsigned char initialize TSRMLS_DC ) /* { { { */
1999-04-08 02:10:10 +08:00
{
int if_end_op_number = get_next_op_number ( CG ( active_op_array ) ) ;
2001-07-28 18:51:54 +08:00
zend_op * opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
1999-04-08 02:10:10 +08:00
zend_llist * jmp_list_ptr ;
opline - > opcode = ZEND_JMP ;
/* save for backpatching */
if ( initialize ) {
zend_llist jmp_list ;
zend_llist_init ( & jmp_list , sizeof ( int ) , NULL , 0 ) ;
zend_stack_push ( & CG ( bp_stack ) , ( void * ) & jmp_list , sizeof ( zend_llist ) ) ;
}
zend_stack_top ( & CG ( bp_stack ) , ( void * * ) & jmp_list_ptr ) ;
zend_llist_add_element ( jmp_list_ptr , & if_end_op_number ) ;
2006-05-10 07:53:23 +08:00
2010-04-20 18:57:45 +08:00
CG ( active_op_array ) - > opcodes [ closing_bracket_token - > u . op . opline_num ] . op2 . opline_num = if_end_op_number + 1 ;
1999-04-08 02:10:10 +08:00
SET_UNUSED ( opline - > op1 ) ;
SET_UNUSED ( opline - > op2 ) ;
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_if_end ( TSRMLS_D ) /* { { { */
1999-04-08 02:10:10 +08:00
{
int next_op_number = get_next_op_number ( CG ( active_op_array ) ) ;
zend_llist * jmp_list_ptr ;
zend_llist_element * le ;
zend_stack_top ( & CG ( bp_stack ) , ( void * * ) & jmp_list_ptr ) ;
for ( le = jmp_list_ptr - > head ; le ; le = le - > next ) {
2010-04-20 18:57:45 +08:00
CG ( active_op_array ) - > opcodes [ * ( ( int * ) le - > data ) ] . op1 . opline_num = next_op_number ;
1999-04-08 02:10:10 +08:00
}
zend_llist_destroy ( jmp_list_ptr ) ;
zend_stack_del_top ( & CG ( bp_stack ) ) ;
DEC_BPC ( CG ( active_op_array ) ) ;
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
void zend_check_writable_variable ( const znode * variable ) /* { { { */
2001-12-17 03:45:49 +08:00
{
2010-04-20 18:57:45 +08:00
zend_uint type = variable - > EA ;
2004-07-20 16:58:18 +08:00
2001-12-17 03:45:49 +08:00
if ( type & ZEND_PARSED_METHOD_CALL ) {
2002-06-29 16:42:17 +08:00
zend_error ( E_COMPILE_ERROR , " Can't use method return value in write context " ) ;
2001-12-17 03:45:49 +08:00
}
2001-12-26 22:46:18 +08:00
if ( type = = ZEND_PARSED_FUNCTION_CALL ) {
2002-06-29 16:42:17 +08:00
zend_error ( E_COMPILE_ERROR , " Can't use function return value in write context " ) ;
2001-12-17 03:45:49 +08:00
}
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_begin_variable_parse ( TSRMLS_D ) /* { { { */
1999-04-08 02:10:10 +08:00
{
zend_llist fetch_list ;
1999-09-20 23:44:30 +08:00
zend_llist_init ( & fetch_list , sizeof ( zend_op ) , NULL , 0 ) ;
1999-04-08 02:10:10 +08:00
zend_stack_push ( & CG ( bp_stack ) , ( void * ) & fetch_list , sizeof ( zend_llist ) ) ;
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_end_variable_parse ( znode * variable , int type , int arg_offset TSRMLS_DC ) /* { { { */
1999-04-08 02:10:10 +08:00
{
zend_llist * fetch_list_ptr ;
zend_llist_element * le ;
2008-05-07 20:04:39 +08:00
zend_op * opline = NULL ;
zend_op * opline_ptr ;
zend_uint this_var = - 1 ;
1999-04-08 02:10:10 +08:00
zend_stack_top ( & CG ( bp_stack ) , ( void * * ) & fetch_list_ptr ) ;
le = fetch_list_ptr - > head ;
2004-07-20 16:58:18 +08:00
/* TODO: $foo->x->y->z = 1 should fetch "x" and "y" for R or RW, not just W */
2006-05-10 07:53:23 +08:00
2005-07-19 15:33:00 +08:00
if ( le ) {
1999-09-20 23:44:30 +08:00
opline_ptr = ( zend_op * ) le - > data ;
2005-07-19 15:33:00 +08:00
if ( opline_is_fetch_this ( opline_ptr TSRMLS_CC ) ) {
2008-05-07 20:04:39 +08:00
/* convert to FETCH_?(this) into IS_CV */
if ( CG ( active_op_array ) - > last = = 0 | |
CG ( active_op_array ) - > opcodes [ CG ( active_op_array ) - > last - 1 ] . opcode ! = ZEND_BEGIN_SILENCE ) {
2010-04-20 18:57:45 +08:00
this_var = opline_ptr - > result . var ;
2008-05-07 20:04:39 +08:00
if ( CG ( active_op_array ) - > this_var = = - 1 ) {
2010-04-20 19:05:54 +08:00
CG ( active_op_array ) - > this_var = lookup_cv ( CG ( active_op_array ) , Z_STRVAL ( CONSTANT ( opline_ptr - > op1 . constant ) ) , Z_STRLEN ( CONSTANT ( opline_ptr - > op1 . constant ) ) TSRMLS_CC ) ;
2010-04-20 18:57:45 +08:00
Z_TYPE ( CONSTANT ( opline_ptr - > op1 . constant ) ) = IS_NULL ;
2008-05-07 20:04:39 +08:00
} else {
2010-04-20 18:57:45 +08:00
zend_del_literal ( CG ( active_op_array ) , opline_ptr - > op1 . constant ) ;
2008-05-07 20:04:39 +08:00
}
le = le - > next ;
if ( variable - > op_type = = IS_VAR & &
2010-04-20 18:57:45 +08:00
variable - > u . op . var = = this_var ) {
2008-05-07 20:04:39 +08:00
variable - > op_type = IS_CV ;
2010-04-20 18:57:45 +08:00
variable - > u . op . var = CG ( active_op_array ) - > this_var ;
2008-05-07 20:04:39 +08:00
}
} else if ( CG ( active_op_array ) - > this_var = = - 1 ) {
2010-04-20 19:05:54 +08:00
CG ( active_op_array ) - > this_var = lookup_cv ( CG ( active_op_array ) , estrndup ( " this " , sizeof ( " this " ) - 1 ) , sizeof ( " this " ) - 1 TSRMLS_CC ) ;
2008-05-07 20:04:39 +08:00
}
1999-04-08 02:10:10 +08:00
}
2002-03-15 23:09:46 +08:00
2008-05-07 20:04:39 +08:00
while ( le ) {
opline_ptr = ( zend_op * ) le - > data ;
2005-07-19 15:33:00 +08:00
opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
memcpy ( opline , opline_ptr , sizeof ( zend_op ) ) ;
2010-04-20 18:57:45 +08:00
if ( opline - > op1_type = = IS_VAR & &
opline - > op1 . var = = this_var ) {
opline - > op1_type = IS_CV ;
opline - > op1 . var = CG ( active_op_array ) - > this_var ;
2008-05-07 20:04:39 +08:00
}
2005-07-19 15:33:00 +08:00
switch ( type ) {
case BP_VAR_R :
2010-04-20 18:57:45 +08:00
if ( opline - > opcode = = ZEND_FETCH_DIM_W & & opline - > op2_type = = IS_UNUSED ) {
2005-07-19 15:33:00 +08:00
zend_error ( E_COMPILE_ERROR , " Cannot use [] for reading " ) ;
}
opline - > opcode - = 3 ;
break ;
case BP_VAR_W :
break ;
case BP_VAR_RW :
opline - > opcode + = 3 ;
break ;
case BP_VAR_IS :
2010-04-20 18:57:45 +08:00
if ( opline - > opcode = = ZEND_FETCH_DIM_W & & opline - > op2_type = = IS_UNUSED ) {
2005-07-19 15:33:00 +08:00
zend_error ( E_COMPILE_ERROR , " Cannot use [] for reading " ) ;
}
opline - > opcode + = 6 ; /* 3+3 */
break ;
case BP_VAR_FUNC_ARG :
opline - > opcode + = 9 ; /* 3+3+3 */
2010-04-20 18:57:45 +08:00
opline - > extended_value | = arg_offset ;
2005-07-19 15:33:00 +08:00
break ;
case BP_VAR_UNSET :
2010-04-20 18:57:45 +08:00
if ( opline - > opcode = = ZEND_FETCH_DIM_W & & opline - > op2_type = = IS_UNUSED ) {
2005-07-19 15:33:00 +08:00
zend_error ( E_COMPILE_ERROR , " Cannot use [] for unsetting " ) ;
}
opline - > opcode + = 12 ; /* 3+3+3+3 */
break ;
}
le = le - > next ;
}
2008-05-07 20:04:39 +08:00
if ( opline & & type = = BP_VAR_W & & arg_offset ) {
2010-04-20 18:57:45 +08:00
opline - > extended_value | = ZEND_FETCH_MAKE_REF ;
2007-10-23 17:55:11 +08:00
}
1999-04-08 02:10:10 +08:00
}
zend_llist_destroy ( fetch_list_ptr ) ;
zend_stack_del_top ( & CG ( bp_stack ) ) ;
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_add_string ( znode * result , const znode * op1 , znode * op2 TSRMLS_DC ) /* { { { */
1999-04-08 02:10:10 +08:00
{
2007-05-18 21:12:05 +08:00
zend_op * opline ;
if ( Z_STRLEN ( op2 - > u . constant ) > 1 ) {
opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
opline - > opcode = ZEND_ADD_STRING ;
} else if ( Z_STRLEN ( op2 - > u . constant ) = = 1 ) {
int ch = * Z_STRVAL ( op2 - > u . constant ) ;
1999-04-08 02:10:10 +08:00
2007-05-18 21:12:05 +08:00
/* Free memory and use ZEND_ADD_CHAR in case of 1 character strings */
efree ( Z_STRVAL ( op2 - > u . constant ) ) ;
ZVAL_LONG ( & op2 - > u . constant , ch ) ;
opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
opline - > opcode = ZEND_ADD_CHAR ;
} else { /* String can be empty after a variable at the end of a heredoc */
efree ( Z_STRVAL ( op2 - > u . constant ) ) ;
return ;
}
2008-07-26 23:30:28 +08:00
if ( op1 ) {
2010-04-20 18:57:45 +08:00
SET_NODE ( opline - > op1 , op1 ) ;
SET_NODE ( opline - > result , op1 ) ;
2008-07-26 23:30:28 +08:00
} else {
SET_UNUSED ( opline - > op1 ) ;
2010-04-20 18:57:45 +08:00
opline - > result_type = IS_TMP_VAR ;
opline - > result . var = get_temporary_variable ( CG ( active_op_array ) ) ;
2008-07-26 23:30:28 +08:00
}
2010-04-20 18:57:45 +08:00
SET_NODE ( opline - > op2 , op2 ) ;
GET_NODE ( result , opline - > result ) ;
1999-04-08 02:10:10 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_add_variable ( znode * result , const znode * op1 , const znode * op2 TSRMLS_DC ) /* { { { */
1999-04-08 02:10:10 +08:00
{
2008-07-26 23:30:28 +08:00
zend_op * opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
1999-04-08 02:10:10 +08:00
2008-07-26 23:30:28 +08:00
opline - > opcode = ZEND_ADD_VAR ;
1999-04-08 02:10:10 +08:00
2008-07-26 23:30:28 +08:00
if ( op1 ) {
2010-04-20 18:57:45 +08:00
SET_NODE ( opline - > op1 , op1 ) ;
SET_NODE ( opline - > result , op1 ) ;
1999-04-08 02:10:10 +08:00
} else {
2008-07-26 23:30:28 +08:00
SET_UNUSED ( opline - > op1 ) ;
2010-04-20 18:57:45 +08:00
opline - > result_type = IS_TMP_VAR ;
opline - > result . var = get_temporary_variable ( CG ( active_op_array ) ) ;
1999-04-08 02:10:10 +08:00
}
2010-04-20 18:57:45 +08:00
SET_NODE ( opline - > op2 , op2 ) ;
GET_NODE ( result , opline - > result ) ;
1999-04-08 02:10:10 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_free ( znode * op1 TSRMLS_DC ) /* { { { */
1999-04-08 02:10:10 +08:00
{
if ( op1 - > op_type = = IS_TMP_VAR ) {
2001-07-28 18:51:54 +08:00
zend_op * opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
1999-11-17 02:46:32 +08:00
1999-12-21 04:01:19 +08:00
opline - > opcode = ZEND_FREE ;
2010-04-20 18:57:45 +08:00
SET_NODE ( opline - > op1 , op1 ) ;
1999-12-21 04:01:19 +08:00
SET_UNUSED ( opline - > op2 ) ;
1999-07-24 19:24:19 +08:00
} else if ( op1 - > op_type = = IS_VAR ) {
1999-04-08 02:10:10 +08:00
zend_op * opline = & CG ( active_op_array ) - > opcodes [ CG ( active_op_array ) - > last - 1 ] ;
2003-01-27 23:13:01 +08:00
while ( opline - > opcode = = ZEND_END_SILENCE | | opline - > opcode = = ZEND_EXT_FCALL_END | | opline - > opcode = = ZEND_OP_DATA ) {
1999-12-21 04:01:19 +08:00
opline - - ;
}
2010-04-20 18:57:45 +08:00
if ( opline - > result_type = = IS_VAR
& & opline - > result . var = = op1 - > u . op . var ) {
opline - > result_type | = EXT_TYPE_UNUSED ;
1999-04-08 02:10:10 +08:00
} else {
while ( opline > CG ( active_op_array ) - > opcodes ) {
2005-06-10 15:56:40 +08:00
if ( opline - > opcode = = ZEND_FETCH_DIM_R
2010-04-20 18:57:45 +08:00
& & opline - > op1_type = = IS_VAR
& & opline - > op1 . var = = op1 - > u . op . var ) {
2000-02-03 00:47:43 +08:00
/* This should the end of a list() construct
* Mark its result as unused
*/
opline - > extended_value = ZEND_FETCH_STANDARD ;
break ;
2010-04-20 18:57:45 +08:00
} else if ( opline - > result_type = = IS_VAR
& & opline - > result . var = = op1 - > u . op . var ) {
2005-06-10 15:56:40 +08:00
if ( opline - > opcode = = ZEND_NEW ) {
2010-04-20 18:57:45 +08:00
opline - > result_type | = EXT_TYPE_UNUSED ;
2005-06-10 15:56:40 +08:00
}
2000-02-03 00:47:43 +08:00
break ;
1999-04-08 02:10:10 +08:00
}
opline - - ;
}
}
2000-12-18 21:28:32 +08:00
} else if ( op1 - > op_type = = IS_CONST ) {
zval_dtor ( & op1 - > u . constant ) ;
2006-05-10 07:53:23 +08:00
}
1999-04-08 02:10:10 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
int zend_do_verify_access_types ( const znode * current_access_type , const znode * new_modifier ) /* { { { */
2002-12-07 01:09:44 +08:00
{
2006-05-10 07:53:23 +08:00
if ( ( Z_LVAL ( current_access_type - > u . constant ) & ZEND_ACC_PPP_MASK )
2007-11-14 00:52:14 +08:00
& & ( Z_LVAL ( new_modifier - > u . constant ) & ZEND_ACC_PPP_MASK ) ) {
2002-12-07 01:09:44 +08:00
zend_error ( E_COMPILE_ERROR , " Multiple access type modifiers are not allowed " ) ;
}
2007-11-14 00:52:14 +08:00
if ( ( Z_LVAL ( current_access_type - > u . constant ) & ZEND_ACC_ABSTRACT )
& & ( Z_LVAL ( new_modifier - > u . constant ) & ZEND_ACC_ABSTRACT ) ) {
zend_error ( E_COMPILE_ERROR , " Multiple abstract modifiers are not allowed " ) ;
}
if ( ( Z_LVAL ( current_access_type - > u . constant ) & ZEND_ACC_STATIC )
& & ( Z_LVAL ( new_modifier - > u . constant ) & ZEND_ACC_STATIC ) ) {
zend_error ( E_COMPILE_ERROR , " Multiple static modifiers are not allowed " ) ;
}
if ( ( Z_LVAL ( current_access_type - > u . constant ) & ZEND_ACC_FINAL )
& & ( Z_LVAL ( new_modifier - > u . constant ) & ZEND_ACC_FINAL ) ) {
zend_error ( E_COMPILE_ERROR , " Multiple final modifiers are not allowed " ) ;
}
2006-05-10 07:53:23 +08:00
if ( ( ( Z_LVAL ( current_access_type - > u . constant ) | Z_LVAL ( new_modifier - > u . constant ) ) & ( ZEND_ACC_ABSTRACT | ZEND_ACC_FINAL ) ) = = ( ZEND_ACC_ABSTRACT | ZEND_ACC_FINAL ) ) {
2003-03-01 22:57:49 +08:00
zend_error ( E_COMPILE_ERROR , " Cannot use the final modifier on an abstract class member " ) ;
}
2006-05-10 07:53:23 +08:00
return ( Z_LVAL ( current_access_type - > u . constant ) | Z_LVAL ( new_modifier - > u . constant ) ) ;
2002-12-07 01:09:44 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
2002-12-07 01:09:44 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_begin_function_declaration ( znode * function_token , znode * function_name , int is_method , int return_reference , znode * fn_flags_znode TSRMLS_DC ) /* { { { */
1999-04-08 02:10:10 +08:00
{
zend_op_array op_array ;
char * name = function_name - > u . constant . value . str . val ;
2003-01-16 04:35:06 +08:00
int name_len = function_name - > u . constant . value . str . len ;
2010-04-20 18:57:45 +08:00
int function_begin_line = function_token - > u . op . opline_num ;
2003-03-05 19:14:44 +08:00
zend_uint fn_flags ;
2003-09-17 18:14:12 +08:00
char * lcname ;
2007-02-15 18:38:28 +08:00
zend_bool orig_interactive ;
2007-11-22 21:27:13 +08:00
ALLOCA_FLAG ( use_heap )
2003-03-05 19:14:44 +08:00
if ( is_method ) {
if ( CG ( active_class_entry ) - > ce_flags & ZEND_ACC_INTERFACE ) {
2006-05-10 07:53:23 +08:00
if ( ( Z_LVAL ( fn_flags_znode - > u . constant ) & ~ ( ZEND_ACC_STATIC | ZEND_ACC_PUBLIC ) ) ) {
2004-10-13 09:04:08 +08:00
zend_error ( E_COMPILE_ERROR , " Access type for interface method %s::%s() must be omitted " , CG ( active_class_entry ) - > name , function_name - > u . constant . value . str . val ) ;
2003-03-05 19:14:44 +08:00
}
2006-05-10 07:53:23 +08:00
Z_LVAL ( fn_flags_znode - > u . constant ) | = ZEND_ACC_ABSTRACT ; /* propagates to the rest of the parser */
2003-03-05 19:14:44 +08:00
}
2006-05-10 07:53:23 +08:00
fn_flags = Z_LVAL ( fn_flags_znode - > u . constant ) ; /* must be done *after* the above check */
2003-03-05 19:14:44 +08:00
} else {
fn_flags = 0 ;
}
2006-05-10 07:53:23 +08:00
if ( ( fn_flags & ZEND_ACC_STATIC ) & & ( fn_flags & ZEND_ACC_ABSTRACT ) & & ! ( CG ( active_class_entry ) - > ce_flags & ZEND_ACC_INTERFACE ) ) {
2006-10-20 10:44:26 +08:00
zend_error ( E_STRICT , " Static function %s%s%s() should not be abstract " , is_method ? CG ( active_class_entry ) - > name : " " , is_method ? " :: " : " " , Z_STRVAL ( function_name - > u . constant ) ) ;
2006-05-10 07:53:23 +08:00
}
1999-04-08 02:10:10 +08:00
function_token - > u . op_array = CG ( active_op_array ) ;
2007-02-15 18:38:28 +08:00
orig_interactive = CG ( interactive ) ;
CG ( interactive ) = 0 ;
2001-07-28 18:51:54 +08:00
init_op_array ( & op_array , ZEND_USER_FUNCTION , INITIAL_OP_ARRAY_SIZE TSRMLS_CC ) ;
2007-02-15 18:38:28 +08:00
CG ( interactive ) = orig_interactive ;
1999-12-16 04:15:32 +08:00
1999-04-08 02:10:10 +08:00
op_array . function_name = name ;
1999-12-16 04:15:32 +08:00
op_array . return_reference = return_reference ;
2007-02-15 18:38:28 +08:00
op_array . fn_flags | = fn_flags ;
2003-08-04 06:03:52 +08:00
op_array . pass_rest_by_reference = 0 ;
1999-04-08 02:10:10 +08:00
2005-04-27 21:30:53 +08:00
op_array . scope = is_method ? CG ( active_class_entry ) : NULL ;
2003-03-30 03:20:35 +08:00
op_array . prototype = NULL ;
2002-03-01 22:04:51 +08:00
2003-04-01 04:42:01 +08:00
op_array . line_start = zend_get_compiled_lineno ( TSRMLS_C ) ;
2003-03-20 05:17:47 +08:00
1999-04-08 02:10:10 +08:00
if ( is_method ) {
2010-04-20 19:05:54 +08:00
lcname = CG ( new_interned_string ) ( zend_str_tolower_dup ( name , name_len ) , name_len + 1 , 1 TSRMLS_CC ) ;
2003-09-17 18:14:12 +08:00
if ( zend_hash_add ( & CG ( active_class_entry ) - > function_table , lcname , name_len + 1 , & op_array , sizeof ( zend_op_array ) , ( void * * ) & CG ( active_op_array ) ) = = FAILURE ) {
2009-04-08 21:17:09 +08:00
zend_error ( E_COMPILE_ERROR , " Cannot redeclare %s::%s() " , CG ( active_class_entry ) - > name , name ) ;
2002-11-11 06:02:28 +08:00
}
2003-02-13 00:28:34 +08:00
2002-12-07 01:09:44 +08:00
if ( fn_flags & ZEND_ACC_ABSTRACT ) {
2004-03-10 00:38:37 +08:00
CG ( active_class_entry ) - > ce_flags | = ZEND_ACC_IMPLICIT_ABSTRACT_CLASS ;
2002-12-07 01:09:44 +08:00
}
2002-11-11 06:02:28 +08:00
2003-02-13 00:28:34 +08:00
if ( ! ( fn_flags & ZEND_ACC_PPP_MASK ) ) {
fn_flags | = ZEND_ACC_PUBLIC ;
}
2008-06-03 22:07:15 +08:00
if ( CG ( active_class_entry ) - > ce_flags & ZEND_ACC_INTERFACE ) {
if ( ( name_len = = sizeof ( ZEND_CALL_FUNC_NAME ) - 1 ) & & ( ! memcmp ( lcname , ZEND_CALL_FUNC_NAME , sizeof ( ZEND_CALL_FUNC_NAME ) - 1 ) ) ) {
if ( fn_flags & ( ( ZEND_ACC_PPP_MASK | ZEND_ACC_STATIC ) ^ ZEND_ACC_PUBLIC ) ) {
2009-06-07 23:46:54 +08:00
zend_error ( E_WARNING , " The magic method __call() must have public visibility and cannot be static " ) ;
2008-06-03 22:07:15 +08:00
}
} else if ( ( name_len = = sizeof ( ZEND_CALLSTATIC_FUNC_NAME ) - 1 ) & & ( ! memcmp ( lcname , ZEND_CALLSTATIC_FUNC_NAME , sizeof ( ZEND_CALLSTATIC_FUNC_NAME ) - 1 ) ) ) {
if ( ( fn_flags & ( ZEND_ACC_PPP_MASK ^ ZEND_ACC_PUBLIC ) ) | | ( fn_flags & ZEND_ACC_STATIC ) = = 0 ) {
2008-06-11 07:09:09 +08:00
zend_error ( E_WARNING , " The magic method __callStatic() must have public visibility and be static " ) ;
2008-06-03 22:07:15 +08:00
}
} else if ( ( name_len = = sizeof ( ZEND_GET_FUNC_NAME ) - 1 ) & & ( ! memcmp ( lcname , ZEND_GET_FUNC_NAME , sizeof ( ZEND_GET_FUNC_NAME ) - 1 ) ) ) {
if ( fn_flags & ( ( ZEND_ACC_PPP_MASK | ZEND_ACC_STATIC ) ^ ZEND_ACC_PUBLIC ) ) {
2009-06-07 23:46:54 +08:00
zend_error ( E_WARNING , " The magic method __get() must have public visibility and cannot be static " ) ;
2008-06-03 22:07:15 +08:00
}
} else if ( ( name_len = = sizeof ( ZEND_SET_FUNC_NAME ) - 1 ) & & ( ! memcmp ( lcname , ZEND_SET_FUNC_NAME , sizeof ( ZEND_SET_FUNC_NAME ) - 1 ) ) ) {
if ( fn_flags & ( ( ZEND_ACC_PPP_MASK | ZEND_ACC_STATIC ) ^ ZEND_ACC_PUBLIC ) ) {
2009-06-07 23:46:54 +08:00
zend_error ( E_WARNING , " The magic method __set() must have public visibility and cannot be static " ) ;
2008-06-03 22:07:15 +08:00
}
} else if ( ( name_len = = sizeof ( ZEND_UNSET_FUNC_NAME ) - 1 ) & & ( ! memcmp ( lcname , ZEND_UNSET_FUNC_NAME , sizeof ( ZEND_UNSET_FUNC_NAME ) - 1 ) ) ) {
if ( fn_flags & ( ( ZEND_ACC_PPP_MASK | ZEND_ACC_STATIC ) ^ ZEND_ACC_PUBLIC ) ) {
2009-06-07 23:46:54 +08:00
zend_error ( E_WARNING , " The magic method __unset() must have public visibility and cannot be static " ) ;
2008-06-03 22:07:15 +08:00
}
} else if ( ( name_len = = sizeof ( ZEND_ISSET_FUNC_NAME ) - 1 ) & & ( ! memcmp ( lcname , ZEND_ISSET_FUNC_NAME , sizeof ( ZEND_ISSET_FUNC_NAME ) - 1 ) ) ) {
if ( fn_flags & ( ( ZEND_ACC_PPP_MASK | ZEND_ACC_STATIC ) ^ ZEND_ACC_PUBLIC ) ) {
2009-06-07 23:46:54 +08:00
zend_error ( E_WARNING , " The magic method __isset() must have public visibility and cannot be static " ) ;
2008-06-03 22:07:15 +08:00
}
} else if ( ( name_len = = sizeof ( ZEND_TOSTRING_FUNC_NAME ) - 1 ) & & ( ! memcmp ( lcname , ZEND_TOSTRING_FUNC_NAME , sizeof ( ZEND_TOSTRING_FUNC_NAME ) - 1 ) ) ) {
if ( fn_flags & ( ( ZEND_ACC_PPP_MASK | ZEND_ACC_STATIC ) ^ ZEND_ACC_PUBLIC ) ) {
2009-06-07 23:46:54 +08:00
zend_error ( E_WARNING , " The magic method __toString() must have public visibility and cannot be static " ) ;
2008-06-03 22:07:15 +08:00
}
}
} else {
2010-04-05 07:28:20 +08:00
char * class_lcname ;
class_lcname = do_alloca ( CG ( active_class_entry ) - > name_length + 1 , use_heap ) ;
zend_str_tolower_copy ( class_lcname , CG ( active_class_entry ) - > name , CG ( active_class_entry ) - > name_length ) ;
2004-08-27 06:25:55 +08:00
/* Improve after RC: cache the lowercase class name */
2010-04-05 07:28:20 +08:00
if ( ( CG ( active_class_entry ) - > name_length = = name_len ) & & ( ! memcmp ( class_lcname , lcname , name_len ) ) ) {
2004-08-27 06:25:55 +08:00
if ( CG ( active_class_entry ) - > constructor ) {
zend_error ( E_STRICT , " Redefining already defined constructor for class %s " , CG ( active_class_entry ) - > name ) ;
} else {
CG ( active_class_entry ) - > constructor = ( zend_function * ) CG ( active_op_array ) ;
}
} else if ( ( name_len = = sizeof ( ZEND_CONSTRUCTOR_FUNC_NAME ) - 1 ) & & ( ! memcmp ( lcname , ZEND_CONSTRUCTOR_FUNC_NAME , sizeof ( ZEND_CONSTRUCTOR_FUNC_NAME ) ) ) ) {
if ( CG ( active_class_entry ) - > constructor ) {
zend_error ( E_STRICT , " Redefining already defined constructor for class %s " , CG ( active_class_entry ) - > name ) ;
}
2004-03-26 02:00:50 +08:00
CG ( active_class_entry ) - > constructor = ( zend_function * ) CG ( active_op_array ) ;
2007-09-29 16:52:40 +08:00
} else if ( ( name_len = = sizeof ( ZEND_DESTRUCTOR_FUNC_NAME ) - 1 ) & & ( ! memcmp ( lcname , ZEND_DESTRUCTOR_FUNC_NAME , sizeof ( ZEND_DESTRUCTOR_FUNC_NAME ) - 1 ) ) ) {
2004-08-27 06:25:55 +08:00
CG ( active_class_entry ) - > destructor = ( zend_function * ) CG ( active_op_array ) ;
2007-09-29 16:52:40 +08:00
} else if ( ( name_len = = sizeof ( ZEND_CLONE_FUNC_NAME ) - 1 ) & & ( ! memcmp ( lcname , ZEND_CLONE_FUNC_NAME , sizeof ( ZEND_CLONE_FUNC_NAME ) - 1 ) ) ) {
2004-08-27 06:25:55 +08:00
CG ( active_class_entry ) - > clone = ( zend_function * ) CG ( active_op_array ) ;
2007-09-29 16:52:40 +08:00
} else if ( ( name_len = = sizeof ( ZEND_CALL_FUNC_NAME ) - 1 ) & & ( ! memcmp ( lcname , ZEND_CALL_FUNC_NAME , sizeof ( ZEND_CALL_FUNC_NAME ) - 1 ) ) ) {
2008-06-03 22:07:15 +08:00
if ( fn_flags & ( ( ZEND_ACC_PPP_MASK | ZEND_ACC_STATIC ) ^ ZEND_ACC_PUBLIC ) ) {
2009-06-07 23:46:54 +08:00
zend_error ( E_WARNING , " The magic method __call() must have public visibility and cannot be static " ) ;
2008-06-03 22:07:15 +08:00
}
2004-08-27 06:25:55 +08:00
CG ( active_class_entry ) - > __call = ( zend_function * ) CG ( active_op_array ) ;
2007-09-29 16:52:40 +08:00
} else if ( ( name_len = = sizeof ( ZEND_CALLSTATIC_FUNC_NAME ) - 1 ) & & ( ! memcmp ( lcname , ZEND_CALLSTATIC_FUNC_NAME , sizeof ( ZEND_CALLSTATIC_FUNC_NAME ) - 1 ) ) ) {
2008-06-03 22:07:15 +08:00
if ( ( fn_flags & ( ZEND_ACC_PPP_MASK ^ ZEND_ACC_PUBLIC ) ) | | ( fn_flags & ZEND_ACC_STATIC ) = = 0 ) {
2008-06-11 07:09:09 +08:00
zend_error ( E_WARNING , " The magic method __callStatic() must have public visibility and be static " ) ;
2008-06-03 22:07:15 +08:00
}
2007-09-29 16:52:40 +08:00
CG ( active_class_entry ) - > __callstatic = ( zend_function * ) CG ( active_op_array ) ;
} else if ( ( name_len = = sizeof ( ZEND_GET_FUNC_NAME ) - 1 ) & & ( ! memcmp ( lcname , ZEND_GET_FUNC_NAME , sizeof ( ZEND_GET_FUNC_NAME ) - 1 ) ) ) {
2008-06-03 22:07:15 +08:00
if ( fn_flags & ( ( ZEND_ACC_PPP_MASK | ZEND_ACC_STATIC ) ^ ZEND_ACC_PUBLIC ) ) {
2009-06-07 23:46:54 +08:00
zend_error ( E_WARNING , " The magic method __get() must have public visibility and cannot be static " ) ;
2008-06-03 22:07:15 +08:00
}
2004-08-27 06:25:55 +08:00
CG ( active_class_entry ) - > __get = ( zend_function * ) CG ( active_op_array ) ;
2007-09-29 16:52:40 +08:00
} else if ( ( name_len = = sizeof ( ZEND_SET_FUNC_NAME ) - 1 ) & & ( ! memcmp ( lcname , ZEND_SET_FUNC_NAME , sizeof ( ZEND_SET_FUNC_NAME ) - 1 ) ) ) {
2008-06-03 22:07:15 +08:00
if ( fn_flags & ( ( ZEND_ACC_PPP_MASK | ZEND_ACC_STATIC ) ^ ZEND_ACC_PUBLIC ) ) {
2009-06-07 23:46:54 +08:00
zend_error ( E_WARNING , " The magic method __set() must have public visibility and cannot be static " ) ;
2008-06-03 22:07:15 +08:00
}
2004-08-27 06:25:55 +08:00
CG ( active_class_entry ) - > __set = ( zend_function * ) CG ( active_op_array ) ;
2007-09-29 16:52:40 +08:00
} else if ( ( name_len = = sizeof ( ZEND_UNSET_FUNC_NAME ) - 1 ) & & ( ! memcmp ( lcname , ZEND_UNSET_FUNC_NAME , sizeof ( ZEND_UNSET_FUNC_NAME ) - 1 ) ) ) {
2008-06-03 22:07:15 +08:00
if ( fn_flags & ( ( ZEND_ACC_PPP_MASK | ZEND_ACC_STATIC ) ^ ZEND_ACC_PUBLIC ) ) {
2009-06-07 23:46:54 +08:00
zend_error ( E_WARNING , " The magic method __unset() must have public visibility and cannot be static " ) ;
2008-06-03 22:07:15 +08:00
}
2005-07-08 00:07:09 +08:00
CG ( active_class_entry ) - > __unset = ( zend_function * ) CG ( active_op_array ) ;
2007-09-29 16:52:40 +08:00
} else if ( ( name_len = = sizeof ( ZEND_ISSET_FUNC_NAME ) - 1 ) & & ( ! memcmp ( lcname , ZEND_ISSET_FUNC_NAME , sizeof ( ZEND_ISSET_FUNC_NAME ) - 1 ) ) ) {
2008-06-03 22:07:15 +08:00
if ( fn_flags & ( ( ZEND_ACC_PPP_MASK | ZEND_ACC_STATIC ) ^ ZEND_ACC_PUBLIC ) ) {
2009-06-07 23:46:54 +08:00
zend_error ( E_WARNING , " The magic method __isset() must have public visibility and cannot be static " ) ;
2008-06-03 22:07:15 +08:00
}
2005-07-08 00:07:09 +08:00
CG ( active_class_entry ) - > __isset = ( zend_function * ) CG ( active_op_array ) ;
2007-09-29 16:52:40 +08:00
} else if ( ( name_len = = sizeof ( ZEND_TOSTRING_FUNC_NAME ) - 1 ) & & ( ! memcmp ( lcname , ZEND_TOSTRING_FUNC_NAME , sizeof ( ZEND_TOSTRING_FUNC_NAME ) - 1 ) ) ) {
2008-06-03 22:07:15 +08:00
if ( fn_flags & ( ( ZEND_ACC_PPP_MASK | ZEND_ACC_STATIC ) ^ ZEND_ACC_PUBLIC ) ) {
2009-06-07 23:46:54 +08:00
zend_error ( E_WARNING , " The magic method __toString() must have public visibility and cannot be static " ) ;
2008-06-03 22:07:15 +08:00
}
2006-05-10 07:53:23 +08:00
CG ( active_class_entry ) - > __tostring = ( zend_function * ) CG ( active_op_array ) ;
2004-08-27 06:25:55 +08:00
} else if ( ! ( fn_flags & ZEND_ACC_STATIC ) ) {
CG ( active_op_array ) - > fn_flags | = ZEND_ACC_ALLOW_STATIC ;
2004-03-04 18:02:59 +08:00
}
2010-04-05 07:28:20 +08:00
free_alloca ( class_lcname , use_heap ) ;
2001-11-03 19:59:14 +08:00
}
2003-09-17 18:14:12 +08:00
2010-04-20 19:05:54 +08:00
str_efree ( lcname ) ;
1999-04-08 02:10:10 +08:00
} else {
2001-07-28 18:51:54 +08:00
zend_op * opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
2010-04-20 18:57:45 +08:00
zval key ;
1999-05-15 23:47:24 +08:00
2007-09-29 03:52:53 +08:00
if ( CG ( current_namespace ) ) {
/* Prefix function name with current namespcae name */
znode tmp ;
tmp . u . constant = * CG ( current_namespace ) ;
zval_copy_ctor ( & tmp . u . constant ) ;
zend_do_build_namespace_name ( & tmp , & tmp , function_name TSRMLS_CC ) ;
op_array . function_name = Z_STRVAL ( tmp . u . constant ) ;
name_len = Z_STRLEN ( tmp . u . constant ) ;
lcname = zend_str_tolower_dup ( Z_STRVAL ( tmp . u . constant ) , name_len ) ;
2010-04-20 19:05:54 +08:00
} else {
lcname = zend_str_tolower_dup ( name , name_len ) ;
2007-09-29 03:52:53 +08:00
}
2002-09-25 03:05:53 +08:00
opline - > opcode = ZEND_DECLARE_FUNCTION ;
2010-04-20 18:57:45 +08:00
opline - > op1_type = IS_CONST ;
build_runtime_defined_function_key ( & key , lcname , name_len TSRMLS_CC ) ;
2010-04-21 22:58:33 +08:00
opline - > op1 . constant = zend_add_literal ( CG ( active_op_array ) , & key ) ;
2010-04-20 18:57:45 +08:00
Z_HASH_P ( & CONSTANT ( opline - > op1 . constant ) ) = zend_hash_func ( Z_STRVAL ( CONSTANT ( opline - > op1 . constant ) ) , Z_STRLEN ( CONSTANT ( opline - > op1 . constant ) ) ) ;
opline - > op2_type = IS_CONST ;
LITERAL_STRINGL ( opline - > op2 , lcname , name_len , 0 ) ;
2010-04-21 22:58:33 +08:00
CALCULATE_LITERAL_HASH ( opline - > op2 . constant ) ;
1999-05-15 23:47:24 +08:00
opline - > extended_value = ZEND_DECLARE_FUNCTION ;
2010-04-20 18:57:45 +08:00
zend_hash_quick_update ( CG ( function_table ) , Z_STRVAL ( key ) , Z_STRLEN ( key ) , Z_HASH_P ( & CONSTANT ( opline - > op1 . constant ) ) , & op_array , sizeof ( zend_op_array ) , ( void * * ) & CG ( active_op_array ) ) ;
1999-04-08 02:10:10 +08:00
}
2008-03-18 16:36:30 +08:00
if ( CG ( compiler_options ) & ZEND_COMPILE_EXTENDED_INFO ) {
2001-07-28 18:51:54 +08:00
zend_op * opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
1999-04-08 02:10:10 +08:00
opline - > opcode = ZEND_EXT_NOP ;
opline - > lineno = function_begin_line ;
SET_UNUSED ( opline - > op1 ) ;
SET_UNUSED ( opline - > op2 ) ;
}
2006-05-10 07:53:23 +08:00
1999-09-09 22:15:17 +08:00
{
/* Push a seperator to the switch and foreach stacks */
zend_switch_entry switch_entry ;
2006-05-10 07:53:23 +08:00
2004-03-06 01:18:34 +08:00
switch_entry . cond . op_type = IS_UNUSED ;
1999-09-09 22:15:17 +08:00
switch_entry . default_case = 0 ;
switch_entry . control_var = 0 ;
zend_stack_push ( & CG ( switch_cond_stack ) , ( void * ) & switch_entry , sizeof ( switch_entry ) ) ;
2004-03-05 21:04:21 +08:00
{
/* Foreach stack separator */
zend_op dummy_opline ;
2010-04-20 18:57:45 +08:00
dummy_opline . result_type = IS_UNUSED ;
dummy_opline . op1_type = IS_UNUSED ;
2004-03-05 21:04:21 +08:00
zend_stack_push ( & CG ( foreach_copy_stack ) , ( void * ) & dummy_opline , sizeof ( zend_op ) ) ;
}
1999-09-09 22:15:17 +08:00
}
2003-04-03 00:51:49 +08:00
if ( CG ( doc_comment ) ) {
2005-06-08 02:11:56 +08:00
CG ( active_op_array ) - > doc_comment = CG ( doc_comment ) ;
2003-04-03 00:51:49 +08:00
CG ( active_op_array ) - > doc_comment_len = CG ( doc_comment_len ) ;
2005-06-08 02:11:56 +08:00
CG ( doc_comment ) = NULL ;
CG ( doc_comment_len ) = 0 ;
2003-04-03 00:51:49 +08:00
}
2008-03-28 22:35:01 +08:00
zend_stack_push ( & CG ( labels_stack ) , ( void * ) & CG ( labels ) , sizeof ( HashTable * ) ) ;
CG ( labels ) = NULL ;
1999-04-08 02:10:10 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2010-04-20 03:45:03 +08:00
void zend_do_begin_lambda_function_declaration ( znode * result , znode * function_token , int return_reference , int is_static TSRMLS_DC ) /* { { { */
2008-07-14 17:49:03 +08:00
{
znode function_name ;
zend_op_array * current_op_array = CG ( active_op_array ) ;
int current_op_number = get_next_op_number ( CG ( active_op_array ) ) ;
zend_op * current_op ;
function_name . op_type = IS_CONST ;
2008-11-04 03:28:32 +08:00
ZVAL_STRINGL ( & function_name . u . constant , " {closure} " , sizeof ( " {closure} " ) - 1 , 1 ) ;
2008-07-14 17:49:03 +08:00
zend_do_begin_function_declaration ( function_token , & function_name , 0 , return_reference , NULL TSRMLS_CC ) ;
result - > op_type = IS_TMP_VAR ;
2010-04-20 18:57:45 +08:00
result - > u . op . var = get_temporary_variable ( current_op_array ) ;
2008-07-14 17:49:03 +08:00
current_op = & current_op_array - > opcodes [ current_op_number ] ;
current_op - > opcode = ZEND_DECLARE_LAMBDA_FUNCTION ;
2010-04-20 18:57:45 +08:00
zend_del_literal ( current_op_array , current_op - > op2 . constant ) ;
SET_UNUSED ( current_op - > op2 ) ;
SET_NODE ( current_op - > result , result ) ;
2010-04-20 03:45:03 +08:00
if ( is_static ) {
CG ( active_op_array ) - > fn_flags | = ZEND_ACC_STATIC ;
}
2008-11-04 03:28:32 +08:00
CG ( active_op_array ) - > fn_flags | = ZEND_ACC_CLOSURE ;
2008-07-14 17:49:03 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
2008-07-14 17:49:03 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_handle_exception ( TSRMLS_D ) /* { { { */
2004-02-03 20:17:09 +08:00
{
zend_op * opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
opline - > opcode = ZEND_HANDLE_EXCEPTION ;
SET_UNUSED ( opline - > op1 ) ;
SET_UNUSED ( opline - > op2 ) ;
}
2009-07-27 22:11:53 +08:00
/* }}} */
2004-02-03 20:17:09 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_end_function_declaration ( const znode * function_token TSRMLS_DC ) /* { { { */
1999-04-08 02:10:10 +08:00
{
2004-08-02 16:27:57 +08:00
char lcname [ 16 ] ;
int name_len ;
2001-07-28 18:51:54 +08:00
zend_do_extended_info ( TSRMLS_C ) ;
zend_do_return ( NULL , 0 TSRMLS_CC ) ;
2004-02-03 20:17:09 +08:00
2001-07-31 12:53:54 +08:00
pass_two ( CG ( active_op_array ) TSRMLS_CC ) ;
2008-03-28 22:35:01 +08:00
zend_release_labels ( TSRMLS_C ) ;
2004-02-02 20:28:19 +08:00
2004-08-02 16:27:57 +08:00
if ( CG ( active_class_entry ) ) {
2004-09-10 00:51:45 +08:00
zend_check_magic_method_implementation ( CG ( active_class_entry ) , ( zend_function * ) CG ( active_op_array ) , E_COMPILE_ERROR TSRMLS_CC ) ;
2004-08-02 16:27:57 +08:00
} else {
2004-09-10 00:51:45 +08:00
/* we don't care if the function name is longer, in fact lowercasing only
* the beginning of the name speeds up the check process */
name_len = strlen ( CG ( active_op_array ) - > function_name ) ;
zend_str_tolower_copy ( lcname , CG ( active_op_array ) - > function_name , MIN ( name_len , sizeof ( lcname ) - 1 ) ) ;
lcname [ sizeof ( lcname ) - 1 ] = ' \0 ' ; /* zend_str_tolower_copy won't necessarily set the zero byte */
2004-08-03 06:41:34 +08:00
if ( name_len = = sizeof ( ZEND_AUTOLOAD_FUNC_NAME ) - 1 & & ! memcmp ( lcname , ZEND_AUTOLOAD_FUNC_NAME , sizeof ( ZEND_AUTOLOAD_FUNC_NAME ) ) & & CG ( active_op_array ) - > num_args ! = 1 ) {
2004-08-02 16:27:57 +08:00
zend_error ( E_COMPILE_ERROR , " %s() must take exactly 1 argument " , ZEND_AUTOLOAD_FUNC_NAME ) ;
}
2004-02-02 20:28:19 +08:00
}
2003-04-01 04:42:01 +08:00
CG ( active_op_array ) - > line_end = zend_get_compiled_lineno ( TSRMLS_C ) ;
1999-04-08 02:10:10 +08:00
CG ( active_op_array ) = function_token - > u . op_array ;
1999-09-09 22:15:17 +08:00
2004-02-02 20:28:19 +08:00
1999-09-09 22:15:17 +08:00
/* Pop the switch and foreach seperators */
zend_stack_del_top ( & CG ( switch_cond_stack ) ) ;
zend_stack_del_top ( & CG ( foreach_copy_stack ) ) ;
1999-04-08 02:10:10 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2010-04-20 19:16:39 +08:00
void zend_do_receive_arg ( zend_uchar op , znode * varname , const znode * offset , const znode * initialization , znode * class_type , zend_uchar pass_by_reference TSRMLS_DC ) /* { { { */
1999-04-08 02:10:10 +08:00
{
2007-05-04 14:18:53 +08:00
zend_op * opline ;
ntroduce infrastructure for supplying information about arguments,
including:
- Whether or not to pass by ref (replaces the old arg_types, with arg_info)
- Argument name (for future use, maybe introspection)
- Class/Interface name (for type hints)
- If a class/interface name is available, whether to allow a null instance
Both user and builtin functions share the same data structures.
To declare a builtin function that expects its first arg to be an instance
of class 'Person', its second argument as a regular arg, and its third by
reference, use:
ZEND_BEGIN_ARG_INFO(my_func_arg_info, 0)
ZEND_ARG_OBJ_INFO(0, someone, Person, 1)
ZEND_ARG_PASS_INFO(0)
ZEND_ARG_PASS_INFO(1)
ZEND_END_ARG_INFO();
and use my_func_arg_info as the arg_info parameter to the ZEND_FE() family
of macros.
The first arg to each ZEND_ARG_*() macro is whether or not to pass by ref.
The boolean arg to ZEND_BEGIN_ARG_INFO() tells the engine whether to treat
the arguments for which there's no explicit information as pass by reference
or not.
The boolean argument to ZEND_ARG_OBJ_INFO() (4th arg) is whether or not to allownull values.
2003-08-04 01:40:44 +08:00
zend_arg_info * cur_arg_info ;
2010-04-20 19:16:39 +08:00
znode var ;
1999-04-08 02:10:10 +08:00
2007-11-20 16:53:02 +08:00
if ( class_type - > op_type = = IS_CONST & &
2007-12-08 01:11:24 +08:00
Z_TYPE ( class_type - > u . constant ) = = IS_STRING & &
2007-11-20 16:53:02 +08:00
Z_STRLEN ( class_type - > u . constant ) = = 0 ) {
/* Usage of namespace as class name not in namespace */
zval_dtor ( & class_type - > u . constant ) ;
zend_error ( E_COMPILE_ERROR , " Cannot use 'namespace' as a class name " ) ;
return ;
}
2010-04-20 19:16:39 +08:00
if ( zend_is_auto_global ( Z_STRVAL ( varname - > u . constant ) , Z_STRLEN ( varname - > u . constant ) TSRMLS_CC ) ) {
zend_error ( E_COMPILE_ERROR , " Cannot re-assign auto-global variable %s " , Z_STRVAL ( varname - > u . constant ) ) ;
} else {
var . op_type = IS_CV ;
var . u . op . var = lookup_cv ( CG ( active_op_array ) , varname - > u . constant . value . str . val , varname - > u . constant . value . str . len TSRMLS_CC ) ;
varname - > u . constant . value . str . val = CG ( active_op_array ) - > vars [ var . u . op . var ] . name ;
var . EA = 0 ;
if ( Z_STRLEN ( varname - > u . constant ) = = sizeof ( " this " ) - 1 & &
! memcmp ( Z_STRVAL ( varname - > u . constant ) , " this " , sizeof ( " this " ) - 1 ) ) {
if ( CG ( active_op_array ) - > scope & &
( CG ( active_op_array ) - > fn_flags & ZEND_ACC_STATIC ) = = 0 ) {
zend_error ( E_COMPILE_ERROR , " Cannot re-assign $this " ) ;
}
CG ( active_op_array ) - > this_var = var . u . op . var ;
}
2007-05-04 14:18:53 +08:00
}
opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
ntroduce infrastructure for supplying information about arguments,
including:
- Whether or not to pass by ref (replaces the old arg_types, with arg_info)
- Argument name (for future use, maybe introspection)
- Class/Interface name (for type hints)
- If a class/interface name is available, whether to allow a null instance
Both user and builtin functions share the same data structures.
To declare a builtin function that expects its first arg to be an instance
of class 'Person', its second argument as a regular arg, and its third by
reference, use:
ZEND_BEGIN_ARG_INFO(my_func_arg_info, 0)
ZEND_ARG_OBJ_INFO(0, someone, Person, 1)
ZEND_ARG_PASS_INFO(0)
ZEND_ARG_PASS_INFO(1)
ZEND_END_ARG_INFO();
and use my_func_arg_info as the arg_info parameter to the ZEND_FE() family
of macros.
The first arg to each ZEND_ARG_*() macro is whether or not to pass by ref.
The boolean arg to ZEND_BEGIN_ARG_INFO() tells the engine whether to treat
the arguments for which there's no explicit information as pass by reference
or not.
The boolean argument to ZEND_ARG_OBJ_INFO() (4th arg) is whether or not to allownull values.
2003-08-04 01:40:44 +08:00
CG ( active_op_array ) - > num_args + + ;
1999-04-08 02:10:10 +08:00
opline - > opcode = op ;
2010-04-20 19:16:39 +08:00
SET_NODE ( opline - > result , & var ) ;
2010-04-20 18:57:45 +08:00
SET_NODE ( opline - > op1 , offset ) ;
2004-02-25 17:25:37 +08:00
if ( op = = ZEND_RECV_INIT ) {
2010-04-20 18:57:45 +08:00
SET_NODE ( opline - > op2 , initialization ) ;
2000-02-02 00:02:38 +08:00
} else {
2004-02-25 17:25:37 +08:00
CG ( active_op_array ) - > required_num_args = CG ( active_op_array ) - > num_args ;
2000-02-02 00:02:38 +08:00
SET_UNUSED ( opline - > op2 ) ;
1999-04-08 02:10:10 +08:00
}
ntroduce infrastructure for supplying information about arguments,
including:
- Whether or not to pass by ref (replaces the old arg_types, with arg_info)
- Argument name (for future use, maybe introspection)
- Class/Interface name (for type hints)
- If a class/interface name is available, whether to allow a null instance
Both user and builtin functions share the same data structures.
To declare a builtin function that expects its first arg to be an instance
of class 'Person', its second argument as a regular arg, and its third by
reference, use:
ZEND_BEGIN_ARG_INFO(my_func_arg_info, 0)
ZEND_ARG_OBJ_INFO(0, someone, Person, 1)
ZEND_ARG_PASS_INFO(0)
ZEND_ARG_PASS_INFO(1)
ZEND_END_ARG_INFO();
and use my_func_arg_info as the arg_info parameter to the ZEND_FE() family
of macros.
The first arg to each ZEND_ARG_*() macro is whether or not to pass by ref.
The boolean arg to ZEND_BEGIN_ARG_INFO() tells the engine whether to treat
the arguments for which there's no explicit information as pass by reference
or not.
The boolean argument to ZEND_ARG_OBJ_INFO() (4th arg) is whether or not to allownull values.
2003-08-04 01:40:44 +08:00
CG ( active_op_array ) - > arg_info = erealloc ( CG ( active_op_array ) - > arg_info , sizeof ( zend_arg_info ) * ( CG ( active_op_array ) - > num_args ) ) ;
cur_arg_info = & CG ( active_op_array ) - > arg_info [ CG ( active_op_array ) - > num_args - 1 ] ;
2010-04-20 19:05:54 +08:00
cur_arg_info - > name = CG ( new_interned_string ) ( estrndup ( varname - > u . constant . value . str . val , varname - > u . constant . value . str . len ) , varname - > u . constant . value . str . len + 1 , 1 TSRMLS_CC ) ;
ntroduce infrastructure for supplying information about arguments,
including:
- Whether or not to pass by ref (replaces the old arg_types, with arg_info)
- Argument name (for future use, maybe introspection)
- Class/Interface name (for type hints)
- If a class/interface name is available, whether to allow a null instance
Both user and builtin functions share the same data structures.
To declare a builtin function that expects its first arg to be an instance
of class 'Person', its second argument as a regular arg, and its third by
reference, use:
ZEND_BEGIN_ARG_INFO(my_func_arg_info, 0)
ZEND_ARG_OBJ_INFO(0, someone, Person, 1)
ZEND_ARG_PASS_INFO(0)
ZEND_ARG_PASS_INFO(1)
ZEND_END_ARG_INFO();
and use my_func_arg_info as the arg_info parameter to the ZEND_FE() family
of macros.
The first arg to each ZEND_ARG_*() macro is whether or not to pass by ref.
The boolean arg to ZEND_BEGIN_ARG_INFO() tells the engine whether to treat
the arguments for which there's no explicit information as pass by reference
or not.
The boolean argument to ZEND_ARG_OBJ_INFO() (4th arg) is whether or not to allownull values.
2003-08-04 01:40:44 +08:00
cur_arg_info - > name_len = varname - > u . constant . value . str . len ;
2005-05-26 21:46:17 +08:00
cur_arg_info - > array_type_hint = 0 ;
ntroduce infrastructure for supplying information about arguments,
including:
- Whether or not to pass by ref (replaces the old arg_types, with arg_info)
- Argument name (for future use, maybe introspection)
- Class/Interface name (for type hints)
- If a class/interface name is available, whether to allow a null instance
Both user and builtin functions share the same data structures.
To declare a builtin function that expects its first arg to be an instance
of class 'Person', its second argument as a regular arg, and its third by
reference, use:
ZEND_BEGIN_ARG_INFO(my_func_arg_info, 0)
ZEND_ARG_OBJ_INFO(0, someone, Person, 1)
ZEND_ARG_PASS_INFO(0)
ZEND_ARG_PASS_INFO(1)
ZEND_END_ARG_INFO();
and use my_func_arg_info as the arg_info parameter to the ZEND_FE() family
of macros.
The first arg to each ZEND_ARG_*() macro is whether or not to pass by ref.
The boolean arg to ZEND_BEGIN_ARG_INFO() tells the engine whether to treat
the arguments for which there's no explicit information as pass by reference
or not.
The boolean argument to ZEND_ARG_OBJ_INFO() (4th arg) is whether or not to allownull values.
2003-08-04 01:40:44 +08:00
cur_arg_info - > allow_null = 1 ;
cur_arg_info - > pass_by_reference = pass_by_reference ;
2007-12-04 20:38:42 +08:00
cur_arg_info - > class_name = NULL ;
cur_arg_info - > class_name_len = 0 ;
2003-05-08 06:19:43 +08:00
if ( class_type - > op_type ! = IS_UNUSED ) {
2005-11-15 23:59:49 +08:00
cur_arg_info - > allow_null = 0 ;
2005-05-26 21:46:17 +08:00
if ( class_type - > u . constant . type = = IS_STRING ) {
2007-12-03 22:15:43 +08:00
if ( ZEND_FETCH_CLASS_DEFAULT = = zend_get_class_fetch_type ( Z_STRVAL ( class_type - > u . constant ) , Z_STRLEN ( class_type - > u . constant ) ) ) {
zend_resolve_class_name ( class_type , & opline - > extended_value , 1 TSRMLS_CC ) ;
}
2010-04-20 19:05:54 +08:00
class_type - > u . constant . value . str . val = CG ( new_interned_string ) ( class_type - > u . constant . value . str . val , class_type - > u . constant . value . str . len + 1 , 1 TSRMLS_CC ) ;
2005-05-26 21:46:17 +08:00
cur_arg_info - > class_name = class_type - > u . constant . value . str . val ;
cur_arg_info - > class_name_len = class_type - > u . constant . value . str . len ;
2005-11-15 23:59:49 +08:00
if ( op = = ZEND_RECV_INIT ) {
if ( Z_TYPE ( initialization - > u . constant ) = = IS_NULL | | ( Z_TYPE ( initialization - > u . constant ) = = IS_CONSTANT & & ! strcasecmp ( Z_STRVAL ( initialization - > u . constant ) , " NULL " ) ) ) {
cur_arg_info - > allow_null = 1 ;
} else {
zend_error ( E_COMPILE_ERROR , " Default value for parameters with a class type hint can only be NULL " ) ;
}
}
2005-05-26 21:46:17 +08:00
} else {
cur_arg_info - > array_type_hint = 1 ;
cur_arg_info - > class_name = NULL ;
cur_arg_info - > class_name_len = 0 ;
2005-11-16 00:49:40 +08:00
if ( op = = ZEND_RECV_INIT ) {
if ( Z_TYPE ( initialization - > u . constant ) = = IS_NULL | | ( Z_TYPE ( initialization - > u . constant ) = = IS_CONSTANT & & ! strcasecmp ( Z_STRVAL ( initialization - > u . constant ) , " NULL " ) ) ) {
cur_arg_info - > allow_null = 1 ;
} else if ( Z_TYPE ( initialization - > u . constant ) ! = IS_ARRAY & & Z_TYPE ( initialization - > u . constant ) ! = IS_CONSTANT_ARRAY ) {
zend_error ( E_COMPILE_ERROR , " Default value for parameters with array type hint can only be an array or NULL " ) ;
}
}
2005-05-26 21:46:17 +08:00
}
2003-03-06 22:31:17 +08:00
}
1999-04-08 02:10:10 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
int zend_do_begin_function_call ( znode * function_name , zend_bool check_namespace TSRMLS_DC ) /* { { { */
1999-04-08 02:10:10 +08:00
{
zend_function * function ;
2003-09-17 19:06:11 +08:00
char * lcname ;
2008-11-12 03:45:29 +08:00
char * is_compound = memchr ( Z_STRVAL ( function_name - > u . constant ) , ' \\ ' , Z_STRLEN ( function_name - > u . constant ) ) ;
2007-09-29 03:52:53 +08:00
2008-11-12 03:45:29 +08:00
zend_resolve_non_class_name ( function_name , check_namespace TSRMLS_CC ) ;
2007-09-29 03:52:53 +08:00
2008-11-12 03:45:29 +08:00
if ( check_namespace & & CG ( current_namespace ) & & ! is_compound ) {
2008-11-04 23:58:55 +08:00
/* We assume we call function from the current namespace
if it is not prefixed . */
2007-09-29 03:52:53 +08:00
2008-11-04 23:58:55 +08:00
/* In run-time PHP will check for function with full name and
internal function with short name */
2008-11-12 03:45:29 +08:00
zend_do_begin_dynamic_function_call ( function_name , 1 TSRMLS_CC ) ;
return 1 ;
}
2003-09-17 19:06:11 +08:00
lcname = zend_str_tolower_dup ( function_name - > u . constant . value . str . val , function_name - > u . constant . value . str . len ) ;
2008-03-18 16:36:30 +08:00
if ( ( zend_hash_find ( CG ( function_table ) , lcname , function_name - > u . constant . value . str . len + 1 , ( void * * ) & function ) = = FAILURE ) | |
2008-11-12 03:45:29 +08:00
( ( CG ( compiler_options ) & ZEND_COMPILE_IGNORE_INTERNAL_FUNCTIONS ) & &
( function - > type = = ZEND_INTERNAL_FUNCTION ) ) ) {
zend_do_begin_dynamic_function_call ( function_name , 0 TSRMLS_CC ) ;
efree ( lcname ) ;
return 1 ; /* Dynamic */
}
2003-09-17 19:06:11 +08:00
efree ( function_name - > u . constant . value . str . val ) ;
function_name - > u . constant . value . str . val = lcname ;
1999-04-08 02:10:10 +08:00
2008-03-12 18:40:02 +08:00
zend_stack_push ( & CG ( function_call_stack ) , ( void * ) & function , sizeof ( zend_function * ) ) ;
2006-05-10 07:53:23 +08:00
zend_do_extended_fcall_begin ( TSRMLS_C ) ;
1999-07-29 20:24:58 +08:00
return 0 ;
1999-04-08 02:10:10 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_begin_method_call ( znode * left_bracket TSRMLS_DC ) /* { { { */
1999-04-08 02:10:10 +08:00
{
2001-09-07 22:46:12 +08:00
zend_op * last_op ;
int last_op_number ;
1999-04-08 02:10:10 +08:00
unsigned char * ptr = NULL ;
2000-07-04 03:17:57 +08:00
2008-05-07 20:04:39 +08:00
zend_do_end_variable_parse ( left_bracket , BP_VAR_R , 0 TSRMLS_CC ) ;
2001-08-08 23:07:11 +08:00
zend_do_begin_variable_parse ( TSRMLS_C ) ;
2006-05-10 07:53:23 +08:00
2001-09-07 22:46:12 +08:00
last_op_number = get_next_op_number ( CG ( active_op_array ) ) - 1 ;
last_op = & CG ( active_op_array ) - > opcodes [ last_op_number ] ;
2001-12-27 01:49:22 +08:00
2010-04-20 18:57:45 +08:00
if ( ( last_op - > op2_type = = IS_CONST ) & & ( Z_TYPE ( CONSTANT ( last_op - > op2 . constant ) ) = = IS_STRING ) & & ( Z_STRLEN ( CONSTANT ( last_op - > op2 . constant ) ) = = sizeof ( ZEND_CLONE_FUNC_NAME ) - 1 )
& & ! zend_binary_strcasecmp ( Z_STRVAL ( CONSTANT ( last_op - > op2 . constant ) ) , Z_STRLEN ( CONSTANT ( last_op - > op2 . constant ) ) , ZEND_CLONE_FUNC_NAME , sizeof ( ZEND_CLONE_FUNC_NAME ) - 1 ) ) {
2004-02-02 20:28:19 +08:00
zend_error ( E_COMPILE_ERROR , " Cannot call __clone() method on objects - use 'clone $obj' instead " ) ;
2001-12-27 01:49:22 +08:00
}
2002-01-04 16:07:39 +08:00
2003-12-17 22:45:59 +08:00
if ( last_op - > opcode = = ZEND_FETCH_OBJ_R ) {
2010-04-20 18:57:45 +08:00
if ( last_op - > op2_type = = IS_CONST ) {
2010-04-21 22:58:33 +08:00
last_op - > op2 . constant =
2010-04-21 23:08:10 +08:00
zend_add_func_name_literal ( CG ( active_op_array ) , & CONSTANT ( last_op - > op2 . constant ) TSRMLS_CC ) ;
2010-04-20 18:57:45 +08:00
}
2003-12-17 22:45:59 +08:00
last_op - > opcode = ZEND_INIT_METHOD_CALL ;
2007-12-27 21:52:05 +08:00
SET_UNUSED ( last_op - > result ) ;
2006-05-10 07:53:23 +08:00
Z_LVAL ( left_bracket - > u . constant ) = ZEND_INIT_FCALL_BY_NAME ;
2003-12-17 22:45:59 +08:00
} else {
2004-02-02 20:28:19 +08:00
zend_op * opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
2003-12-17 22:45:59 +08:00
opline - > opcode = ZEND_INIT_FCALL_BY_NAME ;
2010-04-27 20:09:13 +08:00
SET_UNUSED ( opline - > op1 ) ;
if ( left_bracket - > op_type = = IS_CONST ) {
opline - > op2_type = IS_CONST ;
opline - > op2 . constant = zend_add_func_name_literal ( CG ( active_op_array ) , & left_bracket - > u . constant TSRMLS_CC ) ;
2010-04-27 20:17:32 +08:00
} else {
2010-04-27 20:09:13 +08:00
SET_NODE ( opline - > op2 , left_bracket ) ;
2010-04-27 20:17:32 +08:00
}
2003-12-17 22:45:59 +08:00
}
2001-08-08 23:07:11 +08:00
zend_stack_push ( & CG ( function_call_stack ) , ( void * ) & ptr , sizeof ( zend_function * ) ) ;
2006-05-10 07:53:23 +08:00
zend_do_extended_fcall_begin ( TSRMLS_C ) ;
2001-08-08 23:07:11 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
2006-05-10 07:53:23 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_clone ( znode * result , const znode * expr TSRMLS_DC ) /* { { { */
2004-02-02 20:28:19 +08:00
{
zend_op * opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
opline - > opcode = ZEND_CLONE ;
2010-04-20 18:57:45 +08:00
SET_NODE ( opline - > op1 , expr ) ;
2004-02-02 20:28:19 +08:00
SET_UNUSED ( opline - > op2 ) ;
2010-04-20 18:57:45 +08:00
opline - > result_type = IS_VAR ;
opline - > result . var = get_temporary_variable ( CG ( active_op_array ) ) ;
GET_NODE ( result , opline - > result ) ;
2004-02-02 20:28:19 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
2004-02-02 20:28:19 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_begin_dynamic_function_call ( znode * function_name , int ns_call TSRMLS_DC ) /* { { { */
2001-08-08 23:07:11 +08:00
{
unsigned char * ptr = NULL ;
2010-04-27 20:09:13 +08:00
zend_op * opline ;
2001-08-08 23:07:11 +08:00
opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
2008-11-12 03:45:29 +08:00
if ( ns_call ) {
2007-09-29 03:52:53 +08:00
/* In run-time PHP will check for function with full name and
internal function with short name */
2007-11-22 17:02:55 +08:00
opline - > opcode = ZEND_INIT_NS_FCALL_BY_NAME ;
2010-04-27 20:09:13 +08:00
SET_UNUSED ( opline - > op1 ) ;
opline - > op2_type = IS_CONST ;
opline - > op2 . constant = zend_add_ns_func_name_literal ( CG ( active_op_array ) , & function_name - > u . constant TSRMLS_CC ) ;
2007-09-29 03:52:53 +08:00
} else {
2007-11-22 17:02:55 +08:00
opline - > opcode = ZEND_INIT_FCALL_BY_NAME ;
2010-04-27 20:09:13 +08:00
SET_UNUSED ( opline - > op1 ) ;
if ( function_name - > op_type = = IS_CONST ) {
opline - > op2_type = IS_CONST ;
opline - > op2 . constant = zend_add_func_name_literal ( CG ( active_op_array ) , & function_name - > u . constant TSRMLS_CC ) ;
2010-04-27 20:17:32 +08:00
} else {
2010-04-27 20:09:13 +08:00
SET_NODE ( opline - > op2 , function_name ) ;
2010-04-27 20:17:32 +08:00
}
2007-09-29 03:52:53 +08:00
}
2001-08-08 23:07:11 +08:00
1999-06-07 01:35:42 +08:00
zend_stack_push ( & CG ( function_call_stack ) , ( void * ) & ptr , sizeof ( zend_function * ) ) ;
2006-05-10 07:53:23 +08:00
zend_do_extended_fcall_begin ( TSRMLS_C ) ;
1999-04-08 02:10:10 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
void zend_resolve_non_class_name ( znode * element_name , zend_bool check_namespace TSRMLS_DC ) /* { { { */
2008-11-04 23:58:55 +08:00
{
znode tmp ;
int len ;
zval * * ns ;
2008-11-12 03:45:29 +08:00
char * lcname , * compound = memchr ( Z_STRVAL ( element_name - > u . constant ) , ' \\ ' , Z_STRLEN ( element_name - > u . constant ) ) ;
2008-11-04 23:58:55 +08:00
if ( Z_STRVAL ( element_name - > u . constant ) [ 0 ] = = ' \\ ' ) {
2008-11-12 03:45:29 +08:00
/* name starts with \ so it is known and unambiguos, nothing to do here but shorten it */
memmove ( Z_STRVAL ( element_name - > u . constant ) , Z_STRVAL ( element_name - > u . constant ) + 1 , Z_STRLEN ( element_name - > u . constant ) ) ;
- - Z_STRLEN ( element_name - > u . constant ) ;
return ;
}
if ( ! check_namespace ) {
return ;
2008-11-04 23:58:55 +08:00
}
2008-11-12 03:45:29 +08:00
if ( compound & & CG ( current_import ) ) {
len = compound - Z_STRVAL ( element_name - > u . constant ) ;
2008-11-04 23:58:55 +08:00
lcname = zend_str_tolower_dup ( Z_STRVAL ( element_name - > u . constant ) , len ) ;
/* Check if first part of compound name is an import name */
if ( zend_hash_find ( CG ( current_import ) , lcname , len + 1 , ( void * * ) & ns ) = = SUCCESS ) {
/* Substitute import name */
tmp . op_type = IS_CONST ;
tmp . u . constant = * * ns ;
zval_copy_ctor ( & tmp . u . constant ) ;
len + = 1 ;
Z_STRLEN ( element_name - > u . constant ) - = len ;
memmove ( Z_STRVAL ( element_name - > u . constant ) , Z_STRVAL ( element_name - > u . constant ) + len , Z_STRLEN ( element_name - > u . constant ) + 1 ) ;
zend_do_build_namespace_name ( & tmp , & tmp , element_name TSRMLS_CC ) ;
2008-11-12 03:45:29 +08:00
* element_name = tmp ;
2008-11-04 23:58:55 +08:00
efree ( lcname ) ;
return ;
}
efree ( lcname ) ;
}
2008-11-12 03:45:29 +08:00
if ( CG ( current_namespace ) ) {
2008-11-04 23:58:55 +08:00
tmp = * element_name ;
Z_STRLEN ( tmp . u . constant ) = sizeof ( " \\ " ) - 1 + Z_STRLEN ( element_name - > u . constant ) + Z_STRLEN_P ( CG ( current_namespace ) ) ;
Z_STRVAL ( tmp . u . constant ) = ( char * ) emalloc ( Z_STRLEN ( tmp . u . constant ) + 1 ) ;
memcpy ( Z_STRVAL ( tmp . u . constant ) , Z_STRVAL_P ( CG ( current_namespace ) ) , Z_STRLEN_P ( CG ( current_namespace ) ) ) ;
memcpy ( & ( Z_STRVAL ( tmp . u . constant ) [ Z_STRLEN_P ( CG ( current_namespace ) ) ] ) , " \\ " , sizeof ( " \\ " ) - 1 ) ;
memcpy ( & ( Z_STRVAL ( tmp . u . constant ) [ Z_STRLEN_P ( CG ( current_namespace ) ) + sizeof ( " \\ " ) - 1 ] ) , Z_STRVAL ( element_name - > u . constant ) , Z_STRLEN ( element_name - > u . constant ) + 1 ) ;
STR_FREE ( Z_STRVAL ( element_name - > u . constant ) ) ;
* element_name = tmp ;
}
}
2009-07-27 22:11:53 +08:00
/* }}} */
2008-11-04 23:58:55 +08:00
2009-07-27 22:11:53 +08:00
void zend_resolve_class_name ( znode * class_name , ulong * fetch_type , int check_ns_name TSRMLS_DC ) /* { { { */
2007-09-29 03:52:53 +08:00
{
char * compound ;
char * lcname ;
zval * * ns ;
znode tmp ;
int len ;
2008-11-04 23:58:55 +08:00
compound = memchr ( Z_STRVAL ( class_name - > u . constant ) , ' \\ ' , Z_STRLEN ( class_name - > u . constant ) ) ;
2007-09-29 03:52:53 +08:00
if ( compound ) {
2008-11-04 23:58:55 +08:00
/* This is a compound class name that contains namespace prefix */
2008-11-12 03:45:29 +08:00
if ( Z_STRVAL ( class_name - > u . constant ) [ 0 ] = = ' \\ ' ) {
2008-11-04 23:58:55 +08:00
/* The STRING name has "\" prefix */
Z_STRLEN ( class_name - > u . constant ) - = 1 ;
memmove ( Z_STRVAL ( class_name - > u . constant ) , Z_STRVAL ( class_name - > u . constant ) + 1 , Z_STRLEN ( class_name - > u . constant ) + 1 ) ;
2007-09-29 03:52:53 +08:00
Z_STRVAL ( class_name - > u . constant ) = erealloc (
Z_STRVAL ( class_name - > u . constant ) ,
Z_STRLEN ( class_name - > u . constant ) + 1 ) ;
2007-12-03 22:15:43 +08:00
if ( ZEND_FETCH_CLASS_DEFAULT ! = zend_get_class_fetch_type ( Z_STRVAL ( class_name - > u . constant ) , Z_STRLEN ( class_name - > u . constant ) ) ) {
2008-11-04 23:58:55 +08:00
zend_error ( E_COMPILE_ERROR , " ' \\ %s' is an invalid class name " , Z_STRVAL ( class_name - > u . constant ) ) ;
}
} else {
if ( CG ( current_import ) ) {
len = compound - Z_STRVAL ( class_name - > u . constant ) ;
lcname = zend_str_tolower_dup ( Z_STRVAL ( class_name - > u . constant ) , len ) ;
/* Check if first part of compound name is an import name */
if ( zend_hash_find ( CG ( current_import ) , lcname , len + 1 , ( void * * ) & ns ) = = SUCCESS ) {
/* Substitute import name */
tmp . op_type = IS_CONST ;
tmp . u . constant = * * ns ;
zval_copy_ctor ( & tmp . u . constant ) ;
len + = 1 ;
Z_STRLEN ( class_name - > u . constant ) - = len ;
memmove ( Z_STRVAL ( class_name - > u . constant ) , Z_STRVAL ( class_name - > u . constant ) + len , Z_STRLEN ( class_name - > u . constant ) + 1 ) ;
zend_do_build_namespace_name ( & tmp , & tmp , class_name TSRMLS_CC ) ;
* class_name = tmp ;
2008-11-12 03:45:29 +08:00
efree ( lcname ) ;
return ;
2008-11-04 23:58:55 +08:00
}
efree ( lcname ) ;
2007-12-03 22:15:43 +08:00
}
2008-11-12 03:45:29 +08:00
/* Here name is not prefixed with \ and not imported */
2008-11-04 23:58:55 +08:00
if ( CG ( current_namespace ) ) {
2007-09-29 03:52:53 +08:00
tmp . op_type = IS_CONST ;
2008-11-04 23:58:55 +08:00
tmp . u . constant = * CG ( current_namespace ) ;
2007-09-29 03:52:53 +08:00
zval_copy_ctor ( & tmp . u . constant ) ;
zend_do_build_namespace_name ( & tmp , & tmp , class_name TSRMLS_CC ) ;
* class_name = tmp ;
}
}
} else if ( CG ( current_import ) | | CG ( current_namespace ) ) {
2008-11-04 23:58:55 +08:00
/* this is a plain name (without \) */
2007-09-29 03:52:53 +08:00
lcname = zend_str_tolower_dup ( Z_STRVAL ( class_name - > u . constant ) , Z_STRLEN ( class_name - > u . constant ) ) ;
if ( CG ( current_import ) & &
zend_hash_find ( CG ( current_import ) , lcname , Z_STRLEN ( class_name - > u . constant ) + 1 , ( void * * ) & ns ) = = SUCCESS ) {
/* The given name is an import name. Substitute it. */
zval_dtor ( & class_name - > u . constant ) ;
class_name - > u . constant = * * ns ;
zval_copy_ctor ( & class_name - > u . constant ) ;
} else if ( CG ( current_namespace ) ) {
2008-11-12 03:45:29 +08:00
/* plain name, no import - prepend current namespace to it */
2007-09-29 03:52:53 +08:00
tmp . op_type = IS_CONST ;
tmp . u . constant = * CG ( current_namespace ) ;
zval_copy_ctor ( & tmp . u . constant ) ;
zend_do_build_namespace_name ( & tmp , & tmp , class_name TSRMLS_CC ) ;
* class_name = tmp ;
}
efree ( lcname ) ;
}
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_fetch_class ( znode * result , znode * class_name TSRMLS_DC ) /* { { { */
2001-10-30 01:19:02 +08:00
{
2002-01-14 04:21:55 +08:00
long fetch_class_op_number ;
zend_op * opline ;
2007-11-20 16:53:02 +08:00
if ( class_name - > op_type = = IS_CONST & &
2007-12-08 01:11:24 +08:00
Z_TYPE ( class_name - > u . constant ) = = IS_STRING & &
2007-11-20 16:53:02 +08:00
Z_STRLEN ( class_name - > u . constant ) = = 0 ) {
/* Usage of namespace as class name not in namespace */
zval_dtor ( & class_name - > u . constant ) ;
zend_error ( E_COMPILE_ERROR , " Cannot use 'namespace' as a class name " ) ;
return ;
}
2002-01-14 04:21:55 +08:00
fetch_class_op_number = get_next_op_number ( CG ( active_op_array ) ) ;
opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
2001-10-30 01:19:02 +08:00
opline - > opcode = ZEND_FETCH_CLASS ;
2003-06-02 20:13:11 +08:00
SET_UNUSED ( opline - > op1 ) ;
opline - > extended_value = ZEND_FETCH_CLASS_GLOBAL ;
2003-02-24 21:35:47 +08:00
CG ( catch_begin ) = fetch_class_op_number ;
2003-03-10 06:02:37 +08:00
if ( class_name - > op_type = = IS_CONST ) {
2003-08-03 16:21:08 +08:00
int fetch_type ;
fetch_type = zend_get_class_fetch_type ( class_name - > u . constant . value . str . val , class_name - > u . constant . value . str . len ) ;
switch ( fetch_type ) {
case ZEND_FETCH_CLASS_SELF :
case ZEND_FETCH_CLASS_PARENT :
2007-09-29 15:28:34 +08:00
case ZEND_FETCH_CLASS_STATIC :
2003-08-03 16:21:08 +08:00
SET_UNUSED ( opline - > op2 ) ;
opline - > extended_value = fetch_type ;
zval_dtor ( & class_name - > u . constant ) ;
break ;
default :
2007-09-29 03:52:53 +08:00
zend_resolve_class_name ( class_name , & opline - > extended_value , 0 TSRMLS_CC ) ;
2010-04-21 22:58:33 +08:00
opline - > op2_type = IS_CONST ;
opline - > op2 . constant =
2010-04-21 23:08:10 +08:00
zend_add_class_name_literal ( CG ( active_op_array ) , & class_name - > u . constant TSRMLS_CC ) ;
2003-08-03 16:21:08 +08:00
break ;
2003-03-05 21:25:33 +08:00
}
} else {
2010-04-20 18:57:45 +08:00
SET_NODE ( opline - > op2 , class_name ) ;
2001-12-13 01:38:37 +08:00
}
2010-04-20 18:57:45 +08:00
opline - > result . var = get_temporary_variable ( CG ( active_op_array ) ) ;
opline - > result_type = IS_VAR ; /* FIXME: Hack so that INIT_FCALL_BY_NAME still knows this is a class */
GET_NODE ( result , opline - > result ) ;
result - > EA = opline - > extended_value ;
2001-10-30 01:19:02 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
2001-10-30 01:19:02 +08:00
2008-03-28 22:35:01 +08:00
void zend_do_label ( znode * label TSRMLS_DC ) /* { { { */
{
zend_op_array * oparray = CG ( active_op_array ) ;
zend_label dest ;
if ( ! CG ( labels ) ) {
ALLOC_HASHTABLE ( CG ( labels ) ) ;
zend_hash_init ( CG ( labels ) , 4 , NULL , NULL , 0 ) ;
}
dest . brk_cont = oparray - > current_brk_cont ;
dest . opline_num = get_next_op_number ( oparray ) ;
if ( zend_hash_add ( CG ( labels ) , Z_STRVAL ( label - > u . constant ) , Z_STRLEN ( label - > u . constant ) + 1 , ( void * * ) & dest , sizeof ( zend_label ) , NULL ) = = FAILURE ) {
zend_error ( E_COMPILE_ERROR , " Label '%s' already defined " , Z_STRVAL ( label - > u . constant ) ) ;
}
/* Done with label now */
zval_dtor ( & label - > u . constant ) ;
}
/* }}} */
void zend_resolve_goto_label ( zend_op_array * op_array , zend_op * opline , int pass2 TSRMLS_DC ) /* { { { */
{
zend_label * dest ;
long current , distance ;
2010-04-20 18:57:45 +08:00
zval * label ;
2008-03-28 22:35:01 +08:00
2010-04-20 18:57:45 +08:00
if ( pass2 ) {
label = opline - > op2 . zv ;
} else {
label = & CONSTANT_EX ( op_array , opline - > op2 . constant ) ;
}
2008-03-28 22:35:01 +08:00
if ( CG ( labels ) = = NULL | |
2010-04-20 18:57:45 +08:00
zend_hash_find ( CG ( labels ) , Z_STRVAL_P ( label ) , Z_STRLEN_P ( label ) + 1 , ( void * * ) & dest ) = = FAILURE ) {
2008-03-28 22:35:01 +08:00
if ( pass2 ) {
CG ( in_compilation ) = 1 ;
CG ( active_op_array ) = op_array ;
CG ( zend_lineno ) = opline - > lineno ;
2010-04-20 18:57:45 +08:00
zend_error ( E_COMPILE_ERROR , " 'goto' to undefined label '%s' " , Z_STRVAL_P ( label ) ) ;
2008-03-28 22:35:01 +08:00
} else {
/* Label is not defined. Delay to pass 2. */
INC_BPC ( op_array ) ;
return ;
}
}
2010-04-20 18:57:45 +08:00
opline - > op1 . opline_num = dest - > opline_num ;
zval_dtor ( label ) ;
Z_TYPE_P ( label ) = IS_NULL ;
2008-03-28 22:35:01 +08:00
/* Check that we are not moving into loop or switch */
current = opline - > extended_value ;
for ( distance = 0 ; current ! = dest - > brk_cont ; distance + + ) {
if ( current = = - 1 ) {
if ( pass2 ) {
CG ( in_compilation ) = 1 ;
CG ( active_op_array ) = op_array ;
CG ( zend_lineno ) = opline - > lineno ;
}
zend_error ( E_COMPILE_ERROR , " 'goto' into loop or switch statement is disallowed " ) ;
}
current = op_array - > brk_cont_array [ current ] . parent ;
}
if ( distance = = 0 ) {
/* Nothing to break out of, optimize to ZEND_JMP */
opline - > opcode = ZEND_JMP ;
opline - > extended_value = 0 ;
SET_UNUSED ( opline - > op2 ) ;
} else {
/* Set real break distance */
2010-04-20 18:57:45 +08:00
ZVAL_LONG ( label , distance ) ;
2008-03-28 22:35:01 +08:00
}
if ( pass2 ) {
DEC_BPC ( op_array ) ;
}
}
/* }}} */
2008-08-13 01:20:25 +08:00
void zend_do_goto ( const znode * label TSRMLS_DC ) /* { { { */
2008-03-28 22:35:01 +08:00
{
zend_op * opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
opline - > opcode = ZEND_GOTO ;
opline - > extended_value = CG ( active_op_array ) - > current_brk_cont ;
SET_UNUSED ( opline - > op1 ) ;
2010-04-20 18:57:45 +08:00
SET_NODE ( opline - > op2 , label ) ;
2008-03-28 22:35:01 +08:00
zend_resolve_goto_label ( CG ( active_op_array ) , opline , 0 TSRMLS_CC ) ;
}
/* }}} */
void zend_release_labels ( TSRMLS_D ) /* { { { */
{
if ( CG ( labels ) ) {
zend_hash_destroy ( CG ( labels ) ) ;
FREE_HASHTABLE ( CG ( labels ) ) ;
}
if ( ! zend_stack_is_empty ( & CG ( labels_stack ) ) ) {
HashTable * * pht ;
zend_stack_top ( & CG ( labels_stack ) , ( void * * ) & pht ) ;
CG ( labels ) = * pht ;
zend_stack_del_top ( & CG ( labels_stack ) ) ;
} else {
CG ( labels ) = NULL ;
}
}
/* }}} */
2001-12-01 00:29:47 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_build_full_name ( znode * result , znode * prefix , znode * name , int is_class_member TSRMLS_DC ) /* { { { */
2001-12-01 00:29:47 +08:00
{
zend_uint length ;
if ( ! result ) {
2008-03-26 02:08:37 +08:00
result = prefix ;
2001-12-01 00:29:47 +08:00
} else {
2008-03-26 02:08:37 +08:00
* result = * prefix ;
2001-12-01 00:29:47 +08:00
}
2002-07-16 02:09:56 +08:00
2008-11-04 23:58:55 +08:00
if ( is_class_member ) {
length = sizeof ( " :: " ) - 1 + result - > u . constant . value . str . len + name - > u . constant . value . str . len ;
result - > u . constant . value . str . val = erealloc ( result - > u . constant . value . str . val , length + 1 ) ;
memcpy ( & result - > u . constant . value . str . val [ result - > u . constant . value . str . len ] , " :: " , sizeof ( " :: " ) - 1 ) ;
memcpy ( & result - > u . constant . value . str . val [ result - > u . constant . value . str . len + sizeof ( " :: " ) - 1 ] , name - > u . constant . value . str . val , name - > u . constant . value . str . len + 1 ) ;
STR_FREE ( name - > u . constant . value . str . val ) ;
result - > u . constant . value . str . len = length ;
} else {
length = sizeof ( " \\ " ) - 1 + result - > u . constant . value . str . len + name - > u . constant . value . str . len ;
result - > u . constant . value . str . val = erealloc ( result - > u . constant . value . str . val , length + 1 ) ;
memcpy ( & result - > u . constant . value . str . val [ result - > u . constant . value . str . len ] , " \\ " , sizeof ( " \\ " ) - 1 ) ;
memcpy ( & result - > u . constant . value . str . val [ result - > u . constant . value . str . len + sizeof ( " \\ " ) - 1 ] , name - > u . constant . value . str . val , name - > u . constant . value . str . len + 1 ) ;
STR_FREE ( name - > u . constant . value . str . val ) ;
result - > u . constant . value . str . len = length ;
}
2001-12-01 00:29:47 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
2001-12-01 00:29:47 +08:00
2009-07-27 22:11:53 +08:00
int zend_do_begin_class_member_function_call ( znode * class_name , znode * method_name TSRMLS_DC ) /* { { { */
1999-04-08 02:10:10 +08:00
{
2007-09-29 03:52:53 +08:00
znode class_node ;
1999-04-08 02:10:10 +08:00
unsigned char * ptr = NULL ;
2007-09-29 03:52:53 +08:00
zend_op * opline ;
ulong fetch_type = 0 ;
2004-09-16 08:40:38 +08:00
2007-09-29 03:52:53 +08:00
if ( method_name - > op_type = = IS_CONST ) {
char * lcname = zend_str_tolower_dup ( Z_STRVAL ( method_name - > u . constant ) , Z_STRLEN ( method_name - > u . constant ) ) ;
if ( ( sizeof ( ZEND_CONSTRUCTOR_FUNC_NAME ) - 1 ) = = Z_STRLEN ( method_name - > u . constant ) & &
2007-03-12 21:10:40 +08:00
memcmp ( lcname , ZEND_CONSTRUCTOR_FUNC_NAME , sizeof ( ZEND_CONSTRUCTOR_FUNC_NAME ) - 1 ) = = 0 ) {
2007-09-29 03:52:53 +08:00
zval_dtor ( & method_name - > u . constant ) ;
2010-04-20 18:57:45 +08:00
method_name - > op_type = IS_UNUSED ;
2003-10-09 21:44:44 +08:00
}
2008-06-04 02:11:12 +08:00
efree ( lcname ) ;
2003-07-03 20:03:11 +08:00
}
2002-01-04 16:07:39 +08:00
2007-09-29 03:52:53 +08:00
if ( class_name - > op_type = = IS_CONST & &
ZEND_FETCH_CLASS_DEFAULT = = zend_get_class_fetch_type ( Z_STRVAL ( class_name - > u . constant ) , Z_STRLEN ( class_name - > u . constant ) ) ) {
fetch_type = ZEND_FETCH_CLASS_GLOBAL ;
zend_resolve_class_name ( class_name , & fetch_type , 1 TSRMLS_CC ) ;
class_node = * class_name ;
2010-04-20 18:57:45 +08:00
opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
2007-09-29 03:52:53 +08:00
} else {
zend_do_fetch_class ( & class_node , class_name TSRMLS_CC ) ;
2010-04-20 18:57:45 +08:00
opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
opline - > extended_value = class_node . EA ;
2007-09-29 03:52:53 +08:00
}
opline - > opcode = ZEND_INIT_STATIC_METHOD_CALL ;
2010-04-21 22:58:33 +08:00
if ( class_node . op_type = = IS_CONST ) {
opline - > op1_type = IS_CONST ;
opline - > op1 . constant =
2010-04-21 23:08:10 +08:00
zend_add_class_name_literal ( CG ( active_op_array ) , & class_node . u . constant TSRMLS_CC ) ;
2010-04-21 22:58:33 +08:00
} else {
SET_NODE ( opline - > op1 , & class_node ) ;
2010-04-20 18:57:45 +08:00
}
2010-04-21 22:58:33 +08:00
if ( method_name - > op_type = = IS_CONST ) {
opline - > op2_type = IS_CONST ;
opline - > op2 . constant =
2010-04-21 23:08:10 +08:00
zend_add_func_name_literal ( CG ( active_op_array ) , & method_name - > u . constant TSRMLS_CC ) ;
2010-04-21 22:58:33 +08:00
} else {
SET_NODE ( opline - > op2 , method_name ) ;
2010-04-20 18:57:45 +08:00
}
2007-09-29 03:52:53 +08:00
1999-06-07 01:35:42 +08:00
zend_stack_push ( & CG ( function_call_stack ) , ( void * ) & ptr , sizeof ( zend_function * ) ) ;
2004-09-16 08:40:38 +08:00
zend_do_extended_fcall_begin ( TSRMLS_C ) ;
2007-11-20 16:53:02 +08:00
return 1 ; /* Dynamic */
1999-04-08 02:10:10 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_end_function_call ( znode * function_name , znode * result , const znode * argument_list , int is_method , int is_dynamic_fcall TSRMLS_DC ) /* { { { */
1999-04-08 02:10:10 +08:00
{
2001-12-27 01:49:22 +08:00
zend_op * opline ;
2006-05-10 07:53:23 +08:00
2003-07-01 04:22:35 +08:00
if ( is_method & & function_name & & function_name - > op_type = = IS_UNUSED ) {
2003-05-20 01:12:56 +08:00
/* clone */
2006-05-10 07:53:23 +08:00
if ( Z_LVAL ( argument_list - > u . constant ) ! = 0 ) {
2003-05-20 01:12:56 +08:00
zend_error ( E_WARNING , " Clone method does not require arguments " ) ;
}
2006-05-10 07:53:23 +08:00
opline = & CG ( active_op_array ) - > opcodes [ Z_LVAL ( function_name - > u . constant ) ] ;
1999-06-09 02:33:31 +08:00
} else {
2003-05-20 01:12:56 +08:00
opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
if ( ! is_method & & ! is_dynamic_fcall & & function_name - > op_type = = IS_CONST ) {
opline - > opcode = ZEND_DO_FCALL ;
2010-04-20 18:57:45 +08:00
SET_NODE ( opline - > op1 , function_name ) ;
2010-04-21 22:58:33 +08:00
CALCULATE_LITERAL_HASH ( opline - > op1 . constant ) ;
2003-05-20 01:12:56 +08:00
} else {
opline - > opcode = ZEND_DO_FCALL_BY_NAME ;
SET_UNUSED ( opline - > op1 ) ;
}
1999-06-09 02:33:31 +08:00
}
2006-05-10 07:53:23 +08:00
2010-04-20 18:57:45 +08:00
opline - > result . var = get_temporary_variable ( CG ( active_op_array ) ) ;
opline - > result_type = IS_VAR ;
GET_NODE ( result , opline - > result ) ;
1999-04-08 02:10:10 +08:00
SET_UNUSED ( opline - > op2 ) ;
2001-08-30 23:26:30 +08:00
1999-04-08 02:10:10 +08:00
zend_stack_del_top ( & CG ( function_call_stack ) ) ;
2006-05-10 07:53:23 +08:00
opline - > extended_value = Z_LVAL ( argument_list - > u . constant ) ;
1999-04-08 02:10:10 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_pass_param ( znode * param , zend_uchar op , int offset TSRMLS_DC ) /* { { { */
1999-04-08 02:10:10 +08:00
{
1999-09-22 04:00:01 +08:00
zend_op * opline ;
1999-04-08 02:10:10 +08:00
int original_op = op ;
1999-06-07 01:35:42 +08:00
zend_function * * function_ptr_ptr , * function_ptr ;
2000-11-28 02:46:23 +08:00
int send_by_reference ;
2005-06-22 16:33:00 +08:00
int send_function = 0 ;
2006-05-10 07:53:23 +08:00
1999-06-07 01:35:42 +08:00
zend_stack_top ( & CG ( function_call_stack ) , ( void * * ) & function_ptr_ptr ) ;
function_ptr = * function_ptr_ptr ;
1999-12-16 04:15:32 +08:00
2010-04-26 08:13:34 +08:00
if ( original_op = = ZEND_SEND_REF ) {
2008-06-16 02:27:37 +08:00
if ( function_ptr & &
function_ptr - > common . function_name & &
function_ptr - > common . type = = ZEND_USER_FUNCTION & &
! ARG_SHOULD_BE_SENT_BY_REF ( function_ptr , ( zend_uint ) offset ) ) {
2010-04-26 08:13:34 +08:00
zend_error ( E_COMPILE_ERROR ,
" Call-time pass-by-reference has been removed; "
" If you would like to pass argument by reference, modify the declaration of %s(). " ,
function_ptr - > common . function_name ) ;
2008-06-16 02:27:37 +08:00
} else {
2010-04-26 08:13:34 +08:00
zend_error ( E_COMPILE_ERROR , " Call-time pass-by-reference has been removed " ) ;
2008-06-16 02:27:37 +08:00
}
2010-04-26 08:13:34 +08:00
return ;
}
1999-06-07 01:35:42 +08:00
if ( function_ptr ) {
2005-08-10 20:02:52 +08:00
if ( ARG_MAY_BE_SENT_BY_REF ( function_ptr , ( zend_uint ) offset ) ) {
2009-01-20 19:22:45 +08:00
if ( param - > op_type & ( IS_VAR | IS_CV ) ) {
send_by_reference = 1 ;
if ( op = = ZEND_SEND_VAR & & zend_is_function_or_method_call ( param ) ) {
/* Method call */
op = ZEND_SEND_VAR_NO_REF ;
2009-01-20 21:21:52 +08:00
send_function = ZEND_ARG_SEND_FUNCTION | ZEND_ARG_SEND_SILENT ;
2009-01-20 19:22:45 +08:00
}
} else {
op = ZEND_SEND_VAL ;
send_by_reference = 0 ;
}
2005-08-10 20:02:52 +08:00
} else {
2006-05-10 07:53:23 +08:00
send_by_reference = ARG_SHOULD_BE_SENT_BY_REF ( function_ptr , ( zend_uint ) offset ) ? ZEND_ARG_SEND_BY_REF : 0 ;
2005-08-10 20:02:52 +08:00
}
1999-06-07 01:35:42 +08:00
} else {
ntroduce infrastructure for supplying information about arguments,
including:
- Whether or not to pass by ref (replaces the old arg_types, with arg_info)
- Argument name (for future use, maybe introspection)
- Class/Interface name (for type hints)
- If a class/interface name is available, whether to allow a null instance
Both user and builtin functions share the same data structures.
To declare a builtin function that expects its first arg to be an instance
of class 'Person', its second argument as a regular arg, and its third by
reference, use:
ZEND_BEGIN_ARG_INFO(my_func_arg_info, 0)
ZEND_ARG_OBJ_INFO(0, someone, Person, 1)
ZEND_ARG_PASS_INFO(0)
ZEND_ARG_PASS_INFO(1)
ZEND_END_ARG_INFO();
and use my_func_arg_info as the arg_info parameter to the ZEND_FE() family
of macros.
The first arg to each ZEND_ARG_*() macro is whether or not to pass by ref.
The boolean arg to ZEND_BEGIN_ARG_INFO() tells the engine whether to treat
the arguments for which there's no explicit information as pass by reference
or not.
The boolean argument to ZEND_ARG_OBJ_INFO() (4th arg) is whether or not to allownull values.
2003-08-04 01:40:44 +08:00
send_by_reference = 0 ;
1999-06-07 01:35:42 +08:00
}
1999-04-08 02:10:10 +08:00
2001-12-26 22:46:18 +08:00
if ( op = = ZEND_SEND_VAR & & zend_is_function_or_method_call ( param ) ) {
2001-08-08 23:07:11 +08:00
/* Method call */
op = ZEND_SEND_VAR_NO_REF ;
2005-06-22 16:33:00 +08:00
send_function = ZEND_ARG_SEND_FUNCTION ;
2004-10-05 03:54:35 +08:00
} else if ( op = = ZEND_SEND_VAL & & ( param - > op_type & ( IS_VAR | IS_CV ) ) ) {
2001-08-08 23:07:11 +08:00
op = ZEND_SEND_VAR_NO_REF ;
1999-04-08 02:10:10 +08:00
}
2000-11-28 02:46:23 +08:00
2004-10-09 06:17:32 +08:00
if ( op ! = ZEND_SEND_VAR_NO_REF & & send_by_reference = = ZEND_ARG_SEND_BY_REF ) {
1999-04-08 02:10:10 +08:00
/* change to passing by reference */
switch ( param - > op_type ) {
case IS_VAR :
2004-10-05 03:54:35 +08:00
case IS_CV :
1999-04-08 02:10:10 +08:00
op = ZEND_SEND_REF ;
break ;
default :
2004-10-09 06:17:32 +08:00
zend_error ( E_COMPILE_ERROR , " Only variables can be passed by reference " ) ;
1999-04-08 02:10:10 +08:00
break ;
}
}
2000-11-28 02:46:23 +08:00
if ( original_op = = ZEND_SEND_VAR ) {
2002-11-30 19:20:25 +08:00
switch ( op ) {
2000-10-30 03:16:29 +08:00
case ZEND_SEND_VAR_NO_REF :
2008-05-07 20:04:39 +08:00
zend_do_end_variable_parse ( param , BP_VAR_R , 0 TSRMLS_CC ) ;
2000-11-28 02:46:23 +08:00
break ;
1999-04-08 02:10:10 +08:00
case ZEND_SEND_VAR :
1999-06-07 01:35:42 +08:00
if ( function_ptr ) {
2008-05-07 20:04:39 +08:00
zend_do_end_variable_parse ( param , BP_VAR_R , 0 TSRMLS_CC ) ;
1999-06-07 01:35:42 +08:00
} else {
2008-05-07 20:04:39 +08:00
zend_do_end_variable_parse ( param , BP_VAR_FUNC_ARG , offset TSRMLS_CC ) ;
1999-06-07 01:35:42 +08:00
}
1999-04-08 02:10:10 +08:00
break ;
case ZEND_SEND_REF :
2008-05-07 20:04:39 +08:00
zend_do_end_variable_parse ( param , BP_VAR_W , 0 TSRMLS_CC ) ;
1999-04-08 02:10:10 +08:00
break ;
}
}
2001-07-28 18:51:54 +08:00
opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
2000-11-28 02:46:23 +08:00
if ( op = = ZEND_SEND_VAR_NO_REF ) {
if ( function_ptr ) {
2005-06-22 16:33:00 +08:00
opline - > extended_value = ZEND_ARG_COMPILE_TIME_BOUND | send_by_reference | send_function ;
2000-11-28 02:46:23 +08:00
} else {
2005-06-22 16:33:00 +08:00
opline - > extended_value = send_function ;
2000-11-28 02:46:23 +08:00
}
1999-09-22 04:00:01 +08:00
} else {
2000-11-28 02:46:23 +08:00
if ( function_ptr ) {
opline - > extended_value = ZEND_DO_FCALL ;
} else {
opline - > extended_value = ZEND_DO_FCALL_BY_NAME ;
}
1999-09-22 04:00:01 +08:00
}
1999-04-08 02:10:10 +08:00
opline - > opcode = op ;
2010-04-20 18:57:45 +08:00
SET_NODE ( opline - > op1 , param ) ;
opline - > op2 . opline_num = offset ;
1999-04-08 02:10:10 +08:00
SET_UNUSED ( opline - > op2 ) ;
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
static int generate_free_switch_expr ( const zend_switch_entry * switch_entry TSRMLS_DC ) /* { { { */
1999-04-08 02:10:10 +08:00
{
1999-09-09 22:15:17 +08:00
zend_op * opline ;
2006-05-10 07:53:23 +08:00
2004-03-02 19:03:48 +08:00
if ( switch_entry - > cond . op_type ! = IS_VAR & & switch_entry - > cond . op_type ! = IS_TMP_VAR ) {
2005-04-25 17:56:42 +08:00
return ( switch_entry - > cond . op_type = = IS_UNUSED ) ;
1999-09-09 22:15:17 +08:00
}
2006-05-10 07:53:23 +08:00
2001-07-28 18:51:54 +08:00
opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
1999-04-08 02:10:10 +08:00
2008-05-05 19:03:35 +08:00
opline - > opcode = ( switch_entry - > cond . op_type = = IS_TMP_VAR ) ? ZEND_FREE : ZEND_SWITCH_FREE ;
2010-04-20 18:57:45 +08:00
SET_NODE ( opline - > op1 , & switch_entry - > cond ) ;
1999-07-24 19:43:21 +08:00
SET_UNUSED ( opline - > op2 ) ;
2000-01-25 03:04:07 +08:00
opline - > extended_value = 0 ;
1999-09-09 22:15:17 +08:00
return 0 ;
1999-07-24 19:43:21 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-07-24 19:43:21 +08:00
2009-07-27 22:11:53 +08:00
static int generate_free_foreach_copy ( const zend_op * foreach_copy TSRMLS_DC ) /* { { { */
1999-09-09 22:15:17 +08:00
{
zend_op * opline ;
2004-03-05 21:04:21 +08:00
/* If we reach the seperator then stop applying the stack */
2010-04-20 18:57:45 +08:00
if ( foreach_copy - > result_type = = IS_UNUSED & & foreach_copy - > op1_type = = IS_UNUSED ) {
2004-03-05 21:04:21 +08:00
return 1 ;
2006-05-10 07:53:23 +08:00
}
1999-09-09 22:15:17 +08:00
2001-07-28 18:51:54 +08:00
opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
1999-09-09 22:15:17 +08:00
2010-04-20 18:57:45 +08:00
opline - > opcode = ( foreach_copy - > result_type = = IS_TMP_VAR ) ? ZEND_FREE : ZEND_SWITCH_FREE ;
COPY_NODE ( opline - > op1 , foreach_copy - > result ) ;
1999-09-09 22:15:17 +08:00
SET_UNUSED ( opline - > op2 ) ;
2000-01-25 03:04:07 +08:00
opline - > extended_value = 1 ;
2004-03-05 21:04:21 +08:00
2010-04-20 18:57:45 +08:00
if ( foreach_copy - > op1_type ! = IS_UNUSED ) {
2004-03-05 21:04:21 +08:00
opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
2010-04-20 18:57:45 +08:00
opline - > opcode = ( foreach_copy - > op1_type = = IS_TMP_VAR ) ? ZEND_FREE : ZEND_SWITCH_FREE ;
COPY_NODE ( opline - > op1 , foreach_copy - > op1 ) ;
2004-03-05 21:04:21 +08:00
SET_UNUSED ( opline - > op2 ) ;
opline - > extended_value = 0 ;
}
2006-05-10 07:53:23 +08:00
1999-09-09 22:15:17 +08:00
return 0 ;
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-07-24 19:43:21 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_return ( znode * expr , int do_end_vparse TSRMLS_DC ) /* { { { */
1999-07-24 19:43:21 +08:00
{
zend_op * opline ;
2008-05-05 19:03:35 +08:00
int start_op_number , end_op_number ;
2006-05-10 07:53:23 +08:00
2001-07-19 23:53:38 +08:00
if ( do_end_vparse ) {
2002-03-19 04:27:03 +08:00
if ( CG ( active_op_array ) - > return_reference & & ! zend_is_function_or_method_call ( expr ) ) {
2008-05-07 20:04:39 +08:00
zend_do_end_variable_parse ( expr , BP_VAR_W , 0 TSRMLS_CC ) ;
2001-07-19 23:53:38 +08:00
} else {
2008-05-07 20:04:39 +08:00
zend_do_end_variable_parse ( expr , BP_VAR_R , 0 TSRMLS_CC ) ;
1999-12-16 04:15:32 +08:00
}
}
2001-07-19 23:53:38 +08:00
2008-05-05 19:03:35 +08:00
start_op_number = get_next_op_number ( CG ( active_op_array ) ) ;
1999-07-24 19:43:21 +08:00
# ifdef ZTS
2001-07-28 18:51:54 +08:00
zend_stack_apply_with_argument ( & CG ( switch_cond_stack ) , ZEND_STACK_APPLY_TOPDOWN , ( int ( * ) ( void * element , void * ) ) generate_free_switch_expr TSRMLS_CC ) ;
zend_stack_apply_with_argument ( & CG ( foreach_copy_stack ) , ZEND_STACK_APPLY_TOPDOWN , ( int ( * ) ( void * element , void * ) ) generate_free_foreach_copy TSRMLS_CC ) ;
1999-07-24 19:43:21 +08:00
# else
2000-03-30 06:28:04 +08:00
zend_stack_apply ( & CG ( switch_cond_stack ) , ZEND_STACK_APPLY_TOPDOWN , ( int ( * ) ( void * element ) ) generate_free_switch_expr ) ;
zend_stack_apply ( & CG ( foreach_copy_stack ) , ZEND_STACK_APPLY_TOPDOWN , ( int ( * ) ( void * element ) ) generate_free_foreach_copy ) ;
1999-07-24 19:43:21 +08:00
# endif
2008-05-05 19:03:35 +08:00
end_op_number = get_next_op_number ( CG ( active_op_array ) ) ;
while ( start_op_number < end_op_number ) {
2010-04-20 18:57:45 +08:00
CG ( active_op_array ) - > opcodes [ start_op_number ] . extended_value | = EXT_TYPE_FREE_ON_RETURN ;
2008-05-05 19:03:35 +08:00
start_op_number + + ;
}
2001-07-28 18:51:54 +08:00
opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
1999-07-24 19:43:21 +08:00
2010-04-22 19:56:45 +08:00
opline - > opcode = ( CG ( active_op_array ) - > return_reference = = ZEND_RETURN_REF ) ? ZEND_RETURN_BY_REF : ZEND_RETURN ;
2006-05-10 07:53:23 +08:00
1999-04-08 02:10:10 +08:00
if ( expr ) {
2010-04-20 18:57:45 +08:00
SET_NODE ( opline - > op1 , expr ) ;
2003-12-15 00:09:07 +08:00
2007-02-01 23:23:46 +08:00
if ( do_end_vparse & & zend_is_function_or_method_call ( expr ) ) {
2006-05-10 07:53:23 +08:00
opline - > extended_value = ZEND_RETURNS_FUNCTION ;
2003-12-15 00:09:07 +08:00
}
2007-02-01 23:23:46 +08:00
} else {
2010-04-20 18:57:45 +08:00
opline - > op1_type = IS_CONST ;
LITERAL_NULL ( opline - > op1 ) ;
2003-12-15 00:09:07 +08:00
}
1999-04-08 02:10:10 +08:00
SET_UNUSED ( opline - > op2 ) ;
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
static int zend_add_try_element ( zend_uint try_op TSRMLS_DC ) /* { { { */
2004-02-03 20:17:09 +08:00
{
int try_catch_offset = CG ( active_op_array ) - > last_try_catch + + ;
CG ( active_op_array ) - > try_catch_array = erealloc ( CG ( active_op_array ) - > try_catch_array , sizeof ( zend_try_catch_element ) * CG ( active_op_array ) - > last_try_catch ) ;
CG ( active_op_array ) - > try_catch_array [ try_catch_offset ] . try_op = try_op ;
return try_catch_offset ;
}
2009-07-27 22:11:53 +08:00
/* }}} */
2004-02-03 20:17:09 +08:00
2009-07-27 22:11:53 +08:00
static void zend_add_catch_element ( int offset , zend_uint catch_op TSRMLS_DC ) /* { { { */
2004-02-03 20:17:09 +08:00
{
CG ( active_op_array ) - > try_catch_array [ offset ] . catch_op = catch_op ;
}
2009-07-27 22:11:53 +08:00
/* }}} */
2004-02-03 20:17:09 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_first_catch ( znode * open_parentheses TSRMLS_DC ) /* { { { */
2004-02-03 20:17:09 +08:00
{
2010-04-20 18:57:45 +08:00
open_parentheses - > u . op . opline_num = get_next_op_number ( CG ( active_op_array ) ) ;
2004-02-03 20:17:09 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
2004-02-03 20:17:09 +08:00
2009-07-27 22:11:53 +08:00
void zend_initialize_try_catch_element ( const znode * try_token TSRMLS_DC ) /* { { { */
2004-02-03 20:17:09 +08:00
{
2007-01-09 23:06:07 +08:00
int jmp_op_number = get_next_op_number ( CG ( active_op_array ) ) ;
zend_op * opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
zend_llist jmp_list ;
zend_llist * jmp_list_ptr ;
opline - > opcode = ZEND_JMP ;
SET_UNUSED ( opline - > op1 ) ;
SET_UNUSED ( opline - > op2 ) ;
/* save for backpatching */
zend_llist_init ( & jmp_list , sizeof ( int ) , NULL , 0 ) ;
zend_stack_push ( & CG ( bp_stack ) , ( void * ) & jmp_list , sizeof ( zend_llist ) ) ;
zend_stack_top ( & CG ( bp_stack ) , ( void * * ) & jmp_list_ptr ) ;
zend_llist_add_element ( jmp_list_ptr , & jmp_op_number ) ;
2010-04-20 18:57:45 +08:00
zend_add_catch_element ( try_token - > u . op . opline_num , get_next_op_number ( CG ( active_op_array ) ) TSRMLS_CC ) ;
2004-02-03 20:17:09 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
2004-02-03 20:17:09 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_mark_last_catch ( const znode * first_catch , const znode * last_additional_catch TSRMLS_DC ) /* { { { */
2004-02-03 20:17:09 +08:00
{
2007-01-09 23:06:07 +08:00
CG ( active_op_array ) - > last - - ;
zend_do_if_end ( TSRMLS_C ) ;
2010-04-20 18:57:45 +08:00
if ( last_additional_catch - > u . op . opline_num = = - 1 ) {
CG ( active_op_array ) - > opcodes [ first_catch - > u . op . opline_num ] . result . num = 1 ;
CG ( active_op_array ) - > opcodes [ first_catch - > u . op . opline_num ] . extended_value = get_next_op_number ( CG ( active_op_array ) ) ;
2004-02-03 20:17:09 +08:00
} else {
2010-04-20 18:57:45 +08:00
CG ( active_op_array ) - > opcodes [ last_additional_catch - > u . op . opline_num ] . result . num = 1 ;
CG ( active_op_array ) - > opcodes [ last_additional_catch - > u . op . opline_num ] . extended_value = get_next_op_number ( CG ( active_op_array ) ) ;
2004-02-03 20:17:09 +08:00
}
2005-11-24 19:33:11 +08:00
DEC_BPC ( CG ( active_op_array ) ) ;
2004-02-03 20:17:09 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
2004-02-03 20:17:09 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_try ( znode * try_token TSRMLS_DC ) /* { { { */
2001-08-30 23:26:30 +08:00
{
2010-04-20 18:57:45 +08:00
try_token - > u . op . opline_num = zend_add_try_element ( get_next_op_number ( CG ( active_op_array ) ) TSRMLS_CC ) ;
2005-11-24 19:33:11 +08:00
INC_BPC ( CG ( active_op_array ) ) ;
2001-08-30 23:26:30 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
2001-08-30 23:26:30 +08:00
2010-04-20 19:05:54 +08:00
void zend_do_begin_catch ( znode * try_token , znode * class_name , znode * catch_var , znode * first_catch TSRMLS_DC ) /* { { { */
2001-08-30 23:26:30 +08:00
{
2007-09-29 03:52:53 +08:00
long catch_op_number ;
2001-08-30 23:26:30 +08:00
zend_op * opline ;
2007-09-29 03:52:53 +08:00
znode catch_class ;
2006-05-10 07:53:23 +08:00
2010-04-20 19:16:39 +08:00
if ( class_name - > op_type = = IS_CONST & &
ZEND_FETCH_CLASS_DEFAULT = = zend_get_class_fetch_type ( Z_STRVAL ( class_name - > u . constant ) , Z_STRLEN ( class_name - > u . constant ) ) ) {
ulong fetch_type = ZEND_FETCH_CLASS_GLOBAL ;
2007-09-29 03:52:53 +08:00
2010-04-20 19:16:39 +08:00
zend_resolve_class_name ( class_name , & fetch_type , 1 TSRMLS_CC ) ;
catch_class = * class_name ;
} else {
zend_error ( E_COMPILE_ERROR , " Bad class name in the catch statement " ) ;
2005-09-09 14:48:49 +08:00
}
2010-04-20 19:16:39 +08:00
catch_op_number = get_next_op_number ( CG ( active_op_array ) ) ;
2007-09-29 03:52:53 +08:00
if ( first_catch ) {
2010-04-20 18:57:45 +08:00
first_catch - > u . op . opline_num = catch_op_number ;
2007-09-29 03:52:53 +08:00
}
2001-08-31 01:27:43 +08:00
opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
2001-08-30 23:26:30 +08:00
opline - > opcode = ZEND_CATCH ;
2010-04-21 22:58:33 +08:00
opline - > op1_type = IS_CONST ;
2010-04-21 23:08:10 +08:00
opline - > op1 . constant = zend_add_class_name_literal ( CG ( active_op_array ) , & catch_class . u . constant TSRMLS_CC ) ;
2010-04-20 18:57:45 +08:00
opline - > op2_type = IS_CV ;
2010-04-20 19:05:54 +08:00
opline - > op2 . var = lookup_cv ( CG ( active_op_array ) , catch_var - > u . constant . value . str . val , catch_var - > u . constant . value . str . len TSRMLS_CC ) ;
catch_var - > u . constant . value . str . val = CG ( active_op_array ) - > vars [ opline - > op2 . var ] . name ;
2010-04-20 18:57:45 +08:00
opline - > result . num = 0 ; /* 1 means it's the last catch in the block */
2001-08-30 23:26:30 +08:00
2010-04-20 18:57:45 +08:00
try_token - > u . op . opline_num = catch_op_number ;
2001-08-30 23:26:30 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
2001-08-30 23:26:30 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_end_catch ( const znode * try_token TSRMLS_DC ) /* { { { */
2001-08-30 23:26:30 +08:00
{
2007-01-09 23:06:07 +08:00
int jmp_op_number = get_next_op_number ( CG ( active_op_array ) ) ;
zend_op * opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
zend_llist * jmp_list_ptr ;
opline - > opcode = ZEND_JMP ;
SET_UNUSED ( opline - > op1 ) ;
SET_UNUSED ( opline - > op2 ) ;
/* save for backpatching */
zend_stack_top ( & CG ( bp_stack ) , ( void * * ) & jmp_list_ptr ) ;
zend_llist_add_element ( jmp_list_ptr , & jmp_op_number ) ;
2010-04-20 18:57:45 +08:00
CG ( active_op_array ) - > opcodes [ try_token - > u . op . opline_num ] . extended_value = get_next_op_number ( CG ( active_op_array ) ) ;
2001-08-30 23:26:30 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
2001-08-30 23:26:30 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_throw ( const znode * expr TSRMLS_DC ) /* { { { */
2001-08-30 23:26:30 +08:00
{
zend_op * opline ;
2006-05-10 07:53:23 +08:00
2001-08-31 01:27:43 +08:00
opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
2001-08-30 23:26:30 +08:00
opline - > opcode = ZEND_THROW ;
2010-04-20 18:57:45 +08:00
SET_NODE ( opline - > op1 , expr ) ;
2001-08-30 23:26:30 +08:00
SET_UNUSED ( opline - > op2 ) ;
}
2009-07-27 22:11:53 +08:00
/* }}} */
2001-08-30 23:26:30 +08:00
2009-07-27 22:11:53 +08:00
ZEND_API void function_add_ref ( zend_function * function ) /* { { { */
1999-07-09 00:55:27 +08:00
{
if ( function - > type = = ZEND_USER_FUNCTION ) {
1999-12-24 03:23:36 +08:00
zend_op_array * op_array = & function - > op_array ;
( * op_array - > refcount ) + + ;
if ( op_array - > static_variables ) {
HashTable * static_variables = op_array - > static_variables ;
zval * tmp_zval ;
1999-12-28 03:07:33 +08:00
ALLOC_HASHTABLE ( op_array - > static_variables ) ;
2006-10-03 19:10:33 +08:00
zend_hash_init ( op_array - > static_variables , zend_hash_num_elements ( static_variables ) , NULL , ZVAL_PTR_DTOR , 0 ) ;
2000-02-05 23:11:24 +08:00
zend_hash_copy ( op_array - > static_variables , static_variables , ( copy_ctor_func_t ) zval_add_ref , ( void * ) & tmp_zval , sizeof ( zval * ) ) ;
1999-12-24 03:23:36 +08:00
}
1999-07-09 00:55:27 +08:00
}
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-07-09 00:55:27 +08:00
2009-07-27 22:11:53 +08:00
static void do_inherit_parent_constructor ( zend_class_entry * ce ) /* { { { */
1999-07-28 04:34:31 +08:00
{
2003-09-04 05:21:18 +08:00
zend_function * function ;
2003-05-08 23:04:43 +08:00
if ( ! ce - > parent ) {
return ;
}
2003-09-04 04:26:01 +08:00
/* You cannot change create_object */
ce - > create_object = ce - > parent - > create_object ;
2005-02-23 19:15:51 +08:00
2003-09-04 04:26:01 +08:00
/* Inherit special functions if needed */
2003-10-19 02:43:31 +08:00
if ( ! ce - > get_iterator ) {
ce - > get_iterator = ce - > parent - > get_iterator ;
}
if ( ! ce - > iterator_funcs . funcs ) {
ce - > iterator_funcs . funcs = ce - > parent - > iterator_funcs . funcs ;
}
2003-08-17 20:16:10 +08:00
if ( ! ce - > __get ) {
ce - > __get = ce - > parent - > __get ;
}
if ( ! ce - > __set ) {
ce - > __set = ce - > parent - > __set ;
}
2005-07-08 00:07:09 +08:00
if ( ! ce - > __unset ) {
ce - > __unset = ce - > parent - > __unset ;
}
if ( ! ce - > __isset ) {
ce - > __isset = ce - > parent - > __isset ;
}
2003-08-17 20:16:10 +08:00
if ( ! ce - > __call ) {
ce - > __call = ce - > parent - > __call ;
}
2007-09-29 16:52:40 +08:00
if ( ! ce - > __callstatic ) {
ce - > __callstatic = ce - > parent - > __callstatic ;
}
2006-05-10 07:53:23 +08:00
if ( ! ce - > __tostring ) {
ce - > __tostring = ce - > parent - > __tostring ;
}
2003-09-04 04:26:01 +08:00
if ( ! ce - > clone ) {
ce - > clone = ce - > parent - > clone ;
}
2005-02-23 19:15:51 +08:00
if ( ! ce - > serialize ) {
ce - > serialize = ce - > parent - > serialize ;
}
if ( ! ce - > unserialize ) {
ce - > unserialize = ce - > parent - > unserialize ;
}
2003-08-17 20:16:10 +08:00
if ( ! ce - > destructor ) {
ce - > destructor = ce - > parent - > destructor ;
}
2003-09-04 05:21:18 +08:00
if ( ce - > constructor ) {
2005-12-17 23:51:52 +08:00
if ( ce - > parent - > constructor & & ce - > parent - > constructor - > common . fn_flags & ZEND_ACC_FINAL ) {
zend_error ( E_ERROR , " Cannot override final %s::%s() with %s::%s() " ,
ce - > parent - > name , ce - > parent - > constructor - > common . function_name ,
ce - > name , ce - > constructor - > common . function_name
) ;
}
2003-09-04 05:21:18 +08:00
return ;
2002-06-23 23:46:58 +08:00
}
2006-05-10 07:53:23 +08:00
2003-09-04 05:21:18 +08:00
if ( zend_hash_find ( & ce - > parent - > function_table , ZEND_CONSTRUCTOR_FUNC_NAME , sizeof ( ZEND_CONSTRUCTOR_FUNC_NAME ) , ( void * * ) & function ) = = SUCCESS ) {
/* inherit parent's constructor */
zend_hash_update ( & ce - > function_table , ZEND_CONSTRUCTOR_FUNC_NAME , sizeof ( ZEND_CONSTRUCTOR_FUNC_NAME ) , function , sizeof ( zend_function ) , NULL ) ;
function_add_ref ( function ) ;
} else {
2004-03-16 23:22:44 +08:00
/* Don't inherit the old style constructor if we already have the new style constructor */
char * lc_class_name ;
char * lc_parent_class_name ;
lc_class_name = zend_str_tolower_dup ( ce - > name , ce - > name_length ) ;
if ( ! zend_hash_exists ( & ce - > function_table , lc_class_name , ce - > name_length + 1 ) ) {
lc_parent_class_name = zend_str_tolower_dup ( ce - > parent - > name , ce - > parent - > name_length ) ;
2008-03-17 22:54:42 +08:00
if ( ! zend_hash_exists ( & ce - > function_table , lc_parent_class_name , ce - > parent - > name_length + 1 ) & &
zend_hash_find ( & ce - > parent - > function_table , lc_parent_class_name , ce - > parent - > name_length + 1 , ( void * * ) & function ) = = SUCCESS ) {
2004-08-27 06:25:55 +08:00
if ( function - > common . fn_flags & ZEND_ACC_CTOR ) {
/* inherit parent's constructor */
2008-03-17 22:54:42 +08:00
zend_hash_update ( & ce - > function_table , lc_parent_class_name , ce - > parent - > name_length + 1 , function , sizeof ( zend_function ) , NULL ) ;
2004-08-27 06:25:55 +08:00
function_add_ref ( function ) ;
}
2003-09-04 05:21:18 +08:00
}
2004-03-16 23:22:44 +08:00
efree ( lc_parent_class_name ) ;
2003-09-04 05:21:18 +08:00
}
2004-03-16 23:22:44 +08:00
efree ( lc_class_name ) ;
2003-09-04 05:21:18 +08:00
}
ce - > constructor = ce - > parent - > constructor ;
1999-07-28 04:34:31 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-07-28 04:34:31 +08:00
2009-07-27 22:11:53 +08:00
char * zend_visibility_string ( zend_uint fn_flags ) /* { { { */
2002-12-07 01:09:44 +08:00
{
if ( fn_flags & ZEND_ACC_PRIVATE ) {
return " private " ;
}
if ( fn_flags & ZEND_ACC_PROTECTED ) {
return " protected " ;
}
if ( fn_flags & ZEND_ACC_PUBLIC ) {
return " public " ;
}
return " " ;
}
2009-07-27 22:11:53 +08:00
/* }}} */
2002-12-07 01:09:44 +08:00
2009-07-27 22:11:53 +08:00
static void do_inherit_method ( zend_function * function ) /* { { { */
2002-12-07 01:09:44 +08:00
{
/* The class entry of the derived function intentionally remains the same
* as that of the parent class . That allows us to know in which context
* we ' re running , and handle private method calls properly .
*/
function_add_ref ( function ) ;
}
2009-07-27 22:11:53 +08:00
/* }}} */
2002-12-07 01:09:44 +08:00
2009-07-27 22:11:53 +08:00
static zend_bool zend_do_perform_implementation_check ( const zend_function * fe , const zend_function * proto ) /* { { { */
2003-03-29 19:19:38 +08:00
{
ntroduce infrastructure for supplying information about arguments,
including:
- Whether or not to pass by ref (replaces the old arg_types, with arg_info)
- Argument name (for future use, maybe introspection)
- Class/Interface name (for type hints)
- If a class/interface name is available, whether to allow a null instance
Both user and builtin functions share the same data structures.
To declare a builtin function that expects its first arg to be an instance
of class 'Person', its second argument as a regular arg, and its third by
reference, use:
ZEND_BEGIN_ARG_INFO(my_func_arg_info, 0)
ZEND_ARG_OBJ_INFO(0, someone, Person, 1)
ZEND_ARG_PASS_INFO(0)
ZEND_ARG_PASS_INFO(1)
ZEND_END_ARG_INFO();
and use my_func_arg_info as the arg_info parameter to the ZEND_FE() family
of macros.
The first arg to each ZEND_ARG_*() macro is whether or not to pass by ref.
The boolean arg to ZEND_BEGIN_ARG_INFO() tells the engine whether to treat
the arguments for which there's no explicit information as pass by reference
or not.
The boolean argument to ZEND_ARG_OBJ_INFO() (4th arg) is whether or not to allownull values.
2003-08-04 01:40:44 +08:00
zend_uint i ;
2003-03-29 19:19:38 +08:00
2004-03-01 02:25:50 +08:00
/* If it's a user function then arg_info == NULL means we don't have any parameters but we still need to do the arg number checks. We are only willing to ignore this for internal functions because extensions don't always define arg_info. */
if ( ! proto | | ( ! proto - > common . arg_info & & proto - > common . type ! = ZEND_USER_FUNCTION ) ) {
2003-03-29 19:19:38 +08:00
return 1 ;
}
2006-05-10 09:23:55 +08:00
/* Checks for constructors only if they are declared in an interface */
if ( ( fe - > common . fn_flags & ZEND_ACC_CTOR ) & & ! ( proto - > common . scope - > ce_flags & ZEND_ACC_INTERFACE ) ) {
2004-04-21 16:44:37 +08:00
return 1 ;
}
2004-02-25 17:25:37 +08:00
/* check number of arguments */
2008-03-01 21:53:32 +08:00
if ( proto - > common . required_num_args < fe - > common . required_num_args
2004-02-25 17:25:37 +08:00
| | proto - > common . num_args > fe - > common . num_args ) {
return 0 ;
}
if ( proto - > common . pass_rest_by_reference
& & ! fe - > common . pass_rest_by_reference ) {
2003-03-29 19:19:38 +08:00
return 0 ;
}
2005-06-16 22:56:13 +08:00
if ( fe - > common . return_reference ! = proto - > common . return_reference ) {
return 0 ;
2004-02-12 21:49:55 +08:00
}
2004-02-25 17:25:37 +08:00
for ( i = 0 ; i < proto - > common . num_args ; i + + ) {
if ( ZEND_LOG_XOR ( fe - > common . arg_info [ i ] . class_name , proto - > common . arg_info [ i ] . class_name ) ) {
2003-08-04 15:53:09 +08:00
/* Only one has a type hint and the other one doesn't */
return 0 ;
}
if ( fe - > common . arg_info [ i ] . class_name
2008-01-09 15:57:42 +08:00
& & strcasecmp ( fe - > common . arg_info [ i ] . class_name , proto - > common . arg_info [ i ] . class_name ) ! = 0 ) {
2007-11-09 21:34:39 +08:00
char * colon ;
2008-03-18 19:37:00 +08:00
if ( fe - > common . type ! = ZEND_USER_FUNCTION | |
2008-11-04 23:58:55 +08:00
strchr ( proto - > common . arg_info [ i ] . class_name , ' \\ ' ) ! = NULL | |
( colon = zend_memrchr ( fe - > common . arg_info [ i ] . class_name , ' \\ ' , fe - > common . arg_info [ i ] . class_name_len ) ) = = NULL | |
2008-03-18 19:37:00 +08:00
strcasecmp ( colon + 1 , proto - > common . arg_info [ i ] . class_name ) ! = 0 ) {
2007-11-09 21:34:39 +08:00
return 0 ;
}
2003-03-29 19:19:38 +08:00
}
2005-05-26 21:46:17 +08:00
if ( fe - > common . arg_info [ i ] . array_type_hint ! = proto - > common . arg_info [ i ] . array_type_hint ) {
/* Only one has an array type hint and the other one doesn't */
return 0 ;
}
2004-02-25 17:25:37 +08:00
if ( fe - > common . arg_info [ i ] . pass_by_reference ! = proto - > common . arg_info [ i ] . pass_by_reference ) {
2003-03-29 19:19:38 +08:00
return 0 ;
}
2003-07-16 17:13:47 +08:00
}
2004-02-25 17:25:37 +08:00
if ( proto - > common . pass_rest_by_reference ) {
for ( i = proto - > common . num_args ; i < fe - > common . num_args ; i + + ) {
if ( ! fe - > common . arg_info [ i ] . pass_by_reference ) {
return 0 ;
}
}
}
ntroduce infrastructure for supplying information about arguments,
including:
- Whether or not to pass by ref (replaces the old arg_types, with arg_info)
- Argument name (for future use, maybe introspection)
- Class/Interface name (for type hints)
- If a class/interface name is available, whether to allow a null instance
Both user and builtin functions share the same data structures.
To declare a builtin function that expects its first arg to be an instance
of class 'Person', its second argument as a regular arg, and its third by
reference, use:
ZEND_BEGIN_ARG_INFO(my_func_arg_info, 0)
ZEND_ARG_OBJ_INFO(0, someone, Person, 1)
ZEND_ARG_PASS_INFO(0)
ZEND_ARG_PASS_INFO(1)
ZEND_END_ARG_INFO();
and use my_func_arg_info as the arg_info parameter to the ZEND_FE() family
of macros.
The first arg to each ZEND_ARG_*() macro is whether or not to pass by ref.
The boolean arg to ZEND_BEGIN_ARG_INFO() tells the engine whether to treat
the arguments for which there's no explicit information as pass by reference
or not.
The boolean argument to ZEND_ARG_OBJ_INFO() (4th arg) is whether or not to allownull values.
2003-08-04 01:40:44 +08:00
return 1 ;
2003-03-29 19:19:38 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
2003-03-29 19:19:38 +08:00
2010-04-23 07:16:15 +08:00
static void do_inheritance_check_on_method ( zend_function * child , zend_function * parent TSRMLS_DC )
2002-12-07 01:09:44 +08:00
{
2003-02-04 20:12:34 +08:00
zend_uint child_flags ;
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
zend_uint parent_flags = parent - > common . fn_flags ;
2004-01-29 06:27:39 +08:00
if ( parent - > common . fn_flags & ZEND_ACC_ABSTRACT
& & parent - > common . scope ! = ( child - > common . prototype ? child - > common . prototype - > common . scope : child - > common . scope )
2004-01-28 18:25:45 +08:00
& & child - > common . fn_flags & ( ZEND_ACC_ABSTRACT | ZEND_ACC_IMPLEMENTED_ABSTRACT ) ) {
zend_error ( E_COMPILE_ERROR , " Can't inherit abstract function %s::%s() (previously declared abstract in %s) " ,
parent - > common . scope - > name ,
child - > common . function_name ,
2004-01-28 19:53:52 +08:00
child - > common . prototype ? child - > common . prototype - > common . scope - > name : child - > common . scope - > name ) ;
2004-01-28 18:25:45 +08:00
}
2003-02-24 20:05:58 +08:00
if ( parent_flags & ZEND_ACC_FINAL ) {
zend_error ( E_COMPILE_ERROR , " Cannot override final method %s::%s() " , ZEND_FN_SCOPE_NAME ( parent ) , child - > common . function_name ) ;
}
2003-02-04 20:12:34 +08:00
child_flags = child - > common . fn_flags ;
2002-11-24 04:44:12 +08:00
/* You cannot change from static to non static and vice versa.
*/
2002-12-07 01:09:44 +08:00
if ( ( child_flags & ZEND_ACC_STATIC ) ! = ( parent_flags & ZEND_ACC_STATIC ) ) {
if ( child - > common . fn_flags & ZEND_ACC_STATIC ) {
zend_error ( E_COMPILE_ERROR , " Cannot make non static method %s::%s() static in class %s " , ZEND_FN_SCOPE_NAME ( parent ) , child - > common . function_name , ZEND_FN_SCOPE_NAME ( child ) ) ;
2002-11-24 04:44:12 +08:00
} else {
2002-12-07 01:09:44 +08:00
zend_error ( E_COMPILE_ERROR , " Cannot make static method %s::%s() non static in class %s " , ZEND_FN_SCOPE_NAME ( parent ) , child - > common . function_name , ZEND_FN_SCOPE_NAME ( child ) ) ;
2002-11-24 04:44:12 +08:00
}
}
2002-12-08 05:46:12 +08:00
2003-02-23 21:58:48 +08:00
/* Disallow making an inherited method abstract. */
2002-12-08 05:46:12 +08:00
if ( ( child_flags & ZEND_ACC_ABSTRACT ) & & ! ( parent_flags & ZEND_ACC_ABSTRACT ) ) {
zend_error ( E_COMPILE_ERROR , " Cannot make non abstract method %s::%s() abstract in class %s " , ZEND_FN_SCOPE_NAME ( parent ) , child - > common . function_name , ZEND_FN_SCOPE_NAME ( child ) ) ;
2002-11-24 04:44:12 +08:00
}
2002-12-09 20:10:17 +08:00
2003-01-02 21:58:08 +08:00
if ( parent_flags & ZEND_ACC_CHANGED ) {
child - > common . fn_flags | = ZEND_ACC_CHANGED ;
} else {
/* Prevent derived classes from restricting access that was available in parent classes
*/
if ( ( child_flags & ZEND_ACC_PPP_MASK ) > ( parent_flags & ZEND_ACC_PPP_MASK ) ) {
2003-02-06 08:39:58 +08:00
zend_error ( E_COMPILE_ERROR , " Access level to %s::%s() must be %s (as in class %s)%s " , ZEND_FN_SCOPE_NAME ( child ) , child - > common . function_name , zend_visibility_string ( parent_flags ) , ZEND_FN_SCOPE_NAME ( parent ) , ( parent_flags & ZEND_ACC_PUBLIC ) ? " " : " or weaker " ) ;
2006-05-12 05:07:39 +08:00
} else if ( ( ( child_flags & ZEND_ACC_PPP_MASK ) < ( parent_flags & ZEND_ACC_PPP_MASK ) )
2003-01-02 21:58:08 +08:00
& & ( ( parent_flags & ZEND_ACC_PPP_MASK ) & ZEND_ACC_PRIVATE ) ) {
child - > common . fn_flags | = ZEND_ACC_CHANGED ;
}
2002-12-09 20:10:17 +08:00
}
2002-12-27 00:27:59 +08:00
2006-09-12 19:01:16 +08:00
if ( parent_flags & ZEND_ACC_PRIVATE ) {
child - > common . prototype = NULL ;
} else if ( parent_flags & ZEND_ACC_ABSTRACT ) {
2004-03-01 02:25:50 +08:00
child - > common . fn_flags | = ZEND_ACC_IMPLEMENTED_ABSTRACT ;
2004-04-21 16:44:37 +08:00
child - > common . prototype = parent ;
2006-05-31 07:10:28 +08:00
} else if ( ! ( parent - > common . fn_flags & ZEND_ACC_CTOR ) | | ( parent - > common . prototype & & ( parent - > common . prototype - > common . scope - > ce_flags & ZEND_ACC_INTERFACE ) ) ) {
2006-05-30 04:06:43 +08:00
/* ctors only have a prototype if it comes from an interface */
child - > common . prototype = parent - > common . prototype ? parent - > common . prototype : parent ;
2003-03-29 19:19:38 +08:00
}
2006-05-31 07:10:28 +08:00
if ( child - > common . prototype & & ( child - > common . prototype - > common . fn_flags & ZEND_ACC_ABSTRACT ) ) {
2004-04-21 16:44:37 +08:00
if ( ! zend_do_perform_implementation_check ( child , child - > common . prototype ) ) {
zend_error ( E_COMPILE_ERROR , " Declaration of %s::%s() must be compatible with that of %s::%s() " , ZEND_FN_SCOPE_NAME ( child ) , child - > common . function_name , ZEND_FN_SCOPE_NAME ( child - > common . prototype ) , child - > common . prototype - > common . function_name ) ;
}
2009-04-16 21:48:01 +08:00
} else if ( EG ( error_reporting ) & E_STRICT | | EG ( user_error_handler ) ) { /* Check E_STRICT (or custom error handler) before the check so that we save some time */
2004-04-21 16:44:37 +08:00
if ( ! zend_do_perform_implementation_check ( child , parent ) ) {
zend_error ( E_STRICT , " Declaration of %s::%s() should be compatible with that of %s::%s() " , ZEND_FN_SCOPE_NAME ( child ) , child - > common . function_name , ZEND_FN_SCOPE_NAME ( parent ) , parent - > common . function_name ) ;
2004-03-01 02:25:50 +08:00
}
2003-03-29 19:19:38 +08:00
}
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
}
2003-03-29 19:19:38 +08:00
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
static zend_bool do_inherit_method_check ( HashTable * child_function_table , zend_function * parent , const zend_hash_key * hash_key , zend_class_entry * child_ce ) /* { { { */
{
zend_uint parent_flags = parent - > common . fn_flags ;
zend_function * child ;
TSRMLS_FETCH ( ) ;
if ( zend_hash_quick_find ( child_function_table , hash_key - > arKey , hash_key - > nKeyLength , hash_key - > h , ( void * * ) & child ) = = FAILURE ) {
if ( parent_flags & ( ZEND_ACC_ABSTRACT ) ) {
child_ce - > ce_flags | = ZEND_ACC_IMPLICIT_ABSTRACT_CLASS ;
}
return 1 ; /* method doesn't exist in child, copy from parent */
}
2010-04-23 07:16:15 +08:00
do_inheritance_check_on_method ( child , parent TSRMLS_CC ) ;
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
2002-12-27 00:27:59 +08:00
return 0 ;
2002-11-24 04:44:12 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
2002-11-24 04:44:12 +08:00
2009-07-27 22:11:53 +08:00
static zend_bool do_inherit_property_access_check ( HashTable * target_ht , zend_property_info * parent_info , const zend_hash_key * hash_key , zend_class_entry * ce ) /* { { { */
2003-02-04 20:12:34 +08:00
{
zend_property_info * child_info ;
zend_class_entry * parent_ce = ce - > parent ;
2005-06-10 01:20:44 +08:00
if ( parent_info - > flags & ( ZEND_ACC_PRIVATE | ZEND_ACC_SHADOW ) ) {
2003-03-19 00:30:23 +08:00
if ( zend_hash_quick_find ( & ce - > properties_info , hash_key - > arKey , hash_key - > nKeyLength , hash_key - > h , ( void * * ) & child_info ) = = SUCCESS ) {
2003-09-09 07:25:57 +08:00
child_info - > flags | = ZEND_ACC_CHANGED ;
2005-06-10 01:20:44 +08:00
} else {
zend_hash_quick_update ( & ce - > properties_info , hash_key - > arKey , hash_key - > nKeyLength , hash_key - > h , parent_info , sizeof ( zend_property_info ) , ( void * * ) & child_info ) ;
if ( ce - > type & ZEND_INTERNAL_CLASS ) {
zend_duplicate_property_info_internal ( child_info ) ;
} else {
zend_duplicate_property_info ( child_info ) ;
}
child_info - > flags & = ~ ZEND_ACC_PRIVATE ; /* it's not private anymore */
child_info - > flags | = ZEND_ACC_SHADOW ; /* but it's a shadow of private */
2003-03-19 00:30:23 +08:00
}
2003-02-04 20:12:34 +08:00
return 0 ; /* don't copy access information to child */
}
2003-02-05 21:35:52 +08:00
if ( zend_hash_quick_find ( & ce - > properties_info , hash_key - > arKey , hash_key - > nKeyLength , hash_key - > h , ( void * * ) & child_info ) = = SUCCESS ) {
2003-09-04 03:28:46 +08:00
if ( ( parent_info - > flags & ZEND_ACC_STATIC ) ! = ( child_info - > flags & ZEND_ACC_STATIC ) ) {
zend_error ( E_COMPILE_ERROR , " Cannot redeclare %s%s::$%s as %s%s::$%s " ,
( parent_info - > flags & ZEND_ACC_STATIC ) ? " static " : " non static " , parent_ce - > name , hash_key - > arKey ,
( child_info - > flags & ZEND_ACC_STATIC ) ? " static " : " non static " , ce - > name , hash_key - > arKey ) ;
}
2006-05-12 05:07:39 +08:00
2005-06-10 01:20:44 +08:00
if ( parent_info - > flags & ZEND_ACC_CHANGED ) {
child_info - > flags | = ZEND_ACC_CHANGED ;
}
2006-05-12 05:07:39 +08:00
2003-02-04 20:12:34 +08:00
if ( ( child_info - > flags & ZEND_ACC_PPP_MASK ) > ( parent_info - > flags & ZEND_ACC_PPP_MASK ) ) {
zend_error ( E_COMPILE_ERROR , " Access level to %s::$%s must be %s (as in class %s)%s " , ce - > name , hash_key - > arKey , zend_visibility_string ( parent_info - > flags ) , parent_ce - > name , ( parent_info - > flags & ZEND_ACC_PUBLIC ) ? " " : " or weaker " ) ;
2003-02-10 20:46:58 +08:00
} else if ( child_info - > flags & ZEND_ACC_IMPLICIT_PUBLIC ) {
2003-09-09 07:25:57 +08:00
if ( ! ( parent_info - > flags & ZEND_ACC_IMPLICIT_PUBLIC ) ) {
/* Explicitly copy the default value from the parent (if it has one) */
zval * * pvalue ;
2004-10-17 07:17:58 +08:00
if ( zend_hash_quick_find ( & parent_ce - > default_properties , parent_info - > name , parent_info - > name_length + 1 , parent_info - > h , ( void * * ) & pvalue ) = = SUCCESS ) {
2007-10-07 13:22:07 +08:00
Z_ADDREF_PP ( pvalue ) ;
2008-04-29 16:15:20 +08:00
zend_hash_quick_del ( & ce - > default_properties , child_info - > name , child_info - > name_length + 1 , parent_info - > h ) ;
2003-09-09 07:25:57 +08:00
zend_hash_quick_update ( & ce - > default_properties , parent_info - > name , parent_info - > name_length + 1 , parent_info - > h , pvalue , sizeof ( zval * ) , NULL ) ;
}
2003-03-23 16:14:41 +08:00
}
2003-02-10 20:46:58 +08:00
return 1 ; /* Inherit from the parent */
2003-09-03 18:58:55 +08:00
} else if ( ( child_info - > flags & ZEND_ACC_PUBLIC ) & & ( parent_info - > flags & ZEND_ACC_PROTECTED ) ) {
char * prot_name ;
int prot_name_length ;
2004-01-03 21:51:02 +08:00
zend_mangle_property_name ( & prot_name , & prot_name_length , " * " , 1 , child_info - > name , child_info - > name_length , ce - > type & ZEND_INTERNAL_CLASS ) ;
2003-09-03 18:58:55 +08:00
if ( child_info - > flags & ZEND_ACC_STATIC ) {
2003-09-04 00:13:40 +08:00
zval * * prop ;
2005-09-01 18:05:32 +08:00
HashTable * ht ;
if ( parent_ce - > type ! = ce - > type ) {
/* User class extends internal class */
2005-12-01 19:48:17 +08:00
TSRMLS_FETCH ( ) ;
ht = CE_STATIC_MEMBERS ( parent_ce ) ;
2005-09-01 18:05:32 +08:00
} else {
ht = & parent_ce - > default_static_members ;
}
if ( zend_hash_find ( ht , prot_name , prot_name_length + 1 , ( void * * ) & prop ) = = SUCCESS ) {
zend_hash_del ( & ce - > default_static_members , prot_name , prot_name_length + 1 ) ;
2003-09-04 00:13:40 +08:00
}
2003-09-03 18:58:55 +08:00
} else {
zend_hash_del ( & ce - > default_properties , prot_name , prot_name_length + 1 ) ;
}
pefree ( prot_name , ce - > type & ZEND_INTERNAL_CLASS ) ;
2003-02-04 20:12:34 +08:00
}
return 0 ; /* Don't copy from parent */
} else {
return 1 ; /* Copy from parent */
}
}
2009-07-27 22:11:53 +08:00
/* }}} */
2003-02-04 20:12:34 +08:00
2009-07-27 22:11:53 +08:00
static inline void do_implement_interface ( zend_class_entry * ce , zend_class_entry * iface TSRMLS_DC ) /* { { { */
2003-05-30 03:00:40 +08:00
{
2003-12-07 03:02:12 +08:00
if ( ! ( ce - > ce_flags & ZEND_ACC_INTERFACE ) & & iface - > interface_gets_implemented & & iface - > interface_gets_implemented ( iface , ce TSRMLS_CC ) = = FAILURE ) {
2003-10-23 03:59:58 +08:00
zend_error ( E_CORE_ERROR , " Class %s could not implement interface %s " , ce - > name , iface - > name ) ;
}
2004-12-06 19:53:30 +08:00
if ( ce = = iface ) {
2008-01-29 08:07:26 +08:00
zend_error ( E_ERROR , " Interface %s cannot implement itself " , ce - > name ) ;
2004-12-06 19:53:30 +08:00
}
2003-10-23 03:59:58 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
2003-05-30 03:00:40 +08:00
2009-07-27 22:11:53 +08:00
ZEND_API void zend_do_inherit_interfaces ( zend_class_entry * ce , const zend_class_entry * iface TSRMLS_DC ) /* { { { */
2003-10-23 03:59:58 +08:00
{
/* expects interface to be contained in ce's interface list already */
2004-09-27 16:46:12 +08:00
zend_uint i , ce_num , if_num = iface - > num_interfaces ;
2003-10-23 03:59:58 +08:00
zend_class_entry * entry ;
2004-01-28 17:13:41 +08:00
if ( if_num = = 0 ) {
return ;
}
ce_num = ce - > num_interfaces ;
2003-10-23 03:59:58 +08:00
2004-01-28 17:13:41 +08:00
if ( ce - > type = = ZEND_INTERNAL_CLASS ) {
ce - > interfaces = ( zend_class_entry * * ) realloc ( ce - > interfaces , sizeof ( zend_class_entry * ) * ( ce_num + if_num ) ) ;
} else {
ce - > interfaces = ( zend_class_entry * * ) erealloc ( ce - > interfaces , sizeof ( zend_class_entry * ) * ( ce_num + if_num ) ) ;
}
/* Inherit the interfaces, only if they're not already inherited by the class */
while ( if_num - - ) {
entry = iface - > interfaces [ if_num ] ;
for ( i = 0 ; i < ce_num ; i + + ) {
if ( ce - > interfaces [ i ] = = entry ) {
break ;
2003-10-23 03:59:58 +08:00
}
}
2004-01-28 17:13:41 +08:00
if ( i = = ce_num ) {
ce - > interfaces [ ce - > num_interfaces + + ] = entry ;
2003-05-30 03:00:40 +08:00
}
}
2004-01-28 17:13:41 +08:00
/* and now call the implementing handlers */
while ( ce_num < ce - > num_interfaces ) {
do_implement_interface ( ce , ce - > interfaces [ ce_num + + ] TSRMLS_CC ) ;
}
2003-05-30 03:00:40 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
2003-05-30 03:00:40 +08:00
2009-07-27 22:11:53 +08:00
static int inherit_static_prop ( zval * * p TSRMLS_DC , int num_args , va_list args , const zend_hash_key * key ) /* { { { */
2003-09-04 00:13:40 +08:00
{
2006-12-06 03:03:51 +08:00
HashTable * target = va_arg ( args , HashTable * ) ;
2003-09-04 00:13:40 +08:00
2006-12-06 03:03:51 +08:00
if ( ! zend_hash_quick_exists ( target , key - > arKey , key - > nKeyLength , key - > h ) ) {
SEPARATE_ZVAL_TO_MAKE_IS_REF ( p ) ;
if ( zend_hash_quick_add ( target , key - > arKey , key - > nKeyLength , key - > h , p , sizeof ( zval * ) , NULL ) = = SUCCESS ) {
2007-10-07 13:22:07 +08:00
Z_ADDREF_PP ( p ) ;
2006-12-06 03:03:51 +08:00
}
}
return ZEND_HASH_APPLY_KEEP ;
}
2009-07-27 22:11:53 +08:00
/* }}} */
2003-09-04 00:13:40 +08:00
2009-07-27 22:11:53 +08:00
ZEND_API void zend_do_inheritance ( zend_class_entry * ce , zend_class_entry * parent_ce TSRMLS_DC ) /* { { { */
2000-06-09 22:40:14 +08:00
{
2003-03-05 19:14:44 +08:00
if ( ( ce - > ce_flags & ZEND_ACC_INTERFACE )
& & ! ( parent_ce - > ce_flags & ZEND_ACC_INTERFACE ) ) {
2003-09-03 16:39:43 +08:00
zend_error ( E_COMPILE_ERROR , " Interface %s may not inherit from class (%s) " , ce - > name , parent_ce - > name ) ;
2003-03-05 19:14:44 +08:00
}
2003-06-22 05:56:06 +08:00
if ( parent_ce - > ce_flags & ZEND_ACC_FINAL_CLASS ) {
2003-09-03 16:39:43 +08:00
zend_error ( E_COMPILE_ERROR , " Class %s may not inherit from final class (%s) " , ce - > name , parent_ce - > name ) ;
2003-06-22 05:56:06 +08:00
}
2003-03-05 19:14:44 +08:00
2003-02-04 20:12:34 +08:00
ce - > parent = parent_ce ;
2008-08-25 02:22:33 +08:00
/* Copy serialize/unserialize callbacks */
if ( ! ce - > serialize ) {
ce - > serialize = parent_ce - > serialize ;
}
if ( ! ce - > unserialize ) {
ce - > unserialize = parent_ce - > unserialize ;
}
2003-05-30 03:00:40 +08:00
/* Inherit interfaces */
2003-10-23 03:59:58 +08:00
zend_do_inherit_interfaces ( ce , parent_ce TSRMLS_CC ) ;
2000-06-09 22:40:14 +08:00
2003-02-04 20:12:34 +08:00
/* Inherit properties */
zend_hash_merge ( & ce - > default_properties , & parent_ce - > default_properties , ( void ( * ) ( void * ) ) zval_add_ref , NULL , sizeof ( zval * ) , 0 ) ;
2005-09-01 18:05:32 +08:00
if ( parent_ce - > type ! = ce - > type ) {
/* User class extends internal class */
zend_update_class_constants ( parent_ce TSRMLS_CC ) ;
2008-07-25 03:52:24 +08:00
zend_hash_apply_with_arguments ( CE_STATIC_MEMBERS ( parent_ce ) TSRMLS_CC , ( apply_func_args_t ) inherit_static_prop , 1 , & ce - > default_static_members ) ;
2005-09-01 18:05:32 +08:00
} else {
2008-08-22 23:54:21 +08:00
zend_hash_apply_with_arguments ( & parent_ce - > default_static_members TSRMLS_CC , ( apply_func_args_t ) inherit_static_prop , 1 , & ce - > default_static_members ) ;
2005-09-01 18:05:32 +08:00
}
2003-08-24 03:37:39 +08:00
zend_hash_merge_ex ( & ce - > properties_info , & parent_ce - > properties_info , ( copy_ctor_func_t ) ( ce - > type & ZEND_INTERNAL_CLASS ? zend_duplicate_property_info_internal : zend_duplicate_property_info ) , sizeof ( zend_property_info ) , ( merge_checker_func_t ) do_inherit_property_access_check , ce ) ;
2002-07-16 02:09:56 +08:00
2003-02-04 20:12:34 +08:00
zend_hash_merge ( & ce - > constants_table , & parent_ce - > constants_table , ( void ( * ) ( void * ) ) zval_add_ref , NULL , sizeof ( zval * ) , 0 ) ;
2003-02-23 20:51:40 +08:00
zend_hash_merge_ex ( & ce - > function_table , & parent_ce - > function_table , ( copy_ctor_func_t ) do_inherit_method , sizeof ( zend_function ) , ( merge_checker_func_t ) do_inherit_method_check , ce ) ;
2000-06-09 22:40:14 +08:00
do_inherit_parent_constructor ( ce ) ;
2006-05-12 05:07:39 +08:00
if ( ce - > ce_flags & ZEND_ACC_IMPLICIT_ABSTRACT_CLASS & & ce - > type = = ZEND_INTERNAL_CLASS ) {
ce - > ce_flags | = ZEND_ACC_EXPLICIT_ABSTRACT_CLASS ;
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
} else if ( ! ( ce - > ce_flags & ZEND_ACC_IMPLEMENT_INTERFACES )
& & ! ( ce - > ce_flags & ZEND_ACC_IMPLEMENT_TRAITS ) ) {
2008-03-12 17:46:42 +08:00
/* The verification will be done in runtime by ZEND_VERIFY_ABSTRACT_CLASS */
2006-05-12 05:07:39 +08:00
zend_verify_abstract_class ( ce TSRMLS_CC ) ;
}
2000-06-09 22:40:14 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
2000-06-09 22:40:14 +08:00
2009-07-27 22:11:53 +08:00
static zend_bool do_inherit_constant_check ( HashTable * child_constants_table , const zval * * parent_constant , const zend_hash_key * hash_key , const zend_class_entry * iface ) /* { { { */
2004-01-28 19:53:52 +08:00
{
2005-11-23 17:26:43 +08:00
zval * * old_constant ;
if ( zend_hash_quick_find ( child_constants_table , hash_key - > arKey , hash_key - > nKeyLength , hash_key - > h , ( void * * ) & old_constant ) = = SUCCESS ) {
2009-12-03 20:34:50 +08:00
if ( * old_constant ! = * parent_constant ) {
zend_error ( E_COMPILE_ERROR , " Cannot inherit previously-inherited or override constant %s from interface %s " , hash_key - > arKey , iface - > name ) ;
2005-11-23 17:26:43 +08:00
}
2006-05-10 07:53:23 +08:00
return 0 ;
2004-01-28 19:53:52 +08:00
}
return 1 ;
}
2009-07-27 22:11:53 +08:00
/* }}} */
2004-01-28 19:53:52 +08:00
2009-12-03 20:34:50 +08:00
static int do_interface_constant_check ( zval * * val TSRMLS_DC , int num_args , va_list args , const zend_hash_key * key ) /* { { { */
{
zend_class_entry * * iface = va_arg ( args , zend_class_entry * * ) ;
do_inherit_constant_check ( & ( * iface ) - > constants_table , ( const zval * * ) val , key , * iface ) ;
return ZEND_HASH_APPLY_KEEP ;
}
/* }}} */
2009-07-27 22:11:53 +08:00
ZEND_API void zend_do_implement_interface ( zend_class_entry * ce , zend_class_entry * iface TSRMLS_DC ) /* { { { */
2003-03-05 19:14:44 +08:00
{
2006-06-07 17:21:06 +08:00
zend_uint i , ignore = 0 ;
zend_uint current_iface_num = ce - > num_interfaces ;
zend_uint parent_iface_num = ce - > parent ? ce - > parent - > num_interfaces : 0 ;
for ( i = 0 ; i < ce - > num_interfaces ; i + + ) {
if ( ce - > interfaces [ i ] = = NULL ) {
memmove ( ce - > interfaces + i , ce - > interfaces + i + 1 , sizeof ( zend_class_entry * ) * ( - - ce - > num_interfaces - i ) ) ;
i - - ;
} else if ( ce - > interfaces [ i ] = = iface ) {
if ( i < parent_iface_num ) {
ignore = 1 ;
} else {
zend_error ( E_COMPILE_ERROR , " Class %s cannot implement previously implemented interface %s " , ce - > name , iface - > name ) ;
}
}
}
2009-12-03 20:34:50 +08:00
if ( ignore ) {
/* Check for attempt to redeclare interface constants */
zend_hash_apply_with_arguments ( & ce - > constants_table TSRMLS_CC , ( apply_func_args_t ) do_interface_constant_check , 1 , & iface ) ;
} else {
2006-06-07 17:21:06 +08:00
if ( ce - > num_interfaces > = current_iface_num ) {
if ( ce - > type = = ZEND_INTERNAL_CLASS ) {
ce - > interfaces = ( zend_class_entry * * ) realloc ( ce - > interfaces , sizeof ( zend_class_entry * ) * ( + + current_iface_num ) ) ;
} else {
ce - > interfaces = ( zend_class_entry * * ) erealloc ( ce - > interfaces , sizeof ( zend_class_entry * ) * ( + + current_iface_num ) ) ;
}
}
ce - > interfaces [ ce - > num_interfaces + + ] = iface ;
zend_hash_merge_ex ( & ce - > constants_table , & iface - > constants_table , ( copy_ctor_func_t ) zval_add_ref , sizeof ( zval * ) , ( merge_checker_func_t ) do_inherit_constant_check , iface ) ;
zend_hash_merge_ex ( & ce - > function_table , & iface - > function_table , ( copy_ctor_func_t ) do_inherit_method , sizeof ( zend_function ) , ( merge_checker_func_t ) do_inherit_method_check , ce ) ;
do_implement_interface ( ce , iface TSRMLS_CC ) ;
zend_do_inherit_interfaces ( ce , iface TSRMLS_CC ) ;
}
2003-03-05 19:14:44 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
2003-03-05 19:14:44 +08:00
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
ZEND_API void zend_do_implement_trait ( zend_class_entry * ce , zend_class_entry * trait TSRMLS_DC ) /* { { { */
{
zend_uint i , ignore = 0 ;
zend_uint current_trait_num = ce - > num_traits ;
zend_uint parent_trait_num = ce - > parent ? ce - > parent - > num_traits : 0 ;
for ( i = 0 ; i < ce - > num_traits ; i + + ) {
if ( ce - > traits [ i ] = = NULL ) {
memmove ( ce - > traits + i , ce - > traits + i + 1 , sizeof ( zend_class_entry * ) * ( - - ce - > num_traits - i ) ) ;
i - - ;
} else if ( ce - > traits [ i ] = = trait ) {
if ( i < parent_trait_num ) {
ignore = 1 ;
}
}
}
if ( ! ignore ) {
if ( ce - > num_traits > = current_trait_num ) {
if ( ce - > type = = ZEND_INTERNAL_CLASS ) {
ce - > traits = ( zend_class_entry * * ) realloc ( ce - > traits , sizeof ( zend_class_entry * ) * ( + + current_trait_num ) ) ;
} else {
ce - > traits = ( zend_class_entry * * ) erealloc ( ce - > traits , sizeof ( zend_class_entry * ) * ( + + current_trait_num ) ) ;
}
}
ce - > traits [ ce - > num_traits + + ] = trait ;
}
}
/* }}} */
2010-04-23 09:43:27 +08:00
static int _merge_functions ( zend_function * fn TSRMLS_DC , int num_args , va_list args , zend_hash_key * hash_key ) /* { { { */
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
{
size_t current ;
size_t i ;
size_t count ;
HashTable * resulting_table ;
HashTable * * function_tables ;
zend_class_entry * ce ;
size_t collision = 0 ;
size_t abstract_solved = 0 ;
2010-04-23 07:16:15 +08:00
/* unsigned int name_len; */
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
zend_function * other_trait_fn ;
2010-04-23 07:16:15 +08:00
current = va_arg ( args , size_t ) ; /* index of current trait */
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
count = va_arg ( args , size_t ) ;
resulting_table = va_arg ( args , HashTable * ) ;
function_tables = va_arg ( args , HashTable * * ) ;
ce = va_arg ( args , zend_class_entry * ) ;
for ( i = 0 ; i < count ; i + + ) {
if ( i = = current ) {
2010-04-23 07:16:15 +08:00
continue ; /* just skip this, cause its the table this function is applied on */
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
}
2010-05-03 01:39:24 +08:00
if ( zend_hash_quick_find ( function_tables [ i ] , hash_key - > arKey , hash_key - > nKeyLength , hash_key - > h , ( void * * ) & other_trait_fn ) = = SUCCESS ) {
2010-04-23 07:16:15 +08:00
/* if it is an abstract method, there is no collision */
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
if ( other_trait_fn - > common . fn_flags & ZEND_ACC_ABSTRACT ) {
2010-04-23 07:16:15 +08:00
/* we can savely free and remove it from other table */
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
zend_function_dtor ( other_trait_fn ) ;
2010-05-03 01:39:24 +08:00
zend_hash_quick_del ( function_tables [ i ] , hash_key - > arKey , hash_key - > nKeyLength , hash_key - > h ) ;
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
} else {
2010-04-23 07:16:15 +08:00
/* if it is not an abstract method, there is still no collision */
/* if fn is an abstract method */
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
if ( fn - > common . fn_flags & ZEND_ACC_ABSTRACT ) {
2010-04-23 07:16:15 +08:00
/* just mark as solved, will be added if its own trait is processed */
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
abstract_solved = 1 ;
} else {
2010-04-23 07:16:15 +08:00
/* but else, we have a collision of non-abstract methods */
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
collision + + ;
zend_function_dtor ( other_trait_fn ) ;
2010-05-03 01:39:24 +08:00
zend_hash_quick_del ( function_tables [ i ] , hash_key - > arKey , hash_key - > nKeyLength , hash_key - > h ) ;
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
}
}
}
}
if ( collision ) {
zend_function * class_fn ;
2010-04-23 07:16:15 +08:00
/* make sure method is not already overridden in class */
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
2010-05-03 01:39:24 +08:00
if ( zend_hash_quick_find ( & ce - > function_table , hash_key - > arKey , hash_key - > nKeyLength , hash_key - > h , ( void * * ) & class_fn ) = = FAILURE
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
| | class_fn - > common . scope ! = ce ) {
zend_error ( E_WARNING , " Trait method %s has not been applied, because there are collisions with other trait methods on %s " ,
fn - > common . function_name , ce - > name ) ;
}
zend_function_dtor ( fn ) ;
} else if ( abstract_solved ) {
zend_function_dtor ( fn ) ;
} else {
2010-04-23 07:16:15 +08:00
/* Add it to result function table */
2010-05-03 01:39:24 +08:00
if ( zend_hash_quick_add ( resulting_table , hash_key - > arKey , hash_key - > nKeyLength , hash_key - > h , fn , sizeof ( zend_function ) , NULL ) = = FAILURE ) {
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
zend_error ( E_ERROR , " Trait method %s has not been applied, because failure occured during updating resulting trait method table. " ,
fn - > common . function_name ) ;
}
}
return ZEND_HASH_APPLY_REMOVE ;
}
/* }}} */
# define IS_EQUAL(mname, mname_len, str) \
strncmp ( mname , str , mname_len )
# define _ADD_MAGIC_METHOD(ce, mname, mname_len, fe) { \
if ( IS_EQUAL ( mname , mname_len , " __clone " ) ) { ( ce ) - > clone = ( fe ) ; ( fe ) - > common . fn_flags = ZEND_ACC_CLONE ; } \
else if ( IS_EQUAL ( mname , mname_len , " __get " ) ) ( ce ) - > __get = ( fe ) ; \
else if ( IS_EQUAL ( mname , mname_len , " __set " ) ) ( ce ) - > __set = ( fe ) ; \
else if ( IS_EQUAL ( mname , mname_len , " __call " ) ) ( ce ) - > __call = ( fe ) ; \
else if ( IS_EQUAL ( mname , mname_len , " __unset " ) ) ( ce ) - > __unset = ( fe ) ; \
else if ( IS_EQUAL ( mname , mname_len , " __isset " ) ) ( ce ) - > __isset = ( fe ) ; \
else if ( IS_EQUAL ( mname , mname_len , " __callstatic " ) ) ( ce ) - > __callstatic = ( fe ) ; \
else if ( IS_EQUAL ( mname , mname_len , " __tostring " ) ) ( ce ) - > __tostring = ( fe ) ; \
else if ( IS_EQUAL ( mname , mname_len , " serialize_func " ) ) ( ce ) - > serialize_func = ( fe ) ; \
else if ( IS_EQUAL ( mname , mname_len , " unserialize_func " ) ) ( ce ) - > unserialize_func = ( fe ) ; \
}
2010-05-03 00:32:25 +08:00
/* {{{ Originates from php_runkit_function_copy_ctor
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
Duplicate structures in an op_array where necessary to make an outright duplicate */
2010-05-03 00:32:25 +08:00
void _duplicate_function ( zend_function * fe , char * newname )
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
{
2010-04-23 09:56:03 +08:00
zend_literal * literals_copy ;
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
zend_compiled_variable * dupvars ;
zend_op * opcode_copy ;
int i ;
if ( fe - > op_array . static_variables ) {
HashTable * tmpHash ;
zval tmpZval ;
ALLOC_HASHTABLE ( tmpHash ) ;
zend_hash_init ( tmpHash , 2 , NULL , ZVAL_PTR_DTOR , 0 ) ;
zend_hash_copy ( tmpHash , fe - > op_array . static_variables , ZVAL_COPY_CTOR , & tmpZval , sizeof ( zval ) ) ;
fe - > op_array . static_variables = tmpHash ;
}
fe - > op_array . refcount = emalloc ( sizeof ( zend_uint ) ) ;
* ( fe - > op_array . refcount ) = 1 ;
i = fe - > op_array . last_var ;
dupvars = safe_emalloc ( fe - > op_array . last_var , sizeof ( zend_compiled_variable ) , 0 ) ;
while ( i > 0 ) {
i - - ;
dupvars [ i ] . name = estrndup ( fe - > op_array . vars [ i ] . name , fe - > op_array . vars [ i ] . name_len ) ;
dupvars [ i ] . name_len = fe - > op_array . vars [ i ] . name_len ;
dupvars [ i ] . hash_value = fe - > op_array . vars [ i ] . hash_value ;
}
fe - > op_array . vars = dupvars ;
opcode_copy = safe_emalloc ( sizeof ( zend_op ) , fe - > op_array . last , 0 ) ;
for ( i = 0 ; i < fe - > op_array . last ; i + + ) {
opcode_copy [ i ] = fe - > op_array . opcodes [ i ] ;
2010-04-23 10:57:49 +08:00
if ( opcode_copy [ i ] . op1_type ! = IS_CONST ) {
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
if ( opcode_copy [ i ] . op1 . jmp_addr > = fe - > op_array . opcodes & &
opcode_copy [ i ] . op1 . jmp_addr < fe - > op_array . opcodes + fe - > op_array . last ) {
opcode_copy [ i ] . op1 . jmp_addr = opcode_copy + ( fe - > op_array . opcodes [ i ] . op1 . jmp_addr - fe - > op_array . opcodes ) ;
}
}
2010-04-23 10:57:49 +08:00
if ( opcode_copy [ i ] . op2_type ! = IS_CONST ) {
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
if ( opcode_copy [ i ] . op2 . jmp_addr > = fe - > op_array . opcodes & &
opcode_copy [ i ] . op2 . jmp_addr < fe - > op_array . opcodes + fe - > op_array . last ) {
opcode_copy [ i ] . op2 . jmp_addr = opcode_copy + ( fe - > op_array . opcodes [ i ] . op2 . jmp_addr - fe - > op_array . opcodes ) ;
}
}
}
fe - > op_array . opcodes = opcode_copy ;
fe - > op_array . start_op = fe - > op_array . opcodes ;
fe - > op_array . function_name = newname ;
2010-04-23 07:16:15 +08:00
/* was setting it to fe which does not work since fe is stack allocated and not a stable address */
/* fe->op_array.prototype = fe->op_array.prototype; */
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
if ( fe - > op_array . arg_info ) {
zend_arg_info * tmpArginfo ;
tmpArginfo = safe_emalloc ( sizeof ( zend_arg_info ) , fe - > op_array . num_args , 0 ) ;
for ( i = 0 ; i < fe - > op_array . num_args ; i + + ) {
tmpArginfo [ i ] = fe - > op_array . arg_info [ i ] ;
tmpArginfo [ i ] . name = estrndup ( tmpArginfo [ i ] . name , tmpArginfo [ i ] . name_len ) ;
if ( tmpArginfo [ i ] . class_name ) {
tmpArginfo [ i ] . class_name = estrndup ( tmpArginfo [ i ] . class_name , tmpArginfo [ i ] . class_name_len ) ;
}
}
fe - > op_array . arg_info = tmpArginfo ;
}
2010-04-23 09:56:03 +08:00
fe - > op_array . doc_comment = estrndup ( fe - > op_array . doc_comment , fe - > op_array . doc_comment_len ) ;
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
fe - > op_array . try_catch_array = ( zend_try_catch_element * ) estrndup ( ( char * ) fe - > op_array . try_catch_array , sizeof ( zend_try_catch_element ) * fe - > op_array . last_try_catch ) ;
fe - > op_array . brk_cont_array = ( zend_brk_cont_element * ) estrndup ( ( char * ) fe - > op_array . brk_cont_array , sizeof ( zend_brk_cont_element ) * fe - > op_array . last_brk_cont ) ;
2010-04-23 08:54:51 +08:00
/* TODO: check whether there is something similar and whether that is ok */
2010-04-23 09:56:03 +08:00
literals_copy = ( zend_literal * ) emalloc ( fe - > op_array . size_literal * sizeof ( zend_literal ) ) ;
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
2010-04-23 08:54:51 +08:00
for ( i = 0 ; i < fe - > op_array . size_literal ; i + + ) {
literals_copy [ i ] = fe - > op_array . literals [ i ] ;
2010-04-23 10:57:49 +08:00
zval_copy_ctor ( & literals_copy [ i ] . constant ) ;
2010-04-23 08:54:51 +08:00
}
fe - > op_array . literals = literals_copy ;
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
}
/* }}}} */
2010-04-23 09:43:27 +08:00
static int _merge_functions_to_class ( zend_function * fn TSRMLS_DC , int num_args , va_list args , zend_hash_key * hash_key )
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
{
zend_class_entry * ce = va_arg ( args , zend_class_entry * ) ;
int add = 0 ;
zend_function * existing_fn ;
2010-04-23 07:16:15 +08:00
zend_function * prototype = NULL ; /* is used to determine the prototype according to the inheritance chain */
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
if ( zend_hash_quick_find ( & ce - > function_table , hash_key - > arKey , hash_key - > nKeyLength , hash_key - > h , ( void * * ) & existing_fn ) = = FAILURE ) {
2010-04-23 07:16:15 +08:00
add = 1 ; /* not found */
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
} else if ( existing_fn - > common . scope ! = ce ) {
2010-04-23 07:16:15 +08:00
add = 1 ; /* or inherited from other class or interface */
/* it is just a reference which was added to the subclass while doing the inheritance */
/* prototype = existing_fn; */
/* memory is scrambled anyway???? */
/* function_add_ref(prototype); */
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
zend_hash_quick_del ( & ce - > function_table , hash_key - > arKey , hash_key - > nKeyLength , hash_key - > h ) ;
}
if ( add ) {
zend_function * parent_function ;
if ( ce - > parent & & zend_hash_quick_find ( & ce - > parent - > function_table , hash_key - > arKey , hash_key - > nKeyLength , hash_key - > h , ( void * * ) & parent_function ) ! = FAILURE ) {
2010-04-23 07:16:15 +08:00
prototype = parent_function ; /* ->common.fn_flags |= ZEND_ACC_ABSTRACT; */
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
}
fn - > common . scope = ce ;
fn - > common . prototype = prototype ;
if ( prototype
& & ( prototype - > common . fn_flags & ZEND_ACC_IMPLEMENTED_ABSTRACT
| | prototype - > common . fn_flags & ZEND_ACC_ABSTRACT ) ) {
fn - > common . fn_flags | = ZEND_ACC_IMPLEMENTED_ABSTRACT ;
2010-04-23 07:16:15 +08:00
} else if ( fn - > common . fn_flags & ZEND_ACC_IMPLEMENTED_ABSTRACT ) {
/* remove ZEND_ACC_IMPLEMENTED_ABSTRACT flag, think it shouldn't be copied to class */
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
fn - > common . fn_flags = fn - > common . fn_flags - ZEND_ACC_IMPLEMENTED_ABSTRACT ;
}
2010-04-23 07:16:15 +08:00
/* check whether the trait method fullfills the inheritance requirements */
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
if ( prototype ) {
2010-04-23 07:16:15 +08:00
do_inheritance_check_on_method ( fn , prototype TSRMLS_CC ) ;
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
}
if ( fn - > common . fn_flags & ZEND_ACC_ABSTRACT ) {
ce - > ce_flags | = ZEND_ACC_IMPLICIT_ABSTRACT_CLASS ;
}
2010-05-03 01:39:24 +08:00
if ( zend_hash_quick_update ( & ce - > function_table , hash_key - > arKey , hash_key - > nKeyLength , hash_key - > h , fn , sizeof ( zend_function ) , NULL ) = = FAILURE ) {
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
zend_error ( E_ERROR , " Trait method %s has not been applied, because failure occured during updating class method table. " , hash_key - > arKey ) ;
}
_ADD_MAGIC_METHOD ( ce , hash_key - > arKey , hash_key - > nKeyLength , fn ) ;
2010-04-23 07:16:15 +08:00
/* it could be necessary to update child classes as well */
2010-04-23 07:25:05 +08:00
/* zend_hash_apply_with_arguments(EG(class_table) TSRMLS_CC, (apply_func_args_t)php_runkit_update_children_methods, 5, dce, dce, &dfe, dfunc, dfunc_len); */
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
} else {
zend_function_dtor ( fn ) ;
2010-04-23 07:16:15 +08:00
/* efree(fn); */
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
}
return ZEND_HASH_APPLY_REMOVE ;
}
2010-04-23 08:54:51 +08:00
static int _copy_functions ( zend_function * fn TSRMLS_DC , int num_args , va_list args , zend_hash_key * hash_key )
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
{
HashTable * target ;
zend_trait_alias * * aliases ;
HashTable * exclude_table ;
char * lcname ;
unsigned int lcname_len ;
unsigned int fnname_len ;
zend_function fn_copy ;
2010-04-23 09:56:03 +08:00
void * dummy ;
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
size_t i = 0 ;
target = va_arg ( args , HashTable * ) ;
aliases = va_arg ( args , zend_trait_alias * * ) ;
exclude_table = va_arg ( args , HashTable * ) ;
fnname_len = strlen ( fn - > common . function_name ) ;
2010-04-23 07:16:15 +08:00
/* apply aliases which are qualified with a class name, there should not be any ambiguatty */
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
if ( aliases ) {
while ( aliases [ i ] ) {
if ( fn - > common . scope = = aliases [ i ] - > trait_method - > ce & &
( zend_binary_strcasecmp ( aliases [ i ] - > trait_method - > method_name ,
aliases [ i ] - > trait_method - > mname_len ,
fn - > common . function_name , fnname_len ) = = 0 ) ) {
if ( aliases [ i ] - > alias ) {
fn_copy = * fn ;
2010-05-03 00:32:25 +08:00
_duplicate_function ( & fn_copy , estrndup ( aliases [ i ] - > alias , aliases [ i ] - > alias_len ) ) ;
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
2010-04-23 07:16:15 +08:00
if ( aliases [ i ] - > modifiers ) { /* if it is 0, no modifieres has been changed */
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
fn_copy . common . fn_flags = aliases [ i ] - > modifiers ;
if ( ! ( aliases [ i ] - > modifiers & ZEND_ACC_PPP_MASK ) ) {
fn_copy . common . fn_flags | = ZEND_ACC_PUBLIC ;
}
}
lcname_len = aliases [ i ] - > alias_len ;
lcname = zend_str_tolower_dup ( aliases [ i ] - > alias , lcname_len ) ;
if ( zend_hash_add ( target , lcname , lcname_len + 1 , & fn_copy , sizeof ( zend_function ) , NULL ) = = FAILURE ) {
zend_error ( E_ERROR , " Failed to added aliased trait method (%s) to trait table. Propably there is already a trait method with same name \n " ,
fn_copy . common . function_name ) ;
}
2010-04-23 07:16:15 +08:00
/* aliases[i]->function = fn_copy; */
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
efree ( lcname ) ;
}
}
i + + ;
}
}
lcname = zend_str_tolower_dup ( fn - > common . function_name , fnname_len ) ;
2010-04-23 09:56:03 +08:00
2010-05-03 01:39:24 +08:00
if ( zend_hash_find ( exclude_table , lcname , fnname_len , & dummy ) = = FAILURE ) {
2010-04-23 07:16:15 +08:00
/* is not in hashtable, thus, function is not to be excluded */
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
fn_copy = * fn ;
2010-05-03 00:32:25 +08:00
_duplicate_function ( & fn_copy , estrndup ( fn - > common . function_name , fnname_len ) ) ;
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
2010-04-23 07:16:15 +08:00
/* apply aliases which are not qualified by a class name, or which have not alias name, just setting visibility */
/* TODO: i am still not sure, that there will be no ambigousities... */
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
if ( aliases ) {
i = 0 ;
while ( aliases [ i ] ) {
if ( ( ! aliases [ i ] - > trait_method - > ce | | fn - > common . scope = = aliases [ i ] - > trait_method - > ce ) & &
( zend_binary_strcasecmp ( aliases [ i ] - > trait_method - > method_name ,
aliases [ i ] - > trait_method - > mname_len ,
fn - > common . function_name , fnname_len ) = = 0 ) ) {
if ( aliases [ i ] - > alias ) {
2010-04-23 09:56:03 +08:00
zend_uint lcname2_len ;
char * lcname2 ;
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
zend_function fn_copy2 = * fn ;
2010-05-03 02:47:27 +08:00
2010-05-03 00:32:25 +08:00
_duplicate_function ( & fn_copy2 , estrndup ( aliases [ i ] - > alias , aliases [ i ] - > alias_len ) ) ;
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
2010-04-23 07:16:15 +08:00
if ( aliases [ i ] - > modifiers ) { /* if it is 0, no modifieres has been changed */
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
fn_copy2 . common . fn_flags = aliases [ i ] - > modifiers ;
if ( ! ( aliases [ i ] - > modifiers & ZEND_ACC_PPP_MASK ) ) {
fn_copy2 . common . fn_flags | = ZEND_ACC_PUBLIC ;
}
}
2010-04-23 09:56:03 +08:00
lcname2_len = aliases [ i ] - > alias_len ;
lcname2 = zend_str_tolower_dup ( aliases [ i ] - > alias , lcname2_len ) ;
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
if ( zend_hash_add ( target , lcname2 , lcname2_len + 1 , & fn_copy2 , sizeof ( zend_function ) , NULL ) = = FAILURE ) {
zend_error ( E_ERROR , " Failed to added aliased trait method (%s) to trait table. Propably there is already a trait method with same name \n " ,
fn_copy2 . common . function_name ) ;
}
efree ( lcname2 ) ;
} else {
2010-04-23 07:16:15 +08:00
if ( aliases [ i ] - > modifiers ) { /* if it is 0, no modifieres has been changed */
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
fn_copy . common . fn_flags = aliases [ i ] - > modifiers ;
if ( ! ( aliases [ i ] - > modifiers & ZEND_ACC_PPP_MASK ) ) {
fn_copy . common . fn_flags | = ZEND_ACC_PUBLIC ;
}
}
}
}
i + + ;
}
}
2010-05-03 01:39:24 +08:00
if ( zend_hash_add ( target , lcname , fnname_len + 1 , & fn_copy , sizeof ( zend_function ) , NULL ) = = FAILURE ) {
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
zend_error ( E_ERROR , " Failed to added trait method (%s) to trait table. Propably there is already a trait method with same name \n " ,
fn_copy . common . function_name ) ;
}
}
efree ( lcname ) ;
return ZEND_HASH_APPLY_KEEP ;
}
/**
* Copies function table entries to target function table with applied aliasing
*/
2010-04-23 07:25:05 +08:00
void copy_trait_function_table ( HashTable * target , HashTable * source , zend_trait_alias * * aliases , HashTable * exclude_table TSRMLS_DC ) {
zend_hash_apply_with_arguments ( source TSRMLS_CC , ( apply_func_args_t ) _copy_functions , 3 , /* 3 is number of args for apply_func */
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
target , aliases , exclude_table ) ;
}
void init_trait_structures ( zend_class_entry * ce TSRMLS_DC ) /* { { { */
{
size_t i , j = 0 ;
zend_trait_precedence * cur_precedence ;
zend_trait_method_reference * cur_method_ref ;
zend_class_entry * cur_ce ;
2010-04-23 07:16:15 +08:00
/* resolve class references */
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
if ( ce - > trait_precedences ) {
i = 0 ;
while ( ( cur_precedence = ce - > trait_precedences [ i ] ) ) {
cur_ce = zend_fetch_class ( cur_precedence - > trait_method - > class_name , cur_precedence - > trait_method - > cname_len , ZEND_FETCH_CLASS_TRAIT TSRMLS_CC ) ;
cur_precedence - > trait_method - > ce = cur_ce ;
if ( cur_precedence - > exclude_from_classes ) {
j = 0 ;
while ( cur_precedence - > exclude_from_classes [ j ] ) {
char * class_name = ( char * ) cur_precedence - > exclude_from_classes [ j ] ;
2010-05-03 02:47:27 +08:00
zend_uint name_length = strlen ( class_name ) ;
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
cur_ce = zend_fetch_class ( class_name , name_length , ZEND_FETCH_CLASS_TRAIT TSRMLS_CC ) ;
efree ( class_name ) ;
cur_precedence - > exclude_from_classes [ j ] = cur_ce ;
j + + ;
}
}
i + + ;
}
}
if ( ce - > trait_aliases ) {
i = 0 ;
while ( ce - > trait_aliases [ i ] ) {
cur_method_ref = ce - > trait_aliases [ i ] - > trait_method ;
if ( cur_method_ref - > class_name ) {
cur_method_ref - > ce = zend_fetch_class ( cur_method_ref - > class_name , cur_method_ref - > cname_len , ZEND_FETCH_CLASS_TRAIT TSRMLS_CC ) ;
}
i + + ;
}
}
}
/* }}} */
void compile_exclude_table ( HashTable * exclude_table , zend_trait_precedence * * precedences , zend_class_entry * trait ) {
size_t i , j ;
if ( precedences ) {
i = 0 ;
while ( precedences [ i ] ) {
if ( precedences [ i ] - > exclude_from_classes ) {
j = 0 ;
while ( precedences [ i ] - > exclude_from_classes [ j ] ) {
if ( precedences [ i ] - > exclude_from_classes [ j ] = = trait ) {
zend_uint lcname_len = precedences [ i ] - > trait_method - > mname_len ;
char * lcname = zend_str_tolower_dup ( precedences [ i ] - > trait_method - > method_name ,
lcname_len ) ;
if ( zend_hash_add ( exclude_table , lcname , lcname_len , NULL , 0 , NULL ) = = FAILURE ) {
zend_error ( E_ERROR , " Failed to evaluate a trait precedence (%s). Method of trait %s was defined to be excluded multiple times. \n " ,
precedences [ i ] - > trait_method - > method_name , trait - > name ) ;
}
efree ( lcname ) ;
}
j + + ;
}
}
i + + ;
}
}
}
ZEND_API void zend_do_bind_traits ( zend_class_entry * ce TSRMLS_DC ) /* { { { */
{
HashTable * * function_tables ;
HashTable * resulting_table ;
HashTable exclude_table ;
size_t i ;
2010-05-03 01:39:24 +08:00
if ( ce - > num_traits < = 0 ) {
return ;
}
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
2010-04-23 07:16:15 +08:00
/* zend_error(E_NOTICE, "Do bind Traits on %v with %d traits.\n Class has already %d methods.\n",
ce - > name . s , ce - > num_traits , ce - > function_table . nNumOfElements ) ; */
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
2010-04-23 07:16:15 +08:00
/* complete initialization of trait strutures in ce */
init_trait_structures ( ce TSRMLS_CC ) ;
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
2010-04-23 07:16:15 +08:00
/* prepare copies of trait function tables for combination */
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
function_tables = malloc ( sizeof ( HashTable * ) * ce - > num_traits ) ;
resulting_table = ( HashTable * ) malloc ( sizeof ( HashTable ) ) ;
2010-04-23 07:16:15 +08:00
zend_hash_init_ex ( resulting_table , 10 , /* TODO: revisit this start size, may be its not optimal */
/* NULL, ZEND_FUNCTION_DTOR, 0, 0); */
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
NULL , NULL , 0 , 0 ) ;
for ( i = 0 ; i < ce - > num_traits ; i + + ) {
function_tables [ i ] = ( HashTable * ) malloc ( sizeof ( HashTable ) ) ;
zend_hash_init_ex ( function_tables [ i ] , ce - > traits [ i ] - > function_table . nNumOfElements ,
2010-04-23 07:16:15 +08:00
/* NULL, ZEND_FUNCTION_DTOR, 0, 0); */
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
NULL , NULL , 0 , 0 ) ;
2010-04-23 07:16:15 +08:00
zend_hash_init_ex ( & exclude_table , 2 , /* TODO: revisit this start size, may be its not optimal */
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
NULL , NULL , 0 , 0 ) ;
compile_exclude_table ( & exclude_table , ce - > trait_precedences , ce - > traits [ i ] ) ;
2010-04-23 07:16:15 +08:00
/* copies functions, applies defined aliasing, and excludes unused trait methods */
2010-04-23 07:25:05 +08:00
copy_trait_function_table ( function_tables [ i ] , & ce - > traits [ i ] - > function_table , ce - > trait_aliases , & exclude_table TSRMLS_CC ) ;
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
zend_hash_graceful_destroy ( & exclude_table ) ;
}
2010-04-23 07:16:15 +08:00
/* now merge trait methods */
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
for ( i = 0 ; i < ce - > num_traits ; i + + ) {
2010-04-23 07:25:05 +08:00
zend_hash_apply_with_arguments ( function_tables [ i ] TSRMLS_CC , ( apply_func_args_t ) _merge_functions , 5 , /* 5 is number of args for apply_func */
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
i , ce - > num_traits , resulting_table , function_tables , ce ) ;
}
2010-04-23 07:16:15 +08:00
/* now the resulting_table contains all trait methods we would have to
add to the class
in the following step the methods are inserted into the method table
if there is already a method with the same name it is replaced iff ce ! = fn . scope
- - > all inherited methods are overridden , methods defined in the class are leaved
untouched */
2010-04-23 07:25:05 +08:00
zend_hash_apply_with_arguments ( resulting_table TSRMLS_CC , ( apply_func_args_t ) _merge_functions_to_class , 1 , ce ) ;
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
2010-04-23 07:16:15 +08:00
/* free temporary function tables */
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
for ( i = 0 ; i < ce - > num_traits ; i + + ) {
2010-04-23 07:16:15 +08:00
/* zend_hash_destroy(function_tables[i]); */
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
zend_hash_graceful_destroy ( function_tables [ i ] ) ;
free ( function_tables [ i ] ) ;
}
free ( function_tables ) ;
2010-04-23 07:16:15 +08:00
/* free temporary resulting table */
/* zend_hash_destroy(resulting_table); */
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
zend_hash_graceful_destroy ( resulting_table ) ;
free ( resulting_table ) ;
zend_verify_abstract_class ( ce TSRMLS_CC ) ;
2010-04-23 07:16:15 +08:00
/* now everything should be fine and an added ZEND_ACC_IMPLICIT_ABSTRACT_CLASS should be removed */
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
if ( ce - > ce_flags & ZEND_ACC_IMPLICIT_ABSTRACT_CLASS ) {
ce - > ce_flags - = ZEND_ACC_IMPLICIT_ABSTRACT_CLASS ;
}
}
/* }}} */
2010-04-20 18:57:45 +08:00
ZEND_API int do_bind_function ( const zend_op_array * op_array , zend_op * opline , HashTable * function_table , zend_bool compile_time ) /* { { { */
2002-09-25 03:05:53 +08:00
{
zend_function * function ;
2010-04-20 18:57:45 +08:00
zval * op1 , * op2 ;
if ( compile_time ) {
op1 = & CONSTANT_EX ( op_array , opline - > op1 . constant ) ;
op2 = & CONSTANT_EX ( op_array , opline - > op2 . constant ) ;
} else {
op1 = opline - > op1 . zv ;
op2 = opline - > op2 . zv ;
}
1999-05-15 23:47:24 +08:00
2010-04-20 18:57:45 +08:00
zend_hash_quick_find ( function_table , Z_STRVAL_P ( op1 ) , Z_STRLEN_P ( op1 ) , Z_HASH_P ( op1 ) , ( void * ) & function ) ;
if ( zend_hash_quick_add ( function_table , Z_STRVAL_P ( op2 ) , Z_STRLEN_P ( op2 ) + 1 , Z_HASH_P ( op2 ) , function , sizeof ( zend_function ) , NULL ) = = FAILURE ) {
2003-09-03 16:39:43 +08:00
int error_level = compile_time ? E_COMPILE_ERROR : E_ERROR ;
2009-04-08 21:17:09 +08:00
zend_function * old_function ;
2002-09-25 03:05:53 +08:00
2010-04-20 18:57:45 +08:00
if ( zend_hash_quick_find ( function_table , Z_STRVAL_P ( op2 ) , Z_STRLEN_P ( op2 ) + 1 , Z_HASH_P ( op2 ) , ( void * ) & old_function ) = = SUCCESS
2009-04-08 21:17:09 +08:00
& & old_function - > type = = ZEND_USER_FUNCTION
& & old_function - > op_array . last > 0 ) {
2002-09-25 03:05:53 +08:00
zend_error ( error_level , " Cannot redeclare %s() (previously declared in %s:%d) " ,
2009-04-08 21:17:09 +08:00
function - > common . function_name ,
old_function - > op_array . filename ,
old_function - > op_array . opcodes [ 0 ] . lineno ) ;
2002-09-25 03:05:53 +08:00
} else {
2009-04-08 21:17:09 +08:00
zend_error ( error_level , " Cannot redeclare %s() " , function - > common . function_name ) ;
2002-09-25 03:05:53 +08:00
}
return FAILURE ;
} else {
( * function - > op_array . refcount ) + + ;
function - > op_array . static_variables = NULL ; /* NULL out the unbound function */
return SUCCESS ;
}
}
2009-07-27 22:11:53 +08:00
/* }}} */
2006-05-10 07:53:23 +08:00
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
void zend_add_trait_precedence ( znode * precedence_znode TSRMLS_DC ) /* { { { */
{
zend_class_entry * ce = CG ( active_class_entry ) ;
2010-04-23 07:16:15 +08:00
zend_add_to_list ( & ce - > trait_precedences , precedence_znode - > u . op . ptr TSRMLS_CC ) ;
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
}
/* }}} */
void zend_add_trait_alias ( znode * alias_znode TSRMLS_DC ) /* { { { */
{
zend_class_entry * ce = CG ( active_class_entry ) ;
2010-04-23 07:16:15 +08:00
zend_add_to_list ( & ce - > trait_aliases , alias_znode - > u . op . ptr TSRMLS_CC ) ;
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
}
/* }}} */
void zend_prepare_reference ( znode * result , znode * class_name , znode * method_name TSRMLS_DC ) /* { { { */
{
zend_trait_method_reference * method_ref = emalloc ( sizeof ( zend_trait_method_reference ) ) ;
method_ref - > ce = NULL ;
2010-04-23 07:16:15 +08:00
/* REM: There should not be a need for copying,
zend_do_begin_class_declaration is also just using that string */
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
if ( class_name ) {
method_ref - > class_name = Z_STRVAL ( class_name - > u . constant ) ;
method_ref - > cname_len = Z_STRLEN ( class_name - > u . constant ) ;
} else {
method_ref - > class_name = NULL ;
method_ref - > cname_len = 0 ;
}
method_ref - > method_name = Z_STRVAL ( method_name - > u . constant ) ;
method_ref - > mname_len = Z_STRLEN ( method_name - > u . constant ) ;
result - > u . op . ptr = method_ref ;
2010-05-03 02:47:27 +08:00
result - > op_type = IS_TMP_VAR ;
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
}
/* }}} */
void zend_prepare_trait_alias ( znode * result , znode * method_reference , znode * modifiers , znode * alias TSRMLS_DC ) /* { { { */
{
zend_trait_alias * trait_alias = emalloc ( sizeof ( zend_trait_alias ) ) ;
trait_alias - > trait_method = ( zend_trait_method_reference * ) method_reference - > u . op . ptr ;
trait_alias - > modifiers = Z_LVAL ( modifiers - > u . constant ) ;
if ( alias ) {
trait_alias - > alias = Z_STRVAL ( alias - > u . constant ) ;
trait_alias - > alias_len = Z_STRLEN ( alias - > u . constant ) ;
} else {
trait_alias - > alias = NULL ;
}
trait_alias - > function = NULL ;
result - > u . op . ptr = trait_alias ;
}
/* }}} */
2010-04-23 07:16:15 +08:00
/*void init_trait_alias(znode* result, const znode* method_name, const znode* alias, const znode* modifiers TSRMLS_DC)*/ /* {{{ */
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
/*{
zend_trait_alias * trait_alias = emalloc ( sizeof ( zend_trait_alias ) ) ;
trait_alias - > method_name = Z_UNIVAL ( method_name - > u . constant ) ;
// may be method is only excluded, then the alias node is NULL
if ( alias ) {
trait_alias - > alias = Z_UNIVAL ( alias - > u . constant ) ;
trait_alias - > modifiers = Z_LVAL ( modifiers - > u . constant ) ;
} else {
}
trait_alias - > function = NULL ;
result - > u . var = trait_alias ;
}
2010-04-23 07:16:15 +08:00
*/
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
/* }}} */
void zend_prepare_trait_precedence ( znode * result , znode * method_reference , znode * trait_list TSRMLS_DC ) /* { { { */
{
zend_trait_precedence * trait_precedence = emalloc ( sizeof ( zend_trait_precedence ) ) ;
trait_precedence - > trait_method = ( zend_trait_method_reference * ) method_reference - > u . op . ptr ;
trait_precedence - > exclude_from_classes = ( zend_class_entry * * ) trait_list - > u . op . ptr ;
trait_precedence - > function = NULL ;
result - > u . op . ptr = trait_precedence ;
}
/* }}} */
2010-04-20 18:57:45 +08:00
ZEND_API zend_class_entry * do_bind_class ( const zend_op_array * op_array , const zend_op * opline , HashTable * class_table , zend_bool compile_time TSRMLS_DC ) /* { { { */
2002-09-25 03:05:53 +08:00
{
zend_class_entry * ce , * * pce ;
2010-04-20 18:57:45 +08:00
zval * op1 , * op2 ;
2002-09-25 03:05:53 +08:00
2010-04-20 18:57:45 +08:00
if ( compile_time ) {
op1 = & CONSTANT_EX ( op_array , opline - > op1 . constant ) ;
op2 = & CONSTANT_EX ( op_array , opline - > op2 . constant ) ;
} else {
op1 = opline - > op1 . zv ;
op2 = opline - > op2 . zv ;
}
if ( zend_hash_quick_find ( class_table , Z_STRVAL_P ( op1 ) , Z_STRLEN_P ( op1 ) , Z_HASH_P ( op1 ) , ( void * * ) & pce ) = = FAILURE ) {
zend_error ( E_COMPILE_ERROR , " Internal Zend error - Missing class information for %s " , Z_STRVAL_P ( op1 ) ) ;
2003-03-05 19:14:44 +08:00
return NULL ;
2002-09-25 03:05:53 +08:00
} else {
ce = * pce ;
}
ce - > refcount + + ;
2010-04-20 18:57:45 +08:00
if ( zend_hash_quick_add ( class_table , Z_STRVAL_P ( op2 ) , Z_STRLEN_P ( op2 ) + 1 , Z_HASH_P ( op2 ) , & ce , sizeof ( zend_class_entry * ) , NULL ) = = FAILURE ) {
2002-09-25 03:05:53 +08:00
ce - > refcount - - ;
2004-02-05 00:30:15 +08:00
if ( ! compile_time ) {
/* If we're in compile time, in practice, it's quite possible
* that we ' ll never reach this class declaration at runtime ,
* so we shut up about it . This allows the if ( ! defined ( ' FOO ' ) ) { return ; }
* approach to work .
*/
2006-05-28 02:23:48 +08:00
zend_error ( E_COMPILE_ERROR , " Cannot redeclare class %s " , ce - > name ) ;
2004-02-05 00:30:15 +08:00
}
2003-03-05 19:14:44 +08:00
return NULL ;
2002-09-25 03:05:53 +08:00
} else {
2008-03-12 17:46:42 +08:00
if ( ! ( ce - > ce_flags & ( ZEND_ACC_INTERFACE | ZEND_ACC_IMPLEMENT_INTERFACES ) ) ) {
2004-02-05 01:10:44 +08:00
zend_verify_abstract_class ( ce TSRMLS_CC ) ;
}
2003-03-05 19:14:44 +08:00
return ce ;
2002-09-25 03:05:53 +08:00
}
}
2009-07-27 22:11:53 +08:00
/* }}} */
2000-05-26 00:26:22 +08:00
2010-04-20 18:57:45 +08:00
ZEND_API zend_class_entry * do_bind_inherited_class ( const zend_op_array * op_array , const zend_op * opline , HashTable * class_table , zend_class_entry * parent_ce , zend_bool compile_time TSRMLS_DC ) /* { { { */
2002-09-25 03:05:53 +08:00
{
zend_class_entry * ce , * * pce ;
int found_ce ;
2010-04-20 18:57:45 +08:00
zval * op1 , * op2 ;
if ( compile_time ) {
op1 = & CONSTANT_EX ( op_array , opline - > op1 . constant ) ;
op2 = & CONSTANT_EX ( op_array , opline - > op2 . constant ) ;
} else {
op1 = opline - > op1 . zv ;
op2 = opline - > op2 . zv ;
}
2002-09-15 15:54:01 +08:00
2010-04-20 18:57:45 +08:00
found_ce = zend_hash_quick_find ( class_table , Z_STRVAL_P ( op1 ) , Z_STRLEN_P ( op1 ) , Z_HASH_P ( op1 ) , ( void * * ) & pce ) ;
2002-09-25 03:05:53 +08:00
if ( found_ce = = FAILURE ) {
2004-02-05 00:30:15 +08:00
if ( ! compile_time ) {
/* If we're in compile time, in practice, it's quite possible
* that we ' ll never reach this class declaration at runtime ,
* so we shut up about it . This allows the if ( ! defined ( ' FOO ' ) ) { return ; }
* approach to work .
*/
2010-04-20 18:57:45 +08:00
zend_error ( E_COMPILE_ERROR , " Cannot redeclare class %s " , Z_STRVAL_P ( op2 ) ) ;
2004-02-05 00:30:15 +08:00
}
2003-03-05 19:14:44 +08:00
return NULL ;
2002-09-25 03:05:53 +08:00
} else {
ce = * pce ;
}
2004-08-27 06:59:25 +08:00
if ( parent_ce - > ce_flags & ZEND_ACC_INTERFACE ) {
2004-09-03 01:27:44 +08:00
zend_error ( E_COMPILE_ERROR , " Class %s cannot extend from interface %s " , ce - > name , parent_ce - > name ) ;
2004-08-27 06:59:25 +08:00
}
2003-10-23 03:59:58 +08:00
zend_do_inheritance ( ce , parent_ce TSRMLS_CC ) ;
2002-09-25 03:05:53 +08:00
2002-09-25 03:32:01 +08:00
ce - > refcount + + ;
2002-09-25 03:05:53 +08:00
/* Register the derived class */
2010-04-20 18:57:45 +08:00
if ( zend_hash_quick_add ( class_table , Z_STRVAL_P ( op2 ) , Z_STRLEN_P ( op2 ) + 1 , Z_HASH_P ( op2 ) , pce , sizeof ( zend_class_entry * ) , NULL ) = = FAILURE ) {
2006-05-28 02:23:48 +08:00
zend_error ( E_COMPILE_ERROR , " Cannot redeclare class %s " , ce - > name ) ;
2002-09-25 03:05:53 +08:00
}
2003-03-05 19:14:44 +08:00
return ce ;
1999-05-15 23:47:24 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-05-15 23:47:24 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_early_binding ( TSRMLS_D ) /* { { { */
1999-05-15 23:47:24 +08:00
{
zend_op * opline = & CG ( active_op_array ) - > opcodes [ CG ( active_op_array ) - > last - 1 ] ;
2004-02-05 00:30:15 +08:00
HashTable * table ;
1999-05-15 23:47:24 +08:00
2003-01-28 19:34:24 +08:00
while ( opline - > opcode = = ZEND_TICKS & & opline > CG ( active_op_array ) - > opcodes ) {
opline - - ;
}
2004-02-05 00:30:15 +08:00
switch ( opline - > opcode ) {
case ZEND_DECLARE_FUNCTION :
2010-04-20 18:57:45 +08:00
if ( do_bind_function ( CG ( active_op_array ) , opline , CG ( function_table ) , 1 ) = = FAILURE ) {
2004-02-05 00:30:15 +08:00
return ;
}
table = CG ( function_table ) ;
break ;
2004-02-05 01:10:44 +08:00
case ZEND_DECLARE_CLASS :
2010-04-20 18:57:45 +08:00
if ( do_bind_class ( CG ( active_op_array ) , opline , CG ( class_table ) , 1 TSRMLS_CC ) = = NULL ) {
2008-03-18 16:36:30 +08:00
return ;
}
table = CG ( class_table ) ;
break ;
2004-02-05 01:10:44 +08:00
case ZEND_DECLARE_INHERITED_CLASS :
2008-03-18 16:36:30 +08:00
{
zend_op * fetch_class_opline = opline - 1 ;
2010-04-20 18:57:45 +08:00
zval * parent_name ;
2008-03-18 16:36:30 +08:00
zend_class_entry * * pce ;
2010-04-20 18:57:45 +08:00
parent_name = & CONSTANT ( fetch_class_opline - > op2 . constant ) ;
2008-03-18 16:36:30 +08:00
if ( ( zend_lookup_class ( Z_STRVAL_P ( parent_name ) , Z_STRLEN_P ( parent_name ) , & pce TSRMLS_CC ) = = FAILURE ) | |
( ( CG ( compiler_options ) & ZEND_COMPILE_IGNORE_INTERNAL_CLASSES ) & &
( ( * pce ) - > type = = ZEND_INTERNAL_CLASS ) ) ) {
if ( CG ( compiler_options ) & ZEND_COMPILE_DELAYED_BINDING ) {
zend_uint * opline_num = & CG ( active_op_array ) - > early_binding ;
while ( * opline_num ! = - 1 ) {
2010-04-20 18:57:45 +08:00
opline_num = & CG ( active_op_array ) - > opcodes [ * opline_num ] . result . opline_num ;
2008-03-18 16:36:30 +08:00
}
* opline_num = opline - CG ( active_op_array ) - > opcodes ;
opline - > opcode = ZEND_DECLARE_INHERITED_CLASS_DELAYED ;
2010-04-20 18:57:45 +08:00
opline - > result_type = IS_UNUSED ;
opline - > result . opline_num = - 1 ;
2004-02-05 00:30:15 +08:00
}
return ;
}
2010-04-20 18:57:45 +08:00
if ( do_bind_inherited_class ( CG ( active_op_array ) , opline , CG ( class_table ) , * pce , 1 TSRMLS_CC ) = = NULL ) {
2008-03-18 16:36:30 +08:00
return ;
2004-02-05 01:10:44 +08:00
}
2008-03-18 16:36:30 +08:00
/* clear unnecessary ZEND_FETCH_CLASS opcode */
2010-04-20 18:57:45 +08:00
zend_del_literal ( CG ( active_op_array ) , fetch_class_opline - > op2 . constant ) ;
2008-03-18 16:36:30 +08:00
MAKE_NOP ( fetch_class_opline ) ;
2004-02-05 00:30:15 +08:00
2008-03-18 16:36:30 +08:00
table = CG ( class_table ) ;
break ;
}
case ZEND_VERIFY_ABSTRACT_CLASS :
2004-02-05 02:24:44 +08:00
case ZEND_ADD_INTERFACE :
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
case ZEND_ADD_TRAIT :
case ZEND_BIND_TRAITS :
2004-02-05 02:24:44 +08:00
/* We currently don't early-bind classes that implement interfaces */
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
/* Classes with traits are handled exactly the same, no early-bind here */
2004-02-05 02:24:44 +08:00
return ;
2004-02-05 00:30:15 +08:00
default :
zend_error ( E_COMPILE_ERROR , " Invalid binding type " ) ;
return ;
1999-07-27 05:18:35 +08:00
}
2002-09-25 03:05:53 +08:00
2010-04-20 18:57:45 +08:00
zend_hash_quick_del ( table , Z_STRVAL ( CONSTANT ( opline - > op1 . constant ) ) , Z_STRLEN ( CONSTANT ( opline - > op1 . constant ) ) , Z_HASH_P ( & CONSTANT ( opline - > op1 . constant ) ) ) ;
zend_del_literal ( CG ( active_op_array ) , opline - > op1 . constant ) ;
zend_del_literal ( CG ( active_op_array ) , opline - > op2 . constant ) ;
2008-03-18 16:36:30 +08:00
MAKE_NOP ( opline ) ;
1999-05-15 23:47:24 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-05-15 23:47:24 +08:00
2009-07-27 22:11:53 +08:00
ZEND_API void zend_do_delayed_early_binding ( const zend_op_array * op_array TSRMLS_DC ) /* { { { */
2008-03-18 16:36:30 +08:00
{
if ( op_array - > early_binding ! = - 1 ) {
zend_bool orig_in_compilation = CG ( in_compilation ) ;
zend_uint opline_num = op_array - > early_binding ;
zend_class_entry * * pce ;
CG ( in_compilation ) = 1 ;
while ( opline_num ! = - 1 ) {
2010-04-20 18:57:45 +08:00
if ( zend_lookup_class ( Z_STRVAL_P ( op_array - > opcodes [ opline_num - 1 ] . op2 . zv ) , Z_STRLEN_P ( op_array - > opcodes [ opline_num - 1 ] . op2 . zv ) , & pce TSRMLS_CC ) = = SUCCESS ) {
do_bind_inherited_class ( op_array , & op_array - > opcodes [ opline_num ] , EG ( class_table ) , * pce , 0 TSRMLS_CC ) ;
2008-03-18 16:36:30 +08:00
}
2010-04-20 18:57:45 +08:00
opline_num = op_array - > opcodes [ opline_num ] . result . opline_num ;
2008-03-18 16:36:30 +08:00
}
CG ( in_compilation ) = orig_in_compilation ;
}
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-05-15 23:47:24 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_boolean_or_begin ( znode * expr1 , znode * op_token TSRMLS_DC ) /* { { { */
1999-04-08 02:10:10 +08:00
{
int next_op_number = get_next_op_number ( CG ( active_op_array ) ) ;
2001-07-28 18:51:54 +08:00
zend_op * opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
1999-04-08 02:10:10 +08:00
opline - > opcode = ZEND_JMPNZ_EX ;
if ( expr1 - > op_type = = IS_TMP_VAR ) {
2010-04-20 18:57:45 +08:00
SET_NODE ( opline - > result , expr1 ) ;
1999-04-08 02:10:10 +08:00
} else {
2010-04-20 18:57:45 +08:00
opline - > result . var = get_temporary_variable ( CG ( active_op_array ) ) ;
opline - > result_type = IS_TMP_VAR ;
1999-04-08 02:10:10 +08:00
}
2010-04-20 18:57:45 +08:00
SET_NODE ( opline - > op1 , expr1 ) ;
1999-04-08 02:10:10 +08:00
SET_UNUSED ( opline - > op2 ) ;
2006-05-10 07:53:23 +08:00
2010-04-20 18:57:45 +08:00
op_token - > u . op . opline_num = next_op_number ;
1999-04-08 02:10:10 +08:00
2010-04-20 18:57:45 +08:00
GET_NODE ( expr1 , opline - > result ) ;
1999-04-08 02:10:10 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_boolean_or_end ( znode * result , const znode * expr1 , const znode * expr2 , znode * op_token TSRMLS_DC ) /* { { { */
1999-04-08 02:10:10 +08:00
{
2001-07-28 18:51:54 +08:00
zend_op * opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
1999-04-08 02:10:10 +08:00
* result = * expr1 ; /* we saved the original result in expr1 */
opline - > opcode = ZEND_BOOL ;
2010-04-20 18:57:45 +08:00
SET_NODE ( opline - > result , result ) ;
SET_NODE ( opline - > op1 , expr2 ) ;
1999-04-08 02:10:10 +08:00
SET_UNUSED ( opline - > op2 ) ;
2010-04-20 18:57:45 +08:00
CG ( active_op_array ) - > opcodes [ op_token - > u . op . opline_num ] . op2 . opline_num = get_next_op_number ( CG ( active_op_array ) ) ;
1999-04-08 02:10:10 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_boolean_and_begin ( znode * expr1 , znode * op_token TSRMLS_DC ) /* { { { */
1999-04-08 02:10:10 +08:00
{
int next_op_number = get_next_op_number ( CG ( active_op_array ) ) ;
2001-07-28 18:51:54 +08:00
zend_op * opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
1999-04-08 02:10:10 +08:00
opline - > opcode = ZEND_JMPZ_EX ;
if ( expr1 - > op_type = = IS_TMP_VAR ) {
2010-04-20 18:57:45 +08:00
SET_NODE ( opline - > result , expr1 ) ;
1999-04-08 02:10:10 +08:00
} else {
2010-04-20 18:57:45 +08:00
opline - > result . var = get_temporary_variable ( CG ( active_op_array ) ) ;
opline - > result_type = IS_TMP_VAR ;
1999-04-08 02:10:10 +08:00
}
2010-04-20 18:57:45 +08:00
SET_NODE ( opline - > op1 , expr1 ) ;
1999-04-08 02:10:10 +08:00
SET_UNUSED ( opline - > op2 ) ;
2006-05-10 07:53:23 +08:00
2010-04-20 18:57:45 +08:00
op_token - > u . op . opline_num = next_op_number ;
1999-04-08 02:10:10 +08:00
2010-04-20 18:57:45 +08:00
GET_NODE ( expr1 , opline - > result ) ;
1999-04-08 02:10:10 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_boolean_and_end ( znode * result , const znode * expr1 , const znode * expr2 , const znode * op_token TSRMLS_DC ) /* { { { */
1999-04-08 02:10:10 +08:00
{
2001-07-28 18:51:54 +08:00
zend_op * opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
1999-04-08 02:10:10 +08:00
* result = * expr1 ; /* we saved the original result in expr1 */
opline - > opcode = ZEND_BOOL ;
2010-04-20 18:57:45 +08:00
SET_NODE ( opline - > result , result ) ;
SET_NODE ( opline - > op1 , expr2 ) ;
1999-04-08 02:10:10 +08:00
SET_UNUSED ( opline - > op2 ) ;
2010-04-20 18:57:45 +08:00
CG ( active_op_array ) - > opcodes [ op_token - > u . op . opline_num ] . op2 . opline_num = get_next_op_number ( CG ( active_op_array ) ) ;
1999-04-08 02:10:10 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_do_while_begin ( TSRMLS_D ) /* { { { */
1999-04-08 02:10:10 +08:00
{
2001-07-28 18:51:54 +08:00
do_begin_loop ( TSRMLS_C ) ;
1999-04-08 02:10:10 +08:00
INC_BPC ( CG ( active_op_array ) ) ;
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_do_while_end ( const znode * do_token , const znode * expr_open_bracket , const znode * expr TSRMLS_DC ) /* { { { */
1999-04-08 02:10:10 +08:00
{
2001-07-28 18:51:54 +08:00
zend_op * opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
1999-04-08 02:10:10 +08:00
opline - > opcode = ZEND_JMPNZ ;
2010-04-20 18:57:45 +08:00
SET_NODE ( opline - > op1 , expr ) ;
opline - > op2 . opline_num = do_token - > u . op . opline_num ;
1999-06-23 04:24:51 +08:00
SET_UNUSED ( opline - > op2 ) ;
1999-04-08 02:10:10 +08:00
2010-04-20 18:57:45 +08:00
do_end_loop ( expr_open_bracket - > u . op . opline_num , 0 TSRMLS_CC ) ;
1999-04-08 02:10:10 +08:00
DEC_BPC ( CG ( active_op_array ) ) ;
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_brk_cont ( zend_uchar op , const znode * expr TSRMLS_DC ) /* { { { */
1999-04-08 02:10:10 +08:00
{
2001-07-28 18:51:54 +08:00
zend_op * opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
1999-04-08 02:10:10 +08:00
opline - > opcode = op ;
2010-04-20 18:57:45 +08:00
opline - > op1 . opline_num = CG ( active_op_array ) - > current_brk_cont ;
1999-04-08 02:10:10 +08:00
SET_UNUSED ( opline - > op1 ) ;
if ( expr ) {
2010-04-20 18:57:45 +08:00
SET_NODE ( opline - > op2 , expr ) ;
1999-04-08 02:10:10 +08:00
} else {
2010-04-20 18:57:45 +08:00
LITERAL_LONG ( opline - > op2 , 1 ) ;
opline - > op2_type = IS_CONST ;
1999-04-08 02:10:10 +08:00
}
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_switch_cond ( const znode * cond TSRMLS_DC ) /* { { { */
1999-04-08 02:10:10 +08:00
{
zend_switch_entry switch_entry ;
2000-08-11 04:42:57 +08:00
1999-04-08 02:10:10 +08:00
switch_entry . cond = * cond ;
switch_entry . default_case = - 1 ;
2004-08-24 04:16:35 +08:00
switch_entry . control_var = - 1 ;
1999-04-08 02:10:10 +08:00
zend_stack_push ( & CG ( switch_cond_stack ) , ( void * ) & switch_entry , sizeof ( switch_entry ) ) ;
2001-07-28 18:51:54 +08:00
do_begin_loop ( TSRMLS_C ) ;
1999-04-08 02:10:10 +08:00
INC_BPC ( CG ( active_op_array ) ) ;
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_switch_end ( const znode * case_list TSRMLS_DC ) /* { { { */
1999-04-08 02:10:10 +08:00
{
zend_op * opline ;
zend_switch_entry * switch_entry_ptr ;
2006-05-10 07:53:23 +08:00
1999-04-08 02:10:10 +08:00
zend_stack_top ( & CG ( switch_cond_stack ) , ( void * * ) & switch_entry_ptr ) ;
2004-10-09 02:50:00 +08:00
/* add code to jmp to default case */
if ( switch_entry_ptr - > default_case ! = - 1 ) {
opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
opline - > opcode = ZEND_JMP ;
SET_UNUSED ( opline - > op1 ) ;
SET_UNUSED ( opline - > op2 ) ;
2010-04-20 18:57:45 +08:00
opline - > op1 . opline_num = switch_entry_ptr - > default_case ;
2004-10-09 02:50:00 +08:00
}
2000-04-11 02:24:38 +08:00
if ( case_list - > op_type ! = IS_UNUSED ) { /* non-empty switch */
1999-04-08 02:10:10 +08:00
int next_op_number = get_next_op_number ( CG ( active_op_array ) ) ;
2010-04-20 18:57:45 +08:00
CG ( active_op_array ) - > opcodes [ case_list - > u . op . opline_num ] . op1 . opline_num = next_op_number ;
1999-04-08 02:10:10 +08:00
}
/* remember break/continue loop information */
CG ( active_op_array ) - > brk_cont_array [ CG ( active_op_array ) - > current_brk_cont ] . cont = CG ( active_op_array ) - > brk_cont_array [ CG ( active_op_array ) - > current_brk_cont ] . brk = get_next_op_number ( CG ( active_op_array ) ) ;
CG ( active_op_array ) - > current_brk_cont = CG ( active_op_array ) - > brk_cont_array [ CG ( active_op_array ) - > current_brk_cont ] . parent ;
2000-03-30 06:28:04 +08:00
if ( switch_entry_ptr - > cond . op_type = = IS_VAR | | switch_entry_ptr - > cond . op_type = = IS_TMP_VAR ) {
/* emit free for the switch condition*/
2001-07-28 18:51:54 +08:00
opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
2008-05-05 19:03:35 +08:00
opline - > opcode = ( switch_entry_ptr - > cond . op_type = = IS_TMP_VAR ) ? ZEND_FREE : ZEND_SWITCH_FREE ;
2010-04-20 18:57:45 +08:00
SET_NODE ( opline - > op1 , & switch_entry_ptr - > cond ) ;
2000-03-30 06:28:04 +08:00
SET_UNUSED ( opline - > op2 ) ;
}
1999-04-08 02:10:10 +08:00
if ( switch_entry_ptr - > cond . op_type = = IS_CONST ) {
zval_dtor ( & switch_entry_ptr - > cond . u . constant ) ;
}
zend_stack_del_top ( & CG ( switch_cond_stack ) ) ;
DEC_BPC ( CG ( active_op_array ) ) ;
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_case_before_statement ( const znode * case_list , znode * case_token , const znode * case_expr TSRMLS_DC ) /* { { { */
1999-04-08 02:10:10 +08:00
{
2001-07-28 18:51:54 +08:00
zend_op * opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
1999-04-08 02:10:10 +08:00
int next_op_number ;
zend_switch_entry * switch_entry_ptr ;
znode result ;
zend_stack_top ( & CG ( switch_cond_stack ) , ( void * * ) & switch_entry_ptr ) ;
2004-08-24 04:16:35 +08:00
if ( switch_entry_ptr - > control_var = = - 1 ) {
switch_entry_ptr - > control_var = get_temporary_variable ( CG ( active_op_array ) ) ;
}
1999-04-08 02:10:10 +08:00
opline - > opcode = ZEND_CASE ;
2010-04-20 18:57:45 +08:00
opline - > result . var = switch_entry_ptr - > control_var ;
opline - > result_type = IS_TMP_VAR ;
SET_NODE ( opline - > op1 , & switch_entry_ptr - > cond ) ;
SET_NODE ( opline - > op2 , case_expr ) ;
if ( opline - > op1_type = = IS_CONST ) {
zval_copy_ctor ( & CONSTANT ( opline - > op1 . constant ) ) ;
1999-04-08 02:10:10 +08:00
}
2010-04-20 18:57:45 +08:00
GET_NODE ( & result , opline - > result ) ;
2006-05-10 07:53:23 +08:00
1999-04-08 02:10:10 +08:00
next_op_number = get_next_op_number ( CG ( active_op_array ) ) ;
2001-07-28 18:51:54 +08:00
opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
1999-04-08 02:10:10 +08:00
opline - > opcode = ZEND_JMPZ ;
2010-04-20 18:57:45 +08:00
SET_NODE ( opline - > op1 , & result ) ;
1999-04-08 02:10:10 +08:00
SET_UNUSED ( opline - > op2 ) ;
2010-04-20 18:57:45 +08:00
case_token - > u . op . opline_num = next_op_number ;
1999-04-08 02:10:10 +08:00
2000-04-11 02:02:40 +08:00
if ( case_list - > op_type = = IS_UNUSED ) {
1999-04-08 02:10:10 +08:00
return ;
}
next_op_number = get_next_op_number ( CG ( active_op_array ) ) ;
2010-04-20 18:57:45 +08:00
CG ( active_op_array ) - > opcodes [ case_list - > u . op . opline_num ] . op1 . opline_num = next_op_number ;
1999-04-08 02:10:10 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_case_after_statement ( znode * result , const znode * case_token TSRMLS_DC ) /* { { { */
1999-04-08 02:10:10 +08:00
{
int next_op_number = get_next_op_number ( CG ( active_op_array ) ) ;
2001-07-28 18:51:54 +08:00
zend_op * opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
1999-04-08 02:10:10 +08:00
opline - > opcode = ZEND_JMP ;
SET_UNUSED ( opline - > op1 ) ;
SET_UNUSED ( opline - > op2 ) ;
2010-04-20 18:57:45 +08:00
result - > u . op . opline_num = next_op_number ;
1999-04-08 02:10:10 +08:00
2010-04-20 18:57:45 +08:00
switch ( CG ( active_op_array ) - > opcodes [ case_token - > u . op . opline_num ] . opcode ) {
2004-10-09 02:50:00 +08:00
case ZEND_JMP :
2010-04-20 18:57:45 +08:00
CG ( active_op_array ) - > opcodes [ case_token - > u . op . opline_num ] . op1 . opline_num = get_next_op_number ( CG ( active_op_array ) ) ;
2004-10-09 02:50:00 +08:00
break ;
case ZEND_JMPZ :
2010-04-20 18:57:45 +08:00
CG ( active_op_array ) - > opcodes [ case_token - > u . op . opline_num ] . op2 . opline_num = get_next_op_number ( CG ( active_op_array ) ) ;
2004-10-09 02:50:00 +08:00
break ;
1999-04-08 02:10:10 +08:00
}
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_default_before_statement ( const znode * case_list , znode * default_token TSRMLS_DC ) /* { { { */
1999-04-08 02:10:10 +08:00
{
int next_op_number = get_next_op_number ( CG ( active_op_array ) ) ;
2004-10-09 02:50:00 +08:00
zend_op * opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
1999-04-08 02:10:10 +08:00
zend_switch_entry * switch_entry_ptr ;
zend_stack_top ( & CG ( switch_cond_stack ) , ( void * * ) & switch_entry_ptr ) ;
2004-10-09 02:50:00 +08:00
opline - > opcode = ZEND_JMP ;
SET_UNUSED ( opline - > op1 ) ;
SET_UNUSED ( opline - > op2 ) ;
2010-04-20 18:57:45 +08:00
default_token - > u . op . opline_num = next_op_number ;
2004-10-09 02:50:00 +08:00
next_op_number = get_next_op_number ( CG ( active_op_array ) ) ;
1999-04-08 02:10:10 +08:00
switch_entry_ptr - > default_case = next_op_number ;
2000-04-11 02:24:38 +08:00
if ( case_list - > op_type = = IS_UNUSED ) {
1999-04-08 02:10:10 +08:00
return ;
}
2010-04-20 18:57:45 +08:00
CG ( active_op_array ) - > opcodes [ case_list - > u . op . opline_num ] . op1 . opline_num = next_op_number ;
1999-04-08 02:10:10 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_begin_class_declaration ( const znode * class_token , znode * class_name , const znode * parent_class_name TSRMLS_DC ) /* { { { */
1999-04-08 02:10:10 +08:00
{
2001-10-30 01:19:02 +08:00
zend_op * opline ;
2002-09-25 03:05:53 +08:00
int doing_inheritance = 0 ;
2006-12-21 10:09:36 +08:00
zend_class_entry * new_class_entry ;
char * lcname ;
2007-11-12 23:52:22 +08:00
int error = 0 ;
2010-04-20 18:57:45 +08:00
zval * * ns_name , key ;
2001-10-30 01:19:02 +08:00
2004-01-12 15:15:55 +08:00
if ( CG ( active_class_entry ) ) {
zend_error ( E_COMPILE_ERROR , " Class declarations may not be nested " ) ;
return ;
}
2006-12-21 10:09:36 +08:00
lcname = zend_str_tolower_dup ( class_name - > u . constant . value . str . val , class_name - > u . constant . value . str . len ) ;
2003-12-23 06:34:28 +08:00
if ( ! ( strcmp ( lcname , " self " ) & & strcmp ( lcname , " parent " ) ) ) {
efree ( lcname ) ;
2002-12-07 23:51:25 +08:00
zend_error ( E_COMPILE_ERROR , " Cannot use '%s' as class name as it is reserved " , class_name - > u . constant . value . str . val ) ;
}
2003-02-11 00:11:24 +08:00
2007-09-29 03:52:53 +08:00
/* Class name must not conflict with import names */
if ( CG ( current_import ) & &
2007-11-12 23:52:22 +08:00
zend_hash_find ( CG ( current_import ) , lcname , Z_STRLEN ( class_name - > u . constant ) + 1 , ( void * * ) & ns_name ) = = SUCCESS ) {
error = 1 ;
2007-09-29 03:52:53 +08:00
}
if ( CG ( current_namespace ) ) {
/* Prefix class name with name of current namespace */
znode tmp ;
tmp . u . constant = * CG ( current_namespace ) ;
zval_copy_ctor ( & tmp . u . constant ) ;
zend_do_build_namespace_name ( & tmp , & tmp , class_name TSRMLS_CC ) ;
class_name = & tmp ;
efree ( lcname ) ;
lcname = zend_str_tolower_dup ( Z_STRVAL ( class_name - > u . constant ) , Z_STRLEN ( class_name - > u . constant ) ) ;
}
2007-11-12 23:52:22 +08:00
if ( error ) {
char * tmp = zend_str_tolower_dup ( Z_STRVAL_PP ( ns_name ) , Z_STRLEN_PP ( ns_name ) ) ;
if ( Z_STRLEN_PP ( ns_name ) ! = Z_STRLEN ( class_name - > u . constant ) | |
memcmp ( tmp , lcname , Z_STRLEN ( class_name - > u . constant ) ) ) {
2007-11-13 01:52:15 +08:00
zend_error ( E_COMPILE_ERROR , " Cannot declare class %s because the name is already in use " , Z_STRVAL ( class_name - > u . constant ) ) ;
2007-11-12 23:52:22 +08:00
}
efree ( tmp ) ;
}
2006-12-21 10:09:36 +08:00
new_class_entry = emalloc ( sizeof ( zend_class_entry ) ) ;
2003-03-10 11:04:02 +08:00
new_class_entry - > type = ZEND_USER_CLASS ;
2003-03-04 21:36:16 +08:00
new_class_entry - > name = class_name - > u . constant . value . str . val ;
new_class_entry - > name_length = class_name - > u . constant . value . str . len ;
1999-04-08 02:10:10 +08:00
2003-02-16 19:34:49 +08:00
zend_initialize_class_data ( new_class_entry , 1 TSRMLS_CC ) ;
2003-04-01 04:42:01 +08:00
new_class_entry - > filename = zend_get_compiled_filename ( TSRMLS_C ) ;
2010-04-20 18:57:45 +08:00
new_class_entry - > line_start = class_token - > u . op . opline_num ;
new_class_entry - > ce_flags | = class_token - > EA ;
1999-04-08 02:10:10 +08:00
2004-02-12 06:13:39 +08:00
if ( parent_class_name & & parent_class_name - > op_type ! = IS_UNUSED ) {
2010-04-20 18:57:45 +08:00
switch ( parent_class_name - > EA ) {
2005-06-24 16:45:17 +08:00
case ZEND_FETCH_CLASS_SELF :
zend_error ( E_COMPILE_ERROR , " Cannot use 'self' as class name as it is reserved " ) ;
break ;
case ZEND_FETCH_CLASS_PARENT :
zend_error ( E_COMPILE_ERROR , " Cannot use 'parent' as class name as it is reserved " ) ;
break ;
2007-09-29 15:28:34 +08:00
case ZEND_FETCH_CLASS_STATIC :
zend_error ( E_COMPILE_ERROR , " Cannot use 'static' as class name as it is reserved " ) ;
break ;
2005-06-24 16:45:17 +08:00
default :
break ;
}
2002-09-25 03:05:53 +08:00
doing_inheritance = 1 ;
2001-10-30 01:19:02 +08:00
}
opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
2010-04-20 18:57:45 +08:00
opline - > op1_type = IS_CONST ;
build_runtime_defined_function_key ( & key , lcname , new_class_entry - > name_length TSRMLS_CC ) ;
opline - > op1 . constant = zend_add_literal ( CG ( active_op_array ) , & key ) ;
Z_HASH_P ( & CONSTANT ( opline - > op1 . constant ) ) = zend_hash_func ( Z_STRVAL ( CONSTANT ( opline - > op1 . constant ) ) , Z_STRLEN ( CONSTANT ( opline - > op1 . constant ) ) ) ;
2002-09-25 03:05:53 +08:00
2010-04-20 18:57:45 +08:00
opline - > op2_type = IS_CONST ;
2006-05-10 07:53:23 +08:00
2002-09-25 03:05:53 +08:00
if ( doing_inheritance ) {
2010-04-20 18:57:45 +08:00
opline - > extended_value = parent_class_name - > u . op . var ;
2002-09-25 03:05:53 +08:00
opline - > opcode = ZEND_DECLARE_INHERITED_CLASS ;
1999-07-09 00:55:27 +08:00
} else {
2002-09-25 03:05:53 +08:00
opline - > opcode = ZEND_DECLARE_CLASS ;
1999-07-09 00:55:27 +08:00
}
2002-09-25 03:05:53 +08:00
2010-04-20 18:57:45 +08:00
LITERAL_STRINGL ( opline - > op2 , lcname , new_class_entry - > name_length , 0 ) ;
2010-04-21 22:58:33 +08:00
CALCULATE_LITERAL_HASH ( opline - > op2 . constant ) ;
2003-02-16 19:12:43 +08:00
2010-04-20 18:57:45 +08:00
zend_hash_quick_update ( CG ( class_table ) , Z_STRVAL ( key ) , Z_STRLEN ( key ) , Z_HASH_P ( & CONSTANT ( opline - > op1 . constant ) ) , & new_class_entry , sizeof ( zend_class_entry * ) , NULL ) ;
2002-03-12 18:08:47 +08:00
CG ( active_class_entry ) = new_class_entry ;
2003-03-05 19:14:44 +08:00
2010-04-20 18:57:45 +08:00
opline - > result . var = get_temporary_variable ( CG ( active_op_array ) ) ;
opline - > result_type = IS_VAR ;
GET_NODE ( & CG ( implementing_class ) , opline - > result ) ;
2003-04-03 00:51:49 +08:00
if ( CG ( doc_comment ) ) {
2005-06-08 02:11:56 +08:00
CG ( active_class_entry ) - > doc_comment = CG ( doc_comment ) ;
2003-04-03 00:51:49 +08:00
CG ( active_class_entry ) - > doc_comment_len = CG ( doc_comment_len ) ;
2005-06-08 02:11:56 +08:00
CG ( doc_comment ) = NULL ;
CG ( doc_comment_len ) = 0 ;
2003-04-03 00:51:49 +08:00
}
1999-04-08 02:10:10 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
static void do_verify_abstract_class ( TSRMLS_D ) /* { { { */
2003-03-07 06:53:23 +08:00
{
zend_op * opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
opline - > opcode = ZEND_VERIFY_ABSTRACT_CLASS ;
2010-04-20 18:57:45 +08:00
SET_NODE ( opline - > op1 , & CG ( implementing_class ) ) ;
2004-03-22 23:51:07 +08:00
SET_UNUSED ( opline - > op2 ) ;
2003-03-07 06:53:23 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
2003-03-07 06:53:23 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_end_class_declaration ( const znode * class_token , const znode * parent_token TSRMLS_DC ) /* { { { */
1999-04-08 02:10:10 +08:00
{
2003-05-30 03:00:40 +08:00
zend_class_entry * ce = CG ( active_class_entry ) ;
2003-08-23 23:38:58 +08:00
if ( ce - > constructor ) {
2004-01-25 00:59:24 +08:00
ce - > constructor - > common . fn_flags | = ZEND_ACC_CTOR ;
2003-09-02 21:26:25 +08:00
if ( ce - > constructor - > common . fn_flags & ZEND_ACC_STATIC ) {
2003-09-03 07:12:37 +08:00
zend_error ( E_COMPILE_ERROR , " Constructor %s::%s() cannot be static " , ce - > name , ce - > constructor - > common . function_name ) ;
2003-09-02 21:26:25 +08:00
}
2003-08-23 23:38:58 +08:00
}
if ( ce - > destructor ) {
2004-01-25 00:59:24 +08:00
ce - > destructor - > common . fn_flags | = ZEND_ACC_DTOR ;
2003-09-02 21:26:25 +08:00
if ( ce - > destructor - > common . fn_flags & ZEND_ACC_STATIC ) {
2003-09-03 07:12:37 +08:00
zend_error ( E_COMPILE_ERROR , " Destructor %s::%s() cannot be static " , ce - > name , ce - > destructor - > common . function_name ) ;
2003-09-02 21:26:25 +08:00
}
2003-08-23 23:38:58 +08:00
}
2004-01-24 04:52:39 +08:00
if ( ce - > clone ) {
2004-01-25 00:59:24 +08:00
ce - > clone - > common . fn_flags | = ZEND_ACC_CLONE ;
2004-01-24 04:52:39 +08:00
if ( ce - > clone - > common . fn_flags & ZEND_ACC_STATIC ) {
zend_error ( E_COMPILE_ERROR , " Clone method %s::%s() cannot be static " , ce - > name , ce - > clone - > common . function_name ) ;
}
}
2003-08-23 23:38:58 +08:00
2003-05-30 03:00:40 +08:00
ce - > line_end = zend_get_compiled_lineno ( TSRMLS_C ) ;
2003-04-01 04:42:01 +08:00
2004-03-10 00:38:37 +08:00
if ( ! ( ce - > ce_flags & ( ZEND_ACC_INTERFACE | ZEND_ACC_EXPLICIT_ABSTRACT_CLASS ) )
2003-05-30 03:00:40 +08:00
& & ( ( parent_token - > op_type ! = IS_UNUSED ) | | ( ce - > num_interfaces > 0 ) ) ) {
2004-02-27 17:14:55 +08:00
zend_verify_abstract_class ( ce TSRMLS_CC ) ;
2009-04-08 21:17:09 +08:00
if ( ce - > num_interfaces ) {
2004-02-27 17:14:55 +08:00
do_verify_abstract_class ( TSRMLS_C ) ;
}
2003-03-07 06:53:23 +08:00
}
2006-06-07 17:21:06 +08:00
/* Inherit interfaces; reset number to zero, we need it for above check and
2008-03-12 17:46:42 +08:00
* will restore it during actual implementation .
* The ZEND_ACC_IMPLEMENT_INTERFACES flag disables double call to
* zend_verify_abstract_class ( ) */
2006-06-07 17:21:06 +08:00
if ( ce - > num_interfaces > 0 ) {
ce - > interfaces = NULL ;
ce - > num_interfaces = 0 ;
2008-03-12 17:46:42 +08:00
ce - > ce_flags | = ZEND_ACC_IMPLEMENT_INTERFACES ;
2006-06-07 17:21:06 +08:00
}
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
/* Check for traits and proceed like with interfaces.
* The only difference will be a combined handling of them in the end .
* Thus , we need another opcode here . */
if ( ce - > num_traits > 0 ) {
2010-04-23 09:56:03 +08:00
zend_op * opline ;
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
ce - > traits = NULL ;
ce - > num_traits = 0 ;
ce - > ce_flags | = ZEND_ACC_IMPLEMENT_TRAITS ;
2010-04-23 07:16:15 +08:00
/* opcode generation: */
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
opline - > opcode = ZEND_BIND_TRAITS ;
2010-04-23 09:56:03 +08:00
SET_NODE ( opline - > op1 , & CG ( implementing_class ) ) ;
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
}
2003-02-16 19:12:43 +08:00
CG ( active_class_entry ) = NULL ;
1999-04-08 02:10:10 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_implements_interface ( znode * interface_name TSRMLS_DC ) /* { { { */
2003-03-05 19:14:44 +08:00
{
2005-06-24 16:45:17 +08:00
zend_op * opline ;
2003-03-05 19:14:44 +08:00
2010-04-23 07:16:15 +08:00
/* Traits can not implement interfaces */
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
if ( ( CG ( active_class_entry ) - > ce_flags & ZEND_ACC_TRAIT ) = = ZEND_ACC_TRAIT ) {
zend_error ( E_COMPILE_ERROR , " Cannot use '%s' as interface on '%s' since it is a Trait " ,
Z_STRVAL ( interface_name - > u . constant ) ,
CG ( active_class_entry ) - > name ) ;
}
2008-03-12 18:32:12 +08:00
switch ( zend_get_class_fetch_type ( Z_STRVAL ( interface_name - > u . constant ) , Z_STRLEN ( interface_name - > u . constant ) ) ) {
2005-06-24 16:45:17 +08:00
case ZEND_FETCH_CLASS_SELF :
case ZEND_FETCH_CLASS_PARENT :
2007-09-29 15:28:34 +08:00
case ZEND_FETCH_CLASS_STATIC :
2008-03-12 18:32:12 +08:00
zend_error ( E_COMPILE_ERROR , " Cannot use '%s' as interface name as it is reserved " , Z_STRVAL ( interface_name - > u . constant ) ) ;
2007-09-29 15:28:34 +08:00
break ;
2005-06-24 16:45:17 +08:00
default :
break ;
}
2006-05-10 07:53:23 +08:00
2005-06-24 16:45:17 +08:00
opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
2003-03-05 19:14:44 +08:00
opline - > opcode = ZEND_ADD_INTERFACE ;
2010-04-20 18:57:45 +08:00
SET_NODE ( opline - > op1 , & CG ( implementing_class ) ) ;
2008-03-12 18:32:12 +08:00
zend_resolve_class_name ( interface_name , & opline - > extended_value , 0 TSRMLS_CC ) ;
opline - > extended_value = ( opline - > extended_value & ~ ZEND_FETCH_CLASS_MASK ) | ZEND_FETCH_CLASS_INTERFACE ;
2010-04-21 22:58:33 +08:00
opline - > op2_type = IS_CONST ;
2010-04-21 23:08:10 +08:00
opline - > op2 . constant = zend_add_class_name_literal ( CG ( active_op_array ) , & interface_name - > u . constant TSRMLS_CC ) ;
2008-03-12 18:32:12 +08:00
CG ( active_class_entry ) - > num_interfaces + + ;
2003-03-05 19:14:44 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
2003-03-05 19:14:44 +08:00
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
void zend_do_implements_trait ( znode * trait_name TSRMLS_DC ) /* { { { */
{
zend_op * opline ;
switch ( zend_get_class_fetch_type ( Z_STRVAL ( trait_name - > u . constant ) , Z_STRLEN ( trait_name - > u . constant ) ) ) {
case ZEND_FETCH_CLASS_SELF :
case ZEND_FETCH_CLASS_PARENT :
case ZEND_FETCH_CLASS_STATIC :
zend_error ( E_COMPILE_ERROR , " Cannot use '%s' as trait name as it is reserved " , Z_STRVAL ( trait_name - > u . constant ) ) ;
break ;
default :
break ;
}
opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
opline - > opcode = ZEND_ADD_TRAIT ;
2010-04-23 16:56:03 +08:00
SET_NODE ( opline - > op1 , & CG ( implementing_class ) ) ;
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
zend_resolve_class_name ( trait_name , & opline - > extended_value , 0 TSRMLS_CC ) ;
opline - > extended_value = ( opline - > extended_value & ~ ZEND_FETCH_CLASS_MASK ) | ZEND_FETCH_CLASS_TRAIT ;
2010-04-23 16:56:03 +08:00
opline - > op2_type = IS_CONST ;
opline - > op2 . constant = zend_add_class_name_literal ( CG ( active_op_array ) , & trait_name - > u . constant TSRMLS_CC ) ;
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
CG ( active_class_entry ) - > num_traits + + ;
}
/* }}} */
2009-07-27 22:11:53 +08:00
ZEND_API void zend_mangle_property_name ( char * * dest , int * dest_length , const char * src1 , int src1_length , const char * src2 , int src2_length , int internal ) /* { { { */
2002-02-21 19:50:44 +08:00
{
2002-07-16 02:09:56 +08:00
char * prop_name ;
int prop_name_length ;
2006-05-10 07:53:23 +08:00
2002-07-16 02:09:56 +08:00
prop_name_length = 1 + src1_length + 1 + src2_length ;
2003-08-25 00:17:45 +08:00
prop_name = pemalloc ( prop_name_length + 1 , internal ) ;
2002-07-16 02:09:56 +08:00
prop_name [ 0 ] = ' \0 ' ;
memcpy ( prop_name + 1 , src1 , src1_length + 1 ) ;
memcpy ( prop_name + 1 + src1_length + 1 , src2 , src2_length + 1 ) ;
2002-02-21 19:50:44 +08:00
2002-07-16 02:09:56 +08:00
* dest = prop_name ;
* dest_length = prop_name_length ;
2002-02-21 19:50:44 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
static int zend_strnlen ( const char * s , int maxlen ) /* { { { */
2005-09-17 01:05:09 +08:00
{
2006-07-25 01:58:32 +08:00
int len = 0 ;
while ( * s + + & & maxlen - - ) len + + ;
return len ;
2005-09-17 01:05:09 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
2005-09-17 01:05:09 +08:00
2009-07-27 22:11:53 +08:00
ZEND_API int zend_unmangle_property_name ( char * mangled_property , int len , char * * class_name , char * * prop_name ) /* { { { */
2003-06-09 02:53:58 +08:00
{
2006-07-25 01:58:32 +08:00
int class_name_len ;
* class_name = NULL ;
2003-06-09 02:53:58 +08:00
if ( mangled_property [ 0 ] ! = 0 ) {
* prop_name = mangled_property ;
2006-07-25 01:58:32 +08:00
return SUCCESS ;
}
2006-07-25 06:36:21 +08:00
if ( len < 3 | | mangled_property [ 1 ] = = 0 ) {
2006-07-25 01:58:32 +08:00
zend_error ( E_NOTICE , " Illegal member variable name " ) ;
* prop_name = mangled_property ;
return FAILURE ;
2003-06-09 02:53:58 +08:00
}
2006-07-25 01:58:32 +08:00
class_name_len = zend_strnlen ( mangled_property + 1 , - - len - 1 ) + 1 ;
if ( class_name_len > = len | | mangled_property [ class_name_len ] ! = 0 ) {
zend_error ( E_NOTICE , " Corrupt member variable name " ) ;
* prop_name = mangled_property ;
return FAILURE ;
}
2003-06-09 02:53:58 +08:00
* class_name = mangled_property + 1 ;
2006-07-25 01:58:32 +08:00
* prop_name = ( * class_name ) + class_name_len ;
return SUCCESS ;
2003-06-09 02:53:58 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
2003-06-09 02:53:58 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_declare_property ( const znode * var_name , const znode * value , zend_uint access_type TSRMLS_DC ) /* { { { */
1999-04-08 02:10:10 +08:00
{
2002-02-14 12:01:53 +08:00
zval * property ;
2003-02-10 20:46:58 +08:00
zend_property_info * existing_property_info ;
2005-04-20 06:04:59 +08:00
char * comment = NULL ;
int comment_len = 0 ;
1999-12-27 05:21:33 +08:00
2003-03-05 19:14:44 +08:00
if ( CG ( active_class_entry ) - > ce_flags & ZEND_ACC_INTERFACE ) {
zend_error ( E_COMPILE_ERROR , " Interfaces may not include member variables " ) ;
}
2003-02-11 17:48:37 +08:00
if ( access_type & ZEND_ACC_ABSTRACT ) {
zend_error ( E_COMPILE_ERROR , " Properties cannot be declared abstract " ) ;
}
2003-04-20 22:52:40 +08:00
if ( access_type & ZEND_ACC_FINAL ) {
2007-11-13 00:55:44 +08:00
zend_error ( E_COMPILE_ERROR , " Cannot declare property %s::$%s final, the final modifier is allowed only for methods and classes " ,
2003-04-20 22:52:40 +08:00
CG ( active_class_entry ) - > name , var_name - > u . constant . value . str . val ) ;
}
2003-02-10 20:46:58 +08:00
if ( zend_hash_find ( & CG ( active_class_entry ) - > properties_info , var_name - > u . constant . value . str . val , var_name - > u . constant . value . str . len + 1 , ( void * * ) & existing_property_info ) = = SUCCESS ) {
if ( ! ( existing_property_info - > flags & ZEND_ACC_IMPLICIT_PUBLIC ) ) {
zend_error ( E_COMPILE_ERROR , " Cannot redeclare %s::$%s " , CG ( active_class_entry ) - > name , var_name - > u . constant . value . str . val ) ;
}
}
2002-02-14 12:01:53 +08:00
ALLOC_ZVAL ( property ) ;
1999-04-08 02:10:10 +08:00
2002-02-14 12:01:53 +08:00
if ( value ) {
1999-04-08 02:10:10 +08:00
* property = value - > u . constant ;
2002-02-14 12:01:53 +08:00
} else {
INIT_PZVAL ( property ) ;
2006-05-10 07:53:23 +08:00
Z_TYPE_P ( property ) = IS_NULL ;
2002-02-14 12:01:53 +08:00
}
2005-04-20 06:04:59 +08:00
if ( CG ( doc_comment ) ) {
2005-06-08 02:11:56 +08:00
comment = CG ( doc_comment ) ;
2005-04-20 06:04:59 +08:00
comment_len = CG ( doc_comment_len ) ;
2005-06-08 02:11:56 +08:00
CG ( doc_comment ) = NULL ;
CG ( doc_comment_len ) = 0 ;
2005-04-20 06:04:59 +08:00
}
2010-04-20 19:05:54 +08:00
zend_declare_property_ex ( CG ( active_class_entry ) , CG ( new_interned_string ) ( var_name - > u . constant . value . str . val , var_name - > u . constant . value . str . len + 1 , 0 TSRMLS_CC ) , var_name - > u . constant . value . str . len , property , access_type , comment , comment_len TSRMLS_CC ) ;
2003-07-08 00:22:56 +08:00
efree ( var_name - > u . constant . value . str . val ) ;
1999-04-08 02:10:10 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_declare_class_constant ( znode * var_name , const znode * value TSRMLS_DC ) /* { { { */
2002-12-07 01:09:44 +08:00
{
zval * property ;
2006-05-10 07:53:23 +08:00
2004-01-09 22:02:33 +08:00
if ( Z_TYPE ( value - > u . constant ) = = IS_CONSTANT_ARRAY ) {
zend_error ( E_COMPILE_ERROR , " Arrays are not allowed in class constants " ) ;
}
2006-05-10 07:53:23 +08:00
2002-12-07 01:09:44 +08:00
ALLOC_ZVAL ( property ) ;
2006-03-20 23:13:41 +08:00
* property = value - > u . constant ;
2002-12-07 01:09:44 +08:00
2010-04-20 19:05:54 +08:00
if ( zend_hash_add ( & CG ( active_class_entry ) - > constants_table , CG ( new_interned_string ) ( var_name - > u . constant . value . str . val , var_name - > u . constant . value . str . len + 1 , 0 TSRMLS_CC ) , var_name - > u . constant . value . str . len + 1 , & property , sizeof ( zval * ) , NULL ) = = FAILURE ) {
2004-09-10 14:13:13 +08:00
FREE_ZVAL ( property ) ;
zend_error ( E_COMPILE_ERROR , " Cannot redefine class constant %s::%s " , CG ( active_class_entry ) - > name , var_name - > u . constant . value . str . val ) ;
}
2002-12-07 01:09:44 +08:00
FREE_PNODE ( var_name ) ;
2009-11-15 03:17:22 +08:00
if ( CG ( doc_comment ) ) {
efree ( CG ( doc_comment ) ) ;
CG ( doc_comment ) = NULL ;
CG ( doc_comment_len ) = 0 ;
}
2002-12-07 01:09:44 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
2002-12-07 01:09:44 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_fetch_property ( znode * result , znode * object , const znode * property TSRMLS_DC ) /* { { { */
1999-04-08 02:10:10 +08:00
{
1999-09-20 23:44:30 +08:00
zend_op opline ;
1999-04-08 02:10:10 +08:00
zend_llist * fetch_list_ptr ;
2006-05-10 07:53:23 +08:00
2002-01-05 23:18:30 +08:00
zend_stack_top ( & CG ( bp_stack ) , ( void * * ) & fetch_list_ptr ) ;
2006-05-10 07:53:23 +08:00
2008-05-07 20:04:39 +08:00
if ( object - > op_type = = IS_CV ) {
2010-04-20 18:57:45 +08:00
if ( object - > u . op . var = = CG ( active_op_array ) - > this_var ) {
object - > op_type = IS_UNUSED ; /* this means $this for objects */
2008-05-07 20:04:39 +08:00
}
} else if ( fetch_list_ptr - > count = = 1 ) {
zend_llist_element * le = fetch_list_ptr - > head ;
zend_op * opline_ptr = ( zend_op * ) le - > data ;
2002-10-08 05:20:23 +08:00
2004-02-04 22:25:25 +08:00
if ( opline_is_fetch_this ( opline_ptr TSRMLS_CC ) ) {
2010-04-20 18:57:45 +08:00
zend_del_literal ( CG ( active_op_array ) , opline_ptr - > op1 . constant ) ;
2002-10-17 02:06:36 +08:00
SET_UNUSED ( opline_ptr - > op1 ) ; /* this means $this for objects */
2010-04-20 18:57:45 +08:00
SET_NODE ( opline_ptr - > op2 , property ) ;
2002-10-17 02:06:36 +08:00
/* if it was usual fetch, we change it to object fetch */
2002-11-30 19:20:25 +08:00
switch ( opline_ptr - > opcode ) {
2002-10-17 02:06:36 +08:00
case ZEND_FETCH_W :
opline_ptr - > opcode = ZEND_FETCH_OBJ_W ;
break ;
case ZEND_FETCH_R :
opline_ptr - > opcode = ZEND_FETCH_OBJ_R ;
break ;
case ZEND_FETCH_RW :
opline_ptr - > opcode = ZEND_FETCH_OBJ_RW ;
break ;
case ZEND_FETCH_IS :
opline_ptr - > opcode = ZEND_FETCH_OBJ_IS ;
break ;
case ZEND_FETCH_UNSET :
opline_ptr - > opcode = ZEND_FETCH_OBJ_UNSET ;
break ;
case ZEND_FETCH_FUNC_ARG :
opline_ptr - > opcode = ZEND_FETCH_OBJ_FUNC_ARG ;
break ;
}
2010-04-20 18:57:45 +08:00
if ( opline_ptr - > op2_type = = IS_CONST & & Z_TYPE ( CONSTANT ( opline_ptr - > op2 . constant ) ) = = IS_STRING ) {
2010-04-21 22:58:33 +08:00
CALCULATE_LITERAL_HASH ( opline_ptr - > op2 . constant ) ;
2010-04-20 18:57:45 +08:00
}
GET_NODE ( result , opline_ptr - > result ) ;
2002-01-05 23:18:30 +08:00
return ;
}
}
1999-04-08 02:10:10 +08:00
2001-07-28 18:51:54 +08:00
init_op ( & opline TSRMLS_CC ) ;
1999-09-20 23:44:30 +08:00
opline . opcode = ZEND_FETCH_OBJ_W ; /* the backpatching routine assumes W */
2010-04-20 18:57:45 +08:00
opline . result_type = IS_VAR ;
opline . result . var = get_temporary_variable ( CG ( active_op_array ) ) ;
SET_NODE ( opline . op1 , object ) ;
SET_NODE ( opline . op2 , property ) ;
if ( opline . op2_type = = IS_CONST & & Z_TYPE ( CONSTANT ( opline . op2 . constant ) ) = = IS_STRING ) {
2010-04-21 22:58:33 +08:00
CALCULATE_LITERAL_HASH ( opline . op2 . constant ) ;
2010-04-20 18:57:45 +08:00
}
GET_NODE ( result , opline . result ) ;
1999-04-08 02:10:10 +08:00
1999-09-20 23:44:30 +08:00
zend_llist_add_element ( fetch_list_ptr , & opline ) ;
1999-04-08 02:10:10 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_halt_compiler_register ( TSRMLS_D ) /* { { { */
2007-04-04 08:42:42 +08:00
{
char * name , * cfilename ;
char haltoff [ ] = " __COMPILER_HALT_OFFSET__ " ;
int len , clen ;
cfilename = zend_get_compiled_filename ( TSRMLS_C ) ;
clen = strlen ( cfilename ) ;
2008-03-22 02:20:07 +08:00
zend_mangle_property_name ( & name , & len , haltoff , sizeof ( haltoff ) - 1 , cfilename , clen , 0 ) ;
2007-04-04 08:42:42 +08:00
zend_register_long_constant ( name , len + 1 , zend_get_scanned_file_offset ( TSRMLS_C ) , CONST_CS , 0 TSRMLS_CC ) ;
pefree ( name , 0 ) ;
}
2009-07-27 22:11:53 +08:00
/* }}} */
2003-06-09 21:51:53 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_push_object ( const znode * object TSRMLS_DC ) /* { { { */
1999-04-08 02:10:10 +08:00
{
zend_stack_push ( & CG ( object_stack ) , object , sizeof ( znode ) ) ;
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_pop_object ( znode * object TSRMLS_DC ) /* { { { */
1999-04-08 02:10:10 +08:00
{
2003-06-09 21:51:53 +08:00
if ( object ) {
znode * tmp ;
1999-04-08 02:10:10 +08:00
2003-06-09 21:51:53 +08:00
zend_stack_top ( & CG ( object_stack ) , ( void * * ) & tmp ) ;
* object = * tmp ;
}
1999-04-08 02:10:10 +08:00
zend_stack_del_top ( & CG ( object_stack ) ) ;
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_begin_new_object ( znode * new_token , znode * class_type TSRMLS_DC ) /* { { { */
1999-04-08 02:10:10 +08:00
{
2005-06-10 15:56:40 +08:00
zend_op * opline ;
1999-04-08 02:10:10 +08:00
unsigned char * ptr = NULL ;
2006-05-10 07:53:23 +08:00
2010-04-20 18:57:45 +08:00
new_token - > u . op . opline_num = get_next_op_number ( CG ( active_op_array ) ) ;
2005-06-10 15:56:40 +08:00
opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
1999-04-08 02:10:10 +08:00
opline - > opcode = ZEND_NEW ;
2010-04-20 18:57:45 +08:00
opline - > result_type = IS_VAR ;
opline - > result . var = get_temporary_variable ( CG ( active_op_array ) ) ;
SET_NODE ( opline - > op1 , class_type ) ;
1999-04-08 02:10:10 +08:00
SET_UNUSED ( opline - > op2 ) ;
2006-05-10 07:53:23 +08:00
1999-04-08 02:10:10 +08:00
zend_stack_push ( & CG ( function_call_stack ) , ( void * ) & ptr , sizeof ( unsigned char * ) ) ;
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_end_new_object ( znode * result , const znode * new_token , const znode * argument_list TSRMLS_DC ) /* { { { */
1999-04-08 02:10:10 +08:00
{
znode ctor_result ;
2001-11-05 03:30:49 +08:00
zend_do_end_function_call ( NULL , & ctor_result , argument_list , 1 , 0 TSRMLS_CC ) ;
2001-07-28 18:51:54 +08:00
zend_do_free ( & ctor_result TSRMLS_CC ) ;
1999-04-08 02:10:10 +08:00
2010-04-20 18:57:45 +08:00
CG ( active_op_array ) - > opcodes [ new_token - > u . op . opline_num ] . op2 . opline_num = get_next_op_number ( CG ( active_op_array ) ) ;
GET_NODE ( result , CG ( active_op_array ) - > opcodes [ new_token - > u . op . opline_num ] . result ) ;
1999-04-08 02:10:10 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2008-08-29 18:17:08 +08:00
static zend_constant * zend_get_ct_const ( const zval * const_name , int all_internal_constants_substitution TSRMLS_DC ) /* { { { */
2006-03-15 17:04:36 +08:00
{
zend_constant * c = NULL ;
2008-11-04 23:58:55 +08:00
if ( Z_STRVAL_P ( const_name ) [ 0 ] = = ' \\ ' ) {
if ( zend_hash_find ( EG ( zend_constants ) , Z_STRVAL_P ( const_name ) + 1 , Z_STRLEN_P ( const_name ) , ( void * * ) & c ) = = FAILURE ) {
char * lookup_name = zend_str_tolower_dup ( Z_STRVAL_P ( const_name ) + 1 , Z_STRLEN_P ( const_name ) - 1 ) ;
if ( zend_hash_find ( EG ( zend_constants ) , lookup_name , Z_STRLEN_P ( const_name ) , ( void * * ) & c ) = = SUCCESS ) {
if ( ( c - > flags & CONST_CT_SUBST ) & & ! ( c - > flags & CONST_CS ) ) {
efree ( lookup_name ) ;
return c ;
}
}
efree ( lookup_name ) ;
return NULL ;
}
} else if ( zend_hash_find ( EG ( zend_constants ) , Z_STRVAL_P ( const_name ) , Z_STRLEN_P ( const_name ) + 1 , ( void * * ) & c ) = = FAILURE ) {
2006-03-15 17:04:36 +08:00
char * lookup_name = zend_str_tolower_dup ( Z_STRVAL_P ( const_name ) , Z_STRLEN_P ( const_name ) ) ;
if ( zend_hash_find ( EG ( zend_constants ) , lookup_name , Z_STRLEN_P ( const_name ) + 1 , ( void * * ) & c ) = = SUCCESS ) {
2008-07-28 22:12:19 +08:00
if ( ( c - > flags & CONST_CT_SUBST ) & & ! ( c - > flags & CONST_CS ) ) {
2007-09-29 03:52:53 +08:00
efree ( lookup_name ) ;
2008-07-28 22:12:19 +08:00
return c ;
2006-03-15 17:04:36 +08:00
}
}
efree ( lookup_name ) ;
2008-07-28 22:12:19 +08:00
return NULL ;
2006-03-15 17:04:36 +08:00
}
2007-09-29 03:52:53 +08:00
if ( c - > flags & CONST_CT_SUBST ) {
return c ;
}
2008-08-29 18:17:08 +08:00
if ( all_internal_constants_substitution & &
2008-07-31 22:27:43 +08:00
( c - > flags & CONST_PERSISTENT ) & &
2008-07-25 12:54:08 +08:00
! ( CG ( compiler_options ) & ZEND_COMPILE_NO_CONSTANT_SUBSTITUTION ) & &
Z_TYPE ( c - > value ) ! = IS_CONSTANT & &
Z_TYPE ( c - > value ) ! = IS_CONSTANT_ARRAY ) {
return c ;
}
2007-09-29 03:52:53 +08:00
return NULL ;
}
/* }}} */
2008-08-29 18:17:08 +08:00
static int zend_constant_ct_subst ( znode * result , zval * const_name , int all_internal_constants_substitution TSRMLS_DC ) /* { { { */
2007-09-29 03:52:53 +08:00
{
2008-08-29 18:17:08 +08:00
zend_constant * c = zend_get_ct_const ( const_name , all_internal_constants_substitution TSRMLS_CC ) ;
2007-09-29 03:52:53 +08:00
if ( c ) {
2006-03-15 17:04:36 +08:00
zval_dtor ( const_name ) ;
result - > op_type = IS_CONST ;
result - > u . constant = c - > value ;
zval_copy_ctor ( & result - > u . constant ) ;
INIT_PZVAL ( & result - > u . constant ) ;
return 1 ;
}
return 0 ;
}
2007-09-29 03:52:53 +08:00
/* }}} */
2006-03-15 17:04:36 +08:00
2007-09-29 03:52:53 +08:00
void zend_do_fetch_constant ( znode * result , znode * constant_container , znode * constant_name , int mode , zend_bool check_namespace TSRMLS_DC ) /* { { { */
1999-04-08 02:10:10 +08:00
{
2007-09-29 03:52:53 +08:00
znode tmp ;
2008-11-04 23:58:55 +08:00
zend_op * opline ;
int type ;
2008-11-12 03:45:29 +08:00
char * compound ;
2008-11-04 23:58:55 +08:00
ulong fetch_type = 0 ;
2007-09-29 03:52:53 +08:00
2008-11-04 23:58:55 +08:00
if ( constant_container ) {
switch ( mode ) {
case ZEND_CT :
/* this is a class constant */
type = zend_get_class_fetch_type ( Z_STRVAL ( constant_container - > u . constant ) , Z_STRLEN ( constant_container - > u . constant ) ) ;
2007-10-01 19:25:14 +08:00
if ( ZEND_FETCH_CLASS_STATIC = = type ) {
zend_error ( E_ERROR , " \" static:: \" is not allowed in compile-time constants " ) ;
} else if ( ZEND_FETCH_CLASS_DEFAULT = = type ) {
2007-09-29 03:52:53 +08:00
zend_resolve_class_name ( constant_container , & fetch_type , 1 TSRMLS_CC ) ;
}
2008-11-04 23:58:55 +08:00
zend_do_build_full_name ( NULL , constant_container , constant_name , 1 TSRMLS_CC ) ;
2001-12-01 00:29:47 +08:00
* result = * constant_container ;
2007-09-29 03:52:53 +08:00
result - > u . constant . type = IS_CONSTANT | fetch_type ;
2008-11-04 23:58:55 +08:00
break ;
case ZEND_RT :
if ( constant_container - > op_type = = IS_CONST & &
ZEND_FETCH_CLASS_DEFAULT = = zend_get_class_fetch_type ( Z_STRVAL ( constant_container - > u . constant ) , Z_STRLEN ( constant_container - > u . constant ) ) ) {
zend_resolve_class_name ( constant_container , & fetch_type , 1 TSRMLS_CC ) ;
} else {
zend_do_fetch_class ( & tmp , constant_container TSRMLS_CC ) ;
2007-09-29 03:52:53 +08:00
constant_container = & tmp ;
}
opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
1999-04-08 02:10:10 +08:00
opline - > opcode = ZEND_FETCH_CONSTANT ;
2010-04-20 18:57:45 +08:00
opline - > result_type = IS_TMP_VAR ;
opline - > result . var = get_temporary_variable ( CG ( active_op_array ) ) ;
2010-04-21 22:58:33 +08:00
if ( constant_container - > op_type = = IS_CONST ) {
opline - > op1_type = IS_CONST ;
2010-04-21 23:08:10 +08:00
opline - > op1 . constant = zend_add_class_name_literal ( CG ( active_op_array ) , & constant_container - > u . constant TSRMLS_CC ) ;
2010-04-21 22:58:33 +08:00
} else {
SET_NODE ( opline - > op1 , constant_container ) ;
2010-04-20 18:57:45 +08:00
}
SET_NODE ( opline - > op2 , constant_name ) ;
2010-04-21 22:58:33 +08:00
CALCULATE_LITERAL_HASH ( opline - > op2 . constant ) ;
2010-04-20 18:57:45 +08:00
GET_NODE ( result , opline - > result ) ;
2008-11-04 23:58:55 +08:00
break ;
}
return ;
}
/* namespace constant */
2008-11-12 03:45:29 +08:00
/* only one that did not contain \ from the start can be converted to string if unknown */
2008-11-04 23:58:55 +08:00
switch ( mode ) {
case ZEND_CT :
2008-11-12 03:45:29 +08:00
compound = memchr ( Z_STRVAL ( constant_name - > u . constant ) , ' \\ ' , Z_STRLEN ( constant_name - > u . constant ) ) ;
2008-11-04 23:58:55 +08:00
/* this is a namespace constant, or an unprefixed constant */
if ( zend_constant_ct_subst ( result , & constant_name - > u . constant , 0 TSRMLS_CC ) ) {
break ;
}
2008-11-12 03:45:29 +08:00
zend_resolve_non_class_name ( constant_name , check_namespace TSRMLS_CC ) ;
if ( ! compound ) {
fetch_type | = IS_CONSTANT_UNQUALIFIED ;
2008-11-04 23:58:55 +08:00
}
2008-11-12 03:45:29 +08:00
2008-11-04 23:58:55 +08:00
* result = * constant_name ;
result - > u . constant . type = IS_CONSTANT | fetch_type ;
break ;
case ZEND_RT :
2008-11-12 03:45:29 +08:00
compound = memchr ( Z_STRVAL ( constant_name - > u . constant ) , ' \\ ' , Z_STRLEN ( constant_name - > u . constant ) ) ;
2008-11-04 23:58:55 +08:00
2008-11-12 03:45:29 +08:00
zend_resolve_non_class_name ( constant_name , check_namespace TSRMLS_CC ) ;
if ( zend_constant_ct_subst ( result , & constant_name - > u . constant , 1 TSRMLS_CC ) ) {
break ;
2008-11-04 23:58:55 +08:00
}
opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
opline - > opcode = ZEND_FETCH_CONSTANT ;
2010-04-20 18:57:45 +08:00
opline - > result_type = IS_TMP_VAR ;
opline - > result . var = get_temporary_variable ( CG ( active_op_array ) ) ;
GET_NODE ( result , opline - > result ) ;
2008-11-04 23:58:55 +08:00
SET_UNUSED ( opline - > op1 ) ;
2010-04-22 23:03:17 +08:00
opline - > op2_type = IS_CONST ;
if ( compound ) {
2008-11-12 03:45:29 +08:00
/* the name is unambiguous */
opline - > extended_value = 0 ;
2010-04-22 23:03:17 +08:00
opline - > op2 . constant = zend_add_const_name_literal ( CG ( active_op_array ) , & constant_name - > u . constant , 0 TSRMLS_CC ) ;
} else {
2008-11-12 03:45:29 +08:00
opline - > extended_value = IS_CONSTANT_UNQUALIFIED ;
2010-04-22 23:03:17 +08:00
if ( CG ( current_namespace ) ) {
opline - > extended_value | = IS_CONSTANT_IN_NAMESPACE ;
opline - > op2 . constant = zend_add_const_name_literal ( CG ( active_op_array ) , & constant_name - > u . constant , 1 TSRMLS_CC ) ;
} else {
opline - > op2 . constant = zend_add_const_name_literal ( CG ( active_op_array ) , & constant_name - > u . constant , 0 TSRMLS_CC ) ;
}
2008-11-12 03:45:29 +08:00
}
1999-04-08 02:10:10 +08:00
break ;
}
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_shell_exec ( znode * result , const znode * cmd TSRMLS_DC ) /* { { { */
1999-04-08 02:10:10 +08:00
{
2001-07-28 18:51:54 +08:00
zend_op * opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
1999-04-08 02:10:10 +08:00
switch ( cmd - > op_type ) {
2008-07-26 23:30:28 +08:00
case IS_CONST :
1999-04-08 02:10:10 +08:00
case IS_TMP_VAR :
opline - > opcode = ZEND_SEND_VAL ;
break ;
default :
opline - > opcode = ZEND_SEND_VAR ;
break ;
}
2010-04-20 18:57:45 +08:00
SET_NODE ( opline - > op1 , cmd ) ;
opline - > op2 . opline_num = 0 ;
1999-06-09 02:33:31 +08:00
opline - > extended_value = ZEND_DO_FCALL ;
1999-04-08 02:10:10 +08:00
SET_UNUSED ( opline - > op2 ) ;
2001-08-30 23:31:35 +08:00
/* FIXME: exception support not added to this op2 */
2001-07-28 18:51:54 +08:00
opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
1999-04-08 02:10:10 +08:00
opline - > opcode = ZEND_DO_FCALL ;
2010-04-20 18:57:45 +08:00
opline - > result . var = get_temporary_variable ( CG ( active_op_array ) ) ;
opline - > result_type = IS_VAR ;
LITERAL_STRINGL ( opline - > op1 , estrndup ( " shell_exec " , sizeof ( " shell_exec " ) - 1 ) , sizeof ( " shell_exec " ) - 1 , 0 ) ;
2010-04-21 22:58:33 +08:00
CALCULATE_LITERAL_HASH ( opline - > op1 . constant ) ;
2010-04-20 18:57:45 +08:00
opline - > op1_type = IS_CONST ;
1999-06-01 02:33:12 +08:00
opline - > extended_value = 1 ;
1999-04-08 02:10:10 +08:00
SET_UNUSED ( opline - > op2 ) ;
2010-04-20 18:57:45 +08:00
GET_NODE ( result , opline - > result ) ;
1999-04-08 02:10:10 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_init_array ( znode * result , const znode * expr , const znode * offset , zend_bool is_ref TSRMLS_DC ) /* { { { */
1999-04-08 02:10:10 +08:00
{
2001-07-28 18:51:54 +08:00
zend_op * opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
1999-04-08 02:10:10 +08:00
opline - > opcode = ZEND_INIT_ARRAY ;
2010-04-20 18:57:45 +08:00
opline - > result . var = get_temporary_variable ( CG ( active_op_array ) ) ;
opline - > result_type = IS_TMP_VAR ;
GET_NODE ( result , opline - > result ) ;
1999-04-08 02:10:10 +08:00
if ( expr ) {
2010-04-20 18:57:45 +08:00
SET_NODE ( opline - > op1 , expr ) ;
1999-04-08 02:10:10 +08:00
if ( offset ) {
2010-04-20 18:57:45 +08:00
SET_NODE ( opline - > op2 , offset ) ;
if ( opline - > op2_type = = IS_CONST & & Z_TYPE ( CONSTANT ( opline - > op2 . constant ) ) = = IS_STRING ) {
long index ;
int numeric = 0 ;
ZEND_HANDLE_NUMERIC_EX ( Z_STRVAL ( CONSTANT ( opline - > op2 . constant ) ) , Z_STRLEN ( CONSTANT ( opline - > op2 . constant ) ) + 1 , index , numeric = 1 ) ;
if ( numeric ) {
zval_dtor ( & CONSTANT ( opline - > op2 . constant ) ) ;
ZVAL_LONG ( & CONSTANT ( opline - > op2 . constant ) , index ) ;
} else {
2010-04-21 22:58:33 +08:00
CALCULATE_LITERAL_HASH ( opline - > op2 . constant ) ;
2010-04-20 18:57:45 +08:00
}
}
1999-04-08 02:10:10 +08:00
} else {
SET_UNUSED ( opline - > op2 ) ;
}
} else {
SET_UNUSED ( opline - > op1 ) ;
SET_UNUSED ( opline - > op2 ) ;
}
1999-10-01 18:00:05 +08:00
opline - > extended_value = is_ref ;
1999-04-08 02:10:10 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_add_array_element ( znode * result , const znode * expr , const znode * offset , zend_bool is_ref TSRMLS_DC ) /* { { { */
1999-04-08 02:10:10 +08:00
{
2001-07-28 18:51:54 +08:00
zend_op * opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
1999-04-08 02:10:10 +08:00
opline - > opcode = ZEND_ADD_ARRAY_ELEMENT ;
2010-04-20 18:57:45 +08:00
SET_NODE ( opline - > result , result ) ;
SET_NODE ( opline - > op1 , expr ) ;
1999-04-08 02:10:10 +08:00
if ( offset ) {
2010-04-20 18:57:45 +08:00
SET_NODE ( opline - > op2 , offset ) ;
if ( opline - > op2_type = = IS_CONST & & Z_TYPE ( CONSTANT ( opline - > op2 . constant ) ) = = IS_STRING ) {
long index ;
int numeric = 0 ;
ZEND_HANDLE_NUMERIC_EX ( Z_STRVAL ( CONSTANT ( opline - > op2 . constant ) ) , Z_STRLEN ( CONSTANT ( opline - > op2 . constant ) ) + 1 , index , numeric = 1 ) ;
if ( numeric ) {
zval_dtor ( & CONSTANT ( opline - > op2 . constant ) ) ;
ZVAL_LONG ( & CONSTANT ( opline - > op2 . constant ) , index ) ;
} else {
2010-04-21 22:58:33 +08:00
CALCULATE_LITERAL_HASH ( opline - > op2 . constant ) ;
2010-04-20 18:57:45 +08:00
}
}
1999-04-08 02:10:10 +08:00
} else {
SET_UNUSED ( opline - > op2 ) ;
}
1999-10-01 18:00:05 +08:00
opline - > extended_value = is_ref ;
1999-04-08 02:10:10 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_add_static_array_element ( znode * result , znode * offset , const znode * expr ) /* { { { */
1999-04-08 02:10:10 +08:00
{
1999-12-27 05:21:33 +08:00
zval * element ;
1999-04-08 02:10:10 +08:00
1999-12-27 05:21:33 +08:00
ALLOC_ZVAL ( element ) ;
1999-04-08 02:10:10 +08:00
* element = expr - > u . constant ;
if ( offset ) {
2007-09-29 03:52:53 +08:00
switch ( offset - > u . constant . type & IS_CONSTANT_TYPE_MASK ) {
2000-12-07 18:36:45 +08:00
case IS_CONSTANT :
2001-07-16 19:41:06 +08:00
/* Ugly hack to denote that this value has a constant index */
2006-05-12 05:07:39 +08:00
Z_TYPE_P ( element ) | = IS_CONSTANT_INDEX ;
2007-11-02 18:11:42 +08:00
Z_STRVAL ( offset - > u . constant ) = erealloc ( Z_STRVAL ( offset - > u . constant ) , Z_STRLEN ( offset - > u . constant ) + 3 ) ;
Z_STRVAL ( offset - > u . constant ) [ Z_STRLEN ( offset - > u . constant ) + 1 ] = Z_TYPE ( offset - > u . constant ) ;
Z_STRVAL ( offset - > u . constant ) [ Z_STRLEN ( offset - > u . constant ) + 2 ] = 0 ;
zend_symtable_update ( result - > u . constant . value . ht , Z_STRVAL ( offset - > u . constant ) , Z_STRLEN ( offset - > u . constant ) + 3 , & element , sizeof ( zval * ) , NULL ) ;
zval_dtor ( & offset - > u . constant ) ;
break ;
1999-04-08 02:10:10 +08:00
case IS_STRING :
2003-07-24 21:14:57 +08:00
zend_symtable_update ( result - > u . constant . value . ht , offset - > u . constant . value . str . val , offset - > u . constant . value . str . len + 1 , & element , sizeof ( zval * ) , NULL ) ;
1999-06-23 03:05:40 +08:00
zval_dtor ( & offset - > u . constant ) ;
1999-04-08 02:10:10 +08:00
break ;
2005-06-24 17:24:42 +08:00
case IS_NULL :
2006-05-12 05:07:39 +08:00
zend_symtable_update ( Z_ARRVAL ( result - > u . constant ) , " " , 1 , & element , sizeof ( zval * ) , NULL ) ;
2005-06-24 17:24:42 +08:00
break ;
1999-04-08 02:10:10 +08:00
case IS_LONG :
2006-05-12 05:07:39 +08:00
case IS_BOOL :
zend_hash_index_update ( Z_ARRVAL ( result - > u . constant ) , Z_LVAL ( offset - > u . constant ) , & element , sizeof ( zval * ) , NULL ) ;
2005-06-24 17:24:42 +08:00
break ;
case IS_DOUBLE :
2009-06-05 02:20:45 +08:00
zend_hash_index_update ( Z_ARRVAL ( result - > u . constant ) , zend_dval_to_lval ( Z_DVAL ( offset - > u . constant ) ) , & element , sizeof ( zval * ) , NULL ) ;
1999-04-08 02:10:10 +08:00
break ;
2006-02-24 02:06:47 +08:00
case IS_CONSTANT_ARRAY :
zend_error ( E_ERROR , " Illegal offset type " ) ;
break ;
1999-04-08 02:10:10 +08:00
}
} else {
2006-05-12 05:07:39 +08:00
zend_hash_next_index_insert ( Z_ARRVAL ( result - > u . constant ) , & element , sizeof ( zval * ) , NULL ) ;
1999-04-08 02:10:10 +08:00
}
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_add_list_element ( const znode * element TSRMLS_DC ) /* { { { */
1999-04-08 02:10:10 +08:00
{
list_llist_element lle ;
1999-05-23 00:10:51 +08:00
if ( element ) {
2002-03-18 03:13:46 +08:00
zend_check_writable_variable ( element ) ;
1999-04-08 02:10:10 +08:00
lle . var = * element ;
zend_llist_copy ( & lle . dimensions , & CG ( dimension_llist ) ) ;
2000-02-06 04:19:46 +08:00
zend_llist_prepend_element ( & CG ( list_llist ) , & lle ) ;
1999-04-08 02:10:10 +08:00
}
( * ( ( int * ) CG ( dimension_llist ) . tail - > data ) ) + + ;
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_new_list_begin ( TSRMLS_D ) /* { { { */
1999-04-08 02:10:10 +08:00
{
int current_dimension = 0 ;
zend_llist_add_element ( & CG ( dimension_llist ) , & current_dimension ) ;
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_new_list_end ( TSRMLS_D ) /* { { { */
1999-04-08 02:10:10 +08:00
{
zend_llist_remove_tail ( & CG ( dimension_llist ) ) ;
( * ( ( int * ) CG ( dimension_llist ) . tail - > data ) ) + + ;
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_list_init ( TSRMLS_D ) /* { { { */
1999-04-08 02:10:10 +08:00
{
2001-07-16 23:48:31 +08:00
zend_stack_push ( & CG ( list_stack ) , & CG ( list_llist ) , sizeof ( zend_llist ) ) ;
zend_stack_push ( & CG ( list_stack ) , & CG ( dimension_llist ) , sizeof ( zend_llist ) ) ;
1999-04-08 02:10:10 +08:00
zend_llist_init ( & CG ( list_llist ) , sizeof ( list_llist_element ) , NULL , 0 ) ;
zend_llist_init ( & CG ( dimension_llist ) , sizeof ( int ) , NULL , 0 ) ;
2001-07-28 18:51:54 +08:00
zend_do_new_list_begin ( TSRMLS_C ) ;
1999-04-08 02:10:10 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_list_end ( znode * result , znode * expr TSRMLS_DC ) /* { { { */
1999-04-08 02:10:10 +08:00
{
zend_llist_element * le ;
zend_llist_element * dimension ;
zend_op * opline ;
znode last_container ;
le = CG ( list_llist ) . head ;
while ( le ) {
zend_llist * tmp_dimension_llist = & ( ( list_llist_element * ) le - > data ) - > dimensions ;
dimension = tmp_dimension_llist - > head ;
while ( dimension ) {
2001-07-28 18:51:54 +08:00
opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
1999-04-08 02:10:10 +08:00
if ( dimension = = tmp_dimension_llist - > head ) { /* first */
last_container = * expr ;
2002-11-30 19:20:25 +08:00
switch ( expr - > op_type ) {
1999-04-08 02:10:10 +08:00
case IS_VAR :
2004-10-05 03:54:35 +08:00
case IS_CV :
1999-04-08 02:10:10 +08:00
opline - > opcode = ZEND_FETCH_DIM_R ;
break ;
case IS_TMP_VAR :
2000-10-19 17:36:53 +08:00
opline - > opcode = ZEND_FETCH_DIM_TMP_VAR ;
break ;
1999-04-08 02:10:10 +08:00
case IS_CONST : /* fetch_dim_tmp_var will handle this bogus fetch */
2000-10-19 17:36:53 +08:00
zval_copy_ctor ( & expr - > u . constant ) ;
1999-04-08 02:10:10 +08:00
opline - > opcode = ZEND_FETCH_DIM_TMP_VAR ;
break ;
}
2010-04-20 18:57:45 +08:00
opline - > extended_value | = ZEND_FETCH_ADD_LOCK ;
1999-04-08 02:10:10 +08:00
} else {
opline - > opcode = ZEND_FETCH_DIM_R ;
}
2010-04-20 18:57:45 +08:00
opline - > result_type = IS_VAR ;
opline - > result . var = get_temporary_variable ( CG ( active_op_array ) ) ;
SET_NODE ( opline - > op1 , & last_container ) ;
opline - > op2_type = IS_CONST ;
LITERAL_LONG ( opline - > op2 , * ( ( int * ) dimension - > data ) ) ;
GET_NODE ( & last_container , opline - > result ) ;
1999-04-08 02:10:10 +08:00
dimension = dimension - > next ;
}
( ( list_llist_element * ) le - > data ) - > value = last_container ;
zend_llist_destroy ( & ( ( list_llist_element * ) le - > data ) - > dimensions ) ;
2001-07-28 18:51:54 +08:00
zend_do_assign ( result , & ( ( list_llist_element * ) le - > data ) - > var , & ( ( list_llist_element * ) le - > data ) - > value TSRMLS_CC ) ;
2005-06-03 21:57:26 +08:00
zend_do_free ( result TSRMLS_CC ) ;
1999-04-08 02:10:10 +08:00
le = le - > next ;
}
zend_llist_destroy ( & CG ( dimension_llist ) ) ;
zend_llist_destroy ( & CG ( list_llist ) ) ;
* result = * expr ;
2001-07-16 23:48:31 +08:00
{
zend_llist * p ;
/* restore previous lists */
zend_stack_top ( & CG ( list_stack ) , ( void * * ) & p ) ;
CG ( dimension_llist ) = * p ;
zend_stack_del_top ( & CG ( list_stack ) ) ;
zend_stack_top ( & CG ( list_stack ) , ( void * * ) & p ) ;
CG ( list_llist ) = * p ;
zend_stack_del_top ( & CG ( list_stack ) ) ;
}
1999-04-08 02:10:10 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
void zend_init_list ( void * result , void * item TSRMLS_DC ) /* { { { */
{
void * * list = emalloc ( sizeof ( void * ) * 2 ) ;
list [ 0 ] = item ;
list [ 1 ] = NULL ;
* ( void * * ) result = list ;
}
/* }}} */
void zend_add_to_list ( void * result , void * item TSRMLS_DC ) /* { { { */
{
void * * list = * ( void * * ) result ;
size_t n = 0 ;
while ( list & & list [ n ] ) { n + + ; }
list = erealloc ( list , sizeof ( void * ) * ( n + 2 ) ) ;
list [ n ] = item ;
list [ n + 1 ] = NULL ;
* ( void * * ) result = list ;
}
/* }}} */
2009-07-27 22:11:53 +08:00
void zend_do_fetch_static_variable ( znode * varname , const znode * static_assignment , int fetch_type TSRMLS_DC ) /* { { { */
1999-04-08 02:10:10 +08:00
{
2002-11-11 01:50:27 +08:00
zval * tmp ;
2003-01-20 01:25:39 +08:00
zend_op * opline ;
znode lval ;
znode result ;
1999-04-08 02:10:10 +08:00
2002-11-11 01:50:27 +08:00
ALLOC_ZVAL ( tmp ) ;
2003-12-06 00:28:54 +08:00
2002-11-30 19:20:25 +08:00
if ( static_assignment ) {
1999-04-08 02:10:10 +08:00
* tmp = static_assignment - > u . constant ;
2002-11-11 01:50:27 +08:00
} else {
INIT_ZVAL ( * tmp ) ;
}
if ( ! CG ( active_op_array ) - > static_variables ) {
ALLOC_HASHTABLE ( CG ( active_op_array ) - > static_variables ) ;
zend_hash_init ( CG ( active_op_array ) - > static_variables , 2 , NULL , ZVAL_PTR_DTOR , 0 ) ;
1999-04-08 02:10:10 +08:00
}
2002-11-11 01:50:27 +08:00
zend_hash_update ( CG ( active_op_array ) - > static_variables , varname - > u . constant . value . str . val , varname - > u . constant . value . str . len + 1 , & tmp , sizeof ( zval * ) , NULL ) ;
2003-01-20 01:25:39 +08:00
2008-01-24 01:55:55 +08:00
if ( varname - > op_type = = IS_CONST ) {
if ( Z_TYPE ( varname - > u . constant ) ! = IS_STRING ) {
convert_to_string ( & varname - > u . constant ) ;
}
}
2003-01-20 01:25:39 +08:00
opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
2008-07-14 17:49:03 +08:00
opline - > opcode = ( fetch_type = = ZEND_FETCH_LEXICAL ) ? ZEND_FETCH_R : ZEND_FETCH_W ; /* the default mode must be Write, since fetch_simple_variable() is used to define function arguments */
2010-04-20 18:57:45 +08:00
opline - > result_type = IS_VAR ;
opline - > result . var = get_temporary_variable ( CG ( active_op_array ) ) ;
SET_NODE ( opline - > op1 , varname ) ;
if ( opline - > op1_type = = IS_CONST ) {
2010-04-21 22:58:33 +08:00
CALCULATE_LITERAL_HASH ( opline - > op1 . constant ) ;
2010-04-20 18:57:45 +08:00
}
2003-01-20 01:25:39 +08:00
SET_UNUSED ( opline - > op2 ) ;
2010-04-20 18:57:45 +08:00
opline - > extended_value = ZEND_FETCH_STATIC ;
GET_NODE ( & result , opline - > result ) ;
2003-01-20 01:25:39 +08:00
if ( varname - > op_type = = IS_CONST ) {
zval_copy_ctor ( & varname - > u . constant ) ;
}
fetch_simple_variable ( & lval , varname , 0 TSRMLS_CC ) ; /* Relies on the fact that the default fetch is BP_VAR_W */
2008-07-14 17:49:03 +08:00
if ( fetch_type = = ZEND_FETCH_LEXICAL ) {
znode dummy ;
zend_do_begin_variable_parse ( TSRMLS_C ) ;
zend_do_assign ( & dummy , & lval , & result TSRMLS_CC ) ;
zend_do_free ( & dummy TSRMLS_CC ) ;
} else {
zend_do_assign_ref ( NULL , & lval , & result TSRMLS_CC ) ;
}
2010-04-20 18:57:45 +08:00
CG ( active_op_array ) - > opcodes [ CG ( active_op_array ) - > last - 1 ] . result_type | = EXT_TYPE_UNUSED ;
2002-11-11 01:50:27 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_fetch_lexical_variable ( znode * varname , zend_bool is_ref TSRMLS_DC ) /* { { { */
2008-07-14 17:49:03 +08:00
{
znode value ;
if ( Z_STRLEN ( varname - > u . constant ) = = sizeof ( " this " ) - 1 & &
memcmp ( Z_STRVAL ( varname - > u . constant ) , " this " , sizeof ( " this " ) - 1 ) = = 0 ) {
zend_error ( E_COMPILE_ERROR , " Cannot use $this as lexical variable " ) ;
return ;
}
value . op_type = IS_CONST ;
ZVAL_NULL ( & value . u . constant ) ;
Z_TYPE ( value . u . constant ) | = is_ref ? IS_LEXICAL_REF : IS_LEXICAL_VAR ;
Z_SET_REFCOUNT_P ( & value . u . constant , 1 ) ;
Z_UNSET_ISREF_P ( & value . u . constant ) ;
zend_do_fetch_static_variable ( varname , & value , is_ref ? ZEND_FETCH_STATIC : ZEND_FETCH_LEXICAL TSRMLS_CC ) ;
}
2009-07-27 22:11:53 +08:00
/* }}} */
2008-07-14 17:49:03 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_fetch_global_variable ( znode * varname , const znode * static_assignment , int fetch_type TSRMLS_DC ) /* { { { */
2002-11-11 01:50:27 +08:00
{
zend_op * opline ;
znode lval ;
znode result ;
1999-04-08 02:10:10 +08:00
2008-01-24 01:55:55 +08:00
if ( varname - > op_type = = IS_CONST ) {
if ( Z_TYPE ( varname - > u . constant ) ! = IS_STRING ) {
convert_to_string ( & varname - > u . constant ) ;
}
}
2002-11-11 01:50:27 +08:00
opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
1999-04-08 02:10:10 +08:00
opline - > opcode = ZEND_FETCH_W ; /* the default mode must be Write, since fetch_simple_variable() is used to define function arguments */
2010-04-20 18:57:45 +08:00
opline - > result_type = IS_VAR ;
opline - > result . var = get_temporary_variable ( CG ( active_op_array ) ) ;
SET_NODE ( opline - > op1 , varname ) ;
if ( opline - > op1_type = = IS_CONST ) {
2010-04-21 22:58:33 +08:00
CALCULATE_LITERAL_HASH ( opline - > op1 . constant ) ;
2010-04-20 18:57:45 +08:00
}
1999-04-08 02:10:10 +08:00
SET_UNUSED ( opline - > op2 ) ;
2010-04-20 18:57:45 +08:00
opline - > extended_value = fetch_type ;
GET_NODE ( & result , opline - > result ) ;
1999-04-08 02:10:10 +08:00
if ( varname - > op_type = = IS_CONST ) {
zval_copy_ctor ( & varname - > u . constant ) ;
}
2001-07-28 18:51:54 +08:00
fetch_simple_variable ( & lval , varname , 0 TSRMLS_CC ) ; /* Relies on the fact that the default fetch is BP_VAR_W */
1999-04-08 02:10:10 +08:00
2001-07-28 18:51:54 +08:00
zend_do_assign_ref ( NULL , & lval , & result TSRMLS_CC ) ;
2010-04-20 18:57:45 +08:00
CG ( active_op_array ) - > opcodes [ CG ( active_op_array ) - > last - 1 ] . result_type | = EXT_TYPE_UNUSED ;
1999-04-08 02:10:10 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_cast ( znode * result , const znode * expr , int type TSRMLS_DC ) /* { { { */
1999-04-08 02:10:10 +08:00
{
2001-07-28 18:51:54 +08:00
zend_op * opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
1999-04-08 02:10:10 +08:00
opline - > opcode = ZEND_CAST ;
2010-04-20 18:57:45 +08:00
opline - > result_type = IS_TMP_VAR ;
opline - > result . var = get_temporary_variable ( CG ( active_op_array ) ) ;
SET_NODE ( opline - > op1 , expr ) ;
1999-04-08 02:10:10 +08:00
SET_UNUSED ( opline - > op2 ) ;
2000-05-04 01:56:44 +08:00
opline - > extended_value = type ;
2010-04-20 18:57:45 +08:00
GET_NODE ( result , opline - > result ) ;
1999-04-08 02:10:10 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_include_or_eval ( int type , znode * result , const znode * op1 TSRMLS_DC ) /* { { { */
1999-04-08 02:10:10 +08:00
{
2001-07-28 18:51:54 +08:00
zend_do_extended_fcall_begin ( TSRMLS_C ) ;
2000-09-11 23:15:57 +08:00
{
2001-07-28 18:51:54 +08:00
zend_op * opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
1999-04-08 02:10:10 +08:00
2000-09-11 23:15:57 +08:00
opline - > opcode = ZEND_INCLUDE_OR_EVAL ;
2010-04-20 18:57:45 +08:00
opline - > result_type = IS_VAR ;
opline - > result . var = get_temporary_variable ( CG ( active_op_array ) ) ;
SET_NODE ( opline - > op1 , op1 ) ;
2000-09-11 23:15:57 +08:00
SET_UNUSED ( opline - > op2 ) ;
2010-04-20 18:57:45 +08:00
opline - > extended_value = type ;
GET_NODE ( result , opline - > result ) ;
1999-12-23 23:03:25 +08:00
}
2001-07-28 18:51:54 +08:00
zend_do_extended_fcall_end ( TSRMLS_C ) ;
1999-12-23 23:03:25 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-12-23 23:03:25 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_indirect_references ( znode * result , const znode * num_references , znode * variable TSRMLS_DC ) /* { { { */
1999-04-08 02:10:10 +08:00
{
int i ;
2008-05-07 20:04:39 +08:00
zend_do_end_variable_parse ( variable , BP_VAR_R , 0 TSRMLS_CC ) ;
1999-09-20 22:45:36 +08:00
for ( i = 1 ; i < num_references - > u . constant . value . lval ; i + + ) {
2001-07-28 18:51:54 +08:00
fetch_simple_variable_ex ( result , variable , 0 , ZEND_FETCH_R TSRMLS_CC ) ;
1999-04-08 02:10:10 +08:00
* variable = * result ;
}
2001-07-28 18:51:54 +08:00
zend_do_begin_variable_parse ( TSRMLS_C ) ;
fetch_simple_variable ( result , variable , 1 TSRMLS_CC ) ;
1999-04-08 02:10:10 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_unset ( const znode * variable TSRMLS_DC ) /* { { { */
1999-04-08 02:10:10 +08:00
{
zend_op * last_op ;
2002-03-18 03:13:46 +08:00
zend_check_writable_variable ( variable ) ;
2004-10-05 03:54:35 +08:00
if ( variable - > op_type = = IS_CV ) {
zend_op * opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
opline - > opcode = ZEND_UNSET_VAR ;
2010-04-20 18:57:45 +08:00
SET_NODE ( opline - > op1 , variable ) ;
2004-10-05 03:54:35 +08:00
SET_UNUSED ( opline - > op2 ) ;
SET_UNUSED ( opline - > result ) ;
2010-04-20 18:57:45 +08:00
opline - > extended_value = ZEND_FETCH_LOCAL | ZEND_QUICK_SET ;
2004-10-05 03:54:35 +08:00
} else {
last_op = & CG ( active_op_array ) - > opcodes [ get_next_op_number ( CG ( active_op_array ) ) - 1 ] ;
1999-04-08 02:10:10 +08:00
2004-10-05 03:54:35 +08:00
switch ( last_op - > opcode ) {
case ZEND_FETCH_UNSET :
last_op - > opcode = ZEND_UNSET_VAR ;
break ;
case ZEND_FETCH_DIM_UNSET :
2005-06-16 20:17:39 +08:00
last_op - > opcode = ZEND_UNSET_DIM ;
2004-10-05 03:54:35 +08:00
break ;
case ZEND_FETCH_OBJ_UNSET :
2005-06-16 20:17:39 +08:00
last_op - > opcode = ZEND_UNSET_OBJ ;
2004-10-05 03:54:35 +08:00
break ;
1999-04-08 02:10:10 +08:00
2004-10-05 03:54:35 +08:00
}
2006-05-10 07:53:23 +08:00
}
1999-04-08 02:10:10 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_isset_or_isempty ( int type , znode * result , znode * variable TSRMLS_DC ) /* { { { */
1999-04-08 02:10:10 +08:00
{
2002-02-05 03:29:56 +08:00
zend_op * last_op ;
1999-04-08 02:10:10 +08:00
2008-05-07 20:04:39 +08:00
zend_do_end_variable_parse ( variable , BP_VAR_IS , 0 TSRMLS_CC ) ;
2001-12-26 22:46:18 +08:00
2002-03-18 03:13:46 +08:00
zend_check_writable_variable ( variable ) ;
2006-05-10 07:53:23 +08:00
2004-10-05 03:54:35 +08:00
if ( variable - > op_type = = IS_CV ) {
last_op = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
last_op - > opcode = ZEND_ISSET_ISEMPTY_VAR ;
2010-04-20 18:57:45 +08:00
SET_NODE ( last_op - > op1 , variable ) ;
2004-10-05 03:54:35 +08:00
SET_UNUSED ( last_op - > op2 ) ;
2010-04-20 18:57:45 +08:00
last_op - > result . var = get_temporary_variable ( CG ( active_op_array ) ) ;
last_op - > extended_value = ZEND_FETCH_LOCAL | ZEND_QUICK_SET ;
2004-10-05 03:54:35 +08:00
} else {
last_op = & CG ( active_op_array ) - > opcodes [ get_next_op_number ( CG ( active_op_array ) ) - 1 ] ;
2006-05-10 07:53:23 +08:00
2004-10-05 03:54:35 +08:00
switch ( last_op - > opcode ) {
case ZEND_FETCH_IS :
last_op - > opcode = ZEND_ISSET_ISEMPTY_VAR ;
break ;
case ZEND_FETCH_DIM_IS :
last_op - > opcode = ZEND_ISSET_ISEMPTY_DIM_OBJ ;
break ;
case ZEND_FETCH_OBJ_IS :
last_op - > opcode = ZEND_ISSET_ISEMPTY_PROP_OBJ ;
break ;
}
2002-02-05 03:29:56 +08:00
}
2010-04-20 18:57:45 +08:00
last_op - > result_type = IS_TMP_VAR ;
2008-05-12 17:09:05 +08:00
last_op - > extended_value | = type ;
2002-02-05 03:29:56 +08:00
2010-04-20 18:57:45 +08:00
GET_NODE ( result , last_op - > result ) ;
1999-04-08 02:10:10 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_instanceof ( znode * result , const znode * expr , const znode * class_znode , int type TSRMLS_DC ) /* { { { */
2002-07-30 12:07:15 +08:00
{
2005-09-09 14:48:49 +08:00
int last_op_number = get_next_op_number ( CG ( active_op_array ) ) ;
zend_op * opline ;
2002-07-30 12:07:15 +08:00
2005-09-09 14:48:49 +08:00
if ( last_op_number > 0 ) {
opline = & CG ( active_op_array ) - > opcodes [ last_op_number - 1 ] ;
if ( opline - > opcode = = ZEND_FETCH_CLASS ) {
opline - > extended_value | = ZEND_FETCH_CLASS_NO_AUTOLOAD ;
}
}
2006-12-13 07:25:23 +08:00
if ( expr - > op_type = = IS_CONST ) {
zend_error ( E_COMPILE_ERROR , " instanceof expects an object instance, constant given " ) ;
}
2005-09-09 14:48:49 +08:00
opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
2003-01-15 05:29:23 +08:00
opline - > opcode = ZEND_INSTANCEOF ;
2010-04-20 18:57:45 +08:00
opline - > result_type = IS_TMP_VAR ;
opline - > result . var = get_temporary_variable ( CG ( active_op_array ) ) ;
SET_NODE ( opline - > op1 , expr ) ;
2002-07-30 12:07:15 +08:00
2010-04-20 18:57:45 +08:00
SET_NODE ( opline - > op2 , class_znode ) ;
2002-07-30 12:07:15 +08:00
2010-04-20 18:57:45 +08:00
GET_NODE ( result , opline - > result ) ;
2002-07-30 12:07:15 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
2002-07-30 12:07:15 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_foreach_begin ( znode * foreach_token , znode * open_brackets_token , znode * array , znode * as_token , int variable TSRMLS_DC ) /* { { { */
1999-04-08 02:10:10 +08:00
{
2001-08-08 23:07:11 +08:00
zend_op * opline ;
zend_bool is_variable ;
2004-02-26 01:23:50 +08:00
zend_bool push_container = 0 ;
2007-01-10 23:58:08 +08:00
zend_op dummy_opline ;
2001-08-08 23:07:11 +08:00
if ( variable ) {
2001-12-26 22:46:18 +08:00
if ( zend_is_function_or_method_call ( array ) ) {
2001-08-08 23:07:11 +08:00
is_variable = 0 ;
} else {
is_variable = 1 ;
}
2007-01-10 23:58:08 +08:00
/* save the location of FETCH_W instruction(s) */
2010-04-20 18:57:45 +08:00
open_brackets_token - > u . op . opline_num = get_next_op_number ( CG ( active_op_array ) ) ;
2008-05-07 20:04:39 +08:00
zend_do_end_variable_parse ( array , BP_VAR_W , 0 TSRMLS_CC ) ;
2005-08-04 22:04:36 +08:00
if ( CG ( active_op_array ) - > last > 0 & &
CG ( active_op_array ) - > opcodes [ CG ( active_op_array ) - > last - 1 ] . opcode = = ZEND_FETCH_OBJ_W ) {
2004-03-02 16:28:12 +08:00
/* Only lock the container if we are fetching from a real container and not $this */
2010-04-20 18:57:45 +08:00
if ( CG ( active_op_array ) - > opcodes [ CG ( active_op_array ) - > last - 1 ] . op1_type = = IS_VAR ) {
2004-03-02 16:28:12 +08:00
CG ( active_op_array ) - > opcodes [ CG ( active_op_array ) - > last - 1 ] . extended_value | = ZEND_FETCH_ADD_LOCK ;
push_container = 1 ;
}
2004-02-25 22:56:45 +08:00
}
2001-08-08 23:07:11 +08:00
} else {
is_variable = 0 ;
2010-04-20 18:57:45 +08:00
open_brackets_token - > u . op . opline_num = get_next_op_number ( CG ( active_op_array ) ) ;
2001-08-08 23:07:11 +08:00
}
2005-02-12 06:26:45 +08:00
/* save the location of FE_RESET */
2010-04-20 18:57:45 +08:00
foreach_token - > u . op . opline_num = get_next_op_number ( CG ( active_op_array ) ) ;
2005-02-12 06:26:45 +08:00
2001-08-08 23:07:11 +08:00
opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
1999-04-08 02:10:10 +08:00
/* Preform array reset */
opline - > opcode = ZEND_FE_RESET ;
2010-04-20 18:57:45 +08:00
opline - > result_type = IS_VAR ;
opline - > result . var = get_temporary_variable ( CG ( active_op_array ) ) ;
SET_NODE ( opline - > op1 , array ) ;
1999-04-08 02:10:10 +08:00
SET_UNUSED ( opline - > op2 ) ;
2006-05-12 05:07:39 +08:00
opline - > extended_value = is_variable ? ZEND_FE_RESET_VARIABLE : 0 ;
1999-04-08 02:10:10 +08:00
2010-04-20 18:57:45 +08:00
COPY_NODE ( dummy_opline . result , opline - > result ) ;
2007-01-10 23:58:08 +08:00
if ( push_container ) {
2010-04-20 18:57:45 +08:00
COPY_NODE ( dummy_opline . op1 , CG ( active_op_array ) - > opcodes [ CG ( active_op_array ) - > last - 2 ] . op1 ) ;
2007-01-10 23:58:08 +08:00
} else {
2010-04-20 18:57:45 +08:00
dummy_opline . op1_type = IS_UNUSED ;
2004-02-26 01:23:50 +08:00
}
2007-01-10 23:58:08 +08:00
zend_stack_push ( & CG ( foreach_copy_stack ) , ( void * ) & dummy_opline , sizeof ( zend_op ) ) ;
2006-05-10 07:53:23 +08:00
2005-02-12 06:26:45 +08:00
/* save the location of FE_FETCH */
2010-04-20 18:57:45 +08:00
as_token - > u . op . opline_num = get_next_op_number ( CG ( active_op_array ) ) ;
2005-02-07 23:22:38 +08:00
2001-07-28 18:51:54 +08:00
opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
1999-04-08 02:10:10 +08:00
opline - > opcode = ZEND_FE_FETCH ;
2010-04-20 18:57:45 +08:00
opline - > result_type = IS_VAR ;
opline - > result . var = get_temporary_variable ( CG ( active_op_array ) ) ;
COPY_NODE ( opline - > op1 , dummy_opline . result ) ;
2003-07-24 20:38:33 +08:00
opline - > extended_value = 0 ;
1999-04-08 02:10:10 +08:00
SET_UNUSED ( opline - > op2 ) ;
2005-02-08 00:09:54 +08:00
opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
opline - > opcode = ZEND_OP_DATA ;
2005-02-12 06:26:45 +08:00
SET_UNUSED ( opline - > op1 ) ;
SET_UNUSED ( opline - > op2 ) ;
SET_UNUSED ( opline - > result ) ;
2005-02-07 23:22:38 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
2005-02-07 23:22:38 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_foreach_cont ( znode * foreach_token , const znode * open_brackets_token , const znode * as_token , znode * value , znode * key TSRMLS_DC ) /* { { { */
2005-02-07 23:22:38 +08:00
{
zend_op * opline ;
2005-02-20 18:19:11 +08:00
znode dummy , value_node ;
2005-02-07 23:22:38 +08:00
zend_bool assign_by_ref = 0 ;
1999-04-08 02:10:10 +08:00
2010-04-20 18:57:45 +08:00
opline = & CG ( active_op_array ) - > opcodes [ as_token - > u . op . opline_num ] ;
1999-04-08 02:10:10 +08:00
if ( key - > op_type ! = IS_UNUSED ) {
znode * tmp ;
/* switch between the key and value... */
tmp = key ;
key = value ;
value = tmp ;
2004-08-12 13:41:01 +08:00
/* Mark extended_value in case both key and value are being used */
2005-02-08 00:09:54 +08:00
opline - > extended_value | = ZEND_FE_FETCH_WITH_KEY ;
1999-04-08 02:10:10 +08:00
}
2010-04-20 18:57:45 +08:00
if ( ( key - > op_type ! = IS_UNUSED ) & & ( key - > EA & ZEND_PARSED_REFERENCE_VARIABLE ) ) {
2003-07-24 20:38:33 +08:00
zend_error ( E_COMPILE_ERROR , " Key element cannot be a reference " ) ;
}
2005-02-08 00:09:54 +08:00
2010-04-20 18:57:45 +08:00
if ( value - > EA & ZEND_PARSED_REFERENCE_VARIABLE ) {
2003-07-24 20:38:33 +08:00
assign_by_ref = 1 ;
2005-02-08 00:09:54 +08:00
if ( ! ( opline - 1 ) - > extended_value ) {
2003-07-24 20:38:33 +08:00
zend_error ( E_COMPILE_ERROR , " Cannot create references to elements of a temporary array expression " ) ;
}
2004-08-12 13:41:01 +08:00
/* Mark extended_value for assign-by-reference */
2005-02-08 00:09:54 +08:00
opline - > extended_value | = ZEND_FE_FETCH_BYREF ;
2010-04-20 18:57:45 +08:00
CG ( active_op_array ) - > opcodes [ foreach_token - > u . op . opline_num ] . extended_value | = ZEND_FE_RESET_REFERENCE ;
2007-01-10 23:58:08 +08:00
} else {
zend_op * foreach_copy ;
2010-04-20 18:57:45 +08:00
zend_op * fetch = & CG ( active_op_array ) - > opcodes [ foreach_token - > u . op . opline_num ] ;
zend_op * end = & CG ( active_op_array ) - > opcodes [ open_brackets_token - > u . op . opline_num ] ;
2007-01-10 23:58:08 +08:00
/* Change "write context" into "read context" */
fetch - > extended_value = 0 ; /* reset ZEND_FE_RESET_VARIABLE */
while ( fetch ! = end ) {
2007-05-11 17:39:36 +08:00
- - fetch ;
2010-04-20 18:57:45 +08:00
if ( fetch - > opcode = = ZEND_FETCH_DIM_W & & fetch - > op2_type = = IS_UNUSED ) {
2007-05-11 17:39:36 +08:00
zend_error ( E_COMPILE_ERROR , " Cannot use [] for reading " ) ;
}
fetch - > opcode - = 3 ; /* FETCH_W -> FETCH_R */
2007-01-10 23:58:08 +08:00
}
/* prevent double SWITCH_FREE */
zend_stack_top ( & CG ( foreach_copy_stack ) , ( void * * ) & foreach_copy ) ;
2010-04-20 18:57:45 +08:00
foreach_copy - > op1_type = IS_UNUSED ;
1999-04-08 02:10:10 +08:00
}
2010-04-20 18:57:45 +08:00
GET_NODE ( & value_node , opline - > result ) ;
2006-05-12 05:07:39 +08:00
if ( assign_by_ref ) {
2008-05-07 20:04:39 +08:00
zend_do_end_variable_parse ( value , BP_VAR_W , 0 TSRMLS_CC ) ;
2005-02-08 00:09:54 +08:00
/* Mark FE_FETCH as IS_VAR as it holds the data directly as a value */
2005-02-20 18:19:11 +08:00
zend_do_assign_ref ( NULL , value , & value_node TSRMLS_CC ) ;
2003-07-24 20:38:33 +08:00
} else {
2005-02-20 18:19:11 +08:00
zend_do_assign ( & dummy , value , & value_node TSRMLS_CC ) ;
2005-02-08 00:09:54 +08:00
zend_do_free ( & dummy TSRMLS_CC ) ;
2003-07-24 20:38:33 +08:00
}
2005-02-08 00:09:54 +08:00
1999-04-08 02:10:10 +08:00
if ( key - > op_type ! = IS_UNUSED ) {
2005-02-20 18:19:11 +08:00
znode key_node ;
2010-04-20 18:57:45 +08:00
opline = & CG ( active_op_array ) - > opcodes [ as_token - > u . op . opline_num + 1 ] ;
opline - > result_type = IS_TMP_VAR ;
opline - > result . opline_num = get_temporary_variable ( CG ( active_op_array ) ) ;
GET_NODE ( & key_node , opline - > result ) ;
2005-02-08 00:09:54 +08:00
2005-02-20 18:19:11 +08:00
zend_do_assign ( & dummy , key , & key_node TSRMLS_CC ) ;
2005-02-08 00:09:54 +08:00
zend_do_free ( & dummy TSRMLS_CC ) ;
1999-04-08 02:10:10 +08:00
}
2001-07-28 18:51:54 +08:00
do_begin_loop ( TSRMLS_C ) ;
1999-04-08 02:10:10 +08:00
INC_BPC ( CG ( active_op_array ) ) ;
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_foreach_end ( const znode * foreach_token , const znode * as_token TSRMLS_DC ) /* { { { */
1999-04-08 02:10:10 +08:00
{
2004-03-05 21:04:21 +08:00
zend_op * container_ptr ;
2001-07-28 18:51:54 +08:00
zend_op * opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
1999-04-08 02:10:10 +08:00
opline - > opcode = ZEND_JMP ;
2010-04-20 18:57:45 +08:00
opline - > op1 . opline_num = as_token - > u . op . opline_num ;
1999-10-28 23:53:31 +08:00
SET_UNUSED ( opline - > op1 ) ;
1999-04-08 02:10:10 +08:00
SET_UNUSED ( opline - > op2 ) ;
2010-04-20 18:57:45 +08:00
CG ( active_op_array ) - > opcodes [ foreach_token - > u . op . opline_num ] . op2 . opline_num = get_next_op_number ( CG ( active_op_array ) ) ; /* FE_RESET */
CG ( active_op_array ) - > opcodes [ as_token - > u . op . opline_num ] . op2 . opline_num = get_next_op_number ( CG ( active_op_array ) ) ; /* FE_FETCH */
1999-04-08 02:10:10 +08:00
2010-04-20 18:57:45 +08:00
do_end_loop ( as_token - > u . op . opline_num , 1 TSRMLS_CC ) ;
1999-04-08 02:10:10 +08:00
2004-03-01 02:25:50 +08:00
zend_stack_top ( & CG ( foreach_copy_stack ) , ( void * * ) & container_ptr ) ;
2004-02-26 01:23:50 +08:00
generate_free_foreach_copy ( container_ptr TSRMLS_CC ) ;
zend_stack_del_top ( & CG ( foreach_copy_stack ) ) ;
1999-04-08 02:10:10 +08:00
DEC_BPC ( CG ( active_op_array ) ) ;
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_declare_begin ( TSRMLS_D ) /* { { { */
2000-01-25 03:00:30 +08:00
{
zend_stack_push ( & CG ( declare_stack ) , & CG ( declarables ) , sizeof ( zend_declarables ) ) ;
}
2009-07-27 22:11:53 +08:00
/* }}} */
2000-01-25 03:00:30 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_declare_stmt ( znode * var , znode * val TSRMLS_DC ) /* { { { */
2000-01-25 03:00:30 +08:00
{
if ( ! zend_binary_strcasecmp ( var - > u . constant . value . str . val , var - > u . constant . value . str . len , " ticks " , sizeof ( " ticks " ) - 1 ) ) {
convert_to_long ( & val - > u . constant ) ;
CG ( declarables ) . ticks = val - > u . constant ;
2008-06-29 16:21:35 +08:00
# ifdef ZEND_MULTIBYTE
} else if ( ! zend_binary_strcasecmp ( var - > u . constant . value . str . val , var - > u . constant . value . str . len , " encoding " , sizeof ( " encoding " ) - 1 ) ) {
zend_encoding * new_encoding , * old_encoding ;
zend_encoding_filter old_input_filter ;
2009-01-09 06:36:03 +08:00
if ( ( Z_TYPE ( val - > u . constant ) & IS_CONSTANT_TYPE_MASK ) = = IS_CONSTANT ) {
2008-06-29 16:21:35 +08:00
zend_error ( E_COMPILE_ERROR , " Cannot use constants as encoding " ) ;
}
/*
* Check that the pragma comes before any opcodes . If the compilation
* got as far as this , the previous portion of the script must have been
* parseable according to the . ini script_encoding setting . We still
* want to tell them to put declare ( ) at the top .
*/
2008-07-25 06:21:41 +08:00
{
2008-06-29 16:21:35 +08:00
int num = CG ( active_op_array ) - > last ;
2008-07-25 06:21:41 +08:00
/* ignore ZEND_EXT_STMT and ZEND_TICKS */
2008-06-29 16:21:35 +08:00
while ( num > 0 & &
( CG ( active_op_array ) - > opcodes [ num - 1 ] . opcode = = ZEND_EXT_STMT | |
CG ( active_op_array ) - > opcodes [ num - 1 ] . opcode = = ZEND_TICKS ) ) {
- - num ;
}
2008-07-25 06:21:41 +08:00
2009-12-13 23:18:58 +08:00
if ( num > 0 ) {
2008-07-25 06:21:41 +08:00
zend_error ( E_COMPILE_ERROR , " Encoding declaration pragma must be the very first statement in the script " ) ;
2008-06-29 16:21:35 +08:00
}
2008-07-25 06:21:41 +08:00
}
CG ( encoding_declared ) = 1 ;
2008-06-29 16:21:35 +08:00
convert_to_string ( & val - > u . constant ) ;
new_encoding = zend_multibyte_fetch_encoding ( val - > u . constant . value . str . val ) ;
if ( ! new_encoding ) {
zend_error ( E_COMPILE_WARNING , " Unsupported encoding [%s] " , val - > u . constant . value . str . val ) ;
} else {
old_input_filter = LANG_SCNG ( input_filter ) ;
old_encoding = LANG_SCNG ( script_encoding ) ;
zend_multibyte_set_filter ( new_encoding TSRMLS_CC ) ;
/* need to re-scan if input filter changed */
if ( old_input_filter ! = LANG_SCNG ( input_filter ) | |
( ( old_input_filter = = zend_multibyte_script_encoding_filter ) & &
( new_encoding ! = old_encoding ) ) ) {
zend_multibyte_yyinput_again ( old_input_filter , old_encoding TSRMLS_CC ) ;
}
}
2008-11-12 00:15:53 +08:00
efree ( val - > u . constant . value . str . val ) ;
2008-06-29 16:21:35 +08:00
# else /* !ZEND_MULTIBYTE */
2003-08-11 13:24:42 +08:00
} else if ( ! zend_binary_strcasecmp ( var - > u . constant . value . str . val , var - > u . constant . value . str . len , " encoding " , sizeof ( " encoding " ) - 1 ) ) {
2008-03-17 05:06:55 +08:00
/* Do not generate any kind of warning for encoding declares */
/* zend_error(E_COMPILE_WARNING, "Declare encoding [%s] not supported", val->u.constant.value.str.val); */
2008-11-12 00:15:53 +08:00
zval_dtor ( & val - > u . constant ) ;
2008-06-29 16:21:35 +08:00
# endif /* ZEND_MULTIBYTE */
2006-03-27 16:09:18 +08:00
} else {
2008-03-17 05:06:55 +08:00
zend_error ( E_COMPILE_WARNING , " Unsupported declare '%s' " , var - > u . constant . value . str . val ) ;
2006-03-27 16:09:18 +08:00
zval_dtor ( & val - > u . constant ) ;
2000-01-25 03:00:30 +08:00
}
zval_dtor ( & var - > u . constant ) ;
}
2009-07-27 22:11:53 +08:00
/* }}} */
2000-01-25 03:00:30 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_declare_end ( const znode * declare_token TSRMLS_DC ) /* { { { */
2000-01-25 03:00:30 +08:00
{
zend_declarables * declarables ;
zend_stack_top ( & CG ( declare_stack ) , ( void * * ) & declarables ) ;
2002-07-31 06:19:50 +08:00
/* We should restore if there was more than (current - start) - (ticks?1:0) opcodes */
2010-04-20 18:57:45 +08:00
if ( ( get_next_op_number ( CG ( active_op_array ) ) - declare_token - > u . op . opline_num ) - ( ( Z_LVAL ( CG ( declarables ) . ticks ) ) ? 1 : 0 ) ) {
2002-07-31 06:19:50 +08:00
CG ( declarables ) = * declarables ;
}
2000-01-25 03:00:30 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
2000-01-25 03:00:30 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_exit ( znode * result , const znode * message TSRMLS_DC ) /* { { { */
1999-04-08 02:10:10 +08:00
{
2001-07-28 18:51:54 +08:00
zend_op * opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
1999-04-08 02:10:10 +08:00
opline - > opcode = ZEND_EXIT ;
2010-04-20 18:57:45 +08:00
SET_NODE ( opline - > op1 , message ) ;
1999-04-08 02:10:10 +08:00
SET_UNUSED ( opline - > op2 ) ;
result - > op_type = IS_CONST ;
2006-05-10 07:53:23 +08:00
Z_TYPE ( result - > u . constant ) = IS_BOOL ;
Z_LVAL ( result - > u . constant ) = 1 ;
1999-04-08 02:10:10 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_begin_silence ( znode * strudel_token TSRMLS_DC ) /* { { { */
1999-04-08 02:10:10 +08:00
{
2001-07-28 18:51:54 +08:00
zend_op * opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
1999-04-08 02:10:10 +08:00
opline - > opcode = ZEND_BEGIN_SILENCE ;
2010-04-20 18:57:45 +08:00
opline - > result_type = IS_TMP_VAR ;
opline - > result . var = get_temporary_variable ( CG ( active_op_array ) ) ;
1999-04-08 02:10:10 +08:00
SET_UNUSED ( opline - > op1 ) ;
SET_UNUSED ( opline - > op2 ) ;
2010-04-20 18:57:45 +08:00
GET_NODE ( strudel_token , opline - > result ) ;
1999-04-08 02:10:10 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_end_silence ( const znode * strudel_token TSRMLS_DC ) /* { { { */
1999-04-08 02:10:10 +08:00
{
2001-07-28 18:51:54 +08:00
zend_op * opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
1999-04-08 02:10:10 +08:00
opline - > opcode = ZEND_END_SILENCE ;
2010-04-20 18:57:45 +08:00
SET_NODE ( opline - > op1 , strudel_token ) ;
1999-04-08 02:10:10 +08:00
SET_UNUSED ( opline - > op2 ) ;
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_jmp_set ( const znode * value , znode * jmp_token , znode * colon_token TSRMLS_DC ) /* { { { */
2007-11-21 17:41:35 +08:00
{
int op_number = get_next_op_number ( CG ( active_op_array ) ) ;
zend_op * opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
opline - > opcode = ZEND_JMP_SET ;
2010-04-20 18:57:45 +08:00
opline - > result_type = IS_TMP_VAR ;
opline - > result . var = get_temporary_variable ( CG ( active_op_array ) ) ;
SET_NODE ( opline - > op1 , value ) ;
2007-11-21 17:41:35 +08:00
SET_UNUSED ( opline - > op2 ) ;
2010-04-20 18:57:45 +08:00
GET_NODE ( colon_token , opline - > result ) ;
2007-11-21 17:41:35 +08:00
2010-04-20 18:57:45 +08:00
jmp_token - > u . op . opline_num = op_number ;
2007-11-21 17:41:35 +08:00
INC_BPC ( CG ( active_op_array ) ) ;
}
2009-07-27 22:11:53 +08:00
/* }}} */
2007-11-21 17:41:35 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_jmp_set_else ( znode * result , const znode * false_value , const znode * jmp_token , const znode * colon_token TSRMLS_DC ) /* { { { */
2007-11-21 17:41:35 +08:00
{
zend_op * opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
opline - > opcode = ZEND_QM_ASSIGN ;
opline - > extended_value = 0 ;
2010-04-20 18:57:45 +08:00
SET_NODE ( opline - > result , colon_token ) ;
SET_NODE ( opline - > op1 , false_value ) ;
2007-11-21 17:41:35 +08:00
SET_UNUSED ( opline - > op2 ) ;
2010-04-20 18:57:45 +08:00
GET_NODE ( result , opline - > result ) ;
2007-11-21 17:41:35 +08:00
2010-04-20 18:57:45 +08:00
CG ( active_op_array ) - > opcodes [ jmp_token - > u . op . opline_num ] . op2 . opline_num = get_next_op_number ( CG ( active_op_array ) ) ;
2007-11-21 17:41:35 +08:00
DEC_BPC ( CG ( active_op_array ) ) ;
}
2009-07-27 22:11:53 +08:00
/* }}} */
2007-11-21 17:41:35 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_begin_qm_op ( const znode * cond , znode * qm_token TSRMLS_DC ) /* { { { */
1999-04-08 02:10:10 +08:00
{
int jmpz_op_number = get_next_op_number ( CG ( active_op_array ) ) ;
zend_op * opline ;
2006-05-10 07:53:23 +08:00
2001-07-28 18:51:54 +08:00
opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
1999-04-08 02:10:10 +08:00
opline - > opcode = ZEND_JMPZ ;
2010-04-20 18:57:45 +08:00
SET_NODE ( opline - > op1 , cond ) ;
1999-04-08 02:10:10 +08:00
SET_UNUSED ( opline - > op2 ) ;
2010-04-20 18:57:45 +08:00
opline - > op2 . opline_num = jmpz_op_number ;
GET_NODE ( qm_token , opline - > op2 ) ;
1999-04-08 02:10:10 +08:00
INC_BPC ( CG ( active_op_array ) ) ;
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_qm_true ( const znode * true_value , znode * qm_token , znode * colon_token TSRMLS_DC ) /* { { { */
1999-04-08 02:10:10 +08:00
{
2001-07-28 18:51:54 +08:00
zend_op * opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
1999-04-08 02:10:10 +08:00
2010-04-20 18:57:45 +08:00
CG ( active_op_array ) - > opcodes [ qm_token - > u . op . opline_num ] . op2 . opline_num = get_next_op_number ( CG ( active_op_array ) ) + 1 ; /* jmp over the ZEND_JMP */
1999-04-08 02:10:10 +08:00
opline - > opcode = ZEND_QM_ASSIGN ;
2010-04-20 18:57:45 +08:00
opline - > result_type = IS_TMP_VAR ;
opline - > result . var = get_temporary_variable ( CG ( active_op_array ) ) ;
SET_NODE ( opline - > op1 , true_value ) ;
1999-04-08 02:10:10 +08:00
SET_UNUSED ( opline - > op2 ) ;
2010-04-20 18:57:45 +08:00
GET_NODE ( qm_token , opline - > result ) ;
colon_token - > u . op . opline_num = get_next_op_number ( CG ( active_op_array ) ) ;
1999-04-08 02:10:10 +08:00
2001-07-28 18:51:54 +08:00
opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
1999-04-08 02:10:10 +08:00
opline - > opcode = ZEND_JMP ;
SET_UNUSED ( opline - > op1 ) ;
SET_UNUSED ( opline - > op2 ) ;
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_qm_false ( znode * result , const znode * false_value , const znode * qm_token , const znode * colon_token TSRMLS_DC ) /* { { { */
1999-04-08 02:10:10 +08:00
{
2001-07-28 18:51:54 +08:00
zend_op * opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
1999-04-08 02:10:10 +08:00
opline - > opcode = ZEND_QM_ASSIGN ;
2010-04-20 18:57:45 +08:00
SET_NODE ( opline - > result , qm_token ) ;
SET_NODE ( opline - > op1 , false_value ) ;
1999-04-08 02:10:10 +08:00
SET_UNUSED ( opline - > op2 ) ;
2006-05-10 07:53:23 +08:00
2010-04-20 18:57:45 +08:00
CG ( active_op_array ) - > opcodes [ colon_token - > u . op . opline_num ] . op1 . opline_num = get_next_op_number ( CG ( active_op_array ) ) ;
1999-04-08 02:10:10 +08:00
2010-04-20 18:57:45 +08:00
GET_NODE ( result , opline - > result ) ;
1999-04-08 02:10:10 +08:00
DEC_BPC ( CG ( active_op_array ) ) ;
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_extended_info ( TSRMLS_D ) /* { { { */
1999-04-08 02:10:10 +08:00
{
zend_op * opline ;
2006-05-10 07:53:23 +08:00
2008-03-18 16:36:30 +08:00
if ( ! ( CG ( compiler_options ) & ZEND_COMPILE_EXTENDED_INFO ) ) {
1999-04-08 02:10:10 +08:00
return ;
}
2006-05-10 07:53:23 +08:00
2001-07-28 18:51:54 +08:00
opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
1999-04-08 02:10:10 +08:00
opline - > opcode = ZEND_EXT_STMT ;
SET_UNUSED ( opline - > op1 ) ;
SET_UNUSED ( opline - > op2 ) ;
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_extended_fcall_begin ( TSRMLS_D ) /* { { { */
1999-04-08 02:10:10 +08:00
{
zend_op * opline ;
2006-05-10 07:53:23 +08:00
2008-03-18 16:36:30 +08:00
if ( ! ( CG ( compiler_options ) & ZEND_COMPILE_EXTENDED_INFO ) ) {
1999-04-08 02:10:10 +08:00
return ;
}
2006-05-10 07:53:23 +08:00
2001-07-28 18:51:54 +08:00
opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
1999-04-08 02:10:10 +08:00
opline - > opcode = ZEND_EXT_FCALL_BEGIN ;
SET_UNUSED ( opline - > op1 ) ;
SET_UNUSED ( opline - > op2 ) ;
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_extended_fcall_end ( TSRMLS_D ) /* { { { */
1999-04-08 02:10:10 +08:00
{
zend_op * opline ;
2006-05-10 07:53:23 +08:00
2008-03-18 16:36:30 +08:00
if ( ! ( CG ( compiler_options ) & ZEND_COMPILE_EXTENDED_INFO ) ) {
1999-04-08 02:10:10 +08:00
return ;
}
2006-05-10 07:53:23 +08:00
2001-07-28 18:51:54 +08:00
opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
1999-04-08 02:10:10 +08:00
opline - > opcode = ZEND_EXT_FCALL_END ;
SET_UNUSED ( opline - > op1 ) ;
SET_UNUSED ( opline - > op2 ) ;
}
2009-07-27 22:11:53 +08:00
/* }}} */
1999-04-08 02:10:10 +08:00
2009-07-27 22:11:53 +08:00
void zend_do_ticks ( TSRMLS_D ) /* { { { */
2000-01-25 03:00:30 +08:00
{
2006-05-10 07:53:23 +08:00
if ( Z_LVAL ( CG ( declarables ) . ticks ) ) {
2001-07-28 18:51:54 +08:00
zend_op * opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
2000-01-25 03:00:30 +08:00
opline - > opcode = ZEND_TICKS ;
2010-04-20 18:57:45 +08:00
SET_UNUSED ( opline - > op1 ) ;
2000-01-25 03:00:30 +08:00
SET_UNUSED ( opline - > op2 ) ;
2010-04-20 18:57:45 +08:00
opline - > extended_value = Z_LVAL ( CG ( declarables ) . ticks ) ;
2000-01-25 03:00:30 +08:00
}
}
2009-07-27 22:11:53 +08:00
/* }}} */
2000-01-25 03:00:30 +08:00
2009-07-27 22:11:53 +08:00
void zend_auto_global_dtor ( zend_auto_global * auto_global ) /* { { { */
2003-03-02 18:04:53 +08:00
{
free ( auto_global - > name ) ;
}
2009-07-27 22:11:53 +08:00
/* }}} */
2003-03-02 18:04:53 +08:00
2009-07-27 22:11:53 +08:00
zend_bool zend_is_auto_global ( const char * name , uint name_len TSRMLS_DC ) /* { { { */
2003-03-02 21:33:31 +08:00
{
zend_auto_global * auto_global ;
if ( zend_hash_find ( CG ( auto_globals ) , name , name_len + 1 , ( void * * ) & auto_global ) = = SUCCESS ) {
if ( auto_global - > armed ) {
auto_global - > armed = auto_global - > auto_global_callback ( auto_global - > name , auto_global - > name_len TSRMLS_CC ) ;
}
return 1 ;
}
return 0 ;
}
2009-07-27 22:11:53 +08:00
/* }}} */
2003-03-02 21:33:31 +08:00
2009-07-27 22:11:53 +08:00
int zend_register_auto_global ( const char * name , uint name_len , zend_auto_global_callback auto_global_callback TSRMLS_DC ) /* { { { */
2001-08-09 01:18:16 +08:00
{
2003-03-02 18:04:53 +08:00
zend_auto_global auto_global ;
auto_global . name = zend_strndup ( name , name_len ) ;
auto_global . name_len = name_len ;
auto_global . auto_global_callback = auto_global_callback ;
return zend_hash_add ( CG ( auto_globals ) , name , name_len + 1 , & auto_global , sizeof ( zend_auto_global ) , NULL ) ;
2001-08-09 01:18:16 +08:00
}
2009-07-27 22:11:53 +08:00
/* }}} */
2001-08-09 01:18:16 +08:00
2009-07-27 22:11:53 +08:00
int zendlex ( znode * zendlval TSRMLS_DC ) /* { { { */
1999-04-08 02:10:10 +08:00
{
int retval ;
2001-08-06 22:36:46 +08:00
if ( CG ( increment_lineno ) ) {
CG ( zend_lineno ) + + ;
CG ( increment_lineno ) = 0 ;
}
2007-05-18 21:12:05 +08:00
again :
2006-05-10 07:53:23 +08:00
Z_TYPE ( zendlval - > u . constant ) = IS_LONG ;
2001-07-28 18:51:54 +08:00
retval = lex_scan ( & zendlval - > u . constant TSRMLS_CC ) ;
2002-11-30 19:20:25 +08:00
switch ( retval ) {
1999-04-23 07:08:42 +08:00
case T_COMMENT :
2003-04-03 00:13:12 +08:00
case T_DOC_COMMENT :
1999-04-23 07:08:42 +08:00
case T_OPEN_TAG :
1999-04-08 02:10:10 +08:00
case T_WHITESPACE :
2003-05-29 17:01:55 +08:00
goto again ;
1999-04-23 07:08:42 +08:00
case T_CLOSE_TAG :
2007-05-18 21:12:05 +08:00
if ( LANG_SCNG ( yy_text ) [ LANG_SCNG ( yy_leng ) - 1 ] ! = ' > ' ) {
2001-08-06 22:36:46 +08:00
CG ( increment_lineno ) = 1 ;
}
2008-11-25 17:56:32 +08:00
if ( CG ( has_bracketed_namespaces ) & & ! CG ( in_namespace ) ) {
goto again ;
}
1999-04-08 02:10:10 +08:00
retval = ' ; ' ; /* implicit ; */
break ;
1999-04-23 07:08:42 +08:00
case T_OPEN_TAG_WITH_ECHO :
retval = T_ECHO ;
1999-04-08 02:10:10 +08:00
break ;
2000-02-08 02:39:14 +08:00
case T_END_HEREDOC :
2006-05-12 05:07:39 +08:00
efree ( Z_STRVAL ( zendlval - > u . constant ) ) ;
2000-02-08 02:39:14 +08:00
break ;
1999-04-08 02:10:10 +08:00
}
2006-05-10 07:53:23 +08:00
1999-07-10 04:43:59 +08:00
INIT_PZVAL ( & zendlval - > u . constant ) ;
1999-04-08 02:10:10 +08:00
zendlval - > op_type = IS_CONST ;
return retval ;
}
2009-07-27 22:11:53 +08:00
/* }}} */
2003-02-01 09:49:15 +08:00
2009-07-27 22:11:53 +08:00
ZEND_API void zend_initialize_class_data ( zend_class_entry * ce , zend_bool nullify_handlers TSRMLS_DC ) /* { { { */
2003-02-11 00:11:24 +08:00
{
2003-02-11 00:46:05 +08:00
zend_bool persistent_hashes = ( ce - > type = = ZEND_INTERNAL_CLASS ) ? 1 : 0 ;
2003-11-13 14:40:34 +08:00
dtor_func_t zval_ptr_dtor_func = ( ( persistent_hashes ) ? ZVAL_INTERNAL_PTR_DTOR : ZVAL_PTR_DTOR ) ;
2003-02-11 00:46:05 +08:00
2003-02-11 00:11:24 +08:00
ce - > refcount = 1 ;
ce - > constants_updated = 0 ;
ce - > ce_flags = 0 ;
2003-04-01 04:42:01 +08:00
ce - > doc_comment = NULL ;
ce - > doc_comment_len = 0 ;
2003-02-11 00:11:24 +08:00
2003-11-13 14:40:34 +08:00
zend_hash_init_ex ( & ce - > default_properties , 0 , NULL , zval_ptr_dtor_func , persistent_hashes , 0 ) ;
2003-08-24 03:37:39 +08:00
zend_hash_init_ex ( & ce - > properties_info , 0 , NULL , ( dtor_func_t ) ( persistent_hashes ? zend_destroy_property_info_internal : zend_destroy_property_info ) , persistent_hashes , 0 ) ;
2005-09-01 18:05:32 +08:00
zend_hash_init_ex ( & ce - > default_static_members , 0 , NULL , zval_ptr_dtor_func , persistent_hashes , 0 ) ;
2003-11-13 14:40:34 +08:00
zend_hash_init_ex ( & ce - > constants_table , 0 , NULL , zval_ptr_dtor_func , persistent_hashes , 0 ) ;
2003-02-11 00:46:05 +08:00
zend_hash_init_ex ( & ce - > function_table , 0 , NULL , ZEND_FUNCTION_DTOR , persistent_hashes , 0 ) ;
2003-02-11 00:11:24 +08:00
2005-12-01 19:48:17 +08:00
if ( ce - > type = = ZEND_INTERNAL_CLASS ) {
# ifdef ZTS
int n = zend_hash_num_elements ( CG ( class_table ) ) ;
if ( CG ( static_members ) & & n > = CG ( last_static_member ) ) {
/* Support for run-time declaration: dl() */
CG ( last_static_member ) = n + 1 ;
CG ( static_members ) = realloc ( CG ( static_members ) , ( n + 1 ) * sizeof ( HashTable * ) ) ;
CG ( static_members ) [ n ] = NULL ;
}
2007-04-16 16:09:56 +08:00
ce - > static_members = ( HashTable * ) ( zend_intptr_t ) n ;
2005-12-01 19:48:17 +08:00
# else
ce - > static_members = NULL ;
# endif
} else {
ce - > static_members = & ce - > default_static_members ;
}
2005-09-01 18:05:32 +08:00
2003-02-11 00:11:24 +08:00
if ( nullify_handlers ) {
ce - > constructor = NULL ;
ce - > destructor = NULL ;
ce - > clone = NULL ;
ce - > __get = NULL ;
ce - > __set = NULL ;
2005-07-08 00:07:09 +08:00
ce - > __unset = NULL ;
ce - > __isset = NULL ;
2003-02-11 00:11:24 +08:00
ce - > __call = NULL ;
2007-09-29 16:52:40 +08:00
ce - > __callstatic = NULL ;
2006-05-10 07:53:23 +08:00
ce - > __tostring = NULL ;
2003-02-11 00:11:24 +08:00
ce - > create_object = NULL ;
2003-12-28 23:18:05 +08:00
ce - > get_iterator = NULL ;
ce - > iterator_funcs . funcs = NULL ;
ce - > interface_gets_implemented = NULL ;
2007-09-29 17:34:24 +08:00
ce - > get_static_method = NULL ;
2003-12-28 23:18:05 +08:00
ce - > parent = NULL ;
ce - > num_interfaces = 0 ;
ce - > interfaces = NULL ;
Implemented Traits for PHP as proposed in the RFC [TRAITS]
# RFC http://wiki.php.net/rfc/horizontalreuse#traits_-_reuse_of_behavior
# Ok, here we go, I guess that will result in more discussion, which is fine
# by me. But now, the patch is here, and properly archived.
#
# See below a list of notes to the patch, it also includes a list of
# points which should be fixed
#
# Internals of the Traits Patch
# -----------------------------
#
# Open TODOs
# """"""""""
#
# - Reflection API
# - support for traits for internal classes
# - currently destroy_zend_class does not handle that case
#
# Introduced Structures
# """""""""""""""""""""
#
# Data structures to encode the composition information specified in the
# source:
# - zend_trait_method_reference
# - zend_trait_precedence
# - zend_trait_alias
#
# Changes
# """""""
#
# zend_class_entry
# - uses NULL terminated lists of pointers for
# - trait_aliases
# - trait_precedences
# - do you prefer an explicit counter?
# - the information is only necessary during class composition
# but might be interesting for reflection
# - did not want to blow up class further with not really necessary length counters
#
# added keywords
# - trait
# - insteadof
#
# Added opcodes
# ZEND_ADD_TRAIT
# - similar to ZEND_ADD_INTERFACE
# - adds the trait to the list of traits of a class, no actual composition done
# ZEND_BIND_TRAITS
# - emitted in zend_do_end_class_declaration
# - concludes the class definition and will initiate the trait composition
# when the class definition is encountered during runtime
#
# Added Flags
# ZEND_ACC_TRAIT = 0x120
# ZEND_ACC_IMPLEMENT_TRAITS = 0x400000
# ZEND_FETCH_CLASS_TRAIT = 14
#
# zend_vm_execute.h
# - not sure whether the handler initialization (ZEND_ADD_TRAIT_SPEC_HANDLER,
# ZEND_BIND_TRAITS_SPEC_HANDLER) is correct, maybe it should be more selective
#
# zend_compile.c
# - refactored do_inherit_method_check
# split into do_inherit_method_check and do_inheritance_check_on_method
# - added helper functions use a '_' as prefix and are not mentioned in the
# headers
# - _copy_functions
# prepare hash-maps of functions which should be merged into a class
# here the aliases are handled
# - _merge_functions
# builds a hash-table of the methods which need to be added to a class
# does the conflict detection
# - reused php_runkit_function_copy_ctor
# - it is not identical with the original code anymore, needed to update it
# think I fixed some bugs, not sure whether all have been reported back to runkit
# - has to be renamed, left the name for the moment, to make its origin obvious
# - here might be optimization potential
# - not sure whether everything needs to be copied
# - copying the literals might be broken
# - added it since the literals array is freed by efree and gave problems
# with doubled frees
# - all immutable parts of the zend_op array should not be copied
# - am not sure which parts are immutable
# - and not sure how to avoid doubled frees on the same arrays on shutdown
# - _merge_functions_to_class
# does the final merging with the target class to handle inherited
# and overridden methods
# - small helper for NULL terminated lists
# zend_init_list, zend_add_to_list
#
# zend_language_parser.y
# - reused class definition for traits
# - there should be something with regard to properties
# - if they get explicitly defined, it might be worthwhile to
# check that there are no collisions with other traits in a composition
# (however, I would not introduce elaborate language features to control that
# but a notice for such conflicts might be nice to the developers)
2010-04-23 06:05:56 +08:00
ce - > num_traits = 0 ;
ce - > traits = NULL ;
ce - > trait_aliases = NULL ;
ce - > trait_precedences = NULL ;
2004-06-01 05:38:01 +08:00
ce - > module = NULL ;
2005-02-23 19:15:51 +08:00
ce - > serialize = NULL ;
ce - > unserialize = NULL ;
2005-03-08 06:23:14 +08:00
ce - > serialize_func = NULL ;
ce - > unserialize_func = NULL ;
2005-07-18 03:17:10 +08:00
ce - > builtin_functions = NULL ;
2003-02-11 00:11:24 +08:00
}
}
2009-07-27 22:11:53 +08:00
/* }}} */
2003-02-11 00:11:24 +08:00
2009-07-27 22:11:53 +08:00
int zend_get_class_fetch_type ( const char * class_name , uint class_name_len ) /* { { { */
2003-08-03 16:21:08 +08:00
{
if ( ( class_name_len = = sizeof ( " self " ) - 1 ) & &
2007-09-29 03:52:53 +08:00
! memcmp ( class_name , " self " , sizeof ( " self " ) - 1 ) ) {
2003-08-03 16:21:08 +08:00
return ZEND_FETCH_CLASS_SELF ;
} else if ( ( class_name_len = = sizeof ( " parent " ) - 1 ) & &
2007-09-29 03:52:53 +08:00
! memcmp ( class_name , " parent " , sizeof ( " parent " ) - 1 ) ) {
2003-08-03 16:21:08 +08:00
return ZEND_FETCH_CLASS_PARENT ;
2007-09-29 15:28:34 +08:00
} else if ( ( class_name_len = = sizeof ( " static " ) - 1 ) & &
! memcmp ( class_name , " static " , sizeof ( " static " ) - 1 ) ) {
return ZEND_FETCH_CLASS_STATIC ;
2003-08-03 16:21:08 +08:00
} else {
return ZEND_FETCH_CLASS_DEFAULT ;
}
}
2009-07-27 22:11:53 +08:00
/* }}} */
2003-08-03 16:21:08 +08:00
2009-07-27 22:11:53 +08:00
ZEND_API char * zend_get_compiled_variable_name ( const zend_op_array * op_array , zend_uint var , int * name_len ) /* { { { */
2005-01-22 10:29:18 +08:00
{
if ( name_len ) {
* name_len = op_array - > vars [ var ] . name_len ;
}
return op_array - > vars [ var ] . name ;
}
2009-07-27 22:11:53 +08:00
/* }}} */
2005-01-22 10:29:18 +08:00
2007-09-29 03:52:53 +08:00
void zend_do_build_namespace_name ( znode * result , znode * prefix , znode * name TSRMLS_DC ) /* { { { */
{
if ( prefix ) {
* result = * prefix ;
2007-12-08 01:11:24 +08:00
if ( Z_TYPE ( result - > u . constant ) = = IS_STRING & &
Z_STRLEN ( result - > u . constant ) = = 0 ) {
2008-11-04 23:58:55 +08:00
/* namespace\ */
2007-12-08 01:11:24 +08:00
if ( CG ( current_namespace ) ) {
znode tmp ;
zval_dtor ( & result - > u . constant ) ;
tmp . op_type = IS_CONST ;
tmp . u . constant = * CG ( current_namespace ) ;
zval_copy_ctor ( & tmp . u . constant ) ;
zend_do_build_namespace_name ( result , NULL , & tmp TSRMLS_CC ) ;
}
}
2007-09-29 03:52:53 +08:00
} else {
result - > op_type = IS_CONST ;
Z_TYPE ( result - > u . constant ) = IS_STRING ;
Z_STRVAL ( result - > u . constant ) = NULL ;
Z_STRLEN ( result - > u . constant ) = 0 ;
}
2008-03-26 02:08:37 +08:00
/* prefix = result */
2008-11-04 23:58:55 +08:00
zend_do_build_full_name ( NULL , result , name , 0 TSRMLS_CC ) ;
2007-09-29 03:52:53 +08:00
}
/* }}} */
2008-11-25 17:56:32 +08:00
void zend_do_begin_namespace ( const znode * name , zend_bool with_bracket TSRMLS_DC ) /* { { { */
2007-09-29 03:52:53 +08:00
{
char * lcname ;
2008-11-25 17:56:32 +08:00
/* handle mixed syntax declaration or nested namespaces */
if ( ! CG ( has_bracketed_namespaces ) ) {
if ( CG ( current_namespace ) ) {
/* previous namespace declarations were unbracketed */
if ( with_bracket ) {
zend_error ( E_COMPILE_ERROR , " Cannot mix bracketed namespace declarations with unbracketed namespace declarations " ) ;
}
}
} else {
/* previous namespace declarations were bracketed */
if ( ! with_bracket ) {
zend_error ( E_COMPILE_ERROR , " Cannot mix bracketed namespace declarations with unbracketed namespace declarations " ) ;
} else if ( CG ( current_namespace ) | | CG ( in_namespace ) ) {
zend_error ( E_COMPILE_ERROR , " Namespace declarations cannot be nested " ) ;
}
}
if ( ( ( ! with_bracket & & ! CG ( current_namespace ) ) | | ( with_bracket & & ! CG ( has_bracketed_namespaces ) ) ) & & CG ( active_op_array ) - > last > 0 ) {
2007-11-01 19:58:58 +08:00
/* ignore ZEND_EXT_STMT and ZEND_TICKS */
2007-09-29 03:52:53 +08:00
int num = CG ( active_op_array ) - > last ;
while ( num > 0 & &
2007-11-01 19:58:58 +08:00
( CG ( active_op_array ) - > opcodes [ num - 1 ] . opcode = = ZEND_EXT_STMT | |
CG ( active_op_array ) - > opcodes [ num - 1 ] . opcode = = ZEND_TICKS ) ) {
2007-09-29 03:52:53 +08:00
- - num ;
}
2008-11-25 17:56:32 +08:00
if ( num > 0 ) {
2007-09-29 03:52:53 +08:00
zend_error ( E_COMPILE_ERROR , " Namespace declaration statement has to be the very first statement in the script " ) ;
}
}
2008-11-25 17:56:32 +08:00
CG ( in_namespace ) = 1 ;
if ( with_bracket ) {
CG ( has_bracketed_namespaces ) = 1 ;
2007-09-29 03:52:53 +08:00
}
2008-11-25 17:56:32 +08:00
if ( name ) {
lcname = zend_str_tolower_dup ( Z_STRVAL ( name - > u . constant ) , Z_STRLEN ( name - > u . constant ) ) ;
if ( ( ( Z_STRLEN ( name - > u . constant ) = = sizeof ( " self " ) - 1 ) & &
! memcmp ( lcname , " self " , sizeof ( " self " ) - 1 ) ) | |
( ( Z_STRLEN ( name - > u . constant ) = = sizeof ( " parent " ) - 1 ) & &
! memcmp ( lcname , " parent " , sizeof ( " parent " ) - 1 ) ) ) {
zend_error ( E_COMPILE_ERROR , " Cannot use '%s' as namespace name " , Z_STRVAL ( name - > u . constant ) ) ;
}
efree ( lcname ) ;
if ( CG ( current_namespace ) ) {
zval_dtor ( CG ( current_namespace ) ) ;
} else {
ALLOC_ZVAL ( CG ( current_namespace ) ) ;
}
* CG ( current_namespace ) = name - > u . constant ;
2007-12-13 16:57:52 +08:00
} else {
2008-11-25 17:56:32 +08:00
if ( CG ( current_namespace ) ) {
zval_dtor ( CG ( current_namespace ) ) ;
FREE_ZVAL ( CG ( current_namespace ) ) ;
CG ( current_namespace ) = NULL ;
}
2007-12-13 16:57:52 +08:00
}
2008-11-25 17:56:32 +08:00
2007-12-13 16:57:52 +08:00
if ( CG ( current_import ) ) {
zend_hash_destroy ( CG ( current_import ) ) ;
efree ( CG ( current_import ) ) ;
CG ( current_import ) = NULL ;
}
2007-09-29 03:52:53 +08:00
}
/* }}} */
2007-12-13 18:02:03 +08:00
void zend_do_use ( znode * ns_name , znode * new_name , int is_global TSRMLS_DC ) /* { { { */
2007-09-29 03:52:53 +08:00
{
char * lcname ;
zval * name , * ns , tmp ;
zend_bool warn = 0 ;
2008-06-21 01:17:05 +08:00
zend_class_entry * * pce ;
2007-09-29 03:52:53 +08:00
if ( ! CG ( current_import ) ) {
CG ( current_import ) = emalloc ( sizeof ( HashTable ) ) ;
zend_hash_init ( CG ( current_import ) , 0 , NULL , ZVAL_PTR_DTOR , 0 ) ;
}
ALLOC_ZVAL ( ns ) ;
* ns = ns_name - > u . constant ;
if ( new_name ) {
name = & new_name - > u . constant ;
} else {
char * p ;
2008-11-04 23:58:55 +08:00
/* The form "use A\B" is eqivalent to "use A\B as B".
So we extract the last part of compound name to use as a new_name */
2007-09-29 03:52:53 +08:00
name = & tmp ;
2008-11-04 23:58:55 +08:00
p = zend_memrchr ( Z_STRVAL_P ( ns ) , ' \\ ' , Z_STRLEN_P ( ns ) ) ;
2007-09-29 03:52:53 +08:00
if ( p ) {
ZVAL_STRING ( name , p + 1 , 1 ) ;
} else {
* name = * ns ;
zval_copy_ctor ( name ) ;
2009-01-14 21:57:42 +08:00
warn = ! is_global & & ! CG ( current_namespace ) ;
2007-09-29 03:52:53 +08:00
}
}
lcname = zend_str_tolower_dup ( Z_STRVAL_P ( name ) , Z_STRLEN_P ( name ) ) ;
if ( ( ( Z_STRLEN_P ( name ) = = sizeof ( " self " ) - 1 ) & &
! memcmp ( lcname , " self " , sizeof ( " self " ) - 1 ) ) | |
( ( Z_STRLEN_P ( name ) = = sizeof ( " parent " ) - 1 ) & &
! memcmp ( lcname , " parent " , sizeof ( " parent " ) - 1 ) ) ) {
2007-11-13 01:52:15 +08:00
zend_error ( E_COMPILE_ERROR , " Cannot use %s as %s because '%s' is a special class name " , Z_STRVAL_P ( ns ) , Z_STRVAL_P ( name ) , Z_STRVAL_P ( name ) ) ;
2007-09-29 03:52:53 +08:00
}
2007-10-17 18:01:22 +08:00
if ( CG ( current_namespace ) ) {
/* Prefix import name with current namespace name to avoid conflicts with classes */
2009-09-06 05:16:05 +08:00
char * c_ns_name = emalloc ( Z_STRLEN_P ( CG ( current_namespace ) ) + 1 + Z_STRLEN_P ( name ) + 1 ) ;
2007-10-17 18:01:22 +08:00
2009-09-06 05:16:05 +08:00
zend_str_tolower_copy ( c_ns_name , Z_STRVAL_P ( CG ( current_namespace ) ) , Z_STRLEN_P ( CG ( current_namespace ) ) ) ;
c_ns_name [ Z_STRLEN_P ( CG ( current_namespace ) ) ] = ' \\ ' ;
memcpy ( c_ns_name + Z_STRLEN_P ( CG ( current_namespace ) ) + 1 , lcname , Z_STRLEN_P ( name ) + 1 ) ;
if ( zend_hash_exists ( CG ( class_table ) , c_ns_name , Z_STRLEN_P ( CG ( current_namespace ) ) + 1 + Z_STRLEN_P ( name ) + 1 ) ) {
2007-11-12 23:52:22 +08:00
char * tmp = zend_str_tolower_dup ( Z_STRVAL_P ( ns ) , Z_STRLEN_P ( ns ) ) ;
2008-11-04 23:58:55 +08:00
if ( Z_STRLEN_P ( ns ) ! = Z_STRLEN_P ( CG ( current_namespace ) ) + 1 + Z_STRLEN_P ( name ) | |
2009-09-06 05:16:05 +08:00
memcmp ( tmp , c_ns_name , Z_STRLEN_P ( ns ) ) ) {
2007-11-13 01:52:15 +08:00
zend_error ( E_COMPILE_ERROR , " Cannot use %s as %s because the name is already in use " , Z_STRVAL_P ( ns ) , Z_STRVAL_P ( name ) ) ;
2007-11-12 23:52:22 +08:00
}
efree ( tmp ) ;
2007-10-17 18:01:22 +08:00
}
2009-09-06 05:16:05 +08:00
efree ( c_ns_name ) ;
2008-06-21 01:17:05 +08:00
} else if ( zend_hash_find ( CG ( class_table ) , lcname , Z_STRLEN_P ( name ) + 1 , ( void * * ) & pce ) = = SUCCESS & &
( * pce ) - > type = = ZEND_USER_CLASS & &
( * pce ) - > filename = = CG ( compiled_filename ) ) {
2009-09-06 05:16:05 +08:00
char * c_tmp = zend_str_tolower_dup ( Z_STRVAL_P ( ns ) , Z_STRLEN_P ( ns ) ) ;
2007-11-12 23:52:22 +08:00
if ( Z_STRLEN_P ( ns ) ! = Z_STRLEN_P ( name ) | |
2009-09-06 05:16:05 +08:00
memcmp ( c_tmp , lcname , Z_STRLEN_P ( ns ) ) ) {
2007-11-13 01:52:15 +08:00
zend_error ( E_COMPILE_ERROR , " Cannot use %s as %s because the name is already in use " , Z_STRVAL_P ( ns ) , Z_STRVAL_P ( name ) ) ;
2007-11-12 23:52:22 +08:00
}
2009-09-06 05:16:05 +08:00
efree ( c_tmp ) ;
2007-09-29 03:52:53 +08:00
}
if ( zend_hash_add ( CG ( current_import ) , lcname , Z_STRLEN_P ( name ) + 1 , & ns , sizeof ( zval * ) , NULL ) ! = SUCCESS ) {
2007-11-13 01:52:15 +08:00
zend_error ( E_COMPILE_ERROR , " Cannot use %s as %s because the name is already in use " , Z_STRVAL_P ( ns ) , Z_STRVAL_P ( name ) ) ;
2007-09-29 03:52:53 +08:00
}
if ( warn ) {
2007-11-07 17:13:50 +08:00
zend_error ( E_WARNING , " The use statement with non-compound name '%s' has no effect " , Z_STRVAL_P ( name ) ) ;
2007-09-29 03:52:53 +08:00
}
efree ( lcname ) ;
zval_dtor ( name ) ;
}
/* }}} */
void zend_do_declare_constant ( znode * name , znode * value TSRMLS_DC ) /* { { { */
{
zend_op * opline ;
2007-11-22 18:46:26 +08:00
if ( Z_TYPE ( value - > u . constant ) = = IS_CONSTANT_ARRAY ) {
zend_error ( E_COMPILE_ERROR , " Arrays are not allowed as constants " ) ;
}
2008-08-29 18:17:08 +08:00
if ( zend_get_ct_const ( & name - > u . constant , 0 TSRMLS_CC ) ) {
2007-09-29 03:52:53 +08:00
zend_error ( E_COMPILE_ERROR , " Cannot redeclare constant '%s' " , Z_STRVAL ( name - > u . constant ) ) ;
}
if ( CG ( current_namespace ) ) {
2008-11-12 03:45:29 +08:00
/* Prefix constant name with name of current namespace, lowercased */
2007-09-29 03:52:53 +08:00
znode tmp ;
tmp . op_type = IS_CONST ;
tmp . u . constant = * CG ( current_namespace ) ;
Z_STRVAL ( tmp . u . constant ) = zend_str_tolower_dup ( Z_STRVAL ( tmp . u . constant ) , Z_STRLEN ( tmp . u . constant ) ) ;
zend_do_build_namespace_name ( & tmp , & tmp , name TSRMLS_CC ) ;
* name = tmp ;
}
opline = get_next_op ( CG ( active_op_array ) TSRMLS_CC ) ;
opline - > opcode = ZEND_DECLARE_CONST ;
SET_UNUSED ( opline - > result ) ;
2010-04-20 18:57:45 +08:00
SET_NODE ( opline - > op1 , name ) ;
SET_NODE ( opline - > op2 , value ) ;
2007-09-29 03:52:53 +08:00
}
/* }}} */
2008-11-25 17:56:32 +08:00
void zend_verify_namespace ( TSRMLS_D ) /* { { { */
2007-09-29 03:52:53 +08:00
{
2008-11-25 17:56:32 +08:00
if ( CG ( has_bracketed_namespaces ) & & ! CG ( in_namespace ) ) {
zend_error ( E_COMPILE_ERROR , " No code may exist outside of namespace {} " ) ;
}
}
/* }}} */
void zend_do_end_namespace ( TSRMLS_D ) /* { { { */
{
CG ( in_namespace ) = 0 ;
2007-09-29 03:52:53 +08:00
if ( CG ( current_namespace ) ) {
zval_dtor ( CG ( current_namespace ) ) ;
2007-12-13 16:57:52 +08:00
FREE_ZVAL ( CG ( current_namespace ) ) ;
2007-09-29 03:52:53 +08:00
CG ( current_namespace ) = NULL ;
}
if ( CG ( current_import ) ) {
zend_hash_destroy ( CG ( current_import ) ) ;
efree ( CG ( current_import ) ) ;
CG ( current_import ) = NULL ;
}
}
/* }}} */
2008-11-25 17:56:32 +08:00
void zend_do_end_compilation ( TSRMLS_D ) /* { { { */
{
CG ( has_bracketed_namespaces ) = 0 ;
zend_do_end_namespace ( TSRMLS_C ) ;
}
/* }}} */
2008-02-12 09:17:48 +08:00
/* {{{ zend_dirname
Returns directory name component of path */
2008-02-12 08:21:15 +08:00
ZEND_API size_t zend_dirname ( char * path , size_t len )
{
register char * end = path + len - 1 ;
unsigned int len_adjust = 0 ;
# ifdef PHP_WIN32
/* Note that on Win32 CWD is per drive (heritage from CP/M).
* This means dirname ( " c:foo " ) maps to " c:. " or " c: " - which means CWD on C : drive .
*/
if ( ( 2 < = len ) & & isalpha ( ( int ) ( ( unsigned char * ) path ) [ 0 ] ) & & ( ' : ' = = path [ 1 ] ) ) {
/* Skip over the drive spec (if any) so as not to change */
path + = 2 ;
len_adjust + = 2 ;
if ( 2 = = len ) {
/* Return "c:" on Win32 for dirname("c:").
* It would be more consistent to return " c:. "
* but that would require making the string * longer * .
*/
return len ;
}
}
# elif defined(NETWARE)
/*
* Find the first occurence of : from the left
* move the path pointer to the position just after :
* increment the len_adjust to the length of path till colon character ( inclusive )
* If there is no character beyond : simple return len
*/
char * colonpos = NULL ;
colonpos = strchr ( path , ' : ' ) ;
if ( colonpos ! = NULL ) {
len_adjust = ( ( colonpos - path ) + 1 ) ;
path + = len_adjust ;
if ( len_adjust = = len ) {
return len ;
}
}
# endif
if ( len = = 0 ) {
/* Illegal use of this function */
return 0 ;
}
/* Strip trailing slashes */
while ( end > = path & & IS_SLASH_P ( end ) ) {
end - - ;
}
if ( end < path ) {
/* The path only contained slashes */
path [ 0 ] = DEFAULT_SLASH ;
path [ 1 ] = ' \0 ' ;
return 1 + len_adjust ;
}
/* Strip filename */
while ( end > = path & & ! IS_SLASH_P ( end ) ) {
end - - ;
}
if ( end < path ) {
/* No slash found, therefore return '.' */
# ifdef NETWARE
if ( len_adjust = = 0 ) {
path [ 0 ] = ' . ' ;
path [ 1 ] = ' \0 ' ;
2008-08-03 19:48:20 +08:00
return 1 ; /* only one character */
2008-02-12 08:21:15 +08:00
} else {
path [ 0 ] = ' \0 ' ;
return len_adjust ;
}
# else
path [ 0 ] = ' . ' ;
path [ 1 ] = ' \0 ' ;
return 1 + len_adjust ;
# endif
}
/* Strip slashes which came before the file name */
while ( end > = path & & IS_SLASH_P ( end ) ) {
end - - ;
}
if ( end < path ) {
path [ 0 ] = DEFAULT_SLASH ;
path [ 1 ] = ' \0 ' ;
return 1 + len_adjust ;
}
* ( end + 1 ) = ' \0 ' ;
return ( size_t ) ( end + 1 - path ) + len_adjust ;
}
/* }}} */
2008-02-12 09:17:48 +08:00
2003-02-01 09:49:15 +08:00
/*
* Local variables :
* tab - width : 4
* c - basic - offset : 4
* indent - tabs - mode : t
* End :
*/