Faça o Linux carregar um driver específico para determinado dispositivo (Realtek NIC)

4

Este é um problema de driver em uma máquina vintage (AMD K8, Nvidia chipset) rodando Linux 3.11 (Mint distro, mas eu não acho que isso importe).

claudius ~ # uname -a
Linux claudius 3.11.0-12-generic #19-Ubuntu SMP Wed Oct 9 16:20:46 UTC 2013 x86_64 x86_64 x86_64 GNU/Linux
claudius ~ # lspci
00:00.0 Host bridge: NVIDIA Corporation nForce3 250Gb Host Bridge (rev a1)
00:01.0 ISA bridge: NVIDIA Corporation nForce3 250Gb LPC Bridge (rev a2)
00:01.1 SMBus: NVIDIA Corporation nForce 250Gb PCI System Management (rev a1)
00:02.0 USB controller: NVIDIA Corporation CK8S USB Controller (rev a1)
00:02.1 USB controller: NVIDIA Corporation CK8S USB Controller (rev a1)
00:02.2 USB controller: NVIDIA Corporation nForce3 EHCI USB 2.0 Controller (rev a2)
00:05.0 Bridge: NVIDIA Corporation CK8S Ethernet Controller (rev a2)
00:08.0 IDE interface: NVIDIA Corporation CK8S Parallel ATA Controller (v2.5) (rev a2)
00:0a.0 IDE interface: NVIDIA Corporation nForce3 Serial ATA Controller (rev a2)
00:0b.0 PCI bridge: NVIDIA Corporation nForce3 250Gb AGP Host to PCI Bridge (rev a2)
00:0e.0 PCI bridge: NVIDIA Corporation nForce3 250Gb PCI-to-PCI Bridge (rev a2)
00:18.0 Host bridge: Advanced Micro Devices, Inc. [AMD] K8 [Athlon64/Opteron] HyperTransport Technology Configuration
00:18.1 Host bridge: Advanced Micro Devices, Inc. [AMD] K8 [Athlon64/Opteron] Address Map
00:18.2 Host bridge: Advanced Micro Devices, Inc. [AMD] K8 [Athlon64/Opteron] DRAM Controller
00:18.3 Host bridge: Advanced Micro Devices, Inc. [AMD] K8 [Athlon64/Opteron] Miscellaneous Control
01:00.0 VGA compatible controller: NVIDIA Corporation NV34 [GeForce FX 5200] (rev a1)
02:05.0 Ethernet controller: Realtek Semiconductor Co., Ltd. RTL8169 PCI Gigabit Ethernet Controller (rev 10)

Eu quero que o kernel carregue um driver alternativo para minha placa ethernet PCI Realtek 8169, que você pode vê-lo no final da lista como 02:05.0 . Aqui está o detalhe:

claudius ~ # lspci -s 02:05.0 -vv -nn
02:05.0 Ethernet controller [0200]: Realtek Semiconductor Co., Ltd. RTL8169 PCI Gigabit Ethernet Controller [10ec:8169] (rev 10)
    Subsystem: Realtek Semiconductor Co., Ltd. RTL8169/8110 Family PCI Gigabit Ethernet NIC [10ec:8169]
    Control: I/O+ Mem+ BusMaster- SpecCycle- MemWINV+ VGASnoop- ParErr- Stepping- SERR+ FastB2B- DisINTx-
    Status: Cap+ 66MHz+ UDF- FastB2B+ ParErr- DEVSEL=medium >TAbort- <TAbort- <MAbort- >SERR- <PERR- INTx-
    Interrupt: pin A routed to IRQ 19
    Region 0: I/O ports at e800 [size=256]
    Region 1: Memory at febffc00 (32-bit, non-prefetchable) [size=256]
    Expansion ROM at febc0000 [disabled] [size=128K]
    Capabilities: [dc] Power Management version 2
        Flags: PMEClk- DSI- D1+ D2+ AuxCurrent=375mA PME(D0-,D1+,D2+,D3hot+,D3cold+)
        Status: D0 NoSoftRst- PME-Enable- DSel=0 DScale=0 PME-

