Message ID | 20240813-i3c_fix-v2-10-68fe4a050188@nxp.com (mailing list archive) |
---|---|
State | Superseded |
Headers | show |
Series | i3c: master: some fix and improvemnt for hotjoin | expand |
Hi Frank, kernel test robot noticed the following build warnings: url: https://github.com/intel-lab-lkp/linux/commits/Frank-Li/i3c-master-Remove-i3c_dev_disable_ibi_locked-olddev-on-device-hotjoin/20240814-234209 base: 41c196e567fb1ea97f68a2ffb7faab451cd90854 patch link: https://lore.kernel.org/r/20240813-i3c_fix-v2-10-68fe4a050188%40nxp.com patch subject: [PATCH v2 10/11] i3c: master: svc: wait for Manual ACK/NACK Done before next step config: x86_64-randconfig-161-20240817 (https://download.01.org/0day-ci/archive/20240818/202408180012.ifcIOjgX-lkp@intel.com/config) compiler: clang version 18.1.5 (https://github.com/llvm/llvm-project 617a15a9eac96088ae5e9134248d8236e34b91b1) If you fix the issue in a separate patch/commit (i.e. not just a new version of the same patch/commit), kindly add following tags | Reported-by: kernel test robot <lkp@intel.com> | Reported-by: Dan Carpenter <dan.carpenter@linaro.org> | Closes: https://lore.kernel.org/r/202408180012.ifcIOjgX-lkp@intel.com/ New smatch warnings: drivers/i3c/master/svc-i3c-master.c:1165 svc_i3c_master_xfer() warn: missing error code 'ret' vim +/ret +1165 drivers/i3c/master/svc-i3c-master.c dd3c52846d5954a Miquel Raynal 2021-01-21 1123 static int svc_i3c_master_xfer(struct svc_i3c_master *master, dd3c52846d5954a Miquel Raynal 2021-01-21 1124 bool rnw, unsigned int xfer_type, u8 addr, dd3c52846d5954a Miquel Raynal 2021-01-21 1125 u8 *in, const u8 *out, unsigned int xfer_len, 6fb61734a74eaa3 Frank Li 2023-12-01 1126 unsigned int *actual_len, bool continued) dd3c52846d5954a Miquel Raynal 2021-01-21 1127 { 2d15862dfba6bf1 Frank Li 2024-06-03 1128 int retry = 2; dd3c52846d5954a Miquel Raynal 2021-01-21 1129 u32 reg; dd3c52846d5954a Miquel Raynal 2021-01-21 1130 int ret; dd3c52846d5954a Miquel Raynal 2021-01-21 1131 5e5e3c92e748a6d Frank Li 2023-10-23 1132 /* clean SVC_I3C_MINT_IBIWON w1c bits */ 5e5e3c92e748a6d Frank Li 2023-10-23 1133 writel(SVC_I3C_MINT_IBIWON, master->regs + SVC_I3C_MSTATUS); 5e5e3c92e748a6d Frank Li 2023-10-23 1134 2d15862dfba6bf1 Frank Li 2024-06-03 1135 2d15862dfba6bf1 Frank Li 2024-06-03 1136 while (retry--) { dd3c52846d5954a Miquel Raynal 2021-01-21 1137 writel(SVC_I3C_MCTRL_REQUEST_START_ADDR | dd3c52846d5954a Miquel Raynal 2021-01-21 1138 xfer_type | dd3c52846d5954a Miquel Raynal 2021-01-21 1139 SVC_I3C_MCTRL_IBIRESP_NACK | dd3c52846d5954a Miquel Raynal 2021-01-21 1140 SVC_I3C_MCTRL_DIR(rnw) | dd3c52846d5954a Miquel Raynal 2021-01-21 1141 SVC_I3C_MCTRL_ADDR(addr) | 6fb61734a74eaa3 Frank Li 2023-12-01 1142 SVC_I3C_MCTRL_RDTERM(*actual_len), dd3c52846d5954a Miquel Raynal 2021-01-21 1143 master->regs + SVC_I3C_MCTRL); dd3c52846d5954a Miquel Raynal 2021-01-21 1144 dd3c52846d5954a Miquel Raynal 2021-01-21 1145 ret = readl_poll_timeout(master->regs + SVC_I3C_MSTATUS, reg, dd3c52846d5954a Miquel Raynal 2021-01-21 1146 SVC_I3C_MSTATUS_MCTRLDONE(reg), 0, 1000); dd3c52846d5954a Miquel Raynal 2021-01-21 1147 if (ret) dd3c52846d5954a Miquel Raynal 2021-01-21 1148 goto emit_stop; dd3c52846d5954a Miquel Raynal 2021-01-21 1149 28a9dc69921b36a Frank Li 2024-08-13 1150 /* 28a9dc69921b36a Frank Li 2024-08-13 1151 * According to I3C spec ver 1.1.1, 5.1.2.2.3 Consequence of Controller Starting a 28a9dc69921b36a Frank Li 2024-08-13 1152 * Frame with I3C Target Address. 28a9dc69921b36a Frank Li 2024-08-13 1153 * 28a9dc69921b36a Frank Li 2024-08-13 1154 * The I3C Controller normally should start a Frame, the Address may be arbitrated, 28a9dc69921b36a Frank Li 2024-08-13 1155 * and so the Controller shall monitor to see whether an In-Band Interrupt request, 28a9dc69921b36a Frank Li 2024-08-13 1156 * a Controller Role Request (i.e., Secondary Controller requests to become the 28a9dc69921b36a Frank Li 2024-08-13 1157 * Active Controller), or a Hot-Join Request has been made. 28a9dc69921b36a Frank Li 2024-08-13 1158 * 28a9dc69921b36a Frank Li 2024-08-13 1159 * If missed IBIWON check, the wrong data will be return. When IBIWON happen, issue 28a9dc69921b36a Frank Li 2024-08-13 1160 * repeat start. Address arbitrate only happen at START, never happen at REPEAT 28a9dc69921b36a Frank Li 2024-08-13 1161 * start. 28a9dc69921b36a Frank Li 2024-08-13 1162 */ 28a9dc69921b36a Frank Li 2024-08-13 1163 if (SVC_I3C_MSTATUS_IBIWON(reg)) { 4809f19e89760fb Frank Li 2024-08-13 1164 if (svc_i3c_master_handle_ibi_won(master, reg)) 4809f19e89760fb Frank Li 2024-08-13 @1165 goto emit_stop; Does this need an error code? 28a9dc69921b36a Frank Li 2024-08-13 1166 continue; 28a9dc69921b36a Frank Li 2024-08-13 1167 } 28a9dc69921b36a Frank Li 2024-08-13 1168 49b472ebc61de3d Clark Wang 2023-05-17 1169 if (readl(master->regs + SVC_I3C_MERRWARN) & SVC_I3C_MERRWARN_NACK) { 2d15862dfba6bf1 Frank Li 2024-06-03 1170 /* 2d15862dfba6bf1 Frank Li 2024-06-03 1171 * According to I3C Spec 1.1.1, 11-Jun-2021, section: 5.1.2.2.3. 2d15862dfba6bf1 Frank Li 2024-06-03 1172 * If the Controller chooses to start an I3C Message with an I3C Dynamic 2d15862dfba6bf1 Frank Li 2024-06-03 1173 * Address, then special provisions shall be made because that same I3C 2d15862dfba6bf1 Frank Li 2024-06-03 1174 * Target may be initiating an IBI or a Controller Role Request. So, one of 2d15862dfba6bf1 Frank Li 2024-06-03 1175 * three things may happen: (skip 1, 2) 2d15862dfba6bf1 Frank Li 2024-06-03 1176 * 2d15862dfba6bf1 Frank Li 2024-06-03 1177 * 3. The Addresses match and the RnW bits also match, and so neither 2d15862dfba6bf1 Frank Li 2024-06-03 1178 * Controller nor Target will ACK since both are expecting the other side to 2d15862dfba6bf1 Frank Li 2024-06-03 1179 * provide ACK. As a result, each side might think it had "won" arbitration, 2d15862dfba6bf1 Frank Li 2024-06-03 1180 * but neither side would continue, as each would subsequently see that the 2d15862dfba6bf1 Frank Li 2024-06-03 1181 * other did not provide ACK. 2d15862dfba6bf1 Frank Li 2024-06-03 1182 * ... 2d15862dfba6bf1 Frank Li 2024-06-03 1183 * For either value of RnW: Due to the NACK, the Controller shall defer the 2d15862dfba6bf1 Frank Li 2024-06-03 1184 * Private Write or Private Read, and should typically transmit the Target 2d15862dfba6bf1 Frank Li 2024-06-03 1185 * Address again after a Repeated START (i.e., the next one or any one prior 2d15862dfba6bf1 Frank Li 2024-06-03 1186 * to a STOP in the Frame). Since the Address Header following a Repeated 2d15862dfba6bf1 Frank Li 2024-06-03 1187 * START is not arbitrated, the Controller will always win (see Section 2d15862dfba6bf1 Frank Li 2024-06-03 1188 * 5.1.2.2.4). 2d15862dfba6bf1 Frank Li 2024-06-03 1189 */ 2d15862dfba6bf1 Frank Li 2024-06-03 1190 if (retry && addr != 0x7e) { 2d15862dfba6bf1 Frank Li 2024-06-03 1191 writel(SVC_I3C_MERRWARN_NACK, master->regs + SVC_I3C_MERRWARN); 2d15862dfba6bf1 Frank Li 2024-06-03 1192 } else { 49b472ebc61de3d Clark Wang 2023-05-17 1193 ret = -ENXIO; 6d1a19d34e2cc07 Frank Li 2023-12-01 1194 *actual_len = 0; 49b472ebc61de3d Clark Wang 2023-05-17 1195 goto emit_stop; 49b472ebc61de3d Clark Wang 2023-05-17 1196 } 2d15862dfba6bf1 Frank Li 2024-06-03 1197 } else { 2d15862dfba6bf1 Frank Li 2024-06-03 1198 break; 2d15862dfba6bf1 Frank Li 2024-06-03 1199 } 2d15862dfba6bf1 Frank Li 2024-06-03 1200 } 49b472ebc61de3d Clark Wang 2023-05-17 1201 dd3c52846d5954a Miquel Raynal 2021-01-21 1202 if (rnw) dd3c52846d5954a Miquel Raynal 2021-01-21 1203 ret = svc_i3c_master_read(master, in, xfer_len); dd3c52846d5954a Miquel Raynal 2021-01-21 1204 else dd3c52846d5954a Miquel Raynal 2021-01-21 1205 ret = svc_i3c_master_write(master, out, xfer_len); d5e512574dd2eb0 Clark Wang 2021-12-27 1206 if (ret < 0) dd3c52846d5954a Miquel Raynal 2021-01-21 1207 goto emit_stop; dd3c52846d5954a Miquel Raynal 2021-01-21 1208 d5e512574dd2eb0 Clark Wang 2021-12-27 1209 if (rnw) 6fb61734a74eaa3 Frank Li 2023-12-01 1210 *actual_len = ret; d5e512574dd2eb0 Clark Wang 2021-12-27 1211 dd3c52846d5954a Miquel Raynal 2021-01-21 1212 ret = readl_poll_timeout(master->regs + SVC_I3C_MSTATUS, reg, dd3c52846d5954a Miquel Raynal 2021-01-21 1213 SVC_I3C_MSTATUS_COMPLETE(reg), 0, 1000); dd3c52846d5954a Miquel Raynal 2021-01-21 1214 if (ret) dd3c52846d5954a Miquel Raynal 2021-01-21 1215 goto emit_stop; dd3c52846d5954a Miquel Raynal 2021-01-21 1216 d5e512574dd2eb0 Clark Wang 2021-12-27 1217 writel(SVC_I3C_MINT_COMPLETE, master->regs + SVC_I3C_MSTATUS); d5e512574dd2eb0 Clark Wang 2021-12-27 1218 d5e512574dd2eb0 Clark Wang 2021-12-27 1219 if (!continued) { dd3c52846d5954a Miquel Raynal 2021-01-21 1220 svc_i3c_master_emit_stop(master); dd3c52846d5954a Miquel Raynal 2021-01-21 1221 d5e512574dd2eb0 Clark Wang 2021-12-27 1222 /* Wait idle if stop is sent. */ d5e512574dd2eb0 Clark Wang 2021-12-27 1223 readl_poll_timeout(master->regs + SVC_I3C_MSTATUS, reg, d5e512574dd2eb0 Clark Wang 2021-12-27 1224 SVC_I3C_MSTATUS_STATE_IDLE(reg), 0, 1000); d5e512574dd2eb0 Clark Wang 2021-12-27 1225 } d5e512574dd2eb0 Clark Wang 2021-12-27 1226 dd3c52846d5954a Miquel Raynal 2021-01-21 1227 return 0; dd3c52846d5954a Miquel Raynal 2021-01-21 1228 dd3c52846d5954a Miquel Raynal 2021-01-21 1229 emit_stop: dd3c52846d5954a Miquel Raynal 2021-01-21 1230 svc_i3c_master_emit_stop(master); dd3c52846d5954a Miquel Raynal 2021-01-21 1231 svc_i3c_master_clear_merrwarn(master); dd3c52846d5954a Miquel Raynal 2021-01-21 1232 dd3c52846d5954a Miquel Raynal 2021-01-21 1233 return ret; dd3c52846d5954a Miquel Raynal 2021-01-21 1234 }
diff --git a/drivers/i3c/master/svc-i3c-master.c b/drivers/i3c/master/svc-i3c-master.c index fbb6cef405577..0de95f406c95b 100644 --- a/drivers/i3c/master/svc-i3c-master.c +++ b/drivers/i3c/master/svc-i3c-master.c @@ -384,10 +384,12 @@ static int svc_i3c_master_handle_ibi(struct svc_i3c_master *master, return 0; } -static void svc_i3c_master_ack_ibi(struct svc_i3c_master *master, +static int svc_i3c_master_ack_ibi(struct svc_i3c_master *master, bool mandatory_byte) { unsigned int ibi_ack_nack; + int ret; + u32 reg; ibi_ack_nack = SVC_I3C_MCTRL_REQUEST_IBI_ACKNACK; if (mandatory_byte) @@ -396,18 +398,31 @@ static void svc_i3c_master_ack_ibi(struct svc_i3c_master *master, ibi_ack_nack |= SVC_I3C_MCTRL_IBIRESP_ACK_WITHOUT_BYTE; writel(ibi_ack_nack, master->regs + SVC_I3C_MCTRL); + + ret = readl_poll_timeout_atomic(master->regs + SVC_I3C_MSTATUS, reg, + SVC_I3C_MSTATUS_MCTRLDONE(reg), 1, 1000); + return ret; + } -static void svc_i3c_master_nack_ibi(struct svc_i3c_master *master) +static int svc_i3c_master_nack_ibi(struct svc_i3c_master *master) { + int ret; + u32 reg; + writel(SVC_I3C_MCTRL_REQUEST_IBI_ACKNACK | SVC_I3C_MCTRL_IBIRESP_NACK, master->regs + SVC_I3C_MCTRL); + + ret = readl_poll_timeout_atomic(master->regs + SVC_I3C_MSTATUS, reg, + SVC_I3C_MSTATUS_MCTRLDONE(reg), 1, 1000); + return ret; } static int svc_i3c_master_handle_ibi_won(struct svc_i3c_master *master, u32 mstatus) { u32 ibitype; + int ret = 0; ibitype = SVC_I3C_MSTATUS_IBITYPE(mstatus); @@ -417,10 +432,10 @@ static int svc_i3c_master_handle_ibi_won(struct svc_i3c_master *master, u32 msta switch (ibitype) { case SVC_I3C_MSTATUS_IBITYPE_HOT_JOIN: case SVC_I3C_MSTATUS_IBITYPE_MASTER_REQUEST: - svc_i3c_master_nack_ibi(master); + ret = svc_i3c_master_nack_ibi(master); } - return 0; + return ret; } static void svc_i3c_master_ibi_work(struct work_struct *work) @@ -871,7 +886,8 @@ static int svc_i3c_master_do_daa_locked(struct svc_i3c_master *master, if (ret) break; } else if (SVC_I3C_MSTATUS_IBIWON(reg)) { - svc_i3c_master_handle_ibi_won(master, reg); + if (svc_i3c_master_handle_ibi_won(master, reg)) + break; continue; } else if (SVC_I3C_MSTATUS_MCTRLDONE(reg)) { if (SVC_I3C_MSTATUS_STATE_IDLE(reg) && @@ -1145,7 +1161,8 @@ static int svc_i3c_master_xfer(struct svc_i3c_master *master, * start. */ if (SVC_I3C_MSTATUS_IBIWON(reg)) { - svc_i3c_master_handle_ibi_won(master, reg); + if (svc_i3c_master_handle_ibi_won(master, reg)) + goto emit_stop; continue; }
Wait for the controller to complete emitting ACK/NACK, otherwise the next command may be omitted by the hardware. Add command done check at svc_i3c_master_nack(ack)_ibi() and change return type to int to indicate wait done timeout. Signed-off-by: Frank Li <Frank.Li@nxp.com> --- drivers/i3c/master/svc-i3c-master.c | 29 +++++++++++++++++++++++------ 1 file changed, 23 insertions(+), 6 deletions(-)