/* * V4L2 subdev interface library * * Copyright (C) 2010-2014 Ideas on board SPRL * * Contact: Laurent Pinchart * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU Lesser General Public License as published * by the Free Software Foundation; either version 2.1 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public License * along with this program. If not, see . */ #include #include #include #include #include #include #include #include #include #include #include #include #include "mediactl-priv.h" #include "mediactl.h" #include "tools.h" #include "v4l2subdev.h" #ifdef LOG_TAG #undef LOG_TAG #endif #define LOG_TAG "aiqtool" int v4l2_subdev_open(struct media_entity* entity) { if (entity->fd != -1) { return 0; } entity->fd = open(entity->devname, O_RDWR); if (entity->fd == -1) { int ret = -errno; media_dbg(entity->media, "%s: Failed to open subdev device node %s\n", __func__, entity->devname); return ret; } return 0; } void v4l2_subdev_close(struct media_entity* entity) { close(entity->fd); entity->fd = -1; } int v4l2_subdev_get_format(struct media_entity* entity, struct v4l2_mbus_framefmt* format, unsigned int pad, enum v4l2_subdev_format_whence which) { struct v4l2_subdev_format fmt; int ret; ret = v4l2_subdev_open(entity); if (ret < 0) { return ret; } memset(&fmt, 0, sizeof(fmt)); fmt.pad = pad; fmt.which = which; ret = ioctl(entity->fd, VIDIOC_SUBDEV_G_FMT, &fmt); if (ret < 0) { return -errno; } *format = fmt.format; return 0; } int v4l2_subdev_set_format(struct media_entity* entity, struct v4l2_mbus_framefmt* format, unsigned int pad, enum v4l2_subdev_format_whence which) { struct v4l2_subdev_format fmt; int ret; ret = v4l2_subdev_open(entity); if (ret < 0) { return ret; } memset(&fmt, 0, sizeof(fmt)); fmt.pad = pad; fmt.which = which; fmt.format = *format; ret = ioctl(entity->fd, VIDIOC_SUBDEV_S_FMT, &fmt); if (ret < 0) { return -errno; } *format = fmt.format; return 0; } int v4l2_subdev_get_selection(struct media_entity* entity, struct v4l2_rect* rect, unsigned int pad, unsigned int target, enum v4l2_subdev_format_whence which) { union { struct v4l2_subdev_selection sel; struct v4l2_subdev_crop crop; } u; int ret; ret = v4l2_subdev_open(entity); if (ret < 0) { return ret; } memset(&u.sel, 0, sizeof(u.sel)); u.sel.pad = pad; u.sel.target = target; u.sel.which = which; ret = ioctl(entity->fd, VIDIOC_SUBDEV_G_SELECTION, &u.sel); if (ret >= 0) { *rect = u.sel.r; return 0; } if (errno != ENOTTY || target != V4L2_SEL_TGT_CROP) { return -errno; } memset(&u.crop, 0, sizeof(u.crop)); u.crop.pad = pad; u.crop.which = which; ret = ioctl(entity->fd, VIDIOC_SUBDEV_G_CROP, &u.crop); if (ret < 0) { return -errno; } *rect = u.crop.rect; return 0; } int v4l2_subdev_set_selection(struct media_entity* entity, struct v4l2_rect* rect, unsigned int pad, unsigned int target, enum v4l2_subdev_format_whence which) { union { struct v4l2_subdev_selection sel; struct v4l2_subdev_crop crop; } u; int ret; ret = v4l2_subdev_open(entity); if (ret < 0) { return ret; } memset(&u.sel, 0, sizeof(u.sel)); u.sel.pad = pad; u.sel.target = target; u.sel.which = which; u.sel.r = *rect; ret = ioctl(entity->fd, VIDIOC_SUBDEV_S_SELECTION, &u.sel); if (ret >= 0) { *rect = u.sel.r; return 0; } if (errno != ENOTTY || target != V4L2_SEL_TGT_CROP) { return -errno; } memset(&u.crop, 0, sizeof(u.crop)); u.crop.pad = pad; u.crop.which = which; u.crop.rect = *rect; ret = ioctl(entity->fd, VIDIOC_SUBDEV_S_CROP, &u.crop); if (ret < 0) { return -errno; } *rect = u.crop.rect; return 0; } int v4l2_subdev_get_dv_timings_caps(struct media_entity* entity, struct v4l2_dv_timings_cap* caps) { unsigned int pad = caps->pad; int ret; ret = v4l2_subdev_open(entity); if (ret < 0) { return ret; } memset(caps, 0, sizeof(*caps)); caps->pad = pad; ret = ioctl(entity->fd, VIDIOC_SUBDEV_DV_TIMINGS_CAP, caps); if (ret < 0) { return -errno; } return 0; } int v4l2_subdev_query_dv_timings(struct media_entity* entity, struct v4l2_dv_timings* timings) { int ret; ret = v4l2_subdev_open(entity); if (ret < 0) { return ret; } memset(timings, 0, sizeof(*timings)); ret = ioctl(entity->fd, VIDIOC_SUBDEV_QUERY_DV_TIMINGS, timings); if (ret < 0) { return -errno; } return 0; } int v4l2_subdev_get_dv_timings(struct media_entity* entity, struct v4l2_dv_timings* timings) { int ret; ret = v4l2_subdev_open(entity); if (ret < 0) { return ret; } memset(timings, 0, sizeof(*timings)); ret = ioctl(entity->fd, VIDIOC_SUBDEV_G_DV_TIMINGS, timings); if (ret < 0) { return -errno; } return 0; } int v4l2_subdev_set_dv_timings(struct media_entity* entity, struct v4l2_dv_timings* timings) { int ret; ret = v4l2_subdev_open(entity); if (ret < 0) { return ret; } ret = ioctl(entity->fd, VIDIOC_SUBDEV_S_DV_TIMINGS, timings); if (ret < 0) { return -errno; } return 0; } int v4l2_subdev_get_frame_interval(struct media_entity* entity, struct v4l2_fract* interval, unsigned int pad) { struct v4l2_subdev_frame_interval ival; int ret; ret = v4l2_subdev_open(entity); if (ret < 0) { return ret; } memset(&ival, 0, sizeof(ival)); ival.pad = pad; ret = ioctl(entity->fd, VIDIOC_SUBDEV_G_FRAME_INTERVAL, &ival); if (ret < 0) { return -errno; } *interval = ival.interval; return 0; } int v4l2_subdev_set_frame_interval(struct media_entity* entity, struct v4l2_fract* interval, unsigned int pad) { struct v4l2_subdev_frame_interval ival; int ret; ret = v4l2_subdev_open(entity); if (ret < 0) { return ret; } memset(&ival, 0, sizeof(ival)); ival.pad = pad; ival.interval = *interval; ret = ioctl(entity->fd, VIDIOC_SUBDEV_S_FRAME_INTERVAL, &ival); if (ret < 0) { return -errno; } *interval = ival.interval; return 0; } static int v4l2_subdev_parse_format(struct media_device* media, struct v4l2_mbus_framefmt* format, const char* p, char** endp) { enum v4l2_mbus_pixelcode code; unsigned int width, height; char* fmt; char* end; /* * Compatibility with the old syntax: consider space as valid * separator between the media bus pixel code and the size. */ for (; isspace(*p); ++p) ; for (end = (char*)p; *end != '/' && *end != ' ' && *end != '\0'; ++end) ; fmt = strndup(p, end - p); if (!fmt) { return -ENOMEM; } code = v4l2_subdev_string_to_pixelcode(fmt); free(fmt); if (code == (enum v4l2_mbus_pixelcode) - 1) { media_dbg(media, "Invalid pixel code '%.*s'\n", end - p, p); return -EINVAL; } p = end + 1; width = strtoul(p, &end, 10); if (*end != 'x') { media_dbg(media, "Expected 'x'\n"); return -EINVAL; } p = end + 1; height = strtoul(p, &end, 10); *endp = end; memset(format, 0, sizeof(*format)); format->width = width; format->height = height; format->code = code; return 0; } static int v4l2_subdev_parse_rectangle(struct media_device* media, struct v4l2_rect* r, const char* p, char** endp) { char* end; if (*p++ != '(') { media_dbg(media, "Expected '('\n"); *endp = (char*)p - 1; return -EINVAL; } r->left = strtoul(p, &end, 10); if (*end != ',') { media_dbg(media, "Expected ','\n"); *endp = end; return -EINVAL; } p = end + 1; r->top = strtoul(p, &end, 10); if (*end++ != ')') { media_dbg(media, "Expected ')'\n"); *endp = end - 1; return -EINVAL; } if (*end != '/') { media_dbg(media, "Expected '/'\n"); *endp = end; return -EINVAL; } p = end + 1; r->width = strtoul(p, &end, 10); if (*end != 'x') { media_dbg(media, "Expected 'x'\n"); *endp = end; return -EINVAL; } p = end + 1; r->height = strtoul(p, &end, 10); *endp = end; return 0; } static int v4l2_subdev_parse_frame_interval(struct media_device* media, struct v4l2_fract* interval, const char* p, char** endp) { char* end; for (; isspace(*p); ++p) ; interval->numerator = strtoul(p, &end, 10); for (p = end; isspace(*p); ++p) ; if (*p++ != '/') { media_dbg(media, "Expected '/'\n"); *endp = (char*)p - 1; return -EINVAL; } for (; isspace(*p); ++p) ; interval->denominator = strtoul(p, &end, 10); *endp = end; return 0; } /* * The debate over whether this function should be named icanhasstr() instead * has been strong and heated. If you feel like this would be an important * change, patches are welcome (or not). */ static bool strhazit(const char* str, const char** p) { int len = strlen(str); if (strncmp(str, *p, len)) { return false; } for (*p += len; isspace(**p); ++*p) ; return true; } static struct media_pad* v4l2_subdev_parse_pad_format(struct media_device* media, struct v4l2_mbus_framefmt* format, struct v4l2_rect* crop, struct v4l2_rect* compose, struct v4l2_fract* interval, const char* p, char** endp) { struct media_pad* pad; bool first; char* end; int ret; for (; isspace(*p); ++p) ; pad = media_parse_pad(media, p, &end); if (pad == NULL) { *endp = end; return NULL; } for (p = end; isspace(*p); ++p) ; if (*p++ != '[') { media_dbg(media, "Expected '['\n"); *endp = (char*)p - 1; return NULL; } for (first = true;; first = false) { for (; isspace(*p); p++) ; /* * Backward compatibility: if the first property starts with an * uppercase later, process it as a format description. */ if (strhazit("fmt:", &p) || (first && isupper(*p))) { ret = v4l2_subdev_parse_format(media, format, p, &end); if (ret < 0) { *endp = end; return NULL; } p = end; continue; } if (strhazit("field:", &p)) { enum v4l2_field field; char* strfield; for (end = (char*)p; isalpha(*end) || *end == '-'; ++end) ; strfield = strndup(p, end - p); if (!strfield) { *endp = (char*)p; return NULL; } field = v4l2_subdev_string_to_field(strfield); free(strfield); if (field == (enum v4l2_field) - 1) { media_dbg(media, "Invalid field value '%*s'\n", end - p, p); *endp = (char*)p; return NULL; } format->field = field; p = end; continue; } if (strhazit("colorspace:", &p)) { enum v4l2_colorspace colorspace; char* strfield; for (end = (char*)p; isalnum(*end) || *end == '-'; ++end) ; strfield = strndup(p, end - p); if (!strfield) { *endp = (char*)p; return NULL; } colorspace = v4l2_subdev_string_to_colorspace(strfield); free(strfield); if (colorspace == (enum v4l2_colorspace) - 1) { media_dbg(media, "Invalid colorspace value '%*s'\n", end - p, p); *endp = (char*)p; return NULL; } format->colorspace = colorspace; p = end; continue; } if (strhazit("xfer:", &p)) { enum v4l2_xfer_func xfer_func; char* strfield; for (end = (char*)p; isalnum(*end) || *end == '-'; ++end) ; strfield = strndup(p, end - p); if (!strfield) { *endp = (char*)p; return NULL; } xfer_func = v4l2_subdev_string_to_xfer_func(strfield); free(strfield); if (xfer_func == (enum v4l2_xfer_func) - 1) { media_dbg(media, "Invalid transfer function value '%*s'\n", end - p, p); *endp = (char*)p; return NULL; } format->xfer_func = xfer_func; p = end; continue; } if (strhazit("ycbcr:", &p)) { enum v4l2_ycbcr_encoding ycbcr_enc; char* strfield; for (end = (char*)p; isalnum(*end) || *end == '-'; ++end) ; strfield = strndup(p, end - p); if (!strfield) { *endp = (char*)p; return NULL; } ycbcr_enc = v4l2_subdev_string_to_ycbcr_encoding(strfield); free(strfield); if (ycbcr_enc == (enum v4l2_ycbcr_encoding) - 1) { media_dbg(media, "Invalid YCbCr encoding value '%*s'\n", end - p, p); *endp = (char*)p; return NULL; } format->ycbcr_enc = ycbcr_enc; p = end; continue; } if (strhazit("quantization:", &p)) { enum v4l2_quantization quantization; char* strfield; for (end = (char*)p; isalnum(*end) || *end == '-'; ++end) ; strfield = strndup(p, end - p); if (!strfield) { *endp = (char*)p; return NULL; } quantization = v4l2_subdev_string_to_quantization(strfield); free(strfield); if (quantization == (enum v4l2_quantization) - 1) { media_dbg(media, "Invalid quantization value '%*s'\n", end - p, p); *endp = (char*)p; return NULL; } format->quantization = quantization; p = end; continue; } /* * Backward compatibility: crop rectangles can be specified * implicitly without the 'crop:' property name. */ if (strhazit("crop:", &p) || *p == '(') { ret = v4l2_subdev_parse_rectangle(media, crop, p, &end); if (ret < 0) { *endp = end; return NULL; } p = end; continue; } if (strhazit("compose:", &p)) { ret = v4l2_subdev_parse_rectangle(media, compose, p, &end); if (ret < 0) { *endp = end; return NULL; } for (p = end; isspace(*p); p++) ; continue; } if (*p == '@') { ret = v4l2_subdev_parse_frame_interval(media, interval, ++p, &end); if (ret < 0) { *endp = end; return NULL; } p = end; continue; } break; } if (*p != ']') { media_dbg(media, "Expected ']'\n"); *endp = (char*)p; return NULL; } *endp = (char*)p + 1; return pad; } static int set_format(struct media_pad* pad, struct v4l2_mbus_framefmt* format) { int ret; if (format->width == 0 || format->height == 0) { return 0; } media_dbg(pad->entity->media, "Setting up format %s %ux%u on pad %s/%u\n", v4l2_subdev_pixelcode_to_string(format->code), format->width, format->height, pad->entity->info.name, pad->index); ret = v4l2_subdev_set_format(pad->entity, format, pad->index, V4L2_SUBDEV_FORMAT_ACTIVE); if (ret < 0) { media_dbg(pad->entity->media, "Unable to set format: %s (%d)\n", strerror(-ret), ret); return ret; } media_dbg(pad->entity->media, "Format set: %s %ux%u\n", v4l2_subdev_pixelcode_to_string(format->code), format->width, format->height); return 0; } static int set_selection(struct media_pad* pad, unsigned int target, struct v4l2_rect* rect) { int ret; if (rect->left == -1 || rect->top == -1) { return 0; } media_dbg(pad->entity->media, "Setting up selection target %u rectangle (%u,%u)/%ux%u on pad %s/%u\n", target, rect->left, rect->top, rect->width, rect->height, pad->entity->info.name, pad->index); ret = v4l2_subdev_set_selection(pad->entity, rect, pad->index, target, V4L2_SUBDEV_FORMAT_ACTIVE); if (ret < 0) { media_dbg(pad->entity->media, "Unable to set selection rectangle: %s (%d)\n", strerror(-ret), ret); return ret; } media_dbg(pad->entity->media, "Selection rectangle set: (%u,%u)/%ux%u\n", rect->left, rect->top, rect->width, rect->height); return 0; } static int set_frame_interval(struct media_pad* pad, struct v4l2_fract* interval) { int ret; if (interval->numerator == 0) { return 0; } media_dbg(pad->entity->media, "Setting up frame interval %u/%u on pad %s/%u\n", interval->numerator, interval->denominator, pad->entity->info.name, pad->index); ret = v4l2_subdev_set_frame_interval(pad->entity, interval, pad->index); if (ret < 0) { media_dbg(pad->entity->media, "Unable to set frame interval: %s (%d)", strerror(-ret), ret); return ret; } media_dbg(pad->entity->media, "Frame interval set: %u/%u\n", interval->numerator, interval->denominator); return 0; } static int v4l2_subdev_parse_setup_format(struct media_device* media, const char* p, char** endp) { struct v4l2_mbus_framefmt format = {0, 0, 0}; struct media_pad* pad; struct v4l2_rect crop = {-1, -1, -1, -1}; struct v4l2_rect compose = crop; struct v4l2_fract interval = {0, 0}; unsigned int i; char* end; int ret; pad = v4l2_subdev_parse_pad_format(media, &format, &crop, &compose, &interval, p, &end); if (pad == NULL) { media_print_streampos(media, p, end); media_dbg(media, "Unable to parse format\n"); return -EINVAL; } if (pad->flags & MEDIA_PAD_FL_SINK) { ret = set_format(pad, &format); if (ret < 0) { return ret; } } ret = set_selection(pad, V4L2_SEL_TGT_CROP, &crop); if (ret < 0) { return ret; } ret = set_selection(pad, V4L2_SEL_TGT_COMPOSE, &compose); if (ret < 0) { return ret; } if (pad->flags & MEDIA_PAD_FL_SOURCE) { ret = set_format(pad, &format); if (ret < 0) { return ret; } } ret = set_frame_interval(pad, &interval); if (ret < 0) { return ret; } /* If the pad is an output pad, automatically set the same format and * frame interval on the remote subdev input pads, if any. */ if (pad->flags & MEDIA_PAD_FL_SOURCE) { for (i = 0; i < pad->entity->num_links; ++i) { struct media_link* link = &pad->entity->links[i]; struct v4l2_mbus_framefmt remote_format; if (!(link->flags & MEDIA_LNK_FL_ENABLED)) { continue; } if (link->source == pad && link->sink->entity->info.type == MEDIA_ENT_T_V4L2_SUBDEV) { remote_format = format; set_format(link->sink, &remote_format); ret = set_frame_interval(link->sink, &interval); if (ret < 0 && ret != -EINVAL && ret != -ENOTTY) { return ret; } } } } *endp = end; return 0; } int v4l2_subdev_parse_setup_formats(struct media_device* media, const char* p) { char* end; int ret; do { ret = v4l2_subdev_parse_setup_format(media, p, &end); if (ret < 0) { return ret; } for (; isspace(*end); end++) ; p = end + 1; } while (*end == ','); return *end ? -EINVAL : 0; } static const struct { const char* name; enum v4l2_mbus_pixelcode code; } mbus_formats[] = { #include "media-bus-format-names.h" {"Y8", MEDIA_BUS_FMT_Y8_1X8}, {"Y10", MEDIA_BUS_FMT_Y10_1X10}, {"Y12", MEDIA_BUS_FMT_Y12_1X12}, {"YUYV", MEDIA_BUS_FMT_YUYV8_1X16}, {"YUYV1_5X8", MEDIA_BUS_FMT_YUYV8_1_5X8}, {"YUYV2X8", MEDIA_BUS_FMT_YUYV8_2X8}, {"UYVY", MEDIA_BUS_FMT_UYVY8_1X16}, {"UYVY1_5X8", MEDIA_BUS_FMT_UYVY8_1_5X8}, {"UYVY2X8", MEDIA_BUS_FMT_UYVY8_2X8}, {"VUY24", MEDIA_BUS_FMT_VUY8_1X24}, {"SBGGR8", MEDIA_BUS_FMT_SBGGR8_1X8}, {"SGBRG8", MEDIA_BUS_FMT_SGBRG8_1X8}, {"SGRBG8", MEDIA_BUS_FMT_SGRBG8_1X8}, {"SRGGB8", MEDIA_BUS_FMT_SRGGB8_1X8}, {"SBGGR10", MEDIA_BUS_FMT_SBGGR10_1X10}, {"SGBRG10", MEDIA_BUS_FMT_SGBRG10_1X10}, {"SGRBG10", MEDIA_BUS_FMT_SGRBG10_1X10}, {"SRGGB10", MEDIA_BUS_FMT_SRGGB10_1X10}, {"SBGGR10_DPCM8", MEDIA_BUS_FMT_SBGGR10_DPCM8_1X8}, {"SGBRG10_DPCM8", MEDIA_BUS_FMT_SGBRG10_DPCM8_1X8}, {"SGRBG10_DPCM8", MEDIA_BUS_FMT_SGRBG10_DPCM8_1X8}, {"SRGGB10_DPCM8", MEDIA_BUS_FMT_SRGGB10_DPCM8_1X8}, {"SBGGR12", MEDIA_BUS_FMT_SBGGR12_1X12}, {"SGBRG12", MEDIA_BUS_FMT_SGBRG12_1X12}, {"SGRBG12", MEDIA_BUS_FMT_SGRBG12_1X12}, {"SRGGB12", MEDIA_BUS_FMT_SRGGB12_1X12}, {"AYUV32", MEDIA_BUS_FMT_AYUV8_1X32}, {"RBG24", MEDIA_BUS_FMT_RBG888_1X24}, {"RGB32", MEDIA_BUS_FMT_RGB888_1X32_PADHI}, {"ARGB32", MEDIA_BUS_FMT_ARGB8888_1X32}, }; const char* v4l2_subdev_pixelcode_to_string(enum v4l2_mbus_pixelcode code) { unsigned int i; for (i = 0; i < ARRAY_SIZE(mbus_formats); ++i) { if (mbus_formats[i].code == code) { return mbus_formats[i].name; } } return "unknown"; } enum v4l2_mbus_pixelcode v4l2_subdev_string_to_pixelcode(const char* string) { unsigned int i; for (i = 0; i < ARRAY_SIZE(mbus_formats); ++i) { if (strcmp(mbus_formats[i].name, string) == 0) { return mbus_formats[i].code; } } return (enum v4l2_mbus_pixelcode) - 1; } static struct { const char* name; enum v4l2_field field; } fields[] = { {"any", V4L2_FIELD_ANY}, {"none", V4L2_FIELD_NONE}, {"top", V4L2_FIELD_TOP}, {"bottom", V4L2_FIELD_BOTTOM}, {"interlaced", V4L2_FIELD_INTERLACED}, {"seq-tb", V4L2_FIELD_SEQ_TB}, {"seq-bt", V4L2_FIELD_SEQ_BT}, {"alternate", V4L2_FIELD_ALTERNATE}, {"interlaced-tb", V4L2_FIELD_INTERLACED_TB}, {"interlaced-bt", V4L2_FIELD_INTERLACED_BT}, }; const char* v4l2_subdev_field_to_string(enum v4l2_field field) { unsigned int i; for (i = 0; i < ARRAY_SIZE(fields); ++i) { if (fields[i].field == field) { return fields[i].name; } } return "unknown"; } enum v4l2_field v4l2_subdev_string_to_field(const char* string) { unsigned int i; for (i = 0; i < ARRAY_SIZE(fields); ++i) { if (strcasecmp(fields[i].name, string) == 0) { return fields[i].field; } } return (enum v4l2_field) - 1; } static struct { const char* name; enum v4l2_colorspace colorspace; } colorspaces[] = { {"default", V4L2_COLORSPACE_DEFAULT}, {"smpte170m", V4L2_COLORSPACE_SMPTE170M}, {"smpte240m", V4L2_COLORSPACE_SMPTE240M}, {"rec709", V4L2_COLORSPACE_REC709}, {"470m", V4L2_COLORSPACE_470_SYSTEM_M}, {"470bg", V4L2_COLORSPACE_470_SYSTEM_BG}, {"jpeg", V4L2_COLORSPACE_JPEG}, {"srgb", V4L2_COLORSPACE_SRGB}, #ifdef __ANDROID__ {"oprgb", V4L2_COLORSPACE_ADOBERGB}, #else {"oprgb", V4L2_COLORSPACE_OPRGB}, #endif {"bt2020", V4L2_COLORSPACE_BT2020}, {"dcip3", V4L2_COLORSPACE_DCI_P3}, }; const char* v4l2_subdev_colorspace_to_string(enum v4l2_colorspace colorspace) { unsigned int i; for (i = 0; i < ARRAY_SIZE(colorspaces); ++i) { if (colorspaces[i].colorspace == colorspace) { return colorspaces[i].name; } } return "unknown"; } enum v4l2_colorspace v4l2_subdev_string_to_colorspace(const char* string) { unsigned int i; for (i = 0; i < ARRAY_SIZE(colorspaces); ++i) { if (strcasecmp(colorspaces[i].name, string) == 0) { return colorspaces[i].colorspace; } } return (enum v4l2_colorspace) - 1; } static struct { const char* name; enum v4l2_xfer_func xfer_func; } xfer_funcs[] = { {"default", V4L2_XFER_FUNC_DEFAULT}, {"709", V4L2_XFER_FUNC_709}, {"srgb", V4L2_XFER_FUNC_SRGB}, #ifdef __ANDROID__ {"oprgb", V4L2_XFER_FUNC_ADOBERGB}, #else {"oprgb", V4L2_XFER_FUNC_OPRGB}, #endif {"smpte240m", V4L2_XFER_FUNC_SMPTE240M}, {"smpte2084", V4L2_XFER_FUNC_SMPTE2084}, {"dcip3", V4L2_XFER_FUNC_DCI_P3}, {"none", V4L2_XFER_FUNC_NONE}, }; const char* v4l2_subdev_xfer_func_to_string(enum v4l2_xfer_func xfer_func) { unsigned int i; for (i = 0; i < ARRAY_SIZE(xfer_funcs); ++i) { if (xfer_funcs[i].xfer_func == xfer_func) { return xfer_funcs[i].name; } } return "unknown"; } enum v4l2_xfer_func v4l2_subdev_string_to_xfer_func(const char* string) { unsigned int i; for (i = 0; i < ARRAY_SIZE(xfer_funcs); ++i) { if (strcasecmp(xfer_funcs[i].name, string) == 0) { return xfer_funcs[i].xfer_func; } } return (enum v4l2_xfer_func) - 1; } static struct { const char* name; enum v4l2_ycbcr_encoding ycbcr_enc; } ycbcr_encs[] = { {"default", V4L2_YCBCR_ENC_DEFAULT}, {"601", V4L2_YCBCR_ENC_601}, {"709", V4L2_YCBCR_ENC_709}, {"xv601", V4L2_YCBCR_ENC_XV601}, {"xv709", V4L2_YCBCR_ENC_XV709}, {"bt2020", V4L2_YCBCR_ENC_BT2020}, {"bt2020c", V4L2_YCBCR_ENC_BT2020_CONST_LUM}, {"smpte240m", V4L2_YCBCR_ENC_SMPTE240M}, }; const char* v4l2_subdev_ycbcr_encoding_to_string(enum v4l2_ycbcr_encoding ycbcr_enc) { unsigned int i; for (i = 0; i < ARRAY_SIZE(ycbcr_encs); ++i) { if (ycbcr_encs[i].ycbcr_enc == ycbcr_enc) { return ycbcr_encs[i].name; } } return "unknown"; } enum v4l2_ycbcr_encoding v4l2_subdev_string_to_ycbcr_encoding(const char* string) { unsigned int i; for (i = 0; i < ARRAY_SIZE(ycbcr_encs); ++i) { if (strcasecmp(ycbcr_encs[i].name, string) == 0) { return ycbcr_encs[i].ycbcr_enc; } } return (enum v4l2_ycbcr_encoding) - 1; } static struct { const char* name; enum v4l2_quantization quantization; } quantizations[] = { {"default", V4L2_QUANTIZATION_DEFAULT}, {"full-range", V4L2_QUANTIZATION_FULL_RANGE}, {"lim-range", V4L2_QUANTIZATION_LIM_RANGE}, }; const char* v4l2_subdev_quantization_to_string(enum v4l2_quantization quantization) { unsigned int i; for (i = 0; i < ARRAY_SIZE(quantizations); ++i) { if (quantizations[i].quantization == quantization) { return quantizations[i].name; } } return "unknown"; } enum v4l2_quantization v4l2_subdev_string_to_quantization(const char* string) { unsigned int i; for (i = 0; i < ARRAY_SIZE(quantizations); ++i) { if (strcasecmp(quantizations[i].name, string) == 0) { return quantizations[i].quantization; } } return (enum v4l2_quantization) - 1; } static const enum v4l2_mbus_pixelcode mbus_codes[] = { #include "media-bus-format-codes.h" }; const enum v4l2_mbus_pixelcode* v4l2_subdev_pixelcode_list(unsigned int* length) { *length = ARRAY_SIZE(mbus_codes); return mbus_codes; }