| /* |
| * Copyright (C) 2011 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. |
| */ |
| |
| //#define LOG_NDEBUG 0 |
| #define LOG_TAG "ICameraRecordingProxyListener" |
| #include <camera/CameraUtils.h> |
| #include <camera/ICameraRecordingProxyListener.h> |
| #include <binder/IMemory.h> |
| #include <binder/Parcel.h> |
| #include <media/hardware/HardwareAPI.h> |
| #include <utils/Log.h> |
| |
| namespace android { |
| |
| enum { |
| DATA_CALLBACK_TIMESTAMP = IBinder::FIRST_CALL_TRANSACTION, |
| RECORDING_FRAME_HANDLE_CALLBACK_TIMESTAMP, |
| RECORDING_FRAME_HANDLE_CALLBACK_TIMESTAMP_BATCH |
| }; |
| |
| class BpCameraRecordingProxyListener: public BpInterface<ICameraRecordingProxyListener> |
| { |
| public: |
| explicit BpCameraRecordingProxyListener(const sp<IBinder>& impl) |
| : BpInterface<ICameraRecordingProxyListener>(impl) |
| { |
| } |
| |
| void dataCallbackTimestamp(nsecs_t timestamp, int32_t msgType, const sp<IMemory>& imageData) |
| { |
| ALOGV("dataCallback"); |
| Parcel data, reply; |
| data.writeInterfaceToken(ICameraRecordingProxyListener::getInterfaceDescriptor()); |
| data.writeInt64(timestamp); |
| data.writeInt32(msgType); |
| data.writeStrongBinder(IInterface::asBinder(imageData)); |
| remote()->transact(DATA_CALLBACK_TIMESTAMP, data, &reply, IBinder::FLAG_ONEWAY); |
| } |
| |
| void recordingFrameHandleCallbackTimestamp(nsecs_t timestamp, native_handle_t* handle) { |
| ALOGV("recordingFrameHandleCallbackTimestamp"); |
| Parcel data, reply; |
| data.writeInterfaceToken(ICameraRecordingProxyListener::getInterfaceDescriptor()); |
| data.writeInt64(timestamp); |
| data.writeNativeHandle(handle); |
| remote()->transact(RECORDING_FRAME_HANDLE_CALLBACK_TIMESTAMP, data, &reply, |
| IBinder::FLAG_ONEWAY); |
| |
| // The native handle is dupped in ICameraClient so we need to free it here. |
| native_handle_close(handle); |
| native_handle_delete(handle); |
| } |
| |
| void recordingFrameHandleCallbackTimestampBatch( |
| const std::vector<nsecs_t>& timestamps, |
| const std::vector<native_handle_t*>& handles) { |
| ALOGV("recordingFrameHandleCallbackTimestampBatch"); |
| Parcel data, reply; |
| data.writeInterfaceToken(ICameraRecordingProxyListener::getInterfaceDescriptor()); |
| |
| uint32_t n = timestamps.size(); |
| if (n != handles.size()) { |
| ALOGE("%s: size of timestamps(%zu) and handles(%zu) mismatch!", |
| __FUNCTION__, timestamps.size(), handles.size()); |
| return; |
| } |
| data.writeUint32(n); |
| for (auto ts : timestamps) { |
| data.writeInt64(ts); |
| } |
| for (auto& handle : handles) { |
| data.writeNativeHandle(handle); |
| } |
| remote()->transact(RECORDING_FRAME_HANDLE_CALLBACK_TIMESTAMP_BATCH, data, &reply, |
| IBinder::FLAG_ONEWAY); |
| |
| // The native handle is dupped in ICameraClient so we need to free it here. |
| for (auto& handle : handles) { |
| native_handle_close(handle); |
| native_handle_delete(handle); |
| } |
| } |
| }; |
| |
| IMPLEMENT_META_INTERFACE(CameraRecordingProxyListener, "android.hardware.ICameraRecordingProxyListener"); |
| |
| // ---------------------------------------------------------------------- |
| |
| status_t BnCameraRecordingProxyListener::onTransact( |
| uint32_t code, const Parcel& data, Parcel* reply, uint32_t flags) |
| { |
| switch(code) { |
| case DATA_CALLBACK_TIMESTAMP: { |
| ALOGV("DATA_CALLBACK_TIMESTAMP"); |
| CHECK_INTERFACE(ICameraRecordingProxyListener, data, reply); |
| nsecs_t timestamp = data.readInt64(); |
| int32_t msgType = data.readInt32(); |
| sp<IMemory> imageData = interface_cast<IMemory>(data.readStrongBinder()); |
| dataCallbackTimestamp(timestamp, msgType, imageData); |
| return NO_ERROR; |
| } break; |
| case RECORDING_FRAME_HANDLE_CALLBACK_TIMESTAMP: { |
| ALOGV("RECORDING_FRAME_HANDLE_CALLBACK_TIMESTAMP"); |
| CHECK_INTERFACE(ICameraRecordingProxyListener, data, reply); |
| nsecs_t timestamp; |
| status_t res = data.readInt64(×tamp); |
| if (res != OK) { |
| ALOGE("%s: Failed to read timestamp: %s (%d)", __FUNCTION__, strerror(-res), res); |
| return BAD_VALUE; |
| } |
| |
| native_handle_t* handle = data.readNativeHandle(); |
| if (handle == nullptr) { |
| ALOGE("%s: Received a null native handle", __FUNCTION__); |
| return BAD_VALUE; |
| } |
| // The native handle will be freed in |
| // BpCameraRecordingProxy::releaseRecordingFrameHandle. |
| recordingFrameHandleCallbackTimestamp(timestamp, handle); |
| return NO_ERROR; |
| } break; |
| case RECORDING_FRAME_HANDLE_CALLBACK_TIMESTAMP_BATCH: { |
| ALOGV("RECORDING_FRAME_HANDLE_CALLBACK_TIMESTAMP_BATCH"); |
| CHECK_INTERFACE(ICameraRecordingProxyListener, data, reply); |
| uint32_t n = 0; |
| status_t res = data.readUint32(&n); |
| if (res != OK) { |
| ALOGE("%s: Failed to read batch size: %s (%d)", __FUNCTION__, strerror(-res), res); |
| return BAD_VALUE; |
| } |
| std::vector<nsecs_t> timestamps; |
| std::vector<native_handle_t*> handles; |
| timestamps.reserve(n); |
| handles.reserve(n); |
| for (uint32_t i = 0; i < n; i++) { |
| nsecs_t t; |
| res = data.readInt64(&t); |
| if (res != OK) { |
| ALOGE("%s: Failed to read timestamp[%d]: %s (%d)", |
| __FUNCTION__, i, strerror(-res), res); |
| return BAD_VALUE; |
| } |
| timestamps.push_back(t); |
| } |
| for (uint32_t i = 0; i < n; i++) { |
| native_handle_t* handle = data.readNativeHandle(); |
| if (handle == nullptr) { |
| ALOGE("%s: Received a null native handle at handles[%d]", |
| __FUNCTION__, i); |
| return BAD_VALUE; |
| } |
| handles.push_back(handle); |
| } |
| // The native handle will be freed in |
| // BpCameraRecordingProxy::releaseRecordingFrameHandleBatch. |
| recordingFrameHandleCallbackTimestampBatch(timestamps, handles); |
| return NO_ERROR; |
| } break; |
| default: |
| return BBinder::onTransact(code, data, reply, flags); |
| } |
| } |
| |
| // ---------------------------------------------------------------------------- |
| |
| }; // namespace android |
| |