18c2ecf20Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0
28c2ecf20Sopenharmony_ci/*
38c2ecf20Sopenharmony_ci * PCI driver for the Intel SCU.
48c2ecf20Sopenharmony_ci *
58c2ecf20Sopenharmony_ci * Copyright (C) 2008-2010, 2015, 2020 Intel Corporation
68c2ecf20Sopenharmony_ci * Authors: Sreedhara DS (sreedhara.ds@intel.com)
78c2ecf20Sopenharmony_ci *	    Mika Westerberg <mika.westerberg@linux.intel.com>
88c2ecf20Sopenharmony_ci */
98c2ecf20Sopenharmony_ci
108c2ecf20Sopenharmony_ci#include <linux/errno.h>
118c2ecf20Sopenharmony_ci#include <linux/init.h>
128c2ecf20Sopenharmony_ci#include <linux/pci.h>
138c2ecf20Sopenharmony_ci
148c2ecf20Sopenharmony_ci#include <asm/intel-mid.h>
158c2ecf20Sopenharmony_ci#include <asm/intel_scu_ipc.h>
168c2ecf20Sopenharmony_ci
178c2ecf20Sopenharmony_cistatic int intel_scu_pci_probe(struct pci_dev *pdev,
188c2ecf20Sopenharmony_ci			       const struct pci_device_id *id)
198c2ecf20Sopenharmony_ci{
208c2ecf20Sopenharmony_ci	void (*setup_fn)(void) = (void (*)(void))id->driver_data;
218c2ecf20Sopenharmony_ci	struct intel_scu_ipc_data scu_data = {};
228c2ecf20Sopenharmony_ci	struct intel_scu_ipc_dev *scu;
238c2ecf20Sopenharmony_ci	int ret;
248c2ecf20Sopenharmony_ci
258c2ecf20Sopenharmony_ci	ret = pcim_enable_device(pdev);
268c2ecf20Sopenharmony_ci	if (ret)
278c2ecf20Sopenharmony_ci		return ret;
288c2ecf20Sopenharmony_ci
298c2ecf20Sopenharmony_ci	scu_data.mem = pdev->resource[0];
308c2ecf20Sopenharmony_ci	scu_data.irq = pdev->irq;
318c2ecf20Sopenharmony_ci
328c2ecf20Sopenharmony_ci	scu = intel_scu_ipc_register(&pdev->dev, &scu_data);
338c2ecf20Sopenharmony_ci	if (IS_ERR(scu))
348c2ecf20Sopenharmony_ci		return PTR_ERR(scu);
358c2ecf20Sopenharmony_ci
368c2ecf20Sopenharmony_ci	if (setup_fn)
378c2ecf20Sopenharmony_ci		setup_fn();
388c2ecf20Sopenharmony_ci	return 0;
398c2ecf20Sopenharmony_ci}
408c2ecf20Sopenharmony_ci
418c2ecf20Sopenharmony_cistatic void intel_mid_scu_setup(void)
428c2ecf20Sopenharmony_ci{
438c2ecf20Sopenharmony_ci	intel_scu_devices_create();
448c2ecf20Sopenharmony_ci}
458c2ecf20Sopenharmony_ci
468c2ecf20Sopenharmony_cistatic const struct pci_device_id pci_ids[] = {
478c2ecf20Sopenharmony_ci	{ PCI_VDEVICE(INTEL, 0x080e),
488c2ecf20Sopenharmony_ci	  .driver_data = (kernel_ulong_t)intel_mid_scu_setup },
498c2ecf20Sopenharmony_ci	{ PCI_VDEVICE(INTEL, 0x08ea),
508c2ecf20Sopenharmony_ci	  .driver_data = (kernel_ulong_t)intel_mid_scu_setup },
518c2ecf20Sopenharmony_ci	{ PCI_VDEVICE(INTEL, 0x0a94) },
528c2ecf20Sopenharmony_ci	{ PCI_VDEVICE(INTEL, 0x11a0),
538c2ecf20Sopenharmony_ci	  .driver_data = (kernel_ulong_t)intel_mid_scu_setup },
548c2ecf20Sopenharmony_ci	{ PCI_VDEVICE(INTEL, 0x1a94) },
558c2ecf20Sopenharmony_ci	{ PCI_VDEVICE(INTEL, 0x5a94) },
568c2ecf20Sopenharmony_ci	{}
578c2ecf20Sopenharmony_ci};
588c2ecf20Sopenharmony_ci
598c2ecf20Sopenharmony_cistatic struct pci_driver intel_scu_pci_driver = {
608c2ecf20Sopenharmony_ci	.driver = {
618c2ecf20Sopenharmony_ci		.suppress_bind_attrs = true,
628c2ecf20Sopenharmony_ci	},
638c2ecf20Sopenharmony_ci	.name = "intel_scu",
648c2ecf20Sopenharmony_ci	.id_table = pci_ids,
658c2ecf20Sopenharmony_ci	.probe = intel_scu_pci_probe,
668c2ecf20Sopenharmony_ci};
678c2ecf20Sopenharmony_ci
688c2ecf20Sopenharmony_cibuiltin_pci_driver(intel_scu_pci_driver);
69