162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-only
262306a36Sopenharmony_ci/*
362306a36Sopenharmony_ci * The driver for Freescale MPC512x LocalPlus Bus FIFO
462306a36Sopenharmony_ci * (called SCLPC in the Reference Manual).
562306a36Sopenharmony_ci *
662306a36Sopenharmony_ci * Copyright (C) 2013-2015 Alexander Popov <alex.popov@linux.com>.
762306a36Sopenharmony_ci */
862306a36Sopenharmony_ci
962306a36Sopenharmony_ci#include <linux/interrupt.h>
1062306a36Sopenharmony_ci#include <linux/kernel.h>
1162306a36Sopenharmony_ci#include <linux/module.h>
1262306a36Sopenharmony_ci#include <linux/of.h>
1362306a36Sopenharmony_ci#include <linux/of_address.h>
1462306a36Sopenharmony_ci#include <linux/of_irq.h>
1562306a36Sopenharmony_ci#include <linux/platform_device.h>
1662306a36Sopenharmony_ci#include <asm/mpc5121.h>
1762306a36Sopenharmony_ci#include <asm/io.h>
1862306a36Sopenharmony_ci#include <linux/spinlock.h>
1962306a36Sopenharmony_ci#include <linux/slab.h>
2062306a36Sopenharmony_ci#include <linux/dmaengine.h>
2162306a36Sopenharmony_ci#include <linux/dma-direction.h>
2262306a36Sopenharmony_ci#include <linux/dma-mapping.h>
2362306a36Sopenharmony_ci
2462306a36Sopenharmony_ci#define DRV_NAME "mpc512x_lpbfifo"
2562306a36Sopenharmony_ci
2662306a36Sopenharmony_cistruct cs_range {
2762306a36Sopenharmony_ci	u32 csnum;
2862306a36Sopenharmony_ci	u32 base; /* must be zero */
2962306a36Sopenharmony_ci	u32 addr;
3062306a36Sopenharmony_ci	u32 size;
3162306a36Sopenharmony_ci};
3262306a36Sopenharmony_ci
3362306a36Sopenharmony_cistatic struct lpbfifo_data {
3462306a36Sopenharmony_ci	spinlock_t lock; /* for protecting lpbfifo_data */
3562306a36Sopenharmony_ci	phys_addr_t regs_phys;
3662306a36Sopenharmony_ci	resource_size_t regs_size;
3762306a36Sopenharmony_ci	struct mpc512x_lpbfifo __iomem *regs;
3862306a36Sopenharmony_ci	int irq;
3962306a36Sopenharmony_ci	struct cs_range *cs_ranges;
4062306a36Sopenharmony_ci	size_t cs_n;
4162306a36Sopenharmony_ci	struct dma_chan *chan;
4262306a36Sopenharmony_ci	struct mpc512x_lpbfifo_request *req;
4362306a36Sopenharmony_ci	dma_addr_t ram_bus_addr;
4462306a36Sopenharmony_ci	bool wait_lpbfifo_irq;
4562306a36Sopenharmony_ci	bool wait_lpbfifo_callback;
4662306a36Sopenharmony_ci} lpbfifo;
4762306a36Sopenharmony_ci
4862306a36Sopenharmony_ci/*
4962306a36Sopenharmony_ci * A data transfer from RAM to some device on LPB is finished
5062306a36Sopenharmony_ci * when both mpc512x_lpbfifo_irq() and mpc512x_lpbfifo_callback()
5162306a36Sopenharmony_ci * have been called. We execute the callback registered in
5262306a36Sopenharmony_ci * mpc512x_lpbfifo_request just after that.
5362306a36Sopenharmony_ci * But for a data transfer from some device on LPB to RAM we don't enable
5462306a36Sopenharmony_ci * LPBFIFO interrupt because clearing MPC512X_SCLPC_SUCCESS interrupt flag
5562306a36Sopenharmony_ci * automatically disables LPBFIFO reading request to the DMA controller
5662306a36Sopenharmony_ci * and the data transfer hangs. So the callback registered in
5762306a36Sopenharmony_ci * mpc512x_lpbfifo_request is executed at the end of mpc512x_lpbfifo_callback().
5862306a36Sopenharmony_ci */
5962306a36Sopenharmony_ci
6062306a36Sopenharmony_ci/*
6162306a36Sopenharmony_ci * mpc512x_lpbfifo_irq - IRQ handler for LPB FIFO
6262306a36Sopenharmony_ci */
6362306a36Sopenharmony_cistatic irqreturn_t mpc512x_lpbfifo_irq(int irq, void *param)
6462306a36Sopenharmony_ci{
6562306a36Sopenharmony_ci	struct device *dev = (struct device *)param;
6662306a36Sopenharmony_ci	struct mpc512x_lpbfifo_request *req = NULL;
6762306a36Sopenharmony_ci	unsigned long flags;
6862306a36Sopenharmony_ci	u32 status;
6962306a36Sopenharmony_ci
7062306a36Sopenharmony_ci	spin_lock_irqsave(&lpbfifo.lock, flags);
7162306a36Sopenharmony_ci
7262306a36Sopenharmony_ci	if (!lpbfifo.regs)
7362306a36Sopenharmony_ci		goto end;
7462306a36Sopenharmony_ci
7562306a36Sopenharmony_ci	req = lpbfifo.req;
7662306a36Sopenharmony_ci	if (!req || req->dir == MPC512X_LPBFIFO_REQ_DIR_READ) {
7762306a36Sopenharmony_ci		dev_err(dev, "bogus LPBFIFO IRQ\n");
7862306a36Sopenharmony_ci		goto end;
7962306a36Sopenharmony_ci	}
8062306a36Sopenharmony_ci
8162306a36Sopenharmony_ci	status = in_be32(&lpbfifo.regs->status);
8262306a36Sopenharmony_ci	if (status != MPC512X_SCLPC_SUCCESS) {
8362306a36Sopenharmony_ci		dev_err(dev, "DMA transfer from RAM to peripheral failed\n");
8462306a36Sopenharmony_ci		out_be32(&lpbfifo.regs->enable,
8562306a36Sopenharmony_ci				MPC512X_SCLPC_RESET | MPC512X_SCLPC_FIFO_RESET);
8662306a36Sopenharmony_ci		goto end;
8762306a36Sopenharmony_ci	}
8862306a36Sopenharmony_ci	/* Clear the interrupt flag */
8962306a36Sopenharmony_ci	out_be32(&lpbfifo.regs->status, MPC512X_SCLPC_SUCCESS);
9062306a36Sopenharmony_ci
9162306a36Sopenharmony_ci	lpbfifo.wait_lpbfifo_irq = false;
9262306a36Sopenharmony_ci
9362306a36Sopenharmony_ci	if (lpbfifo.wait_lpbfifo_callback)
9462306a36Sopenharmony_ci		goto end;
9562306a36Sopenharmony_ci
9662306a36Sopenharmony_ci	/* Transfer is finished, set the FIFO as idle */
9762306a36Sopenharmony_ci	lpbfifo.req = NULL;
9862306a36Sopenharmony_ci
9962306a36Sopenharmony_ci	spin_unlock_irqrestore(&lpbfifo.lock, flags);
10062306a36Sopenharmony_ci
10162306a36Sopenharmony_ci	if (req->callback)
10262306a36Sopenharmony_ci		req->callback(req);
10362306a36Sopenharmony_ci
10462306a36Sopenharmony_ci	return IRQ_HANDLED;
10562306a36Sopenharmony_ci
10662306a36Sopenharmony_ci end:
10762306a36Sopenharmony_ci	spin_unlock_irqrestore(&lpbfifo.lock, flags);
10862306a36Sopenharmony_ci	return IRQ_HANDLED;
10962306a36Sopenharmony_ci}
11062306a36Sopenharmony_ci
11162306a36Sopenharmony_ci/*
11262306a36Sopenharmony_ci * mpc512x_lpbfifo_callback is called by DMA driver when
11362306a36Sopenharmony_ci * DMA transaction is finished.
11462306a36Sopenharmony_ci */
11562306a36Sopenharmony_cistatic void mpc512x_lpbfifo_callback(void *param)
11662306a36Sopenharmony_ci{
11762306a36Sopenharmony_ci	unsigned long flags;
11862306a36Sopenharmony_ci	struct mpc512x_lpbfifo_request *req = NULL;
11962306a36Sopenharmony_ci	enum dma_data_direction dir;
12062306a36Sopenharmony_ci
12162306a36Sopenharmony_ci	spin_lock_irqsave(&lpbfifo.lock, flags);
12262306a36Sopenharmony_ci
12362306a36Sopenharmony_ci	if (!lpbfifo.regs) {
12462306a36Sopenharmony_ci		spin_unlock_irqrestore(&lpbfifo.lock, flags);
12562306a36Sopenharmony_ci		return;
12662306a36Sopenharmony_ci	}
12762306a36Sopenharmony_ci
12862306a36Sopenharmony_ci	req = lpbfifo.req;
12962306a36Sopenharmony_ci	if (!req) {
13062306a36Sopenharmony_ci		pr_err("bogus LPBFIFO callback\n");
13162306a36Sopenharmony_ci		spin_unlock_irqrestore(&lpbfifo.lock, flags);
13262306a36Sopenharmony_ci		return;
13362306a36Sopenharmony_ci	}
13462306a36Sopenharmony_ci
13562306a36Sopenharmony_ci	/* Release the mapping */
13662306a36Sopenharmony_ci	if (req->dir == MPC512X_LPBFIFO_REQ_DIR_WRITE)
13762306a36Sopenharmony_ci		dir = DMA_TO_DEVICE;
13862306a36Sopenharmony_ci	else
13962306a36Sopenharmony_ci		dir = DMA_FROM_DEVICE;
14062306a36Sopenharmony_ci	dma_unmap_single(lpbfifo.chan->device->dev,
14162306a36Sopenharmony_ci			lpbfifo.ram_bus_addr, req->size, dir);
14262306a36Sopenharmony_ci
14362306a36Sopenharmony_ci	lpbfifo.wait_lpbfifo_callback = false;
14462306a36Sopenharmony_ci
14562306a36Sopenharmony_ci	if (!lpbfifo.wait_lpbfifo_irq) {
14662306a36Sopenharmony_ci		/* Transfer is finished, set the FIFO as idle */
14762306a36Sopenharmony_ci		lpbfifo.req = NULL;
14862306a36Sopenharmony_ci
14962306a36Sopenharmony_ci		spin_unlock_irqrestore(&lpbfifo.lock, flags);
15062306a36Sopenharmony_ci
15162306a36Sopenharmony_ci		if (req->callback)
15262306a36Sopenharmony_ci			req->callback(req);
15362306a36Sopenharmony_ci	} else {
15462306a36Sopenharmony_ci		spin_unlock_irqrestore(&lpbfifo.lock, flags);
15562306a36Sopenharmony_ci	}
15662306a36Sopenharmony_ci}
15762306a36Sopenharmony_ci
15862306a36Sopenharmony_cistatic int mpc512x_lpbfifo_kick(void)
15962306a36Sopenharmony_ci{
16062306a36Sopenharmony_ci	u32 bits;
16162306a36Sopenharmony_ci	bool no_incr = false;
16262306a36Sopenharmony_ci	u32 bpt = 32; /* max bytes per LPBFIFO transaction involving DMA */
16362306a36Sopenharmony_ci	u32 cs = 0;
16462306a36Sopenharmony_ci	size_t i;
16562306a36Sopenharmony_ci	struct dma_device *dma_dev = NULL;
16662306a36Sopenharmony_ci	struct scatterlist sg;
16762306a36Sopenharmony_ci	enum dma_data_direction dir;
16862306a36Sopenharmony_ci	struct dma_slave_config dma_conf = {};
16962306a36Sopenharmony_ci	struct dma_async_tx_descriptor *dma_tx = NULL;
17062306a36Sopenharmony_ci	dma_cookie_t cookie;
17162306a36Sopenharmony_ci	int ret;
17262306a36Sopenharmony_ci
17362306a36Sopenharmony_ci	/*
17462306a36Sopenharmony_ci	 * 1. Fit the requirements:
17562306a36Sopenharmony_ci	 * - the packet size must be a multiple of 4 since FIFO Data Word
17662306a36Sopenharmony_ci	 *    Register allows only full-word access according the Reference
17762306a36Sopenharmony_ci	 *    Manual;
17862306a36Sopenharmony_ci	 * - the physical address of the device on LPB and the packet size
17962306a36Sopenharmony_ci	 *    must be aligned on BPT (bytes per transaction) or 8-bytes
18062306a36Sopenharmony_ci	 *    boundary according the Reference Manual;
18162306a36Sopenharmony_ci	 * - but we choose DMA maxburst equal (or very close to) BPT to prevent
18262306a36Sopenharmony_ci	 *    DMA controller from overtaking FIFO and causing FIFO underflow
18362306a36Sopenharmony_ci	 *    error. So we force the packet size to be aligned on BPT boundary
18462306a36Sopenharmony_ci	 *    not to confuse DMA driver which requires the packet size to be
18562306a36Sopenharmony_ci	 *    aligned on maxburst boundary;
18662306a36Sopenharmony_ci	 * - BPT should be set to the LPB device port size for operation with
18762306a36Sopenharmony_ci	 *    disabled auto-incrementing according Reference Manual.
18862306a36Sopenharmony_ci	 */
18962306a36Sopenharmony_ci	if (lpbfifo.req->size == 0 || !IS_ALIGNED(lpbfifo.req->size, 4))
19062306a36Sopenharmony_ci		return -EINVAL;
19162306a36Sopenharmony_ci
19262306a36Sopenharmony_ci	if (lpbfifo.req->portsize != LPB_DEV_PORTSIZE_UNDEFINED) {
19362306a36Sopenharmony_ci		bpt = lpbfifo.req->portsize;
19462306a36Sopenharmony_ci		no_incr = true;
19562306a36Sopenharmony_ci	}
19662306a36Sopenharmony_ci
19762306a36Sopenharmony_ci	while (bpt > 1) {
19862306a36Sopenharmony_ci		if (IS_ALIGNED(lpbfifo.req->dev_phys_addr, min(bpt, 0x8u)) &&
19962306a36Sopenharmony_ci					IS_ALIGNED(lpbfifo.req->size, bpt)) {
20062306a36Sopenharmony_ci			break;
20162306a36Sopenharmony_ci		}
20262306a36Sopenharmony_ci
20362306a36Sopenharmony_ci		if (no_incr)
20462306a36Sopenharmony_ci			return -EINVAL;
20562306a36Sopenharmony_ci
20662306a36Sopenharmony_ci		bpt >>= 1;
20762306a36Sopenharmony_ci	}
20862306a36Sopenharmony_ci	dma_conf.dst_maxburst = max(bpt, 0x4u) / 4;
20962306a36Sopenharmony_ci	dma_conf.src_maxburst = max(bpt, 0x4u) / 4;
21062306a36Sopenharmony_ci
21162306a36Sopenharmony_ci	for (i = 0; i < lpbfifo.cs_n; i++) {
21262306a36Sopenharmony_ci		phys_addr_t cs_start = lpbfifo.cs_ranges[i].addr;
21362306a36Sopenharmony_ci		phys_addr_t cs_end = cs_start + lpbfifo.cs_ranges[i].size;
21462306a36Sopenharmony_ci		phys_addr_t access_start = lpbfifo.req->dev_phys_addr;
21562306a36Sopenharmony_ci		phys_addr_t access_end = access_start + lpbfifo.req->size;
21662306a36Sopenharmony_ci
21762306a36Sopenharmony_ci		if (access_start >= cs_start && access_end <= cs_end) {
21862306a36Sopenharmony_ci			cs = lpbfifo.cs_ranges[i].csnum;
21962306a36Sopenharmony_ci			break;
22062306a36Sopenharmony_ci		}
22162306a36Sopenharmony_ci	}
22262306a36Sopenharmony_ci	if (i == lpbfifo.cs_n)
22362306a36Sopenharmony_ci		return -EFAULT;
22462306a36Sopenharmony_ci
22562306a36Sopenharmony_ci	/* 2. Prepare DMA */
22662306a36Sopenharmony_ci	dma_dev = lpbfifo.chan->device;
22762306a36Sopenharmony_ci
22862306a36Sopenharmony_ci	if (lpbfifo.req->dir == MPC512X_LPBFIFO_REQ_DIR_WRITE) {
22962306a36Sopenharmony_ci		dir = DMA_TO_DEVICE;
23062306a36Sopenharmony_ci		dma_conf.direction = DMA_MEM_TO_DEV;
23162306a36Sopenharmony_ci		dma_conf.dst_addr = lpbfifo.regs_phys +
23262306a36Sopenharmony_ci				offsetof(struct mpc512x_lpbfifo, data_word);
23362306a36Sopenharmony_ci	} else {
23462306a36Sopenharmony_ci		dir = DMA_FROM_DEVICE;
23562306a36Sopenharmony_ci		dma_conf.direction = DMA_DEV_TO_MEM;
23662306a36Sopenharmony_ci		dma_conf.src_addr = lpbfifo.regs_phys +
23762306a36Sopenharmony_ci				offsetof(struct mpc512x_lpbfifo, data_word);
23862306a36Sopenharmony_ci	}
23962306a36Sopenharmony_ci	dma_conf.dst_addr_width = DMA_SLAVE_BUSWIDTH_4_BYTES;
24062306a36Sopenharmony_ci	dma_conf.src_addr_width = DMA_SLAVE_BUSWIDTH_4_BYTES;
24162306a36Sopenharmony_ci
24262306a36Sopenharmony_ci	/* Make DMA channel work with LPB FIFO data register */
24362306a36Sopenharmony_ci	if (dma_dev->device_config(lpbfifo.chan, &dma_conf)) {
24462306a36Sopenharmony_ci		ret = -EINVAL;
24562306a36Sopenharmony_ci		goto err_dma_prep;
24662306a36Sopenharmony_ci	}
24762306a36Sopenharmony_ci
24862306a36Sopenharmony_ci	sg_init_table(&sg, 1);
24962306a36Sopenharmony_ci
25062306a36Sopenharmony_ci	sg_dma_address(&sg) = dma_map_single(dma_dev->dev,
25162306a36Sopenharmony_ci			lpbfifo.req->ram_virt_addr, lpbfifo.req->size, dir);
25262306a36Sopenharmony_ci	if (dma_mapping_error(dma_dev->dev, sg_dma_address(&sg)))
25362306a36Sopenharmony_ci		return -EFAULT;
25462306a36Sopenharmony_ci
25562306a36Sopenharmony_ci	lpbfifo.ram_bus_addr = sg_dma_address(&sg); /* For freeing later */
25662306a36Sopenharmony_ci
25762306a36Sopenharmony_ci	sg_dma_len(&sg) = lpbfifo.req->size;
25862306a36Sopenharmony_ci
25962306a36Sopenharmony_ci	dma_tx = dmaengine_prep_slave_sg(lpbfifo.chan, &sg,
26062306a36Sopenharmony_ci						1, dma_conf.direction, 0);
26162306a36Sopenharmony_ci	if (!dma_tx) {
26262306a36Sopenharmony_ci		ret = -ENOSPC;
26362306a36Sopenharmony_ci		goto err_dma_prep;
26462306a36Sopenharmony_ci	}
26562306a36Sopenharmony_ci	dma_tx->callback = mpc512x_lpbfifo_callback;
26662306a36Sopenharmony_ci	dma_tx->callback_param = NULL;
26762306a36Sopenharmony_ci
26862306a36Sopenharmony_ci	/* 3. Prepare FIFO */
26962306a36Sopenharmony_ci	out_be32(&lpbfifo.regs->enable,
27062306a36Sopenharmony_ci				MPC512X_SCLPC_RESET | MPC512X_SCLPC_FIFO_RESET);
27162306a36Sopenharmony_ci	out_be32(&lpbfifo.regs->enable, 0x0);
27262306a36Sopenharmony_ci
27362306a36Sopenharmony_ci	/*
27462306a36Sopenharmony_ci	 * Configure the watermarks for write operation (RAM->DMA->FIFO->dev):
27562306a36Sopenharmony_ci	 * - high watermark 7 words according the Reference Manual,
27662306a36Sopenharmony_ci	 * - low watermark 512 bytes (half of the FIFO).
27762306a36Sopenharmony_ci	 * These watermarks don't work for read operation since the
27862306a36Sopenharmony_ci	 * MPC512X_SCLPC_FLUSH bit is set (according the Reference Manual).
27962306a36Sopenharmony_ci	 */
28062306a36Sopenharmony_ci	out_be32(&lpbfifo.regs->fifo_ctrl, MPC512X_SCLPC_FIFO_CTRL(0x7));
28162306a36Sopenharmony_ci	out_be32(&lpbfifo.regs->fifo_alarm, MPC512X_SCLPC_FIFO_ALARM(0x200));
28262306a36Sopenharmony_ci
28362306a36Sopenharmony_ci	/*
28462306a36Sopenharmony_ci	 * Start address is a physical address of the region which belongs
28562306a36Sopenharmony_ci	 * to the device on the LocalPlus Bus
28662306a36Sopenharmony_ci	 */
28762306a36Sopenharmony_ci	out_be32(&lpbfifo.regs->start_addr, lpbfifo.req->dev_phys_addr);
28862306a36Sopenharmony_ci
28962306a36Sopenharmony_ci	/*
29062306a36Sopenharmony_ci	 * Configure chip select, transfer direction, address increment option
29162306a36Sopenharmony_ci	 * and bytes per transaction option
29262306a36Sopenharmony_ci	 */
29362306a36Sopenharmony_ci	bits = MPC512X_SCLPC_CS(cs);
29462306a36Sopenharmony_ci	if (lpbfifo.req->dir == MPC512X_LPBFIFO_REQ_DIR_READ)
29562306a36Sopenharmony_ci		bits |= MPC512X_SCLPC_READ | MPC512X_SCLPC_FLUSH;
29662306a36Sopenharmony_ci	if (no_incr)
29762306a36Sopenharmony_ci		bits |= MPC512X_SCLPC_DAI;
29862306a36Sopenharmony_ci	bits |= MPC512X_SCLPC_BPT(bpt);
29962306a36Sopenharmony_ci	out_be32(&lpbfifo.regs->ctrl, bits);
30062306a36Sopenharmony_ci
30162306a36Sopenharmony_ci	/* Unmask irqs */
30262306a36Sopenharmony_ci	bits = MPC512X_SCLPC_ENABLE | MPC512X_SCLPC_ABORT_INT_ENABLE;
30362306a36Sopenharmony_ci	if (lpbfifo.req->dir == MPC512X_LPBFIFO_REQ_DIR_WRITE)
30462306a36Sopenharmony_ci		bits |= MPC512X_SCLPC_NORM_INT_ENABLE;
30562306a36Sopenharmony_ci	else
30662306a36Sopenharmony_ci		lpbfifo.wait_lpbfifo_irq = false;
30762306a36Sopenharmony_ci
30862306a36Sopenharmony_ci	out_be32(&lpbfifo.regs->enable, bits);
30962306a36Sopenharmony_ci
31062306a36Sopenharmony_ci	/* 4. Set packet size and kick FIFO off */
31162306a36Sopenharmony_ci	bits = lpbfifo.req->size | MPC512X_SCLPC_START;
31262306a36Sopenharmony_ci	out_be32(&lpbfifo.regs->pkt_size, bits);
31362306a36Sopenharmony_ci
31462306a36Sopenharmony_ci	/* 5. Finally kick DMA off */
31562306a36Sopenharmony_ci	cookie = dma_tx->tx_submit(dma_tx);
31662306a36Sopenharmony_ci	if (dma_submit_error(cookie)) {
31762306a36Sopenharmony_ci		ret = -ENOSPC;
31862306a36Sopenharmony_ci		goto err_dma_submit;
31962306a36Sopenharmony_ci	}
32062306a36Sopenharmony_ci
32162306a36Sopenharmony_ci	return 0;
32262306a36Sopenharmony_ci
32362306a36Sopenharmony_ci err_dma_submit:
32462306a36Sopenharmony_ci	out_be32(&lpbfifo.regs->enable,
32562306a36Sopenharmony_ci				MPC512X_SCLPC_RESET | MPC512X_SCLPC_FIFO_RESET);
32662306a36Sopenharmony_ci err_dma_prep:
32762306a36Sopenharmony_ci	dma_unmap_single(dma_dev->dev, sg_dma_address(&sg),
32862306a36Sopenharmony_ci						lpbfifo.req->size, dir);
32962306a36Sopenharmony_ci	return ret;
33062306a36Sopenharmony_ci}
33162306a36Sopenharmony_ci
33262306a36Sopenharmony_cistatic int mpc512x_lpbfifo_submit_locked(struct mpc512x_lpbfifo_request *req)
33362306a36Sopenharmony_ci{
33462306a36Sopenharmony_ci	int ret = 0;
33562306a36Sopenharmony_ci
33662306a36Sopenharmony_ci	if (!lpbfifo.regs)
33762306a36Sopenharmony_ci		return -ENODEV;
33862306a36Sopenharmony_ci
33962306a36Sopenharmony_ci	/* Check whether a transfer is in progress */
34062306a36Sopenharmony_ci	if (lpbfifo.req)
34162306a36Sopenharmony_ci		return -EBUSY;
34262306a36Sopenharmony_ci
34362306a36Sopenharmony_ci	lpbfifo.wait_lpbfifo_irq = true;
34462306a36Sopenharmony_ci	lpbfifo.wait_lpbfifo_callback = true;
34562306a36Sopenharmony_ci	lpbfifo.req = req;
34662306a36Sopenharmony_ci
34762306a36Sopenharmony_ci	ret = mpc512x_lpbfifo_kick();
34862306a36Sopenharmony_ci	if (ret != 0)
34962306a36Sopenharmony_ci		lpbfifo.req = NULL; /* Set the FIFO as idle */
35062306a36Sopenharmony_ci
35162306a36Sopenharmony_ci	return ret;
35262306a36Sopenharmony_ci}
35362306a36Sopenharmony_ci
35462306a36Sopenharmony_ciint mpc512x_lpbfifo_submit(struct mpc512x_lpbfifo_request *req)
35562306a36Sopenharmony_ci{
35662306a36Sopenharmony_ci	unsigned long flags;
35762306a36Sopenharmony_ci	int ret = 0;
35862306a36Sopenharmony_ci
35962306a36Sopenharmony_ci	spin_lock_irqsave(&lpbfifo.lock, flags);
36062306a36Sopenharmony_ci	ret = mpc512x_lpbfifo_submit_locked(req);
36162306a36Sopenharmony_ci	spin_unlock_irqrestore(&lpbfifo.lock, flags);
36262306a36Sopenharmony_ci
36362306a36Sopenharmony_ci	return ret;
36462306a36Sopenharmony_ci}
36562306a36Sopenharmony_ciEXPORT_SYMBOL(mpc512x_lpbfifo_submit);
36662306a36Sopenharmony_ci
36762306a36Sopenharmony_ci/*
36862306a36Sopenharmony_ci * LPBFIFO driver uses "ranges" property of "localbus" device tree node
36962306a36Sopenharmony_ci * for being able to determine the chip select number of a client device
37062306a36Sopenharmony_ci * ordering a DMA transfer.
37162306a36Sopenharmony_ci */
37262306a36Sopenharmony_cistatic int get_cs_ranges(struct device *dev)
37362306a36Sopenharmony_ci{
37462306a36Sopenharmony_ci	int ret = -ENODEV;
37562306a36Sopenharmony_ci	struct device_node *lb_node;
37662306a36Sopenharmony_ci	size_t i = 0;
37762306a36Sopenharmony_ci	struct of_range_parser parser;
37862306a36Sopenharmony_ci	struct of_range range;
37962306a36Sopenharmony_ci
38062306a36Sopenharmony_ci	lb_node = of_find_compatible_node(NULL, NULL, "fsl,mpc5121-localbus");
38162306a36Sopenharmony_ci	if (!lb_node)
38262306a36Sopenharmony_ci		return ret;
38362306a36Sopenharmony_ci
38462306a36Sopenharmony_ci	of_range_parser_init(&parser, lb_node);
38562306a36Sopenharmony_ci	lpbfifo.cs_n = of_range_count(&parser);
38662306a36Sopenharmony_ci
38762306a36Sopenharmony_ci	lpbfifo.cs_ranges = devm_kcalloc(dev, lpbfifo.cs_n,
38862306a36Sopenharmony_ci					sizeof(struct cs_range), GFP_KERNEL);
38962306a36Sopenharmony_ci	if (!lpbfifo.cs_ranges)
39062306a36Sopenharmony_ci		goto end;
39162306a36Sopenharmony_ci
39262306a36Sopenharmony_ci	for_each_of_range(&parser, &range) {
39362306a36Sopenharmony_ci		u32 base = lower_32_bits(range.bus_addr);
39462306a36Sopenharmony_ci		if (base)
39562306a36Sopenharmony_ci			goto end;
39662306a36Sopenharmony_ci
39762306a36Sopenharmony_ci		lpbfifo.cs_ranges[i].csnum = upper_32_bits(range.bus_addr);
39862306a36Sopenharmony_ci		lpbfifo.cs_ranges[i].base = base;
39962306a36Sopenharmony_ci		lpbfifo.cs_ranges[i].addr = range.cpu_addr;
40062306a36Sopenharmony_ci		lpbfifo.cs_ranges[i].size = range.size;
40162306a36Sopenharmony_ci		i++;
40262306a36Sopenharmony_ci	}
40362306a36Sopenharmony_ci
40462306a36Sopenharmony_ci	ret = 0;
40562306a36Sopenharmony_ci
40662306a36Sopenharmony_ci end:
40762306a36Sopenharmony_ci	of_node_put(lb_node);
40862306a36Sopenharmony_ci	return ret;
40962306a36Sopenharmony_ci}
41062306a36Sopenharmony_ci
41162306a36Sopenharmony_cistatic int mpc512x_lpbfifo_probe(struct platform_device *pdev)
41262306a36Sopenharmony_ci{
41362306a36Sopenharmony_ci	struct resource r;
41462306a36Sopenharmony_ci	int ret = 0;
41562306a36Sopenharmony_ci
41662306a36Sopenharmony_ci	memset(&lpbfifo, 0, sizeof(struct lpbfifo_data));
41762306a36Sopenharmony_ci	spin_lock_init(&lpbfifo.lock);
41862306a36Sopenharmony_ci
41962306a36Sopenharmony_ci	lpbfifo.chan = dma_request_chan(&pdev->dev, "rx-tx");
42062306a36Sopenharmony_ci	if (IS_ERR(lpbfifo.chan))
42162306a36Sopenharmony_ci		return PTR_ERR(lpbfifo.chan);
42262306a36Sopenharmony_ci
42362306a36Sopenharmony_ci	if (of_address_to_resource(pdev->dev.of_node, 0, &r) != 0) {
42462306a36Sopenharmony_ci		dev_err(&pdev->dev, "bad 'reg' in 'sclpc' device tree node\n");
42562306a36Sopenharmony_ci		ret = -ENODEV;
42662306a36Sopenharmony_ci		goto err0;
42762306a36Sopenharmony_ci	}
42862306a36Sopenharmony_ci
42962306a36Sopenharmony_ci	lpbfifo.regs_phys = r.start;
43062306a36Sopenharmony_ci	lpbfifo.regs_size = resource_size(&r);
43162306a36Sopenharmony_ci
43262306a36Sopenharmony_ci	if (!devm_request_mem_region(&pdev->dev, lpbfifo.regs_phys,
43362306a36Sopenharmony_ci					lpbfifo.regs_size, DRV_NAME)) {
43462306a36Sopenharmony_ci		dev_err(&pdev->dev, "unable to request region\n");
43562306a36Sopenharmony_ci		ret = -EBUSY;
43662306a36Sopenharmony_ci		goto err0;
43762306a36Sopenharmony_ci	}
43862306a36Sopenharmony_ci
43962306a36Sopenharmony_ci	lpbfifo.regs = devm_ioremap(&pdev->dev,
44062306a36Sopenharmony_ci					lpbfifo.regs_phys, lpbfifo.regs_size);
44162306a36Sopenharmony_ci	if (!lpbfifo.regs) {
44262306a36Sopenharmony_ci		dev_err(&pdev->dev, "mapping registers failed\n");
44362306a36Sopenharmony_ci		ret = -ENOMEM;
44462306a36Sopenharmony_ci		goto err0;
44562306a36Sopenharmony_ci	}
44662306a36Sopenharmony_ci
44762306a36Sopenharmony_ci	out_be32(&lpbfifo.regs->enable,
44862306a36Sopenharmony_ci				MPC512X_SCLPC_RESET | MPC512X_SCLPC_FIFO_RESET);
44962306a36Sopenharmony_ci
45062306a36Sopenharmony_ci	if (get_cs_ranges(&pdev->dev) != 0) {
45162306a36Sopenharmony_ci		dev_err(&pdev->dev, "bad '/localbus' device tree node\n");
45262306a36Sopenharmony_ci		ret = -ENODEV;
45362306a36Sopenharmony_ci		goto err0;
45462306a36Sopenharmony_ci	}
45562306a36Sopenharmony_ci
45662306a36Sopenharmony_ci	lpbfifo.irq = irq_of_parse_and_map(pdev->dev.of_node, 0);
45762306a36Sopenharmony_ci	if (!lpbfifo.irq) {
45862306a36Sopenharmony_ci		dev_err(&pdev->dev, "mapping irq failed\n");
45962306a36Sopenharmony_ci		ret = -ENODEV;
46062306a36Sopenharmony_ci		goto err0;
46162306a36Sopenharmony_ci	}
46262306a36Sopenharmony_ci
46362306a36Sopenharmony_ci	if (request_irq(lpbfifo.irq, mpc512x_lpbfifo_irq, 0,
46462306a36Sopenharmony_ci						DRV_NAME, &pdev->dev) != 0) {
46562306a36Sopenharmony_ci		dev_err(&pdev->dev, "requesting irq failed\n");
46662306a36Sopenharmony_ci		ret = -ENODEV;
46762306a36Sopenharmony_ci		goto err1;
46862306a36Sopenharmony_ci	}
46962306a36Sopenharmony_ci
47062306a36Sopenharmony_ci	dev_info(&pdev->dev, "probe succeeded\n");
47162306a36Sopenharmony_ci	return 0;
47262306a36Sopenharmony_ci
47362306a36Sopenharmony_ci err1:
47462306a36Sopenharmony_ci	irq_dispose_mapping(lpbfifo.irq);
47562306a36Sopenharmony_ci err0:
47662306a36Sopenharmony_ci	dma_release_channel(lpbfifo.chan);
47762306a36Sopenharmony_ci	return ret;
47862306a36Sopenharmony_ci}
47962306a36Sopenharmony_ci
48062306a36Sopenharmony_cistatic void mpc512x_lpbfifo_remove(struct platform_device *pdev)
48162306a36Sopenharmony_ci{
48262306a36Sopenharmony_ci	unsigned long flags;
48362306a36Sopenharmony_ci	struct dma_device *dma_dev = lpbfifo.chan->device;
48462306a36Sopenharmony_ci	struct mpc512x_lpbfifo __iomem *regs = NULL;
48562306a36Sopenharmony_ci
48662306a36Sopenharmony_ci	spin_lock_irqsave(&lpbfifo.lock, flags);
48762306a36Sopenharmony_ci	regs = lpbfifo.regs;
48862306a36Sopenharmony_ci	lpbfifo.regs = NULL;
48962306a36Sopenharmony_ci	spin_unlock_irqrestore(&lpbfifo.lock, flags);
49062306a36Sopenharmony_ci
49162306a36Sopenharmony_ci	dma_dev->device_terminate_all(lpbfifo.chan);
49262306a36Sopenharmony_ci	out_be32(&regs->enable, MPC512X_SCLPC_RESET | MPC512X_SCLPC_FIFO_RESET);
49362306a36Sopenharmony_ci
49462306a36Sopenharmony_ci	free_irq(lpbfifo.irq, &pdev->dev);
49562306a36Sopenharmony_ci	irq_dispose_mapping(lpbfifo.irq);
49662306a36Sopenharmony_ci	dma_release_channel(lpbfifo.chan);
49762306a36Sopenharmony_ci}
49862306a36Sopenharmony_ci
49962306a36Sopenharmony_cistatic const struct of_device_id mpc512x_lpbfifo_match[] = {
50062306a36Sopenharmony_ci	{ .compatible = "fsl,mpc512x-lpbfifo", },
50162306a36Sopenharmony_ci	{},
50262306a36Sopenharmony_ci};
50362306a36Sopenharmony_ciMODULE_DEVICE_TABLE(of, mpc512x_lpbfifo_match);
50462306a36Sopenharmony_ci
50562306a36Sopenharmony_cistatic struct platform_driver mpc512x_lpbfifo_driver = {
50662306a36Sopenharmony_ci	.probe = mpc512x_lpbfifo_probe,
50762306a36Sopenharmony_ci	.remove_new = mpc512x_lpbfifo_remove,
50862306a36Sopenharmony_ci	.driver = {
50962306a36Sopenharmony_ci		.name = DRV_NAME,
51062306a36Sopenharmony_ci		.of_match_table = mpc512x_lpbfifo_match,
51162306a36Sopenharmony_ci	},
51262306a36Sopenharmony_ci};
51362306a36Sopenharmony_ci
51462306a36Sopenharmony_cimodule_platform_driver(mpc512x_lpbfifo_driver);
51562306a36Sopenharmony_ci
51662306a36Sopenharmony_ciMODULE_AUTHOR("Alexander Popov <alex.popov@linux.com>");
51762306a36Sopenharmony_ciMODULE_DESCRIPTION("MPC512x LocalPlus Bus FIFO device driver");
51862306a36Sopenharmony_ciMODULE_LICENSE("GPL v2");
519