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

Omair Majid omajid at redhat.com
Fri Aug 15 09:03:11 PDT 2008


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

	    * build.xml: added PulseAudioTargetDataLine as a javah target
	    * src/java/org/classpath/icedtea/pulseaudio/ContextEvent.java: spacing
	      changes (modified to fit with eclipse's guidelines
	    * src/java/org/classpath/icedtea/pulseaudio/ContextListener.java: same
	    * src/java/org/classpath/icedtea/pulseaudio/EventLoop.java: same
	    * src/java/org/classpath/icedtea/pulseaudio/Operaiton.java: same
	    * src/java/org/classpath/icedtea/pulseaudio/PulseAudioClip.java: same
	    * src/java/org/classpath/icedtea/pulseaudio/PulseAudioMixer.java: merged
	    * src/java/org/classpath/icedtea/pulseausio/PulseAUdioMixerInfo.java:
	      spacing changes
	    * src/java/org/classpath/icedtea/pulseaudio/PulseAudioMixerProvider.java:
	      same
	    * src/java/org/classpath/icedtea/pulseaudio/PulseAudioMuteControl.java:
	      same
	    * src/java/org/classpath/icedtea/pulseaudio/PulseAudioSourceDataLine.java:
	      merged file
	    * src/java/org/classpath/icedtea/pulseaudio/PulseAudioStreamMuteControl.java:
	      spacing changes
	    * src/java/org/classpath/icedtea/pulseaudio/PulseAudioStreamVolumeControl.java:
	      same
	    * src/java/org/classpath/icedtea/pulseaudio/PulseAudioVolumeContro.java:
	      same
	    * src/java/org/classpath/icedtea/pulseaudio/PulseAudioRecorder.java:
	      merged file
	    * src/native/Makefile.am: merged file

diffstat:

20 files changed, 959 insertions(+), 599 deletions(-)
ChangeLog                                                                    |   21 
build.xml                                                                    |    1 
src/java/org/classpath/icedtea/pulseaudio/ContextEvent.java                  |    2 
src/java/org/classpath/icedtea/pulseaudio/ContextListener.java               |    4 
src/java/org/classpath/icedtea/pulseaudio/EventLoop.java                     |   18 
src/java/org/classpath/icedtea/pulseaudio/Operation.java                     |   15 
src/java/org/classpath/icedtea/pulseaudio/PulseAudioClip.java                |    3 
src/java/org/classpath/icedtea/pulseaudio/PulseAudioMixer.java               |  379 +++++++---
src/java/org/classpath/icedtea/pulseaudio/PulseAudioMixerInfo.java           |    3 
src/java/org/classpath/icedtea/pulseaudio/PulseAudioMixerProvider.java       |    2 
src/java/org/classpath/icedtea/pulseaudio/PulseAudioMuteControl.java         |    4 
src/java/org/classpath/icedtea/pulseaudio/PulseAudioSourceDataLine.java      |  227 -----
src/java/org/classpath/icedtea/pulseaudio/PulseAudioStreamMuteControl.java   |   14 
src/java/org/classpath/icedtea/pulseaudio/PulseAudioStreamVolumeControl.java |   35 
src/java/org/classpath/icedtea/pulseaudio/PulseAudioTargetDataLine.java      |  278 ++++++-
src/java/org/classpath/icedtea/pulseaudio/PulseAudioVolumeControl.java       |   10 
src/java/org/classpath/icedtea/pulseaudio/SimpleAudioRecorder.java           |  208 +++++
src/native/Makefile.am                                                       |    2 
src/native/org_classpath_icedtea_pulseaudio_Operation.c                      |    2 
src/native/org_classpath_icedtea_pulseaudio_PulseAudioTargetDataLine.c       |  330 +++-----

diffs (truncated from 2120 to 500 lines):

diff -r 2e3dd470d097 -r 3cf90f7c6931 ChangeLog
--- a/ChangeLog	Fri Aug 15 10:51:54 2008 -0400
+++ b/ChangeLog	Fri Aug 15 12:03:05 2008 -0400
@@ -1,3 +1,24 @@ 2008-08-11  Joshua Sumali  <jsumali at redh
+2008-08-13 Ioana Ivan  <iivan at redhat.com>
+
+        * src/java/org/classpath/icedtea/pulseaudio/PulseAudioMixer.java:
+	modified getLine() to allow us to obtain a TargetDataLine
+	* src/java/org/classpath/icedtea/pulseaudio/PulseAudioTargetDataLine.java: 
+	added open(), read() and close()
+	* src/java/org/classpath/icedtea/pulseaudio/PulseAudioSourceDataLine.java
+	marged the streamListeners into a single listener
+
+
+
+2008-08-13 Ioana Ivan  <iivan at redhat.com>
+
+        * src/java/org/classpath/icedtea/pulseaudio/PulseAudioMixer.java: the
+	list of formats recognized by PulseAudio is being set here, so it can
+	be used by all DataLines. Also made some changes to get*LineInfo() and
+	isLineSupported()
+	* src/java/org/classpath/icedtea/pulseaudio/PulseAudioSourceDataLine.java:
+	changed the constructor 
+
+
 2008-08-11  Joshua Sumali  <jsumali at redhat.com>
 
 	* configure.ac: Add check for libpulse.
diff -r 2e3dd470d097 -r 3cf90f7c6931 build.xml
--- a/build.xml	Fri Aug 15 10:51:54 2008 -0400
+++ b/build.xml	Fri Aug 15 12:03:05 2008 -0400
@@ -38,6 +38,7 @@
 			<class name="org.classpath.icedtea.pulseaudio.EventLoop"/>
 			<class name="org.classpath.icedtea.pulseaudio.Operation"/>
 			<class name="org.classpath.icedtea.pulseaudio.Stream"/>
+			<class name="org.classpath.icedtea.pulseaudio.PulseAudioTargetDataLine"/>
 			<class name="org.classpath.icedtea.pulseaudio.PulseAudioStreamVolumeControl"/>
 		</javah>
 	</target>
diff -r 2e3dd470d097 -r 3cf90f7c6931 src/java/org/classpath/icedtea/pulseaudio/ContextEvent.java
--- a/src/java/org/classpath/icedtea/pulseaudio/ContextEvent.java	Fri Aug 15 10:51:54 2008 -0400
+++ b/src/java/org/classpath/icedtea/pulseaudio/ContextEvent.java	Fri Aug 15 12:03:05 2008 -0400
@@ -33,7 +33,7 @@ this exception to your version of the li
 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;
 
diff -r 2e3dd470d097 -r 3cf90f7c6931 src/java/org/classpath/icedtea/pulseaudio/ContextListener.java
--- a/src/java/org/classpath/icedtea/pulseaudio/ContextListener.java	Fri Aug 15 10:51:54 2008 -0400
+++ b/src/java/org/classpath/icedtea/pulseaudio/ContextListener.java	Fri Aug 15 12:03:05 2008 -0400
@@ -33,12 +33,12 @@ this exception to your version of the li
 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;
 
 public interface ContextListener {
 
 	public void update(ContextEvent e);
-	
+
 }
diff -r 2e3dd470d097 -r 3cf90f7c6931 src/java/org/classpath/icedtea/pulseaudio/EventLoop.java
--- a/src/java/org/classpath/icedtea/pulseaudio/EventLoop.java	Fri Aug 15 10:51:54 2008 -0400
+++ b/src/java/org/classpath/icedtea/pulseaudio/EventLoop.java	Fri Aug 15 12:03:05 2008 -0400
@@ -105,8 +105,8 @@ public class EventLoop implements Runnab
 	static {
 		try {
 			String library = new java.io.File(".").getCanonicalPath()
-					+ java.io.File.separatorChar
-					+ System.mapLibraryName("pulse-java");
+
+			+ java.io.File.separatorChar + System.mapLibraryName("pulse-java");
 			System.out.println(library);
 			System.load(library);
 		} catch (IOException e) {
@@ -154,12 +154,12 @@ public class EventLoop implements Runnab
 					native_shutdown();
 					// System.out.println(this.getClass().getName()
 					// + ": shutting down");
-					
+
 					// clean up the listeners
 					synchronized (contextListeners) {
 						contextListeners.clear();
 					}
-					
+
 					return;
 
 				}
@@ -216,11 +216,11 @@ public class EventLoop implements Runnab
 	}
 
 	private void fireEvent(final ContextEvent e) {
-//		System.out.println(this.getClass().getName() + "firing event: "
-//				+ e.getType().toString());
-
-		synchronized (contextListeners) {
-//			System.out.println(contextListeners.size());
+		// System.out.println(this.getClass().getName() + "firing event: "
+		// + e.getType().toString());
+
+		synchronized (contextListeners) {
+			// System.out.println(contextListeners.size());
 			for (ContextListener listener : contextListeners) {
 				listener.update(e);
 			}
diff -r 2e3dd470d097 -r 3cf90f7c6931 src/java/org/classpath/icedtea/pulseaudio/Operation.java
--- a/src/java/org/classpath/icedtea/pulseaudio/Operation.java	Fri Aug 15 10:51:54 2008 -0400
+++ b/src/java/org/classpath/icedtea/pulseaudio/Operation.java	Fri Aug 15 12:03:05 2008 -0400
@@ -33,7 +33,7 @@ this exception to your version of the li
 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;
 
@@ -43,7 +43,8 @@ import java.io.IOException;
  * Encapsulates a pa_operation object
  * 
  * 
- * This is really needed only so that we can deallocate the reference counted object
+ * This is really needed only so that we can deallocate the reference counted
+ * object
  * 
  * 
  */
@@ -100,9 +101,9 @@ public class Operation {
 		}
 		return false;
 	}
-	
+
 	public State getState() {
-		assert (operationPointer != 0); 
+		assert (operationPointer != 0);
 		int state;
 		synchronized (eventLoop.threadLock) {
 			state = native_get_state();
@@ -120,11 +121,11 @@ public class Operation {
 
 	}
 
-	public void waitForCompletion()  throws InterruptedException {
-		
+	public void waitForCompletion() throws InterruptedException {
+
 		synchronized (eventLoop.threadLock) {
 			eventLoop.threadLock.wait();
 		}
-		
+
 	}
 }
diff -r 2e3dd470d097 -r 3cf90f7c6931 src/java/org/classpath/icedtea/pulseaudio/PulseAudioClip.java
--- a/src/java/org/classpath/icedtea/pulseaudio/PulseAudioClip.java	Fri Aug 15 10:51:54 2008 -0400
+++ b/src/java/org/classpath/icedtea/pulseaudio/PulseAudioClip.java	Fri Aug 15 12:03:05 2008 -0400
@@ -234,8 +234,7 @@ public class PulseAudioClip implements C
 	@Override
 	public void open(AudioFormat format, byte[] data, int offset, int bufferSize)
 			throws LineUnavailableException {
-		
-		
+
 		long contextPointer = EventLoop.getEventLoop().getContextPointer();
 		int channels = 2;
 		int sampleRate = 22050;
diff -r 2e3dd470d097 -r 3cf90f7c6931 src/java/org/classpath/icedtea/pulseaudio/PulseAudioMixer.java
--- a/src/java/org/classpath/icedtea/pulseaudio/PulseAudioMixer.java	Fri Aug 15 10:51:54 2008 -0400
+++ b/src/java/org/classpath/icedtea/pulseaudio/PulseAudioMixer.java	Fri Aug 15 12:03:05 2008 -0400
@@ -41,7 +41,10 @@ import java.net.InetAddress;
 import java.net.InetAddress;
 import java.net.UnknownHostException;
 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;
@@ -57,6 +60,8 @@ import javax.sound.sampled.LineUnavailab
 import javax.sound.sampled.LineUnavailableException;
 import javax.sound.sampled.Mixer;
 import javax.sound.sampled.SourceDataLine;
+import javax.sound.sampled.TargetDataLine;
+import javax.sound.sampled.AudioFormat.Encoding;
 import javax.sound.sampled.Control.Type;
 
 public class PulseAudioMixer implements javax.sound.sampled.Mixer {
@@ -65,16 +70,20 @@ public class PulseAudioMixer implements 
 	public EventLoop eventLoop;
 	public Thread eventLoopThread;
 
+	private Line.Info[] sourceLineInfos;
+	private Line.Info[] targetLineInfos;
+
 	private static PulseAudioMixer _instance = null;
 
 	private static final String DEFAULT_APP_NAME = "Java App";
+	private static final String PULSEAUDIO_FORMAT_KEY = "PulseAudioFormatKey";
 
 	private boolean isOpen = false;
 
 	private List<PulseAudioSourceDataLine> sourceLines = new ArrayList<PulseAudioSourceDataLine>();
 	// private List<PulseAudioTargetDataLine> targetLines = null;
 	private List<PulseAudioClip> clips = new ArrayList<PulseAudioClip>();
-	
+
 	// private Line.Info targetDataLineInfo = new
 	// Line.Info(PulseAudioTargetDataLine.class);
 
@@ -83,6 +92,13 @@ public class PulseAudioMixer implements 
 	private PulseAudioMixer() {
 		lineListeners = new ArrayList<LineListener>();
 		sourceLines = new ArrayList<PulseAudioSourceDataLine>();
+		AudioFormat[] formats = getSupportedFormats();
+		sourceLineInfos = new Line.Info[] { new DataLine.Info(
+				SourceDataLine.class, formats, AudioSystem.NOT_SPECIFIED,
+				AudioSystem.NOT_SPECIFIED) };
+		targetLineInfos = new Line.Info[] { new DataLine.Info(
+				TargetDataLine.class, formats, AudioSystem.NOT_SPECIFIED,
+				AudioSystem.NOT_SPECIFIED) };
 		// _targetLines = new ArrayList<PulseAudioTargetDataLine>();
 	}
 
@@ -93,6 +109,173 @@ public class PulseAudioMixer implements 
 		return _instance;
 	}
 
+	private AudioFormat[] getSupportedFormats() {
+
+		List<AudioFormat> supportedFormats = new ArrayList<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);
+		}
+
+		return supportedFormats.toArray(new AudioFormat[0]);
+	}
+
 	@Override
 	public Line getLine(javax.sound.sampled.Line.Info info)
 			throws LineUnavailableException {
@@ -101,59 +284,54 @@ public class PulseAudioMixer implements 
 			throw new LineUnavailableException();
 		}
 
-		PulseAudioSourceDataLine sourceLine = null;
-		sourceLine = new PulseAudioSourceDataLine(eventLoop);
-		Line.Info sourceDataLineInfo = sourceLine.getLineInfo();
-
-		if (DataLine.class.isAssignableFrom(info.getLineClass())
-				&& info instanceof DataLine.Info) {
-			System.out
-					.println("DEBUG: trying to find data line with last matching format");
-
-			DataLine.Info dataLineInfo = (DataLine.Info) info;
-			// need to find the DataLine with the last valid format
-			AudioFormat[] wantedFormats = dataLineInfo.getFormats();
-			AudioFormat[] availableFormats = ((DataLine.Info) sourceDataLineInfo)
+		if (!isLineSupported(info)) {
+			throw new IllegalArgumentException("Line unsupported: " + info);
+		}
+
+		AudioFormat[] formats = null;
+		AudioFormat defaultFormat = null;
+
+		if (DataLine.Info.class.isInstance(info)) {
+			ArrayList<AudioFormat> formatList = new ArrayList<AudioFormat>();
+			AudioFormat[] requestedFormats = ((DataLine.Info) info)
 					.getFormats();
-			for (int i = wantedFormats.length - 1; i > -1; i--) {
-				for (AudioFormat aFormat : availableFormats) {
-					if (wantedFormats[i].matches(aFormat)) {
-						System.out.println("DEBUG: found a matching format");
-						System.out.println("wanted: " + wantedFormats[i]);
-						System.out.println("available: " + aFormat);
-						sourceLine.setDefaultFormat(wantedFormats[i]);
-						sourceLines.add(sourceLine);
-						return sourceLine;
+			for (int i = 0; i < requestedFormats.length; i++) {
+				AudioFormat f1 = requestedFormats[i];
+				for (AudioFormat f2 : getSupportedFormats()) {
+
+					if (f1.matches(f2)) {
+						formatList.add(f2);
+						defaultFormat = f1;
 					}
 				}
 			}
-
-			System.out.println("DEBUG: no matches found");
-			// no format matches, so return any line
-			throw new IllegalArgumentException("No matching format found");
-
-		}
-
-		if (info.matches(sourceDataLineInfo)) {
-			sourceLines.add(sourceLine);
-			return sourceLine;
-		}
-
-		// if (info.matches(_targetDataLineInfo)) {
-		// PulseAudioTargetDataLine targetLine = new PulseAudioTargetDataLine();
-		// _targetLines.add(targetLine);
-		// return targetLine;
-		// }
+			formats = formatList.toArray(new AudioFormat[0]);
+
+		} else {
+			formats = getSupportedFormats();
+			defaultFormat = new AudioFormat(Encoding.PCM_UNSIGNED, 22050, 8, 2,
+					2, AudioSystem.NOT_SPECIFIED, false);
+		}
+
+		if ((info.getLineClass() == SourceDataLine.class)) {
+			return new PulseAudioSourceDataLine(eventLoop, formats,
+					defaultFormat);
+		}
+
+		if ((info.getLineClass() == TargetDataLine.class)) {
+			return new PulseAudioTargetDataLine(eventLoop, formats,
+					defaultFormat);
+		}
 
 		PulseAudioClip clip = new PulseAudioClip();
-		
+
 		if (info.matches(clip.getLineInfo())) {
 			clips.add(clip);
 			return clip;
 		}
-		



More information about the distro-pkg-dev mailing list