| /* Copyright (c) 2017-2018, The Linux Foundation. All rights reserved. |
| * |
| * This program is free software; you can redistribute it and/or modify |
| * it under the terms of the GNU General Public License version 2 and |
| * only version 2 as published by the Free Software Foundation. |
| * |
| * This program is distributed in the hope that it will be useful, |
| * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
| * GNU General Public License for more details. |
| */ |
| |
| #include <linux/kernel.h> |
| #include <linux/err.h> |
| #include <linux/io.h> |
| #include <linux/module.h> |
| #include <linux/platform_device.h> |
| #include <linux/of.h> |
| #include <linux/interrupt.h> |
| #include <linux/of_gpio.h> |
| #include <linux/delay.h> |
| #include <linux/slab.h> |
| #include <linux/dma-mapping.h> |
| #include <linux/reboot.h> |
| #include <soc/qcom/subsystem_restart.h> |
| #include <soc/qcom/ramdump.h> |
| #include <soc/qcom/subsystem_notif.h> |
| #include <linux/highmem.h> |
| |
| #include "peripheral-loader.h" |
| #include "../../misc/qseecom_kernel.h" |
| #include "pil_bg_intf.h" |
| #include "bgcom_interface.h" |
| |
| #define INVALID_GPIO -1 |
| #define NUM_GPIOS 4 |
| #define SECURE_APP "bgapp" |
| #define desc_to_data(d) container_of(d, struct pil_bg_data, desc) |
| #define subsys_to_data(d) container_of(d, struct pil_bg_data, subsys_desc) |
| #define BG_RAMDUMP_SZ 0x00102000 |
| #define BG_CRASH_IN_TWM -2 |
| /** |
| * struct pil_bg_data |
| * @qseecom_handle: handle of TZ app |
| * @bg_queue: private queue to schedule worker threads for bottom half |
| * @restart_work: work struct for executing ssr |
| * @reboot_blk: notification block for reboot event |
| * @subsys_desc: subsystem descriptor |
| * @subsys: subsystem device pointer |
| * @gpios: array to hold all gpio handle |
| * @desc: PIL descriptor |
| * @address_fw: address where firmware binaries loaded |
| * @ramdump_dev: ramdump device pointer |
| * @size_fw: size of bg firmware binaries |
| * @errfatal_irq: irq number to indicate bg crash or shutdown |
| * @status_irq: irq to indicate bg status |
| * @app_status: status of tz app loading |
| * @is_ready: Is BG chip up |
| * @err_ready: The error ready signal |
| */ |
| |
| struct pil_bg_data { |
| struct qseecom_handle *qseecom_handle; |
| struct workqueue_struct *bg_queue; |
| struct work_struct restart_work; |
| struct notifier_block reboot_blk; |
| struct subsys_desc subsys_desc; |
| struct subsys_device *subsys; |
| unsigned int gpios[NUM_GPIOS]; |
| int errfatal_irq; |
| int status_irq; |
| struct pil_desc desc; |
| phys_addr_t address_fw; |
| void *ramdump_dev; |
| u32 cmd_status; |
| size_t size_fw; |
| int app_status; |
| bool is_ready; |
| struct completion err_ready; |
| }; |
| |
| static irqreturn_t bg_status_change(int irq, void *dev_id); |
| |
| /** |
| * bg_app_shutdown_notify() - Toggle AP2BG err fatal gpio when |
| * called by SSR framework. |
| * @subsys: struct containing private BG data. |
| * |
| * Return: none. |
| */ |
| static void bg_app_shutdown_notify(const struct subsys_desc *subsys) |
| { |
| struct pil_bg_data *bg_data = subsys_to_data(subsys); |
| /* Toggle AP2BG err fatal gpio here to inform apps err fatal event */ |
| if (gpio_is_valid(bg_data->gpios[2])) |
| gpio_set_value(bg_data->gpios[2], 1); |
| } |
| |
| /** |
| * bg_app_reboot_notify() - Toggle AP2BG err fatal gpio. |
| * @nb: struct containing private BG data. |
| * |
| * Return: NOTIFY_DONE indicating success. |
| */ |
| static int bg_app_reboot_notify(struct notifier_block *nb, |
| unsigned long code, void *unused) |
| { |
| struct pil_bg_data *bg_data = container_of(nb, |
| struct pil_bg_data, reboot_blk); |
| /* Toggle AP2BG err fatal gpio here to inform apps err fatal event */ |
| if (gpio_is_valid(bg_data->gpios[2])) |
| gpio_set_value(bg_data->gpios[2], 1); |
| return NOTIFY_DONE; |
| } |
| |
| /** |
| * get_cmd_rsp_buffers() - Function sets cmd & rsp buffer pointers and |
| * aligns buffer lengths |
| * @hdl: index of qseecom_handle |
| * @cmd: req buffer - set to qseecom_handle.sbuf |
| * @cmd_len: ptr to req buffer len |
| * @rsp: rsp buffer - set to qseecom_handle.sbuf + offset |
| * @rsp_len: ptr to rsp buffer len |
| * |
| * Return: Success always . |
| */ |
| static int get_cmd_rsp_buffers(struct qseecom_handle *handle, void **cmd, |
| uint32_t *cmd_len, void **rsp, uint32_t *rsp_len) |
| { |
| *cmd = handle->sbuf; |
| if (*cmd_len & QSEECOM_ALIGN_MASK) |
| *cmd_len = QSEECOM_ALIGN(*cmd_len); |
| |
| *rsp = handle->sbuf + *cmd_len; |
| if (*rsp_len & QSEECOM_ALIGN_MASK) |
| *rsp_len = QSEECOM_ALIGN(*rsp_len); |
| |
| return 0; |
| } |
| |
| /** |
| * pil_load_bg_tzapp() - Called to load TZ app. |
| * @pbd: struct containing private BG data. |
| * |
| * Return: 0 on success. Error code on failure. |
| */ |
| static int pil_load_bg_tzapp(struct pil_bg_data *pbd) |
| { |
| int rc; |
| |
| /* return success if already loaded */ |
| if (pbd->qseecom_handle && !pbd->app_status) |
| return 0; |
| /* Load the APP */ |
| rc = qseecom_start_app(&pbd->qseecom_handle, SECURE_APP, SZ_4K); |
| if (rc < 0) { |
| dev_err(pbd->desc.dev, "BG TZ app load failure\n"); |
| pbd->app_status = RESULT_FAILURE; |
| return -EIO; |
| } |
| pbd->app_status = RESULT_SUCCESS; |
| return 0; |
| } |
| |
| /** |
| * bgpil_tzapp_comm() - Function called to communicate with TZ APP. |
| * @req: struct containing command and parameters. |
| * |
| * Return: 0 on success. Error code on failure. |
| */ |
| static long bgpil_tzapp_comm(struct pil_bg_data *pbd, |
| struct tzapp_bg_req *req) |
| { |
| struct tzapp_bg_req *bg_tz_req; |
| struct tzapp_bg_rsp *bg_tz_rsp; |
| int rc, req_len, rsp_len; |
| |
| /* Fill command structure */ |
| req_len = sizeof(struct tzapp_bg_req); |
| rsp_len = sizeof(struct tzapp_bg_rsp); |
| rc = get_cmd_rsp_buffers(pbd->qseecom_handle, |
| (void **)&bg_tz_req, &req_len, |
| (void **)&bg_tz_rsp, &rsp_len); |
| if (rc) |
| goto end; |
| |
| bg_tz_req->tzapp_bg_cmd = req->tzapp_bg_cmd; |
| bg_tz_req->address_fw = req->address_fw; |
| bg_tz_req->size_fw = req->size_fw; |
| rc = qseecom_send_command(pbd->qseecom_handle, |
| (void *)bg_tz_req, req_len, (void *)bg_tz_rsp, rsp_len); |
| pr_debug("BG PIL qseecom returned with value 0x%x and status 0x%x\n", |
| rc, bg_tz_rsp->status); |
| if (rc || bg_tz_rsp->status) |
| pbd->cmd_status = bg_tz_rsp->status; |
| else |
| pbd->cmd_status = 0; |
| end: |
| return rc; |
| } |
| |
| /** |
| * wait_for_err_ready() - Called in power_up to wait for error ready. |
| * Signal waiting function. |
| * @bg_data: BG PIL private structure. |
| * |
| * Return: 0 on success. Error code on failure. |
| */ |
| static int wait_for_err_ready(struct pil_bg_data *bg_data) |
| { |
| int ret; |
| |
| if ((!bg_data->status_irq)) |
| return 0; |
| |
| ret = wait_for_completion_timeout(&bg_data->err_ready, |
| msecs_to_jiffies(10000)); |
| if (!ret) { |
| pr_err("[%s]: Error ready timed out\n", bg_data->desc.name); |
| return -ETIMEDOUT; |
| } |
| return 0; |
| } |
| |
| /** |
| * bg_powerup() - Called by SSR framework on userspace invocation. |
| * does load tz app and call peripheral loader. |
| * @subsys: struct containing private BG data. |
| * |
| * Return: 0 on success. Error code on failure. |
| */ |
| static int bg_powerup(const struct subsys_desc *subsys) |
| { |
| struct pil_bg_data *bg_data = subsys_to_data(subsys); |
| int ret; |
| |
| init_completion(&bg_data->err_ready); |
| if (!bg_data->qseecom_handle) { |
| ret = pil_load_bg_tzapp(bg_data); |
| if (ret) { |
| dev_err(bg_data->desc.dev, |
| "%s: BG TZ app load failure\n", |
| __func__); |
| return ret; |
| } |
| } |
| pr_debug("bgapp loaded\n"); |
| bg_data->desc.fw_name = subsys->fw_name; |
| |
| ret = devm_request_irq(bg_data->desc.dev, bg_data->status_irq, |
| bg_status_change, |
| IRQF_TRIGGER_RISING | IRQF_TRIGGER_FALLING | IRQF_ONESHOT, |
| "bg2ap_status", bg_data); |
| if (ret < 0) { |
| dev_err(bg_data->desc.dev, |
| "%s: BG2AP_STATUS IRQ#%d re registration failed, err=%d", |
| __func__, bg_data->status_irq, ret); |
| return ret; |
| } |
| disable_irq(bg_data->status_irq); |
| |
| /* Enable status and err fatal irqs */ |
| ret = pil_boot(&bg_data->desc); |
| if (ret) { |
| dev_err(bg_data->desc.dev, |
| "%s: BG PIL Boot failed\n", __func__); |
| return ret; |
| } |
| enable_irq(bg_data->status_irq); |
| enable_irq(bg_data->errfatal_irq); |
| ret = wait_for_err_ready(bg_data); |
| if (ret) { |
| dev_err(bg_data->desc.dev, |
| "[%s:%d]: Timed out waiting for error ready: %s!\n", |
| current->comm, current->pid, bg_data->desc.name); |
| return ret; |
| } |
| return ret; |
| } |
| |
| /** |
| * bg_shutdown() - Called by SSR framework on userspace invocation. |
| * disable status interrupt to avoid spurious signal during PRM exit. |
| * @subsys: struct containing private BG data. |
| * @force_stop: unused |
| * |
| * Return: always success |
| */ |
| static int bg_shutdown(const struct subsys_desc *subsys, bool force_stop) |
| { |
| struct pil_bg_data *bg_data = subsys_to_data(subsys); |
| |
| disable_irq(bg_data->status_irq); |
| devm_free_irq(bg_data->desc.dev, bg_data->status_irq, bg_data); |
| disable_irq(bg_data->errfatal_irq); |
| bg_data->is_ready = false; |
| return 0; |
| } |
| |
| /** |
| * bg_auth_metadata() - Called by Peripheral loader framework |
| * send command to tz app for authentication of metadata. |
| * @pil: pil descriptor. |
| * @metadata: metadata load address |
| * @size: size of metadata |
| * |
| * Return: 0 on success. Error code on failure. |
| */ |
| static int bg_auth_metadata(struct pil_desc *pil, |
| const u8 *metadata, size_t size, |
| phys_addr_t addr, void *sz) |
| { |
| struct pil_bg_data *bg_data = desc_to_data(pil); |
| struct tzapp_bg_req bg_tz_req; |
| void *mdata_buf; |
| dma_addr_t mdata_phys; |
| unsigned long attrs = 0; |
| struct device dev = {0}; |
| int ret; |
| |
| arch_setup_dma_ops(&dev, 0, 0, NULL, 0); |
| |
| dev.coherent_dma_mask = DMA_BIT_MASK(sizeof(dma_addr_t) * 8); |
| attrs |= DMA_ATTR_STRONGLY_ORDERED; |
| mdata_buf = dma_alloc_attrs(&dev, size, |
| &mdata_phys, GFP_KERNEL, attrs); |
| |
| if (!mdata_buf) { |
| pr_err("BG_PIL: Allocation for metadata failed.\n"); |
| return -ENOMEM; |
| } |
| |
| /* Make sure there are no mappings in PKMAP and fixmap */ |
| kmap_flush_unused(); |
| kmap_atomic_flush_unused(); |
| |
| memcpy(mdata_buf, metadata, size); |
| |
| bg_tz_req.tzapp_bg_cmd = BGPIL_AUTH_MDT; |
| bg_tz_req.address_fw = (phys_addr_t)mdata_phys; |
| bg_tz_req.size_fw = size; |
| |
| ret = bgpil_tzapp_comm(bg_data, &bg_tz_req); |
| if (ret || bg_data->cmd_status) { |
| dev_err(pil->dev, |
| "%s: BGPIL_AUTH_MDT qseecom call failed\n", |
| __func__); |
| return bg_data->cmd_status; |
| } |
| dma_free_attrs(&dev, size, mdata_buf, mdata_phys, attrs); |
| pr_debug("BG MDT Authenticated\n"); |
| return 0; |
| } |
| |
| /** |
| * bg_get_firmware_addr() - Called by Peripheral loader framework |
| * to get address and size of bg firmware binaries. |
| * @pil: pil descriptor. |
| * @addr: fw load address |
| * @size: size of fw |
| * |
| * Return: 0 on success. |
| */ |
| static int bg_get_firmware_addr(struct pil_desc *pil, |
| phys_addr_t addr, size_t size) |
| { |
| struct pil_bg_data *bg_data = desc_to_data(pil); |
| |
| bg_data->address_fw = addr; |
| bg_data->size_fw = size; |
| pr_debug("BG PIL loads firmware blobs at 0x%x with size 0x%x\n", |
| addr, size); |
| return 0; |
| } |
| |
| |
| /** |
| * bg_auth_and_xfer() - Called by Peripheral loader framework |
| * to signal tz app to authenticate and boot bg chip. |
| * @pil: pil descriptor. |
| * |
| * Return: 0 on success. Error code on failure. |
| */ |
| static int bg_auth_and_xfer(struct pil_desc *pil) |
| { |
| struct pil_bg_data *bg_data = desc_to_data(pil); |
| struct tzapp_bg_req bg_tz_req; |
| int ret; |
| |
| bg_tz_req.tzapp_bg_cmd = BGPIL_IMAGE_LOAD; |
| bg_tz_req.address_fw = bg_data->address_fw; |
| bg_tz_req.size_fw = bg_data->size_fw; |
| |
| ret = bgpil_tzapp_comm(bg_data, &bg_tz_req); |
| if (bg_data->cmd_status == BG_CRASH_IN_TWM) { |
| /* Do ramdump and resend boot cmd */ |
| if (is_twm_exit()) |
| bg_data->subsys_desc.ramdump(true, |
| &bg_data->subsys_desc); |
| bg_tz_req.tzapp_bg_cmd = BGPIL_DLOAD_CONT; |
| ret = bgpil_tzapp_comm(bg_data, &bg_tz_req); |
| } |
| if (ret || bg_data->cmd_status) { |
| dev_err(pil->dev, |
| "%s: BGPIL_IMAGE_LOAD qseecom call failed\n", |
| __func__); |
| pil_free_memory(&bg_data->desc); |
| return bg_data->cmd_status; |
| } |
| /* BG Transfer of image is complete, free up the memory */ |
| pr_debug("BG Firmware authentication and transfer done\n"); |
| pil_free_memory(&bg_data->desc); |
| return 0; |
| } |
| |
| /** |
| * bg_ramdump() - Called by SSR framework to save dump of BG internal |
| * memory, BG PIL does allocate region from dynamic memory and pass this |
| * region to tz to dump memory content of BG. |
| * @subsys: subsystem descriptor. |
| * |
| * Return: 0 on success. Error code on failure. |
| */ |
| static int bg_ramdump(int enable, const struct subsys_desc *subsys) |
| { |
| struct pil_bg_data *bg_data = subsys_to_data(subsys); |
| struct pil_desc desc = bg_data->desc; |
| struct ramdump_segment *ramdump_segments; |
| struct tzapp_bg_req bg_tz_req; |
| phys_addr_t start_addr; |
| void *region; |
| int ret; |
| struct device dev = {0}; |
| |
| arch_setup_dma_ops(&dev, 0, 0, NULL, 0); |
| |
| desc.attrs = 0; |
| desc.attrs |= DMA_ATTR_SKIP_ZEROING; |
| desc.attrs |= DMA_ATTR_STRONGLY_ORDERED; |
| |
| region = dma_alloc_attrs(desc.dev, BG_RAMDUMP_SZ, |
| &start_addr, GFP_KERNEL, desc.attrs); |
| |
| if (region == NULL) { |
| dev_dbg(desc.dev, |
| "BG PIL failure to allocate ramdump region of size %zx\n", |
| BG_RAMDUMP_SZ); |
| return -ENOMEM; |
| } |
| |
| ramdump_segments = kcalloc(1, sizeof(*ramdump_segments), GFP_KERNEL); |
| if (!ramdump_segments) |
| return -ENOMEM; |
| |
| bg_tz_req.tzapp_bg_cmd = BGPIL_RAMDUMP; |
| bg_tz_req.address_fw = start_addr; |
| bg_tz_req.size_fw = BG_RAMDUMP_SZ; |
| |
| ret = bgpil_tzapp_comm(bg_data, &bg_tz_req); |
| if (ret || bg_data->cmd_status) { |
| dev_dbg(desc.dev, "%s: BG PIL ramdump collection failed\n", |
| __func__); |
| return bg_data->cmd_status; |
| } |
| |
| ramdump_segments->address = start_addr; |
| ramdump_segments->size = BG_RAMDUMP_SZ; |
| |
| do_ramdump(bg_data->ramdump_dev, ramdump_segments, 1); |
| kfree(ramdump_segments); |
| dma_free_attrs(desc.dev, BG_RAMDUMP_SZ, region, |
| start_addr, desc.attrs); |
| return 0; |
| } |
| |
| static struct pil_reset_ops pil_ops_trusted = { |
| .init_image = bg_auth_metadata, |
| .mem_setup = bg_get_firmware_addr, |
| .auth_and_reset = bg_auth_and_xfer, |
| .shutdown = NULL, |
| .proxy_vote = NULL, |
| .proxy_unvote = NULL, |
| }; |
| |
| /** |
| * bg_restart_work() - scheduled by interrupt handler to carry |
| * out ssr sequence |
| * @work: work struct. |
| * |
| * Return: none. |
| */ |
| static void bg_restart_work(struct work_struct *work) |
| { |
| struct pil_bg_data *drvdata = |
| container_of(work, struct pil_bg_data, restart_work); |
| subsystem_restart_dev(drvdata->subsys); |
| } |
| |
| static irqreturn_t bg_errfatal(int irq, void *dev_id) |
| { |
| struct pil_bg_data *drvdata = (struct pil_bg_data *)dev_id; |
| |
| if (!drvdata) |
| return IRQ_HANDLED; |
| |
| dev_dbg(drvdata->desc.dev, "BG s/w err fatal\n"); |
| |
| queue_work(drvdata->bg_queue, &drvdata->restart_work); |
| |
| return IRQ_HANDLED; |
| } |
| |
| static irqreturn_t bg_status_change(int irq, void *dev_id) |
| { |
| bool value; |
| struct pil_bg_data *drvdata = (struct pil_bg_data *)dev_id; |
| |
| if (!drvdata) |
| return IRQ_HANDLED; |
| |
| value = gpio_get_value(drvdata->gpios[0]); |
| if (value == true && !drvdata->is_ready) { |
| dev_info(drvdata->desc.dev, |
| "BG services are up and running: irq state changed 0->1\n"); |
| drvdata->is_ready = true; |
| complete(&drvdata->err_ready); |
| } else if (value == false && drvdata->is_ready) { |
| dev_err(drvdata->desc.dev, |
| "BG got unexpected reset: irq state changed 1->0\n"); |
| queue_work(drvdata->bg_queue, &drvdata->restart_work); |
| } else { |
| dev_err(drvdata->desc.dev, |
| "BG status irq: unknown status\n"); |
| } |
| |
| return IRQ_HANDLED; |
| } |
| |
| /** |
| * setup_bg_gpio_irq() - called in probe to configure input/ |
| * output gpio. |
| * @drvdata: private data struct for BG. |
| * |
| * Return: 0 on success. Error code on failure. |
| */ |
| static int setup_bg_gpio_irq(struct platform_device *pdev, |
| struct pil_bg_data *drvdata) |
| { |
| int ret = -1; |
| int irq, i; |
| |
| if (gpio_request(drvdata->gpios[0], "BG2AP_STATUS")) { |
| dev_err(&pdev->dev, |
| "%s Failed to configure BG2AP_STATUS gpio\n", |
| __func__); |
| goto err; |
| } |
| if (gpio_request(drvdata->gpios[1], "BG2AP_ERRFATAL")) { |
| dev_err(&pdev->dev, |
| "%s Failed to configure BG2AP_ERRFATAL gpio\n", |
| __func__); |
| goto err; |
| } |
| gpio_direction_input(drvdata->gpios[0]); |
| gpio_direction_input(drvdata->gpios[1]); |
| /* BG2AP STATUS IRQ */ |
| irq = gpio_to_irq(drvdata->gpios[0]); |
| if (irq < 0) { |
| dev_err(&pdev->dev, |
| "%s: bad BG2AP_STATUS IRQ resource, err = %d\n", |
| __func__, irq); |
| goto err; |
| } |
| |
| drvdata->status_irq = irq; |
| /* BG2AP ERR_FATAL irq. */ |
| irq = gpio_to_irq(drvdata->gpios[1]); |
| if (irq < 0) { |
| dev_err(&pdev->dev, "bad BG2AP_ERRFATAL IRQ resource\n"); |
| goto err; |
| } |
| ret = request_irq(irq, bg_errfatal, |
| IRQF_TRIGGER_RISING | IRQF_ONESHOT, "bg2ap_errfatal", drvdata); |
| if (ret < 0) { |
| dev_err(&pdev->dev, |
| "%s: BG2AP_ERRFATAL IRQ#%d request failed,\n", |
| __func__, irq); |
| goto err; |
| } |
| drvdata->errfatal_irq = irq; |
| enable_irq(drvdata->errfatal_irq); |
| /* Configure outgoing GPIO's */ |
| if (gpio_request(drvdata->gpios[2], "AP2BG_ERRFATAL")) { |
| dev_err(&pdev->dev, |
| "%s Failed to configure AP2BG_ERRFATAL gpio\n", |
| __func__); |
| goto err; |
| } |
| if (gpio_request(drvdata->gpios[3], "AP2BG_STATUS")) { |
| dev_err(&pdev->dev, |
| "%s Failed to configure AP2BG_STATUS gpio\n", |
| __func__); |
| goto err; |
| } |
| /* |
| * Put status gpio in default high state which will |
| * make transition to low on any sudden reset case of msm |
| */ |
| gpio_direction_output(drvdata->gpios[2], 0); |
| gpio_direction_output(drvdata->gpios[3], 1); |
| /* Inform BG that AP is up */ |
| gpio_set_value(drvdata->gpios[3], 1); |
| return 0; |
| err: |
| for (i = 0; i < NUM_GPIOS; ++i) { |
| if (gpio_is_valid(drvdata->gpios[i])) |
| gpio_free(drvdata->gpios[i]); |
| } |
| return ret; |
| } |
| |
| /** |
| * bg_dt_parse_gpio() - called in probe to parse gpio's |
| * @drvdata: private data struct for BG. |
| * |
| * Return: 0 on success. Error code on failure. |
| */ |
| static int bg_dt_parse_gpio(struct platform_device *pdev, |
| struct pil_bg_data *drvdata) |
| { |
| int i, val; |
| |
| for (i = 0; i < NUM_GPIOS; i++) |
| drvdata->gpios[i] = INVALID_GPIO; |
| val = of_get_named_gpio(pdev->dev.of_node, |
| "qcom,bg2ap-status-gpio", 0); |
| if (val >= 0) |
| drvdata->gpios[0] = val; |
| else { |
| pr_err("BG status gpio not found, error=%d\n", val); |
| return -EINVAL; |
| } |
| val = of_get_named_gpio(pdev->dev.of_node, |
| "qcom,bg2ap-errfatal-gpio", 0); |
| if (val >= 0) |
| drvdata->gpios[1] = val; |
| else { |
| pr_err("BG err-fatal gpio not found, error=%d\n", val); |
| return -EINVAL; |
| } |
| val = of_get_named_gpio(pdev->dev.of_node, |
| "qcom,ap2bg-errfatal-gpio", 0); |
| if (val >= 0) |
| drvdata->gpios[2] = val; |
| else { |
| pr_err("ap2bg err-fatal gpio not found, error=%d\n", val); |
| return -EINVAL; |
| } |
| val = of_get_named_gpio(pdev->dev.of_node, |
| "qcom,ap2bg-status-gpio", 0); |
| if (val >= 0) |
| drvdata->gpios[3] = val; |
| else { |
| pr_err("ap2bg status gpio not found, error=%d\n", val); |
| return -EINVAL; |
| } |
| return 0; |
| } |
| |
| static int pil_bg_driver_probe(struct platform_device *pdev) |
| { |
| struct pil_bg_data *bg_data; |
| int rc; |
| |
| bg_data = devm_kzalloc(&pdev->dev, sizeof(*bg_data), GFP_KERNEL); |
| if (!bg_data) |
| return -ENOMEM; |
| platform_set_drvdata(pdev, bg_data); |
| rc = of_property_read_string(pdev->dev.of_node, |
| "qcom,firmware-name", &bg_data->desc.name); |
| if (rc) |
| return rc; |
| bg_data->desc.dev = &pdev->dev; |
| bg_data->desc.owner = THIS_MODULE; |
| bg_data->desc.ops = &pil_ops_trusted; |
| rc = pil_desc_init(&bg_data->desc); |
| if (rc) |
| return rc; |
| /* Read gpio configuration */ |
| rc = bg_dt_parse_gpio(pdev, bg_data); |
| if (rc) |
| return rc; |
| rc = setup_bg_gpio_irq(pdev, bg_data); |
| if (rc < 0) |
| return rc; |
| bg_data->subsys_desc.name = bg_data->desc.name; |
| bg_data->subsys_desc.owner = THIS_MODULE; |
| bg_data->subsys_desc.dev = &pdev->dev; |
| bg_data->subsys_desc.shutdown = bg_shutdown; |
| bg_data->subsys_desc.powerup = bg_powerup; |
| bg_data->subsys_desc.ramdump = bg_ramdump; |
| bg_data->subsys_desc.free_memory = NULL; |
| bg_data->subsys_desc.crash_shutdown = bg_app_shutdown_notify; |
| bg_data->ramdump_dev = |
| create_ramdump_device(bg_data->subsys_desc.name, &pdev->dev); |
| if (!bg_data->ramdump_dev) { |
| rc = -ENOMEM; |
| goto err_ramdump; |
| } |
| bg_data->subsys = subsys_register(&bg_data->subsys_desc); |
| if (IS_ERR(bg_data->subsys)) { |
| rc = PTR_ERR(bg_data->subsys); |
| goto err_subsys; |
| } |
| |
| bg_data->reboot_blk.notifier_call = bg_app_reboot_notify; |
| register_reboot_notifier(&bg_data->reboot_blk); |
| |
| bg_data->bg_queue = alloc_workqueue("bg_queue", 0, 0); |
| if (!bg_data->bg_queue) { |
| dev_err(&pdev->dev, "could not create bg_queue\n"); |
| subsys_unregister(bg_data->subsys); |
| goto err_subsys; |
| } |
| INIT_WORK(&bg_data->restart_work, bg_restart_work); |
| return 0; |
| err_subsys: |
| destroy_ramdump_device(bg_data->ramdump_dev); |
| err_ramdump: |
| pil_desc_release(&bg_data->desc); |
| return rc; |
| } |
| |
| static int pil_bg_driver_exit(struct platform_device *pdev) |
| { |
| struct pil_bg_data *bg_data = platform_get_drvdata(pdev); |
| |
| subsys_unregister(bg_data->subsys); |
| destroy_ramdump_device(bg_data->ramdump_dev); |
| pil_desc_release(&bg_data->desc); |
| |
| return 0; |
| } |
| |
| const struct of_device_id pil_bg_match_table[] = { |
| {.compatible = "qcom,pil-blackghost"}, |
| {} |
| }; |
| |
| static struct platform_driver pil_bg_driver = { |
| .probe = pil_bg_driver_probe, |
| .remove = pil_bg_driver_exit, |
| .driver = { |
| .name = "subsys-pil-bg", |
| .of_match_table = pil_bg_match_table, |
| .owner = THIS_MODULE, |
| }, |
| }; |
| |
| static int __init pil_bg_init(void) |
| { |
| return platform_driver_register(&pil_bg_driver); |
| } |
| module_init(pil_bg_init); |
| |
| static void __exit pil_bg_exit(void) |
| { |
| platform_driver_unregister(&pil_bg_driver); |
| } |
| module_exit(pil_bg_exit); |
| |
| MODULE_DESCRIPTION("Support for booting QTI Blackghost SoC"); |
| MODULE_LICENSE("GPL v2"); |