diff mbox series

Wrong piix4_smbus address / slow trackpoint on Thinkpad P14s gen 2 (AMD)

Message ID CAPoEpV0ZSidL6aMXvB6LN1uS-3CUHS4ggT8RwFgmkzzCiYJ-XQ@mail.gmail.com
State New
Headers show
Series Wrong piix4_smbus address / slow trackpoint on Thinkpad P14s gen 2 (AMD) | expand

Commit Message

Miroslav Bendík Dec. 11, 2021, 12:14 p.m. UTC
On Thinkpad P14s gen 2 with AMD (model 21A00003CK, BIOS R1MET43W - 1.13) SMBus
address detection don't work and trackpoint has slow rate (30 Hz) because can't
use SMBus. Email is divided to SMBus and trackpoint section, there are issues in
both modules.

I am using amd-pstate-dev-v5 branch of kernel, but problem is in all versions.


SMBUS
=====


After inserting i2c_piix4 i have address 0xff00 / 0xff20 in log, but
correct address is 0xb20:

[    0.349576] ACPI: \_SB_.PCI0.SMB1: Unsupported CMI method: _STA
[    0.349629] piix4_smbus 0000:00:14.0: Using IRQ for SMBus
[    0.349630] piix4_smbus 0000:00:14.0: SMBus Host Controller at
0xff00, revision 15
[    0.349632] piix4_smbus 0000:00:14.0: Using register 0x02 for SMBus
port selection
[    0.349652] i2c_dev: adapter [SMBus PIIX4 adapter port 0 at ff00]
registered as minor 2
[    0.349658] i2c i2c-2: adapter [SMBus PIIX4 adapter port 0 at ff00]
registered
[    0.349702] i2c_dev: adapter [SMBus PIIX4 adapter port 2 at ff00]
registered as minor 3
[    0.349705] i2c i2c-3: adapter [SMBus PIIX4 adapter port 2 at ff00]
registered
[    0.349720] piix4_smbus 0000:00:14.0: Auxiliary SMBus Host
Controller at 0xff20
[    0.349734] i2c_dev: adapter [SMBus PIIX4 adapter port 1 at ff20]
registered as minor 4
[    0.349737] i2c i2c-4: adapter [SMBus PIIX4 adapter port 1 at ff20]
registered

Here is relevant section of ACPI DSDT:

Scope (_SB.PCI0)
{
    Device (SMB1)
    {
        Name (_HID, "SMB0001")  // _HID: Hardware ID
        Name (_CRS, ResourceTemplate ()  // _CRS: Current Resource Settings
        {
            IO (Decode16,
                0x0B20,             // Range Minimum
                0x0B20,             // Range Maximum
                0x20,               // Alignment
                0x20,               // Length
                )
            IRQ (Level, ActiveLow, Shared, )
                {7}
        })
        Method (_STA, 0, NotSerialized)  // _STA: Status
        {
            Return (0x0F)
        }
    }
}

Here is SMBus section of lspci:

# lspci -vvv -b -x -nn -PP -P -s 00:14.0
00:14.0 SMBus [0c05]: Advanced Micro Devices, Inc. [AMD] FCH SMBus
Controller [1022:790b] (rev 51)
    Subsystem: Lenovo FCH SMBus Controller [17aa:5094]
    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-
    IOMMU group: 9
    Kernel driver in use: piix4_smbus
    Kernel modules: i2c_piix4, sp5100_tco
00: 22 10 0b 79 00 04 20 02 51 00 05 0c 00 00 80 00
10: 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00
20: 00 00 00 00 00 00 00 00 00 00 00 00 aa 17 94 50
30: 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00

Command i2cdetect -q -y 4 don't detect anything and log contains this messages:

[  813.156253] i2c i2c-9: Transaction (pre): CNT=ff, CMD=ff, ADD=ff,
DAT0=ff, DAT1=ff
[  813.156259] i2c i2c-9: SMBus busy (ff). Resetting...
[  813.156265] i2c i2c-9: Failed! (ff)
[  813.156272] i2c i2c-9: Probing failed, no device found

I tried to temporary fix problem on my machine. Relevant function is
piix4_setup_sb800 in drivers/i2c/buses/i2c-piix4.c. I have added debug messages
and changed address in this function.

     smba_en_lo = inb_p(SB800_PIIX4_SMB_IDX + 1);
     outb_p(smb_en + 1, SB800_PIIX4_SMB_IDX);
     smba_en_hi = inb_p(SB800_PIIX4_SMB_IDX + 1);
+    printk(KERN_INFO "piix4 smba_en_lo %u", smba_en_lo);
+    printk(KERN_INFO "piix4 smba_en_hi %u", smba_en_hi);
+    smba_en_hi = 0x0b;

...

+    printk(KERN_INFO "piix4 irq %d", PIIX4_dev->irq);
+    printk(KERN_INFO "piix4 i2ccfg %u", i2ccfg);

Then log looks like this:

[    0.809897] piix4 smba_en_lo 255
[    0.809899] piix4 smba_en_hi 255
[    0.811592] piix4 irq 0
[    0.811594] piix4 i2ccfg 4
[    0.811595] piix4_smbus 0000:00:14.0: SMBus Host Controller at
0xb00, revision 0
[    0.811598] piix4_smbus 0000:00:14.0: Using register 0x02 for SMBus
port selection
[    0.822338] piix4 smba_en_lo 255
[    0.822340] piix4 smba_en_hi 255
[    0.822345] piix4_smbus 0000:00:14.0: Auxiliary SMBus Host
Controller at 0xb20

And touchpad / trackpoint is detected:

# i2cdetect -q -y 4
     0  1  2  3  4  5  6  7  8  9  a  b  c  d  e  f
