All Downloads are FREE. Search and download functionalities are using the official Maven repository.

android.hardware.camera2.legacy.CaptureCollector Maven / Gradle / Ivy

Go to download

A library jar that provides APIs for Applications written for the Google Android Platform.

There is a newer version: 14-robolectric-10818077
Show newest version
/*
 * Copyright (C) 2014 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.hardware.camera2.legacy;

import android.hardware.camera2.impl.CameraDeviceImpl;
import android.util.Log;
import android.util.MutableLong;
import android.util.Pair;

import java.util.ArrayDeque;
import java.util.ArrayList;
import java.util.TreeSet;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.locks.Condition;
import java.util.concurrent.locks.ReentrantLock;

/**
 * Collect timestamps and state for each {@link CaptureRequest} as it passes through
 * the Legacy camera pipeline.
 */
public class CaptureCollector {
    private static final String TAG = "CaptureCollector";

    private static final boolean DEBUG = Log.isLoggable(LegacyCameraDevice.DEBUG_PROP, Log.DEBUG);

    private static final int FLAG_RECEIVED_JPEG = 1;
    private static final int FLAG_RECEIVED_JPEG_TS = 2;
    private static final int FLAG_RECEIVED_PREVIEW = 4;
    private static final int FLAG_RECEIVED_PREVIEW_TS = 8;
    private static final int FLAG_RECEIVED_ALL_JPEG = FLAG_RECEIVED_JPEG | FLAG_RECEIVED_JPEG_TS;
    private static final int FLAG_RECEIVED_ALL_PREVIEW = FLAG_RECEIVED_PREVIEW |
            FLAG_RECEIVED_PREVIEW_TS;

    private static final int MAX_JPEGS_IN_FLIGHT = 1;

    private class CaptureHolder implements Comparable{
        private final RequestHolder mRequest;
        private final LegacyRequest mLegacy;
        public final boolean needsJpeg;
        public final boolean needsPreview;

        private long mTimestamp = 0;
        private int mReceivedFlags = 0;
        private boolean mHasStarted = false;
        private boolean mFailedJpeg = false;
        private boolean mFailedPreview = false;
        private boolean mCompleted = false;
        private boolean mPreviewCompleted = false;

        public CaptureHolder(RequestHolder request, LegacyRequest legacyHolder) {
            mRequest = request;
            mLegacy = legacyHolder;
            needsJpeg = request.hasJpegTargets();
            needsPreview = request.hasPreviewTargets();
        }

        public boolean isPreviewCompleted() {
            return (mReceivedFlags & FLAG_RECEIVED_ALL_PREVIEW) == FLAG_RECEIVED_ALL_PREVIEW;
        }

        public  boolean isJpegCompleted() {
            return (mReceivedFlags & FLAG_RECEIVED_ALL_JPEG) == FLAG_RECEIVED_ALL_JPEG;
        }

        public boolean isCompleted() {
            return (needsJpeg == isJpegCompleted()) && (needsPreview == isPreviewCompleted());
        }

        public void tryComplete() {
            if (!mPreviewCompleted && needsPreview && isPreviewCompleted()) {
                CaptureCollector.this.onPreviewCompleted();
                mPreviewCompleted = true;
            }

            if (isCompleted() && !mCompleted) {
                if (mFailedPreview || mFailedJpeg) {
                    if (!mHasStarted) {
                        // Send a request error if the capture has not yet started.
                        mRequest.failRequest();
                        CaptureCollector.this.mDeviceState.setCaptureStart(mRequest, mTimestamp,
                                CameraDeviceImpl.CameraDeviceCallbacks.ERROR_CAMERA_REQUEST);
                    } else {
                        // Send buffer dropped errors for each pending buffer if the request has
                        // started.
                        if (mFailedPreview) {
                            Log.w(TAG, "Preview buffers dropped for request: " +
                                    mRequest.getRequestId());
                            for (int i = 0; i < mRequest.numPreviewTargets(); i++) {
                                CaptureCollector.this.mDeviceState.setCaptureResult(mRequest,
                                    /*result*/null,
                                        CameraDeviceImpl.CameraDeviceCallbacks.ERROR_CAMERA_BUFFER);
                            }
                        }
                        if (mFailedJpeg) {
                            Log.w(TAG, "Jpeg buffers dropped for request: " +
                                    mRequest.getRequestId());
                            for (int i = 0; i < mRequest.numJpegTargets(); i++) {
                                CaptureCollector.this.mDeviceState.setCaptureResult(mRequest,
                                    /*result*/null,
                                        CameraDeviceImpl.CameraDeviceCallbacks.ERROR_CAMERA_BUFFER);
                            }
                        }
                    }
                }
                CaptureCollector.this.onRequestCompleted(CaptureHolder.this);
                mCompleted = true;
            }
        }

