/* Copyright (c) 2014-2015, 2017-2020, 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 "esoc-mdm.h" /* This function can be called from atomic context. */ static int mdm9x55_toggle_soft_reset(struct mdm_ctrl *mdm, bool atomic) { int soft_reset_direction_assert = 0, soft_reset_direction_de_assert = 1; uint32_t reset_time_us = mdm->reset_time_ms * 1000; if (mdm->soft_reset_inverted) { soft_reset_direction_assert = 1; soft_reset_direction_de_assert = 0; } gpio_direction_output(MDM_GPIO(mdm, AP2MDM_SOFT_RESET), soft_reset_direction_assert); /* * Allow PS hold assert to be detected */ if (!atomic) usleep_range(reset_time_us, reset_time_us + 100000); else /* * The flow falls through this path as a part of the * panic handler, which has to executed atomically. */ mdelay(mdm->reset_time_ms); gpio_direction_output(MDM_GPIO(mdm, AP2MDM_SOFT_RESET), soft_reset_direction_de_assert); return 0; } /* This function can be called from atomic context. */ static int sdx50m_toggle_soft_reset(struct mdm_ctrl *mdm, bool atomic) { int soft_reset_direction_assert = 0, soft_reset_direction_de_assert = 1; if (mdm->soft_reset_inverted) { soft_reset_direction_assert = 1; soft_reset_direction_de_assert = 0; } esoc_mdm_log("RESET GPIO value (before doing a reset): %d\n", gpio_get_value(MDM_GPIO(mdm, AP2MDM_SOFT_RESET))); esoc_mdm_log("Setting AP2MDM_SOFT_RESET = %d\n", soft_reset_direction_assert); gpio_direction_output(MDM_GPIO(mdm, AP2MDM_SOFT_RESET), soft_reset_direction_assert); /* * Allow PS hold assert to be detected */ if (!atomic) usleep_range(120000, 180000); else /* * The flow falls through this path as a part of the * panic handler, which has to executed atomically. */ mdelay(100); esoc_mdm_log("Setting AP2MDM_SOFT_RESET = %d\n", soft_reset_direction_de_assert); gpio_direction_output(MDM_GPIO(mdm, AP2MDM_SOFT_RESET), soft_reset_direction_de_assert); return 0; } static int mdm4x_do_first_power_on(struct mdm_ctrl *mdm) { int i; int pblrdy; struct device *dev = mdm->dev; esoc_mdm_log("Powering on modem for the first time\n"); dev_dbg(dev, "Powering on modem for the first time\n"); if (mdm->esoc->auto_boot) return 0; mdm_toggle_soft_reset(mdm, false); /* Add a delay to allow PON sequence to complete*/ msleep(150); esoc_mdm_log("Setting AP2MDM_STATUS = 1\n"); gpio_direction_output(MDM_GPIO(mdm, AP2MDM_STATUS), 1); if (gpio_is_valid(MDM_GPIO(mdm, MDM2AP_PBLRDY))) { for (i = 0; i < MDM_PBLRDY_CNT; i++) { pblrdy = gpio_get_value(MDM_GPIO(mdm, MDM2AP_PBLRDY)); if (pblrdy) break; usleep_range(5000, 6000); } dev_dbg(dev, "pblrdy i:%d\n", i); msleep(200); } /* * No PBLRDY gpio associated with this modem * Send request for image. Let userspace confirm establishment of * link to external modem. */ else { esoc_mdm_log("Queueing the request: ESOC_REQ_IMG\n"); esoc_clink_queue_request(ESOC_REQ_IMG, mdm->esoc); } return 0; } static int sdxmarmot_do_first_power_on(struct mdm_ctrl *mdm) { int i; int pblrdy; struct device *dev = mdm->dev; esoc_mdm_log("Powering on modem for the first time\n"); dev_dbg(dev, "Powering on modem for the first time\n"); if (mdm->esoc->auto_boot) return 0; mdm_toggle_soft_reset(mdm, false); /* Add a delay to allow PON sequence to complete*/ msleep(325); esoc_mdm_log("Setting AP2MDM_STATUS = 1\n"); gpio_direction_output(MDM_GPIO(mdm, AP2MDM_STATUS), 1); if (gpio_is_valid(MDM_GPIO(mdm, MDM2AP_PBLRDY))) { for (i = 0; i < MDM_PBLRDY_CNT; i++) { pblrdy = gpio_get_value(MDM_GPIO(mdm, MDM2AP_PBLRDY)); if (pblrdy) break; usleep_range(5000, 6000); } dev_dbg(dev, "pblrdy i:%d\n", i); msleep(200); } /* * No PBLRDY gpio associated with this modem * Send request for image. Let userspace confirm establishment of * link to external modem. */ else { esoc_mdm_log("Queueing the request: ESOC_REQ_IMG\n"); esoc_clink_queue_request(ESOC_REQ_IMG, mdm->esoc); } return 0; } static int mdm9x55_power_down(struct mdm_ctrl *mdm) { struct device *dev = mdm->dev; int soft_reset_direction_assert = 0, soft_reset_direction_de_assert = 1; if (mdm->soft_reset_inverted) { soft_reset_direction_assert = 1; soft_reset_direction_de_assert = 0; } /* Assert the soft reset line whether mdm2ap_status went low or not */ gpio_direction_output(MDM_GPIO(mdm, AP2MDM_SOFT_RESET), soft_reset_direction_assert); dev_dbg(dev, "Doing a hard reset\n"); /* * Currently, there is a debounce timer on the charm PMIC. It is * necessary to hold the PMIC RESET low for 406ms * for the reset to fully take place. Sleep here to ensure the * reset has occurred before the function exits. */ msleep(406); gpio_direction_output(MDM_GPIO(mdm, AP2MDM_SOFT_RESET), soft_reset_direction_de_assert); return 0; } static int sdx50m_power_down(struct mdm_ctrl *mdm) { struct device *dev = mdm->dev; int soft_reset_direction = mdm->soft_reset_inverted ? 1 : 0; /* Assert the soft reset line whether mdm2ap_status went low or not */ esoc_mdm_log("Setting AP2MDM_SOFT_RESET = %d\n", soft_reset_direction); gpio_direction_output(MDM_GPIO(mdm, AP2MDM_SOFT_RESET), soft_reset_direction); dev_dbg(dev, "Doing a hard reset\n"); gpio_direction_output(MDM_GPIO(mdm, AP2MDM_SOFT_RESET), soft_reset_direction); /* * Currently, there is a debounce timer on the charm PMIC. It is * necessary to hold the PMIC RESET low for 406ms * for the reset to fully take place. Sleep here to ensure the * reset has occurred before the function exits. */ msleep(300); return 0; } static void mdm9x55_cold_reset(struct mdm_ctrl *mdm) { dev_dbg(mdm->dev, "Triggering mdm cold reset"); gpio_direction_output(MDM_GPIO(mdm, AP2MDM_SOFT_RESET), !!mdm->soft_reset_inverted); /* * The function is executed as a part of the atomic reboot handler. * Hence, go with a busy loop instead of sleep. */ mdelay(334); gpio_direction_output(MDM_GPIO(mdm, AP2MDM_SOFT_RESET), !mdm->soft_reset_inverted); } static void sdx50m_cold_reset(struct mdm_ctrl *mdm) { dev_dbg(mdm->dev, "Triggering mdm cold reset"); esoc_mdm_log("Setting AP2MDM_SOFT_RESET = %d\n", !!mdm->soft_reset_inverted); gpio_direction_output(MDM_GPIO(mdm, AP2MDM_SOFT_RESET), !!mdm->soft_reset_inverted); /* * The function is executed as a part of the atomic reboot handler. * Hence, go with a busy loop instead of sleep. */ mdelay(600); esoc_mdm_log("Setting AP2MDM_SOFT_RESET = %d\n", !!mdm->soft_reset_inverted); gpio_direction_output(MDM_GPIO(mdm, AP2MDM_SOFT_RESET), !mdm->soft_reset_inverted); } static int mdm9x55_pon_dt_init(struct mdm_ctrl *mdm) { int val; struct device_node *node = mdm->dev->of_node; enum of_gpio_flags flags = OF_GPIO_ACTIVE_LOW; val = of_property_read_u32(node, "qcom,reset-time-ms", &mdm->reset_time_ms); if (val) mdm->reset_time_ms = DEF_MDM9X55_RESET_TIME; val = of_get_named_gpio_flags(node, "qcom,ap2mdm-soft-reset-gpio", 0, &flags); if (val >= 0) { MDM_GPIO(mdm, AP2MDM_SOFT_RESET) = val; if (flags & OF_GPIO_ACTIVE_LOW) mdm->soft_reset_inverted = 1; return 0; } else return -EIO; } static int mdm4x_pon_dt_init(struct mdm_ctrl *mdm) { int val; struct device_node *node = mdm->dev->of_node; enum of_gpio_flags flags = OF_GPIO_ACTIVE_LOW; val = of_get_named_gpio_flags(node, "qcom,ap2mdm-soft-reset-gpio", 0, &flags); if (val >= 0) { MDM_GPIO(mdm, AP2MDM_SOFT_RESET) = val; if (flags & OF_GPIO_ACTIVE_LOW) mdm->soft_reset_inverted = 1; return 0; } else return -EIO; } static int mdm4x_pon_setup(struct mdm_ctrl *mdm) { struct device *dev = mdm->dev; if (gpio_is_valid(MDM_GPIO(mdm, AP2MDM_SOFT_RESET))) { if (gpio_request(MDM_GPIO(mdm, AP2MDM_SOFT_RESET), "AP2MDM_SOFT_RESET")) { dev_err(dev, "Cannot config AP2MDM_SOFT_RESET gpio\n"); return -EIO; } } return 0; } struct mdm_pon_ops mdm9x55_pon_ops = { .pon = mdm4x_do_first_power_on, .soft_reset = mdm9x55_toggle_soft_reset, .poff_force = mdm9x55_power_down, .cold_reset = mdm9x55_cold_reset, .dt_init = mdm9x55_pon_dt_init, .setup = mdm4x_pon_setup, }; struct mdm_pon_ops sdx50m_pon_ops = { .pon = mdm4x_do_first_power_on, .soft_reset = sdx50m_toggle_soft_reset, .poff_force = sdx50m_power_down, .cold_reset = sdx50m_cold_reset, .dt_init = mdm4x_pon_dt_init, .setup = mdm4x_pon_setup, }; struct mdm_pon_ops sdxmarmot_pon_ops = { .pon = sdxmarmot_do_first_power_on, .soft_reset = sdx50m_toggle_soft_reset, .poff_force = sdx50m_power_down, .cold_reset = sdx50m_cold_reset, .dt_init = mdm4x_pon_dt_init, .setup = mdm4x_pon_setup, };