1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
|
From 4e62a443aa8ffecca7918db93329d8bd8210a92c Mon Sep 17 00:00:00 2001
From: Phil Elwell <phil@raspberrypi.org>
Date: Thu, 20 Aug 2015 13:50:18 +0100
Subject: [PATCH 165/171] bcm2708-dmaengine: Use more DMA channels (but not 12)
1) Only the bcm2708_fb drivers uses the legacy DMA API, and
it requires a BULK-capable channel, so all other types
(FAST, NORMAL and LITE) can be made available to the regular
DMA API.
2) DMA channels 11-14 share an interrupt. The driver can't
handle this, so don't use channels 12-14 (12 was used, probably
because it appears to have an interrupt, but in reality that
interrupt is for activity on ANY channel). This may explain
a lockup encountered when running out of DMA channels.
The combined effect of this patch is to leave 7 DMA channels
available + channel 0 for bcm2708_fb via the legacy API.
See: https://github.com/raspberrypi/linux/issues/1110
https://github.com/raspberrypi/linux/issues/1108
---
arch/arm/boot/dts/bcm2708_common.dtsi | 5 ++--
drivers/dma/bcm2708-dmaengine.c | 43 +++++++++++++++++++++++------------
2 files changed, 31 insertions(+), 17 deletions(-)
--- a/arch/arm/boot/dts/bcm2708_common.dtsi
+++ b/arch/arm/boot/dts/bcm2708_common.dtsi
@@ -59,11 +59,10 @@
<1 24>,
<1 25>,
<1 26>,
- <1 27>,
- <1 28>;
+ <1 27>;
#dma-cells = <1>;
- brcm,dma-channel-mask = <0x7f35>;
+ brcm,dma-channel-mask = <0x0f35>;
};
intc: interrupt-controller {
--- a/drivers/dma/bcm2708-dmaengine.c
+++ b/drivers/dma/bcm2708-dmaengine.c
@@ -184,7 +184,7 @@ static void vc_dmaman_init(struct vc_dma
}
static int vc_dmaman_chan_alloc(struct vc_dmaman *dmaman,
- unsigned preferred_feature_set)
+ unsigned required_feature_set)
{
u32 chans;
int chan = 0;
@@ -193,10 +193,8 @@ static int vc_dmaman_chan_alloc(struct v
chans = dmaman->chan_available;
for (feature = 0; feature < BCM_DMA_FEATURE_COUNT; feature++)
/* select the subset of available channels with the desired
- feature so long as some of the candidate channels have that
- feature */
- if ((preferred_feature_set & (1 << feature)) &&
- (chans & dmaman->has_feature[feature]))
+ features */
+ if (required_feature_set & (1 << feature))
chans &= dmaman->has_feature[feature];
if (!chans)
@@ -228,7 +226,7 @@ static int vc_dmaman_chan_free(struct vc
/* DMA Manager Monitor */
-extern int bcm_dma_chan_alloc(unsigned preferred_feature_set,
+extern int bcm_dma_chan_alloc(unsigned required_feature_set,
void __iomem **out_dma_base, int *out_dma_irq)
{
struct vc_dmaman *dmaman = g_dmaman;
@@ -240,7 +238,7 @@ extern int bcm_dma_chan_alloc(unsigned p
return -ENODEV;
mutex_lock(&dmaman->lock);
- chan = vc_dmaman_chan_alloc(dmaman, preferred_feature_set);
+ chan = vc_dmaman_chan_alloc(dmaman, required_feature_set);
if (chan < 0)
goto out;
@@ -442,6 +440,7 @@ static inline struct bcm2835_desc *to_bc
return container_of(t, struct bcm2835_desc, vd.tx);
}
+#if 0
static void dma_dumpregs(struct bcm2835_chan *c)
{
pr_debug("-------------DMA DUMPREGS-------------\n");
@@ -457,6 +456,7 @@ static void dma_dumpregs(struct bcm2835_
readl(c->chan_base + BCM2835_DMA_NEXTCB));
pr_debug("--------------------------------------\n");
}
+#endif
static void bcm2835_dma_desc_free(struct virt_dma_desc *vd)
{
@@ -862,6 +862,7 @@ static struct dma_async_tx_descriptor *b
uint32_t len = sg_dma_len(sgent);
for (j = 0; j < len; j += max_size) {
+ u32 waits;
struct bcm2835_dma_cb *control_block =
&d->control_block_base[i+splitct];
@@ -879,7 +880,7 @@ static struct dma_async_tx_descriptor *b
}
/* Common part */
- u32 waits = SDHCI_BCM_DMA_WAITS;
+ waits = SDHCI_BCM_DMA_WAITS;
if ((dma_debug >> 0) & 0x1f)
waits = (dma_debug >> 0) & 0x1f;
control_block->info |= BCM2835_DMA_WAITS(waits);
@@ -1074,6 +1075,14 @@ static int bcm2835_dma_probe(struct plat
int rc;
int i;
int irq;
+#ifdef CONFIG_DMA_BCM2708_LEGACY
+ static const u32 wanted_features[] = {
+ BCM_DMA_FEATURE_FAST,
+ BCM_DMA_FEATURE_NORMAL,
+ BCM_DMA_FEATURE_LITE
+ };
+ int j;
+#endif
if (!pdev->dev.dma_mask)
@@ -1120,20 +1129,24 @@ static int bcm2835_dma_probe(struct plat
platform_set_drvdata(pdev, od);
- for (i = 0; i < 5; i++) {
+ for (i = 0, j = 0; j < ARRAY_SIZE(wanted_features);) {
+
void __iomem *chan_base;
int chan_id;
- chan_id = bcm_dma_chan_alloc(BCM_DMA_FEATURE_LITE,
- &chan_base,
- &irq);
-
- if (chan_id < 0)
- break;
+ chan_id = bcm_dma_chan_alloc(wanted_features[j],
+ &chan_base,
+ &irq);
+
+ if (chan_id < 0) {
+ j++;
+ continue;
+ }
rc = bcm2708_dma_chan_init(od, chan_base, chan_id, irq);
if (rc)
goto err_no_dma;
+ i++;
}
if (pdev->dev.of_node) {
@@ -1146,6 +1159,8 @@ static int bcm2835_dma_probe(struct plat
}
}
+ dev_info(&pdev->dev, "Initialized %i DMA channels (+ 1 legacy)\n", i);
+
#else
rc = dma_set_mask_and_coherent(&pdev->dev, DMA_BIT_MASK(32));
if (rc)
|