summaryrefslogtreecommitdiff
path: root/common/usb_pd_protocol.c
blob: b5c64d83f802def3e7e880391a676cf7119c0da8 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
/* Copyright (c) 2014 The Chromium OS Authors. All rights reserved.
 * Use of this source code is governed by a BSD-style license that can be
 * found in the LICENSE file.
 */

#include "adc.h"
#include "board.h"
#include "common.h"
#include "console.h"
#include "crc.h"
#include "gpio.h"
#include "hooks.h"
#include "registers.h"
#include "task.h"
#include "timer.h"
#include "util.h"
#include "usb_pd.h"
#include "usb_pd_config.h"

#ifdef CONFIG_COMMON_RUNTIME
#define CPRINTF(format, args...) cprintf(CC_USBPD, format, ## args)

/* dump full packet on RX error */
static int debug_dump;
#else
#define CPRINTF(format, args...)
const int debug_dump;
#endif

/* Control Message type */
enum {
	/* 0 Reserved */
	PD_CTRL_GOOD_CRC = 1,
	PD_CTRL_GOTO_MIN = 2,
	PD_CTRL_ACCEPT = 3,
	PD_CTRL_REJECT = 4,
	PD_CTRL_PING = 5,
	PD_CTRL_PS_RDY = 6,
	PD_CTRL_GET_SOURCE_CAP = 7,
	PD_CTRL_GET_SINK_CAP = 8,
	PD_CTRL_PROTOCOL_ERR = 9,
	PD_CTRL_SWAP = 10,
	/* 11 Reserved */
	PD_CTRL_WAIT = 12,
	PD_CTRL_SOFT_RESET = 13,
	/* 14-15 Reserved */
};

/* Data message type */
enum {
	/* 0 Reserved */
	PD_DATA_SOURCE_CAP = 1,
	PD_DATA_REQUEST = 2,
	PD_DATA_BIST = 3,
	PD_DATA_SINK_CAP = 4,
	/* 5-14 Reserved */
	PD_DATA_VENDOR_DEF = 15,
};

/* Protocol revision */
#define PD_REV10 0

/* Port role */
#define PD_ROLE_SINK   0
#define PD_ROLE_SOURCE 1

/* build message header */
#define PD_HEADER(type, role, id, cnt) \
	((type) | (PD_REV10 << 6) | \
	 ((role) << 8) | ((id) << 9) | ((cnt) << 12))

#define PD_HEADER_CNT(header)  (((header) >> 12) & 7)
#define PD_HEADER_TYPE(header) ((header) & 0xF)
#define PD_HEADER_ID(header)   (((header) >> 9) & 7)

/* Encode 5 bits using Biphase Mark Coding */
#define BMC(x)   ((x &  1 ? 0x001 : 0x3FF) \
		^ (x &  2 ? 0x004 : 0x3FC) \
		^ (x &  4 ? 0x010 : 0x3F0) \
		^ (x &  8 ? 0x040 : 0x3C0) \
		^ (x & 16 ? 0x100 : 0x300))