        public void setJpegTimestamp(long timestamp) {
            if (DEBUG) {
                Log.d(TAG, "setJpegTimestamp - called for request " + mRequest.getRequestId());
            }
            if (!needsJpeg) {
                throw new IllegalStateException(
                        "setJpegTimestamp called for capture with no jpeg targets.");
            }
            if (isCompleted()) {
                throw new IllegalStateException(
                        "setJpegTimestamp called on already completed request.");
            }

            mReceivedFlags |= FLAG_RECEIVED_JPEG_TS;

            if (mTimestamp == 0) {
                mTimestamp = timestamp;
            }

            if (!mHasStarted) {
                mHasStarted = true;
                CaptureCollector.this.mDeviceState.setCaptureStart(mRequest, mTimestamp,
                        CameraDeviceState.NO_CAPTURE_ERROR);
            }

            tryComplete();
        }

        public void setJpegProduced() {
            if (DEBUG) {
                Log.d(TAG, "setJpegProduced - called for request " + mRequest.getRequestId());
            }
            if (!needsJpeg) {
                throw new IllegalStateException(
                        "setJpegProduced called for capture with no jpeg targets.");
            }
            if (isCompleted()) {
                throw new IllegalStateException(
                        "setJpegProduced called on already completed request.");
            }

            mReceivedFlags |= FLAG_RECEIVED_JPEG;
            tryComplete();
        }

        public void setJpegFailed() {
            if (DEBUG) {
                Log.d(TAG, "setJpegFailed - called for request " + mRequest.getRequestId());
            }
            if (!needsJpeg || isJpegCompleted()) {
                return;
            }
            mFailedJpeg = true;

            mReceivedFlags |= FLAG_RECEIVED_JPEG;
            mReceivedFlags |= FLAG_RECEIVED_JPEG_TS;
            tryComplete();
        }

        public void setPreviewTimestamp(long timestamp) {
            if (DEBUG) {
                Log.d(TAG, "setPreviewTimestamp - called for request " + mRequest.getRequestId());
            }
            if (!needsPreview) {
                throw new IllegalStateException(
                        "setPreviewTimestamp called for capture with no preview targets.");
            }
            if (isCompleted()) {
                throw new IllegalStateException(
                        "setPreviewTimestamp called on already completed request.");
            }

            mReceivedFlags |= FLAG_RECEIVED_PREVIEW_TS;

            if (mTimestamp == 0) {
                mTimestamp = timestamp;
            }

            if (!needsJpeg) {
                if (!mHasStarted) {
                    mHasStarted = true;
                    CaptureCollector.this.mDeviceState.setCaptureStart(mRequest, mTimestamp,
                            CameraDeviceState.NO_CAPTURE_ERROR);
                }
            }

            tryComplete();
        }

        public void setPreviewProduced() {
            if (DEBUG) {
                Log.d(TAG, "setPreviewProduced - called for request " + mRequest.getRequestId());
            }
            if (!needsPreview) {
                throw new IllegalStateException(
                        "setPreviewProduced called for capture with no preview targets.");
            }
            if (isCompleted()) {
                throw new IllegalStateException(
                        "setPreviewProduced called on already completed request.");
            }

            mReceivedFlags |= FLAG_RECEIVED_PREVIEW;
            tryComplete();
        }

