xref: /linux/drivers/remoteproc/qcom_common.c (revision 743cfae79d2458e241b06ed523c28a09f1449b75)
1 // SPDX-License-Identifier: GPL-2.0-only
2 /*
3  * Qualcomm Peripheral Image Loader helpers
4  *
5  * Copyright (C) 2016 Linaro Ltd
6  * Copyright (C) 2015 Sony Mobile Communications Inc
7  * Copyright (c) 2012-2013, The Linux Foundation. All rights reserved.
8  */
9 
10 #include <linux/firmware.h>
11 #include <linux/kernel.h>
12 #include <linux/module.h>
13 #include <linux/notifier.h>
14 #include <linux/remoteproc.h>
15 #include <linux/remoteproc/qcom_rproc.h>
16 #include <linux/auxiliary_bus.h>
17 #include <linux/rpmsg/qcom_glink.h>
18 #include <linux/rpmsg/qcom_smd.h>
19 #include <linux/slab.h>
20 #include <linux/soc/qcom/mdt_loader.h>
21 #include <linux/soc/qcom/smem.h>
22 
23 #include "remoteproc_internal.h"
24 #include "qcom_common.h"
25 
26 #define to_glink_subdev(d) container_of(d, struct qcom_rproc_glink, subdev)
27 #define to_smd_subdev(d) container_of(d, struct qcom_rproc_subdev, subdev)
28 #define to_ssr_subdev(d) container_of(d, struct qcom_rproc_ssr, subdev)
29 #define to_pdm_subdev(d) container_of(d, struct qcom_rproc_pdm, subdev)
30 
31 #define MAX_REGION_NAME_LENGTH  16
32 #define SBL_MINIDUMP_SMEM_ID	602
33 #define MINIDUMP_REGION_VALID		('V' << 24 | 'A' << 16 | 'L' << 8 | 'I' << 0)
34 #define MINIDUMP_SS_ENCR_DONE		('D' << 24 | 'O' << 16 | 'N' << 8 | 'E' << 0)
35 #define MINIDUMP_SS_ENABLED		('E' << 24 | 'N' << 16 | 'B' << 8 | 'L' << 0)
36 
37 /**
38  * struct minidump_region - Minidump region
39  * @name		: Name of the region to be dumped
40  * @seq_num:		: Use to differentiate regions with same name.
41  * @valid		: This entry to be dumped (if set to 1)
42  * @address		: Physical address of region to be dumped
43  * @size		: Size of the region
44  */
45 struct minidump_region {
46 	char	name[MAX_REGION_NAME_LENGTH];
47 	__le32	seq_num;
48 	__le32	valid;
49 	__le64	address;
50 	__le64	size;
51 };
52 
53 /**
54  * struct minidump_subsystem - Subsystem's SMEM Table of content
55  * @status : Subsystem toc init status
56  * @enabled : if set to 1, this region would be copied during coredump
57  * @encryption_status: Encryption status for this subsystem
58  * @encryption_required : Decides to encrypt the subsystem regions or not
59  * @region_count : Number of regions added in this subsystem toc
60  * @regions_baseptr : regions base pointer of the subsystem
61  */
62 struct minidump_subsystem {
63 	__le32	status;
64 	__le32	enabled;
65 	__le32	encryption_status;
66 	__le32	encryption_required;
67 	__le32	region_count;
68 	__le64	regions_baseptr;
69 };
70 
71 /**
72  * struct minidump_global_toc - Global Table of Content
73  * @status : Global Minidump init status
74  * @md_revision : Minidump revision
75  * @enabled : Minidump enable status
76  * @subsystems : Array of subsystems toc
77  */
78 struct minidump_global_toc {
79 	__le32				status;
80 	__le32				md_revision;
81 	__le32				enabled;
82 	struct minidump_subsystem	subsystems[];
83 };
84 
85 struct qcom_ssr_subsystem {
86 	const char *name;
87 	struct srcu_notifier_head notifier_list;
88 	struct list_head list;
89 };
90 
91 static LIST_HEAD(qcom_ssr_subsystem_list);
92 static DEFINE_MUTEX(qcom_ssr_subsys_lock);
93 
94 static void qcom_minidump_cleanup(struct rproc *rproc)
95 {
96 	struct rproc_dump_segment *entry, *tmp;
97 
98 	list_for_each_entry_safe(entry, tmp, &rproc->dump_segments, node) {
99 		list_del(&entry->node);
100 		kfree(entry->priv);
101 		kfree(entry);
102 	}
103 }
104 
105 static int qcom_add_minidump_segments(struct rproc *rproc, struct minidump_subsystem *subsystem,
106 			void (*rproc_dumpfn_t)(struct rproc *rproc, struct rproc_dump_segment *segment,
107 				void *dest, size_t offset, size_t size))
108 {
109 	struct minidump_region __iomem *ptr;
110 	struct minidump_region region;
111 	int seg_cnt, i;
112 	dma_addr_t da;
113 	size_t size;
114 	char *name;
115 
116 	if (WARN_ON(!list_empty(&rproc->dump_segments))) {
117 		dev_err(&rproc->dev, "dump segment list already populated\n");
118 		return -EUCLEAN;
119 	}
120 
121 	seg_cnt = le32_to_cpu(subsystem->region_count);
122 	ptr = ioremap((unsigned long)le64_to_cpu(subsystem->regions_baseptr),
123 		      seg_cnt * sizeof(struct minidump_region));
124 	if (!ptr)
125 		return -EFAULT;
126 
127 	for (i = 0; i < seg_cnt; i++) {
128 		memcpy_fromio(&region, ptr + i, sizeof(region));
129 		if (le32_to_cpu(region.valid) == MINIDUMP_REGION_VALID) {
130 			name = kstrndup(region.name, MAX_REGION_NAME_LENGTH - 1, GFP_KERNEL);
131 			if (!name) {
132 				iounmap(ptr);
133 				return -ENOMEM;
134 			}
135 			da = le64_to_cpu(region.address);
136 			size = le64_to_cpu(region.size);
137 			rproc_coredump_add_custom_segment(rproc, da, size, rproc_dumpfn_t, name);
138 		}
139 	}
140 
141 	iounmap(ptr);
142 	return 0;
143 }
144 
145 void qcom_minidump(struct rproc *rproc, unsigned int minidump_id,
146 		void (*rproc_dumpfn_t)(struct rproc *rproc,
147 		struct rproc_dump_segment *segment, void *dest, size_t offset,
148 		size_t size))
149 {
150 	int ret;
151 	struct minidump_subsystem *subsystem;
152 	struct minidump_global_toc *toc;
153 	unsigned int num_ss;
154 	size_t toc_size;
155 
156 	/* Get Global minidump ToC*/
157 	toc = qcom_smem_get(QCOM_SMEM_HOST_ANY, SBL_MINIDUMP_SMEM_ID, &toc_size);
158 
159 	/* check if global table pointer exists and init is set */
160 	if (IS_ERR(toc) || !toc->status) {
161 		dev_err(&rproc->dev, "Minidump TOC not found in SMEM\n");
162 		return;
163 	}
164 
165 	/* Derive the number of subsystems from the actual SMEM item size */
166 	num_ss = (toc_size - offsetof(struct minidump_global_toc, subsystems)) /
167 		 sizeof(struct minidump_subsystem);
168 
169 	if (minidump_id >= num_ss) {
170 		dev_err(&rproc->dev, "Minidump id %d is out of range: %d\n",
171 			minidump_id, num_ss);
172 		return;
173 	}
174 
175 	/* Get subsystem table of contents using the minidump id */
176 	subsystem = &toc->subsystems[minidump_id];
177 
178 	/**
179 	 * Collect minidump if SS ToC is valid and segment table
180 	 * is initialized in memory and encryption status is set.
181 	 */
182 	if (subsystem->regions_baseptr == 0 ||
183 	    le32_to_cpu(subsystem->status) != 1 ||
184 	    le32_to_cpu(subsystem->enabled) != MINIDUMP_SS_ENABLED) {
185 		return rproc_coredump(rproc);
186 	}
187 
188 	if (le32_to_cpu(subsystem->encryption_status) != MINIDUMP_SS_ENCR_DONE) {
189 		dev_err(&rproc->dev, "Minidump not ready, skipping\n");
190 		return;
191 	}
192 
193 	/**
194 	 * Clear out the dump segments populated by parse_fw before
195 	 * re-populating them with minidump segments.
196 	 */
197 	rproc_coredump_cleanup(rproc);
198 
199 	ret = qcom_add_minidump_segments(rproc, subsystem, rproc_dumpfn_t);
200 	if (ret) {
201 		dev_err(&rproc->dev, "Failed with error: %d while adding minidump entries\n", ret);
202 		goto clean_minidump;
203 	}
204 	rproc_coredump_using_sections(rproc);
205 clean_minidump:
206 	qcom_minidump_cleanup(rproc);
207 }
208 EXPORT_SYMBOL_GPL(qcom_minidump);
209 
210 static int glink_subdev_start(struct rproc_subdev *subdev)
211 {
212 	struct qcom_rproc_glink *glink = to_glink_subdev(subdev);
213 
214 	glink->edge = qcom_glink_smem_register(glink->dev, glink->node);
215 
216 	return PTR_ERR_OR_ZERO(glink->edge);
217 }
218 
219 static void glink_subdev_stop(struct rproc_subdev *subdev, bool crashed)
220 {
221 	struct qcom_rproc_glink *glink = to_glink_subdev(subdev);
222 
223 	qcom_glink_smem_unregister(glink->edge);
224 	glink->edge = NULL;
225 }
226 
227 static void glink_subdev_unprepare(struct rproc_subdev *subdev)
228 {
229 	struct qcom_rproc_glink *glink = to_glink_subdev(subdev);
230 
231 	qcom_glink_ssr_notify(glink->ssr_name);
232 }
233 
234 /**
235  * qcom_add_glink_subdev() - try to add a GLINK subdevice to rproc
236  * @rproc:	rproc handle to parent the subdevice
237  * @glink:	reference to a GLINK subdev context
238  * @ssr_name:	identifier of the associated remoteproc for ssr notifications
239  */
240 void qcom_add_glink_subdev(struct rproc *rproc, struct qcom_rproc_glink *glink,
241 			   const char *ssr_name)
242 {
243 	struct device *dev = &rproc->dev;
244 
245 	glink->node = of_get_child_by_name(dev->parent->of_node, "glink-edge");
246 	if (!glink->node)
247 		return;
248 
249 	glink->ssr_name = kstrdup_const(ssr_name, GFP_KERNEL);
250 	if (!glink->ssr_name)
251 		return;
252 
253 	glink->dev = dev;
254 	glink->subdev.start = glink_subdev_start;
255 	glink->subdev.stop = glink_subdev_stop;
256 	glink->subdev.unprepare = glink_subdev_unprepare;
257 
258 	rproc_add_subdev(rproc, &glink->subdev);
259 }
260 EXPORT_SYMBOL_GPL(qcom_add_glink_subdev);
261 
262 /**
263  * qcom_remove_glink_subdev() - remove a GLINK subdevice from rproc
264  * @rproc:	rproc handle
265  * @glink:	reference to a GLINK subdev context
266  */
267 void qcom_remove_glink_subdev(struct rproc *rproc, struct qcom_rproc_glink *glink)
268 {
269 	if (!glink->node)
270 		return;
271 
272 	rproc_remove_subdev(rproc, &glink->subdev);
273 	kfree_const(glink->ssr_name);
274 	of_node_put(glink->node);
275 }
276 EXPORT_SYMBOL_GPL(qcom_remove_glink_subdev);
277 
278 /**
279  * qcom_register_dump_segments() - register segments for coredump
280  * @rproc:	remoteproc handle
281  * @fw:		firmware header
282  *
283  * Register all segments of the ELF in the remoteproc coredump segment list
284  *
285  * Return: 0 on success, negative errno on failure.
286  */
287 int qcom_register_dump_segments(struct rproc *rproc,
288 				const struct firmware *fw)
289 {
290 	const struct elf32_phdr *phdrs;
291 	const struct elf32_phdr *phdr;
292 	const struct elf32_hdr *ehdr;
293 	int ret;
294 	int i;
295 
296 	ehdr = (struct elf32_hdr *)fw->data;
297 	phdrs = (struct elf32_phdr *)(ehdr + 1);
298 
299 	for (i = 0; i < ehdr->e_phnum; i++) {
300 		phdr = &phdrs[i];
301 
302 		if (phdr->p_type != PT_LOAD)
303 			continue;
304 
305 		if ((phdr->p_flags & QCOM_MDT_TYPE_MASK) == QCOM_MDT_TYPE_HASH)
306 			continue;
307 
308 		if (!phdr->p_memsz)
309 			continue;
310 
311 		ret = rproc_coredump_add_segment(rproc, phdr->p_paddr,
312 						 phdr->p_memsz);
313 		if (ret)
314 			return ret;
315 	}
316 
317 	return 0;
318 }
319 EXPORT_SYMBOL_GPL(qcom_register_dump_segments);
320 
321 static int smd_subdev_start(struct rproc_subdev *subdev)
322 {
323 	struct qcom_rproc_subdev *smd = to_smd_subdev(subdev);
324 
325 	smd->edge = qcom_smd_register_edge(smd->dev, smd->node);
326 
327 	return PTR_ERR_OR_ZERO(smd->edge);
328 }
329 
330 static void smd_subdev_stop(struct rproc_subdev *subdev, bool crashed)
331 {
332 	struct qcom_rproc_subdev *smd = to_smd_subdev(subdev);
333 
334 	qcom_smd_unregister_edge(smd->edge);
335 	smd->edge = NULL;
336 }
337 
338 /**
339  * qcom_add_smd_subdev() - try to add a SMD subdevice to rproc
340  * @rproc:	rproc handle to parent the subdevice
341  * @smd:	reference to a Qualcomm subdev context
342  */
343 void qcom_add_smd_subdev(struct rproc *rproc, struct qcom_rproc_subdev *smd)
344 {
345 	struct device *dev = &rproc->dev;
346 
347 	smd->node = of_get_child_by_name(dev->parent->of_node, "smd-edge");
348 	if (!smd->node)
349 		return;
350 
351 	smd->dev = dev;
352 	smd->subdev.start = smd_subdev_start;
353 	smd->subdev.stop = smd_subdev_stop;
354 
355 	rproc_add_subdev(rproc, &smd->subdev);
356 }
357 EXPORT_SYMBOL_GPL(qcom_add_smd_subdev);
358 
359 /**
360  * qcom_remove_smd_subdev() - remove the smd subdevice from rproc
361  * @rproc:	rproc handle
362  * @smd:	the SMD subdevice to remove
363  */
364 void qcom_remove_smd_subdev(struct rproc *rproc, struct qcom_rproc_subdev *smd)
365 {
366 	if (!smd->node)
367 		return;
368 
369 	rproc_remove_subdev(rproc, &smd->subdev);
370 	of_node_put(smd->node);
371 }
372 EXPORT_SYMBOL_GPL(qcom_remove_smd_subdev);
373 
374 static struct qcom_ssr_subsystem *qcom_ssr_get_subsys(const char *name)
375 {
376 	struct qcom_ssr_subsystem *info;
377 
378 	mutex_lock(&qcom_ssr_subsys_lock);
379 	/* Match in the global qcom_ssr_subsystem_list with name */
380 	list_for_each_entry(info, &qcom_ssr_subsystem_list, list)
381 		if (!strcmp(info->name, name))
382 			goto out;
383 
384 	info = kzalloc_obj(*info);
385 	if (!info) {
386 		info = ERR_PTR(-ENOMEM);
387 		goto out;
388 	}
389 	info->name = kstrdup_const(name, GFP_KERNEL);
390 	srcu_init_notifier_head(&info->notifier_list);
391 
392 	/* Add to global notification list */
393 	list_add_tail(&info->list, &qcom_ssr_subsystem_list);
394 
395 out:
396 	mutex_unlock(&qcom_ssr_subsys_lock);
397 	return info;
398 }
399 
400 /**
401  * qcom_register_ssr_notifier() - register SSR notification handler
402  * @name:	Subsystem's SSR name
403  * @nb:		notifier_block to be invoked upon subsystem's state change
404  *
405  * This registers the @nb notifier block as part the notifier chain for a
406  * remoteproc associated with @name. The notifier block's callback
407  * will be invoked when the remote processor's SSR events occur
408  * (pre/post startup and pre/post shutdown).
409  *
410  * Return: a subsystem cookie on success, ERR_PTR on failure.
411  */
412 void *qcom_register_ssr_notifier(const char *name, struct notifier_block *nb)
413 {
414 	struct qcom_ssr_subsystem *info;
415 
416 	info = qcom_ssr_get_subsys(name);
417 	if (IS_ERR(info))
418 		return info;
419 
420 	srcu_notifier_chain_register(&info->notifier_list, nb);
421 
422 	return &info->notifier_list;
423 }
424 EXPORT_SYMBOL_GPL(qcom_register_ssr_notifier);
425 
426 /**
427  * qcom_unregister_ssr_notifier() - unregister SSR notification handler
428  * @notify:	subsystem cookie returned from qcom_register_ssr_notifier
429  * @nb:		notifier_block to unregister
430  *
431  * This function will unregister the notifier from the particular notifier
432  * chain.
433  *
434  * Return: 0 on success, %ENOENT otherwise.
435  */
436 int qcom_unregister_ssr_notifier(void *notify, struct notifier_block *nb)
437 {
438 	return srcu_notifier_chain_unregister(notify, nb);
439 }
440 EXPORT_SYMBOL_GPL(qcom_unregister_ssr_notifier);
441 
442 static int ssr_notify_prepare(struct rproc_subdev *subdev)
443 {
444 	struct qcom_rproc_ssr *ssr = to_ssr_subdev(subdev);
445 	struct qcom_ssr_notify_data data = {
446 		.name = ssr->info->name,
447 		.crashed = false,
448 	};
449 
450 	srcu_notifier_call_chain(&ssr->info->notifier_list,
451 				 QCOM_SSR_BEFORE_POWERUP, &data);
452 	return 0;
453 }
454 
455 static int ssr_notify_start(struct rproc_subdev *subdev)
456 {
457 	struct qcom_rproc_ssr *ssr = to_ssr_subdev(subdev);
458 	struct qcom_ssr_notify_data data = {
459 		.name = ssr->info->name,
460 		.crashed = false,
461 	};
462 
463 	srcu_notifier_call_chain(&ssr->info->notifier_list,
464 				 QCOM_SSR_AFTER_POWERUP, &data);
465 	return 0;
466 }
467 
468 static void ssr_notify_stop(struct rproc_subdev *subdev, bool crashed)
469 {
470 	struct qcom_rproc_ssr *ssr = to_ssr_subdev(subdev);
471 	struct qcom_ssr_notify_data data = {
472 		.name = ssr->info->name,
473 		.crashed = crashed,
474 	};
475 
476 	srcu_notifier_call_chain(&ssr->info->notifier_list,
477 				 QCOM_SSR_BEFORE_SHUTDOWN, &data);
478 }
479 
480 static void ssr_notify_unprepare(struct rproc_subdev *subdev)
481 {
482 	struct qcom_rproc_ssr *ssr = to_ssr_subdev(subdev);
483 	struct qcom_ssr_notify_data data = {
484 		.name = ssr->info->name,
485 		.crashed = false,
486 	};
487 
488 	srcu_notifier_call_chain(&ssr->info->notifier_list,
489 				 QCOM_SSR_AFTER_SHUTDOWN, &data);
490 }
491 
492 /**
493  * qcom_add_ssr_subdev() - register subdevice as restart notification source
494  * @rproc:	rproc handle
495  * @ssr:	SSR subdevice handle
496  * @ssr_name:	identifier to use for notifications originating from @rproc
497  *
498  * As the @ssr is registered with the @rproc SSR events will be sent to all
499  * registered listeners for the remoteproc when it's SSR events occur
500  * (pre/post startup and pre/post shutdown).
501  */
502 void qcom_add_ssr_subdev(struct rproc *rproc, struct qcom_rproc_ssr *ssr,
503 			 const char *ssr_name)
504 {
505 	struct qcom_ssr_subsystem *info;
506 
507 	info = qcom_ssr_get_subsys(ssr_name);
508 	if (IS_ERR(info)) {
509 		dev_err(&rproc->dev, "Failed to add ssr subdevice\n");
510 		return;
511 	}
512 
513 	ssr->info = info;
514 	ssr->subdev.prepare = ssr_notify_prepare;
515 	ssr->subdev.start = ssr_notify_start;
516 	ssr->subdev.stop = ssr_notify_stop;
517 	ssr->subdev.unprepare = ssr_notify_unprepare;
518 
519 	rproc_add_subdev(rproc, &ssr->subdev);
520 }
521 EXPORT_SYMBOL_GPL(qcom_add_ssr_subdev);
522 
523 /**
524  * qcom_remove_ssr_subdev() - remove subdevice as restart notification source
525  * @rproc:	rproc handle
526  * @ssr:	SSR subdevice handle
527  */
528 void qcom_remove_ssr_subdev(struct rproc *rproc, struct qcom_rproc_ssr *ssr)
529 {
530 	rproc_remove_subdev(rproc, &ssr->subdev);
531 	ssr->info = NULL;
532 }
533 EXPORT_SYMBOL_GPL(qcom_remove_ssr_subdev);
534 
535 static void pdm_dev_release(struct device *dev)
536 {
537 	struct auxiliary_device *adev = to_auxiliary_dev(dev);
538 
539 	kfree(adev);
540 }
541 
542 static int pdm_notify_prepare(struct rproc_subdev *subdev)
543 {
544 	struct qcom_rproc_pdm *pdm = to_pdm_subdev(subdev);
545 	struct auxiliary_device *adev;
546 	int ret;
547 
548 	adev = kzalloc_obj(*adev);
549 	if (!adev)
550 		return -ENOMEM;
551 
552 	adev->dev.parent = pdm->dev;
553 	adev->dev.release = pdm_dev_release;
554 	adev->name = "pd-mapper";
555 	adev->id = pdm->index;
556 
557 	ret = auxiliary_device_init(adev);
558 	if (ret) {
559 		kfree(adev);
560 		return ret;
561 	}
562 
563 	ret = auxiliary_device_add(adev);
564 	if (ret) {
565 		auxiliary_device_uninit(adev);
566 		return ret;
567 	}
568 
569 	pdm->adev = adev;
570 
571 	return 0;
572 }
573 
574 
575 static void pdm_notify_unprepare(struct rproc_subdev *subdev)
576 {
577 	struct qcom_rproc_pdm *pdm = to_pdm_subdev(subdev);
578 
579 	if (!pdm->adev)
580 		return;
581 
582 	auxiliary_device_delete(pdm->adev);
583 	auxiliary_device_uninit(pdm->adev);
584 	pdm->adev = NULL;
585 }
586 
587 /**
588  * qcom_add_pdm_subdev() - register PD Mapper subdevice
589  * @rproc:	rproc handle
590  * @pdm:	PDM subdevice handle
591  *
592  * Register @pdm so that Protection Device mapper service is started when the
593  * DSP is started too.
594  */
595 void qcom_add_pdm_subdev(struct rproc *rproc, struct qcom_rproc_pdm *pdm)
596 {
597 	pdm->dev = &rproc->dev;
598 	pdm->index = rproc->index;
599 
600 	pdm->subdev.prepare = pdm_notify_prepare;
601 	pdm->subdev.unprepare = pdm_notify_unprepare;
602 
603 	rproc_add_subdev(rproc, &pdm->subdev);
604 }
605 EXPORT_SYMBOL_GPL(qcom_add_pdm_subdev);
606 
607 /**
608  * qcom_remove_pdm_subdev() - remove PD Mapper subdevice
609  * @rproc:	rproc handle
610  * @pdm:	PDM subdevice handle
611  *
612  * Remove the PD Mapper subdevice.
613  */
614 void qcom_remove_pdm_subdev(struct rproc *rproc, struct qcom_rproc_pdm *pdm)
615 {
616 	rproc_remove_subdev(rproc, &pdm->subdev);
617 }
618 EXPORT_SYMBOL_GPL(qcom_remove_pdm_subdev);
619 
620 MODULE_DESCRIPTION("Qualcomm Remoteproc helper driver");
621 MODULE_LICENSE("GPL v2");
622