| /* | 
 | ** | 
 | ** Copyright 2008, 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 "ICamera" | 
 | #include <utils/Log.h> | 
 | #include <stdint.h> | 
 | #include <sys/types.h> | 
 | #include <binder/Parcel.h> | 
 | #include <camera/CameraUtils.h> | 
 | #include <android/hardware/ICamera.h> | 
 | #include <android/hardware/ICameraClient.h> | 
 | #include <gui/IGraphicBufferProducer.h> | 
 | #include <gui/Surface.h> | 
 | #include <media/hardware/HardwareAPI.h> | 
 |  | 
 | namespace android { | 
 | namespace hardware { | 
 |  | 
 | enum { | 
 |     DISCONNECT = IBinder::FIRST_CALL_TRANSACTION, | 
 |     SET_PREVIEW_TARGET, | 
 |     SET_PREVIEW_CALLBACK_FLAG, | 
 |     SET_PREVIEW_CALLBACK_TARGET, | 
 |     START_PREVIEW, | 
 |     STOP_PREVIEW, | 
 |     AUTO_FOCUS, | 
 |     CANCEL_AUTO_FOCUS, | 
 |     TAKE_PICTURE, | 
 |     SET_PARAMETERS, | 
 |     GET_PARAMETERS, | 
 |     SEND_COMMAND, | 
 |     CONNECT, | 
 |     LOCK, | 
 |     UNLOCK, | 
 |     PREVIEW_ENABLED, | 
 |     START_RECORDING, | 
 |     STOP_RECORDING, | 
 |     RECORDING_ENABLED, | 
 |     RELEASE_RECORDING_FRAME, | 
 |     SET_VIDEO_BUFFER_MODE, | 
 |     SET_VIDEO_BUFFER_TARGET, | 
 |     RELEASE_RECORDING_FRAME_HANDLE, | 
 |     RELEASE_RECORDING_FRAME_HANDLE_BATCH, | 
 |     SET_AUDIO_RESTRICTION, | 
 |     GET_GLOBAL_AUDIO_RESTRICTION, | 
 | }; | 
 |  | 
 | class BpCamera: public BpInterface<ICamera> | 
 | { | 
 | public: | 
 |     explicit BpCamera(const sp<IBinder>& impl) | 
 |         : BpInterface<ICamera>(impl) | 
 |     { | 
 |     } | 
 |  | 
 |     // disconnect from camera service | 
 |     binder::Status disconnect() | 
 |     { | 
 |         ALOGV("disconnect"); | 
 |         Parcel data, reply; | 
 |         data.writeInterfaceToken(ICamera::getInterfaceDescriptor()); | 
 |         remote()->transact(DISCONNECT, data, &reply); | 
 |         reply.readExceptionCode(); | 
 |         return binder::Status::ok(); | 
 |     } | 
 |  | 
 |     // pass the buffered IGraphicBufferProducer to the camera service | 
 |     status_t setPreviewTarget(const sp<IGraphicBufferProducer>& bufferProducer) | 
 |     { | 
 |         ALOGV("setPreviewTarget"); | 
 |         Parcel data, reply; | 
 |         data.writeInterfaceToken(ICamera::getInterfaceDescriptor()); | 
 |         sp<IBinder> b(IInterface::asBinder(bufferProducer)); | 
 |         data.writeStrongBinder(b); | 
 |         remote()->transact(SET_PREVIEW_TARGET, data, &reply); | 
 |         return reply.readInt32(); | 
 |     } | 
 |  | 
 |     // set the preview callback flag to affect how the received frames from | 
 |     // preview are handled. See Camera.h for details. | 
 |     void setPreviewCallbackFlag(int flag) | 
 |     { | 
 |         ALOGV("setPreviewCallbackFlag(%d)", flag); | 
 |         Parcel data, reply; | 
 |         data.writeInterfaceToken(ICamera::getInterfaceDescriptor()); | 
 |         data.writeInt32(flag); | 
 |         remote()->transact(SET_PREVIEW_CALLBACK_FLAG, data, &reply); | 
 |     } | 
 |  | 
 |     status_t setPreviewCallbackTarget( | 
 |             const sp<IGraphicBufferProducer>& callbackProducer) | 
 |     { | 
 |         ALOGV("setPreviewCallbackTarget"); | 
 |         Parcel data, reply; | 
 |         data.writeInterfaceToken(ICamera::getInterfaceDescriptor()); | 
 |         sp<IBinder> b(IInterface::asBinder(callbackProducer)); | 
 |         data.writeStrongBinder(b); | 
 |         remote()->transact(SET_PREVIEW_CALLBACK_TARGET, data, &reply); | 
 |         return reply.readInt32(); | 
 |     } | 
 |  | 
 |     // start preview mode, must call setPreviewTarget first | 
 |     status_t startPreview() | 
 |     { | 
 |         ALOGV("startPreview"); | 
 |         Parcel data, reply; | 
 |         data.writeInterfaceToken(ICamera::getInterfaceDescriptor()); | 
 |         remote()->transact(START_PREVIEW, data, &reply); | 
 |         return reply.readInt32(); | 
 |     } | 
 |  | 
 |     // start recording mode, must call setPreviewTarget first | 
 |     status_t startRecording() | 
 |     { | 
 |         ALOGV("startRecording"); | 
 |         Parcel data, reply; | 
 |         data.writeInterfaceToken(ICamera::getInterfaceDescriptor()); | 
 |         remote()->transact(START_RECORDING, data, &reply); | 
 |         return reply.readInt32(); | 
 |     } | 
 |  | 
 |     // stop preview mode | 
 |     void stopPreview() | 
 |     { | 
 |         ALOGV("stopPreview"); | 
 |         Parcel data, reply; | 
 |         data.writeInterfaceToken(ICamera::getInterfaceDescriptor()); | 
 |         remote()->transact(STOP_PREVIEW, data, &reply); | 
 |     } | 
 |  | 
 |     // stop recording mode | 
 |     void stopRecording() | 
 |     { | 
 |         ALOGV("stopRecording"); | 
 |         Parcel data, reply; | 
 |         data.writeInterfaceToken(ICamera::getInterfaceDescriptor()); | 
 |         remote()->transact(STOP_RECORDING, data, &reply); | 
 |     } | 
 |  | 
 |     void releaseRecordingFrame(const sp<IMemory>& mem) | 
 |     { | 
 |         ALOGV("releaseRecordingFrame"); | 
 |         Parcel data, reply; | 
 |         data.writeInterfaceToken(ICamera::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(ICamera::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(ICamera::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); | 
 |         } | 
 |     } | 
 |  | 
 |     status_t setAudioRestriction(int32_t mode) { | 
 |         Parcel data, reply; | 
 |         data.writeInterfaceToken(ICamera::getInterfaceDescriptor()); | 
 |         data.writeInt32(mode); | 
 |         remote()->transact(SET_AUDIO_RESTRICTION, data, &reply); | 
 |         return reply.readInt32(); | 
 |     } | 
 |  | 
 |     int32_t getGlobalAudioRestriction() { | 
 |         Parcel data, reply; | 
 |         data.writeInterfaceToken(ICamera::getInterfaceDescriptor()); | 
 |         remote()->transact(GET_GLOBAL_AUDIO_RESTRICTION, data, &reply); | 
 |         return reply.readInt32(); | 
 |     } | 
 |  | 
 |     status_t setVideoBufferMode(int32_t videoBufferMode) | 
 |     { | 
 |         ALOGV("setVideoBufferMode: %d", videoBufferMode); | 
 |         Parcel data, reply; | 
 |         data.writeInterfaceToken(ICamera::getInterfaceDescriptor()); | 
 |         data.writeInt32(videoBufferMode); | 
 |         remote()->transact(SET_VIDEO_BUFFER_MODE, data, &reply); | 
 |         return reply.readInt32(); | 
 |     } | 
 |  | 
 |     // check preview state | 
 |     bool previewEnabled() | 
 |     { | 
 |         ALOGV("previewEnabled"); | 
 |         Parcel data, reply; | 
 |         data.writeInterfaceToken(ICamera::getInterfaceDescriptor()); | 
 |         remote()->transact(PREVIEW_ENABLED, data, &reply); | 
 |         return reply.readInt32(); | 
 |     } | 
 |  | 
 |     // check recording state | 
 |     bool recordingEnabled() | 
 |     { | 
 |         ALOGV("recordingEnabled"); | 
 |         Parcel data, reply; | 
 |         data.writeInterfaceToken(ICamera::getInterfaceDescriptor()); | 
 |         remote()->transact(RECORDING_ENABLED, data, &reply); | 
 |         return reply.readInt32(); | 
 |     } | 
 |  | 
 |     // auto focus | 
 |     status_t autoFocus() | 
 |     { | 
 |         ALOGV("autoFocus"); | 
 |         Parcel data, reply; | 
 |         data.writeInterfaceToken(ICamera::getInterfaceDescriptor()); | 
 |         remote()->transact(AUTO_FOCUS, data, &reply); | 
 |         status_t ret = reply.readInt32(); | 
 |         return ret; | 
 |     } | 
 |  | 
 |     // cancel focus | 
 |     status_t cancelAutoFocus() | 
 |     { | 
 |         ALOGV("cancelAutoFocus"); | 
 |         Parcel data, reply; | 
 |         data.writeInterfaceToken(ICamera::getInterfaceDescriptor()); | 
 |         remote()->transact(CANCEL_AUTO_FOCUS, data, &reply); | 
 |         status_t ret = reply.readInt32(); | 
 |         return ret; | 
 |     } | 
 |  | 
 |     // take a picture - returns an IMemory (ref-counted mmap) | 
 |     status_t takePicture(int msgType) | 
 |     { | 
 |         ALOGV("takePicture: 0x%x", msgType); | 
 |         Parcel data, reply; | 
 |         data.writeInterfaceToken(ICamera::getInterfaceDescriptor()); | 
 |         data.writeInt32(msgType); | 
 |         remote()->transact(TAKE_PICTURE, data, &reply); | 
 |         status_t ret = reply.readInt32(); | 
 |         return ret; | 
 |     } | 
 |  | 
 |     // set preview/capture parameters - key/value pairs | 
 |     status_t setParameters(const String8& params) | 
 |     { | 
 |         ALOGV("setParameters"); | 
 |         Parcel data, reply; | 
 |         data.writeInterfaceToken(ICamera::getInterfaceDescriptor()); | 
 |         data.writeString8(params); | 
 |         remote()->transact(SET_PARAMETERS, data, &reply); | 
 |         return reply.readInt32(); | 
 |     } | 
 |  | 
 |     // get preview/capture parameters - key/value pairs | 
 |     String8 getParameters() const | 
 |     { | 
 |         ALOGV("getParameters"); | 
 |         Parcel data, reply; | 
 |         data.writeInterfaceToken(ICamera::getInterfaceDescriptor()); | 
 |         remote()->transact(GET_PARAMETERS, data, &reply); | 
 |         return reply.readString8(); | 
 |     } | 
 |     virtual status_t sendCommand(int32_t cmd, int32_t arg1, int32_t arg2) | 
 |     { | 
 |         ALOGV("sendCommand"); | 
 |         Parcel data, reply; | 
 |         data.writeInterfaceToken(ICamera::getInterfaceDescriptor()); | 
 |         data.writeInt32(cmd); | 
 |         data.writeInt32(arg1); | 
 |         data.writeInt32(arg2); | 
 |         remote()->transact(SEND_COMMAND, data, &reply); | 
 |         return reply.readInt32(); | 
 |     } | 
 |     virtual status_t connect(const sp<ICameraClient>& cameraClient) | 
 |     { | 
 |         Parcel data, reply; | 
 |         data.writeInterfaceToken(ICamera::getInterfaceDescriptor()); | 
 |         data.writeStrongBinder(IInterface::asBinder(cameraClient)); | 
 |         remote()->transact(CONNECT, data, &reply); | 
 |         return reply.readInt32(); | 
 |     } | 
 |     virtual status_t lock() | 
 |     { | 
 |         Parcel data, reply; | 
 |         data.writeInterfaceToken(ICamera::getInterfaceDescriptor()); | 
 |         remote()->transact(LOCK, data, &reply); | 
 |         return reply.readInt32(); | 
 |     } | 
 |     virtual status_t unlock() | 
 |     { | 
 |         Parcel data, reply; | 
 |         data.writeInterfaceToken(ICamera::getInterfaceDescriptor()); | 
 |         remote()->transact(UNLOCK, data, &reply); | 
 |         return reply.readInt32(); | 
 |     } | 
 |  | 
 |     status_t setVideoTarget(const sp<IGraphicBufferProducer>& bufferProducer) | 
 |     { | 
 |         ALOGV("setVideoTarget"); | 
 |         Parcel data, reply; | 
 |         data.writeInterfaceToken(ICamera::getInterfaceDescriptor()); | 
 |         sp<IBinder> b(IInterface::asBinder(bufferProducer)); | 
 |         data.writeStrongBinder(b); | 
 |         remote()->transact(SET_VIDEO_BUFFER_TARGET, data, &reply); | 
 |         return reply.readInt32(); | 
 |     } | 
 | }; | 
 |  | 
 | IMPLEMENT_META_INTERFACE(Camera, "android.hardware.ICamera"); | 
 |  | 
 | // ---------------------------------------------------------------------- | 
 |  | 
 | status_t BnCamera::onTransact( | 
 |     uint32_t code, const Parcel& data, Parcel* reply, uint32_t flags) | 
 | { | 
 |     switch(code) { | 
 |         case DISCONNECT: { | 
 |             ALOGV("DISCONNECT"); | 
 |             CHECK_INTERFACE(ICamera, data, reply); | 
 |             disconnect(); | 
 |             reply->writeNoException(); | 
 |             return NO_ERROR; | 
 |         } break; | 
 |         case SET_PREVIEW_TARGET: { | 
 |             ALOGV("SET_PREVIEW_TARGET"); | 
 |             CHECK_INTERFACE(ICamera, data, reply); | 
 |             sp<IGraphicBufferProducer> st = | 
 |                 interface_cast<IGraphicBufferProducer>(data.readStrongBinder()); | 
 |             reply->writeInt32(setPreviewTarget(st)); | 
 |             return NO_ERROR; | 
 |         } break; | 
 |         case SET_PREVIEW_CALLBACK_FLAG: { | 
 |             ALOGV("SET_PREVIEW_CALLBACK_TYPE"); | 
 |             CHECK_INTERFACE(ICamera, data, reply); | 
 |             int callback_flag = data.readInt32(); | 
 |             setPreviewCallbackFlag(callback_flag); | 
 |             return NO_ERROR; | 
 |         } break; | 
 |         case SET_PREVIEW_CALLBACK_TARGET: { | 
 |             ALOGV("SET_PREVIEW_CALLBACK_TARGET"); | 
 |             CHECK_INTERFACE(ICamera, data, reply); | 
 |             sp<IGraphicBufferProducer> cp = | 
 |                 interface_cast<IGraphicBufferProducer>(data.readStrongBinder()); | 
 |             reply->writeInt32(setPreviewCallbackTarget(cp)); | 
 |             return NO_ERROR; | 
 |         } | 
 |         case START_PREVIEW: { | 
 |             ALOGV("START_PREVIEW"); | 
 |             CHECK_INTERFACE(ICamera, data, reply); | 
 |             reply->writeInt32(startPreview()); | 
 |             return NO_ERROR; | 
 |         } break; | 
 |         case START_RECORDING: { | 
 |             ALOGV("START_RECORDING"); | 
 |             CHECK_INTERFACE(ICamera, data, reply); | 
 |             reply->writeInt32(startRecording()); | 
 |             return NO_ERROR; | 
 |         } break; | 
 |         case STOP_PREVIEW: { | 
 |             ALOGV("STOP_PREVIEW"); | 
 |             CHECK_INTERFACE(ICamera, data, reply); | 
 |             stopPreview(); | 
 |             return NO_ERROR; | 
 |         } break; | 
 |         case STOP_RECORDING: { | 
 |             ALOGV("STOP_RECORDING"); | 
 |             CHECK_INTERFACE(ICamera, data, reply); | 
 |             stopRecording(); | 
 |             return NO_ERROR; | 
 |         } break; | 
 |         case RELEASE_RECORDING_FRAME: { | 
 |             ALOGV("RELEASE_RECORDING_FRAME"); | 
 |             CHECK_INTERFACE(ICamera, 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(ICamera, 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(ICamera, data, reply); | 
 |             // releaseRecordingFrameHandle will be responsble to close the native handle. | 
 |             uint32_t n = data.readUint32(); | 
 |             std::vector<native_handle_t*> handles; | 
 |             handles.reserve(n); | 
 |             for (uint32_t i = 0; i < n; i++) { | 
 |                 handles.push_back(data.readNativeHandle()); | 
 |             } | 
 |             releaseRecordingFrameHandleBatch(handles); | 
 |             return NO_ERROR; | 
 |         } break; | 
 |         case SET_VIDEO_BUFFER_MODE: { | 
 |             ALOGV("SET_VIDEO_BUFFER_MODE"); | 
 |             CHECK_INTERFACE(ICamera, data, reply); | 
 |             int32_t mode = data.readInt32(); | 
 |             reply->writeInt32(setVideoBufferMode(mode)); | 
 |             return NO_ERROR; | 
 |         } break; | 
 |         case PREVIEW_ENABLED: { | 
 |             ALOGV("PREVIEW_ENABLED"); | 
 |             CHECK_INTERFACE(ICamera, data, reply); | 
 |             reply->writeInt32(previewEnabled()); | 
 |             return NO_ERROR; | 
 |         } break; | 
 |         case RECORDING_ENABLED: { | 
 |             ALOGV("RECORDING_ENABLED"); | 
 |             CHECK_INTERFACE(ICamera, data, reply); | 
 |             reply->writeInt32(recordingEnabled()); | 
 |             return NO_ERROR; | 
 |         } break; | 
 |         case AUTO_FOCUS: { | 
 |             ALOGV("AUTO_FOCUS"); | 
 |             CHECK_INTERFACE(ICamera, data, reply); | 
 |             reply->writeInt32(autoFocus()); | 
 |             return NO_ERROR; | 
 |         } break; | 
 |         case CANCEL_AUTO_FOCUS: { | 
 |             ALOGV("CANCEL_AUTO_FOCUS"); | 
 |             CHECK_INTERFACE(ICamera, data, reply); | 
 |             reply->writeInt32(cancelAutoFocus()); | 
 |             return NO_ERROR; | 
 |         } break; | 
 |         case TAKE_PICTURE: { | 
 |             ALOGV("TAKE_PICTURE"); | 
 |             CHECK_INTERFACE(ICamera, data, reply); | 
 |             int msgType = data.readInt32(); | 
 |             reply->writeInt32(takePicture(msgType)); | 
 |             return NO_ERROR; | 
 |         } break; | 
 |         case SET_PARAMETERS: { | 
 |             ALOGV("SET_PARAMETERS"); | 
 |             CHECK_INTERFACE(ICamera, data, reply); | 
 |             String8 params(data.readString8()); | 
 |             reply->writeInt32(setParameters(params)); | 
 |             return NO_ERROR; | 
 |          } break; | 
 |         case GET_PARAMETERS: { | 
 |             ALOGV("GET_PARAMETERS"); | 
 |             CHECK_INTERFACE(ICamera, data, reply); | 
 |              reply->writeString8(getParameters()); | 
 |             return NO_ERROR; | 
 |          } break; | 
 |         case SEND_COMMAND: { | 
 |             ALOGV("SEND_COMMAND"); | 
 |             CHECK_INTERFACE(ICamera, data, reply); | 
 |             int command = data.readInt32(); | 
 |             int arg1 = data.readInt32(); | 
 |             int arg2 = data.readInt32(); | 
 |             reply->writeInt32(sendCommand(command, arg1, arg2)); | 
 |             return NO_ERROR; | 
 |          } break; | 
 |         case CONNECT: { | 
 |             CHECK_INTERFACE(ICamera, data, reply); | 
 |             sp<ICameraClient> cameraClient = interface_cast<ICameraClient>(data.readStrongBinder()); | 
 |             reply->writeInt32(connect(cameraClient)); | 
 |             return NO_ERROR; | 
 |         } break; | 
 |         case LOCK: { | 
 |             CHECK_INTERFACE(ICamera, data, reply); | 
 |             reply->writeInt32(lock()); | 
 |             return NO_ERROR; | 
 |         } break; | 
 |         case UNLOCK: { | 
 |             CHECK_INTERFACE(ICamera, data, reply); | 
 |             reply->writeInt32(unlock()); | 
 |             return NO_ERROR; | 
 |         } break; | 
 |         case SET_VIDEO_BUFFER_TARGET: { | 
 |             ALOGV("SET_VIDEO_BUFFER_TARGET"); | 
 |             CHECK_INTERFACE(ICamera, data, reply); | 
 |             sp<IGraphicBufferProducer> st = | 
 |                 interface_cast<IGraphicBufferProducer>(data.readStrongBinder()); | 
 |             reply->writeInt32(setVideoTarget(st)); | 
 |             return NO_ERROR; | 
 |         } break; | 
 |         case SET_AUDIO_RESTRICTION: { | 
 |             CHECK_INTERFACE(ICamera, data, reply); | 
 |             int32_t mode = data.readInt32(); | 
 |             reply->writeInt32(setAudioRestriction(mode)); | 
 |             return NO_ERROR; | 
 |         } break; | 
 |         case GET_GLOBAL_AUDIO_RESTRICTION: { | 
 |             CHECK_INTERFACE(ICamera, data, reply); | 
 |             reply->writeInt32(getGlobalAudioRestriction()); | 
 |             return NO_ERROR; | 
 |         } break; | 
 |         default: | 
 |             return BBinder::onTransact(code, data, reply, flags); | 
 |     } | 
 | } | 
 |  | 
 | // ---------------------------------------------------------------------------- | 
 |  | 
 | } // namespace hardware | 
 | } // namespace android |