18c2ecf20Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-only
28c2ecf20Sopenharmony_ci/*
38c2ecf20Sopenharmony_ci * Intel PCH/PCU SPI flash PCI driver.
48c2ecf20Sopenharmony_ci *
58c2ecf20Sopenharmony_ci * Copyright (C) 2016, Intel Corporation
68c2ecf20Sopenharmony_ci * Author: Mika Westerberg <mika.westerberg@linux.intel.com>
78c2ecf20Sopenharmony_ci */
88c2ecf20Sopenharmony_ci
98c2ecf20Sopenharmony_ci#include <linux/ioport.h>
108c2ecf20Sopenharmony_ci#include <linux/kernel.h>
118c2ecf20Sopenharmony_ci#include <linux/module.h>
128c2ecf20Sopenharmony_ci#include <linux/pci.h>
138c2ecf20Sopenharmony_ci
148c2ecf20Sopenharmony_ci#include "intel-spi.h"
158c2ecf20Sopenharmony_ci
168c2ecf20Sopenharmony_ci#define BCR		0xdc
178c2ecf20Sopenharmony_ci#define BCR_WPD		BIT(0)
188c2ecf20Sopenharmony_ci
198c2ecf20Sopenharmony_cistatic bool intel_spi_pci_set_writeable(void __iomem *base, void *data)
208c2ecf20Sopenharmony_ci{
218c2ecf20Sopenharmony_ci	struct pci_dev *pdev = data;
228c2ecf20Sopenharmony_ci	u32 bcr;
238c2ecf20Sopenharmony_ci
248c2ecf20Sopenharmony_ci	/* Try to make the chip read/write */
258c2ecf20Sopenharmony_ci	pci_read_config_dword(pdev, BCR, &bcr);
268c2ecf20Sopenharmony_ci	if (!(bcr & BCR_WPD)) {
278c2ecf20Sopenharmony_ci		bcr |= BCR_WPD;
288c2ecf20Sopenharmony_ci		pci_write_config_dword(pdev, BCR, bcr);
298c2ecf20Sopenharmony_ci		pci_read_config_dword(pdev, BCR, &bcr);
308c2ecf20Sopenharmony_ci	}
318c2ecf20Sopenharmony_ci
328c2ecf20Sopenharmony_ci	return bcr & BCR_WPD;
338c2ecf20Sopenharmony_ci}
348c2ecf20Sopenharmony_ci
358c2ecf20Sopenharmony_cistatic const struct intel_spi_boardinfo bxt_info = {
368c2ecf20Sopenharmony_ci	.type = INTEL_SPI_BXT,
378c2ecf20Sopenharmony_ci	.set_writeable = intel_spi_pci_set_writeable,
388c2ecf20Sopenharmony_ci};
398c2ecf20Sopenharmony_ci
408c2ecf20Sopenharmony_cistatic const struct intel_spi_boardinfo cnl_info = {
418c2ecf20Sopenharmony_ci	.type = INTEL_SPI_CNL,
428c2ecf20Sopenharmony_ci	.set_writeable = intel_spi_pci_set_writeable,
438c2ecf20Sopenharmony_ci};
448c2ecf20Sopenharmony_ci
458c2ecf20Sopenharmony_cistatic int intel_spi_pci_probe(struct pci_dev *pdev,
468c2ecf20Sopenharmony_ci			       const struct pci_device_id *id)
478c2ecf20Sopenharmony_ci{
488c2ecf20Sopenharmony_ci	struct intel_spi_boardinfo *info;
498c2ecf20Sopenharmony_ci	struct intel_spi *ispi;
508c2ecf20Sopenharmony_ci	int ret;
518c2ecf20Sopenharmony_ci
528c2ecf20Sopenharmony_ci	ret = pcim_enable_device(pdev);
538c2ecf20Sopenharmony_ci	if (ret)
548c2ecf20Sopenharmony_ci		return ret;
558c2ecf20Sopenharmony_ci
568c2ecf20Sopenharmony_ci	info = devm_kmemdup(&pdev->dev, (void *)id->driver_data, sizeof(*info),
578c2ecf20Sopenharmony_ci			    GFP_KERNEL);
588c2ecf20Sopenharmony_ci	if (!info)
598c2ecf20Sopenharmony_ci		return -ENOMEM;
608c2ecf20Sopenharmony_ci
618c2ecf20Sopenharmony_ci	info->data = pdev;
628c2ecf20Sopenharmony_ci	ispi = intel_spi_probe(&pdev->dev, &pdev->resource[0], info);
638c2ecf20Sopenharmony_ci	if (IS_ERR(ispi))
648c2ecf20Sopenharmony_ci		return PTR_ERR(ispi);
658c2ecf20Sopenharmony_ci
668c2ecf20Sopenharmony_ci	pci_set_drvdata(pdev, ispi);
678c2ecf20Sopenharmony_ci	return 0;
688c2ecf20Sopenharmony_ci}
698c2ecf20Sopenharmony_ci
708c2ecf20Sopenharmony_cistatic void intel_spi_pci_remove(struct pci_dev *pdev)
718c2ecf20Sopenharmony_ci{
728c2ecf20Sopenharmony_ci	intel_spi_remove(pci_get_drvdata(pdev));
738c2ecf20Sopenharmony_ci}
748c2ecf20Sopenharmony_ci
758c2ecf20Sopenharmony_cistatic const struct pci_device_id intel_spi_pci_ids[] = {
768c2ecf20Sopenharmony_ci	{ PCI_VDEVICE(INTEL, 0x02a4), (unsigned long)&bxt_info },
778c2ecf20Sopenharmony_ci	{ PCI_VDEVICE(INTEL, 0x06a4), (unsigned long)&bxt_info },
788c2ecf20Sopenharmony_ci	{ PCI_VDEVICE(INTEL, 0x18e0), (unsigned long)&bxt_info },
798c2ecf20Sopenharmony_ci	{ PCI_VDEVICE(INTEL, 0x19e0), (unsigned long)&bxt_info },
808c2ecf20Sopenharmony_ci	{ PCI_VDEVICE(INTEL, 0x1bca), (unsigned long)&bxt_info },
818c2ecf20Sopenharmony_ci	{ PCI_VDEVICE(INTEL, 0x34a4), (unsigned long)&bxt_info },
828c2ecf20Sopenharmony_ci	{ PCI_VDEVICE(INTEL, 0x43a4), (unsigned long)&cnl_info },
838c2ecf20Sopenharmony_ci	{ PCI_VDEVICE(INTEL, 0x4b24), (unsigned long)&bxt_info },
848c2ecf20Sopenharmony_ci	{ PCI_VDEVICE(INTEL, 0x4da4), (unsigned long)&bxt_info },
858c2ecf20Sopenharmony_ci	{ PCI_VDEVICE(INTEL, 0x7aa4), (unsigned long)&cnl_info },
868c2ecf20Sopenharmony_ci	{ PCI_VDEVICE(INTEL, 0xa0a4), (unsigned long)&bxt_info },
878c2ecf20Sopenharmony_ci	{ PCI_VDEVICE(INTEL, 0xa1a4), (unsigned long)&bxt_info },
888c2ecf20Sopenharmony_ci	{ PCI_VDEVICE(INTEL, 0xa224), (unsigned long)&bxt_info },
898c2ecf20Sopenharmony_ci	{ PCI_VDEVICE(INTEL, 0xa324), (unsigned long)&cnl_info },
908c2ecf20Sopenharmony_ci	{ PCI_VDEVICE(INTEL, 0xa3a4), (unsigned long)&bxt_info },
918c2ecf20Sopenharmony_ci	{ },
928c2ecf20Sopenharmony_ci};
938c2ecf20Sopenharmony_ciMODULE_DEVICE_TABLE(pci, intel_spi_pci_ids);
948c2ecf20Sopenharmony_ci
958c2ecf20Sopenharmony_cistatic struct pci_driver intel_spi_pci_driver = {
968c2ecf20Sopenharmony_ci	.name = "intel-spi",
978c2ecf20Sopenharmony_ci	.id_table = intel_spi_pci_ids,
988c2ecf20Sopenharmony_ci	.probe = intel_spi_pci_probe,
998c2ecf20Sopenharmony_ci	.remove = intel_spi_pci_remove,
1008c2ecf20Sopenharmony_ci};
1018c2ecf20Sopenharmony_ci
1028c2ecf20Sopenharmony_cimodule_pci_driver(intel_spi_pci_driver);
1038c2ecf20Sopenharmony_ci
1048c2ecf20Sopenharmony_ciMODULE_DESCRIPTION("Intel PCH/PCU SPI flash PCI driver");
1058c2ecf20Sopenharmony_ciMODULE_AUTHOR("Mika Westerberg <mika.westerberg@linux.intel.com>");
1068c2ecf20Sopenharmony_ciMODULE_LICENSE("GPL v2");
107