A questão é que o r8169 simplesmente não funciona. Pode até fazer o Linux falhar. Eu coloquei na blacklist em /etc/modprobe.d/ .

A solução parece ser usando o driver r8168 . (Há também uma pergunta relacionada neste site .) Você pode baixar o código-fonte do Realtek, compilá-lo e instalá-lo, o que eu fiz. Se você modprobe r8168 , ele carrega bem, mas não se associa ao hardware, então o cartão não aparece em ifconfig .

Acho que isso tem a ver com o modalias (codifica as informações de hardware) e o mapeamento de dispositivo / driver em /lib/modules/*/modules.alias .

claudius ~ # cat /sys/devices/pci0000\:00/0000\:00\:0e.0/0000\:02\:05.0/modalias 
pci:v000010ECd00008169sv000010ECsd00008169bc02sc00i00

Este é o meu cartão. Aqui estão os mapeamentos do driver Realtek no sistema:

claudius ~ # grep r816 /lib/modules/3.11.0-12-generic/modules.alias
alias pci:v00001186d00004300sv00001186sd00004C00bc*sc*i* r8169
alias pci:v000010ECd00008169sv*sd*bc*sc*i* r8169
alias pci:v000010ECd00008167sv*sd*bc*sc*i* r8169
alias pci:v00001186d00004300sv00001186sd00004B10bc*sc*i* r8168
alias pci:v000010ECd00008168sv*sd*bc*sc*i* r8168

Eu tentei adicionar esta linha ao final do arquivo (sabendo que você não deveria fazer edições manuais):

alias pci:v000010ECd00008169sv000010ECsd00008169bc02sc00i00 r8168

Em seguida, descarregamos e recarregamos r8168 , mas isso não configura a placa e não há nada em dmesg .

Qual é a maneira correta de mapear minha placa de rede para o driver r8168 ? Eu tenho que reconstruir o driver para que ele possa lidar com o meu cartão? Ou há alguns dados de configuração que tenho que adicionar no lugar certo?

    
por Lumi 06.06.2014 / 17:02

2 respostas

6

Você pode forçar um dispositivo a usar um determinado dispositivo usando a vinculação. Se o dispositivo já pertence a um driver diferente, você primeiro tem que desvinculá-lo.

Se uma ID de fornecedor de PCI ( 10ec para Realtek) e combinação de ID de dispositivo não for reconhecida, você poderá torná-la reconhecida em tempo de execução com:

# echo 10ec 8169 > /sys/bus/pci/drivers/r8169/new_id

Exemplo:

# lspci -s 04: -nnvvv
04:00.0 Ethernet controller [0200]: Realtek Semiconductor Co., Ltd. RTL-8129 [10ec:8129] (rev 10)
        Subsystem: Coreco Inc RTL8111/8168 PCIe Gigabit Ethernet (misconfigured) [11ec:8129]
        Control: I/O+ Mem+ BusMaster- SpecCycle- MemWINV- VGASnoop- ParErr- Stepping- SERR- FastB2B- DisINTx-
        Status: Cap+ 66MHz+ UDF- FastB2B+ ParErr- DEVSEL=medium >TAbort- <TAbort- <MAbort- >SERR- <PERR- INTx-
        Interrupt: pin A routed to IRQ 3
        Region 0: I/O ports at c000 [size=256]
        Region 1: Memory at f7b40000 (32-bit, non-prefetchable) [size=256]
        [virtual] Expansion ROM at f7b00000 [disabled] [size=256K]
        Capabilities: [dc] Power Management version 1
                Flags: PMEClk- DSI- D1- D2- AuxCurrent=55mA PME(D0-,D1-,D2-,D3hot-,D3cold+)
                Status: D0 NoSoftRst- PME-Enable- DSel=0 DScale=0 PME-
        Kernel driver in use: pci-stub

# echo 0000:04:00.0 > /sys/bus/pci/drivers/pci-stub/unbind
# echo 0000:04:00.0 > /sys/bus/pci/drivers/r8169/bind
# lspci -s 04: -nnvvv
04:00.0 Ethernet controller [0200]: Realtek Semiconductor Co., Ltd. RTL-8129 [10ec:8129] (rev 10)
        Subsystem: Coreco Inc RTL8111/8168 PCIe Gigabit Ethernet (misconfigured) [11ec:8129]
        Control: I/O+ Mem+ BusMaster+ SpecCycle- MemWINV+ VGASnoop- ParErr- Stepping- SERR- FastB2B- DisINTx-
        Status: Cap+ 66MHz+ UDF- FastB2B+ ParErr- DEVSEL=medium >TAbort- <TAbort- <MAbort- >SERR- <PERR- INTx-
        Latency: 32 (8000ns min, 8000ns max), Cache Line Size: 64 bytes
        Interrupt: pin A routed to IRQ 19
        Region 0: I/O ports at c000 [size=256]
        Region 1: Memory at f7b40000 (32-bit, non-prefetchable) [size=256]
        [virtual] Expansion ROM at f7b00000 [disabled] [size=256K]
        Capabilities: [dc] Power Management version 1
                Flags: PMEClk- DSI- D1- D2- AuxCurrent=55mA PME(D0-,D1-,D2-,D3hot-,D3cold+)
                Status: D0 NoSoftRst- PME-Enable- DSel=0 DScale=0 PME-
        Kernel driver in use: r8169
    
por 08.07.2014 / 23:05
0

Eu consegui trabalhar. (Veja abaixo.)

Eu não teria perdido tanto tempo se não tivesse saltado para as conclusões . Eu tinha uma placa de rede PCI RTL8169 para a qual o Linux atribuiu o mecanismo de árvore% kernelr8169 . Este driver é disfuncional para o meu PCI RTL8169 e pode até travar o sistema. Encontrei então a página do Hetzner (veja o link em questão) e assumi o conselho para usar o Realtek r8168 aplicado à minha situação. (Não o fez.) Quando o motorista não se associou com o meu cartão, eu teimosamente pensei que deveria haver uma maneira de fazê-lo para que o cartão funcionasse. Essa era uma rua sem saída.

A solução é navegar novamente no site da Realtek . Eles também têm um driver chamado r8169 , assim como o driver disfuncional do kernel do Linux. Eu tentei isso meio ano atrás. Aparentemente, não teve sucesso (não se lembra exatamente); provavelmente devido a minha inépcia. De qualquer forma, a Realtek lançou um driver atualizado em março de 2014, e esse driver funciona! Obrigado Realtek!

O resultado feliz é o seguinte (usando smbclient ):

smb: \> get "Zwölf Uhr mittags.ts"
getting file \Zwölf Uhr mittags.ts of size 4703186788 as Zwölf Uhr mittags.ts
(47839,3 KiloBytes/sec) (average 47839,3 KiloBytes/sec)

smb: \> get "Die heiße Spur.ts"
getting file \Die heiße Spur.ts of size 6841251568 as Die heiße Spur.ts
(48516,1 KiloBytes/sec) (average 48516,1 KiloBytes/sec)

Eu não esperava obter uma taxa de transferência de 100 MB / s. O disco rígido WDC WD800AAJS-00B4A0 gira em torno de 95 MB / s ( dd if=/dev/zero of=test.out bs=8k count=200k ), mas primeiro há um barramento PCI (não PCIe!) compartilhado largura de banda de 133 MB / s que pode afetar minha configuração ( veja Gigabit Ethernet: Cara, onde está minha largura de banda? ) e, segundo, O CPU é um velho AMD Athlon 64 3200+ de 2003 com 2200 MHz, e é comum em minha experiência ver a taxa de transferência de Gigabit Ethernet significativamente reduzida por CPUs relativamente modestas, seja x86 / 64 ou ARM. Então, 50 MB / s é bom o suficiente.

    
por 07.06.2014 / 13:39