/* 4b/5b + Bimark Phase encoding */
static const uint16_t bmc4b5b[] = {
/* 0 = 0000 */ BMC(0x1E) /* 11110 */,
/* 1 = 0001 */ BMC(0x09) /* 01001 */,
/* 2 = 0010 */ BMC(0x14) /* 10100 */,
/* 3 = 0011 */ BMC(0x15) /* 10101 */,
/* 4 = 0100 */ BMC(0x0A) /* 01010 */,
/* 5 = 0101 */ BMC(0x0B) /* 01011 */,
/* 6 = 0110 */ BMC(0x0E) /* 01110 */,
/* 7 = 0111 */ BMC(0x0F) /* 01111 */,
/* 8 = 1000 */ BMC(0x12) /* 10010 */,
/* 9 = 1001 */ BMC(0x13) /* 10011 */,
/* A = 1010 */ BMC(0x16) /* 10110 */,
/* B = 1011 */ BMC(0x17) /* 10111 */,
/* C = 1100 */ BMC(0x1A) /* 11010 */,
/* D = 1101 */ BMC(0x1B) /* 11011 */,
/* E = 1110 */ BMC(0x1C) /* 11100 */,
/* F = 1111 */ BMC(0x1D) /* 11101 */,
/* Sync-1      K-code       11000 Startsynch #1 */
/* Sync-2      K-code       10001 Startsynch #2 */
/* RST-1       K-code       00111 Hard Reset #1 */
/* RST-2       K-code       11001 Hard Reset #2 */
/* EOP         K-code       01101 EOP End Of Packet */
/* Reserved    Error        00000 */
/* Reserved    Error        00001 */
/* Reserved    Error        00010 */
/* Reserved    Error        00011 */
/* Reserved    Error        00100 */
/* Reserved    Error        00101 */
/* Reserved    Error        00110 */
/* Reserved    Error        01000 */
/* Reserved    Error        01100 */
/* Reserved    Error        10000 */
/* Reserved    Error        11111 */
};
#define PD_SYNC1 0x18
#define PD_SYNC2 0x11
#define PD_RST1  0x07
#define PD_RST2  0x19
#define PD_EOP   0x0D

static const uint8_t dec4b5b[] = {
/* Error    */ 0x10 /* 00000 */,
/* Error    */ 0x10 /* 00001 */,
/* Error    */ 0x10 /* 00010 */,
/* Error    */ 0x10 /* 00011 */,
/* Error    */ 0x10 /* 00100 */,
/* Error    */ 0x10 /* 00101 */,
/* Error    */ 0x10 /* 00110 */,
/* RST-1    */ 0x13 /* 00111 K-code: Hard Reset #1 */,
/* Error    */ 0x10 /* 01000 */,
/* 1 = 0001 */ 0x01 /* 01001 */,
/* 4 = 0100 */ 0x04 /* 01010 */,
/* 5 = 0101 */ 0x05 /* 01011 */,
/* Error    */ 0x10 /* 01100 */,
/* EOP      */ 0x15 /* 01101 K-code: EOP End Of Packet */,
/* 6 = 0110 */ 0x06 /* 01110 */,
/* 7 = 0111 */ 0x07 /* 01111 */,
/* Error    */ 0x10 /* 10000 */,
/* Sync-2   */ 0x12 /* 10001 K-code: Startsynch #2 */,
/* 8 = 1000 */ 0x08 /* 10010 */,
/* 9 = 1001 */ 0x09 /* 10011 */,
/* 2 = 0010 */ 0x02 /* 10100 */,
/* 3 = 0011 */ 0x03 /* 10101 */,
/* A = 1010 */ 0x0A /* 10110 */,
/* B = 1011 */ 0x0B /* 10111 */,
/* Sync-1   */ 0x11 /* 11000 K-code: Startsynch #1 */,
/* RST-2    */ 0x14 /* 11001 K-code: Hard Reset #2 */,
/* C = 1100 */ 0x0C /* 11010 */,
/* D = 1101 */ 0x0D /* 11011 */,
/* E = 1110 */ 0x0E /* 11100 */,
/* F = 1111 */ 0x0F /* 11101 */,
/* 0 = 0000 */ 0x00 /* 11110 */,
/* Error    */ 0x10 /* 11111 */,
};

/* Start of Packet sequence : three Sync-1 K-codes, then one Sync-2 K-code */
#define PD_SOP (PD_SYNC1 | (PD_SYNC1<<5) | (PD_SYNC1<<10) | (PD_SYNC2<<15))

/* Hard Reset sequence : three RST-1 K-codes, then one RST-2 K-code */
#define PD_HARD_RESET (PD_RST1 | (PD_RST1 << 5) |\
		      (PD_RST1 << 10) | (PD_RST2 << 15))

/* PD counter definitions */
#define PD_MESSAGE_ID_COUNT 7
#define PD_RETRY_COUNT 2
#define PD_HARD_RESET_COUNT 2
#define PD_CAPS_COUNT 50

