f3ae7d1bd9
Reviewed-by: serb, jdv
339 lines
11 KiB
Java
339 lines
11 KiB
Java
/*
|
|
* Copyright (c) 2007, 2018, 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 6678218 6681745 6691737 8198613
|
|
* @summary Tests that v-synced BufferStrategies works (if vsync is supported)
|
|
* @author Dmitri.Trembovetski@sun.com: area=Graphics
|
|
* @modules java.desktop/sun.java2d.pipe.hw
|
|
* @compile -XDignore.symbol.file=true VSyncedBufferStrategyTest.java
|
|
* @run main/manual/othervm VSyncedBufferStrategyTest
|
|
*/
|
|
|
|
import java.awt.AWTException;
|
|
import java.awt.BufferCapabilities;
|
|
import java.awt.BufferCapabilities.FlipContents;
|
|
import java.awt.Button;
|
|
import java.awt.Canvas;
|
|
import java.awt.Color;
|
|
import java.awt.Dimension;
|
|
import java.awt.EventQueue;
|
|
import java.awt.FlowLayout;
|
|
import java.awt.Font;
|
|
import java.awt.Frame;
|
|
import java.awt.Graphics;
|
|
import java.awt.HeadlessException;
|
|
import java.awt.ImageCapabilities;
|
|
import java.awt.Panel;
|
|
import java.awt.event.ActionEvent;
|
|
import java.awt.event.ActionListener;
|
|
import java.awt.event.WindowAdapter;
|
|
import java.awt.event.WindowEvent;
|
|
import java.awt.image.BufferStrategy;
|
|
import java.util.concurrent.CountDownLatch;
|
|
import javax.swing.JButton;
|
|
import javax.swing.JFrame;
|
|
import javax.swing.JPanel;
|
|
import javax.swing.JScrollPane;
|
|
import javax.swing.JTextArea;
|
|
|
|
public class VSyncedBufferStrategyTest extends Canvas implements Runnable {
|
|
|
|
private static final int BLOCK_W = 50;
|
|
private static final int BLOCK_H = 200;
|
|
|
|
BufferStrategy bs;
|
|
Thread renderThread;
|
|
|
|
int blockX = 10;
|
|
int blockY = 10;
|
|
|
|
private volatile boolean done = false;
|
|
private volatile boolean requestVSync;
|
|
private boolean currentBSVSynced;
|
|
|
|
public VSyncedBufferStrategyTest(boolean requestVSync) {
|
|
this.requestVSync = requestVSync;
|
|
this.currentBSVSynced = !requestVSync;
|
|
renderThread = new Thread(this);
|
|
renderThread.start();
|
|
}
|
|
|
|
private static final BufferCapabilities defaultBC =
|
|
new BufferCapabilities(
|
|
new ImageCapabilities(true),
|
|
new ImageCapabilities(true),
|
|
null);
|
|
|
|
private void createBS(boolean requestVSync) {
|
|
if (bs != null && requestVSync == currentBSVSynced) {
|
|
return;
|
|
}
|
|
|
|
BufferCapabilities bc = defaultBC;
|
|
if (requestVSync) {
|
|
bc = new sun.java2d.pipe.hw.ExtendedBufferCapabilities(
|
|
new ImageCapabilities(true),
|
|
new ImageCapabilities(true),
|
|
FlipContents.COPIED,
|
|
sun.java2d.pipe.hw.ExtendedBufferCapabilities.VSyncType.VSYNC_ON);
|
|
}
|
|
try {
|
|
createBufferStrategy(2, bc);
|
|
} catch (AWTException e) {
|
|
System.err.println("Warning: cap is not supported: "+bc);
|
|
e.printStackTrace();
|
|
createBufferStrategy(2);
|
|
}
|
|
currentBSVSynced = requestVSync;
|
|
bs = getBufferStrategy();
|
|
String s =
|
|
getParent() instanceof Frame ?
|
|
((Frame)getParent()).getTitle() : "parent";
|
|
System.out.println("Created BS for \"" + s + "\" frame, bs="+bs);
|
|
}
|
|
|
|
@Override
|
|
public void paint(Graphics g) {
|
|
}
|
|
@Override
|
|
public void update(Graphics g) {
|
|
}
|
|
|
|
@Override
|
|
public void run() {
|
|
while (!isShowing()) {
|
|
try { Thread.sleep(5); } catch (InterruptedException e) {}
|
|
}
|
|
try { Thread.sleep(2000); } catch (InterruptedException e) {}
|
|
|
|
try {
|
|
while (!done && isShowing()) {
|
|
createBS(requestVSync);
|
|
do {
|
|
step();
|
|
Graphics g = bs.getDrawGraphics();
|
|
render(g);
|
|
if (!bs.contentsRestored()) {
|
|
bs.show();
|
|
}
|
|
} while (bs.contentsLost());
|
|
Thread.yield();
|
|
}
|
|
} catch (Throwable e) {
|
|
// since we're not bothering with proper synchronization, exceptions
|
|
// may be thrown when the frame is closed
|
|
if (isShowing()) {
|
|
throw new RuntimeException(e);
|
|
}
|
|
}
|
|
}
|
|
|
|
int inc = 5;
|
|
private void step() {
|
|
blockX += inc;
|
|
if (blockX > getWidth() - BLOCK_W - 10) {
|
|
inc = -inc;
|
|
blockX += inc;
|
|
}
|
|
if (blockX < 10) {
|
|
inc = -inc;
|
|
blockX += inc;
|
|
}
|
|
}
|
|
|
|
private void render(Graphics g) {
|
|
g.setColor(Color.white);
|
|
g.fillRect(0, 0, getWidth(), getHeight());
|
|
|
|
g.setColor(Color.black);
|
|
g.fillRect(blockX, blockY, BLOCK_W, BLOCK_H);
|
|
}
|
|
|
|
private void setRequestVSync(boolean reqVSync) {
|
|
requestVSync = reqVSync;
|
|
}
|
|
|
|
@Override
|
|
public Dimension getPreferredSize() {
|
|
return new Dimension(BLOCK_W*10+20, BLOCK_H+20);
|
|
}
|
|
|
|
private static int frameNum = 0;
|
|
private static Frame createAndShowBSFrame() {
|
|
final Frame f = new Frame("Not V-Synced");
|
|
|
|
int myNum;
|
|
synchronized (VSyncedBufferStrategyTest.class) {
|
|
myNum = frameNum++;
|
|
}
|
|
|
|
final VSyncedBufferStrategyTest component =
|
|
new VSyncedBufferStrategyTest(false);
|
|
f.setIgnoreRepaint(true);
|
|
f.add("Center", component);
|
|
|
|
Panel p = new Panel();
|
|
|
|
Button b = new Button("Request VSync");
|
|
b.addActionListener(new ActionListener() {
|
|
public void actionPerformed(ActionEvent e) {
|
|
f.setTitle("Possibly V-Synced");
|
|
component.setRequestVSync(true);
|
|
}
|
|
});
|
|
p.add(b);
|
|
|
|
b = new Button("Relinquish VSync");
|
|
b.addActionListener(new ActionListener() {
|
|
int inc = 1;
|
|
public void actionPerformed(ActionEvent e) {
|
|
f.setTitle("Not V-Synced");
|
|
component.setRequestVSync(false);
|
|
f.setSize(f.getWidth()+inc, f.getHeight());
|
|
inc = -inc;
|
|
}
|
|
});
|
|
p.add(b);
|
|
|
|
f.add("South", p);
|
|
|
|
f.pack();
|
|
f.setLocation(10, myNum * f.getHeight());
|
|
f.setVisible(true);
|
|
f.addWindowListener(new WindowAdapter() {
|
|
@Override
|
|
public void windowClosing(WindowEvent e) {
|
|
component.done = true;
|
|
f.dispose();
|
|
}
|
|
@Override
|
|
public void windowClosed(WindowEvent e) {
|
|
component.done = true;
|
|
}
|
|
});
|
|
|
|
return f;
|
|
}
|
|
|
|
private static final String description =
|
|
"Tests that v-synced BufferStrategy works. Note that it in some\n" +
|
|
"cases the v-sync can not be enabled, and it is accepted.\n" +
|
|
"The following however is true: only one buffer strategy at a time can\n"+
|
|
"be created v-synced. In order for other BS to become v-synced, the one\n"+
|
|
"that currently is v-synched (or its window) needs to be disposed.\n" +
|
|
"Try the following scenarios:\n" +
|
|
" - click the \"Request VSync\" button in one of the frames. If the\n"+
|
|
" behavior of the animation changes - the animation becomes smooth\n" +
|
|
" it had successfully created a v-synced BS. Note that the animation\n" +
|
|
" in other frames may also become smoother - this is a side-effect\n"+
|
|
" of one of the BS-es becoming v-synched\n" +
|
|
" - click the \"Relinquish VSync\" button on the same frame. If the\n"+
|
|
" behavior changes to the original (tearing)- it had successfully\n" +
|
|
" created a non-vsynced strategy.\n" +
|
|
" - next, try making another one v-synced. It should succeed.\n" +
|
|
" - next, try making another one v-synced - while there's already\n" +
|
|
" a v-synced frame. It should not succeed - meaning, it shouldn't\n" +
|
|
" appear to become smoother, and the behavior of the current v-synced\n" +
|
|
" frame shouldn't change.\n" +
|
|
"\n" +
|
|
"If there aren't any BufferStrategy-related exceptions or other\n" +
|
|
"issues, and the scenarios worked, the test passed, otherwise it\n"+
|
|
"failed.\n";
|
|
|
|
private static void createAndShowDescGUI(final Frame f3, final Frame f1,
|
|
final Frame f2)
|
|
throws HeadlessException, RuntimeException
|
|
{
|
|
final JFrame desc =
|
|
new JFrame("VSyncedBufferStrategyTest - Description");
|
|
desc.addWindowListener(new WindowAdapter() {
|
|
|
|
@Override
|
|
public void windowClosing(WindowEvent e) {
|
|
f1.dispose();
|
|
f2.dispose();
|
|
f3.dispose();
|
|
l.countDown();
|
|
}
|
|
});
|
|
JPanel p = new JPanel();
|
|
JButton bPassed = new JButton("Passed");
|
|
bPassed.addActionListener(new ActionListener() {
|
|
public void actionPerformed(ActionEvent e) {
|
|
desc.dispose();
|
|
f1.dispose();
|
|
f2.dispose();
|
|
f3.dispose();
|
|
l.countDown();
|
|
}
|
|
});
|
|
JButton bFailed = new JButton("Failed");
|
|
bFailed.addActionListener(new ActionListener() {
|
|
public void actionPerformed(ActionEvent e) {
|
|
failed = true;
|
|
desc.dispose();
|
|
f1.dispose();
|
|
f2.dispose();
|
|
f3.dispose();
|
|
l.countDown();
|
|
}
|
|
});
|
|
p.setLayout(new FlowLayout());
|
|
p.add(bPassed);
|
|
p.add(bFailed);
|
|
JTextArea ta = new JTextArea(24, 75);
|
|
ta.setFont(new Font(Font.MONOSPACED, Font.PLAIN, 12));
|
|
ta.setEditable(false);
|
|
ta.setText(description);
|
|
desc.add("Center", new JScrollPane(ta));
|
|
desc.add("South", p);
|
|
desc.pack();
|
|
desc.setLocation(BLOCK_W*10+50, 0);
|
|
desc.setVisible(true);
|
|
}
|
|
|
|
private static void createTestFrames() {
|
|
Frame f1 = createAndShowBSFrame();
|
|
Frame f2 = createAndShowBSFrame();
|
|
Frame f3 = createAndShowBSFrame();
|
|
createAndShowDescGUI(f1, f2, f3);
|
|
}
|
|
|
|
static boolean failed = false;
|
|
static CountDownLatch l = new CountDownLatch(1);
|
|
public static void main(String[] args) throws Exception {
|
|
EventQueue.invokeLater(new Runnable() {
|
|
public void run() {
|
|
createTestFrames();
|
|
}
|
|
});
|
|
l.await();
|
|
if (failed) {
|
|
throw new RuntimeException("Test FAILED");
|
|
}
|
|
System.out.println("Test PASSED");
|
|
}
|
|
|
|
}
|