Package davaguine.jmac.spi

Source Code of davaguine.jmac.spi.APEAudioInputStream

/*
*  21.04.2004 Original verion. davagin@udm.ru.
*-----------------------------------------------------------------------
*  This program 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; either version 2 of the License, or
*  (at your option) any later version.
*
*  This program 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 this program; if not, write to the Free Software
*  Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
*----------------------------------------------------------------------
*/

package davaguine.jmac.spi;

import davaguine.jmac.decoder.IAPEDecompress;
import davaguine.jmac.tools.File;
import davaguine.jmac.tools.InputStreamFile;

import javax.sound.sampled.AudioFormat;
import javax.sound.sampled.AudioInputStream;
import javax.sound.sampled.AudioSystem;
import java.io.IOException;
import java.io.InputStream;

/**
* Author: Dmitry Vaguine
* Date: 12.03.2004
* Time: 13:35:13
*/

/**
* Decoded APE audio input stream.
*/
public class APEAudioInputStream extends AudioInputStream {
    private final static int BLOCKS_PER_DECODE = 9216;
    private IAPEDecompress m_decoder = null;
    private File file = null;
    private byte[] buffer = null;
    private int nBlocksLeft;
    private int blockAlign;
    private int pos = 0;
    private int size = 0;

    /**
     * Constructs an audio input stream that has the requested format, using audio data
     * from the specified input stream.
     *
     * @param format - the format of this stream's audio data
     * @param stream - the stream on which this APEAudioInputStream object is based
     */
    public APEAudioInputStream(AudioFormat format, InputStream stream) {
        super(stream, format, AudioSystem.NOT_SPECIFIED);
        try {
            file = new InputStreamFile(stream);
            m_decoder = IAPEDecompress.CreateIAPEDecompress(file);

            nBlocksLeft = m_decoder.getApeInfoDecompressTotalBlocks();
            blockAlign = m_decoder.getApeInfoBlockAlign();

            // allocate space for decompression
            buffer = new byte[blockAlign * BLOCKS_PER_DECODE];
        } catch (IOException e) {
            e.printStackTrace();
        }
    }

    private void ensureOpen() throws IOException {
        if (m_decoder == null)
            throw new IOException("Stream closed");
    }

    private void fill() throws IOException {
        pos = 0;
        if (nBlocksLeft > 0) {
            int nBlocksDecoded = m_decoder.GetData(buffer, BLOCKS_PER_DECODE);

            nBlocksLeft -= nBlocksDecoded;
            size = nBlocksDecoded * blockAlign;
        } else
            size = 0;
    }

    /**
     * Reads the next byte of data from the audio input stream.
     *
     * @return the next byte of data, or -1 if the end of the stream is reached
     * @throws IOException - if an input or output error occurs
     */
    public synchronized int read() throws IOException {
        ensureOpen();
        if (pos < size)
            return buffer[pos++] & 0xff;
        else {
            fill();
            return pos < size ? (buffer[pos++] & 0xff) : -1;
        }
    }

    /**
     * Reads up to a specified maximum number of bytes of data from the audio stream, putting
     * them into the given byte array.
     *
     * @param b   - the buffer into which the data is read
     * @param off - the offset, from the beginning of array b, at which the data will be written
     * @param len - the maximum number of bytes to read
     * @return the total number of bytes read into the buffer, or -1 if there is no more data because the end of the stream has been reached
     * @throws IOException if an input or output error occurs
     */
    public synchronized int read(byte[] b, int off, int len) throws IOException {
        ensureOpen();
        if (b == null) {
            throw new NullPointerException();
        } else if ((off < 0) || (off > b.length) || (len < 0) ||
                ((off + len) > b.length) || ((off + len) < 0)) {
            throw new IndexOutOfBoundsException();
        }
        if (pos >= size)
            fill();
        if (pos >= size)
            return -1;
        if (pos + len < size) {
            System.arraycopy(buffer, pos, b, off, len);
            pos += len;
            return len;
        }
        int avail = size - pos;
        System.arraycopy(buffer, pos, b, off, avail);
        fill();
        if (pos >= size)
            return avail;
        if (pos + len - avail < size) {
            System.arraycopy(buffer, pos, b, off + avail, len - avail);
            pos += (len - avail);
            return len;
        }
        len = size - pos;
        System.arraycopy(buffer, pos, b, off + avail, len);
        pos += len;
        return len + avail;
    }

    /**
     * Skips over and discards a specified number of bytes from this audio input stream.
     *
     * @param n - the requested number of bytes to be skipped
     * @return the actual number of bytes skipped
     * @throws IOException - if an input or output error occurs
     */
    public synchronized long skip(long n) throws IOException {
        ensureOpen();
        if (n <= 0) {
            return 0;
        }
        if (pos >= size)
            fill();
        if (pos >= size)
            return 0;
        if (pos + n < size) {
            pos += n;
            return n;
        }
        int avail = size - pos;
        fill();
        if (pos >= size)
            return avail;
        if (pos + n - avail < size) {
            pos += (n - avail);
            return n;
        }
        n = size - pos;
        pos += n;
        return n + avail;
    }

    /**
     * Returns the maximum number of bytes that can be read (or skipped over) from this
     * audio input stream without blocking. This limit applies only to the next invocation
     * of a read or skip method for this audio input stream; the limit can vary each time
     * these methods are invoked. Depending on the underlying stream,an IOException may be
     * thrown if this stream is closed.
     *
     * @return the number of bytes that can be read from this audio input stream without blocking
     * @throws IOException - if an input or output error occurs
     */
    public int available() throws IOException {
        ensureOpen();
        return size - pos;
    }

    /**
     * Tests whether this audio input stream supports the mark and reset methods.
     * This method always returns false.
     *
     * @return returns false.
     */
    public boolean markSupported() {
        return false;
    }

    /**
     * Marks the current position in this audio input stream. This method does nothing.
     *
     * @param readlimit - the maximum number of bytes that can be read before the mark position becomes invalid.
     */
    public void mark(int readlimit) {
    }

    /**
     * Repositions this audio input stream to the position it had at the time its mark method was last invoked.
     * This method always throws IOException since this stream doesn't support mark feature.
     *
     * @throws IOException - if an input or output error occurs.
     */
    public void reset() throws IOException {
        throw new IOException("mark not supported");
    }

    /**
     * Closes this audio input stream and releases any system resources associated with the stream.
     *
     * @throws IOException - if an input or output error occurs
     */
    public void close() throws IOException {
        if (file == null)
            return;
        file.close();
        file = null;
        m_decoder = null;
        buffer = null;
    }

}
TOP

Related Classes of davaguine.jmac.spi.APEAudioInputStream

TOP
Copyright © 2018 www.massapi.com. All rights reserved.
All source code are property of their respective owners. Java is a trademark of Sun Microsystems, Inc and owned by ORACLE Inc. Contact coftware#gmail.com.