162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0+ 262306a36Sopenharmony_ci/* 362306a36Sopenharmony_ci * Copyright (C) 2010 ST-Ericsson AB 462306a36Sopenharmony_ci * Mian Yousaf Kaukab <mian.yousaf.kaukab@stericsson.com> 562306a36Sopenharmony_ci * 662306a36Sopenharmony_ci * Based on omap2430.c 762306a36Sopenharmony_ci */ 862306a36Sopenharmony_ci 962306a36Sopenharmony_ci#include <linux/module.h> 1062306a36Sopenharmony_ci#include <linux/kernel.h> 1162306a36Sopenharmony_ci#include <linux/clk.h> 1262306a36Sopenharmony_ci#include <linux/err.h> 1362306a36Sopenharmony_ci#include <linux/io.h> 1462306a36Sopenharmony_ci#include <linux/of.h> 1562306a36Sopenharmony_ci#include <linux/platform_device.h> 1662306a36Sopenharmony_ci#include <linux/usb/musb-ux500.h> 1762306a36Sopenharmony_ci 1862306a36Sopenharmony_ci#include "musb_core.h" 1962306a36Sopenharmony_ci 2062306a36Sopenharmony_cistatic const struct musb_hdrc_config ux500_musb_hdrc_config = { 2162306a36Sopenharmony_ci .multipoint = true, 2262306a36Sopenharmony_ci .dyn_fifo = true, 2362306a36Sopenharmony_ci .num_eps = 16, 2462306a36Sopenharmony_ci .ram_bits = 16, 2562306a36Sopenharmony_ci}; 2662306a36Sopenharmony_ci 2762306a36Sopenharmony_cistruct ux500_glue { 2862306a36Sopenharmony_ci struct device *dev; 2962306a36Sopenharmony_ci struct platform_device *musb; 3062306a36Sopenharmony_ci struct clk *clk; 3162306a36Sopenharmony_ci}; 3262306a36Sopenharmony_ci#define glue_to_musb(g) platform_get_drvdata(g->musb) 3362306a36Sopenharmony_ci 3462306a36Sopenharmony_cistatic void ux500_musb_set_vbus(struct musb *musb, int is_on) 3562306a36Sopenharmony_ci{ 3662306a36Sopenharmony_ci u8 devctl; 3762306a36Sopenharmony_ci unsigned long timeout = jiffies + msecs_to_jiffies(1000); 3862306a36Sopenharmony_ci /* HDRC controls CPEN, but beware current surges during device 3962306a36Sopenharmony_ci * connect. They can trigger transient overcurrent conditions 4062306a36Sopenharmony_ci * that must be ignored. 4162306a36Sopenharmony_ci */ 4262306a36Sopenharmony_ci 4362306a36Sopenharmony_ci devctl = musb_readb(musb->mregs, MUSB_DEVCTL); 4462306a36Sopenharmony_ci 4562306a36Sopenharmony_ci if (is_on) { 4662306a36Sopenharmony_ci if (musb->xceiv->otg->state == OTG_STATE_A_IDLE) { 4762306a36Sopenharmony_ci /* start the session */ 4862306a36Sopenharmony_ci devctl |= MUSB_DEVCTL_SESSION; 4962306a36Sopenharmony_ci musb_writeb(musb->mregs, MUSB_DEVCTL, devctl); 5062306a36Sopenharmony_ci /* 5162306a36Sopenharmony_ci * Wait for the musb to set as A device to enable the 5262306a36Sopenharmony_ci * VBUS 5362306a36Sopenharmony_ci */ 5462306a36Sopenharmony_ci while (musb_readb(musb->mregs, MUSB_DEVCTL) & 0x80) { 5562306a36Sopenharmony_ci 5662306a36Sopenharmony_ci if (time_after(jiffies, timeout)) { 5762306a36Sopenharmony_ci dev_err(musb->controller, 5862306a36Sopenharmony_ci "configured as A device timeout"); 5962306a36Sopenharmony_ci break; 6062306a36Sopenharmony_ci } 6162306a36Sopenharmony_ci } 6262306a36Sopenharmony_ci 6362306a36Sopenharmony_ci } else { 6462306a36Sopenharmony_ci musb->is_active = 1; 6562306a36Sopenharmony_ci musb->xceiv->otg->state = OTG_STATE_A_WAIT_VRISE; 6662306a36Sopenharmony_ci devctl |= MUSB_DEVCTL_SESSION; 6762306a36Sopenharmony_ci MUSB_HST_MODE(musb); 6862306a36Sopenharmony_ci } 6962306a36Sopenharmony_ci } else { 7062306a36Sopenharmony_ci musb->is_active = 0; 7162306a36Sopenharmony_ci 7262306a36Sopenharmony_ci /* NOTE: we're skipping A_WAIT_VFALL -> A_IDLE and jumping 7362306a36Sopenharmony_ci * right to B_IDLE... 7462306a36Sopenharmony_ci */ 7562306a36Sopenharmony_ci devctl &= ~MUSB_DEVCTL_SESSION; 7662306a36Sopenharmony_ci MUSB_DEV_MODE(musb); 7762306a36Sopenharmony_ci } 7862306a36Sopenharmony_ci musb_writeb(musb->mregs, MUSB_DEVCTL, devctl); 7962306a36Sopenharmony_ci 8062306a36Sopenharmony_ci /* 8162306a36Sopenharmony_ci * Devctl values will be updated after vbus goes below 8262306a36Sopenharmony_ci * session_valid. The time taken depends on the capacitance 8362306a36Sopenharmony_ci * on VBUS line. The max discharge time can be upto 1 sec 8462306a36Sopenharmony_ci * as per the spec. Typically on our platform, it is 200ms 8562306a36Sopenharmony_ci */ 8662306a36Sopenharmony_ci if (!is_on) 8762306a36Sopenharmony_ci mdelay(200); 8862306a36Sopenharmony_ci 8962306a36Sopenharmony_ci dev_dbg(musb->controller, "VBUS %s, devctl %02x\n", 9062306a36Sopenharmony_ci usb_otg_state_string(musb->xceiv->otg->state), 9162306a36Sopenharmony_ci musb_readb(musb->mregs, MUSB_DEVCTL)); 9262306a36Sopenharmony_ci} 9362306a36Sopenharmony_ci 9462306a36Sopenharmony_cistatic int musb_otg_notifications(struct notifier_block *nb, 9562306a36Sopenharmony_ci unsigned long event, void *unused) 9662306a36Sopenharmony_ci{ 9762306a36Sopenharmony_ci struct musb *musb = container_of(nb, struct musb, nb); 9862306a36Sopenharmony_ci 9962306a36Sopenharmony_ci dev_dbg(musb->controller, "musb_otg_notifications %ld %s\n", 10062306a36Sopenharmony_ci event, usb_otg_state_string(musb->xceiv->otg->state)); 10162306a36Sopenharmony_ci 10262306a36Sopenharmony_ci switch (event) { 10362306a36Sopenharmony_ci case UX500_MUSB_ID: 10462306a36Sopenharmony_ci dev_dbg(musb->controller, "ID GND\n"); 10562306a36Sopenharmony_ci ux500_musb_set_vbus(musb, 1); 10662306a36Sopenharmony_ci break; 10762306a36Sopenharmony_ci case UX500_MUSB_VBUS: 10862306a36Sopenharmony_ci dev_dbg(musb->controller, "VBUS Connect\n"); 10962306a36Sopenharmony_ci break; 11062306a36Sopenharmony_ci case UX500_MUSB_NONE: 11162306a36Sopenharmony_ci dev_dbg(musb->controller, "VBUS Disconnect\n"); 11262306a36Sopenharmony_ci if (is_host_active(musb)) 11362306a36Sopenharmony_ci ux500_musb_set_vbus(musb, 0); 11462306a36Sopenharmony_ci else 11562306a36Sopenharmony_ci musb->xceiv->otg->state = OTG_STATE_B_IDLE; 11662306a36Sopenharmony_ci break; 11762306a36Sopenharmony_ci default: 11862306a36Sopenharmony_ci dev_dbg(musb->controller, "ID float\n"); 11962306a36Sopenharmony_ci return NOTIFY_DONE; 12062306a36Sopenharmony_ci } 12162306a36Sopenharmony_ci return NOTIFY_OK; 12262306a36Sopenharmony_ci} 12362306a36Sopenharmony_ci 12462306a36Sopenharmony_cistatic irqreturn_t ux500_musb_interrupt(int irq, void *__hci) 12562306a36Sopenharmony_ci{ 12662306a36Sopenharmony_ci unsigned long flags; 12762306a36Sopenharmony_ci irqreturn_t retval = IRQ_NONE; 12862306a36Sopenharmony_ci struct musb *musb = __hci; 12962306a36Sopenharmony_ci 13062306a36Sopenharmony_ci spin_lock_irqsave(&musb->lock, flags); 13162306a36Sopenharmony_ci 13262306a36Sopenharmony_ci musb->int_usb = musb_readb(musb->mregs, MUSB_INTRUSB); 13362306a36Sopenharmony_ci musb->int_tx = musb_readw(musb->mregs, MUSB_INTRTX); 13462306a36Sopenharmony_ci musb->int_rx = musb_readw(musb->mregs, MUSB_INTRRX); 13562306a36Sopenharmony_ci 13662306a36Sopenharmony_ci if (musb->int_usb || musb->int_tx || musb->int_rx) 13762306a36Sopenharmony_ci retval = musb_interrupt(musb); 13862306a36Sopenharmony_ci 13962306a36Sopenharmony_ci spin_unlock_irqrestore(&musb->lock, flags); 14062306a36Sopenharmony_ci 14162306a36Sopenharmony_ci return retval; 14262306a36Sopenharmony_ci} 14362306a36Sopenharmony_ci 14462306a36Sopenharmony_cistatic int ux500_musb_init(struct musb *musb) 14562306a36Sopenharmony_ci{ 14662306a36Sopenharmony_ci int status; 14762306a36Sopenharmony_ci 14862306a36Sopenharmony_ci musb->xceiv = usb_get_phy(USB_PHY_TYPE_USB2); 14962306a36Sopenharmony_ci if (IS_ERR_OR_NULL(musb->xceiv)) { 15062306a36Sopenharmony_ci pr_err("HS USB OTG: no transceiver configured\n"); 15162306a36Sopenharmony_ci return -EPROBE_DEFER; 15262306a36Sopenharmony_ci } 15362306a36Sopenharmony_ci 15462306a36Sopenharmony_ci musb->nb.notifier_call = musb_otg_notifications; 15562306a36Sopenharmony_ci status = usb_register_notifier(musb->xceiv, &musb->nb); 15662306a36Sopenharmony_ci if (status < 0) { 15762306a36Sopenharmony_ci dev_dbg(musb->controller, "notification register failed\n"); 15862306a36Sopenharmony_ci return status; 15962306a36Sopenharmony_ci } 16062306a36Sopenharmony_ci 16162306a36Sopenharmony_ci musb->isr = ux500_musb_interrupt; 16262306a36Sopenharmony_ci 16362306a36Sopenharmony_ci return 0; 16462306a36Sopenharmony_ci} 16562306a36Sopenharmony_ci 16662306a36Sopenharmony_cistatic int ux500_musb_exit(struct musb *musb) 16762306a36Sopenharmony_ci{ 16862306a36Sopenharmony_ci usb_unregister_notifier(musb->xceiv, &musb->nb); 16962306a36Sopenharmony_ci 17062306a36Sopenharmony_ci usb_put_phy(musb->xceiv); 17162306a36Sopenharmony_ci 17262306a36Sopenharmony_ci return 0; 17362306a36Sopenharmony_ci} 17462306a36Sopenharmony_ci 17562306a36Sopenharmony_cistatic const struct musb_platform_ops ux500_ops = { 17662306a36Sopenharmony_ci .quirks = MUSB_DMA_UX500 | MUSB_INDEXED_EP, 17762306a36Sopenharmony_ci#ifdef CONFIG_USB_UX500_DMA 17862306a36Sopenharmony_ci .dma_init = ux500_dma_controller_create, 17962306a36Sopenharmony_ci .dma_exit = ux500_dma_controller_destroy, 18062306a36Sopenharmony_ci#endif 18162306a36Sopenharmony_ci .init = ux500_musb_init, 18262306a36Sopenharmony_ci .exit = ux500_musb_exit, 18362306a36Sopenharmony_ci .fifo_mode = 5, 18462306a36Sopenharmony_ci 18562306a36Sopenharmony_ci .set_vbus = ux500_musb_set_vbus, 18662306a36Sopenharmony_ci}; 18762306a36Sopenharmony_ci 18862306a36Sopenharmony_cistatic struct musb_hdrc_platform_data * 18962306a36Sopenharmony_ciux500_of_probe(struct platform_device *pdev, struct device_node *np) 19062306a36Sopenharmony_ci{ 19162306a36Sopenharmony_ci struct musb_hdrc_platform_data *pdata; 19262306a36Sopenharmony_ci const char *mode; 19362306a36Sopenharmony_ci int strlen; 19462306a36Sopenharmony_ci 19562306a36Sopenharmony_ci pdata = devm_kzalloc(&pdev->dev, sizeof(*pdata), GFP_KERNEL); 19662306a36Sopenharmony_ci if (!pdata) 19762306a36Sopenharmony_ci return NULL; 19862306a36Sopenharmony_ci 19962306a36Sopenharmony_ci mode = of_get_property(np, "dr_mode", &strlen); 20062306a36Sopenharmony_ci if (!mode) { 20162306a36Sopenharmony_ci dev_err(&pdev->dev, "No 'dr_mode' property found\n"); 20262306a36Sopenharmony_ci return NULL; 20362306a36Sopenharmony_ci } 20462306a36Sopenharmony_ci 20562306a36Sopenharmony_ci if (strlen > 0) { 20662306a36Sopenharmony_ci if (!strcmp(mode, "host")) 20762306a36Sopenharmony_ci pdata->mode = MUSB_HOST; 20862306a36Sopenharmony_ci if (!strcmp(mode, "otg")) 20962306a36Sopenharmony_ci pdata->mode = MUSB_OTG; 21062306a36Sopenharmony_ci if (!strcmp(mode, "peripheral")) 21162306a36Sopenharmony_ci pdata->mode = MUSB_PERIPHERAL; 21262306a36Sopenharmony_ci } 21362306a36Sopenharmony_ci 21462306a36Sopenharmony_ci return pdata; 21562306a36Sopenharmony_ci} 21662306a36Sopenharmony_ci 21762306a36Sopenharmony_cistatic int ux500_probe(struct platform_device *pdev) 21862306a36Sopenharmony_ci{ 21962306a36Sopenharmony_ci struct musb_hdrc_platform_data *pdata = dev_get_platdata(&pdev->dev); 22062306a36Sopenharmony_ci struct device_node *np = pdev->dev.of_node; 22162306a36Sopenharmony_ci struct platform_device *musb; 22262306a36Sopenharmony_ci struct ux500_glue *glue; 22362306a36Sopenharmony_ci struct clk *clk; 22462306a36Sopenharmony_ci int ret = -ENOMEM; 22562306a36Sopenharmony_ci 22662306a36Sopenharmony_ci if (!pdata) { 22762306a36Sopenharmony_ci if (np) { 22862306a36Sopenharmony_ci pdata = ux500_of_probe(pdev, np); 22962306a36Sopenharmony_ci if (!pdata) 23062306a36Sopenharmony_ci goto err0; 23162306a36Sopenharmony_ci 23262306a36Sopenharmony_ci pdev->dev.platform_data = pdata; 23362306a36Sopenharmony_ci } else { 23462306a36Sopenharmony_ci dev_err(&pdev->dev, "no pdata or device tree found\n"); 23562306a36Sopenharmony_ci goto err0; 23662306a36Sopenharmony_ci } 23762306a36Sopenharmony_ci } 23862306a36Sopenharmony_ci 23962306a36Sopenharmony_ci glue = devm_kzalloc(&pdev->dev, sizeof(*glue), GFP_KERNEL); 24062306a36Sopenharmony_ci if (!glue) 24162306a36Sopenharmony_ci goto err0; 24262306a36Sopenharmony_ci 24362306a36Sopenharmony_ci musb = platform_device_alloc("musb-hdrc", PLATFORM_DEVID_AUTO); 24462306a36Sopenharmony_ci if (!musb) { 24562306a36Sopenharmony_ci dev_err(&pdev->dev, "failed to allocate musb device\n"); 24662306a36Sopenharmony_ci goto err0; 24762306a36Sopenharmony_ci } 24862306a36Sopenharmony_ci 24962306a36Sopenharmony_ci clk = devm_clk_get(&pdev->dev, NULL); 25062306a36Sopenharmony_ci if (IS_ERR(clk)) { 25162306a36Sopenharmony_ci dev_err(&pdev->dev, "failed to get clock\n"); 25262306a36Sopenharmony_ci ret = PTR_ERR(clk); 25362306a36Sopenharmony_ci goto err1; 25462306a36Sopenharmony_ci } 25562306a36Sopenharmony_ci 25662306a36Sopenharmony_ci ret = clk_prepare_enable(clk); 25762306a36Sopenharmony_ci if (ret) { 25862306a36Sopenharmony_ci dev_err(&pdev->dev, "failed to enable clock\n"); 25962306a36Sopenharmony_ci goto err1; 26062306a36Sopenharmony_ci } 26162306a36Sopenharmony_ci 26262306a36Sopenharmony_ci musb->dev.parent = &pdev->dev; 26362306a36Sopenharmony_ci musb->dev.dma_mask = &pdev->dev.coherent_dma_mask; 26462306a36Sopenharmony_ci musb->dev.coherent_dma_mask = pdev->dev.coherent_dma_mask; 26562306a36Sopenharmony_ci device_set_of_node_from_dev(&musb->dev, &pdev->dev); 26662306a36Sopenharmony_ci 26762306a36Sopenharmony_ci glue->dev = &pdev->dev; 26862306a36Sopenharmony_ci glue->musb = musb; 26962306a36Sopenharmony_ci glue->clk = clk; 27062306a36Sopenharmony_ci 27162306a36Sopenharmony_ci pdata->platform_ops = &ux500_ops; 27262306a36Sopenharmony_ci pdata->config = &ux500_musb_hdrc_config; 27362306a36Sopenharmony_ci 27462306a36Sopenharmony_ci platform_set_drvdata(pdev, glue); 27562306a36Sopenharmony_ci 27662306a36Sopenharmony_ci ret = platform_device_add_resources(musb, pdev->resource, pdev->num_resources); 27762306a36Sopenharmony_ci if (ret) { 27862306a36Sopenharmony_ci dev_err(&pdev->dev, "failed to add resources\n"); 27962306a36Sopenharmony_ci goto err2; 28062306a36Sopenharmony_ci } 28162306a36Sopenharmony_ci 28262306a36Sopenharmony_ci ret = platform_device_add_data(musb, pdata, sizeof(*pdata)); 28362306a36Sopenharmony_ci if (ret) { 28462306a36Sopenharmony_ci dev_err(&pdev->dev, "failed to add platform_data\n"); 28562306a36Sopenharmony_ci goto err2; 28662306a36Sopenharmony_ci } 28762306a36Sopenharmony_ci 28862306a36Sopenharmony_ci ret = platform_device_add(musb); 28962306a36Sopenharmony_ci if (ret) { 29062306a36Sopenharmony_ci dev_err(&pdev->dev, "failed to register musb device\n"); 29162306a36Sopenharmony_ci goto err2; 29262306a36Sopenharmony_ci } 29362306a36Sopenharmony_ci 29462306a36Sopenharmony_ci return 0; 29562306a36Sopenharmony_ci 29662306a36Sopenharmony_cierr2: 29762306a36Sopenharmony_ci clk_disable_unprepare(clk); 29862306a36Sopenharmony_ci 29962306a36Sopenharmony_cierr1: 30062306a36Sopenharmony_ci platform_device_put(musb); 30162306a36Sopenharmony_ci 30262306a36Sopenharmony_cierr0: 30362306a36Sopenharmony_ci return ret; 30462306a36Sopenharmony_ci} 30562306a36Sopenharmony_ci 30662306a36Sopenharmony_cistatic void ux500_remove(struct platform_device *pdev) 30762306a36Sopenharmony_ci{ 30862306a36Sopenharmony_ci struct ux500_glue *glue = platform_get_drvdata(pdev); 30962306a36Sopenharmony_ci 31062306a36Sopenharmony_ci platform_device_unregister(glue->musb); 31162306a36Sopenharmony_ci clk_disable_unprepare(glue->clk); 31262306a36Sopenharmony_ci} 31362306a36Sopenharmony_ci 31462306a36Sopenharmony_ci#ifdef CONFIG_PM_SLEEP 31562306a36Sopenharmony_cistatic int ux500_suspend(struct device *dev) 31662306a36Sopenharmony_ci{ 31762306a36Sopenharmony_ci struct ux500_glue *glue = dev_get_drvdata(dev); 31862306a36Sopenharmony_ci struct musb *musb = glue_to_musb(glue); 31962306a36Sopenharmony_ci 32062306a36Sopenharmony_ci if (musb) 32162306a36Sopenharmony_ci usb_phy_set_suspend(musb->xceiv, 1); 32262306a36Sopenharmony_ci 32362306a36Sopenharmony_ci clk_disable_unprepare(glue->clk); 32462306a36Sopenharmony_ci 32562306a36Sopenharmony_ci return 0; 32662306a36Sopenharmony_ci} 32762306a36Sopenharmony_ci 32862306a36Sopenharmony_cistatic int ux500_resume(struct device *dev) 32962306a36Sopenharmony_ci{ 33062306a36Sopenharmony_ci struct ux500_glue *glue = dev_get_drvdata(dev); 33162306a36Sopenharmony_ci struct musb *musb = glue_to_musb(glue); 33262306a36Sopenharmony_ci int ret; 33362306a36Sopenharmony_ci 33462306a36Sopenharmony_ci ret = clk_prepare_enable(glue->clk); 33562306a36Sopenharmony_ci if (ret) { 33662306a36Sopenharmony_ci dev_err(dev, "failed to enable clock\n"); 33762306a36Sopenharmony_ci return ret; 33862306a36Sopenharmony_ci } 33962306a36Sopenharmony_ci 34062306a36Sopenharmony_ci if (musb) 34162306a36Sopenharmony_ci usb_phy_set_suspend(musb->xceiv, 0); 34262306a36Sopenharmony_ci 34362306a36Sopenharmony_ci return 0; 34462306a36Sopenharmony_ci} 34562306a36Sopenharmony_ci#endif 34662306a36Sopenharmony_ci 34762306a36Sopenharmony_cistatic SIMPLE_DEV_PM_OPS(ux500_pm_ops, ux500_suspend, ux500_resume); 34862306a36Sopenharmony_ci 34962306a36Sopenharmony_cistatic const struct of_device_id ux500_match[] = { 35062306a36Sopenharmony_ci { .compatible = "stericsson,db8500-musb", }, 35162306a36Sopenharmony_ci {} 35262306a36Sopenharmony_ci}; 35362306a36Sopenharmony_ci 35462306a36Sopenharmony_ciMODULE_DEVICE_TABLE(of, ux500_match); 35562306a36Sopenharmony_ci 35662306a36Sopenharmony_cistatic struct platform_driver ux500_driver = { 35762306a36Sopenharmony_ci .probe = ux500_probe, 35862306a36Sopenharmony_ci .remove_new = ux500_remove, 35962306a36Sopenharmony_ci .driver = { 36062306a36Sopenharmony_ci .name = "musb-ux500", 36162306a36Sopenharmony_ci .pm = &ux500_pm_ops, 36262306a36Sopenharmony_ci .of_match_table = ux500_match, 36362306a36Sopenharmony_ci }, 36462306a36Sopenharmony_ci}; 36562306a36Sopenharmony_ci 36662306a36Sopenharmony_ciMODULE_DESCRIPTION("UX500 MUSB Glue Layer"); 36762306a36Sopenharmony_ciMODULE_AUTHOR("Mian Yousaf Kaukab <mian.yousaf.kaukab@stericsson.com>"); 36862306a36Sopenharmony_ciMODULE_LICENSE("GPL v2"); 36962306a36Sopenharmony_cimodule_platform_driver(ux500_driver); 370