Home | History | Annotate | Download | only in autofocus
      1 /*
      2  * Copyright (C) 2014 The Android Open Source Project
      3  *
      4  * Licensed under the Apache License, Version 2.0 (the "License");
      5  * you may not use this file except in compliance with the License.
      6  * You may obtain a copy of the License at
      7  *
      8  *      http://www.apache.org/licenses/LICENSE-2.0
      9  *
     10  * Unless required by applicable law or agreed to in writing, software
     11  * distributed under the License is distributed on an "AS IS" BASIS,
     12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
     13  * See the License for the specific language governing permissions and
     14  * limitations under the License.
     15  */
     16 
     17 package com.android.camera.one.v2.autofocus;
     18 
     19 import android.hardware.camera2.CaptureRequest;
     20 import android.hardware.camera2.CaptureResult;
     21 
     22 import com.android.camera.async.Updatable;
     23 import com.android.camera.one.v2.camera2proxy.CaptureResultProxy;
     24 import com.google.common.base.Objects;
     25 import com.google.common.collect.ImmutableSet;
     26 import com.google.common.util.concurrent.SettableFuture;
     27 
     28 import java.util.Set;
     29 import java.util.concurrent.ExecutionException;
     30 import java.util.concurrent.TimeUnit;
     31 import java.util.concurrent.TimeoutException;
     32 
     33 import javax.annotation.ParametersAreNonnullByDefault;
     34 
     35 /**
     36  * Listens for image metadata and returns the result of an AF scan caused by an
     37  * AF_TRIGGER_START.
     38  * <p>
     39  * This maintains/implements the subset of the finite state machine of
     40  * {@link android.hardware.camera2.CaptureResult#CONTROL_AF_STATE} which relates
     41  * to AF_TRIGGER.
     42  * <p>
     43  * That is, it invokes the given callback when a scan is complete, according to
     44  * the following sequence:
     45  *
     46  * <pre>
     47  * .* CONTROL_AF_TRIGGER_START .* (STATE_INACTIVE|STATE_FOCUSED_LOCKED|STATE_NOT_FOCUSED_LOCKED)
     48  * </pre>
     49  * <p>
     50  * See the android documentation for {@link CaptureResult#CONTROL_AF_STATE} for
     51  * further documentation on the state machine this class implements.
     52  */
     53 @ParametersAreNonnullByDefault
     54 public final class AFTriggerResult implements Updatable<CaptureResultProxy> {
     55     private static final Set<Integer> TRIGGER_DONE_STATES = ImmutableSet.of(
     56             CaptureResult.CONTROL_AF_STATE_INACTIVE,
     57             CaptureResult.CONTROL_AF_STATE_FOCUSED_LOCKED,
     58             CaptureResult.CONTROL_AF_STATE_NOT_FOCUSED_LOCKED);
     59 
     60     private final TriggerStateMachine mStateMachine;
     61     private final SettableFuture<Boolean> mFutureResult;
     62 
     63     public AFTriggerResult() {
     64         mFutureResult = SettableFuture.create();
     65         mStateMachine = new TriggerStateMachine(
     66                 CaptureRequest.CONTROL_AF_TRIGGER_START,
     67                 TRIGGER_DONE_STATES);
     68     }
     69 
     70     @Override
     71     public void update(CaptureResultProxy result) {
     72         Integer afState = result.get(CaptureResult.CONTROL_AF_STATE);
     73         boolean done = mStateMachine.update(
     74                 result.getFrameNumber(),
     75                 result.getRequest().get(CaptureRequest.CONTROL_AF_TRIGGER),
     76                 afState);
     77         if (done) {
     78             boolean inFocus = Objects.equal(afState, CaptureResult.CONTROL_AF_STATE_PASSIVE_FOCUSED)
     79                     || Objects.equal(afState, CaptureResult.CONTROL_AF_STATE_FOCUSED_LOCKED);
     80             mFutureResult.set(inFocus);
     81         }
     82     }
     83 
     84     /**
     85      * Blocks until the AF scan is complete.
     86      *
     87      * @return Whether the scene is in-focus or not, based on the camera driver.
     88      * @throws InterruptedException
     89      */
     90     public boolean get() throws InterruptedException {
     91         try {
     92             return mFutureResult.get();
     93         } catch (ExecutionException impossible) {
     94             throw new RuntimeException(impossible);
     95         }
     96     }
     97 
     98     /**
     99      * Blocks until the AF scan is complete.
    100      *
    101      * @return Whether the scene is in-focus or not, based on the camera driver.
    102      * @throws InterruptedException
    103      */
    104     public boolean get(long timeout, TimeUnit timeUnit) throws InterruptedException,
    105             TimeoutException {
    106         try {
    107             return mFutureResult.get(timeout, timeUnit);
    108         } catch (ExecutionException impossible) {
    109             throw new RuntimeException(impossible);
    110         }
    111     }
    112 }
    113