00:                         -- -- -- -- -- -- -- --
10: -- -- -- -- -- -- -- -- -- -- -- -- 1c -- -- --
20: -- -- -- -- -- -- -- -- -- -- -- -- 2c 2d 2e 2f
30: 30 31 32 33 34 35 36 37 38 39 3a 3b 3c 3d 3e 3f
40: 40 41 42 43 44 45 46 47 48 49 4a 4b 4c 4d 4e 4f
50: 50 51 52 53 54 55 56 57 58 59 5a 5b 5c 5d 5e 5f
60: 60 61 62 63 64 65 66 67 68 69 6a 6b 6c 6d 6e 6f
70: 70 71 72 73 74 75 76 77


Trackpoint (RMI4)
=================

SMBus is now working (probably).

Without any module parameters i have this output:

psmouse serio1: synaptics: Your touchpad (PNP: LEN2073 PNP0f13) says
it can support a different bus. If i2c-hid and hid-rmi are not used,
you might want to try setting psmouse.synaptics_intertouch to 1 and
report this to linux-input@vger.kernel.org.

When i try with synaptics_intertouch=1 i have this in dmesg:

psmouse serio1: synaptics: Trying to set up SMBus access
psmouse serio1: synaptics: SMbus companion is not ready yet

I have disabled I2C_FUNC_SMBUS_HOST_NOTIFY checks in psmouse-smbus.c,
synaptics.c and
rmi_smbus.c. Module i2c_piix4 dont export I2C_FUNC_SMBUS_HOST_NOTIFY. Exact
changes are:


Now i have working trackpoint. It's not reliable. Sometimes it's generating
keyboard events instead mouse events, sometimes starts working for cca 1 second
after first move and then stops (no IRQs are generated), sometimes dmesg
contains many "Failed to read irq" messages.

[    2.058579] psmouse serio1: synaptics: queried max coordinates: x
[..5678], y [..4694]
[    2.092853] psmouse serio1: synaptics: queried min coordinates: x
[1266..], y [1162..]
[    2.092855] psmouse serio1: synaptics: Trying to set up SMBus access
[    2.116391] rmi4_smbus 4-002c: registering SMbus-connected sensor
[    2.186473] rmi4_f01 rmi4-00.fn01: found RMI device, manufacturer:
Synaptics, product: TM3471-030, fw id: 3418235
[    2.226009] input: Synaptics TM3471-030 as /devices/rmi4-00/input/input54
[    2.239455] serio: RMI4 PS/2 pass-through port at rmi4-00.fn03
[    2.347012] psmouse serio2: trackpoint: Elan TrackPoint firmware:
0x12, buttons: 3/3
[    2.388030] input: TPPS/2 Elan TrackPoint as
/devices/rmi4-00/rmi4-00.fn03/serio2/input/input55
[ 1507.883750] rmi4_physical rmi4-00: Failed to read irqs, code=-5
[ 3263.258033] rmi4_physical rmi4-00: Failed to read irqs, code=-5
[ 3263.258587] rmi4_f03 rmi4-00.fn03: rmi_f03_pt_write: Failed to
write to F03 TX register (-5).
[ 3263.259036] rmi4_physical rmi4-00: Failed to read irqs, code=-6
[ 3263.259579] rmi4_physical rmi4-00: Failed to read irqs, code=-6
[ 3263.267027] rmi4_physical rmi4-00: Failed to read irqs, code=-5
[ 3263.267581] rmi4_f01 rmi4-00.fn01: Failed to write sleep mode: -5.
[ 3263.267582] rmi4_f01 rmi4-00.fn01: Suspend failed with code -5.


Changes needed to fix issues
============================

Address detection in piix4 is really broken for this machine. I don't know how
to fix this, without breaking detection on other machines.

I think, that piix4 should implement I2C_FUNC_SMBUS_HOST_NOTIFY like i2c-i801.
Or rmi4 can be reliably enabled without I2C_FUNC_SMBUS_HOST_NOTIFY, just using
irq?

If bus access is fixed, then should be LEN2073 whitelisted for SMBus access.


Files
=====

Kernel config - https://mireq.linuxos.sk/kernel/p14s_gen2_amd_kernel_config
ACPI tables - https://mireq.linuxos.sk/kernel/p14s_gen2_amd_acpi_tables.tar.xz
My really really dangerous changes -
https://mireq.linuxos.sk/kernel/p14s_gen2_amd_smbus.patch

Comments

Wolfram Sang Jan. 11, 2022, 12:35 p.m. UTC | #1
> Summary of synaptics_intertouch problem on Thinkpad with Ryzen 5850U:

A pity the threads are a bit mixed, but some more information about
previous attempts can be found in this thread:

https://lore.kernel.org/r/42c83ec8-bbac-85e2-9ab5-87e59a679f95@redhat.com

> Problem with wrong base address is fixed using this patch:
> https://lore.kernel.org/all/20210715221828.244536-1-Terry.Bowman@amd.com/

This patch is under discussion again.
Miroslav Bendík Feb. 12, 2022, 5:42 p.m. UTC | #2
Hello,
i think, that SMBus works now pretty good and last problem is screaming 
interrupt from synaptics (1000 irq/s). I need little help to solve this 
problem.

Little summary first:

On this thinkpad is synaptics trackpoint/touchpad connected to PIIX4. To 
enable RMI4 mode, SMBus driver should support host notify protocol. I 
have added support of host notify and replaced active waiting 
transaction with completer + interrupt. Driver is now pretty stable and 
works way better, than old implementation. For example i2c-detect shows 
real devices (previous transaction code showed all addresses from 0x1c 
as active). Patch on following link is still hack, has hardcoded IRQ and 
supports host notifications and interrupts only on auxiliary port. I can 
implement other ports later.

