changeset in /hg/pulseaudio: 2008-10-08 Ioana Ivan <iivan at redhat...
Ioana Ivan
iivan at redhat.com
Wed Oct 8 11:35:49 PDT 2008
changeset 4260a476a101 in /hg/pulseaudio
details: http://icedtea.classpath.org/hg/pulseaudio?cmd=changeset;node=4260a476a101
description:
2008-10-08 Ioana Ivan <iivan at redhat.com>
* src/java/org/classpath/icedtea/pulseaudio/PulseAudioDataLine.java
(addStreamListener): startedListener always fires a START event
strtedListener notifies drain that there is data
on the line
(stop): sets writeInterrupted to true
(start): doesn't send any events
(getBytesInBuffer): new function, returns the number of bytes
currently present in a stream's buffer
* src/java/org/classpath/icedtea/pulseaudio/PulseAudioSourceDataLine.java:
(write): -checks writeInterrupted instead of drain and flush
-if stop() was called before write() it writes data to the
buffer until the buffer fills, then blocks
-moved some code from the synchronized(this) block, since it
was causing one test to hang
(close): sets writeInterrupted to true
(drain): sets writeInterrupted to true
if the line is stopped and there is no data on the line,
returns immediately, if there is data, blocks until the line
is started
* src/java/org/classpath/icedtea/pulseaudio/Stream.java
(native_pa_stream_updateTimingInfo): new function
( bytesInBuffer): new function
* src/native/org_classpath_icedtea_pulseaudio_Stream.c
(JNICALL Java_org_classpath_icedtea_pulseaudio_Stream_bytesInBuffer):
new function
(Java_org_classpath_icedtea_pulseaudio_Stream_native_1pa_1stream_1updateTimingInfo):
new function
(update_timing_info_callback): new function
(Java_org_classpath_icedtea_pulseaudio_Stream_native_1pa_1stream_1writable_1size):
return 0 if the stream has been closed
diffstat:
12 files changed, 254 insertions(+), 102 deletions(-)
ChangeLog | 36 ++
configure.ac | 6
src/java/org/classpath/icedtea/pulseaudio/EventLoop.java | 4
src/java/org/classpath/icedtea/pulseaudio/Operation.java | 1
src/java/org/classpath/icedtea/pulseaudio/PulseAudioDataLine.java | 41 +--
src/java/org/classpath/icedtea/pulseaudio/PulseAudioMixer.java | 1
src/java/org/classpath/icedtea/pulseaudio/PulseAudioSourceDataLine.java | 136 +++++-----
src/java/org/classpath/icedtea/pulseaudio/PulseAudioSourcePort.java | 2
src/java/org/classpath/icedtea/pulseaudio/PulseAudioTargetPort.java | 2
src/java/org/classpath/icedtea/pulseaudio/Stream.java | 16 +
src/native/org_classpath_icedtea_pulseaudio_Stream.c | 44 +++
unittests/org/classpath/icedtea/pulseaudio/PulseAudioSourceDataLineTest.java | 67 +++-
diffs (truncated from 695 to 500 lines):
diff -r a3a8e9e19967 -r 4260a476a101 ChangeLog
--- a/ChangeLog Wed Oct 08 10:38:29 2008 -0400
+++ b/ChangeLog Wed Oct 08 14:27:39 2008 -0400
@@ -1,3 +1,39 @@ 2008-09-25 Ioana Ivan <iivan at redhat.com>
+2008-10-08 Ioana Ivan <iivan at redhat.com>
+ * src/java/org/classpath/icedtea/pulseaudio/PulseAudioDataLine.java
+ (addStreamListener): startedListener always fires a START event
+ strtedListener notifies drain that there is data
+ on the line
+ (stop): sets writeInterrupted to true
+ (start): doesn't send any events
+ (getBytesInBuffer): new function, returns the number of bytes
+ currently present in a stream's buffer
+
+ * src/java/org/classpath/icedtea/pulseaudio/PulseAudioSourceDataLine.java:
+ (write): -checks writeInterrupted instead of drain and flush
+ -if stop() was called before write() it writes data to the
+ buffer until the buffer fills, then blocks
+ -moved some code from the synchronized(this) block, since it
+ was causing one test to hang
+ (close): sets writeInterrupted to true
+ (drain): sets writeInterrupted to true
+ if the line is stopped and there is no data on the line,
+ returns immediately, if there is data, blocks until the line
+ is started
+
+ * src/java/org/classpath/icedtea/pulseaudio/Stream.java
+ (native_pa_stream_updateTimingInfo): new function
+ ( bytesInBuffer): new function
+
+ * src/native/org_classpath_icedtea_pulseaudio_Stream.c
+ (JNICALL Java_org_classpath_icedtea_pulseaudio_Stream_bytesInBuffer):
+ new function
+ (Java_org_classpath_icedtea_pulseaudio_Stream_native_1pa_1stream_1updateTimingInfo):
+ new function
+ (update_timing_info_callback): new function
+ (Java_org_classpath_icedtea_pulseaudio_Stream_native_1pa_1stream_1writable_1size):
+ return 0 if the stream has been closed
+
+
2008-09-25 Ioana Ivan <iivan at redhat.com>
* src/java/org/classpath/icedtea/pulseaudio/PulseAudioDataLine.java:
removed all references to the boolean variable corked since it was
diff -r a3a8e9e19967 -r 4260a476a101 configure.ac
--- a/configure.ac Wed Oct 08 10:38:29 2008 -0400
+++ b/configure.ac Wed Oct 08 14:27:39 2008 -0400
@@ -4,12 +4,12 @@ AC_PROG_LIBTOOL
AC_PROG_LIBTOOL
dnl Check for pulseaudio libraries.
-PKG_CHECK_MODULES(LIBPULSE,[libpulse >= 0.9.11],[LIBPULSE_FOUND=yes]
+PKG_CHECK_MODULES(LIBPULSE,libpulse,[LIBPULSE_FOUND=yes]
,[LIBPULSE_FOUND=no])
if test "x${LIBPULSE_FOUND}" = xno
then
- AC_MSG_ERROR([Could not find pulseaudio>=0.9.11 libraries - \
- Try installing pulseaudio-libs-devel>=0.9.11.])
+ AC_MSG_ERROR([Could not find pulseaudio libraries - \
+ Try installing pulseaudio-libs-devel.])
fi
AC_SUBST(LIBPULSE_CFLAGS)
AC_SUBST(LIBPULSE_LIBS)
diff -r a3a8e9e19967 -r 4260a476a101 src/java/org/classpath/icedtea/pulseaudio/EventLoop.java
--- a/src/java/org/classpath/icedtea/pulseaudio/EventLoop.java Wed Oct 08 10:38:29 2008 -0400
+++ b/src/java/org/classpath/icedtea/pulseaudio/EventLoop.java Wed Oct 08 14:27:39 2008 -0400
@@ -37,6 +37,7 @@ exception statement from your version.
package org.classpath.icedtea.pulseaudio;
+import java.io.IOException;
import java.util.ArrayList;
import java.util.List;
import java.util.concurrent.Semaphore;
@@ -105,7 +106,7 @@ public class EventLoop implements Runnab
*/
static {
- System.loadLibrary("pulse-java");
+ System.loadLibrary("pulse-java");
}
private EventLoop() {
@@ -197,6 +198,7 @@ public class EventLoop implements Runnab
break;
case 5:
fireEvent(new ContextEvent(Type.FAILED));
+ System.out.println("context failed");
break;
case 6:
fireEvent(new ContextEvent(Type.TERMINATED));
diff -r a3a8e9e19967 -r 4260a476a101 src/java/org/classpath/icedtea/pulseaudio/Operation.java
--- a/src/java/org/classpath/icedtea/pulseaudio/Operation.java Wed Oct 08 10:38:29 2008 -0400
+++ b/src/java/org/classpath/icedtea/pulseaudio/Operation.java Wed Oct 08 14:27:39 2008 -0400
@@ -37,6 +37,7 @@ exception statement from your version.
package org.classpath.icedtea.pulseaudio;
+import java.io.IOException;
/*
* Encapsulates a pa_operation object
diff -r a3a8e9e19967 -r 4260a476a101 src/java/org/classpath/icedtea/pulseaudio/PulseAudioDataLine.java
--- a/src/java/org/classpath/icedtea/pulseaudio/PulseAudioDataLine.java Wed Oct 08 10:38:29 2008 -0400
+++ b/src/java/org/classpath/icedtea/pulseaudio/PulseAudioDataLine.java Wed Oct 08 14:27:39 2008 -0400
@@ -76,6 +76,7 @@ public abstract class PulseAudioDataLine
protected EventLoop eventLoop = null;
protected Semaphore semaphore = new Semaphore(0);
protected Stream stream;
+ boolean writeInterrupted = false;
protected void open(AudioFormat format, int bufferSize)
throws LineUnavailableException {
@@ -149,6 +150,9 @@ public abstract class PulseAudioDataLine
PulseAudioDataLine.this,
LineEvent.Type.CLOSE, framesSinceOpen)));
}
+ synchronized (this) {
+ this.notifyAll();
+ }
semaphore.release();
}
@@ -174,15 +178,13 @@ public abstract class PulseAudioDataLine
Stream.PlaybackStartedListener startedListener = new Stream.PlaybackStartedListener() {
@Override
public void update() {
-
- // only send a START event in the beginning and following
- // an underflow
- if (!dataWritten) {
- fireLineEvent(new LineEvent(PulseAudioDataLine.this,
- LineEvent.Type.START, framesSinceOpen));
- }
+ fireLineEvent(new LineEvent(PulseAudioDataLine.this,
+ LineEvent.Type.START, framesSinceOpen));
dataWritten = true;
+ synchronized (this) {
+ this.notifyAll();
+ }
}
};
@@ -260,6 +262,7 @@ public abstract class PulseAudioDataLine
}
public void close() {
+
if (!isOpen) {
throw new IllegalStateException(
"Line must be open for close() to work");
@@ -280,10 +283,10 @@ public abstract class PulseAudioDataLine
}
super.close();
+
isStarted = false;
-
- }
-
+ }
+
public void reconnectforSynchronization(Stream masterStream)
throws LineUnavailableException {
sendEvents = false;
@@ -319,10 +322,6 @@ public abstract class PulseAudioDataLine
Operation op;
synchronized (eventLoop.threadLock) {
op = stream.unCork();
- if (dataWritten && (!isStarted)) {
- fireLineEvent(new LineEvent(PulseAudioDataLine.this,
- LineEvent.Type.START, framesSinceOpen));
- }
}
op.waitForCompletion();
@@ -331,16 +330,18 @@ public abstract class PulseAudioDataLine
}
+
public synchronized void stop() {
if (!isOpen) {
throw new IllegalStateException(
"Line must be open()ed before it can be start()ed");
}
+ writeInterrupted = true;
if (!isStarted) {
return;
}
- isStarted = true;
+
Operation op;
synchronized (eventLoop.threadLock) {
op = stream.cork();
@@ -442,4 +443,14 @@ public abstract class PulseAudioDataLine
return streamName;
}
+ public int getBytesInBuffer() {
+ Operation o;
+ synchronized (eventLoop.threadLock) {
+ o = stream.updateTimingInfo();
+ }
+ o.waitForCompletion();
+ o.releaseReference();
+ return stream.bytesInBuffer();
+ }
+
}
diff -r a3a8e9e19967 -r 4260a476a101 src/java/org/classpath/icedtea/pulseaudio/PulseAudioMixer.java
--- a/src/java/org/classpath/icedtea/pulseaudio/PulseAudioMixer.java Wed Oct 08 10:38:29 2008 -0400
+++ b/src/java/org/classpath/icedtea/pulseaudio/PulseAudioMixer.java Wed Oct 08 14:27:39 2008 -0400
@@ -401,6 +401,7 @@ public class PulseAudioMixer implements
@Override
public boolean isSynchronizationSupported(Line[] lines, boolean maintainSync) {
+
return false;
}
diff -r a3a8e9e19967 -r 4260a476a101 src/java/org/classpath/icedtea/pulseaudio/PulseAudioSourceDataLine.java
--- a/src/java/org/classpath/icedtea/pulseaudio/PulseAudioSourceDataLine.java Wed Oct 08 10:38:29 2008 -0400
+++ b/src/java/org/classpath/icedtea/pulseaudio/PulseAudioSourceDataLine.java Wed Oct 08 14:27:39 2008 -0400
@@ -44,6 +44,8 @@ import javax.sound.sampled.LineUnavailab
import javax.sound.sampled.LineUnavailableException;
import javax.sound.sampled.SourceDataLine;
+import org.classpath.icedtea.pulseaudio.Operation.State;
+
public class PulseAudioSourceDataLine extends PulseAudioDataLine implements
SourceDataLine, PulseAudioPlaybackLine {
@@ -51,9 +53,6 @@ public class PulseAudioSourceDataLine ex
private PulseAudioVolumeControl volumeControl;
private boolean muted;
private float volume;
-
- boolean flushed = false;
- boolean drained = false;
public PulseAudioSourceDataLine(EventLoop eventLoop, AudioFormat[] formats,
AudioFormat defaultFormat) {
@@ -134,6 +133,10 @@ public class PulseAudioSourceDataLine ex
public int write(byte[] data, int offset, int length) {
// can't call write() without open()ing first, but can call write()
// without start()ing
+ synchronized (this) {
+ writeInterrupted = false;
+ }
+
if (!isOpen) {
throw new IllegalStateException("must call open() before write()");
}
@@ -157,62 +160,57 @@ public class PulseAudioSourceDataLine ex
int sizeWritten = 0;
- flushed = false;
-
boolean interrupted = false;
while (remainingLength != 0) {
- synchronized (this) {
-
- if (!isStarted || !isOpen) {
- return sizeWritten;
- }
-
- if (flushed) {
- flushed = false;
- return sizeWritten;
- }
-
- synchronized (eventLoop.threadLock) {
-
- do {
- availableSize = stream.getWritableSize();
-
- if (availableSize < 0) {
- return sizeWritten;
+ synchronized (eventLoop.threadLock) {
+
+ do {
+ if (writeInterrupted) {
+ return sizeWritten;
+ }
+
+ if (availableSize == -1) {
+ return sizeWritten;
+ }
+ availableSize = stream.getWritableSize();
+
+ if (availableSize == 0) {
+ try {
+ eventLoop.threadLock.wait(100);
+ } catch (InterruptedException e) {
+ // ignore for now
+ interrupted = true;
}
- if (availableSize == 0) {
- try {
- eventLoop.threadLock.wait();
- } catch (InterruptedException e) {
- // ignore for now
- interrupted = true;
- }
-
- }
-
- } while (availableSize == 0);
-
- if (availableSize > remainingLength) {
- availableSize = remainingLength;
- }
-
- // only write entire frames, so round down avialableSize to
- // a
- // multiple of frameSize
- availableSize = (availableSize / frameSize) * frameSize;
-
+ }
+
+ } while (availableSize == 0);
+
+ if (availableSize > remainingLength) {
+ availableSize = remainingLength;
+ }
+
+ // only write entire frames, so round down avialableSize to
+ // a
+ // multiple of frameSize
+ availableSize = (availableSize / frameSize) * frameSize;
+
+ synchronized (this) {
+ if (writeInterrupted) {
+ return sizeWritten;
+ }
/* write a little bit of the buffer */
stream.write(data, position, availableSize);
-
- sizeWritten += availableSize;
- position += availableSize;
- remainingLength -= availableSize;
-
- framesSinceOpen += availableSize / frameSize;
- }
+ }
+
+ sizeWritten += availableSize;
+ position += availableSize;
+ remainingLength -= availableSize;
+
+ framesSinceOpen += availableSize / frameSize;
+
}
}
@@ -260,6 +258,29 @@ public class PulseAudioSourceDataLine ex
}
+ synchronized (this) {
+ writeInterrupted = true;
+ }
+
+ do {
+ synchronized (this) {
+ if (!isOpen) {
+ return;
+ }
+ if (getBytesInBuffer() == 0) {
+ return;
+ }
+ if (isStarted || !isOpen) {
+ break;
+ }
+ try {
+ this.wait(100);
+ } catch (InterruptedException e) {
+ return;
+ }
+ }
+ } while (!isStarted);
+
Operation operation;
synchronized (eventLoop.threadLock) {
@@ -269,10 +290,6 @@ public class PulseAudioSourceDataLine ex
operation.waitForCompletion();
operation.releaseReference();
- synchronized (this) {
- drained = true;
- }
-
}
@Override
@@ -281,6 +298,9 @@ public class PulseAudioSourceDataLine ex
throw new IllegalStateException(
"Line must be open before it can be flush()ed");
}
+ synchronized (this) {
+ writeInterrupted = true;
+ }
Operation operation;
synchronized (eventLoop.threadLock) {
@@ -290,10 +310,6 @@ public class PulseAudioSourceDataLine ex
operation.waitForCompletion();
operation.releaseReference();
- synchronized (this) {
- flushed = true;
- }
-
}
@Override
@@ -301,6 +317,8 @@ public class PulseAudioSourceDataLine ex
if (!isOpen) {
throw new IllegalStateException("not open so cant close");
}
+
+ writeInterrupted = true;
PulseAudioMixer parent = PulseAudioMixer.getInstance();
parent.removeSourceLine(this);
diff -r a3a8e9e19967 -r 4260a476a101 src/java/org/classpath/icedtea/pulseaudio/PulseAudioSourcePort.java
--- a/src/java/org/classpath/icedtea/pulseaudio/PulseAudioSourcePort.java Wed Oct 08 10:38:29 2008 -0400
+++ b/src/java/org/classpath/icedtea/pulseaudio/PulseAudioSourcePort.java Wed Oct 08 14:27:39 2008 -0400
@@ -37,6 +37,8 @@ exception statement from your version.
package org.classpath.icedtea.pulseaudio;
+import java.io.IOException;
+
import javax.sound.sampled.Port;
public class PulseAudioSourcePort extends PulseAudioPort {
diff -r a3a8e9e19967 -r 4260a476a101 src/java/org/classpath/icedtea/pulseaudio/PulseAudioTargetPort.java
--- a/src/java/org/classpath/icedtea/pulseaudio/PulseAudioTargetPort.java Wed Oct 08 10:38:29 2008 -0400
+++ b/src/java/org/classpath/icedtea/pulseaudio/PulseAudioTargetPort.java Wed Oct 08 14:27:39 2008 -0400
@@ -37,6 +37,8 @@ exception statement from your version.
package org.classpath.icedtea.pulseaudio;
+import java.io.IOException;
+
import javax.sound.sampled.Port;
public class PulseAudioTargetPort extends PulseAudioPort {
diff -r a3a8e9e19967 -r 4260a476a101 src/java/org/classpath/icedtea/pulseaudio/Stream.java
--- a/src/java/org/classpath/icedtea/pulseaudio/Stream.java Wed Oct 08 10:38:29 2008 -0400
+++ b/src/java/org/classpath/icedtea/pulseaudio/Stream.java Wed Oct 08 14:27:39 2008 -0400
@@ -37,6 +37,7 @@ exception statement from your version.
package org.classpath.icedtea.pulseaudio;
+import java.io.IOException;
import java.util.LinkedList;
import java.util.List;
@@ -83,6 +84,10 @@ public class Stream {
}
public interface MovedListener {
+ public void update();
+ }
+
+ public interface UpdateTimingInfoListener {
public void update();
}
@@ -163,6 +168,10 @@ public class Stream {
private native int native_pa_stream_readable_size();
private native byte[] native_pa_stream_drain();
+
+ private native byte[] native_pa_stream_updateTimingInfo();
+
+ public native int bytesInBuffer();
/*
* pa_operation pa_stream_update_timing_info (pa_stream *p,
@@ -370,6 +379,7 @@ public class Stream {
suspendedListeners.remove(listener);
}
}
+
public Stream.State getState() {
int state = native_pa_stream_get_state();
@@ -534,6 +544,12 @@ public class Stream {
return drainOperation;
}
More information about the distro-pkg-dev
mailing list