ov2640_probe_dt从设备树中获取ov2640的GPIO引脚并进行初始化
ov2640_probe_dt从设备树中获取ov2640的GPIO引脚并进行初始化
这个函数用于从设备树中获取OV2640摄像头的GPIO引脚并进行初始化。下面是对该函数的概括总结:
请求并获取重置引脚(resetb_gpio)。如果重置引脚未分配,则打印调试信息。如果获取重置引脚失败,则返回错误码。
请求并获取电源引脚(pwdn_gpio)。如果电源引脚未分配,则打印调试信息。如果获取电源引脚失败,则返回错误码。
初始化soc_camera_subdev_desc结构体。
设置电源控制函数为ov2640_hw_power。
设置复位函数为ov2640_hw_reset。
将soc_camera_subdev_desc结构体的指针赋值给i2c_client的platform_data字段。
该函数的主要作用是通过设备树获取OV2640摄像头的GPIO引脚,并将相关信息存储在ov2640_priv结构体中,以便后续的初始化操作使用。
// 从设备树中获取ov2640的GPIO引脚并进行初始化 static int ov2640_probe_dt(struct i2c_client *client, struct ov2640_priv *priv) { /* 请求重置GPIO引脚 */ priv->resetb_gpio = devm_gpiod_get_optional(&client->dev, "resetb", GPIOD_OUT_LOW); if (!priv->resetb_gpio) dev_dbg(&client->dev, "resetb gpio Not allocated!\n"); else if (IS_ERR(priv->resetb_gpio)) return PTR_ERR(priv->resetb_gpio); /* 请求电源GPIO引脚 */ priv->pwdn_gpio = devm_gpiod_get_optional(&client->dev, "pwdn", GPIOD_OUT_HIGH); if (!priv->pwdn_gpio) dev_dbg(&client->dev, "pwdn gpio Not allocated!\n"); else if (IS_ERR(priv->pwdn_gpio)) return PTR_ERR(priv->pwdn_gpio); /* 初始化soc_camera_subdev_desc */ priv->ssdd_dt.power = ov2640_hw_power; // 设置电源控制函数 priv->ssdd_dt.reset = ov2640_hw_reset; // 设置复位函数 client->dev.platform_data = &priv->ssdd_dt; // 设置platform_data return 0; }
v4l2_i2c_subdev_init初始化v4l2子设备
这个函数用于初始化v4l2_subdev结构体,并与i2c_client进行关联。下面是对该函数的概括总结:
使用提供的v4l2_subdev_ops初始化v4l2_subdev结构体。
设置标志位,表示该子设备是I2C设备。
将v4l2_subdev的owner字段设置为i2c_client的driver owner。
将v4l2_subdev的dev字段设置为指向i2c_client的dev。
使用v4l2_set_subdevdata函数将v4l2_subdev的私有数据设置为i2c_client。
使用i2c_set_clientdata函数将i2c_client的私有数据设置为v4l2_subdev。
初始化name字段,格式为"driver_name adapter_id-addr",其中driver_name是i2c_client的driver name,adapter_id是i2c_adapter的ID,addr是i2c_client的地址。
该函数的主要作用是将v4l2_subdev结构体与i2c_client进行关联,并初始化相应的字段,以便后续的操作可以方便地访问和管理I2C子设备。
void v4l2_i2c_subdev_init(struct v4l2_subdev *sd, struct i2c_client *client, const struct v4l2_subdev_ops *ops) // 初始化 v4l2_subdev 结构体 { v4l2_subdev_init(sd, ops); // 初始化 v4l2_subdev 结构体 sd->flags |= V4L2_SUBDEV_FL_IS_I2C; // 设置标志位,表示是 I2C 设备 /* the owner is the same as the i2c_client's driver owner */ sd->owner = client->dev.driver->owner; // 设置 owner,与 i2c_client 的 driver owner 相同 sd->dev = &client->dev; // 设置 dev,指向 i2c_client 的 dev /* i2c_client and v4l2_subdev point to one another */ v4l2_set_subdevdata(sd, client); // 设置 v4l2_subdev 的私有数据为 i2c_client i2c_set_clientdata(client, sd); // 设置 i2c_client 的私有数据为 v4l2_subdev /* initialize name */ snprintf(sd->name, sizeof(sd->name), "%s %d-%04x", client->dev.driver->name, i2c_adapter_id(client->adapter), client->addr); // 初始化 name,格式为 "driver_name adapter_id-addr" }
ov2640_subdev_ops定义了OV2640摄像头子设备的操作函数。下面是对该代码的概括总结:
ov2640_subdev_core_ops结构体定义了OV2640子设备的核心操作函数,包括获取/设置寄存器值和设置电源状态等。这些函数在配置和控制OV2640摄像头时起作用。
ov2640_subdev_video_ops结构体定义了OV2640子设备的视频操作函数,包括开始流、获取/设置视频格式、获取/设置裁剪参数、枚举视频格式和获取总线配置等。这些函数用于处理与视频数据相关的操作。
ov2640_subdev_ops结构体定义了OV2640子设备的全部操作函数,包括核心操作和视频操作。这些函数将被用于初始化和管理OV2640摄像头子设备。
这些操作函数提供了对OV2640摄像头子设备的核心功能和视频功能的支持,使得应用程序可以方便地配置、控制和获取摄像头的数据。
// ov2640子设备核心操作 static struct v4l2_subdev_core_ops ov2640_subdev_core_ops = { #ifdef CONFIG_VIDEO_ADV_DEBUG .g_register = ov2640_g_register, // 获取寄存器值 .s_register = ov2640_s_register, // 设置寄存器值 #endif .s_power = ov2640_s_power, // 设置ov2640的电源 }; static struct v4l2_subdev_video_ops ov2640_subdev_video_ops = { .s_stream = ov2640_s_stream, // 开始流 .g_mbus_fmt = ov2640_g_fmt, // 获取格式 .s_mbus_fmt = ov2640_s_fmt, // 设置格式 .try_mbus_fmt = ov2640_try_fmt, // 尝试格式 .cropcap = ov2640_cropcap, // 裁剪能力 .g_crop = ov2640_g_crop, // 获取裁剪 .enum_mbus_fmt = ov2640_enum_fmt, // 枚举格式 .g_mbus_config = ov2640_g_mbus_config, // 获取总线配置 }; // ov2640子设备操作 static struct v4l2_subdev_ops ov2640_subdev_ops = { .core = &ov2640_subdev_core_ops, // 核心操作 .video = &ov2640_subdev_video_ops, // 视频操作 };
v4l2_ctrl_new_std添加vflip控制器
这个函数用于创建一个新的V4L2控件(control)。下面是对该函数的概括总结:
接受一个V4L2控制器处理器(v4l2_ctrl_handler)指针(hdl),控件操作(ops),控件的ID(id),最小值(min),最大值(max),步长(step)和默认值(def)作为参数。
函数内部定义了控件名称(name)、控件类型(type)和控件标志(flags)的变量。
调用v4l2_ctrl_fill函数来填充控件信息,包括名称、类型、最小值、最大值、步长、默认值和标志。
如果控件类型是菜单(V4L2_CTRL_TYPE_MENU)、整数菜单(V4L2_CTRL_TYPE_INTEGER_MENU)或者复合类型(V4L2_CTRL_COMPOUND_TYPES),则设置处理器的错误信息为EINVAL,并返回空指针。
调用v4l2_ctrl_new函数来创建一个新的控件,使用填充的控件信息和提供的参数。
返回新创建的控件的指针。
该函数的主要作用是通过提供的参数创建一个新的V4L2控件,并将其添加到V4L2控制器处理器中。控件用于管理和控制设备的各种参数和功能。
struct v4l2_ctrl *v4l2_ctrl_new_std(struct v4l2_ctrl_handler *hdl, // 创建一个新的控件 const struct v4l2_ctrl_ops *ops, // 控件操作 u32 id, s64 min, s64 max, u64 step, s64 def) // 控件的id,最小值,最大值,步长,缺省值 { const char *name; // 控件名称 enum v4l2_ctrl_type type; // 控件类型 u32 flags; // 控件标志 v4l2_ctrl_fill(id, &name, &type, &min, &max, &step, &def, &flags); // 填充控件信息 if (type == V4L2_CTRL_TYPE_MENU || // 如果控件类型是菜单 type == V4L2_CTRL_TYPE_INTEGER_MENU || // 或者是整数菜单 type >= V4L2_CTRL_COMPOUND_TYPES) { // 或者是复合类型 handler_set_err(hdl, -EINVAL); // 设置错误信息 return NULL; // 返回空指针 } return v4l2_ctrl_new(hdl, ops, NULL, id, name, type, // 创建新的控件 min, max, step, def, NULL, 0, flags, NULL, NULL, NULL); }
// ov2640控制器操作 static const struct v4l2_ctrl_ops ov2640_ctrl_ops = { .s_ctrl = ov2640_s_ctrl, // 设置ov2640_s_ctrl函数为s_ctrl操作 };
如果文章对您有帮助,点赞👍支持,感谢🤝