changeset in /hg/pulseaudio: 2008-08-11 Omair Majid <omajid at redh...

Omair Majid omajid at redhat.com
Tue Aug 12 12:17:20 PDT 2008


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

	    * src/java/org/classpath/icedtea/pulseaudio/PulseAudioSourceDataLine.java: now notifies any linelisteners of OPEN event. write() now tests the arguments passed to it
	    * src/native/org_classpath_icedtea_pulseaudio/PulseAudioSourceDataLine.c: enabled a little debug output
	    * unittests/org/classpath/icedtea/pulseaudio/PulseSourceDataLineTest.java: added tests to confirm write's argument checking. also added tests to check that listeners are notified of OPEN and CLOSE LineEvents

diffstat:

3 files changed, 159 insertions(+), 17 deletions(-)
src/java/org/classpath/icedtea/pulseaudio/PulseAudioSourceDataLine.java |   63 ++++-
src/native/org_classpath_icedtea_pulseaudio_PulseAudioSourceDataLine.c  |    2 
unittests/org/classpath/icedtea/pulseaudio/PulseSourceDataLineTest.java |  111 ++++++++++

diffs (299 lines):

diff -r 8f4e01b67c92 -r 856e11044f24 src/java/org/classpath/icedtea/pulseaudio/PulseAudioSourceDataLine.java
--- a/src/java/org/classpath/icedtea/pulseaudio/PulseAudioSourceDataLine.java	Tue Aug 12 11:28:22 2008 -0400
+++ b/src/java/org/classpath/icedtea/pulseaudio/PulseAudioSourceDataLine.java	Tue Aug 12 15:17:14 2008 -0400
@@ -73,7 +73,7 @@ public class PulseAudioSourceDataLine im
 	private AudioFormat currentFormat = null;
 	private AudioFormat defaultFormat = null;
 
-	private List<LineListener> listeners;
+	private List<LineListener> lineListeners;
 
 	private Control[] controls = null;
 	private PulseAudioStreamMuteControl muteControl;