Patch: 
https://lore.kernel.org/all/c9b0b147-2907-ff41-4f13-464b3b891c50@wisdomtech.sk/
This patch includes PM register access using MMIO: 
https://lore.kernel.org/all/20210715221828.244536-1-Terry.Bowman@amd.com/

Now i can load psmouse synaptics_intertouch=1 and everything works 
great, but it uses 5% CPU and interrupt is called 1000/s. I have changed 
interrupt from rising edge to active low (it's PCIE, PCIE has active 
low) and i have many times checked if all interrupt bits are cleared in 
interrupt request. Yes, they are always cleared. Interrupts are 
generated only after first touch if i have compiled only F12. If i 
compile F03, then interrupts are generated immediately after load of 
psmouse. After unload, interrupts are not generated (i2c-piix4 still 
loaded).

On this machine I2C is accessible using GPIO 19(SCL), 20(SDA). Using 
kernel thread with RT priority on isolated core i have tried to record 
pin values on GPIO pins. Latency is too high to record all transferred 
data. Some state changes are lost (approximately 1/50 bits). Not too low 
to read reliably all data, but good enough to see what happens at bus 
level. Here is recorded file: 
https://mireq.linuxos.sk/kernel/thinkpad_p14s/i2c_scl_sda.xz.

Every byte is sample, first bit is SCL, second SDA. Sample rate is cca 
500 000 Hz, but often drops under 100 000 (lost bit).

On this screenshot is typical activity on bus: 
https://mireq.linuxos.sk/kernel/thinkpad_p14s/i2c_1.png (pulseview with 
imported raw file)

Zoom to two packet is here: 
https://mireq.linuxos.sk/kernel/thinkpad_p14s/i2c_2.png

First packet is SMBus host notify. Address 0x08 is SMBus host address 
and 0x58 is address of synaptics (0x2c << 1). Second packet is reading 
of interrupt status registers. Data 02 is length of interrupt status 
register (9 bits) and last 2 bytes are zero (idle, when moving cursor, 
then interrupt status register contains one bit set).

Zoomed out: https://mireq.linuxos.sk/kernel/thinkpad_p14s/i2c_3.png

Before transaction SMBus slave state machine is disabled and after 
transaction enabled. If notification is received when state machine is 
disabled, then device writes only address and don't get response. If 
driver runs with always enabled slave state machine, then output will 
contain only notify + read interrupt status pairs and no separate 
addresses, but with this mode bus collisions occur more often.

Here is dmesg output: https://pastebin.com/RdDYHJn0

Cursor is moved until 2862.8, then i have not touched trackpoint.

Idle device don't produce bus collisions. Moving cursor produces 
collisions, but sample rate is stable 100Hz, which is way better, than 
<40 Hz with PS/2 mode. I don't know how to solve collisions. Maybe they 
are related to not silenced host notifications.

If i were to be optimistic, then i would say that clearing interrupt 
vector will solve all problems. According old RMI4 documentation, 
reading from interrupt status register should clear interrupts (status 
register is cleared), but this don't prevent device form sending host 
notifications. Maybe exists new way to disable interrupts. I don't know, 
i have no access to current documentation.

My device has this signature:
Synaptics, product: TM3471-030, fw id: 3418235

Any help welcome.
Mario Limonciello Oct. 10, 2024, 4:25 p.m. UTC | #3
On 2/12/2022 11:42, Miroslav Bendík wrote:
> Hello,
> i think, that SMBus works now pretty good and last problem is screaming 
> interrupt from synaptics (1000 irq/s). I need little help to solve this 
> problem.
> 
> Little summary first:
> 
> On this thinkpad is synaptics trackpoint/touchpad connected to PIIX4. To 
> enable RMI4 mode, SMBus driver should support host notify protocol. I 
> have added support of host notify and replaced active waiting 
> transaction with completer + interrupt. Driver is now pretty stable and 
> works way better, than old implementation. For example i2c-detect shows 
> real devices (previous transaction code showed all addresses from 0x1c 
> as active). Patch on following link is still hack, has hardcoded IRQ and 
> supports host notifications and interrupts only on auxiliary port. I can 
> implement other ports later.
> 
> Patch: https://lore.kernel.org/all/c9b0b147-2907- 
> ff41-4f13-464b3b891c50@wisdomtech.sk/
> This patch includes PM register access using MMIO: https:// 
> lore.kernel.org/all/20210715221828.244536-1-Terry.Bowman@amd.com/
> 
> Now i can load psmouse synaptics_intertouch=1 and everything works 
> great, but it uses 5% CPU and interrupt is called 1000/s. I have changed 
> interrupt from rising edge to active low (it's PCIE, PCIE has active 
> low) and i have many times checked if all interrupt bits are cleared in 
> interrupt request. Yes, they are always cleared. Interrupts are 
> generated only after first touch if i have compiled only F12. If i 
> compile F03, then interrupts are generated immediately after load of 
> psmouse. After unload, interrupts are not generated (i2c-piix4 still 
> loaded).
> 
> On this machine I2C is accessible using GPIO 19(SCL), 20(SDA). Using 
> kernel thread with RT priority on isolated core i have tried to record 
> pin values on GPIO pins. Latency is too high to record all transferred 
> data. Some state changes are lost (approximately 1/50 bits). Not too low 
> to read reliably all data, but good enough to see what happens at bus 
> level. Here is recorded file: https://mireq.linuxos.sk/kernel/ 
> thinkpad_p14s/i2c_scl_sda.xz.
> 
> Every byte is sample, first bit is SCL, second SDA. Sample rate is cca 
> 500 000 Hz, but often drops under 100 000 (lost bit).
> 
> On this screenshot is typical activity on bus: https://mireq.linuxos.sk/ 
> kernel/thinkpad_p14s/i2c_1.png (pulseview with imported raw file)
> 
> Zoom to two packet is here: https://mireq.linuxos.sk/kernel/ 
> thinkpad_p14s/i2c_2.png
> 
> First packet is SMBus host notify. Address 0x08 is SMBus host address 
> and 0x58 is address of synaptics (0x2c << 1). Second packet is reading 
> of interrupt status registers. Data 02 is length of interrupt status 
> register (9 bits) and last 2 bytes are zero (idle, when moving cursor, 
> then interrupt status register contains one bit set).
> 
> Zoomed out: https://mireq.linuxos.sk/kernel/thinkpad_p14s/i2c_3.png
> 
> Before transaction SMBus slave state machine is disabled and after 
> transaction enabled. If notification is received when state machine is 
> disabled, then device writes only address and don't get response. If 
> driver runs with always enabled slave state machine, then output will 
> contain only notify + read interrupt status pairs and no separate 
> addresses, but with this mode bus collisions occur more often.
> 
> Here is dmesg output: https://pastebin.com/RdDYHJn0
> 
> Cursor is moved until 2862.8, then i have not touched trackpoint.
> 
> Idle device don't produce bus collisions. Moving cursor produces 
> collisions, but sample rate is stable 100Hz, which is way better, than 
> <40 Hz with PS/2 mode. I don't know how to solve collisions. Maybe they 
> are related to not silenced host notifications.
> 
> If i were to be optimistic, then i would say that clearing interrupt 
> vector will solve all problems. According old RMI4 documentation, 
> reading from interrupt status register should clear interrupts (status 
> register is cleared), but this don't prevent device form sending host 
> notifications. Maybe exists new way to disable interrupts. I don't know, 
> i have no access to current documentation.
> 
> My device has this signature:
> Synaptics, product: TM3471-030, fw id: 3418235
> 
> Any help welcome.
> 

