Merge "Revert "Remove AmrInputStream""
am: e2d8581529
Change-Id: I55615b328bedf67819d076ccb21b9bc1f13a07a8
This commit is contained in:
195
media/java/android/media/AmrInputStream.java
Normal file
195
media/java/android/media/AmrInputStream.java
Normal file
@@ -0,0 +1,195 @@
|
||||
/*
|
||||
* Copyright (C) 2008 The Android Open Source Project
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
package android.media;
|
||||
|
||||
import java.io.InputStream;
|
||||
import java.io.IOException;
|
||||
import java.nio.ByteBuffer;
|
||||
|
||||
import android.media.MediaCodec.BufferInfo;
|
||||
import android.util.Log;
|
||||
|
||||
|
||||
/**
|
||||
* AmrInputStream
|
||||
* @hide
|
||||
*/
|
||||
public final class AmrInputStream extends InputStream {
|
||||
private final static String TAG = "AmrInputStream";
|
||||
|
||||
// frame is 20 msec at 8.000 khz
|
||||
private final static int SAMPLES_PER_FRAME = 8000 * 20 / 1000;
|
||||
|
||||
MediaCodec mCodec;
|
||||
BufferInfo mInfo;
|
||||
boolean mSawOutputEOS;
|
||||
boolean mSawInputEOS;
|
||||
|
||||
// pcm input stream
|
||||
private InputStream mInputStream;
|
||||
|
||||
// result amr stream
|
||||
private final byte[] mBuf = new byte[SAMPLES_PER_FRAME * 2];
|
||||
private int mBufIn = 0;
|
||||
private int mBufOut = 0;
|
||||
|
||||
// helper for bytewise read()
|
||||
private byte[] mOneByte = new byte[1];
|
||||
|
||||
/**
|
||||
* Create a new AmrInputStream, which converts 16 bit PCM to AMR
|
||||
* @param inputStream InputStream containing 16 bit PCM.
|
||||
*/
|
||||
public AmrInputStream(InputStream inputStream) {
|
||||
mInputStream = inputStream;
|
||||
|
||||
MediaFormat format = new MediaFormat();
|
||||
format.setString(MediaFormat.KEY_MIME, MediaFormat.MIMETYPE_AUDIO_AMR_NB);
|
||||
format.setInteger(MediaFormat.KEY_SAMPLE_RATE, 8000);
|
||||
format.setInteger(MediaFormat.KEY_CHANNEL_COUNT, 1);
|
||||
format.setInteger(MediaFormat.KEY_BIT_RATE, 12200);
|
||||
|
||||
MediaCodecList mcl = new MediaCodecList(MediaCodecList.REGULAR_CODECS);
|
||||
String name = mcl.findEncoderForFormat(format);
|
||||
if (name != null) {
|
||||
try {
|
||||
mCodec = MediaCodec.createByCodecName(name);
|
||||
mCodec.configure(format,
|
||||
null /* surface */,
|
||||
null /* crypto */,
|
||||
MediaCodec.CONFIGURE_FLAG_ENCODE);
|
||||
mCodec.start();
|
||||
} catch (IOException e) {
|
||||
if (mCodec != null) {
|
||||
mCodec.release();
|
||||
}
|
||||
mCodec = null;
|
||||
}
|
||||
}
|
||||
mInfo = new BufferInfo();
|
||||
}
|
||||
|
||||
@Override
|
||||
public int read() throws IOException {
|
||||
int rtn = read(mOneByte, 0, 1);
|
||||
return rtn == 1 ? (0xff & mOneByte[0]) : -1;
|
||||
}
|
||||
|
||||
@Override
|
||||
public int read(byte[] b) throws IOException {
|
||||
return read(b, 0, b.length);
|
||||
}
|
||||
|
||||
@Override
|
||||
public int read(byte[] b, int offset, int length) throws IOException {
|
||||
if (mCodec == null) {
|
||||
throw new IllegalStateException("not open");
|
||||
}
|
||||
|
||||
if (mBufOut >= mBufIn && !mSawOutputEOS) {
|
||||
// no data left in buffer, refill it
|
||||
mBufOut = 0;
|
||||
mBufIn = 0;
|
||||
|
||||
// first push as much data into the encoder as possible
|
||||
while (!mSawInputEOS) {
|
||||
int index = mCodec.dequeueInputBuffer(0);
|
||||
if (index < 0) {
|
||||
// no input buffer currently available
|
||||
break;
|
||||
} else {
|
||||
int numRead;
|
||||
for (numRead = 0; numRead < SAMPLES_PER_FRAME * 2; ) {
|
||||
int n = mInputStream.read(mBuf, numRead, SAMPLES_PER_FRAME * 2 - numRead);
|
||||
if (n == -1) {
|
||||
mSawInputEOS = true;
|
||||
break;
|
||||
}
|
||||
numRead += n;
|
||||
}
|
||||
ByteBuffer buf = mCodec.getInputBuffer(index);
|
||||
buf.put(mBuf, 0, numRead);
|
||||
mCodec.queueInputBuffer(index,
|
||||
0 /* offset */,
|
||||
numRead,
|
||||
0 /* presentationTimeUs */,
|
||||
mSawInputEOS ? MediaCodec.BUFFER_FLAG_END_OF_STREAM : 0 /* flags */);
|
||||
}
|
||||
}
|
||||
|
||||
// now read encoded data from the encoder (blocking, since we just filled up the
|
||||
// encoder's input with data it should be able to output at least one buffer)
|
||||
while (true) {
|
||||
int index = mCodec.dequeueOutputBuffer(mInfo, -1);
|
||||
if (index >= 0) {
|
||||
mBufIn = mInfo.size;
|
||||
ByteBuffer out = mCodec.getOutputBuffer(index);
|
||||
out.get(mBuf, 0 /* offset */, mBufIn /* length */);
|
||||
mCodec.releaseOutputBuffer(index, false /* render */);
|
||||
if ((mInfo.flags & MediaCodec.BUFFER_FLAG_END_OF_STREAM) != 0) {
|
||||
mSawOutputEOS = true;
|
||||
}
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (mBufOut < mBufIn) {
|
||||
// there is data in the buffer
|
||||
if (length > mBufIn - mBufOut) {
|
||||
length = mBufIn - mBufOut;
|
||||
}
|
||||
System.arraycopy(mBuf, mBufOut, b, offset, length);
|
||||
mBufOut += length;
|
||||
return length;
|
||||
}
|
||||
|
||||
if (mSawInputEOS && mSawOutputEOS) {
|
||||
// no more data available in buffer, codec or input stream
|
||||
return -1;
|
||||
}
|
||||
|
||||
// caller should try again
|
||||
return 0;
|
||||
}
|
||||
|
||||
@Override
|
||||
public void close() throws IOException {
|
||||
try {
|
||||
if (mInputStream != null) {
|
||||
mInputStream.close();
|
||||
}
|
||||
} finally {
|
||||
mInputStream = null;
|
||||
try {
|
||||
if (mCodec != null) {
|
||||
mCodec.release();
|
||||
}
|
||||
} finally {
|
||||
mCodec = null;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void finalize() throws Throwable {
|
||||
if (mCodec != null) {
|
||||
Log.w(TAG, "AmrInputStream wasn't closed");
|
||||
mCodec.release();
|
||||
}
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user