changeset in /hg/pulseaudio: Added support for querying the audi...
Omair Majid
omajid at redhat.com
Fri Aug 8 08:19:52 PDT 2008
changeset 2785b9eba70d in /hg/pulseaudio
details: http://icedtea.classpath.org/hg/pulseaudio?cmd=changeset;node=2785b9eba70d
description:
Added support for querying the audio formats available to pulseaudio
and obtaining a line based on those formats
also added test cases
diffstat:
5 files changed, 323 insertions(+), 106 deletions(-)
src/org/classpath/icedtea/pulseaudio/PulseAudioMixer.java | 39 -
src/org/classpath/icedtea/pulseaudio/PulseAudioSourceDataLine.java | 270 ++++++++--
src/org_classpath_icedtea_pulseaudio_PulseAudioSourceDataLine.c | 45 +
unittests/org/classpath/icedtea/pulseaudio/PulseAudioMixerTest.java | 3
unittests/org/classpath/icedtea/pulseaudio/PulseSourceDataLineTest.java | 72 +-
diffs (truncated from 719 to 500 lines):
diff -r 2331da2e5f6a -r 2785b9eba70d src/org/classpath/icedtea/pulseaudio/PulseAudioMixer.java
--- a/src/org/classpath/icedtea/pulseaudio/PulseAudioMixer.java Wed Aug 06 16:30:43 2008 -0400
+++ b/src/org/classpath/icedtea/pulseaudio/PulseAudioMixer.java Fri Aug 08 11:11:03 2008 -0400
@@ -48,6 +48,7 @@ import javax.sound.sampled.AudioInputStr
import javax.sound.sampled.AudioInputStream;
import javax.sound.sampled.AudioSystem;
import javax.sound.sampled.Control;
+import javax.sound.sampled.DataLine;
import javax.sound.sampled.Line;
import javax.sound.sampled.LineEvent;
import javax.sound.sampled.LineListener;
@@ -71,8 +72,6 @@ public class PulseAudioMixer implements
private List<PulseAudioSourceDataLine> sourceLines = new ArrayList<PulseAudioSourceDataLine>();
// private List<PulseAudioTargetDataLine> targetLines = null;
- private Line.Info sourceDataLineInfo = new Line.Info(
- PulseAudioSourceDataLine.class);
// private Line.Info targetDataLineInfo = new
// Line.Info(PulseAudioTargetDataLine.class);
@@ -99,11 +98,14 @@ public class PulseAudioMixer implements
throw new LineUnavailableException();
}
- if (info.matches(sourceDataLineInfo)) {
- PulseAudioSourceDataLine sourceLine = null;
- sourceLine = new PulseAudioSourceDataLine(eventLoop);
- sourceLines.add(sourceLine);
- return sourceLine;
+ PulseAudioSourceDataLine sourceLine = null;
+ sourceLine = new PulseAudioSourceDataLine(eventLoop);
+ Line.Info sourceDataLineInfo = sourceLine.getLineInfo();
+ if (info instanceof DataLine.Info) {
+ if (info.matches(sourceDataLineInfo)) {
+ sourceLines.add(sourceLine);
+ return sourceLine;
+ }
}
// if (info.matches(_targetDataLineInfo)) {
@@ -127,8 +129,14 @@ public class PulseAudioMixer implements
@Override
public javax.sound.sampled.Line.Info[] getSourceLineInfo() {
- Line.Info[] info = { new Line.Info(PulseAudioSourceDataLine.class), };
- return info;
+ if (isOpen) {
+ SourceDataLine sourceLine = new PulseAudioSourceDataLine(eventLoop);
+ Line.Info[] info = new Line.Info[] { sourceLine.getLineInfo() };
+ return info;
+ }
+
+ // if not open, then return empty array
+ return new Line.Info[] {};
}
@Override
@@ -347,7 +355,7 @@ public class PulseAudioMixer implements
eventLoopThread.start();
try {
-// System.out.println("waiting...");
+ // System.out.println("waiting...");
ready.acquire();
if (eventLoop.getStatus() != 4) {
/*
@@ -362,12 +370,13 @@ public class PulseAudioMixer implements
throw new LineUnavailableException();
}
eventLoop.removeContextListener(initListener);
-// System.out.println("got signal");
+ // System.out.println("got signal");
} catch (InterruptedException e) {
- System.out.println("PulseAudioMixer: got interrupted while waiting for the EventLoop to initialize");
- }
-
-// System.out.println(this.getClass().getName() + ": ready");
+ System.out
+ .println("PulseAudioMixer: got interrupted while waiting for the EventLoop to initialize");
+ }
+
+ // System.out.println(this.getClass().getName() + ": ready");
this.isOpen = true;
diff -r 2331da2e5f6a -r 2785b9eba70d src/org/classpath/icedtea/pulseaudio/PulseAudioSourceDataLine.java
--- a/src/org/classpath/icedtea/pulseaudio/PulseAudioSourceDataLine.java Wed Aug 06 16:30:43 2008 -0400
+++ b/src/org/classpath/icedtea/pulseaudio/PulseAudioSourceDataLine.java Fri Aug 08 11:11:03 2008 -0400
@@ -39,22 +39,29 @@ package org.classpath.icedtea.pulseaudio
import java.io.IOException;
import java.util.ArrayList;
+import java.util.HashMap;
+import java.util.LinkedList;
import java.util.List;
+import java.util.Map;
import java.util.concurrent.Semaphore;
import javax.sound.sampled.AudioFormat;
+import javax.sound.sampled.AudioSystem;
import javax.sound.sampled.BooleanControl;
import javax.sound.sampled.Control;
-import javax.sound.sampled.Line;
+import javax.sound.sampled.DataLine;
import javax.sound.sampled.LineEvent;
import javax.sound.sampled.LineListener;
import javax.sound.sampled.LineUnavailableException;
import javax.sound.sampled.SourceDataLine;
+import javax.sound.sampled.AudioFormat.Encoding;
import javax.sound.sampled.Control.Type;
public class PulseAudioSourceDataLine implements SourceDataLine {
private static final int DEFAULT_BUFFER_SIZE = 1000;
+ private static final String PULSEAUDIO_FORMAT_KEY = "PulseAudioFormatKey";
+
private String streamName = "Java Stream";
private List<StreamListener> streamListeners = new ArrayList<StreamListener>();
@@ -63,7 +70,8 @@ public class PulseAudioSourceDataLine im
private boolean isOpen = false;
private boolean isPaused = false;
- private AudioFormat format = null;
+ private List<AudioFormat> supportedFormats = null;
+ private AudioFormat currentFormat = null;
private Control[] controls;
private Mute muteControl;
@@ -79,9 +87,8 @@ public class PulseAudioSourceDataLine im
@SuppressWarnings("unused")
private long streamPointer;
- private native void native_open(long contextPointer, String name,
- String encoding, float rate, int size, int channels,
- boolean bigEndian, int bufferSize);
+ private native void native_open(long contextPointer, String streamName,
+ String encoding, int sampleRate, int channels, int bufferSize);
private native void native_write(byte[] data, int offset, int length);
@@ -113,6 +120,9 @@ public class PulseAudioSourceDataLine im
assert ("Loading failed".endsWith("library"));
}
}
+
+
+
public PulseAudioSourceDataLine(EventLoop eventLoop) {
this.eventLoop = eventLoop;
@@ -121,6 +131,189 @@ public class PulseAudioSourceDataLine im
volumeControl = new StreamVolume(this);
muteControl = new Mute();
controls = new Control[] { volumeControl, muteControl };
+
+ /*
+ * FIXME puselaudio supports any sample rate (it can covert between
+ * sample rates without a problem). it calculates the frame size and the
+ * frame rate based on that.
+ *
+ * Java's AudioSystem interface accepts NOT_SPECIFIED only for sample
+ * rate and frame rate. eg: cant say that it supports any number of
+ * audio channels
+ *
+ * sample size in bytes [PA_SAMPLE_U8] = 1, [PA_SAMPLE_ULAW] = 1,
+ * [PA_SAMPLE_ALAW] = 1, [PA_SAMPLE_S16LE] = 2, [PA_SAMPLE_S16BE] = 2,
+ * [PA_SAMPLE_FLOAT32LE] = 4, [PA_SAMPLE_FLOAT32BE] = 4,
+ * [PA_SAMPLE_S32LE] = 4, [PA_SAMPLE_S32BE] = 4,
+ *
+ *
+ */
+
+ supportedFormats = new LinkedList<AudioFormat>();
+
+ Map<String, Object> properties;
+
+ int[] channelSizes = new int[] { 1, 2, 5 };
+ for (int channelSize : channelSizes) {
+ properties = new HashMap<String, Object>();
+ properties.put(PULSEAUDIO_FORMAT_KEY, "PA_SAMPLE_U8");
+
+ // frameSize = sample size (in bytes, not bits) x # of channels
+ // ^ that's from PulseAudio sources, so it will pretty much break
+ // as soon as they change something
+ // FIXME ^
+ int sampleSize = 8; // in bits
+ AudioFormat PA_SAMPLE_U8 = new AudioFormat(
+ Encoding.PCM_UNSIGNED, // encoding
+ AudioSystem.NOT_SPECIFIED, // sample rate
+ sampleSize, // sample size
+ channelSize, // channels
+ sampleSize / 8 * channelSize, // frame size in bytes
+ AudioSystem.NOT_SPECIFIED, // frame rate
+ false, // big endian?
+ properties);
+
+ supportedFormats.add(PA_SAMPLE_U8);
+ }
+
+ for (int channelSize : channelSizes) {
+ properties = new HashMap<String, Object>();
+ properties.put(PULSEAUDIO_FORMAT_KEY, "PA_SAMPLE_ALAW");
+
+ // frameSize = sample size (in bytes, not bits) x # of channels
+ // ^ that's from PulseAudio sources, so it will pretty much break
+ // as soon as they change something
+ // FIXME ^
+
+ int sampleSize = 8;
+ final AudioFormat PA_SAMPLE_ALAW = new AudioFormat(Encoding.ALAW, // encoding
+ AudioSystem.NOT_SPECIFIED, // sample rate
+ sampleSize, // sample size
+ channelSize, // channels
+ sampleSize / 8 * channelSize, // frame size
+ AudioSystem.NOT_SPECIFIED, // frame rate
+ false, // big endian?
+ properties);
+
+ supportedFormats.add(PA_SAMPLE_ALAW);
+ }
+
+ for (int channelSize : channelSizes) {
+ properties = new HashMap<String, Object>();
+ properties.put(PULSEAUDIO_FORMAT_KEY, "PA_SAMPLE_ULAW");
+
+ // frameSize = sample size (in bytes, not bits) x # of channels
+ // ^ that's from PulseAudio sources, so it will pretty much break
+ // as soon as they change something
+ // FIXME ^
+
+ int sampleSize = 8;
+ final AudioFormat PA_SAMPLE_ULAW = new AudioFormat(Encoding.ULAW, // encoding
+ AudioSystem.NOT_SPECIFIED, // sample rate
+ sampleSize, // sample size
+ channelSize, // channels
+ sampleSize / 8 * channelSize, // frame size
+ AudioSystem.NOT_SPECIFIED, // frame rate
+ false, // big endian?
+ properties);
+
+ supportedFormats.add(PA_SAMPLE_ULAW);
+ }
+
+ for (int channelSize : channelSizes) {
+ properties = new HashMap<String, Object>();
+ properties.put(PULSEAUDIO_FORMAT_KEY, "PA_SAMPLE_S16BE");
+
+ // frameSize = sample size (in bytes, not bits) x # of channels
+ // ^ that's from PulseAudio sources, so it will pretty much break
+ // as soon as they change something
+ // FIXME ^
+
+ int sampleSize = 16;
+ final AudioFormat PA_SAMPLE_S16BE = new AudioFormat(
+ Encoding.PCM_SIGNED, // encoding
+ AudioSystem.NOT_SPECIFIED, // sample rate
+ sampleSize, // sample size
+ channelSize, // channels
+ sampleSize / 8 * channelSize, // frame size
+ AudioSystem.NOT_SPECIFIED, // frame rate
+ true, // big endian?
+ properties);
+
+ supportedFormats.add(PA_SAMPLE_S16BE);
+ }
+
+ for (int channelSize : channelSizes) {
+ properties = new HashMap<String, Object>();
+ properties.put(PULSEAUDIO_FORMAT_KEY, "PA_SAMPLE_S16LE");
+
+ // frameSize = sample size (in bytes, not bits) x # of channels
+ // ^ that's from PulseAudio sources, so it will pretty much break
+ // as soon as they change something
+ // FIXME ^
+
+ int sampleSize = 16;
+ final AudioFormat A_SAMPLE_S16LE = new AudioFormat(
+ Encoding.PCM_SIGNED, // encoding
+ AudioSystem.NOT_SPECIFIED, // sample rate
+ sampleSize, // sample size
+ channelSize, // channels
+ sampleSize / 8 * channelSize, // frame size
+ AudioSystem.NOT_SPECIFIED, // frame rate
+ false, // big endian?
+ properties);
+
+ supportedFormats.add(A_SAMPLE_S16LE);
+ }
+
+ for (int channelSize : channelSizes) {
+ properties = new HashMap<String, Object>();
+ properties.put(PULSEAUDIO_FORMAT_KEY, "PA_SAMPLE_S32BE");
+
+ // frameSize = sample size (in bytes, not bits) x # of channels
+ // ^ that's from PulseAudio sources, so it will pretty much break
+ // as soon as they change something
+ // FIXME ^
+
+ int sampleSize = 32;
+ final AudioFormat PA_SAMPLE_S32BE = new AudioFormat(
+ Encoding.PCM_SIGNED, // encoding
+ AudioSystem.NOT_SPECIFIED, // sample rate
+ sampleSize, // sample size
+ channelSize, // channels
+ sampleSize / 8 * channelSize, // frame size
+ AudioSystem.NOT_SPECIFIED, // frame rate
+ true, // big endian?
+ properties);
+
+ supportedFormats.add(PA_SAMPLE_S32BE);
+ }
+
+ for (int channelSize : channelSizes) {
+ properties = new HashMap<String, Object>();
+ properties.put(PULSEAUDIO_FORMAT_KEY, "PA_SAMPLE_S32LE");
+
+ // frameSize = sample size (in bytes, not bits) x # of channels
+ // ^ that's from PulseAudio sources, so it will pretty much break
+ // as soon as they change something
+ // FIXME ^
+
+ int sampleSize = 32;
+ final AudioFormat PA_SAMPLE_S32LE = new AudioFormat(
+ Encoding.PCM_SIGNED, // encoding
+ AudioSystem.NOT_SPECIFIED, // sample rate
+ sampleSize, // sample size
+ channelSize, // channels
+ sampleSize / 8 * channelSize, // frame size
+ AudioSystem.NOT_SPECIFIED, // frame rate
+ false, // big endian?
+ properties);
+
+ supportedFormats.add(PA_SAMPLE_S32LE);
+ }
+
+ currentFormat = null;
+
}
public void open(AudioFormat format, int bufferSize)
@@ -128,19 +321,23 @@ public class PulseAudioSourceDataLine im
if (isOpen) {
throw new IllegalStateException("Line is already open");
}
-
- isOpen = true;
-
- int channels = format.getChannels();
- float rate = format.getSampleRate();
- int sampleSize = format.getSampleSizeInBits();
- String encoding = format.getEncoding().toString();
- boolean bigEndian = format.isBigEndian();
-
- synchronized (eventLoop.threadLock) {
- native_open(eventLoop.getContextPointer(), streamName, encoding,
- rate, sampleSize, channels, bigEndian, bufferSize);
- }
+
+ // ignore suggested buffer size
+
+ for (AudioFormat myFormat : supportedFormats) {
+ if (format.matches(myFormat)) {
+ native_open(eventLoop.getContextPointer(), streamName,
+ (String) myFormat.getProperty(PULSEAUDIO_FORMAT_KEY),
+ (int) format.getSampleRate(), format.getChannels(),
+ bufferSize);
+ currentFormat = format;
+ isOpen = true;
+ return;
+ }
+ }
+
+ throw new IllegalArgumentException("invalid format");
+
}
public void open(AudioFormat format) throws LineUnavailableException {
@@ -149,7 +346,9 @@ public class PulseAudioSourceDataLine im
}
public void open() throws LineUnavailableException {
- format = new AudioFormat(44100, 16, 2, true, false);
+ // pick a random format
+ AudioFormat format = new AudioFormat(Encoding.PCM_UNSIGNED, 22050, 8,
+ 2, 2, AudioSystem.NOT_SPECIFIED, false);
open(format, DEFAULT_BUFFER_SIZE);
}
@@ -260,6 +459,8 @@ public class PulseAudioSourceDataLine im
};
public void close() {
+ assert (isOpen);
+
synchronized (eventLoop.threadLock) {
native_close();
}
@@ -271,35 +472,35 @@ public class PulseAudioSourceDataLine im
}
public int getBufferSize() {
+ // FIXME!
+ return 10000;
+ }
+
+ public AudioFormat getFormat() {
+ return currentFormat;
+ }
+
+ public int getFramePosition() {
// TODO Auto-generated method stub
return 0;
}
- public AudioFormat getFormat() {
- // TODO Auto-generated method stub
- return null;
- }
-
- public int getFramePosition() {
+ public float getLevel() {
// TODO Auto-generated method stub
return 0;
}
- public float getLevel() {
+ public long getLongFramePosition() {
// TODO Auto-generated method stub
return 0;
}
- public long getLongFramePosition() {
+ public long getMicrosecondPosition() {
// TODO Auto-generated method stub
return 0;
}
- public long getMicrosecondPosition() {
- // TODO Auto-generated method stub
- return 0;
- }
-
+
public boolean isActive() {
// TODO Auto-generated method stub
return false;
@@ -312,7 +513,7 @@ public class PulseAudioSourceDataLine im
public Control getControl(Type control) {
for (int i = 0; i < controls.length; i++) {
- if (controls[i].getType() == control){
+ if (controls[i].getType() == control) {
return controls[i];
}
}
@@ -324,7 +525,8 @@ public class PulseAudioSourceDataLine im
}
public javax.sound.sampled.Line.Info getLineInfo() {
- return new Line.Info(SourceDataLine.class);
+ return new DataLine.Info(SourceDataLine.class,
+ supportedFormats.toArray(new AudioFormat[0]), 0, 100000);
}
public boolean isControlSupported(Type control) {
diff -r 2331da2e5f6a -r 2785b9eba70d src/org_classpath_icedtea_pulseaudio_PulseAudioSourceDataLine.c
--- a/src/org_classpath_icedtea_pulseaudio_PulseAudioSourceDataLine.c Wed Aug 06 16:30:43 2008 -0400
+++ b/src/org_classpath_icedtea_pulseaudio_PulseAudioSourceDataLine.c Fri Aug 08 11:11:03 2008 -0400
@@ -98,14 +98,13 @@ static void stream_state_change_callback
return;
}
-
/*
* Class: org_classpath_icedtea_pulseaudio_PulseAudioSourceDataLine
* Method: native_open
- * Signature: (JLjava/lang/String;Ljava/lang/String;FIIZI)V
+ * Signature: (JLjava/lang/String;Ljava/lang/String;III)V
*/
JNIEXPORT void JNICALL Java_org_classpath_icedtea_pulseaudio_PulseAudioSourceDataLine_native_1open
-(JNIEnv* env, jobject obj, jlong contextPointer, jstring name, jstring encodingString, jfloat rate, jint size, jint channels, jboolean bigEndian, jint bufferSize) {
+(JNIEnv* env, jobject obj, jlong contextPointer, jstring name, jstring encodingString, jint sampleRate, jint channels, jint bufferSize) {
//TODO: Need to deal with the buffer size. Currently ignored
@@ -121,32 +120,43 @@ JNIEXPORT void JNICALL Java_org_classpat
const char *encoding = (*env)->GetStringUTFChars(env, encodingString, NULL);
- if( (strcmp(encoding, "PCM_UNSIGNED") == 0) && (size == 8)) {
+ if (strcmp(encoding, "PA_SAMPLE_U8") == 0) {
sample_spec.format = PA_SAMPLE_U8;
- } else if( (strcmp(encoding, "ALAW") == 0) && (size == 8)) {
+ } else if (strcmp(encoding, "PA_SAMPLE_ALAW") == 0) {
sample_spec.format = PA_SAMPLE_ALAW;
- } else if( (strcmp(encoding, "ULAW") == 0) && (size == 8)) {
+ } else if (strcmp(encoding, "PA_SAMPLE_ULAW;") == 0) {
sample_spec.format = PA_SAMPLE_ULAW;
- } else if ( (strcmp(encoding, "PCM_SIGNED") == 0) && (size == 16) && (bigEndian == 1)) {
More information about the distro-pkg-dev
mailing list