Hi Benjamin, 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 ? > >> + > >> +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; > >> +} -- Regards, Laurent Pinchart