Donate to e Foundation | Murena handsets with /e/OS | Own a part of Murena! Learn more

Commit 869c5978 authored by Daniel Mack's avatar Daniel Mack Committed by Felipe Balbi
Browse files

usb: musb: dsps: add support for suspend and resume



The dsps platform needs to save save some registers at suspend time and
restore them after resume. This patch adds a struct for these registers,
and also lets the musb core know that the core registers need to be
saved as well.

We also have to explicitly de-assert the port reset upon resume on this
platform, but musb_port_reset() should not be called from glue layers.

Hence, introduce a flag in struct musb_hdrc_config for this.

Signed-off-by: default avatarDaniel Mack <zonque@gmail.com>
Signed-off-by: default avatarFelipe Balbi <balbi@ti.com>
parent b991f9b7
Loading
Loading
Loading
Loading
+57 −0
Original line number Original line Diff line number Diff line
@@ -113,6 +113,19 @@ struct dsps_musb_wrapper {
	u8		poll_seconds;
	u8		poll_seconds;
};
};


/*
 * register shadow for suspend
 */
struct dsps_context {
	u32 control;
	u32 epintr;
	u32 coreintr;
	u32 phy_utmi;
	u32 mode;
	u32 tx_mode;
	u32 rx_mode;
};

/**
/**
 * DSPS glue structure.
 * DSPS glue structure.
 */
 */
@@ -122,6 +135,8 @@ struct dsps_glue {
	const struct dsps_musb_wrapper *wrp; /* wrapper register offsets */
	const struct dsps_musb_wrapper *wrp; /* wrapper register offsets */
	struct timer_list timer;	/* otg_workaround timer */
	struct timer_list timer;	/* otg_workaround timer */
	unsigned long last_timer;    /* last timer data for each instance */
	unsigned long last_timer;    /* last timer data for each instance */

	struct dsps_context context;
};
};


static void dsps_musb_try_idle(struct musb *musb, unsigned long timeout)
static void dsps_musb_try_idle(struct musb *musb, unsigned long timeout)
@@ -559,6 +574,7 @@ static int dsps_create_musb_pdev(struct dsps_glue *glue,


	config->num_eps = get_int_prop(dn, "mentor,num-eps");
	config->num_eps = get_int_prop(dn, "mentor,num-eps");
	config->ram_bits = get_int_prop(dn, "mentor,ram-bits");
	config->ram_bits = get_int_prop(dn, "mentor,ram-bits");
	config->host_port_deassert_reset_at_resume = 1;
	pdata.mode = get_musb_port_mode(dev);
	pdata.mode = get_musb_port_mode(dev);
	/* DT keeps this entry in mA, musb expects it as per USB spec */
	/* DT keeps this entry in mA, musb expects it as per USB spec */
	pdata.power = get_int_prop(dn, "mentor,power") / 2;
	pdata.power = get_int_prop(dn, "mentor,power") / 2;
@@ -683,11 +699,52 @@ static const struct of_device_id musb_dsps_of_match[] = {
};
};
MODULE_DEVICE_TABLE(of, musb_dsps_of_match);
MODULE_DEVICE_TABLE(of, musb_dsps_of_match);


#ifdef CONFIG_PM
static int dsps_suspend(struct device *dev)
{
	struct dsps_glue *glue = dev_get_drvdata(dev);
	const struct dsps_musb_wrapper *wrp = glue->wrp;
	struct musb *musb = platform_get_drvdata(glue->musb);
	void __iomem *mbase = musb->ctrl_base;

	glue->context.control = dsps_readl(mbase, wrp->control);
	glue->context.epintr = dsps_readl(mbase, wrp->epintr_set);
	glue->context.coreintr = dsps_readl(mbase, wrp->coreintr_set);
	glue->context.phy_utmi = dsps_readl(mbase, wrp->phy_utmi);
	glue->context.mode = dsps_readl(mbase, wrp->mode);
	glue->context.tx_mode = dsps_readl(mbase, wrp->tx_mode);
	glue->context.rx_mode = dsps_readl(mbase, wrp->rx_mode);

	return 0;
}

static int dsps_resume(struct device *dev)
{
	struct dsps_glue *glue = dev_get_drvdata(dev);
	const struct dsps_musb_wrapper *wrp = glue->wrp;
	struct musb *musb = platform_get_drvdata(glue->musb);
	void __iomem *mbase = musb->ctrl_base;

	dsps_writel(mbase, wrp->control, glue->context.control);
	dsps_writel(mbase, wrp->epintr_set, glue->context.epintr);
	dsps_writel(mbase, wrp->coreintr_set, glue->context.coreintr);
	dsps_writel(mbase, wrp->phy_utmi, glue->context.phy_utmi);
	dsps_writel(mbase, wrp->mode, glue->context.mode);
	dsps_writel(mbase, wrp->tx_mode, glue->context.tx_mode);
	dsps_writel(mbase, wrp->rx_mode, glue->context.rx_mode);

	return 0;
}
#endif

