Yin-Chia Yeh | 6a6fe0f | 2018-09-06 15:38:34 -0700 | [diff] [blame] | 1 | /* |
| 2 | * Copyright (C) 2018 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 | #define LOG_TAG "CamDevSession@3.5-impl" |
Yin-Chia Yeh | c2d3d1d | 2018-09-20 15:06:13 -0700 | [diff] [blame] | 18 | #define ATRACE_TAG ATRACE_TAG_CAMERA |
Yin-Chia Yeh | 6a6fe0f | 2018-09-06 15:38:34 -0700 | [diff] [blame] | 19 | #include <android/log.h> |
| 20 | |
Yin-Chia Yeh | c2d3d1d | 2018-09-20 15:06:13 -0700 | [diff] [blame] | 21 | #include <vector> |
Yin-Chia Yeh | 6a6fe0f | 2018-09-06 15:38:34 -0700 | [diff] [blame] | 22 | #include <utils/Trace.h> |
| 23 | #include "CameraDeviceSession.h" |
| 24 | |
| 25 | namespace android { |
| 26 | namespace hardware { |
| 27 | namespace camera { |
| 28 | namespace device { |
| 29 | namespace V3_5 { |
| 30 | namespace implementation { |
| 31 | |
| 32 | CameraDeviceSession::CameraDeviceSession( |
| 33 | camera3_device_t* device, |
| 34 | const camera_metadata_t* deviceInfo, |
| 35 | const sp<V3_2::ICameraDeviceCallback>& callback) : |
| 36 | V3_4::implementation::CameraDeviceSession(device, deviceInfo, callback) { |
| 37 | |
Yin-Chia Yeh | c2d3d1d | 2018-09-20 15:06:13 -0700 | [diff] [blame] | 38 | mCallback_3_5 = nullptr; |
Yin-Chia Yeh | 6a6fe0f | 2018-09-06 15:38:34 -0700 | [diff] [blame] | 39 | |
| 40 | auto castResult = ICameraDeviceCallback::castFrom(callback); |
| 41 | if (castResult.isOk()) { |
| 42 | sp<ICameraDeviceCallback> callback3_5 = castResult; |
| 43 | if (callback3_5 != nullptr) { |
Yin-Chia Yeh | c2d3d1d | 2018-09-20 15:06:13 -0700 | [diff] [blame] | 44 | mCallback_3_5 = callback3_5; |
| 45 | } |
| 46 | } |
| 47 | |
| 48 | if (mCallback_3_5 != nullptr) { |
| 49 | camera_metadata_entry bufMgrVersion = mDeviceInfo.find( |
| 50 | ANDROID_INFO_SUPPORTED_BUFFER_MANAGEMENT_VERSION); |
| 51 | if (bufMgrVersion.count > 0) { |
| 52 | mSupportBufMgr = (bufMgrVersion.data.u8[0] == |
| 53 | ANDROID_INFO_SUPPORTED_BUFFER_MANAGEMENT_VERSION_HIDL_DEVICE_3_5); |
| 54 | if (mSupportBufMgr) { |
| 55 | request_stream_buffers = sRequestStreamBuffers; |
| 56 | return_stream_buffers = sReturnStreamBuffers; |
| 57 | } |
Yin-Chia Yeh | 6a6fe0f | 2018-09-06 15:38:34 -0700 | [diff] [blame] | 58 | } |
| 59 | } |
| 60 | } |
| 61 | |
| 62 | CameraDeviceSession::~CameraDeviceSession() { |
| 63 | } |
| 64 | |
| 65 | Return<void> CameraDeviceSession::configureStreams_3_5( |
Yin-Chia Yeh | 86a7d96 | 2018-10-15 14:20:15 -0700 | [diff] [blame] | 66 | const StreamConfiguration& requestedConfiguration, |
Yin-Chia Yeh | 6a6fe0f | 2018-09-06 15:38:34 -0700 | [diff] [blame] | 67 | ICameraDeviceSession::configureStreams_3_5_cb _hidl_cb) { |
Yin-Chia Yeh | c2d3d1d | 2018-09-20 15:06:13 -0700 | [diff] [blame] | 68 | configureStreams_3_4_Impl(requestedConfiguration.v3_4, _hidl_cb, |
| 69 | requestedConfiguration.streamConfigCounter); |
| 70 | return Void(); |
Yin-Chia Yeh | 6a6fe0f | 2018-09-06 15:38:34 -0700 | [diff] [blame] | 71 | } |
| 72 | |
| 73 | Return<void> CameraDeviceSession::signalStreamFlush( |
Yin-Chia Yeh | c2d3d1d | 2018-09-20 15:06:13 -0700 | [diff] [blame] | 74 | const hidl_vec<int32_t>& streamIds, uint32_t streamConfigCounter) { |
| 75 | std::vector<camera3_stream_t*> streams(streamIds.size()); |
| 76 | { |
| 77 | Mutex::Autolock _l(mInflightLock); |
| 78 | for (size_t i = 0; i < streamIds.size(); i++) { |
| 79 | int32_t id = streamIds[i]; |
| 80 | if (mStreamMap.count(id) == 0) { |
| 81 | ALOGE("%s: unknown streamId %d", __FUNCTION__, id); |
| 82 | return Void(); |
| 83 | } |
| 84 | streams[i] = &mStreamMap[id]; |
| 85 | } |
| 86 | } |
| 87 | if (mDevice->ops->signal_stream_flush != nullptr) { |
| 88 | mDevice->ops->signal_stream_flush(mDevice, |
| 89 | streamConfigCounter, streams.size(), streams.data()); |
| 90 | } |
Yin-Chia Yeh | 6a6fe0f | 2018-09-06 15:38:34 -0700 | [diff] [blame] | 91 | return Void(); |
| 92 | } |
| 93 | |
Yin-Chia Yeh | c2d3d1d | 2018-09-20 15:06:13 -0700 | [diff] [blame] | 94 | Status CameraDeviceSession::importRequest( |
| 95 | const CaptureRequest& request, |
| 96 | hidl_vec<buffer_handle_t*>& allBufPtrs, |
| 97 | hidl_vec<int>& allFences) { |
| 98 | if (mSupportBufMgr) { |
| 99 | return importRequestImpl(request, allBufPtrs, allFences, /*allowEmptyBuf*/ true); |
| 100 | } |
| 101 | return importRequestImpl(request, allBufPtrs, allFences, /*allowEmptyBuf*/ false); |
| 102 | } |
| 103 | |
| 104 | void CameraDeviceSession::pushBufferId( |
| 105 | const buffer_handle_t& buf, uint64_t bufferId, int streamId) { |
| 106 | std::lock_guard<std::mutex> lock(mBufferIdMapLock); |
| 107 | |
| 108 | // emplace will return existing entry if there is one. |
| 109 | auto pair = mBufferIdMaps.emplace(streamId, BufferIdMap{}); |
| 110 | BufferIdMap& bIdMap = pair.first->second; |
| 111 | bIdMap[buf] = bufferId; |
| 112 | } |
| 113 | |
| 114 | uint64_t CameraDeviceSession::popBufferId( |
| 115 | const buffer_handle_t& buf, int streamId) { |
| 116 | std::lock_guard<std::mutex> lock(mBufferIdMapLock); |
| 117 | |
| 118 | auto streamIt = mBufferIdMaps.find(streamId); |
| 119 | if (streamIt == mBufferIdMaps.end()) { |
| 120 | return BUFFER_ID_NO_BUFFER; |
| 121 | } |
| 122 | BufferIdMap& bIdMap = streamIt->second; |
| 123 | auto it = bIdMap.find(buf); |
| 124 | if (it == bIdMap.end()) { |
| 125 | return BUFFER_ID_NO_BUFFER; |
| 126 | } |
| 127 | uint64_t bufId = it->second; |
| 128 | bIdMap.erase(it); |
| 129 | if (bIdMap.empty()) { |
| 130 | mBufferIdMaps.erase(streamIt); |
| 131 | } |
| 132 | return bufId; |
| 133 | } |
| 134 | |
| 135 | uint64_t CameraDeviceSession::getCapResultBufferId(const buffer_handle_t& buf, int streamId) { |
| 136 | if (mSupportBufMgr) { |
| 137 | return popBufferId(buf, streamId); |
| 138 | } |
| 139 | return BUFFER_ID_NO_BUFFER; |
| 140 | } |
| 141 | |
| 142 | Camera3Stream* CameraDeviceSession::getStreamPointer(int32_t streamId) { |
| 143 | Mutex::Autolock _l(mInflightLock); |
| 144 | if (mStreamMap.count(streamId) == 0) { |
| 145 | ALOGE("%s: unknown streamId %d", __FUNCTION__, streamId); |
| 146 | return nullptr; |
| 147 | } |
| 148 | return &mStreamMap[streamId]; |
| 149 | } |
| 150 | |
| 151 | void CameraDeviceSession::cleanupInflightBufferFences( |
| 152 | std::vector<int>& fences, std::vector<std::pair<buffer_handle_t, int>>& bufs) { |
| 153 | hidl_vec<int> hFences = fences; |
| 154 | cleanupInflightFences(hFences, fences.size()); |
| 155 | for (auto& p : bufs) { |
| 156 | popBufferId(p.first, p.second); |
| 157 | } |
| 158 | } |
| 159 | |
| 160 | camera3_buffer_request_status_t CameraDeviceSession::requestStreamBuffers( |
| 161 | uint32_t num_buffer_reqs, |
| 162 | const camera3_buffer_request_t *buffer_reqs, |
| 163 | /*out*/uint32_t *num_returned_buf_reqs, |
| 164 | /*out*/camera3_stream_buffer_ret_t *returned_buf_reqs) { |
| 165 | ATRACE_CALL(); |
| 166 | *num_returned_buf_reqs = 0; |
| 167 | hidl_vec<BufferRequest> hBufReqs(num_buffer_reqs); |
| 168 | for (size_t i = 0; i < num_buffer_reqs; i++) { |
| 169 | hBufReqs[i].streamId = |
| 170 | static_cast<Camera3Stream*>(buffer_reqs[i].stream)->mId; |
| 171 | hBufReqs[i].numBuffersRequested = buffer_reqs[i].num_buffers_requested; |
| 172 | } |
| 173 | |
| 174 | ATRACE_BEGIN("HIDL requestStreamBuffers"); |
| 175 | BufferRequestStatus status; |
| 176 | hidl_vec<StreamBufferRet> bufRets; |
| 177 | auto err = mCallback_3_5->requestStreamBuffers(hBufReqs, |
| 178 | [&status, &bufRets] |
| 179 | (BufferRequestStatus s, const hidl_vec<StreamBufferRet>& rets) { |
| 180 | status = s; |
| 181 | bufRets = std::move(rets); |
| 182 | }); |
| 183 | if (!err.isOk()) { |
| 184 | ALOGE("%s: Transaction error: %s", __FUNCTION__, err.description().c_str()); |
| 185 | return CAMERA3_BUF_REQ_FAILED_UNKNOWN; |
| 186 | } |
| 187 | ATRACE_END(); |
| 188 | |
| 189 | if (status == BufferRequestStatus::OK || status == BufferRequestStatus::FAILED_PARTIAL) { |
| 190 | if (bufRets.size() != num_buffer_reqs) { |
| 191 | ALOGE("%s: expect %d buffer requests returned, only got %zu", |
| 192 | __FUNCTION__, num_buffer_reqs, bufRets.size()); |
| 193 | return CAMERA3_BUF_REQ_FAILED_UNKNOWN; |
| 194 | } |
| 195 | |
| 196 | for (size_t i = 0; i < num_buffer_reqs; i++) { |
| 197 | // maybe we can query all streams in one call to avoid frequent locking device here? |
| 198 | Camera3Stream* stream = getStreamPointer(bufRets[i].streamId); |
| 199 | if (stream == nullptr) { |
| 200 | ALOGE("%s: unknown streamId %d", __FUNCTION__, bufRets[i].streamId); |
| 201 | return CAMERA3_BUF_REQ_FAILED_UNKNOWN; |
| 202 | } |
| 203 | returned_buf_reqs[i].stream = stream; |
| 204 | } |
| 205 | |
| 206 | std::vector<int> importedFences; |
| 207 | std::vector<std::pair<buffer_handle_t, int>> importedBuffers; |
| 208 | for (size_t i = 0; i < num_buffer_reqs; i++) { |
| 209 | int streamId = bufRets[i].streamId; |
| 210 | switch (bufRets[i].val.getDiscriminator()) { |
| 211 | case StreamBuffersVal::hidl_discriminator::error: |
| 212 | returned_buf_reqs[i].num_output_buffers = 0; |
| 213 | switch (bufRets[i].val.error()) { |
| 214 | case StreamBufferRequestError::NO_BUFFER_AVAILABLE: |
| 215 | returned_buf_reqs[i].status = CAMERA3_PS_BUF_REQ_NO_BUFFER_AVAILABLE; |
| 216 | break; |
| 217 | case StreamBufferRequestError::MAX_BUFFER_EXCEEDED: |
| 218 | returned_buf_reqs[i].status = CAMERA3_PS_BUF_REQ_MAX_BUFFER_EXCEEDED; |
| 219 | break; |
| 220 | case StreamBufferRequestError::STREAM_DISCONNECTED: |
| 221 | returned_buf_reqs[i].status = CAMERA3_PS_BUF_REQ_STREAM_DISCONNECTED; |
| 222 | break; |
| 223 | case StreamBufferRequestError::UNKNOWN_ERROR: |
| 224 | returned_buf_reqs[i].status = CAMERA3_PS_BUF_REQ_UNKNOWN_ERROR; |
| 225 | break; |
| 226 | default: |
| 227 | ALOGE("%s: Unknown StreamBufferRequestError %d", |
| 228 | __FUNCTION__, bufRets[i].val.error()); |
| 229 | cleanupInflightBufferFences(importedFences, importedBuffers); |
| 230 | return CAMERA3_BUF_REQ_FAILED_UNKNOWN; |
| 231 | } |
| 232 | break; |
| 233 | case StreamBuffersVal::hidl_discriminator::buffers: { |
| 234 | const hidl_vec<StreamBuffer>& hBufs = bufRets[i].val.buffers(); |
| 235 | camera3_stream_buffer_t* outBufs = returned_buf_reqs[i].output_buffers; |
| 236 | for (size_t b = 0; b < hBufs.size(); b++) { |
| 237 | const StreamBuffer& hBuf = hBufs[b]; |
| 238 | camera3_stream_buffer_t& outBuf = outBufs[b]; |
| 239 | // maybe add importBuffers API to avoid frequent locking device? |
| 240 | Status s = importBuffer(streamId, |
| 241 | hBuf.bufferId, hBuf.buffer.getNativeHandle(), |
| 242 | /*out*/&(outBuf.buffer), |
| 243 | /*allowEmptyBuf*/false); |
| 244 | if (s != Status::OK) { |
| 245 | ALOGE("%s: import stream %d bufferId %" PRIu64 " failed!", |
| 246 | __FUNCTION__, streamId, hBuf.bufferId); |
| 247 | cleanupInflightBufferFences(importedFences, importedBuffers); |
| 248 | // Buffer import should never fail - restart HAL since something is very |
| 249 | // wrong. |
| 250 | assert(false); |
| 251 | return CAMERA3_BUF_REQ_FAILED_UNKNOWN; |
| 252 | } |
| 253 | |
| 254 | pushBufferId(*(outBuf.buffer), hBuf.bufferId, streamId); |
| 255 | importedBuffers.push_back(std::make_pair(*(outBuf.buffer), streamId)); |
| 256 | |
| 257 | if (!sHandleImporter.importFence( |
| 258 | hBuf.acquireFence, |
| 259 | outBuf.acquire_fence)) { |
| 260 | ALOGE("%s: stream %d bufferId %" PRIu64 "acquire fence is invalid", |
| 261 | __FUNCTION__, streamId, hBuf.bufferId); |
| 262 | cleanupInflightBufferFences(importedFences, importedBuffers); |
| 263 | return CAMERA3_BUF_REQ_FAILED_UNKNOWN; |
| 264 | } |
| 265 | importedFences.push_back(outBuf.acquire_fence); |
| 266 | outBuf.stream = returned_buf_reqs[i].stream; |
| 267 | outBuf.status = CAMERA3_BUFFER_STATUS_OK; |
| 268 | outBuf.release_fence = -1; |
| 269 | } |
| 270 | returned_buf_reqs[i].status = CAMERA3_PS_BUF_REQ_OK; |
| 271 | } break; |
| 272 | default: |
| 273 | ALOGE("%s: unknown StreamBuffersVal discrimator!", __FUNCTION__); |
| 274 | cleanupInflightBufferFences(importedFences, importedBuffers); |
| 275 | return CAMERA3_BUF_REQ_FAILED_UNKNOWN; |
| 276 | } |
| 277 | } |
| 278 | |
| 279 | *num_returned_buf_reqs = num_buffer_reqs; |
| 280 | |
| 281 | return (status == BufferRequestStatus::OK) ? |
| 282 | CAMERA3_BUF_REQ_OK : CAMERA3_BUF_REQ_FAILED_PARTIAL; |
| 283 | } |
| 284 | |
| 285 | switch (status) { |
| 286 | case BufferRequestStatus::FAILED_CONFIGURING: |
| 287 | return CAMERA3_BUF_REQ_FAILED_CONFIGURING; |
| 288 | case BufferRequestStatus::FAILED_ILLEGAL_ARGUMENTS: |
| 289 | return CAMERA3_BUF_REQ_FAILED_ILLEGAL_ARGUMENTS; |
| 290 | case BufferRequestStatus::FAILED_UNKNOWN: |
| 291 | default: |
| 292 | return CAMERA3_BUF_REQ_FAILED_UNKNOWN; |
| 293 | } |
| 294 | } |
| 295 | |
| 296 | void CameraDeviceSession::returnStreamBuffers( |
| 297 | uint32_t num_buffers, |
| 298 | const camera3_stream_buffer_t* const* buffers) { |
| 299 | ATRACE_CALL(); |
| 300 | hidl_vec<StreamBuffer> hBufs(num_buffers); |
| 301 | |
| 302 | for (size_t i = 0; i < num_buffers; i++) { |
| 303 | hBufs[i].streamId = |
| 304 | static_cast<Camera3Stream*>(buffers[i]->stream)->mId; |
| 305 | hBufs[i].buffer = nullptr; // use bufferId |
| 306 | hBufs[i].bufferId = popBufferId(*(buffers[i]->buffer), hBufs[i].streamId); |
| 307 | if (hBufs[i].bufferId == BUFFER_ID_NO_BUFFER) { |
| 308 | ALOGE("%s: unknown buffer is returned to stream %d", |
| 309 | __FUNCTION__, hBufs[i].streamId); |
| 310 | } |
| 311 | // ERROR since the buffer is not for application to consume |
| 312 | hBufs[i].status = BufferStatus::ERROR; |
| 313 | // skip acquire fence since it's of no use to camera service |
| 314 | if (buffers[i]->release_fence != -1) { |
| 315 | native_handle_t* handle = native_handle_create(/*numFds*/1, /*numInts*/0); |
| 316 | handle->data[0] = buffers[i]->release_fence; |
| 317 | hBufs[i].releaseFence.setTo(handle, /*shouldOwn*/true); |
| 318 | } |
| 319 | } |
| 320 | |
| 321 | mCallback_3_5->returnStreamBuffers(hBufs); |
| 322 | return; |
| 323 | } |
| 324 | |
| 325 | /** |
| 326 | * Static callback forwarding methods from HAL to instance |
| 327 | */ |
| 328 | camera3_buffer_request_status_t CameraDeviceSession::sRequestStreamBuffers( |
| 329 | const struct camera3_callback_ops *cb, |
| 330 | uint32_t num_buffer_reqs, |
| 331 | const camera3_buffer_request_t *buffer_reqs, |
| 332 | /*out*/uint32_t *num_returned_buf_reqs, |
| 333 | /*out*/camera3_stream_buffer_ret_t *returned_buf_reqs) { |
| 334 | CameraDeviceSession *d = |
| 335 | const_cast<CameraDeviceSession*>(static_cast<const CameraDeviceSession*>(cb)); |
| 336 | |
| 337 | if (num_buffer_reqs == 0 || buffer_reqs == nullptr || num_returned_buf_reqs == nullptr || |
| 338 | returned_buf_reqs == nullptr) { |
| 339 | ALOGE("%s: bad argument: numBufReq %d, bufReqs %p, numRetBufReq %p, retBufReqs %p", |
| 340 | __FUNCTION__, num_buffer_reqs, buffer_reqs, |
| 341 | num_returned_buf_reqs, returned_buf_reqs); |
| 342 | return CAMERA3_BUF_REQ_FAILED_ILLEGAL_ARGUMENTS; |
| 343 | } |
| 344 | |
| 345 | return d->requestStreamBuffers(num_buffer_reqs, buffer_reqs, |
| 346 | num_returned_buf_reqs, returned_buf_reqs); |
| 347 | } |
| 348 | |
| 349 | void CameraDeviceSession::sReturnStreamBuffers( |
| 350 | const struct camera3_callback_ops *cb, |
| 351 | uint32_t num_buffers, |
| 352 | const camera3_stream_buffer_t* const* buffers) { |
| 353 | CameraDeviceSession *d = |
| 354 | const_cast<CameraDeviceSession*>(static_cast<const CameraDeviceSession*>(cb)); |
| 355 | |
| 356 | d->returnStreamBuffers(num_buffers, buffers); |
| 357 | } |
| 358 | |
Emilian Peev | 22eac5f | 2019-01-23 11:14:11 -0800 | [diff] [blame] | 359 | Return<void> CameraDeviceSession::isReconfigurationRequired( |
| 360 | const V3_2::CameraMetadata& oldSessionParams, const V3_2::CameraMetadata& newSessionParams, |
| 361 | ICameraDeviceSession::isReconfigurationRequired_cb _hidl_cb) { |
| 362 | if (mDevice->ops->is_reconfiguration_required != nullptr) { |
| 363 | const camera_metadata_t *oldParams, *newParams; |
| 364 | V3_2::implementation::convertFromHidl(oldSessionParams, &oldParams); |
| 365 | V3_2::implementation::convertFromHidl(newSessionParams, &newParams); |
| 366 | auto ret = mDevice->ops->is_reconfiguration_required(mDevice, oldParams, newParams); |
| 367 | switch (ret) { |
| 368 | case 0: |
| 369 | _hidl_cb(Status::OK, true); |
| 370 | break; |
| 371 | case -EINVAL: |
| 372 | _hidl_cb(Status::OK, false); |
| 373 | break; |
| 374 | case -ENOSYS: |
| 375 | _hidl_cb(Status::METHOD_NOT_SUPPORTED, true); |
| 376 | break; |
| 377 | default: |
| 378 | _hidl_cb(Status::INTERNAL_ERROR, true); |
| 379 | break; |
| 380 | }; |
| 381 | } else { |
| 382 | _hidl_cb(Status::METHOD_NOT_SUPPORTED, true); |
| 383 | } |
| 384 | |
| 385 | return Void(); |
| 386 | } |
| 387 | |
Yin-Chia Yeh | 6a6fe0f | 2018-09-06 15:38:34 -0700 | [diff] [blame] | 388 | } // namespace implementation |
| 389 | } // namespace V3_5 |
| 390 | } // namespace device |
| 391 | } // namespace camera |
| 392 | } // namespace hardware |
| 393 | } // namespace android |