Received: by 2002:ac0:a5b6:0:0:0:0:0 with SMTP id m51-v6csp3100702imm; Tue, 29 May 2018 00:38:47 -0700 (PDT) X-Google-Smtp-Source: AB8JxZrPnEiX34jMrPMrP/LETlWiYP4yWxZK8sE++cfbReEfbuA2M34JAqPsRUGqeb3FPZANPkBG X-Received: by 2002:a65:4642:: with SMTP id k2-v6mr13159965pgr.305.1527579527610; Tue, 29 May 2018 00:38:47 -0700 (PDT) ARC-Seal: i=1; a=rsa-sha256; t=1527579527; cv=none; d=google.com; s=arc-20160816; b=OZHPaZhC9xo5fc9ggIjuhzWpoCxjMY3VR+IlomhaIcxbMXuaKiCJxN63DH2spl621P obsOkssh5QrjJcbLheBvHbECAkfR2LaKCLq4YEQxC/v3JxOweqSP/uI7zfohTf/f3lQM Cz1KOsI8k/HLw5/CmCNzwglPGUzNthGimWEieR2li6qNtu8VudB8YqgzU2R7fTjA5D+C ClBUc1gQYCJVhBSWS7xya3tiBgtdPkIqyzsmOzI18hf5HEi/YktHDu0ZOgZIYS+VxUI2 CzlotRJWtpuCdx/z0RIKjX9+j6BWMFpof/ZqxAotzQSbtrfLwc9WjuUi7Ihjw7wuuryB 6sJA== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:sender:references:in-reply-to:message-id:date :subject:cc:to:from:dkim-signature:arc-authentication-results; bh=o6VBY2In02ak1x+I4n40tXfYttjs3sZ0fJCuvnKzjIA=; b=G/cJoRfAv3ak6mJEpn5TaSI+7AHoEh3ovLTJEqwncn7BmhKg9sctKvnHCJV5A8GPu7 NyZjExzlYhTSzHAcpN1wPkFB1Dz/SJ1NLjatm7nC/j7pxFss4+hsHz7+DRjg1AEogRJU CWtI0QmePahjtLtgVf168qFWtF9iuHuPla0AzYiWS8fu7v2zYwWKlQNZBIxxcxfOVqTA Dl02t5peyFmqZRpSgHQ6Lg7GzKVUtrPHdFshZxdWQPP4NfOmqijgUj/tkOfzt/qP98Ak rokILagJx4/3kg/2Taw4tfDEq2ZVGfwB+RfRG26tzmkAScpqcSKPY1Enpkt+MLEBRhgB ungg== ARC-Authentication-Results: i=1; mx.google.com; dkim=pass header.i=@google.com header.s=20161025 header.b=LCqcPCkK; 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=REJECT sp=REJECT dis=NONE) header.from=google.com Return-Path: Received: from vger.kernel.org (vger.kernel.org. [209.132.180.67]) by mx.google.com with ESMTP id m21-v6si8007945pls.217.2018.05.29.00.38.33; Tue, 29 May 2018 00:38:47 -0700 (PDT) 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=@google.com header.s=20161025 header.b=LCqcPCkK; 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=REJECT sp=REJECT dis=NONE) header.from=google.com Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1754942AbeE2Hgu (ORCPT + 99 others); Tue, 29 May 2018 03:36:50 -0400 Received: from mail-pf0-f196.google.com ([209.85.192.196]:44697 "EHLO mail-pf0-f196.google.com" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1754759AbeE2Hgq (ORCPT ); Tue, 29 May 2018 03:36:46 -0400 Received: by mail-pf0-f196.google.com with SMTP id q22-v6so6864134pff.11 for ; Tue, 29 May 2018 00:36:46 -0700 (PDT) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=20161025; h=from:to:cc:subject:date:message-id:in-reply-to:references; bh=o6VBY2In02ak1x+I4n40tXfYttjs3sZ0fJCuvnKzjIA=; b=LCqcPCkKN+xv/1QZr/uUJnvGnbyumCTe0j7ML7v+Ypmj6RlIglOd5NBaxtIoXpMxRg eHI7LFEAt1fHKedadY73y5KJ657qa6lEIZLksLRv1qTZ4BgHejF/ZY4hw6FBgAExyZIE P/+bSL4qDufsKy6Jnh913ZIsiiUrvhmU2F1MncvHgVFPwXgMA3XTcvMG59/ZquMkZ8Ep NAaLK6XMCL3O/O6v2UlNI6xFyPoD4YGJlzCiM0soOHSDinUgtQekhP6BcWdZnUaGu6Mj QGg+cGLFzQ2R+pyu3OCFAWlWIJkYASNOQ5nDgmjrdszOgZhMq6ZIJYdTlYcGr4Pdksl1 9P3A== X-Google-DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=1e100.net; s=20161025; h=x-gm-message-state:from:to:cc:subject:date:message-id:in-reply-to :references; bh=o6VBY2In02ak1x+I4n40tXfYttjs3sZ0fJCuvnKzjIA=; b=Z6PPMNgG0h6kZd1s6VbH4qK8q9SQmDdpPc0PQbRzqO9ut2JjyL5mFpdgaanHg/xycE eFR1p1HkY/FXrxNpuLCdjBgfJVhkhntR5jzeYjO3T0xesNr55PWMavFVxYSW9SoK4Ybb l7C30rHvxuLl4qbkt3en4pV1LKTD6mLeQ8dlHe5gOSnOBHSrFvASRxCCnTzQ70uFuUwM ON8CHsOd5AOBxpHqiTe7CuxNIbCnHYWvdeDejCTj3RypBIKfCq2bgvCHlSFN6S5pqylK UuGy1rYe01Aj6NqgnQPV0ihNnBtT1wzUNlcTsetvwdMU6QT0vLcgEBC6SuudAOvnyIzi LF4A== X-Gm-Message-State: ALKqPwc+40Q31cOEAdaBPB7AJqHk6CnNLYv2Y0UTmEXtCtjyHhOLuscD QIh0fwqor3Udakxh7BPoHFBcZQ== X-Received: by 2002:a62:3e11:: with SMTP id l17-v6mr16417275pfa.18.1527579405337; Tue, 29 May 2018 00:36:45 -0700 (PDT) Received: from liumartin.ntc.corp.google.com ([2401:fa00:fc:202:ac57:d0e0:7094:5]) by smtp.gmail.com with ESMTPSA id 69-v6sm72128436pfh.33.2018.05.29.00.36.42 (version=TLS1_2 cipher=ECDHE-RSA-AES128-GCM-SHA256 bits=128/128); Tue, 29 May 2018 00:36:44 -0700 (PDT) From: martin_liu To: stern@rowland.harvard.edu Cc: heikki.krogerus@linux.intel.com, johan@kernel.org, gregkh@linuxfoundation.org, linux-kernel@vger.kernel.org, linux-usb@vger.kernel.org, jenhaochen@google.com, martin_liu Subject: [RFC PATCH v2] driver core: hold dev's parent lock when needed Date: Tue, 29 May 2018 15:07:20 +0800 Message-Id: <20180529070719.164626-1-liumartin@google.com> X-Mailer: git-send-email 2.17.0.921.gf22659ad46-goog In-Reply-To: References: Sender: linux-kernel-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org SOC have internal I/O buses that can't be proved for devices. The devices on the buses can be accessed directly without additinal configuration required. This type of bus is represented as "simple-bus". In some platforms, we name "soc" with "simple-bus" attribute and many devices are hooked under it desribed in DT (device tree). In commit 'bf74ad5bc417 introduce ("[PATCH] Hold the device's parent's lock during probe and remove")' to solve USB subsystem lock sequence since usb device's characteristic. Thus "soc" needs to be locked whenever a device and driver's probing happen under "soc" bus. During this period, an async driver tries to probe a device which is under the "soc" bus would be blocked until previous driver finish the probing and release "soc" lock. And the next probing under the "soc" bus need to wait for async finish. Because of that, driver's async probe for init time improvement will be shadowed. Since many devices don't have USB devices' characteristic, they actually don't need parent's lock. Thus, we introduce a lock flag in device struct and driver core would lock the parent lock base on the flag. For usbsystem, we set this flag when its device and driver is matched and to keep original lock behavior in driver core. Async probe could have more benefit after this patch. Signed-off-by: martin_liu Suggested-by: Alan Stern --- Changes in v2: - take Alan's suggestion to introudce a flag to guide driver core to hold device parent's lock. [v1]: https://lkml.org/lkml/2018/5/22/545 Currently, I have the flag set in USB subsystem match function. Since I'm not familar with USB part, need some feedback to know if they cover all the cases that original case driver core protects. drivers/base/bus.c | 16 ++++++++-------- drivers/base/dd.c | 8 ++++---- drivers/usb/common/ulpi.c | 16 ++++++++++++---- drivers/usb/core/driver.c | 9 +++++++-- drivers/usb/core/usb-acpi.c | 6 +++++- drivers/usb/serial/bus.c | 4 +++- include/linux/device.h | 1 + 7 files changed, 40 insertions(+), 20 deletions(-) diff --git a/drivers/base/bus.c b/drivers/base/bus.c index ef6183306b40..18ea94caec02 100644 --- a/drivers/base/bus.c +++ b/drivers/base/bus.c @@ -184,10 +184,10 @@ static ssize_t unbind_store(struct device_driver *drv, const char *buf, dev = bus_find_device_by_name(bus, NULL, buf); if (dev && dev->driver == drv) { - if (dev->parent) /* Needed for USB */ + if (dev->parent && dev->need_parent_lock)/* Needed for USB */ device_lock(dev->parent); device_release_driver(dev); - if (dev->parent) + if (dev->parent && dev->need_parent_lock) device_unlock(dev->parent); err = count; } @@ -211,12 +211,12 @@ static ssize_t bind_store(struct device_driver *drv, const char *buf, dev = bus_find_device_by_name(bus, NULL, buf); if (dev && dev->driver == NULL && driver_match_device(drv, dev)) { - if (dev->parent) /* Needed for USB */ + if (dev->parent && dev->need_parent_lock)/* Needed for USB */ device_lock(dev->parent); device_lock(dev); err = driver_probe_device(drv, dev); device_unlock(dev); - if (dev->parent) + if (dev->parent && dev->need_parent_lock) device_unlock(dev->parent); if (err > 0) { @@ -735,10 +735,10 @@ static int __must_check bus_rescan_devices_helper(struct device *dev, int ret = 0; if (!dev->driver) { - if (dev->parent) /* Needed for USB */ + if (dev->parent && dev->need_parent_lock)/* Needed for USB */ device_lock(dev->parent); ret = device_attach(dev); - if (dev->parent) + if (dev->parent && dev->need_parent_lock) device_unlock(dev->parent); } return ret < 0 ? ret : 0; @@ -770,10 +770,10 @@ EXPORT_SYMBOL_GPL(bus_rescan_devices); int device_reprobe(struct device *dev) { if (dev->driver) { - if (dev->parent) /* Needed for USB */ + if (dev->parent && dev->need_parent_lock)/* Needed for USB */ device_lock(dev->parent); device_release_driver(dev); - if (dev->parent) + if (dev->parent && dev->need_parent_lock) device_unlock(dev->parent); } return bus_rescan_devices_helper(dev, NULL); diff --git a/drivers/base/dd.c b/drivers/base/dd.c index c9f54089429b..c9a118568775 100644 --- a/drivers/base/dd.c +++ b/drivers/base/dd.c @@ -817,13 +817,13 @@ static int __driver_attach(struct device *dev, void *data) return ret; } /* ret > 0 means positive match */ - if (dev->parent) /* Needed for USB */ + if (dev->parent && dev->need_parent_lock)/* Needed for USB */ device_lock(dev->parent); device_lock(dev); if (!dev->driver) driver_probe_device(drv, dev); device_unlock(dev); - if (dev->parent) + if (dev->parent && dev->need_parent_lock) device_unlock(dev->parent); return 0; @@ -919,7 +919,7 @@ void device_release_driver_internal(struct device *dev, struct device_driver *drv, struct device *parent) { - if (parent) + if (parent && dev->need_parent_lock) device_lock(parent); device_lock(dev); @@ -927,7 +927,7 @@ void device_release_driver_internal(struct device *dev, __device_release_driver(dev, parent); device_unlock(dev); - if (parent) + if (parent && dev->need_parent_lock) device_unlock(parent); } diff --git a/drivers/usb/common/ulpi.c b/drivers/usb/common/ulpi.c index 9a2ab6751a23..609566396bf8 100644 --- a/drivers/usb/common/ulpi.c +++ b/drivers/usb/common/ulpi.c @@ -40,13 +40,21 @@ static int ulpi_match(struct device *dev, struct device_driver *driver) const struct ulpi_device_id *id; /* Some ULPI devices don't have a vendor id so rely on OF match */ - if (ulpi->id.vendor == 0) - return of_driver_match_device(dev, driver); + if (ulpi->id.vendor == 0) { + if (of_driver_match_device(dev, driver)) { + dev->need_parent_lock = 1; + return 1; + } + return o; + } - for (id = drv->id_table; id->vendor; id++) + for (id = drv->id_table; id->vendor; id++) { if (id->vendor == ulpi->id.vendor && - id->product == ulpi->id.product) + id->product == ulpi->id.product) { + dev->need_parent_lock = 1; return 1; + } + } return 0; } diff --git a/drivers/usb/core/driver.c b/drivers/usb/core/driver.c index 9792cedfc351..3dbabf8d3cb8 100644 --- a/drivers/usb/core/driver.c +++ b/drivers/usb/core/driver.c @@ -808,6 +808,7 @@ static int usb_device_match(struct device *dev, struct device_driver *drv) return 0; /* TODO: Add real matching code */ + dev->need_parent_lock = 1; return 1; } else if (is_usb_interface(dev)) { @@ -823,12 +824,16 @@ static int usb_device_match(struct device *dev, struct device_driver *drv) usb_drv = to_usb_driver(drv); id = usb_match_id(intf, usb_drv->id_table); - if (id) + if (id) { + dev->need_parent_lock = 1; return 1; + } id = usb_match_dynamic_id(intf, usb_drv); - if (id) + if (id) { + dev->need_parent_lock = 1; return 1; + } } return 0; diff --git a/drivers/usb/core/usb-acpi.c b/drivers/usb/core/usb-acpi.c index e221861b3187..03901fc86227 100644 --- a/drivers/usb/core/usb-acpi.c +++ b/drivers/usb/core/usb-acpi.c @@ -224,7 +224,11 @@ static struct acpi_device *usb_acpi_find_companion(struct device *dev) static bool usb_acpi_bus_match(struct device *dev) { - return is_usb_device(dev) || is_usb_port(dev); + if (is_usb_device(dev) || is_usb_port(dev)) { + dev->need_parent_lock = 1; + return 1; + } + return 0; } static struct acpi_bus_type usb_acpi_bus = { diff --git a/drivers/usb/serial/bus.c b/drivers/usb/serial/bus.c index 9e265eb92611..599ba0046f8e 100644 --- a/drivers/usb/serial/bus.c +++ b/drivers/usb/serial/bus.c @@ -29,8 +29,10 @@ static int usb_serial_device_match(struct device *dev, driver = to_usb_serial_driver(drv); - if (driver == port->serial->type) + if (driver == port->serial->type) { + dev->need_parent_lock = 1; return 1; + } return 0; } diff --git a/include/linux/device.h b/include/linux/device.h index 477956990f5e..a96194f39dbe 100644 --- a/include/linux/device.h +++ b/include/linux/device.h @@ -992,6 +992,7 @@ struct device { bool offline_disabled:1; bool offline:1; bool of_node_reused:1; + bool need_parent_lock:1; }; static inline struct device *kobj_to_dev(struct kobject *kobj) -- 2.17.0.921.gf22659ad46-goog