Loading...
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 | // SPDX-License-Identifier: GPL-2.0 /* * Thunderbolt driver - quirks * * Copyright (c) 2020 Mario Limonciello <mario.limonciello@dell.com> */ #include "tb.h" static void quirk_force_power_link(struct tb_switch *sw) { sw->quirks |= QUIRK_FORCE_POWER_LINK_CONTROLLER; tb_sw_dbg(sw, "forcing power to link controller\n"); } static void quirk_dp_credit_allocation(struct tb_switch *sw) { if (sw->credit_allocation && sw->min_dp_main_credits == 56) { sw->min_dp_main_credits = 18; tb_sw_dbg(sw, "quirked DP main: %u\n", sw->min_dp_main_credits); } } static void quirk_clx_disable(struct tb_switch *sw) { sw->quirks |= QUIRK_NO_CLX; tb_sw_dbg(sw, "disabling CL states\n"); } static void quirk_usb3_maximum_bandwidth(struct tb_switch *sw) { struct tb_port *port; if (tb_switch_is_icm(sw)) return; tb_switch_for_each_port(sw, port) { if (!tb_port_is_usb3_down(port)) continue; port->max_bw = 16376; tb_port_dbg(port, "USB3 maximum bandwidth limited to %u Mb/s\n", port->max_bw); } } static void quirk_block_rpm_in_redrive(struct tb_switch *sw) { sw->quirks |= QUIRK_KEEP_POWER_IN_DP_REDRIVE; tb_sw_dbg(sw, "preventing runtime PM in DP redrive mode\n"); } struct tb_quirk { u16 hw_vendor_id; u16 hw_device_id; u16 vendor; u16 device; void (*hook)(struct tb_switch *sw); }; static const struct tb_quirk tb_quirks[] = { /* Dell WD19TB supports self-authentication on unplug */ { 0x0000, 0x0000, 0x00d4, 0xb070, quirk_force_power_link }, { 0x0000, 0x0000, 0x00d4, 0xb071, quirk_force_power_link }, /* * Intel Goshen Ridge NVM 27 and before report wrong number of * DP buffers. */ { 0x8087, 0x0b26, 0x0000, 0x0000, quirk_dp_credit_allocation }, /* * Limit the maximum USB3 bandwidth for the following Intel USB4 * host routers due to a hardware issue. */ { 0x8087, PCI_DEVICE_ID_INTEL_ADL_NHI0, 0x0000, 0x0000, quirk_usb3_maximum_bandwidth }, { 0x8087, PCI_DEVICE_ID_INTEL_ADL_NHI1, 0x0000, 0x0000, quirk_usb3_maximum_bandwidth }, { 0x8087, PCI_DEVICE_ID_INTEL_RPL_NHI0, 0x0000, 0x0000, quirk_usb3_maximum_bandwidth }, { 0x8087, PCI_DEVICE_ID_INTEL_RPL_NHI1, 0x0000, 0x0000, quirk_usb3_maximum_bandwidth }, { 0x8087, PCI_DEVICE_ID_INTEL_MTL_M_NHI0, 0x0000, 0x0000, quirk_usb3_maximum_bandwidth }, { 0x8087, PCI_DEVICE_ID_INTEL_MTL_P_NHI0, 0x0000, 0x0000, quirk_usb3_maximum_bandwidth }, { 0x8087, PCI_DEVICE_ID_INTEL_MTL_P_NHI1, 0x0000, 0x0000, quirk_usb3_maximum_bandwidth }, { 0x8087, PCI_DEVICE_ID_INTEL_BARLOW_RIDGE_HOST_80G_NHI, 0x0000, 0x0000, quirk_usb3_maximum_bandwidth }, { 0x8087, PCI_DEVICE_ID_INTEL_BARLOW_RIDGE_HOST_40G_NHI, 0x0000, 0x0000, quirk_usb3_maximum_bandwidth }, { 0x8087, PCI_DEVICE_ID_INTEL_BARLOW_RIDGE_HUB_80G_BRIDGE, 0x0000, 0x0000, quirk_usb3_maximum_bandwidth }, { 0x8087, PCI_DEVICE_ID_INTEL_BARLOW_RIDGE_HUB_40G_BRIDGE, 0x0000, 0x0000, quirk_usb3_maximum_bandwidth }, /* * Block Runtime PM in DP redrive mode for Intel Barlow Ridge host * controllers. */ { 0x8087, PCI_DEVICE_ID_INTEL_BARLOW_RIDGE_HOST_80G_NHI, 0x0000, 0x0000, quirk_block_rpm_in_redrive }, { 0x8087, PCI_DEVICE_ID_INTEL_BARLOW_RIDGE_HOST_40G_NHI, 0x0000, 0x0000, quirk_block_rpm_in_redrive }, /* * CLx is not supported on AMD USB4 Yellow Carp and Pink Sardine platforms. */ { 0x0438, 0x0208, 0x0000, 0x0000, quirk_clx_disable }, { 0x0438, 0x0209, 0x0000, 0x0000, quirk_clx_disable }, { 0x0438, 0x020a, 0x0000, 0x0000, quirk_clx_disable }, { 0x0438, 0x020b, 0x0000, 0x0000, quirk_clx_disable }, }; /** * tb_check_quirks() - Check for quirks to apply * @sw: Thunderbolt switch * * Apply any quirks for the Thunderbolt controller. */ void tb_check_quirks(struct tb_switch *sw) { int i; for (i = 0; i < ARRAY_SIZE(tb_quirks); i++) { const struct tb_quirk *q = &tb_quirks[i]; if (q->hw_vendor_id && q->hw_vendor_id != sw->config.vendor_id) continue; if (q->hw_device_id && q->hw_device_id != sw->config.device_id) continue; if (q->vendor && q->vendor != sw->vendor) continue; if (q->device && q->device != sw->device) continue; tb_sw_dbg(sw, "running %ps\n", q->hook); q->hook(sw); } } |