2 * Intel Quark MFD PCI driver for I2C & GPIO
4 * Copyright(c) 2014 Intel Corporation.
6 * This program is free software; you can redistribute it and/or modify it
7 * under the terms and conditions of the GNU General Public License,
8 * version 2, as published by the Free Software Foundation.
10 * This program is distributed in the hope it will be useful, but WITHOUT
11 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
12 * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
15 * Intel Quark PCI device for I2C and GPIO controller sharing the same
16 * PCI function. This PCI driver will split the 2 devices into their
20 #include <linux/kernel.h>
21 #include <linux/module.h>
22 #include <linux/pci.h>
23 #include <linux/mfd/core.h>
24 #include <linux/clkdev.h>
25 #include <linux/clk-provider.h>
26 #include <linux/dmi.h>
27 #include <linux/platform_data/gpio-dwapb.h>
28 #include <linux/platform_data/i2c-designware.h>
30 /* PCI BAR for register base address */
32 #define MFD_GPIO_BAR 1
34 /* The base GPIO number under GPIOLIB framework */
35 #define INTEL_QUARK_MFD_GPIO_BASE 8
37 /* The default number of South-Cluster GPIO on Quark. */
38 #define INTEL_QUARK_MFD_NGPIO 8
40 /* The DesignWare GPIO ports on Quark. */
41 #define INTEL_QUARK_GPIO_NPORTS 1
43 #define INTEL_QUARK_IORES_MEM 0
44 #define INTEL_QUARK_IORES_IRQ 1
46 #define INTEL_QUARK_I2C_CONTROLLER_CLK "i2c_designware.0"
48 /* The Quark I2C controller source clock */
49 #define INTEL_QUARK_I2C_CLK_HZ 33000000
51 #define INTEL_QUARK_I2C_NCLK 1
53 struct intel_quark_mfd
{
56 struct clk_lookup
*i2c_clk_lookup
;
59 struct i2c_mode_info
{
61 unsigned int i2c_scl_freq
;
64 static const struct i2c_mode_info platform_i2c_mode_info
[] = {
67 .i2c_scl_freq
= 100000,
70 .name
= "GalileoGen2",
71 .i2c_scl_freq
= 400000,
76 static struct resource intel_quark_i2c_res
[] = {
77 [INTEL_QUARK_IORES_MEM
] = {
78 .flags
= IORESOURCE_MEM
,
80 [INTEL_QUARK_IORES_IRQ
] = {
81 .flags
= IORESOURCE_IRQ
,
85 static struct resource intel_quark_gpio_res
[] = {
86 [INTEL_QUARK_IORES_MEM
] = {
87 .flags
= IORESOURCE_MEM
,
91 static struct mfd_cell intel_quark_mfd_cells
[] = {
94 .name
= "i2c_designware",
95 .num_resources
= ARRAY_SIZE(intel_quark_i2c_res
),
96 .resources
= intel_quark_i2c_res
,
97 .ignore_resource_conflicts
= true,
101 .name
= "gpio-dwapb",
102 .num_resources
= ARRAY_SIZE(intel_quark_gpio_res
),
103 .resources
= intel_quark_gpio_res
,
104 .ignore_resource_conflicts
= true,
108 static const struct pci_device_id intel_quark_mfd_ids
[] = {
109 { PCI_VDEVICE(INTEL
, 0x0934), },
112 MODULE_DEVICE_TABLE(pci
, intel_quark_mfd_ids
);
114 static int intel_quark_register_i2c_clk(struct intel_quark_mfd
*quark_mfd
)
116 struct pci_dev
*pdev
= quark_mfd
->pdev
;
117 struct clk_lookup
*i2c_clk_lookup
;
121 i2c_clk_lookup
= devm_kcalloc(&pdev
->dev
, INTEL_QUARK_I2C_NCLK
,
122 sizeof(*i2c_clk_lookup
), GFP_KERNEL
);
126 i2c_clk_lookup
[0].dev_id
= INTEL_QUARK_I2C_CONTROLLER_CLK
;
128 i2c_clk
= clk_register_fixed_rate(&pdev
->dev
,
129 INTEL_QUARK_I2C_CONTROLLER_CLK
, NULL
,
130 CLK_IS_ROOT
, INTEL_QUARK_I2C_CLK_HZ
);
132 quark_mfd
->i2c_clk_lookup
= i2c_clk_lookup
;
133 quark_mfd
->i2c_clk
= i2c_clk
;
135 ret
= clk_register_clkdevs(i2c_clk
, i2c_clk_lookup
,
136 INTEL_QUARK_I2C_NCLK
);
138 dev_err(&pdev
->dev
, "Fixed clk register failed: %d\n", ret
);
143 static void intel_quark_unregister_i2c_clk(struct pci_dev
*pdev
)
145 struct intel_quark_mfd
*quark_mfd
= dev_get_drvdata(&pdev
->dev
);
147 if (!quark_mfd
->i2c_clk
|| !quark_mfd
->i2c_clk_lookup
)
150 clkdev_drop(quark_mfd
->i2c_clk_lookup
);
151 clk_unregister(quark_mfd
->i2c_clk
);
154 static int intel_quark_i2c_setup(struct pci_dev
*pdev
, struct mfd_cell
*cell
)
156 const char *board_name
= dmi_get_system_info(DMI_BOARD_NAME
);
157 const struct i2c_mode_info
*info
;
158 struct dw_i2c_platform_data
*pdata
;
159 struct resource
*res
= (struct resource
*)cell
->resources
;
160 struct device
*dev
= &pdev
->dev
;
162 res
[INTEL_QUARK_IORES_MEM
].start
=
163 pci_resource_start(pdev
, MFD_I2C_BAR
);
164 res
[INTEL_QUARK_IORES_MEM
].end
=
165 pci_resource_end(pdev
, MFD_I2C_BAR
);
167 res
[INTEL_QUARK_IORES_IRQ
].start
= pdev
->irq
;
168 res
[INTEL_QUARK_IORES_IRQ
].end
= pdev
->irq
;
170 pdata
= devm_kzalloc(dev
, sizeof(*pdata
), GFP_KERNEL
);
174 /* Normal mode by default */
175 pdata
->i2c_scl_freq
= 100000;
178 for (info
= platform_i2c_mode_info
; info
->name
; info
++) {
179 if (!strcmp(board_name
, info
->name
)) {
180 pdata
->i2c_scl_freq
= info
->i2c_scl_freq
;
186 cell
->platform_data
= pdata
;
187 cell
->pdata_size
= sizeof(*pdata
);
192 static int intel_quark_gpio_setup(struct pci_dev
*pdev
, struct mfd_cell
*cell
)
194 struct dwapb_platform_data
*pdata
;
195 struct resource
*res
= (struct resource
*)cell
->resources
;
196 struct device
*dev
= &pdev
->dev
;
198 res
[INTEL_QUARK_IORES_MEM
].start
=
199 pci_resource_start(pdev
, MFD_GPIO_BAR
);
200 res
[INTEL_QUARK_IORES_MEM
].end
=
201 pci_resource_end(pdev
, MFD_GPIO_BAR
);
203 pdata
= devm_kzalloc(dev
, sizeof(*pdata
), GFP_KERNEL
);
207 /* For intel quark x1000, it has only one port: portA */
208 pdata
->nports
= INTEL_QUARK_GPIO_NPORTS
;
209 pdata
->properties
= devm_kcalloc(dev
, pdata
->nports
,
210 sizeof(*pdata
->properties
),
212 if (!pdata
->properties
)
215 /* Set the properties for portA */
216 pdata
->properties
->node
= NULL
;
217 pdata
->properties
->name
= "intel-quark-x1000-gpio-portA";
218 pdata
->properties
->idx
= 0;
219 pdata
->properties
->ngpio
= INTEL_QUARK_MFD_NGPIO
;
220 pdata
->properties
->gpio_base
= INTEL_QUARK_MFD_GPIO_BASE
;
221 pdata
->properties
->irq
= pdev
->irq
;
222 pdata
->properties
->irq_shared
= true;
224 cell
->platform_data
= pdata
;
225 cell
->pdata_size
= sizeof(*pdata
);
230 static int intel_quark_mfd_probe(struct pci_dev
*pdev
,
231 const struct pci_device_id
*id
)
233 struct intel_quark_mfd
*quark_mfd
;
236 ret
= pcim_enable_device(pdev
);
240 quark_mfd
= devm_kzalloc(&pdev
->dev
, sizeof(*quark_mfd
), GFP_KERNEL
);
243 quark_mfd
->pdev
= pdev
;
245 ret
= intel_quark_register_i2c_clk(quark_mfd
);
249 dev_set_drvdata(&pdev
->dev
, quark_mfd
);
251 ret
= intel_quark_i2c_setup(pdev
, &intel_quark_mfd_cells
[MFD_I2C_BAR
]);
255 ret
= intel_quark_gpio_setup(pdev
,
256 &intel_quark_mfd_cells
[MFD_GPIO_BAR
]);
260 return mfd_add_devices(&pdev
->dev
, 0, intel_quark_mfd_cells
,
261 ARRAY_SIZE(intel_quark_mfd_cells
), NULL
, 0,
265 static void intel_quark_mfd_remove(struct pci_dev
*pdev
)
267 intel_quark_unregister_i2c_clk(pdev
);
268 mfd_remove_devices(&pdev
->dev
);
271 static struct pci_driver intel_quark_mfd_driver
= {
272 .name
= "intel_quark_mfd_i2c_gpio",
273 .id_table
= intel_quark_mfd_ids
,
274 .probe
= intel_quark_mfd_probe
,
275 .remove
= intel_quark_mfd_remove
,
278 module_pci_driver(intel_quark_mfd_driver
);
280 MODULE_AUTHOR("Raymond Tan <raymond.tan@intel.com>");
281 MODULE_DESCRIPTION("Intel Quark MFD PCI driver for I2C & GPIO");
282 MODULE_LICENSE("GPL v2");