Skip to content

Commit 976e7e9

Browse files
Christian A. Ehrhardtgregkh
Christian A. Ehrhardt
authored andcommitted
acpi: typec: ucsi: Introduce a ->poll_cci method
For the ACPI backend of UCSI the UCSI "registers" are just a memory copy of the register values in an opregion. The ACPI implementation in the BIOS ensures that the opregion contents are synced to the embedded controller and it ensures that the registers (in particular CCI) are synced back to the opregion on notifications. While there is an ACPI call that syncs the actual registers to the opregion there is rarely a need to do this and on some ACPI implementations it actually breaks in various interesting ways. The only reason to force a sync from the embedded controller is to poll CCI while notifications are disabled. Only the ucsi core knows if this is the case and guessing based on the current command is suboptimal, i.e. leading to the following spurious assertion splat: WARNING: CPU: 3 PID: 76 at drivers/usb/typec/ucsi/ucsi.c:1388 ucsi_reset_ppm+0x1b4/0x1c0 [typec_ucsi] CPU: 3 UID: 0 PID: 76 Comm: kworker/3:0 Not tainted 6.12.11-200.fc41.x86_64 #1 Hardware name: LENOVO 21D0/LNVNB161216, BIOS J6CN45WW 03/17/2023 Workqueue: events_long ucsi_init_work [typec_ucsi] RIP: 0010:ucsi_reset_ppm+0x1b4/0x1c0 [typec_ucsi] Call Trace: <TASK> ucsi_init_work+0x3c/0xac0 [typec_ucsi] process_one_work+0x179/0x330 worker_thread+0x252/0x390 kthread+0xd2/0x100 ret_from_fork+0x34/0x50 ret_from_fork_asm+0x1a/0x30 </TASK> Thus introduce a ->poll_cci() method that works like ->read_cci() with an additional forced sync and document that this should be used when polling with notifications disabled. For all other backends that presumably don't have this issue use the same implementation for both methods. Fixes: fa48d7e ("usb: typec: ucsi: Do not call ACPI _DSM method for UCSI read operations") Cc: stable <stable@kernel.org> Signed-off-by: Christian A. Ehrhardt <lk@c--e.de> Tested-by: Fedor Pchelkin <boddah8794@gmail.com> Signed-off-by: Fedor Pchelkin <boddah8794@gmail.com> Reviewed-by: Heikki Krogerus <heikki.krogerus@linux.intel.com> Link: https://lore.kernel.org/r/20250217105442.113486-2-boddah8794@gmail.com Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
1 parent d6b82da commit 976e7e9

File tree

7 files changed

+25
-12
lines changed

7 files changed

+25
-12
lines changed

drivers/usb/typec/ucsi/ucsi.c

+5-5
Original file line numberDiff line numberDiff line change
@@ -1346,7 +1346,7 @@ static int ucsi_reset_ppm(struct ucsi *ucsi)
13461346

13471347
mutex_lock(&ucsi->ppm_lock);
13481348

1349-
ret = ucsi->ops->read_cci(ucsi, &cci);
1349+
ret = ucsi->ops->poll_cci(ucsi, &cci);
13501350
if (ret < 0)
13511351
goto out;
13521352

@@ -1364,7 +1364,7 @@ static int ucsi_reset_ppm(struct ucsi *ucsi)
13641364

13651365
tmo = jiffies + msecs_to_jiffies(UCSI_TIMEOUT_MS);
13661366
do {
1367-
ret = ucsi->ops->read_cci(ucsi, &cci);
1367+
ret = ucsi->ops->poll_cci(ucsi, &cci);
13681368
if (ret < 0)
13691369
goto out;
13701370
if (cci & UCSI_CCI_COMMAND_COMPLETE)
@@ -1393,7 +1393,7 @@ static int ucsi_reset_ppm(struct ucsi *ucsi)
13931393
/* Give the PPM time to process a reset before reading CCI */
13941394
msleep(20);
13951395

1396-
ret = ucsi->ops->read_cci(ucsi, &cci);
1396+
ret = ucsi->ops->poll_cci(ucsi, &cci);
13971397
if (ret)
13981398
goto out;
13991399

@@ -1929,8 +1929,8 @@ struct ucsi *ucsi_create(struct device *dev, const struct ucsi_operations *ops)
19291929
struct ucsi *ucsi;
19301930

19311931
if (!ops ||
1932-
!ops->read_version || !ops->read_cci || !ops->read_message_in ||
1933-
!ops->sync_control || !ops->async_control)
1932+
!ops->read_version || !ops->read_cci || !ops->poll_cci ||
1933+
!ops->read_message_in || !ops->sync_control || !ops->async_control)
19341934
return ERR_PTR(-EINVAL);
19351935

19361936
ucsi = kzalloc(sizeof(*ucsi), GFP_KERNEL);

drivers/usb/typec/ucsi/ucsi.h

