/* * 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 #include #include #include #include #include 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 { public: explicit BpCameraRecordingProxyListener(const sp& impl) : BpInterface(impl) { } void dataCallbackTimestamp(nsecs_t timestamp, int32_t msgType, const sp& 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& timestamps, const std::vector& 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 imageData = interface_cast(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 timestamps; std::vector 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