changeset in /hg/pulseaudio: 2008-09-23 Omair Majid <omajid at redh...

Omair Majid omajid at redhat.com
Tue Sep 23 12:08:59 PDT 2008


changeset c46f6e0e7959 in /hg/pulseaudio
details: http://icedtea.classpath.org/hg/pulseaudio?cmd=changeset;node=c46f6e0e7959
description:
	2008-09-23 Omair Majid <omajid at redhat.com>

	    * src/java/org/classpath/icedtea/pulseaudio/PulseAudioClip.java
	    Removed clipThreadStarted. Use clipThread.isAlive() etc.
	    Removed unused clipSemaphore
	    (ClipThread.run): Fixed to run without clipSemaphore.
	    (close): Reset currentFrame and framesSinceOpen on close.
	    (getFramePosition): Synchronized on framesSinceOpen.
	    (loop): Replaced use of clipThreadStarted with clipThread.isAlive.
	    (open): Initialize currentFrame, framesSinceOpen and endFrame.
	    (stop): Remember which frame we stopped at.

	    * src/java/org/classpath/icedtea/pulseaudio/PulseAudioDataLine.java
	    (open): Clean up properly on error.

	    * src/java/org/classpath/icedtea/pulseaudio/PulseAudioSourceDataLine.java
	    (write): On interrupt, return asap and mark the thread as being interrupted.

	    * unittests/org/classpath/icedtea/pulseaudio/PulseAudioClipTest.java
	    (testFramePosition): Renamed to testFramePositionBeforeClose.
	    (testFramePositionWithStartStop): New function. Checks that starting
	    and stopping doesnt change the frame counts.
	    (testFramePositionWithLoop): Checks that looping doesnt interfere with
	    the frame position.

	    * unittests/org/classpath/icedtea/pulseaudio/PulseAudioSourceDataLineRawTest.java
	    (ThreadWriter.run): Removed useless TODO comment.

	    * unittests/org/classpath/icedtea/pulseaudio/PulseAudioSourceDataLineTest.java
	    (ThreadWriter.run): Removed useless TODO.
	    (setUp): Initialize listeners.

diffstat:

6 files changed, 168 insertions(+), 48 deletions(-)
src/java/org/classpath/icedtea/pulseaudio/PulseAudioClip.java                   |   86 +++++---
src/java/org/classpath/icedtea/pulseaudio/PulseAudioDataLine.java               |   13 -
src/java/org/classpath/icedtea/pulseaudio/PulseAudioSourceDataLine.java         |   10 
unittests/org/classpath/icedtea/pulseaudio/PulseAudioClipTest.java              |  103 +++++++++-
unittests/org/classpath/icedtea/pulseaudio/PulseAudioSourceDataLineRawTest.java |    1 
unittests/org/classpath/icedtea/pulseaudio/PulseAudioSourceDataLineTest.java    |    3 

diffs (482 lines):

diff -r 381f212496eb -r c46f6e0e7959 src/java/org/classpath/icedtea/pulseaudio/PulseAudioClip.java
--- a/src/java/org/classpath/icedtea/pulseaudio/PulseAudioClip.java	Tue Sep 23 11:38:18 2008 -0400
+++ b/src/java/org/classpath/icedtea/pulseaudio/PulseAudioClip.java	Tue Sep 23 15:02:55 2008 -0400
@@ -38,7 +38,6 @@ package org.classpath.icedtea.pulseaudio
 package org.classpath.icedtea.pulseaudio;
 
 import java.io.IOException;
-import java.util.concurrent.Semaphore;
 
 import javax.sound.sampled.AudioFormat;
 import javax.sound.sampled.AudioInputStream;
@@ -57,53 +56,57 @@ public class PulseAudioClip extends Puls
 	private float volume;
 
 	// these are frame indices. so counted from 0
+	// the current frame index
 	private int currentFrame = 0;
+
+	// total number of frames in this clip
 	private int frameCount = 0;
+
+	// the starting frame of the loop
 	private int startFrame = 0;
+	// the ending frame of the loop
 	private int endFrame = 0;
+
+	// the total number of frames played since this line was opened
 	private int framesSinceOpen = 0;
 
 	public static final String DEFAULT_CLIP_NAME = "Clip";
 
 	private Object clipLock = new Object();
-	private boolean clipThreadStarted;
 	private int loopsLeft;
