提交 f9a1bf8c 编写于 作者: M malenkov

8030118: Document listeners fired outside document lock

Reviewed-by: art, serb
上级 23bd4b84
...@@ -697,7 +697,6 @@ public abstract class AbstractDocument implements Document, Serializable { ...@@ -697,7 +697,6 @@ public abstract class AbstractDocument implements Document, Serializable {
return; return;
} }
DocumentFilter filter = getDocumentFilter(); DocumentFilter filter = getDocumentFilter();
InsertStringResult insertStringResult = null;
writeLock(); writeLock();
...@@ -705,23 +704,21 @@ public abstract class AbstractDocument implements Document, Serializable { ...@@ -705,23 +704,21 @@ public abstract class AbstractDocument implements Document, Serializable {
if (filter != null) { if (filter != null) {
filter.insertString(getFilterBypass(), offs, str, a); filter.insertString(getFilterBypass(), offs, str, a);
} else { } else {
insertStringResult = handleInsertString(offs, str, a); handleInsertString(offs, str, a);
} }
} finally { } finally {
writeUnlock(); writeUnlock();
} }
processInsertStringResult(insertStringResult);
} }
/** /**
* Performs the actual work of inserting the text; it is assumed the * Performs the actual work of inserting the text; it is assumed the
* caller has obtained a write lock before invoking this. * caller has obtained a write lock before invoking this.
*/ */
private InsertStringResult handleInsertString(int offs, String str, AttributeSet a) private void handleInsertString(int offs, String str, AttributeSet a)
throws BadLocationException { throws BadLocationException {
if ((str == null) || (str.length() == 0)) { if ((str == null) || (str.length() == 0)) {
return null; return;
} }
UndoableEdit u = data.insertString(offs, str); UndoableEdit u = data.insertString(offs, str);
DefaultDocumentEvent e = DefaultDocumentEvent e =
...@@ -748,29 +745,11 @@ public abstract class AbstractDocument implements Document, Serializable { ...@@ -748,29 +745,11 @@ public abstract class AbstractDocument implements Document, Serializable {
insertUpdate(e, a); insertUpdate(e, a);
// Mark the edit as done. // Mark the edit as done.
e.end(); e.end();
fireInsertUpdate(e);
InsertStringResult result = new InsertStringResult();
result.documentEvent = e;
// only fire undo if Content implementation supports it // only fire undo if Content implementation supports it
// undo for the composed text is not supported for now // undo for the composed text is not supported for now
if (u != null && (a == null || !a.isDefined(StyleConstants.ComposedTextAttribute))) { if (u != null && (a == null || !a.isDefined(StyleConstants.ComposedTextAttribute))) {
result.undoableEditEvent = new UndoableEditEvent(this, e); fireUndoableEditUpdate(new UndoableEditEvent(this, e));
}
return result;
}
private void processInsertStringResult(InsertStringResult insertStringResult) {
if (insertStringResult == null) {
return;
}
fireInsertUpdate(insertStringResult.documentEvent);
if (insertStringResult.undoableEditEvent != null) {
fireUndoableEditUpdate(insertStringResult.undoableEditEvent);
} }
} }
...@@ -3125,23 +3104,13 @@ public abstract class AbstractDocument implements Document, Serializable { ...@@ -3125,23 +3104,13 @@ public abstract class AbstractDocument implements Document, Serializable {
public void insertString(int offset, String string, public void insertString(int offset, String string,
AttributeSet attr) throws AttributeSet attr) throws
BadLocationException { BadLocationException {
InsertStringResult insertStringResult = handleInsertString(offset, string, attr); handleInsertString(offset, string, attr);
processInsertStringResult(insertStringResult);
} }
public void replace(int offset, int length, String text, public void replace(int offset, int length, String text,
AttributeSet attrs) throws BadLocationException { AttributeSet attrs) throws BadLocationException {
handleRemove(offset, length); handleRemove(offset, length);
handleInsertString(offset, text, attrs);
InsertStringResult insertStringResult = handleInsertString(offset, text, attrs);
processInsertStringResult(insertStringResult);
} }
} }
private static class InsertStringResult {
DefaultDocumentEvent documentEvent;
UndoableEditEvent undoableEditEvent;
}
} }
/* /*
* Copyright (c) 2012, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 2013, 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
...@@ -21,57 +21,63 @@ ...@@ -21,57 +21,63 @@
* questions. * questions.
*/ */
/* @test import javax.swing.event.DocumentEvent;
@bug 7146146 import javax.swing.event.DocumentListener;
@summary Deadlock between subclass of AbstractDocument and UndoManager
@author Pavel Porvatov
*/
import javax.swing.text.BadLocationException; import javax.swing.text.BadLocationException;
import javax.swing.text.PlainDocument; import javax.swing.text.PlainDocument;
import javax.swing.text.StringContent; import java.util.concurrent.CountDownLatch;
import javax.swing.undo.UndoManager; import java.util.concurrent.TimeUnit;
public class bug7146146 { /*
public static void main(String[] args) throws Exception { * @test
for (int i = 0; i < 1000; i++) { * @bug 8030118
System.out.print("Iteration " + i); * @summary Tests that AbstractDocument cannot be modified from another thread
* @author Sergey Malenkov
*/
test(); public class Test8030118 implements DocumentListener, Runnable {
private final CountDownLatch latch = new CountDownLatch(1);
private final PlainDocument doc = new PlainDocument();
System.out.print(" passed"); private Test8030118(String string) throws Exception {
} this.doc.addDocumentListener(this);
this.doc.insertString(0, string, null);
} }
private static void test() throws Exception {
final PlainDocument doc = new PlainDocument(new StringContent());
final UndoManager undoManager = new UndoManager();
doc.addUndoableEditListener(undoManager);
doc.insertString(0, "<Test 1>", null);
Thread t1 = new Thread("Thread 1") {
@Override @Override
public void run() { public void run() {
try { try {
doc.insertString(0, "<Test 2>", null); this.doc.remove(0, this.doc.getLength());
} catch (BadLocationException e) { } catch (BadLocationException exception) {
throw new RuntimeException(e); throw new Error("unexpected", exception);
} }
this.latch.countDown();
} }
};
Thread t2 = new Thread("Thread 2") {
@Override @Override
public void run() { public void insertUpdate(DocumentEvent event) {
undoManager.undo(); new Thread(this).start();
try {
this.latch.await(10, TimeUnit.SECONDS);
} catch (InterruptedException exception) {
throw new Error("unexpected", exception);
}
try {
event.getDocument().getText(event.getOffset(), event.getLength());
} catch (BadLocationException exception) {
throw new Error("concurrent modification", exception);
}
} }
};
t1.start(); @Override
t2.start(); public void removeUpdate(DocumentEvent event) {
}
t1.join(); @Override
t2.join(); public void changedUpdate(DocumentEvent event) {
}
public static void main(String[] args) throws Exception {
new Test8030118("string");
} }
} }
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册