diff options
Diffstat (limited to 'drivers/base')
| -rw-r--r-- | drivers/base/core.c | 308 | ||||
| -rw-r--r-- | drivers/base/cpu.c | 17 | ||||
| -rw-r--r-- | drivers/base/firmware_loader/Kconfig | 14 | ||||
| -rw-r--r-- | drivers/base/firmware_loader/builtin/Makefile | 3 | ||||
| -rw-r--r-- | drivers/base/firmware_loader/main.c | 9 | ||||
| -rw-r--r-- | drivers/base/memory.c | 76 | ||||
| -rw-r--r-- | drivers/base/platform.c | 393 | ||||
| -rw-r--r-- | drivers/base/power/common.c | 20 | ||||
| -rw-r--r-- | drivers/base/power/domain.c | 40 | ||||
| -rw-r--r-- | drivers/base/power/power.h | 30 | ||||
| -rw-r--r-- | drivers/base/power/wakeirq.c | 4 | ||||
| -rw-r--r-- | drivers/base/property.c | 83 | ||||
| -rw-r--r-- | drivers/base/regmap/regmap-w1.c | 4 | ||||
| -rw-r--r-- | drivers/base/soc.c | 30 | ||||
| -rw-r--r-- | drivers/base/swnode.c | 258 | 
15 files changed, 690 insertions, 599 deletions
| diff --git a/drivers/base/core.c b/drivers/base/core.c index 7bd9cd366d41..42a672456432 100644 --- a/drivers/base/core.c +++ b/drivers/base/core.c @@ -45,6 +45,10 @@ early_param("sysfs.deprecated", sysfs_deprecated_setup);  #endif  /* Device links support. */ +static LIST_HEAD(wait_for_suppliers); +static DEFINE_MUTEX(wfs_lock); +static LIST_HEAD(deferred_sync); +static unsigned int defer_sync_state_count = 1;  #ifdef CONFIG_SRCU  static DEFINE_MUTEX(device_links_lock); @@ -127,6 +131,9 @@ static int device_is_dependent(struct device *dev, void *target)  		return ret;  	list_for_each_entry(link, &dev->links.consumers, s_node) { +		if (link->flags == (DL_FLAG_SYNC_STATE_ONLY | DL_FLAG_MANAGED)) +			continue; +  		if (link->consumer == target)  			return 1; @@ -196,8 +203,11 @@ static int device_reorder_to_tail(struct device *dev, void *not_used)  		device_pm_move_last(dev);  	device_for_each_child(dev, NULL, device_reorder_to_tail); -	list_for_each_entry(link, &dev->links.consumers, s_node) +	list_for_each_entry(link, &dev->links.consumers, s_node) { +		if (link->flags == (DL_FLAG_SYNC_STATE_ONLY | DL_FLAG_MANAGED)) +			continue;  		device_reorder_to_tail(link->consumer, NULL); +	}  	return 0;  } @@ -224,7 +234,8 @@ void device_pm_move_to_tail(struct device *dev)  #define DL_MANAGED_LINK_FLAGS (DL_FLAG_AUTOREMOVE_CONSUMER | \  			       DL_FLAG_AUTOREMOVE_SUPPLIER | \ -			       DL_FLAG_AUTOPROBE_CONSUMER) +			       DL_FLAG_AUTOPROBE_CONSUMER  | \ +			       DL_FLAG_SYNC_STATE_ONLY)  #define DL_ADD_VALID_FLAGS (DL_MANAGED_LINK_FLAGS | DL_FLAG_STATELESS | \  			    DL_FLAG_PM_RUNTIME | DL_FLAG_RPM_ACTIVE) @@ -292,6 +303,8 @@ struct device_link *device_link_add(struct device *consumer,  	if (!consumer || !supplier || flags & ~DL_ADD_VALID_FLAGS ||  	    (flags & DL_FLAG_STATELESS && flags & DL_MANAGED_LINK_FLAGS) || +	    (flags & DL_FLAG_SYNC_STATE_ONLY && +	     flags != DL_FLAG_SYNC_STATE_ONLY) ||  	    (flags & DL_FLAG_AUTOPROBE_CONSUMER &&  	     flags & (DL_FLAG_AUTOREMOVE_CONSUMER |  		      DL_FLAG_AUTOREMOVE_SUPPLIER))) @@ -312,11 +325,14 @@ struct device_link *device_link_add(struct device *consumer,  	/*  	 * If the supplier has not been fully registered yet or there is a -	 * reverse dependency between the consumer and the supplier already in -	 * the graph, return NULL. +	 * reverse (non-SYNC_STATE_ONLY) dependency between the consumer and +	 * the supplier already in the graph, return NULL. If the link is a +	 * SYNC_STATE_ONLY link, we don't check for reverse dependencies +	 * because it only affects sync_state() callbacks.  	 */  	if (!device_pm_initialized(supplier) -	    || device_is_dependent(consumer, supplier)) { +	    || (!(flags & DL_FLAG_SYNC_STATE_ONLY) && +		  device_is_dependent(consumer, supplier))) {  		link = NULL;  		goto out;  	} @@ -343,9 +359,14 @@ struct device_link *device_link_add(struct device *consumer,  		}  		if (flags & DL_FLAG_STATELESS) { -			link->flags |= DL_FLAG_STATELESS;  			kref_get(&link->kref); -			goto out; +			if (link->flags & DL_FLAG_SYNC_STATE_ONLY && +			    !(link->flags & DL_FLAG_STATELESS)) { +				link->flags |= DL_FLAG_STATELESS; +				goto reorder; +			} else { +				goto out; +			}  		}  		/* @@ -367,6 +388,12 @@ struct device_link *device_link_add(struct device *consumer,  			link->flags |= DL_FLAG_MANAGED;  			device_link_init_status(link, consumer, supplier);  		} +		if (link->flags & DL_FLAG_SYNC_STATE_ONLY && +		    !(flags & DL_FLAG_SYNC_STATE_ONLY)) { +			link->flags &= ~DL_FLAG_SYNC_STATE_ONLY; +			goto reorder; +		} +  		goto out;  	} @@ -406,6 +433,13 @@ struct device_link *device_link_add(struct device *consumer,  	    flags & DL_FLAG_PM_RUNTIME)  		pm_runtime_resume(supplier); +	if (flags & DL_FLAG_SYNC_STATE_ONLY) { +		dev_dbg(consumer, +			"Linked as a sync state only consumer to %s\n", +			dev_name(supplier)); +		goto out; +	} +reorder:  	/*  	 * Move the consumer and all of the devices depending on it to the end  	 * of dpm_list and the devices_kset list. @@ -431,6 +465,70 @@ struct device_link *device_link_add(struct device *consumer,  }  EXPORT_SYMBOL_GPL(device_link_add); +/** + * device_link_wait_for_supplier - Add device to wait_for_suppliers list + * @consumer: Consumer device + * + * Marks the @consumer device as waiting for suppliers to become available by + * adding it to the wait_for_suppliers list. The consumer device will never be + * probed until it's removed from the wait_for_suppliers list. + * + * The caller is responsible for adding the links to the supplier devices once + * they are available and removing the @consumer device from the + * wait_for_suppliers list once links to all the suppliers have been created. + * + * This function is NOT meant to be called from the probe function of the + * consumer but rather from code that creates/adds the consumer device. + */ +static void device_link_wait_for_supplier(struct device *consumer, +					  bool need_for_probe) +{ +	mutex_lock(&wfs_lock); +	list_add_tail(&consumer->links.needs_suppliers, &wait_for_suppliers); +	consumer->links.need_for_probe = need_for_probe; +	mutex_unlock(&wfs_lock); +} + +static void device_link_wait_for_mandatory_supplier(struct device *consumer) +{ +	device_link_wait_for_supplier(consumer, true); +} + +static void device_link_wait_for_optional_supplier(struct device *consumer) +{ +	device_link_wait_for_supplier(consumer, false); +} + +/** + * device_link_add_missing_supplier_links - Add links from consumer devices to + *					    supplier devices, leaving any + *					    consumer with inactive suppliers on + *					    the wait_for_suppliers list + * + * Loops through all consumers waiting on suppliers and tries to add all their + * supplier links. If that succeeds, the consumer device is removed from + * wait_for_suppliers list. Otherwise, they are left in the wait_for_suppliers + * list.  Devices left on the wait_for_suppliers list will not be probed. + * + * The fwnode add_links callback is expected to return 0 if it has found and + * added all the supplier links for the consumer device. It should return an + * error if it isn't able to do so. + * + * The caller of device_link_wait_for_supplier() is expected to call this once + * it's aware of potential suppliers becoming available. + */ +static void device_link_add_missing_supplier_links(void) +{ +	struct device *dev, *tmp; + +	mutex_lock(&wfs_lock); +	list_for_each_entry_safe(dev, tmp, &wait_for_suppliers, +				 links.needs_suppliers) +		if (!fwnode_call_int_op(dev->fwnode, add_links, dev)) +			list_del_init(&dev->links.needs_suppliers); +	mutex_unlock(&wfs_lock); +} +  static void device_link_free(struct device_link *link)  {  	while (refcount_dec_not_one(&link->rpm_active)) @@ -565,10 +663,23 @@ int device_links_check_suppliers(struct device *dev)  	struct device_link *link;  	int ret = 0; +	/* +	 * Device waiting for supplier to become available is not allowed to +	 * probe. +	 */ +	mutex_lock(&wfs_lock); +	if (!list_empty(&dev->links.needs_suppliers) && +	    dev->links.need_for_probe) { +		mutex_unlock(&wfs_lock); +		return -EPROBE_DEFER; +	} +	mutex_unlock(&wfs_lock); +  	device_links_write_lock();  	list_for_each_entry(link, &dev->links.suppliers, c_node) { -		if (!(link->flags & DL_FLAG_MANAGED)) +		if (!(link->flags & DL_FLAG_MANAGED) || +		    link->flags & DL_FLAG_SYNC_STATE_ONLY)  			continue;  		if (link->status != DL_STATE_AVAILABLE) { @@ -585,6 +696,128 @@ int device_links_check_suppliers(struct device *dev)  }  /** + * __device_links_queue_sync_state - Queue a device for sync_state() callback + * @dev: Device to call sync_state() on + * @list: List head to queue the @dev on + * + * Queues a device for a sync_state() callback when the device links write lock + * isn't held. This allows the sync_state() execution flow to use device links + * APIs.  The caller must ensure this function is called with + * device_links_write_lock() held. + * + * This function does a get_device() to make sure the device is not freed while + * on this list. + * + * So the caller must also ensure that device_links_flush_sync_list() is called + * as soon as the caller releases device_links_write_lock().  This is necessary + * to make sure the sync_state() is called in a timely fashion and the + * put_device() is called on this device. + */ +static void __device_links_queue_sync_state(struct device *dev, +					    struct list_head *list) +{ +	struct device_link *link; + +	if (dev->state_synced) +		return; + +	list_for_each_entry(link, &dev->links.consumers, s_node) { +		if (!(link->flags & DL_FLAG_MANAGED)) +			continue; +		if (link->status != DL_STATE_ACTIVE) +			return; +	} + +	/* +	 * Set the flag here to avoid adding the same device to a list more +	 * than once. This can happen if new consumers get added to the device +	 * and probed before the list is flushed. +	 */ +	dev->state_synced = true; + +	if (WARN_ON(!list_empty(&dev->links.defer_sync))) +		return; + +	get_device(dev); +	list_add_tail(&dev->links.defer_sync, list); +} + +/** + * device_links_flush_sync_list - Call sync_state() on a list of devices + * @list: List of devices to call sync_state() on + * + * Calls sync_state() on all the devices that have been queued for it. This + * function is used in conjunction with __device_links_queue_sync_state(). + */ +static void device_links_flush_sync_list(struct list_head *list) +{ +	struct device *dev, *tmp; + +	list_for_each_entry_safe(dev, tmp, list, links.defer_sync) { +		list_del_init(&dev->links.defer_sync); + +		device_lock(dev); + +		if (dev->bus->sync_state) +			dev->bus->sync_state(dev); +		else if (dev->driver && dev->driver->sync_state) +			dev->driver->sync_state(dev); + +		device_unlock(dev); + +		put_device(dev); +	} +} + +void device_links_supplier_sync_state_pause(void) +{ +	device_links_write_lock(); +	defer_sync_state_count++; +	device_links_write_unlock(); +} + +void device_links_supplier_sync_state_resume(void) +{ +	struct device *dev, *tmp; +	LIST_HEAD(sync_list); + +	device_links_write_lock(); +	if (!defer_sync_state_count) { +		WARN(true, "Unmatched sync_state pause/resume!"); +		goto out; +	} +	defer_sync_state_count--; +	if (defer_sync_state_count) +		goto out; + +	list_for_each_entry_safe(dev, tmp, &deferred_sync, links.defer_sync) { +		/* +		 * Delete from deferred_sync list before queuing it to +		 * sync_list because defer_sync is used for both lists. +		 */ +		list_del_init(&dev->links.defer_sync); +		__device_links_queue_sync_state(dev, &sync_list); +	} +out: +	device_links_write_unlock(); + +	device_links_flush_sync_list(&sync_list); +} + +static int sync_state_resume_initcall(void) +{ +	device_links_supplier_sync_state_resume(); +	return 0; +} +late_initcall(sync_state_resume_initcall); + +static void __device_links_supplier_defer_sync(struct device *sup) +{ +	if (list_empty(&sup->links.defer_sync)) +		list_add_tail(&sup->links.defer_sync, &deferred_sync); +} + +/**   * device_links_driver_bound - Update device links after probing its driver.   * @dev: Device to update the links for.   * @@ -598,6 +831,16 @@ int device_links_check_suppliers(struct device *dev)  void device_links_driver_bound(struct device *dev)  {  	struct device_link *link; +	LIST_HEAD(sync_list); + +	/* +	 * If a device probes successfully, it's expected to have created all +	 * the device links it needs to or make new device links as it needs +	 * them. So, it no longer needs to wait on any suppliers. +	 */ +	mutex_lock(&wfs_lock); +	list_del_init(&dev->links.needs_suppliers); +	mutex_unlock(&wfs_lock);  	device_links_write_lock(); @@ -628,11 +871,19 @@ void device_links_driver_bound(struct device *dev)  		WARN_ON(link->status != DL_STATE_CONSUMER_PROBE);  		WRITE_ONCE(link->status, DL_STATE_ACTIVE); + +		if (defer_sync_state_count) +			__device_links_supplier_defer_sync(link->supplier); +		else +			__device_links_queue_sync_state(link->supplier, +							&sync_list);  	}  	dev->links.status = DL_DEV_DRIVER_BOUND;  	device_links_write_unlock(); + +	device_links_flush_sync_list(&sync_list);  }  static void device_link_drop_managed(struct device_link *link) @@ -744,6 +995,7 @@ void device_links_driver_cleanup(struct device *dev)  		WRITE_ONCE(link->status, DL_STATE_DORMANT);  	} +	list_del_init(&dev->links.defer_sync);  	__device_links_no_driver(dev);  	device_links_write_unlock(); @@ -813,7 +1065,8 @@ void device_links_unbind_consumers(struct device *dev)  	list_for_each_entry(link, &dev->links.consumers, s_node) {  		enum device_link_state status; -		if (!(link->flags & DL_FLAG_MANAGED)) +		if (!(link->flags & DL_FLAG_MANAGED) || +		    link->flags & DL_FLAG_SYNC_STATE_ONLY)  			continue;  		status = link->status; @@ -849,6 +1102,10 @@ static void device_links_purge(struct device *dev)  {  	struct device_link *link, *ln; +	mutex_lock(&wfs_lock); +	list_del(&dev->links.needs_suppliers); +	mutex_unlock(&wfs_lock); +  	/*  	 * Delete all of the remaining links from this device to any other  	 * devices (either consumers or suppliers). @@ -1713,6 +1970,8 @@ void device_initialize(struct device *dev)  #endif  	INIT_LIST_HEAD(&dev->links.consumers);  	INIT_LIST_HEAD(&dev->links.suppliers); +	INIT_LIST_HEAD(&dev->links.needs_suppliers); +	INIT_LIST_HEAD(&dev->links.defer_sync);  	dev->links.status = DL_DEV_NO_DRIVER;  }  EXPORT_SYMBOL_GPL(device_initialize); @@ -2101,7 +2360,7 @@ int device_add(struct device *dev)  	struct device *parent;  	struct kobject *kobj;  	struct class_interface *class_intf; -	int error = -EINVAL; +	int error = -EINVAL, fw_ret;  	struct kobject *glue_dir = NULL;  	dev = get_device(dev); @@ -2199,6 +2458,32 @@ int device_add(struct device *dev)  					     BUS_NOTIFY_ADD_DEVICE, dev);  	kobject_uevent(&dev->kobj, KOBJ_ADD); + +	if (dev->fwnode && !dev->fwnode->dev) +		dev->fwnode->dev = dev; + +	/* +	 * Check if any of the other devices (consumers) have been waiting for +	 * this device (supplier) to be added so that they can create a device +	 * link to it. +	 * +	 * This needs to happen after device_pm_add() because device_link_add() +	 * requires the supplier be registered before it's called. +	 * +	 * But this also needs to happe before bus_probe_device() to make sure +	 * waiting consumers can link to it before the driver is bound to the +	 * device and the driver sync_state callback is called for this device. +	 */ +	device_link_add_missing_supplier_links(); + +	if (fwnode_has_op(dev->fwnode, add_links)) { +		fw_ret = fwnode_call_int_op(dev->fwnode, add_links, dev); +		if (fw_ret == -ENODEV) +			device_link_wait_for_mandatory_supplier(dev); +		else if (fw_ret) +			device_link_wait_for_optional_supplier(dev); +	} +  	bus_probe_device(dev);  	if (parent)  		klist_add_tail(&dev->p->knode_parent, @@ -2343,6 +2628,9 @@ void device_del(struct device *dev)  	kill_device(dev);  	device_unlock(dev); +	if (dev->fwnode && dev->fwnode->dev == dev) +		dev->fwnode->dev = NULL; +  	/* Notify clients of device removal.  This call must come  	 * before dpm_sysfs_remove().  	 */ diff --git a/drivers/base/cpu.c b/drivers/base/cpu.c index cc37511de866..6265871a4af2 100644 --- a/drivers/base/cpu.c +++ b/drivers/base/cpu.c @@ -554,12 +554,27 @@ ssize_t __weak cpu_show_mds(struct device *dev,  	return sprintf(buf, "Not affected\n");  } +ssize_t __weak cpu_show_tsx_async_abort(struct device *dev, +					struct device_attribute *attr, +					char *buf) +{ +	return sprintf(buf, "Not affected\n"); +} + +ssize_t __weak cpu_show_itlb_multihit(struct device *dev, +			    struct device_attribute *attr, char *buf) +{ +	return sprintf(buf, "Not affected\n"); +} +  static DEVICE_ATTR(meltdown, 0444, cpu_show_meltdown, NULL);  static DEVICE_ATTR(spectre_v1, 0444, cpu_show_spectre_v1, NULL);  static DEVICE_ATTR(spectre_v2, 0444, cpu_show_spectre_v2, NULL);  static DEVICE_ATTR(spec_store_bypass, 0444, cpu_show_spec_store_bypass, NULL);  static DEVICE_ATTR(l1tf, 0444, cpu_show_l1tf, NULL);  static DEVICE_ATTR(mds, 0444, cpu_show_mds, NULL); +static DEVICE_ATTR(tsx_async_abort, 0444, cpu_show_tsx_async_abort, NULL); +static DEVICE_ATTR(itlb_multihit, 0444, cpu_show_itlb_multihit, NULL);  static struct attribute *cpu_root_vulnerabilities_attrs[] = {  	&dev_attr_meltdown.attr, @@ -568,6 +583,8 @@ static struct attribute *cpu_root_vulnerabilities_attrs[] = {  	&dev_attr_spec_store_bypass.attr,  	&dev_attr_l1tf.attr,  	&dev_attr_mds.attr, +	&dev_attr_tsx_async_abort.attr, +	&dev_attr_itlb_multihit.attr,  	NULL  }; diff --git a/drivers/base/firmware_loader/Kconfig b/drivers/base/firmware_loader/Kconfig index 3f9e274e2ed3..5b24f3959255 100644 --- a/drivers/base/firmware_loader/Kconfig +++ b/drivers/base/firmware_loader/Kconfig @@ -148,7 +148,7 @@ config FW_LOADER_USER_HELPER_FALLBACK  	  to be used for all firmware requests which explicitly do not disable a  	  a fallback mechanism. Firmware calls which do prohibit a fallback  	  mechanism is request_firmware_direct(). This option is kept for -          backward compatibility purposes given this precise mechanism can also +	  backward compatibility purposes given this precise mechanism can also  	  be enabled by setting the proc sysctl value to true:  	       /proc/sys/kernel/firmware_config/force_sysfs_fallback @@ -169,5 +169,17 @@ config FW_LOADER_COMPRESS  	  be compressed with either none or crc32 integrity check type (pass  	  "-C crc32" option to xz command). +config FW_CACHE +	bool "Enable firmware caching during suspend" +	depends on PM_SLEEP +	default y if PM_SLEEP +	help +	  Because firmware caching generates uevent messages that are sent +	  over a netlink socket, it can prevent suspend on many platforms. +	  It is also not always useful, so on such platforms we have the +	  option. + +	  If unsure, say Y. +  endif # FW_LOADER  endmenu diff --git a/drivers/base/firmware_loader/builtin/Makefile b/drivers/base/firmware_loader/builtin/Makefile index 37e5ae387400..4a66888e7253 100644 --- a/drivers/base/firmware_loader/builtin/Makefile +++ b/drivers/base/firmware_loader/builtin/Makefile @@ -8,7 +8,8 @@ fwdir := $(addprefix $(srctree)/,$(filter-out /%,$(fwdir)))$(filter /%,$(fwdir))  obj-y  := $(addsuffix .gen.o, $(subst $(quote),,$(CONFIG_EXTRA_FIRMWARE)))  FWNAME    = $(patsubst $(obj)/%.gen.S,%,$@) -FWSTR     = $(subst /,_,$(subst .,_,$(subst -,_,$(FWNAME)))) +comma     := , +FWSTR     = $(subst $(comma),_,$(subst /,_,$(subst .,_,$(subst -,_,$(FWNAME)))))  ASM_WORD  = $(if $(CONFIG_64BIT),.quad,.long)  ASM_ALIGN = $(if $(CONFIG_64BIT),3,2)  PROGBITS  = $(if $(CONFIG_ARM),%,@)progbits diff --git a/drivers/base/firmware_loader/main.c b/drivers/base/firmware_loader/main.c index bf44c79beae9..249add8c5e05 100644 --- a/drivers/base/firmware_loader/main.c +++ b/drivers/base/firmware_loader/main.c @@ -4,7 +4,7 @@   *   * Copyright (c) 2003 Manuel Estrada Sainz   * - * Please see Documentation/firmware_class/ for more information. + * Please see Documentation/driver-api/firmware/ for more information.   *   */ @@ -51,7 +51,7 @@ struct firmware_cache {  	struct list_head head;  	int state; -#ifdef CONFIG_PM_SLEEP +#ifdef CONFIG_FW_CACHE  	/*  	 * Names of firmware images which have been cached successfully  	 * will be added into the below list so that device uncache @@ -504,6 +504,7 @@ fw_get_filesystem_firmware(struct device *device, struct fw_priv *fw_priv,  					 path);  			continue;  		} +		dev_dbg(device, "Loading firmware from %s\n", path);  		if (decompress) {  			dev_dbg(device, "f/w decompressing %s\n",  				fw_priv->fw_name); @@ -556,7 +557,7 @@ static void fw_set_page_data(struct fw_priv *fw_priv, struct firmware *fw)  		 (unsigned int)fw_priv->size);  } -#ifdef CONFIG_PM_SLEEP +#ifdef CONFIG_FW_CACHE  static void fw_name_devm_release(struct device *dev, void *res)  {  	struct fw_name_devm *fwn = res; @@ -1046,7 +1047,7 @@ request_firmware_nowait(  }  EXPORT_SYMBOL(request_firmware_nowait); -#ifdef CONFIG_PM_SLEEP +#ifdef CONFIG_FW_CACHE  static ASYNC_DOMAIN_EXCLUSIVE(fw_cache_domain);  /** diff --git a/drivers/base/memory.c b/drivers/base/memory.c index 55907c27075b..799b43191dea 100644 --- a/drivers/base/memory.c +++ b/drivers/base/memory.c @@ -19,15 +19,12 @@  #include <linux/memory.h>  #include <linux/memory_hotplug.h>  #include <linux/mm.h> -#include <linux/mutex.h>  #include <linux/stat.h>  #include <linux/slab.h>  #include <linux/atomic.h>  #include <linux/uaccess.h> -static DEFINE_MUTEX(mem_sysfs_mutex); -  #define MEMORY_CLASS_NAME	"memory"  #define to_memory_block(dev) container_of(dev, struct memory_block, dev) @@ -538,12 +535,7 @@ static ssize_t soft_offline_page_store(struct device *dev,  	if (kstrtoull(buf, 0, &pfn) < 0)  		return -EINVAL;  	pfn >>= PAGE_SHIFT; -	if (!pfn_valid(pfn)) -		return -ENXIO; -	/* Only online pages can be soft-offlined (esp., not ZONE_DEVICE). */ -	if (!pfn_to_online_page(pfn)) -		return -EIO; -	ret = soft_offline_page(pfn_to_page(pfn), 0); +	ret = soft_offline_page(pfn, 0);  	return ret == 0 ? count : ret;  } @@ -705,6 +697,8 @@ static void unregister_memory(struct memory_block *memory)   * Create memory block devices for the given memory area. Start and size   * have to be aligned to memory block granularity. Memory block devices   * will be initialized as offline. + * + * Called under device_hotplug_lock.   */  int create_memory_block_devices(unsigned long start, unsigned long size)  { @@ -718,7 +712,6 @@ int create_memory_block_devices(unsigned long start, unsigned long size)  			 !IS_ALIGNED(size, memory_block_size_bytes())))  		return -EINVAL; -	mutex_lock(&mem_sysfs_mutex);  	for (block_id = start_block_id; block_id != end_block_id; block_id++) {  		ret = init_memory_block(&mem, block_id, MEM_OFFLINE);  		if (ret) @@ -730,11 +723,12 @@ int create_memory_block_devices(unsigned long start, unsigned long size)  		for (block_id = start_block_id; block_id != end_block_id;  		     block_id++) {  			mem = find_memory_block_by_id(block_id); +			if (WARN_ON_ONCE(!mem)) +				continue;  			mem->section_count = 0;  			unregister_memory(mem);  		}  	} -	mutex_unlock(&mem_sysfs_mutex);  	return ret;  } @@ -742,6 +736,8 @@ int create_memory_block_devices(unsigned long start, unsigned long size)   * Remove memory block devices for the given memory area. Start and size   * have to be aligned to memory block granularity. Memory block devices   * have to be offline. + * + * Called under device_hotplug_lock.   */  void remove_memory_block_devices(unsigned long start, unsigned long size)  { @@ -754,7 +750,6 @@ void remove_memory_block_devices(unsigned long start, unsigned long size)  			 !IS_ALIGNED(size, memory_block_size_bytes())))  		return; -	mutex_lock(&mem_sysfs_mutex);  	for (block_id = start_block_id; block_id != end_block_id; block_id++) {  		mem = find_memory_block_by_id(block_id);  		if (WARN_ON_ONCE(!mem)) @@ -763,7 +758,6 @@ void remove_memory_block_devices(unsigned long start, unsigned long size)  		unregister_memory_block_under_nodes(mem);  		unregister_memory(mem);  	} -	mutex_unlock(&mem_sysfs_mutex);  }  /* return true if the memory block is offlined, otherwise, return false */ @@ -797,12 +791,13 @@ static const struct attribute_group *memory_root_attr_groups[] = {  };  /* - * Initialize the sysfs support for memory devices... + * Initialize the sysfs support for memory devices. At the time this function + * is called, we cannot have concurrent creation/deletion of memory block + * devices, the device_hotplug_lock is not needed.   */  void __init memory_dev_init(void)  {  	int ret; -	int err;  	unsigned long block_sz, nr;  	/* Validate the configured memory block size */ @@ -813,24 +808,19 @@ void __init memory_dev_init(void)  	ret = subsys_system_register(&memory_subsys, memory_root_attr_groups);  	if (ret) -		goto out; +		panic("%s() failed to register subsystem: %d\n", __func__, ret);  	/*  	 * Create entries for memory sections that were found  	 * during boot and have been initialized  	 */ -	mutex_lock(&mem_sysfs_mutex);  	for (nr = 0; nr <= __highest_present_section_nr;  	     nr += sections_per_block) { -		err = add_memory_block(nr); -		if (!ret) -			ret = err; +		ret = add_memory_block(nr); +		if (ret) +			panic("%s() failed to add memory block: %d\n", __func__, +			      ret);  	} -	mutex_unlock(&mem_sysfs_mutex); - -out: -	if (ret) -		panic("%s() failed: %d\n", __func__, ret);  }  /** @@ -872,3 +862,39 @@ int walk_memory_blocks(unsigned long start, unsigned long size,  	}  	return ret;  } + +struct for_each_memory_block_cb_data { +	walk_memory_blocks_func_t func; +	void *arg; +}; + +static int for_each_memory_block_cb(struct device *dev, void *data) +{ +	struct memory_block *mem = to_memory_block(dev); +	struct for_each_memory_block_cb_data *cb_data = data; + +	return cb_data->func(mem, cb_data->arg); +} + +/** + * for_each_memory_block - walk through all present memory blocks + * + * @arg: argument passed to func + * @func: callback for each memory block walked + * + * This function walks through all present memory blocks, calling func on + * each memory block. + * + * In case func() returns an error, walking is aborted and the error is + * returned. + */ +int for_each_memory_block(void *arg, walk_memory_blocks_func_t func) +{ +	struct for_each_memory_block_cb_data cb_data = { +		.func = func, +		.arg = arg, +	}; + +	return bus_for_each_dev(&memory_subsys, NULL, &cb_data, +				for_each_memory_block_cb); +} diff --git a/drivers/base/platform.c b/drivers/base/platform.c index b230beb6ccb4..7c532548b0a6 100644 --- a/drivers/base/platform.c +++ b/drivers/base/platform.c @@ -60,6 +60,7 @@ struct resource *platform_get_resource(struct platform_device *dev,  }  EXPORT_SYMBOL_GPL(platform_get_resource); +#ifdef CONFIG_HAS_IOMEM  /**   * devm_platform_ioremap_resource - call devm_ioremap_resource() for a platform   *				    device @@ -68,7 +69,6 @@ EXPORT_SYMBOL_GPL(platform_get_resource);   *        resource management   * @index: resource index   */ -#ifdef CONFIG_HAS_IOMEM  void __iomem *devm_platform_ioremap_resource(struct platform_device *pdev,  					     unsigned int index)  { @@ -78,9 +78,63 @@ void __iomem *devm_platform_ioremap_resource(struct platform_device *pdev,  	return devm_ioremap_resource(&pdev->dev, res);  }  EXPORT_SYMBOL_GPL(devm_platform_ioremap_resource); + +/** + * devm_platform_ioremap_resource_wc - write-combined variant of + *                                     devm_platform_ioremap_resource() + * + * @pdev: platform device to use both for memory resource lookup as well as + *        resource management + * @index: resource index + */ +void __iomem *devm_platform_ioremap_resource_wc(struct platform_device *pdev, +						unsigned int index) +{ +	struct resource *res; + +	res = platform_get_resource(pdev, IORESOURCE_MEM, index); +	return devm_ioremap_resource_wc(&pdev->dev, res); +} + +/** + * devm_platform_ioremap_resource_byname - call devm_ioremap_resource for + *					   a platform device, retrieve the + *					   resource by name + * + * @pdev: platform device to use both for memory resource lookup as well as + *	  resource management + * @name: name of the resource + */ +void __iomem * +devm_platform_ioremap_resource_byname(struct platform_device *pdev, +				      const char *name) +{ +	struct resource *res; + +	res = platform_get_resource_byname(pdev, IORESOURCE_MEM, name); +	return devm_ioremap_resource(&pdev->dev, res); +} +EXPORT_SYMBOL_GPL(devm_platform_ioremap_resource_byname);  #endif /* CONFIG_HAS_IOMEM */ -static int __platform_get_irq(struct platform_device *dev, unsigned int num) +/** + * platform_get_irq_optional - get an optional IRQ for a device + * @dev: platform device + * @num: IRQ number index + * + * Gets an IRQ for a platform device. Device drivers should check the return + * value for errors so as to not pass a negative integer value to the + * request_irq() APIs. This is the same as platform_get_irq(), except that it + * does not print an error message if an IRQ can not be obtained. + * + * Example: + *		int irq = platform_get_irq_optional(pdev, 0); + *		if (irq < 0) + *			return irq; + * + * Return: IRQ number on success, negative error number on failure. + */ +int platform_get_irq_optional(struct platform_device *dev, unsigned int num)  {  #ifdef CONFIG_SPARC  	/* sparc does not have irqs represented as IORESOURCE_IRQ resources */ @@ -89,9 +143,9 @@ static int __platform_get_irq(struct platform_device *dev, unsigned int num)  	return dev->archdata.irqs[num];  #else  	struct resource *r; -	if (IS_ENABLED(CONFIG_OF_IRQ) && dev->dev.of_node) { -		int ret; +	int ret; +	if (IS_ENABLED(CONFIG_OF_IRQ) && dev->dev.of_node) {  		ret = of_irq_get(dev->dev.of_node, num);  		if (ret > 0 || ret == -EPROBE_DEFER)  			return ret; @@ -100,8 +154,6 @@ static int __platform_get_irq(struct platform_device *dev, unsigned int num)  	r = platform_get_resource(dev, IORESOURCE_IRQ, num);  	if (has_acpi_companion(&dev->dev)) {  		if (r && r->flags & IORESOURCE_DISABLED) { -			int ret; -  			ret = acpi_irq_get(ACPI_HANDLE(&dev->dev), num, r);  			if (ret)  				return ret; @@ -134,8 +186,7 @@ static int __platform_get_irq(struct platform_device *dev, unsigned int num)  	 * allows a common code path across either kind of resource.  	 */  	if (num == 0 && has_acpi_companion(&dev->dev)) { -		int ret = acpi_dev_gpio_irq_get(ACPI_COMPANION(&dev->dev), num); - +		ret = acpi_dev_gpio_irq_get(ACPI_COMPANION(&dev->dev), num);  		/* Our callers expect -ENXIO for missing IRQs. */  		if (ret >= 0 || ret == -EPROBE_DEFER)  			return ret; @@ -144,6 +195,7 @@ static int __platform_get_irq(struct platform_device *dev, unsigned int num)  	return -ENXIO;  #endif  } +EXPORT_SYMBOL_GPL(platform_get_irq_optional);  /**   * platform_get_irq - get an IRQ for a device @@ -165,7 +217,7 @@ int platform_get_irq(struct platform_device *dev, unsigned int num)  {  	int ret; -	ret = __platform_get_irq(dev, num); +	ret = platform_get_irq_optional(dev, num);  	if (ret < 0 && ret != -EPROBE_DEFER)  		dev_err(&dev->dev, "IRQ index %u not found\n", num); @@ -174,29 +226,6 @@ int platform_get_irq(struct platform_device *dev, unsigned int num)  EXPORT_SYMBOL_GPL(platform_get_irq);  /** - * platform_get_irq_optional - get an optional IRQ for a device - * @dev: platform device - * @num: IRQ number index - * - * Gets an IRQ for a platform device. Device drivers should check the return - * value for errors so as to not pass a negative integer value to the - * request_irq() APIs. This is the same as platform_get_irq(), except that it - * does not print an error message if an IRQ can not be obtained. - * - * Example: - *		int irq = platform_get_irq_optional(pdev, 0); - *		if (irq < 0) - *			return irq; - * - * Return: IRQ number on success, negative error number on failure. - */ -int platform_get_irq_optional(struct platform_device *dev, unsigned int num) -{ -	return __platform_get_irq(dev, num); -} -EXPORT_SYMBOL_GPL(platform_get_irq_optional); - -/**   * platform_irq_count - Count the number of IRQs a platform device uses   * @dev: platform device   * @@ -206,7 +235,7 @@ int platform_irq_count(struct platform_device *dev)  {  	int ret, nr = 0; -	while ((ret = __platform_get_irq(dev, nr)) >= 0) +	while ((ret = platform_get_irq_optional(dev, nr)) >= 0)  		nr++;  	if (ret == -EPROBE_DEFER) @@ -245,10 +274,9 @@ static int __platform_get_irq_byname(struct platform_device *dev,  				     const char *name)  {  	struct resource *r; +	int ret;  	if (IS_ENABLED(CONFIG_OF_IRQ) && dev->dev.of_node) { -		int ret; -  		ret = of_irq_get_byname(dev->dev.of_node, name);  		if (ret > 0 || ret == -EPROBE_DEFER)  			return ret; @@ -1278,6 +1306,11 @@ struct bus_type platform_bus_type = {  };  EXPORT_SYMBOL_GPL(platform_bus_type); +static inline int __platform_match(struct device *dev, const void *drv) +{ +	return platform_match(dev, (struct device_driver *)drv); +} +  /**   * platform_find_device_by_driver - Find a platform device with a given   * driver. @@ -1288,7 +1321,7 @@ struct device *platform_find_device_by_driver(struct device *start,  					      const struct device_driver *drv)  {  	return bus_find_device(&platform_bus_type, start, drv, -			       (void *)platform_match); +			       __platform_match);  }  EXPORT_SYMBOL_GPL(platform_find_device_by_driver); @@ -1296,8 +1329,6 @@ int __init platform_bus_init(void)  {  	int error; -	early_platform_cleanup(); -  	error = device_register(&platform_bus);  	if (error) {  		put_device(&platform_bus); @@ -1309,289 +1340,3 @@ int __init platform_bus_init(void)  	of_platform_register_reconfig_notifier();  	return error;  } - -static __initdata LIST_HEAD(early_platform_driver_list); -static __initdata LIST_HEAD(early_platform_device_list); - -/** - * early_platform_driver_register - register early platform driver - * @epdrv: early_platform driver structure - * @buf: string passed from early_param() - * - * Helper function for early_platform_init() / early_platform_init_buffer() - */ -int __init early_platform_driver_register(struct early_platform_driver *epdrv, -					  char *buf) -{ -	char *tmp; -	int n; - -	/* Simply add the driver to the end of the global list. -	 * Drivers will by default be put on the list in compiled-in order. -	 */ -	if (!epdrv->list.next) { -		INIT_LIST_HEAD(&epdrv->list); -		list_add_tail(&epdrv->list, &early_platform_driver_list); -	} - -	/* If the user has specified device then make sure the driver -	 * gets prioritized. The driver of the last device specified on -	 * command line will be put first on the list. -	 */ -	n = strlen(epdrv->pdrv->driver.name); -	if (buf && !strncmp(buf, epdrv->pdrv->driver.name, n)) { -		list_move(&epdrv->list, &early_platform_driver_list); - -		/* Allow passing parameters after device name */ -		if (buf[n] == '\0' || buf[n] == ',') -			epdrv->requested_id = -1; -		else { -			epdrv->requested_id = simple_strtoul(&buf[n + 1], -							     &tmp, 10); - -			if (buf[n] != '.' || (tmp == &buf[n + 1])) { -				epdrv->requested_id = EARLY_PLATFORM_ID_ERROR; -				n = 0; -			} else -				n += strcspn(&buf[n + 1], ",") + 1; -		} - -		if (buf[n] == ',') -			n++; - -		if (epdrv->bufsize) { -			memcpy(epdrv->buffer, &buf[n], -			       min_t(int, epdrv->bufsize, strlen(&buf[n]) + 1)); -			epdrv->buffer[epdrv->bufsize - 1] = '\0'; -		} -	} - -	return 0; -} - -/** - * early_platform_add_devices - adds a number of early platform devices - * @devs: array of early platform devices to add - * @num: number of early platform devices in array - * - * Used by early architecture code to register early platform devices and - * their platform data. - */ -void __init early_platform_add_devices(struct platform_device **devs, int num) -{ -	struct device *dev; -	int i; - -	/* simply add the devices to list */ -	for (i = 0; i < num; i++) { -		dev = &devs[i]->dev; - -		if (!dev->devres_head.next) { -			pm_runtime_early_init(dev); -			INIT_LIST_HEAD(&dev->devres_head); -			list_add_tail(&dev->devres_head, -				      &early_platform_device_list); -		} -	} -} - -/** - * early_platform_driver_register_all - register early platform drivers - * @class_str: string to identify early platform driver class - * - * Used by architecture code to register all early platform drivers - * for a certain class. If omitted then only early platform drivers - * with matching kernel command line class parameters will be registered. - */ -void __init early_platform_driver_register_all(char *class_str) -{ -	/* The "class_str" parameter may or may not be present on the kernel -	 * command line. If it is present then there may be more than one -	 * matching parameter. -	 * -	 * Since we register our early platform drivers using early_param() -	 * we need to make sure that they also get registered in the case -	 * when the parameter is missing from the kernel command line. -	 * -	 * We use parse_early_options() to make sure the early_param() gets -	 * called at least once. The early_param() may be called more than -	 * once since the name of the preferred device may be specified on -	 * the kernel command line. early_platform_driver_register() handles -	 * this case for us. -	 */ -	parse_early_options(class_str); -} - -/** - * early_platform_match - find early platform device matching driver - * @epdrv: early platform driver structure - * @id: id to match against - */ -static struct platform_device * __init -early_platform_match(struct early_platform_driver *epdrv, int id) -{ -	struct platform_device *pd; - -	list_for_each_entry(pd, &early_platform_device_list, dev.devres_head) -		if (platform_match(&pd->dev, &epdrv->pdrv->driver)) -			if (pd->id == id) -				return pd; - -	return NULL; -} - -/** - * early_platform_left - check if early platform driver has matching devices - * @epdrv: early platform driver structure - * @id: return true if id or above exists - */ -static int __init early_platform_left(struct early_platform_driver *epdrv, -				       int id) -{ -	struct platform_device *pd; - -	list_for_each_entry(pd, &early_platform_device_list, dev.devres_head) -		if (platform_match(&pd->dev, &epdrv->pdrv->driver)) -			if (pd->id >= id) -				return 1; - -	return 0; -} - -/** - * early_platform_driver_probe_id - probe drivers matching class_str and id - * @class_str: string to identify early platform driver class - * @id: id to match against - * @nr_probe: number of platform devices to successfully probe before exiting - */ -static int __init early_platform_driver_probe_id(char *class_str, -						 int id, -						 int nr_probe) -{ -	struct early_platform_driver *epdrv; -	struct platform_device *match; -	int match_id; -	int n = 0; -	int left = 0; - -	list_for_each_entry(epdrv, &early_platform_driver_list, list) { -		/* only use drivers matching our class_str */ -		if (strcmp(class_str, epdrv->class_str)) -			continue; - -		if (id == -2) { -			match_id = epdrv->requested_id; -			left = 1; - -		} else { -			match_id = id; -			left += early_platform_left(epdrv, id); - -			/* skip requested id */ -			switch (epdrv->requested_id) { -			case EARLY_PLATFORM_ID_ERROR: -			case EARLY_PLATFORM_ID_UNSET: -				break; -			default: -				if (epdrv->requested_id == id) -					match_id = EARLY_PLATFORM_ID_UNSET; -			} -		} - -		switch (match_id) { -		case EARLY_PLATFORM_ID_ERROR: -			pr_warn("%s: unable to parse %s parameter\n", -				class_str, epdrv->pdrv->driver.name); -			/* fall-through */ -		case EARLY_PLATFORM_ID_UNSET: -			match = NULL; -			break; -		default: -			match = early_platform_match(epdrv, match_id); -		} - -		if (match) { -			/* -			 * Set up a sensible init_name to enable -			 * dev_name() and others to be used before the -			 * rest of the driver core is initialized. -			 */ -			if (!match->dev.init_name && slab_is_available()) { -				if (match->id != -1) -					match->dev.init_name = -						kasprintf(GFP_KERNEL, "%s.%d", -							  match->name, -							  match->id); -				else -					match->dev.init_name = -						kasprintf(GFP_KERNEL, "%s", -							  match->name); - -				if (!match->dev.init_name) -					return -ENOMEM; -			} - -			if (epdrv->pdrv->probe(match)) -				pr_warn("%s: unable to probe %s early.\n", -					class_str, match->name); -			else -				n++; -		} - -		if (n >= nr_probe) -			break; -	} - -	if (left) -		return n; -	else -		return -ENODEV; -} - -/** - * early_platform_driver_probe - probe a class of registered drivers - * @class_str: string to identify early platform driver class - * @nr_probe: number of platform devices to successfully probe before exiting - * @user_only: only probe user specified early platform devices - * - * Used by architecture code to probe registered early platform drivers - * within a certain class. For probe to happen a registered early platform - * device matching a registered early platform driver is needed. - */ -int __init early_platform_driver_probe(char *class_str, -				       int nr_probe, -				       int user_only) -{ -	int k, n, i; - -	n = 0; -	for (i = -2; n < nr_probe; i++) { -		k = early_platform_driver_probe_id(class_str, i, nr_probe - n); - -		if (k < 0) -			break; - -		n += k; - -		if (user_only) -			break; -	} - -	return n; -} - -/** - * early_platform_cleanup - clean up early platform code - */ -void __init early_platform_cleanup(void) -{ -	struct platform_device *pd, *pd2; - -	/* clean up the devres list used to chain devices */ -	list_for_each_entry_safe(pd, pd2, &early_platform_device_list, -				 dev.devres_head) { -		list_del(&pd->dev.devres_head); -		memset(&pd->dev.devres_head, 0, sizeof(pd->dev.devres_head)); -	} -} - diff --git a/drivers/base/power/common.c b/drivers/base/power/common.c index 8db98a1f83dc..bbddb267c2e6 100644 --- a/drivers/base/power/common.c +++ b/drivers/base/power/common.c @@ -188,6 +188,26 @@ void dev_pm_domain_detach(struct device *dev, bool power_off)  EXPORT_SYMBOL_GPL(dev_pm_domain_detach);  /** + * dev_pm_domain_start - Start the device through its PM domain. + * @dev: Device to start. + * + * This function should typically be called during probe by a subsystem/driver, + * when it needs to start its device from the PM domain's perspective. Note + * that, it's assumed that the PM domain is already powered on when this + * function is called. + * + * Returns 0 on success and negative error values on failures. + */ +int dev_pm_domain_start(struct device *dev) +{ +	if (dev->pm_domain && dev->pm_domain->start) +		return dev->pm_domain->start(dev); + +	return 0; +} +EXPORT_SYMBOL_GPL(dev_pm_domain_start); + +/**   * dev_pm_domain_set - Set PM domain of a device.   * @dev: Device whose PM domain is to be set.   * @pd: PM domain to be set, or NULL. diff --git a/drivers/base/power/domain.c b/drivers/base/power/domain.c index cc85e87eaf05..8e5725b11ee8 100644 --- a/drivers/base/power/domain.c +++ b/drivers/base/power/domain.c @@ -634,6 +634,13 @@ static int genpd_power_on(struct generic_pm_domain *genpd, unsigned int depth)  	return ret;  } +static int genpd_dev_pm_start(struct device *dev) +{ +	struct generic_pm_domain *genpd = dev_to_genpd(dev); + +	return genpd_start_dev(genpd, dev); +} +  static int genpd_dev_pm_qos_notifier(struct notifier_block *nb,  				     unsigned long val, void *ptr)  { @@ -922,24 +929,6 @@ static int __init genpd_power_off_unused(void)  }  late_initcall(genpd_power_off_unused); -#if defined(CONFIG_PM_SLEEP) || defined(CONFIG_PM_GENERIC_DOMAINS_OF) - -static bool genpd_present(const struct generic_pm_domain *genpd) -{ -	const struct generic_pm_domain *gpd; - -	if (IS_ERR_OR_NULL(genpd)) -		return false; - -	list_for_each_entry(gpd, &gpd_list, gpd_list_node) -		if (gpd == genpd) -			return true; - -	return false; -} - -#endif -  #ifdef CONFIG_PM_SLEEP  /** @@ -1354,8 +1343,8 @@ static void genpd_syscore_switch(struct device *dev, bool suspend)  {  	struct generic_pm_domain *genpd; -	genpd = dev_to_genpd(dev); -	if (!genpd_present(genpd)) +	genpd = dev_to_genpd_safe(dev); +	if (!genpd)  		return;  	if (suspend) { @@ -1805,6 +1794,7 @@ int pm_genpd_init(struct generic_pm_domain *genpd,  	genpd->domain.ops.poweroff_noirq = genpd_poweroff_noirq;  	genpd->domain.ops.restore_noirq = genpd_restore_noirq;  	genpd->domain.ops.complete = genpd_complete; +	genpd->domain.start = genpd_dev_pm_start;  	if (genpd->flags & GENPD_FLAG_PM_CLK) {  		genpd->dev_ops.stop = pm_clk_suspend; @@ -2020,6 +2010,16 @@ static int genpd_add_provider(struct device_node *np, genpd_xlate_t xlate,  	return 0;  } +static bool genpd_present(const struct generic_pm_domain *genpd) +{ +	const struct generic_pm_domain *gpd; + +	list_for_each_entry(gpd, &gpd_list, gpd_list_node) +		if (gpd == genpd) +			return true; +	return false; +} +  /**   * of_genpd_add_provider_simple() - Register a simple PM domain provider   * @np: Device node pointer associated with the PM domain provider. diff --git a/drivers/base/power/power.h b/drivers/base/power/power.h index 39a06a0cfdaa..444f5c169a0b 100644 --- a/drivers/base/power/power.h +++ b/drivers/base/power/power.h @@ -117,6 +117,13 @@ static inline bool device_pm_initialized(struct device *dev)  	return dev->power.in_dpm_list;  } +/* drivers/base/power/wakeup_stats.c */ +extern int wakeup_source_sysfs_add(struct device *parent, +				   struct wakeup_source *ws); +extern void wakeup_source_sysfs_remove(struct wakeup_source *ws); + +extern int pm_wakeup_source_sysfs_add(struct device *parent); +  #else /* !CONFIG_PM_SLEEP */  static inline void device_pm_sleep_init(struct device *dev) {} @@ -141,6 +148,11 @@ static inline bool device_pm_initialized(struct device *dev)  	return device_is_registered(dev);  } +static inline int pm_wakeup_source_sysfs_add(struct device *parent) +{ +	return 0; +} +  #endif /* !CONFIG_PM_SLEEP */  static inline void device_pm_init(struct device *dev) @@ -149,21 +161,3 @@ static inline void device_pm_init(struct device *dev)  	device_pm_sleep_init(dev);  	pm_runtime_init(dev);  } - -#ifdef CONFIG_PM_SLEEP - -/* drivers/base/power/wakeup_stats.c */ -extern int wakeup_source_sysfs_add(struct device *parent, -				   struct wakeup_source *ws); -extern void wakeup_source_sysfs_remove(struct wakeup_source *ws); - -extern int pm_wakeup_source_sysfs_add(struct device *parent); - -#else /* !CONFIG_PM_SLEEP */ - -static inline int pm_wakeup_source_sysfs_add(struct device *parent) -{ -	return 0; -} - -#endif /* CONFIG_PM_SLEEP */ diff --git a/drivers/base/power/wakeirq.c b/drivers/base/power/wakeirq.c index 5ce77d1ef9fc..8e021082dba8 100644 --- a/drivers/base/power/wakeirq.c +++ b/drivers/base/power/wakeirq.c @@ -272,7 +272,7 @@ void dev_pm_enable_wake_irq_check(struct device *dev,  {  	struct wake_irq *wirq = dev->power.wakeirq; -	if (!wirq || !((wirq->status & WAKE_IRQ_DEDICATED_MASK))) +	if (!wirq || !(wirq->status & WAKE_IRQ_DEDICATED_MASK))  		return;  	if (likely(wirq->status & WAKE_IRQ_DEDICATED_MANAGED)) { @@ -299,7 +299,7 @@ void dev_pm_disable_wake_irq_check(struct device *dev)  {  	struct wake_irq *wirq = dev->power.wakeirq; -	if (!wirq || !((wirq->status & WAKE_IRQ_DEDICATED_MASK))) +	if (!wirq || !(wirq->status & WAKE_IRQ_DEDICATED_MASK))  		return;  	if (wirq->status & WAKE_IRQ_DEDICATED_MANAGED) diff --git a/drivers/base/property.c b/drivers/base/property.c index 81bd01ed4042..511f6d7acdfe 100644 --- a/drivers/base/property.c +++ b/drivers/base/property.c @@ -557,6 +557,42 @@ int device_add_properties(struct device *dev,  EXPORT_SYMBOL_GPL(device_add_properties);  /** + * fwnode_get_name - Return the name of a node + * @fwnode: The firmware node + * + * Returns a pointer to the node name. + */ +const char *fwnode_get_name(const struct fwnode_handle *fwnode) +{ +	return fwnode_call_ptr_op(fwnode, get_name); +} + +/** + * fwnode_get_name_prefix - Return the prefix of node for printing purposes + * @fwnode: The firmware node + * + * Returns the prefix of a node, intended to be printed right before the node. + * The prefix works also as a separator between the nodes. + */ +const char *fwnode_get_name_prefix(const struct fwnode_handle *fwnode) +{ +	return fwnode_call_ptr_op(fwnode, get_name_prefix); +} + +/** + * fwnode_get_parent - Return parent firwmare node + * @fwnode: Firmware whose parent is retrieved + * + * Return parent firmware node of the given node if possible or %NULL if no + * parent was available. + */ +struct fwnode_handle *fwnode_get_parent(const struct fwnode_handle *fwnode) +{ +	return fwnode_call_ptr_op(fwnode, get_parent); +} +EXPORT_SYMBOL_GPL(fwnode_get_parent); + +/**   * fwnode_get_next_parent - Iterate to the node's parent   * @fwnode: Firmware whose parent is retrieved   * @@ -578,17 +614,50 @@ struct fwnode_handle *fwnode_get_next_parent(struct fwnode_handle *fwnode)  EXPORT_SYMBOL_GPL(fwnode_get_next_parent);  /** - * fwnode_get_parent - Return parent firwmare node - * @fwnode: Firmware whose parent is retrieved + * fwnode_count_parents - Return the number of parents a node has + * @fwnode: The node the parents of which are to be counted   * - * Return parent firmware node of the given node if possible or %NULL if no - * parent was available. + * Returns the number of parents a node has.   */ -struct fwnode_handle *fwnode_get_parent(const struct fwnode_handle *fwnode) +unsigned int fwnode_count_parents(const struct fwnode_handle *fwnode)  { -	return fwnode_call_ptr_op(fwnode, get_parent); +	struct fwnode_handle *__fwnode; +	unsigned int count; + +	__fwnode = fwnode_get_parent(fwnode); + +	for (count = 0; __fwnode; count++) +		__fwnode = fwnode_get_next_parent(__fwnode); + +	return count;  } -EXPORT_SYMBOL_GPL(fwnode_get_parent); +EXPORT_SYMBOL_GPL(fwnode_count_parents); + +/** + * fwnode_get_nth_parent - Return an nth parent of a node + * @fwnode: The node the parent of which is requested + * @depth: Distance of the parent from the node + * + * Returns the nth parent of a node. If there is no parent at the requested + * @depth, %NULL is returned. If @depth is 0, the functionality is equivalent to + * fwnode_handle_get(). For @depth == 1, it is fwnode_get_parent() and so on. + * + * The caller is responsible for calling fwnode_handle_put() for the returned + * node. + */ +struct fwnode_handle *fwnode_get_nth_parent(struct fwnode_handle *fwnode, +					    unsigned int depth) +{ +	unsigned int i; + +	fwnode_handle_get(fwnode); + +	for (i = 0; i < depth && fwnode; i++) +		fwnode = fwnode_get_next_parent(fwnode); + +	return fwnode; +} +EXPORT_SYMBOL_GPL(fwnode_get_nth_parent);  /**   * fwnode_get_next_child_node - Return the next child node handle for a node diff --git a/drivers/base/regmap/regmap-w1.c b/drivers/base/regmap/regmap-w1.c index 3a7d30b8c3ac..1fbaaad71ca5 100644 --- a/drivers/base/regmap/regmap-w1.c +++ b/drivers/base/regmap/regmap-w1.c @@ -215,8 +215,6 @@ struct regmap *__regmap_init_w1(struct device *w1_dev,  	return __regmap_init(w1_dev, bus, w1_dev, config,  			 lock_key, lock_name); - -	return NULL;  }  EXPORT_SYMBOL_GPL(__regmap_init_w1); @@ -233,8 +231,6 @@ struct regmap *__devm_regmap_init_w1(struct device *w1_dev,  	return __devm_regmap_init(w1_dev, bus, w1_dev, config,  				 lock_key, lock_name); - -	return NULL;  }  EXPORT_SYMBOL_GPL(__devm_regmap_init_w1); diff --git a/drivers/base/soc.c b/drivers/base/soc.c index 7c0c5ca5953d..4af11a423475 100644 --- a/drivers/base/soc.c +++ b/drivers/base/soc.c @@ -104,15 +104,12 @@ static const struct attribute_group soc_attr_group = {  	.is_visible = soc_attribute_mode,  }; -static const struct attribute_group *soc_attr_groups[] = { -	&soc_attr_group, -	NULL, -}; -  static void soc_release(struct device *dev)  {  	struct soc_device *soc_dev = container_of(dev, struct soc_device, dev); +	ida_simple_remove(&soc_ida, soc_dev->soc_dev_num); +	kfree(soc_dev->dev.groups);  	kfree(soc_dev);  } @@ -121,6 +118,7 @@ static struct soc_device_attribute *early_soc_dev_attr;  struct soc_device *soc_device_register(struct soc_device_attribute *soc_dev_attr)  {  	struct soc_device *soc_dev; +	const struct attribute_group **soc_attr_groups;  	int ret;  	if (!soc_bus_type.p) { @@ -136,10 +134,18 @@ struct soc_device *soc_device_register(struct soc_device_attribute *soc_dev_attr  		goto out1;  	} +	soc_attr_groups = kcalloc(3, sizeof(*soc_attr_groups), GFP_KERNEL); +	if (!soc_attr_groups) { +		ret = -ENOMEM; +		goto out2; +	} +	soc_attr_groups[0] = &soc_attr_group; +	soc_attr_groups[1] = soc_dev_attr->custom_attr_group; +  	/* Fetch a unique (reclaimable) SOC ID. */  	ret = ida_simple_get(&soc_ida, 0, 0, GFP_KERNEL);  	if (ret < 0) -		goto out2; +		goto out3;  	soc_dev->soc_dev_num = ret;  	soc_dev->attr = soc_dev_attr; @@ -150,15 +156,15 @@ struct soc_device *soc_device_register(struct soc_device_attribute *soc_dev_attr  	dev_set_name(&soc_dev->dev, "soc%d", soc_dev->soc_dev_num);  	ret = device_register(&soc_dev->dev); -	if (ret) -		goto out3; +	if (ret) { +		put_device(&soc_dev->dev); +		return ERR_PTR(ret); +	}  	return soc_dev;  out3: -	ida_simple_remove(&soc_ida, soc_dev->soc_dev_num); -	put_device(&soc_dev->dev); -	soc_dev = NULL; +	kfree(soc_attr_groups);  out2:  	kfree(soc_dev);  out1: @@ -169,8 +175,6 @@ EXPORT_SYMBOL_GPL(soc_device_register);  /* Ensure soc_dev->attr is freed prior to calling soc_device_unregister. */  void soc_device_unregister(struct soc_device *soc_dev)  { -	ida_simple_remove(&soc_ida, soc_dev->soc_dev_num); -  	device_unregister(&soc_dev->dev);  	early_soc_dev_attr = NULL;  } diff --git a/drivers/base/swnode.c b/drivers/base/swnode.c index a1f3f0994f9f..d8d0dc0ca5ac 100644 --- a/drivers/base/swnode.c +++ b/drivers/base/swnode.c @@ -71,9 +71,9 @@ software_node_to_swnode(const struct software_node *node)  	return swnode;  } -const struct software_node *to_software_node(struct fwnode_handle *fwnode) +const struct software_node *to_software_node(const struct fwnode_handle *fwnode)  { -	struct swnode *swnode = to_swnode(fwnode); +	const struct swnode *swnode = to_swnode(fwnode);  	return swnode ? swnode->node : NULL;  } @@ -103,71 +103,15 @@ property_entry_get(const struct property_entry *prop, const char *name)  	return NULL;  } -static void -property_set_pointer(struct property_entry *prop, const void *pointer) -{ -	switch (prop->type) { -	case DEV_PROP_U8: -		if (prop->is_array) -			prop->pointer.u8_data = pointer; -		else -			prop->value.u8_data = *((u8 *)pointer); -		break; -	case DEV_PROP_U16: -		if (prop->is_array) -			prop->pointer.u16_data = pointer; -		else -			prop->value.u16_data = *((u16 *)pointer); -		break; -	case DEV_PROP_U32: -		if (prop->is_array) -			prop->pointer.u32_data = pointer; -		else -			prop->value.u32_data = *((u32 *)pointer); -		break; -	case DEV_PROP_U64: -		if (prop->is_array) -			prop->pointer.u64_data = pointer; -		else -			prop->value.u64_data = *((u64 *)pointer); -		break; -	case DEV_PROP_STRING: -		if (prop->is_array) -			prop->pointer.str = pointer; -		else -			prop->value.str = pointer; -		break; -	default: -		break; -	} -} -  static const void *property_get_pointer(const struct property_entry *prop)  { -	switch (prop->type) { -	case DEV_PROP_U8: -		if (prop->is_array) -			return prop->pointer.u8_data; -		return &prop->value.u8_data; -	case DEV_PROP_U16: -		if (prop->is_array) -			return prop->pointer.u16_data; -		return &prop->value.u16_data; -	case DEV_PROP_U32: -		if (prop->is_array) -			return prop->pointer.u32_data; -		return &prop->value.u32_data; -	case DEV_PROP_U64: -		if (prop->is_array) -			return prop->pointer.u64_data; -		return &prop->value.u64_data; -	case DEV_PROP_STRING: -		if (prop->is_array) -			return prop->pointer.str; -		return &prop->value.str; -	default: +	if (!prop->length)  		return NULL; -	} + +	if (prop->is_array) +		return prop->pointer; + +	return &prop->value;  }  static const void *property_entry_find(const struct property_entry *props, @@ -187,66 +131,6 @@ static const void *property_entry_find(const struct property_entry *props,  	return pointer;  } -static int property_entry_read_u8_array(const struct property_entry *props, -					const char *propname, -					u8 *values, size_t nval) -{ -	const void *pointer; -	size_t length = nval * sizeof(*values); - -	pointer = property_entry_find(props, propname, length); -	if (IS_ERR(pointer)) -		return PTR_ERR(pointer); - -	memcpy(values, pointer, length); -	return 0; -} - -static int property_entry_read_u16_array(const struct property_entry *props, -					 const char *propname, -					 u16 *values, size_t nval) -{ -	const void *pointer; -	size_t length = nval * sizeof(*values); - -	pointer = property_entry_find(props, propname, length); -	if (IS_ERR(pointer)) -		return PTR_ERR(pointer); - -	memcpy(values, pointer, length); -	return 0; -} - -static int property_entry_read_u32_array(const struct property_entry *props, -					 const char *propname, -					 u32 *values, size_t nval) -{ -	const void *pointer; -	size_t length = nval * sizeof(*values); - -	pointer = property_entry_find(props, propname, length); -	if (IS_ERR(pointer)) -		return PTR_ERR(pointer); - -	memcpy(values, pointer, length); -	return 0; -} - -static int property_entry_read_u64_array(const struct property_entry *props, -					 const char *propname, -					 u64 *values, size_t nval) -{ -	const void *pointer; -	size_t length = nval * sizeof(*values); - -	pointer = property_entry_find(props, propname, length); -	if (IS_ERR(pointer)) -		return PTR_ERR(pointer); - -	memcpy(values, pointer, length); -	return 0; -} -  static int  property_entry_count_elems_of_size(const struct property_entry *props,  				   const char *propname, size_t length) @@ -265,49 +149,45 @@ static int property_entry_read_int_array(const struct property_entry *props,  					 unsigned int elem_size, void *val,  					 size_t nval)  { +	const void *pointer; +	size_t length; +  	if (!val)  		return property_entry_count_elems_of_size(props, name,  							  elem_size); -	switch (elem_size) { -	case sizeof(u8): -		return property_entry_read_u8_array(props, name, val, nval); -	case sizeof(u16): -		return property_entry_read_u16_array(props, name, val, nval); -	case sizeof(u32): -		return property_entry_read_u32_array(props, name, val, nval); -	case sizeof(u64): -		return property_entry_read_u64_array(props, name, val, nval); -	} -	return -ENXIO; +	if (!is_power_of_2(elem_size) || elem_size > sizeof(u64)) +		return -ENXIO; + +	length = nval * elem_size; + +	pointer = property_entry_find(props, name, length); +	if (IS_ERR(pointer)) +		return PTR_ERR(pointer); + +	memcpy(val, pointer, length); +	return 0;  }  static int property_entry_read_string_array(const struct property_entry *props,  					    const char *propname,  					    const char **strings, size_t nval)  { -	const struct property_entry *prop;  	const void *pointer; -	size_t array_len, length; +	size_t length; +	int array_len;  	/* Find out the array length. */ -	prop = property_entry_get(props, propname); -	if (!prop) -		return -EINVAL; - -	if (prop->is_array) -		/* Find the length of an array. */ -		array_len = property_entry_count_elems_of_size(props, propname, -							  sizeof(const char *)); -	else -		/* The array length for a non-array string property is 1. */ -		array_len = 1; +	array_len = property_entry_count_elems_of_size(props, propname, +						       sizeof(const char *)); +	if (array_len < 0) +		return array_len;  	/* Return how many there are if strings is NULL. */  	if (!strings)  		return array_len; -	array_len = min(nval, array_len); +	array_len = min_t(size_t, nval, array_len);  	length = array_len * sizeof(*strings);  	pointer = property_entry_find(props, propname, length); @@ -322,13 +202,15 @@ static int property_entry_read_string_array(const struct property_entry *props,  static void property_entry_free_data(const struct property_entry *p)  {  	const void *pointer = property_get_pointer(p); +	const char * const *src_str;  	size_t i, nval;  	if (p->is_array) { -		if (p->type == DEV_PROP_STRING && p->pointer.str) { +		if (p->type == DEV_PROP_STRING && p->pointer) { +			src_str = p->pointer;  			nval = p->length / sizeof(const char *);  			for (i = 0; i < nval; i++) -				kfree(p->pointer.str[i]); +				kfree(src_str[i]);  		}  		kfree(pointer);  	} else if (p->type == DEV_PROP_STRING) { @@ -337,29 +219,29 @@ static void property_entry_free_data(const struct property_entry *p)  	kfree(p->name);  } -static int property_copy_string_array(struct property_entry *dst, -				      const struct property_entry *src) +static const char * const * +property_copy_string_array(const struct property_entry *src)  {  	const char **d; +	const char * const *src_str = src->pointer;  	size_t nval = src->length / sizeof(*d);  	int i;  	d = kcalloc(nval, sizeof(*d), GFP_KERNEL);  	if (!d) -		return -ENOMEM; +		return NULL;  	for (i = 0; i < nval; i++) { -		d[i] = kstrdup(src->pointer.str[i], GFP_KERNEL); -		if (!d[i] && src->pointer.str[i]) { +		d[i] = kstrdup(src_str[i], GFP_KERNEL); +		if (!d[i] && src_str[i]) {  			while (--i >= 0)  				kfree(d[i]);  			kfree(d); -			return -ENOMEM; +			return NULL;  		}  	} -	dst->pointer.str = d; -	return 0; +	return d;  }  static int property_entry_copy_data(struct property_entry *dst, @@ -367,36 +249,35 @@ static int property_entry_copy_data(struct property_entry *dst,  {  	const void *pointer = property_get_pointer(src);  	const void *new; -	int error;  	if (src->is_array) {  		if (!src->length)  			return -ENODATA;  		if (src->type == DEV_PROP_STRING) { -			error = property_copy_string_array(dst, src); -			if (error) -				return error; -			new = dst->pointer.str; +			new = property_copy_string_array(src); +			if (!new) +				return -ENOMEM;  		} else {  			new = kmemdup(pointer, src->length, GFP_KERNEL);  			if (!new)  				return -ENOMEM;  		} + +		dst->is_array = true; +		dst->pointer = new;  	} else if (src->type == DEV_PROP_STRING) {  		new = kstrdup(src->value.str, GFP_KERNEL);  		if (!new && src->value.str)  			return -ENOMEM; + +		dst->value.str = new;  	} else { -		new = pointer; +		dst->value = src->value;  	}  	dst->length = src->length; -	dst->is_array = src->is_array;  	dst->type = src->type; - -	property_set_pointer(dst, new); -  	dst->name = kstrdup(src->name, GFP_KERNEL);  	if (!dst->name)  		goto out_free_data; @@ -515,12 +396,47 @@ static int software_node_read_string_array(const struct fwnode_handle *fwnode,  						propname, val, nval);  } +static const char * +software_node_get_name(const struct fwnode_handle *fwnode) +{ +	const struct swnode *swnode = to_swnode(fwnode); + +	if (!swnode) +		return "(null)"; + +	return kobject_name(&swnode->kobj); +} + +static const char * +software_node_get_name_prefix(const struct fwnode_handle *fwnode) +{ +	struct fwnode_handle *parent; +	const char *prefix; + +	parent = fwnode_get_parent(fwnode); +	if (!parent) +		return ""; + +	/* Figure out the prefix from the parents. */ +	while (is_software_node(parent)) +		parent = fwnode_get_next_parent(parent); + +	prefix = fwnode_get_name_prefix(parent); +	fwnode_handle_put(parent); + +	/* Guess something if prefix was NULL. */ +	return prefix ?: "/"; +} +  static struct fwnode_handle *  software_node_get_parent(const struct fwnode_handle *fwnode)  {  	struct swnode *swnode = to_swnode(fwnode); -	return swnode ? (swnode->parent ? &swnode->parent->fwnode : NULL) : NULL; +	if (!swnode || !swnode->parent) +		return NULL; + +	return fwnode_handle_get(&swnode->parent->fwnode);  }  static struct fwnode_handle * @@ -612,6 +528,8 @@ static const struct fwnode_operations software_node_ops = {  	.property_present = software_node_property_present,  	.property_read_int_array = software_node_read_int_array,  	.property_read_string_array = software_node_read_string_array, +	.get_name = software_node_get_name, +	.get_name_prefix = software_node_get_name_prefix,  	.get_parent = software_node_get_parent,  	.get_next_child_node = software_node_get_next_child,  	.get_named_child_node = software_node_get_named_child_node, | 

