d07a59e651
Change-Id: I1b52f72beb953bd957952b5b7a660c269c2ef4b3
626 lines
19 KiB
C++
626 lines
19 KiB
C++
/*
|
|
**
|
|
** Copyright 2013, 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 "ICameraDeviceUser"
|
|
#include <utils/Log.h>
|
|
#include <stdint.h>
|
|
#include <sys/types.h>
|
|
#include <binder/Parcel.h>
|
|
#include <camera/camera2/ICameraDeviceUser.h>
|
|
#include <gui/IGraphicBufferProducer.h>
|
|
#include <gui/Surface.h>
|
|
#include <camera/CameraMetadata.h>
|
|
#include <camera/camera2/CaptureRequest.h>
|
|
#include <camera/camera2/OutputConfiguration.h>
|
|
|
|
namespace android {
|
|
|
|
typedef Parcel::WritableBlob WritableBlob;
|
|
typedef Parcel::ReadableBlob ReadableBlob;
|
|
|
|
enum {
|
|
DISCONNECT = IBinder::FIRST_CALL_TRANSACTION,
|
|
SUBMIT_REQUEST,
|
|
SUBMIT_REQUEST_LIST,
|
|
CANCEL_REQUEST,
|
|
BEGIN_CONFIGURE,
|
|
END_CONFIGURE,
|
|
DELETE_STREAM,
|
|
CREATE_STREAM,
|
|
CREATE_INPUT_STREAM,
|
|
GET_INPUT_SURFACE,
|
|
CREATE_DEFAULT_REQUEST,
|
|
GET_CAMERA_INFO,
|
|
WAIT_UNTIL_IDLE,
|
|
FLUSH,
|
|
PREPARE,
|
|
TEAR_DOWN,
|
|
PREPARE2
|
|
};
|
|
|
|
namespace {
|
|
// Read empty strings without printing a false error message.
|
|
String16 readMaybeEmptyString16(const Parcel& parcel) {
|
|
size_t len;
|
|
const char16_t* str = parcel.readString16Inplace(&len);
|
|
if (str != NULL) {
|
|
return String16(str, len);
|
|
} else {
|
|
return String16();
|
|
}
|
|
}
|
|
};
|
|
|
|
class BpCameraDeviceUser : public BpInterface<ICameraDeviceUser>
|
|
{
|
|
public:
|
|
BpCameraDeviceUser(const sp<IBinder>& impl)
|
|
: BpInterface<ICameraDeviceUser>(impl)
|
|
{
|
|
}
|
|
|
|
// disconnect from camera service
|
|
void disconnect()
|
|
{
|
|
ALOGV("disconnect");
|
|
Parcel data, reply;
|
|
data.writeInterfaceToken(ICameraDeviceUser::getInterfaceDescriptor());
|
|
remote()->transact(DISCONNECT, data, &reply);
|
|
reply.readExceptionCode();
|
|
}
|
|
|
|
virtual int submitRequest(sp<CaptureRequest> request, bool repeating,
|
|
int64_t *lastFrameNumber)
|
|
{
|
|
Parcel data, reply;
|
|
data.writeInterfaceToken(ICameraDeviceUser::getInterfaceDescriptor());
|
|
|
|
// arg0 = CaptureRequest
|
|
if (request != 0) {
|
|
data.writeInt32(1);
|
|
request->writeToParcel(&data);
|
|
} else {
|
|
data.writeInt32(0);
|
|
}
|
|
|
|
// arg1 = streaming (bool)
|
|
data.writeInt32(repeating);
|
|
|
|
remote()->transact(SUBMIT_REQUEST, data, &reply);
|
|
|
|
reply.readExceptionCode();
|
|
status_t res = reply.readInt32();
|
|
|
|
status_t resFrameNumber = BAD_VALUE;
|
|
if (reply.readInt32() != 0) {
|
|
if (lastFrameNumber != NULL) {
|
|
resFrameNumber = reply.readInt64(lastFrameNumber);
|
|
}
|
|
}
|
|
|
|
if (res < 0 || (resFrameNumber != NO_ERROR)) {
|
|
res = FAILED_TRANSACTION;
|
|
}
|
|
return res;
|
|
}
|
|
|
|
virtual int submitRequestList(List<sp<CaptureRequest> > requestList, bool repeating,
|
|
int64_t *lastFrameNumber)
|
|
{
|
|
Parcel data, reply;
|
|
data.writeInterfaceToken(ICameraDeviceUser::getInterfaceDescriptor());
|
|
|
|
data.writeInt32(requestList.size());
|
|
|
|
for (List<sp<CaptureRequest> >::iterator it = requestList.begin();
|
|
it != requestList.end(); ++it) {
|
|
sp<CaptureRequest> request = *it;
|
|
if (request != 0) {
|
|
data.writeInt32(1);
|
|
if (request->writeToParcel(&data) != OK) {
|
|
return BAD_VALUE;
|
|
}
|
|
} else {
|
|
data.writeInt32(0);
|
|
}
|
|
}
|
|
|
|
data.writeInt32(repeating);
|
|
|
|
remote()->transact(SUBMIT_REQUEST_LIST, data, &reply);
|
|
|
|
reply.readExceptionCode();
|
|
status_t res = reply.readInt32();
|
|
|
|
status_t resFrameNumber = BAD_VALUE;
|
|
if (reply.readInt32() != 0) {
|
|
if (lastFrameNumber != NULL) {
|
|
resFrameNumber = reply.readInt64(lastFrameNumber);
|
|
}
|
|
}
|
|
if (res < 0 || (resFrameNumber != NO_ERROR)) {
|
|
res = FAILED_TRANSACTION;
|
|
}
|
|
return res;
|
|
}
|
|
|
|
virtual status_t cancelRequest(int requestId, int64_t *lastFrameNumber)
|
|
{
|
|
Parcel data, reply;
|
|
data.writeInterfaceToken(ICameraDeviceUser::getInterfaceDescriptor());
|
|
data.writeInt32(requestId);
|
|
|
|
remote()->transact(CANCEL_REQUEST, data, &reply);
|
|
|
|
reply.readExceptionCode();
|
|
status_t res = reply.readInt32();
|
|
|
|
status_t resFrameNumber = BAD_VALUE;
|
|
if (reply.readInt32() != 0) {
|
|
if (lastFrameNumber != NULL) {
|
|
resFrameNumber = reply.readInt64(lastFrameNumber);
|
|
}
|
|
}
|
|
if ((res != NO_ERROR) || (resFrameNumber != NO_ERROR)) {
|
|
res = FAILED_TRANSACTION;
|
|
}
|
|
return res;
|
|
}
|
|
|
|
virtual status_t beginConfigure()
|
|
{
|
|
ALOGV("beginConfigure");
|
|
Parcel data, reply;
|
|
data.writeInterfaceToken(ICameraDeviceUser::getInterfaceDescriptor());
|
|
remote()->transact(BEGIN_CONFIGURE, data, &reply);
|
|
reply.readExceptionCode();
|
|
return reply.readInt32();
|
|
}
|
|
|
|
virtual status_t endConfigure(bool isConstrainedHighSpeed)
|
|
{
|
|
ALOGV("endConfigure");
|
|
Parcel data, reply;
|
|
data.writeInterfaceToken(ICameraDeviceUser::getInterfaceDescriptor());
|
|
data.writeInt32(isConstrainedHighSpeed);
|
|
|
|
remote()->transact(END_CONFIGURE, data, &reply);
|
|
reply.readExceptionCode();
|
|
return reply.readInt32();
|
|
}
|
|
|
|
virtual status_t deleteStream(int streamId)
|
|
{
|
|
Parcel data, reply;
|
|
data.writeInterfaceToken(ICameraDeviceUser::getInterfaceDescriptor());
|
|
data.writeInt32(streamId);
|
|
|
|
remote()->transact(DELETE_STREAM, data, &reply);
|
|
|
|
reply.readExceptionCode();
|
|
return reply.readInt32();
|
|
}
|
|
|
|
virtual status_t createStream(const OutputConfiguration& outputConfiguration)
|
|
{
|
|
Parcel data, reply;
|
|
data.writeInterfaceToken(ICameraDeviceUser::getInterfaceDescriptor());
|
|
if (outputConfiguration.getGraphicBufferProducer() != NULL) {
|
|
data.writeInt32(1); // marker that OutputConfiguration is not null. Mimic aidl behavior
|
|
outputConfiguration.writeToParcel(data);
|
|
} else {
|
|
data.writeInt32(0);
|
|
}
|
|
remote()->transact(CREATE_STREAM, data, &reply);
|
|
|
|
reply.readExceptionCode();
|
|
return reply.readInt32();
|
|
}
|
|
|
|
virtual status_t createInputStream(int width, int height, int format)
|
|
{
|
|
Parcel data, reply;
|
|
data.writeInterfaceToken(ICameraDeviceUser::getInterfaceDescriptor());
|
|
data.writeInt32(width);
|
|
data.writeInt32(height);
|
|
data.writeInt32(format);
|
|
|
|
remote()->transact(CREATE_INPUT_STREAM, data, &reply);
|
|
|
|
reply.readExceptionCode();
|
|
return reply.readInt32();
|
|
}
|
|
|
|
// get the buffer producer of the input stream
|
|
virtual status_t getInputBufferProducer(
|
|
sp<IGraphicBufferProducer> *producer) {
|
|
if (producer == NULL) {
|
|
return BAD_VALUE;
|
|
}
|
|
|
|
Parcel data, reply;
|
|
data.writeInterfaceToken(ICameraDeviceUser::getInterfaceDescriptor());
|
|
|
|
remote()->transact(GET_INPUT_SURFACE, data, &reply);
|
|
|
|
reply.readExceptionCode();
|
|
status_t result = reply.readInt32() ;
|
|
if (result != OK) {
|
|
return result;
|
|
}
|
|
|
|
sp<IGraphicBufferProducer> bp = NULL;
|
|
if (reply.readInt32() != 0) {
|
|
String16 name = readMaybeEmptyString16(reply);
|
|
bp = interface_cast<IGraphicBufferProducer>(
|
|
reply.readStrongBinder());
|
|
}
|
|
|
|
*producer = bp;
|
|
|
|
return *producer == NULL ? INVALID_OPERATION : OK;
|
|
}
|
|
|
|
// Create a request object from a template.
|
|
virtual status_t createDefaultRequest(int templateId,
|
|
/*out*/
|
|
CameraMetadata* request)
|
|
{
|
|
Parcel data, reply;
|
|
data.writeInterfaceToken(ICameraDeviceUser::getInterfaceDescriptor());
|
|
data.writeInt32(templateId);
|
|
remote()->transact(CREATE_DEFAULT_REQUEST, data, &reply);
|
|
|
|
reply.readExceptionCode();
|
|
status_t result = reply.readInt32();
|
|
|
|
CameraMetadata out;
|
|
if (reply.readInt32() != 0) {
|
|
out.readFromParcel(&reply);
|
|
}
|
|
|
|
if (request != NULL) {
|
|
request->swap(out);
|
|
}
|
|
return result;
|
|
}
|
|
|
|
|
|
virtual status_t getCameraInfo(CameraMetadata* info)
|
|
{
|
|
Parcel data, reply;
|
|
data.writeInterfaceToken(ICameraDeviceUser::getInterfaceDescriptor());
|
|
remote()->transact(GET_CAMERA_INFO, data, &reply);
|
|
|
|
reply.readExceptionCode();
|
|
status_t result = reply.readInt32();
|
|
|
|
CameraMetadata out;
|
|
if (reply.readInt32() != 0) {
|
|
out.readFromParcel(&reply);
|
|
}
|
|
|
|
if (info != NULL) {
|
|
info->swap(out);
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
virtual status_t waitUntilIdle()
|
|
{
|
|
ALOGV("waitUntilIdle");
|
|
Parcel data, reply;
|
|
data.writeInterfaceToken(ICameraDeviceUser::getInterfaceDescriptor());
|
|
remote()->transact(WAIT_UNTIL_IDLE, data, &reply);
|
|
reply.readExceptionCode();
|
|
return reply.readInt32();
|
|
}
|
|
|
|
virtual status_t flush(int64_t *lastFrameNumber)
|
|
{
|
|
ALOGV("flush");
|
|
Parcel data, reply;
|
|
data.writeInterfaceToken(ICameraDeviceUser::getInterfaceDescriptor());
|
|
remote()->transact(FLUSH, data, &reply);
|
|
reply.readExceptionCode();
|
|
status_t res = reply.readInt32();
|
|
|
|
status_t resFrameNumber = BAD_VALUE;
|
|
if (reply.readInt32() != 0) {
|
|
if (lastFrameNumber != NULL) {
|
|
resFrameNumber = reply.readInt64(lastFrameNumber);
|
|
}
|
|
}
|
|
if ((res != NO_ERROR) || (resFrameNumber != NO_ERROR)) {
|
|
res = FAILED_TRANSACTION;
|
|
}
|
|
return res;
|
|
}
|
|
|
|
virtual status_t prepare(int streamId)
|
|
{
|
|
ALOGV("prepare");
|
|
Parcel data, reply;
|
|
|
|
data.writeInterfaceToken(ICameraDeviceUser::getInterfaceDescriptor());
|
|
data.writeInt32(streamId);
|
|
|
|
remote()->transact(PREPARE, data, &reply);
|
|
|
|
reply.readExceptionCode();
|
|
return reply.readInt32();
|
|
}
|
|
|
|
virtual status_t prepare2(int maxCount, int streamId)
|
|
{
|
|
ALOGV("prepare2");
|
|
Parcel data, reply;
|
|
|
|
data.writeInterfaceToken(ICameraDeviceUser::getInterfaceDescriptor());
|
|
data.writeInt32(maxCount);
|
|
data.writeInt32(streamId);
|
|
|
|
remote()->transact(PREPARE2, data, &reply);
|
|
|
|
reply.readExceptionCode();
|
|
return reply.readInt32();
|
|
}
|
|
|
|
virtual status_t tearDown(int streamId)
|
|
{
|
|
ALOGV("tearDown");
|
|
Parcel data, reply;
|
|
|
|
data.writeInterfaceToken(ICameraDeviceUser::getInterfaceDescriptor());
|
|
data.writeInt32(streamId);
|
|
|
|
remote()->transact(TEAR_DOWN, data, &reply);
|
|
|
|
reply.readExceptionCode();
|
|
return reply.readInt32();
|
|
}
|
|
|
|
private:
|
|
|
|
|
|
};
|
|
|
|
IMPLEMENT_META_INTERFACE(CameraDeviceUser,
|
|
"android.hardware.camera2.ICameraDeviceUser");
|
|
|
|
// ----------------------------------------------------------------------
|
|
|
|
status_t BnCameraDeviceUser::onTransact(
|
|
uint32_t code, const Parcel& data, Parcel* reply, uint32_t flags)
|
|
{
|
|
switch(code) {
|
|
case DISCONNECT: {
|
|
ALOGV("DISCONNECT");
|
|
CHECK_INTERFACE(ICameraDeviceUser, data, reply);
|
|
disconnect();
|
|
reply->writeNoException();
|
|
return NO_ERROR;
|
|
} break;
|
|
case SUBMIT_REQUEST: {
|
|
CHECK_INTERFACE(ICameraDeviceUser, data, reply);
|
|
|
|
// arg0 = request
|
|
sp<CaptureRequest> request;
|
|
if (data.readInt32() != 0) {
|
|
request = new CaptureRequest();
|
|
request->readFromParcel(const_cast<Parcel*>(&data));
|
|
}
|
|
|
|
// arg1 = streaming (bool)
|
|
bool repeating = data.readInt32();
|
|
|
|
// return code: requestId (int32)
|
|
reply->writeNoException();
|
|
int64_t lastFrameNumber = -1;
|
|
reply->writeInt32(submitRequest(request, repeating, &lastFrameNumber));
|
|
reply->writeInt32(1);
|
|
reply->writeInt64(lastFrameNumber);
|
|
|
|
return NO_ERROR;
|
|
} break;
|
|
case SUBMIT_REQUEST_LIST: {
|
|
CHECK_INTERFACE(ICameraDeviceUser, data, reply);
|
|
|
|
List<sp<CaptureRequest> > requestList;
|
|
int requestListSize = data.readInt32();
|
|
for (int i = 0; i < requestListSize; i++) {
|
|
if (data.readInt32() != 0) {
|
|
sp<CaptureRequest> request = new CaptureRequest();
|
|
if (request->readFromParcel(const_cast<Parcel*>(&data)) != OK) {
|
|
return BAD_VALUE;
|
|
}
|
|
requestList.push_back(request);
|
|
} else {
|
|
sp<CaptureRequest> request = 0;
|
|
requestList.push_back(request);
|
|
ALOGE("A request is missing. Sending in null request.");
|
|
}
|
|
}
|
|
|
|
bool repeating = data.readInt32();
|
|
|
|
reply->writeNoException();
|
|
int64_t lastFrameNumber = -1;
|
|
reply->writeInt32(submitRequestList(requestList, repeating, &lastFrameNumber));
|
|
reply->writeInt32(1);
|
|
reply->writeInt64(lastFrameNumber);
|
|
|
|
return NO_ERROR;
|
|
} break;
|
|
case CANCEL_REQUEST: {
|
|
CHECK_INTERFACE(ICameraDeviceUser, data, reply);
|
|
int requestId = data.readInt32();
|
|
reply->writeNoException();
|
|
int64_t lastFrameNumber = -1;
|
|
reply->writeInt32(cancelRequest(requestId, &lastFrameNumber));
|
|
reply->writeInt32(1);
|
|
reply->writeInt64(lastFrameNumber);
|
|
return NO_ERROR;
|
|
} break;
|
|
case DELETE_STREAM: {
|
|
CHECK_INTERFACE(ICameraDeviceUser, data, reply);
|
|
int streamId = data.readInt32();
|
|
reply->writeNoException();
|
|
reply->writeInt32(deleteStream(streamId));
|
|
return NO_ERROR;
|
|
} break;
|
|
case CREATE_STREAM: {
|
|
CHECK_INTERFACE(ICameraDeviceUser, data, reply);
|
|
|
|
status_t ret = BAD_VALUE;
|
|
if (data.readInt32() != 0) {
|
|
OutputConfiguration outputConfiguration(data);
|
|
ret = createStream(outputConfiguration);
|
|
} else {
|
|
ALOGE("%s: cannot take an empty OutputConfiguration", __FUNCTION__);
|
|
}
|
|
|
|
reply->writeNoException();
|
|
ALOGV("%s: CREATE_STREAM: write noException", __FUNCTION__);
|
|
reply->writeInt32(ret);
|
|
ALOGV("%s: CREATE_STREAM: write ret = %d", __FUNCTION__, ret);
|
|
|
|
return NO_ERROR;
|
|
} break;
|
|
case CREATE_INPUT_STREAM: {
|
|
CHECK_INTERFACE(ICameraDeviceUser, data, reply);
|
|
int width, height, format;
|
|
|
|
width = data.readInt32();
|
|
height = data.readInt32();
|
|
format = data.readInt32();
|
|
status_t ret = createInputStream(width, height, format);
|
|
|
|
reply->writeNoException();
|
|
reply->writeInt32(ret);
|
|
return NO_ERROR;
|
|
|
|
} break;
|
|
case GET_INPUT_SURFACE: {
|
|
CHECK_INTERFACE(ICameraDeviceUser, data, reply);
|
|
|
|
sp<IGraphicBufferProducer> bp;
|
|
status_t ret = getInputBufferProducer(&bp);
|
|
sp<IBinder> b(IInterface::asBinder(ret == OK ? bp : NULL));
|
|
|
|
reply->writeNoException();
|
|
reply->writeInt32(ret);
|
|
reply->writeInt32(1);
|
|
reply->writeString16(String16("camera input")); // name of surface
|
|
reply->writeStrongBinder(b);
|
|
|
|
return NO_ERROR;
|
|
} break;
|
|
case CREATE_DEFAULT_REQUEST: {
|
|
CHECK_INTERFACE(ICameraDeviceUser, data, reply);
|
|
|
|
int templateId = data.readInt32();
|
|
|
|
CameraMetadata request;
|
|
status_t ret;
|
|
ret = createDefaultRequest(templateId, &request);
|
|
|
|
reply->writeNoException();
|
|
reply->writeInt32(ret);
|
|
|
|
// out-variables are after exception and return value
|
|
reply->writeInt32(1); // to mark presence of metadata object
|
|
request.writeToParcel(const_cast<Parcel*>(reply));
|
|
|
|
return NO_ERROR;
|
|
} break;
|
|
case GET_CAMERA_INFO: {
|
|
CHECK_INTERFACE(ICameraDeviceUser, data, reply);
|
|
|
|
CameraMetadata info;
|
|
status_t ret;
|
|
ret = getCameraInfo(&info);
|
|
|
|
reply->writeNoException();
|
|
reply->writeInt32(ret);
|
|
|
|
// out-variables are after exception and return value
|
|
reply->writeInt32(1); // to mark presence of metadata object
|
|
info.writeToParcel(reply);
|
|
|
|
return NO_ERROR;
|
|
} break;
|
|
case WAIT_UNTIL_IDLE: {
|
|
CHECK_INTERFACE(ICameraDeviceUser, data, reply);
|
|
reply->writeNoException();
|
|
reply->writeInt32(waitUntilIdle());
|
|
return NO_ERROR;
|
|
} break;
|
|
case FLUSH: {
|
|
CHECK_INTERFACE(ICameraDeviceUser, data, reply);
|
|
reply->writeNoException();
|
|
int64_t lastFrameNumber = -1;
|
|
reply->writeInt32(flush(&lastFrameNumber));
|
|
reply->writeInt32(1);
|
|
reply->writeInt64(lastFrameNumber);
|
|
return NO_ERROR;
|
|
}
|
|
case BEGIN_CONFIGURE: {
|
|
CHECK_INTERFACE(ICameraDeviceUser, data, reply);
|
|
reply->writeNoException();
|
|
reply->writeInt32(beginConfigure());
|
|
return NO_ERROR;
|
|
} break;
|
|
case END_CONFIGURE: {
|
|
CHECK_INTERFACE(ICameraDeviceUser, data, reply);
|
|
bool isConstrainedHighSpeed = data.readInt32();
|
|
reply->writeNoException();
|
|
reply->writeInt32(endConfigure(isConstrainedHighSpeed));
|
|
return NO_ERROR;
|
|
} break;
|
|
case PREPARE: {
|
|
CHECK_INTERFACE(ICameraDeviceUser, data, reply);
|
|
int streamId = data.readInt32();
|
|
reply->writeNoException();
|
|
reply->writeInt32(prepare(streamId));
|
|
return NO_ERROR;
|
|
} break;
|
|
case TEAR_DOWN: {
|
|
CHECK_INTERFACE(ICameraDeviceUser, data, reply);
|
|
int streamId = data.readInt32();
|
|
reply->writeNoException();
|
|
reply->writeInt32(tearDown(streamId));
|
|
return NO_ERROR;
|
|
} break;
|
|
case PREPARE2: {
|
|
CHECK_INTERFACE(ICameraDeviceUser, data, reply);
|
|
int maxCount = data.readInt32();
|
|
int streamId = data.readInt32();
|
|
reply->writeNoException();
|
|
reply->writeInt32(prepare2(maxCount, streamId));
|
|
return NO_ERROR;
|
|
} break;
|
|
default:
|
|
return BBinder::onTransact(code, data, reply, flags);
|
|
}
|
|
}
|
|
|
|
// ----------------------------------------------------------------------------
|
|
|
|
}; // namespace android
|