Hello Sara,

  nice catch, fix looks pretty correct to me

best regards
marcus

Wednesday, July 5, 2006, 10:59:35 PM, you wrote:

> When executing ZEND_FETCH_OBJ_IS on a non-object, the handler dispatches
> to zend_fetch_property_address_read_helper with type==BP_VAR_IS and ends
> up generating an error (isset() is supposed to be quiet) here:

>     if (Z_TYPE_P(container) != IS_OBJECT ||
> !Z_OBJ_HT_P(container)->read_property) {
>         zend_error(E_NOTICE, "Trying to get property of non-object");
>         *retval = EG(uninitialized_zval_ptr);
>         SELECTIVE_PZVAL_LOCK(*retval, &opline->result);
>         AI_USE_PTR(EX_T(opline->result.u.var).var);


> It's cousin ZEND_FETCH_DIM_IS winds up at zend_fetch_dimension_address()
> where the non-arrayness of op1 is handled via these case statements:
>         case IS_NULL: {
>             /* for read-mode only */
>             if (result) {
>                 result->var.ptr_ptr = &EG(uninitialized_zval_ptr);
>                 PZVAL_LOCK(*result->var.ptr_ptr);
>             }
>             if (type==BP_VAR_W || type==BP_VAR_RW) {
>                 zend_error(E_WARNING, "Cannot use a NULL value as an array");
>             }
>             break;
>         }
>         default: {
>                 switch (type) {
>                     case BP_VAR_UNSET:
>                         zend_error(E_WARNING, "Cannot unset offset in a 
> non-array variable");
>                         /* break missing intentionally */
>                     case BP_VAR_R:
>                     case BP_VAR_IS:
>                         retval = &EG(uninitialized_zval_ptr);
>                         break;
>                     default:
>                         retval = &EG(error_zval_ptr);
>                         break;
>                 }
>                 if (result) {
>                     result->var.ptr_ptr = retval;
>                     PZVAL_LOCK(*result->var.ptr_ptr);
>                 }
>                 if (type==BP_VAR_W || type==BP_VAR_RW) {
>                     zend_error(E_WARNING, "Cannot use a scalar value as an 
> array");
>                 }
>             }

> Which indicate the expected behavior of silent failure.  It's probably
> gone undetected till now as it requires fetching a property or dimension
> offset from a property in a third (non-object) variable within an isset() 
> statement:

> $foo = NULL;
> isset($foo->bar->baz);

> Unless someone can suggest why Objects should be noisy inside isset(),
> I'd like to suggest the following fix:

> Index: Zend/zend_vm_def.h
> ===================================================================
> RCS file: /repository/ZendEngine2/zend_vm_def.h,v
> retrieving revision 1.120
> diff -u -r1.120 zend_vm_def.h
> --- Zend/zend_vm_def.h  13 Jun 2006 12:56:20 -0000      1.120
> +++ Zend/zend_vm_def.h  5 Jul 2006 20:52:41 -0000
> @@ -1176,7 +1176,9 @@


>         if (Z_TYPE_P(container) != IS_OBJECT ||
> !Z_OBJ_HT_P(container)->read_property) {
> -               zend_error(E_NOTICE, "Trying to get property of non-object");
> +               if (type != BP_VAR_IS) {
> +                       zend_error(E_NOTICE, "Trying to get property of 
> non-object");
> +               }
>                 *retval = EG(uninitialized_zval_ptr);
>                 SELECTIVE_PZVAL_LOCK(*retval, &opline->result);
>                 AI_USE_PTR(EX_T(opline->result.u.var).var);


> I suggest only excluding BP_VAR_IS since the matching behavior for
> unset()/DIM is to throw a notice.  I could potentially see excluding
> BP_VAR_R in order to match read/DIM, but personally I never liked that 
> behavior :)

> -Sara



Best regards,
 Marcus

-- 
PHP Internals - PHP Runtime Development Mailing List
To unsubscribe, visit: http://www.php.net/unsub.php

Reply via email to