changeset in /hg/pulseaudio: 2008-09-29 Omair Majid <omajid at redh...
Omair Majid
omajid at redhat.com
Mon Sep 29 12:57:41 PDT 2008
changeset b185dd9217bb in /hg/pulseaudio
details: http://icedtea.classpath.org/hg/pulseaudio?cmd=changeset;node=b185dd9217bb
description:
2008-09-29 Omair Majid <omajid at redhat.com>
* src/java/org/classpath/icedtea/pulseaudio/PulseAudioTargetDataLine.java
New class variables flushed and drained.
(close): Now synchronized.
(open): Likewise.
(open): Likewise.
(read): Likewise. Return when flush()ed, drain()ed, stop()ed or close()d.
(drain): Synchronized. Sets drained to true.
(flush): Sets flushed to true.
(start): Synchronized.
(stop): Synchronized.
* unittests/org/classpath/icedtea/pulseaudio/PulseAudioTargetDataLineTest.java
(testReadAndDrain): Fixed test and made it active.
diffstat:
2 files changed, 127 insertions(+), 66 deletions(-)
src/java/org/classpath/icedtea/pulseaudio/PulseAudioTargetDataLine.java | 173 ++++++----
unittests/org/classpath/icedtea/pulseaudio/PulseAudioTargetDataLineTest.java | 20 -
diffs (290 lines):
diff -r 350bc72ba495 -r b185dd9217bb src/java/org/classpath/icedtea/pulseaudio/PulseAudioTargetDataLine.java
--- a/src/java/org/classpath/icedtea/pulseaudio/PulseAudioTargetDataLine.java Mon Sep 29 15:16:15 2008 -0400
+++ b/src/java/org/classpath/icedtea/pulseaudio/PulseAudioTargetDataLine.java Mon Sep 29 15:56:36 2008 -0400
@@ -52,6 +52,13 @@ public class PulseAudioTargetDataLine ex
*/
byte[] fragmentBuffer;
+ /*
+ * these are set to true only by the respective functions (flush(), drain())
+ * set to false only by read()
+ */
+ boolean flushed = false;
+ boolean drained = false;
+
public PulseAudioTargetDataLine(EventLoop eventLoop, AudioFormat[] formats,
AudioFormat defaultFormat) {
supportedFormats = formats;
@@ -62,7 +69,7 @@ public class PulseAudioTargetDataLine ex
}
@Override
- public void close() {
+ synchronized public void close() {
if (!isOpen) {
throw new IllegalStateException(
"Line cant be closed if it isnt open");
@@ -75,7 +82,7 @@ public class PulseAudioTargetDataLine ex
}
@Override
- public void open(AudioFormat format, int bufferSize)
+ synchronized public void open(AudioFormat format, int bufferSize)
throws LineUnavailableException {
if (isOpen) {
throw new IllegalStateException("already open");
@@ -90,7 +97,8 @@ public class PulseAudioTargetDataLine ex
}
@Override
- public void open(AudioFormat format) throws LineUnavailableException {
+ synchronized public void open(AudioFormat format)
+ throws LineUnavailableException {
open(format, DEFAULT_BUFFER_SIZE);
}
@@ -140,72 +148,98 @@ public class PulseAudioTargetDataLine ex
/* on first read() of the line, fragmentBuffer is null */
if (fragmentBuffer != null) {
-
- boolean fragmentBufferSmaller = fragmentBuffer.length < length;
- int smallerBufferLength = Math.min(fragmentBuffer.length, length);
- System.arraycopy(fragmentBuffer, 0, data, position,
- smallerBufferLength);
-
- if (!fragmentBufferSmaller) {
- /*
- * if fragment was larger, then we already have all the data we
- * need. clean up the buffer before returning. Make a new
- * fragmentBuffer from the remaining bytes
- */
- int remainingBytesInFragment = (fragmentBuffer.length - length);
- byte[] newFragmentBuffer = new byte[remainingBytesInFragment];
- System.arraycopy(fragmentBuffer, length, newFragmentBuffer, 0,
- newFragmentBuffer.length);
- fragmentBuffer = newFragmentBuffer;
- return length;
- }
-
- /* done with fragment buffer, remove it */
- bytesRead = smallerBufferLength;
- sizeRead += bytesRead;
- position += bytesRead;
- remainingLength -= bytesRead;
- fragmentBuffer = null;
+ synchronized (this) {
+
+ boolean fragmentBufferSmaller = fragmentBuffer.length < length;
+ int smallerBufferLength = Math.min(fragmentBuffer.length,
+ length);
+ System.arraycopy(fragmentBuffer, 0, data, position,
+ smallerBufferLength);
+ framesSinceOpen += smallerBufferLength
+ / currentFormat.getFrameSize();
+
+ if (!fragmentBufferSmaller) {
+ /*
+ * if fragment was larger, then we already have all the data
+ * we need. clean up the buffer before returning. Make a new
+ * fragmentBuffer from the remaining bytes
+ */
+ int remainingBytesInFragment = (fragmentBuffer.length - length);
+ byte[] newFragmentBuffer = new byte[remainingBytesInFragment];
+ System.arraycopy(fragmentBuffer, length, newFragmentBuffer,
+ 0, newFragmentBuffer.length);
+ fragmentBuffer = newFragmentBuffer;
+ return length;
+ }
+
+ /* done with fragment buffer, remove it */
+ bytesRead = smallerBufferLength;
+ sizeRead += bytesRead;
+ position += bytesRead;
+ remainingLength -= bytesRead;
+ fragmentBuffer = null;
+
+ }
}
/*
* if we need to read more data, then we read from PulseAudio's buffer
*/
while (remainingLength != 0) {
- byte[] currentFragment;
- synchronized (eventLoop.threadLock) {
-
- /* read a fragment, and drop it from the server */
- currentFragment = stream.peek();
- stream.drop();
- if (currentFragment == null) {
- System.out
- .println("DEBUG: PulseAudioTargetDataLine:read(): error in stream.peek()");
+ synchronized (this) {
+
+ if (!isOpen || !isStarted) {
return sizeRead;
}
- bytesRead = Math.min(currentFragment.length, remainingLength);
-
- /*
- * we read more than we required, save the rest of the data in
- * the fragmentBuffer
- */
- if (bytesRead < currentFragment.length) {
- /* allocate a buffer to store unsaved data */
- fragmentBuffer = new byte[currentFragment.length
- - bytesRead];
- /* copy over the unsaved data */
- System.arraycopy(currentFragment, bytesRead,
- fragmentBuffer, 0, fragmentBuffer.length
- - bytesRead);
- }
-
- System.arraycopy(currentFragment, 0, data, position, bytesRead);
-
- sizeRead += bytesRead;
- position += bytesRead;
- remainingLength -= bytesRead;
- framesSinceOpen += bytesRead / currentFormat.getFrameSize();
+ if (flushed) {
+ flushed = false;
+ return sizeRead;
+ }
+
+ if (drained) {
+ drained = false;
+ return sizeRead;
+ }
+
+ byte[] currentFragment;
+ synchronized (eventLoop.threadLock) {
+
+ /* read a fragment, and drop it from the server */
+ currentFragment = stream.peek();
+ stream.drop();
+ if (currentFragment == null) {
+ System.out
+ .println("DEBUG: PulseAudioTargetDataLine:read(): error in stream.peek()");
+ return sizeRead;
+ }
+
+ bytesRead = Math.min(currentFragment.length,
+ remainingLength);
+
+ /*
+ * we read more than we required, save the rest of the data
+ * in the fragmentBuffer
+ */
+ if (bytesRead < currentFragment.length) {
+ /* allocate a buffer to store unsaved data */
+ fragmentBuffer = new byte[currentFragment.length
+ - bytesRead];
+
+ /* copy over the unsaved data */
+ System.arraycopy(currentFragment, bytesRead,
+ fragmentBuffer, 0, currentFragment.length
+ - bytesRead);
+ }
+
+ System.arraycopy(currentFragment, 0, data, position,
+ bytesRead);
+
+ sizeRead += bytesRead;
+ position += bytesRead;
+ remainingLength -= bytesRead;
+ framesSinceOpen += bytesRead / currentFormat.getFrameSize();
+ }
}
}
@@ -227,9 +261,18 @@ public class PulseAudioTargetDataLine ex
throw new IllegalStateException("must call open() before drain()");
}
+ synchronized (this) {
+ drained = true;
+ }
+
// blocks when there is data on the line
// http://www.jsresources.org/faq_audio.html#stop_drain_tdl
- while (isStarted) {
+ while (true) {
+ synchronized (this) {
+ if (!isStarted || !isOpen) {
+ break;
+ }
+ }
try {
Thread.sleep(100);
} catch (InterruptedException e) {
@@ -250,6 +293,10 @@ public class PulseAudioTargetDataLine ex
}
operation.waitForCompletion();
operation.releaseReference();
+
+ synchronized (this) {
+ flushed = true;
+ }
}
@@ -282,14 +329,14 @@ public class PulseAudioTargetDataLine ex
*/
@Override
- public void start() {
+ synchronized public void start() {
super.start();
fireLineEvent(new LineEvent(this, LineEvent.Type.START, framesSinceOpen));
}
@Override
- public void stop() {
+ synchronized public void stop() {
super.stop();
fireLineEvent(new LineEvent(this, LineEvent.Type.STOP, framesSinceOpen));
diff -r 350bc72ba495 -r b185dd9217bb unittests/org/classpath/icedtea/pulseaudio/PulseAudioTargetDataLineTest.java
--- a/unittests/org/classpath/icedtea/pulseaudio/PulseAudioTargetDataLineTest.java Mon Sep 29 15:16:15 2008 -0400
+++ b/unittests/org/classpath/icedtea/pulseaudio/PulseAudioTargetDataLineTest.java Mon Sep 29 15:56:36 2008 -0400
@@ -299,7 +299,6 @@ public class PulseAudioTargetDataLineTes
// this is kind of messed up
// drain should hang on a started data line
// but read should return
- @Ignore
@Test
public void testReadAndDrain() throws LineUnavailableException,
InterruptedException {
@@ -320,12 +319,27 @@ public class PulseAudioTargetDataLineTes
Assert.assertTrue(reader.isAlive());
- targetDataLine.drain();
+ Thread drainer = new Thread() {
+
+ @Override
+ public void run() {
+ targetDataLine.drain();
+
+ }
+
+ };
+
+ drainer.start();
Thread.sleep(100);
Assert.assertFalse(reader.isAlive());
- targetDataLine.stop();
+
+ targetDataLine.stop();
+
+ Thread.sleep(100);
+ Assert.assertFalse(drainer.isAlive());
+
targetDataLine.close();
}
More information about the distro-pkg-dev
mailing list