xref: /linux/drivers/hwtracing/coresight/coresight-tmc-etr.c (revision 0883c2c06fb5bcf5b9e008270827e63c09a88c1e)
1 /*
2  * Copyright(C) 2016 Linaro Limited. All rights reserved.
3  * Author: Mathieu Poirier <mathieu.poirier@linaro.org>
4  *
5  * This program is free software; you can redistribute it and/or modify it
6  * under the terms of the GNU General Public License version 2 as published by
7  * the Free Software Foundation.
8  *
9  * This program is distributed in the hope that it will be useful, but WITHOUT
10  * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
11  * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License for
12  * more details.
13  *
14  * You should have received a copy of the GNU General Public License along with
15  * this program.  If not, see <http://www.gnu.org/licenses/>.
16  */
17 
18 #include <linux/coresight.h>
19 #include <linux/dma-mapping.h>
20 #include "coresight-priv.h"
21 #include "coresight-tmc.h"
22 
23 void tmc_etr_enable_hw(struct tmc_drvdata *drvdata)
24 {
25 	u32 axictl;
26 
27 	/* Zero out the memory to help with debug */
28 	memset(drvdata->vaddr, 0, drvdata->size);
29 
30 	CS_UNLOCK(drvdata->base);
31 
32 	/* Wait for TMCSReady bit to be set */
33 	tmc_wait_for_tmcready(drvdata);
34 
35 	writel_relaxed(drvdata->size / 4, drvdata->base + TMC_RSZ);
36 	writel_relaxed(TMC_MODE_CIRCULAR_BUFFER, drvdata->base + TMC_MODE);
37 
38 	axictl = readl_relaxed(drvdata->base + TMC_AXICTL);
39 	axictl |= TMC_AXICTL_WR_BURST_16;
40 	writel_relaxed(axictl, drvdata->base + TMC_AXICTL);
41 	axictl &= ~TMC_AXICTL_SCT_GAT_MODE;
42 	writel_relaxed(axictl, drvdata->base + TMC_AXICTL);
43 	axictl = (axictl &
44 		  ~(TMC_AXICTL_PROT_CTL_B0 | TMC_AXICTL_PROT_CTL_B1)) |
45 		  TMC_AXICTL_PROT_CTL_B1;
46 	writel_relaxed(axictl, drvdata->base + TMC_AXICTL);
47 
48 	writel_relaxed(drvdata->paddr, drvdata->base + TMC_DBALO);
49 	writel_relaxed(0x0, drvdata->base + TMC_DBAHI);
50 	writel_relaxed(TMC_FFCR_EN_FMT | TMC_FFCR_EN_TI |
51 		       TMC_FFCR_FON_FLIN | TMC_FFCR_FON_TRIG_EVT |
52 		       TMC_FFCR_TRIGON_TRIGIN,
53 		       drvdata->base + TMC_FFCR);
54 	writel_relaxed(drvdata->trigger_cntr, drvdata->base + TMC_TRG);
55 	tmc_enable_hw(drvdata);
56 
57 	CS_LOCK(drvdata->base);
58 }
59 
60 static void tmc_etr_dump_hw(struct tmc_drvdata *drvdata)
61 {
62 	u32 rwp, val;
63 
64 	rwp = readl_relaxed(drvdata->base + TMC_RWP);
65 	val = readl_relaxed(drvdata->base + TMC_STS);
66 
67 	/* How much memory do we still have */
68 	if (val & BIT(0))
69 		drvdata->buf = drvdata->vaddr + rwp - drvdata->paddr;
70 	else
71 		drvdata->buf = drvdata->vaddr;
72 }
73 
74 static void tmc_etr_disable_hw(struct tmc_drvdata *drvdata)
75 {
76 	CS_UNLOCK(drvdata->base);
77 
78 	tmc_flush_and_stop(drvdata);
79 	/*
80 	 * When operating in sysFS mode the content of the buffer needs to be
81 	 * read before the TMC is disabled.
82 	 */
83 	if (local_read(&drvdata->mode) == CS_MODE_SYSFS)
84 		tmc_etr_dump_hw(drvdata);
85 	tmc_disable_hw(drvdata);
86 
87 	CS_LOCK(drvdata->base);
88 }
89 
90 static int tmc_enable_etr_sink_sysfs(struct coresight_device *csdev, u32 mode)
91 {
92 	int ret = 0;
93 	bool used = false;
94 	long val;
95 	unsigned long flags;
96 	void __iomem *vaddr = NULL;
97 	dma_addr_t paddr;
98 	struct tmc_drvdata *drvdata = dev_get_drvdata(csdev->dev.parent);
99 
100 	 /* This shouldn't be happening */
101 	if (WARN_ON(mode != CS_MODE_SYSFS))
102 		return -EINVAL;
103 
104 	/*
105 	 * If we don't have a buffer release the lock and allocate memory.
106 	 * Otherwise keep the lock and move along.
107 	 */
108 	spin_lock_irqsave(&drvdata->spinlock, flags);
109 	if (!drvdata->vaddr) {
110 		spin_unlock_irqrestore(&drvdata->spinlock, flags);
111 
112 		/*
113 		 * Contiguous  memory can't be allocated while a spinlock is
114 		 * held.  As such allocate memory here and free it if a buffer
115 		 * has already been allocated (from a previous session).
116 		 */
117 		vaddr = dma_alloc_coherent(drvdata->dev, drvdata->size,
118 					   &paddr, GFP_KERNEL);
119 		if (!vaddr)
120 			return -ENOMEM;
121 
122 		/* Let's try again */
123 		spin_lock_irqsave(&drvdata->spinlock, flags);
124 	}
125 
126 	if (drvdata->reading) {
127 		ret = -EBUSY;
128 		goto out;
129 	}
130 
131 	val = local_xchg(&drvdata->mode, mode);
132 	/*
133 	 * In sysFS mode we can have multiple writers per sink.  Since this
134 	 * sink is already enabled no memory is needed and the HW need not be
135 	 * touched.
136 	 */
137 	if (val == CS_MODE_SYSFS)
138 		goto out;
139 
140 	/*
141 	 * If drvdata::buf == NULL, use the memory allocated above.
142 	 * Otherwise a buffer still exists from a previous session, so
143 	 * simply use that.
144 	 */
145 	if (drvdata->buf == NULL) {
146 		used = true;
147 		drvdata->vaddr = vaddr;
148 		drvdata->paddr = paddr;
149 		drvdata->buf = drvdata->vaddr;
150 	}
151 
152 	memset(drvdata->vaddr, 0, drvdata->size);
153 
154 	tmc_etr_enable_hw(drvdata);
155 out:
156 	spin_unlock_irqrestore(&drvdata->spinlock, flags);
157 
158 	/* Free memory outside the spinlock if need be */
159 	if (!used && vaddr)
160 		dma_free_coherent(drvdata->dev, drvdata->size, vaddr, paddr);
161 
162 	if (!ret)
163 		dev_info(drvdata->dev, "TMC-ETR enabled\n");
164 
165 	return ret;
166 }
167 
168 static int tmc_enable_etr_sink_perf(struct coresight_device *csdev, u32 mode)
169 {
170 	int ret = 0;
171 	long val;
172 	unsigned long flags;
173 	struct tmc_drvdata *drvdata = dev_get_drvdata(csdev->dev.parent);
174 
175 	 /* This shouldn't be happening */
176 	if (WARN_ON(mode != CS_MODE_PERF))
177 		return -EINVAL;
178 
179 	spin_lock_irqsave(&drvdata->spinlock, flags);
180 	if (drvdata->reading) {
181 		ret = -EINVAL;
182 		goto out;
183 	}
184 
185 	val = local_xchg(&drvdata->mode, mode);
186 	/*
187 	 * In Perf mode there can be only one writer per sink.  There
188 	 * is also no need to continue if the ETR is already operated
189 	 * from sysFS.
190 	 */
191 	if (val != CS_MODE_DISABLED) {
192 		ret = -EINVAL;
193 		goto out;
194 	}
195 
196 	tmc_etr_enable_hw(drvdata);
197 out:
198 	spin_unlock_irqrestore(&drvdata->spinlock, flags);
199 
200 	return ret;
201 }
202 
203 static int tmc_enable_etr_sink(struct coresight_device *csdev, u32 mode)
204 {
205 	switch (mode) {
206 	case CS_MODE_SYSFS:
207 		return tmc_enable_etr_sink_sysfs(csdev, mode);
208 	case CS_MODE_PERF:
209 		return tmc_enable_etr_sink_perf(csdev, mode);
210 	}
211 
212 	/* We shouldn't be here */
213 	return -EINVAL;
214 }
215 
216 static void tmc_disable_etr_sink(struct coresight_device *csdev)
217 {
218 	long val;
219 	unsigned long flags;
220 	struct tmc_drvdata *drvdata = dev_get_drvdata(csdev->dev.parent);
221 
222 	spin_lock_irqsave(&drvdata->spinlock, flags);
223 	if (drvdata->reading) {
224 		spin_unlock_irqrestore(&drvdata->spinlock, flags);
225 		return;
226 	}
227 
228 	val = local_xchg(&drvdata->mode, CS_MODE_DISABLED);
229 	/* Disable the TMC only if it needs to */
230 	if (val != CS_MODE_DISABLED)
231 		tmc_etr_disable_hw(drvdata);
232 
233 	spin_unlock_irqrestore(&drvdata->spinlock, flags);
234 
235 	dev_info(drvdata->dev, "TMC-ETR disabled\n");
236 }
237 
238 static const struct coresight_ops_sink tmc_etr_sink_ops = {
239 	.enable		= tmc_enable_etr_sink,
240 	.disable	= tmc_disable_etr_sink,
241 };
242 
243 const struct coresight_ops tmc_etr_cs_ops = {
244 	.sink_ops	= &tmc_etr_sink_ops,
245 };
246 
247 int tmc_read_prepare_etr(struct tmc_drvdata *drvdata)
248 {
249 	int ret = 0;
250 	long val;
251 	unsigned long flags;
252 
253 	/* config types are set a boot time and never change */
254 	if (WARN_ON_ONCE(drvdata->config_type != TMC_CONFIG_TYPE_ETR))
255 		return -EINVAL;
256 
257 	spin_lock_irqsave(&drvdata->spinlock, flags);
258 	if (drvdata->reading) {
259 		ret = -EBUSY;
260 		goto out;
261 	}
262 
263 	val = local_read(&drvdata->mode);
264 	/* Don't interfere if operated from Perf */
265 	if (val == CS_MODE_PERF) {
266 		ret = -EINVAL;
267 		goto out;
268 	}
269 
270 	/* If drvdata::buf is NULL the trace data has been read already */
271 	if (drvdata->buf == NULL) {
272 		ret = -EINVAL;
273 		goto out;
274 	}
275 
276 	/* Disable the TMC if need be */
277 	if (val == CS_MODE_SYSFS)
278 		tmc_etr_disable_hw(drvdata);
279 
280 	drvdata->reading = true;
281 out:
282 	spin_unlock_irqrestore(&drvdata->spinlock, flags);
283 
284 	return ret;
285 }
286 
287 int tmc_read_unprepare_etr(struct tmc_drvdata *drvdata)
288 {
289 	unsigned long flags;
290 	dma_addr_t paddr;
291 	void __iomem *vaddr = NULL;
292 
293 	/* config types are set a boot time and never change */
294 	if (WARN_ON_ONCE(drvdata->config_type != TMC_CONFIG_TYPE_ETR))
295 		return -EINVAL;
296 
297 	spin_lock_irqsave(&drvdata->spinlock, flags);
298 
299 	/* RE-enable the TMC if need be */
300 	if (local_read(&drvdata->mode) == CS_MODE_SYSFS) {
301 		/*
302 		 * The trace run will continue with the same allocated trace
303 		 * buffer. As such zero-out the buffer so that we don't end
304 		 * up with stale data.
305 		 *
306 		 * Since the tracer is still enabled drvdata::buf
307 		 * can't be NULL.
308 		 */
309 		memset(drvdata->buf, 0, drvdata->size);
310 		tmc_etr_enable_hw(drvdata);
311 	} else {
312 		/*
313 		 * The ETR is not tracing and the buffer was just read.
314 		 * As such prepare to free the trace buffer.
315 		 */
316 		vaddr = drvdata->vaddr;
317 		paddr = drvdata->paddr;
318 		drvdata->buf = NULL;
319 	}
320 
321 	drvdata->reading = false;
322 	spin_unlock_irqrestore(&drvdata->spinlock, flags);
323 
324 	/* Free allocated memory out side of the spinlock */
325 	if (vaddr)
326 		dma_free_coherent(drvdata->dev, drvdata->size, vaddr, paddr);
327 
328 	return 0;
329 }
330