Commit 54dfbb08 authored by Fabio Baltieri's avatar Fabio Baltieri Committed by Felipe Balbi

usb: phy: ab8500-usb: enable/disable regulator on phy events

Add ab8500_usb_regulator_{enable,disable} functions to control USB phy
regulators on corresponding ab8500_usb_phy_{enable,disable} events.

This contains some workaround and optimization for specific AB8500
versions.
Signed-off-by: default avatarMian Yousaf Kaukab <mian.yousaf.kaukab@stericsson.com>
Signed-off-by: default avatarSakethram Bommisetti <sakethram.bommisetti@stericsson.com>
Signed-off-by: default avatarPraveena Nadahally <praveen.nadahally@stericsson.com>
Acked-by: default avatarLinus Walleij <linus.walleij@linaro.org>
Signed-off-by: default avatarFabio Baltieri <fabio.baltieri@linaro.org>
Signed-off-by: default avatarFelipe Balbi <balbi@ti.com>
parent c0ea7064
...@@ -130,6 +130,7 @@ struct ab8500_usb { ...@@ -130,6 +130,7 @@ struct ab8500_usb {
struct regulator *v_ape; struct regulator *v_ape;
struct regulator *v_musb; struct regulator *v_musb;
struct regulator *v_ulpi; struct regulator *v_ulpi;
int saved_v_ulpi;
int previous_link_status_state; int previous_link_status_state;
}; };
...@@ -161,6 +162,67 @@ static void ab8500_usb_wd_workaround(struct ab8500_usb *ab) ...@@ -161,6 +162,67 @@ static void ab8500_usb_wd_workaround(struct ab8500_usb *ab)
0); 0);
} }
static void ab8500_usb_regulator_enable(struct ab8500_usb *ab)
{
int ret, volt;
regulator_enable(ab->v_ape);
if (!is_ab8500_2p0_or_earlier(ab->ab8500)) {
ab->saved_v_ulpi = regulator_get_voltage(ab->v_ulpi);
if (ab->saved_v_ulpi < 0)
dev_err(ab->dev, "Failed to get v_ulpi voltage\n");
ret = regulator_set_voltage(ab->v_ulpi, 1300000, 1350000);
if (ret < 0)
dev_err(ab->dev, "Failed to set the Vintcore to 1.3V, ret=%d\n",
ret);
ret = regulator_set_optimum_mode(ab->v_ulpi, 28000);
if (ret < 0)
dev_err(ab->dev, "Failed to set optimum mode (ret=%d)\n",
ret);
}
regulator_enable(ab->v_ulpi);
if (!is_ab8500_2p0_or_earlier(ab->ab8500)) {
volt = regulator_get_voltage(ab->v_ulpi);
if ((volt != 1300000) && (volt != 1350000))
dev_err(ab->dev, "Vintcore is not set to 1.3V volt=%d\n",
volt);
}
regulator_enable(ab->v_musb);
}
static void ab8500_usb_regulator_disable(struct ab8500_usb *ab)
{
int ret;
regulator_disable(ab->v_musb);
regulator_disable(ab->v_ulpi);
/* USB is not the only consumer of Vintcore, restore old settings */
if (!is_ab8500_2p0_or_earlier(ab->ab8500)) {
if (ab->saved_v_ulpi > 0) {
ret = regulator_set_voltage(ab->v_ulpi,
ab->saved_v_ulpi, ab->saved_v_ulpi);
if (ret < 0)
dev_err(ab->dev, "Failed to set the Vintcore to %duV, ret=%d\n",
ab->saved_v_ulpi, ret);
}
ret = regulator_set_optimum_mode(ab->v_ulpi, 0);
if (ret < 0)
dev_err(ab->dev, "Failed to set optimum mode (ret=%d)\n",
ret);
}
regulator_disable(ab->v_ape);
}
static void ab8500_usb_wd_linkstatus(struct ab8500_usb *ab, u8 bit) static void ab8500_usb_wd_linkstatus(struct ab8500_usb *ab, u8 bit)
{ {
/* Workaround for v2.0 bug # 31952 */ /* Workaround for v2.0 bug # 31952 */
...@@ -178,6 +240,8 @@ static void ab8500_usb_phy_enable(struct ab8500_usb *ab, bool sel_host) ...@@ -178,6 +240,8 @@ static void ab8500_usb_phy_enable(struct ab8500_usb *ab, bool sel_host)
bit = sel_host ? AB8500_BIT_PHY_CTRL_HOST_EN : bit = sel_host ? AB8500_BIT_PHY_CTRL_HOST_EN :
AB8500_BIT_PHY_CTRL_DEVICE_EN; AB8500_BIT_PHY_CTRL_DEVICE_EN;
ab8500_usb_regulator_enable(ab);
abx500_mask_and_set_register_interruptible(ab->dev, abx500_mask_and_set_register_interruptible(ab->dev,
AB8500_USB, AB8500_USB_PHY_CTRL_REG, AB8500_USB, AB8500_USB_PHY_CTRL_REG,
bit, bit); bit, bit);
...@@ -197,6 +261,8 @@ static void ab8500_usb_phy_disable(struct ab8500_usb *ab, bool sel_host) ...@@ -197,6 +261,8 @@ static void ab8500_usb_phy_disable(struct ab8500_usb *ab, bool sel_host)
/* Needed to disable the phy.*/ /* Needed to disable the phy.*/
ab8500_usb_wd_workaround(ab); ab8500_usb_wd_workaround(ab);
ab8500_usb_regulator_disable(ab);
} }
#define ab8500_usb_host_phy_en(ab) ab8500_usb_phy_enable(ab, true) #define ab8500_usb_host_phy_en(ab) ab8500_usb_phy_enable(ab, true)
...@@ -544,7 +610,6 @@ static int ab8500_usb_set_peripheral(struct usb_otg *otg, ...@@ -544,7 +610,6 @@ static int ab8500_usb_set_peripheral(struct usb_otg *otg,
*/ */
if (!gadget) { if (!gadget) {
/* TODO: Disable regulators. */
otg->gadget = NULL; otg->gadget = NULL;
schedule_work(&ab->phy_dis_work); schedule_work(&ab->phy_dis_work);
} else { } else {
...@@ -576,7 +641,6 @@ static int ab8500_usb_set_host(struct usb_otg *otg, struct usb_bus *host) ...@@ -576,7 +641,6 @@ static int ab8500_usb_set_host(struct usb_otg *otg, struct usb_bus *host)
*/ */
if (!host) { if (!host) {
/* TODO: Disable regulators. */
otg->host = NULL; otg->host = NULL;
schedule_work(&ab->phy_dis_work); schedule_work(&ab->phy_dis_work);
} else { } else {
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment