16861d27cSMihai Carabas // SPDX-License-Identifier: GPL-2.0+
26861d27cSMihai Carabas /*
36861d27cSMihai Carabas  *  Pvpanic MMIO Device Support
46861d27cSMihai Carabas  *
56861d27cSMihai Carabas  *  Copyright (C) 2013 Fujitsu.
66861d27cSMihai Carabas  *  Copyright (C) 2018 ZTE.
76861d27cSMihai Carabas  *  Copyright (C) 2021 Oracle.
86861d27cSMihai Carabas  */
96861d27cSMihai Carabas 
106861d27cSMihai Carabas #include <linux/io.h>
116861d27cSMihai Carabas #include <linux/kernel.h>
126861d27cSMihai Carabas #include <linux/kexec.h>
136861d27cSMihai Carabas #include <linux/mod_devicetable.h>
146861d27cSMihai Carabas #include <linux/module.h>
156861d27cSMihai Carabas #include <linux/platform_device.h>
166861d27cSMihai Carabas #include <linux/types.h>
17b3c0f877SMihai Carabas #include <linux/slab.h>
186861d27cSMihai Carabas 
196861d27cSMihai Carabas #include <uapi/misc/pvpanic.h>
206861d27cSMihai Carabas 
216861d27cSMihai Carabas #include "pvpanic.h"
226861d27cSMihai Carabas 
236861d27cSMihai Carabas MODULE_AUTHOR("Hu Tao <hutao@cn.fujitsu.com>");
246861d27cSMihai Carabas MODULE_DESCRIPTION("pvpanic-mmio device driver");
256861d27cSMihai Carabas MODULE_LICENSE("GPL");
266861d27cSMihai Carabas 
pvpanic_mmio_probe(struct platform_device * pdev)276861d27cSMihai Carabas static int pvpanic_mmio_probe(struct platform_device *pdev)
286861d27cSMihai Carabas {
296861d27cSMihai Carabas 	struct device *dev = &pdev->dev;
306861d27cSMihai Carabas 	struct resource *res;
31b3c0f877SMihai Carabas 	void __iomem *base;
326861d27cSMihai Carabas 
336861d27cSMihai Carabas 	res = platform_get_mem_or_io(pdev, 0);
346861d27cSMihai Carabas 	if (!res)
356861d27cSMihai Carabas 		return -EINVAL;
366861d27cSMihai Carabas 
376861d27cSMihai Carabas 	switch (resource_type(res)) {
386861d27cSMihai Carabas 	case IORESOURCE_IO:
396861d27cSMihai Carabas 		base = devm_ioport_map(dev, res->start, resource_size(res));
406861d27cSMihai Carabas 		if (!base)
416861d27cSMihai Carabas 			return -ENOMEM;
426861d27cSMihai Carabas 		break;
436861d27cSMihai Carabas 	case IORESOURCE_MEM:
446861d27cSMihai Carabas 		base = devm_ioremap_resource(dev, res);
456861d27cSMihai Carabas 		if (IS_ERR(base))
466861d27cSMihai Carabas 			return PTR_ERR(base);
476861d27cSMihai Carabas 		break;
486861d27cSMihai Carabas 	default:
496861d27cSMihai Carabas 		return -EINVAL;
506861d27cSMihai Carabas 	}
516861d27cSMihai Carabas 
52e9194a95SThomas Weißschuh 	return devm_pvpanic_probe(dev, base);
536861d27cSMihai Carabas }
546861d27cSMihai Carabas 
556861d27cSMihai Carabas static const struct of_device_id pvpanic_mmio_match[] = {
566861d27cSMihai Carabas 	{ .compatible = "qemu,pvpanic-mmio", },
576861d27cSMihai Carabas 	{}
586861d27cSMihai Carabas };
596861d27cSMihai Carabas MODULE_DEVICE_TABLE(of, pvpanic_mmio_match);
606861d27cSMihai Carabas 
616861d27cSMihai Carabas static const struct acpi_device_id pvpanic_device_ids[] = {
626861d27cSMihai Carabas 	{ "QEMU0001", 0 },
636861d27cSMihai Carabas 	{ "", 0 }
646861d27cSMihai Carabas };
656861d27cSMihai Carabas MODULE_DEVICE_TABLE(acpi, pvpanic_device_ids);
666861d27cSMihai Carabas 
676861d27cSMihai Carabas static struct platform_driver pvpanic_mmio_driver = {
686861d27cSMihai Carabas 	.driver = {
696861d27cSMihai Carabas 		.name = "pvpanic-mmio",
706861d27cSMihai Carabas 		.of_match_table = pvpanic_mmio_match,
716861d27cSMihai Carabas 		.acpi_match_table = pvpanic_device_ids,
72e9194a95SThomas Weißschuh 		.dev_groups = pvpanic_dev_groups,
736861d27cSMihai Carabas 	},
746861d27cSMihai Carabas 	.probe = pvpanic_mmio_probe,
756861d27cSMihai Carabas };
766861d27cSMihai Carabas module_platform_driver(pvpanic_mmio_driver);
77