Received: by 2002:a25:1506:0:0:0:0:0 with SMTP id 6csp1005224ybv; Fri, 7 Feb 2020 12:29:54 -0800 (PST) X-Google-Smtp-Source: APXvYqxYo0hbpmcZKwnMjcUixWsbZCBRRAKWdC6Tiom1A0uq/zs77GhNTkRIXRk5MTZXdvTXclhv X-Received: by 2002:a9d:12a8:: with SMTP id g37mr921313otg.261.1581107394376; Fri, 07 Feb 2020 12:29:54 -0800 (PST) ARC-Seal: i=1; a=rsa-sha256; t=1581107394; cv=none; d=google.com; s=arc-20160816; b=DhKugIPFYD1A1yEr2BSMPY3ZxAfGhk/A4k5mqVeemJE8bCMIKoRLIEUcZ8ldU6nx3e CY9N5BBhE374W/jgwgLF1d3UHs6uGGTgBSQcdp8TpCQa22C2JfUrK27+3aPiurlrRTaL IKSqsUBTBJOxQLFDIIr1WbzJKdSdzOYlt2WQyc26yWH01K3Q3hjQCp/FRPM+eBJa9ywv JwCKkYSlSm42eWB51VHXz4GdXBF16LmorBpuCfshIo29EBDKYv4dCKZ/Svc+GdwgnlM0 BxfDtmkuVWba7bzx90cuDEt5qiDAMY7H9PXJYmHHEr46vU9dbsuC1CcbAnuTf66n54n3 n4BA== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:sender:cc:to:subject:message-id:date:from :in-reply-to:references:mime-version:dkim-signature; bh=fhAGifdB6DtAEywQcYytW5oBhmqL7rZZTE3i/2Vmdhc=; b=I1dDAwCJXVC2Uk55az85OhW0Ozu+r1+Ua2HLmsbZ3lDeMlzYocy1h6b4qJKwUqt18j D4cG8+TTXYZGOOPB2iOLlGG7Hadr7RQ/SzzXvrVJcPQ6ls6G4T+qhwr/4m4xT64rmVm2 QiwiBAzWufwUcwbNOPVTQyfTou6WEHla3LVeb0aAR8+tPcp9O59tLrJjjfFgHoTEqLjT wZXi1RWRybM4hagVEndsxBSH/7j0t5/SIv+HpBpOK8SKsurAK80WT6m8pBRtkjFVifu3 5tOQpsLvz8LgdMs30kbeSXXMIkpQmYFvoiPuriJS69CVE4FplKmt+C7nbzMTtEV74Ahz dnMg== ARC-Authentication-Results: i=1; mx.google.com; dkim=pass header.i=@chromium.org header.s=google header.b=h8g+KlPI; spf=pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=pass (p=NONE sp=NONE dis=NONE) header.from=chromium.org Return-Path: Received: from vger.kernel.org (vger.kernel.org. [209.132.180.67]) by mx.google.com with ESMTP id i9si216503otp.139.2020.02.07.12.29.39; Fri, 07 Feb 2020 12:29:54 -0800 (PST) Received-SPF: pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) client-ip=209.132.180.67; Authentication-Results: mx.google.com; dkim=pass header.i=@chromium.org header.s=google header.b=h8g+KlPI; spf=pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=pass (p=NONE sp=NONE dis=NONE) header.from=chromium.org Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1727234AbgBGU2N (ORCPT + 99 others); Fri, 7 Feb 2020 15:28:13 -0500 Received: from mail-qk1-f193.google.com ([209.85.222.193]:44263 "EHLO mail-qk1-f193.google.com" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1727031AbgBGU2N (ORCPT ); Fri, 7 Feb 2020 15:28:13 -0500 Received: by mail-qk1-f193.google.com with SMTP id v195so358789qkb.11 for ; Fri, 07 Feb 2020 12:28:12 -0800 (PST) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=chromium.org; s=google; h=mime-version:references:in-reply-to:from:date:message-id:subject:to :cc; bh=fhAGifdB6DtAEywQcYytW5oBhmqL7rZZTE3i/2Vmdhc=; b=h8g+KlPIzXBX35U7BH7Wij8+3DXA+rd5R3vQ62pyqvYQAORguSmAbWVcYSBKPWiGy1 /a7jTGB63doNk6vU2NvSfEHWTAyNkw/NIF4fqZy1WGL48C2yMxrPqqxYUe2I8zpmrieD ZdRXwE7EVRUKAdfG9gmcCClzJ5prvew2GZqRs= X-Google-DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=1e100.net; s=20161025; h=x-gm-message-state:mime-version:references:in-reply-to:from:date :message-id:subject:to:cc; bh=fhAGifdB6DtAEywQcYytW5oBhmqL7rZZTE3i/2Vmdhc=; b=h+WO2hZkbQ7UNlbJI7qpJKrcfz+7wb4i6INdjD4mjUjhH0i+lIJbqbvN2p8d7XMgft s1vd1rxI9nc+EQJvBwjSwHDtJkpPVBOYC1a4xOCpmaKv8I/DjU/ymkzfAR8YMr9ys3Su xbSmgW7riSYpvhrifpg0U2aUMCLXPBba3kqcbJ68DwmL7OKC7NdIWpEAMdb/AFZsDhVO xqaiPns/vIJQYyaRuRtYITLvwWHpyEB+rm2oVX+XCGt6+XzfrDww9++9TJCblz5TBcxA 7eC193Rn0iSwWuOneO/28aJ+b5TG49uWPw6E1M8rnJJ5J9GMF2I8BGUMlkZPeJM4EpKq WZbQ== X-Gm-Message-State: APjAAAWgUBWJwf9aqWN+1bUPyQBf6rEpeU4s4TqQnday5rN5q4Ldeqgr oIzmATWdpiOKOwcjUIu0Iphgx6MpjJRdOfceSi/SggFa X-Received: by 2002:a37:9d8c:: with SMTP id g134mr692823qke.128.1581107291689; Fri, 07 Feb 2020 12:28:11 -0800 (PST) MIME-Version: 1.0 References: <20200205205954.84503-1-pmalani@chromium.org> <20200205205954.84503-2-pmalani@chromium.org> <544d31cc-e840-c91e-f65d-7f7b57ba1337@collabora.corp-partner.google.com> <9f3a8777-9bd4-5044-7d3b-5ccf25a9b97e@collabora.com> In-Reply-To: <9f3a8777-9bd4-5044-7d3b-5ccf25a9b97e@collabora.com> From: Prashant Malani Date: Fri, 7 Feb 2020 12:28:00 -0800 Message-ID: Subject: Re: [PATCH 1/3] platform/chrome: Add Type C connector class driver To: Enric Balletbo i Serra Cc: Linux Kernel Mailing List , heikki.krogerus@intel.com, Benson Leung , Guenter Roeck Content-Type: text/plain; charset="UTF-8" Sender: linux-kernel-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org On Fri, Feb 7, 2020 at 9:00 AM Enric Balletbo i Serra wrote: > > Hi Prashant, > > On 7/2/20 0:30, Prashant Malani wrote: > > Hi Enric, > > > > Thanks for taking a look at the patch. Please see my responses inline > > (I will defer sending the next version till the one pending question I > > had is resolved): > > > > On Thu, Feb 6, 2020 at 8:19 AM Enric Balletbo i Serra > > wrote: > >> > >> Hi Prashant, > >> > >> On 5/2/20 21:59, Prashant Malani wrote: > >>> Add a driver to implement the Type C connector class for Chrome OS > >>> devices with ECs (Embedded Controllers). > >>> > >>> The driver relies on firmware device specifications for various port > >>> attributes. On ACPI platforms, this is specified using the logical > >>> device with HID GOOG0014. On DT platforms, this is specified using the > >>> DT node with compatible string "google,cros-ec-typec". > >>> > >> > >> If that's the case you should document this in a binding file. > > > > Done. I will add this in the next version of the series. > > > >> driver a replacement of the cros-ec-extcon driver or is a different thing? > > > > Currently it is distinct. We're hoping to plug in to type C connector > > class work which Heikki is working on (relating to Type C Mux agent). > > Hopefully, we will gradually transition the extcon functionality to > > the Type C port driver. > > > >> > >> There is a device where I can test this? > > > > I think you can try it on kevin if you add the DT node for it (haven't > > tried it myself on kevin). An example will be present in the DT > > Documentation I provide in the next version. > > > >> > >>> This patch reads the device FW node and uses the port attributes to > >>> register the typec ports with the Type C connector class framework, but > >>> doesn't do much else. > >>> > >>> Subsequent patches will add more functionality to the driver, including > >>> obtaining current port information (polarity, vconn role, current power > >>> role etc.) after querying the EC. > >>> > >>> Signed-off-by: Prashant Malani > >>> --- > >>> drivers/platform/chrome/Kconfig | 11 ++ > >>> drivers/platform/chrome/Makefile | 1 + > >>> drivers/platform/chrome/cros_ec_typec.c | 228 ++++++++++++++++++++++++ > >>> 3 files changed, 240 insertions(+) > >>> create mode 100644 drivers/platform/chrome/cros_ec_typec.c > >>> > >>> diff --git a/drivers/platform/chrome/Kconfig b/drivers/platform/chrome/Kconfig > >>> index 5f57282a28da00..1370dfd1ca1565 100644 > >>> --- a/drivers/platform/chrome/Kconfig > >>> +++ b/drivers/platform/chrome/Kconfig > >>> @@ -214,6 +214,17 @@ config CROS_EC_SYSFS > >>> To compile this driver as a module, choose M here: the > >>> module will be called cros_ec_sysfs. > >>> > >>> +config CROS_EC_TYPEC > >>> + tristate "ChromeOS EC Type-C Connector Control" > >>> + depends on MFD_CROS_EC_DEV && TYPEC > >>> + default n > >> > >> Default value is already n, so you don't need to put it here. But I'd say that > >> we might be interested on have default MFD_CROS_EC_DEV like the other drivers. > > > > Done. > > > >> > >>> + help > >>> + If you say Y here, you get support for accessing Type C connector > >>> + information from the Chrome OS EC. > >>> + > >>> + To compile this driver as a module, choose M here: the module will be > >>> + called cros_ec_typec. > >>> + > >>> config CROS_USBPD_LOGGER > >>> tristate "Logging driver for USB PD charger" > >>> depends on CHARGER_CROS_USBPD > >>> diff --git a/drivers/platform/chrome/Makefile b/drivers/platform/chrome/Makefile > >>> index aacd5920d8a180..caf2a9cdb5e6d1 100644 > >>> --- a/drivers/platform/chrome/Makefile > >>> +++ b/drivers/platform/chrome/Makefile > >>> @@ -12,6 +12,7 @@ obj-$(CONFIG_CROS_EC_ISHTP) += cros_ec_ishtp.o > >>> obj-$(CONFIG_CROS_EC_RPMSG) += cros_ec_rpmsg.o > >>> obj-$(CONFIG_CROS_EC_SPI) += cros_ec_spi.o > >>> cros_ec_lpcs-objs := cros_ec_lpc.o cros_ec_lpc_mec.o > >>> +obj-$(CONFIG_CROS_EC_TYPEC) += cros_ec_typec.o > >>> obj-$(CONFIG_CROS_EC_LPC) += cros_ec_lpcs.o > >>> obj-$(CONFIG_CROS_EC_PROTO) += cros_ec_proto.o cros_ec_trace.o > >>> obj-$(CONFIG_CROS_KBD_LED_BACKLIGHT) += cros_kbd_led_backlight.o > >>> diff --git a/drivers/platform/chrome/cros_ec_typec.c b/drivers/platform/chrome/cros_ec_typec.c > >>> new file mode 100644 > >>> index 00000000000000..fe5659171c2a85 > >>> --- /dev/null > >>> +++ b/drivers/platform/chrome/cros_ec_typec.c > >>> @@ -0,0 +1,228 @@ > >>> +// SPDX-License-Identifier: GPL-2.0-only > >>> +/* > >>> + * Copyright 2020 Google LLC > >>> + * > >>> + * This driver provides the ability to view and manage Type C ports through the > >>> + * Chrome OS EC. > >>> + */ > >>> + > >>> +#include > >>> +#include > >>> +#include > >>> +#include > >>> +#include > >>> +#include > >>> +#include > >>> + > >>> +#define DRV_NAME "cros-ec-typec" > >>> + > >>> +/* Platform-specific data for the Chrome OS EC Type C controller. */ > >>> +struct cros_typec_data { > >>> + struct device *dev; > >>> + struct cros_ec_device *ec; > >>> + int num_ports; > >>> + /* Array of ports, indexed by port number. */ > >>> + struct typec_port *ports[EC_USB_PD_MAX_PORTS]; > >>> +}; > >>> + > >>> +int cros_typec_parse_port_props(struct typec_capability *cap, > >> > >> static int > > > > Sorry, done. > > > >> > >>> + struct fwnode_handle *fwnode, > >>> + struct device *dev) > >>> +{ > >>> + const char *buf; > >>> + int ret; > >>> + > >>> + memset(cap, 0, sizeof(*cap)); > >>> + ret = fwnode_property_read_string(fwnode, "power-role", &buf); > >>> + if (ret) { > >>> + dev_err(dev, "power-role not found: %d\n", ret); > >>> + return ret; > >>> + } > >>> + > >>> + ret = typec_find_port_power_role(buf); > >>> + if (ret < 0) > >>> + return ret; > >>> + cap->type = ret; > >>> + > >>> + ret = fwnode_property_read_string(fwnode, "data-role", &buf); > >>> + if (ret) { > >>> + dev_err(dev, "data-role not found: %d\n", ret); > >>> + return ret; > >>> + } > >>> + > >>> + ret = typec_find_port_data_role(buf); > >>> + if (ret < 0) > >>> + return ret; > >>> + cap->data = ret; > >>> + > >>> + ret = fwnode_property_read_string(fwnode, "try-power-role", &buf); > >>> + if (ret) { > >>> + dev_err(dev, "try-power-role not found: %d\n", ret); > >>> + return ret; > >>> + } > >>> + > >>> + ret = typec_find_power_role(buf); > >>> + if (ret < 0) > >>> + return ret; > >>> + cap->prefer_role = ret; > >>> + > >>> + cap->fwnode = fwnode; > >>> + > >>> + return 0; > >>> +} > >>> + > >>> +static int cros_typec_init_ports(struct cros_typec_data *typec) > >>> +{ > >>> + struct device *dev = typec->dev; > >>> + struct typec_capability cap; > >>> + struct fwnode_handle *fwnode; > >>> + int ret; > >>> + int i; > >>> + int nports; > >>> + u32 port_num; > >>> + > >>> + nports = device_get_child_node_count(dev); > >>> + if (nports == 0) { > >>> + dev_err(dev, "No port entries found.\n"); > >>> + return -ENODEV; > >>> + } > >>> + > >>> + device_for_each_child_node(dev, fwnode) { > >>> + if (fwnode_property_read_u32(fwnode, "port-number", > >>> + &port_num)) { > >>> + dev_err(dev, "No port-number for port, skipping.\n"); > >>> + ret = -EINVAL; > >>> + goto unregister_ports; > >>> + } > >>> + > >>> + if (port_num >= typec->num_ports) { > >>> + dev_err(dev, "Invalid port number.\n"); > >>> + ret = -EINVAL; > >>> + goto unregister_ports; > >>> + } > >>> + > >>> + dev_dbg(dev, "Registering port %d\n", port_num); > >>> + ret = cros_typec_parse_port_props(&cap, fwnode, dev); > >>> + if (ret < 0) > >>> + goto unregister_ports; > >>> + typec->ports[port_num] = typec_register_port(dev, &cap); > >>> + if (IS_ERR(typec->ports[port_num])) { > >>> + dev_err(dev, "Failed to register port %d\n", port_num); > >>> + ret = PTR_ERR(typec->ports[port_num]); > >>> + goto unregister_ports; > >>> + } > >>> + } > >>> + > >>> + return 0; > >>> + > >>> +unregister_ports: > >>> + for (i = 0; i < typec->num_ports; i++) > >>> + typec_unregister_port(typec->ports[i]); > >>> + return ret; > >>> +} > >>> + > >>> +static int cros_typec_ec_command(struct cros_typec_data *typec, > >>> + unsigned int version, > >>> + unsigned int command, > >>> + void *outdata, > >>> + unsigned int outsize, > >>> + void *indata, > >>> + unsigned int insize) > >>> +{ > >>> + struct cros_ec_command *msg; > >>> + int ret; > >>> + > >>> + msg = kzalloc(sizeof(*msg) + max(outsize, insize), GFP_KERNEL); > >>> + if (!msg) > >>> + return -ENOMEM; > >>> + > >>> + msg->version = version; > >>> + msg->command = command; > >>> + msg->outsize = outsize; > >>> + msg->insize = insize; > >>> + > >>> + if (outsize) > >>> + memcpy(msg->data, outdata, outsize); > >>> + > >>> + ret = cros_ec_cmd_xfer_status(typec->ec, msg); > >>> + if (ret >= 0 && insize) > >>> + memcpy(indata, msg->data, insize); > >>> + > >>> + kfree(msg); > >>> + return ret; > >>> +} > >>> + > >>> + > >>> +static int cros_typec_get_num_ports(struct cros_typec_data *typec) > >>> +{ > >> > >> This functions is only called once at probe, you can just put the code there. I > >> had some readibility problems trying to follow de code. > > > > Done. > > > >> > >>> + struct ec_response_usb_pd_ports resp; > >>> + int ret; > >>> + > >>> + ret = cros_typec_ec_command(typec, 0, EC_CMD_USB_PD_PORTS, NULL, 0, > >>> + &resp, sizeof(resp)); > >>> + if (ret < 0) > >>> + return ret; > >>> + > >>> + typec->num_ports = resp.num_ports; > >>> + if (typec->num_ports > EC_USB_PD_MAX_PORTS) { > >>> + dev_warn(typec->dev, > >>> + "Too many ports reported: %d, limiting to max: %d\n", > >>> + typec->num_ports, EC_USB_PD_MAX_PORTS); > >> > >> You say that you are limiting the number of ports to max but typec->num_ports is > >> still resp.num_ports, is that correct? > >> > > > > Sorry, thanks for catching this, it should be set to > > EC_USB_PD_MAX_PORTS. I will fix this in the next version. > > > >>> + } > >>> + > >>> + return 0; > >>> +} > >>> + > >>> +#ifdef CONFIG_ACPI > >>> +static const struct acpi_device_id cros_typec_acpi_id[] = { > >>> + { "GOOG0014", 0 }, > >>> + { /* sentinel */ } > >> > >> No need to add /* sentinel */ here, is obvious. > > > > Done. > > > >> > >>> +}; > >>> +MODULE_DEVICE_TABLE(acpi, cros_typec_acpi_id); > >>> +#endif > >>> + > >>> +#ifdef CONFIG_OF > >>> +static const struct of_device_id cros_typec_of_match[] = { > >>> + { .compatible = "google,cros-ec-typec", }, > >>> + { /* sentinel */ }, > >> > >> No need to add /* sentinel */ here, is obvious. And no need for the colon at the > >> end. > > > > Done. > >> > >>> +}; > >>> +MODULE_DEVICE_TABLE(of, cros_typec_of_match); > >>> +#endif > >>> + > >>> +static int cros_typec_probe(struct platform_device *pdev) > >>> +{ > >>> + struct device *dev = &pdev->dev; > >>> + struct cros_typec_data *typec; > >>> + int ret; > >>> + > >>> + typec = devm_kzalloc(dev, sizeof(*typec), GFP_KERNEL); > >>> + if (!typec) > >>> + return -ENOMEM; > >>> + typec->dev = dev; > >>> + typec->ec = dev_get_drvdata(pdev->dev.parent); > >>> + platform_set_drvdata(pdev, typec); > >>> + > >>> + ret = cros_typec_get_num_ports(typec); > >>> + if (ret < 0) > >>> + return ret; > >>> + > >>> + ret = cros_typec_init_ports(typec); > >> > >> both, cros_typec_get_num_ports and cros_typec_init_ports are only called once, > >> as I said I had some problems of readibility because typec->num_ports is set in > >> one function and unregister in another function when fails. > > > > Can we keep cros_typec_init_ports() as a separate function? I was > > hoping to avoid cluttering the _probe() with FW node parsing logic. > > cros_typec_init_ports() registers the ports (and unregisters on > > failure), so seems fairly self-contained. > > Of course, happy to place everything in probe if you still prefer that. > > > > I don't mind clutter probe if is more readable but try it, and lets see how > looks like. You are also registering and unregistering the typec ports there, > not only parsing. Usually what we have is parsing the firmware or dt and then > register and unregister the typec ports, all mixed. I'm more in favour of have > it separated, IMHO. Sounds good, I'll push another version with cros_typec_init_ports() distinct and we can check how it looks. If it's still not readable, I'll push version with the code directly in probe() > > >> > >> I'd just remove those two functions and put the code directly here. > >> > >>> + if (!ret) > >>> + return ret; > >>> + > >>> + return 0; > >>> +} > >>> + > >>> +static struct platform_driver cros_typec_driver = { > >>> + .driver = { > >>> + .name = DRV_NAME, > >> > >> no tab, just space after .-name > > > > Done. > >> > >>> + .acpi_match_table = ACPI_PTR(cros_typec_acpi_id), > >>> + .of_match_table = of_match_ptr(cros_typec_of_match), > >>> + }, > >>> + .probe = cros_typec_probe, > >> > >> no tab, just space after .probe > > > > Done. > > > >> > >>> +}; > >>> + > >>> +module_platform_driver(cros_typec_driver); > >>> + > >>> +MODULE_LICENSE("GPL"); > >>> +MODULE_DESCRIPTION("Chrome OS EC Type C control"); > >>> > >> > >> You probably want to add the MODULE_AUTHOR here > > > > Done. > > > > Best regards, > >