diff options
Diffstat (limited to 'drivers/nvdimm/bus.c')
| -rw-r--r-- | drivers/nvdimm/bus.c | 210 | 
1 files changed, 140 insertions, 70 deletions
| diff --git a/drivers/nvdimm/bus.c b/drivers/nvdimm/bus.c index 2dca3034fee0..798c5c4aea9c 100644 --- a/drivers/nvdimm/bus.c +++ b/drivers/nvdimm/bus.c @@ -26,7 +26,7 @@  int nvdimm_major;  static int nvdimm_bus_major; -static struct class *nd_class; +struct class *nd_class;  static DEFINE_IDA(nd_ida);  static int to_nd_device_type(struct device *dev) @@ -73,7 +73,7 @@ static void nvdimm_bus_probe_end(struct nvdimm_bus *nvdimm_bus)  {  	nvdimm_bus_lock(&nvdimm_bus->dev);  	if (--nvdimm_bus->probe_active == 0) -		wake_up(&nvdimm_bus->probe_wait); +		wake_up(&nvdimm_bus->wait);  	nvdimm_bus_unlock(&nvdimm_bus->dev);  } @@ -91,7 +91,10 @@ static int nvdimm_bus_probe(struct device *dev)  			dev->driver->name, dev_name(dev));  	nvdimm_bus_probe_start(nvdimm_bus); +	debug_nvdimm_lock(dev);  	rc = nd_drv->probe(dev); +	debug_nvdimm_unlock(dev); +  	if (rc == 0)  		nd_region_probe_success(nvdimm_bus, dev);  	else @@ -113,8 +116,11 @@ static int nvdimm_bus_remove(struct device *dev)  	struct nvdimm_bus *nvdimm_bus = walk_to_nvdimm_bus(dev);  	int rc = 0; -	if (nd_drv->remove) +	if (nd_drv->remove) { +		debug_nvdimm_lock(dev);  		rc = nd_drv->remove(dev); +		debug_nvdimm_unlock(dev); +	}  	nd_region_disable(nvdimm_bus, dev);  	dev_dbg(&nvdimm_bus->dev, "%s.remove(%s) = %d\n", dev->driver->name, @@ -140,7 +146,7 @@ static void nvdimm_bus_shutdown(struct device *dev)  void nd_device_notify(struct device *dev, enum nvdimm_event event)  { -	device_lock(dev); +	nd_device_lock(dev);  	if (dev->driver) {  		struct nd_device_driver *nd_drv; @@ -148,7 +154,7 @@ void nd_device_notify(struct device *dev, enum nvdimm_event event)  		if (nd_drv->notify)  			nd_drv->notify(dev, event);  	} -	device_unlock(dev); +	nd_device_unlock(dev);  }  EXPORT_SYMBOL(nd_device_notify); @@ -296,7 +302,7 @@ static void nvdimm_bus_release(struct device *dev)  	kfree(nvdimm_bus);  } -static bool is_nvdimm_bus(struct device *dev) +bool is_nvdimm_bus(struct device *dev)  {  	return dev->release == nvdimm_bus_release;  } @@ -341,7 +347,7 @@ struct nvdimm_bus *nvdimm_bus_register(struct device *parent,  		return NULL;  	INIT_LIST_HEAD(&nvdimm_bus->list);  	INIT_LIST_HEAD(&nvdimm_bus->mapping_list); -	init_waitqueue_head(&nvdimm_bus->probe_wait); +	init_waitqueue_head(&nvdimm_bus->wait);  	nvdimm_bus->id = ida_simple_get(&nd_ida, 0, 0, GFP_KERNEL);  	if (nvdimm_bus->id < 0) {  		kfree(nvdimm_bus); @@ -426,6 +432,9 @@ static int nd_bus_remove(struct device *dev)  	list_del_init(&nvdimm_bus->list);  	mutex_unlock(&nvdimm_bus_list_mutex); +	wait_event(nvdimm_bus->wait, +			atomic_read(&nvdimm_bus->ioctl_active) == 0); +  	nd_synchronize();  	device_for_each_child(&nvdimm_bus->dev, NULL, child_unregister); @@ -547,13 +556,38 @@ EXPORT_SYMBOL(nd_device_register);  void nd_device_unregister(struct device *dev, enum nd_async_mode mode)  { +	bool killed; +  	switch (mode) {  	case ND_ASYNC: +		/* +		 * In the async case this is being triggered with the +		 * device lock held and the unregistration work needs to +		 * be moved out of line iff this is thread has won the +		 * race to schedule the deletion. +		 */ +		if (!kill_device(dev)) +			return; +  		get_device(dev);  		async_schedule_domain(nd_async_device_unregister, dev,  				&nd_async_domain);  		break;  	case ND_SYNC: +		/* +		 * In the sync case the device is being unregistered due +		 * to a state change of the parent. Claim the kill state +		 * to synchronize against other unregistration requests, +		 * or otherwise let the async path handle it if the +		 * unregistration was already queued. +		 */ +		nd_device_lock(dev); +		killed = kill_device(dev); +		nd_device_unlock(dev); + +		if (!killed) +			return; +  		nd_synchronize();  		device_unregister(dev);  		break; @@ -859,10 +893,12 @@ void wait_nvdimm_bus_probe_idle(struct device *dev)  	do {  		if (nvdimm_bus->probe_active == 0)  			break; -		nvdimm_bus_unlock(&nvdimm_bus->dev); -		wait_event(nvdimm_bus->probe_wait, +		nvdimm_bus_unlock(dev); +		nd_device_unlock(dev); +		wait_event(nvdimm_bus->wait,  				nvdimm_bus->probe_active == 0); -		nvdimm_bus_lock(&nvdimm_bus->dev); +		nd_device_lock(dev); +		nvdimm_bus_lock(dev);  	} while (true);  } @@ -945,20 +981,19 @@ static int __nd_ioctl(struct nvdimm_bus *nvdimm_bus, struct nvdimm *nvdimm,  		int read_only, unsigned int ioctl_cmd, unsigned long arg)  {  	struct nvdimm_bus_descriptor *nd_desc = nvdimm_bus->nd_desc; -	static char out_env[ND_CMD_MAX_ENVELOPE]; -	static char in_env[ND_CMD_MAX_ENVELOPE];  	const struct nd_cmd_desc *desc = NULL;  	unsigned int cmd = _IOC_NR(ioctl_cmd);  	struct device *dev = &nvdimm_bus->dev;  	void __user *p = (void __user *) arg; +	char *out_env = NULL, *in_env = NULL;  	const char *cmd_name, *dimm_name;  	u32 in_len = 0, out_len = 0;  	unsigned int func = cmd;  	unsigned long cmd_mask;  	struct nd_cmd_pkg pkg;  	int rc, i, cmd_rc; +	void *buf = NULL;  	u64 buf_len = 0; -	void *buf;  	if (nvdimm) {  		desc = nd_cmd_dimm_desc(cmd); @@ -989,7 +1024,7 @@ static int __nd_ioctl(struct nvdimm_bus *nvdimm_bus, struct nvdimm *nvdimm,  		case ND_CMD_ARS_START:  		case ND_CMD_CLEAR_ERROR:  		case ND_CMD_CALL: -			dev_dbg(&nvdimm_bus->dev, "'%s' command while read-only.\n", +			dev_dbg(dev, "'%s' command while read-only.\n",  					nvdimm ? nvdimm_cmd_name(cmd)  					: nvdimm_bus_cmd_name(cmd));  			return -EPERM; @@ -998,6 +1033,9 @@ static int __nd_ioctl(struct nvdimm_bus *nvdimm_bus, struct nvdimm *nvdimm,  		}  	/* process an input envelope */ +	in_env = kzalloc(ND_CMD_MAX_ENVELOPE, GFP_KERNEL); +	if (!in_env) +		return -ENOMEM;  	for (i = 0; i < desc->in_num; i++) {  		u32 in_size, copy; @@ -1005,14 +1043,17 @@ static int __nd_ioctl(struct nvdimm_bus *nvdimm_bus, struct nvdimm *nvdimm,  		if (in_size == UINT_MAX) {  			dev_err(dev, "%s:%s unknown input size cmd: %s field: %d\n",  					__func__, dimm_name, cmd_name, i); -			return -ENXIO; +			rc = -ENXIO; +			goto out;  		} -		if (in_len < sizeof(in_env)) -			copy = min_t(u32, sizeof(in_env) - in_len, in_size); +		if (in_len < ND_CMD_MAX_ENVELOPE) +			copy = min_t(u32, ND_CMD_MAX_ENVELOPE - in_len, in_size);  		else  			copy = 0; -		if (copy && copy_from_user(&in_env[in_len], p + in_len, copy)) -			return -EFAULT; +		if (copy && copy_from_user(&in_env[in_len], p + in_len, copy)) { +			rc = -EFAULT; +			goto out; +		}  		in_len += in_size;  	} @@ -1024,6 +1065,12 @@ static int __nd_ioctl(struct nvdimm_bus *nvdimm_bus, struct nvdimm *nvdimm,  	}  	/* process an output envelope */ +	out_env = kzalloc(ND_CMD_MAX_ENVELOPE, GFP_KERNEL); +	if (!out_env) { +		rc = -ENOMEM; +		goto out; +	} +  	for (i = 0; i < desc->out_num; i++) {  		u32 out_size = nd_cmd_out_size(nvdimm, cmd, desc, i,  				(u32 *) in_env, (u32 *) out_env, 0); @@ -1032,15 +1079,18 @@ static int __nd_ioctl(struct nvdimm_bus *nvdimm_bus, struct nvdimm *nvdimm,  		if (out_size == UINT_MAX) {  			dev_dbg(dev, "%s unknown output size cmd: %s field: %d\n",  					dimm_name, cmd_name, i); -			return -EFAULT; +			rc = -EFAULT; +			goto out;  		} -		if (out_len < sizeof(out_env)) -			copy = min_t(u32, sizeof(out_env) - out_len, out_size); +		if (out_len < ND_CMD_MAX_ENVELOPE) +			copy = min_t(u32, ND_CMD_MAX_ENVELOPE - out_len, out_size);  		else  			copy = 0;  		if (copy && copy_from_user(&out_env[out_len], -					p + in_len + out_len, copy)) -			return -EFAULT; +					p + in_len + out_len, copy)) { +			rc = -EFAULT; +			goto out; +		}  		out_len += out_size;  	} @@ -1048,19 +1098,23 @@ static int __nd_ioctl(struct nvdimm_bus *nvdimm_bus, struct nvdimm *nvdimm,  	if (buf_len > ND_IOCTL_MAX_BUFLEN) {  		dev_dbg(dev, "%s cmd: %s buf_len: %llu > %d\n", dimm_name,  				cmd_name, buf_len, ND_IOCTL_MAX_BUFLEN); -		return -EINVAL; +		rc = -EINVAL; +		goto out;  	}  	buf = vmalloc(buf_len); -	if (!buf) -		return -ENOMEM; +	if (!buf) { +		rc = -ENOMEM; +		goto out; +	}  	if (copy_from_user(buf, p, buf_len)) {  		rc = -EFAULT;  		goto out;  	} -	nvdimm_bus_lock(&nvdimm_bus->dev); +	nd_device_lock(dev); +	nvdimm_bus_lock(dev);  	rc = nd_cmd_clear_to_send(nvdimm_bus, nvdimm, func, buf);  	if (rc)  		goto out_unlock; @@ -1075,39 +1129,24 @@ static int __nd_ioctl(struct nvdimm_bus *nvdimm_bus, struct nvdimm *nvdimm,  		nvdimm_account_cleared_poison(nvdimm_bus, clear_err->address,  				clear_err->cleared);  	} -	nvdimm_bus_unlock(&nvdimm_bus->dev);  	if (copy_to_user(p, buf, buf_len))  		rc = -EFAULT; -	vfree(buf); -	return rc; - - out_unlock: -	nvdimm_bus_unlock(&nvdimm_bus->dev); - out: +out_unlock: +	nvdimm_bus_unlock(dev); +	nd_device_unlock(dev); +out: +	kfree(in_env); +	kfree(out_env);  	vfree(buf);  	return rc;  } -static long nd_ioctl(struct file *file, unsigned int cmd, unsigned long arg) -{ -	long id = (long) file->private_data; -	int rc = -ENXIO, ro; -	struct nvdimm_bus *nvdimm_bus; - -	ro = ((file->f_flags & O_ACCMODE) == O_RDONLY); -	mutex_lock(&nvdimm_bus_list_mutex); -	list_for_each_entry(nvdimm_bus, &nvdimm_bus_list, list) { -		if (nvdimm_bus->id == id) { -			rc = __nd_ioctl(nvdimm_bus, NULL, ro, cmd, arg); -			break; -		} -	} -	mutex_unlock(&nvdimm_bus_list_mutex); - -	return rc; -} +enum nd_ioctl_mode { +	BUS_IOCTL, +	DIMM_IOCTL, +};  static int match_dimm(struct device *dev, void *data)  { @@ -1122,31 +1161,62 @@ static int match_dimm(struct device *dev, void *data)  	return 0;  } -static long nvdimm_ioctl(struct file *file, unsigned int cmd, unsigned long arg) +static long nd_ioctl(struct file *file, unsigned int cmd, unsigned long arg, +		enum nd_ioctl_mode mode) +  { -	int rc = -ENXIO, ro; -	struct nvdimm_bus *nvdimm_bus; +	struct nvdimm_bus *nvdimm_bus, *found = NULL; +	long id = (long) file->private_data; +	struct nvdimm *nvdimm = NULL; +	int rc, ro;  	ro = ((file->f_flags & O_ACCMODE) == O_RDONLY);  	mutex_lock(&nvdimm_bus_list_mutex);  	list_for_each_entry(nvdimm_bus, &nvdimm_bus_list, list) { -		struct device *dev = device_find_child(&nvdimm_bus->dev, -				file->private_data, match_dimm); -		struct nvdimm *nvdimm; - -		if (!dev) -			continue; +		if (mode == DIMM_IOCTL) { +			struct device *dev; + +			dev = device_find_child(&nvdimm_bus->dev, +					file->private_data, match_dimm); +			if (!dev) +				continue; +			nvdimm = to_nvdimm(dev); +			found = nvdimm_bus; +		} else if (nvdimm_bus->id == id) { +			found = nvdimm_bus; +		} -		nvdimm = to_nvdimm(dev); -		rc = __nd_ioctl(nvdimm_bus, nvdimm, ro, cmd, arg); -		put_device(dev); -		break; +		if (found) { +			atomic_inc(&nvdimm_bus->ioctl_active); +			break; +		}  	}  	mutex_unlock(&nvdimm_bus_list_mutex); +	if (!found) +		return -ENXIO; + +	nvdimm_bus = found; +	rc = __nd_ioctl(nvdimm_bus, nvdimm, ro, cmd, arg); + +	if (nvdimm) +		put_device(&nvdimm->dev); +	if (atomic_dec_and_test(&nvdimm_bus->ioctl_active)) +		wake_up(&nvdimm_bus->wait); +  	return rc;  } +static long bus_ioctl(struct file *file, unsigned int cmd, unsigned long arg) +{ +	return nd_ioctl(file, cmd, arg, BUS_IOCTL); +} + +static long dimm_ioctl(struct file *file, unsigned int cmd, unsigned long arg) +{ +	return nd_ioctl(file, cmd, arg, DIMM_IOCTL); +} +  static int nd_open(struct inode *inode, struct file *file)  {  	long minor = iminor(inode); @@ -1158,16 +1228,16 @@ static int nd_open(struct inode *inode, struct file *file)  static const struct file_operations nvdimm_bus_fops = {  	.owner = THIS_MODULE,  	.open = nd_open, -	.unlocked_ioctl = nd_ioctl, -	.compat_ioctl = nd_ioctl, +	.unlocked_ioctl = bus_ioctl, +	.compat_ioctl = bus_ioctl,  	.llseek = noop_llseek,  };  static const struct file_operations nvdimm_fops = {  	.owner = THIS_MODULE,  	.open = nd_open, -	.unlocked_ioctl = nvdimm_ioctl, -	.compat_ioctl = nvdimm_ioctl, +	.unlocked_ioctl = dimm_ioctl, +	.compat_ioctl = dimm_ioctl,  	.llseek = noop_llseek,  }; | 
