私はテストしています libstreaming 新しいAndroidLollipop、および以前のリリースで機能していたこのコードは、例外を起動するようです。
try {
mMediaRecorder = new MediaRecorder();
mMediaRecorder.setCamera(mCamera);
mMediaRecorder.setVideoSource(MediaRecorder.VideoSource.CAMERA);
mMediaRecorder.setOutputFormat(MediaRecorder.OutputFormat.THREE_GPP);
mMediaRecorder.setVideoEncoder(mVideoEncoder);
mMediaRecorder.setPreviewDisplay(mSurfaceView.getHolder().getSurface());
mMediaRecorder.setVideoSize(mRequestedQuality.resX,mRequestedQuality.resY);
mMediaRecorder.setVideoFrameRate(mRequestedQuality.framerate);
// The bandwidth actually consumed is often above what was requested
mMediaRecorder.setVideoEncodingBitRate((int)(mRequestedQuality.bitrate*0.8));
// We write the ouput of the camera in a local socket instead of a file !
// This one little trick makes streaming feasible quiet simply: data from the camera
// can then be manipulated at the other end of the socket
mMediaRecorder.setOutputFile(mSender.getFileDescriptor());
mMediaRecorder.prepare();
mMediaRecorder.start();
} catch (Exception e) {
throw new ConfNotSupportedException(e.getMessage());
}
起動された例外は次のとおりです。
MediaRecorder:起動に失敗しました-38
11-18 09:50:21.028: W/System.err(15783): net.majorkernelpanic.streaming.exceptions.ConfNotSupportedException
11-18 09:50:21.028: W/System.err(15783): at net.majorkernelpanic.streaming.video.VideoStream.encodeWithMediaRecorder(VideoStream.Java:442)
11-18 09:50:21.028: W/System.err(15783): at net.majorkernelpanic.streaming.MediaStream.start(MediaStream.Java:250)
私はコメントしようとしました:
mMediaRecorder.setOutputFile(mSender.getFileDescriptor());
例外は起動されませんでしたが、ストリーミングを開始すると、ダイアログに出力ファイルが必要であることを通知します。
感謝します。
AOSPにバグレポートを提出しました。 https://code.google.com/p/Android/issues/detail?id=80715
「現在のSELinuxポリシーでは、メディアサーバーがアプリで生成された抽象的なUNIXドメインソケットを処理することを許可していません。
代わりに、=で許可されているパイプペア( http://developer.Android.com/reference/Android/os/ParcelFileDescriptor.html#createPipe() )を作成することをお勧めしますAndroid 5.0ポリシー。 "彼らがこれを行った理由や、私たちがどのように知っているはずだったのか、私にはわかりません。
私は非常に古い/変更された(わかりません)libstreamingのバージョンを使用していますが、mediastreamはまだmediarecorderから拡張されていますが、現在のバージョンを見ると、MediaStreamでcreateSocketsを次のようなものに変更したいと思うでしょう:
ParcelFileDescriptor[] parcelFileDescriptors =ParcelFileDescriptor.createPipe();
parcelRead = new ParcelFileDescriptor(parcelFileDescriptors[0]);
parcelWrite = new ParcelFileDescriptor(parcelFileDescriptors[1]);
その後、あなたのビデオ/オーディオストリームで
setOutputFile(parcelWrite.getFileDescriptor());
そして同じファイルの変更で
// The packetizer encapsulates the bit stream in an RTP stream and send it over the network
mPacketizer.setInputStream(mReceiver.getInputStream());
mPacketizer.start();
に
InputStream is = null;
try{ is = new ParcelFileDescriptor.AutoCloseInputStream(parcelRead);
}
catch (Exception e){}
mPacketizer.setInputStream(is);
Andreasperelliがコメントで指摘したように、closeSockets()またはMediaRecorder.stop()を呼び出す前に、closeSockets()で、または実装とバージョンに応じて、ParcelFileDescriptorsを必ず閉じてください。
Android 6.0でこの問題をコードで解決します
new Thread(new Runnable() {
@Override public void run() {
FileInputStream inputStream = null;
try {
inputStream = new FileInputStream(path);
} catch (FileNotFoundException e) {
e.printStackTrace();
}
while (true) {
byte[] buffer = new byte[0];
try {
buffer = new byte[inputStream.available()];
} catch (IOException e) {
e.printStackTrace();
}
try {
inputStream.read(buffer);
} catch (IOException e) {
e.printStackTrace();
}
try {
mSender.getOutputStream().write(buffer);
mSender.getOutputStream().flush();
} catch (IOException e) {
e.printStackTrace();
}
}
}
}).start();
ファイルをバッファとして使用し、別のスレッドでバイトを書き込みます。MediaRecorderはファイルに出力します。