@@ -123,7 +123,7 @@ public class PulseAudioSourceDataLine im
 
 	public PulseAudioSourceDataLine(EventLoop eventLoop) {
 		this.eventLoop = eventLoop;
-		this.listeners = new ArrayList<LineListener>();
+		this.lineListeners = new ArrayList<LineListener>();
 		this.volume = PulseAudioVolumeControl.MAX_VOLUME;
 
 		/*
@@ -349,6 +349,24 @@ public class PulseAudioSourceDataLine im
 			throw new IllegalArgumentException("Invalid format");
 		}
 
+		StreamListener openCloseListener = new StreamListener() {
+
+			@Override
+			public void update(StreamEvent e) {
+				if (e.getType() == StreamEvent.Type.READY) {
+					fireLineEvent(new LineEvent(PulseAudioSourceDataLine.this,
+							LineEvent.Type.OPEN, AudioSystem.NOT_SPECIFIED));
+				} else if (e.getType() == StreamEvent.Type.TERMINATED
+						|| e.getType() == StreamEvent.Type.FAILED) {
+					fireLineEvent((new LineEvent(PulseAudioSourceDataLine.this,
+							LineEvent.Type.CLOSE, AudioSystem.NOT_SPECIFIED)));
+				}
+			}
+
+		};
+		
+		addStreamListener(openCloseListener);
+
 		final Semaphore semaphore = new Semaphore(0);
 
 		synchronized (eventLoop.threadLock) {
@@ -374,6 +392,7 @@ public class PulseAudioSourceDataLine im
 			// throw new LineUnavailableException("unable to prepare
 			// stream");
 		}
+
 		System.out.println(this.getClass().getName() + "stream is ready");
 
 		controls = new Control[2];
@@ -402,7 +421,18 @@ public class PulseAudioSourceDataLine im
 	@Override
 	public int write(byte[] data, int offset, int length) {
 
-		// TODO add check that the data is an integral number of frames
+		int frameSize = currentFormat.getFrameSize();
+		if (length % frameSize != 0) {
+			throw new IllegalArgumentException(
+					"amount of data to write does not represent an integral number of frames");
+		}
+		if (length < 0) {
+			throw new IllegalArgumentException("length is negative");
+		}
+
+		if (length + offset > data.length) {
+			throw new ArrayIndexOutOfBoundsException(length + offset);
+		}
 
 		int position = offset;
 		int remainingLength = length;
@@ -462,11 +492,11 @@ public class PulseAudioSourceDataLine im
 	}
 
 	public void addLineListener(LineListener listener) {
-		this.listeners.add(listener);
+		this.lineListeners.add(listener);
 	}
 
 	public void removeLineListener(LineListener listener) {
-		this.listeners.remove(listener);
+		this.lineListeners.remove(listener);
 	}
 
 	private void addStreamListener(StreamListener listener) {
@@ -488,10 +518,6 @@ public class PulseAudioSourceDataLine im
 
 		synchronized (eventLoop.threadLock) {
 			native_close();
-		}
-
-		for (LineListener l : this.listeners) {
-			l.update(new LineEvent(this, LineEvent.Type.CLOSE, 0));
 		}
 
 	}
@@ -563,7 +589,7 @@ public class PulseAudioSourceDataLine im
 		} else {
 			return new Control[] {};
 		}
-		
+
 	}
 
 	public javax.sound.sampled.Line.Info getLineInfo() {
@@ -622,16 +648,16 @@ public class PulseAudioSourceDataLine im
 		synchronized (eventLoop.threadLock) {
 			switch (status) {
 			case 0:
-				fireEvent(new StreamEvent(StreamEvent.Type.UNCONNECTED));
+				fireStreamEvent(new StreamEvent(StreamEvent.Type.UNCONNECTED));
 				break;
 			case 1:
-				fireEvent(new StreamEvent(StreamEvent.Type.CREATING));
+				fireStreamEvent(new StreamEvent(StreamEvent.Type.CREATING));
 				break;
 			case 2:
-				fireEvent(new StreamEvent(StreamEvent.Type.READY));
+				fireStreamEvent(new StreamEvent(StreamEvent.Type.READY));
 				break;
 			case 3:
-				fireEvent(new StreamEvent(StreamEvent.Type.FAILED));
+				fireStreamEvent(new StreamEvent(StreamEvent.Type.FAILED));
 				break;
 			case 4:
 				break;
@@ -641,8 +667,13 @@ public class PulseAudioSourceDataLine im
 		}
 	}
 
-	private void fireEvent(StreamEvent e) {
-
+	private void fireLineEvent(LineEvent e) {
+		for (LineListener lineListener : lineListeners) {
+			lineListener.update(e);
+		}
+	}
+
+	private void fireStreamEvent(StreamEvent e) {
 		for (StreamListener streamListener : streamListeners) {
 			streamListener.update(e);
 		}
diff -r 8f4e01b67c92 -r 856e11044f24 src/native/org_classpath_icedtea_pulseaudio_PulseAudioSourceDataLine.c
--- a/src/native/org_classpath_icedtea_pulseaudio_PulseAudioSourceDataLine.c	Tue Aug 12 11:28:22 2008 -0400
+++ b/src/native/org_classpath_icedtea_pulseaudio_PulseAudioSourceDataLine.c	Tue Aug 12 15:17:14 2008 -0400
@@ -75,7 +75,7 @@ static void stream_state_change_callback
 
 	jobject obj = java_context->obj;
 
-	//	printf("stream state changed to %d\n", pa_stream_get_state(stream));
+	printf("----> Stream state changed to %d\n", pa_stream_get_state(stream));
 
 	/* Call the 'update' method in java
 	 * to handle all java-side events
diff -r 8f4e01b67c92 -r 856e11044f24 unittests/org/classpath/icedtea/pulseaudio/PulseSourceDataLineTest.java
--- a/unittests/org/classpath/icedtea/pulseaudio/PulseSourceDataLineTest.java	Tue Aug 12 11:28:22 2008 -0400
+++ b/unittests/org/classpath/icedtea/pulseaudio/PulseSourceDataLineTest.java	Tue Aug 12 15:17:14 2008 -0400
@@ -48,6 +48,8 @@ import javax.sound.sampled.DataLine;
 import javax.sound.sampled.DataLine;
 import javax.sound.sampled.FloatControl;
 import javax.sound.sampled.Line;
+import javax.sound.sampled.LineEvent;
+import javax.sound.sampled.LineListener;
 import javax.sound.sampled.LineUnavailableException;
 import javax.sound.sampled.Mixer;
 import javax.sound.sampled.SourceDataLine;
@@ -63,6 +65,8 @@ public class PulseSourceDataLineTest {
 public class PulseSourceDataLineTest {
 	Mixer mixer;
 
+	private int listenerCalled = 0;
+
 	public static junit.framework.Test suite() {
 		return new JUnit4TestAdapter(PulseSourceDataLineTest.class);
 	}
@@ -103,6 +107,51 @@ public class PulseSourceDataLineTest {
 		line.flush();
 		line.close();
 
+	}
+
+	@Test(expected = IllegalArgumentException.class)
+	public void testPlayLessThanFrameSize() throws LineUnavailableException,
+			UnsupportedAudioFileException, IOException {
+		File soundFile = new File("testsounds/startup.wav");
+		AudioInputStream audioInputStream = AudioSystem
+				.getAudioInputStream(soundFile);
+		AudioFormat audioFormat = audioInputStream.getFormat();
+		// the audio file must have an even number of channels
+		Assert.assertTrue(audioFormat.getChannels() % 2 == 0);
+		SourceDataLine line;
+		line = (SourceDataLine) mixer.getLine(new DataLine.Info(
+				SourceDataLine.class, audioFormat));
+
+		byte[] data = new byte[1];
+		data[0] = (byte) 'a';
+
+		line.open();
+		line.write(data, 0, 1);
+
+		line.close();
+
+	}
+
+	@Test
+	public void testOpenFormat() throws LineUnavailableException,
+			UnsupportedAudioFileException, IOException {
+		/*
+		 * This test makes sure that the default format of a line using open()
+		 * is the same format that was passed to the mixer's getLine() function
+		 * to get the line in the first place
+		 */
+
+		File soundFile = new File("testsounds/startup.wav");
+		AudioInputStream audioInputStream = AudioSystem
+				.getAudioInputStream(soundFile);
+		AudioFormat audioFormat = audioInputStream.getFormat();
+
+		SourceDataLine line;
+		line = (SourceDataLine) mixer.getLine(new DataLine.Info(
+				SourceDataLine.class, audioFormat));
+		Assert.assertNotNull(line);
+		line.open();
+		Assert.assertTrue(line.getFormat().matches(audioFormat));
 	}
 
 	@Test
