1 /*
2 * Copyright 2012-15 Advanced Micro Devices, Inc.
3 *
4 * Permission is hereby granted, free of charge, to any person obtaining a
5 * copy of this software and associated documentation files (the "Software"),
6 * to deal in the Software without restriction, including without limitation
7 * the rights to use, copy, modify, merge, publish, distribute, sublicense,
8 * and/or sell copies of the Software, and to permit persons to whom the
9 * Software is furnished to do so, subject to the following conditions:
10 *
11 * The above copyright notice and this permission notice shall be included in
12 * all copies or substantial portions of the Software.
13 *
14 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
15 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
16 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
17 * THE COPYRIGHT HOLDER(S) OR AUTHOR(S) BE LIABLE FOR ANY CLAIM, DAMAGES OR
18 * OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
19 * ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
20 * OTHER DEALINGS IN THE SOFTWARE.
21 *
22 * Authors: AMD
23 *
24 */
25
26 #include "dm_services.h"
27 #include "dce_aux.h"
28 #include "dce/dce_11_0_sh_mask.h"
29
30 #define CTX \
31 aux110->base.ctx
32 #define REG(reg_name)\
33 (aux110->regs->reg_name)
34
35 #define DC_LOGGER \
36 engine->ctx->logger
37
38 #include "reg_helper.h"
39
40 #define FROM_AUX_ENGINE(ptr) \
41 container_of((ptr), struct aux_engine_dce110, base)
42
43 #define FROM_ENGINE(ptr) \
44 FROM_AUX_ENGINE(container_of((ptr), struct aux_engine, base))
45
46 #define FROM_AUX_ENGINE_ENGINE(ptr) \
47 container_of((ptr), struct aux_engine, base)
48 enum {
49 AUX_INVALID_REPLY_RETRY_COUNTER = 1,
50 AUX_TIMED_OUT_RETRY_COUNTER = 2,
51 AUX_DEFER_RETRY_COUNTER = 6
52 };
release_engine(struct aux_engine * engine)53 static void release_engine(
54 struct aux_engine *engine)
55 {
56 struct aux_engine_dce110 *aux110 = FROM_AUX_ENGINE(engine);
57
58 dal_ddc_close(engine->ddc);
59
60 engine->ddc = NULL;
61
62 REG_UPDATE(AUX_ARB_CONTROL, AUX_SW_DONE_USING_AUX_REG, 1);
63 }
64
65 #define SW_CAN_ACCESS_AUX 1
66 #define DMCU_CAN_ACCESS_AUX 2
67
is_engine_available(struct aux_engine * engine)68 static bool is_engine_available(
69 struct aux_engine *engine)
70 {
71 struct aux_engine_dce110 *aux110 = FROM_AUX_ENGINE(engine);
72
73 uint32_t value = REG_READ(AUX_ARB_CONTROL);
74 uint32_t field = get_reg_field_value(
75 value,
76 AUX_ARB_CONTROL,
77 AUX_REG_RW_CNTL_STATUS);
78
79 return (field != DMCU_CAN_ACCESS_AUX);
80 }
acquire_engine(struct aux_engine * engine)81 static bool acquire_engine(
82 struct aux_engine *engine)
83 {
84 struct aux_engine_dce110 *aux110 = FROM_AUX_ENGINE(engine);
85
86 uint32_t value = REG_READ(AUX_ARB_CONTROL);
87 uint32_t field = get_reg_field_value(
88 value,
89 AUX_ARB_CONTROL,
90 AUX_REG_RW_CNTL_STATUS);
91 if (field == DMCU_CAN_ACCESS_AUX)
92 return false;
93 /* enable AUX before request SW to access AUX */
94 value = REG_READ(AUX_CONTROL);
95 field = get_reg_field_value(value,
96 AUX_CONTROL,
97 AUX_EN);
98
99 if (field == 0) {
100 set_reg_field_value(
101 value,
102 1,
103 AUX_CONTROL,
104 AUX_EN);
105
106 if (REG(AUX_RESET_MASK)) {
107 /*DP_AUX block as part of the enable sequence*/
108 set_reg_field_value(
109 value,
110 1,
111 AUX_CONTROL,
112 AUX_RESET);
113 }
114
115 REG_WRITE(AUX_CONTROL, value);
116
117 if (REG(AUX_RESET_MASK)) {
118 /*poll HW to make sure reset it done*/
119
120 REG_WAIT(AUX_CONTROL, AUX_RESET_DONE, 1,
121 1, 11);
122
123 set_reg_field_value(
124 value,
125 0,
126 AUX_CONTROL,
127 AUX_RESET);
128
129 REG_WRITE(AUX_CONTROL, value);
130
131 REG_WAIT(AUX_CONTROL, AUX_RESET_DONE, 0,
132 1, 11);
133 }
134 } /*if (field)*/
135
136 /* request SW to access AUX */
137 REG_UPDATE(AUX_ARB_CONTROL, AUX_SW_USE_AUX_REG_REQ, 1);
138
139 value = REG_READ(AUX_ARB_CONTROL);
140 field = get_reg_field_value(
141 value,
142 AUX_ARB_CONTROL,
143 AUX_REG_RW_CNTL_STATUS);
144
145 return (field == SW_CAN_ACCESS_AUX);
146 }
147
148 #define COMPOSE_AUX_SW_DATA_16_20(command, address) \
149 ((command) | ((0xF0000 & (address)) >> 16))
150
151 #define COMPOSE_AUX_SW_DATA_8_15(address) \
152 ((0xFF00 & (address)) >> 8)
153
154 #define COMPOSE_AUX_SW_DATA_0_7(address) \
155 (0xFF & (address))
156
submit_channel_request(struct aux_engine * engine,struct aux_request_transaction_data * request)157 static void submit_channel_request(
158 struct aux_engine *engine,
159 struct aux_request_transaction_data *request)
160 {
161 struct aux_engine_dce110 *aux110 = FROM_AUX_ENGINE(engine);
162 uint32_t value;
163 uint32_t length;
164
165 bool is_write =
166 ((request->type == AUX_TRANSACTION_TYPE_DP) &&
167 (request->action == I2CAUX_TRANSACTION_ACTION_DP_WRITE)) ||
168 ((request->type == AUX_TRANSACTION_TYPE_I2C) &&
169 ((request->action == I2CAUX_TRANSACTION_ACTION_I2C_WRITE) ||
170 (request->action == I2CAUX_TRANSACTION_ACTION_I2C_WRITE_MOT)));
171 if (REG(AUXN_IMPCAL)) {
172 /* clear_aux_error */
173 REG_UPDATE_SEQ(AUXN_IMPCAL, AUXN_CALOUT_ERROR_AK,
174 1,
175 0);
176
177 REG_UPDATE_SEQ(AUXP_IMPCAL, AUXP_CALOUT_ERROR_AK,
178 1,
179 0);
180
181 /* force_default_calibrate */
182 REG_UPDATE_1BY1_2(AUXN_IMPCAL,
183 AUXN_IMPCAL_ENABLE, 1,
184 AUXN_IMPCAL_OVERRIDE_ENABLE, 0);
185
186 /* bug? why AUXN update EN and OVERRIDE_EN 1 by 1 while AUX P toggles OVERRIDE? */
187
188 REG_UPDATE_SEQ(AUXP_IMPCAL, AUXP_IMPCAL_OVERRIDE_ENABLE,
189 1,
190 0);
191 }
192 /* set the delay and the number of bytes to write */
193
194 /* The length include
195 * the 4 bit header and the 20 bit address
196 * (that is 3 byte).
197 * If the requested length is non zero this means
198 * an addition byte specifying the length is required.
199 */
200
201 length = request->length ? 4 : 3;
202 if (is_write)
203 length += request->length;
204
205 REG_UPDATE_2(AUX_SW_CONTROL,
206 AUX_SW_START_DELAY, request->delay,
207 AUX_SW_WR_BYTES, length);
208
209 /* program action and address and payload data (if 'is_write') */
210 value = REG_UPDATE_4(AUX_SW_DATA,
211 AUX_SW_INDEX, 0,
212 AUX_SW_DATA_RW, 0,
213 AUX_SW_AUTOINCREMENT_DISABLE, 1,
214 AUX_SW_DATA, COMPOSE_AUX_SW_DATA_16_20(request->action, request->address));
215
216 value = REG_SET_2(AUX_SW_DATA, value,
217 AUX_SW_AUTOINCREMENT_DISABLE, 0,
218 AUX_SW_DATA, COMPOSE_AUX_SW_DATA_8_15(request->address));
219
220 value = REG_SET(AUX_SW_DATA, value,
221 AUX_SW_DATA, COMPOSE_AUX_SW_DATA_0_7(request->address));
222
223 if (request->length) {
224 value = REG_SET(AUX_SW_DATA, value,
225 AUX_SW_DATA, request->length - 1);
226 }
227
228 if (is_write) {
229 /* Load the HW buffer with the Data to be sent.
230 * This is relevant for write operation.
231 * For read, the data recived data will be
232 * processed in process_channel_reply().
233 */
234 uint32_t i = 0;
235
236 while (i < request->length) {
237 value = REG_SET(AUX_SW_DATA, value,
238 AUX_SW_DATA, request->data[i]);
239
240 ++i;
241 }
242 }
243
244 REG_UPDATE(AUX_INTERRUPT_CONTROL, AUX_SW_DONE_ACK, 1);
245 REG_WAIT(AUX_SW_STATUS, AUX_SW_DONE, 0,
246 10, aux110->timeout_period/10);
247 REG_UPDATE(AUX_SW_CONTROL, AUX_SW_GO, 1);
248 }
249
read_channel_reply(struct aux_engine * engine,uint32_t size,uint8_t * buffer,uint8_t * reply_result,uint32_t * sw_status)250 static int read_channel_reply(struct aux_engine *engine, uint32_t size,
251 uint8_t *buffer, uint8_t *reply_result,
252 uint32_t *sw_status)
253 {
254 struct aux_engine_dce110 *aux110 = FROM_AUX_ENGINE(engine);
255 uint32_t bytes_replied;
256 uint32_t reply_result_32;
257
258 *sw_status = REG_GET(AUX_SW_STATUS, AUX_SW_REPLY_BYTE_COUNT,
259 &bytes_replied);
260
261 /* In case HPD is LOW, exit AUX transaction */
262 if ((*sw_status & AUX_SW_STATUS__AUX_SW_HPD_DISCON_MASK))
263 return -1;
264
265 /* Need at least the status byte */
266 if (!bytes_replied)
267 return -1;
268
269 REG_UPDATE_1BY1_3(AUX_SW_DATA,
270 AUX_SW_INDEX, 0,
271 AUX_SW_AUTOINCREMENT_DISABLE, 1,
272 AUX_SW_DATA_RW, 1);
273
274 REG_GET(AUX_SW_DATA, AUX_SW_DATA, &reply_result_32);
275 reply_result_32 = reply_result_32 >> 4;
276 *reply_result = (uint8_t)reply_result_32;
277
278 if (reply_result_32 == 0) { /* ACK */
279 uint32_t i = 0;
280
281 /* First byte was already used to get the command status */
282 --bytes_replied;
283
284 /* Do not overflow buffer */
285 if (bytes_replied > size)
286 return -1;
287
288 while (i < bytes_replied) {
289 uint32_t aux_sw_data_val;
290
291 REG_GET(AUX_SW_DATA, AUX_SW_DATA, &aux_sw_data_val);
292 buffer[i] = aux_sw_data_val;
293 ++i;
294 }
295
296 return i;
297 }
298
299 return 0;
300 }
301
process_channel_reply(struct aux_engine * engine,struct aux_reply_transaction_data * reply)302 static void process_channel_reply(
303 struct aux_engine *engine,
304 struct aux_reply_transaction_data *reply)
305 {
306 int bytes_replied;
307 uint8_t reply_result;
308 uint32_t sw_status;
309
310 bytes_replied = read_channel_reply(engine, reply->length, reply->data,
311 &reply_result, &sw_status);
312
313 /* in case HPD is LOW, exit AUX transaction */
314 if ((sw_status & AUX_SW_STATUS__AUX_SW_HPD_DISCON_MASK)) {
315 reply->status = AUX_CHANNEL_OPERATION_FAILED_HPD_DISCON;
316 return;
317 }
318
319 if (bytes_replied < 0) {
320 /* Need to handle an error case...
321 * Hopefully, upper layer function won't call this function if
322 * the number of bytes in the reply was 0, because there was
323 * surely an error that was asserted that should have been
324 * handled for hot plug case, this could happens
325 */
326 if (!(sw_status & AUX_SW_STATUS__AUX_SW_HPD_DISCON_MASK)) {
327 reply->status = AUX_TRANSACTION_REPLY_INVALID;
328 ASSERT_CRITICAL(false);
329 return;
330 }
331 } else {
332
333 switch (reply_result) {
334 case 0: /* ACK */
335 reply->status = AUX_TRANSACTION_REPLY_AUX_ACK;
336 break;
337 case 1: /* NACK */
338 reply->status = AUX_TRANSACTION_REPLY_AUX_NACK;
339 break;
340 case 2: /* DEFER */
341 reply->status = AUX_TRANSACTION_REPLY_AUX_DEFER;
342 break;
343 case 4: /* AUX ACK / I2C NACK */
344 reply->status = AUX_TRANSACTION_REPLY_I2C_NACK;
345 break;
346 case 8: /* AUX ACK / I2C DEFER */
347 reply->status = AUX_TRANSACTION_REPLY_I2C_DEFER;
348 break;
349 default:
350 reply->status = AUX_TRANSACTION_REPLY_INVALID;
351 }
352 }
353 }
354
get_channel_status(struct aux_engine * engine,uint8_t * returned_bytes)355 static enum aux_channel_operation_result get_channel_status(
356 struct aux_engine *engine,
357 uint8_t *returned_bytes)
358 {
359 struct aux_engine_dce110 *aux110 = FROM_AUX_ENGINE(engine);
360
361 uint32_t value;
362
363 if (returned_bytes == NULL) {
364 /*caller pass NULL pointer*/
365 ASSERT_CRITICAL(false);
366 return AUX_CHANNEL_OPERATION_FAILED_REASON_UNKNOWN;
367 }
368 *returned_bytes = 0;
369
370 /* poll to make sure that SW_DONE is asserted */
371 value = REG_WAIT(AUX_SW_STATUS, AUX_SW_DONE, 1,
372 10, aux110->timeout_period/10);
373
374 /* in case HPD is LOW, exit AUX transaction */
375 if ((value & AUX_SW_STATUS__AUX_SW_HPD_DISCON_MASK))
376 return AUX_CHANNEL_OPERATION_FAILED_HPD_DISCON;
377
378 /* Note that the following bits are set in 'status.bits'
379 * during CTS 4.2.1.2 (FW 3.3.1):
380 * AUX_SW_RX_MIN_COUNT_VIOL, AUX_SW_RX_INVALID_STOP,
381 * AUX_SW_RX_RECV_NO_DET, AUX_SW_RX_RECV_INVALID_H.
382 *
383 * AUX_SW_RX_MIN_COUNT_VIOL is an internal,
384 * HW debugging bit and should be ignored.
385 */
386 if (value & AUX_SW_STATUS__AUX_SW_DONE_MASK) {
387 if ((value & AUX_SW_STATUS__AUX_SW_RX_TIMEOUT_STATE_MASK) ||
388 (value & AUX_SW_STATUS__AUX_SW_RX_TIMEOUT_MASK))
389 return AUX_CHANNEL_OPERATION_FAILED_TIMEOUT;
390
391 else if ((value & AUX_SW_STATUS__AUX_SW_RX_INVALID_STOP_MASK) ||
392 (value & AUX_SW_STATUS__AUX_SW_RX_RECV_NO_DET_MASK) ||
393 (value &
394 AUX_SW_STATUS__AUX_SW_RX_RECV_INVALID_H_MASK) ||
395 (value & AUX_SW_STATUS__AUX_SW_RX_RECV_INVALID_L_MASK))
396 return AUX_CHANNEL_OPERATION_FAILED_INVALID_REPLY;
397
398 *returned_bytes = get_reg_field_value(value,
399 AUX_SW_STATUS,
400 AUX_SW_REPLY_BYTE_COUNT);
401
402 if (*returned_bytes == 0)
403 return
404 AUX_CHANNEL_OPERATION_FAILED_INVALID_REPLY;
405 else {
406 *returned_bytes -= 1;
407 return AUX_CHANNEL_OPERATION_SUCCEEDED;
408 }
409 } else {
410 /*time_elapsed >= aux_engine->timeout_period
411 * AUX_SW_STATUS__AUX_SW_HPD_DISCON = at this point
412 */
413 ASSERT_CRITICAL(false);
414 return AUX_CHANNEL_OPERATION_FAILED_TIMEOUT;
415 }
416 }
process_read_reply(struct aux_engine * engine,struct read_command_context * ctx)417 static void process_read_reply(
418 struct aux_engine *engine,
419 struct read_command_context *ctx)
420 {
421 engine->funcs->process_channel_reply(engine, &ctx->reply);
422
423 switch (ctx->reply.status) {
424 case AUX_TRANSACTION_REPLY_AUX_ACK:
425 ctx->defer_retry_aux = 0;
426 if (ctx->returned_byte > ctx->current_read_length) {
427 ctx->status =
428 I2CAUX_TRANSACTION_STATUS_FAILED_PROTOCOL_ERROR;
429 ctx->operation_succeeded = false;
430 } else if (ctx->returned_byte < ctx->current_read_length) {
431 ctx->current_read_length -= ctx->returned_byte;
432
433 ctx->offset += ctx->returned_byte;
434
435 ++ctx->invalid_reply_retry_aux_on_ack;
436
437 if (ctx->invalid_reply_retry_aux_on_ack >
438 AUX_INVALID_REPLY_RETRY_COUNTER) {
439 ctx->status =
440 I2CAUX_TRANSACTION_STATUS_FAILED_PROTOCOL_ERROR;
441 ctx->operation_succeeded = false;
442 }
443 } else {
444 ctx->status = I2CAUX_TRANSACTION_STATUS_SUCCEEDED;
445 ctx->transaction_complete = true;
446 ctx->operation_succeeded = true;
447 }
448 break;
449 case AUX_TRANSACTION_REPLY_AUX_NACK:
450 ctx->status = I2CAUX_TRANSACTION_STATUS_FAILED_NACK;
451 ctx->operation_succeeded = false;
452 break;
453 case AUX_TRANSACTION_REPLY_AUX_DEFER:
454 ++ctx->defer_retry_aux;
455
456 if (ctx->defer_retry_aux > AUX_DEFER_RETRY_COUNTER) {
457 ctx->status = I2CAUX_TRANSACTION_STATUS_FAILED_TIMEOUT;
458 ctx->operation_succeeded = false;
459 }
460 break;
461 case AUX_TRANSACTION_REPLY_I2C_DEFER:
462 ctx->defer_retry_aux = 0;
463
464 ++ctx->defer_retry_i2c;
465
466 if (ctx->defer_retry_i2c > AUX_DEFER_RETRY_COUNTER) {
467 ctx->status = I2CAUX_TRANSACTION_STATUS_FAILED_TIMEOUT;
468 ctx->operation_succeeded = false;
469 }
470 break;
471 case AUX_TRANSACTION_REPLY_HPD_DISCON:
472 ctx->status = I2CAUX_TRANSACTION_STATUS_FAILED_HPD_DISCON;
473 ctx->operation_succeeded = false;
474 break;
475 default:
476 ctx->status = I2CAUX_TRANSACTION_STATUS_UNKNOWN;
477 ctx->operation_succeeded = false;
478 }
479 }
process_read_request(struct aux_engine * engine,struct read_command_context * ctx)480 static void process_read_request(
481 struct aux_engine *engine,
482 struct read_command_context *ctx)
483 {
484 enum aux_channel_operation_result operation_result;
485
486 engine->funcs->submit_channel_request(engine, &ctx->request);
487
488 operation_result = engine->funcs->get_channel_status(
489 engine, &ctx->returned_byte);
490
491 switch (operation_result) {
492 case AUX_CHANNEL_OPERATION_SUCCEEDED:
493 if (ctx->returned_byte > ctx->current_read_length) {
494 ctx->status =
495 I2CAUX_TRANSACTION_STATUS_FAILED_PROTOCOL_ERROR;
496 ctx->operation_succeeded = false;
497 } else {
498 ctx->timed_out_retry_aux = 0;
499 ctx->invalid_reply_retry_aux = 0;
500
501 ctx->reply.length = ctx->returned_byte;
502 ctx->reply.data = ctx->buffer;
503
504 process_read_reply(engine, ctx);
505 }
506 break;
507 case AUX_CHANNEL_OPERATION_FAILED_INVALID_REPLY:
508 ++ctx->invalid_reply_retry_aux;
509
510 if (ctx->invalid_reply_retry_aux >
511 AUX_INVALID_REPLY_RETRY_COUNTER) {
512 ctx->status =
513 I2CAUX_TRANSACTION_STATUS_FAILED_PROTOCOL_ERROR;
514 ctx->operation_succeeded = false;
515 } else
516 udelay(400);
517 break;
518 case AUX_CHANNEL_OPERATION_FAILED_TIMEOUT:
519 ++ctx->timed_out_retry_aux;
520
521 if (ctx->timed_out_retry_aux > AUX_TIMED_OUT_RETRY_COUNTER) {
522 ctx->status = I2CAUX_TRANSACTION_STATUS_FAILED_TIMEOUT;
523 ctx->operation_succeeded = false;
524 } else {
525 /* DP 1.2a, table 2-58:
526 * "S3: AUX Request CMD PENDING:
527 * retry 3 times, with 400usec wait on each"
528 * The HW timeout is set to 550usec,
529 * so we should not wait here
530 */
531 }
532 break;
533 case AUX_CHANNEL_OPERATION_FAILED_HPD_DISCON:
534 ctx->status = I2CAUX_TRANSACTION_STATUS_FAILED_HPD_DISCON;
535 ctx->operation_succeeded = false;
536 break;
537 default:
538 ctx->status = I2CAUX_TRANSACTION_STATUS_UNKNOWN;
539 ctx->operation_succeeded = false;
540 }
541 }
read_command(struct aux_engine * engine,struct i2caux_transaction_request * request,bool middle_of_transaction)542 static bool read_command(
543 struct aux_engine *engine,
544 struct i2caux_transaction_request *request,
545 bool middle_of_transaction)
546 {
547 struct read_command_context ctx;
548
549 ctx.buffer = request->payload.data;
550 ctx.current_read_length = request->payload.length;
551 ctx.offset = 0;
552 ctx.timed_out_retry_aux = 0;
553 ctx.invalid_reply_retry_aux = 0;
554 ctx.defer_retry_aux = 0;
555 ctx.defer_retry_i2c = 0;
556 ctx.invalid_reply_retry_aux_on_ack = 0;
557 ctx.transaction_complete = false;
558 ctx.operation_succeeded = true;
559
560 if (request->payload.address_space ==
561 I2CAUX_TRANSACTION_ADDRESS_SPACE_DPCD) {
562 ctx.request.type = AUX_TRANSACTION_TYPE_DP;
563 ctx.request.action = I2CAUX_TRANSACTION_ACTION_DP_READ;
564 ctx.request.address = request->payload.address;
565 } else if (request->payload.address_space ==
566 I2CAUX_TRANSACTION_ADDRESS_SPACE_I2C) {
567 ctx.request.type = AUX_TRANSACTION_TYPE_I2C;
568 ctx.request.action = middle_of_transaction ?
569 I2CAUX_TRANSACTION_ACTION_I2C_READ_MOT :
570 I2CAUX_TRANSACTION_ACTION_I2C_READ;
571 ctx.request.address = request->payload.address >> 1;
572 } else {
573 /* in DAL2, there was no return in such case */
574 BREAK_TO_DEBUGGER();
575 return false;
576 }
577
578 ctx.request.delay = 0;
579
580 do {
581 memset(ctx.buffer + ctx.offset, 0, ctx.current_read_length);
582
583 ctx.request.data = ctx.buffer + ctx.offset;
584 ctx.request.length = ctx.current_read_length;
585
586 process_read_request(engine, &ctx);
587
588 request->status = ctx.status;
589
590 if (ctx.operation_succeeded && !ctx.transaction_complete)
591 if (ctx.request.type == AUX_TRANSACTION_TYPE_I2C)
592 msleep(engine->delay);
593 } while (ctx.operation_succeeded && !ctx.transaction_complete);
594
595 if (request->payload.address_space ==
596 I2CAUX_TRANSACTION_ADDRESS_SPACE_DPCD) {
597 DC_LOG_I2C_AUX("READ: addr:0x%x value:0x%x Result:%d",
598 request->payload.address,
599 request->payload.data[0],
600 ctx.operation_succeeded);
601 }
602
603 return ctx.operation_succeeded;
604 }
605
process_write_reply(struct aux_engine * engine,struct write_command_context * ctx)606 static void process_write_reply(
607 struct aux_engine *engine,
608 struct write_command_context *ctx)
609 {
610 engine->funcs->process_channel_reply(engine, &ctx->reply);
611
612 switch (ctx->reply.status) {
613 case AUX_TRANSACTION_REPLY_AUX_ACK:
614 ctx->operation_succeeded = true;
615
616 if (ctx->returned_byte) {
617 ctx->request.action = ctx->mot ?
618 I2CAUX_TRANSACTION_ACTION_I2C_STATUS_REQUEST_MOT :
619 I2CAUX_TRANSACTION_ACTION_I2C_STATUS_REQUEST;
620
621 ctx->current_write_length = 0;
622
623 ++ctx->ack_m_retry;
624
625 if (ctx->ack_m_retry > AUX_DEFER_RETRY_COUNTER) {
626 ctx->status =
627 I2CAUX_TRANSACTION_STATUS_FAILED_TIMEOUT;
628 ctx->operation_succeeded = false;
629 } else
630 udelay(300);
631 } else {
632 ctx->status = I2CAUX_TRANSACTION_STATUS_SUCCEEDED;
633 ctx->defer_retry_aux = 0;
634 ctx->ack_m_retry = 0;
635 ctx->transaction_complete = true;
636 }
637 break;
638 case AUX_TRANSACTION_REPLY_AUX_NACK:
639 ctx->status = I2CAUX_TRANSACTION_STATUS_FAILED_NACK;
640 ctx->operation_succeeded = false;
641 break;
642 case AUX_TRANSACTION_REPLY_AUX_DEFER:
643 ++ctx->defer_retry_aux;
644
645 if (ctx->defer_retry_aux > ctx->max_defer_retry) {
646 ctx->status = I2CAUX_TRANSACTION_STATUS_FAILED_TIMEOUT;
647 ctx->operation_succeeded = false;
648 }
649 break;
650 case AUX_TRANSACTION_REPLY_I2C_DEFER:
651 ctx->defer_retry_aux = 0;
652 ctx->current_write_length = 0;
653
654 ctx->request.action = ctx->mot ?
655 I2CAUX_TRANSACTION_ACTION_I2C_STATUS_REQUEST_MOT :
656 I2CAUX_TRANSACTION_ACTION_I2C_STATUS_REQUEST;
657
658 ++ctx->defer_retry_i2c;
659
660 if (ctx->defer_retry_i2c > ctx->max_defer_retry) {
661 ctx->status = I2CAUX_TRANSACTION_STATUS_FAILED_TIMEOUT;
662 ctx->operation_succeeded = false;
663 }
664 break;
665 case AUX_TRANSACTION_REPLY_HPD_DISCON:
666 ctx->status = I2CAUX_TRANSACTION_STATUS_FAILED_HPD_DISCON;
667 ctx->operation_succeeded = false;
668 break;
669 default:
670 ctx->status = I2CAUX_TRANSACTION_STATUS_UNKNOWN;
671 ctx->operation_succeeded = false;
672 }
673 }
process_write_request(struct aux_engine * engine,struct write_command_context * ctx)674 static void process_write_request(
675 struct aux_engine *engine,
676 struct write_command_context *ctx)
677 {
678 enum aux_channel_operation_result operation_result;
679
680 engine->funcs->submit_channel_request(engine, &ctx->request);
681
682 operation_result = engine->funcs->get_channel_status(
683 engine, &ctx->returned_byte);
684
685 switch (operation_result) {
686 case AUX_CHANNEL_OPERATION_SUCCEEDED:
687 ctx->timed_out_retry_aux = 0;
688 ctx->invalid_reply_retry_aux = 0;
689
690 ctx->reply.length = ctx->returned_byte;
691 ctx->reply.data = ctx->reply_data;
692
693 process_write_reply(engine, ctx);
694 break;
695 case AUX_CHANNEL_OPERATION_FAILED_INVALID_REPLY:
696 ++ctx->invalid_reply_retry_aux;
697
698 if (ctx->invalid_reply_retry_aux >
699 AUX_INVALID_REPLY_RETRY_COUNTER) {
700 ctx->status =
701 I2CAUX_TRANSACTION_STATUS_FAILED_PROTOCOL_ERROR;
702 ctx->operation_succeeded = false;
703 } else
704 udelay(400);
705 break;
706 case AUX_CHANNEL_OPERATION_FAILED_TIMEOUT:
707 ++ctx->timed_out_retry_aux;
708
709 if (ctx->timed_out_retry_aux > AUX_TIMED_OUT_RETRY_COUNTER) {
710 ctx->status = I2CAUX_TRANSACTION_STATUS_FAILED_TIMEOUT;
711 ctx->operation_succeeded = false;
712 } else {
713 /* DP 1.2a, table 2-58:
714 * "S3: AUX Request CMD PENDING:
715 * retry 3 times, with 400usec wait on each"
716 * The HW timeout is set to 550usec,
717 * so we should not wait here
718 */
719 }
720 break;
721 case AUX_CHANNEL_OPERATION_FAILED_HPD_DISCON:
722 ctx->status = I2CAUX_TRANSACTION_STATUS_FAILED_HPD_DISCON;
723 ctx->operation_succeeded = false;
724 break;
725 default:
726 ctx->status = I2CAUX_TRANSACTION_STATUS_UNKNOWN;
727 ctx->operation_succeeded = false;
728 }
729 }
write_command(struct aux_engine * engine,struct i2caux_transaction_request * request,bool middle_of_transaction)730 static bool write_command(
731 struct aux_engine *engine,
732 struct i2caux_transaction_request *request,
733 bool middle_of_transaction)
734 {
735 struct write_command_context ctx;
736
737 ctx.mot = middle_of_transaction;
738 ctx.buffer = request->payload.data;
739 ctx.current_write_length = request->payload.length;
740 ctx.timed_out_retry_aux = 0;
741 ctx.invalid_reply_retry_aux = 0;
742 ctx.defer_retry_aux = 0;
743 ctx.defer_retry_i2c = 0;
744 ctx.ack_m_retry = 0;
745 ctx.transaction_complete = false;
746 ctx.operation_succeeded = true;
747
748 if (request->payload.address_space ==
749 I2CAUX_TRANSACTION_ADDRESS_SPACE_DPCD) {
750 ctx.request.type = AUX_TRANSACTION_TYPE_DP;
751 ctx.request.action = I2CAUX_TRANSACTION_ACTION_DP_WRITE;
752 ctx.request.address = request->payload.address;
753 } else if (request->payload.address_space ==
754 I2CAUX_TRANSACTION_ADDRESS_SPACE_I2C) {
755 ctx.request.type = AUX_TRANSACTION_TYPE_I2C;
756 ctx.request.action = middle_of_transaction ?
757 I2CAUX_TRANSACTION_ACTION_I2C_WRITE_MOT :
758 I2CAUX_TRANSACTION_ACTION_I2C_WRITE;
759 ctx.request.address = request->payload.address >> 1;
760 } else {
761 /* in DAL2, there was no return in such case */
762 BREAK_TO_DEBUGGER();
763 return false;
764 }
765
766 ctx.request.delay = 0;
767
768 ctx.max_defer_retry =
769 (engine->max_defer_write_retry > AUX_DEFER_RETRY_COUNTER) ?
770 engine->max_defer_write_retry : AUX_DEFER_RETRY_COUNTER;
771
772 do {
773 ctx.request.data = ctx.buffer;
774 ctx.request.length = ctx.current_write_length;
775
776 process_write_request(engine, &ctx);
777
778 request->status = ctx.status;
779
780 if (ctx.operation_succeeded && !ctx.transaction_complete)
781 if (ctx.request.type == AUX_TRANSACTION_TYPE_I2C)
782 msleep(engine->delay);
783 } while (ctx.operation_succeeded && !ctx.transaction_complete);
784
785 if (request->payload.address_space ==
786 I2CAUX_TRANSACTION_ADDRESS_SPACE_DPCD) {
787 DC_LOG_I2C_AUX("WRITE: addr:0x%x value:0x%x Result:%d",
788 request->payload.address,
789 request->payload.data[0],
790 ctx.operation_succeeded);
791 }
792
793 return ctx.operation_succeeded;
794 }
end_of_transaction_command(struct aux_engine * engine,struct i2caux_transaction_request * request)795 static bool end_of_transaction_command(
796 struct aux_engine *engine,
797 struct i2caux_transaction_request *request)
798 {
799 struct i2caux_transaction_request dummy_request;
800 uint8_t dummy_data;
801
802 /* [tcheng] We only need to send the stop (read with MOT = 0)
803 * for I2C-over-Aux, not native AUX
804 */
805
806 if (request->payload.address_space !=
807 I2CAUX_TRANSACTION_ADDRESS_SPACE_I2C)
808 return false;
809
810 dummy_request.operation = request->operation;
811 dummy_request.payload.address_space = request->payload.address_space;
812 dummy_request.payload.address = request->payload.address;
813
814 /*
815 * Add a dummy byte due to some receiver quirk
816 * where one byte is sent along with MOT = 0.
817 * Ideally this should be 0.
818 */
819
820 dummy_request.payload.length = 0;
821 dummy_request.payload.data = &dummy_data;
822
823 if (request->operation == I2CAUX_TRANSACTION_READ)
824 return read_command(engine, &dummy_request, false);
825 else
826 return write_command(engine, &dummy_request, false);
827
828 /* according Syed, it does not need now DoDummyMOT */
829 }
submit_request(struct aux_engine * engine,struct i2caux_transaction_request * request,bool middle_of_transaction)830 static bool submit_request(
831 struct aux_engine *engine,
832 struct i2caux_transaction_request *request,
833 bool middle_of_transaction)
834 {
835
836 bool result;
837 bool mot_used = true;
838
839 switch (request->operation) {
840 case I2CAUX_TRANSACTION_READ:
841 result = read_command(engine, request, mot_used);
842 break;
843 case I2CAUX_TRANSACTION_WRITE:
844 result = write_command(engine, request, mot_used);
845 break;
846 default:
847 result = false;
848 }
849
850 /* [tcheng]
851 * need to send stop for the last transaction to free up the AUX
852 * if the above command fails, this would be the last transaction
853 */
854
855 if (!middle_of_transaction || !result)
856 end_of_transaction_command(engine, request);
857
858 /* mask AUX interrupt */
859
860 return result;
861 }
get_engine_type(const struct aux_engine * engine)862 enum i2caux_engine_type get_engine_type(
863 const struct aux_engine *engine)
864 {
865 return I2CAUX_ENGINE_TYPE_AUX;
866 }
867
acquire(struct aux_engine * engine,struct ddc * ddc)868 static bool acquire(
869 struct aux_engine *engine,
870 struct ddc *ddc)
871 {
872
873 enum gpio_result result;
874
875 if (engine->funcs->is_engine_available) {
876 /*check whether SW could use the engine*/
877 if (!engine->funcs->is_engine_available(engine))
878 return false;
879 }
880
881 result = dal_ddc_open(ddc, GPIO_MODE_HARDWARE,
882 GPIO_DDC_CONFIG_TYPE_MODE_AUX);
883
884 if (result != GPIO_RESULT_OK)
885 return false;
886
887 if (!engine->funcs->acquire_engine(engine)) {
888 dal_ddc_close(ddc);
889 return false;
890 }
891
892 engine->ddc = ddc;
893
894 return true;
895 }
896
897 static const struct aux_engine_funcs aux_engine_funcs = {
898 .acquire_engine = acquire_engine,
899 .submit_channel_request = submit_channel_request,
900 .process_channel_reply = process_channel_reply,
901 .read_channel_reply = read_channel_reply,
902 .get_channel_status = get_channel_status,
903 .is_engine_available = is_engine_available,
904 .release_engine = release_engine,
905 .destroy_engine = dce110_engine_destroy,
906 .submit_request = submit_request,
907 .get_engine_type = get_engine_type,
908 .acquire = acquire,
909 };
910
dce110_engine_destroy(struct aux_engine ** engine)911 void dce110_engine_destroy(struct aux_engine **engine)
912 {
913
914 struct aux_engine_dce110 *engine110 = FROM_AUX_ENGINE(*engine);
915
916 kfree(engine110);
917 *engine = NULL;
918
919 }
dce110_aux_engine_construct(struct aux_engine_dce110 * aux_engine110,struct dc_context * ctx,uint32_t inst,uint32_t timeout_period,const struct dce110_aux_registers * regs)920 struct aux_engine *dce110_aux_engine_construct(struct aux_engine_dce110 *aux_engine110,
921 struct dc_context *ctx,
922 uint32_t inst,
923 uint32_t timeout_period,
924 const struct dce110_aux_registers *regs)
925 {
926 aux_engine110->base.ddc = NULL;
927 aux_engine110->base.ctx = ctx;
928 aux_engine110->base.delay = 0;
929 aux_engine110->base.max_defer_write_retry = 0;
930 aux_engine110->base.funcs = &aux_engine_funcs;
931 aux_engine110->base.inst = inst;
932 aux_engine110->timeout_period = timeout_period;
933 aux_engine110->regs = regs;
934
935 return &aux_engine110->base;
936 }
937
938