1b07079f1SVignesh Raghavendra // SPDX-License-Identifier: GPL-2.0
2b07079f1SVignesh Raghavendra //
3614a895fSAlexander A. Klimov // Copyright (C) 2019 Texas Instruments Incorporated - https://www.ti.com/
4b07079f1SVignesh Raghavendra // Author: Vignesh Raghavendra <vigneshr@ti.com>
5b07079f1SVignesh Raghavendra
600c9cf49SVignesh Raghavendra #include <linux/completion.h>
700c9cf49SVignesh Raghavendra #include <linux/dma-direction.h>
800c9cf49SVignesh Raghavendra #include <linux/dma-mapping.h>
900c9cf49SVignesh Raghavendra #include <linux/dmaengine.h>
10b07079f1SVignesh Raghavendra #include <linux/err.h>
11b07079f1SVignesh Raghavendra #include <linux/kernel.h>
12b07079f1SVignesh Raghavendra #include <linux/module.h>
13b07079f1SVignesh Raghavendra #include <linux/mtd/cfi.h>
14b07079f1SVignesh Raghavendra #include <linux/mtd/hyperbus.h>
15b07079f1SVignesh Raghavendra #include <linux/mtd/mtd.h>
16b07079f1SVignesh Raghavendra #include <linux/mux/consumer.h>
17b07079f1SVignesh Raghavendra #include <linux/of.h>
18b6fe8bc6SSergei Shtylyov #include <linux/of_address.h>
19b07079f1SVignesh Raghavendra #include <linux/platform_device.h>
2000c9cf49SVignesh Raghavendra #include <linux/sched/task_stack.h>
21b07079f1SVignesh Raghavendra #include <linux/types.h>
22b07079f1SVignesh Raghavendra
23b07079f1SVignesh Raghavendra #define AM654_HBMC_CALIB_COUNT 25
24b07079f1SVignesh Raghavendra
2500c9cf49SVignesh Raghavendra struct am654_hbmc_device_priv {
2600c9cf49SVignesh Raghavendra struct completion rx_dma_complete;
2700c9cf49SVignesh Raghavendra phys_addr_t device_base;
2800c9cf49SVignesh Raghavendra struct hyperbus_ctlr *ctlr;
2900c9cf49SVignesh Raghavendra struct dma_chan *rx_chan;
3000c9cf49SVignesh Raghavendra };
3100c9cf49SVignesh Raghavendra
32b07079f1SVignesh Raghavendra struct am654_hbmc_priv {
33b07079f1SVignesh Raghavendra struct hyperbus_ctlr ctlr;
34b07079f1SVignesh Raghavendra struct hyperbus_device hbdev;
35b07079f1SVignesh Raghavendra struct mux_control *mux_ctrl;
36b07079f1SVignesh Raghavendra };
37b07079f1SVignesh Raghavendra
am654_hbmc_calibrate(struct hyperbus_device * hbdev)38b07079f1SVignesh Raghavendra static int am654_hbmc_calibrate(struct hyperbus_device *hbdev)
39b07079f1SVignesh Raghavendra {
40b07079f1SVignesh Raghavendra struct map_info *map = &hbdev->map;
41b07079f1SVignesh Raghavendra struct cfi_private cfi;
42b07079f1SVignesh Raghavendra int count = AM654_HBMC_CALIB_COUNT;
43b07079f1SVignesh Raghavendra int pass_count = 0;
44b07079f1SVignesh Raghavendra int ret;
45b07079f1SVignesh Raghavendra
46b07079f1SVignesh Raghavendra cfi.interleave = 1;
47b07079f1SVignesh Raghavendra cfi.device_type = CFI_DEVICETYPE_X16;
48b07079f1SVignesh Raghavendra cfi_send_gen_cmd(0xF0, 0, 0, map, &cfi, cfi.device_type, NULL);
49b07079f1SVignesh Raghavendra cfi_send_gen_cmd(0x98, 0x55, 0, map, &cfi, cfi.device_type, NULL);
50b07079f1SVignesh Raghavendra
51b07079f1SVignesh Raghavendra while (count--) {
52b07079f1SVignesh Raghavendra ret = cfi_qry_present(map, 0, &cfi);
53b07079f1SVignesh Raghavendra if (ret)
54b07079f1SVignesh Raghavendra pass_count++;
55b07079f1SVignesh Raghavendra else
56b07079f1SVignesh Raghavendra pass_count = 0;
57b07079f1SVignesh Raghavendra if (pass_count == 5)
58b07079f1SVignesh Raghavendra break;
59b07079f1SVignesh Raghavendra }
60b07079f1SVignesh Raghavendra
61b07079f1SVignesh Raghavendra cfi_qry_mode_off(0, map, &cfi);
62b07079f1SVignesh Raghavendra
63b07079f1SVignesh Raghavendra return ret;
64b07079f1SVignesh Raghavendra }
65b07079f1SVignesh Raghavendra
am654_hbmc_dma_callback(void * param)6600c9cf49SVignesh Raghavendra static void am654_hbmc_dma_callback(void *param)
6700c9cf49SVignesh Raghavendra {
6800c9cf49SVignesh Raghavendra struct am654_hbmc_device_priv *priv = param;
6900c9cf49SVignesh Raghavendra
7000c9cf49SVignesh Raghavendra complete(&priv->rx_dma_complete);
7100c9cf49SVignesh Raghavendra }
7200c9cf49SVignesh Raghavendra
am654_hbmc_dma_read(struct am654_hbmc_device_priv * priv,void * to,unsigned long from,ssize_t len)7300c9cf49SVignesh Raghavendra static int am654_hbmc_dma_read(struct am654_hbmc_device_priv *priv, void *to,
7400c9cf49SVignesh Raghavendra unsigned long from, ssize_t len)
7500c9cf49SVignesh Raghavendra
7600c9cf49SVignesh Raghavendra {
7700c9cf49SVignesh Raghavendra enum dma_ctrl_flags flags = DMA_CTRL_ACK | DMA_PREP_INTERRUPT;
7800c9cf49SVignesh Raghavendra struct dma_chan *rx_chan = priv->rx_chan;
7900c9cf49SVignesh Raghavendra struct dma_async_tx_descriptor *tx;
8000c9cf49SVignesh Raghavendra dma_addr_t dma_dst, dma_src;
8100c9cf49SVignesh Raghavendra dma_cookie_t cookie;
8200c9cf49SVignesh Raghavendra int ret;
8300c9cf49SVignesh Raghavendra
8400c9cf49SVignesh Raghavendra if (!priv->rx_chan || !virt_addr_valid(to) || object_is_on_stack(to))
8500c9cf49SVignesh Raghavendra return -EINVAL;
8600c9cf49SVignesh Raghavendra
8700c9cf49SVignesh Raghavendra dma_dst = dma_map_single(rx_chan->device->dev, to, len, DMA_FROM_DEVICE);
8800c9cf49SVignesh Raghavendra if (dma_mapping_error(rx_chan->device->dev, dma_dst)) {
8900c9cf49SVignesh Raghavendra dev_dbg(priv->ctlr->dev, "DMA mapping failed\n");
9000c9cf49SVignesh Raghavendra return -EIO;
9100c9cf49SVignesh Raghavendra }
9200c9cf49SVignesh Raghavendra
9300c9cf49SVignesh Raghavendra dma_src = priv->device_base + from;
9400c9cf49SVignesh Raghavendra tx = dmaengine_prep_dma_memcpy(rx_chan, dma_dst, dma_src, len, flags);
9500c9cf49SVignesh Raghavendra if (!tx) {
9600c9cf49SVignesh Raghavendra dev_err(priv->ctlr->dev, "device_prep_dma_memcpy error\n");
9700c9cf49SVignesh Raghavendra ret = -EIO;
9800c9cf49SVignesh Raghavendra goto unmap_dma;
9900c9cf49SVignesh Raghavendra }
10000c9cf49SVignesh Raghavendra
10100c9cf49SVignesh Raghavendra reinit_completion(&priv->rx_dma_complete);
10200c9cf49SVignesh Raghavendra tx->callback = am654_hbmc_dma_callback;
10300c9cf49SVignesh Raghavendra tx->callback_param = priv;
10400c9cf49SVignesh Raghavendra cookie = dmaengine_submit(tx);
10500c9cf49SVignesh Raghavendra
10600c9cf49SVignesh Raghavendra ret = dma_submit_error(cookie);
10700c9cf49SVignesh Raghavendra if (ret) {
10800c9cf49SVignesh Raghavendra dev_err(priv->ctlr->dev, "dma_submit_error %d\n", cookie);
10900c9cf49SVignesh Raghavendra goto unmap_dma;
11000c9cf49SVignesh Raghavendra }
11100c9cf49SVignesh Raghavendra
11200c9cf49SVignesh Raghavendra dma_async_issue_pending(rx_chan);
11300c9cf49SVignesh Raghavendra if (!wait_for_completion_timeout(&priv->rx_dma_complete, msecs_to_jiffies(len + 1000))) {
11400c9cf49SVignesh Raghavendra dmaengine_terminate_sync(rx_chan);
11500c9cf49SVignesh Raghavendra dev_err(priv->ctlr->dev, "DMA wait_for_completion_timeout\n");
11600c9cf49SVignesh Raghavendra ret = -ETIMEDOUT;
11700c9cf49SVignesh Raghavendra }
11800c9cf49SVignesh Raghavendra
11900c9cf49SVignesh Raghavendra unmap_dma:
12000c9cf49SVignesh Raghavendra dma_unmap_single(rx_chan->device->dev, dma_dst, len, DMA_FROM_DEVICE);
12100c9cf49SVignesh Raghavendra return ret;
12200c9cf49SVignesh Raghavendra }
12300c9cf49SVignesh Raghavendra
am654_hbmc_read(struct hyperbus_device * hbdev,void * to,unsigned long from,ssize_t len)12400c9cf49SVignesh Raghavendra static void am654_hbmc_read(struct hyperbus_device *hbdev, void *to,
12500c9cf49SVignesh Raghavendra unsigned long from, ssize_t len)
12600c9cf49SVignesh Raghavendra {
12700c9cf49SVignesh Raghavendra struct am654_hbmc_device_priv *priv = hbdev->priv;
12800c9cf49SVignesh Raghavendra
12900c9cf49SVignesh Raghavendra if (len < SZ_1K || am654_hbmc_dma_read(priv, to, from, len))
13000c9cf49SVignesh Raghavendra memcpy_fromio(to, hbdev->map.virt + from, len);
13100c9cf49SVignesh Raghavendra }
13200c9cf49SVignesh Raghavendra
133b07079f1SVignesh Raghavendra static const struct hyperbus_ops am654_hbmc_ops = {
134b07079f1SVignesh Raghavendra .calibrate = am654_hbmc_calibrate,
13500c9cf49SVignesh Raghavendra .copy_from = am654_hbmc_read,
136b07079f1SVignesh Raghavendra };
137b07079f1SVignesh Raghavendra
am654_hbmc_request_mmap_dma(struct am654_hbmc_device_priv * priv)13800c9cf49SVignesh Raghavendra static int am654_hbmc_request_mmap_dma(struct am654_hbmc_device_priv *priv)
13900c9cf49SVignesh Raghavendra {
14000c9cf49SVignesh Raghavendra struct dma_chan *rx_chan;
14100c9cf49SVignesh Raghavendra dma_cap_mask_t mask;
14200c9cf49SVignesh Raghavendra
14300c9cf49SVignesh Raghavendra dma_cap_zero(mask);
14400c9cf49SVignesh Raghavendra dma_cap_set(DMA_MEMCPY, mask);
14500c9cf49SVignesh Raghavendra
14600c9cf49SVignesh Raghavendra rx_chan = dma_request_chan_by_mask(&mask);
14700c9cf49SVignesh Raghavendra if (IS_ERR(rx_chan)) {
14800c9cf49SVignesh Raghavendra if (PTR_ERR(rx_chan) == -EPROBE_DEFER)
14900c9cf49SVignesh Raghavendra return -EPROBE_DEFER;
15000c9cf49SVignesh Raghavendra dev_dbg(priv->ctlr->dev, "No DMA channel available\n");
15100c9cf49SVignesh Raghavendra return 0;
15200c9cf49SVignesh Raghavendra }
15300c9cf49SVignesh Raghavendra priv->rx_chan = rx_chan;
15400c9cf49SVignesh Raghavendra init_completion(&priv->rx_dma_complete);
15500c9cf49SVignesh Raghavendra
15600c9cf49SVignesh Raghavendra return 0;
15700c9cf49SVignesh Raghavendra }
15800c9cf49SVignesh Raghavendra
am654_hbmc_probe(struct platform_device * pdev)159b07079f1SVignesh Raghavendra static int am654_hbmc_probe(struct platform_device *pdev)
160b07079f1SVignesh Raghavendra {
161b6fe8bc6SSergei Shtylyov struct device_node *np = pdev->dev.of_node;
16200c9cf49SVignesh Raghavendra struct am654_hbmc_device_priv *dev_priv;
163b07079f1SVignesh Raghavendra struct device *dev = &pdev->dev;
164b07079f1SVignesh Raghavendra struct am654_hbmc_priv *priv;
165b6fe8bc6SSergei Shtylyov struct resource res;
166b07079f1SVignesh Raghavendra int ret;
167b07079f1SVignesh Raghavendra
168b07079f1SVignesh Raghavendra priv = devm_kzalloc(dev, sizeof(*priv), GFP_KERNEL);
169b07079f1SVignesh Raghavendra if (!priv)
170b07079f1SVignesh Raghavendra return -ENOMEM;
171b07079f1SVignesh Raghavendra
172b07079f1SVignesh Raghavendra platform_set_drvdata(pdev, priv);
173b07079f1SVignesh Raghavendra
174aca31ce9SVignesh Raghavendra priv->hbdev.np = of_get_next_child(np, NULL);
175aca31ce9SVignesh Raghavendra ret = of_address_to_resource(priv->hbdev.np, 0, &res);
176b6fe8bc6SSergei Shtylyov if (ret)
177b6fe8bc6SSergei Shtylyov return ret;
178b6fe8bc6SSergei Shtylyov
179b07079f1SVignesh Raghavendra if (of_property_read_bool(dev->of_node, "mux-controls")) {
180b07079f1SVignesh Raghavendra struct mux_control *control = devm_mux_control_get(dev, NULL);
181b07079f1SVignesh Raghavendra
182b07079f1SVignesh Raghavendra if (IS_ERR(control))
183b07079f1SVignesh Raghavendra return PTR_ERR(control);
184b07079f1SVignesh Raghavendra
185b07079f1SVignesh Raghavendra ret = mux_control_select(control, 1);
186b07079f1SVignesh Raghavendra if (ret) {
187b07079f1SVignesh Raghavendra dev_err(dev, "Failed to select HBMC mux\n");
188b07079f1SVignesh Raghavendra return ret;
189b07079f1SVignesh Raghavendra }
190b07079f1SVignesh Raghavendra priv->mux_ctrl = control;
191b07079f1SVignesh Raghavendra }
192b07079f1SVignesh Raghavendra
193b6fe8bc6SSergei Shtylyov priv->hbdev.map.size = resource_size(&res);
194b6fe8bc6SSergei Shtylyov priv->hbdev.map.virt = devm_ioremap_resource(dev, &res);
195b6fe8bc6SSergei Shtylyov if (IS_ERR(priv->hbdev.map.virt))
196b6fe8bc6SSergei Shtylyov return PTR_ERR(priv->hbdev.map.virt);
197b6fe8bc6SSergei Shtylyov
198b07079f1SVignesh Raghavendra priv->ctlr.dev = dev;
199b07079f1SVignesh Raghavendra priv->ctlr.ops = &am654_hbmc_ops;
200b07079f1SVignesh Raghavendra priv->hbdev.ctlr = &priv->ctlr;
20100c9cf49SVignesh Raghavendra
20200c9cf49SVignesh Raghavendra dev_priv = devm_kzalloc(dev, sizeof(*dev_priv), GFP_KERNEL);
20300c9cf49SVignesh Raghavendra if (!dev_priv) {
20400c9cf49SVignesh Raghavendra ret = -ENOMEM;
205992df3bbSVignesh Raghavendra goto disable_mux;
206b07079f1SVignesh Raghavendra }
207b07079f1SVignesh Raghavendra
20800c9cf49SVignesh Raghavendra priv->hbdev.priv = dev_priv;
20900c9cf49SVignesh Raghavendra dev_priv->device_base = res.start;
21000c9cf49SVignesh Raghavendra dev_priv->ctlr = &priv->ctlr;
21100c9cf49SVignesh Raghavendra
21200c9cf49SVignesh Raghavendra ret = am654_hbmc_request_mmap_dma(dev_priv);
21300c9cf49SVignesh Raghavendra if (ret)
21400c9cf49SVignesh Raghavendra goto disable_mux;
21500c9cf49SVignesh Raghavendra
21600c9cf49SVignesh Raghavendra ret = hyperbus_register_device(&priv->hbdev);
21700c9cf49SVignesh Raghavendra if (ret) {
21800c9cf49SVignesh Raghavendra dev_err(dev, "failed to register controller\n");
21900c9cf49SVignesh Raghavendra goto release_dma;
22000c9cf49SVignesh Raghavendra }
22100c9cf49SVignesh Raghavendra
222b07079f1SVignesh Raghavendra return 0;
22300c9cf49SVignesh Raghavendra release_dma:
22400c9cf49SVignesh Raghavendra if (dev_priv->rx_chan)
22500c9cf49SVignesh Raghavendra dma_release_channel(dev_priv->rx_chan);
226992df3bbSVignesh Raghavendra disable_mux:
227b07079f1SVignesh Raghavendra if (priv->mux_ctrl)
228b07079f1SVignesh Raghavendra mux_control_deselect(priv->mux_ctrl);
229b07079f1SVignesh Raghavendra return ret;
230b07079f1SVignesh Raghavendra }
231b07079f1SVignesh Raghavendra
am654_hbmc_remove(struct platform_device * pdev)232b07079f1SVignesh Raghavendra static int am654_hbmc_remove(struct platform_device *pdev)
233b07079f1SVignesh Raghavendra {
234b07079f1SVignesh Raghavendra struct am654_hbmc_priv *priv = platform_get_drvdata(pdev);
23500c9cf49SVignesh Raghavendra struct am654_hbmc_device_priv *dev_priv = priv->hbdev.priv;
236b07079f1SVignesh Raghavendra
237*0c90466aSUwe Kleine-König hyperbus_unregister_device(&priv->hbdev);
238*0c90466aSUwe Kleine-König
239b07079f1SVignesh Raghavendra if (priv->mux_ctrl)
240b07079f1SVignesh Raghavendra mux_control_deselect(priv->mux_ctrl);
241b07079f1SVignesh Raghavendra
24200c9cf49SVignesh Raghavendra if (dev_priv->rx_chan)
24300c9cf49SVignesh Raghavendra dma_release_channel(dev_priv->rx_chan);
24400c9cf49SVignesh Raghavendra
245*0c90466aSUwe Kleine-König return 0;
246b07079f1SVignesh Raghavendra }
247b07079f1SVignesh Raghavendra
248b07079f1SVignesh Raghavendra static const struct of_device_id am654_hbmc_dt_ids[] = {
249b07079f1SVignesh Raghavendra {
250b07079f1SVignesh Raghavendra .compatible = "ti,am654-hbmc",
251b07079f1SVignesh Raghavendra },
252b07079f1SVignesh Raghavendra { /* end of table */ }
253b07079f1SVignesh Raghavendra };
254b07079f1SVignesh Raghavendra
255b07079f1SVignesh Raghavendra MODULE_DEVICE_TABLE(of, am654_hbmc_dt_ids);
256b07079f1SVignesh Raghavendra
257b07079f1SVignesh Raghavendra static struct platform_driver am654_hbmc_platform_driver = {
258b07079f1SVignesh Raghavendra .probe = am654_hbmc_probe,
259b07079f1SVignesh Raghavendra .remove = am654_hbmc_remove,
260b07079f1SVignesh Raghavendra .driver = {
261b07079f1SVignesh Raghavendra .name = "hbmc-am654",
262b07079f1SVignesh Raghavendra .of_match_table = am654_hbmc_dt_ids,
263b07079f1SVignesh Raghavendra },
264b07079f1SVignesh Raghavendra };
265b07079f1SVignesh Raghavendra
266b07079f1SVignesh Raghavendra module_platform_driver(am654_hbmc_platform_driver);
267b07079f1SVignesh Raghavendra
268b07079f1SVignesh Raghavendra MODULE_DESCRIPTION("HBMC driver for AM654 SoC");
269b07079f1SVignesh Raghavendra MODULE_LICENSE("GPL v2");
270b07079f1SVignesh Raghavendra MODULE_ALIAS("platform:hbmc-am654");
271b07079f1SVignesh Raghavendra MODULE_AUTHOR("Vignesh Raghavendra <vigneshr@ti.com>");
272