/* Timers */
#define PD_T_SEND_SOURCE_CAP 1500000 /* us (between 1s and 2s) */
#define PD_T_GET_SOURCE_CAP  1500000 /* us (between 1s and 2s) */
#define PD_T_SOURCE_ACTIVITY   45000 /* us (between 40ms and 50ms) */

/* Port role at startup */
#ifdef CONFIG_USB_PD_DUAL_ROLE
#define PD_ROLE_DEFAULT PD_ROLE_SINK
#else
#define PD_ROLE_DEFAULT PD_ROLE_SOURCE
#endif

/* current port role */
static uint8_t pd_role = PD_ROLE_DEFAULT;
/* 3-bit rolling message ID counter */
static uint8_t pd_message_id;
/* Port polarity : 0 => CC1 is CC line, 1 => CC2 is CC line */
static uint8_t pd_polarity;

static enum {
	PD_STATE_DISABLED,
#ifdef CONFIG_USB_PD_DUAL_ROLE
	PD_STATE_SNK_DISCONNECTED,
	PD_STATE_SNK_DISCOVERY,
	PD_STATE_SNK_TRANSITION,
	PD_STATE_SNK_READY,
#endif /* CONFIG_USB_PD_DUAL_ROLE */

	PD_STATE_SRC_DISCONNECTED,
	PD_STATE_SRC_DISCOVERY,
	PD_STATE_SRC_NEGOCIATE,
	PD_STATE_SRC_ACCEPTED,
	PD_STATE_SRC_TRANSITION,
	PD_STATE_SRC_READY,

	PD_STATE_HARD_RESET,
	PD_STATE_BIST,
} pd_task_state = PD_DEFAULT_STATE;

/* increment message ID counter */
static void inc_id(void)
{
	pd_message_id = (pd_message_id + 1) & PD_MESSAGE_ID_COUNT;
}

static inline int encode_short(void *ctxt, int off, uint16_t val16)
{
	off = pd_write_sym(ctxt, off, bmc4b5b[(val16 >> 0) & 0xF]);
	off = pd_write_sym(ctxt, off, bmc4b5b[(val16 >> 4) & 0xF]);
	off = pd_write_sym(ctxt, off, bmc4b5b[(val16 >> 8) & 0xF]);
	return pd_write_sym(ctxt, off, bmc4b5b[(val16 >> 12) & 0xF]);
}

static inline int encode_word(void *ctxt, int off, uint32_t val32)
{
	off = encode_short(ctxt, off, (val32 >> 0) & 0xFFFF);
	return encode_short(ctxt, off, (val32 >> 16) & 0xFFFF);
}

/* prepare a 4b/5b-encoded PD message to send */
static int prepare_message(void *ctxt, uint16_t header, uint8_t cnt,
			   const uint32_t *data)
{
	int off, i;
	crc32_init();
	/* 64-bit preamble */
	off = pd_write_preamble(ctxt);
	/* Start Of Packet: 3x Sync-1 + 1x Sync-2 */
	off = pd_write_sym(ctxt, off, BMC(PD_SYNC1));
	off = pd_write_sym(ctxt, off, BMC(PD_SYNC1));
	off = pd_write_sym(ctxt, off, BMC(PD_SYNC1));
	off = pd_write_sym(ctxt, off, BMC(PD_SYNC2));
	/* header */
	off = encode_short(ctxt, off, header);
	crc32_hash16(header);
	/* data payload */
	for (i = 0; i < cnt; i++) {
		off = encode_word(ctxt, off, data[i]);
		crc32_hash32(data[i]);
	}
	/* CRC */
	off = encode_word(ctxt, off, crc32_result());
	/* End Of Packet */
	off = pd_write_sym(ctxt, off, BMC(PD_EOP));
	/* Ensure that we have a final edge */
	return pd_write_last_edge(ctxt, off);
}

static int analyze_rx(uint32_t *payload);

