web-dev-qa-db-ja.com

Android LollipopでのMediaRecorderの問題

私はテストしています 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());

例外は起動されませんでしたが、ストリーミングを開始すると、ダイアログに出力ファイルが必要であることを通知します。

感謝します。

20
andreasperelli

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を必ず閉じてください。

25
sbaar

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はファイルに出力します。

1
xiaose