00001
00002
00003
00004
00005
00006
00007
00008
00009
00010
00011
00012
00013
00014
00015
00016
00017
00018
00019
00020
00021
00022
#include "kdp.h"
00023
00024 ULONG
00025
KdpComputeChecksum (
00026 IN PUCHAR Buffer,
00027 IN ULONG Length
00028 );
00029
00030 ULONG
00031
KdpReceiveString (
00032 OUT PCHAR Destination,
00033 IN ULONG Length
00034 );
00035
00036
VOID
00037
KdpSendString (
00038 IN PCHAR Source,
00039 IN ULONG Length
00040 );
00041
00042
VOID
00043
KdpSendControlPacket (
00044 IN USHORT PacketType,
00045 IN ULONG PacketId OPTIONAL
00046 );
00047
00048
#ifdef ALLOC_PRAGMA
00049
#pragma alloc_text(PAGEKD, KdpComputeChecksum)
00050
#pragma alloc_text(PAGEKD, KdpReceivePacketLeader)
00051
#pragma alloc_text(PAGEKD, KdpReceiveString)
00052
#pragma alloc_text(PAGEKD, KdpSendString)
00053
#pragma alloc_text(PAGEKD, KdpSendControlPacket)
00054
#pragma alloc_text(PAGEKD, KdpReceivePacket)
00055
#pragma alloc_text(PAGEKD, KdpSendPacket)
00056
#endif
00057
00058
00059 ULONG
00060
KdpComputeChecksum (
00061 IN PUCHAR Buffer,
00062 IN ULONG Length
00063 )
00064
00065
00066
00067
00068
00069
00070
00071
00072
00073
00074
00075
00076
00077
00078
00079
00080
00081
00082
00083 {
00084
00085 ULONG Checksum = 0;
00086
00087
while (Length > 0) {
00088 Checksum = Checksum + (ULONG)*
Buffer++;
00089 Length--;
00090 }
00091
return Checksum;
00092 }
00093
00094
USHORT
00095 KdpReceivePacketLeader (
00096 IN ULONG PacketType,
00097 OUT PULONG PacketLeader
00098 )
00099
00100
00101
00102
00103
00104
00105
00106
00107
00108
00109
00110
00111
00112
00113
00114
00115
00116
00117
00118
00119
00120
00121 {
00122
00123 UCHAR Input, PreviousByte = 0;
00124 ULONG PacketId = 0;
00125 ULONG
Index;
00126 ULONG ReturnCode;
00127 BOOLEAN BreakinDetected =
FALSE;
00128
00129
00130
00131
00132
00133
00134
00135
00136
00137
Index = 0;
00138
do {
00139 ReturnCode =
KdPortGetByte(&Input);
00140
if (ReturnCode ==
CP_GET_NODATA) {
00141
if (BreakinDetected) {
00142
KdpControlCPending =
TRUE;
00143
return KDP_PACKET_RESEND;
00144 }
else {
00145
return KDP_PACKET_TIMEOUT;
00146 }
00147 }
else if (ReturnCode ==
CP_GET_ERROR) {
00148
Index = 0;
00149
continue;
00150 }
else {
00151
if ( Input == PACKET_LEADER_BYTE ||
00152 Input == CONTROL_PACKET_LEADER_BYTE ) {
00153
if (
Index == 0 ) {
00154 PreviousByte = Input;
00155
Index++;
00156 }
else if (Input == PreviousByte ) {
00157
Index++;
00158 }
else {
00159 PreviousByte = Input;
00160
Index = 1;
00161 }
00162 }
else {
00163
00164
00165
00166
00167
00168
00169
00170
00171
00172
00173
00174
00175
if ( Input == BREAKIN_PACKET_BYTE ) {
00176 BreakinDetected =
TRUE;
00177 }
else {
00178
00179
00180
00181
00182
00183 BreakinDetected =
FALSE;
00184 }
00185
Index = 0;
00186 }
00187 }
00188 }
while (
Index < 4 );
00189
00190
if (BreakinDetected) {
00191
KdpControlCPending =
TRUE;
00192 }
00193
00194
00195
00196
00197
00198
if ( Input == PACKET_LEADER_BYTE ) {
00199 *PacketLeader = PACKET_LEADER;
00200 }
else {
00201 *PacketLeader = CONTROL_PACKET_LEADER;
00202 }
00203
00204
KdDebuggerNotPresent =
FALSE;
00205 SharedUserData->KdDebuggerEnabled |= 0x00000002;
00206
return KDP_PACKET_RECEIVED;
00207 }
00208
00209 ULONG
00210
KdpReceiveString (
00211 OUT PCHAR Destination,
00212 IN ULONG Length
00213 )
00214
00215
00216
00217
00218
00219
00220
00221
00222
00223
00224
00225
00226
00227
00228
00229
00230
00231
00232
00233
00234
00235
00236 {
00237
00238 UCHAR Input;
00239 ULONG ReturnCode;
00240
00241
00242
00243
00244
00245
while (Length > 0) {
00246 ReturnCode =
KdPortGetByte(&Input);
00247
if (ReturnCode !=
CP_GET_SUCCESS) {
00248
return ReturnCode;
00249 }
else {
00250 *Destination++ = Input;
00251 Length -= 1;
00252 }
00253 }
00254
return CP_GET_SUCCESS;
00255 }
00256
00257
VOID
00258
KdpSendString (
00259 IN PCHAR Source,
00260 IN ULONG Length
00261 )
00262
00263
00264
00265
00266
00267
00268
00269
00270
00271
00272
00273
00274
00275
00276
00277
00278
00279
00280
00281 {
00282
00283 UCHAR Output;
00284
00285
00286
00287
00288
00289
while (Length > 0) {
00290 Output = *Source++;
00291
KdPortPutByte(Output);
00292 Length -= 1;
00293 }
00294
return;
00295 }
00296
00297
VOID
00298
KdpSendControlPacket (
00299 IN USHORT PacketType,
00300 IN ULONG PacketId OPTIONAL
00301 )
00302
00303
00304
00305
00306
00307
00308
00309
00310
00311
00312
00313
00314
00315
00316
00317
00318
00319
00320
00321
00322 {
00323
00324 KD_PACKET PacketHeader;
00325
00326
00327
00328
00329
00330 PacketHeader.PacketLeader = CONTROL_PACKET_LEADER;
00331
if (ARGUMENT_PRESENT( (PVOID)(ULONG_PTR) PacketId )) {
00332 PacketHeader.PacketId = PacketId;
00333 }
00334 PacketHeader.ByteCount = 0;
00335 PacketHeader.Checksum = 0;
00336 PacketHeader.PacketType = PacketType;
00337
KdpSendString((PCHAR)&PacketHeader,
sizeof(KD_PACKET));
00338
00339
return;
00340 }
00341
00342 ULONG
00343 KdpReceivePacket (
00344 IN ULONG PacketType,
00345 OUT PSTRING MessageHeader,
00346 OUT PSTRING MessageData,
00347 OUT PULONG DataLength
00348 )
00349
00350
00351
00352
00353
00354
00355
00356
00357
00358
00359
00360
00361
00362
00363
00364
00365
00366
00367
00368
00369
00370
00371
00372
00373
00374
00375
00376
00377
00378
00379
00380
00381
00382 {
00383
00384 UCHAR Input;
00385 ULONG MessageLength;
00386 KD_PACKET PacketHeader;
00387 ULONG ReturnCode;
00388 ULONG Checksum;
00389
00390 WaitForPacketLeader:
00391
00392
00393
00394
00395
00396 ReturnCode =
KdpReceivePacketLeader(PacketType, &PacketHeader.PacketLeader);
00397
00398
00399
00400
00401
00402
00403
if (ReturnCode !=
KDP_PACKET_TIMEOUT) {
00404
KdpNumberRetries =
KdpRetryCount;
00405 }
00406
if (ReturnCode !=
KDP_PACKET_RECEIVED) {
00407
return ReturnCode;
00408 }
00409
00410
00411
00412
00413
00414 ReturnCode =
KdpReceiveString((PCHAR)&PacketHeader.PacketType,
00415
sizeof(PacketHeader.PacketType));
00416
if (ReturnCode ==
CP_GET_NODATA) {
00417
return KDP_PACKET_TIMEOUT;
00418 }
else if (ReturnCode ==
CP_GET_ERROR) {
00419
if (PacketHeader.PacketLeader == CONTROL_PACKET_LEADER) {
00420
00421
00422
00423
00424
00425
00426
00427
00428
goto WaitForPacketLeader;
00429 }
else {
00430
00431
00432
00433
00434
00435
00436
goto SendResendPacket;
00437 }
00438 }
00439
00440
00441
00442
00443
00444
00445
if ( PacketHeader.PacketLeader == CONTROL_PACKET_LEADER &&
00446 PacketHeader.PacketType == PACKET_TYPE_KD_RESEND ) {
00447
return KDP_PACKET_RESEND;
00448 }
00449
00450
00451
00452
00453
00454 ReturnCode =
KdpReceiveString((PCHAR)&PacketHeader.ByteCount,
00455
sizeof(PacketHeader.ByteCount));
00456
if (ReturnCode ==
CP_GET_NODATA) {
00457
return KDP_PACKET_TIMEOUT;
00458 }
else if (ReturnCode ==
CP_GET_ERROR) {
00459
if (PacketHeader.PacketLeader == CONTROL_PACKET_LEADER) {
00460
goto WaitForPacketLeader;
00461 }
else {
00462
goto SendResendPacket;
00463 }
00464 }
00465
00466
00467
00468
00469
00470 ReturnCode =
KdpReceiveString((PCHAR)&PacketHeader.PacketId,
00471
sizeof(PacketHeader.PacketId));
00472
00473
if (ReturnCode ==
CP_GET_NODATA) {
00474
return KDP_PACKET_TIMEOUT;
00475 }
else if (ReturnCode ==
CP_GET_ERROR) {
00476
if (PacketHeader.PacketLeader == CONTROL_PACKET_LEADER) {
00477
goto WaitForPacketLeader;
00478 }
else {
00479
goto SendResendPacket;
00480 }
00481 }
00482
00483
00484
00485
00486
00487 ReturnCode =
KdpReceiveString((PCHAR)&PacketHeader.Checksum,
00488
sizeof(PacketHeader.Checksum));
00489
if (ReturnCode ==
CP_GET_NODATA) {
00490
return KDP_PACKET_TIMEOUT;
00491 }
else if (ReturnCode ==
CP_GET_ERROR) {
00492
if (PacketHeader.PacketLeader == CONTROL_PACKET_LEADER) {
00493
goto WaitForPacketLeader;
00494 }
else {
00495
goto SendResendPacket;
00496 }
00497 }
00498
00499
00500
00501
00502
00503
00504
if (PacketHeader.PacketLeader == CONTROL_PACKET_LEADER ) {
00505
if (PacketHeader.PacketType == PACKET_TYPE_KD_ACKNOWLEDGE ) {
00506
00507
00508
00509
00510
00511
00512
00513
00514
00515
if (PacketHeader.PacketId !=
00516 (
KdpNextPacketIdToSend & ~SYNC_PACKET_ID)) {
00517
goto WaitForPacketLeader;
00518 }
else if (PacketType == PACKET_TYPE_KD_ACKNOWLEDGE) {
00519
KdpNextPacketIdToSend ^= 1;
00520
return KDP_PACKET_RECEIVED;
00521 }
else {
00522
goto WaitForPacketLeader;
00523 }
00524 }
else if (PacketHeader.PacketType == PACKET_TYPE_KD_RESET) {
00525
00526
00527
00528
00529
00530
00531
KdpNextPacketIdToSend = INITIAL_PACKET_ID;
00532
KdpPacketIdExpected = INITIAL_PACKET_ID;
00533
KdpSendControlPacket(PACKET_TYPE_KD_RESET, 0
L);
00534
return KDP_PACKET_RESEND;
00535 }
else if (PacketHeader.PacketType == PACKET_TYPE_KD_RESEND) {
00536
return KDP_PACKET_RESEND;
00537 }
else {
00538
00539
00540
00541
00542
00543
goto WaitForPacketLeader;
00544 }
00545
00546
00547
00548
00549
00550 }
else if (PacketType == PACKET_TYPE_KD_ACKNOWLEDGE) {
00551
00552
00553
00554
00555
00556
00557
00558
00559
if (PacketHeader.PacketId ==
KdpPacketIdExpected) {
00560
KdpSendControlPacket(PACKET_TYPE_KD_RESEND, 0
L);
00561
KdpNextPacketIdToSend ^= 1;
00562
return KDP_PACKET_RECEIVED;
00563 }
else {
00564
KdpSendControlPacket(PACKET_TYPE_KD_ACKNOWLEDGE,
00565 PacketHeader.PacketId
00566 );
00567
goto WaitForPacketLeader;
00568 }
00569 }
00570
00571
00572
00573
00574
00575
00576
00577
00578
00579
00580
00581 MessageLength = MessageHeader->MaximumLength;
00582
if ((PacketHeader.ByteCount > (
USHORT)PACKET_MAX_SIZE) ||
00583 (PacketHeader.ByteCount < (
USHORT)MessageLength)) {
00584
goto SendResendPacket;
00585 }
00586 *DataLength = PacketHeader.ByteCount - MessageLength;
00587
00588
00589
00590
00591
00592 ReturnCode =
KdpReceiveString(MessageHeader->Buffer, MessageLength);
00593
if (ReturnCode !=
CP_GET_SUCCESS) {
00594
goto SendResendPacket;
00595 }
00596 MessageHeader->Length = (
USHORT)MessageLength;
00597
00598
00599
00600
00601
00602 ReturnCode =
KdpReceiveString(MessageData->Buffer, *DataLength);
00603
if (ReturnCode !=
CP_GET_SUCCESS) {
00604
goto SendResendPacket;
00605 }
00606 MessageData->Length = (
USHORT)*DataLength;
00607
00608
00609
00610
00611
00612 ReturnCode =
KdPortGetByte(&Input);
00613
if (ReturnCode !=
CP_GET_SUCCESS || Input != PACKET_TRAILING_BYTE) {
00614
goto SendResendPacket;
00615 }
00616
00617
00618
00619
00620
00621
if (PacketType != PacketHeader.PacketType) {
00622
KdpSendControlPacket(PACKET_TYPE_KD_ACKNOWLEDGE,
00623 PacketHeader.PacketId
00624 );
00625
goto WaitForPacketLeader;
00626 }
00627
00628
00629
00630
00631
00632
if (PacketHeader.PacketId == INITIAL_PACKET_ID ||
00633 PacketHeader.PacketId == (INITIAL_PACKET_ID ^ 1)) {
00634
if (PacketHeader.PacketId !=
KdpPacketIdExpected) {
00635
KdpSendControlPacket(PACKET_TYPE_KD_ACKNOWLEDGE,
00636 PacketHeader.PacketId
00637 );
00638
goto WaitForPacketLeader;
00639 }
00640 }
else {
00641
goto SendResendPacket;
00642 }
00643
00644
00645
00646
00647
00648 Checksum =
KdpComputeChecksum(
00649 MessageHeader->Buffer,
00650 MessageHeader->Length
00651 );
00652
00653 Checksum +=
KdpComputeChecksum(
00654 MessageData->Buffer,
00655 MessageData->Length
00656 );
00657
if (Checksum != PacketHeader.Checksum) {
00658
goto SendResendPacket;
00659 }
00660
00661
00662
00663
00664
00665
00666
KdpSendControlPacket(PACKET_TYPE_KD_ACKNOWLEDGE,
00667 PacketHeader.PacketId
00668 );
00669
00670
00671
00672
00673
00674
00675
KdpPacketIdExpected ^= 1;
00676
return KDP_PACKET_RECEIVED;
00677
00678 SendResendPacket:
00679
KdpSendControlPacket(PACKET_TYPE_KD_RESEND, 0
L);
00680
goto WaitForPacketLeader;
00681 }
00682
00683
VOID
00684 KdpSendPacket (
00685 IN ULONG PacketType,
00686 IN PSTRING MessageHeader,
00687 IN PSTRING MessageData OPTIONAL
00688 )
00689
00690
00691
00692
00693
00694
00695
00696
00697
00698
00699
00700
00701
00702
00703
00704
00705
00706
00707
00708
00709
00710
00711
00712
00713 {
00714
00715 KD_PACKET PacketHeader;
00716 ULONG MessageDataLength;
00717 ULONG ReturnCode;
00718 PDBGKD_DEBUG_IO DebugIo;
00719 PDBGKD_WAIT_STATE_CHANGE64 StateChange;
00720
00721
if ( ARGUMENT_PRESENT(MessageData) ) {
00722 MessageDataLength = MessageData->Length;
00723 PacketHeader.Checksum =
KdpComputeChecksum(
00724 MessageData->Buffer,
00725 MessageData->Length
00726 );
00727 }
else {
00728 MessageDataLength = 0;
00729 PacketHeader.Checksum = 0;
00730 }
00731
00732 PacketHeader.Checksum +=
KdpComputeChecksum (
00733 MessageHeader->Buffer,
00734 MessageHeader->Length
00735 );
00736
00737
00738
00739
00740
00741 PacketHeader.PacketLeader = PACKET_LEADER;
00742 PacketHeader.ByteCount = (
USHORT)(MessageHeader->Length + MessageDataLength);
00743 PacketHeader.PacketType = (
USHORT)PacketType;
00744
KdpNumberRetries =
KdpRetryCount;
00745
do {
00746
if (
KdpNumberRetries == 0) {
00747
00748
00749
00750
00751
00752
00753
if (PacketType == PACKET_TYPE_KD_DEBUG_IO) {
00754 DebugIo = (PDBGKD_DEBUG_IO)MessageHeader->Buffer;
00755
if (DebugIo->ApiNumber == DbgKdPrintStringApi) {
00756
KdDebuggerNotPresent =
TRUE;
00757 SharedUserData->KdDebuggerEnabled &= ~0x00000002;
00758
KdpNextPacketIdToSend = INITIAL_PACKET_ID | SYNC_PACKET_ID;
00759
KdpPacketIdExpected = INITIAL_PACKET_ID;
00760
return;
00761 }
00762 }
else if (PacketType == PACKET_TYPE_KD_STATE_CHANGE64) {
00763 StateChange = (PDBGKD_WAIT_STATE_CHANGE64)MessageHeader->Buffer;
00764
if (StateChange->NewState == DbgKdLoadSymbolsStateChange) {
00765
KdDebuggerNotPresent =
TRUE;
00766 SharedUserData->KdDebuggerEnabled &= ~0x00000002;
00767
KdpNextPacketIdToSend = INITIAL_PACKET_ID | SYNC_PACKET_ID;
00768
KdpPacketIdExpected = INITIAL_PACKET_ID;
00769
return;
00770 }
00771 }
00772 }
00773
00774
00775
00776
00777
00778
00779 PacketHeader.PacketId =
KdpNextPacketIdToSend;
00780
KdpSendString((PCHAR)&PacketHeader,
sizeof(KD_PACKET));
00781
00782
00783
00784
00785
00786
KdpSendString(MessageHeader->Buffer, MessageHeader->Length);
00787
00788
00789
00790
00791
00792
if ( MessageDataLength ) {
00793
KdpSendString(MessageData->Buffer, MessageData->Length);
00794 }
00795
00796
00797
00798
00799
00800
KdPortPutByte(PACKET_TRAILING_BYTE);
00801
00802
00803
00804
00805
00806 ReturnCode =
KdpReceivePacket(
00807 PACKET_TYPE_KD_ACKNOWLEDGE,
00808
NULL,
00809
NULL,
00810
NULL
00811 );
00812
if (ReturnCode ==
KDP_PACKET_TIMEOUT) {
00813
KdpNumberRetries--;
00814 }
00815 }
while (ReturnCode !=
KDP_PACKET_RECEIVED);
00816
00817
00818
00819
00820
00821
KdpNextPacketIdToSend &= ~SYNC_PACKET_ID;
00822
00823
00824
00825
00826
00827
00828
KdpRetryCount =
KdpDefaultRetries ;
00829 }