static void send_hard_reset(void *ctxt)
{
	int off;

	/* 64-bit preamble */
	off = pd_write_preamble(ctxt);
	/* Hard-Reset: 3x RST-1 + 1x RST-2 */
	off = pd_write_sym(ctxt, off, BMC(PD_RST1));
	off = pd_write_sym(ctxt, off, BMC(PD_RST1));
	off = pd_write_sym(ctxt, off, BMC(PD_RST1));
	off = pd_write_sym(ctxt, off, BMC(PD_RST2));
	/* Ensure that we have a final edge */
	off = pd_write_last_edge(ctxt, off);
	/* Transmit the packet */
	pd_start_tx(ctxt, off);
	pd_tx_done();
}

static int send_validate_message(void *ctxt, uint16_t header, uint8_t cnt,
				 const uint32_t *data)
{
	int r;
	static uint32_t payload[7];

	/* retry 3 times if we are not getting a valid answer */
	for (r = 0; r <= PD_RETRY_COUNT; r++) {
		int bit_len;
		uint16_t head;
		/* write the encoded packet in the transmission buffer */
		bit_len = prepare_message(ctxt, header, cnt, data);
		/* Transmit the packet */
		pd_start_tx(ctxt, bit_len);
		pd_tx_done();
		/* starting waiting for GoodCrc */
		pd_rx_start();
		/* read the incoming packet if any */
		head = analyze_rx(payload);
		pd_rx_complete();
		if (head > 0) { /* we got a good packet, analyze it */
			int type = PD_HEADER_TYPE(head);
			int nb = PD_HEADER_CNT(head);
			uint8_t id = PD_HEADER_ID(head);
			if (type == PD_CTRL_GOOD_CRC && nb == 0 &&
			   id == pd_message_id) {
				/* got the GoodCRC we were expecting */
				inc_id();
				/* do not catch last edges as a new packet */
				udelay(10);
				return bit_len;
			} else {
				/* CPRINTF("ERR ACK/%d %04x\n", id, head); */
			}
		}
	}
	/* we failed all the re-transmissions */
	/* TODO: try HardReset */
	CPRINTF("TX NO ACK %04x/%d\n", header, cnt);
	return -1;
}

static int send_control(void *ctxt, int type)
{
	int bit_len;
	uint16_t header = PD_HEADER(type, pd_role, pd_message_id, 0);

	bit_len = send_validate_message(ctxt, header, 0, NULL);

	CPRINTF("CTRL[%d]>%d\n", type, bit_len);

	return bit_len;
}

static void send_goodcrc(void *ctxt, int id)
{
	uint16_t header = PD_HEADER(PD_CTRL_GOOD_CRC, pd_role, id, 0);
	int bit_len = prepare_message(ctxt, header, 0, NULL);

	pd_start_tx(ctxt, bit_len);
	pd_tx_done();
}

static int send_source_cap(void *ctxt)
{
	int bit_len;
	uint16_t header = PD_HEADER(PD_DATA_SOURCE_CAP, pd_role, pd_message_id,
				    pd_src_pdo_cnt);

	bit_len = send_validate_message(ctxt, header, pd_src_pdo_cnt,
					pd_src_pdo);
	CPRINTF("srcCAP>%d\n", bit_len);

	return bit_len;
}

#ifdef CONFIG_USB_PD_DUAL_ROLE
static void send_sink_cap(void *ctxt)
{
	int bit_len;
	uint16_t header = PD_HEADER(PD_DATA_SINK_CAP, pd_role, pd_message_id,
				    pd_snk_pdo_cnt);

	bit_len = send_validate_message(ctxt, header, pd_snk_pdo_cnt,
					pd_snk_pdo);
	CPRINTF("snkCAP>%d\n", bit_len);
}

static void send_request(void *ctxt, uint32_t rdo)
{
	int bit_len;
	uint16_t header = PD_HEADER(PD_DATA_REQUEST, pd_role, pd_message_id, 1);

	bit_len = send_validate_message(ctxt, header, 1, &rdo);
	CPRINTF("REQ%d>\n", bit_len);
}
#endif /* CONFIG_USB_PD_DUAL_ROLE */

