LLDB mainline
StringExtractorGDBRemote.cpp
Go to the documentation of this file.
1//===-- StringExtractorGDBRemote.cpp --------------------------------------===//
2//
3// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4// See https://llvm.org/LICENSE.txt for license information.
5// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6//
7//===----------------------------------------------------------------------===//
8
10
11#include <cctype>
12#include <cstring>
13#include <optional>
14
17 if (m_packet.empty())
18 return eUnsupported;
19
20 switch (m_packet[0]) {
21 case 'E':
22 if (isxdigit(m_packet[1]) && isxdigit(m_packet[2])) {
23 if (m_packet.size() == 3)
24 return eError;
25 llvm::StringRef packet_ref(m_packet);
26 if (packet_ref[3] == ';') {
27 auto err_string = packet_ref.substr(4);
28 for (auto e : err_string)
29 if (!isxdigit(e))
30 return eResponse;
31 return eError;
32 }
33 }
34 break;
35
36 case 'O':
37 if (m_packet.size() == 2 && m_packet[1] == 'K')
38 return eOK;
39 break;
40
41 case '+':
42 if (m_packet.size() == 1)
43 return eAck;
44 break;
45
46 case '-':
47 if (m_packet.size() == 1)
48 return eNack;
49 break;
50 }
51 return eResponse;
52}
53
56#define PACKET_MATCHES(s) \
57 ((packet_size == (sizeof(s) - 1)) && (strcmp((packet_cstr), (s)) == 0))
58#define PACKET_STARTS_WITH(s) \
59 ((packet_size >= (sizeof(s) - 1)) && \
60 ::strncmp(packet_cstr, s, (sizeof(s) - 1)) == 0)
61
62 // Empty is not a supported packet...
63 if (m_packet.empty())
65
66 const size_t packet_size = m_packet.size();
67 const char *packet_cstr = m_packet.c_str();
68 switch (m_packet[0]) {
69
70 case '%':
72
73 case '\x03':
74 if (packet_size == 1)
76 break;
77
78 case '-':
79 if (packet_size == 1)
81 break;
82
83 case '+':
84 if (packet_size == 1)
86 break;
87
88 case 'A':
90
91 case 'Q':
92
93 switch (packet_cstr[1]) {
94 case 'E':
95 if (PACKET_STARTS_WITH("QEnvironment:"))
97 if (PACKET_STARTS_WITH("QEnvironmentHexEncoded:"))
99 if (PACKET_STARTS_WITH("QEnableErrorStrings"))
101 break;
102
103 case 'P':
104 if (PACKET_STARTS_WITH("QPassSignals:"))
106 break;
107
108 case 'S':
109 if (PACKET_MATCHES("QStartNoAckMode"))
111 if (PACKET_STARTS_WITH("QSaveRegisterState"))
113 if (PACKET_STARTS_WITH("QSetDisableASLR:"))
115 if (PACKET_STARTS_WITH("QSetDetachOnError:"))
117 if (PACKET_STARTS_WITH("QSetSTDIN:"))
119 if (PACKET_STARTS_WITH("QSetSTDOUT:"))
121 if (PACKET_STARTS_WITH("QSetSTDERR:"))
123 if (PACKET_STARTS_WITH("QSetSTDIOWindowSize:"))
125 if (PACKET_STARTS_WITH("QSetWorkingDir:"))
127 if (PACKET_STARTS_WITH("QSetLogging:"))
129 if (PACKET_STARTS_WITH("QSetIgnoredExceptions"))
131 if (PACKET_STARTS_WITH("QSetMaxPacketSize:"))
133 if (PACKET_STARTS_WITH("QSetMaxPayloadSize:"))
135 if (PACKET_STARTS_WITH("QSetEnableAsyncProfiling;"))
137 if (PACKET_STARTS_WITH("QSyncThreadState:"))
139 break;
140
141 case 'L':
142 if (PACKET_STARTS_WITH("QLaunchArch:"))
144 if (PACKET_MATCHES("QListThreadsInStopReply"))
146 break;
147
148 case 'M':
149 if (PACKET_STARTS_WITH("QMemTags"))
151 break;
152
153 case 'N':
154 if (PACKET_STARTS_WITH("QNonStop:"))
156 break;
157
158 case 'R':
159 if (PACKET_STARTS_WITH("QRestoreRegisterState:"))
161 break;
162
163 case 'T':
164 if (PACKET_MATCHES("QThreadSuffixSupported"))
166 break;
167 }
168 break;
169
170 case 'q':
171 switch (packet_cstr[1]) {
172 case 's':
173 if (PACKET_MATCHES("qsProcessInfo"))
175 if (PACKET_MATCHES("qsThreadInfo"))
177 break;
178
179 case 'f':
180 if (PACKET_STARTS_WITH("qfProcessInfo"))
182 if (PACKET_STARTS_WITH("qfThreadInfo"))
184 break;
185
186 case 'C':
187 if (packet_size == 2)
189 break;
190
191 case 'E':
192 if (PACKET_STARTS_WITH("qEcho:"))
194 break;
195
196 case 'F':
197 if (PACKET_STARTS_WITH("qFileLoadAddress:"))
199 break;
200
201 case 'G':
202 if (PACKET_STARTS_WITH("qGroupName:"))
204 if (PACKET_MATCHES("qGetWorkingDir"))
206 if (PACKET_MATCHES("qGetPid"))
208 if (PACKET_STARTS_WITH("qGetProfileData;"))
210 if (PACKET_MATCHES("qGDBServerVersion"))
212 break;
213
214 case 'H':
215 if (PACKET_MATCHES("qHostInfo"))
217 break;
218
219 case 'K':
220 if (PACKET_STARTS_WITH("qKillSpawnedProcess"))
222 break;
223
224 case 'L':
225 if (PACKET_STARTS_WITH("qLaunchGDBServer"))
227 if (PACKET_MATCHES("qLaunchSuccess"))
229 break;
230
231 case 'M':
232 if (PACKET_STARTS_WITH("qMemoryRegionInfo:"))
234 if (PACKET_MATCHES("qMemoryRegionInfo"))
236 if (PACKET_STARTS_WITH("qModuleInfo:"))
238 if (PACKET_STARTS_WITH("qMemTags:"))
240 break;
241
242 case 'P':
243 if (PACKET_STARTS_WITH("qProcessInfoPID:"))
245 if (PACKET_STARTS_WITH("qPlatform_shell:"))
247 if (PACKET_STARTS_WITH("qPlatform_mkdir:"))
249 if (PACKET_STARTS_WITH("qPlatform_chmod:"))
251 if (PACKET_MATCHES("qProcessInfo"))
253 if (PACKET_STARTS_WITH("qPathComplete:"))
255 break;
256
257 case 'Q':
258 if (PACKET_MATCHES("qQueryGDBServer"))
260 break;
261
262 case 'R':
263 if (PACKET_STARTS_WITH("qRcmd,"))
265 if (PACKET_STARTS_WITH("qRegisterInfo"))
267 break;
268
269 case 'S':
270 if (PACKET_STARTS_WITH("qSaveCore"))
272 if (PACKET_STARTS_WITH("qSpeedTest:"))
274 if (PACKET_MATCHES("qShlibInfoAddr"))
276 if (PACKET_MATCHES("qStepPacketSupported"))
278 if (PACKET_STARTS_WITH("qSupported"))
280 if (PACKET_MATCHES("qSyncThreadStateSupported"))
282 if (PACKET_MATCHES("qStructuredDataPlugins"))
284 break;
285
286 case 'T':
287 if (PACKET_STARTS_WITH("qThreadExtraInfo,"))
289 if (PACKET_STARTS_WITH("qThreadStopInfo"))
291 break;
292
293 case 'U':
294 if (PACKET_STARTS_WITH("qUserName:"))
296 break;
297
298 case 'V':
299 if (PACKET_MATCHES("qVAttachOrWaitSupported"))
301 break;
302
303 case 'W':
304 if (PACKET_STARTS_WITH("qWatchpointSupportInfo:"))
306 if (PACKET_MATCHES("qWatchpointSupportInfo"))
308 break;
309
310 case 'X':
311 if (PACKET_STARTS_WITH("qXfer:"))
313 break;
314 }
315 break;
316
317 case 'j':
318 if (PACKET_STARTS_WITH("jModulesInfo:"))
320 if (PACKET_MATCHES("jSignalsInfo"))
322 if (PACKET_MATCHES("jThreadsInfo"))
324
325 if (PACKET_MATCHES("jLLDBTraceSupported"))
327 if (PACKET_STARTS_WITH("jLLDBTraceStop:"))
329 if (PACKET_STARTS_WITH("jLLDBTraceStart:"))
331 if (PACKET_STARTS_WITH("jLLDBTraceGetState:"))
333 if (PACKET_STARTS_WITH("jLLDBTraceGetBinaryData:"))
335 if (PACKET_STARTS_WITH("jMultiBreakpoint:"))
337 if (PACKET_MATCHES("jAcceleratorPluginInitialize"))
339 if (PACKET_STARTS_WITH("jAcceleratorPluginBreakpointHit:"))
341 break;
342
343 case 'v':
344 if (PACKET_STARTS_WITH("vFile:")) {
345 if (PACKET_STARTS_WITH("vFile:open:"))
347 else if (PACKET_STARTS_WITH("vFile:close:"))
349 else if (PACKET_STARTS_WITH("vFile:pread"))
351 else if (PACKET_STARTS_WITH("vFile:pwrite"))
353 else if (PACKET_STARTS_WITH("vFile:size"))
355 else if (PACKET_STARTS_WITH("vFile:exists"))
357 else if (PACKET_STARTS_WITH("vFile:fstat"))
359 else if (PACKET_STARTS_WITH("vFile:stat"))
361 else if (PACKET_STARTS_WITH("vFile:mode"))
363 else if (PACKET_STARTS_WITH("vFile:MD5"))
365 else if (PACKET_STARTS_WITH("vFile:symlink"))
367 else if (PACKET_STARTS_WITH("vFile:unlink"))
369
370 } else {
371 if (PACKET_STARTS_WITH("vAttach;"))
373 if (PACKET_STARTS_WITH("vAttachWait;"))
375 if (PACKET_STARTS_WITH("vAttachOrWait;"))
377 if (PACKET_STARTS_WITH("vAttachName;"))
379 if (PACKET_STARTS_WITH("vCont;"))
381 if (PACKET_MATCHES("vCont?"))
383 if (PACKET_STARTS_WITH("vKill;"))
385 if (PACKET_STARTS_WITH("vRun;"))
387 if (PACKET_MATCHES("vStopped"))
389 if (PACKET_MATCHES("vCtrlC"))
391 if (PACKET_MATCHES("vStdio"))
393 break;
394
395 }
396 break;
397 case '_':
398 switch (packet_cstr[1]) {
399 case 'M':
401
402 case 'm':
404 }
405 break;
406
407 case '?':
408 if (packet_size == 1)
410 break;
411
412 case 'c':
413 return eServerPacketType_c;
414
415 case 'C':
416 return eServerPacketType_C;
417
418 case 'D':
419 return eServerPacketType_D;
420
421 case 'g':
422 return eServerPacketType_g;
423
424 case 'G':
425 return eServerPacketType_G;
426
427 case 'H':
428 return eServerPacketType_H;
429
430 case 'I':
431 return eServerPacketType_I;
432
433 case 'k':
434 if (packet_size == 1)
435 return eServerPacketType_k;
436 break;
437
438 case 'm':
439 return eServerPacketType_m;
440
441 case 'M':
442 return eServerPacketType_M;
443
444 case 'p':
445 return eServerPacketType_p;
446
447 case 'P':
448 return eServerPacketType_P;
449
450 case 's':
451 if (packet_size == 1)
452 return eServerPacketType_s;
453 break;
454
455 case 'S':
456 return eServerPacketType_S;
457
458 case 'x':
459 return eServerPacketType_x;
460
461 case 'X':
462 return eServerPacketType_X;
463
464 case 'T':
465 return eServerPacketType_T;
466
467 case 'z':
468 if (packet_cstr[1] >= '0' && packet_cstr[1] <= '4')
469 return eServerPacketType_z;
470 break;
471
472 case 'Z':
473 if (packet_cstr[1] >= '0' && packet_cstr[1] <= '4')
474 return eServerPacketType_Z;
475 break;
476 }
478}
479
481 return GetResponseType() == eOK;
482}
483
487
491
493 return GetResponseType() == eError && isxdigit(m_packet[1]) &&
494 isxdigit(m_packet[2]);
495}
496
498 if (GetResponseType() == eError) {
499 SetFilePos(1);
500 return GetHexU8(255);
501 }
502 return 0;
503}
504
507 if (GetResponseType() == eError) {
508 SetFilePos(1);
509 uint8_t errc = GetHexU8(255);
511 std::string error_messg;
512 if (GetChar() == ';') {
513 GetHexByteString(error_messg);
514 error = lldb_private::Status(error_messg);
515 }
516 }
517 return error;
518}
519
521 // Just get the data bytes in the string as
522 // GDBRemoteCommunication::CheckForPacket() already removes any 0x7d escaped
523 // characters. If any 0x7d characters are left in the packet, then they are
524 // supposed to be there...
525 str.clear();
526 const size_t bytes_left = GetBytesLeft();
527 if (bytes_left > 0) {
528 str.assign(m_packet, m_index, bytes_left);
529 m_index += bytes_left;
530 }
531 return str.size();
532}
533
534static bool
536 const StringExtractorGDBRemote &response) {
537 switch (response.GetResponseType()) {
541 return true;
542
546 break;
547 }
548 return false;
549}
550
551static bool JSONResponseValidator(void *,
552 const StringExtractorGDBRemote &response) {
553 switch (response.GetResponseType()) {
556 return true; // Accept unsupported or EXX as valid responses
557
561 break;
562
564 // JSON that is returned in from JSON query packets is currently always
565 // either a dictionary which starts with a '{', or an array which starts
566 // with a '['. This is a quick validator to just make sure the response
567 // could be valid JSON without having to validate all of the
568 // JSON content.
569 switch (response.GetStringRef()[0]) {
570 case '{':
571 return true;
572 case '[':
573 return true;
574 default:
575 break;
576 }
577 break;
578 }
579 return false;
580}
581
582static bool
584 const StringExtractorGDBRemote &response) {
585 switch (response.GetResponseType()) {
588 return true; // Accept unsupported or EXX as valid responses
589
593 break;
594
596 uint32_t valid_count = 0;
597 for (const char ch : response.GetStringRef()) {
598 if (!isxdigit(ch)) {
599 return false;
600 }
601 if (++valid_count >= 16)
602 break; // Don't validate all the characters in case the packet is very
603 // large
604 }
605 return true;
606 } break;
607 }
608 return false;
609}
610
616
618 ResponseValidatorCallback callback, void *baton) {
619 m_validator = callback;
620 m_validator_baton = baton;
621}
622
627
632
637
639 // If we have a validator callback, try to validate the callback
640 if (m_validator)
641 return m_validator(m_validator_baton, *this);
642 else
643 return true; // No validator, so response is valid
644}
645
646std::optional<std::pair<lldb::pid_t, lldb::tid_t>>
648 llvm::StringRef view = llvm::StringRef(m_packet).substr(m_index);
649 size_t initial_length = view.size();
651 lldb::tid_t tid;
652
653 if (view.consume_front("p")) {
654 // process identifier
655 if (view.consume_front("-1")) {
656 // -1 is a special case
657 pid = AllProcesses;
658 } else if (view.consumeInteger(16, pid) || pid == 0) {
659 // not a valid hex integer OR unsupported pid 0
661 return std::nullopt;
662 }
663
664 // "." must follow if we expect TID too; otherwise, we assume -1
665 if (!view.consume_front(".")) {
666 // update m_index
667 m_index += initial_length - view.size();
668
669 return {{pid, AllThreads}};
670 }
671 }
672
673 // thread identifier
674 if (view.consume_front("-1")) {
675 // -1 is a special case
676 tid = AllThreads;
677 } else if (view.consumeInteger(16, tid) || tid == 0 || pid == AllProcesses) {
678 // not a valid hex integer OR tid 0 OR pid -1 + a specific tid
680 return std::nullopt;
681 }
682
683 // update m_index
684 m_index += initial_length - view.size();
685
686 return {{pid != LLDB_INVALID_PROCESS_ID ? pid : default_pid, tid}};
687}
static llvm::raw_ostream & error(Stream &strm)
#define PACKET_STARTS_WITH(s)
#define PACKET_MATCHES(s)
static bool JSONResponseValidator(void *, const StringExtractorGDBRemote &response)
static bool OKErrorNotSupportedResponseValidator(void *, const StringExtractorGDBRemote &response)
static bool ASCIIHexBytesResponseValidator(void *, const StringExtractorGDBRemote &response)
void CopyResponseValidator(const StringExtractorGDBRemote &rhs)
ServerPacketType GetServerPacketType() const
StringExtractorGDBRemote()=default
static constexpr lldb::tid_t AllThreads
size_t GetEscapedBinaryData(std::string &str)
void SetResponseValidator(ResponseValidatorCallback callback, void *baton)
static constexpr lldb::pid_t AllProcesses
ResponseValidatorCallback m_validator
bool(* ResponseValidatorCallback)(void *baton, const StringExtractorGDBRemote &response)
std::optional< std::pair< lldb::pid_t, lldb::tid_t > > GetPidTid(lldb::pid_t default_pid)
void SetFilePos(uint32_t idx)
uint64_t m_index
When extracting data from a packet, this index will march along as things get extracted.
size_t GetHexByteString(std::string &str)
uint8_t GetHexU8(uint8_t fail_value=0, bool set_eof_on_fail=true)
char GetChar(char fail_value='\0')
std::string m_packet
The string in which to extract data.
llvm::StringRef GetStringRef() const
An error handling class.
Definition Status.h:118
static Status FromErrorStringWithFormat(const char *format,...) __attribute__((format(printf
Definition Status.cpp:106
#define UINT64_MAX
#define LLDB_INVALID_PROCESS_ID
uint64_t pid_t
Definition lldb-types.h:83
uint64_t tid_t
Definition lldb-types.h:84