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 int ret = 0; 113 dma_addr_t da; 114 size_t size; 115 char *name; 116 117 if (WARN_ON(!list_empty(&rproc->dump_segments))) { 118 dev_err(&rproc->dev, "dump segment list already populated\n"); 119 return -EUCLEAN; 120 } 121 122 seg_cnt = le32_to_cpu(subsystem->region_count); 123 ptr = ioremap((unsigned long)le64_to_cpu(subsystem->regions_baseptr), 124 seg_cnt * sizeof(struct minidump_region)); 125 if (!ptr) 126 return -EFAULT; 127 128 for (i = 0; i < seg_cnt; i++) { 129 memcpy_fromio(®ion, ptr + i, sizeof(region)); 130 if (le32_to_cpu(region.valid) == MINIDUMP_REGION_VALID) { 131 name = kstrndup(region.name, MAX_REGION_NAME_LENGTH - 1, GFP_KERNEL); 132 if (!name) { 133 ret = -ENOMEM; 134 break; 135 } 136 da = le64_to_cpu(region.address); 137 size = le64_to_cpu(region.size); 138 ret = rproc_coredump_add_custom_segment(rproc, da, size, rproc_dumpfn_t, 139 name); 140 if (ret) { 141 kfree(name); 142 break; 143 } 144 } 145 } 146 147 iounmap(ptr); 148 return ret; 149 } 150 151 void qcom_minidump(struct rproc *rproc, unsigned int minidump_id, 152 void (*rproc_dumpfn_t)(struct rproc *rproc, 153 struct rproc_dump_segment *segment, void *dest, size_t offset, 154 size_t size)) 155 { 156 int ret; 157 struct minidump_subsystem *subsystem; 158 struct minidump_global_toc *toc; 159 unsigned int num_ss; 160 size_t toc_size; 161 162 /* Get Global minidump ToC*/ 163 toc = qcom_smem_get(QCOM_SMEM_HOST_ANY, SBL_MINIDUMP_SMEM_ID, &toc_size); 164 165 /* check if global table pointer exists and init is set */ 166 if (IS_ERR(toc) || !toc->status) { 167 dev_err(&rproc->dev, "Minidump TOC not found in SMEM\n"); 168 return; 169 } 170 171 /* Derive the number of subsystems from the actual SMEM item size */ 172 num_ss = (toc_size - offsetof(struct minidump_global_toc, subsystems)) / 173 sizeof(struct minidump_subsystem); 174 175 if (minidump_id >= num_ss) { 176 dev_err(&rproc->dev, "Minidump id %d is out of range: %d\n", 177 minidump_id, num_ss); 178 return; 179 } 180 181 /* Get subsystem table of contents using the minidump id */ 182 subsystem = &toc->subsystems[minidump_id]; 183 184 /** 185 * Collect minidump if SS ToC is valid and segment table 186 * is initialized in memory and encryption status is set. 187 */ 188 if (subsystem->regions_baseptr == 0 || 189 le32_to_cpu(subsystem->status) != 1 || 190 le32_to_cpu(subsystem->enabled) != MINIDUMP_SS_ENABLED) { 191 return rproc_coredump(rproc); 192 } 193 194 if (le32_to_cpu(subsystem->encryption_status) != MINIDUMP_SS_ENCR_DONE) { 195 dev_err(&rproc->dev, "Minidump not ready, skipping\n"); 196 return; 197 } 198 199 /** 200 * Clear out the dump segments populated by parse_fw before 201 * re-populating them with minidump segments. 202 */ 203 rproc_coredump_cleanup(rproc); 204 205 ret = qcom_add_minidump_segments(rproc, subsystem, rproc_dumpfn_t); 206 if (ret) { 207 dev_err(&rproc->dev, "Failed with error: %d while adding minidump entries\n", ret); 208 goto clean_minidump; 209 } 210 rproc_coredump_using_sections(rproc); 211 clean_minidump: 212 qcom_minidump_cleanup(rproc); 213 } 214 EXPORT_SYMBOL_GPL(qcom_minidump); 215 216 static int glink_subdev_start(struct rproc_subdev *subdev) 217 { 218 struct qcom_rproc_glink *glink = to_glink_subdev(subdev); 219 220 glink->edge = qcom_glink_smem_register(glink->dev, glink->node); 221 222 return PTR_ERR_OR_ZERO(glink->edge); 223 } 224 225 static void glink_subdev_stop(struct rproc_subdev *subdev, bool crashed) 226 { 227 struct qcom_rproc_glink *glink = to_glink_subdev(subdev); 228 229 qcom_glink_smem_unregister(glink->edge); 230 glink->edge = NULL; 231 } 232 233 static void glink_subdev_unprepare(struct rproc_subdev *subdev) 234 { 235 struct qcom_rproc_glink *glink = to_glink_subdev(subdev); 236 237 qcom_glink_ssr_notify(glink->ssr_name); 238 } 239 240 /** 241 * qcom_add_glink_subdev() - try to add a GLINK subdevice to rproc 242 * @rproc: rproc handle to parent the subdevice 243 * @glink: reference to a GLINK subdev context 244 * @ssr_name: identifier of the associated remoteproc for ssr notifications 245 */ 246 void qcom_add_glink_subdev(struct rproc *rproc, struct qcom_rproc_glink *glink, 247 const char *ssr_name) 248 { 249 struct device *dev = &rproc->dev; 250 251 glink->node = of_get_child_by_name(dev->parent->of_node, "glink-edge"); 252 if (!glink->node) 253 return; 254 255 glink->ssr_name = kstrdup_const(ssr_name, GFP_KERNEL); 256 if (!glink->ssr_name) 257 return; 258 259 glink->dev = dev; 260 glink->subdev.start = glink_subdev_start; 261 glink->subdev.stop = glink_subdev_stop; 262 glink->subdev.unprepare = glink_subdev_unprepare; 263 264 rproc_add_subdev(rproc, &glink->subdev); 265 } 266 EXPORT_SYMBOL_GPL(qcom_add_glink_subdev); 267 268 /** 269 * qcom_remove_glink_subdev() - remove a GLINK subdevice from rproc 270 * @rproc: rproc handle 271 * @glink: reference to a GLINK subdev context 272 */ 273 void qcom_remove_glink_subdev(struct rproc *rproc, struct qcom_rproc_glink *glink) 274 { 275 if (!glink->node) 276 return; 277 278 rproc_remove_subdev(rproc, &glink->subdev); 279 kfree_const(glink->ssr_name); 280 of_node_put(glink->node); 281 } 282 EXPORT_SYMBOL_GPL(qcom_remove_glink_subdev); 283 284 /** 285 * qcom_register_dump_segments() - register segments for coredump 286 * @rproc: remoteproc handle 287 * @fw: firmware header 288 * 289 * Register all segments of the ELF in the remoteproc coredump segment list 290 * 291 * Return: 0 on success, negative errno on failure. 292 */ 293 int qcom_register_dump_segments(struct rproc *rproc, 294 const struct firmware *fw) 295 { 296 const struct elf32_phdr *phdrs; 297 const struct elf32_phdr *phdr; 298 const struct elf32_hdr *ehdr; 299 int ret; 300 int i; 301 302 ehdr = (struct elf32_hdr *)fw->data; 303 phdrs = (struct elf32_phdr *)(ehdr + 1); 304 305 for (i = 0; i < ehdr->e_phnum; i++) { 306 phdr = &phdrs[i]; 307 308 if (phdr->p_type != PT_LOAD) 309 continue; 310 311 if ((phdr->p_flags & QCOM_MDT_TYPE_MASK) == QCOM_MDT_TYPE_HASH) 312 continue; 313 314 if (!phdr->p_memsz) 315 continue; 316 317 ret = rproc_coredump_add_segment(rproc, phdr->p_paddr, 318 phdr->p_memsz); 319 if (ret) 320 return ret; 321 } 322 323 return 0; 324 } 325 EXPORT_SYMBOL_GPL(qcom_register_dump_segments); 326 327 static int smd_subdev_start(struct rproc_subdev *subdev) 328 { 329 struct qcom_rproc_subdev *smd = to_smd_subdev(subdev); 330 331 smd->edge = qcom_smd_register_edge(smd->dev, smd->node); 332 333 return PTR_ERR_OR_ZERO(smd->edge); 334 } 335 336 static void smd_subdev_stop(struct rproc_subdev *subdev, bool crashed) 337 { 338 struct qcom_rproc_subdev *smd = to_smd_subdev(subdev); 339 340 qcom_smd_unregister_edge(smd->edge); 341 smd->edge = NULL; 342 } 343 344 /** 345 * qcom_add_smd_subdev() - try to add a SMD subdevice to rproc 346 * @rproc: rproc handle to parent the subdevice 347 * @smd: reference to a Qualcomm subdev context 348 */ 349 void qcom_add_smd_subdev(struct rproc *rproc, struct qcom_rproc_subdev *smd) 350 { 351 struct device *dev = &rproc->dev; 352 353 smd->node = of_get_child_by_name(dev->parent->of_node, "smd-edge"); 354 if (!smd->node) 355 return; 356 357 smd->dev = dev; 358 smd->subdev.start = smd_subdev_start; 359 smd->subdev.stop = smd_subdev_stop; 360 361 rproc_add_subdev(rproc, &smd->subdev); 362 } 363 EXPORT_SYMBOL_GPL(qcom_add_smd_subdev); 364 365 /** 366 * qcom_remove_smd_subdev() - remove the smd subdevice from rproc 367 * @rproc: rproc handle 368 * @smd: the SMD subdevice to remove 369 */ 370 void qcom_remove_smd_subdev(struct rproc *rproc, struct qcom_rproc_subdev *smd) 371 { 372 if (!smd->node) 373 return; 374 375 rproc_remove_subdev(rproc, &smd->subdev); 376 of_node_put(smd->node); 377 } 378 EXPORT_SYMBOL_GPL(qcom_remove_smd_subdev); 379 380 static struct qcom_ssr_subsystem *qcom_ssr_get_subsys(const char *name) 381 { 382 struct qcom_ssr_subsystem *info; 383 384 mutex_lock(&qcom_ssr_subsys_lock); 385 /* Match in the global qcom_ssr_subsystem_list with name */ 386 list_for_each_entry(info, &qcom_ssr_subsystem_list, list) 387 if (!strcmp(info->name, name)) 388 goto out; 389 390 info = kzalloc_obj(*info); 391 if (!info) { 392 info = ERR_PTR(-ENOMEM); 393 goto out; 394 } 395 info->name = kstrdup_const(name, GFP_KERNEL); 396 srcu_init_notifier_head(&info->notifier_list); 397 398 /* Add to global notification list */ 399 list_add_tail(&info->list, &qcom_ssr_subsystem_list); 400 401 out: 402 mutex_unlock(&qcom_ssr_subsys_lock); 403 return info; 404 } 405 406 /** 407 * qcom_register_ssr_notifier() - register SSR notification handler 408 * @name: Subsystem's SSR name 409 * @nb: notifier_block to be invoked upon subsystem's state change 410 * 411 * This registers the @nb notifier block as part the notifier chain for a 412 * remoteproc associated with @name. The notifier block's callback 413 * will be invoked when the remote processor's SSR events occur 414 * (pre/post startup and pre/post shutdown). 415 * 416 * Return: a subsystem cookie on success, ERR_PTR on failure. 417 */ 418 void *qcom_register_ssr_notifier(const char *name, struct notifier_block *nb) 419 { 420 struct qcom_ssr_subsystem *info; 421 422 info = qcom_ssr_get_subsys(name); 423 if (IS_ERR(info)) 424 return info; 425 426 srcu_notifier_chain_register(&info->notifier_list, nb); 427 428 return &info->notifier_list; 429 } 430 EXPORT_SYMBOL_GPL(qcom_register_ssr_notifier); 431 432 /** 433 * qcom_unregister_ssr_notifier() - unregister SSR notification handler 434 * @notify: subsystem cookie returned from qcom_register_ssr_notifier 435 * @nb: notifier_block to unregister 436 * 437 * This function will unregister the notifier from the particular notifier 438 * chain. 439 * 440 * Return: 0 on success, %ENOENT otherwise. 441 */ 442 int qcom_unregister_ssr_notifier(void *notify, struct notifier_block *nb) 443 { 444 return srcu_notifier_chain_unregister(notify, nb); 445 } 446 EXPORT_SYMBOL_GPL(qcom_unregister_ssr_notifier); 447 448 static int ssr_notify_prepare(struct rproc_subdev *subdev) 449 { 450 struct qcom_rproc_ssr *ssr = to_ssr_subdev(subdev); 451 struct qcom_ssr_notify_data data = { 452 .name = ssr->info->name, 453 .crashed = false, 454 }; 455 456 srcu_notifier_call_chain(&ssr->info->notifier_list, 457 QCOM_SSR_BEFORE_POWERUP, &data); 458 return 0; 459 } 460 461 static int ssr_notify_start(struct rproc_subdev *subdev) 462 { 463 struct qcom_rproc_ssr *ssr = to_ssr_subdev(subdev); 464 struct qcom_ssr_notify_data data = { 465 .name = ssr->info->name, 466 .crashed = false, 467 }; 468 469 srcu_notifier_call_chain(&ssr->info->notifier_list, 470 QCOM_SSR_AFTER_POWERUP, &data); 471 return 0; 472 } 473 474 static void ssr_notify_stop(struct rproc_subdev *subdev, bool crashed) 475 { 476 struct qcom_rproc_ssr *ssr = to_ssr_subdev(subdev); 477 struct qcom_ssr_notify_data data = { 478 .name = ssr->info->name, 479 .crashed = crashed, 480 }; 481 482 srcu_notifier_call_chain(&ssr->info->notifier_list, 483 QCOM_SSR_BEFORE_SHUTDOWN, &data); 484 } 485 486 static void ssr_notify_unprepare(struct rproc_subdev *subdev) 487 { 488 struct qcom_rproc_ssr *ssr = to_ssr_subdev(subdev); 489 struct qcom_ssr_notify_data data = { 490 .name = ssr->info->name, 491 .crashed = false, 492 }; 493 494 srcu_notifier_call_chain(&ssr->info->notifier_list, 495 QCOM_SSR_AFTER_SHUTDOWN, &data); 496 } 497 498 /** 499 * qcom_add_ssr_subdev() - register subdevice as restart notification source 500 * @rproc: rproc handle 501 * @ssr: SSR subdevice handle 502 * @ssr_name: identifier to use for notifications originating from @rproc 503 * 504 * As the @ssr is registered with the @rproc SSR events will be sent to all 505 * registered listeners for the remoteproc when it's SSR events occur 506 * (pre/post startup and pre/post shutdown). 507 */ 508 void qcom_add_ssr_subdev(struct rproc *rproc, struct qcom_rproc_ssr *ssr, 509 const char *ssr_name) 510 { 511 struct qcom_ssr_subsystem *info; 512 513 info = qcom_ssr_get_subsys(ssr_name); 514 if (IS_ERR(info)) { 515 dev_err(&rproc->dev, "Failed to add ssr subdevice\n"); 516 return; 517 } 518 519 ssr->info = info; 520 ssr->subdev.prepare = ssr_notify_prepare; 521 ssr->subdev.start = ssr_notify_start; 522 ssr->subdev.stop = ssr_notify_stop; 523 ssr->subdev.unprepare = ssr_notify_unprepare; 524 525 rproc_add_subdev(rproc, &ssr->subdev); 526 } 527 EXPORT_SYMBOL_GPL(qcom_add_ssr_subdev); 528 529 /** 530 * qcom_remove_ssr_subdev() - remove subdevice as restart notification source 531 * @rproc: rproc handle 532 * @ssr: SSR subdevice handle 533 */ 534 void qcom_remove_ssr_subdev(struct rproc *rproc, struct qcom_rproc_ssr *ssr) 535 { 536 rproc_remove_subdev(rproc, &ssr->subdev); 537 ssr->info = NULL; 538 } 539 EXPORT_SYMBOL_GPL(qcom_remove_ssr_subdev); 540 541 static void pdm_dev_release(struct device *dev) 542 { 543 struct auxiliary_device *adev = to_auxiliary_dev(dev); 544 545 kfree(adev); 546 } 547 548 static int pdm_notify_prepare(struct rproc_subdev *subdev) 549 { 550 struct qcom_rproc_pdm *pdm = to_pdm_subdev(subdev); 551 struct auxiliary_device *adev; 552 int ret; 553 554 adev = kzalloc_obj(*adev); 555 if (!adev) 556 return -ENOMEM; 557 558 adev->dev.parent = pdm->dev; 559 adev->dev.release = pdm_dev_release; 560 adev->name = "pd-mapper"; 561 adev->id = pdm->index; 562 563 ret = auxiliary_device_init(adev); 564 if (ret) { 565 kfree(adev); 566 return ret; 567 } 568 569 ret = auxiliary_device_add(adev); 570 if (ret) { 571 auxiliary_device_uninit(adev); 572 return ret; 573 } 574 575 pdm->adev = adev; 576 577 return 0; 578 } 579 580 581 static void pdm_notify_unprepare(struct rproc_subdev *subdev) 582 { 583 struct qcom_rproc_pdm *pdm = to_pdm_subdev(subdev); 584 585 if (!pdm->adev) 586 return; 587 588 auxiliary_device_delete(pdm->adev); 589 auxiliary_device_uninit(pdm->adev); 590 pdm->adev = NULL; 591 } 592 593 /** 594 * qcom_add_pdm_subdev() - register PD Mapper subdevice 595 * @rproc: rproc handle 596 * @pdm: PDM subdevice handle 597 * 598 * Register @pdm so that Protection Device mapper service is started when the 599 * DSP is started too. 600 */ 601 void qcom_add_pdm_subdev(struct rproc *rproc, struct qcom_rproc_pdm *pdm) 602 { 603 pdm->dev = &rproc->dev; 604 pdm->index = rproc->index; 605 606 pdm->subdev.prepare = pdm_notify_prepare; 607 pdm->subdev.unprepare = pdm_notify_unprepare; 608 609 rproc_add_subdev(rproc, &pdm->subdev); 610 } 611 EXPORT_SYMBOL_GPL(qcom_add_pdm_subdev); 612 613 /** 614 * qcom_remove_pdm_subdev() - remove PD Mapper subdevice 615 * @rproc: rproc handle 616 * @pdm: PDM subdevice handle 617 * 618 * Remove the PD Mapper subdevice. 619 */ 620 void qcom_remove_pdm_subdev(struct rproc *rproc, struct qcom_rproc_pdm *pdm) 621 { 622 rproc_remove_subdev(rproc, &pdm->subdev); 623 } 624 EXPORT_SYMBOL_GPL(qcom_remove_pdm_subdev); 625 626 MODULE_DESCRIPTION("Qualcomm Remoteproc helper driver"); 627 MODULE_LICENSE("GPL v2"); 628