static int send_bist(void *ctxt)
{
	uint32_t bdo = BDO(BDO_MODE_TRANSMIT, 0);
	int bit_len;
	uint16_t header = PD_HEADER(PD_DATA_BIST, pd_role, pd_message_id, 1);

	bit_len = send_validate_message(ctxt, header, 1, &bdo);
	CPRINTF("BIST>%d\n", bit_len);

	return bit_len;
}

static void handle_vdm_request(void *ctxt, int cnt, uint32_t *payload)
{
	CPRINTF("Unhandled VDM VID %04x CMD %04x\n", payload[0] >> 16,
		payload[0] & 0xFFFF);
}

static void handle_data_request(void *ctxt, uint16_t head, uint32_t *payload)
{
	int type = PD_HEADER_TYPE(head);
	int cnt = PD_HEADER_CNT(head);

	switch (type) {
#ifdef CONFIG_USB_PD_DUAL_ROLE
	case PD_DATA_SOURCE_CAP:
		if ((pd_task_state == PD_STATE_SNK_DISCOVERY)
			|| (pd_task_state == PD_STATE_SNK_TRANSITION)) {
			uint32_t rdo;
			int res;
			/* we were waiting for them, let's process them */
			res = pd_choose_voltage(cnt, payload, &rdo);
			if (res >= 0) {
				send_request(ctxt, rdo);
				pd_task_state = PD_STATE_SNK_TRANSITION;
			}
		}
		break;
#endif /* CONFIG_USB_PD_DUAL_ROLE */
	case PD_DATA_REQUEST:
		if ((pd_role == PD_ROLE_SOURCE) && (cnt == 1))
			if (!pd_request_voltage(payload[0])) {
				send_control(ctxt, PD_CTRL_ACCEPT);
				pd_task_state = PD_STATE_SRC_ACCEPTED;
				return;
			}
		/* the message was incorrect or cannot be satisfied */
		send_control(ctxt, PD_CTRL_REJECT);
		break;
	case PD_DATA_BIST:
		CPRINTF("BIST not supported\n");
		break;
	case PD_DATA_SINK_CAP:
		break;
	case PD_DATA_VENDOR_DEF:
		handle_vdm_request(ctxt, cnt, payload);
		break;
	default:
		CPRINTF("Unhandled data message type %d\n", type);
	}
}

static void handle_ctrl_request(void *ctxt, uint16_t head, uint32_t *payload)
{
	int type = PD_HEADER_TYPE(head);

	switch (type) {
	case PD_CTRL_GOOD_CRC:
		/* should not get it */
		break;
	case PD_CTRL_PING:
		/* Nothing else to do */
		break;
	case PD_CTRL_GET_SOURCE_CAP:
		send_source_cap(ctxt);
		break;
#ifdef CONFIG_USB_PD_DUAL_ROLE
	case PD_CTRL_GET_SINK_CAP:
		send_sink_cap(ctxt);
		break;
	case PD_CTRL_GOTO_MIN:
		break;
	case PD_CTRL_PS_RDY:
		if (pd_role == PD_ROLE_SINK)
			pd_task_state = PD_STATE_SNK_READY;
		break;
#endif /* CONFIG_USB_PD_DUAL_ROLE */
	case PD_CTRL_ACCEPT:
		break;
	case PD_CTRL_REJECT:
		break;
	case PD_CTRL_PROTOCOL_ERR:
	case PD_CTRL_SWAP:
	case PD_CTRL_WAIT:
	case PD_CTRL_SOFT_RESET:
	default:
		CPRINTF("Unhandled ctrl message type %d\n", type);
	}
}

static void handle_request(void *ctxt, uint16_t head, uint32_t *payload)
{
	int cnt = PD_HEADER_CNT(head);
	int p;

	if (PD_HEADER_TYPE(head) != 1 || cnt)
		send_goodcrc(ctxt, PD_HEADER_ID(head));

	/* dump received packet content */
	CPRINTF("RECV %04x/%d ", head, cnt);
	for (p = 0; p < cnt; p++)
		CPRINTF("[%d]%08x ", p, payload[p]);
	CPRINTF("\n");

	if (cnt)
		handle_data_request(ctxt, head, payload);
	else
		handle_ctrl_request(ctxt, head, payload);
}

