Hi Laurent, On 10/7/22 15:19, Laurent Pinchart wrote: > Hi Benjamin, > > On Fri, Oct 07, 2022 at 02:38:33PM +0200, Benjamin MUGNIER wrote: >> On 10/7/22 14:32, Laurent Pinchart wrote: >>> On Fri, Oct 07, 2022 at 02:24:16PM +0200, Benjamin MUGNIER wrote: >>>> Hi Sakari, >>>> >>>> Thank you for your review. >>>> Please consider everything not commented as queued for v7. >>>> >>>> On 10/6/22 21:14, Sakari Ailus wrote: >>>>> Hi Benjamin, >>>>> >>>>> Thanks for the update. A few more comments below... >>>>> >>>>> On Tue, Sep 27, 2022 at 10:37:02AM +0200, Benjamin Mugnier wrote: >>>>>> +static const char * const vgxy61_hdr_mode_menu[] = { >>>>>> + "HDR linearize", >>>>>> + "HDR substraction", >>>>>> + "No HDR", >>>>>> +}; >>>>> >>>>> I think more documentation is needed on the HDR modes. What do these mean? >>>>> For instance, are they something that requires further processing or is the >>>>> result e.g. a ready HDR image? >>>>> >>>>> This should probably make it to driver specific documentation. >>>> >>>> Sure, in fact I did something like this in v3: >>>> https://lore.kernel.org/linux-media/20220512074037.3829926-4-benjamin.mugnier@xxxxxxxxxxx/ >>>> >>>> Since I standardized the control I was unsure what to do with this >>>> documentation, and dropped it. >>>> >>>> I will add back the >>>> Documentation/userspace-api/media/drivers/st-vgxy61.rst file from this >>>> commit to the patchset, while changing the control name to the new >>>> one. >>> >>> The documentation there is about modes for HDR merge on the sensor side. >>> Can the sensor also output the unmerged images, for instance >>> line-interleaved ? >> >> The sensor can not output unmerged images. It can only output a single >> image and frame merging as to be done sensor side. > > I wonder then, should we have two HDR mode controls, one for sensor-side > HDR, and one to control the interleaving of images for host-side HDR ? > The latter would need some standardization I think, as the ISP > configuration needs to match, so there must be some industry de-facto > standards. > I forgot to change it for this version, but Nicolas advised in v5 ircc to change the sensor side control from 'V4L2_CID_HDR_MODE' to 'V4L2_CID_HDR_SENSOR_MODE'. I'll do it for v7. That will give space for the host side HDR control later on. >>>>>> + >>>>>> +static const char * const vgxy61_supply_name[] = { >>>>>> + "VCORE", >>>>>> + "VDDIO", >>>>>> + "VANA", >>>>>> +}; >>>>>> + >>>>>> +#define VGXY61_NUM_SUPPLIES ARRAY_SIZE(vgxy61_supply_name) >>>>> >>>>> Please use plain ARRAY_SIZE() instead. >>>>> >>>>> ... >>>>> >>>>>> +static int vgxy61_poll_reg(struct vgxy61_dev *sensor, u32 reg, u8 poll_val, >>>>>> + unsigned int timeout_ms) >>>>>> +{ >>>>>> + const unsigned int loop_delay_ms = 10; >>>>>> + int ret, timeout; >>>>>> + >>>>>> + timeout = read_poll_timeout(vgxy61_read_reg, ret, >>>>>> + ((ret < 0) || (ret == poll_val)), >>>>>> + loop_delay_ms * 1000, timeout_ms * 1000, >>>>>> + false, sensor, reg); >>>>>> + if (timeout) >>>>>> + return timeout; >>>>>> + >>>>>> + return 0; >>>>> >>>>> "timeout" here is entirely pointless. >>>>> >>>>>> +} >>>>> >>>>> ... >>>>> >>>>>> +static int vgxy61_apply_exposure(struct vgxy61_dev *sensor) >>>>>> +{ >>>>>> + int ret = 0; >>>>>> + >>>>>> + /* We first set expo to zero to avoid forbidden parameters couple */ >>>>>> + ret = vgxy61_write_reg(sensor, VGXY61_REG_COARSE_EXPOSURE_SHORT, >>>>>> + 0, &ret); >>>>>> + ret = vgxy61_write_reg(sensor, VGXY61_REG_COARSE_EXPOSURE_LONG, >>>>>> + sensor->expo_long, &ret); >>>>>> + ret = vgxy61_write_reg(sensor, VGXY61_REG_COARSE_EXPOSURE_SHORT, >>>>>> + sensor->expo_short, &ret); >>>>> >>>>> return vgxy61_write_reg(...); >>>>> >>>>>> + if (ret) >>>>>> + return ret; >>>>>> + >>>>>> + return 0; >>>>>> +} >>>>> >>>>> ... >>>>> >>>>>> +static int vgxy61_init_controls(struct vgxy61_dev *sensor) >>>>>> +{ >>>>>> + const struct v4l2_ctrl_ops *ops = &vgxy61_ctrl_ops; >>>>>> + struct v4l2_ctrl_handler *hdl = &sensor->ctrl_handler; >>>>>> + const struct vgxy61_mode_info *cur_mode = sensor->current_mode; >>>>>> + struct v4l2_ctrl *ctrl; >>>>>> + int ret; >>>>>> + >>>>>> + v4l2_ctrl_handler_init(hdl, 16); >>>>>> + /* We can use our own mutex for the ctrl lock */ >>>>>> + hdl->lock = &sensor->lock; >>>>>> + v4l2_ctrl_new_std(hdl, ops, V4L2_CID_ANALOGUE_GAIN, 0, 0x1c, 1, >>>>>> + sensor->analog_gain); >>>>>> + v4l2_ctrl_new_std(hdl, ops, V4L2_CID_DIGITAL_GAIN, 0, 0xfff, 1, >>>>>> + sensor->digital_gain); >>>>>> + v4l2_ctrl_new_std_menu_items(hdl, ops, V4L2_CID_TEST_PATTERN, >>>>>> + ARRAY_SIZE(vgxy61_test_pattern_menu) - 1, >>>>>> + 0, 0, vgxy61_test_pattern_menu); >>>>>> + ctrl = v4l2_ctrl_new_std(hdl, ops, V4L2_CID_HBLANK, 0, >>>>>> + sensor->line_length, 1, >>>>>> + sensor->line_length - cur_mode->width); >>>>>> + if (ctrl) >>>>>> + ctrl->flags |= V4L2_CTRL_FLAG_READ_ONLY; >>>>>> + ctrl = v4l2_ctrl_new_int_menu(hdl, ops, V4L2_CID_LINK_FREQ, >>>>>> + ARRAY_SIZE(link_freq) - 1, 0, link_freq); >>>>>> + if (ctrl) >>>>>> + ctrl->flags |= V4L2_CTRL_FLAG_READ_ONLY; >>>>>> + v4l2_ctrl_new_std_menu_items(hdl, ops, V4L2_CID_HDR_MODE, >>>>>> + ARRAY_SIZE(vgxy61_hdr_mode_menu) - 1, 0, >>>>>> + VGXY61_NO_HDR, vgxy61_hdr_mode_menu); >>>>>> + >>>>>> + /* >>>>>> + * Keep a pointer to these controls as we need to update them when >>>>>> + * setting the format >>>>>> + */ >>>>>> + sensor->pixel_rate_ctrl = v4l2_ctrl_new_std(hdl, ops, >>>>>> + V4L2_CID_PIXEL_RATE, 1, >>>>>> + INT_MAX, 1, >>>>>> + get_pixel_rate(sensor)); >>>>>> + sensor->pixel_rate_ctrl->flags |= V4L2_CTRL_FLAG_READ_ONLY; >>>>> >>>>> Also sensor->pixel_rate_ctrl may be NULL here. >>>>> >>>>>> + sensor->expo_ctrl = v4l2_ctrl_new_std(hdl, ops, V4L2_CID_EXPOSURE, >>>>>> + sensor->expo_min, >>>>>> + sensor->expo_max, 1, >>>>>> + sensor->expo_long); >>>>>> + sensor->vblank_ctrl = v4l2_ctrl_new_std(hdl, ops, V4L2_CID_VBLANK, >>>>>> + sensor->vblank_min, >>>>>> + 0xffff - cur_mode->crop.height, >>>>>> + 1, sensor->vblank); >>>>>> + sensor->vflip_ctrl = v4l2_ctrl_new_std(hdl, ops, V4L2_CID_VFLIP, >>>>>> + 0, 1, 1, sensor->vflip); >>>>>> + sensor->hflip_ctrl = v4l2_ctrl_new_std(hdl, ops, V4L2_CID_HFLIP, >>>>>> + 0, 1, 1, sensor->hflip); >>>>>> + >>>>>> + if (hdl->error) { >>>>>> + ret = hdl->error; >>>>>> + goto free_ctrls; >>>>>> + } >>>>>> + >>>>>> + sensor->sd.ctrl_handler = hdl; >>>>>> + return 0; >>>>>> + >>>>>> +free_ctrls: >>>>>> + v4l2_ctrl_handler_free(hdl); >>>>>> + return ret; >>>>>> +} >>>>> >>>>> ... >>>>> >>>>>> +static int vgxy61_probe(struct i2c_client *client) >>>>>> +{ >>>>>> + struct device *dev = &client->dev; >>>>>> + struct fwnode_handle *handle; >>>>>> + struct vgxy61_dev *sensor; >>>>>> + int ret; >>>>>> + >>>>>> + sensor = devm_kzalloc(dev, sizeof(*sensor), GFP_KERNEL); >>>>>> + if (!sensor) >>>>>> + return -ENOMEM; >>>>>> + >>>>>> + sensor->i2c_client = client; >>>>>> + sensor->streaming = false; >>>>>> + sensor->hdr = VGXY61_NO_HDR; >>>>>> + sensor->expo_long = 200; >>>>>> + sensor->expo_short = 0; >>>>>> + sensor->hflip = false; >>>>>> + sensor->vflip = false; >>>>>> + sensor->analog_gain = 0; >>>>>> + sensor->digital_gain = 256; >>>>>> + >>>>>> + handle = fwnode_graph_get_next_endpoint(of_fwnode_handle(dev->of_node), >>>>>> + NULL); >>>>> >>>>> handle = fwnode_graph_get_endpoint_by_id(dev_fwnode(dev), 0, 0, 0); >>>>> >>>>>> + if (!handle) { >>>>>> + dev_err(dev, "handle node not found\n"); >>>>>> + return -EINVAL; >>>>>> + } >>>>>> + >>>>>> + ret = vgxy61_tx_from_ep(sensor, handle); >>>>>> + fwnode_handle_put(handle); >>>>>> + if (ret) { >>>>>> + dev_err(dev, "Failed to parse handle %d\n", ret); >>>>>> + return ret; >>>>>> + } >>>>>> + >>>>>> + sensor->xclk = devm_clk_get(dev, NULL); >>>>>> + if (IS_ERR(sensor->xclk)) { >>>>>> + dev_err(dev, "failed to get xclk\n"); >>>>>> + return PTR_ERR(sensor->xclk); >>>>>> + } >>>>>> + sensor->clk_freq = clk_get_rate(sensor->xclk); >>>>>> + if (sensor->clk_freq < 6 * HZ_PER_MHZ || >>>>>> + sensor->clk_freq > 27 * HZ_PER_MHZ) { >>>>>> + dev_err(dev, "Only 6Mhz-27Mhz clock range supported. provide %lu MHz\n", >>>>>> + sensor->clk_freq / HZ_PER_MHZ); >>>>>> + return -EINVAL; >>>>>> + } >>>>>> + sensor->gpios_polarity = of_property_read_bool(dev->of_node, >>>>>> + "invert-gpios-polarity"); >>>>> >>>>> I thought we did discuss dropping support for sensor synchronisation in >>>>> this version? >>>> >>>> This properties affects strobing lights gpios polarities as you can >>>> see in vgxy61_update_gpios_strobe_polarity. If set to '1' all strobing >>>> gpios are inverted. This has nothing to do with the sensor >>>> synchronization. >>>> >>>> Now I realize this is poorly named, and I even forgot to document it >>>> in the device tree bindings file. I apologize. >>>> >>>> I would like to rename it to 'st,strobe-polarity' since this is vendor >>>> specific and to better reflect that it affects strobing gpios. >>>> I'll make this change for v7 and document this in the bindings file >>>> too. Tell me if there is any issues with that. >>>> >>>>>> + >>>>>> + v4l2_i2c_subdev_init(&sensor->sd, client, &vgxy61_subdev_ops); >>>>>> + sensor->sd.flags |= V4L2_SUBDEV_FL_HAS_DEVNODE; >>>>>> + sensor->pad.flags = MEDIA_PAD_FL_SOURCE; >>>>>> + sensor->sd.entity.ops = &vgxy61_subdev_entity_ops; >>>>>> + sensor->sd.entity.function = MEDIA_ENT_F_CAM_SENSOR; >>>>>> + >>>>>> + sensor->reset_gpio = devm_gpiod_get_optional(dev, "reset", >>>>>> + GPIOD_OUT_HIGH); >>>>>> + >>>>>> + ret = vgxy61_get_regulators(sensor); >>>>>> + if (ret) { >>>>>> + dev_err(&client->dev, "failed to get regulators %d\n", ret); >>>>>> + return ret; >>>>>> + } >>>>>> + >>>>>> + ret = regulator_bulk_enable(VGXY61_NUM_SUPPLIES, sensor->supplies); >>>>>> + if (ret) { >>>>>> + dev_err(&client->dev, "failed to enable regulators %d\n", ret); >>>>>> + return ret; >>>>>> + } >>>>>> + >>>>>> + ret = vgxy61_power_on(dev); >>>>>> + if (ret) >>>>>> + return ret; >>>>>> + >>>>>> + ret = vgxy61_detect(sensor); >>>>>> + if (ret) { >>>>>> + dev_err(&client->dev, "sensor detect failed %d\n", ret); >>>>>> + return ret; >>>>>> + } >>>>>> + >>>>>> + vgxy61_fill_sensor_param(sensor); >>>>>> + vgxy61_fill_framefmt(sensor, sensor->current_mode, &sensor->fmt, >>>>>> + VGXY61_MEDIA_BUS_FMT_DEF); >>>>>> + >>>>>> + ret = vgxy61_update_hdr(sensor, sensor->hdr); >>>>>> + if (ret) >>>>>> + return ret; >>>>>> + >>>>>> + mutex_init(&sensor->lock); >>>>>> + >>>>>> + ret = vgxy61_init_controls(sensor); >>>>>> + if (ret) { >>>>>> + dev_err(&client->dev, "controls initialization failed %d\n", >>>>>> + ret); >>>>>> + goto error_power_off; >>>>>> + } >>>>>> + >>>>>> + ret = media_entity_pads_init(&sensor->sd.entity, 1, &sensor->pad); >>>>>> + if (ret) { >>>>>> + dev_err(&client->dev, "pads init failed %d\n", ret); >>>>>> + goto error_handler_free; >>>>>> + } >>>>>> + >>>>>> + /* Enable runtime PM and turn off the device */ >>>>>> + pm_runtime_set_active(dev); >>>>>> + pm_runtime_enable(dev); >>>>>> + pm_runtime_idle(dev); >>>>>> + >>>>>> + ret = v4l2_async_register_subdev(&sensor->sd); >>>>>> + if (ret) { >>>>>> + dev_err(&client->dev, "async subdev register failed %d\n", ret); >>>>>> + goto error_pm_runtime; >>>>>> + } >>>>>> + >>>>>> + pm_runtime_set_autosuspend_delay(&client->dev, 1000); >>>>>> + pm_runtime_use_autosuspend(&client->dev); >>>>>> + >>>>>> + dev_dbg(&client->dev, "vgxy61 probe successfully\n"); >>>>>> + >>>>>> + return 0; >>>>>> + >>>>>> +error_pm_runtime: >>>>>> + pm_runtime_disable(&client->dev); >>>>>> + pm_runtime_set_suspended(&client->dev); >>>>>> + media_entity_cleanup(&sensor->sd.entity); >>>>>> +error_handler_free: >>>>>> + v4l2_ctrl_handler_free(sensor->sd.ctrl_handler); >>>>>> + mutex_destroy(&sensor->lock); >>>>>> +error_power_off: >>>>>> + vgxy61_power_off(dev); >>>>>> + >>>>>> + return ret; >>>>>> +} >>> >