dp_aux.c 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945
  1. // SPDX-License-Identifier: GPL-2.0-only
  2. /*
  3. * Copyright (c) 2021-2023, Qualcomm Innovation Center, Inc. All rights reserved.
  4. * Copyright (c) 2012-2021, The Linux Foundation. All rights reserved.
  5. */
  6. #include <linux/delay.h>
  7. #if IS_ENABLED(CONFIG_QCOM_FSA4480_I2C)
  8. #include <linux/soc/qcom/fsa4480-i2c.h>
  9. #elif IS_ENABLED(CONFIG_QCOM_WCD939X_I2C)
  10. #include <linux/soc/qcom/wcd939x-i2c.h>
  11. #endif
  12. #include "dp_aux.h"
  13. #include "dp_hpd.h"
  14. #include "dp_debug.h"
  15. #define DP_AUX_ENUM_STR(x) #x
  16. #define DP_AUX_IPC_NUM_PAGES 10
  17. #define DP_AUX_DEBUG(dp_aux, fmt, ...) \
  18. do { \
  19. if (dp_aux) \
  20. ipc_log_string(dp_aux->ipc_log_context, "[d][%-4d]"fmt,\
  21. current->pid, ##__VA_ARGS__); \
  22. DP_DEBUG_V(fmt, ##__VA_ARGS__); \
  23. } while (0)
  24. #define DP_AUX_WARN(dp_aux, fmt, ...) \
  25. do { \
  26. if (dp_aux) \
  27. ipc_log_string(dp_aux->ipc_log_context, "[w][%-4d]"fmt,\
  28. current->pid, ##__VA_ARGS__); \
  29. DP_WARN_V(fmt, ##__VA_ARGS__); \
  30. } while (0)
  31. #define DP_AUX_WARN_RATELIMITED(dp_aux, fmt, ...) \
  32. do { \
  33. if (dp_aux) \
  34. ipc_log_string(dp_aux->ipc_log_context, "[w][%-4d]"fmt,\
  35. current->pid, ##__VA_ARGS__); \
  36. DP_WARN_RATELIMITED_V(fmt, ##__VA_ARGS__); \
  37. } while (0)
  38. #define DP_AUX_ERR(dp_aux, fmt, ...) \
  39. do { \
  40. if (dp_aux) \
  41. ipc_log_string(dp_aux->ipc_log_context, "[e][%-4d]"fmt,\
  42. current->pid, ##__VA_ARGS__); \
  43. DP_ERR_V(fmt, ##__VA_ARGS__); \
  44. } while (0)
  45. #define DP_AUX_ERR_RATELIMITED(dp_aux, fmt, ...) \
  46. do { \
  47. if (dp_aux) \
  48. ipc_log_string(dp_aux->ipc_log_context, "[e][%-4d]"fmt,\
  49. current->pid, ##__VA_ARGS__); \
  50. DP_ERR_RATELIMITED_V(fmt, ##__VA_ARGS__); \
  51. } while (0)
  52. enum {
  53. DP_AUX_DATA_INDEX_WRITE = BIT(31),
  54. };
  55. struct dp_aux_private {
  56. struct device *dev;
  57. struct dp_aux dp_aux;
  58. struct dp_catalog_aux *catalog;
  59. struct dp_aux_cfg *cfg;
  60. struct device_node *aux_switch_node;
  61. struct mutex mutex;
  62. struct completion comp;
  63. struct drm_dp_aux drm_aux;
  64. struct dp_aux_bridge *aux_bridge;
  65. struct dp_aux_bridge *sim_bridge;
  66. bool bridge_in_transfer;
  67. bool sim_in_transfer;
  68. bool cmd_busy;
  69. bool native;
  70. bool read;
  71. bool no_send_addr;
  72. bool no_send_stop;
  73. bool enabled;
  74. u32 offset;
  75. u32 segment;
  76. u32 aux_error_num;
  77. u32 retry_cnt;
  78. atomic_t aborted;
  79. };
  80. #if IS_ENABLED(CONFIG_DYNAMIC_DEBUG)
  81. static void dp_aux_hex_dump(struct drm_dp_aux *drm_aux,
  82. struct drm_dp_aux_msg *msg)
  83. {
  84. char prefix[64];
  85. int i, linelen, remaining = msg->size;
  86. const int rowsize = 16;
  87. u8 linebuf[64];
  88. struct dp_aux_private *aux = container_of(drm_aux,
  89. struct dp_aux_private, drm_aux);
  90. struct dp_aux *dp_aux = &aux->dp_aux;
  91. snprintf(prefix, sizeof(prefix), "%s %s %4xh(%2zu): ",
  92. (msg->request & DP_AUX_I2C_MOT) ? "I2C" : "NAT",
  93. (msg->request & DP_AUX_I2C_READ) ? "RD" : "WR",
  94. msg->address, msg->size);
  95. for (i = 0; i < msg->size; i += rowsize) {
  96. linelen = min(remaining, rowsize);
  97. remaining -= rowsize;
  98. hex_dump_to_buffer(msg->buffer + i, linelen, rowsize, 1,
  99. linebuf, sizeof(linebuf), false);
  100. if (msg->size == 1 && msg->address == 0)
  101. DP_DEBUG_V("%s%s\n", prefix, linebuf);
  102. else
  103. DP_AUX_DEBUG(dp_aux, "%s%s\n", prefix, linebuf);
  104. }
  105. }
  106. #else
  107. static void dp_aux_hex_dump(struct drm_dp_aux *drm_aux,
  108. struct drm_dp_aux_msg *msg)
  109. {
  110. }
  111. #endif /* CONFIG_DYNAMIC_DEBUG */
  112. static char *dp_aux_get_error(u32 aux_error)
  113. {
  114. switch (aux_error) {
  115. case DP_AUX_ERR_NONE:
  116. return DP_AUX_ENUM_STR(DP_AUX_ERR_NONE);
  117. case DP_AUX_ERR_ADDR:
  118. return DP_AUX_ENUM_STR(DP_AUX_ERR_ADDR);
  119. case DP_AUX_ERR_TOUT:
  120. return DP_AUX_ENUM_STR(DP_AUX_ERR_TOUT);
  121. case DP_AUX_ERR_NACK:
  122. return DP_AUX_ENUM_STR(DP_AUX_ERR_NACK);
  123. case DP_AUX_ERR_DEFER:
  124. return DP_AUX_ENUM_STR(DP_AUX_ERR_DEFER);
  125. case DP_AUX_ERR_NACK_DEFER:
  126. return DP_AUX_ENUM_STR(DP_AUX_ERR_NACK_DEFER);
  127. default:
  128. return "unknown";
  129. }
  130. }
  131. static u32 dp_aux_write(struct dp_aux_private *aux,
  132. struct drm_dp_aux_msg *msg)
  133. {
  134. u32 data[4], reg, len;
  135. u8 *msgdata = msg->buffer;
  136. int const aux_cmd_fifo_len = 128;
  137. int i = 0;
  138. struct dp_aux *dp_aux = &aux->dp_aux;
  139. if (aux->read)
  140. len = 4;
  141. else
  142. len = msg->size + 4;
  143. /*
  144. * cmd fifo only has depth of 144 bytes
  145. * limit buf length to 128 bytes here
  146. */
  147. if (len > aux_cmd_fifo_len) {
  148. DP_AUX_ERR(dp_aux, "buf len error\n");
  149. return 0;
  150. }
  151. /* Pack cmd and write to HW */
  152. data[0] = (msg->address >> 16) & 0xf; /* addr[19:16] */
  153. if (aux->read)
  154. data[0] |= BIT(4); /* R/W */
  155. data[1] = (msg->address >> 8) & 0xff; /* addr[15:8] */
  156. data[2] = msg->address & 0xff; /* addr[7:0] */
  157. data[3] = (msg->size - 1) & 0xff; /* len[7:0] */
  158. for (i = 0; i < len; i++) {
  159. reg = (i < 4) ? data[i] : msgdata[i - 4];
  160. reg = ((reg) << 8) & 0x0000ff00; /* index = 0, write */
  161. if (i == 0)
  162. reg |= DP_AUX_DATA_INDEX_WRITE;
  163. aux->catalog->data = reg;
  164. aux->catalog->write_data(aux->catalog);
  165. }
  166. aux->catalog->clear_trans(aux->catalog, false);
  167. aux->catalog->clear_hw_interrupts(aux->catalog);
  168. reg = 0; /* Transaction number == 1 */
  169. if (!aux->native) { /* i2c */
  170. reg |= BIT(8);
  171. if (aux->no_send_addr)
  172. reg |= BIT(10);
  173. if (aux->no_send_stop)
  174. reg |= BIT(11);
  175. }
  176. reg |= BIT(9);
  177. aux->catalog->data = reg;
  178. aux->catalog->write_trans(aux->catalog);
  179. return len;
  180. }
  181. static int dp_aux_cmd_fifo_tx(struct dp_aux_private *aux,
  182. struct drm_dp_aux_msg *msg)
  183. {
  184. u32 ret = 0, len = 0, timeout;
  185. int const aux_timeout_ms = HZ/4;
  186. struct dp_aux *dp_aux = &aux->dp_aux;
  187. char prefix[64];
  188. snprintf(prefix, sizeof(prefix), "%s %s %4xh(%2zu): ",
  189. (msg->request & DP_AUX_I2C_MOT) ? "I2C" : "NAT",
  190. (msg->request & DP_AUX_I2C_READ) ? "RD" : "WR",
  191. msg->address, msg->size);
  192. reinit_completion(&aux->comp);
  193. len = dp_aux_write(aux, msg);
  194. if (len == 0) {
  195. DP_AUX_ERR(dp_aux, "DP AUX write failed: %s\n", prefix);
  196. return -EINVAL;
  197. }
  198. timeout = wait_for_completion_timeout(&aux->comp, aux_timeout_ms);
  199. if (!timeout) {
  200. DP_AUX_WARN_RATELIMITED(dp_aux, "aux timeout during [%s]\n", prefix);
  201. return -ETIMEDOUT;
  202. }
  203. if (aux->aux_error_num == DP_AUX_ERR_NONE) {
  204. ret = len;
  205. } else {
  206. DP_AUX_WARN_RATELIMITED(dp_aux, "aux err [%s] during [%s]\n",
  207. dp_aux_get_error(aux->aux_error_num), prefix);
  208. ret = -EINVAL;
  209. }
  210. return ret;
  211. }
  212. static void dp_aux_cmd_fifo_rx(struct dp_aux_private *aux,
  213. struct drm_dp_aux_msg *msg)
  214. {
  215. u32 data;
  216. u8 *dp;
  217. u32 i, actual_i;
  218. u32 len = msg->size;
  219. struct dp_aux *dp_aux = &aux->dp_aux;
  220. aux->catalog->clear_trans(aux->catalog, true);
  221. data = 0;
  222. data |= DP_AUX_DATA_INDEX_WRITE; /* INDEX_WRITE */
  223. data |= BIT(0); /* read */
  224. aux->catalog->data = data;
  225. aux->catalog->write_data(aux->catalog);
  226. dp = msg->buffer;
  227. /* discard first byte */
  228. data = aux->catalog->read_data(aux->catalog);
  229. for (i = 0; i < len; i++) {
  230. data = aux->catalog->read_data(aux->catalog);
  231. *dp++ = (u8)((data >> 8) & 0xff);
  232. actual_i = (data >> 16) & 0xFF;
  233. if (i != actual_i)
  234. DP_AUX_WARN(dp_aux, "Index mismatch: expected %d, found %d\n",
  235. i, actual_i);
  236. }
  237. }
  238. static void dp_aux_native_handler(struct dp_aux_private *aux)
  239. {
  240. u32 isr = aux->catalog->isr;
  241. if (isr & DP_INTR_AUX_I2C_DONE)
  242. aux->aux_error_num = DP_AUX_ERR_NONE;
  243. else if (isr & DP_INTR_WRONG_ADDR)
  244. aux->aux_error_num = DP_AUX_ERR_ADDR;
  245. else if (isr & DP_INTR_TIMEOUT)
  246. aux->aux_error_num = DP_AUX_ERR_TOUT;
  247. if (isr & DP_INTR_NACK_DEFER)
  248. aux->aux_error_num = DP_AUX_ERR_NACK;
  249. if (isr & DP_INTR_AUX_ERROR) {
  250. aux->aux_error_num = DP_AUX_ERR_PHY;
  251. aux->catalog->clear_hw_interrupts(aux->catalog);
  252. }
  253. complete(&aux->comp);
  254. }
  255. static void dp_aux_i2c_handler(struct dp_aux_private *aux)
  256. {
  257. u32 isr = aux->catalog->isr;
  258. if (isr & DP_INTR_AUX_I2C_DONE) {
  259. if (isr & (DP_INTR_I2C_NACK | DP_INTR_I2C_DEFER))
  260. aux->aux_error_num = DP_AUX_ERR_NACK;
  261. else
  262. aux->aux_error_num = DP_AUX_ERR_NONE;
  263. } else {
  264. if (isr & DP_INTR_WRONG_ADDR)
  265. aux->aux_error_num = DP_AUX_ERR_ADDR;
  266. else if (isr & DP_INTR_TIMEOUT)
  267. aux->aux_error_num = DP_AUX_ERR_TOUT;
  268. if (isr & DP_INTR_NACK_DEFER)
  269. aux->aux_error_num = DP_AUX_ERR_NACK_DEFER;
  270. if (isr & DP_INTR_I2C_NACK)
  271. aux->aux_error_num = DP_AUX_ERR_NACK;
  272. if (isr & DP_INTR_I2C_DEFER)
  273. aux->aux_error_num = DP_AUX_ERR_DEFER;
  274. if (isr & DP_INTR_AUX_ERROR) {
  275. aux->aux_error_num = DP_AUX_ERR_PHY;
  276. aux->catalog->clear_hw_interrupts(aux->catalog);
  277. }
  278. }
  279. complete(&aux->comp);
  280. }
  281. static void dp_aux_isr(struct dp_aux *dp_aux)
  282. {
  283. struct dp_aux_private *aux;
  284. if (!dp_aux) {
  285. DP_AUX_ERR(dp_aux, "invalid input\n");
  286. return;
  287. }
  288. aux = container_of(dp_aux, struct dp_aux_private, dp_aux);
  289. aux->catalog->get_irq(aux->catalog, aux->cmd_busy);
  290. if (!aux->cmd_busy)
  291. return;
  292. if (aux->native)
  293. dp_aux_native_handler(aux);
  294. else
  295. dp_aux_i2c_handler(aux);
  296. }
  297. static void dp_aux_reconfig(struct dp_aux *dp_aux)
  298. {
  299. struct dp_aux_private *aux;
  300. if (!dp_aux) {
  301. DP_AUX_ERR(dp_aux, "invalid input\n");
  302. return;
  303. }
  304. aux = container_of(dp_aux, struct dp_aux_private, dp_aux);
  305. aux->catalog->update_aux_cfg(aux->catalog,
  306. aux->cfg, PHY_AUX_CFG1);
  307. aux->catalog->reset(aux->catalog);
  308. }
  309. static void dp_aux_abort_transaction(struct dp_aux *dp_aux, bool abort)
  310. {
  311. struct dp_aux_private *aux;
  312. if (!dp_aux) {
  313. DP_AUX_ERR(dp_aux, "invalid input\n");
  314. return;
  315. }
  316. aux = container_of(dp_aux, struct dp_aux_private, dp_aux);
  317. atomic_set(&aux->aborted, abort);
  318. }
  319. static void dp_aux_update_offset_and_segment(struct dp_aux_private *aux,
  320. struct drm_dp_aux_msg *input_msg)
  321. {
  322. u32 const edid_address = 0x50;
  323. u32 const segment_address = 0x30;
  324. bool i2c_read = input_msg->request &
  325. (DP_AUX_I2C_READ & DP_AUX_NATIVE_READ);
  326. u8 *data = NULL;
  327. if (aux->native || i2c_read || ((input_msg->address != edid_address) &&
  328. (input_msg->address != segment_address)))
  329. return;
  330. data = input_msg->buffer;
  331. if (input_msg->address == segment_address)
  332. aux->segment = *data;
  333. else
  334. aux->offset = *data;
  335. }
  336. /**
  337. * dp_aux_transfer_helper() - helper function for EDID read transactions
  338. *
  339. * @aux: DP AUX private structure
  340. * @input_msg: input message from DRM upstream APIs
  341. * @send_seg: send the seg to sink
  342. *
  343. * return: void
  344. *
  345. * This helper function is used to fix EDID reads for non-compliant
  346. * sinks that do not handle the i2c middle-of-transaction flag correctly.
  347. */
  348. static void dp_aux_transfer_helper(struct dp_aux_private *aux,
  349. struct drm_dp_aux_msg *input_msg, bool send_seg)
  350. {
  351. struct drm_dp_aux_msg helper_msg;
  352. u32 const message_size = 0x10;
  353. u32 const segment_address = 0x30;
  354. u32 const edid_block_length = 0x80;
  355. bool i2c_mot = input_msg->request & DP_AUX_I2C_MOT;
  356. bool i2c_read = input_msg->request &
  357. (DP_AUX_I2C_READ & DP_AUX_NATIVE_READ);
  358. if (!i2c_mot || !i2c_read || (input_msg->size == 0))
  359. return;
  360. /*
  361. * Sending the segment value and EDID offset will be performed
  362. * from the DRM upstream EDID driver for each block. Avoid
  363. * duplicate AUX transactions related to this while reading the
  364. * first 16 bytes of each block.
  365. */
  366. if (!(aux->offset % edid_block_length) || !send_seg)
  367. goto end;
  368. aux->read = false;
  369. aux->cmd_busy = true;
  370. aux->no_send_addr = true;
  371. aux->no_send_stop = true;
  372. /*
  373. * Send the segment address for i2c reads for segment > 0 and for which
  374. * the middle-of-transaction flag is set. This is required to support
  375. * EDID reads of more than 2 blocks as the segment address is reset to 0
  376. * since we are overriding the middle-of-transaction flag for read
  377. * transactions.
  378. */
  379. if (aux->segment) {
  380. memset(&helper_msg, 0, sizeof(helper_msg));
  381. helper_msg.address = segment_address;
  382. helper_msg.buffer = &aux->segment;
  383. helper_msg.size = 1;
  384. dp_aux_cmd_fifo_tx(aux, &helper_msg);
  385. }
  386. /*
  387. * Send the offset address for every i2c read in which the
  388. * middle-of-transaction flag is set. This will ensure that the sink
  389. * will update its read pointer and return the correct portion of the
  390. * EDID buffer in the subsequent i2c read trasntion triggered in the
  391. * native AUX transfer function.
  392. */
  393. memset(&helper_msg, 0, sizeof(helper_msg));
  394. helper_msg.address = input_msg->address;
  395. helper_msg.buffer = &aux->offset;
  396. helper_msg.size = 1;
  397. dp_aux_cmd_fifo_tx(aux, &helper_msg);
  398. end:
  399. aux->offset += message_size;
  400. if (aux->offset == 0x80 || aux->offset == 0x100)
  401. aux->segment = 0x0; /* reset segment at end of block */
  402. }
  403. static int dp_aux_transfer_ready(struct dp_aux_private *aux,
  404. struct drm_dp_aux_msg *msg, bool send_seg)
  405. {
  406. int ret = 0;
  407. int const aux_cmd_native_max = 16;
  408. int const aux_cmd_i2c_max = 128;
  409. struct dp_aux *dp_aux = &aux->dp_aux;
  410. if (atomic_read(&aux->aborted)) {
  411. ret = -ETIMEDOUT;
  412. goto error;
  413. }
  414. aux->native = msg->request & (DP_AUX_NATIVE_WRITE & DP_AUX_NATIVE_READ);
  415. /* Ignore address only message */
  416. if ((msg->size == 0) || (msg->buffer == NULL)) {
  417. msg->reply = aux->native ?
  418. DP_AUX_NATIVE_REPLY_ACK : DP_AUX_I2C_REPLY_ACK;
  419. goto error;
  420. }
  421. /* msg sanity check */
  422. if ((aux->native && (msg->size > aux_cmd_native_max)) ||
  423. (msg->size > aux_cmd_i2c_max)) {
  424. DP_AUX_ERR(dp_aux, "%s: invalid msg: size(%zu), request(%x)\n",
  425. __func__, msg->size, msg->request);
  426. ret = -EINVAL;
  427. goto error;
  428. }
  429. dp_aux_update_offset_and_segment(aux, msg);
  430. dp_aux_transfer_helper(aux, msg, send_seg);
  431. aux->read = msg->request & (DP_AUX_I2C_READ & DP_AUX_NATIVE_READ);
  432. if (aux->read) {
  433. aux->no_send_addr = true;
  434. aux->no_send_stop = false;
  435. } else {
  436. aux->no_send_addr = true;
  437. aux->no_send_stop = true;
  438. }
  439. aux->cmd_busy = true;
  440. error:
  441. return ret;
  442. }
  443. static inline bool dp_aux_is_sideband_msg(u32 address, size_t size)
  444. {
  445. return (address >= 0x1000 && address + size < 0x1800) ||
  446. (address >= 0x2000 && address + size < 0x2200);
  447. }
  448. /*
  449. * This function does the real job to process an AUX transaction.
  450. * It will call aux_reset() function to reset the AUX channel,
  451. * if the waiting is timeout.
  452. */
  453. static ssize_t dp_aux_transfer(struct drm_dp_aux *drm_aux,
  454. struct drm_dp_aux_msg *msg)
  455. {
  456. ssize_t ret;
  457. int const retry_count = 5;
  458. struct dp_aux_private *aux = container_of(drm_aux,
  459. struct dp_aux_private, drm_aux);
  460. mutex_lock(&aux->mutex);
  461. ret = dp_aux_transfer_ready(aux, msg, true);
  462. if (ret)
  463. goto unlock_exit;
  464. if (!aux->cmd_busy) {
  465. ret = msg->size;
  466. goto unlock_exit;
  467. }
  468. ret = dp_aux_cmd_fifo_tx(aux, msg);
  469. if ((ret < 0) && !atomic_read(&aux->aborted)) {
  470. aux->retry_cnt++;
  471. if (!(aux->retry_cnt % retry_count))
  472. aux->catalog->update_aux_cfg(aux->catalog,
  473. aux->cfg, PHY_AUX_CFG1);
  474. aux->catalog->reset(aux->catalog);
  475. goto unlock_exit;
  476. } else if (ret < 0) {
  477. goto unlock_exit;
  478. }
  479. if (aux->aux_error_num == DP_AUX_ERR_NONE) {
  480. if (aux->read)
  481. dp_aux_cmd_fifo_rx(aux, msg);
  482. dp_aux_hex_dump(drm_aux, msg);
  483. msg->reply = aux->native ?
  484. DP_AUX_NATIVE_REPLY_ACK : DP_AUX_I2C_REPLY_ACK;
  485. } else {
  486. /* Reply defer to retry */
  487. msg->reply = aux->native ?
  488. DP_AUX_NATIVE_REPLY_DEFER : DP_AUX_I2C_REPLY_DEFER;
  489. }
  490. /* Return requested size for success or retry */
  491. ret = msg->size;
  492. aux->retry_cnt = 0;
  493. unlock_exit:
  494. aux->cmd_busy = false;
  495. mutex_unlock(&aux->mutex);
  496. return ret;
  497. }
  498. static ssize_t dp_aux_bridge_transfer(struct drm_dp_aux *drm_aux,
  499. struct drm_dp_aux_msg *msg)
  500. {
  501. struct dp_aux_private *aux = container_of(drm_aux,
  502. struct dp_aux_private, drm_aux);
  503. ssize_t size;
  504. if (aux->bridge_in_transfer) {
  505. size = dp_aux_transfer(drm_aux, msg);
  506. } else {
  507. aux->bridge_in_transfer = true;
  508. size = aux->aux_bridge->transfer(aux->aux_bridge,
  509. drm_aux, msg);
  510. aux->bridge_in_transfer = false;
  511. dp_aux_hex_dump(drm_aux, msg);
  512. }
  513. return size;
  514. }
  515. static ssize_t dp_aux_transfer_debug(struct drm_dp_aux *drm_aux,
  516. struct drm_dp_aux_msg *msg)
  517. {
  518. struct dp_aux_private *aux = container_of(drm_aux,
  519. struct dp_aux_private, drm_aux);
  520. ssize_t size;
  521. int aborted;
  522. mutex_lock(&aux->mutex);
  523. aborted = atomic_read(&aux->aborted);
  524. mutex_unlock(&aux->mutex);
  525. if (aborted) {
  526. size = -ETIMEDOUT;
  527. goto end;
  528. }
  529. if (aux->sim_in_transfer) {
  530. if (aux->aux_bridge && aux->aux_bridge->transfer)
  531. size = dp_aux_bridge_transfer(drm_aux, msg);
  532. else
  533. size = dp_aux_transfer(drm_aux, msg);
  534. } else {
  535. aux->sim_in_transfer = true;
  536. size = aux->sim_bridge->transfer(aux->sim_bridge,
  537. drm_aux, msg);
  538. aux->sim_in_transfer = false;
  539. dp_aux_hex_dump(drm_aux, msg);
  540. }
  541. end:
  542. return size;
  543. }
  544. static void dp_aux_reset_phy_config_indices(struct dp_aux_cfg *aux_cfg)
  545. {
  546. int i = 0;
  547. for (i = 0; i < PHY_AUX_CFG_MAX; i++)
  548. aux_cfg[i].current_index = 0;
  549. }
  550. static void dp_aux_init(struct dp_aux *dp_aux, struct dp_aux_cfg *aux_cfg)
  551. {
  552. struct dp_aux_private *aux;
  553. if (!dp_aux || !aux_cfg) {
  554. DP_AUX_ERR(dp_aux, "invalid input\n");
  555. return;
  556. }
  557. aux = container_of(dp_aux, struct dp_aux_private, dp_aux);
  558. if (aux->enabled)
  559. return;
  560. dp_aux_reset_phy_config_indices(aux_cfg);
  561. aux->catalog->setup(aux->catalog, aux_cfg);
  562. aux->catalog->reset(aux->catalog);
  563. aux->catalog->enable(aux->catalog, true);
  564. atomic_set(&aux->aborted, 0);
  565. aux->retry_cnt = 0;
  566. aux->enabled = true;
  567. }
  568. static void dp_aux_deinit(struct dp_aux *dp_aux)
  569. {
  570. struct dp_aux_private *aux;
  571. if (!dp_aux) {
  572. DP_AUX_ERR(dp_aux, "invalid input\n");
  573. return;
  574. }
  575. aux = container_of(dp_aux, struct dp_aux_private, dp_aux);
  576. if (!aux->enabled)
  577. return;
  578. atomic_set(&aux->aborted, 1);
  579. aux->catalog->enable(aux->catalog, false);
  580. aux->enabled = false;
  581. }
  582. static int dp_aux_register(struct dp_aux *dp_aux, struct drm_device *drm_dev)
  583. {
  584. struct dp_aux_private *aux;
  585. int ret = 0;
  586. if (!dp_aux) {
  587. DP_AUX_ERR(dp_aux, "invalid input\n");
  588. ret = -EINVAL;
  589. goto exit;
  590. }
  591. aux = container_of(dp_aux, struct dp_aux_private, dp_aux);
  592. aux->drm_aux.name = "sde_dp_aux";
  593. aux->drm_aux.dev = aux->dev;
  594. aux->drm_aux.transfer = dp_aux_transfer;
  595. #if (KERNEL_VERSION(5, 15, 0) <= LINUX_VERSION_CODE)
  596. aux->drm_aux.drm_dev = drm_dev;
  597. #endif
  598. atomic_set(&aux->aborted, 1);
  599. ret = drm_dp_aux_register(&aux->drm_aux);
  600. if (ret) {
  601. DP_AUX_ERR(dp_aux, "%s: failed to register drm aux: %d\n", __func__, ret);
  602. goto exit;
  603. }
  604. dp_aux->drm_aux = &aux->drm_aux;
  605. /* if bridge is defined, override transfer function */
  606. if (aux->aux_bridge && aux->aux_bridge->transfer)
  607. aux->drm_aux.transfer = dp_aux_bridge_transfer;
  608. exit:
  609. return ret;
  610. }
  611. static void dp_aux_deregister(struct dp_aux *dp_aux)
  612. {
  613. struct dp_aux_private *aux;
  614. if (!dp_aux) {
  615. DP_AUX_ERR(dp_aux, "invalid input\n");
  616. return;
  617. }
  618. aux = container_of(dp_aux, struct dp_aux_private, dp_aux);
  619. drm_dp_aux_unregister(&aux->drm_aux);
  620. }
  621. static void dp_aux_set_sim_mode(struct dp_aux *dp_aux,
  622. struct dp_aux_bridge *sim_bridge)
  623. {
  624. struct dp_aux_private *aux;
  625. if (!dp_aux) {
  626. DP_AUX_ERR(dp_aux, "invalid input\n");
  627. return;
  628. }
  629. aux = container_of(dp_aux, struct dp_aux_private, dp_aux);
  630. mutex_lock(&aux->mutex);
  631. aux->sim_bridge = sim_bridge;
  632. if (sim_bridge) {
  633. atomic_set(&aux->aborted, 0);
  634. aux->drm_aux.transfer = dp_aux_transfer_debug;
  635. } else if (aux->aux_bridge && aux->aux_bridge->transfer) {
  636. aux->drm_aux.transfer = dp_aux_bridge_transfer;
  637. } else {
  638. aux->drm_aux.transfer = dp_aux_transfer;
  639. }
  640. mutex_unlock(&aux->mutex);
  641. }
  642. #if IS_ENABLED(CONFIG_QCOM_FSA4480_I2C)
  643. static int dp_aux_configure_fsa_switch(struct dp_aux *dp_aux,
  644. bool enable, int orientation)
  645. {
  646. struct dp_aux_private *aux;
  647. int rc = 0;
  648. enum fsa_function event = FSA_USBC_DISPLAYPORT_DISCONNECTED;
  649. if (!dp_aux) {
  650. DP_AUX_ERR(dp_aux, "invalid input\n");
  651. rc = -EINVAL;
  652. goto end;
  653. }
  654. aux = container_of(dp_aux, struct dp_aux_private, dp_aux);
  655. if (!aux->aux_switch_node) {
  656. DP_AUX_DEBUG(dp_aux, "undefined fsa4480 handle\n");
  657. rc = -EINVAL;
  658. goto end;
  659. }
  660. if (enable) {
  661. switch (orientation) {
  662. case ORIENTATION_CC1:
  663. event = FSA_USBC_ORIENTATION_CC1;
  664. break;
  665. case ORIENTATION_CC2:
  666. event = FSA_USBC_ORIENTATION_CC2;
  667. break;
  668. default:
  669. DP_AUX_ERR(dp_aux, "invalid orientation\n");
  670. rc = -EINVAL;
  671. goto end;
  672. }
  673. }
  674. DP_AUX_DEBUG(dp_aux, "enable=%d, orientation=%d, event=%d\n",
  675. enable, orientation, event);
  676. rc = fsa4480_switch_event(aux->aux_switch_node, event);
  677. if (rc)
  678. DP_AUX_ERR(dp_aux, "failed to configure fsa4480 i2c device (%d)\n", rc);
  679. end:
  680. return rc;
  681. }
  682. #endif
  683. #if IS_ENABLED(CONFIG_QCOM_WCD939X_I2C)
  684. static int dp_aux_configure_wcd_switch(struct dp_aux *dp_aux,
  685. bool enable, int orientation)
  686. {
  687. struct dp_aux_private *aux;
  688. int rc = 0;
  689. enum wcd_usbss_cable_status status = WCD_USBSS_CABLE_DISCONNECT;
  690. enum wcd_usbss_cable_types event = WCD_USBSS_DP_AUX_CC1;
  691. if (!dp_aux) {
  692. DP_AUX_ERR(dp_aux, "invalid input\n");
  693. rc = -EINVAL;
  694. goto end;
  695. }
  696. aux = container_of(dp_aux, struct dp_aux_private, dp_aux);
  697. if (!aux->aux_switch_node) {
  698. DP_AUX_DEBUG(dp_aux, "undefined wcd939x switch handle\n");
  699. rc = -EINVAL;
  700. goto end;
  701. }
  702. if (enable) {
  703. status = WCD_USBSS_CABLE_CONNECT;
  704. switch (orientation) {
  705. case ORIENTATION_CC1:
  706. event = WCD_USBSS_DP_AUX_CC1;
  707. break;
  708. case ORIENTATION_CC2:
  709. event = WCD_USBSS_DP_AUX_CC2;
  710. break;
  711. default:
  712. DP_AUX_ERR(dp_aux, "invalid orientation\n");
  713. rc = -EINVAL;
  714. goto end;
  715. }
  716. }
  717. DP_AUX_DEBUG(dp_aux, "enable=%d, orientation=%d, event=%d\n",
  718. enable, orientation, event);
  719. rc = wcd_usbss_switch_update(event, status);
  720. if (rc)
  721. DP_AUX_ERR(dp_aux, "failed to configure wcd939x i2c device (%d)\n", rc);
  722. end:
  723. return rc;
  724. }
  725. #endif
  726. struct dp_aux *dp_aux_get(struct device *dev, struct dp_catalog_aux *catalog,
  727. struct dp_parser *parser, struct device_node *aux_switch,
  728. struct dp_aux_bridge *aux_bridge, void *ipc_log_context)
  729. {
  730. int rc = 0;
  731. struct dp_aux_private *aux;
  732. struct dp_aux *dp_aux = NULL;
  733. if (!catalog || !parser) {
  734. DP_AUX_ERR(dp_aux, "invalid input\n");
  735. rc = -ENODEV;
  736. goto error;
  737. }
  738. aux = devm_kzalloc(dev, sizeof(*aux), GFP_KERNEL);
  739. if (!aux) {
  740. rc = -ENOMEM;
  741. goto error;
  742. }
  743. init_completion(&aux->comp);
  744. aux->cmd_busy = false;
  745. mutex_init(&aux->mutex);
  746. aux->dev = dev;
  747. aux->catalog = catalog;
  748. aux->cfg = parser->aux_cfg;
  749. aux->aux_switch_node = aux_switch;
  750. aux->aux_bridge = aux_bridge;
  751. dp_aux = &aux->dp_aux;
  752. aux->retry_cnt = 0;
  753. dp_aux->isr = dp_aux_isr;
  754. dp_aux->init = dp_aux_init;
  755. dp_aux->deinit = dp_aux_deinit;
  756. dp_aux->drm_aux_register = dp_aux_register;
  757. dp_aux->drm_aux_deregister = dp_aux_deregister;
  758. dp_aux->reconfig = dp_aux_reconfig;
  759. dp_aux->abort = dp_aux_abort_transaction;
  760. dp_aux->set_sim_mode = dp_aux_set_sim_mode;
  761. dp_aux->ipc_log_context = ipc_log_context;
  762. #if IS_ENABLED(CONFIG_QCOM_FSA4480_I2C)
  763. dp_aux->switch_configure = dp_aux_configure_fsa_switch;
  764. dp_aux->switch_register_notifier = fsa4480_reg_notifier;
  765. dp_aux->switch_unregister_notifier = fsa4480_unreg_notifier;
  766. #elif IS_ENABLED(CONFIG_QCOM_WCD939X_I2C)
  767. dp_aux->switch_configure = dp_aux_configure_wcd_switch;
  768. dp_aux->switch_register_notifier = wcd_usbss_reg_notifier;
  769. dp_aux->switch_unregister_notifier = wcd_usbss_unreg_notifier;
  770. #endif
  771. return dp_aux;
  772. error:
  773. return ERR_PTR(rc);
  774. }
  775. void dp_aux_put(struct dp_aux *dp_aux)
  776. {
  777. struct dp_aux_private *aux;
  778. if (!dp_aux)
  779. return;
  780. aux = container_of(dp_aux, struct dp_aux_private, dp_aux);
  781. mutex_destroy(&aux->mutex);
  782. devm_kfree(aux->dev, aux);
  783. }