239 lines
		
	
	
	
		
			5.4 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			239 lines
		
	
	
	
		
			5.4 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| // SPDX-License-Identifier: GPL-2.0-only
 | |
| /* Copyright(c) 2021 Intel Corporation. All rights reserved. */
 | |
| #include <linux/libnvdimm.h>
 | |
| #include <linux/device.h>
 | |
| #include <linux/module.h>
 | |
| #include <linux/ndctl.h>
 | |
| #include <linux/async.h>
 | |
| #include <linux/slab.h>
 | |
| #include "cxlmem.h"
 | |
| #include "cxl.h"
 | |
| 
 | |
| /*
 | |
|  * Ordered workqueue for cxl nvdimm device arrival and departure
 | |
|  * to coordinate bus rescans when a bridge arrives and trigger remove
 | |
|  * operations when the bridge is removed.
 | |
|  */
 | |
| static struct workqueue_struct *cxl_pmem_wq;
 | |
| 
 | |
| static void unregister_nvdimm(void *nvdimm)
 | |
| {
 | |
| 	nvdimm_delete(nvdimm);
 | |
| }
 | |
| 
 | |
| static int match_nvdimm_bridge(struct device *dev, const void *data)
 | |
| {
 | |
| 	return strcmp(dev_name(dev), "nvdimm-bridge") == 0;
 | |
| }
 | |
| 
 | |
| static struct cxl_nvdimm_bridge *cxl_find_nvdimm_bridge(void)
 | |
| {
 | |
| 	struct device *dev;
 | |
| 
 | |
| 	dev = bus_find_device(&cxl_bus_type, NULL, NULL, match_nvdimm_bridge);
 | |
| 	if (!dev)
 | |
| 		return NULL;
 | |
| 	return to_cxl_nvdimm_bridge(dev);
 | |
| }
 | |
| 
 | |
| static int cxl_nvdimm_probe(struct device *dev)
 | |
| {
 | |
| 	struct cxl_nvdimm *cxl_nvd = to_cxl_nvdimm(dev);
 | |
| 	struct cxl_nvdimm_bridge *cxl_nvb;
 | |
| 	unsigned long flags = 0;
 | |
| 	struct nvdimm *nvdimm;
 | |
| 	int rc = -ENXIO;
 | |
| 
 | |
| 	cxl_nvb = cxl_find_nvdimm_bridge();
 | |
| 	if (!cxl_nvb)
 | |
| 		return -ENXIO;
 | |
| 
 | |
| 	device_lock(&cxl_nvb->dev);
 | |
| 	if (!cxl_nvb->nvdimm_bus)
 | |
| 		goto out;
 | |
| 
 | |
| 	set_bit(NDD_LABELING, &flags);
 | |
| 	nvdimm = nvdimm_create(cxl_nvb->nvdimm_bus, cxl_nvd, NULL, flags, 0, 0,
 | |
| 			       NULL);
 | |
| 	if (!nvdimm)
 | |
| 		goto out;
 | |
| 
 | |
| 	rc = devm_add_action_or_reset(dev, unregister_nvdimm, nvdimm);
 | |
| out:
 | |
| 	device_unlock(&cxl_nvb->dev);
 | |
| 	put_device(&cxl_nvb->dev);
 | |
| 
 | |
| 	return rc;
 | |
| }
 | |
| 
 | |
| static struct cxl_driver cxl_nvdimm_driver = {
 | |
| 	.name = "cxl_nvdimm",
 | |
| 	.probe = cxl_nvdimm_probe,
 | |
| 	.id = CXL_DEVICE_NVDIMM,
 | |
| };
 | |
| 
 | |
| static int cxl_pmem_ctl(struct nvdimm_bus_descriptor *nd_desc,
 | |
| 			struct nvdimm *nvdimm, unsigned int cmd, void *buf,
 | |
| 			unsigned int buf_len, int *cmd_rc)
 | |
