- 
                Notifications
    You must be signed in to change notification settings 
- Fork 8k
GH-18572: infinite stack recursion in fallback object comparison. #18577
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Conversation
With nested objects and recursive comparisons, it is for now unavoidable to have a stack overflow we do some early damage control attempt early on with zend.max_allowed_stack_size check but ultimately more a band-aid than a definitive solution.
| As mentioned by @nielsdos the performance drop here should be minimal, zend_hash_compare however would have been more impactful I assume. | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This looks good to me
        
          
                Zend/tests/gh18519.phpt
              
                Outdated
          
        
      | --FILE-- | ||
| <?php | ||
|  | ||
| error_reporting(E_ALL & ~E_DEPRECATED); | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Nit: Add #[AllowDynamicProperties] on the class to avoid the warning
        
          
                Zend/zend_object_handlers.c
              
                Outdated
          
        
      | zend_object *zobj1, *zobj2; | ||
|  | ||
| if (zend_objects_check_stack_limit()) { | ||
| zend_throw_error(NULL, "Object compare - stack limit reached"); | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Nit:
| zend_throw_error(NULL, "Object compare - stack limit reached"); | |
| zend_throw_error(NULL, "Maximum call stack size reached during object comparison"); | 
To get a similar message as in
Line 114 in 05618e7
| "Maximum call stack size of %zu bytes (zend.max_allowed_stack_size - zend.reserved_stack_size) reached during compilation. Try splitting expression", | 
|  | ||
| $cur = $first; | ||
|  | ||
| for ($i = 0; $i < 50000; $i++) { | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The test "fail" even for i=0 (no loop at all) - https://3v4l.org/gkKqd. The comparison probably cannot find cycles correctly.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
it does trigger the Nesting level too deep warning indeed but it does not trigger the Maximum call stack size reached one, which is the whole point of this test. (just tried locally with master)
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I have simplified the code and opened #18585. I belive the recursion in weak comparasion should be handled.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
With nested objects and recursive comparisons, it is for now unavoidable to have a stack overflow we do some early damage control attempt early on with zend.max_allowed_stack_size check but ultimately more a band-aid than a definitive solution.