+2
Original file line numberDiff line numberDiff line change
@@ -62,6 +62,7 @@ struct dentry;
6262
* struct ucsi_operations - UCSI I/O operations
6363
* @read_version: Read implemented UCSI version
6464
* @read_cci: Read CCI register
65+
* @poll_cci: Read CCI register while polling with notifications disabled
6566
* @read_message_in: Read message data from UCSI
6667
* @sync_control: Blocking control operation
6768
* @async_control: Non-blocking control operation
@@ -76,6 +77,7 @@ struct dentry;
7677
struct ucsi_operations {
7778
int (*read_version)(struct ucsi *ucsi, u16 *version);
7879
int (*read_cci)(struct ucsi *ucsi, u32 *cci);
80+
int (*poll_cci)(struct ucsi *ucsi, u32 *cci);
7981
int (*read_message_in)(struct ucsi *ucsi, void *val, size_t val_len);
8082
int (*sync_control)(struct ucsi *ucsi, u64 command);
8183
int (*async_control)(struct ucsi *ucsi, u64 command);

drivers/usb/typec/ucsi/ucsi_acpi.c

+14-7
Original file line numberDiff line numberDiff line change
@@ -59,19 +59,24 @@ static int ucsi_acpi_read_version(struct ucsi *ucsi, u16 *version)
5959
static int ucsi_acpi_read_cci(struct ucsi *ucsi, u32 *cci)
6060
{
6161
struct ucsi_acpi *ua = ucsi_get_drvdata(ucsi);
62-
int ret;
63-
64-
if (UCSI_COMMAND(ua->cmd) == UCSI_PPM_RESET) {
65-
ret = ucsi_acpi_dsm(ua, UCSI_DSM_FUNC_READ);
66-
if (ret)
67-
return ret;
68-
}
6962

7063
memcpy(cci, ua->base + UCSI_CCI, sizeof(*cci));
7164

7265
return 0;
7366
}
7467

68+
static int ucsi_acpi_poll_cci(struct ucsi *ucsi, u32 *cci)
69+
{
70+
struct ucsi_acpi *ua = ucsi_get_drvdata(ucsi);
71+
int ret;
72+
73+
ret = ucsi_acpi_dsm(ua, UCSI_DSM_FUNC_READ);
74+
if (ret)
75+
return ret;
76+
77+
return ucsi_acpi_read_cci(ucsi, cci);
78+
}
79+
7580
static int ucsi_acpi_read_message_in(struct ucsi *ucsi, void *val, size_t val_len)
7681
{
7782
struct ucsi_acpi *ua = ucsi_get_drvdata(ucsi);
@@ -94,6 +99,7 @@ static int ucsi_acpi_async_control(struct ucsi *ucsi, u64 command)
9499
static const struct ucsi_operations ucsi_acpi_ops = {
95100
.read_version = ucsi_acpi_read_version,
96101
.read_cci = ucsi_acpi_read_cci,
102+
.poll_cci = ucsi_acpi_poll_cci,
97103
.read_message_in = ucsi_acpi_read_message_in,
98104
.sync_control = ucsi_sync_control_common,
99105
.async_control = ucsi_acpi_async_control
@@ -142,6 +148,7 @@ static int ucsi_gram_sync_control(struct ucsi *ucsi, u64 command)
142148
static const struct ucsi_operations ucsi_gram_ops = {
143149
.read_version = ucsi_acpi_read_version,
144150
.read_cci = ucsi_acpi_read_cci,
151+
.poll_cci = ucsi_acpi_poll_cci,
145152
.read_message_in = ucsi_gram_read_message_in,
146153
.sync_control = ucsi_gram_sync_control,
147154
.async_control = ucsi_acpi_async_control

drivers/usb/typec/ucsi/ucsi_ccg.c

+1
Original file line numberDiff line numberDiff line change
@@ -664,6 +664,7 @@ static int ucsi_ccg_sync_control(struct ucsi *ucsi, u64 command)
664664
static const struct ucsi_operations ucsi_ccg_ops = {
665665
.read_version = ucsi_ccg_read_version,
666666
.read_cci = ucsi_ccg_read_cci,
667+
.poll_cci = ucsi_ccg_read_cci,
667668
.read_message_in = ucsi_ccg_read_message_in,
668669
.sync_control = ucsi_ccg_sync_control,
669670
.async_control = ucsi_ccg_async_control,

drivers/usb/typec/ucsi/ucsi_glink.c

+1
Original file line numberDiff line numberDiff line change
@@ -206,6 +206,7 @@ static void pmic_glink_ucsi_connector_status(struct ucsi_connector *con)
206206
static const struct ucsi_operations pmic_glink_ucsi_ops = {
207207
.read_version = pmic_glink_ucsi_read_version,
208208
.read_cci = pmic_glink_ucsi_read_cci,
209+
.poll_cci = pmic_glink_ucsi_read_cci,
209210
.read_message_in = pmic_glink_ucsi_read_message_in,
210211
.sync_control = ucsi_sync_control_common,
211212
.async_control = pmic_glink_ucsi_async_control,

drivers/usb/typec/ucsi/ucsi_stm32g0.c

+1
Original file line numberDiff line numberDiff line change
@@ -424,6 +424,7 @@ static irqreturn_t ucsi_stm32g0_irq_handler(int irq, void *data)
424424
static const struct ucsi_operations ucsi_stm32g0_ops = {
425425
.read_version = ucsi_stm32g0_read_version,
426426
.read_cci = ucsi_stm32g0_read_cci,
427+
.poll_cci = ucsi_stm32g0_read_cci,
427428
.read_message_in = ucsi_stm32g0_read_message_in,
428429
.sync_control = ucsi_sync_control_common,
429430
.async_control = ucsi_stm32g0_async_control,

drivers/usb/typec/ucsi/ucsi_yoga_c630.c

+1
Original file line numberDiff line numberDiff line change
@@ -74,6 +74,7 @@ static int yoga_c630_ucsi_async_control(struct ucsi *ucsi, u64 command)
7474
static const struct ucsi_operations yoga_c630_ucsi_ops = {
7575
.read_version = yoga_c630_ucsi_read_version,
7676
.read_cci = yoga_c630_ucsi_read_cci,
77+
.poll_cci = yoga_c630_ucsi_read_cci,
7778
.read_message_in = yoga_c630_ucsi_read_message_in,
7879
.sync_control = ucsi_sync_control_common,
7980
.async_control = yoga_c630_ucsi_async_control,

0 commit comments

Comments
 (0)