-
Notifications
You must be signed in to change notification settings - Fork 59
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
N^2 Performance on decoding large definite bytestrings #240
Comments
I can see why this would have N^2 performance. But if I were to preallocate the entire thing, it would allow easy attacks using malicious data that specifies a gigantic bytestring without actually providing that data. On the other hand, if I just store an array of buffers, I would eventually need to copy them into a larger one, briefly requiring 2x the amount of memory. Suggestions? |
Briefly requiring 2x the memory is not an issue for us. I appreciate the guard against the attacks, and the desire not to regress there. Could we:
Disclaimer: no idea how to do 2. |
Note: After realizing that the other cbor library doesn't support RFC 8949 and only supports 7049, we switched back to this one using the pure Python implementation, but made our own decoder subclass that doesn't use a chunked buffer copy. This should be an acceptable solution for a while (perhaps forever), but it would be nice to use a C implementation here :). The difference does not completely break our workflow, though. |
I suggest using an left = length
- buffer = bytearray()
+ try:
+ buffer = mmap(-1, left)
+ except (OSError, OverflowError):
+ raise CBORDecodeValueError("invalid length for bytestring 0x%x" % left)
while left:
- chunk_size = min(left, 65536)
- buffer.extend(self.read(chunk_size))
- left -= chunk_size
+ chunk = self.read(min(left, 65536))
+ buffer.write(chunk)
+ left -= len(chunk)
result = bytes(buffer) and its equivalent in C. |
In pure Python the “conversion” to a bytestring at the end is actually a copy, so both the current code and the one I suggested require 2× the memory. In C it should be possible to create a bytestring pointing directly to the mmapped data, thus avoiding the copy. |
Things to check first
I have searched the existing issues and didn't find my bug already reported there
I have checked that my bug is still present in the latest release
cbor2 version
5.6.4
Python version
3.9, 3.10, 3.12.1, 3.12.3
What happened?
I work with a lab that receives fairly high frequency, large, compressed images in a CBOR format off of a piece of hardware.
Frequencies range from 10hz to 200hz, and the size of the compressed images range from 1MB to 24MB.
After failing to read messages quickly enough, we discovered that
cbor2.loads
was the culprit and >90% of the time was spent decode.c:613/614.I understand that this N^2 memcopy was implemented in #204 to address some memory crash related issues with malicious inputs. However this implementation is unacceptable for us and we have switched to a different python cbor implementation.
I would prefer to continue using this one, as it is clearly more active/supported :).
How can we reproduce the bug?
Easy to reproduce, and easy to see from the code.
Here is a script that makes a plot, though, comparing it to
cbor
:).The text was updated successfully, but these errors were encountered: