drm/msm/dp: Handle aux timeouts, nacks, defers

Let's look at the irq status bits after a transfer and see if we got a
nack or a defer or a timeout, instead of telling drm layers that
everything was fine, while still printing an error message. I wasn't
sure about NACK+DEFER so I lumped all those various errors along with a
nack so that the drm core can figure out that things are just not going
well. The important thing is that we're now returning -ETIMEDOUT when
the message times out and nacks for bad addresses.

Cc: Dmitry Baryshkov <dmitry.baryshkov@linaro.org>
Cc: Abhinav Kumar <abhinavk@codeaurora.org>
Cc: Kuogee Hsieh <khsieh@codeaurora.org>
Cc: aravindh@codeaurora.org
Cc: Sean Paul <sean@poorly.run>
Signed-off-by: Stephen Boyd <swboyd@chromium.org>
Reviewed-by: Kuogee Hsieh <khsieh@codeaurora.org>
Link: https://lore.kernel.org/r/20210507212505.1224111-4-swboyd@chromium.org
Signed-off-by: Rob Clark <robdclark@chromium.org>
This commit is contained in:
Stephen Boyd 2021-05-07 14:25:05 -07:00 committed by Rob Clark
parent 47327fdd7e
commit e305f678e9
2 changed files with 61 additions and 87 deletions

View File