-	private Semaphore clipSemaphore = new Semaphore(1);
+
+	// private Semaphore clipSemaphore = new Semaphore(1);
 
 	private class ClipThread extends Thread {
 		@Override
 		public void run() {
 
-			clipThreadStarted = true;
 			while (loopsLeft >= 0) {
 				writeFrames(currentFrame, endFrame + 1);
 				if (Thread.interrupted()) {
 					// Thread.currentThread().interrupt();
-					clipThreadStarted = false;
+					// System.out.println("returned from interrupted
+					// writeFrames");
 					break;
 				}
 
-				try {
-					// if loop(0) has been called from the mainThread,
-					// wait until loopsLeft has been set
-					clipSemaphore.acquire();
-					if (loopsLeft == 0) {
-						System.out.println("Reading to the end of the file");
-						writeFrames(endFrame, getFrameLength());
-						break;
-					} else {
-						synchronized (clipLock) {
-							currentFrame = startFrame;
-							loopsLeft--;
-						}
+				// if loop(0) has been called from the mainThread,
+				// wait until loopsLeft has been set
+				if (loopsLeft == 0) {
+					// System.out.println("Reading to the end of the file");
+					// System.out.println("endFrame: " + endFrame);
+					writeFrames(endFrame, getFrameLength());
+					break;
+				} else {
+					synchronized (clipLock) {
+						currentFrame = startFrame;
+						loopsLeft--;
 					}
-					clipSemaphore.release();
-				} catch (InterruptedException e) {
-					break;
 				}
 
 			}
 
+			// drain
 			Operation operation;
 
 			synchronized (eventLoop.threadLock) {
@@ -145,6 +148,9 @@ public class PulseAudioClip extends Puls
 						try {
 							eventLoop.threadLock.wait();
 						} catch (InterruptedException e) {
+							// System.out
+							// .println("interrupted while waiting for
+							// getWritableSize");
 							// clean up and return
 							Thread.currentThread().interrupt();
 							stream.removeWriteListener(writeListener);
@@ -160,16 +166,18 @@ public class PulseAudioClip extends Puls
 						framesToWrite * getFormat().getFrameSize());
 				remainingFrames -= framesToWrite;
 				currentFrame += framesToWrite;
+				framesSinceOpen += framesToWrite;
 				if (Thread.interrupted()) {
 					Thread.currentThread().interrupt();
 					break;
 				}
 				// System.out.println("remaining frames" + remainingFrames);
+				// System.out.println("currentFrame: " + currentFrame);
+				// System.out.println("framesSinceOpen: " + framesSinceOpen);
 			}
 		}
 
 		stream.removeWriteListener(writeListener);
-
 	}
 
 	public PulseAudioClip(EventLoop eventLoop, AudioFormat[] formats,
@@ -210,6 +218,9 @@ public class PulseAudioClip extends Puls
 			e.printStackTrace();
 		}
 
+		currentFrame = 0;
+		framesSinceOpen = 0;
+
 		PulseAudioMixer mixer = PulseAudioMixer.getInstance();
 		mixer.removeSourceLine(this);
 
@@ -235,8 +246,7 @@ public class PulseAudioClip extends Puls
 			try {
 				clipThread.join();
 			} catch (InterruptedException e) {
-				// TODO Auto-generated catch block
-				e.printStackTrace();
+				// ignore
 			}
 		}
 
@@ -280,8 +290,9 @@ public class PulseAudioClip extends Puls
 		if (!isOpen) {
 			throw new IllegalStateException("Line not open");
 		}
-
-		return (int) framesSinceOpen;
+		synchronized (clipLock) {
+			return (int) framesSinceOpen;
+		}
 	}
 
 	@Override
@@ -324,7 +335,7 @@ public class PulseAudioClip extends Puls
 
 		System.out.println("Loop " + count + " called");
 