static inline int decode_short(void *ctxt, int off, uint16_t *val16)
{
	uint32_t w;
	int end;

	end = pd_dequeue_bits(ctxt, off, 20, &w);

#if 0 /* DEBUG */
	CPRINTF("%d-%d: %05x %x:%x:%x:%x\n",
		off, end, w,
		dec4b5b[(w >> 15) & 0x1f], dec4b5b[(w >> 10) & 0x1f],
		dec4b5b[(w >>  5) & 0x1f], dec4b5b[(w >>  0) & 0x1f]);
#endif
	*val16 = dec4b5b[w & 0x1f] |
		(dec4b5b[(w >>  5) & 0x1f] << 4) |
		(dec4b5b[(w >> 10) & 0x1f] << 8) |
		(dec4b5b[(w >> 15) & 0x1f] << 12);
	return end;
}

static inline int decode_word(void *ctxt, int off, uint32_t *val32)
{
	off = decode_short(ctxt, off, (uint16_t *)val32);
	return decode_short(ctxt, off, ((uint16_t *)val32 + 1));
}

static int analyze_rx(uint32_t *payload)
{
	int bit;
	char *msg = "---";
	uint32_t val = 0;
	uint16_t header;
	uint32_t pcrc, ccrc;
	int p, cnt;
	/* uint32_t eop; */
	void *ctxt;

	crc32_init();
	ctxt = pd_init_dequeue();

	/* Detect preamble */
	bit = pd_find_preamble(ctxt);
	if (bit < 0) {
		msg = "Preamble";
		goto packet_err;
	}

	/* Find the Start Of Packet sequence */
	while (bit > 0) {
		bit = pd_dequeue_bits(ctxt, bit, 20, &val);
		if (val == PD_SOP)
			break;
		/* TODO: detect SOP with 1 error code */
		/* TODO: detect Hard reset */
	}
	if (bit < 0) {
		msg = "SOP";
		goto packet_err;
	}

	/* read header */
	bit = decode_short(ctxt, bit, &header);
	crc32_hash16(header);
	cnt = PD_HEADER_CNT(header);

	/* read payload data */
	for (p = 0; p < cnt && bit > 0; p++) {
		bit = decode_word(ctxt, bit, payload+p);
		crc32_hash32(payload[p]);
	}
	if (bit < 0) {
		msg = "len";
		goto packet_err;
	}

	/* check transmitted CRC */
	bit = decode_word(ctxt, bit, &pcrc);
	ccrc = crc32_result();
	if (bit < 0 || pcrc != ccrc) {
		msg = "CRC";
		if (pcrc != ccrc)
			bit = PD_ERR_CRC;
		/* DEBUG */CPRINTF("CRC %08x <> %08x\n", pcrc, crc32_result());
		goto packet_err;
	}

	/* check End Of Packet */
	/* SKIP EOP for now
	bit = pd_dequeue_bits(ctxt, bit, 5, &eop);
	if (bit < 0 || eop != PD_EOP) {
		msg = "EOP";
		goto packet_err;
	}
	*/

	return header;
packet_err:
	if (debug_dump)
		pd_dump_packet(ctxt, msg);
	else
		CPRINTF("RX ERR (%d)\n", bit);
	return bit;
}

static void execute_hard_reset(void)
{
	pd_message_id = 0;
#ifdef CONFIG_USB_PD_DUAL_ROLE
	pd_task_state = pd_role == PD_ROLE_SINK ? PD_STATE_SNK_DISCONNECTED
						: PD_STATE_SRC_DISCONNECTED;
#else
	pd_task_state = PD_STATE_SRC_DISCONNECTED;
#endif
	pd_power_supply_reset();
	CPRINTF("HARD RESET!\n");
}

