| /* |
| * Copyright (c) 1999, 2013, Oracle and/or its affiliates. All rights reserved. |
| * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. |
| * |
| * This code is free software; you can redistribute it and/or modify it |
| * under the terms of the GNU General Public License version 2 only, as |
| * published by the Free Software Foundation. Oracle designates this |
| * particular file as subject to the "Classpath" exception as provided |
| * by Oracle in the LICENSE file that accompanied this code. |
| * |
| * This code 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 |
| * version 2 for more details (a copy is included in the LICENSE file that |
| * accompanied this code). |
| * |
| * You should have received a copy of the GNU General Public License version |
| * 2 along with this work; if not, write to the Free Software Foundation, |
| * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. |
| * |
| * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA |
| * or visit www.oracle.com if you need additional information or have any |
| * questions. |
| */ |
| |
| package com.sun.media.sound; |
| |
| import javax.sound.sampled.AudioFormat; |
| import javax.sound.sampled.AudioSystem; |
| import javax.sound.sampled.Control; |
| import javax.sound.sampled.DataLine; |
| import javax.sound.sampled.LineEvent; |
| import javax.sound.sampled.LineUnavailableException; |
| |
| |
| /** |
| * AbstractDataLine |
| * |
| * @author Kara Kytle |
| */ |
| abstract class AbstractDataLine extends AbstractLine implements DataLine { |
| |
| // DEFAULTS |
| |
| // default format |
| private final AudioFormat defaultFormat; |
| |
| // default buffer size in bytes |
| private final int defaultBufferSize; |
| |
| // the lock for synchronization |
| protected final Object lock = new Object(); |
| |
| // STATE |
| |
| // current format |
| protected AudioFormat format; |
| |
| // current buffer size in bytes |
| protected int bufferSize; |
| |
| protected boolean running = false; |
| private boolean started = false; |
| private boolean active = false; |
| |
| |
| /** |
| * Constructs a new AbstractLine. |
| */ |
| protected AbstractDataLine(DataLine.Info info, AbstractMixer mixer, Control[] controls) { |
| this(info, mixer, controls, null, AudioSystem.NOT_SPECIFIED); |
| } |
| |
| /** |
| * Constructs a new AbstractLine. |
| */ |
| protected AbstractDataLine(DataLine.Info info, AbstractMixer mixer, Control[] controls, AudioFormat format, int bufferSize) { |
| |
| super(info, mixer, controls); |
| |
| // record the default values |
| if (format != null) { |
| defaultFormat = format; |
| } else { |
| // default CD-quality |
| defaultFormat = new AudioFormat(44100.0f, 16, 2, true, Platform.isBigEndian()); |
| } |
| if (bufferSize > 0) { |
| defaultBufferSize = bufferSize; |
| } else { |
| // 0.5 seconds buffer |
| defaultBufferSize = ((int) (defaultFormat.getFrameRate() / 2)) * defaultFormat.getFrameSize(); |
| } |
| |
| // set the initial values to the defaults |
| this.format = defaultFormat; |
| this.bufferSize = defaultBufferSize; |
| } |
| |
| |
| // DATA LINE METHODS |
| |
| public final void open(AudioFormat format, int bufferSize) throws LineUnavailableException { |
| //$$fb 2001-10-09: Bug #4517739: avoiding deadlock by synchronizing to mixer ! |
| synchronized (mixer) { |
| if (Printer.trace) Printer.trace("> AbstractDataLine.open(format, bufferSize) (class: "+getClass().getName()); |
| |
| // if the line is not currently open, try to open it with this format and buffer size |
| if (!isOpen()) { |
| // make sure that the format is specified correctly |
| // $$fb part of fix for 4679187: Clip.open() throws unexpected Exceptions |
| Toolkit.isFullySpecifiedAudioFormat(format); |
| |
| if (Printer.debug) Printer.debug(" need to open the mixer..."); |
| // reserve mixer resources for this line |
| //mixer.open(this, format, bufferSize); |
| mixer.open(this); |
| |
| try { |
| // open the data line. may throw LineUnavailableException. |
| implOpen(format, bufferSize); |
| |
| // if we succeeded, set the open state to true and send events |
| setOpen(true); |
| |
| } catch (LineUnavailableException e) { |
| // release mixer resources for this line and then throw the exception |
| mixer.close(this); |
| throw e; |
| } |
| } else { |
| if (Printer.debug) Printer.debug(" dataline already open"); |
| |
| // if the line is already open and the requested format differs from the |
| // current settings, throw an IllegalStateException |
| //$$fb 2002-04-02: fix for 4661602: Buffersize is checked when re-opening line |
| if (!format.matches(getFormat())) { |
| throw new IllegalStateException("Line is already open with format " + getFormat() + |
| " and bufferSize " + getBufferSize()); |
| } |
| //$$fb 2002-07-26: allow changing the buffersize of already open lines |
| if (bufferSize > 0) { |
| setBufferSize(bufferSize); |
| } |
| } |
| |
| if (Printer.trace) Printer.trace("< AbstractDataLine.open(format, bufferSize) completed"); |
| } |
| } |
| |
| |
| public final void open(AudioFormat format) throws LineUnavailableException { |
| open(format, AudioSystem.NOT_SPECIFIED); |
| } |
| |
| |
| /** |
| * This implementation always returns 0. |
| */ |
| public int available() { |
| return 0; |
| } |
| |
| |
| /** |
| * This implementation does nothing. |
| */ |
| public void drain() { |
| if (Printer.trace) Printer.trace("AbstractDataLine: drain"); |
| } |
| |
| |
| /** |
| * This implementation does nothing. |
| */ |
| public void flush() { |
| if (Printer.trace) Printer.trace("AbstractDataLine: flush"); |
| } |
| |
| |
| public final void start() { |
| //$$fb 2001-10-09: Bug #4517739: avoiding deadlock by synchronizing to mixer ! |
| synchronized(mixer) { |
| if (Printer.trace) Printer.trace("> "+getClass().getName()+".start() - AbstractDataLine"); |
| |
| // $$kk: 06.06.99: if not open, this doesn't work....??? |
| if (isOpen()) { |
| |
| if (!isStartedRunning()) { |
| mixer.start(this); |
| implStart(); |
| running = true; |
| } |
| } |
| } |
| |
| synchronized(lock) { |
| lock.notifyAll(); |
| } |
| |
| if (Printer.trace) Printer.trace("< "+getClass().getName()+".start() - AbstractDataLine"); |
| } |
| |
| |
| public final void stop() { |
| |
| //$$fb 2001-10-09: Bug #4517739: avoiding deadlock by synchronizing to mixer ! |
| synchronized(mixer) { |
| if (Printer.trace) Printer.trace("> "+getClass().getName()+".stop() - AbstractDataLine"); |
| |
| // $$kk: 06.06.99: if not open, this doesn't work. |
| if (isOpen()) { |
| |
| if (isStartedRunning()) { |
| |
| implStop(); |
| mixer.stop(this); |
| |
| running = false; |
| |
| // $$kk: 11.10.99: this is not exactly correct, but will probably work |
| if (started && (!isActive())) { |
| setStarted(false); |
| } |
| } |
| } |
| } |
| |
| synchronized(lock) { |
| lock.notifyAll(); |
| } |
| |
| if (Printer.trace) Printer.trace("< "+getClass().getName()+".stop() - AbstractDataLine"); |
| } |
| |
| // $$jb: 12.10.99: The official API for this is isRunning(). |
| // Per the denied RFE 4297981, |
| // the change to isStarted() is technically an unapproved API change. |
| // The 'started' variable is false when playback of data stops. |
| // It is changed throughout the implementation with setStarted(). |
| // This state is what should be returned by isRunning() in the API. |
| // Note that the 'running' variable is true between calls to |
| // start() and stop(). This state is accessed now through the |
| // isStartedRunning() method, defined below. I have not changed |
| // the variable names at this point, since 'running' is accessed |
| // in MixerSourceLine and MixerClip, and I want to touch as little |
| // code as possible to change isStarted() back to isRunning(). |
| |
| public final boolean isRunning() { |
| return started; |
| } |
| |
| public final boolean isActive() { |
| return active; |
| } |
| |
| |
| public final long getMicrosecondPosition() { |
| |
| long microseconds = getLongFramePosition(); |
| if (microseconds != AudioSystem.NOT_SPECIFIED) { |
| microseconds = Toolkit.frames2micros(getFormat(), microseconds); |
| } |
| return microseconds; |
| } |
| |
| |
| public final AudioFormat getFormat() { |
| return format; |
| } |
| |
| |
| public final int getBufferSize() { |
| return bufferSize; |
| } |
| |
| /** |
| * This implementation does NOT change the buffer size |
| */ |
| public final int setBufferSize(int newSize) { |
| return getBufferSize(); |
| } |
| |
| /** |
| * This implementation returns AudioSystem.NOT_SPECIFIED. |
| */ |
| public final float getLevel() { |
| return (float)AudioSystem.NOT_SPECIFIED; |
| } |
| |
| |
| // HELPER METHODS |
| |
| /** |
| * running is true after start is called and before stop is called, |
| * regardless of whether data is actually being presented. |
| */ |
| // $$jb: 12.10.99: calling this method isRunning() conflicts with |
| // the official API that was once called isStarted(). Since we |
| // use this method throughout the implementation, I am renaming |
| // it to isStartedRunning(). This is part of backing out the |
| // change denied in RFE 4297981. |
| |
| final boolean isStartedRunning() { |
| return running; |
| } |
| |
| /** |
| * This method sets the active state and generates |
| * events if it changes. |
| */ |
| final void setActive(boolean active) { |
| |
| if (Printer.trace) Printer.trace("> AbstractDataLine: setActive(" + active + ")"); |
| |
| //boolean sendEvents = false; |
| //long position = getLongFramePosition(); |
| |
| synchronized (this) { |
| |
| //if (Printer.debug) Printer.debug(" AbstractDataLine: setActive: this.active: " + this.active); |
| //if (Printer.debug) Printer.debug(" active: " + active); |
| |
| if (this.active != active) { |
| this.active = active; |
| //sendEvents = true; |
| } |
| } |
| |
| //if (Printer.debug) Printer.debug(" this.active: " + this.active); |
| //if (Printer.debug) Printer.debug(" sendEvents: " + sendEvents); |
| |
| |
| // $$kk: 11.19.99: take ACTIVE / INACTIVE / EOM events out; |
| // putting them in is technically an API change. |
| // do not generate ACTIVE / INACTIVE events for now |
| // if (sendEvents) { |
| // |
| // if (active) { |
| // sendEvents(new LineEvent(this, LineEvent.Type.ACTIVE, position)); |
| // } else { |
| // sendEvents(new LineEvent(this, LineEvent.Type.INACTIVE, position)); |
| // } |
| //} |
| } |
| |
| /** |
| * This method sets the started state and generates |
| * events if it changes. |
| */ |
| final void setStarted(boolean started) { |
| |
| if (Printer.trace) Printer.trace("> AbstractDataLine: setStarted(" + started + ")"); |
| |
| boolean sendEvents = false; |
| long position = getLongFramePosition(); |
| |
| synchronized (this) { |
| |
| //if (Printer.debug) Printer.debug(" AbstractDataLine: setStarted: this.started: " + this.started); |
| //if (Printer.debug) Printer.debug(" started: " + started); |
| |
| if (this.started != started) { |
| this.started = started; |
| sendEvents = true; |
| } |
| } |
| |
| //if (Printer.debug) Printer.debug(" this.started: " + this.started); |
| //if (Printer.debug) Printer.debug(" sendEvents: " + sendEvents); |
| |
| if (sendEvents) { |
| |
| if (started) { |
| sendEvents(new LineEvent(this, LineEvent.Type.START, position)); |
| } else { |
| sendEvents(new LineEvent(this, LineEvent.Type.STOP, position)); |
| } |
| } |
| if (Printer.trace) Printer.trace("< AbstractDataLine: setStarted completed"); |
| } |
| |
| |
| /** |
| * This method generates a STOP event and sets the started state to false. |
| * It is here for historic reasons when an EOM event existed. |
| */ |
| final void setEOM() { |
| |
| if (Printer.trace) Printer.trace("> AbstractDataLine: setEOM()"); |
| //$$fb 2002-04-21: sometimes, 2 STOP events are generated. |
| // better use setStarted() to send STOP event. |
| setStarted(false); |
| if (Printer.trace) Printer.trace("< AbstractDataLine: setEOM() completed"); |
| } |
| |
| |
| |
| |
| // OVERRIDES OF ABSTRACT LINE METHODS |
| |
| /** |
| * Try to open the line with the current format and buffer size values. |
| * If the line is not open, these will be the defaults. If the |
| * line is open, this should return quietly because the values |
| * requested will match the current ones. |
| */ |
| public final void open() throws LineUnavailableException { |
| |
| if (Printer.trace) Printer.trace("> "+getClass().getName()+".open() - AbstractDataLine"); |
| |
| // this may throw a LineUnavailableException. |
| open(format, bufferSize); |
| if (Printer.trace) Printer.trace("< "+getClass().getName()+".open() - AbstractDataLine"); |
| } |
| |
| |
| /** |
| * This should also stop the line. The closed line should not be running or active. |
| * After we close the line, we reset the format and buffer size to the defaults. |
| */ |
| public final void close() { |
| //$$fb 2001-10-09: Bug #4517739: avoiding deadlock by synchronizing to mixer ! |
| synchronized (mixer) { |
| if (Printer.trace) Printer.trace("> "+getClass().getName()+".close() - in AbstractDataLine."); |
| |
| if (isOpen()) { |
| |
| // stop |
| stop(); |
| |
| // set the open state to false and send events |
| setOpen(false); |
| |
| // close resources for this line |
| implClose(); |
| |
| // release mixer resources for this line |
| mixer.close(this); |
| |
| // reset format and buffer size to the defaults |
| format = defaultFormat; |
| bufferSize = defaultBufferSize; |
| } |
| } |
| if (Printer.trace) Printer.trace("< "+getClass().getName()+".close() - in AbstractDataLine"); |
| } |
| |
| |
| // IMPLEMENTATIONS OF ABSTRACT LINE ABSTRACE METHODS |
| |
| |
| // ABSTRACT METHODS |
| |
| abstract void implOpen(AudioFormat format, int bufferSize) throws LineUnavailableException; |
| abstract void implClose(); |
| |
| abstract void implStart(); |
| abstract void implStop(); |
| } |