@ -9,7 +9,15 @@
#include "dp_reg.h" #include "dp_reg.h"
#include "dp_aux.h" #include "dp_aux.h"
#define DP_AUX_ENUM_STR(x) #x enum msm_dp_aux_err {
DP_AUX_ERR_NONE,
DP_AUX_ERR_ADDR,
DP_AUX_ERR_TOUT,
DP_AUX_ERR_NACK,
DP_AUX_ERR_DEFER,
DP_AUX_ERR_NACK_DEFER,
DP_AUX_ERR_PHY,
};
struct dp_aux_private { struct dp_aux_private {
struct device *dev; struct device *dev;
@ -18,7 +26,7 @@ struct dp_aux_private {
struct mutex mutex; struct mutex mutex;
struct completion comp; struct completion comp;
u32 aux_error_num; enum msm_dp_aux_err aux_error_num;
u32 retry_cnt; u32 retry_cnt;
bool cmd_busy; bool cmd_busy;
bool native; bool native;
@ -33,62 +41,45 @@ struct dp_aux_private {
#define MAX_AUX_RETRIES 5 #define MAX_AUX_RETRIES 5
static const char *dp_aux_get_error(u32 aux_error) static ssize_t dp_aux_write(struct dp_aux_private *aux,
{
switch (aux_error) {
case DP_AUX_ERR_NONE:
return DP_AUX_ENUM_STR(DP_AUX_ERR_NONE);
case DP_AUX_ERR_ADDR:
return DP_AUX_ENUM_STR(DP_AUX_ERR_ADDR);
case DP_AUX_ERR_TOUT:
return DP_AUX_ENUM_STR(DP_AUX_ERR_TOUT);
case DP_AUX_ERR_NACK:
return DP_AUX_ENUM_STR(DP_AUX_ERR_NACK);
case DP_AUX_ERR_DEFER:
return DP_AUX_ENUM_STR(DP_AUX_ERR_DEFER);
case DP_AUX_ERR_NACK_DEFER:
return DP_AUX_ENUM_STR(DP_AUX_ERR_NACK_DEFER);
default:
return "unknown";
}
}
static u32 dp_aux_write(struct dp_aux_private *aux,
struct drm_dp_aux_msg *msg) struct drm_dp_aux_msg *msg)
{ {
u32 data[4], reg, len; u8 data[4];
u32 reg;
ssize_t len;
u8 *msgdata = msg->buffer; u8 *msgdata = msg->buffer;
int const AUX_CMD_FIFO_LEN = 128; int const AUX_CMD_FIFO_LEN = 128;
int i = 0; int i = 0;
if (aux->read) if (aux->read)
len = 4; len = 0;
else else
len = msg->size + 4; len = msg->size;
/* /*
* cmd fifo only has depth of 144 bytes * cmd fifo only has depth of 144 bytes
* limit buf length to 128 bytes here * limit buf length to 128 bytes here
*/ */
if (len > AUX_CMD_FIFO_LEN) { if (len > AUX_CMD_FIFO_LEN - 4) {
DRM_ERROR("buf size greater than allowed size of 128 bytes\n"); DRM_ERROR("buf size greater than allowed size of 128 bytes\n");
return 0; return -EINVAL;
} }
/* Pack cmd and write to HW */ /* Pack cmd and write to HW */
data[0] = (msg->address >> 16) & 0xf; /* addr[19:16] */ data[0] = (msg->address >> 16) & 0xf; /* addr[19:16] */
if (aux->read) if (aux->read)
data[0] |= BIT(4); /* R/W */ data[0] |= BIT(4); /* R/W */
data[1] = (msg->address >> 8) & 0xff; /* addr[15:8] */ data[1] = msg->address >> 8; /* addr[15:8] */
data[2] = msg->address & 0xff; /* addr[7:0] */ data[2] = msg->address; /* addr[7:0] */
data[3] = (msg->size - 1) & 0xff; /* len[7:0] */ data[3] = msg->size - 1; /* len[7:0] */
for (i = 0; i < len; i++) { for (i = 0; i < len + 4; i++) {
reg = (i < 4) ? data[i] : msgdata[i - 4]; reg = (i < 4) ? data[i] : msgdata[i - 4];
reg <<= DP_AUX_DATA_OFFSET;
reg &= DP_AUX_DATA_MASK;
reg |= DP_AUX_DATA_WRITE;
/* index = 0, write */ /* index = 0, write */
reg = (((reg) << DP_AUX_DATA_OFFSET)
& DP_AUX_DATA_MASK) | DP_AUX_DATA_WRITE;
if (i == 0) if (i == 0)
reg |= DP_AUX_DATA_INDEX_WRITE; reg |= DP_AUX_DATA_INDEX_WRITE;
aux->catalog->aux_data = reg; aux->catalog->aux_data = reg;
@ -116,39 +107,27 @@ static u32 dp_aux_write(struct dp_aux_private *aux,
return len; return len;
} }
static int dp_aux_cmd_fifo_tx(struct dp_aux_private *aux, static ssize_t dp_aux_cmd_fifo_tx(struct dp_aux_private *aux,
struct drm_dp_aux_msg *msg) struct drm_dp_aux_msg *msg)
{ {
u32 ret, len, timeout; ssize_t ret;
int aux_timeout_ms = HZ/4; unsigned long time_left;
reinit_completion(&aux->comp); reinit_completion(&aux->comp);
len = dp_aux_write(aux, msg); ret = dp_aux_write(aux, msg);
if (len == 0) { if (ret < 0)
DRM_ERROR("DP AUX write failed\n"); return ret;
return -EINVAL;
}
timeout = wait_for_completion_timeout(&aux->comp, aux_timeout_ms); time_left = wait_for_completion_timeout(&aux->comp,
if (!timeout) { msecs_to_jiffies(250));
DRM_ERROR("aux %s timeout\n", (aux->read ? "read" : "write")); if (!time_left)
return -ETIMEDOUT; return -ETIMEDOUT;
}
if (aux->aux_error_num == DP_AUX_ERR_NONE) {
ret = len;
} else {
DRM_ERROR_RATELIMITED("aux err: %s\n",
dp_aux_get_error(aux->aux_error_num));
ret = -EINVAL;
}
return ret; return ret;
} }
static void dp_aux_cmd_fifo_rx(struct dp_aux_private *aux, static ssize_t dp_aux_cmd_fifo_rx(struct dp_aux_private *aux,
struct drm_dp_aux_msg *msg) struct drm_dp_aux_msg *msg)
{ {
u32 data; u32 data;
@ -175,9 +154,10 @@ static void dp_aux_cmd_fifo_rx(struct dp_aux_private *aux,
actual_i = (data >> DP_AUX_DATA_INDEX_OFFSET) & 0xFF; actual_i = (data >> DP_AUX_DATA_INDEX_OFFSET) & 0xFF;
if (i != actual_i) if (i != actual_i)
DRM_ERROR("Index mismatch: expected %d, found %d\n", break;
i, actual_i);
} }
return i;
} }
static void dp_aux_native_handler(struct dp_aux_private *aux, u32 isr) static void dp_aux_native_handler(struct dp_aux_private *aux, u32 isr)
@ -367,36 +347,38 @@ static ssize_t dp_aux_transfer(struct drm_dp_aux *dp_aux,
} }
ret = dp_aux_cmd_fifo_tx(aux, msg); ret = dp_aux_cmd_fifo_tx(aux, msg);
if (ret < 0) { if (ret < 0) {
if (aux->native) { if (aux->native) {
aux->retry_cnt++; aux->retry_cnt++;
if (!(aux->retry_cnt % MAX_AUX_RETRIES)) if (!(aux->retry_cnt % MAX_AUX_RETRIES))
dp_catalog_aux_update_cfg(aux->catalog); dp_catalog_aux_update_cfg(aux->catalog);
} }
usleep_range(400, 500); /* at least 400us to next try */
goto unlock_exit;
}
if (aux->aux_error_num == DP_AUX_ERR_NONE) {
if (aux->read)
dp_aux_cmd_fifo_rx(aux, msg);
msg->reply = aux->native ?
DP_AUX_NATIVE_REPLY_ACK : DP_AUX_I2C_REPLY_ACK;
} else { } else {
/* Reply defer to retry */ aux->retry_cnt = 0;
msg->reply = aux->native ? switch (aux->aux_error_num) {
DP_AUX_NATIVE_REPLY_DEFER : DP_AUX_I2C_REPLY_DEFER; case DP_AUX_ERR_NONE:
if (aux->read)
ret = dp_aux_cmd_fifo_rx(aux, msg);
msg->reply = aux->native ? DP_AUX_NATIVE_REPLY_ACK : DP_AUX_I2C_REPLY_ACK;
break;
case DP_AUX_ERR_DEFER:
msg->reply = aux->native ? DP_AUX_NATIVE_REPLY_DEFER : DP_AUX_I2C_REPLY_DEFER;
break;
case DP_AUX_ERR_PHY:
case DP_AUX_ERR_ADDR:
case DP_AUX_ERR_NACK:
case DP_AUX_ERR_NACK_DEFER:
msg->reply = aux->native ? DP_AUX_NATIVE_REPLY_NACK : DP_AUX_I2C_REPLY_NACK;
break;
case DP_AUX_ERR_TOUT:
ret = -ETIMEDOUT;
break;
}
} }
/* Return requested size for success or retry */
ret = msg->size;
aux->retry_cnt = 0;
unlock_exit:
aux->cmd_busy = false; aux->cmd_busy = false;
mutex_unlock(&aux->mutex); mutex_unlock(&aux->mutex);
return ret; return ret;
} }

View File

@ -9,14 +9,6 @@
#include "dp_catalog.h" #include "dp_catalog.h"
#include <drm/drm_dp_helper.h> #include <drm/drm_dp_helper.h>
#define DP_AUX_ERR_NONE 0
#define DP_AUX_ERR_ADDR -1
#define DP_AUX_ERR_TOUT -2
#define DP_AUX_ERR_NACK -3
#define DP_AUX_ERR_DEFER -4
#define DP_AUX_ERR_NACK_DEFER -5
#define DP_AUX_ERR_PHY -6
int dp_aux_register(struct drm_dp_aux *dp_aux); int dp_aux_register(struct drm_dp_aux *dp_aux);
void dp_aux_unregister(struct drm_dp_aux *dp_aux); void dp_aux_unregister(struct drm_dp_aux *dp_aux);
void dp_aux_isr(struct drm_dp_aux *dp_aux); void dp_aux_isr(struct drm_dp_aux *dp_aux);