 Chromium Code Reviews
 Chromium Code Reviews Issue 2273573003:
  Support for video file instead of camera and output video out to file  (Closed) 
  Base URL: https://chromium.googlesource.com/external/webrtc.git@master
    
  
    Issue 2273573003:
  Support for video file instead of camera and output video out to file  (Closed) 
  Base URL: https://chromium.googlesource.com/external/webrtc.git@master| Index: webrtc/api/android/java/src/org/webrtc/FileVideoCapturer.java | 
| diff --git a/webrtc/api/android/java/src/org/webrtc/FileVideoCapturer.java b/webrtc/api/android/java/src/org/webrtc/FileVideoCapturer.java | 
| new file mode 100644 | 
| index 0000000000000000000000000000000000000000..2d44f518fe24393d15c8fc8ad7bf0255b3880c56 | 
| --- /dev/null | 
| +++ b/webrtc/api/android/java/src/org/webrtc/FileVideoCapturer.java | 
| @@ -0,0 +1,239 @@ | 
| +/* | 
| + * Copyright 2016 The WebRTC Project Authors. All rights reserved. | 
| + * | 
| + * Use of this source code is governed by a BSD-style license | 
| + * that can be found in the LICENSE file in the root of the source | 
| + * tree. An additional intellectual property rights grant can be found | 
| + * in the file PATENTS. All contributing project authors may | 
| + * be found in the AUTHORS file in the root of the source tree. | 
| + */ | 
| + | 
| +package org.webrtc; | 
| + | 
| +import android.content.Context; | 
| +import android.os.SystemClock; | 
| + | 
| +import java.util.concurrent.TimeUnit; | 
| +import java.util.Timer; | 
| +import java.util.TimerTask; | 
| +import java.io.File; | 
| +import java.io.RandomAccessFile; | 
| +import java.io.IOException; | 
| + | 
| +public class FileVideoCapturer implements VideoCapturer { | 
| + private interface VideoReader { | 
| + int getFrameWidth(); | 
| + int getFrameHeight(); | 
| + byte[] getNextFrame(); | 
| + void close(); | 
| + } | 
| + | 
| + /** | 
| + * Read video data from file for the .y4m container. | 
| + */ | 
| + private static class VideoReaderY4M implements VideoReader { | 
| + private final static String TAG = "VideoReaderY4M"; | 
| + private final int frameWidth; | 
| + private final int frameHeight; | 
| + private final int frameSize; | 
| + | 
| + // First char after header | 
| + private final long videoStart; | 
| + | 
| + private static final String Y4M_FRAME_DELIMETER = "FRAME"; | 
| + | 
| + private final RandomAccessFile mediaFileStream; | 
| + | 
| + public int getFrameWidth() { | 
| + return frameWidth; | 
| + } | 
| + | 
| + public int getFrameHeight() { | 
| + return frameHeight; | 
| + } | 
| + | 
| + public VideoReaderY4M(String file) throws IOException { | 
| + mediaFileStream = new RandomAccessFile(file, "r"); | 
| + StringBuilder builder = new StringBuilder(); | 
| + for (;;) { | 
| + int c = mediaFileStream.read(); | 
| + if (c == -1) { | 
| + // End of file reached. | 
| + throw new RuntimeException("Found end of file before end of header for file: " + file); | 
| + } | 
| + if (c == '\n') { | 
| + // End of header found. | 
| + break; | 
| + } | 
| + builder.append((char)c); | 
| + } | 
| + videoStart = mediaFileStream.getFilePointer(); | 
| + String header = builder.toString(); | 
| + String[] headerTokens = header.split("[\n ]"); | 
| 
sakal
2016/10/05 13:28:40
Header cannot contain \n because we break on it, r
 
mandermo
2016/10/07 11:33:40
Done.
 | 
| + Logging.d(TAG, "header: " + header + ", headerTokens" + headerTokens); | 
| + int w = 0; | 
| + int h = 0; | 
| + String colorSpace = ""; | 
| + for (String tok : headerTokens) { | 
| + char c = tok.charAt(0); | 
| + switch (c) { | 
| + case 'W': | 
| + w = Integer.parseInt(tok.substring(1)); | 
| + break; | 
| + case 'H': | 
| + h = Integer.parseInt(tok.substring(1)); | 
| + break; | 
| + case 'C': | 
| + colorSpace = tok.substring(1); | 
| + break; | 
| + } | 
| + } | 
| + Logging.d(TAG, "Color space: " + colorSpace); | 
| + if (!colorSpace.equals("420")) { | 
| + throw new IllegalArgumentException("Does not support any other color space than I420"); | 
| + } | 
| + if ((w % 2) == 1 || (h % 2) == 1) { | 
| + throw new IllegalArgumentException("Does not support odd width or height"); | 
| + } | 
| + frameWidth = w; | 
| + frameHeight = h; | 
| + frameSize = w * h * 3 / 2; | 
| + Logging.d(TAG, "frame dim: (" + w + ", " + h + ") frameSize: " + frameSize); | 
| + } | 
| + | 
| + public byte[] getNextFrame() { | 
| + byte[] frame = new byte[frameSize]; | 
| + try { | 
| + byte[] frameDelim = new byte[Y4M_FRAME_DELIMETER.length() + 1]; | 
| + if (mediaFileStream.read(frameDelim) < frameDelim.length) | 
| + { | 
| + // We reach end of file, loop | 
| + //mediaFileStream.seek(videoStart + Y4M_FRAME_DELIMETER.length() + 1); | 
| 
sakal
2016/10/05 13:28:40
Please remove commented out code.
 
mandermo
2016/10/07 11:33:40
Done.
 | 
| + mediaFileStream.seek(videoStart); | 
| + if (mediaFileStream.read(frameDelim) < frameDelim.length) { | 
| + throw new RuntimeException("Error looping video"); | 
| + } | 
| + } | 
| + String frameDelimStr = new String(frameDelim); | 
| + if (!frameDelimStr.equals(Y4M_FRAME_DELIMETER + "\n")) { | 
| + Logging.d(TAG, "frameDelim: '" + frameDelimStr + "'"); | 
| + throw new RuntimeException( | 
| + "Frames should be delimited by FRAME plus newline"); | 
| + } | 
| + mediaFileStream.readFully(frame); | 
| + byte[] nv21Frame = new byte[frameSize]; | 
| + nativeI420ToNV21( | 
| + frame, frameWidth, frameHeight, nv21Frame); | 
| + return nv21Frame; | 
| + } catch (IOException e) { | 
| + throw new RuntimeException(e); | 
| + } | 
| + } | 
| + | 
| + public void close() { | 
| + try { | 
| + mediaFileStream.close(); | 
| + } catch (IOException e) { | 
| + Logging.e(TAG, "Problem closing file"); | 
| + } | 
| + } | 
| + } | 
| + | 
| + private final static String TAG = "FileVideoCapturer"; | 
| + | 
| + private VideoReader videoReader; | 
| 
sakal
2016/10/05 13:28:40
final
 
mandermo
2016/10/07 11:33:40
Done.
 | 
| + | 
| 
sakal
2016/10/05 13:28:40
Please reduce the amount of empty lines.
 
mandermo
2016/10/07 11:33:40
Done.
 | 
| + private CapturerObserver capturerObserver; | 
| + | 
| + private Timer timer = new Timer(); | 
| 
sakal
2016/10/05 13:28:40
final
 
mandermo
2016/10/07 11:33:40
Done.
 | 
| + | 
| + private TimerTask tickTask = new TimerTask() { | 
| 
sakal
2016/10/05 13:28:40
final
 
mandermo
2016/10/07 11:33:40
Done.
 | 
| + @Override | 
| + public void run() { | 
| + tick(); | 
| + } | 
| + }; | 
| + | 
| + private int getFrameWidth() { | 
| + return videoReader.getFrameWidth(); | 
| + } | 
| + | 
| + private int getFrameHeight() { | 
| + return videoReader.getFrameHeight(); | 
| + } | 
| + | 
| + public FileVideoCapturer(String inputFile) throws IOException { | 
| + try { | 
| + videoReader = new VideoReaderY4M(inputFile); | 
| + } catch (IOException e) { | 
| + Logging.d(TAG, "Could not open video file: " + inputFile); | 
| + throw e; | 
| + } | 
| + } | 
| + | 
| + public static FileVideoCapturer create(String inputFile) { | 
| + try { | 
| + return new FileVideoCapturer( | 
| + inputFile); | 
| + } catch (IOException e) { | 
| + return null; | 
| + } | 
| + } | 
| + | 
| + private byte[] getNextFrame() { | 
| + return videoReader.getNextFrame(); | 
| + } | 
| + | 
| + public void tick() { | 
| + final long captureTimeNs = | 
| + TimeUnit.MILLISECONDS.toNanos(SystemClock.elapsedRealtime()); | 
| + | 
| + byte[] frameData = getNextFrame(); | 
| + capturerObserver.onByteBufferFrameCaptured( | 
| + frameData, | 
| + getFrameWidth(), | 
| + getFrameHeight(), | 
| + 0, | 
| + captureTimeNs); | 
| + } | 
| + | 
| + @Override | 
| + public void initialize(SurfaceTextureHelper surfaceTextureHelper, Context applicationContext, | 
| + CapturerObserver capturerObserver) { | 
| + this.capturerObserver = capturerObserver; | 
| + } | 
| + | 
| + @Override | 
| + public void startCapture( | 
| + int width, int height, int framerate) { | 
| + timer.schedule(tickTask, 0, 1000/framerate); | 
| + } | 
| + | 
| + @Override | 
| + public void stopCapture() throws InterruptedException { | 
| + timer.cancel(); | 
| + } | 
| + | 
| + @Override | 
| + public void onOutputFormatRequest(int width, int height, int framerate) { | 
| + // Empty on purpose | 
| + } | 
| + | 
| + @Override | 
| + public void changeCaptureFormat(int width, int height, int framerate) { | 
| + // Empty on purpose | 
| + } | 
| + | 
| + @Override | 
| + public void dispose() { | 
| + videoReader.close(); | 
| + } | 
| + | 
| + @Override | 
| + public boolean isScreencast() { | 
| + return false; | 
| + } | 
| + | 
| + public static native void nativeI420ToNV21(byte[] src, int width, | 
| + int height, byte[] dst); | 
| +} |