        public void setPreviewFailed() {
            if (DEBUG) {
                Log.d(TAG, "setPreviewFailed - called for request " + mRequest.getRequestId());
            }
            if (!needsPreview || isPreviewCompleted()) {
                return;
            }
            mFailedPreview = true;

            mReceivedFlags |= FLAG_RECEIVED_PREVIEW;
            mReceivedFlags |= FLAG_RECEIVED_PREVIEW_TS;
            tryComplete();
        }

        // Comparison and equals based on frame number.
        @Override
        public int compareTo(CaptureHolder captureHolder) {
            return (mRequest.getFrameNumber() > captureHolder.mRequest.getFrameNumber()) ? 1 :
                    ((mRequest.getFrameNumber() == captureHolder.mRequest.getFrameNumber()) ? 0 :
                            -1);
        }

        // Comparison and equals based on frame number.
        @Override
        public boolean equals(Object o) {
            return o instanceof CaptureHolder && compareTo((CaptureHolder) o) == 0;
        }
    }

    private final TreeSet mActiveRequests;
    private final ArrayDeque mJpegCaptureQueue;
    private final ArrayDeque mJpegProduceQueue;
    private final ArrayDeque mPreviewCaptureQueue;
    private final ArrayDeque mPreviewProduceQueue;
    private final ArrayList mCompletedRequests = new ArrayList<>();

    private final ReentrantLock mLock = new ReentrantLock();
    private final Condition mIsEmpty;
    private final Condition mPreviewsEmpty;
    private final Condition mNotFull;
    private final CameraDeviceState mDeviceState;
    private int mInFlight = 0;
    private int mInFlightPreviews = 0;
    private final int mMaxInFlight;

    /**
     * Create a new {@link CaptureCollector} that can modify the given {@link CameraDeviceState}.
     *
     * @param maxInFlight max allowed in-flight requests.
     * @param deviceState the {@link CameraDeviceState} to update as requests are processed.
     */
    public CaptureCollector(int maxInFlight, CameraDeviceState deviceState) {
        mMaxInFlight = maxInFlight;
        mJpegCaptureQueue = new ArrayDeque<>(MAX_JPEGS_IN_FLIGHT);
        mJpegProduceQueue = new ArrayDeque<>(MAX_JPEGS_IN_FLIGHT);
        mPreviewCaptureQueue = new ArrayDeque<>(mMaxInFlight);
        mPreviewProduceQueue = new ArrayDeque<>(mMaxInFlight);
        mActiveRequests = new TreeSet<>();
        mIsEmpty = mLock.newCondition();
        mNotFull = mLock.newCondition();
        mPreviewsEmpty = mLock.newCondition();
        mDeviceState = deviceState;
    }

