libcamera: support Rectangle and Size types

Fixes: https://github.com/ayufan/camera-streamer/issues/28
Replaces: https://github.com/ayufan/camera-streamer/pull/30
This commit is contained in:
Kamil Trzcinski 2023-02-22 17:59:29 +01:00
parent d963fdd16d
commit aafd120af3
3 changed files with 70 additions and 26 deletions

View File

@ -13,42 +13,48 @@ extern unsigned char html_webrtc_html[];
extern unsigned int html_webrtc_html_len; extern unsigned int html_webrtc_html_len;
extern camera_t *camera; extern camera_t *camera;
void *camera_http_set_option(http_worker_t *worker, FILE *stream, const char *key, const char *value, void *headersp) static void http_once(FILE *stream, void (*fn)(FILE *stream, const char *data), void *headersp)
{ {
bool *headers = headersp; bool *headers = headersp;
if (!camera) {
if (!*headers) { if (!*headers) {
http_500(stream, ""); fn(stream, "");
*headers = true; *headers = true;
} }
}
void *camera_http_set_option(http_worker_t *worker, FILE *stream, const char *key, const char *value, void *headersp)
{
if (!camera) {
http_once(stream, http_500, headersp);
fprintf(stream, "No camera attached.\r\n"); fprintf(stream, "No camera attached.\r\n");
return NULL; return NULL;
} }
bool set = false; bool found = false;
for (int i = 0; i < MAX_DEVICES; i++) { for (int i = 0; i < MAX_DEVICES; i++) {
if (device_set_option_string(camera->devices[i], key, value) == 0) { device_t *dev = camera->devices[i];
set = true; if (!dev) {
break; continue;
}
} }
if (set) { int ret = device_set_option_string(dev, key, value);
if (!*headers) { if (ret > 0) {
http_200(stream, ""); http_once(stream, http_200, headersp);
*headers = true; fprintf(stream, "%s: The '%s' was set to '%s'.\r\n", dev->name, key, value);
} else if (ret < 0) {
http_once(stream, http_500, headersp);
fprintf(stream, "%s: Cannot set '%s' to '%s'.\r\n", dev->name, key, value);
} }
fprintf(stream, "The '%s' was set to '%s'.\r\n", key, value); found = true;
} else {
if (!*headers) {
http_500(stream, "");
*headers = true;
}
fprintf(stream, "Cannot set '%s' to '%s'.\r\n", key, value);
} }
if (found)
return NULL;
http_once(stream, http_404, headersp);
fprintf(stream, "The '%s' was set not found.\r\n", key);
return NULL; return NULL;
} }
@ -56,7 +62,9 @@ void camera_http_option(http_worker_t *worker, FILE *stream)
{ {
bool headers = false; bool headers = false;
http_enum_params(worker, stream, camera_http_set_option, &headers); http_enum_params(worker, stream, camera_http_set_option, &headers);
if (!headers) { if (headers) {
fprintf(stream, "---\r\n");
} else {
http_404(stream, ""); http_404(stream, "");
fprintf(stream, "No options passed.\r\n"); fprintf(stream, "No options passed.\r\n");
} }

View File

@ -258,7 +258,7 @@ int device_set_rotation(device_t *dev, bool vflip, bool hflip)
int device_set_option_string(device_t *dev, const char *key, const char *value) int device_set_option_string(device_t *dev, const char *key, const char *value)
{ {
if (!dev) { if (!dev) {
return -1; return 0;
} }
return dev->hw->device_set_option(dev, key, value); return dev->hw->device_set_option(dev, key, value);

View File

@ -173,9 +173,43 @@ int libcamera_device_set_option(device_t *dev, const char *keyp, const char *val
control_value.set<float>(atof(value)); control_value.set<float>(atof(value));
break; break;
case libcamera::ControlTypeString:
case libcamera::ControlTypeRectangle: case libcamera::ControlTypeRectangle:
static const char *RECTANGLE_PATTERNS[] = {
"(%d,%d)/%ux%u",
"%d,%d,%u,%u",
NULL
};
for (int i = 0; RECTANGLE_PATTERNS[i]; i++) {
libcamera::Rectangle rectangle;
if (4 == sscanf(value, RECTANGLE_PATTERNS[i],
&rectangle.x, &rectangle.y,
&rectangle.width, &rectangle.height)) {
control_value.set(rectangle);
break;
}
}
break;
case libcamera::ControlTypeSize: case libcamera::ControlTypeSize:
static const char *SIZE_PATTERNS[] = {
"%ux%u",
"%u,%u",
NULL
};
for (int i = 0; SIZE_PATTERNS[i]; i++) {
libcamera::Size size;
if (2 == sscanf(value, SIZE_PATTERNS[i], &size.width, &size.height)) {
control_value.set(size);
break;
}
}
break;
case libcamera::ControlTypeString:
break; break;
} }
@ -187,9 +221,11 @@ int libcamera_device_set_option(device_t *dev, const char *keyp, const char *val
control_key.c_str(), control_id->id(), control_id->type(), control_key.c_str(), control_id->id(), control_id->type(),
control_value.toString().c_str()); control_value.toString().c_str());
dev->libcamera->controls.set(control_id->id(), control_value); dev->libcamera->controls.set(control_id->id(), control_value);
return 0; return 1;
} }
return 0;
error: error:
return -1; return -1;
} }