blob: 0737a81ff377181aa642eef317e51574968efd7e [file] [log] [blame]
/* Copyright (c) 2011, 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 <linux/module.h>
#include <linux/platform_device.h>
#include <linux/err.h>
#include <linux/slab.h>
#include <linux/io.h>
#include <linux/mutex.h>
#include <linux/miscdevice.h>
#include <linux/fs.h>
#include <linux/gpio.h>
#include <linux/kernel.h>
#include <linux/irq.h>
#include <linux/ioctl.h>
#include <linux/delay.h>
#include <linux/reboot.h>
#include <linux/debugfs.h>
#include <linux/completion.h>
#include <linux/workqueue.h>
#include <linux/clk.h>
#include <linux/mfd/pmic8058.h>
#include <asm/mach-types.h>
#include <asm/uaccess.h>
#include <mach/mdm2.h>
#include <mach/restart.h>
#include <mach/subsystem_notif.h>
#include <mach/subsystem_restart.h>
#include <linux/msm_charm.h>
#include "msm_watchdog.h"
#include "devices.h"
#include "clock.h"
#include "mdm_private.h"
#define MDM_MODEM_TIMEOUT 6000
#define MDM_HOLD_TIME 4000
#define MDM_MODEM_DELTA 100
#define IFLINE_UP 1
#define IFLINE_DOWN 0
static int mdm_debug_on;
static int ifline_status = IFLINE_UP;
static struct mdm_callbacks mdm_cb;
#define MDM_DBG(...) do { if (mdm_debug_on) \
pr_info(__VA_ARGS__); \
} while (0);
static void power_on_mdm(struct mdm_modem_drv *mdm_drv)
{
/* Remove hsic driver before powering on the modem. */
if (ifline_status == IFLINE_UP) {
MDM_DBG("%s: Removing hsic device\n", __func__);
platform_device_del(&msm_device_hsic_host);
ifline_status = IFLINE_DOWN;
}
/* Pull both ERR_FATAL and RESET low */
MDM_DBG("Pulling PWR and RESET gpio's low\n");
gpio_direction_output(mdm_drv->ap2mdm_pmic_reset_n_gpio, 0);
gpio_direction_output(mdm_drv->ap2mdm_kpdpwr_n_gpio, 0);
/* Wait for them to settle. */
usleep(1000);
/* Deassert RESET first and wait for ir to settle. */
MDM_DBG("%s: Pulling RESET gpio high\n", __func__);
gpio_direction_output(mdm_drv->ap2mdm_pmic_reset_n_gpio, 1);
usleep(1000);
/* Pull PWR gpio high and wait for it to settle. */
MDM_DBG("%s: Powering on mdm modem\n", __func__);
gpio_direction_output(mdm_drv->ap2mdm_kpdpwr_n_gpio, 1);
usleep(1000);
/* Add back hsic device after modem power up */
MDM_DBG("%s: Adding hsic device\n", __func__);
platform_device_add(&msm_device_hsic_host);
ifline_status = IFLINE_UP;
msleep(200);
}
static void power_down_mdm(struct mdm_modem_drv *mdm_drv)
{
int i;
for (i = MDM_MODEM_TIMEOUT; i > 0; i -= MDM_MODEM_DELTA) {
pet_watchdog();
msleep(MDM_MODEM_DELTA);
if (gpio_get_value(mdm_drv->mdm2ap_status_gpio) == 0)
break;
}
if (i <= 0) {
pr_err("%s: MDM2AP_STATUS never went low.\n",
__func__);
gpio_direction_output(mdm_drv->ap2mdm_pmic_reset_n_gpio, 0);
for (i = MDM_HOLD_TIME; i > 0; i -= MDM_MODEM_DELTA) {
pet_watchdog();
msleep(MDM_MODEM_DELTA);
}
}
/* Also remove the hsic device on 9k power down. */
MDM_DBG("%s: Removing hsic device\n", __func__);
if (ifline_status == IFLINE_UP) {
platform_device_del(&msm_device_hsic_host);
ifline_status = IFLINE_DOWN;
}
}
static void normal_boot_done(struct mdm_modem_drv *mdm_drv)
{
}
static void debug_state_changed(int value)
{
mdm_debug_on = value;
}
static int __init mdm_modem_probe(struct platform_device *pdev)
{
/* Instantiate driver object. */
mdm_cb.power_on_mdm_cb = power_on_mdm;
mdm_cb.power_down_mdm_cb = power_down_mdm;
mdm_cb.normal_boot_done_cb = normal_boot_done;
mdm_cb.debug_state_changed_cb = debug_state_changed;
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);
}
static struct platform_driver mdm_modem_driver = {
.remove = mdm_modem_remove,
.shutdown = mdm_modem_shutdown,
.driver = {
.name = "mdm2_modem",
.owner = THIS_MODULE
},
};
static int __init mdm_modem_init(void)
{
return platform_driver_probe(&mdm_modem_driver, mdm_modem_probe);
}
static void __exit mdm_modem_exit(void)
{
platform_driver_unregister(&mdm_modem_driver);
}
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");