Sorry to bump such an old thread, but AFAIK you never came up with a 
good solution here.  I did want to point out that there was a very 
recent submission by Shyam (CC'ed) [1] that adds an ASF driver (which is 
an extension to PIIX4).  By default it's going to bind to an ACPI ID 
that isn't present on your system (present on newer systems only) but 
the hardware for ASF /should/ be present even on yours.

So I was going to suggest if you still are interested in this to play 
with that series and come up with a way to force using ASF (perhaps by a 
DMI match for your system) and see how that goes.

[1] 
https://lore.kernel.org/all/20240923080401.2167310-1-Shyam-sundar.S-k@amd.com/
Miroslav Bendík Oct. 13, 2024, 5:20 p.m. UTC | #4
On 10. 10. 2024 18:25, Mario Limonciello wrote:
> On 2/12/2022 11:42, Miroslav Bendík wrote:
>> Hello,
>> i think, that SMBus works now pretty good and last problem is 
>> screaming interrupt from synaptics (1000 irq/s). I need little help 
>> to solve this problem.
>>
>> Little summary first:
>>
>> On this thinkpad is synaptics trackpoint/touchpad connected to PIIX4. 
>> To enable RMI4 mode, SMBus driver should support host notify 
>> protocol. I have added support of host notify and replaced active 
>> waiting transaction with completer + interrupt. Driver is now pretty 
>> stable and works way better, than old implementation. For example 
>> i2c-detect shows real devices (previous transaction code showed all 
>> addresses from 0x1c as active). Patch on following link is still 
>> hack, has hardcoded IRQ and supports host notifications and 
>> interrupts only on auxiliary port. I can implement other ports later.
>>
>> Patch: https://lore.kernel.org/all/c9b0b147-2907- 
>> ff41-4f13-464b3b891c50@wisdomtech.sk/
>> This patch includes PM register access using MMIO: https:// 
>> lore.kernel.org/all/20210715221828.244536-1-Terry.Bowman@amd.com/
>>
>> Now i can load psmouse synaptics_intertouch=1 and everything works 
>> great, but it uses 5% CPU and interrupt is called 1000/s. I have 
>> changed interrupt from rising edge to active low (it's PCIE, PCIE has 
>> active low) and i have many times checked if all interrupt bits are 
>> cleared in interrupt request. Yes, they are always cleared. 
>> Interrupts are generated only after first touch if i have compiled 
>> only F12. If i compile F03, then interrupts are generated immediately 
>> after load of psmouse. After unload, interrupts are not generated 
>> (i2c-piix4 still loaded).
>>
>> On this machine I2C is accessible using GPIO 19(SCL), 20(SDA). Using 
>> kernel thread with RT priority on isolated core i have tried to 
>> record pin values on GPIO pins. Latency is too high to record all 
>> transferred data. Some state changes are lost (approximately 1/50 
>> bits). Not too low to read reliably all data, but good enough to see 
>> what happens at bus level. Here is recorded file: 
>> https://mireq.linuxos.sk/kernel/ thinkpad_p14s/i2c_scl_sda.xz.
>>
>> Every byte is sample, first bit is SCL, second SDA. Sample rate is 
>> cca 500 000 Hz, but often drops under 100 000 (lost bit).
>>
>> On this screenshot is typical activity on bus: 
>> https://mireq.linuxos.sk/ kernel/thinkpad_p14s/i2c_1.png (pulseview 
>> with imported raw file)
>>
>> Zoom to two packet is here: https://mireq.linuxos.sk/kernel/ 
>> thinkpad_p14s/i2c_2.png
>>
>> First packet is SMBus host notify. Address 0x08 is SMBus host address 
>> and 0x58 is address of synaptics (0x2c << 1). Second packet is 
>> reading of interrupt status registers. Data 02 is length of interrupt 
>> status register (9 bits) and last 2 bytes are zero (idle, when moving 
>> cursor, then interrupt status register contains one bit set).
>>
>> Zoomed out: https://mireq.linuxos.sk/kernel/thinkpad_p14s/i2c_3.png
>>
>> Before transaction SMBus slave state machine is disabled and after 
>> transaction enabled. If notification is received when state machine 
>> is disabled, then device writes only address and don't get response. 
>> If driver runs with always enabled slave state machine, then output 
>> will contain only notify + read interrupt status pairs and no 
>> separate addresses, but with this mode bus collisions occur more often.
>>
>> Here is dmesg output: https://pastebin.com/RdDYHJn0
>>
>> Cursor is moved until 2862.8, then i have not touched trackpoint.
>>
>> Idle device don't produce bus collisions. Moving cursor produces 
>> collisions, but sample rate is stable 100Hz, which is way better, 
>> than <40 Hz with PS/2 mode. I don't know how to solve collisions. 
>> Maybe they are related to not silenced host notifications.
>>
>> If i were to be optimistic, then i would say that clearing interrupt 
>> vector will solve all problems. According old RMI4 documentation, 
>> reading from interrupt status register should clear interrupts 
>> (status register is cleared), but this don't prevent device form 
>> sending host notifications. Maybe exists new way to disable 
>> interrupts. I don't know, i have no access to current documentation.
>>
>> My device has this signature:
>> Synaptics, product: TM3471-030, fw id: 3418235
>>
>> Any help welcome.
>>
>
> Sorry to bump such an old thread, but AFAIK you never came up with a 
> good solution here.  I did want to point out that there was a very 
> recent submission by Shyam (CC'ed) [1] that adds an ASF driver (which 
> is an extension to PIIX4).  By default it's going to bind to an ACPI 
> ID that isn't present on your system (present on newer systems only) 
> but the hardware for ASF /should/ be present even on yours.
>
> So I was going to suggest if you still are interested in this to play 
> with that series and come up with a way to force using ASF (perhaps by 
> a DMI match for your system) and see how that goes.
>
> [1] 
> https://lore.kernel.org/all/20240923080401.2167310-1-Shyam-sundar.S-k@amd.com/
Hello.

Thanks for the update. It looks good as a separate driver. I had 
intended to split this driver, replace polling with interrupts, and 
convert all I/O calls to MMIO, similar to how a Windows driver operates. 
I paused the work because I needed documentation and fixes from other 
companies, and I resolved my issue using a different approach:

- I have not received a response from Synaptics.
- I have not received a response from Lenovo.
- I have fixed the original issue - 
https://patchwork.kernel.org/project/linux-input/patch/71d9dc66-9576-c26f-c9d9-129217f50255@gmail.com/#24848525
- Too many subsystems are affected, some of which are currently hardly 
fixable.

The biggest issue is interrupt support, which cannot be resolved with 
quirks alone.

My device has this DSDT ACPI entry:

  Name (_HID, "SMB0001")  // _HID: Hardware ID
  Name (_CRS, ResourceTemplate ()  // _CRS: Current Resource Settings
  {
      IO (Decode16,
          0x0B20,             // Range Minimum
          0x0B20,             // Range Maximum
          0x20,               // Alignment
          0x20,               // Length
          )
      IRQ (Level, ActiveLow, Shared, )
          {7}
  })

This entry defines the IRQ number, trigger, and polarity. However, the 
kernel ignores this entry and only uses the "Interrupt Source Override" 
from the MADT table. Here are some potential solutions:

- Scan ACPI for interrupts and include them in overrides
   - Scanning might break or fix many drivers. This is a significant 
change that could affect numerous buggy BIOS ACPI implementations.

- Change the interrupt trigger and polarity on the fly in the ASF driver
   - The interrupt trigger cannot be changed with irq_set_type because 
it lacks an ioapic_ir_chip.irq_set_type implementation

- Implement a new quirk or hook to modify interrupts
   - This would involve extending the quirk system to allow changes to 
polarity/trigger before APIC initialization.

- Add the missing MADT table
   - Lenovo ignores this requirement.

Fixing this bug might break trackpoint support on ThinkPads, as the 
Synaptics firmware does not behave as described in the documentation.

Given the high potential for system instability with minimal gain, my 
requested feature may not be worth pursuing.

I can maybe help with testing this new ASF driver on older hardware 
without specific ACPI ID.
Hans de Goede Oct. 13, 2024, 5:53 p.m. UTC | #5
Hi,

