<AWT Dev> [16] RFR 8249183: JVM crash in "AwtFrame::WmSize" method
Anton Litvinov
anton.litvinov at oracle.com
Fri Aug 28 13:16:50 UTC 2020
Thank you, Sergey, for review and approval of this fix.
Hello reviewers,
Can anybody else become the second reviewer of this fix?
Thank you,
Anton
On 27/08/2020 03:03, Sergey Bylokhov wrote:
> Looks fine.
>
> On 24.08.2020 16:26, Anton Litvinov wrote:
>> Hi Sergey,
>>
>> Thank you for confirmation about which code in the function
>> "AwtFrame::WmSize" you suggested to move to Java code level. I agree
>> with your suggestion and also came to a conclusion in my previous
>> e-mail that only this code can be moved out. The new version of the
>> fix, which implements your suggestion, was created and tested. Could
>> you please look at the second fix version.
>>
>> Webrev (the 2nd version):
>> http://cr.openjdk.java.net/~alitvinov/8249183/jdk16/webrev.01
>>
>> I studied the involved C++ code flow and confirm that the next 3 C++
>> function calls from the defined by you part of the function
>> "AwtFrame::WmSize" do not alter any fields of the involved C++ object
>> "AwtFrame", "AwtDialog" and are just nothing else then the next 3
>> corresponding Java expressions:
>>
>> 1. C++ function call
>> "SendWindowEvent(java_awt_event_WindowEvent_WINDOW_ICONIFIED);" -->
>> Java expression "sun.awt.windows.WComponentPeer.postEvent(new
>> TimedWindowEvent((Window) target, WindowEvent.WINDOW_ICONIFIED, null,
>> 0, 0, System.currentTimeMillis()));"
>>
>> 2. C++ function call
>> "SendWindowEvent(java_awt_event_WindowEvent_WINDOW_DEICONIFIED);" -->
>> Java expression "sun.awt.windows.WComponentPeer.postEvent(new
>> TimedWindowEvent((Window) target, WindowEvent.WINDOW_DEICONIFIED,
>> null, 0, 0, System.currentTimeMillis()));"
>>
>> 3. C++ function call "SendWindowStateEvent(oldState, newState);" -->
>> Java expression "sun.awt.windows.WComponentPeer.postEvent(new
>> TimedWindowEvent((Window) target, WindowEvent.WINDOW_STATE_CHANGED,
>> null, oldState, newState, System.currentTimeMillis()));"
>>
>> Regarding your remark about the need to take into account
>> "getExtendedState" Java method, I agree with you and confirm that it
>> is impossible that this method will be called on not instance of
>> "WFramePeer" Java class and agree that there is no need to change
>> code related to "getExtendedState" method.
>>
>> CHANGES IN THE SECOND FIX VERSION:
>>
>> Change 1 - The pointed by you part of "AwtFrame::WmSize" function was
>> reimplemented in Java language as the new method
>> "sun.awt.windows.WWindowPeer.notifyWindowStateChanged(int oldState,
>> int newState)" in the file "WWindowPeer.java". JNI Invocation of this
>> new Java method was wrapped in the new C++ function "void
>> AwtWindow::NotifyWindowStateChanged(jint oldState, jint newState)" in
>> the file "awt_Window.cpp".
>>
>> Change 2 - The function "void SendWindowStateEvent(int oldState, int
>> newState);" was removed from the C++ class "AwtFrame" in the files
>> "awt_Frame.h", "awt_Frame.cpp", because it is not called from
>> anywhere in JDK source code after this fix.
>>
>> Change 3 - Java method ID variable "static jmethodID
>> setExtendedStateMID;" with code using it was removed from the C++
>> class "AwtFrame" in the files "awt_Frame.h", "awt_Frame.cpp", because
>> it is not called from anywhere in JDK source code after this fix.
>>
>> Thank you,
>> Anton
>>
>> On 20/08/2020 05:21, Sergey Bylokhov wrote:
>>> On 19.08.2020 09:16, Anton Litvinov wrote:
>>>> Thank you very much for review of this fix. I have been evaluating
>>>> your suggestion and my opinion is following. On macOS that code was
>>>> implemented already from the moment JDK 7 code for macOS appeared.
>>>> Although macOS-specific platform-dependent code on Java level has
>>>> some similarities with Windows platform-dependent code on Java
>>>> level, native platform-dependent code is very different between
>>>> these two OSes.
>>>>
>>>> Of course it is technically possible to move some part of
>>>> "AwtFrame::WmSize" function
>>>> (http://hg.openjdk.java.net/jdk/client/file/2c7b5ae39c19/src/java.desktop/windows/native/libawt/windows/awt_Frame.cpp#l915)
>>>> to some new method or methods in "sun.awt.windows.WWindowPeer", but
>>>> that fix will be much bigger, will require different changes both
>>>> in C++ and Java code, and will be necessary to prove that such
>>>> refactoring did not break anything previously working well.
>>>> "AwtFrame::WmSize" function depends on:
>>>> - "AwtFrame" class fields: "m_iconic", "m_zoomed" through the
>>>> functions: "isIconic()", "isZoomed()", "setIconic(BOOL)",
>>>> "setZoomed(BOOL)".
>>>> - Win32 API constants: "SIZE_MINIMIZED", "SIZE_MAXIMIZED".
>>>> - Call to the function "AwtWindow::UpdateSecurityWarningVisibility"
>>>> is not movable to Java level.
>>>
>>>
>>> No I did not meant to move the whole "AwtFrame::WmSize" to the java,
>>> but only the part related for notification, same as on mac.
>>> So instead of calling " AwtFrame::setExtendedStateMID," from the
>>> native, just call something like "WWindowPeer.notifyState(int old,
>>> int new, zoom, iconify,... etc)", and on
>>> the java side call all necessary things like on mac:
>>> http://hg.openjdk.java.net/jdk/jdk/file/6b984aa424e3/src/java.desktop/macosx/classes/sun/lwawt/LWWindowPeer.java#l661
>>>
>>> - postEvent iconify/zoom;
>>> - postWindowStateChangedEvent(newWindowState);
>>> etc
>>>
>>> Probably this part of the code in awt_Frame.cpp can be moved to
>>> "WWindowPeer.notifyState(...)?
>>>
>>> DTRACE_PRINTLN2("AwtFrame::WmSize: reporting state change %x
>>> -> %x",
>>> oldState, newState);
>>>
>>> // sync target with peer
>>> JNIEnv *env = (JNIEnv *)JNU_GetEnv(jvm, JNI_VERSION_1_2);
>>> env->CallVoidMethod(GetPeer(env),
>>> AwtFrame::setExtendedStateMID, newState);
>>>
>>> // report (de)iconification to old clients
>>> if (changed & java_awt_Frame_ICONIFIED) {
>>> if (newState & java_awt_Frame_ICONIFIED) {
>>> SendWindowEvent(java_awt_event_WindowEvent_WINDOW_ICONIFIED);
>>> } else {
>>> SendWindowEvent(java_awt_event_WindowEvent_WINDOW_DEICONIFIED);
>>> }
>>> }
>>>
>>> // New (since 1.4) state change event
>>> SendWindowStateEvent(oldState, newState);
>>>
>>>
>>>
>>>>
>>>> It looks that only contents of "if (changed != 0) {" block
>>>> (http://hg.openjdk.java.net/jdk/client/file/2c7b5ae39c19/src/java.desktop/windows/native/libawt/windows/awt_Frame.cpp#l975)
>>>> can be moved to some new Java method. But what to do with JNI
>>>> invocation of "getExtendedState()" method
>>>> (http://hg.openjdk.java.net/jdk/client/file/2c7b5ae39c19/src/java.desktop/windows/native/libawt/windows/awt_Frame.cpp#l281).
>>>>
>>>> To my mind, such refactoring will not give much benefit, new Java
>>>> method or methods will not be used from anywhere else, but it will
>>>> be required to change code which currently works stably and maybe
>>>> the changes will bring some new problems. I would prefer to make
>>>> the fix as narrow as possible for this not very usual scenario
>>>> leading to the crash.
>>>>
>>>> ALTERNATIVE SUGGESTION FROM ME:
>>>>
>>>> By doing this minimal version of the fix, I tried to avoid even JNI
>>>> usage without the need. If you do not like this custom flag, then
>>>> would you agree for its change to a normal check of peer object
>>>> class using "instanceof" operator?
>>>>
>>>> For example, to use instead of this flag the next if condition in
>>>> "awt_Frame.cpp":
>>>>
>>>> "jobject peer = GetPeer();
>>>> if ((peer != NULL) && (JNU_IsInstanceOfByName(env, peer,
>>>> "sun/awt/windows/WFramePeer") > 0)) {"
>>>>
>>>> Or if you do not like variant with "instanceof" operator, then
>>>> instead of introduction any checks anywhere, I can suggest you as
>>>> another alternative option to implement 2 new empty private
>>>> methods: "int getExtendedState()", "void setExtendedState(int)" in
>>>> the class "sun.awt.windows.WWindowPeer".
>>>
>>> Do we really need to change the code related to the
>>> "getExtendedState" during creation of AWTFrame?
>>> That one should never be used by the AwtDialog, so the crash is kind
>>> of assertion that something go wrong already.
>>>
>>>
>>
>
>
More information about the awt-dev
mailing list