@@ -199,6 +248,68 @@ public class PulseSourceDataLineTest {
 		sourceLine.close();
 	}
 
+	@Test
+	public void testOpenEvent() throws LineUnavailableException {
+
+		listenerCalled = 0;
+		LineListener openListener = new LineListener() {
+			public void update(LineEvent event) {
+				Assert.assertTrue(event.getType() == LineEvent.Type.OPEN);
+				PulseSourceDataLineTest.this.listenerCalled++;
+			}
+		};
+
+		SourceDataLine line = (SourceDataLine) mixer.getLine(new Line.Info(
+				SourceDataLine.class));
+		line.addLineListener(openListener);
+		line.open();
+		line.removeLineListener(openListener);
+		line.close();
+		Assert.assertEquals(1, listenerCalled);
+		listenerCalled = 0;
+	}
+
+	@Test
+	public void testCloseEvent() throws LineUnavailableException {
+		listenerCalled = 0;
+		LineListener closeListener = new LineListener() {
+			public void update(LineEvent event) {
+				Assert.assertTrue(event.getType() == LineEvent.Type.CLOSE);
+				PulseSourceDataLineTest.this.listenerCalled++;
+			}
+		};
+
+		SourceDataLine line = (SourceDataLine) mixer.getLine(new Line.Info(
+				SourceDataLine.class));
+		line.open();
+		line.addLineListener(closeListener);
+		line.close();
+		line.removeLineListener(closeListener);
+		Assert.assertEquals(1, listenerCalled);
+		listenerCalled = 0;
+	}
+
+	@Test
+	public void testCloseEventWrongListener() throws LineUnavailableException {
+		listenerCalled = 0;
+		LineListener closeListener = new LineListener() {
+			public void update(LineEvent event) {
+				PulseSourceDataLineTest.this.listenerCalled++;
+			}
+		};
+
+		SourceDataLine line = (SourceDataLine) mixer.getLine(new Line.Info(
+				SourceDataLine.class));
+
+		line.open();
+		line.addLineListener(closeListener);
+		line.removeLineListener(closeListener);
+		line.close();
+		Assert.assertEquals(0, listenerCalled);
+		listenerCalled = 0;
+
+	}
+
 	@After
 	public void tearDown() throws Exception {
 



More information about the distro-pkg-dev mailing list