static SIMPLE_DEV_PM_OPS(dsps_pm_ops, dsps_suspend, dsps_resume);

static struct platform_driver dsps_usbss_driver = {
static struct platform_driver dsps_usbss_driver = {
	.probe		= dsps_probe,
	.probe		= dsps_probe,
	.remove         = dsps_remove,
	.remove         = dsps_remove,
	.driver         = {
	.driver         = {
		.name   = "musb-dsps",
		.name   = "musb-dsps",
		.pm	= &dsps_pm_ops,
		.of_match_table	= musb_dsps_of_match,
		.of_match_table	= musb_dsps_of_match,
	},
	},
};
};
+6 −1
Original line number Original line Diff line number Diff line
@@ -2464,7 +2464,12 @@ static int musb_bus_suspend(struct usb_hcd *hcd)


static int musb_bus_resume(struct usb_hcd *hcd)
static int musb_bus_resume(struct usb_hcd *hcd)
{
{
	/* resuming child port does the work */
	struct musb *musb = hcd_to_musb(hcd);

	if (musb->config &&
	    musb->config->host_port_deassert_reset_at_resume)
		musb_port_reset(musb, false);

	return 0;
	return 0;
}
}


+2 −0
Original line number Original line Diff line number Diff line
@@ -93,6 +93,7 @@ extern void musb_root_disconnect(struct musb *musb);
extern void musb_host_resume_root_hub(struct musb *musb);
extern void musb_host_resume_root_hub(struct musb *musb);
extern void musb_host_poke_root_hub(struct musb *musb);
extern void musb_host_poke_root_hub(struct musb *musb);
extern void musb_port_suspend(struct musb *musb, bool do_suspend);
extern void musb_port_suspend(struct musb *musb, bool do_suspend);
extern void musb_port_reset(struct musb *musb, bool do_reset);
#else
#else
static inline struct musb *hcd_to_musb(struct usb_hcd *hcd)
static inline struct musb *hcd_to_musb(struct usb_hcd *hcd)
{
{
@@ -123,6 +124,7 @@ static inline void musb_host_resume_root_hub(struct musb *musb) {}
static inline void musb_host_poll_rh_status(struct musb *musb)	{}
static inline void musb_host_poll_rh_status(struct musb *musb)	{}
static inline void musb_host_poke_root_hub(struct musb *musb)	{}
static inline void musb_host_poke_root_hub(struct musb *musb)	{}
static inline void musb_port_suspend(struct musb *musb, bool do_suspend) {}
static inline void musb_port_suspend(struct musb *musb, bool do_suspend) {}
static inline void musb_port_reset(struct musb *musb)		{}
#endif
#endif


struct usb_hcd;
struct usb_hcd;
+1 −1
Original line number Original line Diff line number Diff line
@@ -109,7 +109,7 @@ void musb_port_suspend(struct musb *musb, bool do_suspend)
	}
	}
}
}


static void musb_port_reset(struct musb *musb, bool do_reset)
void musb_port_reset(struct musb *musb, bool do_reset)
{
{
	u8		power;
	u8		power;
	void __iomem	*mbase = musb->mregs;
	void __iomem	*mbase = musb->mregs;
+3 −0
Original line number Original line Diff line number Diff line
@@ -76,6 +76,9 @@ struct musb_hdrc_config {
	unsigned	dma:1 __deprecated; /* supports DMA */
	unsigned	dma:1 __deprecated; /* supports DMA */
	unsigned	vendor_req:1 __deprecated; /* vendor registers required */
	unsigned	vendor_req:1 __deprecated; /* vendor registers required */


	/* need to explicitly de-assert the port reset after resume? */
	unsigned	host_port_deassert_reset_at_resume:1;

	u8		num_eps;	/* number of endpoints _with_ ep0 */
	u8		num_eps;	/* number of endpoints _with_ ep0 */
	u8		dma_channels __deprecated; /* number of dma channels */
	u8		dma_channels __deprecated; /* number of dma channels */
	u8		dyn_fifo_size;	/* dynamic size in bytes */
	u8		dyn_fifo_size;	/* dynamic size in bytes */