Commit dc4c1aa7 authored by Chunfeng Yun's avatar Chunfeng Yun Committed by Greg Kroah-Hartman

usb: mtu3: add ->udc_set_speed()

Make use of the method to make sure connect on speeds
supported by the gadget driver.
Signed-off-by: default avatarChunfeng Yun <chunfeng.yun@mediatek.com>
Link: https://lore.kernel.org/r/1595834101-13094-6-git-send-email-chunfeng.yun@mediatek.comSigned-off-by: default avatarGreg Kroah-Hartman <gregkh@linuxfoundation.org>
parent 1258450e
...@@ -348,7 +348,8 @@ struct mtu3 { ...@@ -348,7 +348,8 @@ struct mtu3 {
struct usb_gadget_driver *gadget_driver; struct usb_gadget_driver *gadget_driver;
struct mtu3_request ep0_req; struct mtu3_request ep0_req;
u8 setup_buf[EP0_RESPONSE_BUF]; u8 setup_buf[EP0_RESPONSE_BUF];
u32 max_speed; enum usb_device_speed max_speed;
enum usb_device_speed speed;
unsigned is_active:1; unsigned is_active:1;
unsigned may_wakeup:1; unsigned may_wakeup:1;
...@@ -433,6 +434,7 @@ void mtu3_ep0_setup(struct mtu3 *mtu); ...@@ -433,6 +434,7 @@ void mtu3_ep0_setup(struct mtu3 *mtu);
void mtu3_start(struct mtu3 *mtu); void mtu3_start(struct mtu3 *mtu);
void mtu3_stop(struct mtu3 *mtu); void mtu3_stop(struct mtu3 *mtu);
void mtu3_dev_on_off(struct mtu3 *mtu, int is_on); void mtu3_dev_on_off(struct mtu3 *mtu, int is_on);
void mtu3_set_speed(struct mtu3 *mtu, enum usb_device_speed speed);
int mtu3_gadget_setup(struct mtu3 *mtu); int mtu3_gadget_setup(struct mtu3 *mtu);
void mtu3_gadget_cleanup(struct mtu3 *mtu); void mtu3_gadget_cleanup(struct mtu3 *mtu);
......
...@@ -203,7 +203,42 @@ static void mtu3_intr_enable(struct mtu3 *mtu) ...@@ -203,7 +203,42 @@ static void mtu3_intr_enable(struct mtu3 *mtu)
mtu3_writel(mbase, U3D_DEV_LINK_INTR_ENABLE, SSUSB_DEV_SPEED_CHG_INTR); mtu3_writel(mbase, U3D_DEV_LINK_INTR_ENABLE, SSUSB_DEV_SPEED_CHG_INTR);
} }
static void mtu3_set_speed(struct mtu3 *mtu); void mtu3_set_speed(struct mtu3 *mtu, enum usb_device_speed speed)
{
void __iomem *mbase = mtu->mac_base;
if (speed > mtu->max_speed)
speed = mtu->max_speed;
switch (speed) {
case USB_SPEED_FULL:
/* disable U3 SS function */
mtu3_clrbits(mbase, U3D_USB3_CONFIG, USB3_EN);
/* disable HS function */
mtu3_clrbits(mbase, U3D_POWER_MANAGEMENT, HS_ENABLE);
break;
case USB_SPEED_HIGH:
mtu3_clrbits(mbase, U3D_USB3_CONFIG, USB3_EN);
/* HS/FS detected by HW */
mtu3_setbits(mbase, U3D_POWER_MANAGEMENT, HS_ENABLE);
break;
case USB_SPEED_SUPER:
mtu3_clrbits(mtu->ippc_base, SSUSB_U3_CTRL(0),
SSUSB_U3_PORT_SSP_SPEED);
break;
case USB_SPEED_SUPER_PLUS:
mtu3_setbits(mtu->ippc_base, SSUSB_U3_CTRL(0),
SSUSB_U3_PORT_SSP_SPEED);
break;
default:
dev_err(mtu->dev, "invalid speed: %s\n",
usb_speed_string(speed));
return;
}
mtu->speed = speed;
dev_dbg(mtu->dev, "set speed: %s\n", usb_speed_string(speed));
}
/* CSR registers will be reset to default value if port is disabled */ /* CSR registers will be reset to default value if port is disabled */
static void mtu3_csr_init(struct mtu3 *mtu) static void mtu3_csr_init(struct mtu3 *mtu)
...@@ -225,8 +260,6 @@ static void mtu3_csr_init(struct mtu3 *mtu) ...@@ -225,8 +260,6 @@ static void mtu3_csr_init(struct mtu3 *mtu)
mtu3_setbits(mbase, U3D_U3U2_SWITCH_CTRL, SOFTCON_CLR_AUTO_EN); mtu3_setbits(mbase, U3D_U3U2_SWITCH_CTRL, SOFTCON_CLR_AUTO_EN);
} }
mtu3_set_speed(mtu);
/* delay about 0.1us from detecting reset to send chirp-K */ /* delay about 0.1us from detecting reset to send chirp-K */
mtu3_clrbits(mbase, U3D_LINK_RESET_INFO, WTCHRP_MSK); mtu3_clrbits(mbase, U3D_LINK_RESET_INFO, WTCHRP_MSK);
/* enable automatical HWRW from L1 */ /* enable automatical HWRW from L1 */
...@@ -280,13 +313,13 @@ void mtu3_ep_stall_set(struct mtu3_ep *mep, bool set) ...@@ -280,13 +313,13 @@ void mtu3_ep_stall_set(struct mtu3_ep *mep, bool set)
void mtu3_dev_on_off(struct mtu3 *mtu, int is_on) void mtu3_dev_on_off(struct mtu3 *mtu, int is_on)
{ {
if (mtu->is_u3_ip && mtu->max_speed >= USB_SPEED_SUPER) if (mtu->is_u3_ip && mtu->speed >= USB_SPEED_SUPER)
mtu3_ss_func_set(mtu, is_on); mtu3_ss_func_set(mtu, is_on);
else else
mtu3_hs_softconn_set(mtu, is_on); mtu3_hs_softconn_set(mtu, is_on);
dev_info(mtu->dev, "gadget (%s) pullup D%s\n", dev_info(mtu->dev, "gadget (%s) pullup D%s\n",
usb_speed_string(mtu->max_speed), is_on ? "+" : "-"); usb_speed_string(mtu->speed), is_on ? "+" : "-");
} }
void mtu3_start(struct mtu3 *mtu) void mtu3_start(struct mtu3 *mtu)
...@@ -299,6 +332,7 @@ void mtu3_start(struct mtu3 *mtu) ...@@ -299,6 +332,7 @@ void mtu3_start(struct mtu3 *mtu)
mtu3_clrbits(mtu->ippc_base, U3D_SSUSB_IP_PW_CTRL2, SSUSB_IP_DEV_PDN); mtu3_clrbits(mtu->ippc_base, U3D_SSUSB_IP_PW_CTRL2, SSUSB_IP_DEV_PDN);
mtu3_csr_init(mtu); mtu3_csr_init(mtu);
mtu3_set_speed(mtu, mtu->speed);
/* Initialize the default interrupts */ /* Initialize the default interrupts */
mtu3_intr_enable(mtu); mtu3_intr_enable(mtu);
...@@ -569,28 +603,6 @@ static void mtu3_mem_free(struct mtu3 *mtu) ...@@ -569,28 +603,6 @@ static void mtu3_mem_free(struct mtu3 *mtu)
kfree(mtu->ep_array); kfree(mtu->ep_array);
} }
static void mtu3_set_speed(struct mtu3 *mtu)
{
void __iomem *mbase = mtu->mac_base;
if (mtu->max_speed == USB_SPEED_FULL) {
/* disable U3 SS function */
mtu3_clrbits(mbase, U3D_USB3_CONFIG, USB3_EN);
/* disable HS function */
mtu3_clrbits(mbase, U3D_POWER_MANAGEMENT, HS_ENABLE);
} else if (mtu->max_speed == USB_SPEED_HIGH) {
mtu3_clrbits(mbase, U3D_USB3_CONFIG, USB3_EN);
/* HS/FS detected by HW */
mtu3_setbits(mbase, U3D_POWER_MANAGEMENT, HS_ENABLE);
} else if (mtu->max_speed == USB_SPEED_SUPER) {
mtu3_clrbits(mtu->ippc_base, SSUSB_U3_CTRL(0),
SSUSB_U3_PORT_SSP_SPEED);
}
dev_info(mtu->dev, "max_speed: %s\n",
usb_speed_string(mtu->max_speed));
}
static void mtu3_regs_init(struct mtu3 *mtu) static void mtu3_regs_init(struct mtu3 *mtu)
{ {
void __iomem *mbase = mtu->mac_base; void __iomem *mbase = mtu->mac_base;
...@@ -780,6 +792,8 @@ static void mtu3_check_params(struct mtu3 *mtu) ...@@ -780,6 +792,8 @@ static void mtu3_check_params(struct mtu3 *mtu)
if (!mtu->is_u3_ip && (mtu->max_speed > USB_SPEED_HIGH)) if (!mtu->is_u3_ip && (mtu->max_speed > USB_SPEED_HIGH))
mtu->max_speed = USB_SPEED_HIGH; mtu->max_speed = USB_SPEED_HIGH;
mtu->speed = mtu->max_speed;
dev_info(mtu->dev, "max_speed: %s\n", dev_info(mtu->dev, "max_speed: %s\n",
usb_speed_string(mtu->max_speed)); usb_speed_string(mtu->max_speed));
} }
......
...@@ -577,6 +577,19 @@ static int mtu3_gadget_stop(struct usb_gadget *g) ...@@ -577,6 +577,19 @@ static int mtu3_gadget_stop(struct usb_gadget *g)
return 0; return 0;
} }
static void
mtu3_gadget_set_speed(struct usb_gadget *g, enum usb_device_speed speed)
{
struct mtu3 *mtu = gadget_to_mtu3(g);
unsigned long flags;
dev_dbg(mtu->dev, "%s %s\n", __func__, usb_speed_string(speed));
spin_lock_irqsave(&mtu->lock, flags);
mtu3_set_speed(mtu, speed);
spin_unlock_irqrestore(&mtu->lock, flags);
}
static const struct usb_gadget_ops mtu3_gadget_ops = { static const struct usb_gadget_ops mtu3_gadget_ops = {
.get_frame = mtu3_gadget_get_frame, .get_frame = mtu3_gadget_get_frame,
.wakeup = mtu3_gadget_wakeup, .wakeup = mtu3_gadget_wakeup,
...@@ -584,6 +597,7 @@ static const struct usb_gadget_ops mtu3_gadget_ops = { ...@@ -584,6 +597,7 @@ static const struct usb_gadget_ops mtu3_gadget_ops = {
.pullup = mtu3_gadget_pullup, .pullup = mtu3_gadget_pullup,
.udc_start = mtu3_gadget_start, .udc_start = mtu3_gadget_start,
.udc_stop = mtu3_gadget_stop, .udc_stop = mtu3_gadget_stop,
.udc_set_speed = mtu3_gadget_set_speed,
}; };
static void mtu3_state_reset(struct mtu3 *mtu) static void mtu3_state_reset(struct mtu3 *mtu)
......
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