1 // SPDX-License-Identifier: GPL-2.0-or-later
3 * OpenFirmware bindings for the MMC-over-SPI driver
5 * Copyright (c) MontaVista Software, Inc. 2008.
7 * Author: Anton Vorontsov <avorontsov@ru.mvista.com>
10 #include <linux/kernel.h>
11 #include <linux/module.h>
12 #include <linux/device.h>
13 #include <linux/slab.h>
14 #include <linux/irq.h>
16 #include <linux/of_irq.h>
17 #include <linux/spi/spi.h>
18 #include <linux/spi/mmc_spi.h>
19 #include <linux/mmc/core.h>
20 #include <linux/mmc/host.h>
22 MODULE_LICENSE("GPL");
25 struct mmc_spi_platform_data pdata
;
29 static struct of_mmc_spi
*to_of_mmc_spi(struct device
*dev
)
31 return container_of(dev
->platform_data
, struct of_mmc_spi
, pdata
);
34 static int of_mmc_spi_init(struct device
*dev
,
35 irqreturn_t (*irqhandler
)(int, void *), void *mmc
)
37 struct of_mmc_spi
*oms
= to_of_mmc_spi(dev
);
39 return request_threaded_irq(oms
->detect_irq
, NULL
, irqhandler
,
40 IRQF_ONESHOT
, dev_name(dev
), mmc
);
43 static void of_mmc_spi_exit(struct device
*dev
, void *mmc
)
45 struct of_mmc_spi
*oms
= to_of_mmc_spi(dev
);
47 free_irq(oms
->detect_irq
, mmc
);
50 struct mmc_spi_platform_data
*mmc_spi_get_pdata(struct spi_device
*spi
)
52 struct mmc_host
*mmc
= dev_get_drvdata(&spi
->dev
);
53 struct device
*dev
= &spi
->dev
;
54 struct of_mmc_spi
*oms
;
56 if (dev
->platform_data
|| !dev_fwnode(dev
))
57 return dev
->platform_data
;
59 oms
= kzalloc(sizeof(*oms
), GFP_KERNEL
);
63 if (mmc_of_parse_voltage(mmc
, &oms
->pdata
.ocr_mask
) < 0)
66 oms
->detect_irq
= spi
->irq
;
67 if (oms
->detect_irq
> 0) {
68 oms
->pdata
.init
= of_mmc_spi_init
;
69 oms
->pdata
.exit
= of_mmc_spi_exit
;
71 oms
->pdata
.caps
|= MMC_CAP_NEEDS_POLL
;
74 dev
->platform_data
= &oms
->pdata
;
75 return dev
->platform_data
;
80 EXPORT_SYMBOL(mmc_spi_get_pdata
);
82 void mmc_spi_put_pdata(struct spi_device
*spi
)
84 struct device
*dev
= &spi
->dev
;
85 struct of_mmc_spi
*oms
= to_of_mmc_spi(dev
);
87 if (!dev
->platform_data
|| !dev_fwnode(dev
))
91 dev
->platform_data
= NULL
;
93 EXPORT_SYMBOL(mmc_spi_put_pdata
);