1 /*
2  * Copyright (C) 2013 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 #ifndef ANDROID_SERVERS_CAMERA3_STATUSTRACKER_H
18 #define ANDROID_SERVERS_CAMERA3_STATUSTRACKER_H
19 
20 #include <string>
21 #include <utils/Condition.h>
22 #include <utils/Errors.h>
23 #include <utils/List.h>
24 #include <utils/Mutex.h>
25 #include <utils/Thread.h>
26 #include <utils/KeyedVector.h>
27 
28 #include "common/CameraDeviceBase.h"
29 
30 namespace android {
31 
32 class Camera3Device;
33 class Fence;
34 
35 namespace camera3 {
36 
37 /**
38  * State tracking for idle and other collective state transitions.
39  * Collects idle notifications from different sources and calls the
40  * parent when all of them become idle.
41  *
42  * The parent is responsible for synchronizing the status updates with its
43  * internal state correctly, which means the notifyStatus call to the parent may
44  * block for a while.
45  */
46 class StatusTracker: public Thread {
47   public:
48     explicit StatusTracker(wp<Camera3Device> parent);
49     ~StatusTracker();
50 
51     // An always-invalid component ID
52     static const int NO_STATUS_ID = -1;
53 
54     // Add a component to track; returns non-negative unique ID for the new
55     // component on success, negative error code on failure.
56     // New components start in the idle state.
57     int addComponent(std::string componentName);
58 
59     // Remove existing component from idle tracking. Ignores unknown IDs
60     void removeComponent(int id);
61 
62     // Set the state of a tracked component to be idle. Ignores unknown IDs; can
63     // accept a fence to wait on to complete idle.  The fence is merged with any
64     // previous fences given, which means they all must signal before the
65     // component is considered idle.
66     void markComponentIdle(int id, const sp<Fence>& componentFence);
67 
68     // Set the state of a tracked component to be active. Ignores unknown IDs.
69     void markComponentActive(int id);
70 
71     void dumpActiveComponents();
72 
73     // Flush all pending states inflight in the tracker, and return upon
74     // completion.
75     void flushPendingStates();
76 
77     virtual void requestExit();
78   protected:
79 
80     virtual bool threadLoop();
81 
82   private:
83     enum ComponentState {
84         IDLE,
85         ACTIVE
86     };
87 
88     void markComponent(int id, ComponentState state,
89             const sp<Fence>& componentFence);
90 
91     // Guards mPendingChange, mPendingStates, mComponentsChanged
92     Mutex mPendingLock;
93 
94     Condition mPendingChangeSignal;
95 
96     struct StateChange {
97         int id;
98         ComponentState state;
99         sp<Fence> fence;
100     };
101     // A queue of yet-to-be-processed state changes to components
102     Vector<StateChange> mPendingChangeQueue;
103     bool mComponentsChanged;
104 
105     wp<Camera3Device> mParent;
106 
107     // Guards rest of internals. Must be locked after mPendingLock if both used.
108     Mutex mLock;
109 
110     int mNextComponentId;
111 
112     // Current component states
113     KeyedVector<int, ComponentState> mStates;
114     KeyedVector<int, std::string> mComponentNames;
115     // Merged fence for all processed state changes
116     sp<Fence> mIdleFence;
117     // Current overall device state
118     ComponentState mDeviceState;
119 
120     // For flushing all pending states transitions
121     bool mFlushed;
122     Mutex mFlushLock;
123     Condition mFlushCondition;
124 
125     // Private to threadLoop
126 
127     // Determine current overall device state
128     // We're IDLE iff
129     // - All components are currently IDLE
130     // - The merged fence for all component updates has signalled
131     ComponentState getDeviceStateLocked();
132 
133     Vector<ComponentState> mStateTransitions;
134 
135     static const nsecs_t kWaitDuration = 250000000LL; // 250 ms
136 };
137 
138 } // namespace camera3
139 
140 } // namespace android
141 
142 #endif
143