/* * 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 "ICameraRecordingProxy" #include <camera/CameraUtils.h> #include <camera/ICameraRecordingProxy.h> #include <camera/ICameraRecordingProxyListener.h> #include <binder/IMemory.h> #include <binder/Parcel.h> #include <media/hardware/HardwareAPI.h> #include <stdint.h> #include <utils/Log.h> namespace android { enum { START_RECORDING = IBinder::FIRST_CALL_TRANSACTION, STOP_RECORDING, RELEASE_RECORDING_FRAME, RELEASE_RECORDING_FRAME_HANDLE, RELEASE_RECORDING_FRAME_HANDLE_BATCH, }; class BpCameraRecordingProxy: public BpInterface<ICameraRecordingProxy> { public: explicit BpCameraRecordingProxy(const sp<IBinder>& impl) : BpInterface<ICameraRecordingProxy>(impl) { } status_t startRecording(const sp<ICameraRecordingProxyListener>& listener) { ALOGV("startRecording"); Parcel data, reply; data.writeInterfaceToken(ICameraRecordingProxy::getInterfaceDescriptor()); data.writeStrongBinder(IInterface::asBinder(listener)); remote()->transact(START_RECORDING, data, &reply); return reply.readInt32(); } void stopRecording() { ALOGV("stopRecording"); Parcel data, reply; data.writeInterfaceToken(ICameraRecordingProxy::getInterfaceDescriptor()); remote()->transact(STOP_RECORDING, data, &reply); } void releaseRecordingFrame(const sp<IMemory>& mem) { ALOGV("releaseRecordingFrame"); Parcel data, reply; data.writeInterfaceToken(ICameraRecordingProxy::getInterfaceDescriptor()); data.writeStrongBinder(IInterface::asBinder(mem)); remote()->transact(RELEASE_RECORDING_FRAME, data, &reply); } void releaseRecordingFrameHandle(native_handle_t *handle) { ALOGV("releaseRecordingFrameHandle"); Parcel data, reply; data.writeInterfaceToken(ICameraRecordingProxy::getInterfaceDescriptor()); data.writeNativeHandle(handle); remote()->transact(RELEASE_RECORDING_FRAME_HANDLE, data, &reply); // Close the native handle because camera received a dup copy. native_handle_close(handle); native_handle_delete(handle); } void releaseRecordingFrameHandleBatch(const std::vector<native_handle_t*>& handles) { ALOGV("releaseRecordingFrameHandleBatch"); Parcel data, reply; data.writeInterfaceToken(ICameraRecordingProxy::getInterfaceDescriptor()); uint32_t n = handles.size(); data.writeUint32(n); for (auto& handle : handles) { data.writeNativeHandle(handle); } remote()->transact(RELEASE_RECORDING_FRAME_HANDLE_BATCH, data, &reply); // Close the native handle because camera received a dup copy. for (auto& handle : handles) { native_handle_close(handle); native_handle_delete(handle); } } }; IMPLEMENT_META_INTERFACE(CameraRecordingProxy, "android.hardware.ICameraRecordingProxy"); // ---------------------------------------------------------------------- status_t BnCameraRecordingProxy::onTransact( uint32_t code, const Parcel& data, Parcel* reply, uint32_t flags) { switch(code) { case START_RECORDING: { ALOGV("START_RECORDING"); CHECK_INTERFACE(ICameraRecordingProxy, data, reply); sp<ICameraRecordingProxyListener> listener = interface_cast<ICameraRecordingProxyListener>(data.readStrongBinder()); reply->writeInt32(startRecording(listener)); return NO_ERROR; } break; case STOP_RECORDING: { ALOGV("STOP_RECORDING"); CHECK_INTERFACE(ICameraRecordingProxy, data, reply); stopRecording(); return NO_ERROR; } break; case RELEASE_RECORDING_FRAME: { ALOGV("RELEASE_RECORDING_FRAME"); CHECK_INTERFACE(ICameraRecordingProxy, data, reply); sp<IMemory> mem = interface_cast<IMemory>(data.readStrongBinder()); releaseRecordingFrame(mem); return NO_ERROR; } break; case RELEASE_RECORDING_FRAME_HANDLE: { ALOGV("RELEASE_RECORDING_FRAME_HANDLE"); CHECK_INTERFACE(ICameraRecordingProxy, data, reply); // releaseRecordingFrameHandle will be responsble to close the native handle. releaseRecordingFrameHandle(data.readNativeHandle()); return NO_ERROR; } break; case RELEASE_RECORDING_FRAME_HANDLE_BATCH: { ALOGV("RELEASE_RECORDING_FRAME_HANDLE_BATCH"); CHECK_INTERFACE(ICameraRecordingProxy, 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<native_handle_t*> handles; handles.reserve(n); 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); } // releaseRecordingFrameHandleBatch will be responsble to close the native handle. releaseRecordingFrameHandleBatch(handles); return NO_ERROR; } break; default: return BBinder::onTransact(code, data, reply, flags); } } // ---------------------------------------------------------------------------- }; // namespace android