Commit 47175e5f authored by Bjørn Mork's avatar Bjørn Mork Committed by David S. Miller

net: cdc_ncm: refactoring cdc_ncm_setup

Rewriting the "set max datagram" part of dc_ncm_setup to
separate the selection and validatation of the size from
the code which optionally informs the device of this
value.  This ensures that we use the correct value
regardless of device support for the get and set commands.

Removing some of the many indent levels while doing this
to make the code more readable.

Cc: Alexey Orishko <alexey.orishko@gmail.com>
Signed-off-by: default avatarBjørn Mork <bjorn@mork.no>
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parent 6dd13e83
...@@ -76,8 +76,7 @@ static u8 cdc_ncm_setup(struct usbnet *dev) ...@@ -76,8 +76,7 @@ static u8 cdc_ncm_setup(struct usbnet *dev)
int err; int err;
int eth_hlen; int eth_hlen;
u16 ntb_fmt_supported; u16 ntb_fmt_supported;
u32 min_dgram_size; __le16 max_datagram_size;
u32 min_hdr_size;
iface_no = ctx->control->cur_altsetting->desc.bInterfaceNumber; iface_no = ctx->control->cur_altsetting->desc.bInterfaceNumber;
...@@ -101,20 +100,29 @@ static u8 cdc_ncm_setup(struct usbnet *dev) ...@@ -101,20 +100,29 @@ static u8 cdc_ncm_setup(struct usbnet *dev)
ctx->tx_max_datagrams = le16_to_cpu(ncm_parm.wNtbOutMaxDatagrams); ctx->tx_max_datagrams = le16_to_cpu(ncm_parm.wNtbOutMaxDatagrams);
ntb_fmt_supported = le16_to_cpu(ncm_parm.bmNtbFormatsSupported); ntb_fmt_supported = le16_to_cpu(ncm_parm.bmNtbFormatsSupported);
eth_hlen = ETH_HLEN; /* there are some minor differences in NCM and MBIM defaults */
min_dgram_size = CDC_NCM_MIN_DATAGRAM_SIZE; if (cdc_ncm_comm_intf_is_mbim(ctx->control->cur_altsetting)) {
min_hdr_size = CDC_NCM_MIN_HDR_SIZE; if (!ctx->mbim_desc)
if (ctx->mbim_desc != NULL) { return -EINVAL;
flags = ctx->mbim_desc->bmNetworkCapabilities;
eth_hlen = 0; eth_hlen = 0;
min_dgram_size = CDC_MBIM_MIN_DATAGRAM_SIZE; flags = ctx->mbim_desc->bmNetworkCapabilities;
min_hdr_size = 0; ctx->max_datagram_size = le16_to_cpu(ctx->mbim_desc->wMaxSegmentSize);
} else if (ctx->func_desc != NULL) { if (ctx->max_datagram_size < CDC_MBIM_MIN_DATAGRAM_SIZE)
flags = ctx->func_desc->bmNetworkCapabilities; ctx->max_datagram_size = CDC_MBIM_MIN_DATAGRAM_SIZE;
} else { } else {
flags = 0; if (!ctx->func_desc)
return -EINVAL;
eth_hlen = ETH_HLEN;
flags = ctx->func_desc->bmNetworkCapabilities;
ctx->max_datagram_size = le16_to_cpu(ctx->ether_desc->wMaxSegmentSize);
if (ctx->max_datagram_size < CDC_NCM_MIN_DATAGRAM_SIZE)
ctx->max_datagram_size = CDC_NCM_MIN_DATAGRAM_SIZE;
} }
/* common absolute max for NCM and MBIM */
if (ctx->max_datagram_size > CDC_NCM_MAX_DATAGRAM_SIZE)
ctx->max_datagram_size = CDC_NCM_MAX_DATAGRAM_SIZE;
dev_dbg(&dev->intf->dev, dev_dbg(&dev->intf->dev,
"dwNtbInMaxSize=%u dwNtbOutMaxSize=%u wNdpOutPayloadRemainder=%u wNdpOutDivisor=%u wNdpOutAlignment=%u wNtbOutMaxDatagrams=%u flags=0x%x\n", "dwNtbInMaxSize=%u dwNtbOutMaxSize=%u wNdpOutPayloadRemainder=%u wNdpOutDivisor=%u wNdpOutAlignment=%u wNtbOutMaxDatagrams=%u flags=0x%x\n",
ctx->rx_max, ctx->tx_max, ctx->tx_remainder, ctx->tx_modulus, ctx->rx_max, ctx->tx_max, ctx->tx_remainder, ctx->tx_modulus,
...@@ -151,8 +159,7 @@ static u8 cdc_ncm_setup(struct usbnet *dev) ...@@ -151,8 +159,7 @@ static u8 cdc_ncm_setup(struct usbnet *dev)
} }
/* verify maximum size of transmitted NTB in bytes */ /* verify maximum size of transmitted NTB in bytes */
if ((ctx->tx_max < if ((ctx->tx_max < (CDC_NCM_MIN_HDR_SIZE + ctx->max_datagram_size)) ||
(min_hdr_size + min_dgram_size)) ||
(ctx->tx_max > CDC_NCM_NTB_MAX_SIZE_TX)) { (ctx->tx_max > CDC_NCM_NTB_MAX_SIZE_TX)) {
dev_dbg(&dev->intf->dev, "Using default maximum transmit length=%d\n", dev_dbg(&dev->intf->dev, "Using default maximum transmit length=%d\n",
CDC_NCM_NTB_MAX_SIZE_TX); CDC_NCM_NTB_MAX_SIZE_TX);
...@@ -229,60 +236,33 @@ static u8 cdc_ncm_setup(struct usbnet *dev) ...@@ -229,60 +236,33 @@ static u8 cdc_ncm_setup(struct usbnet *dev)
dev_dbg(&dev->intf->dev, "Setting NTB format to 16-bit failed\n"); dev_dbg(&dev->intf->dev, "Setting NTB format to 16-bit failed\n");
} }
ctx->max_datagram_size = min_dgram_size; /* inform the device about the selected Max Datagram Size */
if (!(flags & USB_CDC_NCM_NCAP_MAX_DATAGRAM_SIZE))
goto out;
/* set Max Datagram Size (MTU) */ /* read current mtu value from device */
if (flags & USB_CDC_NCM_NCAP_MAX_DATAGRAM_SIZE) { err = usbnet_read_cmd(dev, USB_CDC_GET_MAX_DATAGRAM_SIZE,
__le16 max_datagram_size; USB_TYPE_CLASS | USB_DIR_IN | USB_RECIP_INTERFACE,
u16 eth_max_sz; 0, iface_no, &max_datagram_size, 2);
if (ctx->ether_desc != NULL) if (err < 0) {
eth_max_sz = le16_to_cpu(ctx->ether_desc->wMaxSegmentSize); dev_dbg(&dev->intf->dev, "GET_MAX_DATAGRAM_SIZE failed\n");
else if (ctx->mbim_desc != NULL) goto out;
eth_max_sz = le16_to_cpu(ctx->mbim_desc->wMaxSegmentSize);
else
goto max_dgram_err;
err = usbnet_read_cmd(dev, USB_CDC_GET_MAX_DATAGRAM_SIZE,
USB_TYPE_CLASS | USB_DIR_IN
| USB_RECIP_INTERFACE,
0, iface_no, &max_datagram_size, 2);
if (err < 0) {
dev_dbg(&dev->intf->dev, "GET_MAX_DATAGRAM_SIZE failed, use size=%u\n",
min_dgram_size);
} else {
ctx->max_datagram_size =
le16_to_cpu(max_datagram_size);
/* Check Eth descriptor value */
if (ctx->max_datagram_size > eth_max_sz)
ctx->max_datagram_size = eth_max_sz;
if (ctx->max_datagram_size > CDC_NCM_MAX_DATAGRAM_SIZE)
ctx->max_datagram_size = CDC_NCM_MAX_DATAGRAM_SIZE;
if (ctx->max_datagram_size < min_dgram_size)
ctx->max_datagram_size = min_dgram_size;
/* if value changed, update device */
if (ctx->max_datagram_size !=
le16_to_cpu(max_datagram_size)) {
max_datagram_size = cpu_to_le16(ctx->max_datagram_size);
err = usbnet_write_cmd(dev,
USB_CDC_SET_MAX_DATAGRAM_SIZE,
USB_TYPE_CLASS | USB_DIR_OUT
| USB_RECIP_INTERFACE,
0,
iface_no, &max_datagram_size,
2);
if (err < 0)
dev_dbg(&dev->intf->dev, "SET_MAX_DGRAM_SIZE failed\n");
}
}
} }
max_dgram_err: if (le16_to_cpu(max_datagram_size) == ctx->max_datagram_size)
if (dev->net->mtu != (ctx->max_datagram_size - eth_hlen)) goto out;
dev->net->mtu = ctx->max_datagram_size - eth_hlen;
max_datagram_size = cpu_to_le16(ctx->max_datagram_size);
err = usbnet_write_cmd(dev, USB_CDC_SET_MAX_DATAGRAM_SIZE,
USB_TYPE_CLASS | USB_DIR_OUT | USB_RECIP_INTERFACE,
0, iface_no, &max_datagram_size, 2);
if (err < 0)
dev_dbg(&dev->intf->dev, "SET_MAX_DATAGRAM_SIZE failed\n");
out:
/* set MTU to max supported by the device if necessary */
if (dev->net->mtu > ctx->max_datagram_size - eth_hlen)
dev->net->mtu = ctx->max_datagram_size - eth_hlen;
return 0; return 0;
} }
......
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