void pd_task(void)
{
	int head;
	void *ctxt = pd_hw_init();
	uint32_t payload[7];
	int timeout = 10000;
	uint32_t evt;
	int cc1_volt, cc2_volt;
	int res;

	/* Ensure the power supply is in the default state */
	pd_power_supply_reset();

	while (1) {
		/* monitor for incoming packet */
		pd_rx_enable_monitoring();
		/* Verify board specific health status : current, voltages... */
		pd_board_checks();
		/* wait for next event/packet or timeout expiration */
		evt = task_wait_event(timeout);
		/* incoming packet ? */
		if (evt & PD_EVENT_RX) {
			head = analyze_rx(payload);
			pd_rx_complete();
			if (head > 0)
				handle_request(ctxt, head, payload);
			else if (head == PD_ERR_HARD_RESET)
				execute_hard_reset();
		}
		/* if nothing to do, verify the state of the world in 500ms */
		timeout = 500*MSEC;
		switch (pd_task_state) {
		case PD_STATE_DISABLED:
			/* Nothing to do */
			break;
		case PD_STATE_SRC_DISCONNECTED:
			/* Vnc monitoring */
			cc1_volt = adc_read_channel(ADC_CH_CC1_PD);
			cc2_volt = adc_read_channel(ADC_CH_CC2_PD);
			if ((cc1_volt < PD_SRC_VNC) ||
			    (cc2_volt < PD_SRC_VNC)) {
				pd_polarity = !(cc1_volt < PD_SRC_VNC);
				pd_task_state = PD_STATE_SRC_DISCOVERY;
			}
			timeout = 10000;
			break;
		case PD_STATE_SRC_DISCOVERY:
			/* Query capabilites of the other side */
			res = send_source_cap(ctxt);
			/* packet was acked => PD capable device) */
			if (res >= 0) {
				pd_task_state = PD_STATE_SRC_NEGOCIATE;
			} else { /* failed, retry later */
				timeout = PD_T_SEND_SOURCE_CAP;
			}
			break;
		case PD_STATE_SRC_NEGOCIATE:
			/* wait for a "Request" message */
			break;
		case PD_STATE_SRC_ACCEPTED:
			/* Accept sent, wait for the end of transition */
			timeout = PD_POWER_SUPPLY_TRANSITION_DELAY;
			pd_task_state = PD_STATE_SRC_TRANSITION;
			break;
		case PD_STATE_SRC_TRANSITION:
			res = pd_set_power_supply_ready();
			/* TODO error fallback */
			/* the voltage output is good, notify the source */
			res = send_control(ctxt, PD_CTRL_PS_RDY);
			if (res >= 0) {
				timeout =  PD_T_SEND_SOURCE_CAP;
				/* it'a time to ping regularly the sink */
				pd_task_state = PD_STATE_SRC_READY;
			}
			/* TODO error fallback */
			break;
		case PD_STATE_SRC_READY:
			/* Verify that the sink is alive */
			res = send_control(ctxt, PD_CTRL_PING);
			if (res < 0) {
				/* The sink died ... TODO */
				pd_task_state = PD_STATE_SRC_DISCOVERY;
				timeout = PD_T_SEND_SOURCE_CAP;
			} else { /* schedule next keep-alive */
				timeout = PD_T_SOURCE_ACTIVITY;
			}
			break;
#ifdef CONFIG_USB_PD_DUAL_ROLE
		case PD_STATE_SNK_DISCONNECTED:
			/* Source connection monitoring */
			cc1_volt = adc_read_channel(ADC_CH_CC1_PD);
			cc2_volt = adc_read_channel(ADC_CH_CC2_PD);
			if ((cc1_volt > PD_SNK_VA) ||
			    (cc2_volt > PD_SNK_VA)) {
				pd_polarity = !(cc1_volt > PD_SNK_VA);
				pd_task_state = PD_STATE_SNK_DISCOVERY;
			}
			timeout = 10000;
			break;
		case PD_STATE_SNK_DISCOVERY:
			res = send_control(ctxt, PD_CTRL_GET_SOURCE_CAP);
			/* packet was acked => PD capable device) */
			if (res >= 0) {
				pd_task_state = PD_STATE_SNK_TRANSITION;
			} else { /* failed, retry later */
				timeout = PD_T_GET_SOURCE_CAP;
			}
			break;
		case PD_STATE_SNK_TRANSITION:
			break;
		case PD_STATE_SNK_READY:
			/* we have power and we are happy */
			/* check vital parameters from time to time */
			timeout = 100*MSEC;
			break;
#endif /* CONFIG_USB_PD_DUAL_ROLE */
		case PD_STATE_HARD_RESET:
			send_hard_reset(ctxt);
			/* reset our own state machine */
			execute_hard_reset();
			break;
		case PD_STATE_BIST:
			send_bist(ctxt);
			pd_task_state = PD_STATE_DISABLED;
			break;
		}
	}
}