-		if (clipThreadStarted && count != 0) {
+		if (clipThread.isAlive() && count != 0) {
 			// Do nothing; behavior not specified by the Java API
 			return;
 		}
@@ -358,7 +369,9 @@ public class PulseAudioClip extends Puls
 		this.data = new byte[bufferSize];
 		System.arraycopy(data, offset, this.data, 0, bufferSize);
 		frameCount = bufferSize / format.getFrameSize();
-
+		currentFrame = 0;
+		framesSinceOpen = 0;
+		endFrame = frameCount - 1;
 		PulseAudioVolumeControl volumeControl = new PulseAudioVolumeControl(
 				this, eventLoop);
 		PulseAudioMuteControl muteControl = new PulseAudioMuteControl(this,
@@ -373,6 +386,20 @@ public class PulseAudioClip extends Puls
 
 	}
 
+	// FIXME This is not exposed by the Clip interface, but becomes available if
+	// using PulseAudioClip
+	// @Override
+	// public void open(AudioFormat format, int bufferSize)
+	// throws LineUnavailableException {
+	// throw new IllegalArgumentException("open(AudioFormat, int) on a Clip is
+	// not allowed");
+	// }
+
+	@Override
+	// public void open(AudioFormat format) throws LineUnavailableException {
+	// throw new IllegalArgumentException("open(AudioFormat) on a Clip is not
+	// allowed");
+	// }
 	public byte[] native_setVolume(float value) {
 		return stream.native_setVolume(value);
 	}
@@ -428,7 +455,7 @@ public class PulseAudioClip extends Puls
 		}
 
 		if (end == -1) {
-			end = frameCount;
+			end = frameCount - 1;
 		}
 
 		if (end < start) {
@@ -496,7 +523,6 @@ public class PulseAudioClip extends Puls
 			e.printStackTrace();
 		}
 		synchronized (clipLock) {
-			currentFrame = 0;
 			loopsLeft = 0;
 		}
 
diff -r 381f212496eb -r c46f6e0e7959 src/java/org/classpath/icedtea/pulseaudio/PulseAudioDataLine.java
--- a/src/java/org/classpath/icedtea/pulseaudio/PulseAudioDataLine.java	Tue Sep 23 11:38:18 2008 -0400
+++ b/src/java/org/classpath/icedtea/pulseaudio/PulseAudioDataLine.java	Tue Sep 23 15:02:55 2008 -0400
@@ -160,7 +160,11 @@ public abstract class PulseAudioDataLine
 
 		} catch (LineUnavailableException e) {
 			// error connecting to the server!
-			// FIXME clean up
+			stream.removePlaybackStartedListener(startedListener);
+			stream.removeUnderflowListener(stoppedListener);
+			stream.removeStateListener(openCloseListener);
+			stream.free();
+			stream = null;
 			throw e;
 		}
 		this.bufferSize = bufferSize;
@@ -169,6 +173,7 @@ public abstract class PulseAudioDataLine
 			synchronized (eventLoop.threadLock) {
 				if (stream.getState() != Stream.State.READY) {
 					stream.disconnect();
+					stream.free();
 					throw new LineUnavailableException(
 							"unable to obtain a line");
 				}
@@ -230,9 +235,9 @@ public abstract class PulseAudioDataLine
 		isStarted = false;
 	}
 
-	// A BIG FIXME !
-
 	/*
+	 * TODO
+	 * 
 	 * http://bugs.sun.com/bugdatabase/view_bug.do?bug_id=4791152 :
 	 * 
 	 * a line is active in between calls to start() and stop(). In that sense,
@@ -242,6 +247,8 @@ public abstract class PulseAudioDataLine
 	 * running. This also means that a line should become not running on buffer
 	 * underrun/overflow.
 	 * 
+	 * 
+	 * HOWEVER, the javadocs say the opposite thing!
 	 * 
 	 */
 
diff -r 381f212496eb -r c46f6e0e7959 src/java/org/classpath/icedtea/pulseaudio/PulseAudioSourceDataLine.java
--- a/src/java/org/classpath/icedtea/pulseaudio/PulseAudioSourceDataLine.java	Tue Sep 23 11:38:18 2008 -0400
+++ b/src/java/org/classpath/icedtea/pulseaudio/PulseAudioSourceDataLine.java	Tue Sep 23 15:02:55 2008 -0400
@@ -151,6 +151,8 @@ public class PulseAudioSourceDataLine ex
 		};
 		stream.addWriteListener(writeNotifier);
 
+		boolean interrupted = false;
+
 		while (remainingLength != 0) {
 
 			synchronized (eventLoop.threadLock) {
@@ -164,8 +166,8 @@ public class PulseAudioSourceDataLine ex
 						try {
 							eventLoop.threadLock.wait(100);
 						} catch (InterruptedException e) {
-							// FIXME
-							assert (false);
+							// ignore for now
+							interrupted = true;
 						}
 
 					}
@@ -195,7 +197,9 @@ public class PulseAudioSourceDataLine ex
 		 */
 
 		stream.removeWriteListener(writeNotifier);
-
+		if (interrupted) {
+			Thread.currentThread().interrupt();
+		}
 		return sizeWritten;
 	}
 
