You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
231 lines
9.0 KiB
231 lines
9.0 KiB
/*
|
|
**
|
|
** 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 "ICameraClient"
|
|
#include <utils/Log.h>
|
|
#include <stdint.h>
|
|
#include <sys/types.h>
|
|
#include <camera/CameraUtils.h>
|
|
#include <android/hardware/ICameraClient.h>
|
|
#include <media/hardware/HardwareAPI.h>
|
|
|
|
namespace android {
|
|
namespace hardware {
|
|
|
|
enum {
|
|
NOTIFY_CALLBACK = IBinder::FIRST_CALL_TRANSACTION,
|
|
DATA_CALLBACK,
|
|
DATA_CALLBACK_TIMESTAMP,
|
|
RECORDING_FRAME_HANDLE_CALLBACK_TIMESTAMP,
|
|
RECORDING_FRAME_HANDLE_CALLBACK_TIMESTAMP_BATCH,
|
|
};
|
|
|
|
class BpCameraClient: public BpInterface<ICameraClient>
|
|
{
|
|
public:
|
|
explicit BpCameraClient(const sp<IBinder>& impl)
|
|
: BpInterface<ICameraClient>(impl)
|
|
{
|
|
}
|
|
|
|
// generic callback from camera service to app
|
|
void notifyCallback(int32_t msgType, int32_t ext1, int32_t ext2)
|
|
{
|
|
ALOGV("notifyCallback");
|
|
Parcel data, reply;
|
|
data.writeInterfaceToken(ICameraClient::getInterfaceDescriptor());
|
|
data.writeInt32(msgType);
|
|
data.writeInt32(ext1);
|
|
data.writeInt32(ext2);
|
|
remote()->transact(NOTIFY_CALLBACK, data, &reply, IBinder::FLAG_ONEWAY);
|
|
}
|
|
|
|
// generic data callback from camera service to app with image data
|
|
void dataCallback(int32_t msgType, const sp<IMemory>& imageData,
|
|
camera_frame_metadata_t *metadata)
|
|
{
|
|
ALOGV("dataCallback");
|
|
Parcel data, reply;
|
|
data.writeInterfaceToken(ICameraClient::getInterfaceDescriptor());
|
|
data.writeInt32(msgType);
|
|
data.writeStrongBinder(IInterface::asBinder(imageData));
|
|
if (metadata) {
|
|
data.writeInt32(metadata->number_of_faces);
|
|
data.write(metadata->faces, sizeof(camera_face_t) * metadata->number_of_faces);
|
|
}
|
|
remote()->transact(DATA_CALLBACK, data, &reply, IBinder::FLAG_ONEWAY);
|
|
}
|
|
|
|
// generic data callback from camera service to app with image data
|
|
void dataCallbackTimestamp(nsecs_t timestamp, int32_t msgType, const sp<IMemory>& imageData)
|
|
{
|
|
ALOGV("dataCallback");
|
|
Parcel data, reply;
|
|
data.writeInterfaceToken(ICameraClient::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(ICameraClient::getInterfaceDescriptor());
|
|
data.writeInt64(timestamp);
|
|
data.writeNativeHandle(handle);
|
|
remote()->transact(RECORDING_FRAME_HANDLE_CALLBACK_TIMESTAMP, data, &reply,
|
|
IBinder::FLAG_ONEWAY);
|
|
}
|
|
|
|
void recordingFrameHandleCallbackTimestampBatch(
|
|
const std::vector<nsecs_t>& timestamps,
|
|
const std::vector<native_handle_t*>& handles) {
|
|
ALOGV("recordingFrameHandleCallbackTimestampBatch");
|
|
Parcel data, reply;
|
|
data.writeInterfaceToken(ICameraClient::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);
|
|
}
|
|
};
|
|
|
|
IMPLEMENT_META_INTERFACE(CameraClient, "android.hardware.ICameraClient");
|
|
|
|
// ----------------------------------------------------------------------
|
|
|
|
status_t BnCameraClient::onTransact(
|
|
uint32_t code, const Parcel& data, Parcel* reply, uint32_t flags)
|
|
{
|
|
switch(code) {
|
|
case NOTIFY_CALLBACK: {
|
|
ALOGV("NOTIFY_CALLBACK");
|
|
CHECK_INTERFACE(ICameraClient, data, reply);
|
|
int32_t msgType = data.readInt32();
|
|
int32_t ext1 = data.readInt32();
|
|
int32_t ext2 = data.readInt32();
|
|
notifyCallback(msgType, ext1, ext2);
|
|
return NO_ERROR;
|
|
} break;
|
|
case DATA_CALLBACK: {
|
|
ALOGV("DATA_CALLBACK");
|
|
CHECK_INTERFACE(ICameraClient, data, reply);
|
|
int32_t msgType = data.readInt32();
|
|
sp<IMemory> imageData = interface_cast<IMemory>(data.readStrongBinder());
|
|
camera_frame_metadata_t metadata;
|
|
if (data.dataAvail() > 0) {
|
|
metadata.number_of_faces = data.readInt32();
|
|
// Zero faces is a valid case, to notify clients that no faces are now visible
|
|
if (metadata.number_of_faces < 0 ||
|
|
metadata.number_of_faces > (int32_t)(INT32_MAX / sizeof(camera_face_t))) {
|
|
ALOGE("%s: Too large face count: %d", __FUNCTION__, metadata.number_of_faces);
|
|
return BAD_VALUE;
|
|
}
|
|
metadata.faces = (camera_face_t *) data.readInplace(
|
|
sizeof(camera_face_t) * metadata.number_of_faces);
|
|
}
|
|
dataCallback(msgType, imageData, &metadata);
|
|
return NO_ERROR;
|
|
} break;
|
|
case DATA_CALLBACK_TIMESTAMP: {
|
|
ALOGV("DATA_CALLBACK_TIMESTAMP");
|
|
CHECK_INTERFACE(ICameraClient, 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(ICameraClient, 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 BpCamera::releaseRecordingFrameHandle.
|
|
recordingFrameHandleCallbackTimestamp(timestamp, handle);
|
|
return NO_ERROR;
|
|
} break;
|
|
case RECORDING_FRAME_HANDLE_CALLBACK_TIMESTAMP_BATCH: {
|
|
ALOGV("RECORDING_FRAME_HANDLE_CALLBACK_TIMESTAMP_BATCH");
|
|
CHECK_INTERFACE(ICameraClient, 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 BpCamera::releaseRecordingFrameHandleBatch.
|
|
recordingFrameHandleCallbackTimestampBatch(timestamps, handles);
|
|
return NO_ERROR;
|
|
} break;
|
|
default:
|
|
return BBinder::onTransact(code, data, reply, flags);
|
|
}
|
|
}
|
|
|
|
// ----------------------------------------------------------------------------
|
|
|
|
} // namespace hardware
|
|
} // namespace android
|