我試圖讓這個在Android 4.1上工作(使用升級的華碩變壓器平板電腦)。感謝Alex's response to my previous question,我已經能夠將一些原始的H.264數據寫入文件,但這個文件只能在ffplay -f h264
下播放,並且好像丟失了有關幀速率(極快的播放)的所有信息。此外,顏色空間看起來不正確(在編碼器側使用相機的默認值atm)。使用Android MediaCodec從相機編碼H.264
public class AvcEncoder {
private MediaCodec mediaCodec;
private BufferedOutputStream outputStream;
public AvcEncoder() {
File f = new File(Environment.getExternalStorageDirectory(), "Download/video_encoded.264");
touch (f);
try {
outputStream = new BufferedOutputStream(new FileOutputStream(f));
Log.i("AvcEncoder", "outputStream initialized");
} catch (Exception e){
e.printStackTrace();
}
mediaCodec = MediaCodec.createEncoderByType("video/avc");
MediaFormat mediaFormat = MediaFormat.createVideoFormat("video/avc", 320, 240);
mediaFormat.setInteger(MediaFormat.KEY_BIT_RATE, 125000);
mediaFormat.setInteger(MediaFormat.KEY_FRAME_RATE, 15);
mediaFormat.setInteger(MediaFormat.KEY_COLOR_FORMAT, MediaCodecInfo.CodecCapabilities.COLOR_FormatYUV420Planar);
mediaFormat.setInteger(MediaFormat.KEY_I_FRAME_INTERVAL, 5);
mediaCodec.configure(mediaFormat, null, null, MediaCodec.CONFIGURE_FLAG_ENCODE);
mediaCodec.start();
}
public void close() {
try {
mediaCodec.stop();
mediaCodec.release();
outputStream.flush();
outputStream.close();
} catch (Exception e){
e.printStackTrace();
}
}
// called from Camera.setPreviewCallbackWithBuffer(...) in other class
public void offerEncoder(byte[] input) {
try {
ByteBuffer[] inputBuffers = mediaCodec.getInputBuffers();
ByteBuffer[] outputBuffers = mediaCodec.getOutputBuffers();
int inputBufferIndex = mediaCodec.dequeueInputBuffer(-1);
if (inputBufferIndex >= 0) {
ByteBuffer inputBuffer = inputBuffers[inputBufferIndex];
inputBuffer.clear();
inputBuffer.put(input);
mediaCodec.queueInputBuffer(inputBufferIndex, 0, input.length, 0, 0);
}
MediaCodec.BufferInfo bufferInfo = new MediaCodec.BufferInfo();
int outputBufferIndex = mediaCodec.dequeueOutputBuffer(bufferInfo,0);
while (outputBufferIndex >= 0) {
ByteBuffer outputBuffer = outputBuffers[outputBufferIndex];
byte[] outData = new byte[bufferInfo.size];
outputBuffer.get(outData);
outputStream.write(outData, 0, outData.length);
Log.i("AvcEncoder", outData.length + " bytes written");
mediaCodec.releaseOutputBuffer(outputBufferIndex, false);
outputBufferIndex = mediaCodec.dequeueOutputBuffer(bufferInfo, 0);
}
} catch (Throwable t) {
t.printStackTrace();
}
}
更改編碼器類型爲「視頻/ MP4」顯然解決了幀率-問題,但由於主要目標是使流媒體服務,這不是一個很好的解決方案。
我知道我放棄了Alex的一些代碼,考慮到SPS和PPS NALU的代碼,但我希望這不是必須的,因爲這些信息也來自outData
,我假定編碼器會正確格式化。如果情況並非如此,我應該如何在文件/流中安排不同類型的NALU?
那麼,我爲了製作一個有效的,正在運行的H.264數據流而錯過了什麼?我應該使用哪些設置來使相機的色彩空間和編碼器的色彩空間匹配?
我有一種感覺,這是一個比Android/MediaCodec主題更多的H.264相關問題。或者我仍然沒有正確使用MediaCodec API?
在此先感謝。
我已經遇到了很多有關Android媒體播放器的問題,以及Android手機的不同行爲不同。你有可能做轉換服務器端嗎? –
我正在處理類似的功能,但是當從Camera.setPreviewCallbackWithBuffer(...)調用offerEncoder(...)時,我當前正在獲取java.nio.BufferOverflowException,是否可以共享您在Camera.setPreviewCallbackWithBuffer中執行的方法(......)。非常感謝。 – lucasweb
代碼似乎沒問題,除了發生在inputBuffer.put(input)上的java.nio.BufferOverflowException;聲明。試圖將字節拆分爲buffer.capacity()塊,最後在MediaCodec.queueInputBuffer調用時出現IllegalStateException錯誤。 任何想法如何可以糾正? –