xref: /openbmc/linux/drivers/media/pci/cobalt/cobalt-omnitek.c (revision cbecf716ca618fd44feda6bd9a64a8179d031fc5)
16884db3cSHans Verkuil // SPDX-License-Identifier: GPL-2.0-only
285756a06SHans Verkuil /*
385756a06SHans Verkuil  *  Omnitek Scatter-Gather DMA Controller
485756a06SHans Verkuil  *
585756a06SHans Verkuil  *  Copyright 2012-2015 Cisco Systems, Inc. and/or its affiliates.
685756a06SHans Verkuil  *  All rights reserved.
785756a06SHans Verkuil  */
885756a06SHans Verkuil 
985756a06SHans Verkuil #include <linux/string.h>
1085756a06SHans Verkuil #include <linux/io.h>
1185756a06SHans Verkuil #include <linux/pci_regs.h>
1285756a06SHans Verkuil #include <linux/spinlock.h>
1385756a06SHans Verkuil 
1485756a06SHans Verkuil #include "cobalt-driver.h"
1585756a06SHans Verkuil #include "cobalt-omnitek.h"
1685756a06SHans Verkuil 
1785756a06SHans Verkuil /* descriptor */
1885756a06SHans Verkuil #define END_OF_CHAIN		(1 << 1)
1985756a06SHans Verkuil #define INTERRUPT_ENABLE	(1 << 2)
2085756a06SHans Verkuil #define WRITE_TO_PCI		(1 << 3)
2185756a06SHans Verkuil #define READ_FROM_PCI		(0 << 3)
2285756a06SHans Verkuil #define DESCRIPTOR_FLAG_MSK	(END_OF_CHAIN | INTERRUPT_ENABLE | WRITE_TO_PCI)
2385756a06SHans Verkuil #define NEXT_ADRS_MSK		0xffffffe0
2485756a06SHans Verkuil 
2585756a06SHans Verkuil /* control/status register */
2685756a06SHans Verkuil #define ENABLE                  (1 << 0)
2785756a06SHans Verkuil #define START                   (1 << 1)
2885756a06SHans Verkuil #define ABORT                   (1 << 2)
2985756a06SHans Verkuil #define DONE                    (1 << 4)
3085756a06SHans Verkuil #define SG_INTERRUPT            (1 << 5)
3185756a06SHans Verkuil #define EVENT_INTERRUPT         (1 << 6)
3285756a06SHans Verkuil #define SCATTER_GATHER_MODE     (1 << 8)
3385756a06SHans Verkuil #define DISABLE_VIDEO_RESYNC    (1 << 9)
3485756a06SHans Verkuil #define EVENT_INTERRUPT_ENABLE  (1 << 10)
3585756a06SHans Verkuil #define DIRECTIONAL_MSK         (3 << 16)
3685756a06SHans Verkuil #define INPUT_ONLY              (0 << 16)
3785756a06SHans Verkuil #define OUTPUT_ONLY             (1 << 16)
3885756a06SHans Verkuil #define BIDIRECTIONAL           (2 << 16)
3985756a06SHans Verkuil #define DMA_TYPE_MEMORY         (0 << 18)
4085756a06SHans Verkuil #define DMA_TYPE_FIFO		(1 << 18)
4185756a06SHans Verkuil 
4285756a06SHans Verkuil #define BASE			(cobalt->bar0)
4385756a06SHans Verkuil #define CAPABILITY_HEADER	(BASE)
4485756a06SHans Verkuil #define CAPABILITY_REGISTER	(BASE + 0x04)
4585756a06SHans Verkuil #define PCI_64BIT		(1 << 8)
4685756a06SHans Verkuil #define LOCAL_64BIT		(1 << 9)
4785756a06SHans Verkuil #define INTERRUPT_STATUS	(BASE + 0x08)
4885756a06SHans Verkuil #define PCI(c)			(BASE + 0x40 + ((c) * 0x40))
4985756a06SHans Verkuil #define SIZE(c)			(BASE + 0x58 + ((c) * 0x40))
5085756a06SHans Verkuil #define DESCRIPTOR(c)		(BASE + 0x50 + ((c) * 0x40))
5185756a06SHans Verkuil #define CS_REG(c)		(BASE + 0x60 + ((c) * 0x40))
5285756a06SHans Verkuil #define BYTES_TRANSFERRED(c)	(BASE + 0x64 + ((c) * 0x40))
5385756a06SHans Verkuil 
5485756a06SHans Verkuil 
get_dma_direction(u32 status)5585756a06SHans Verkuil static char *get_dma_direction(u32 status)
5685756a06SHans Verkuil {
5785756a06SHans Verkuil 	switch (status & DIRECTIONAL_MSK) {
5885756a06SHans Verkuil 	case INPUT_ONLY: return "Input";
5985756a06SHans Verkuil 	case OUTPUT_ONLY: return "Output";
6085756a06SHans Verkuil 	case BIDIRECTIONAL: return "Bidirectional";
6185756a06SHans Verkuil 	}
6285756a06SHans Verkuil 	return "";
6385756a06SHans Verkuil }
6485756a06SHans Verkuil 
show_dma_capability(struct cobalt * cobalt)6585756a06SHans Verkuil static void show_dma_capability(struct cobalt *cobalt)
6685756a06SHans Verkuil {
6785756a06SHans Verkuil 	u32 header = ioread32(CAPABILITY_HEADER);
6885756a06SHans Verkuil 	u32 capa = ioread32(CAPABILITY_REGISTER);
6985756a06SHans Verkuil 	u32 i;
7085756a06SHans Verkuil 
7185756a06SHans Verkuil 	cobalt_info("Omnitek DMA capability: ID 0x%02x Version 0x%02x Next 0x%x Size 0x%x\n",
7285756a06SHans Verkuil 		    header & 0xff, (header >> 8) & 0xff,
7385756a06SHans Verkuil 		    (header >> 16) & 0xffff, (capa >> 24) & 0xff);
7485756a06SHans Verkuil 
7585756a06SHans Verkuil 	switch ((capa >> 8) & 0x3) {
7685756a06SHans Verkuil 	case 0:
7785756a06SHans Verkuil 		cobalt_info("Omnitek DMA: 32 bits PCIe and Local\n");
7885756a06SHans Verkuil 		break;
7985756a06SHans Verkuil 	case 1:
8085756a06SHans Verkuil 		cobalt_info("Omnitek DMA: 64 bits PCIe, 32 bits Local\n");
8185756a06SHans Verkuil 		break;
8285756a06SHans Verkuil 	case 3:
8385756a06SHans Verkuil 		cobalt_info("Omnitek DMA: 64 bits PCIe and Local\n");
8485756a06SHans Verkuil 		break;
8585756a06SHans Verkuil 	}
8685756a06SHans Verkuil 
8785756a06SHans Verkuil 	for (i = 0;  i < (capa & 0xf);  i++) {
8885756a06SHans Verkuil 		u32 status = ioread32(CS_REG(i));
8985756a06SHans Verkuil 
9085756a06SHans Verkuil 		cobalt_info("Omnitek DMA channel #%d: %s %s\n", i,
9185756a06SHans Verkuil 			    status & DMA_TYPE_FIFO ? "FIFO" : "MEMORY",
9285756a06SHans Verkuil 			    get_dma_direction(status));
9385756a06SHans Verkuil 	}
9485756a06SHans Verkuil }
9585756a06SHans Verkuil 
omni_sg_dma_start(struct cobalt_stream * s,struct sg_dma_desc_info * desc)9685756a06SHans Verkuil void omni_sg_dma_start(struct cobalt_stream *s, struct sg_dma_desc_info *desc)
9785756a06SHans Verkuil {
9885756a06SHans Verkuil 	struct cobalt *cobalt = s->cobalt;
9985756a06SHans Verkuil 
10040c942a1SHans Verkuil 	iowrite32((u32)((u64)desc->bus >> 32), DESCRIPTOR(s->dma_channel) + 4);
10185756a06SHans Verkuil 	iowrite32((u32)desc->bus & NEXT_ADRS_MSK, DESCRIPTOR(s->dma_channel));
10285756a06SHans Verkuil 	iowrite32(ENABLE | SCATTER_GATHER_MODE | START, CS_REG(s->dma_channel));
10385756a06SHans Verkuil }
10485756a06SHans Verkuil 
is_dma_done(struct cobalt_stream * s)10585756a06SHans Verkuil bool is_dma_done(struct cobalt_stream *s)
10685756a06SHans Verkuil {
10785756a06SHans Verkuil 	struct cobalt *cobalt = s->cobalt;
10885756a06SHans Verkuil 
10985756a06SHans Verkuil 	if (ioread32(CS_REG(s->dma_channel)) & DONE)
11085756a06SHans Verkuil 		return true;
11185756a06SHans Verkuil 
11285756a06SHans Verkuil 	return false;
11385756a06SHans Verkuil }
11485756a06SHans Verkuil 
omni_sg_dma_abort_channel(struct cobalt_stream * s)11585756a06SHans Verkuil void omni_sg_dma_abort_channel(struct cobalt_stream *s)
11685756a06SHans Verkuil {
11785756a06SHans Verkuil 	struct cobalt *cobalt = s->cobalt;
11885756a06SHans Verkuil 
119*52cdded0SDaniel W. S. Almeida 	if (!is_dma_done(s))
12085756a06SHans Verkuil 		iowrite32(ABORT, CS_REG(s->dma_channel));
12185756a06SHans Verkuil }
12285756a06SHans Verkuil 
omni_sg_dma_init(struct cobalt * cobalt)12385756a06SHans Verkuil int omni_sg_dma_init(struct cobalt *cobalt)
12485756a06SHans Verkuil {
12585756a06SHans Verkuil 	u32 capa = ioread32(CAPABILITY_REGISTER);
12685756a06SHans Verkuil 	int i;
12785756a06SHans Verkuil 
12885756a06SHans Verkuil 	cobalt->first_fifo_channel = 0;
12985756a06SHans Verkuil 	cobalt->dma_channels = capa & 0xf;
13085756a06SHans Verkuil 	if (capa & PCI_64BIT)
13185756a06SHans Verkuil 		cobalt->pci_32_bit = false;
13285756a06SHans Verkuil 	else
13385756a06SHans Verkuil 		cobalt->pci_32_bit = true;
13485756a06SHans Verkuil 
13585756a06SHans Verkuil 	for (i = 0; i < cobalt->dma_channels; i++) {
13685756a06SHans Verkuil 		u32 status = ioread32(CS_REG(i));
13785756a06SHans Verkuil 		u32 ctrl = ioread32(CS_REG(i));
13885756a06SHans Verkuil 
13985756a06SHans Verkuil 		if (!(ctrl & DONE))
14085756a06SHans Verkuil 			iowrite32(ABORT, CS_REG(i));
14185756a06SHans Verkuil 
14285756a06SHans Verkuil 		if (!(status & DMA_TYPE_FIFO))
14385756a06SHans Verkuil 			cobalt->first_fifo_channel++;
14485756a06SHans Verkuil 	}
14585756a06SHans Verkuil 	show_dma_capability(cobalt);
14685756a06SHans Verkuil 	return 0;
14785756a06SHans Verkuil }
14885756a06SHans Verkuil 
descriptor_list_create(struct cobalt * cobalt,struct scatterlist * scatter_list,bool to_pci,unsigned sglen,unsigned size,unsigned width,unsigned stride,struct sg_dma_desc_info * desc)14985756a06SHans Verkuil int descriptor_list_create(struct cobalt *cobalt,
15085756a06SHans Verkuil 		struct scatterlist *scatter_list, bool to_pci, unsigned sglen,
15185756a06SHans Verkuil 		unsigned size, unsigned width, unsigned stride,
15285756a06SHans Verkuil 		struct sg_dma_desc_info *desc)
15385756a06SHans Verkuil {
15485756a06SHans Verkuil 	struct sg_dma_descriptor *d = (struct sg_dma_descriptor *)desc->virt;
15585756a06SHans Verkuil 	dma_addr_t next = desc->bus;
15685756a06SHans Verkuil 	unsigned offset = 0;
15785756a06SHans Verkuil 	unsigned copy_bytes = width;
15885756a06SHans Verkuil 	unsigned copied = 0;
15985756a06SHans Verkuil 	bool first = true;
16085756a06SHans Verkuil 
16185756a06SHans Verkuil 	/* Must be 4-byte aligned */
16285756a06SHans Verkuil 	WARN_ON(sg_dma_address(scatter_list) & 3);
16385756a06SHans Verkuil 	WARN_ON(size & 3);
16485756a06SHans Verkuil 	WARN_ON(next & 3);
16585756a06SHans Verkuil 	WARN_ON(stride & 3);
16685756a06SHans Verkuil 	WARN_ON(stride < width);
16785756a06SHans Verkuil 	if (width >= stride)
16885756a06SHans Verkuil 		copy_bytes = stride = size;
16985756a06SHans Verkuil 
17085756a06SHans Verkuil 	while (size) {
17185756a06SHans Verkuil 		dma_addr_t addr = sg_dma_address(scatter_list) + offset;
17285756a06SHans Verkuil 		unsigned bytes;
17385756a06SHans Verkuil 
17485756a06SHans Verkuil 		if (addr == 0)
17585756a06SHans Verkuil 			return -EFAULT;
17685756a06SHans Verkuil 		if (cobalt->pci_32_bit) {
17785756a06SHans Verkuil 			WARN_ON((u64)addr >> 32);
17885756a06SHans Verkuil 			if ((u64)addr >> 32)
17985756a06SHans Verkuil 				return -EFAULT;
18085756a06SHans Verkuil 		}
18185756a06SHans Verkuil 
18285756a06SHans Verkuil 		/* PCIe address */
18385756a06SHans Verkuil 		d->pci_l = addr & 0xffffffff;
18485756a06SHans Verkuil 		/* If dma_addr_t is 32 bits, then addr >> 32 is actually the
18585756a06SHans Verkuil 		   equivalent of addr >> 0 in gcc. So must cast to u64. */
18685756a06SHans Verkuil 		d->pci_h = (u64)addr >> 32;
18785756a06SHans Verkuil 
18885756a06SHans Verkuil 		/* Sync to start of streaming frame */
18985756a06SHans Verkuil 		d->local = 0;
19085756a06SHans Verkuil 		d->reserved0 = 0;
19185756a06SHans Verkuil 
19285756a06SHans Verkuil 		/* Transfer bytes */
19385756a06SHans Verkuil 		bytes = min(sg_dma_len(scatter_list) - offset,
19485756a06SHans Verkuil 				copy_bytes - copied);
19585756a06SHans Verkuil 
19685756a06SHans Verkuil 		if (first) {
19785756a06SHans Verkuil 			if (to_pci)
19885756a06SHans Verkuil 				d->local = 0x11111111;
19985756a06SHans Verkuil 			first = false;
20085756a06SHans Verkuil 			if (sglen == 1) {
20185756a06SHans Verkuil 				/* Make sure there are always at least two
20285756a06SHans Verkuil 				 * descriptors */
20385756a06SHans Verkuil 				d->bytes = (bytes / 2) & ~3;
20485756a06SHans Verkuil 				d->reserved1 = 0;
20585756a06SHans Verkuil 				size -= d->bytes;
20685756a06SHans Verkuil 				copied += d->bytes;
20785756a06SHans Verkuil 				offset += d->bytes;
20885756a06SHans Verkuil 				addr += d->bytes;
20985756a06SHans Verkuil 				next += sizeof(struct sg_dma_descriptor);
21040c942a1SHans Verkuil 				d->next_h = (u32)((u64)next >> 32);
21185756a06SHans Verkuil 				d->next_l = (u32)next |
21285756a06SHans Verkuil 					(to_pci ? WRITE_TO_PCI : 0);
21385756a06SHans Verkuil 				bytes -= d->bytes;
21485756a06SHans Verkuil 				d++;
21585756a06SHans Verkuil 				/* PCIe address */
21685756a06SHans Verkuil 				d->pci_l = addr & 0xffffffff;
21785756a06SHans Verkuil 				/* If dma_addr_t is 32 bits, then addr >> 32
21885756a06SHans Verkuil 				 * is actually the equivalent of addr >> 0 in
21985756a06SHans Verkuil 				 * gcc. So must cast to u64. */
22085756a06SHans Verkuil 				d->pci_h = (u64)addr >> 32;
22185756a06SHans Verkuil 
22285756a06SHans Verkuil 				/* Sync to start of streaming frame */
22385756a06SHans Verkuil 				d->local = 0;
22485756a06SHans Verkuil 				d->reserved0 = 0;
22585756a06SHans Verkuil 			}
22685756a06SHans Verkuil 		}
22785756a06SHans Verkuil 
22885756a06SHans Verkuil 		d->bytes = bytes;
22985756a06SHans Verkuil 		d->reserved1 = 0;
23085756a06SHans Verkuil 		size -= bytes;
23185756a06SHans Verkuil 		copied += bytes;
23285756a06SHans Verkuil 		offset += bytes;
23385756a06SHans Verkuil 
23485756a06SHans Verkuil 		if (copied == copy_bytes) {
23585756a06SHans Verkuil 			while (copied < stride) {
23685756a06SHans Verkuil 				bytes = min(sg_dma_len(scatter_list) - offset,
23785756a06SHans Verkuil 						stride - copied);
23885756a06SHans Verkuil 				copied += bytes;
23985756a06SHans Verkuil 				offset += bytes;
24085756a06SHans Verkuil 				size -= bytes;
24185756a06SHans Verkuil 				if (sg_dma_len(scatter_list) == offset) {
24285756a06SHans Verkuil 					offset = 0;
24385756a06SHans Verkuil 					scatter_list = sg_next(scatter_list);
24485756a06SHans Verkuil 				}
24585756a06SHans Verkuil 			}
24685756a06SHans Verkuil 			copied = 0;
24785756a06SHans Verkuil 		} else {
24885756a06SHans Verkuil 			offset = 0;
24985756a06SHans Verkuil 			scatter_list = sg_next(scatter_list);
25085756a06SHans Verkuil 		}
25185756a06SHans Verkuil 
25285756a06SHans Verkuil 		/* Next descriptor + control bits */
25385756a06SHans Verkuil 		next += sizeof(struct sg_dma_descriptor);
25485756a06SHans Verkuil 		if (size == 0) {
25585756a06SHans Verkuil 			/* Loopback to the first descriptor */
25640c942a1SHans Verkuil 			d->next_h = (u32)((u64)desc->bus >> 32);
25785756a06SHans Verkuil 			d->next_l = (u32)desc->bus |
25885756a06SHans Verkuil 				(to_pci ? WRITE_TO_PCI : 0) | INTERRUPT_ENABLE;
25985756a06SHans Verkuil 			if (!to_pci)
26085756a06SHans Verkuil 				d->local = 0x22222222;
26185756a06SHans Verkuil 			desc->last_desc_virt = d;
26285756a06SHans Verkuil 		} else {
26340c942a1SHans Verkuil 			d->next_h = (u32)((u64)next >> 32);
26485756a06SHans Verkuil 			d->next_l = (u32)next | (to_pci ? WRITE_TO_PCI : 0);
26585756a06SHans Verkuil 		}
26685756a06SHans Verkuil 		d++;
26785756a06SHans Verkuil 	}
26885756a06SHans Verkuil 	return 0;
26985756a06SHans Verkuil }
27085756a06SHans Verkuil 
descriptor_list_chain(struct sg_dma_desc_info * this,struct sg_dma_desc_info * next)27185756a06SHans Verkuil void descriptor_list_chain(struct sg_dma_desc_info *this,
27285756a06SHans Verkuil 			   struct sg_dma_desc_info *next)
27385756a06SHans Verkuil {
27485756a06SHans Verkuil 	struct sg_dma_descriptor *d = this->last_desc_virt;
27585756a06SHans Verkuil 	u32 direction = d->next_l & WRITE_TO_PCI;
27685756a06SHans Verkuil 
27785756a06SHans Verkuil 	if (next == NULL) {
27885756a06SHans Verkuil 		d->next_h = 0;
27985756a06SHans Verkuil 		d->next_l = direction | INTERRUPT_ENABLE | END_OF_CHAIN;
28085756a06SHans Verkuil 	} else {
28140c942a1SHans Verkuil 		d->next_h = (u32)((u64)next->bus >> 32);
28285756a06SHans Verkuil 		d->next_l = (u32)next->bus | direction | INTERRUPT_ENABLE;
28385756a06SHans Verkuil 	}
28485756a06SHans Verkuil }
28585756a06SHans Verkuil 
descriptor_list_allocate(struct sg_dma_desc_info * desc,size_t bytes)28685756a06SHans Verkuil void *descriptor_list_allocate(struct sg_dma_desc_info *desc, size_t bytes)
28785756a06SHans Verkuil {
28885756a06SHans Verkuil 	desc->size = bytes;
28985756a06SHans Verkuil 	desc->virt = dma_alloc_coherent(desc->dev, bytes,
29085756a06SHans Verkuil 					&desc->bus, GFP_KERNEL);
29185756a06SHans Verkuil 	return desc->virt;
29285756a06SHans Verkuil }
29385756a06SHans Verkuil 
descriptor_list_free(struct sg_dma_desc_info * desc)29485756a06SHans Verkuil void descriptor_list_free(struct sg_dma_desc_info *desc)
29585756a06SHans Verkuil {
29685756a06SHans Verkuil 	if (desc->virt)
29785756a06SHans Verkuil 		dma_free_coherent(desc->dev, desc->size,
29885756a06SHans Verkuil 				  desc->virt, desc->bus);
29985756a06SHans Verkuil 	desc->virt = NULL;
30085756a06SHans Verkuil }
30185756a06SHans Verkuil 
descriptor_list_interrupt_enable(struct sg_dma_desc_info * desc)30285756a06SHans Verkuil void descriptor_list_interrupt_enable(struct sg_dma_desc_info *desc)
30385756a06SHans Verkuil {
30485756a06SHans Verkuil 	struct sg_dma_descriptor *d = desc->last_desc_virt;
30585756a06SHans Verkuil 
30685756a06SHans Verkuil 	d->next_l |= INTERRUPT_ENABLE;
30785756a06SHans Verkuil }
30885756a06SHans Verkuil 
descriptor_list_interrupt_disable(struct sg_dma_desc_info * desc)30985756a06SHans Verkuil void descriptor_list_interrupt_disable(struct sg_dma_desc_info *desc)
31085756a06SHans Verkuil {
31185756a06SHans Verkuil 	struct sg_dma_descriptor *d = desc->last_desc_virt;
31285756a06SHans Verkuil 
31385756a06SHans Verkuil 	d->next_l &= ~INTERRUPT_ENABLE;
31485756a06SHans Verkuil }
31585756a06SHans Verkuil 
descriptor_list_loopback(struct sg_dma_desc_info * desc)31685756a06SHans Verkuil void descriptor_list_loopback(struct sg_dma_desc_info *desc)
31785756a06SHans Verkuil {
31885756a06SHans Verkuil 	struct sg_dma_descriptor *d = desc->last_desc_virt;
31985756a06SHans Verkuil 
32040c942a1SHans Verkuil 	d->next_h = (u32)((u64)desc->bus >> 32);
32185756a06SHans Verkuil 	d->next_l = (u32)desc->bus | (d->next_l & DESCRIPTOR_FLAG_MSK);
32285756a06SHans Verkuil }
32385756a06SHans Verkuil 
descriptor_list_end_of_chain(struct sg_dma_desc_info * desc)32485756a06SHans Verkuil void descriptor_list_end_of_chain(struct sg_dma_desc_info *desc)
32585756a06SHans Verkuil {
32685756a06SHans Verkuil 	struct sg_dma_descriptor *d = desc->last_desc_virt;
32785756a06SHans Verkuil 
32885756a06SHans Verkuil 	d->next_l |= END_OF_CHAIN;
32985756a06SHans Verkuil }
330