2020-06-23 11:14:29 -05:00
// 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 ;
2023-02-27 12:45:09 +02:00
tb_sw_dbg ( sw , " forcing power to link controller \n " ) ;
2020-06-23 11:14:29 -05:00
}
2021-03-23 19:05:23 +02:00
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 ) ;
}
}
2023-02-14 13:13:50 -06:00
static void quirk_clx_disable ( struct tb_switch * sw )
{
sw - > quirks | = QUIRK_NO_CLX ;
tb_sw_dbg ( sw , " disabling CL states \n " ) ;
}
2023-01-31 13:04:52 +02:00
static void quirk_usb3_maximum_bandwidth ( struct tb_switch * sw )
{
struct tb_port * port ;
2023-08-25 10:10:35 +03:00
if ( tb_switch_is_icm ( sw ) )
return ;
2023-01-31 13:04:52 +02:00
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 ) ;
}
}
2020-06-23 11:14:29 -05:00
struct tb_quirk {
2021-03-23 19:05:23 +02:00
u16 hw_vendor_id ;
u16 hw_device_id ;
2020-06-23 11:14:29 -05:00
u16 vendor ;
u16 device ;
void ( * hook ) ( struct tb_switch * sw ) ;
} ;
2020-07-02 16:07:50 +08:00
static const struct tb_quirk tb_quirks [ ] = {
2020-06-23 11:14:29 -05:00
/* Dell WD19TB supports self-authentication on unplug */
2021-03-23 19:05:23 +02:00
{ 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 } ,
2023-01-31 13:04:52 +02:00
/*
* 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 } ,
2023-05-17 10:45:53 +03:00
{ 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 } ,
2023-02-14 13:13:50 -06:00
/*
* 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 } ,
2020-06-23 11:14:29 -05:00
} ;
/**
* tb_check_quirks ( ) - Check for quirks to apply
* @ sw : Thunderbolt switch
*
2020-08-26 08:57:00 +03:00
* Apply any quirks for the Thunderbolt controller .
2020-06-23 11:14:29 -05:00
*/
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 ] ;
2021-03-23 19:05:23 +02:00
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 ;
2023-02-03 15:57:59 +02:00
tb_sw_dbg ( sw , " running %ps \n " , q - > hook ) ;
2021-03-23 19:05:23 +02:00
q - > hook ( sw ) ;
2020-06-23 11:14:29 -05:00
}
}