提交 212cf16d 编写于 作者: A asaha

Merge

...@@ -415,4 +415,5 @@ f732971e3d20664164a3797cf0b1a4cb80470959 jdk8u51-b03 ...@@ -415,4 +415,5 @@ f732971e3d20664164a3797cf0b1a4cb80470959 jdk8u51-b03
286b9a885fcc6245fdf2b20697473ec3b35f2538 jdk8u51-b07 286b9a885fcc6245fdf2b20697473ec3b35f2538 jdk8u51-b07
f7da0b943b9381aaf378d0c7b337dd7654335293 jdk8u51-b08 f7da0b943b9381aaf378d0c7b337dd7654335293 jdk8u51-b08
7e8459e7a45cb5b49de376893e3a95bfa92d0325 jdk8u51-b09 7e8459e7a45cb5b49de376893e3a95bfa92d0325 jdk8u51-b09
dcc75a75d3a30270fbf52d0d0b0504319882e419 jdk8u51-b10
286b9a885fcc6245fdf2b20697473ec3b35f2538 jdk8u65-b00 286b9a885fcc6245fdf2b20697473ec3b35f2538 jdk8u65-b00
...@@ -56,6 +56,18 @@ final class CClipboard extends SunClipboard { ...@@ -56,6 +56,18 @@ final class CClipboard extends SunClipboard {
// Leaving Empty, as WClipboard.clearNativeContext is empty as well. // Leaving Empty, as WClipboard.clearNativeContext is empty as well.
} }
@Override
public synchronized Transferable getContents(Object requestor) {
checkPasteboardAndNotify();
return super.getContents(requestor);
}
@Override
protected synchronized Transferable getContextContents() {
checkPasteboardAndNotify();
return super.getContextContents();
}
@Override @Override
protected void setContentsNative(Transferable contents) { protected void setContentsNative(Transferable contents) {
FlavorTable flavorMap = getDefaultFlavorTable(); FlavorTable flavorMap = getDefaultFlavorTable();
...@@ -116,13 +128,20 @@ final class CClipboard extends SunClipboard { ...@@ -116,13 +128,20 @@ final class CClipboard extends SunClipboard {
private native void declareTypes(long[] formats, SunClipboard newOwner); private native void declareTypes(long[] formats, SunClipboard newOwner);
private native void setData(byte[] data, long format); private native void setData(byte[] data, long format);
void checkPasteboardAndNotify() {
if (checkPasteboardWithoutNotification()) {
notifyChanged();
lostOwnershipNow(null);
}
}
/** /**
* Invokes native check whether a change count on the general pasteboard is different * Invokes native check whether a change count on the general pasteboard is different
* than when we set it. The different count value means the current owner lost * than when we set it. The different count value means the current owner lost
* pasteboard ownership and someone else put data on the clipboard. * pasteboard ownership and someone else put data on the clipboard.
* @since 1.7 * @since 1.7
*/ */
native void checkPasteboard(); native boolean checkPasteboardWithoutNotification();
/*** Native Callbacks ***/ /*** Native Callbacks ***/
private void notifyLostOwnership() { private void notifyLostOwnership() {
......
...@@ -120,7 +120,7 @@ public class CEmbeddedFrame extends EmbeddedFrame { ...@@ -120,7 +120,7 @@ public class CEmbeddedFrame extends EmbeddedFrame {
// it won't be invoced if focuse is moved to a html element // it won't be invoced if focuse is moved to a html element
// on the same page. // on the same page.
CClipboard clipboard = (CClipboard) Toolkit.getDefaultToolkit().getSystemClipboard(); CClipboard clipboard = (CClipboard) Toolkit.getDefaultToolkit().getSystemClipboard();
clipboard.checkPasteboard(); clipboard.checkPasteboardAndNotify();
} }
if (parentWindowActive) { if (parentWindowActive) {
responder.handleWindowFocusEvent(focused, null); responder.handleWindowFocusEvent(focused, null);
......
...@@ -171,6 +171,8 @@ static CClipboard *sClipboard = nil; ...@@ -171,6 +171,8 @@ static CClipboard *sClipboard = nil;
else [args removeLastObject]; else [args removeLastObject];
} }
- (void) checkPasteboard:(id)application { - (void) checkPasteboard:(id)application {
AWT_ASSERT_APPKIT_THREAD; AWT_ASSERT_APPKIT_THREAD;
...@@ -202,6 +204,19 @@ static CClipboard *sClipboard = nil; ...@@ -202,6 +204,19 @@ static CClipboard *sClipboard = nil;
} }
} }
- (BOOL) checkPasteboardWithoutNotification:(id)application {
AWT_ASSERT_APPKIT_THREAD;
NSInteger newChangeCount = [[NSPasteboard generalPasteboard] changeCount];
if (fChangeCount != newChangeCount) {
fChangeCount = newChangeCount;
return YES;
} else {
return NO;
}
}
@end @end
/* /*
...@@ -348,16 +363,17 @@ JNF_COCOA_EXIT(env); ...@@ -348,16 +363,17 @@ JNF_COCOA_EXIT(env);
* Method: checkPasteboard * Method: checkPasteboard
* Signature: ()V * Signature: ()V
*/ */
JNIEXPORT void JNICALL Java_sun_lwawt_macosx_CClipboard_checkPasteboard JNIEXPORT jboolean JNICALL Java_sun_lwawt_macosx_CClipboard_checkPasteboardWithoutNotification
(JNIEnv *env, jobject inObject ) (JNIEnv *env, jobject inObject)
{ {
__block BOOL ret = NO;
JNF_COCOA_ENTER(env); JNF_COCOA_ENTER(env);
[ThreadUtilities performOnMainThreadWaiting:YES block:^(){ [ThreadUtilities performOnMainThreadWaiting:YES block:^(){
[[CClipboard sharedClipboard] checkPasteboard:nil]; ret = [[CClipboard sharedClipboard] checkPasteboardWithoutNotification:nil];
}]; }];
JNF_COCOA_EXIT(env); JNF_COCOA_EXIT(env);
return ret;
} }
...@@ -150,7 +150,7 @@ public abstract class SunClipboard extends Clipboard ...@@ -150,7 +150,7 @@ public abstract class SunClipboard extends Clipboard
* AppContext as it is currently retrieved or null otherwise * AppContext as it is currently retrieved or null otherwise
* @since 1.5 * @since 1.5
*/ */
private synchronized Transferable getContextContents() { protected synchronized Transferable getContextContents() {
AppContext context = AppContext.getAppContext(); AppContext context = AppContext.getAppContext();
return (context == contentsContext) ? contents : null; return (context == contentsContext) ? contents : null;
} }
...@@ -281,42 +281,41 @@ public abstract class SunClipboard extends Clipboard ...@@ -281,42 +281,41 @@ public abstract class SunClipboard extends Clipboard
return; return;
} }
final Runnable runnable = new Runnable() { SunToolkit.postEvent(context, new PeerEvent(this, () -> lostOwnershipNow(disposedContext),
public void run() {
final SunClipboard sunClipboard = SunClipboard.this;
ClipboardOwner owner = null;
Transferable contents = null;
synchronized (sunClipboard) {
final AppContext context = sunClipboard.contentsContext;
if (context == null) {
return;
}
if (disposedContext == null || context == disposedContext) {
owner = sunClipboard.owner;
contents = sunClipboard.contents;
sunClipboard.contentsContext = null;
sunClipboard.owner = null;
sunClipboard.contents = null;
sunClipboard.clearNativeContext();
context.removePropertyChangeListener
(AppContext.DISPOSED_PROPERTY_NAME, sunClipboard);
} else {
return;
}
}
if (owner != null) {
owner.lostOwnership(sunClipboard, contents);
}
}
};
SunToolkit.postEvent(context, new PeerEvent(this, runnable,
PeerEvent.PRIORITY_EVENT)); PeerEvent.PRIORITY_EVENT));
} }
protected void lostOwnershipNow(final AppContext disposedContext) {
final SunClipboard sunClipboard = SunClipboard.this;
ClipboardOwner owner = null;
Transferable contents = null;
synchronized (sunClipboard) {
final AppContext context = sunClipboard.contentsContext;
if (context == null) {
return;
}
if (disposedContext == null || context == disposedContext) {
owner = sunClipboard.owner;
contents = sunClipboard.contents;
sunClipboard.contentsContext = null;
sunClipboard.owner = null;
sunClipboard.contents = null;
sunClipboard.clearNativeContext();
context.removePropertyChangeListener
(AppContext.DISPOSED_PROPERTY_NAME, sunClipboard);
} else {
return;
}
}
if (owner != null) {
owner.lostOwnership(sunClipboard, contents);
}
}
protected abstract void clearNativeContext(); protected abstract void clearNativeContext();
protected abstract void setContentsNative(Transferable contents); protected abstract void setContentsNative(Transferable contents);
......
/*
* Copyright (c) 2015, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* This code is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 only, as
* published by the Free Software Foundation.
*
* This code is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
* version 2 for more details (a copy is included in the LICENSE file that
* accompanied this code).
*
* You should have received a copy of the GNU General Public License version
* 2 along with this work; if not, write to the Free Software Foundation,
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
*
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
* or visit www.oracle.com if you need additional information or have any
* questions.
*/
/*
@test
@bug 8071668
@summary Check whether clipboard see changes from external process after taking ownership
@author Anton Nashatyrev: area=datatransfer
@library /lib/testlibrary
@build jdk.testlibrary.Utils
@run main ClipboardInterVMTest
*/
import jdk.testlibrary.Utils;
import java.awt.*;
import java.awt.datatransfer.*;
import java.io.BufferedReader;
import java.io.File;
import java.io.IOException;
import java.io.Reader;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
import java.util.concurrent.CountDownLatch;
import java.util.concurrent.TimeUnit;
public class ClipboardInterVMTest {
static CountDownLatch lostOwnershipMonitor = new CountDownLatch(1);
static CountDownLatch flavorChangedMonitor = new CountDownLatch(1);
static Process process;
public static void main(String[] args) throws Throwable {
Clipboard clip = Toolkit.getDefaultToolkit().getSystemClipboard();
if (args.length > 0) {
System.out.println("Changing clip...");
clip.setContents(new StringSelection("pong"), null);
System.out.println("done");
// keeping this process running for a while since on Mac the clipboard
// will be invalidated via NSApplicationDidBecomeActiveNotification
// callback in the main process after this child process finishes
Thread.sleep(60 * 1000);
return;
};
clip.setContents(new CustomSelection(), new ClipboardOwner() {
@Override
public void lostOwnership(Clipboard clipboard, Transferable contents) {
System.out.println("ClipboardInterVMTest.lostOwnership");
lostOwnershipMonitor.countDown();
}
});
clip.addFlavorListener(new FlavorListener() {
@Override
public void flavorsChanged(FlavorEvent e) {
System.out.println("ClipboardInterVMTest.flavorsChanged");
flavorChangedMonitor.countDown();
}
});
System.out.println("Starting external clipborad modifier...");
new Thread(() -> runTest(ClipboardInterVMTest.class.getCanonicalName(), "pong")).start();
String content = "";
long startTime = System.currentTimeMillis();
while (System.currentTimeMillis() - startTime < 30 * 1000) {
Transferable c = clip.getContents(null);
if (c.isDataFlavorSupported(DataFlavor.plainTextFlavor)) {
Reader reader = DataFlavor.plainTextFlavor.getReaderForText(c);
content = new BufferedReader(reader).readLine();
System.out.println(content);
if (content.equals("pong")) {
break;
}
}
Thread.sleep(200);
}
if (!lostOwnershipMonitor.await(10, TimeUnit.SECONDS)) {
throw new RuntimeException("No LostOwnership event received.");
};
if (!flavorChangedMonitor.await(10, TimeUnit.SECONDS)) {
throw new RuntimeException("No LostOwnership event received.");
};
if (!content.equals("pong")) {
throw new RuntimeException("Content was not passed.");
}
process.destroy();
System.out.println("Passed.");
}
private static void runTest(String main, String... args) {
try {
List<String> opts = new ArrayList<>();
opts.add(getJavaExe());
opts.addAll(Arrays.asList(Utils.getTestJavaOpts()));
opts.add("-cp");
opts.add(System.getProperty("test.class.path", System.getProperty("java.class.path")));
opts.add(main);
opts.addAll(Arrays.asList(args));
ProcessBuilder pb = new ProcessBuilder(opts.toArray(new String[0]));
process = pb.start();
} catch (Throwable throwable) {
throw new RuntimeException(throwable);
}
}
private static String getJavaExe() throws IOException {
File p = new File(System.getProperty("java.home"), "bin");
File j = new File(p, "java");
if (!j.canRead()) {
j = new File(p, "java.exe");
}
if (!j.canRead()) {
throw new RuntimeException("Can't find java executable in " + p);
}
return j.getCanonicalPath();
}
static class CustomSelection implements Transferable {
private static final DataFlavor[] flavors = { DataFlavor.allHtmlFlavor };
public DataFlavor[] getTransferDataFlavors() {
return flavors;
}
public boolean isDataFlavorSupported(DataFlavor flavor) {
return flavors[0].equals(flavor);
}
public Object getTransferData(DataFlavor flavor)
throws UnsupportedFlavorException, java.io.IOException {
if (isDataFlavorSupported(flavor)) {
return "ping";
} else {
throw new UnsupportedFlavorException(flavor);
}
}
}
}
\ No newline at end of file
/* /*
* Copyright (c) 2002, 2013, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 2002, 2015, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
...@@ -45,7 +45,7 @@ import javax.net.ssl.*; ...@@ -45,7 +45,7 @@ import javax.net.ssl.*;
public class CipherTest { public class CipherTest {
// use any available port for the server socket // use any available port for the server socket
static int serverPort = 0; static volatile int serverPort = 0;
final int THREADS; final int THREADS;
......
/* /*
* Copyright (c) 2002, 2006, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 2002, 2015, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
...@@ -42,7 +42,7 @@ class JSSEServer extends CipherTest.Server { ...@@ -42,7 +42,7 @@ class JSSEServer extends CipherTest.Server {
serverContext.init(new KeyManager[] {cipherTest.keyManager}, new TrustManager[] {cipherTest.trustManager}, cipherTest.secureRandom); serverContext.init(new KeyManager[] {cipherTest.keyManager}, new TrustManager[] {cipherTest.trustManager}, cipherTest.secureRandom);
SSLServerSocketFactory factory = (SSLServerSocketFactory)serverContext.getServerSocketFactory(); SSLServerSocketFactory factory = (SSLServerSocketFactory)serverContext.getServerSocketFactory();
serverSocket = (SSLServerSocket)factory.createServerSocket(cipherTest.serverPort); serverSocket = (SSLServerSocket)factory.createServerSocket(0);
cipherTest.serverPort = serverSocket.getLocalPort(); cipherTest.serverPort = serverSocket.getLocalPort();
serverSocket.setEnabledCipherSuites(factory.getSupportedCipherSuites()); serverSocket.setEnabledCipherSuites(factory.getSupportedCipherSuites());
serverSocket.setWantClientAuth(true); serverSocket.setWantClientAuth(true);
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册