void pd_rx_event(void)
{
	task_set_event(TASK_ID_PD, PD_EVENT_RX, 0);
}

#ifdef CONFIG_COMMON_RUNTIME
void pd_request_source_voltage(int mv)
{
	pd_set_max_voltage(mv);
	pd_role = PD_ROLE_SINK;
	pd_set_host_mode(0);
	pd_task_state = PD_STATE_SNK_DISCONNECTED;
	task_wake(TASK_ID_PD);
}

static int command_pd(int argc, char **argv)
{
	if (argc < 2)
		return EC_ERROR_PARAM1;

	if (!strcasecmp(argv[1], "tx")) {
		pd_task_state = PD_STATE_SNK_DISCOVERY;
		task_wake(TASK_ID_PD);
	} else if (!strcasecmp(argv[1], "rx")) {
		pd_rx_event();
	} else if (!strcasecmp(argv[1], "bist")) {
		pd_task_state = PD_STATE_BIST;
		task_wake(TASK_ID_PD);
	} else if (!strcasecmp(argv[1], "charger")) {
		pd_role = PD_ROLE_SOURCE;
		pd_set_host_mode(1);
		pd_task_state = PD_STATE_SRC_DISCONNECTED;
		task_wake(TASK_ID_PD);
	} else if (!strncasecmp(argv[1], "dev", 3)) {
		int max_volt = -1;
		if (argc >= 3) {
			char *e;
			max_volt = strtoi(argv[2], &e, 10) * 1000;
		}
		pd_request_source_voltage(max_volt);
	} else if (!strcasecmp(argv[1], "clock")) {
		int freq;
		char *e;

		if (argc < 3)
			return EC_ERROR_PARAM2;

		freq = strtoi(argv[2], &e, 10);
		if (*e)
			return EC_ERROR_PARAM2;
		pd_set_clock(freq);
		ccprintf("set TX frequency to %d Hz\n", freq);
	} else if (!strcasecmp(argv[1], "dump")) {
		debug_dump = !debug_dump;
	} else if (!strncasecmp(argv[1], "hard", 4)) {
		pd_task_state = PD_STATE_HARD_RESET;
		task_wake(TASK_ID_PD);
	} else if (!strncasecmp(argv[1], "ping", 4)) {
		pd_role = PD_ROLE_SOURCE;
		pd_set_host_mode(1);
		pd_task_state = PD_STATE_SRC_READY;
		task_wake(TASK_ID_PD);
	} else if (!strncasecmp(argv[1], "state", 5)) {
		const char * const state_names[] = {
			"DISABLED",
			"SNK_DISCONNECTED", "SNK_DISCOVERY", "SNK_TRANSITION",
			"SNK_READY",
			"SRC_DISCONNECTED", "SRC_DISCOVERY", "SRC_NEGOCIATE",
			"SRC_READY",
			"HARD_RESET", "BIST",
		};
		ccprintf("Role: %s Polarity: CC%d State: %s\n",
			pd_role == PD_ROLE_SOURCE ? "SRC" : "SNK",
			pd_polarity + 1, state_names[pd_task_state]);
	} else {
		return EC_ERROR_PARAM1;
	}

	return EC_SUCCESS;
}
DECLARE_CONSOLE_COMMAND(pd, command_pd,
			"[rx|tx|hardreset|clock|connect]",
			"USB PD",
			NULL);
#endif /* CONFIG_COMMON_RUNTIME */