Skip to content
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

bpo-38368: Added fix for ctypes crash when handling arrays in structs… #16589

Merged
merged 8 commits into from
Oct 8, 2019
Merged

bpo-38368: Added fix for ctypes crash when handling arrays in structs… #16589

merged 8 commits into from
Oct 8, 2019

Conversation

vsajip
Copy link
Member

@vsajip vsajip commented Oct 4, 2019

@vsajip vsajip added type-bug An unexpected behavior, bug, or error skip news labels Oct 4, 2019
@vsajip vsajip requested a review from zooba October 4, 2019 23:43
@vsajip vsajip removed the request for review from zooba October 5, 2019 00:04
@vsajip
Copy link
Member Author

vsajip commented Oct 5, 2019

Tests run on Windows x64 but not on Linux. Still investigating.

@vsajip vsajip requested a review from zooba October 5, 2019 11:40
Py_ssize_t length = dict->length;
StgDictObject *edict;
ffi_type * dummy_struct;
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
ffi_type * dummy_struct;
ffi_type *dummy_struct;

Py_ssize_t length = dict->length;
StgDictObject *edict;
ffi_type * dummy_struct;
ffi_type ** dummy_fields;
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
ffi_type ** dummy_fields;
ffi_type **dummy_fields;

/* Do separate allocations for now. */
dummy_struct = PyMem_New(ffi_type, 1);
if (dummy_struct == NULL) {
PyErr_NoMemory();
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think there should be a Py_DECREF(pair); here.

}
dummy_fields = PyMem_New(ffi_type *, length + 1);
if (dummy_fields == NULL) {
PyErr_NoMemory();
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think dummy_struct should be freed here. Also, pair should be decrefed.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Well, the problem is that we're doing these allocations in a loop, and unless we keep some sort of linked list or dynamically allocated structure to keep track of all our allocations (both dummy_struct and dummy_fields in earlier iterations of the loop), we will leak this memory in an Out Of Memory (OOM) situation. I'm not aware that we currently do this type of recovery for these internal allocations in an OOM situation, but happy to be proved wrong.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I will have a think about an alternative allocation strategy which allows an easier recovery, but it will be potentially a little harder to follow the logic compared to what's there now.

Copy link
Member

@zooba zooba left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'd like to have one more person go over this line-by-line as well to track the memory usage, but I didn't spot anything that seemed wrong.

@vsajip
Copy link
Member Author

vsajip commented Oct 7, 2019

I'd like to have one more person go over this line-by-line as well to track the memory usage

Agreed - please request a review from a core developer who you think might have the bandwidth to do this. I asked various people to review the earlier PR (for issue 22273) but didn't get many takers ...

Copy link
Member

@ericsnowcurrently ericsnowcurrently left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

LGTM. I've left some minor comments, but they aren't critical.

I did double-check the memory allocations and everything looks good. The only allocations were the 2 PySequence_GetItem() calls and the one PyMem_Malloc() call. They are cleaned up in all the right spots.

if (pair == NULL) {
PyMem_Free(type_block);
return -1;
}
if (!PyArg_ParseTuple(pair, "UO|i", &name, &desc, &bitsize)) {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This check (but certainly not the call) might be redundant based on the earlier for loop.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Although it shouldn't fail, I suppose one can't guarantee it? For that reason, I'd leave the check in, it makes the code a wee bit bigger but shouldn't really impact performance.

if (!PyArg_ParseTuple(pair, "UO|i", &name, &desc, &bitsize)) {
PyErr_SetString(PyExc_TypeError,
"'_fields_' must be a sequence of (name, C type) pairs");
Py_XDECREF(pair);
PyMem_Free(type_block);
return -1;
}
dict = PyType_stgdict(desc);
if (dict == NULL) {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This check seems redundant based on the earlier for loop.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

As per earlier comment - would prefer to leave the check in for now.

@vsajip
Copy link
Member Author

vsajip commented Oct 8, 2019

Thanks for the feedback, @zooba , @ericsnowcurrently and @ZackerySpytz !

changed a couple of Py_XDECREF to Py_DECREF, and added some comments
about possibly defensive checks.
@vsajip vsajip merged commit e8bedbd into python:master Oct 8, 2019
@vsajip vsajip deleted the fix-38368 branch October 8, 2019 20:59
@vsajip
Copy link
Member Author

vsajip commented Oct 8, 2019

I'll wait for the buildbots to pronounce on this merge before backporting to 3.8/3.7.

@miss-islington
Copy link
Contributor

Thanks @vsajip for the PR 🌮🎉.. I'm working now to backport this PR to: 3.8.
🐍🍒⛏🤖

@miss-islington
Copy link
Contributor

Sorry, @vsajip, I could not cleanly backport this to 3.8 due to a conflict.
Please backport using cherry_picker on command line.
cherry_picker e8bedbddadaa86be6bd86dc32dbdbd53933a4988 3.8

@bedevere-bot
Copy link

GH-16671 is a backport of this pull request to the 3.8 branch.

@miss-islington
Copy link
Contributor

Thanks @vsajip for the PR 🌮🎉.. I'm working now to backport this PR to: 3.7.
🐍🍒⛏🤖

@miss-islington
Copy link
Contributor

Sorry @vsajip, I had trouble checking out the 3.7 backport branch.
Please backport using cherry_picker on command line.
cherry_picker e8bedbddadaa86be6bd86dc32dbdbd53933a4988 3.7

@bedevere-bot
Copy link

GH-16672 is a backport of this pull request to the 3.7 branch.

vsajip added a commit that referenced this pull request Oct 9, 2019
vsajip added a commit that referenced this pull request Oct 9, 2019
@doko42
Copy link
Member

doko42 commented Oct 9, 2019

I checked that at least two packages in Debian and Ubuntu, python-ipaddr and python-pysnmp4 now work again with this patch (only tested on the 3.7 branch).

@vsajip
Copy link
Member Author

vsajip commented Oct 9, 2019

@doko42 Thanks for the feedback.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
skip news type-bug An unexpected behavior, bug, or error
Projects
None yet
Development

Successfully merging this pull request may close these issues.

8 participants