| {
 | |
| 	return -ENOTTY;
 | |
| }
 | |
| 
 | |
| static bool online_nvdimm_bus(struct cxl_nvdimm_bridge *cxl_nvb)
 | |
| {
 | |
| 	if (cxl_nvb->nvdimm_bus)
 | |
| 		return true;
 | |
| 	cxl_nvb->nvdimm_bus =
 | |
| 		nvdimm_bus_register(&cxl_nvb->dev, &cxl_nvb->nd_desc);
 | |
| 	return cxl_nvb->nvdimm_bus != NULL;
 | |
| }
 | |
| 
 | |
| static int cxl_nvdimm_release_driver(struct device *dev, void *data)
 | |
| {
 | |
| 	if (!is_cxl_nvdimm(dev))
 | |
| 		return 0;
 | |
| 	device_release_driver(dev);
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static void offline_nvdimm_bus(struct nvdimm_bus *nvdimm_bus)
 | |
| {
 | |
| 	if (!nvdimm_bus)
 | |
| 		return;
 | |
| 
 | |
| 	/*
 | |
| 	 * Set the state of cxl_nvdimm devices to unbound / idle before
 | |
| 	 * nvdimm_bus_unregister() rips the nvdimm objects out from
 | |
| 	 * underneath them.
 | |
| 	 */
 | |
| 	bus_for_each_dev(&cxl_bus_type, NULL, NULL, cxl_nvdimm_release_driver);
 | |
| 	nvdimm_bus_unregister(nvdimm_bus);
 | |
| }
 | |
| 
 | |
| static void cxl_nvb_update_state(struct work_struct *work)
 | |
| {
 | |
| 	struct cxl_nvdimm_bridge *cxl_nvb =
 | |
| 		container_of(work, typeof(*cxl_nvb), state_work);
 | |
| 	struct nvdimm_bus *victim_bus = NULL;
 | |
| 	bool release = false, rescan = false;
 | |
| 
 | |
| 	device_lock(&cxl_nvb->dev);
 | |
| 	switch (cxl_nvb->state) {
 | |
| 	case CXL_NVB_ONLINE:
 | |
| 		if (!online_nvdimm_bus(cxl_nvb)) {
 | |
| 			dev_err(&cxl_nvb->dev,
 | |
| 				"failed to establish nvdimm bus\n");
 | |
| 			release = true;
 | |
| 		} else
 | |
| 			rescan = true;
 | |
| 		break;
 | |
| 	case CXL_NVB_OFFLINE:
 | |
| 	case CXL_NVB_DEAD:
 | |
| 		victim_bus = cxl_nvb->nvdimm_bus;
 | |
| 		cxl_nvb->nvdimm_bus = NULL;
 | |
| 		break;
 | |
| 	default:
 | |
| 		break;
 | |
| 	}
 | |
| 	device_unlock(&cxl_nvb->dev);
 | |
| 
 | |
| 	if (release)
 | |
| 		device_release_driver(&cxl_nvb->dev);
 | |
| 	if (rescan) {
 | |
| 		int rc = bus_rescan_devices(&cxl_bus_type);
 | |
| 
 | |
| 		dev_dbg(&cxl_nvb->dev, "rescan: %d\n", rc);
 | |
| 	}
 | |
| 	offline_nvdimm_bus(victim_bus);
 | |
| 
 | |
| 	put_device(&cxl_nvb->dev);
 | |
| }
 | |
| 
 | |
| static void cxl_nvdimm_bridge_state_work(struct cxl_nvdimm_bridge *cxl_nvb)
 | |
| {
 | |
| 	/*
 | |
| 	 * Take a reference that the workqueue will drop if new work
 | |
| 	 * gets queued.
 | |
| 	 */
 | |
| 	get_device(&cxl_nvb->dev);
 | |
| 	if (!queue_work(cxl_pmem_wq, &cxl_nvb->state_work))
 | |
| 		put_device(&cxl_nvb->dev);
 | |
| }
 | |
| 
 | |
| static void cxl_nvdimm_bridge_remove(struct device *dev)
 | |
| {
 | |
| 	struct cxl_nvdimm_bridge *cxl_nvb = to_cxl_nvdimm_bridge(dev);
 | |
| 
 | |
| 	if (cxl_nvb->state == CXL_NVB_ONLINE)
 | |
| 		cxl_nvb->state = CXL_NVB_OFFLINE;
 | |
| 	cxl_nvdimm_bridge_state_work(cxl_nvb);
 | |
| }
 | |
| 
 | |
| static int cxl_nvdimm_bridge_probe(struct device *dev)
 | |
| {
 | |
| 	struct cxl_nvdimm_bridge *cxl_nvb = to_cxl_nvdimm_bridge(dev);
 | |
| 
 | |
| 	if (cxl_nvb->state == CXL_NVB_DEAD)
 | |
| 		return -ENXIO;
 | |
| 
 | |
| 	if (cxl_nvb->state == CXL_NVB_NEW) {
 | |
| 		cxl_nvb->nd_desc = (struct nvdimm_bus_descriptor) {
 | |
| 			.provider_name = "CXL",
 | |
| 			.module = THIS_MODULE,
 | |
| 			.ndctl = cxl_pmem_ctl,
 | |
| 		};
 | |
| 
 | |
| 		INIT_WORK(&cxl_nvb->state_work, cxl_nvb_update_state);
 | |
| 	}
 | |
| 
 | |
| 	cxl_nvb->state = CXL_NVB_ONLINE;
 | |
| 	cxl_nvdimm_bridge_state_work(cxl_nvb);
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static struct cxl_driver cxl_nvdimm_bridge_driver = {
 | |
| 	.name = "cxl_nvdimm_bridge",
 | |
| 	.probe = cxl_nvdimm_bridge_probe,
 | |
| 	.remove = cxl_nvdimm_bridge_remove,
 | |
| 	.id = CXL_DEVICE_NVDIMM_BRIDGE,
 | |
| };
 | |
| 
 | |
| static __init int cxl_pmem_init(void)
 | |
| {
 | |
| 	int rc;
 | |
| 
 | |
| 	cxl_pmem_wq = alloc_ordered_workqueue("cxl_pmem", 0);
 | |
| 	if (!cxl_pmem_wq)
 | |
| 		return -ENXIO;
 | |
| 
 | |
| 	rc = cxl_driver_register(&cxl_nvdimm_bridge_driver);
 | |
| 	if (rc)
 | |
| 		goto err_bridge;
 | |
| 
 | |
| 	rc = cxl_driver_register(&cxl_nvdimm_driver);
 | |
| 	if (rc)
 | |
| 		goto err_nvdimm;
 | |
| 
 | |
| 	return 0;
 | |
| 
 | |
| err_nvdimm:
 | |
| 	cxl_driver_unregister(&cxl_nvdimm_bridge_driver);
 | |
| err_bridge:
 | |
| 	destroy_workqueue(cxl_pmem_wq);
 | |
| 	return rc;
 | |
| }
 | |
| 
 | |
| static __exit void cxl_pmem_exit(void)
 | |
| {
 | |
| 	cxl_driver_unregister(&cxl_nvdimm_driver);
 | |
| 	cxl_driver_unregister(&cxl_nvdimm_bridge_driver);
 | |
| 	destroy_workqueue(cxl_pmem_wq);
 | |
| }
 | |
| 
 | |
| MODULE_LICENSE("GPL v2");
 | |
| module_init(cxl_pmem_init);
 | |
| module_exit(cxl_pmem_exit);
 | |
| MODULE_IMPORT_NS(CXL);
 | |
| MODULE_ALIAS_CXL(CXL_DEVICE_NVDIMM_BRIDGE);
 | |
| MODULE_ALIAS_CXL(CXL_DEVICE_NVDIMM);
 |