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