Commit e536e0d4 authored by Linus Torvalds's avatar Linus Torvalds

Merge tag 'usb-6.8-rc8' of git://git.kernel.org/pub/scm/linux/kernel/git/gregkh/usb

Pull USB / Thunderbolt fixes from Greg KH:
 "Here are some small remaining fixes for USB and Thunderbolt drivers.
  Included in here are fixes for:

   - thunderbold NULL dereference fix

   - typec driver fixes

   - xhci driver regression fix

   - usb-storage divide-by-0 fix

   - ncm gadget driver fix

  All of these have been in linux-next with no reported issues"

* tag 'usb-6.8-rc8' of git://git.kernel.org/pub/scm/linux/kernel/git/gregkh/usb:
  xhci: Fix failure to detect ring expansion need.
  usb: port: Don't try to peer unused USB ports based on location
  usb: gadget: ncm: Fix handling of zero block length packets
  usb: typec: altmodes/displayport: create sysfs nodes as driver's default device attribute group
  usb: typec: tpcm: Fix PORT_RESET behavior for self powered devices
  usb: typec: ucsi: fix UCSI on SM8550 & SM8650 Qualcomm devices
  USB: usb-storage: Prevent divide-by-0 error in isd200_ata_command
  thunderbolt: Fix NULL pointer dereference in tb_port_update_credits()
