changeset in /hg/pulseaudio: 2008-09-02 Omair Majid <omajid at redh...
Omair Majid
omajid at redhat.com
Wed Sep 3 07:59:35 PDT 2008
changeset 52c81c12f7d0 in /hg/pulseaudio
details: http://icedtea.classpath.org/hg/pulseaudio?cmd=changeset;node=52c81c12f7d0
description:
2008-09-02 Omair Majid <omajid at redhat.com>
* src/java/org/classpath/icedtea/pulseaudio/EventLoop.java
Formatting fixes.
* src/java/org/classpath/icedtea/pulseaudio/Operation.java
(waitForCompletion): Removed redundant call to getState and the assert.
* src/java/org/classpath/icedtea/pulseaudio/PulseAudioClip.java
(writeFrames): Fixed the function to not hog the cpu completely even when
there is nothing to do. Big performance gain.
(open): Add the clip to the source lines currently open in the mixer.
(close): Remoce the clip from the list of source lines currently open in
the mixer.
* src/java/org/classpath/icedtea/pulseaudio/PulseAudioDataLine.java
Added license text. Removed unneeded imports.
(open): Initial implementation of support for START and STOP events.
* src/java/org/classpath/icedtea/pulseaudio/PulseAudioLine.java.
Added license text. Removed unneeded imports.
* src/java/org/classpath/icedtea/pulseaudio/PulseAudioMixer.java
sourceLines and targetLines are now lists of PulseAudioLine. They contain
all the source and target lines open to the mixer.
(getSourceLines): Return a Line[] instead of a derived class
(close): Added a check to make sure all the lines have been closed.
(addSourceDataLine): Renamed to addSourceLine.
(removeSourceDataLine): Renamed to removeSourceLine.
(addTargetDataLine): Renamed to addTargetLine.
(removeTargetDataLine): Rename to removeTargetLine.
* src/java/org/classpath/icedtea/pulseaudio/PulseAudioPlaybackLine.java
Added license text.
* src/java/org/classpath/icedtea/pulseaudio/PulseAudioPort.java
Likewise.
(getName): Accessor function for the name of the port.
* src/java/org/classpath/icedtea/pulseaudio/PulseAudioSourceDataLine.java
(write): Fixed the function to not use ~90% CPU. Now stays around 10%.
* src/java/org/classpath/icedtea/pulseaudio/PulseAudioSourcePort.java
Added license text.
(native_updateVolumeInfo): Returns a pointer as a long, not as an int.
* src/java/org/classpath/icedtea/pulseaudio/PulseAudioTargetDataLine.java
Formatting fixes.
* src/java/org/classpath/icedtea/pulseaudio/PulseAudioTargetPort.java
Added license text.
(native_updateVolumeInfo): Return a long instead of an int.
* src/java/org/classpath/icedtea/pulseaudio/PulseAudioRecorder.java
Added license text.
* src/java/org/classpath/icedtea/pulseaudio/Stream.java
Likewise.
* src/java/org/classpath/icedtea/pulseaudio/StreamBufferAttributes.java
Likewise.
* src/java/org/classpath/icedtea/pulseaudio/StreamSampleSpecification.java
Likewise.
* unittests/org/classpath/icedtea/pulseaudio/PulseAudioClipTest.java
(playTwoSoundFiles): Fixed test - complete playing and then exit properly.
* unittests/org/classpath/icedtea/pulseaudio/PulseAudioEventLoopOverhead.java
New test class. Tests the performance of the Eventloop when nothing is
happeneing.
* unittests/org/classpath/icedtea/pulseaudio/PulseAudioMixerTest.java
(testSourceLinesOpenAndClose): Changed the test to work with ports.
(testLongWait): Reduce the waiting time. Use PulseAudioEventLoopOverehead
for longer durations.
* unitests/org/classpath/icedtea/pulseaudio/PulseSourceDataLineTest.java
(testStartedStopped): New test. Checks for the START and STOP signals
being emitted.
diffstat:
20 files changed, 751 insertions(+), 210 deletions(-)
src/java/org/classpath/icedtea/pulseaudio/EventLoop.java | 17 -
src/java/org/classpath/icedtea/pulseaudio/Operation.java | 8
src/java/org/classpath/icedtea/pulseaudio/PulseAudioClip.java | 48 ++++
src/java/org/classpath/icedtea/pulseaudio/PulseAudioDataLine.java | 70 ++++++-
src/java/org/classpath/icedtea/pulseaudio/PulseAudioLine.java | 52 +++--
src/java/org/classpath/icedtea/pulseaudio/PulseAudioMixer.java | 75 +++----
src/java/org/classpath/icedtea/pulseaudio/PulseAudioPlaybackLine.java | 51 ++++-
src/java/org/classpath/icedtea/pulseaudio/PulseAudioPort.java | 82 +++++---
src/java/org/classpath/icedtea/pulseaudio/PulseAudioSourceDataLine.java | 61 ++++--
src/java/org/classpath/icedtea/pulseaudio/PulseAudioSourcePort.java | 51 ++++-
src/java/org/classpath/icedtea/pulseaudio/PulseAudioTargetDataLine.java | 8
src/java/org/classpath/icedtea/pulseaudio/PulseAudioTargetPort.java | 59 ++++-
src/java/org/classpath/icedtea/pulseaudio/SimpleAudioRecorder.java | 100 ++++++----
src/java/org/classpath/icedtea/pulseaudio/Stream.java | 44 +++-
src/java/org/classpath/icedtea/pulseaudio/StreamBufferAttributes.java | 41 +++-
src/java/org/classpath/icedtea/pulseaudio/StreamSampleSpecification.java | 40 +++-
unittests/org/classpath/icedtea/pulseaudio/PulseAudioClipTest.java | 13 +
unittests/org/classpath/icedtea/pulseaudio/PulseAudioEventLoopOverhead.java | 67 ++++++
unittests/org/classpath/icedtea/pulseaudio/PulseAudioMixerTest.java | 7
unittests/org/classpath/icedtea/pulseaudio/PulseSourceDataLineTest.java | 67 ++++++
diffs (truncated from 1610 to 500 lines):
diff -r fd7d965da555 -r 52c81c12f7d0 src/java/org/classpath/icedtea/pulseaudio/EventLoop.java
--- a/src/java/org/classpath/icedtea/pulseaudio/EventLoop.java Tue Sep 02 14:02:34 2008 -0400
+++ b/src/java/org/classpath/icedtea/pulseaudio/EventLoop.java Tue Sep 02 17:23:01 2008 -0400
@@ -71,7 +71,7 @@ public class EventLoop implements Runnab
// private boolean eventLoopIsRunning = false;
public Semaphore finished = new Semaphore(0);
-
+
private List<String> targetPortNameList = new ArrayList<String>();
private List<String> sourcePortNameList = new ArrayList<String>();
@@ -243,10 +243,11 @@ public class EventLoop implements Runnab
public long getMainLoopPointer() {
return mainloopPointer;
}
-
+
private native int nativeUpdateTargetPortNameList();
+
private native int nativeUpdateSourcePortNameList();
-
+
protected synchronized List<String> updateTargetPortNameList() {
targetPortNameList = new ArrayList<String>();
Operation op;
@@ -261,7 +262,7 @@ public class EventLoop implements Runnab
op.releaseReference();
return targetPortNameList;
}
-
+
protected synchronized List<String> updateSourcePortNameList() {
sourcePortNameList = new ArrayList<String>();
Operation op;
@@ -276,14 +277,12 @@ public class EventLoop implements Runnab
op.releaseReference();
return sourcePortNameList;
}
-
-
- public void source_callback(String name ) {
+
+ public void source_callback(String name) {
sourcePortNameList.add(name);
}
-
- public void sink_callback(String name ) {
+ public void sink_callback(String name) {
targetPortNameList.add(name);
}
diff -r fd7d965da555 -r 52c81c12f7d0 src/java/org/classpath/icedtea/pulseaudio/Operation.java
--- a/src/java/org/classpath/icedtea/pulseaudio/Operation.java Tue Sep 02 14:02:34 2008 -0400
+++ b/src/java/org/classpath/icedtea/pulseaudio/Operation.java Tue Sep 02 17:23:01 2008 -0400
@@ -63,7 +63,8 @@ public class Operation {
String library = new java.io.File(".").getCanonicalPath()
+ java.io.File.separatorChar
+ System.mapLibraryName("pulse-java");
- System.out.println(Operation.class.getCanonicalName() + ": " + library);
+ System.out.println(Operation.class.getCanonicalName() + ": "
+ + library);
System.load(library);
} catch (IOException e) {
assert ("Loading failed".endsWith("library"));
@@ -77,7 +78,7 @@ public class Operation {
private native int native_get_state();
public Operation(long operationPointer) {
- assert(operationPointer != 0);
+ assert (operationPointer != 0);
this.operationPointer = operationPointer;
this.eventLoop = EventLoop.getEventLoop();
}
@@ -132,9 +133,6 @@ public class Operation {
public void waitForCompletion() {
assert (operationPointer != 0);
- State operationState = getState();
- assert (operationState != State.Done);
-
do {
synchronized (eventLoop.threadLock) {
if (getState() == Operation.State.Done) {
diff -r fd7d965da555 -r 52c81c12f7d0 src/java/org/classpath/icedtea/pulseaudio/PulseAudioClip.java
--- a/src/java/org/classpath/icedtea/pulseaudio/PulseAudioClip.java Tue Sep 02 14:02:34 2008 -0400
+++ b/src/java/org/classpath/icedtea/pulseaudio/PulseAudioClip.java Tue Sep 02 17:23:01 2008 -0400
@@ -53,6 +53,8 @@ import javax.sound.sampled.LineListener;
import javax.sound.sampled.LineListener;
import javax.sound.sampled.LineUnavailableException;
import javax.sound.sampled.Control.Type;
+
+import org.classpath.icedtea.pulseaudio.Stream.WriteListener;
public class PulseAudioClip extends PulseAudioDataLine implements Clip,
PulseAudioPlaybackLine {
@@ -117,10 +119,41 @@ public class PulseAudioClip extends Puls
private void writeFrames(int startingFrame, int lastFrame) {
+ WriteListener writeListener = new WriteListener() {
+ @Override
+ public void update() {
+ eventLoop.threadLock.notifyAll();
+ }
+ };
+
+ stream.addWriteListener(writeListener);
+
int remainingFrames = lastFrame - startingFrame - 1;
while (remainingFrames > 0) {
synchronized (eventLoop.threadLock) {
- int availableSize = stream.getWritableSize();
+ int availableSize;
+
+ do {
+ availableSize = stream.getWritableSize();
+ if (availableSize < 0) {
+ Thread.currentThread().interrupt();
+ stream.removeWriteListener(writeListener);
+ return;
+ }
+ if (availableSize == 0) {
+ try {
+ eventLoop.threadLock.wait();
+ } catch (InterruptedException e) {
+ // FIXME what to do when this happens?
+ // e.printStackTrace();
+ Thread.currentThread().interrupt();
+ stream.removeWriteListener(writeListener);
+ return;
+ }
+ }
+
+ } while (availableSize == 0);
+
int framesToWrite = Math.min(remainingFrames, availableSize
/ getFormat().getFrameSize());
stream.write(data, currentFrame * getFormat().getFrameSize(),
@@ -134,6 +167,9 @@ public class PulseAudioClip extends Puls
// System.out.println("remaining frames" + remainingFrames);
}
}
+
+ stream.removeWriteListener(writeListener);
+
}
public PulseAudioClip(EventLoop eventLoop, AudioFormat[] formats,
@@ -174,6 +210,10 @@ public class PulseAudioClip extends Puls
stream.disconnect();
isOpen = false;
}
+
+ PulseAudioMixer mixer = PulseAudioMixer.getInstance();
+ mixer.removeSourceLine(this);
+
}
@Override
@@ -341,7 +381,6 @@ public class PulseAudioClip extends Puls
this.data = new byte[bufferSize];
System.arraycopy(data, offset, this.data, 0, bufferSize);
frameCount = bufferSize / format.getFrameSize();
- isOpen = true;
PulseAudioVolumeControl volumeControl = new PulseAudioVolumeControl(
this, eventLoop);
@@ -350,6 +389,11 @@ public class PulseAudioClip extends Puls
controls.add(volumeControl);
controls.add(muteControl);
+ PulseAudioMixer mixer = PulseAudioMixer.getInstance();
+ mixer.addSourceLine(this);
+
+ isOpen = true;
+
}
public long native_setVolume(float value) {
diff -r fd7d965da555 -r 52c81c12f7d0 src/java/org/classpath/icedtea/pulseaudio/PulseAudioDataLine.java
--- a/src/java/org/classpath/icedtea/pulseaudio/PulseAudioDataLine.java Tue Sep 02 14:02:34 2008 -0400
+++ b/src/java/org/classpath/icedtea/pulseaudio/PulseAudioDataLine.java Tue Sep 02 17:23:01 2008 -0400
@@ -1,17 +1,52 @@ package org.classpath.icedtea.pulseaudio
+/* PulseAudioClip.java
+ Copyright (C) 2008 Red Hat, Inc.
+
+This file is part of IcedTea.
+
+IcedTea is free software; you can redistribute it and/or
+modify it under the terms of the GNU General Public License as published by
+the Free Software Foundation, version 2.
+
+IcedTea is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with IcedTea; see the file COPYING. If not, write to
+the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
+02110-1301 USA.
+
+Linking this library statically or dynamically with other modules is
+making a combined work based on this library. Thus, the terms and
+conditions of the GNU General Public License cover the whole
+combination.
+
+As a special exception, the copyright holders of this library give you
+permission to link this library with independent modules to produce an
+executable, regardless of the license terms of these independent
+modules, and to copy and distribute the resulting executable under
+terms of your choice, provided that you also meet, for each linked
+independent module, the terms and conditions of the license of that
+module. An independent module is a module which is not derived from
+or based on this library. If you modify this library, you may extend
+this exception to your version of the library, but you are not
+obligated to do so. If you do not wish to do so, delete this
+exception statement from your version.
+ */
+
package org.classpath.icedtea.pulseaudio;
-import java.util.ArrayList;
-import java.util.List;
import java.util.concurrent.Semaphore;
import javax.sound.sampled.AudioFormat;
import javax.sound.sampled.AudioSystem;
import javax.sound.sampled.DataLine;
import javax.sound.sampled.LineEvent;
-import javax.sound.sampled.LineListener;
import javax.sound.sampled.LineUnavailableException;
-public abstract class PulseAudioDataLine extends PulseAudioLine implements DataLine {
+public abstract class PulseAudioDataLine extends PulseAudioLine implements
+ DataLine {
protected static final int DEFAULT_BUFFER_SIZE = StreamBufferAttributes.SANE_DEFAULT;
protected static final String PULSEAUDIO_FORMAT_KEY = "PulseAudioFormatKey";
@@ -20,7 +55,6 @@ public abstract class PulseAudioDataLine
// true between open() and close(). ie represents when a line has acquire
// resources
-
// true between start() and stop()
protected boolean isStarted = false;
@@ -85,6 +119,26 @@ public abstract class PulseAudioDataLine
stream.addStateListener(openCloseListener);
+ Stream.UnderflowListener stoppedListener = new Stream.UnderflowListener() {
+ @Override
+ public void update() {
+ fireLineEvent(new LineEvent(PulseAudioDataLine.this,
+ LineEvent.Type.STOP, AudioSystem.NOT_SPECIFIED));
+ }
+ };
+ stream.addUnderflowListener(stoppedListener);
+
+ Stream.PlaybackStartedListener startedListener = new Stream.PlaybackStartedListener() {
+ @Override
+ public void update() {
+ fireLineEvent(new LineEvent(PulseAudioDataLine.this,
+ LineEvent.Type.START, AudioSystem.NOT_SPECIFIED));
+ }
+
+ };
+
+ stream.addPlaybackStartedListener(startedListener);
+
synchronized (eventLoop.threadLock) {
connectLine(bufferSize);
}
@@ -183,16 +237,12 @@ public abstract class PulseAudioDataLine
return isEngagedInIo;
}
-
-
protected abstract void connectLine(int bufferSize);
public abstract void drain();
-
-
public Stream getStream() {
return stream;
}
-
+
}
diff -r fd7d965da555 -r 52c81c12f7d0 src/java/org/classpath/icedtea/pulseaudio/PulseAudioLine.java
--- a/src/java/org/classpath/icedtea/pulseaudio/PulseAudioLine.java Tue Sep 02 14:02:34 2008 -0400
+++ b/src/java/org/classpath/icedtea/pulseaudio/PulseAudioLine.java Tue Sep 02 17:23:01 2008 -0400
@@ -1,22 +1,53 @@ package org.classpath.icedtea.pulseaudio
+/* PulseAudioClip.java
+ Copyright (C) 2008 Red Hat, Inc.
+
+This file is part of IcedTea.
+
+IcedTea is free software; you can redistribute it and/or
+modify it under the terms of the GNU General Public License as published by
+the Free Software Foundation, version 2.
+
+IcedTea is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with IcedTea; see the file COPYING. If not, write to
+the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
+02110-1301 USA.
+
+Linking this library statically or dynamically with other modules is
+making a combined work based on this library. Thus, the terms and
+conditions of the GNU General Public License cover the whole
+combination.
+
+As a special exception, the copyright holders of this library give you
+permission to link this library with independent modules to produce an
+executable, regardless of the license terms of these independent
+modules, and to copy and distribute the resulting executable under
+terms of your choice, provided that you also meet, for each linked
+independent module, the terms and conditions of the license of that
+module. An independent module is a module which is not derived from
+or based on this library. If you modify this library, you may extend
+this exception to your version of the library, but you are not
+obligated to do so. If you do not wish to do so, delete this
+exception statement from your version.
+ */
+
package org.classpath.icedtea.pulseaudio;
import java.util.ArrayList;
import java.util.List;
-import javax.sound.sampled.Control;
-import javax.sound.sampled.Line;
import javax.sound.sampled.LineEvent;
import javax.sound.sampled.LineListener;
-import javax.sound.sampled.LineUnavailableException;
-import javax.sound.sampled.Control.Type;
-
abstract class PulseAudioLine {
protected List<LineListener> lineListeners = new ArrayList<LineListener>();
protected boolean isOpen = false;
-
-
+
public void addLineListener(LineListener listener) {
this.lineListeners.add(listener);
}
@@ -30,14 +61,9 @@ abstract class PulseAudioLine {
lineListener.update(e);
}
}
-
+
public boolean isOpen() {
return isOpen;
}
-
-
-
-
-
}
diff -r fd7d965da555 -r 52c81c12f7d0 src/java/org/classpath/icedtea/pulseaudio/PulseAudioMixer.java
--- a/src/java/org/classpath/icedtea/pulseaudio/PulseAudioMixer.java Tue Sep 02 14:02:34 2008 -0400
+++ b/src/java/org/classpath/icedtea/pulseaudio/PulseAudioMixer.java Tue Sep 02 17:23:01 2008 -0400
@@ -63,7 +63,6 @@ import javax.sound.sampled.TargetDataLin
import javax.sound.sampled.TargetDataLine;
import javax.sound.sampled.AudioFormat.Encoding;
import javax.sound.sampled.Control.Type;
-import javax.sound.sampled.Port.Info;
public class PulseAudioMixer implements javax.sound.sampled.Mixer {
// singleton
@@ -82,17 +81,15 @@ public class PulseAudioMixer implements
private boolean isOpen = false;
- private final List<PulseAudioSourceDataLine> sourceLines = new ArrayList<PulseAudioSourceDataLine>();
- private final List<PulseAudioTargetDataLine> targetLines = new ArrayList<PulseAudioTargetDataLine>();
- private final List<PulseAudioClip> clips = new ArrayList<PulseAudioClip>();
+ private final List<PulseAudioLine> sourceLines = new ArrayList<PulseAudioLine>();
+ private final List<PulseAudioLine> targetLines = new ArrayList<PulseAudioLine>();
private final List<LineListener> lineListeners = new ArrayList<LineListener>();
private PulseAudioMixer() {
AudioFormat[] formats = getSupportedFormats();
List<Line.Info> sourceLineInfoList = new ArrayList<Line.Info>();
- sourceLineInfoList.add(new DataLine.Info(
- SourceDataLine.class, formats,
+ sourceLineInfoList.add(new DataLine.Info(SourceDataLine.class, formats,
StreamBufferAttributes.MIN_VALUE,
StreamBufferAttributes.MAX_VALUE));
sourceLineInfos = new Line.Info[] { new DataLine.Info(
@@ -331,25 +328,22 @@ public class PulseAudioMixer implements
}
if ((info.getLineClass() == Clip.class)) {
- Clip clip = new PulseAudioClip(eventLoop, formats, defaultFormat);
- clips.add((PulseAudioClip) clip);
- return clip;
- }
-
+ return new PulseAudioClip(eventLoop, formats, defaultFormat);
+ }
+
String portName;
boolean isSource;
-
+
if (Port.Info.class.isInstance(info)) {
Port.Info portInfo = (Port.Info) info;
portName = portInfo.getName();
- if(portInfo.isSource()){
+ if (portInfo.isSource()) {
return new PulseAudioSourcePort(portInfo.getName(), eventLoop);
} else {
return new PulseAudioTargetPort(portInfo.getName(), eventLoop);
}
}
-
-
+
throw new IllegalArgumentException();
}
@@ -386,7 +380,7 @@ public class PulseAudioMixer implements
@Override
public Line[] getSourceLines() {
- return (Line[]) sourceLines.toArray(new PulseAudioSourceDataLine[0]);
+ return (Line[]) sourceLines.toArray(new Line[0]);
}
@@ -482,7 +476,16 @@ public class PulseAudioMixer implements
// System.out.println(this.getClass().getName() + ": closed");
- this.isOpen = false;
+ isOpen = false;
+
+ if (sourceLines.size() > 0) {
+ System.out.println("DEBUG: some source lines have not been closed");
+ assert (sourceLines.size() < 0);
+ }
+ if (targetLines.size() > 0) {
+ System.out.println("DEBUG: some target lines have not been closed");
+ assert (targetLines.size() < 0);
+ }
synchronized (lineListeners) {
lineListeners.clear();
@@ -522,23 +525,23 @@ public class PulseAudioMixer implements
@Override
public void open() throws LineUnavailableException {
openLocal();
- //the sourceLineInfo and targetLineInfo arrays need to be updated with
- //port infos, which can only be obtained after EventLoop had started
-
+ // the sourceLineInfo and targetLineInfo arrays need to be updated with
+ // port infos, which can only be obtained after EventLoop had started
+
ArrayList<Line.Info> sourceLineInfoList = new ArrayList<Line.Info>();
sourceLineInfoList.add(sourceLineInfos[0]);
- for(String portName : eventLoop.updateSourcePortNameList()){
+ for (String portName : eventLoop.updateSourcePortNameList()) {
sourceLineInfoList.add(new Port.Info(Port.class, portName, true));
}
sourceLineInfos = sourceLineInfoList.toArray(new Line.Info[0]);
-
+
ArrayList<Line.Info> targetLineInfoList = new ArrayList<Line.Info>();
targetLineInfoList.add(targetLineInfos[0]);
- for(String portName : eventLoop.updateTargetPortNameList()){
+ for (String portName : eventLoop.updateTargetPortNameList()) {
targetLineInfoList.add(new Port.Info(Port.class, portName, false));
}
targetLineInfos = targetLineInfoList.toArray(new Line.Info[0]);
-
+
}
public void open(String appName, String host) throws UnknownHostException,
@@ -689,38 +692,38 @@ public class PulseAudioMixer implements
AudioInputStream audioInputStream = AudioSystem
.getAudioInputStream(soundFile);
AudioFormat audioFormat = audioInputStream.getFormat();
More information about the distro-pkg-dev
mailing list