Received: by 2002:a05:7412:31a9:b0:e2:908c:2ebd with SMTP id et41csp4887373rdb; Fri, 15 Sep 2023 15:56:33 -0700 (PDT) X-Google-Smtp-Source: AGHT+IHRZN5ltgEl4tq5rCz1jzlHoGPZi3/AGTgRqgMwyiyTvW2AEshssu2gqD3oUuuiAb33mAH6 X-Received: by 2002:a17:903:11c5:b0:1b0:f8:9b2d with SMTP id q5-20020a17090311c500b001b000f89b2dmr3540693plh.29.1694818593044; Fri, 15 Sep 2023 15:56:33 -0700 (PDT) ARC-Seal: i=1; a=rsa-sha256; t=1694818593; cv=none; d=google.com; s=arc-20160816; b=ZFMgjmDdWGh0PZuJSWSAh1hcbSDb0FYA3T+FgeB68cpmFb2MCz9VynKiFXmd8W7LCN 3/wmSzva5IbTsmziH+OjmCkPII3niws8bFdePnMbCZSoIOq/p46ry0WABAFT62LqOZiO v4QqFyTmu+Zcti+wxj2TfzLytPWQA7v3rRgUZHhV8Vzd9ppNGNU8W4A+td+71tARVNA/ 5nBfNXkcjj5fnYYQbG4OUsbx/UY4lp7PUJ6DxNSsPqtrgO/uoi9J+eiemn/PiXprfMZI DJUv0H+HVZTbft4Tf81KCj7UL/OWusSmGWIeDwooJk3VF3KNrP0Uhw12fA+52qjendD9 BfEQ== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:cc:to:subject:message-id:date:from:in-reply-to :references:mime-version:dkim-signature; bh=IrfDCHJCwJUzfWjjYVo0sWuFa6L//oZVEoGV1Odt0OU=; fh=5ah769M4UbaI8/VGyZdxvaSzh5Y0GVMoxzfvrcNbw/s=; b=ThQ98VvUGTABY9goTwbNHhbPF2ca23O9bSxsWqJ0qFsc3MbX8eQ+98pO2lI/aV3AKB GKc2xnYrqjsTRCNruPdthPuhQPWWTqyYoALq5UqM7EQt8pJ2BlnMYKZhFzYON8yt77cF TaNXhVBdmVBvsNM/5TE/59YToGuZqU9VxYrMVuHqKusOpxjpnnVpcD/65HKdkBrPRUG3 0iRCTyHuVeRUifG/2b2tpnvOrO8hMmriKYk1M8WKzqy3P8MkhoLsUsUtekRJ5EwbP5wp PWIzSzCcekwU0MOSGTqkqb2thKR9LGMQTz75PxoeYFVOWbXxUpcqWTHzhR0QNIvN2mL+ dqcg== ARC-Authentication-Results: i=1; mx.google.com; dkim=pass header.i=@raspberrypi.com header.s=google header.b=Y8GYJwrg; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 23.128.96.36 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=pass (p=QUARANTINE sp=NONE dis=NONE) header.from=raspberrypi.com Return-Path: Received: from pete.vger.email (pete.vger.email. [23.128.96.36]) by mx.google.com with ESMTPS id m18-20020a170902f65200b001c3411c9b83si4070910plg.454.2023.09.15.15.56.32 (version=TLS1_3 cipher=TLS_AES_256_GCM_SHA384 bits=256/256); Fri, 15 Sep 2023 15:56:32 -0700 (PDT) Received-SPF: pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 23.128.96.36 as permitted sender) client-ip=23.128.96.36; Authentication-Results: mx.google.com; dkim=pass header.i=@raspberrypi.com header.s=google header.b=Y8GYJwrg; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 23.128.96.36 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=pass (p=QUARANTINE sp=NONE dis=NONE) header.from=raspberrypi.com Received: from out1.vger.email (depot.vger.email [IPv6:2620:137:e000::3:0]) by pete.vger.email (Postfix) with ESMTP id 31E1E807CF47; Fri, 15 Sep 2023 10:45:18 -0700 (PDT) X-Virus-Status: Clean X-Virus-Scanned: clamav-milter 0.103.10 at pete.vger.email Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S235746AbjIORol (ORCPT + 99 others); Fri, 15 Sep 2023 13:44:41 -0400 Received: from lindbergh.monkeyblade.net ([23.128.96.19]:40146 "EHLO lindbergh.monkeyblade.net" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S235965AbjIORo1 (ORCPT ); Fri, 15 Sep 2023 13:44:27 -0400 Received: from mail-ua1-x92f.google.com (mail-ua1-x92f.google.com [IPv6:2607:f8b0:4864:20::92f]) by lindbergh.monkeyblade.net (Postfix) with ESMTPS id CD19110C9 for ; Fri, 15 Sep 2023 10:44:19 -0700 (PDT) Received: by mail-ua1-x92f.google.com with SMTP id a1e0cc1a2514c-79a2216a2d1so969916241.2 for ; Fri, 15 Sep 2023 10:44:19 -0700 (PDT) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=raspberrypi.com; s=google; t=1694799859; x=1695404659; darn=vger.kernel.org; h=cc:to:subject:message-id:date:from:in-reply-to:references :mime-version:from:to:cc:subject:date:message-id:reply-to; bh=IrfDCHJCwJUzfWjjYVo0sWuFa6L//oZVEoGV1Odt0OU=; b=Y8GYJwrgRued7C+d1jIRJQ+U/750X4sgJzcSo5elsxB677g7X4MTyN79TwFSH4cH4o dh/pwvsewvdCQzcqFhHajudpliddYXQqDsdqIDzDTtYVArQOA9tPB+d71uS0cmITWTEK +wvZxmlzBS+K2RXG/yccHjBOoaG8PQnH8TOoqzkCFEcHyKuSrIIDsq/NXm6CGUbxx2GJ NK1Yoqw3LAqtoU1uknDLoyaT4Diqn8M3y9myNmg+o3T3ocTXFh6gwFKYcBLt9uHV3n73 YxbfXKL8SJkivLU575jUXQ9bNjA8wZnhrCXCSCYiICdnI1rJwKBLOqH9XciBIKMpfs56 O1TA== X-Google-DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=1e100.net; s=20230601; t=1694799859; x=1695404659; h=cc:to:subject:message-id:date:from:in-reply-to:references :mime-version:x-gm-message-state:from:to:cc:subject:date:message-id :reply-to; bh=IrfDCHJCwJUzfWjjYVo0sWuFa6L//oZVEoGV1Odt0OU=; b=Fh0zuf3FM5LyIDEtbKMrZyQga/bio/j2BfC0HgZeohGVzqyfJiyjqjvXIdr/uEiwkX yD6XGTVjbp7bQvRt2yVybMtokf+571SjOCrppmRVToYOFGh7FkyYtIp7nE2j+ZZdV00r v+N+b4YRL+oD9JxCL+cyPuDSIvDbaELLFddH1jsVxMGzmT4M/TZTn8zjUE9RGRWw+0vX jEcbSg3LkIxAEZfBADFteQHZ98zvr3PN5DE/SALZTvhxY3oyVqnKaoTA6zD716F6akba /dG2+gvZSMaOhm63ghLicWR+lgLpGZR67yoiZ24bAcSLSvZkRxKoCzdviNgxuB6S3Q5W TMzg== X-Gm-Message-State: AOJu0YxnIHhI4iN7uxkx4+mBydXZbPlpPyI9uxcOSKVxXVUWMmoHIIx9 ss+56YfQYN09lMWbtmMus2HeTkXhXU3fZ4KWD1M5/g== X-Received: by 2002:a05:6102:152:b0:451:40b:2210 with SMTP id a18-20020a056102015200b00451040b2210mr2408170vsr.3.1694799858876; Fri, 15 Sep 2023 10:44:18 -0700 (PDT) MIME-Version: 1.0 References: <20230915165946.4183839-1-kieran.bingham@ideasonboard.com> <20230915165946.4183839-3-kieran.bingham@ideasonboard.com> In-Reply-To: <20230915165946.4183839-3-kieran.bingham@ideasonboard.com> From: Dave Stevenson Date: Fri, 15 Sep 2023 18:44:02 +0100 Message-ID: Subject: Re: [PATCH 2/2] media: i2c: Add ROHM BU64754 Camera Autofocus Actuator To: Kieran Bingham Cc: linux-media@vger.kernel.org, Mauro Carvalho Chehab , Liam Girdwood , Mark Brown , Sakari Ailus , Laurent Pinchart , Hans de Goede , Andy Shevchenko , Tomi Valkeinen , Marco Felsch , Gerald Loacker , Bingbu Cao , Daniel Scally , Linus Walleij , open list Content-Type: text/plain; charset="UTF-8" X-Spam-Status: No, score=-0.9 required=5.0 tests=DKIM_SIGNED,DKIM_VALID, DKIM_VALID_AU,HEADER_FROM_DIFFERENT_DOMAINS,MAILING_LIST_MULTI, SPF_HELO_NONE,SPF_PASS autolearn=unavailable autolearn_force=no version=3.4.6 X-Spam-Checker-Version: SpamAssassin 3.4.6 (2021-04-09) on pete.vger.email Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org X-Greylist: Sender passed SPF test, not delayed by milter-greylist-4.6.4 (pete.vger.email [0.0.0.0]); Fri, 15 Sep 2023 10:45:18 -0700 (PDT) Hi Kieran On Fri, 15 Sept 2023 at 18:02, Kieran Bingham wrote: > > Add support for the ROHM BU64754 Motor Driver for Camera Autofocus. A > V4L2 Subdevice is registered and provides a single > V4L2_CID_FOCUS_ABSOLUTE control. > > Signed-off-by: Kieran Bingham > --- > MAINTAINERS | 1 + > drivers/media/i2c/Kconfig | 13 ++ > drivers/media/i2c/Makefile | 1 + > drivers/media/i2c/bu64754.c | 308 ++++++++++++++++++++++++++++++++++++ > 4 files changed, 323 insertions(+) > create mode 100644 drivers/media/i2c/bu64754.c > > diff --git a/MAINTAINERS b/MAINTAINERS > index f43e0ffcaf56..fd244560c317 100644 > --- a/MAINTAINERS > +++ b/MAINTAINERS > @@ -18576,6 +18576,7 @@ L: linux-media@vger.kernel.org > S: Maintained > T: git git://linuxtv.org/media_tree.git > F: Documentation/devicetree/bindings/media/i2c/rohm,bu64754.yaml > +F: drivers/media/i2c/bu64754.c > > ROHM MULTIFUNCTION BD9571MWV-M PMIC DEVICE DRIVERS > M: Marek Vasut > diff --git a/drivers/media/i2c/Kconfig b/drivers/media/i2c/Kconfig > index 74ff833ff48c..b7b8004816ed 100644 > --- a/drivers/media/i2c/Kconfig > +++ b/drivers/media/i2c/Kconfig > @@ -641,6 +641,19 @@ config VIDEO_AK7375 > capability. This is designed for linear control of > voice coil motors, controlled via I2C serial interface. > > +config VIDEO_BU64754 > + tristate "BU64754 Motor Driver for Camera Autofocus" > + depends on I2C && VIDEO_DEV > + select MEDIA_CONTROLLER > + select VIDEO_V4L2_SUBDEV_API > + select V4L2_ASYNC > + select V4L2_CCI_I2C > + help > + This is a driver for the BU64754 Motor Driver for Camera > + Autofocus. The BU64754 is an actuator driver IC which can > + control the actuator position precisely using an internal > + Hall Sensor. I can't find any data on this driver. Is it still expecting a VCM and hence near instantaneous movement? I was noting your comment on the hall sensor and thinking you potentially needed to be able to report whether the target position had been reached or not. Michael's series at [1] was trying to address that. [1] https://patchwork.linuxtv.org/project/linux-media/cover/20230406-feature-controls-lens-v2-0-faa8ad2bc404@wolfvision.net/ > + > config VIDEO_DW9714 > tristate "DW9714 lens voice coil support" > depends on I2C && VIDEO_DEV > diff --git a/drivers/media/i2c/Makefile b/drivers/media/i2c/Makefile > index 80b00d39b48f..e62aa0df7b1a 100644 > --- a/drivers/media/i2c/Makefile > +++ b/drivers/media/i2c/Makefile > @@ -22,6 +22,7 @@ obj-$(CONFIG_VIDEO_AR0521) += ar0521.o > obj-$(CONFIG_VIDEO_BT819) += bt819.o > obj-$(CONFIG_VIDEO_BT856) += bt856.o > obj-$(CONFIG_VIDEO_BT866) += bt866.o > +obj-$(CONFIG_VIDEO_BU64754) += bu64754.o > obj-$(CONFIG_VIDEO_CCS) += ccs/ > obj-$(CONFIG_VIDEO_CCS_PLL) += ccs-pll.o > obj-$(CONFIG_VIDEO_CS3308) += cs3308.o > diff --git a/drivers/media/i2c/bu64754.c b/drivers/media/i2c/bu64754.c > new file mode 100644 > index 000000000000..3367b6f17660 > --- /dev/null > +++ b/drivers/media/i2c/bu64754.c > @@ -0,0 +1,308 @@ > +// SPDX-License-Identifier: GPL-2.0 > +/* > + * The BU64754 is an actuator driver IC which can control the > + * actuator position precisely using an internal Hall Sensor. > + */ > + > +#include > +#include > +#include > +#include > +#include > + > +#include > +#include > +#include > + > +#define BU64754_REG_ACTIVE CCI_REG16(0x07) > +#define BU64754_ACTIVE_MODE 0x8080 > + > +#define BU64754_REG_SERVE CCI_REG16(0xd9) > +#define BU64754_SERVE_ON 0x0404 > + > +#define BU64754_REG_POSITION CCI_REG16(0x45) > +#define BU64753_POSITION_MAX 1023 /* 0x3ff */ > + > +#define BU64754_POWER_ON_DELAY 800 /* uS : t1, t3 */ > + > +struct bu64754 { > + struct device *dev; > + > + struct v4l2_ctrl_handler ctrls_vcm; > + struct v4l2_subdev sd; > + struct regmap *cci; > + > + u16 current_val; > + struct regulator *vdd; > + struct notifier_block notifier; > +}; > + > +static inline struct bu64754 *sd_to_bu64754(struct v4l2_subdev *subdev) > +{ > + return container_of(subdev, struct bu64754, sd); > +} > + > +static int bu64754_set(struct bu64754 *bu64754, u16 position) > +{ > + int ret; > + > + ret = cci_write(bu64754->cci, BU64754_REG_POSITION, position, NULL); > + if (ret) { > + dev_err(bu64754->dev, "Set position failed ret=%d\n", ret); > + return ret; > + } > + > + return 0; > +} > + > +static int bu64754_active(struct bu64754 *bu64754) > +{ > + int ret; > + > + /* Power on */ > + ret = cci_write(bu64754->cci, BU64754_REG_ACTIVE, BU64754_ACTIVE_MODE, > + NULL); > + if (ret < 0) { > + dev_err(bu64754->dev, "Failed to set active mode ret = %d\n", > + ret); > + return ret; > + } > + > + /* Serve on */ > + ret = cci_write(bu64754->cci, BU64754_REG_SERVE, BU64754_SERVE_ON, > + NULL); > + if (ret < 0) { > + dev_err(bu64754->dev, "Failed to enable serve ret = %d\n", > + ret); > + return ret; > + } > + > + return bu64754_set(bu64754, bu64754->current_val); > +} > + > +static int bu64754_standby(struct bu64754 *bu64754) > +{ > + int ret; > + > + cci_write(bu64754->cci, BU64754_REG_ACTIVE, 0, &ret); > + if (ret < 0) > + dev_err(bu64754->dev, "Failed to set active mode ret = %d\n", > + ret); > + > + return ret; > +} > + > +static int bu64754_regulator_event(struct notifier_block *nb, > + unsigned long action, void *data) > +{ > + struct bu64754 *bu64754 = container_of(nb, struct bu64754, notifier); > + > + if (action & REGULATOR_EVENT_ENABLE) { > + /* > + * Initialisation delay between VDD low->high and availability > + * i2c operation. > + */ > + usleep_range(BU64754_POWER_ON_DELAY, > + BU64754_POWER_ON_DELAY + 100); > + > + bu64754_active(bu64754); > + } else if (action & REGULATOR_EVENT_PRE_DISABLE) { > + bu64754_standby(bu64754); > + } Presumably this is based on the assumption that the same regulator controls sensor and lens, so when the sensor is powered up the lens position gets restored. I'm sure when I suggested doing the same previously it was shot down in flames ... found it [2] Personally I think it makes sense that the lens powers up automagically, and have almost exactly the same code as this in a couple of our VCM drivers, but others disagree. Dave [2] https://lore.kernel.org/all/CAPY8ntBZpZjecHNCMf-eMefcp2EgmbqkXMt4p=UeOe0n-o8WrA@mail.gmail.com/ > + > + return 0; > +} > + > +static int bu64754_set_ctrl(struct v4l2_ctrl *ctrl) > +{ > + struct bu64754 *bu64754 = container_of(ctrl->handler, > + struct bu64754, ctrls_vcm); > + > + if (ctrl->id == V4L2_CID_FOCUS_ABSOLUTE) { > + bu64754->current_val = ctrl->val; > + return bu64754_set(bu64754, ctrl->val); > + } > + > + return -EINVAL; > +} > + > +static const struct v4l2_ctrl_ops bu64754_vcm_ctrl_ops = { > + .s_ctrl = bu64754_set_ctrl, > +}; > + > +static int bu64754_open(struct v4l2_subdev *sd, struct v4l2_subdev_fh *fh) > +{ > + return pm_runtime_resume_and_get(sd->dev); > +} > + > +static int bu64754_close(struct v4l2_subdev *sd, struct v4l2_subdev_fh *fh) > +{ > + pm_runtime_put(sd->dev); > + return 0; > +} > + > +static const struct v4l2_subdev_internal_ops bu64754_int_ops = { > + .open = bu64754_open, > + .close = bu64754_close, > +}; > + > +static const struct v4l2_subdev_ops bu64754_ops = { }; > + > +static void bu64754_subdev_cleanup(struct bu64754 *bu64754) > +{ > + v4l2_async_unregister_subdev(&bu64754->sd); > + v4l2_ctrl_handler_free(&bu64754->ctrls_vcm); > + media_entity_cleanup(&bu64754->sd.entity); > +} > + > +static int bu64754_init_controls(struct bu64754 *bu64754) > +{ > + struct v4l2_ctrl_handler *hdl = &bu64754->ctrls_vcm; > + const struct v4l2_ctrl_ops *ops = &bu64754_vcm_ctrl_ops; > + > + v4l2_ctrl_handler_init(hdl, 1); > + > + v4l2_ctrl_new_std(hdl, ops, V4L2_CID_FOCUS_ABSOLUTE, > + 0, BU64753_POSITION_MAX, 1, 0); > + > + bu64754->current_val = 0; > + > + bu64754->sd.ctrl_handler = hdl; > + if (hdl->error) { > + dev_err(bu64754->dev, "%s fail error: 0x%x\n", > + __func__, hdl->error); > + return hdl->error; > + } > + > + return 0; > +} > + > +static int bu64754_probe(struct i2c_client *client) > +{ > + struct bu64754 *bu64754; > + int ret; > + > + bu64754 = devm_kzalloc(&client->dev, sizeof(*bu64754), GFP_KERNEL); > + if (!bu64754) > + return -ENOMEM; > + > + bu64754->dev = &client->dev; > + > + bu64754->cci = devm_cci_regmap_init_i2c(client, 8); > + if (IS_ERR(bu64754->cci)) { > + dev_err(bu64754->dev, "Failed to initialize CCI\n"); > + return PTR_ERR(bu64754->cci); > + } > + > + bu64754->vdd = devm_regulator_get_optional(&client->dev, "vdd"); > + if (IS_ERR(bu64754->vdd)) { > + if (PTR_ERR(bu64754->vdd) != -ENODEV) > + return PTR_ERR(bu64754->vdd); > + > + bu64754->vdd = NULL; > + } else { > + bu64754->notifier.notifier_call = bu64754_regulator_event; > + > + ret = regulator_register_notifier(bu64754->vdd, > + &bu64754->notifier); > + if (ret) { > + dev_err(bu64754->dev, > + "could not register regulator notifier\n"); > + return ret; > + } > + } > + > + v4l2_i2c_subdev_init(&bu64754->sd, client, &bu64754_ops); > + bu64754->sd.flags |= V4L2_SUBDEV_FL_HAS_DEVNODE; > + bu64754->sd.internal_ops = &bu64754_int_ops; > + bu64754->sd.entity.function = MEDIA_ENT_F_LENS; > + > + ret = bu64754_init_controls(bu64754); > + if (ret) > + goto err_cleanup; > + > + ret = media_entity_pads_init(&bu64754->sd.entity, 0, NULL); > + if (ret < 0) > + goto err_cleanup; > + > + ret = v4l2_async_register_subdev(&bu64754->sd); > + if (ret < 0) > + goto err_cleanup; > + > + if (!bu64754->vdd) > + pm_runtime_set_active(&client->dev); > + > + pm_runtime_enable(&client->dev); > + pm_runtime_idle(&client->dev); > + > + return 0; > + > +err_cleanup: > + v4l2_ctrl_handler_free(&bu64754->ctrls_vcm); > + media_entity_cleanup(&bu64754->sd.entity); > + > + return ret; > +} > + > +static void bu64754_remove(struct i2c_client *client) > +{ > + struct v4l2_subdev *sd = i2c_get_clientdata(client); > + struct bu64754 *bu64754 = sd_to_bu64754(sd); > + > + if (bu64754->vdd) > + regulator_unregister_notifier(bu64754->vdd, > + &bu64754->notifier); > + > + pm_runtime_disable(&client->dev); > + > + bu64754_subdev_cleanup(bu64754); > +} > + > +static int __maybe_unused bu64754_vcm_suspend(struct device *dev) > +{ > + struct i2c_client *client = to_i2c_client(dev); > + struct v4l2_subdev *sd = i2c_get_clientdata(client); > + struct bu64754 *bu64754 = sd_to_bu64754(sd); > + > + if (bu64754->vdd) > + return regulator_disable(bu64754->vdd); > + > + return bu64754_standby(bu64754); > +} > + > +static int __maybe_unused bu64754_vcm_resume(struct device *dev) > +{ > + struct i2c_client *client = to_i2c_client(dev); > + struct v4l2_subdev *sd = i2c_get_clientdata(client); > + struct bu64754 *bu64754 = sd_to_bu64754(sd); > + > + if (bu64754->vdd) > + return regulator_enable(bu64754->vdd); > + > + return bu64754_active(bu64754); > +} > + > +static const struct of_device_id bu64754_of_table[] = { > + { .compatible = "rohm,bu64754", }, > + { /* sentinel */ } > +}; > + > +MODULE_DEVICE_TABLE(of, bu64754_of_table); > + > +static SIMPLE_DEV_PM_OPS(bu64754_pm, bu64754_vcm_suspend, bu64754_vcm_resume); > + > +static struct i2c_driver bu64754_i2c_driver = { > + .driver = { > + .name = "bu64754", > + .pm = &bu64754_pm, > + .of_match_table = bu64754_of_table, > + }, > + .probe = bu64754_probe, > + .remove = bu64754_remove, > +}; > + > +module_i2c_driver(bu64754_i2c_driver); > + > +MODULE_AUTHOR("Kieran Bingham "); > +MODULE_DESCRIPTION("ROHM BU64754 VCM driver"); > +MODULE_LICENSE("GPL"); > -- > 2.34.1 >