On 13-Oct-24 7:20 PM, Miroslav Bendík wrote:
> On 10. 10. 2024 18:25, Mario Limonciello wrote:
>> On 2/12/2022 11:42, Miroslav Bendík wrote:
>>> Hello,
>>> i think, that SMBus works now pretty good and last problem is screaming interrupt from synaptics (1000 irq/s). I need little help to solve this problem.
>>>
>>> Little summary first:
>>>
>>> On this thinkpad is synaptics trackpoint/touchpad connected to PIIX4. To enable RMI4 mode, SMBus driver should support host notify protocol. I have added support of host notify and replaced active waiting transaction with completer + interrupt. Driver is now pretty stable and works way better, than old implementation. For example i2c-detect shows real devices (previous transaction code showed all addresses from 0x1c as active). Patch on following link is still hack, has hardcoded IRQ and supports host notifications and interrupts only on auxiliary port. I can implement other ports later.
>>>
>>> Patch: https://lore.kernel.org/all/c9b0b147-2907- ff41-4f13-464b3b891c50@wisdomtech.sk/
>>> This patch includes PM register access using MMIO: https:// lore.kernel.org/all/20210715221828.244536-1-Terry.Bowman@amd.com/
>>>
>>> Now i can load psmouse synaptics_intertouch=1 and everything works great, but it uses 5% CPU and interrupt is called 1000/s. I have changed interrupt from rising edge to active low (it's PCIE, PCIE has active low) and i have many times checked if all interrupt bits are cleared in interrupt request. Yes, they are always cleared. Interrupts are generated only after first touch if i have compiled only F12. If i compile F03, then interrupts are generated immediately after load of psmouse. After unload, interrupts are not generated (i2c-piix4 still loaded).
>>>
>>> On this machine I2C is accessible using GPIO 19(SCL), 20(SDA). Using kernel thread with RT priority on isolated core i have tried to record pin values on GPIO pins. Latency is too high to record all transferred data. Some state changes are lost (approximately 1/50 bits). Not too low to read reliably all data, but good enough to see what happens at bus level. Here is recorded file: https://mireq.linuxos.sk/kernel/ thinkpad_p14s/i2c_scl_sda.xz.
>>>
>>> Every byte is sample, first bit is SCL, second SDA. Sample rate is cca 500 000 Hz, but often drops under 100 000 (lost bit).
>>>
>>> On this screenshot is typical activity on bus: https://mireq.linuxos.sk/ kernel/thinkpad_p14s/i2c_1.png (pulseview with imported raw file)
>>>
>>> Zoom to two packet is here: https://mireq.linuxos.sk/kernel/ thinkpad_p14s/i2c_2.png
>>>
>>> First packet is SMBus host notify. Address 0x08 is SMBus host address and 0x58 is address of synaptics (0x2c << 1). Second packet is reading of interrupt status registers. Data 02 is length of interrupt status register (9 bits) and last 2 bytes are zero (idle, when moving cursor, then interrupt status register contains one bit set).
>>>
>>> Zoomed out: https://mireq.linuxos.sk/kernel/thinkpad_p14s/i2c_3.png
>>>
>>> Before transaction SMBus slave state machine is disabled and after transaction enabled. If notification is received when state machine is disabled, then device writes only address and don't get response. If driver runs with always enabled slave state machine, then output will contain only notify + read interrupt status pairs and no separate addresses, but with this mode bus collisions occur more often.
>>>
>>> Here is dmesg output: https://pastebin.com/RdDYHJn0
>>>
>>> Cursor is moved until 2862.8, then i have not touched trackpoint.
>>>
>>> Idle device don't produce bus collisions. Moving cursor produces collisions, but sample rate is stable 100Hz, which is way better, than <40 Hz with PS/2 mode. I don't know how to solve collisions. Maybe they are related to not silenced host notifications.
>>>
>>> If i were to be optimistic, then i would say that clearing interrupt vector will solve all problems. According old RMI4 documentation, reading from interrupt status register should clear interrupts (status register is cleared), but this don't prevent device form sending host notifications. Maybe exists new way to disable interrupts. I don't know, i have no access to current documentation.
>>>
>>> My device has this signature:
>>> Synaptics, product: TM3471-030, fw id: 3418235
>>>
>>> Any help welcome.
>>>
>>
>> Sorry to bump such an old thread, but AFAIK you never came up with a good solution here.  I did want to point out that there was a very recent submission by Shyam (CC'ed) [1] that adds an ASF driver (which is an extension to PIIX4).  By default it's going to bind to an ACPI ID that isn't present on your system (present on newer systems only) but the hardware for ASF /should/ be present even on yours.
>>
>> So I was going to suggest if you still are interested in this to play with that series and come up with a way to force using ASF (perhaps by a DMI match for your system) and see how that goes.
>>
>> [1] https://lore.kernel.org/all/20240923080401.2167310-1-Shyam-sundar.S-k@amd.com/
> Hello.
> 
> Thanks for the update. It looks good as a separate driver. I had intended to split this driver, replace polling with interrupts, and convert all I/O calls to MMIO, similar to how a Windows driver operates. I paused the work because I needed documentation and fixes from other companies, and I resolved my issue using a different approach:
> 
> - I have not received a response from Synaptics.
> - I have not received a response from Lenovo.
> - I have fixed the original issue - https://patchwork.kernel.org/project/linux-input/patch/71d9dc66-9576-c26f-c9d9-129217f50255@gmail.com/#24848525
> - Too many subsystems are affected, some of which are currently hardly fixable.
> 
> The biggest issue is interrupt support, which cannot be resolved with quirks alone.
> 
> My device has this DSDT ACPI entry:
> 
>  Name (_HID, "SMB0001")  // _HID: Hardware ID
>  Name (_CRS, ResourceTemplate ()  // _CRS: Current Resource Settings
>  {
>      IO (Decode16,
>          0x0B20,             // Range Minimum
>          0x0B20,             // Range Maximum
>          0x20,               // Alignment
>          0x20,               // Length
>          )
>      IRQ (Level, ActiveLow, Shared, )
>          {7}
>  })
> 
> This entry defines the IRQ number, trigger, and polarity. However, the kernel ignores this entry and only uses the "Interrupt Source Override" from the MADT table.

Note that we already have a quirk table for this because this hits more
interrupts in the legacy ISA interrupt range, see:

https://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git/tree/drivers/acpi/resource.c#n659

note that the MADT table is already alway skipped on AMD systems,
but currently only for IRQs 1/12 which are the PS/2 kbd + mouse
IRQs.

Regards,

Hans
Miroslav Bendík Oct. 14, 2024, 4:43 p.m. UTC | #6
On 14. 10. 2024 18:01, Shyam Sundar S K wrote:
>
> On 10/13/2024 23:23, Hans de Goede wrote:
>> Hi,
>>
>> On 13-Oct-24 7:20 PM, Miroslav Bendík wrote:
>>> On 10. 10. 2024 18:25, Mario Limonciello wrote:
>>>> On 2/12/2022 11:42, Miroslav Bendík wrote:
>>>>> Hello,
>>>>> i think, that SMBus works now pretty good and last problem is screaming interrupt from synaptics (1000 irq/s). I need little help to solve this problem.
>>>>>
>>>>> Little summary first:
>>>>>
>>>>> On this thinkpad is synaptics trackpoint/touchpad connected to PIIX4. To enable RMI4 mode, SMBus driver should support host notify protocol. I have added support of host notify and replaced active waiting transaction with completer + interrupt. Driver is now pretty stable and works way better, than old implementation. For example i2c-detect shows real devices (previous transaction code showed all addresses from 0x1c as active). Patch on following link is still hack, has hardcoded IRQ and supports host notifications and interrupts only on auxiliary port. I can implement other ports later.
>>>>>
>>>>> Patch: https://lore.kernel.org/all/c9b0b147-2907- ff41-4f13-464b3b891c50@wisdomtech.sk/
>>>>> This patch includes PM register access using MMIO: https:// lore.kernel.org/all/20210715221828.244536-1-Terry.Bowman@amd.com/
>>>>>
>>>>> Now i can load psmouse synaptics_intertouch=1 and everything works great, but it uses 5% CPU and interrupt is called 1000/s. I have changed interrupt from rising edge to active low (it's PCIE, PCIE has active low) and i have many times checked if all interrupt bits are cleared in interrupt request. Yes, they are always cleared. Interrupts are generated only after first touch if i have compiled only F12. If i compile F03, then interrupts are generated immediately after load of psmouse. After unload, interrupts are not generated (i2c-piix4 still loaded).
>>>>>
>>>>> On this machine I2C is accessible using GPIO 19(SCL), 20(SDA). Using kernel thread with RT priority on isolated core i have tried to record pin values on GPIO pins. Latency is too high to record all transferred data. Some state changes are lost (approximately 1/50 bits). Not too low to read reliably all data, but good enough to see what happens at bus level. Here is recorded file: https://mireq.linuxos.sk/kernel/ thinkpad_p14s/i2c_scl_sda.xz.
>>>>>
>>>>> Every byte is sample, first bit is SCL, second SDA. Sample rate is cca 500 000 Hz, but often drops under 100 000 (lost bit).
>>>>>
>>>>> On this screenshot is typical activity on bus: https://mireq.linuxos.sk/ kernel/thinkpad_p14s/i2c_1.png (pulseview with imported raw file)
>>>>>
>>>>> Zoom to two packet is here: https://mireq.linuxos.sk/kernel/ thinkpad_p14s/i2c_2.png
>>>>>
>>>>> First packet is SMBus host notify. Address 0x08 is SMBus host address and 0x58 is address of synaptics (0x2c << 1). Second packet is reading of interrupt status registers. Data 02 is length of interrupt status register (9 bits) and last 2 bytes are zero (idle, when moving cursor, then interrupt status register contains one bit set).
>>>>>
>>>>> Zoomed out: https://mireq.linuxos.sk/kernel/thinkpad_p14s/i2c_3.png
>>>>>
>>>>> Before transaction SMBus slave state machine is disabled and after transaction enabled. If notification is received when state machine is disabled, then device writes only address and don't get response. If driver runs with always enabled slave state machine, then output will contain only notify + read interrupt status pairs and no separate addresses, but with this mode bus collisions occur more often.
>>>>>
>>>>> Here is dmesg output: https://pastebin.com/RdDYHJn0
>>>>>
>>>>> Cursor is moved until 2862.8, then i have not touched trackpoint.
>>>>>
>>>>> Idle device don't produce bus collisions. Moving cursor produces collisions, but sample rate is stable 100Hz, which is way better, than <40 Hz with PS/2 mode. I don't know how to solve collisions. Maybe they are related to not silenced host notifications.
>>>>>
>>>>> If i were to be optimistic, then i would say that clearing interrupt vector will solve all problems. According old RMI4 documentation, reading from interrupt status register should clear interrupts (status register is cleared), but this don't prevent device form sending host notifications. Maybe exists new way to disable interrupts. I don't know, i have no access to current documentation.
>>>>>
>>>>> My device has this signature:
>>>>> Synaptics, product: TM3471-030, fw id: 3418235
>>>>>
>>>>> Any help welcome.
>>>>>
>>>> Sorry to bump such an old thread, but AFAIK you never came up with a good solution here.  I did want to point out that there was a very recent submission by Shyam (CC'ed) [1] that adds an ASF driver (which is an extension to PIIX4).  By default it's going to bind to an ACPI ID that isn't present on your system (present on newer systems only) but the hardware for ASF /should/ be present even on yours.
>>>>
>>>> So I was going to suggest if you still are interested in this to play with that series and come up with a way to force using ASF (perhaps by a DMI match for your system) and see how that goes.
>>>>
>>>> [1] https://lore.kernel.org/all/20240923080401.2167310-1-Shyam-sundar.S-k@amd.com/
>>> Hello.
>>>
>>> Thanks for the update. It looks good as a separate driver. I had intended to split this driver, replace polling with interrupts, and convert all I/O calls to MMIO, similar to how a Windows driver operates. I paused the work because I needed documentation and fixes from other companies, and I resolved my issue using a different approach:
>>>
>>> - I have not received a response from Synaptics.
>>> - I have not received a response from Lenovo.
>>> - I have fixed the original issue - https://patchwork.kernel.org/project/linux-input/patch/71d9dc66-9576-c26f-c9d9-129217f50255@gmail.com/#24848525
>>> - Too many subsystems are affected, some of which are currently hardly fixable.
>>>
>>> The biggest issue is interrupt support, which cannot be resolved with quirks alone.
> Do you mean the interrupt support from SMBus controller which happens
> via the piix4 driver?
>
> Note that SMBus controller do not support interrupt and the same has
> been documented in the datasheet:
>
> D14F0x03C [Interrupt Line] (FCH::SMBUSPCI::IntLine)
>
> 15:8 InterruptPin: Interrupt Pin. Read. Reset: Fixed,00h.
>
> ValidValues:
>
>
> Value 			Description
> -------------		-------------
> 00h 			This module does not generate interrupts.
> FFh-01h			Interrupt pin.
>
> Thanks,
> Shyam

Hello,

i am using ASF documentation (48751 Rev 3.03 - February 19, 2015 BKDG 
for AMD Family 16h Models 00h-0Fh Processors - 
https://www.amd.com/en/search/documentation/hub.html#q=BKDG%2048751).

There is a ASFx0A ASFStatus with SlaveIntr bit. Pointing device on my 
machine triggers interrupt 7 and it can be cleared using ASF when i move 
cursor using touchpad or trackpoint. It's possible to implement and 
enable I2C_CLIENT_HOST_NOTIFY flag on this hardware.

Another interrupt bit Intr from ASFx00 HostStatus is 1 after transfer 
(interrupt 7 is triggered). Transfer can be implemented without timers 
and active waiting on HostBusy status (there is a loop in 
piix4_transaction).

With regards
Miroslav Bendík

>
>>> My device has this DSDT ACPI entry:
>>>
>>>   Name (_HID, "SMB0001")  // _HID: Hardware ID
>>>   Name (_CRS, ResourceTemplate ()  // _CRS: Current Resource Settings
>>>   {
>>>       IO (Decode16,
>>>           0x0B20,             // Range Minimum
>>>           0x0B20,             // Range Maximum
>>>           0x20,               // Alignment
>>>           0x20,               // Length
>>>           )
>>>       IRQ (Level, ActiveLow, Shared, )
>>>           {7}
>>>   })
>>>
>>> This entry defines the IRQ number, trigger, and polarity. However, the kernel ignores this entry and only uses the "Interrupt Source Override" from the MADT table.
>> Note that we already have a quirk table for this because this hits more
>> interrupts in the legacy ISA interrupt range, see:
>>
>> https://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git/tree/drivers/acpi/resource.c#n659
>>
>> note that the MADT table is already alway skipped on AMD systems,
>> but currently only for IRQs 1/12 which are the PS/2 kbd + mouse
>> IRQs.
>>
>> Regards,
>>
>> Hans
>>
>>
diff mbox series

Patch

diff --git a/drivers/input/mouse/psmouse-smbus.c
b/drivers/input/mouse/psmouse-smbus.c
index a472489cc..6205f8d65 100644
--- a/drivers/input/mouse/psmouse-smbus.c
+++ b/drivers/input/mouse/psmouse-smbus.c
@@ -30,8 +30,8 @@  static void psmouse_smbus_check_adapter(struct
i2c_adapter *adapter)
 {
     struct psmouse_smbus_dev *smbdev;

-    if (!i2c_check_functionality(adapter, I2C_FUNC_SMBUS_HOST_NOTIFY))
-        return;
+    //if (!i2c_check_functionality(adapter, I2C_FUNC_SMBUS_HOST_NOTIFY))
+    //    return;

     mutex_lock(&psmouse_smbus_mutex);

@@ -196,8 +196,8 @@  static int psmouse_smbus_create_companion(struct
device *dev, void *data)
     if (!adapter)
         return 0;

-    if (!i2c_check_functionality(adapter, I2C_FUNC_SMBUS_HOST_NOTIFY))
-        return 0;
+    //if (!i2c_check_functionality(adapter, I2C_FUNC_SMBUS_HOST_NOTIFY))
+    //    return 0;

     client = i2c_new_scanned_device(adapter, &smbdev->board,
                     addr_list, NULL);
diff --git a/drivers/input/mouse/synaptics.c b/drivers/input/mouse/synaptics.c
index ffad14280..2f476dce7 100644
--- a/drivers/input/mouse/synaptics.c
+++ b/drivers/input/mouse/synaptics.c
@@ -1765,7 +1765,7 @@  static int synaptics_create_intertouch(struct
psmouse *psmouse,
     };
     const struct i2c_board_info intertouch_board = {
         I2C_BOARD_INFO("rmi4_smbus", 0x2c),
-        .flags = I2C_CLIENT_HOST_NOTIFY,
+        //.flags = I2C_CLIENT_HOST_NOTIFY,
     };

     return psmouse_smbus_init(psmouse, &intertouch_board,
diff --git a/drivers/input/rmi4/rmi_smbus.c b/drivers/input/rmi4/rmi_smbus.c
index 2407ea43d..3eb7193b4 100644
--- a/drivers/input/rmi4/rmi_smbus.c
+++ b/drivers/input/rmi4/rmi_smbus.c
@@ -281,14 +281,17 @@  static int rmi_smb_probe(struct i2c_client *client,
         return -ENOMEM;
     }

+    //if (!i2c_check_functionality(client->adapter,
+    //                 I2C_FUNC_SMBUS_READ_BLOCK_DATA |
+    //                 I2C_FUNC_SMBUS_HOST_NOTIFY)) {
     if (!i2c_check_functionality(client->adapter,
-                     I2C_FUNC_SMBUS_READ_BLOCK_DATA |
-                     I2C_FUNC_SMBUS_HOST_NOTIFY)) {
+                     I2C_FUNC_SMBUS_READ_BLOCK_DATA)) {
         dev_err(&client->dev,
             "adapter does not support required functionality\n");
         return -ENODEV;
     }

+    client->irq = 7;
     if (client->irq <= 0) {
         dev_err(&client->dev, "no IRQ provided, giving up\n");
         return client->irq ? client->irq : -ENODEV;