/* Copyright (c) 2011-2012, Code Aurora Forum. 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 #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #ifndef CONFIG_ARCH_EXYNOS #include #endif #include #include #include #include #include #include #ifndef CONFIG_ARCH_EXYNOS #include "msm_watchdog.h" #include "devices.h" #include "clock.h" #endif #include "mdm_private.h" #include #define MDM_PBLRDY_CNT 20 static int mdm_debug_on; static int power_on_count; /* ehci driver already loaded by kernel init */ static int hsic_peripheral_status = 1; static DEFINE_MUTEX(hsic_status_lock); static void mdm_peripheral_connect(struct mdm_modem_drv *mdm_drv) { struct mdm_platform_data *pdata; pr_info("%s\n", __func__); if (!mdm_drv || !mdm_drv->pdata) return; pdata = mdm_drv->pdata; mutex_lock(&hsic_status_lock); if (hsic_peripheral_status) goto out; /* exynos usb phy contgrol seq * ehci on -> ohci on -> ohci off -> ehci on */ if (pdata->peripheral_platform_device_ehci) platform_device_add(pdata->peripheral_platform_device_ehci); /* timing problem occurs when it is controlled after ohci on * control ap2mdm_wake signal right after ehci host power on */ gpio_direction_output(mdm_drv->ap2mdm_wakeup_gpio, 1); if (pdata->peripheral_platform_device_ohci) platform_device_add(pdata->peripheral_platform_device_ohci); hsic_peripheral_status = 1; out: mutex_unlock(&hsic_status_lock); } void mdm_peripheral_disconnect(struct mdm_modem_drv *mdm_drv) { struct mdm_platform_data *pdata; pr_info("%s\n", __func__); if (!mdm_drv || !mdm_drv->pdata) return; pdata = mdm_drv->pdata; mutex_lock(&hsic_status_lock); if (!hsic_peripheral_status) goto out; /* exynos usb phy contgrol seq * ehci on -> ohci on -> ohci off -> ehci on */ if (pdata->peripheral_platform_device_ohci) platform_device_del(pdata->peripheral_platform_device_ohci); if (pdata->peripheral_platform_device_ehci) platform_device_del(pdata->peripheral_platform_device_ehci); hsic_peripheral_status = 0; out: mutex_unlock(&hsic_status_lock); } static void mdm_toggle_soft_reset(struct mdm_modem_drv *mdm_drv) { int soft_reset_direction_assert = 0, soft_reset_direction_de_assert = 1; if (mdm_drv->pdata->soft_reset_inverted) { soft_reset_direction_assert = 1; soft_reset_direction_de_assert = 0; } gpio_direction_output(mdm_drv->ap2mdm_soft_reset_gpio, soft_reset_direction_assert); usleep_range(5000, 10000); gpio_direction_output(mdm_drv->ap2mdm_soft_reset_gpio, soft_reset_direction_de_assert); msleep(20); } static void mdm_power_down_common(struct mdm_modem_drv *mdm_drv) { int i; int soft_reset_direction = mdm_drv->pdata->soft_reset_inverted ? 1 : 0; pr_info("%s\n", __func__); /* Wait for the modem to complete its power down actions. */ for (i = 20; i > 0; i--) { if (gpio_get_value(mdm_drv->mdm2ap_status_gpio) == 0) break; msleep(100); } if (i == 0) { pr_err("%s: MDM2AP_STATUS never went low. Doing a hard reset\n", __func__); gpio_direction_output(mdm_drv->ap2mdm_soft_reset_gpio, soft_reset_direction); /* * Currently, there is a debounce timer on the charm PMIC. It is * necessary to hold the PMIC RESET low for ~3.5 seconds * for the reset to fully take place. Sleep here to ensure the * reset has occured before the function exits. */ msleep(4000); } mdm_peripheral_disconnect(mdm_drv); } static void mdm_do_first_power_on(struct mdm_modem_drv *mdm_drv) { int i; int pblrdy; if (power_on_count != 1) { pr_err("%s: Calling fn when power_on_count != 1\n", __func__); return; } pr_err("%s: Powering on modem for the first time\n", __func__); gpio_direction_output(mdm_drv->ap2mdm_wakeup_gpio, 0); mdm_peripheral_disconnect(mdm_drv); /* If this is the first power-up after a panic, the modem may still * be in a power-on state, in which case we need to toggle the gpio * instead of just de-asserting it. No harm done if the modem was * powered down. */ mdm_toggle_soft_reset(mdm_drv); /* If the device has a kpd pwr gpio then toggle it. */ if (mdm_drv->ap2mdm_kpdpwr_n_gpio > 0) { /* Pull AP2MDM_KPDPWR gpio high and wait for PS_HOLD to settle, * then pull it back low. */ pr_debug("%s: Pulling AP2MDM_KPDPWR gpio high\n", __func__); gpio_direction_output(mdm_drv->ap2mdm_kpdpwr_n_gpio, 1); msleep(1000); gpio_direction_output(mdm_drv->ap2mdm_kpdpwr_n_gpio, 0); } /* first power charged after 10ms */ usleep_range(10000, 15000); gpio_direction_output(mdm_drv->ap2mdm_status_gpio, 1); if (!mdm_drv->mdm2ap_pblrdy) goto start_mdm_peripheral; for (i = 0; i < MDM_PBLRDY_CNT; i++) { pblrdy = gpio_get_value(mdm_drv->mdm2ap_pblrdy); if (pblrdy) break; usleep_range(5000, 5000); } pr_debug("%s: i:%d\n", __func__, i); start_mdm_peripheral: mdm_peripheral_connect(mdm_drv); msleep(200); } static void mdm_do_soft_power_on(struct mdm_modem_drv *mdm_drv) { int i; int pblrdy; pr_err("%s: soft resetting mdm modem\n", __func__); mdm_peripheral_disconnect(mdm_drv); mdm_toggle_soft_reset(mdm_drv); if (!mdm_drv->mdm2ap_pblrdy) goto start_mdm_peripheral; for (i = 0; i < MDM_PBLRDY_CNT; i++) { pblrdy = gpio_get_value(mdm_drv->mdm2ap_pblrdy); if (pblrdy) break; usleep_range(5000, 5000); } pr_debug("%s: i:%d\n", __func__, i); start_mdm_peripheral: mdm_peripheral_connect(mdm_drv); msleep(200); } static void mdm_power_on_common(struct mdm_modem_drv *mdm_drv) { power_on_count++; /* this gpio will be used to indicate apq readiness, * de-assert it now so that it can be asserted later. * May not be used. */ if (mdm_drv->ap2mdm_wakeup_gpio > 0) gpio_direction_output(mdm_drv->ap2mdm_wakeup_gpio, 0); /* * If we did an "early power on" then ignore the very next * power-on request because it would the be first request from * user space but we're already powered on. Ignore it. */ if (mdm_drv->pdata->early_power_on && (power_on_count == 2)) return; if (power_on_count == 1) mdm_do_first_power_on(mdm_drv); else mdm_do_soft_power_on(mdm_drv); } #ifdef CONFIG_ARCH_EXYNOS static void normal_boot_done(struct mdm_modem_drv *mdm_drv) { pr_err("%s\n", __func__); mdm_peripheral_disconnect(mdm_drv); } #endif static void debug_state_changed(int value) { mdm_debug_on = value; } static void mdm_status_changed(struct mdm_modem_drv *mdm_drv, int value) { pr_debug("%s: value:%d\n", __func__, value); if (value) { mdm_peripheral_disconnect(mdm_drv); mdm_peripheral_connect(mdm_drv); if (mdm_drv->ap2mdm_wakeup_gpio > 0) { if (gpio_get_value(mdm_drv->ap2mdm_wakeup_gpio)) { gpio_set_value(mdm_drv->ap2mdm_wakeup_gpio, 0); mdelay(5); } gpio_direction_output(mdm_drv->ap2mdm_wakeup_gpio, 1); } } } static struct mdm_ops mdm_cb = { .power_on_mdm_cb = mdm_power_on_common, .reset_mdm_cb = mdm_power_on_common, .power_down_mdm_cb = mdm_power_down_common, .debug_state_changed_cb = debug_state_changed, .status_cb = mdm_status_changed, #ifdef CONFIG_ARCH_EXYNOS .normal_boot_done_cb = normal_boot_done, #endif }; static int __init mdm_modem_probe(struct platform_device *pdev) { pr_info("%s\n", __func__); return mdm_common_create(pdev, &mdm_cb); } static int __devexit mdm_modem_remove(struct platform_device *pdev) { return mdm_common_modem_remove(pdev); } static void mdm_modem_shutdown(struct platform_device *pdev) { mdm_common_modem_shutdown(pdev); } #ifdef CONFIG_FAST_BOOT static void modem_complete(struct device *pdev) { struct mdm_platform_data *pdata; if (!pdev) { pr_err("pdev is null!!\n"); return; } pdata = pdev->platform_data; if (!pdata) { pr_err("pdata is null!!\n"); return; } if (pdata->modem_complete) pdata->modem_complete(pdev); } static const struct dev_pm_ops mdm2_pm_ops = { .complete = modem_complete, }; #endif static struct platform_driver mdm_modem_driver = { .remove = mdm_modem_remove, /** * shutdown has done at reboot notifier *.shutdown = mdm_modem_shutdown, */ .driver = { .name = "mdm2_modem", #ifdef CONFIG_FAST_BOOT .pm = &mdm2_pm_ops, #endif .owner = THIS_MODULE }, }; static int __init mdm_modem_init(void) { /* in lpm mode, do not load modem driver */ if (lpcharge) return 0; return platform_driver_probe(&mdm_modem_driver, mdm_modem_probe); } static void __exit mdm_modem_exit(void) { platform_driver_unregister(&mdm_modem_driver); } late_initcall(mdm_modem_init); /* module_init(mdm_modem_init); */ module_exit(mdm_modem_exit); MODULE_LICENSE("GPL v2"); MODULE_DESCRIPTION("mdm modem driver"); MODULE_VERSION("2.0"); MODULE_ALIAS("mdm_modem");