0

概括

我目前正在尝试v4l2_ctrl_handler_setup在 Linux 内核驱动程序中调用后将自定义 V4L2 控件添加到 V4L2 设备。但是似乎没有添加控件(运行时不显示v4l2-ctl --list-ctrls)。下面通常是我试图采取的方法。

static int add_custom_v4l2_ctrls(struct tegracam_device *tc_dev)
{
    struct camera_common_data *s_data = tc_dev->s_data;
    struct v4l2_ctrl_handler *ctrl_handler = s_data->ctrl_handler;
    struct v4l2_ctrl *ctrl;
    int err = 0;

    static struct v4l2_ctrl_config my_control = {
        .ops = &my_custom_ctrl_ops,
        .id = TEGRA_CAMERA_CID_BASE+150,
        .name = "My control",
        .type = V4L2_CTRL_TYPE_INTEGER,
        .flags = V4L2_CTRL_FLAG_SLIDER,
        .min = 0,
        .max = 1,
        .def = 0,
        .step = 1,
    };

    // Increment number of controls
    tc_dev->numctrls++;
    s_data->numctrls++;

    ctrl = v4l2_ctrl_new_custom(ctrl_handler, &my_control, NULL);
    if(ctrl == NULL) {
        dev_err(tc_dev->dev, "Failed to init ctrl");
        return -EIO;
    }

    // err = v4l2_ctrl_handler_setup(ctrl_handler);
    if(err) {
        printk("FAILED");
    }

    return 0;
}

此代码片段在有效调用v4l2_ctrl_handler_setupand之后运行v4l2_async_register_subdev

问题

注册设备后是否可以添加自定义 V4L2 控件?如果是这样,我的方法有什么问题导致控件不显示?

更多信息

该驱动程序是使用 NVIDIA 的 Tegracam V2 框架实现的,该框架抽象了 V4L2 设置代码,包括添加控件,目前它不公开添加自定义 V4L2 控件的能力,这是这种方法背后的原因。

4

2 回答 2

1

这可能来晚了,因为您已经标记了一个解决方案,但我相信我有一个可以提供帮助的解决方案。我将尽力解释我是如何将自定义控件添加到 tegracamv2 框架的。

不幸的是,您无法在子设备注册后添加控件,但是您可以使用结构字段中v4l2_async_register_subdev定义的注册回调挂钩到异步框架。这允许您定义自己的控制处理程序,然后将其从 tegracam 框架添加到控制处理程序,然后将其添加到 v4l2 设备。v4l2sd_internal_opstc_dev

我最近为此苦苦挣扎,我发现这是我唯一的解决方案,除了修补v4l2_async_register_subdevtegracam 框架中的调用并在添加控件后手动调用它(这是未经测试的)。

相关代码(这假设了一个数组ctrl_config,但展示了这个想法。我还假设您已经定义了 ctrl_ops 和配置。):

static int my_subdev_register(struct v4l2_subdev *sd)
{
    struct i2c_client *client = v4l2_get_subdevdata(sd);
    struct camera_common_data *s_data = to_camera_common_data(&client->dev);
    struct my_priv *priv = (struct my_priv *)s_data->priv;
    struct v4l2_ctrl *ctrl;
    int err, i, num_ctrls;

    // setup custom controls
    num_ctrls = ARRAY_SIZE(ctrl_config_list);
    v4l2_ctrl_handler_init(&priv->ctrl_handler, num_ctrls);

    for (i = 0; i < num_ctrls; i++) {
        ctrl = v4l2_ctrl_new_custom(&priv->ctrl_handler,
            &ctrl_config_list[i], NULL);
        if (ctrl == NULL) {
            dev_err(&client->dev, "Failed to init %s ctrl\n",
                ctrl_config_list[i].name);
            continue;
        }
        if (err)
            return err;
        // I believe storing the pointer is technically optional
        priv->ctrls[i] = ctrl;
    }

    priv->num_ctrls = num_ctrls;
    //This is where the magic happens
    err = v4l2_ctrl_add_handler(
            sd->ctrl_handler,
            &priv->ctrl_handler,
            NULL);
    if (err)
        goto error;
    return 0;
error:
    v4l2_ctrl_handler_free(&priv->ctrl_handler);
    return err;
}
...
static const struct v4l2_subdev_internal_ops my_subdev_internal_ops = {
    .registered = my_subdev_register,
};

然后在探测函数中包含以下内容tegracam_device_register。(但之前不会调用它tegracam_v4l2subdev_register。)

tc_dev->sensor_ops = &my_subdev_internal_ops;
于 2021-06-16T12:17:41.583 回答
0

在跟踪调用堆栈后,我找到了这个顺序(箭头代表调用,链接指向 Linux 源代码)。

v4l2_async_register_subdev -> v4l2_async_match_notify -> v4l2_device_register_subdev -> v4l2_ctrl_add_handler

最后一个函数 ( v4l2_ctrl_add_handler ) 通过并将 V4L2 控件从一个处理程序复制到另一个处理程序。

因此,如果在调用之前未添加 V4L2 控件v4l2_async_register_subdev,则该控件将不会被复制到不同的设备,因此将不是可用的有效选项。

所以从我发现的总结来看,不,在设备注册后不可能添加 V4L2 控件。

于 2021-05-07T20:02:15.787 回答