diff -r 381f212496eb -r c46f6e0e7959 unittests/org/classpath/icedtea/pulseaudio/PulseAudioClipTest.java
--- a/unittests/org/classpath/icedtea/pulseaudio/PulseAudioClipTest.java	Tue Sep 23 11:38:18 2008 -0400
+++ b/unittests/org/classpath/icedtea/pulseaudio/PulseAudioClipTest.java	Tue Sep 23 15:02:55 2008 -0400
@@ -168,6 +168,8 @@ public class PulseAudioClipTest {
 	@Test
 	public void testOpenEvent() throws LineUnavailableException,
 			UnsupportedAudioFileException, IOException {
+		System.out
+				.println("This tests the OPEN event. You should see an OPEN on the next line");
 		Clip clip = (Clip) mixer.getLine(new Line.Info(Clip.class));
 		File soundFile = new File("testsounds/startup.wav");
 		AudioInputStream audioInputStream = AudioSystem
@@ -327,8 +329,8 @@ public class PulseAudioClipTest {
 	}
 
 	@Test
-	public void testLoop0InterruptsPlayback() throws LineUnavailableException, IOException,
-			UnsupportedAudioFileException {
+	public void testLoop0InterruptsPlayback() throws LineUnavailableException,
+			IOException, UnsupportedAudioFileException {
 		Clip clip = (Clip) mixer.getLine(new Line.Info(Clip.class));
 		File soundFile = new File("testsounds/startup.wav");
 		AudioInputStream audioInputStream = AudioSystem
@@ -410,7 +412,7 @@ public class PulseAudioClipTest {
 	}
 
 	@Test
-	public void testFramePosition() throws LineUnavailableException,
+	public void testFramePositionBeforeClose() throws LineUnavailableException,
 			UnsupportedAudioFileException, IOException {
 		System.out
 				.println("This tests if the Clip provides the correct frame position");
@@ -430,7 +432,7 @@ public class PulseAudioClipTest {
 		clip.close();
 
 		long expected = 136703;
-		long granularity = 100;
+		long granularity = 5;
 		System.out.println("Frames in " + fileName + ": " + expected);
 		System.out.println("Frame position in clip :" + pos);
 		Assert.assertTrue("Expected: " + expected + " got " + pos, Math
@@ -439,9 +441,10 @@ public class PulseAudioClipTest {
 	}
 
 	@Test
-	public void testFramePositionAfterLooping()
+	public void testFramePositionWithStartStop()
 			throws LineUnavailableException, UnsupportedAudioFileException,
-			IOException {
+			IOException, InterruptedException {
+
 		System.out
 				.println("This tests if the Clip provides the correct frame position");
 		Clip clip = (Clip) mixer.getLine(new Line.Info(Clip.class));
@@ -451,7 +454,12 @@ public class PulseAudioClipTest {
 				.getAudioInputStream(soundFile1);
 		clip.open(audioInputStream1);
 
-		clip.loop(1);
+		clip.start();
+
+		Thread.sleep(500);
+		clip.stop();
+		Thread.sleep(5000);
+		clip.start();
 
 		clip.drain();
 
@@ -459,8 +467,8 @@ public class PulseAudioClipTest {
 
 		clip.close();
 
-		long expected = 136703 * 2;
-		long granularity = 100;
+		long expected = 136703;
+		long granularity = 5;
 		System.out.println("Frames in " + fileName + ": " + expected);
 		System.out.println("Frame position in clip :" + pos);
 		Assert.assertTrue("Expected: " + expected + " got " + pos, Math
@@ -469,6 +477,83 @@ public class PulseAudioClipTest {
 	}
 
 	@Test
+	public void testFramePositionWithLoop() throws LineUnavailableException,
+			UnsupportedAudioFileException, IOException, InterruptedException {
+		System.out.println("This tests if the Clip provides the correct frame "
+				+ "position with a bit of looping in the clip");
+		Clip clip = (Clip) mixer.getLine(new Line.Info(Clip.class));
+		String fileName = "testsounds/logout.wav";
+		File soundFile1 = new File(fileName);
+		AudioInputStream audioInputStream1 = AudioSystem
+				.getAudioInputStream(soundFile1);
+		clip.open(audioInputStream1);
+
+		clip.setLoopPoints(0, -1);
+		clip.loop(1);
+		Thread.sleep(500);
+		clip.stop();
+		Thread.sleep(2000);
+		clip.start();
+		Thread.sleep(100);
+		clip.stop();
+		Thread.sleep(4000);
+		clip.start();
+		Thread.sleep(100);
+		clip.stop();
+		Thread.sleep(2000);
+		clip.start();
+		Thread.sleep(100);
+		clip.stop();
+		Thread.sleep(3000);
+		clip.start();
+
+		clip.drain();
+
+		long pos = clip.getFramePosition();
+
+		clip.close();
+
+		long expected = 136703 * 1;
+		long granularity = 5;
+		System.out.println("Frames in " + fileName + ": " + expected);
+		System.out.println("Frame position in clip :" + pos);
+		Assert.assertTrue("Expected: " + expected + " got " + pos, Math
+				.abs(expected - pos) < granularity);
+
+	}
+
+	@Test
+	public void testFramePositionAfterLooping()
+			throws LineUnavailableException, UnsupportedAudioFileException,
+			IOException {
+		System.out
+				.println("This tests if the Clip provides the correct frame position");
+		Clip clip = (Clip) mixer.getLine(new Line.Info(Clip.class));
+		String fileName = "testsounds/logout.wav";
+		File soundFile1 = new File(fileName);
+		AudioInputStream audioInputStream1 = AudioSystem
+				.getAudioInputStream(soundFile1);
+		clip.open(audioInputStream1);
+
+		clip.setLoopPoints(0, -1);
+		clip.loop(1);
+
+		clip.drain();
+
+		long pos = clip.getFramePosition();
+
+		clip.close();
+
+		long expected = 136703 * 2;
+		long granularity = 5;
+		System.out.println("Frames in " + fileName + ": " + expected);
+		System.out.println("Frame position in clip :" + pos);
+		Assert.assertTrue("Expected: " + expected + " got " + pos, Math
+				.abs(expected - pos) < granularity);
+
+	}
+
+	@Test
 	public void testMixerKnowsAboutOpenClips() throws LineUnavailableException,
 			UnsupportedAudioFileException, IOException {
 		Clip clip = (Clip) mixer.getLine(new Line.Info(Clip.class));
diff -r 381f212496eb -r c46f6e0e7959 unittests/org/classpath/icedtea/pulseaudio/PulseAudioSourceDataLineRawTest.java
--- a/unittests/org/classpath/icedtea/pulseaudio/PulseAudioSourceDataLineRawTest.java	Tue Sep 23 11:38:18 2008 -0400
+++ b/unittests/org/classpath/icedtea/pulseaudio/PulseAudioSourceDataLineRawTest.java	Tue Sep 23 15:02:55 2008 -0400
@@ -113,7 +113,6 @@ public class PulseAudioSourceDataLineRaw
 			} catch (LineUnavailableException e) {
 				e.printStackTrace();
 			} catch (IOException e) {
-				// TODO Auto-generated catch block
 				e.printStackTrace();
 			}
 		}
diff -r 381f212496eb -r c46f6e0e7959 unittests/org/classpath/icedtea/pulseaudio/PulseAudioSourceDataLineTest.java
--- a/unittests/org/classpath/icedtea/pulseaudio/PulseAudioSourceDataLineTest.java	Tue Sep 23 11:38:18 2008 -0400
+++ b/unittests/org/classpath/icedtea/pulseaudio/PulseAudioSourceDataLineTest.java	Tue Sep 23 15:02:55 2008 -0400
@@ -119,7 +119,6 @@ public class PulseAudioSourceDataLineTes
 			} catch (LineUnavailableException e) {
 				e.printStackTrace();
 			} catch (IOException e) {
-				// TODO Auto-generated catch block
 				e.printStackTrace();
 			}
 		}
@@ -152,7 +151,7 @@ public class PulseAudioSourceDataLineTes
 		stopped = 0;
 		opened = 0;
 		closed = 0;
-
+		listenerCalled = 0;
 	}
 
 	@Test



More information about the distro-pkg-dev mailing list