    /**
     * Queue a new request.
     *
     * 

* For requests that use the Camera1 API preview output stream, this will block if there are * already {@code maxInFlight} requests in progress (until at least one prior request has * completed). For requests that use the Camera1 API jpeg callbacks, this will block until * all prior requests have been completed to avoid stopping preview for * {@link android.hardware.Camera#takePicture} before prior preview requests have been * completed. *

* @param holder the {@link RequestHolder} for this request. * @param legacy the {@link LegacyRequest} for this request; this will not be mutated. * @param timeout a timeout to use for this call. * @param unit the units to use for the timeout. * @return {@code false} if this method timed out. * @throws InterruptedException if this thread is interrupted. */ public boolean queueRequest(RequestHolder holder, LegacyRequest legacy, long timeout, TimeUnit unit) throws InterruptedException { CaptureHolder h = new CaptureHolder(holder, legacy); long nanos = unit.toNanos(timeout); final ReentrantLock lock = this.mLock; lock.lock(); try { if (DEBUG) { Log.d(TAG, "queueRequest for request " + holder.getRequestId() + " - " + mInFlight + " requests remain in flight."); } if (!(h.needsJpeg || h.needsPreview)) { throw new IllegalStateException("Request must target at least one output surface!"); } if (h.needsJpeg) { // Wait for all current requests to finish before queueing jpeg. while (mInFlight > 0) { if (nanos <= 0) { return false; } nanos = mIsEmpty.awaitNanos(nanos); } mJpegCaptureQueue.add(h); mJpegProduceQueue.add(h); } if (h.needsPreview) { while (mInFlight >= mMaxInFlight) { if (nanos <= 0) { return false; } nanos = mNotFull.awaitNanos(nanos); } mPreviewCaptureQueue.add(h); mPreviewProduceQueue.add(h); mInFlightPreviews++; } mActiveRequests.add(h); mInFlight++; return true; } finally { lock.unlock(); } } /** * Wait all queued requests to complete. * * @param timeout a timeout to use for this call. * @param unit the units to use for the timeout. * @return {@code false} if this method timed out. * @throws InterruptedException if this thread is interrupted. */ public boolean waitForEmpty(long timeout, TimeUnit unit) throws InterruptedException { long nanos = unit.toNanos(timeout); final ReentrantLock lock = this.mLock; lock.lock(); try { while (mInFlight > 0) { if (nanos <= 0) { return false; } nanos = mIsEmpty.awaitNanos(nanos); } return true; } finally { lock.unlock(); } } /** * Wait all queued requests that use the Camera1 API preview output to complete. * * @param timeout a timeout to use for this call. * @param unit the units to use for the timeout. * @return {@code false} if this method timed out. * @throws InterruptedException if this thread is interrupted. */ public boolean waitForPreviewsEmpty(long timeout, TimeUnit unit) throws InterruptedException { long nanos = unit.toNanos(timeout); final ReentrantLock lock = this.mLock; lock.lock(); try { while (mInFlightPreviews > 0) { if (nanos <= 0) { return false; } nanos = mPreviewsEmpty.awaitNanos(nanos); } return true; } finally { lock.unlock(); } } /** * Wait for the specified request to be completed (all buffers available). * *

May not wait for the same request more than once, since a successful wait * will erase the history of that request.

* * @param holder the {@link RequestHolder} for this request. * @param timeout a timeout to use for this call. * @param unit the units to use for the timeout. * @param timestamp the timestamp of the request will be written out to here, in ns * * @return {@code false} if this method timed out. * * @throws InterruptedException if this thread is interrupted. */ public boolean waitForRequestCompleted(RequestHolder holder, long timeout, TimeUnit unit, MutableLong timestamp) throws InterruptedException { long nanos = unit.toNanos(timeout); final ReentrantLock lock = this.mLock; lock.lock(); try { while (!removeRequestIfCompleted(holder, /*out*/timestamp)) { if (nanos <= 0) { return false; } nanos = mNotFull.awaitNanos(nanos); } return true; } finally { lock.unlock(); } } private boolean removeRequestIfCompleted(RequestHolder holder, MutableLong timestamp) { int i = 0; for (CaptureHolder h : mCompletedRequests) { if (h.mRequest.equals(holder)) { timestamp.value = h.mTimestamp; mCompletedRequests.remove(i); return true; } i++; } return false; } /** * Called to alert the {@link CaptureCollector} that the jpeg capture has begun. * * @param timestamp the time of the jpeg capture. * @return the {@link RequestHolder} for the request associated with this capture. */ public RequestHolder jpegCaptured(long timestamp) { final ReentrantLock lock = this.mLock; lock.lock(); try { CaptureHolder h = mJpegCaptureQueue.poll(); if (h == null) { Log.w(TAG, "jpegCaptured called with no jpeg request on queue!"); return null; } h.setJpegTimestamp(timestamp); return h.mRequest; } finally { lock.unlock(); } } /** * Called to alert the {@link CaptureCollector} that the jpeg capture has completed. * * @return a pair containing the {@link RequestHolder} and the timestamp of the capture. */ public Pair jpegProduced() { final ReentrantLock lock = this.mLock; lock.lock(); try { CaptureHolder h = mJpegProduceQueue.poll(); if (h == null) { Log.w(TAG, "jpegProduced called with no jpeg request on queue!"); return null; } h.setJpegProduced(); return new Pair<>(h.mRequest, h.mTimestamp); } finally { lock.unlock(); } } /** * Check if there are any pending capture requests that use the Camera1 API preview output. * * @return {@code true} if there are pending preview requests. */ public boolean hasPendingPreviewCaptures() { final ReentrantLock lock = this.mLock; lock.lock(); try { return !mPreviewCaptureQueue.isEmpty(); } finally { lock.unlock(); } } /** * Called to alert the {@link CaptureCollector} that the preview capture has begun. * * @param timestamp the time of the preview capture. * @return a pair containing the {@link RequestHolder} and the timestamp of the capture. */ public Pair previewCaptured(long timestamp) { final ReentrantLock lock = this.mLock; lock.lock(); try { CaptureHolder h = mPreviewCaptureQueue.poll(); if (h == null) { if (DEBUG) { Log.d(TAG, "previewCaptured called with no preview request on queue!"); } return null; } h.setPreviewTimestamp(timestamp); return new Pair<>(h.mRequest, h.mTimestamp); } finally { lock.unlock(); } } /** * Called to alert the {@link CaptureCollector} that the preview capture has completed. * * @return the {@link RequestHolder} for the request associated with this capture. */ public RequestHolder previewProduced() { final ReentrantLock lock = this.mLock; lock.lock(); try { CaptureHolder h = mPreviewProduceQueue.poll(); if (h == null) { Log.w(TAG, "previewProduced called with no preview request on queue!"); return null; } h.setPreviewProduced(); return h.mRequest; } finally { lock.unlock(); } } /** * Called to alert the {@link CaptureCollector} that the next pending preview capture has failed. */ public void failNextPreview() { final ReentrantLock lock = this.mLock; lock.lock(); try { CaptureHolder h1 = mPreviewCaptureQueue.peek(); CaptureHolder h2 = mPreviewProduceQueue.peek(); // Find the request with the lowest frame number. CaptureHolder h = (h1 == null) ? h2 : ((h2 == null) ? h1 : ((h1.compareTo(h2) <= 0) ? h1 : h2)); if (h != null) { mPreviewCaptureQueue.remove(h); mPreviewProduceQueue.remove(h); mActiveRequests.remove(h); h.setPreviewFailed(); } } finally { lock.unlock(); } } /** * Called to alert the {@link CaptureCollector} that the next pending jpeg capture has failed. */ public void failNextJpeg() { final ReentrantLock lock = this.mLock; lock.lock(); try { CaptureHolder h1 = mJpegCaptureQueue.peek(); CaptureHolder h2 = mJpegProduceQueue.peek(); // Find the request with the lowest frame number. CaptureHolder h = (h1 == null) ? h2 : ((h2 == null) ? h1 : ((h1.compareTo(h2) <= 0) ? h1 : h2)); if (h != null) { mJpegCaptureQueue.remove(h); mJpegProduceQueue.remove(h); mActiveRequests.remove(h); h.setJpegFailed(); } } finally { lock.unlock(); } } /** * Called to alert the {@link CaptureCollector} all pending captures have failed. */ public void failAll() { final ReentrantLock lock = this.mLock; lock.lock(); try { CaptureHolder h; while ((h = mActiveRequests.pollFirst()) != null) { h.setPreviewFailed(); h.setJpegFailed(); } mPreviewCaptureQueue.clear(); mPreviewProduceQueue.clear(); mJpegCaptureQueue.clear(); mJpegProduceQueue.clear(); } finally { lock.unlock(); } } private void onPreviewCompleted() { mInFlightPreviews--; if (mInFlightPreviews < 0) { throw new IllegalStateException( "More preview captures completed than requests queued."); } if (mInFlightPreviews == 0) { mPreviewsEmpty.signalAll(); } } private void onRequestCompleted(CaptureHolder capture) { RequestHolder request = capture.mRequest; mInFlight--; if (DEBUG) { Log.d(TAG, "Completed request " + request.getRequestId() + ", " + mInFlight + " requests remain in flight."); } if (mInFlight < 0) { throw new IllegalStateException( "More captures completed than requests queued."); } mCompletedRequests.add(capture); mActiveRequests.remove(capture); mNotFull.signalAll(); if (mInFlight == 0) { mIsEmpty.signalAll(); } } }




© 2015 - 2024 Weber Informatics LLC | Privacy Policy