soc: qcom: pmic_glink: Handle GLINK intent allocation rejections
Some versions of the pmic_glink firmware does not allow dynamic GLINK
intent allocations, attempting to send a message before the firmware has
allocated its receive buffers and announced these intent allocations
will fail. When this happens something like this showns up in the log:
pmic_glink_altmode.pmic_glink_altmode pmic_glink.altmode.0: failed to send altmode request: 0x10 (-125)
pmic_glink_altmode.pmic_glink_altmode pmic_glink.altmode.0: failed to request altmode notifications: -125
ucsi_glink.pmic_glink_ucsi pmic_glink.ucsi.0: failed to send UCSI read request: -125
qcom_battmgr.pmic_glink_power_supply pmic_glink.power-supply.0: failed to request power notifications
GLINK has been updated to distinguish between the cases where the remote
is going down (-ECANCELED) and the intent allocation being rejected
(-EAGAIN).
Retry the send until intent buffers becomes available, or an actual
error occur.
To avoid infinitely waiting for the firmware in the event that this
misbehaves and no intents arrive, an arbitrary 5 second timeout is
used.
This patch was developed with input from Chris Lew.
Reported-by: Johan Hovold <johan@kernel.org>
Closes: https://lore.kernel.org/all/Zqet8iInnDhnxkT9@hovoldconsulting.com/#t
Cc: stable@vger.kernel.org # rpmsg: glink: Handle rejected intent request better
Fixes: 58ef4ece1e
("soc: qcom: pmic_glink: Introduce base PMIC GLINK driver")
Tested-by: Johan Hovold <johan+linaro@kernel.org>
Reviewed-by: Johan Hovold <johan+linaro@kernel.org>
Signed-off-by: Bjorn Andersson <bjorn.andersson@oss.qualcomm.com>
Reviewed-by: Chris Lew <quic_clew@quicinc.com>
Link: https://lore.kernel.org/r/20241023-pmic-glink-ecancelled-v2-2-ebc268129407@oss.qualcomm.com
Signed-off-by: Bjorn Andersson <andersson@kernel.org>
This commit is contained in:
parent
a387e73fed
commit
f8c8791924
@ -4,6 +4,7 @@
|
|||||||
* Copyright (c) 2022, Linaro Ltd
|
* Copyright (c) 2022, Linaro Ltd
|
||||||
*/
|
*/
|
||||||
#include <linux/auxiliary_bus.h>
|
#include <linux/auxiliary_bus.h>
|
||||||
|
#include <linux/delay.h>
|
||||||
#include <linux/module.h>
|
#include <linux/module.h>
|
||||||
#include <linux/of.h>
|
#include <linux/of.h>
|
||||||
#include <linux/platform_device.h>
|
#include <linux/platform_device.h>
|
||||||
@ -13,6 +14,8 @@
|
|||||||
#include <linux/soc/qcom/pmic_glink.h>
|
#include <linux/soc/qcom/pmic_glink.h>
|
||||||
#include <linux/spinlock.h>
|
#include <linux/spinlock.h>
|
||||||
|
|
||||||
|
#define PMIC_GLINK_SEND_TIMEOUT (5 * HZ)
|
||||||
|
|
||||||
enum {
|
enum {
|
||||||
PMIC_GLINK_CLIENT_BATT = 0,
|
PMIC_GLINK_CLIENT_BATT = 0,
|
||||||
PMIC_GLINK_CLIENT_ALTMODE,
|
PMIC_GLINK_CLIENT_ALTMODE,
|
||||||
@ -112,13 +115,29 @@ EXPORT_SYMBOL_GPL(pmic_glink_client_register);
|
|||||||
int pmic_glink_send(struct pmic_glink_client *client, void *data, size_t len)
|
int pmic_glink_send(struct pmic_glink_client *client, void *data, size_t len)
|
||||||
{
|
{
|
||||||
struct pmic_glink *pg = client->pg;
|
struct pmic_glink *pg = client->pg;
|
||||||
|
bool timeout_reached = false;
|
||||||
|
unsigned long start;
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
mutex_lock(&pg->state_lock);
|
mutex_lock(&pg->state_lock);
|
||||||
if (!pg->ept)
|
if (!pg->ept) {
|
||||||
ret = -ECONNRESET;
|
ret = -ECONNRESET;
|
||||||
else
|
} else {
|
||||||
|
start = jiffies;
|
||||||
|
for (;;) {
|
||||||
ret = rpmsg_send(pg->ept, data, len);
|
ret = rpmsg_send(pg->ept, data, len);
|
||||||
|
if (ret != -EAGAIN)
|
||||||
|
break;
|
||||||
|
|
||||||
|
if (timeout_reached) {
|
||||||
|
ret = -ETIMEDOUT;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
usleep_range(1000, 5000);
|
||||||
|
timeout_reached = time_after(jiffies, start + PMIC_GLINK_SEND_TIMEOUT);
|
||||||
|
}
|
||||||
|
}
|
||||||
mutex_unlock(&pg->state_lock);
|
mutex_unlock(&pg->state_lock);
|
||||||
|
|
||||||
return ret;
|
return ret;
|
||||||
|
Loading…
Reference in New Issue
Block a user