-
Notifications
You must be signed in to change notification settings - Fork 5.8k
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
8303512: Race condition when computing is_loaded property of TypePtr::InterfaceSet #13868
Conversation
👋 Welcome back thartmann! A progress list of the required criteria for merging this PR into |
@TobiHartmann The following label will be automatically applied to this pull request:
When this pull request is ready to be reviewed, an "RFR" email will be sent to the corresponding mailing list. If you would like to change these labels, use the /label pull request command. |
Webrevs
|
It seems to me that if 2 threads access a variable racily without using atomic, it results in undefined behaviours. This patch removes the logic and therefore removes the race condition. Is there any risk with that in the surrounding code, too? I see you reordered |
Thanks for looking at this, @merykitty. I did these reorders because I found it weird that we set On second thought though, I think you are right that this could lead to undefined behavior, especially if the C++ compiler decides to reorder again. One thread could then observe Since lock-free data structures are hard to get rid, and the overhead would only be needed for a few shared types, I propose to eagerly compute |
I have another set of refactoring changes. Will push these later today after some more cleanup / testing. |
Done. |
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.
Looks good to me. Nice refactoring and simplification.
src/hotspot/share/opto/type.cpp
Outdated
int TypePtr::InterfaceSet::hash() const { | ||
if (_hash_computed) { | ||
return _hash; | ||
bool TypePtr::InterfaceSet::eq(ciInstanceKlass* k, InterfaceHandling interface_handling) const { |
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.
Why not remove the interface_handling parameter? It doesn't seem useful.
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.
Right, I'll remove it.
@TobiHartmann This change now passes all automated pre-integration checks. ℹ️ This project also has non-automated pre-integration requirements. Please see the file CONTRIBUTING.md for details. After integration, the commit message for the final commit will be:
You can use pull request commands such as /summary, /contributor and /issue to adjust it as needed. At the time when this comment was updated there had been no new commits pushed to the ➡️ To integrate this PR with the above commit message to the |
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.
Looks good.
Thanks for the reviews, Roland, Quan Anh and Vladimir! |
I'm seeing issues in higher tier testing. Investigating. |
False alarm. I'm integrating this now. |
/integrate |
@TobiHartmann Pushed as commit ad348a8. 💡 You may see a message that your pull request was closed with unmerged commits. This can be safely ignored. |
JDK-8297933 introduced a race condition among compiler threads computing
TypePtr::InterfaceSet::_is_loaded
for shared types (for example, for TypeInstPtr::NOTNULL). One thread can set_is_loaded_computed
before setting_is_loaded
:jdk/src/hotspot/share/opto/type.cpp
Lines 3473 to 3482 in 5726d31
while another thread can already access
is_loaded()
and wrongly observe_is_loaded = false
:jdk/src/hotspot/share/opto/type.cpp
Lines 3464 to 3468 in 5726d31
As a result, the klass of a
TypePtr
can be loaded while the interfaces it implements appear to be not loaded.Another problem is that
TypePtr::InterfaceSet::eq
does not take the_is_loaded
/_is_loaded_computed
fields into account:jdk/src/hotspot/share/opto/type.cpp
Lines 3290 to 3301 in 5726d31
A loaded type can therefore be replaced by an unloaded type during GVN.
In the case of the failure reported by this bug,
LibraryCallKit::inline_native_hashcode
first null checks the receiver and updates the type. Due to the issues described above, the null-free type is GVN'ed with it's unloaded counterpart and propagated to another, redundant null check emitted byLibraryCallKit::generate_method_call
(I filed JDK-8307625 to remove it). Since the type is now unloaded, an uncommon trap is emitted and parsing is stopped(). We crash when trying to de-referenceGraphKit::_map->_jvms
which is NULL (in debug we would hit an assert).Another failure mode is reported by JDK-8305339. Both failures are extremely intermittent and I was never able to reproduce them.
The fix I propose is to completely remove the
_is_loaded
logic because if a klass is loaded, the interface that it implements should be loaded as well. I added an assert to verify this. Higher tier testing is still running.In addition, I noticed some
#ifdef DEBUG
checks whereDEBUG
is never defined. I replaced them by#ifdef ASSERT
and fixed the verification code that failed to build.Thanks,
Tobias
Progress
Issue
Reviewers
Reviewing
Using
git
Checkout this PR locally:
$ git fetch https://git.openjdk.org/jdk.git pull/13868/head:pull/13868
$ git checkout pull/13868
Update a local copy of the PR:
$ git checkout pull/13868
$ git pull https://git.openjdk.org/jdk.git pull/13868/head
Using Skara CLI tools
Checkout this PR locally:
$ git pr checkout 13868
View PR using the GUI difftool:
$ git pr show -t 13868
Using diff file
Download this PR as a diff file:
https://git.openjdk.org/jdk/pull/13868.diff
Webrev
Link to Webrev Comment