1 /*
2  * Copyright (C) 2016 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 #include "EvsStateControl.h"
17 #include "RenderDirectView.h"
18 #include "RenderTopView.h"
19 #include "RenderPixelCopy.h"
20 
21 #include <stdio.h>
22 #include <string.h>
23 
24 #include <log/log.h>
25 #include <inttypes.h>
26 #include <utils/SystemClock.h>
27 #include <binder/IServiceManager.h>
28 
isSfReady()29 static bool isSfReady() {
30     const android::String16 serviceName("SurfaceFlinger");
31     return android::defaultServiceManager()->checkService(serviceName) != nullptr;
32 }
33 
34 // TODO:  Seems like it'd be nice if the Vehicle HAL provided such helpers (but how & where?)
getPropType(VehicleProperty prop)35 inline constexpr VehiclePropertyType getPropType(VehicleProperty prop) {
36     return static_cast<VehiclePropertyType>(
37             static_cast<int32_t>(prop)
38             & static_cast<int32_t>(VehiclePropertyType::MASK));
39 }
40 
41 
EvsStateControl(android::sp<IVehicle> pVnet,android::sp<IEvsEnumerator> pEvs,android::sp<IEvsDisplay> pDisplay,const ConfigManager & config)42 EvsStateControl::EvsStateControl(android::sp <IVehicle>       pVnet,
43                                  android::sp <IEvsEnumerator> pEvs,
44                                  android::sp <IEvsDisplay>    pDisplay,
45                                  const ConfigManager&         config) :
46     mVehicle(pVnet),
47     mEvs(pEvs),
48     mDisplay(pDisplay),
49     mConfig(config),
50     mCurrentState(OFF) {
51 
52     // Initialize the property value containers we'll be updating (they'll be zeroed by default)
53     static_assert(getPropType(VehicleProperty::GEAR_SELECTION) == VehiclePropertyType::INT32,
54                   "Unexpected type for GEAR_SELECTION property");
55     static_assert(getPropType(VehicleProperty::TURN_SIGNAL_STATE) == VehiclePropertyType::INT32,
56                   "Unexpected type for TURN_SIGNAL_STATE property");
57 
58     mGearValue.prop       = static_cast<int32_t>(VehicleProperty::GEAR_SELECTION);
59     mTurnSignalValue.prop = static_cast<int32_t>(VehicleProperty::TURN_SIGNAL_STATE);
60 
61 #if 1
62     // This way we only ever deal with cameras which exist in the system
63     // Build our set of cameras for the states we support
64     ALOGD("Requesting camera list");
65     mEvs->getCameraList([this, &config](hidl_vec<CameraDesc> cameraList) {
66                             ALOGI("Camera list callback received %zu cameras",
67                                   cameraList.size());
68                             for (auto&& cam: cameraList) {
69                                 ALOGD("Found camera %s", cam.cameraId.c_str());
70                                 bool cameraConfigFound = false;
71 
72                                 // Check our configuration for information about this camera
73                                 // Note that a camera can have a compound function string
74                                 // such that a camera can be "right/reverse" and be used for both.
75                                 // If more than one camera is listed for a given function, we'll
76                                 // list all of them and let the UX/rendering logic use one, some
77                                 // or all of them as appropriate.
78                                 for (auto&& info: config.getCameras()) {
79                                     if (cam.cameraId == info.cameraId) {
80                                         // We found a match!
81                                         if (info.function.find("reverse") != std::string::npos) {
82                                             mCameraList[State::REVERSE].push_back(info);
83                                         }
84                                         if (info.function.find("right") != std::string::npos) {
85                                             mCameraList[State::RIGHT].push_back(info);
86                                         }
87                                         if (info.function.find("left") != std::string::npos) {
88                                             mCameraList[State::LEFT].push_back(info);
89                                         }
90                                         if (info.function.find("park") != std::string::npos) {
91                                             mCameraList[State::PARKING].push_back(info);
92                                         }
93                                         cameraConfigFound = true;
94                                         break;
95                                     }
96                                 }
97                                 if (!cameraConfigFound) {
98                                     ALOGW("No config information for hardware camera %s",
99                                           cam.cameraId.c_str());
100                                 }
101                             }
102                         }
103     );
104 #else // This way we use placeholders for cameras in the configuration but not reported by EVS
105     // Build our set of cameras for the states we support
106     ALOGD("Requesting camera list");
107     for (auto&& info: config.getCameras()) {
108         if (info.function.find("reverse") != std::string::npos) {
109             mCameraList[State::REVERSE].push_back(info);
110         }
111         if (info.function.find("right") != std::string::npos) {
112             mCameraList[State::RIGHT].push_back(info);
113         }
114         if (info.function.find("left") != std::string::npos) {
115             mCameraList[State::LEFT].push_back(info);
116         }
117         if (info.function.find("park") != std::string::npos) {
118             mCameraList[State::PARKING].push_back(info);
119         }
120     }
121 #endif
122 
123     ALOGD("State controller ready");
124 }
125 
126 
startUpdateLoop()127 bool EvsStateControl::startUpdateLoop() {
128     // Create the thread and report success if it gets started
129     mRenderThread = std::thread([this](){ updateLoop(); });
130     return mRenderThread.joinable();
131 }
132 
133 
postCommand(const Command & cmd)134 void EvsStateControl::postCommand(const Command& cmd) {
135     // Push the command onto the queue watched by updateLoop
136     mLock.lock();
137     mCommandQueue.push(cmd);
138     mLock.unlock();
139 
140     // Send a signal to wake updateLoop in case it is asleep
141     mWakeSignal.notify_all();
142 }
143 
144 
updateLoop()145 void EvsStateControl::updateLoop() {
146     ALOGD("Starting EvsStateControl update loop");
147 
148     bool run = true;
149     while (run) {
150         // Process incoming commands
151         {
152             std::lock_guard <std::mutex> lock(mLock);
153             while (!mCommandQueue.empty()) {
154                 const Command& cmd = mCommandQueue.front();
155                 switch (cmd.operation) {
156                 case Op::EXIT:
157                     run = false;
158                     break;
159                 case Op::CHECK_VEHICLE_STATE:
160                     // Just running selectStateForCurrentConditions below will take care of this
161                     break;
162                 case Op::TOUCH_EVENT:
163                     // Implement this given the x/y location of the touch event
164                     break;
165                 }
166                 mCommandQueue.pop();
167             }
168         }
169 
170         // Review vehicle state and choose an appropriate renderer
171         if (!selectStateForCurrentConditions()) {
172             ALOGE("selectStateForCurrentConditions failed so we're going to die");
173             break;
174         }
175 
176         // If we have an active renderer, give it a chance to draw
177         if (mCurrentRenderer) {
178             // Get the output buffer we'll use to display the imagery
179             BufferDesc tgtBuffer = {};
180             mDisplay->getTargetBuffer([&tgtBuffer](const BufferDesc& buff) {
181                                           tgtBuffer = buff;
182                                       }
183             );
184 
185             if (tgtBuffer.memHandle == nullptr) {
186                 ALOGE("Didn't get requested output buffer -- skipping this frame.");
187             } else {
188                 // Generate our output image
189                 if (!mCurrentRenderer->drawFrame(tgtBuffer)) {
190                     // If drawing failed, we want to exit quickly so an app restart can happen
191                     run = false;
192                 }
193 
194                 // Send the finished image back for display
195                 mDisplay->returnTargetBufferForDisplay(tgtBuffer);
196             }
197         } else {
198             // No active renderer, so sleep until somebody wakes us with another command
199             std::unique_lock<std::mutex> lock(mLock);
200             mWakeSignal.wait(lock);
201         }
202     }
203 
204     ALOGW("EvsStateControl update loop ending");
205 
206     // TODO:  Fix it so we can exit cleanly from the main thread instead
207     printf("Shutting down app due to state control loop ending\n");
208     ALOGE("KILLING THE APP FROM THE EvsStateControl LOOP ON DRAW FAILURE!!!");
209     exit(1);
210 }
211 
212 
selectStateForCurrentConditions()213 bool EvsStateControl::selectStateForCurrentConditions() {
214     static int32_t sDummyGear   = int32_t(VehicleGear::GEAR_REVERSE);
215     static int32_t sDummySignal = int32_t(VehicleTurnSignal::NONE);
216 
217     if (mVehicle != nullptr) {
218         // Query the car state
219         if (invokeGet(&mGearValue) != StatusCode::OK) {
220             ALOGE("GEAR_SELECTION not available from vehicle.  Exiting.");
221             return false;
222         }
223         if ((mTurnSignalValue.prop == 0) || (invokeGet(&mTurnSignalValue) != StatusCode::OK)) {
224             // Silently treat missing turn signal state as no turn signal active
225             mTurnSignalValue.value.int32Values.setToExternal(&sDummySignal, 1);
226             mTurnSignalValue.prop = 0;
227         }
228     } else {
229         // While testing without a vehicle, behave as if we're in reverse for the first 20 seconds
230         static const int kShowTime = 20;    // seconds
231 
232         // See if it's time to turn off the default reverse camera
233         static std::chrono::steady_clock::time_point start = std::chrono::steady_clock::now();
234         std::chrono::steady_clock::time_point now = std::chrono::steady_clock::now();
235         if (std::chrono::duration_cast<std::chrono::seconds>(now - start).count() > kShowTime) {
236             // Switch to drive (which should turn off the reverse camera)
237             sDummyGear = int32_t(VehicleGear::GEAR_DRIVE);
238         }
239 
240         // Build the placeholder vehicle state values (treating single values as 1 element vectors)
241         mGearValue.value.int32Values.setToExternal(&sDummyGear, 1);
242         mTurnSignalValue.value.int32Values.setToExternal(&sDummySignal, 1);
243     }
244 
245     // Choose our desired EVS state based on the current car state
246     // TODO:  Update this logic, and consider user input when choosing if a view should be presented
247     State desiredState = OFF;
248     if (mGearValue.value.int32Values[0] == int32_t(VehicleGear::GEAR_REVERSE)) {
249         desiredState = REVERSE;
250     } else if (mTurnSignalValue.value.int32Values[0] == int32_t(VehicleTurnSignal::RIGHT)) {
251         desiredState = RIGHT;
252     } else if (mTurnSignalValue.value.int32Values[0] == int32_t(VehicleTurnSignal::LEFT)) {
253         desiredState = LEFT;
254     } else if (mGearValue.value.int32Values[0] == int32_t(VehicleGear::GEAR_PARK)) {
255         desiredState = PARKING;
256     }
257 
258     // Apply the desire state
259     return configureEvsPipeline(desiredState);
260 }
261 
262 
invokeGet(VehiclePropValue * pRequestedPropValue)263 StatusCode EvsStateControl::invokeGet(VehiclePropValue *pRequestedPropValue) {
264     StatusCode status = StatusCode::TRY_AGAIN;
265 
266     // Call the Vehicle HAL, which will block until the callback is complete
267     mVehicle->get(*pRequestedPropValue,
268                   [pRequestedPropValue, &status]
269                   (StatusCode s, const VehiclePropValue& v) {
270                        status = s;
271                        if (s == StatusCode::OK) {
272                            *pRequestedPropValue = v;
273                        }
274                   }
275     );
276 
277     return status;
278 }
279 
280 
configureEvsPipeline(State desiredState)281 bool EvsStateControl::configureEvsPipeline(State desiredState) {
282     static bool isGlReady = false;
283 
284     if (mCurrentState == desiredState) {
285         // Nothing to do here...
286         return true;
287     }
288 
289     ALOGD("Switching to state %d.", desiredState);
290     ALOGD("  Current state %d has %zu cameras", mCurrentState,
291           mCameraList[mCurrentState].size());
292     ALOGD("  Desired state %d has %zu cameras", desiredState,
293           mCameraList[desiredState].size());
294 
295     if (!isGlReady && !isSfReady()) {
296         // Graphics is not ready yet; using CPU renderer.
297         if (mCameraList[desiredState].size() >= 1) {
298             mDesiredRenderer = std::make_unique<RenderPixelCopy>(mEvs,
299                                                                  mCameraList[desiredState][0]);
300             if (!mDesiredRenderer) {
301                 ALOGE("Failed to construct Pixel Copy renderer.  Skipping state change.");
302                 return false;
303             }
304         } else {
305             ALOGD("Unsupported, desiredState %d has %u cameras.",
306                   desiredState, static_cast<unsigned int>(mCameraList[desiredState].size()));
307         }
308     } else {
309         // Assumes that SurfaceFlinger is available always after being launched.
310 
311         // Do we need a new direct view renderer?
312         if (mCameraList[desiredState].size() == 1) {
313             // We have a camera assigned to this state for direct view.
314             mDesiredRenderer = std::make_unique<RenderDirectView>(mEvs,
315                                                                   mCameraList[desiredState][0]);
316             if (!mDesiredRenderer) {
317                 ALOGE("Failed to construct direct renderer.  Skipping state change.");
318                 return false;
319             }
320         } else if (mCameraList[desiredState].size() > 1 || desiredState == PARKING) {
321             mDesiredRenderer = std::make_unique<RenderTopView>(mEvs,
322                                                                mCameraList[desiredState],
323                                                                mConfig);
324             if (!mDesiredRenderer) {
325                 ALOGE("Failed to construct top view renderer.  Skipping state change.");
326                 return false;
327             }
328         } else {
329             ALOGD("Unsupported, desiredState %d has %u cameras.",
330                   desiredState, static_cast<unsigned int>(mCameraList[desiredState].size()));
331         }
332 
333         // GL renderer is now ready.
334         isGlReady = true;
335     }
336 
337     // Since we're changing states, shut down the current renderer
338     if (mCurrentRenderer != nullptr) {
339         mCurrentRenderer->deactivate();
340         mCurrentRenderer = nullptr; // It's a smart pointer, so destructs on assignment to null
341     }
342 
343     // Now set the display state based on whether we have a video feed to show
344     if (mDesiredRenderer == nullptr) {
345         ALOGD("Turning off the display");
346         mDisplay->setDisplayState(DisplayState::NOT_VISIBLE);
347     } else {
348         mCurrentRenderer = std::move(mDesiredRenderer);
349 
350         // Start the camera stream
351         ALOGD("EvsStartCameraStreamTiming start time: %" PRId64 "ms", android::elapsedRealtime());
352         if (!mCurrentRenderer->activate()) {
353             ALOGE("New renderer failed to activate");
354             return false;
355         }
356 
357         // Activate the display
358         ALOGD("EvsActivateDisplayTiming start time: %" PRId64 "ms", android::elapsedRealtime());
359         Return<EvsResult> result = mDisplay->setDisplayState(DisplayState::VISIBLE_ON_NEXT_FRAME);
360         if (result != EvsResult::OK) {
361             ALOGE("setDisplayState returned an error (%d)", (EvsResult)result);
362             return false;
363         }
364     }
365 
366     // Record our current state
367     ALOGI("Activated state %d.", desiredState);
368     mCurrentState = desiredState;
369 
370     return true;
371 }
372