spi-dw-pci.c 3.4 KB
Newer Older
1
/*
G
Grant Likely 已提交
2
 * PCI interface driver for DW SPI Core
3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21
 *
 * Copyright (c) 2009, Intel Corporation.
 *
 * This program is free software; you can redistribute it and/or modify it
 * under the terms and conditions of the GNU General Public License,
 * version 2, as published by the Free Software Foundation.
 *
 * This program is distributed in the hope it will be useful, but WITHOUT
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
 * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License for
 * more details.
 *
 * You should have received a copy of the GNU General Public License along
 * with this program; if not, write to the Free Software Foundation,
 * Inc., 51 Franklin St - Fifth Floor, Boston, MA 02110-1301 USA.
 */

#include <linux/interrupt.h>
#include <linux/pci.h>
22
#include <linux/slab.h>
23
#include <linux/spi/spi.h>
24
#include <linux/module.h>
25

G
Grant Likely 已提交
26
#include "spi-dw.h"
27

28 29 30
#define DRIVER_NAME "dw_spi_pci"

struct dw_spi_pci {
F
Feng Tang 已提交
31 32
	struct pci_dev	*pdev;
	struct dw_spi	dws;
33 34
};

35
static int spi_pci_probe(struct pci_dev *pdev,
36 37 38 39 40 41 42
	const struct pci_device_id *ent)
{
	struct dw_spi_pci *dwpci;
	struct dw_spi *dws;
	int pci_bar = 0;
	int ret;

43
	dev_info(&pdev->dev, "found PCI SPI controller(ID: %04x:%04x)\n",
44 45
		pdev->vendor, pdev->device);

B
Baruch Siach 已提交
46
	ret = pcim_enable_device(pdev);
47 48 49
	if (ret)
		return ret;

B
Baruch Siach 已提交
50 51 52
	dwpci = devm_kzalloc(&pdev-dev, sizeof(struct dw_spi_pci), GFP_KERNEL);
	if (!dwpci)
		return -ENOMEM;
53 54 55 56 57 58 59

	dwpci->pdev = pdev;
	dws = &dwpci->dws;

	/* Get basic io resource and map it */
	dws->paddr = pci_resource_start(pdev, pci_bar);

B
Baruch Siach 已提交
60
	ret = pcim_iomap_regions(pdev, 1, dev_name(&pdev->dev));
61
	if (ret)
B
Baruch Siach 已提交
62
		return ret;
63 64 65 66

	dws->bus_num = 0;
	dws->num_cs = 4;
	dws->irq = pdev->irq;
F
Feng Tang 已提交
67 68 69 70 71 72 73 74

	/*
	 * Specific handling for Intel MID paltforms, like dma setup,
	 * clock rate, FIFO depth.
	 */
	if (pdev->device == 0x0800) {
		ret = dw_spi_mid_init(dws);
		if (ret)
B
Baruch Siach 已提交
75
			return ret;
F
Feng Tang 已提交
76
	}
77

B
Baruch Siach 已提交
78
	ret = dw_spi_add_host(&pdev->dev, dws);
79
	if (ret)
B
Baruch Siach 已提交
80
		return ret;
81 82 83 84

	/* PCI hook and SPI hook use the same drv data */
	pci_set_drvdata(pdev, dwpci);

B
Baruch Siach 已提交
85
	return 0;
86 87
}

88
static void spi_pci_remove(struct pci_dev *pdev)
89 90 91
{
	struct dw_spi_pci *dwpci = pci_get_drvdata(pdev);

92
	dw_spi_remove_host(&dwpci->dws);
93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130
	iounmap(dwpci->dws.regs);
	pci_release_region(pdev, 0);
	kfree(dwpci);
	pci_disable_device(pdev);
}

#ifdef CONFIG_PM
static int spi_suspend(struct pci_dev *pdev, pm_message_t state)
{
	struct dw_spi_pci *dwpci = pci_get_drvdata(pdev);
	int ret;

	ret = dw_spi_suspend_host(&dwpci->dws);
	if (ret)
		return ret;
	pci_save_state(pdev);
	pci_disable_device(pdev);
	pci_set_power_state(pdev, pci_choose_state(pdev, state));
	return ret;
}

static int spi_resume(struct pci_dev *pdev)
{
	struct dw_spi_pci *dwpci = pci_get_drvdata(pdev);
	int ret;

	pci_set_power_state(pdev, PCI_D0);
	pci_restore_state(pdev);
	ret = pci_enable_device(pdev);
	if (ret)
		return ret;
	return dw_spi_resume_host(&dwpci->dws);
}
#else
#define spi_suspend	NULL
#define spi_resume	NULL
#endif

131
static DEFINE_PCI_DEVICE_TABLE(pci_ids) = {
F
Feng Tang 已提交
132
	/* Intel MID platform SPI controller 0 */
133 134 135 136 137 138 139 140
	{ PCI_DEVICE(PCI_VENDOR_ID_INTEL, 0x0800) },
	{},
};

static struct pci_driver dw_spi_driver = {
	.name =		DRIVER_NAME,
	.id_table =	pci_ids,
	.probe =	spi_pci_probe,
141
	.remove =	spi_pci_remove,
142 143 144 145
	.suspend =	spi_suspend,
	.resume	=	spi_resume,
};

A
Axel Lin 已提交
146
module_pci_driver(dw_spi_driver);
147 148 149 150

MODULE_AUTHOR("Feng Tang <feng.tang@intel.com>");
MODULE_DESCRIPTION("PCI interface driver for DW SPI Core");
MODULE_LICENSE("GPL v2");