#include "camera_android.h"
#include "core/os/os.h"
#include "platform/android/display_server_android.h"
#ifndef IF_EQUAL_RETURN
#define MAKE_FORMAT_CONST(suffix) AIMAGE_FORMAT_##suffix
#define IF_EQUAL_RETURN(param, val) \
if (MAKE_FORMAT_CONST(val) == param) \
return #val
#endif
String GetFormatName(const int32_t &format) {
IF_EQUAL_RETURN(format, YUV_420_888);
IF_EQUAL_RETURN(format, RGB_888);
IF_EQUAL_RETURN(format, RGBA_8888);
return "Unsupported";
}
CameraFeedAndroid::CameraFeedAndroid(ACameraManager *manager, ACameraMetadata *metadata, const char *id,
CameraFeed::FeedPosition position, int32_t orientation) :
CameraFeed() {
this->manager = manager;
this->metadata = metadata;
this->orientation = orientation;
_add_formats();
camera_id = id;
set_position(position);
switch (position) {
case CameraFeed::FEED_BACK:
name = vformat("%s | BACK", id);
break;
case CameraFeed::FEED_FRONT:
name = vformat("%s | FRONT", id);
break;
default:
name = vformat("%s", id);
break;
}
image_y.instantiate();
image_uv.instantiate();
}
CameraFeedAndroid::~CameraFeedAndroid() {
if (is_active()) {
deactivate_feed();
}
if (metadata != nullptr) {
ACameraMetadata_free(metadata);
}
}
void CameraFeedAndroid::_set_rotation() {
int display_rotation = DisplayServerAndroid::get_singleton()->get_display_rotation();
switch (display_rotation) {
case 90:
display_rotation = 270;
break;
case 270:
display_rotation = 90;
break;
default:
break;
}
int sign = position == CameraFeed::FEED_FRONT ? 1 : -1;
float imageRotation = (orientation - display_rotation * sign + 360) % 360;
transform.set_rotation(real_t(Math::deg_to_rad(imageRotation)));
}
void CameraFeedAndroid::_add_formats() {
ACameraMetadata_const_entry formats;
camera_status_t status = ACameraMetadata_getConstEntry(metadata, ACAMERA_SCALER_AVAILABLE_STREAM_CONFIGURATIONS, &formats);
if (status == ACAMERA_OK) {
for (uint32_t f = 0; f < formats.count; f += 4) {
int32_t input = formats.data.i32[f + 3];
if (input) {
continue;
}
int32_t format = formats.data.i32[f + 0];
if (format == AIMAGE_FORMAT_YUV_420_888 ||
format == AIMAGE_FORMAT_RGBA_8888 ||
format == AIMAGE_FORMAT_RGB_888) {
CameraFeed::FeedFormat feed_format;
feed_format.width = formats.data.i32[f + 1];
feed_format.height = formats.data.i32[f + 2];
feed_format.format = GetFormatName(format);
feed_format.pixel_format = format;
this->formats.append(feed_format);
}
}
}
}
bool CameraFeedAndroid::activate_feed() {
ERR_FAIL_COND_V_MSG(selected_format == -1, false, "CameraFeed format needs to be set before activating.");
if (is_active()) {
deactivate_feed();
};
if (!OS::get_singleton()->request_permission("CAMERA")) {
return false;
}
static ACameraDevice_stateCallbacks deviceCallbacks = {
.context = this,
.onDisconnected = onDisconnected,
.onError = onError,
};
camera_status_t c_status = ACameraManager_openCamera(manager, camera_id.utf8().get_data(), &deviceCallbacks, &device);
if (c_status != ACAMERA_OK) {
onError(this, device, c_status);
return false;
}
const FeedFormat &feed_format = formats[selected_format];
media_status_t m_status = AImageReader_new(feed_format.width, feed_format.height, feed_format.pixel_format, 1, &reader);
if (m_status != AMEDIA_OK) {
onError(this, device, m_status);
return false;
}
static AImageReader_ImageListener listener{
.context = this,
.onImageAvailable = onImage,
};
m_status = AImageReader_setImageListener(reader, &listener);
if (m_status != AMEDIA_OK) {
onError(this, device, m_status);
return false;
}
ANativeWindow *surface;
m_status = AImageReader_getWindow(reader, &surface);
if (m_status != AMEDIA_OK) {
onError(this, device, m_status);
return false;
}
ACaptureSessionOutput *output = nullptr;
c_status = ACaptureSessionOutput_create(surface, &output);
if (c_status != ACAMERA_OK) {
onError(this, device, c_status);
return false;
}
ACaptureSessionOutputContainer *outputs = nullptr;
c_status = ACaptureSessionOutputContainer_create(&outputs);
if (c_status != ACAMERA_OK) {
onError(this, device, c_status);
return false;
}
c_status = ACaptureSessionOutputContainer_add(outputs, output);
if (c_status != ACAMERA_OK) {
onError(this, device, c_status);
return false;
}
static ACameraCaptureSession_stateCallbacks sessionStateCallbacks{
.context = this,
.onClosed = onSessionClosed,
.onReady = onSessionReady,
.onActive = onSessionActive
};
c_status = ACameraDevice_createCaptureSession(device, outputs, &sessionStateCallbacks, &session);
if (c_status != ACAMERA_OK) {
onError(this, device, c_status);
return false;
}
c_status = ACameraDevice_createCaptureRequest(device, TEMPLATE_PREVIEW, &request);
if (c_status != ACAMERA_OK) {
onError(this, device, c_status);
return false;
}
ACameraOutputTarget *imageTarget = nullptr;
c_status = ACameraOutputTarget_create(surface, &imageTarget);
if (c_status != ACAMERA_OK) {
onError(this, device, c_status);
return false;
}
c_status = ACaptureRequest_addTarget(request, imageTarget);
if (c_status != ACAMERA_OK) {
onError(this, device, c_status);
return false;
}
c_status = ACameraCaptureSession_setRepeatingRequest(session, nullptr, 1, &request, nullptr);
if (c_status != ACAMERA_OK) {
onError(this, device, c_status);
return false;
}
return true;
}
bool CameraFeedAndroid::set_format(int p_index, const Dictionary &p_parameters) {
ERR_FAIL_COND_V_MSG(active, false, "Feed is active.");
ERR_FAIL_INDEX_V_MSG(p_index, formats.size(), false, "Invalid format index.");
selected_format = p_index;
return true;
}
Array CameraFeedAndroid::get_formats() const {
Array result;
for (const FeedFormat &feed_format : formats) {
Dictionary dictionary;
dictionary["width"] = feed_format.width;
dictionary["height"] = feed_format.height;
dictionary["format"] = feed_format.format;
result.push_back(dictionary);
}
return result;
}
CameraFeed::FeedFormat CameraFeedAndroid::get_format() const {
CameraFeed::FeedFormat feed_format = {};
return selected_format == -1 ? feed_format : formats[selected_format];
}
void CameraFeedAndroid::onImage(void *context, AImageReader *p_reader) {
CameraFeedAndroid *feed = static_cast<CameraFeedAndroid *>(context);
Vector<uint8_t> data_y = feed->data_y;
Vector<uint8_t> data_uv = feed->data_uv;
Ref<Image> image_y = feed->image_y;
Ref<Image> image_uv = feed->image_uv;
AImage *image = nullptr;
media_status_t status = AImageReader_acquireNextImage(p_reader, &image);
ERR_FAIL_COND(status != AMEDIA_OK);
uint8_t *data = nullptr;
int len = 0;
int32_t pixel_stride, row_stride;
FeedFormat format = feed->get_format();
int width = format.width;
int height = format.height;
switch (format.pixel_format) {
case AIMAGE_FORMAT_YUV_420_888:
AImage_getPlaneData(image, 0, &data, &len);
if (len <= 0) {
return;
}
if (len != data_y.size()) {
int64_t size = Image::get_image_data_size(width, height, Image::FORMAT_R8, false);
data_y.resize(len > size ? len : size);
}
memcpy(data_y.ptrw(), data, len);
AImage_getPlanePixelStride(image, 1, &pixel_stride);
AImage_getPlaneRowStride(image, 1, &row_stride);
AImage_getPlaneData(image, 1, &data, &len);
if (len <= 0) {
return;
}
if (len != data_uv.size()) {
int64_t size = Image::get_image_data_size(width / 2, height / 2, Image::FORMAT_RG8, false);
data_uv.resize(len > size ? len : size);
}
memcpy(data_uv.ptrw(), data, len);
image_y->initialize_data(width, height, false, Image::FORMAT_R8, data_y);
image_uv->initialize_data(width / 2, height / 2, false, Image::FORMAT_RG8, data_uv);
feed->set_ycbcr_images(image_y, image_uv);
break;
case AIMAGE_FORMAT_RGBA_8888:
AImage_getPlaneData(image, 0, &data, &len);
if (len <= 0) {
return;
}
if (len != data_y.size()) {
int64_t size = Image::get_image_data_size(width, height, Image::FORMAT_RGBA8, false);
data_y.resize(len > size ? len : size);
}
memcpy(data_y.ptrw(), data, len);
image_y->initialize_data(width, height, false, Image::FORMAT_RGBA8, data_y);
feed->set_rgb_image(image_y);
break;
case AIMAGE_FORMAT_RGB_888:
AImage_getPlaneData(image, 0, &data, &len);
if (len <= 0) {
return;
}
if (len != data_y.size()) {
int64_t size = Image::get_image_data_size(width, height, Image::FORMAT_RGB8, false);
data_y.resize(len > size ? len : size);
}
memcpy(data_y.ptrw(), data, len);
image_y->initialize_data(width, height, false, Image::FORMAT_RGB8, data_y);
feed->set_rgb_image(image_y);
break;
default:
return;
}
feed->_set_rotation();
AImage_delete(image);
}
void CameraFeedAndroid::onSessionReady(void *context, ACameraCaptureSession *session) {
print_verbose("Capture session ready");
}
void CameraFeedAndroid::onSessionActive(void *context, ACameraCaptureSession *session) {
print_verbose("Capture session active");
}
void CameraFeedAndroid::onSessionClosed(void *context, ACameraCaptureSession *session) {
print_verbose("Capture session closed");
}
void CameraFeedAndroid::deactivate_feed() {
if (session != nullptr) {
ACameraCaptureSession_stopRepeating(session);
ACameraCaptureSession_close(session);
session = nullptr;
}
if (request != nullptr) {
ACaptureRequest_free(request);
request = nullptr;
}
if (reader != nullptr) {
AImageReader_delete(reader);
reader = nullptr;
}
if (device != nullptr) {
ACameraDevice_close(device);
device = nullptr;
}
}
void CameraFeedAndroid::onError(void *context, ACameraDevice *p_device, int error) {
print_error(vformat("Camera error: %d", error));
onDisconnected(context, p_device);
}
void CameraFeedAndroid::onDisconnected(void *context, ACameraDevice *p_device) {
print_verbose("Camera disconnected");
auto *feed = static_cast<CameraFeedAndroid *>(context);
feed->set_active(false);
}
void CameraAndroid::update_feeds() {
ACameraIdList *cameraIds = nullptr;
camera_status_t c_status = ACameraManager_getCameraIdList(cameraManager, &cameraIds);
ERR_FAIL_COND(c_status != ACAMERA_OK);
for (int i = feeds.size() - 1; i >= 0; i--) {
remove_feed(feeds[i]);
}
for (int c = 0; c < cameraIds->numCameras; ++c) {
const char *id = cameraIds->cameraIds[c];
ACameraMetadata *metadata = nullptr;
ACameraManager_getCameraCharacteristics(cameraManager, id, &metadata);
if (!metadata) {
continue;
}
ACameraMetadata_const_entry orientation;
c_status = ACameraMetadata_getConstEntry(metadata, ACAMERA_SENSOR_ORIENTATION, &orientation);
int32_t cameraOrientation;
if (c_status == ACAMERA_OK) {
cameraOrientation = orientation.data.i32[0];
} else {
cameraOrientation = 0;
print_error(vformat("Unable to get sensor orientation: %s", id));
}
ACameraMetadata_const_entry lensInfo;
CameraFeed::FeedPosition position = CameraFeed::FEED_UNSPECIFIED;
camera_status_t status;
status = ACameraMetadata_getConstEntry(metadata, ACAMERA_LENS_FACING, &lensInfo);
if (status != ACAMERA_OK) {
ACameraMetadata_free(metadata);
continue;
}
uint8_t lens_facing = static_cast<acamera_metadata_enum_android_lens_facing_t>(lensInfo.data.u8[0]);
if (lens_facing == ACAMERA_LENS_FACING_FRONT) {
position = CameraFeed::FEED_FRONT;
} else if (lens_facing == ACAMERA_LENS_FACING_BACK) {
position = CameraFeed::FEED_BACK;
} else {
ACameraMetadata_free(metadata);
continue;
}
Ref<CameraFeedAndroid> feed = memnew(CameraFeedAndroid(cameraManager, metadata, id, position, cameraOrientation));
add_feed(feed);
}
ACameraManager_deleteCameraIdList(cameraIds);
emit_signal(SNAME(CameraServer::feeds_updated_signal_name));
}
void CameraAndroid::remove_all_feeds() {
for (int i = feeds.size() - 1; i >= 0; i--) {
remove_feed(feeds[i]);
}
if (cameraManager != nullptr) {
ACameraManager_delete(cameraManager);
cameraManager = nullptr;
}
}
void CameraAndroid::set_monitoring_feeds(bool p_monitoring_feeds) {
if (p_monitoring_feeds == monitoring_feeds) {
return;
}
CameraServer::set_monitoring_feeds(p_monitoring_feeds);
if (p_monitoring_feeds) {
if (cameraManager == nullptr) {
cameraManager = ACameraManager_create();
}
update_feeds();
} else {
remove_all_feeds();
}
}
CameraAndroid::~CameraAndroid() {
remove_all_feeds();
}