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(®ion, 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