changeset in /hg/pulseaudio: fixed the deadlock which occurred w...
Omair Majid
omajid at redhat.com
Fri Aug 8 08:19:51 PDT 2008
changeset c3802b8ed541 in /hg/pulseaudio
details: http://icedtea.classpath.org/hg/pulseaudio?cmd=changeset;node=c3802b8ed541
description:
fixed the deadlock which occurred when adding multiple listeners during the open() call of the PulseAudioMixer
diffstat:
3 files changed, 76 insertions(+), 62 deletions(-)
.hgignore | 8 +
src/org/classpath/icedtea/pulseaudio/EventLoop.java | 35 +---
src/org/classpath/icedtea/pulseaudio/PulseAudioMixer.java | 95 +++++++------
diffs (198 lines):
diff -r 4a01f1203d48 -r c3802b8ed541 .hgignore
--- /dev/null Thu Jan 01 00:00:00 1970 +0000
+++ b/.hgignore Wed Aug 06 14:55:15 2008 -0400
@@ -0,0 +1,8 @@
+# use glob syntax.
+syntax: glob
+
+*.class
+*.o
+*~
+org_classpath_icedtea_*.h
+*.log
diff -r 4a01f1203d48 -r c3802b8ed541 src/org/classpath/icedtea/pulseaudio/EventLoop.java
--- a/src/org/classpath/icedtea/pulseaudio/EventLoop.java Wed Aug 06 14:08:52 2008 -0400
+++ b/src/org/classpath/icedtea/pulseaudio/EventLoop.java Wed Aug 06 14:55:15 2008 -0400
@@ -45,8 +45,8 @@ import org.classpath.icedtea.pulseaudio.
import org.classpath.icedtea.pulseaudio.ContextEvent.Type;
/*
- * any methods that can obstruct the behaviour of pa_mainloop
- * should run synchronized
+ * any methods that can obstruct the behaviour of pa_mainloop should run
+ * synchronized
*
*
*/
@@ -68,7 +68,7 @@ public class EventLoop implements Runnab
private String serverString;
private int status;
- // private boolean eventLoopIsRunning = false;
+ // private boolean eventLoopIsRunning = false;
public Semaphore finished = new Semaphore(0);
@@ -214,25 +214,16 @@ public class EventLoop implements Runnab
private void fireEvent(final ContextEvent e) {
System.out.println(this.getClass().getName() + "firing event: "
+ e.getType().toString());
- Thread th = new Thread(new Runnable() {
-
- @Override
- public void run() {
- System.out.println("notifying listeners");
-
- synchronized (contextListeners) {
- System.out.println(contextListeners.size());
- for (ContextListener listener : contextListeners) {
- listener.update(e);
- System.out.println("updating listeners");
- }
- }
- }
-
- });
-
- th.start();
-
+ System.out.println("notifying listeners");
+
+ synchronized (contextListeners) {
+ System.out.println(contextListeners.size());
+ for (ContextListener listener : contextListeners) {
+ listener.update(e);
+ System.out.println("updating listeners");
+ }
+ }
+
}
public void setVolume(long streamPointer, int volume) {
diff -r 4a01f1203d48 -r c3802b8ed541 src/org/classpath/icedtea/pulseaudio/PulseAudioMixer.java
--- a/src/org/classpath/icedtea/pulseaudio/PulseAudioMixer.java Wed Aug 06 14:08:52 2008 -0400
+++ b/src/org/classpath/icedtea/pulseaudio/PulseAudioMixer.java Wed Aug 06 14:55:15 2008 -0400
@@ -308,12 +308,38 @@ public class PulseAudioMixer implements
eventLoop.setAppName(appName);
eventLoop.setServer(host);
- final Semaphore ready = new Semaphore(0);
-
- ContextListener initListener = new ContextListener() {
+
+ ContextListener eventListener = new ContextListener() {
@Override
public void update(ContextEvent e) {
+ System.out.println("General listener was notified");
+ System.out.println(" eventType: " + e.getType());
+ if (e.getType() == ContextEvent.Type.READY) {
+ System.out.println("firing event ready..");
+ fireEvent(new LineEvent(PulseAudioMixer.this,
+ LineEvent.Type.OPEN, AudioSystem.NOT_SPECIFIED));
+ System.out.println("done");
+ } else if (e.getType() == ContextEvent.Type.FAILED
+ || e.getType() == ContextEvent.Type.TERMINATED) {
+ fireEvent(new LineEvent(PulseAudioMixer.this,
+ LineEvent.Type.CLOSE, AudioSystem.NOT_SPECIFIED));
+ }
+ System.out.println("general listener returning");
+ }
+
+ };
+
+ eventLoop.addContextListener(eventListener);
+
+
+ final Semaphore ready = new Semaphore(0);
+
+ ContextListener initListener = new ContextListener() {
+
+ @Override
+ public void update(ContextEvent e) {
+ System.out.println("specific listener notifed");
System.out.println(this.getClass().getName()
+ ": Event detected " + e.getType().toString());
if (e.getType() == ContextEvent.Type.READY
@@ -322,6 +348,7 @@ public class PulseAudioMixer implements
ready.release();
System.out.println("realeasing semaphore ready");
}
+ System.out.println("specific listener returning");
}
};
@@ -355,48 +382,36 @@ public class PulseAudioMixer implements
}
System.out.println(this.getClass().getName() + ": ready");
-
+
this.isOpen = true;
-// ContextListener eventListener = new ContextListener() {
-//
-// @Override
-// public void update(ContextEvent e) {
-// if (e.getType() == ContextEvent.Type.READY) {
-//
-// fireEvent(new LineEvent(PulseAudioMixer.this,
-// LineEvent.Type.OPEN, AudioSystem.NOT_SPECIFIED));
-// } else if (e.getType() == ContextEvent.Type.FAILED
-// || e.getType() == ContextEvent.Type.TERMINATED) {
-// fireEvent(new LineEvent(PulseAudioMixer.this,
-// LineEvent.Type.CLOSE, AudioSystem.NOT_SPECIFIED));
-// }
-//
-// }
-//
-// };
-
-// eventLoop.addContextListener(eventListener);
-
- }
-
- @Override
- synchronized public void removeLineListener(LineListener listener) {
+
+
+ }
+
+ @Override
+ public void removeLineListener(LineListener listener) {
lineListeners.remove(listener);
}
- synchronized private void fireEvent(final LineEvent e) {
- Thread th = new Thread(new Runnable() {
- @Override
- public void run() {
- synchronized (lineListeners) {
- for (LineListener lineListener : lineListeners) {
- lineListener.update(e);
- }
- }
- }
- });
- th.start();
+ /*
+ * Should this method be synchronized? I had a few reasons, but i forgot them
+ * Pros:
+ * - Thread safety?
+ *
+ * Cons:
+ * - eventListeners are run from other threads, if those then call fireEvent
+ * while a method is waiting on a listener, this synchronized block wont
+ * be entered: deadlock!
+ *
+ */
+ private void fireEvent(final LineEvent e) {
+ System.out.println(this.getClass().getName() + "fireEvent(): firing event");
+ synchronized (lineListeners) {
+ for (LineListener lineListener : lineListeners) {
+ lineListener.update(e);
+ }
+ }
}
public static void main(String[] args) throws Exception {
More information about the distro-pkg-dev
mailing list