parents 49deb280 b234c70f
...@@ -1249,6 +1249,9 @@ int tb_port_update_credits(struct tb_port *port) ...@@ -1249,6 +1249,9 @@ int tb_port_update_credits(struct tb_port *port)
ret = tb_port_do_update_credits(port); ret = tb_port_do_update_credits(port);
if (ret) if (ret)
return ret; return ret;
if (!port->dual_link_port)
return 0;
return tb_port_do_update_credits(port->dual_link_port); return tb_port_do_update_credits(port->dual_link_port);
} }
......
...@@ -573,7 +573,7 @@ static int match_location(struct usb_device *peer_hdev, void *p) ...@@ -573,7 +573,7 @@ static int match_location(struct usb_device *peer_hdev, void *p)
struct usb_hub *peer_hub = usb_hub_to_struct_hub(peer_hdev); struct usb_hub *peer_hub = usb_hub_to_struct_hub(peer_hdev);
struct usb_device *hdev = to_usb_device(port_dev->dev.parent->parent); struct usb_device *hdev = to_usb_device(port_dev->dev.parent->parent);
if (!peer_hub) if (!peer_hub || port_dev->connect_type == USB_PORT_NOT_USED)
return 0; return 0;
hcd = bus_to_hcd(hdev->bus); hcd = bus_to_hcd(hdev->bus);
...@@ -584,7 +584,8 @@ static int match_location(struct usb_device *peer_hdev, void *p) ...@@ -584,7 +584,8 @@ static int match_location(struct usb_device *peer_hdev, void *p)
for (port1 = 1; port1 <= peer_hdev->maxchild; port1++) { for (port1 = 1; port1 <= peer_hdev->maxchild; port1++) {
peer = peer_hub->ports[port1 - 1]; peer = peer_hub->ports[port1 - 1];
if (peer && peer->location == port_dev->location) { if (peer && peer->connect_type != USB_PORT_NOT_USED &&
peer->location == port_dev->location) {
link_peers_report(port_dev, peer); link_peers_report(port_dev, peer);
return 1; /* done */ return 1; /* done */
} }
......
...@@ -1346,7 +1346,7 @@ static int ncm_unwrap_ntb(struct gether *port, ...@@ -1346,7 +1346,7 @@ static int ncm_unwrap_ntb(struct gether *port,
if (to_process == 1 && if (to_process == 1 &&
(*(unsigned char *)(ntb_ptr + block_len) == 0x00)) { (*(unsigned char *)(ntb_ptr + block_len) == 0x00)) {
to_process--; to_process--;
} else if (to_process > 0) { } else if ((to_process > 0) && (block_len != 0)) {
ntb_ptr = (unsigned char *)(ntb_ptr + block_len); ntb_ptr = (unsigned char *)(ntb_ptr + block_len);
goto parse_ntb; goto parse_ntb;
} }
......
...@@ -326,7 +326,13 @@ static unsigned int xhci_ring_expansion_needed(struct xhci_hcd *xhci, struct xhc ...@@ -326,7 +326,13 @@ static unsigned int xhci_ring_expansion_needed(struct xhci_hcd *xhci, struct xhc
/* how many trbs will be queued past the enqueue segment? */ /* how many trbs will be queued past the enqueue segment? */
trbs_past_seg = enq_used + num_trbs - (TRBS_PER_SEGMENT - 1); trbs_past_seg = enq_used + num_trbs - (TRBS_PER_SEGMENT - 1);
if (trbs_past_seg <= 0) /*
* Consider expanding the ring already if num_trbs fills the current
* segment (i.e. trbs_past_seg == 0), not only when num_trbs goes into
* the next segment. Avoids confusing full ring with special empty ring
* case below
*/
if (trbs_past_seg < 0)
return 0; return 0;
/* Empty ring special case, enqueue stuck on link trb while dequeue advanced */ /* Empty ring special case, enqueue stuck on link trb while dequeue advanced */
......
...@@ -1105,7 +1105,7 @@ static void isd200_dump_driveid(struct us_data *us, u16 *id) ...@@ -1105,7 +1105,7 @@ static void isd200_dump_driveid(struct us_data *us, u16 *id)
static int isd200_get_inquiry_data( struct us_data *us ) static int isd200_get_inquiry_data( struct us_data *us )
{ {
struct isd200_info *info = (struct isd200_info *)us->extra; struct isd200_info *info = (struct isd200_info *)us->extra;
int retStatus = ISD200_GOOD; int retStatus;
u16 *id = info->id; u16 *id = info->id;
usb_stor_dbg(us, "Entering isd200_get_inquiry_data\n"); usb_stor_dbg(us, "Entering isd200_get_inquiry_data\n");
...@@ -1137,6 +1137,13 @@ static int isd200_get_inquiry_data( struct us_data *us ) ...@@ -1137,6 +1137,13 @@ static int isd200_get_inquiry_data( struct us_data *us )
isd200_fix_driveid(id); isd200_fix_driveid(id);
isd200_dump_driveid(us, id); isd200_dump_driveid(us, id);
/* Prevent division by 0 in isd200_scsi_to_ata() */
if (id[ATA_ID_HEADS] == 0 || id[ATA_ID_SECTORS] == 0) {
usb_stor_dbg(us, " Invalid ATA Identify data\n");
retStatus = ISD200_ERROR;
goto Done;
}
memset(&info->InquiryData, 0, sizeof(info->InquiryData)); memset(&info->InquiryData, 0, sizeof(info->InquiryData));
/* Standard IDE interface only supports disks */ /* Standard IDE interface only supports disks */
...@@ -1202,6 +1209,7 @@ static int isd200_get_inquiry_data( struct us_data *us ) ...@@ -1202,6 +1209,7 @@ static int isd200_get_inquiry_data( struct us_data *us )
} }
} }
Done:
usb_stor_dbg(us, "Leaving isd200_get_inquiry_data %08X\n", retStatus); usb_stor_dbg(us, "Leaving isd200_get_inquiry_data %08X\n", retStatus);
return(retStatus); return(retStatus);
...@@ -1481,22 +1489,27 @@ static int isd200_init_info(struct us_data *us) ...@@ -1481,22 +1489,27 @@ static int isd200_init_info(struct us_data *us)
static int isd200_Initialization(struct us_data *us) static int isd200_Initialization(struct us_data *us)
{ {
int rc = 0;
usb_stor_dbg(us, "ISD200 Initialization...\n"); usb_stor_dbg(us, "ISD200 Initialization...\n");
/* Initialize ISD200 info struct */ /* Initialize ISD200 info struct */
if (isd200_init_info(us) == ISD200_ERROR) { if (isd200_init_info(us) < 0) {
usb_stor_dbg(us, "ERROR Initializing ISD200 Info struct\n"); usb_stor_dbg(us, "ERROR Initializing ISD200 Info struct\n");
rc = -ENOMEM;
} else { } else {
/* Get device specific data */ /* Get device specific data */
if (isd200_get_inquiry_data(us) != ISD200_GOOD) if (isd200_get_inquiry_data(us) != ISD200_GOOD) {
usb_stor_dbg(us, "ISD200 Initialization Failure\n"); usb_stor_dbg(us, "ISD200 Initialization Failure\n");
else rc = -EINVAL;
} else {
usb_stor_dbg(us, "ISD200 Initialization complete\n"); usb_stor_dbg(us, "ISD200 Initialization complete\n");
}
} }
return 0; return rc;
} }
......
...@@ -559,16 +559,21 @@ static ssize_t hpd_show(struct device *dev, struct device_attribute *attr, char ...@@ -559,16 +559,21 @@ static ssize_t hpd_show(struct device *dev, struct device_attribute *attr, char
} }
static DEVICE_ATTR_RO(hpd); static DEVICE_ATTR_RO(hpd);
static struct attribute *dp_altmode_attrs[] = { static struct attribute *displayport_attrs[] = {
&dev_attr_configuration.attr, &dev_attr_configuration.attr,
&dev_attr_pin_assignment.attr, &dev_attr_pin_assignment.attr,
&dev_attr_hpd.attr, &dev_attr_hpd.attr,
NULL NULL
}; };
static const struct attribute_group dp_altmode_group = { static const struct attribute_group displayport_group = {
.name = "displayport", .name = "displayport",
.attrs = dp_altmode_attrs, .attrs = displayport_attrs,
};
static const struct attribute_group *displayport_groups[] = {
&displayport_group,
NULL,
}; };
int dp_altmode_probe(struct typec_altmode *alt) int dp_altmode_probe(struct typec_altmode *alt)
...@@ -576,7 +581,6 @@ int dp_altmode_probe(struct typec_altmode *alt) ...@@ -576,7 +581,6 @@ int dp_altmode_probe(struct typec_altmode *alt)
const struct typec_altmode *port = typec_altmode_get_partner(alt); const struct typec_altmode *port = typec_altmode_get_partner(alt);
struct fwnode_handle *fwnode; struct fwnode_handle *fwnode;
struct dp_altmode *dp; struct dp_altmode *dp;
int ret;
/* FIXME: Port can only be DFP_U. */ /* FIXME: Port can only be DFP_U. */
...@@ -587,10 +591,6 @@ int dp_altmode_probe(struct typec_altmode *alt) ...@@ -587,10 +591,6 @@ int dp_altmode_probe(struct typec_altmode *alt)
DP_CAP_PIN_ASSIGN_DFP_D(alt->vdo))) DP_CAP_PIN_ASSIGN_DFP_D(alt->vdo)))
return -ENODEV; return -ENODEV;
ret = sysfs_create_group(&alt->dev.kobj, &dp_altmode_group);
if (ret)
return ret;
dp = devm_kzalloc(&alt->dev, sizeof(*dp), GFP_KERNEL); dp = devm_kzalloc(&alt->dev, sizeof(*dp), GFP_KERNEL);
if (!dp) if (!dp)
return -ENOMEM; return -ENOMEM;
...@@ -624,7 +624,6 @@ void dp_altmode_remove(struct typec_altmode *alt) ...@@ -624,7 +624,6 @@ void dp_altmode_remove(struct typec_altmode *alt)
{ {
struct dp_altmode *dp = typec_altmode_get_drvdata(alt); struct dp_altmode *dp = typec_altmode_get_drvdata(alt);
sysfs_remove_group(&alt->dev.kobj, &dp_altmode_group);
cancel_work_sync(&dp->work); cancel_work_sync(&dp->work);
if (dp->connector_fwnode) { if (dp->connector_fwnode) {
...@@ -649,6 +648,7 @@ static struct typec_altmode_driver dp_altmode_driver = { ...@@ -649,6 +648,7 @@ static struct typec_altmode_driver dp_altmode_driver = {
.driver = { .driver = {
.name = "typec_displayport", .name = "typec_displayport",
.owner = THIS_MODULE, .owner = THIS_MODULE,
.dev_groups = displayport_groups,
}, },
}; };
module_typec_altmode_driver(dp_altmode_driver); module_typec_altmode_driver(dp_altmode_driver);
......
...@@ -4873,8 +4873,11 @@ static void run_state_machine(struct tcpm_port *port) ...@@ -4873,8 +4873,11 @@ static void run_state_machine(struct tcpm_port *port)
break; break;
case PORT_RESET: case PORT_RESET:
tcpm_reset_port(port); tcpm_reset_port(port);
tcpm_set_cc(port, tcpm_default_state(port) == SNK_UNATTACHED ? if (port->self_powered)
TYPEC_CC_RD : tcpm_rp_cc(port)); tcpm_set_cc(port, TYPEC_CC_OPEN);
else
tcpm_set_cc(port, tcpm_default_state(port) == SNK_UNATTACHED ?
TYPEC_CC_RD : tcpm_rp_cc(port));
tcpm_set_state(port, PORT_RESET_WAIT_OFF, tcpm_set_state(port, PORT_RESET_WAIT_OFF,
PD_T_ERROR_RECOVERY); PD_T_ERROR_RECOVERY);
break; break;
......
...@@ -301,6 +301,7 @@ static const struct of_device_id pmic_glink_ucsi_of_quirks[] = { ...@@ -301,6 +301,7 @@ static const struct of_device_id pmic_glink_ucsi_of_quirks[] = {
{ .compatible = "qcom,sc8180x-pmic-glink", .data = (void *)UCSI_NO_PARTNER_PDOS, }, { .compatible = "qcom,sc8180x-pmic-glink", .data = (void *)UCSI_NO_PARTNER_PDOS, },
{ .compatible = "qcom,sc8280xp-pmic-glink", .data = (void *)UCSI_NO_PARTNER_PDOS, }, { .compatible = "qcom,sc8280xp-pmic-glink", .data = (void *)UCSI_NO_PARTNER_PDOS, },
{ .compatible = "qcom,sm8350-pmic-glink", .data = (void *)UCSI_NO_PARTNER_PDOS, }, { .compatible = "qcom,sm8350-pmic-glink", .data = (void *)UCSI_NO_PARTNER_PDOS, },
{ .compatible = "qcom,sm8550-pmic-glink", .data = (void *)UCSI_NO_PARTNER_PDOS, },
{} {}
}; };
......
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