2 public import static "ecere"
3 public import static "ec"
15 #define GDB_DEBUG_CONSOLE
19 extern char * strrchr(const char * s, int c);
22 #define strlen _strlen
26 #include <string.h> // For memchr
34 #include <sys/time.h> // Required on Apple...
48 sprintf(s[0], "%04d", now.year);
49 sprintf(s[1], "%02d", now.month+1);
50 sprintf(s[2], "%02d", now.day);
51 sprintf(s[3], "%02d", now.hour);
52 sprintf(s[4], "%02d", now.minute);
53 sprintf(s[5], "%02d", now.second);
54 time = PrintString("*", s[0], s[1], s[2], "-", s[3], s[4], s[5], "*");
60 // use =0 to disable printing of specific channels
62 static enum dplchan { none, gdbProtoIgnored=0/*1*/, gdbProtoUnknown=2, gdbOutput=3/*3*/, gdbCommand=4/*4*/, debuggerCall=0/*5*/, debuggerProblem=6,
63 debuggerUserAction=7,debuggerState=8, debuggerBreakpoints=9, debuggerWatches=0/*10*/, debuggerTemp=0 };
65 static enum dplchan { none, gdbProtoIgnored=0, gdbProtoUnknown=0, gdbOutput=0, gdbCommand=0, debuggerCall=0, debuggerProblem=0,
66 debuggerUserAction=0,debuggerState=0, debuggerBreakpoints=0, debuggerWatches=0, debuggerTemp=0 };
68 static char * _dpct[] = {
70 "GDB Protocol Ignored",
71 "GDB Protocol ***Unknown***",
75 "Debugger ***Problem***",
76 "Debugger::ChangeUserAction",
77 "Debugger::ChangeState",
80 "-----> Temporary Message",
84 // TODO if(strlen(item.value) < MAX_F_STRING)
88 #define _dpl2(...) __dpl2(__FILE__, __LINE__, ##__VA_ARGS__)
92 static void __dpl2(char * file, int line, char ** channels, int channel, int indent, typed_object object, ...)
94 bool chan = channel && channels && channels[channel];
97 char string[MAX_F_STRING];
99 char * time = PrintNow();
101 //ide.outputView.debugBox.Logf();
102 Logf("%s %s:% 5d: %s%s", time, file, line, chan ? channels[channel] : "", chan && channels[channel][0] ? ": " : "");
103 va_start(args, object);
104 len = PrintStdArgsToBuffer(string, sizeof(string), object, args);
112 #define _dpl(...) __dpl(__FILE__, __LINE__, ##__VA_ARGS__)
113 static void __dpl(char * file, int line, int indent, char * format, ...)
116 char string[MAX_F_STRING];
118 char * time = PrintNow();
119 //static File f = null;
120 va_start(args, format);
121 vsnprintf(string, sizeof(string), format, args);
122 string[sizeof(string)-1] = 0;
125 char * time = PrintNow();
127 logName = PrintString(time, ".log");
129 f = FileOpen(logName, write);
132 /*f.Printf("%s %s:% 5d: ", time, file, line);
133 for(c = 0; c<indent; c++)
135 f.Printf("%s\n", string);*/
136 Logf("%s %s:% 5d: ", time, file, line);
137 for(c = 0; c<indent; c++)
139 Logf("%s\n", string);
144 public char * StripQuotes2(char * string, char * output)
148 bool quoted = false, escaped = false;
150 for(c = 0; ch = string[c]; c++)
154 if(escaped || ch != '\"')
157 escaped = !escaped && ch == '\\';
171 // String Escape Copy
172 static void strescpy(char * d, char * s)
180 case '\n': d[k] = '\\'; d[++k] = 'n'; break;
181 case '\t': d[k] = '\\'; d[++k] = 't'; break;
182 case '\a': d[k] = '\\'; d[++k] = 'a'; break;
183 case '\b': d[k] = '\\'; d[++k] = 'b'; break;
184 case '\f': d[k] = '\\'; d[++k] = 'f'; break;
185 case '\r': d[k] = '\\'; d[++k] = 'r'; break;
186 case '\v': d[k] = '\\'; d[++k] = 'v'; break;
187 case '\\': d[k] = '\\'; d[++k] = '\\'; break;
188 case '\"': d[k] = '\\'; d[++k] = '\"'; break;
189 default: d[k] = s[j];
196 static char * CopyUnescapedSystemPath(char * p)
199 char * d = new char[len + 1];
200 UnescapeString(d, p, len);
201 #if defined(__WIN32__)
202 ChangeCh(d, '/', '\\');
207 static char * CopyUnescapedUnixPath(char * p)
210 char * d = new char[len + 1];
211 UnescapeString(d, p, len);
212 #if defined(__WIN32__)
213 ChangeCh(d, '\\', '/');
218 static char * CopyUnescapedString(char * s)
221 char * d = new char[len + 1];
222 UnescapeString(d, s, len);
226 static char * StripBrackets(char * string)
228 int length = strlen(string);
229 if(length > 1 && *string == '[' && string[length - 1] == ']')
232 string[length - 1] = '\0';
239 static char * StripCurlies(char * string)
241 int length = strlen(string);
242 if(length > 1 && *string == '{' && string[length - 1] == '}')
245 string[length - 1] = '\0';
252 static int StringGetInt(char * string, int start)
255 int i, len = strlen(string);
257 for(i = start; i < len && i < start + 8; i++)
259 if(string[i] == '0' || string[i] == '1' || string[i] == '2' || string[i] == '3' || string[i] == '4' || string[i] == '5' || string[i] == '6' || string[i] == '7' || string[i] == '8' || string[i] == '9')
260 strncat(number, &string[i], 1);
267 static int TokenizeList(char * string, const char seperator, Array<char *> tokens)
271 bool quoted = false, escaped = false;
272 char * start = string, ch;
274 for(; (ch = *string); string++)
281 if(escaped || ch != '\"')
282 escaped = !escaped && ch == '\\';
288 else if(ch == '{' || ch == '[' || ch == '(' || ch == '<')
290 else if(ch == '}' || ch == ']' || ch == ')' || ch == '>')
292 else if(ch == seperator && !level)
301 //tokens[count] = start;
302 //tokens[count++] = start;
309 static bool TokenizeListItem(char * string, DebugListItem item)
311 char * equal = strstr(string, "=");
323 static bool CheckCommandAvailable(const char * command)
325 bool available = false;
327 char * name = new char[MAX_FILENAME];
328 char * pathVar = new char[maxPathLen];
330 GetEnvironment("PATH", pathVar, maxPathLen);
331 count = TokenizeWith(pathVar, sizeof(paths) / sizeof(char *), paths, pathListSep, false);
332 strcpy(name, command);
336 const char * extensions[] = { "exe", "com", "bat", null };
337 for(e=0; extensions[e]; e++)
339 ChangeExtension(name, extensions[e], name);
341 for(c=0; c<count; c++)
343 FileListing fl { paths[c] };
346 if(fl.stats.attribs.isFile && !fstrcmp(fl.name, name))
365 // define GdbGetLineSize = 1638400;
366 define GdbGetLineSize = 5638400;
367 #if defined(__unix__)
368 char progFifoPath[MAX_LOCATION];
369 char progFifoDir[MAX_LOCATION];
372 enum DebuggerState { none, prompt, loaded, running, stopped, terminated, error };
375 none, hit, breakEvent, signal, stepEnd, functionEnd, exit, valgrindStartPause, locationReached;
377 property bool canBeMonitored { get { return (this == hit || this == breakEvent || this == signal || this == stepEnd || this == functionEnd || this == locationReached); } };
379 enum DebuggerAction { none, internal, restart, stop, selectFrame, advance }; //, bpValidation
382 unknown, endSteppingRange, functionFinished, signalReceived, breakpointHit, locationReached
383 //watchpointTrigger, readWatchpointTrigger, accessWatchpointTrigger, watchpointScope,
384 //exited, exitedNormally, exitedSignalled;
388 none, internalMain, internalWinMain, internalModulesLoaded, user, runToCursor, internalModuleLoad, internalEntry;
390 property bool isInternal { get { return (this == internalMain || this == internalWinMain || this == internalModulesLoaded || this == internalModuleLoad || this == internalEntry); } };
391 property bool isUser { get { return (this == user || this == runToCursor); } };
393 enum DebuggerEvaluationError { none, symbolNotFound, memoryCantBeRead, unknown };
394 enum DebuggerUserAction
396 none, start, resume, _break, stop, restart, selectThread, selectFrame, stepInto, stepOver, stepUntil, stepOut, runToCursor;
397 property bool breaksOnInternalBreakpoint { get { return (this == stepInto || this == stepOver || this == stepUntil); } };
401 none, run, _continue, next, until, advance, step, finish;
402 property bool suspendInternalBreakpoints { get { return (this == until || this == advance || this == step || this == finish); } };
405 FileDialog debuggerFileDialog { type = selectDir };
407 static DualPipe vgTargetHandle;
408 static File vgLogFile;
409 static char vgLogPath[MAX_LOCATION];
410 static DualPipe gdbHandle;
411 static DebugEvaluationData eval { };
413 static int targetProcessId;
415 static bool gdbReady;
416 static bool breakpointError;
420 Semaphore serialSemaphore { };
426 bool sentBreakInsert;
427 bool ignoreBreakpoints;
435 int activeFrameLevel;
444 GdbExecution gdbExecution;
445 DebuggerUserAction userAction;
448 DebuggerAction breakType;
450 //DebuggerCommand lastCommand; // THE COMPILER COMPILES STUFF THAT DOES NOT EXIST???
452 GdbDataStop stopItem;
453 GdbDataBreakpoint bpItem;
456 List<Breakpoint> sysBPs { };
457 Breakpoint bpRunToCursor;
458 Breakpoint intBpEntry;
459 Breakpoint intBpMain;
460 Breakpoint intBpWinMain;
464 CompilerConfig currentCompiler;
465 ProjectConfig prjConfig;
468 CodeEditor codeEditor;
470 ValgrindLogThread vgLogThread { debugger = this };
471 ValgrindTargetThread vgTargetThread { debugger = this };
472 GdbThread gdbThread { debugger = this };
475 Map<String, bool> projectsLibraryLoaded { };
479 delay = 0.0, userData = this;
483 bool monitor = false;
484 DebuggerEvent curEvent = event;
485 GdbDataStop stopItem = this.stopItem;
486 Breakpoint bpUser = null;
487 Breakpoint bpInternal = null;
495 this.stopItem = null;
499 DynamicString bpReport { };
501 for(bp : sysBPs; bp.inserted)
503 bpReport.concatx(",", bp.type, "(", s=bp.CopyLocationString(false), ")");
506 if(bpRunToCursor && bpRunToCursor.inserted)
508 Breakpoint bp = bpRunToCursor;
509 bpReport.concatx(",", bp.type, "(", s=bp.CopyLocationString(false), ")");
512 for(bp : ide.workspace.breakpoints; bp.inserted)
514 bpReport.concatx(",", bp.type, "(", s=bp.CopyLocationString(false), ")");
518 _dpl2(_dpct, dplchan::debuggerBreakpoints, 0, "gdbTimer::DelayExpired: ", s+1);
523 Breakpoint bp = GetBreakpointById(stopItem.bkptno, &isInternal);
526 _dpl2(_dpct, dplchan::debuggerBreakpoints, 0, "gdb stopped by a breakpoint: ", bp.type, "(", s=bp.CopyLocationString(false), ")");
537 if(curEvent && curEvent != exit)
539 _dpl(0, "No stop item");
547 Restart(currentCompiler, prjConfig, bitDepth, usingValgrind);
556 GdbCommand(0, false, "-stack-select-frame %d", activeFrameLevel);
557 for(activeFrame = stackFrames.first; activeFrame; activeFrame = activeFrame.next)
558 if(activeFrame.level == activeFrameLevel)
564 // GdbCommand(0, false, "-break-info %s", bpItem.number);
576 Breakpoint bp = stopItem ? GetBreakpointById(stopItem.bkptno, &isInternal) : null;
577 if(bp && bp.inserted && bp.bp.addr)
579 if(bp.type.isInternal)
583 if(stopItem && stopItem.frame)
585 if(bpInternal && bpRunToCursor && bpRunToCursor.inserted && !strcmp(bpRunToCursor.bp.addr, bp.bp.addr))
586 bpUser = bpRunToCursor;
589 for(item : (bpInternal ? ide.workspace.breakpoints : sysBPs); item.inserted)
591 if(item.bp && item.bp.addr && !strcmp(item.bp.addr, bp.bp.addr))
603 _dpl2(_dpct, dplchan::debuggerProblem, 0, "Invalid stopItem!");
604 if(bpUser && stopItem.frame.addr && strcmp(stopItem.frame.addr, bpUser.bp.addr))
605 _dpl2(_dpct, dplchan::debuggerProblem, 0, "Breakpoint bkptno(", stopItem.bkptno, ") address missmatch!");
608 _dpl2(_dpct, dplchan::debuggerProblem, 0, "Breakpoint bkptno(", stopItem.bkptno, ") invalid or not found!");
609 if((bpUser && !ignoreBreakpoints) || (bpInternal && userAction.breaksOnInternalBreakpoint))
611 hitThread = stopItem.threadid;
615 signalThread = stopItem.threadid;
619 case locationReached:
621 ignoreBreakpoints = false;
623 case valgrindStartPause:
624 GdbExecContinue(true);
632 if(curEvent == signal)
636 ide.outputView.debugBox.Logf($"Signal received: %s - %s\n", stopItem.name, stopItem.meaning);
637 ide.outputView.debugBox.Logf(" %s:%d\n", (s = CopySystemPath(stopItem.frame.file)), stopItem.frame.line);
638 ide.outputView.Show();
639 ide.callStackView.Show();
642 else if(curEvent == breakEvent)
644 ide.threadsView.Show();
645 ide.callStackView.Show();
646 ide.callStackView.Activate();
648 else if(curEvent == hit)
650 if(BreakpointHit(stopItem, bpInternal, bpUser))
652 ide.AdjustDebugMenus();
653 if(bpUser && bpUser.type == runToCursor)
655 ignoreBreakpoints = false;
656 UnsetBreakpoint(bpUser);
657 delete bpRunToCursor;
662 if(breakType == advance && bpInternal && (bpInternal.type == internalMain || bpInternal.type == internalEntry))
665 GdbExecAdvance(breakString, 0);
670 GdbExecContinue(false);
676 if(monitor && curEvent.canBeMonitored)
679 activeThread = stopItem.threadid;
680 GdbCommand(0, false, "-thread-list-ids");
681 InternalSelectFrame(activeFrameLevel);
682 GoToStackFrameLine(activeFrameLevel, true, false);
684 ide.ShowCodeEditor();
685 ide.AdjustDebugMenus();
686 ideMainFrame.Activate(); // TOFIX: ide.Activate() is not reliable (app inactive)
699 #ifdef GDB_DEBUG_CONSOLE
700 char lastGdbOutput[GdbGetLineSize];
702 #if defined(__unix__)
703 ProgramThread progThread { };
707 #define _ChangeUserAction(value) ChangeUserAction(__FILE__, __LINE__, value)
708 void ChangeUserAction(char * file, int line, DebuggerUserAction value)
710 bool same = value == userAction;
712 __dpl2(file, line, _dpct, dplchan::debuggerUserAction, 0, userAction, /*same ? " *** == *** " : */" -> ", value);
717 #define _ChangeUserAction(value) userAction = value
721 #define _ChangeState(value) ChangeState(__FILE__, __LINE__, value)
722 void ChangeState(char * file, int line, DebuggerState value)
724 #define _ChangeState(value) ChangeState(value)
725 void ChangeState(DebuggerState value)
728 bool same = value == state;
729 #if 0 //def _DEBUG_INST
730 __dpl2(file, line, _dpct, dplchan::debuggerState, 0, state, same ? " *** == *** " : " -> ", value);
733 if(!same) ide.AdjustDebugMenus();
738 _dpl2(_dpct, dplchan::debuggerCall, 0, "Debugger::CleanUp");
740 stackFrames.Free(Frame::Free);
750 waitingForPID = false;
755 sentBreakInsert = false;
756 ignoreBreakpoints = false;
759 activeFrameLevel = 0;
776 bpRunToCursor = null;
778 delete currentCompiler;
781 WatchesReleaseCodeEditor();
784 projectsLibraryLoaded.Free();
786 /*GdbThread gdbThread
792 _dpl2(_dpct, dplchan::debuggerCall, 0, "Debugger::constructor");
793 ideProcessId = Process_GetCurrentProcessId();
795 sysBPs.Add((intBpEntry = Breakpoint { type = internalEntry, enabled = false, level = -1 }));
796 sysBPs.Add((intBpMain = Breakpoint { type = internalMain, function = "main", enabled = true, level = -1 }));
797 #if defined(__WIN32__)
798 sysBPs.Add((intBpWinMain = Breakpoint { type = internalWinMain, function = "WinMain", enabled = true, level = -1 }));
800 sysBPs.Add(Breakpoint { type = internalModulesLoaded, enabled = true, level = -1 });
801 sysBPs.Add(Breakpoint { type = internalModuleLoad, function = "InternalModuleLoadBreakpoint", enabled = true, level = -1 });
806 _dpl2(_dpct, dplchan::debuggerCall, 0, "Debugger::destructor");
814 property bool isActive { get { return state == running || state == stopped; } }
815 property bool isPrepared { get { return state == loaded || state == running || state == stopped; } }
819 _dpl2(_dpct, dplchan::debuggerCall, 0, "Debugger::Resume");
820 _ChangeUserAction(resume);
821 GdbExecContinue(true);
826 _dpl2(_dpct, dplchan::debuggerCall, 0, "Debugger::Break");
827 _ChangeUserAction(_break);
831 GdbDebugBreak(false);
837 _dpl2(_dpct, dplchan::debuggerCall, 0, "Debugger::Stop");
838 _ChangeUserAction(stop);
845 GdbDebugBreak(false);
859 void Restart(CompilerConfig compiler, ProjectConfig config, int bitDepth, bool useValgrind)
861 _dpl2(_dpct, dplchan::debuggerCall, 0, "Debugger::Restart");
862 _ChangeUserAction(restart);
863 if(StartSession(compiler, config, bitDepth, useValgrind, true, false) == loaded)
867 bool GoToCodeLine(char * location)
870 _dpl2(_dpct, dplchan::debuggerCall, 0, "Debugger::GoToCodeLine(", location, ")");
871 codloc = CodeLocation::ParseCodeLocation(location);
874 CodeEditor editor = (CodeEditor)ide.OpenFile(codloc.absoluteFile, false, true, null, no, normal, false);
877 EditBox editBox = editor.editBox;
878 if(editBox.GoToLineNum(codloc.line - 1))
879 editBox.GoToPosition(editBox.line, codloc.line - 1, 0);
886 bool GoToStackFrameLine(int stackLevel, bool askForLocation, bool fromCallStack)
888 _dpl2(_dpct, dplchan::debuggerCall, 0, "Debugger::GoToStackFrameLine(", stackLevel, ", ", askForLocation, ")");
891 char filePath[MAX_LOCATION];
892 char sourceDir[MAX_LOCATION];
894 CodeEditor editor = null;
895 if(stackLevel == -1) // this (the two lines) is part of that fix that I would not put in for some time
897 for(frame = stackFrames.first; frame; frame = frame.next)
898 if(frame.level == stackLevel)
903 ide.callStackView.Show();
905 if(frame.absoluteFile)
906 editor = (CodeEditor)ide.OpenFile(frame.absoluteFile, false, true, null, no, normal, false);
907 if(!editor && frame.file)
908 frame.absoluteFile = ide.workspace.GetAbsolutePathFromRelative(frame.file);
909 if(!frame.absoluteFile && askForLocation && frame.file)
912 char title[MAX_LOCATION];
913 snprintf(title, sizeof(title), $"Provide source file location for %s", (s = CopySystemPath(frame.file)));
914 title[sizeof(title)-1] = 0;
916 if(SourceDirDialog(title, ide.workspace.projectDir, frame.file, sourceDir))
918 AddSourceDir(sourceDir);
919 frame.absoluteFile = ide.workspace.GetAbsolutePathFromRelative(frame.file);
922 if(!editor && frame.absoluteFile)
923 editor = (CodeEditor)ide.OpenFile(frame.absoluteFile, false, true, null, no, normal, false);
925 ide.RepositionWindows(false);
927 if(editor && frame.line)
929 EditBox editBox = editor.editBox;
930 editBox.GoToLineNum(frame.line - 1);
931 editBox.GoToPosition(editBox.line, frame.line - 1, 0);
939 void SelectThread(int thread)
941 _dpl2(_dpct, dplchan::debuggerCall, 0, "Debugger::SelectThread(", thread, ")");
942 _ChangeUserAction(selectThread);
945 if(thread != activeThread)
947 activeFrameLevel = -1;
948 ide.callStackView.Clear();
949 GdbCommand(0, false, "-thread-select %d", thread);
951 InternalSelectFrame(activeFrameLevel);
952 GoToStackFrameLine(activeFrameLevel, true, false);
956 ide.callStackView.Show();
960 void SelectFrame(int frame)
962 //_dpl2(_dpct, dplchan::debuggerCall, 0, "Debugger::SelectFrame(", frame, ")");
963 _ChangeUserAction(selectFrame);
966 if(frame != activeFrameLevel)
968 InternalSelectFrame(frame);
975 void InternalSelectFrame(int frame)
977 //_dpl2(_dpct, dplchan::debuggerCall, 0, "Debugger::InternalSelectFrame(", frame, ")");
978 activeFrameLevel = frame; // there is no active frame number in the gdb reply
979 GdbCommand(0, false, "-stack-select-frame %d", activeFrameLevel);
980 for(activeFrame = stackFrames.first; activeFrame; activeFrame = activeFrame.next)
981 if(activeFrame.level == activeFrameLevel)
985 void HandleExit(char * reason, char * code)
987 bool returnedExitCode = false;
988 char verboseExitCode[128];
990 _dpl2(_dpct, dplchan::debuggerCall, 0, "Debugger::HandleExit(", reason, ", ", code, ")");
991 _ChangeState(loaded); // this state change seems to be superfluous, might be in case of gdb crash
996 snprintf(verboseExitCode, sizeof(verboseExitCode), $" with exit code %s", code);
997 verboseExitCode[sizeof(verboseExitCode)-1] = 0;
1000 verboseExitCode[0] = '\0';
1004 // ClearBreakDisplay();
1008 for(wh : ide.workspace.watches)
1010 if(wh.type) FreeType(wh.type);
1013 ide.watchesView.UpdateWatch(wh);
1017 #if defined(__unix__)
1020 progThread.terminate = true;
1023 fifoFile.CloseInput();
1033 char program[MAX_LOCATION];
1034 GetSystemPathBuffer(program, targetFile);
1036 ide.outputView.debugBox.Logf($"The program %s has exited%s.\n", program, verboseExitCode);
1037 else if(!strcmp(reason, "exited-normally"))
1038 ide.outputView.debugBox.Logf($"The program %s has exited normally%s.\n", program, verboseExitCode);
1039 else if(!strcmp(reason, "exited"))
1040 ide.outputView.debugBox.Logf($"The program %s has exited%s.\n", program, verboseExitCode);
1041 else if(!strcmp(reason, "exited-signalled"))
1042 ide.outputView.debugBox.Logf($"The program %s has exited with a signal%s.\n", program, verboseExitCode);
1044 ide.outputView.debugBox.Logf($"The program %s has exited (gdb provided an unknown reason)%s.\n", program, verboseExitCode);
1049 DebuggerState StartSession(CompilerConfig compiler, ProjectConfig config, int bitDepth, bool useValgrind, bool restart, bool ignoreBreakpoints)
1051 DebuggerState result = none;
1052 _dpl2(_dpct, dplchan::debuggerCall, 0, "Debugger::StartSession(restart(", restart, "), ignoreBreakpoints(", ignoreBreakpoints, ")");
1053 if(restart && state == running && targetProcessId)
1055 breakType = DebuggerAction::restart;
1056 GdbDebugBreak(false);
1060 if(restart && state == stopped)
1062 if(needReset && state == loaded)
1063 GdbExit(); // this reset is to get a clean state with all the breakpoints until a better state can be maintained on program exit
1065 if(result == none || result == terminated)
1067 ide.outputView.ShowClearSelectTab(debug);
1068 ide.outputView.debugBox.Logf($"Starting debug mode\n");
1075 for(bp : ide.workspace.breakpoints)
1081 if(GdbInit(compiler, config, bitDepth, useValgrind))
1086 this.ignoreBreakpoints = ignoreBreakpoints;
1091 void Start(CompilerConfig compiler, ProjectConfig config, int bitDepth, bool useValgrind)
1093 _dpl2(_dpct, dplchan::debuggerCall, 0, "Debugger::Start()");
1094 _ChangeUserAction(start);
1095 if(StartSession(compiler, config, bitDepth, useValgrind, true, false) == loaded)
1099 void StepInto(CompilerConfig compiler, ProjectConfig config, int bitDepth, bool useValgrind)
1101 _dpl2(_dpct, dplchan::debuggerCall, 0, "Debugger::StepInto()");
1102 _ChangeUserAction(stepInto);
1103 switch(StartSession(compiler, config, bitDepth, useValgrind, false, false))
1105 case loaded: GdbExecRun(); break;
1106 case stopped: GdbExecStep(); break;
1110 void StepOver(CompilerConfig compiler, ProjectConfig config, int bitDepth, bool useValgrind, bool ignoreBreakpoints)
1112 _dpl2(_dpct, dplchan::debuggerCall, 0, "Debugger::StepOver()");
1113 _ChangeUserAction(stepOver);
1114 switch(StartSession(compiler, config, bitDepth, useValgrind, false, ignoreBreakpoints))
1116 case loaded: GdbExecRun(); break;
1117 case stopped: GdbExecNext(); break;
1121 void StepUntil(CompilerConfig compiler, ProjectConfig config, int bitDepth, bool useValgrind, bool ignoreBreakpoints)
1123 _dpl2(_dpct, dplchan::debuggerCall, 0, "Debugger::StepUntil()");
1124 _ChangeUserAction(stepUntil);
1125 switch(StartSession(compiler, config, bitDepth, useValgrind, false, ignoreBreakpoints))
1127 case loaded: GdbExecRun(); break;
1128 case stopped: GdbExecUntil(null, 0); break;
1132 void StepOut(bool ignoreBreakpoints)
1134 _dpl2(_dpct, dplchan::debuggerCall, 0, "Debugger::StepOut()");
1135 _ChangeUserAction(stepOut);
1136 if(state == stopped)
1138 this.ignoreBreakpoints = ignoreBreakpoints;
1142 GdbExecContinue(true);
1146 void RunToCursor(CompilerConfig compiler, ProjectConfig config, int bitDepth, bool useValgrind, char * absoluteFilePath, int lineNumber, bool ignoreBreakpoints, bool atSameLevel, bool oldImplementation)
1148 char relativeFilePath[MAX_LOCATION];
1149 _dpl2(_dpct, dplchan::debuggerCall, 0, "Debugger::RunToCursor()");
1150 _ChangeUserAction(runToCursor);
1151 WorkspaceGetRelativePath(absoluteFilePath, relativeFilePath, null);
1153 if(bpRunToCursor && bpRunToCursor.inserted && symbols)
1155 UnsetBreakpoint(bpRunToCursor);
1156 delete bpRunToCursor;
1159 StartSession(compiler, config, bitDepth, useValgrind, false, ignoreBreakpoints);
1162 if(oldImplementation)
1164 bpRunToCursor = Breakpoint { };
1165 bpRunToCursor.absoluteFilePath = absoluteFilePath;
1166 bpRunToCursor.relativeFilePath = relativeFilePath;
1167 bpRunToCursor.line = lineNumber;
1168 bpRunToCursor.type = runToCursor;
1169 bpRunToCursor.enabled = true;
1170 bpRunToCursor.level = atSameLevel ? frameCount - activeFrameLevel -1 : -1;
1175 breakType = advance;
1176 breakString = PrintString(relativeFilePath, ":", lineNumber);
1179 else if(state == stopped)
1181 if(oldImplementation)
1182 GdbExecContinue(true);
1186 GdbExecUntil(absoluteFilePath, lineNumber);
1188 GdbExecAdvance(absoluteFilePath, lineNumber);
1193 void GetCallStackCursorLine(bool * error, int * lineCursor, int * lineTopFrame)
1195 //_dpl2(_dpct, dplchan::debuggerCall, 0, "Debugger::GetCallStackCursorLine()");
1196 if(activeFrameLevel == -1)
1204 *error = signalOn && activeThread == signalThread;
1205 *lineCursor = activeFrameLevel - ((frameCount > 192 && activeFrameLevel > 191) ? frameCount - 192 - 1 : 0) + 1;
1206 *lineTopFrame = activeFrameLevel ? 1 : 0;
1210 int GetMarginIconsLineNumbers(char * fileName, int lines[], bool enabled[], int max, bool * error, int * lineCursor, int * lineTopFrame)
1212 char winFilePath[MAX_LOCATION];
1213 char * absoluteFilePath = GetSlashPathBuffer(winFilePath, fileName);
1215 Iterator<Breakpoint> it { ide.workspace.breakpoints };
1216 //_dpl2(_dpct, dplchan::debuggerCall, 0, "Debugger::GetMarginIconsLineNumbers()");
1217 while(it.Next() && count < max)
1219 Breakpoint bp = it.data;
1222 if(bp.absoluteFilePath && bp.absoluteFilePath[0] && !fstrcmp(bp.absoluteFilePath, absoluteFilePath))
1224 lines[count] = bp.line;
1225 enabled[count] = bp.enabled;
1230 if(activeFrameLevel == -1)
1238 *error = signalOn && activeThread == signalThread;
1239 if(activeFrame && activeFrame.absoluteFile && !fstrcmp(absoluteFilePath, activeFrame.absoluteFile))
1240 *lineCursor = activeFrame.line;
1243 if(activeFrame && stopItem && stopItem.frame && activeFrame.level == stopItem.frame.level)
1245 else if(stopItem && stopItem.frame && stopItem.frame.absoluteFile && !fstrcmp(absoluteFilePath, stopItem.frame.absoluteFile))
1246 *lineTopFrame = stopItem.frame.line;
1250 if(*lineTopFrame == *lineCursor && *lineTopFrame)
1256 void ChangeWatch(DataRow row, char * expression)
1258 Watch wh = (Watch)row.tag;
1259 //_dpl2(_dpct, dplchan::debuggerCall, 0, "Debugger::ChangeWatch(", expression, ")");
1262 delete wh.expression;
1264 wh.expression = CopyString(expression);
1267 Iterator<Watch> it { ide.workspace.watches };
1269 ide.workspace.watches.Delete(it.pointer);
1275 row.tag = (int64)wh;
1276 ide.workspace.watches.Add(wh);
1278 wh.expression = CopyString(expression);
1280 ide.workspace.Save();
1281 //if(expression && state == stopped)
1286 void MoveIcons(char * fileName, int lineNumber, int move, bool start)
1288 char winFilePath[MAX_LOCATION];
1289 char * absoluteFilePath = GetSlashPathBuffer(winFilePath, fileName);
1292 //_dpl2(_dpct, dplchan::debuggerCall, 0, "Debugger::MoveIcons()");
1293 for(bpLink = ide.workspace.breakpoints.first; bpLink; bpLink = next)
1295 Breakpoint bp = (Breakpoint)bpLink.data;
1298 if(bp.type == user && bp.absoluteFilePath && !fstrcmp(bp.absoluteFilePath, absoluteFilePath))
1300 if(bp.line > lineNumber || (bp.line == lineNumber && start))
1302 if(move < 0 && (bp.line < lineNumber - move))
1303 ide.workspace.RemoveBreakpoint(bp);
1307 ide.breakpointsView.UpdateBreakpoint(bp.row);
1308 ide.workspace.Save();
1314 // moving code cursors is futile, on next step, stop, hit, cursors will be offset anyways
1317 bool SourceDirDialog(char * title, char * startDir, char * test, char * sourceDir)
1321 String srcDir = null;
1323 _dpl2(_dpct, dplchan::debuggerCall, 0, "Debugger::SourceDirDialog()");
1324 debuggerFileDialog.text = title;
1325 debuggerFileDialog.currentDirectory = startDir;
1326 debuggerFileDialog.master = ide;
1328 while(debuggerFileDialog.Modal())
1330 strcpy(sourceDir, debuggerFileDialog.filePath);
1331 if(!fstrcmp(ide.workspace.projectDir, sourceDir) &&
1332 MessageBox { type = yesNo, master = ide,
1333 contents = $"This is the project directory.\nWould you like to try again?",
1334 text = $"Invalid Source Directory" }.Modal() == no)
1338 for(dir : ide.workspace.sourceDirs)
1340 if(!fstrcmp(dir, sourceDir))
1348 MessageBox { type = yesNo, master = ide,
1349 contents = $"This source directory is already specified.\nWould you like to try again?",
1350 text = $"Invalid Source Directory" }.Modal() == no)
1356 char file[MAX_LOCATION];
1357 strcpy(file, sourceDir);
1358 PathCat(file, test);
1359 result = FileExists(file);
1361 MessageBox { type = yesNo, master = ide,
1362 contents = $"Unable to locate source file.\nWould you like to try again?",
1363 text = $"Invalid Source Directory" }.Modal() == no)
1377 void AddSourceDir(char * sourceDir)
1379 _dpl2(_dpct, dplchan::debuggerCall, 0, "Debugger::AddSourceDir(", sourceDir, ")");
1380 ide.workspace.sourceDirs.Add(CopyString(sourceDir));
1381 ide.workspace.Save();
1385 DebuggerState oldState = state;
1390 GdbDebugBreak(true);
1393 GdbCommand(0, false, "-environment-directory \"%s\"", sourceDir);
1396 if(oldState == running)
1397 GdbExecContinue(false);
1401 void ToggleBreakpoint(char * fileName, int lineNumber)
1403 char absolutePath[MAX_LOCATION];
1404 Breakpoint bp = null;
1406 _dpl2(_dpct, dplchan::debuggerBreakpoints, 0, "Debugger::ToggleBreakpoint(", fileName, ":", lineNumber, ")");
1408 GetSlashPathBuffer(absolutePath, fileName);
1409 for(i : ide.workspace.breakpoints; i.type == user && i.absoluteFilePath && !fstrcmp(i.absoluteFilePath, absolutePath) && i.line == lineNumber)
1418 ide.workspace.RemoveBreakpoint(bp);
1427 char relativePath[MAX_LOCATION];
1429 WorkspaceGetRelativePath(absolutePath, relativePath, &owner);
1431 if(!owner && !FileExists(absolutePath))
1433 char title[MAX_LOCATION];
1434 char directory[MAX_LOCATION];
1435 char sourceDir[MAX_LOCATION];
1436 StripLastDirectory(absolutePath, directory);
1437 snprintf(title, sizeof(title), $"Provide source files location directory for %s", relativePath);
1438 title[sizeof(title)-1] = 0;
1441 String srcDir = null;
1442 for(dir : ide.workspace.sourceDirs)
1444 if(IsPathInsideOf(absolutePath, dir))
1446 MakePathRelative(absolutePath, dir, relativePath);
1454 if(SourceDirDialog(title, directory, null, sourceDir))
1456 if(IsPathInsideOf(absolutePath, sourceDir))
1458 AddSourceDir(sourceDir);
1459 MakePathRelative(absolutePath, sourceDir, relativePath);
1462 else if(MessageBox { type = yesNo, master = ide,
1463 contents = $"You must provide a valid source directory in order to place a breakpoint in this file.\nWould you like to try again?",
1464 text = $"Invalid Source Directory" }.Modal() == no)
1471 ide.workspace.bpCount++;
1472 bp = { line = lineNumber, type = user, enabled = true, level = -1, project = owner };
1473 ide.workspace.breakpoints.Add(bp);
1474 bp.absoluteFilePath = absolutePath;
1475 bp.relativeFilePath = relativePath;
1476 ide.breakpointsView.AddBreakpoint(bp);
1481 DebuggerState oldState = state;
1486 GdbDebugBreak(true);
1489 if(!SetBreakpoint(bp, false))
1490 SetBreakpoint(bp, true);
1493 if(oldState == running)
1494 GdbExecContinue(false);
1497 ide.workspace.Save();
1500 void UpdateRemovedBreakpoint(Breakpoint bp)
1502 _dpl2(_dpct, dplchan::debuggerCall, 0, "Debugger::UpdateRemovedBreakpoint()");
1503 if(targeted && bp.inserted)
1505 DebuggerState oldState = state;
1510 GdbDebugBreak(true);
1513 UnsetBreakpoint(bp);
1516 if(oldState == running)
1517 GdbExecContinue(false);
1523 void ParseFrame(Frame frame, char * string)
1526 Array<char *> frameTokens { minAllocSize = 50 };
1527 Array<char *> argsTokens { minAllocSize = 50 };
1528 Array<char *> argumentTokens { minAllocSize = 50 };
1529 DebugListItem item { };
1532 //_dpl2(_dpct, dplchan::debuggerCall, 0, "Debugger::ParseFrame()");
1533 TokenizeList(string, ',', frameTokens);
1534 for(i = 0; i < frameTokens.count; i++)
1536 if(TokenizeListItem(frameTokens[i], item))
1538 StripQuotes(item.value, item.value);
1539 if(!strcmp(item.name, "level"))
1540 frame.level = atoi(item.value);
1541 else if(!strcmp(item.name, "addr"))
1542 frame.addr = item.value;
1543 else if(!strcmp(item.name, "func"))
1544 frame.func = item.value;
1545 else if(!strcmp(item.name, "args"))
1547 if(!strcmp(item.value, "[]"))
1548 frame.argsCount = 0;
1551 item.value = StripBrackets(item.value);
1552 TokenizeList(item.value, ',', argsTokens);
1553 for(j = 0; j < argsTokens.count; j++)
1555 argsTokens[j] = StripCurlies(argsTokens[j]);
1556 TokenizeList(argsTokens[j], ',', argumentTokens);
1557 for(k = 0; k < argumentTokens.count; k++)
1560 frame.args.Add(arg);
1561 if(TokenizeListItem(argumentTokens[k], item))
1563 if(!strcmp(item.name, "name"))
1565 StripQuotes(item.value, item.value);
1566 arg.name = item.value;
1568 else if(!strcmp(item.name, "value"))
1570 StripQuotes(item.value, item.value);
1571 arg.val = item.value;
1574 _dpl2(_dpct, dplchan::gdbProtoUnknown, 0, "frame args item (", item.name, "=", item.value, ") is unheard of");
1577 _dpl(0, "Bad frame args item");
1579 argumentTokens.RemoveAll();
1581 frame.argsCount = argsTokens.count;
1582 argsTokens.RemoveAll();
1585 else if(!strcmp(item.name, "from"))
1586 frame.from = item.value;
1587 else if(!strcmp(item.name, "file"))
1588 frame.file = item.value;
1589 else if(!strcmp(item.name, "line"))
1590 frame.line = atoi(item.value);
1591 else if(!strcmp(item.name, "fullname"))
1593 // GDB 6.3 on OS X is giving "fullname" and "dir", all in absolute, but file name only in 'file'
1594 Workspace ws = ide.workspace;
1597 String path = ide.workspace.GetPathWorkspaceRelativeOrAbsolute(item.value);
1598 if(strcmp(frame.file, path))
1602 frame.absoluteFile = item.value;
1605 _dpl2(_dpct, dplchan::gdbProtoUnknown, 0, "frame member (", item.name, "=", item.value, ") is unheard of");
1608 _dpl(0, "Bad frame");
1613 delete argumentTokens;
1617 Breakpoint GetBreakpointById(int id, bool * isInternal)
1619 Breakpoint bp = null;
1620 //_dpl2(_dpct, dplchan::debuggerBreakpoints, 0, "Debugger::GetBreakpointById(", id, ")");
1622 *isInternal = false;
1625 for(i : sysBPs; i.bp && i.bp.id == id)
1632 if(!bp && bpRunToCursor && bpRunToCursor.bp && bpRunToCursor.bp.id == id)
1636 for(i : ide.workspace.breakpoints; i.bp && i.bp.id == id)
1646 GdbDataBreakpoint ParseBreakpoint(char * string, Array<char *> outTokens)
1649 GdbDataBreakpoint bp { };
1650 DebugListItem item { };
1651 Array<char *> bpTokens { minAllocSize = 16 };
1652 _dpl2(_dpct, dplchan::debuggerCall, 0, "Debugger::ParseBreakpoint()");
1653 string = StripCurlies(string);
1654 TokenizeList(string, ',', bpTokens);
1655 for(i = 0; i < bpTokens.count; i++)
1657 if(TokenizeListItem(bpTokens[i], item))
1659 StripQuotes(item.value, item.value);
1660 if(!strcmp(item.name, "number"))
1662 if(!strchr(item.value, '.'))
1663 bp.id = atoi(item.value);
1664 bp.number = item.value;
1666 else if(!strcmp(item.name, "type"))
1667 bp.type = item.value;
1668 else if(!strcmp(item.name, "disp"))
1669 bp.disp = item.value;
1670 else if(!strcmp(item.name, "enabled"))
1671 bp.enabled = (!strcmpi(item.value, "y"));
1672 else if(!strcmp(item.name, "addr"))
1674 if(outTokens && !strcmp(item.value, "<MULTIPLE>"))
1677 Array<GdbDataBreakpoint> bpArray = bp.multipleBPs = { };
1678 while(outTokens.count > ++c)
1680 GdbDataBreakpoint multBp = ParseBreakpoint(outTokens[c], null);
1681 bpArray.Add(multBp);
1685 bp.addr = item.value;
1687 else if(!strcmp(item.name, "func"))
1688 bp.func = item.value;
1689 else if(!strcmp(item.name, "file"))
1690 bp.file = item.value;
1691 else if(!strcmp(item.name, "fullname"))
1692 bp.fullname = item.value;
1693 else if(!strcmp(item.name, "line"))
1694 bp.line = atoi(item.value);
1695 else if(!strcmp(item.name, "at"))
1697 else if(!strcmp(item.name, "times"))
1698 bp.times = atoi(item.value);
1699 else if(!strcmp(item.name, "original-location") || !strcmp(item.name, "thread-groups"))
1700 _dpl2(_dpct, dplchan::gdbProtoIgnored, 0, "breakpoint member (", item.name, "=", item.value, ") is ignored");
1702 _dpl2(_dpct, dplchan::gdbProtoUnknown, 0, "breakpoint member (", item.name, "=", item.value, ") is unheard of");
1710 void ShowDebuggerViews()
1712 //_dpl2(_dpct, dplchan::debuggerCall, 0, "Debugger::ShowDebuggerViews()");
1713 ide.outputView.Show();
1714 ide.outputView.SelectTab(debug);
1715 ide.threadsView.Show();
1716 ide.callStackView.Show();
1717 ide.watchesView.Show();
1721 void HideDebuggerViews()
1723 //_dpl2(_dpct, dplchan::debuggerCall, 0, "Debugger::HideDebuggerViews()");
1724 ide.RepositionWindows(true);
1727 bool ::GdbCommand(Time timeOut, bool focus, char * format, ...)
1729 bool result = false;
1733 // TODO: Improve this limit
1734 static char string[MAX_F_STRING*4];
1736 va_start(args, format);
1737 vsnprintf(string, sizeof(string), format, args);
1738 string[sizeof(string)-1] = 0;
1742 ide.debugger.serialSemaphore.TryWait();
1744 #ifdef GDB_DEBUG_CONSOLE
1745 _dpl2(_dpct, dplchan::gdbCommand, 0, string);
1747 #ifdef GDB_DEBUG_OUTPUT
1748 ide.outputView.gdbBox.Logf("cmd: %s\n", string);
1750 #ifdef GDB_DEBUG_GUI
1752 ide.gdbDialog.AddCommand(string);
1755 strcat(string,"\n");
1756 gdbHandle.Puts(string);
1759 Process_ShowWindows(targetProcessId);
1765 startTime = GetTime();
1768 if(ide.debugger.serialSemaphore.TryWait())
1775 if(GetTime() - startTime > timeOut)
1783 ide.debugger.serialSemaphore.Wait();
1792 bool ValidateBreakpoint(Breakpoint bp)
1794 _dpl2(_dpct, dplchan::debuggerCall, 0, "Debugger::ValidateBreakpoint()");
1795 if(modules && bp.line && bp.bp)
1797 if(bp.bp.line != bp.line)
1803 ide.outputView.debugBox.Logf("WOULD HAVE -- Invalid breakpoint disabled: %s:%d\n", bp.relativeFilePath, bp.line);
1805 //UnsetBreakpoint(bp);
1806 //bp.enabled = false;
1812 ide.outputView.debugBox.Logf("Debugger Error: ValidateBreakpoint error\n");
1813 bp.line = bp.bp.line;
1820 void BreakpointsMaintenance()
1822 //_dpl2(_dpct, dplchan::debuggerBreakpoints, 0, "Debugger::BreakpointsMaintenance()");
1825 if(gdbExecution.suspendInternalBreakpoints)
1827 for(bp : sysBPs; bp.inserted)
1828 UnsetBreakpoint(bp);
1832 DirExpression objDir = ide.project.GetObjDir(currentCompiler, prjConfig, bitDepth);
1833 for(bp : sysBPs; !bp.inserted)
1835 bool insert = false;
1836 if(bp.type == internalModulesLoaded)
1838 char path[MAX_LOCATION];
1839 char name[MAX_LOCATION];
1840 char fixedModuleName[MAX_FILENAME];
1843 bool moduleLoadBlock = false;
1845 ReplaceSpaces(fixedModuleName, ide.project.moduleName);
1846 snprintf(name, sizeof(name),"%s.main.ec", fixedModuleName);
1847 name[sizeof(name)-1] = 0;
1848 strcpy(path, ide.workspace.projectDir);
1849 PathCatSlash(path, objDir.dir);
1850 PathCatSlash(path, name);
1851 f = FileOpen(path, read);
1854 for(lineNumber = 1; !f.Eof(); lineNumber++)
1856 if(f.GetLine(line, sizeof(line) - 1))
1858 bool moduleLoadLine;
1859 TrimLSpaces(line, line);
1860 moduleLoadLine = !strncmp(line, "eModule_Load", strlen("eModule_Load"));
1861 if(!moduleLoadBlock && moduleLoadLine)
1862 moduleLoadBlock = true;
1863 else if(moduleLoadBlock && !moduleLoadLine && strlen(line) > 0)
1869 char relative[MAX_LOCATION];
1870 bp.absoluteFilePath = path;
1871 MakePathRelative(path, ide.workspace.projectDir, relative);
1872 bp.relativeFilePath = relative;
1873 bp.line = lineNumber;
1879 else if(bp.type == internalModuleLoad)
1883 for(prj : ide.workspace.projects)
1885 if(!strcmp(prj.moduleName, "ecere"))
1887 ProjectNode node = prj.topNode.Find("instance.c", false);
1890 char path[MAX_LOCATION];
1891 char relative[MAX_LOCATION];
1892 node.GetFullFilePath(path);
1893 bp.absoluteFilePath = path;
1894 MakePathRelative(path, prj.topNode.path, relative);
1895 bp.relativeFilePath = relative;
1907 if(!SetBreakpoint(bp, false))
1908 SetBreakpoint(bp, true);
1914 if(userAction != runToCursor && bpRunToCursor && bpRunToCursor.inserted)
1915 UnsetBreakpoint(bpRunToCursor);
1916 if(bpRunToCursor && !bpRunToCursor.inserted)
1918 if(!SetBreakpoint(bpRunToCursor, false))
1919 SetBreakpoint(bpRunToCursor, true);
1922 if(ignoreBreakpoints)
1924 for(bp : ide.workspace.breakpoints; bp.inserted)
1925 UnsetBreakpoint(bp);
1929 for(bp : ide.workspace.breakpoints; !bp.inserted && bp.type == user)
1933 if(!SetBreakpoint(bp, false))
1934 SetBreakpoint(bp, true);
1943 bp.bp = GdbDataBreakpoint { };
1950 void UnsetBreakpoint(Breakpoint bp)
1952 char * s = null; _dpl2(_dpct, dplchan::debuggerBreakpoints, 0, "Debugger::UnsetBreakpoint(", s=bp.CopyLocationString(false), ") -- ", bp.type); delete s;
1953 if(symbols && bp.inserted)
1955 GdbCommand(0, false, "-break-delete %s", bp.bp.number);
1956 bp.inserted = false;
1962 bool SetBreakpoint(Breakpoint bp, bool removePath)
1964 char * s = null; _dpl2(_dpct, dplchan::debuggerBreakpoints, 0, "Debugger::SetBreakpoint(", s=bp.CopyLocationString(false), ", ", removePath ? "**** removePath(true) ****" : "", ") -- ", bp.type); delete s;
1965 breakpointError = false;
1966 if(symbols && bp.enabled && (!bp.project || bp.project.GetTargetType(bp.project.config) == staticLibrary || bp.project == ide.project || projectsLibraryLoaded[bp.project.name]))
1968 sentBreakInsert = true;
1970 GdbCommand(0, false, "-break-insert *%s", bp.address);
1973 char * location = bp.CopyLocationString(removePath);
1974 GdbCommand(0, false, "-break-insert %s", location);
1977 if(!breakpointError)
1979 char * address = null;
1980 if(bpItem && bpItem.multipleBPs && bpItem.multipleBPs.count)
1983 GdbDataBreakpoint first = null;
1984 for(n : bpItem.multipleBPs)
1986 if(!fstrcmp(n.fullname, bp.absoluteFilePath) && !first)
1996 GdbCommand(0, false, "-break-disable %s", n.number);
2000 _dpl2(_dpct, dplchan::debuggerProblem, 0, "Debugger::SetBreakpoint -- error breakpoint already disabled.");
2005 address = CopyString(first.addr);
2006 bpItem.addr = first.addr;
2007 bpItem.func = first.func;
2008 bpItem.file = first.file;
2009 bpItem.fullname = first.fullname;
2010 bpItem.line = first.line;
2011 //bpItem.thread-groups = first.thread-groups;*/
2014 _dpl2(_dpct, dplchan::debuggerProblem, 0, "Debugger::SetBreakpoint -- error multiple breakpoints all disabled.");
2016 _dpl2(_dpct, dplchan::debuggerProblem, 0, "Debugger::SetBreakpoint -- error multiple breakpoints in exact same file not supported.");
2017 bpItem.multipleBPs.Free();
2018 delete bpItem.multipleBPs;
2023 bp.inserted = (bp.bp && bp.bp.number && strcmp(bp.bp.number, "0"));
2025 ValidateBreakpoint(bp);
2029 UnsetBreakpoint(bp);
2030 bp.address = address;
2032 SetBreakpoint(bp, removePath);
2035 return !breakpointError;
2042 _dpl2(_dpct, dplchan::debuggerCall, 0, "Debugger::GdbGetStack()");
2044 stackFrames.Free(Frame::Free);
2045 GdbCommand(0, false, "-stack-info-depth");
2047 GdbCommand(0, false, "-stack-info-depth 192");
2048 if(frameCount && frameCount <= 192)
2049 GdbCommand(0, false, "-stack-list-frames 0 %d", Min(frameCount-1, 191));
2052 GdbCommand(0, false, "-stack-list-frames 0 %d", Min(frameCount-1, 95));
2053 GdbCommand(0, false, "-stack-list-frames %d %d", Max(frameCount - 96, 96), frameCount - 1);
2055 GdbCommand(0, false, "");
2060 _dpl2(_dpct, dplchan::debuggerCall, 0, "Debugger::GdbTargetSet()");
2063 char escaped[MAX_LOCATION];
2064 strescpy(escaped, targetFile);
2065 GdbCommand(0, false, "file \"%s\"", escaped); //GDB/MI Missing Implementation in 5.1.1 but we now have -file-exec-and-symbols / -file-exec-file / -file-symbol-file
2072 const char *vgdbCommand = "/usr/bin/vgdb"; // TODO: vgdb command config option
2073 //GdbCommand(0, false, "-target-select remote | %s --pid=%d", "vgdb", targetProcessId);
2074 printf("target remote | %s --pid=%d\n", vgdbCommand, targetProcessId);
2075 GdbCommand(0, false, "target remote | %s --pid=%d", vgdbCommand, targetProcessId); // TODO: vgdb command config option
2078 GdbCommand(0, false, "info target"); //GDB/MI Missing Implementation -file-list-symbol-files and -file-list-exec-sections
2080 /*for(prj : ide.workspace.projects; prj != ide.workspace.projects.firstIterator.data)
2081 GdbCommand(0, false, "-environment-directory \"%s\"", prj.topNode.path);*/
2083 for(dir : ide.workspace.sourceDirs; dir && dir[0])
2085 bool interference = false;
2086 for(prj : ide.workspace.projects)
2088 if(!fstrcmp(prj.topNode.path, dir))
2090 interference = true;
2094 if(!interference && dir[0])
2095 GdbCommand(0, false, "-environment-directory \"%s\"", dir);
2103 /*void GdbTargetRelease()
2107 BreakpointsDeleteAll();
2108 GdbCommand(0, false, "file"); //GDB/MI Missing Implementation -target-detach
2114 void GdbDebugBreak(bool internal)
2116 _dpl2(_dpct, dplchan::debuggerCall, 0, "Debugger::GdbDebugBreak()");
2120 breakType = DebuggerAction::internal;
2122 if(ide) ide.Update(null);
2124 if(Process_Break(targetProcessId)) //GdbCommand(0, false, "-exec-interrupt");
2125 serialSemaphore.Wait();
2128 _ChangeState(loaded);
2129 targetProcessId = 0;
2134 ide.outputView.debugBox.Logf("Debugger Error: GdbDebugBreak with not target id should never happen\n");
2139 _dpl2(_dpct, dplchan::debuggerCall, 0, "Debugger::GdbExecRun()");
2144 ShowDebuggerViews();
2146 GdbExecContinue(true);
2147 else if(!GdbCommand(3, true, "-exec-run"))
2148 gdbExecution = none;
2151 void GdbExecContinue(bool focus)
2153 _dpl2(_dpct, dplchan::debuggerCall, 0, "Debugger::GdbExecContinue()");
2156 GdbCommand(0, focus, "-exec-continue");
2161 _dpl2(_dpct, dplchan::debuggerCall, 0, "Debugger::GdbExecNext()");
2162 gdbExecution = next;
2164 GdbCommand(0, true, "-exec-next");
2167 void ForceUpdateCurrentFrame()
2170 GdbCommand(0, false, "-thread-list-ids");
2171 InternalSelectFrame(activeFrameLevel);
2172 GoToStackFrameLine(activeFrameLevel, true, false);
2174 ide.ShowCodeEditor();
2175 ide.AdjustDebugMenus();
2176 ideMainFrame.Activate(); // TOFIX: ide.Activate() is not reliable (app inactive)
2180 void GdbExecUntil(char * absoluteFilePath, int lineNumber)
2182 bool forceUpdate = false;
2183 char relativeFilePath[MAX_LOCATION];
2184 _dpl2(_dpct, dplchan::debuggerCall, 0, "Debugger::GdbExecUntil()");
2185 gdbExecution = until;
2187 if(absoluteFilePath)
2189 WorkspaceGetRelativePath(absoluteFilePath, relativeFilePath, null);
2190 if(!GdbCommand(0.1, true, "-exec-until %s:%d", relativeFilePath, lineNumber))
2192 GetLastDirectory(relativeFilePath, relativeFilePath);
2193 if(GdbCommand(1, true, "-exec-until %s:%d", relativeFilePath, lineNumber))
2198 GdbCommand(0, true, "-exec-until");
2200 // This is to handle GDB 6.3 on OS X not giving us *running then *stopped:
2201 // (It may not be ideal, we may need to wait?)
2203 ForceUpdateCurrentFrame();
2206 void GdbExecAdvance(char * absoluteFilePathOrLocation, int lineNumber)
2208 bool forceUpdate = false;
2209 char relativeFilePath[MAX_LOCATION];
2210 _dpl2(_dpct, dplchan::debuggerCall, 0, "Debugger::GdbExecAdvance()");
2211 gdbExecution = advance;
2215 WorkspaceGetRelativePath(absoluteFilePathOrLocation, relativeFilePath, null);
2216 if(!GdbCommand(0.1, true, "advance %s:%d", relativeFilePath, lineNumber)) // should use -exec-advance -- GDB/MI implementation missing
2218 GetLastDirectory(relativeFilePath, relativeFilePath);
2219 if(GdbCommand(1, true, "advance %s:%d", relativeFilePath, lineNumber))
2225 if(!GdbCommand(0.1, true, "advance %s", absoluteFilePathOrLocation)) // should use -exec-advance -- GDB/MI implementation missing
2227 GetLastDirectory(absoluteFilePathOrLocation, relativeFilePath);
2228 if(GdbCommand(1, true, "advance %s", relativeFilePath))
2233 // This is to handle GDB 6.3 on OS X not giving us *running then *stopped:
2234 // (It may not be ideal, we may need to wait?)
2236 ForceUpdateCurrentFrame();
2241 _dpl2(_dpct, dplchan::debuggerCall, 0, "Debugger::GdbExecStep()");
2242 gdbExecution = step;
2244 GdbCommand(0, true, "-exec-step");
2247 void GdbExecFinish()
2249 _dpl2(_dpct, dplchan::debuggerCall, 0, "Debugger::GdbExecFinish()");
2250 gdbExecution = finish;
2252 GdbCommand(0, true, "-exec-finish");
2255 void GdbExecCommon()
2257 //_dpl2(_dpct, dplchan::debuggerCall, 0, "Debugger::GdbExecCommon()");
2258 BreakpointsMaintenance();
2261 #ifdef GDB_DEBUG_GUI
2262 void SendGDBCommand(char * command)
2264 //_dpl2(_dpct, dplchan::debuggerCall, 0, "Debugger::SendGDBCommand()");
2265 DebuggerState oldState = state;
2270 GdbDebugBreak(true);
2273 GdbCommand(0, false, command);
2276 if(oldState == running)
2277 GdbExecContinue(false);
2281 void ClearBreakDisplay()
2283 //_dpl2(_dpct, dplchan::debuggerCall, 0, "Debugger::ClearBreakDisplay()");
2285 activeFrameLevel = -1;
2295 stackFrames.Free(Frame::Free);
2296 ide.callStackView.Clear();
2297 ide.threadsView.Clear();
2303 _dpl2(_dpct, dplchan::debuggerCall, 0, "Debugger::GdbAbortExec()");
2305 GdbCommand(0, false, "-interpreter-exec console \"kill\""); // should use -exec-abort -- GDB/MI implementation incomplete
2309 bool GdbInit(CompilerConfig compiler, ProjectConfig config, int bitDepth, bool useValgrind)
2312 char oldDirectory[MAX_LOCATION];
2313 char tempPath[MAX_LOCATION];
2314 char command[MAX_F_STRING*4];
2315 Project project = ide.project;
2316 DirExpression targetDirExp = project.GetTargetDir(compiler, config, bitDepth);
2317 PathBackup pathBackup { };
2318 Map<String, String> envBackup { };
2320 _dpl2(_dpct, dplchan::debuggerCall, 0, "Debugger::GdbInit()");
2321 if(currentCompiler != compiler)
2323 delete currentCompiler;
2324 currentCompiler = compiler;
2325 incref currentCompiler;
2328 this.bitDepth = bitDepth;
2329 usingValgrind = useValgrind;
2331 _ChangeState(loaded);
2333 sentBreakInsert = false;
2334 breakpointError = false;
2335 ignoreBreakpoints = false;
2341 projectsLibraryLoaded.Free();
2343 ide.outputView.ShowClearSelectTab(debug);
2344 ide.outputView.debugBox.Logf($"Starting debug mode\n");
2346 #ifdef GDB_DEBUG_OUTPUT
2347 ide.outputView.gdbBox.Logf("run: Starting GDB\n");
2350 strcpy(tempPath, ide.workspace.projectDir);
2351 PathCatSlash(tempPath, targetDirExp.dir);
2353 targetDir = CopyString(tempPath);
2354 project.CatTargetFileName(tempPath, compiler, config);
2356 targetFile = CopyString(tempPath);
2358 GetWorkingDir(oldDirectory, MAX_LOCATION);
2359 if(ide.workspace.debugDir && ide.workspace.debugDir[0])
2361 char temp[MAX_LOCATION];
2362 strcpy(temp, ide.workspace.projectDir);
2363 PathCatSlash(temp, ide.workspace.debugDir);
2364 ChangeWorkingDir(temp);
2367 ChangeWorkingDir(ide.workspace.projectDir);
2369 ide.SetPath(true, compiler, config, bitDepth);
2371 // TODO: This pollutes the environment, but at least it works
2372 // It shouldn't really affect the IDE as the PATH gets restored and other variables set for testing will unlikely cause problems
2373 // What is the proper solution for this? DualPipeOpenEnv?
2374 // gdb set environment commands don't seem to take effect
2375 for(e : ide.workspace.environmentVars)
2377 // Backing up the environment variables until we use DualPipeOpenEnv()
2378 envBackup[e.name] = CopyString(getenv(e.name));
2379 SetEnvironment(e.name, e.string);
2384 char * clArgs = ide.workspace.commandLineArgs;
2385 const char *valgrindCommand = "valgrind"; // TODO: valgrind command config option //TODO: valgrind options
2386 ValgrindLeakCheck vgLeakCheck = ide.workspace.vgLeakCheck;
2387 int vgRedzoneSize = ide.workspace.vgRedzoneSize;
2388 bool vgTrackOrigins = ide.workspace.vgTrackOrigins;
2389 vgLogFile = CreateTemporaryFile(vgLogPath, "ecereidevglog");
2393 vgLogThread.Create();
2397 ide.outputView.debugBox.Logf($"Debugger Fatal Error: Couldn't open temporary log file for Valgrind output\n");
2400 if(result && !CheckCommandAvailable(valgrindCommand))
2402 ide.outputView.debugBox.Logf($"Debugger Fatal Error: Command %s for Valgrind is not available.\n", valgrindCommand);
2407 char * vgRedzoneSizeFlag = vgRedzoneSize == -1 ? "" : PrintString(" --redzone-size=", vgRedzoneSize);
2408 sprintf(command, "%s --vgdb=yes --vgdb-error=0 --log-file=%s --leak-check=%s%s --track-origins=%s %s%s%s",
2409 valgrindCommand, vgLogPath, (char*)vgLeakCheck, vgRedzoneSizeFlag, vgTrackOrigins ? "yes" : "no", targetFile, clArgs ? " " : "", clArgs ? clArgs : "");
2410 if(vgRedzoneSize != -1)
2411 delete vgRedzoneSizeFlag;
2412 vgTargetHandle = DualPipeOpen(PipeOpenMode { output = true, /*error = true, */input = true }, command);
2415 ide.outputView.debugBox.Logf($"Debugger Fatal Error: Couldn't start Valgrind\n");
2421 incref vgTargetHandle;
2422 vgTargetThread.Create();
2424 targetProcessId = vgTargetHandle.GetProcessID();
2425 waitingForPID = false;
2426 if(!targetProcessId)
2428 ide.outputView.debugBox.Logf($"Debugger Fatal Error: Couldn't get Valgrind process ID\n");
2435 serialSemaphore.Wait();
2442 if(compiler.targetPlatform == win32)
2445 #if !((defined(__WORDSIZE) && __WORDSIZE == 8) || defined(__x86_64__))
2448 bitDepth == 32 ? "i686-w64-mingw32-gdb" : "gdb"); // x86_64-w64-mingw32-gdb
2451 // We really should have a box to select GDB in the compiler/toolchain options
2452 strcpy(command, "gdb");
2453 if(!CheckCommandAvailable(command))
2455 ide.outputView.debugBox.Logf($"Debugger Fatal Error: Command %s for GDB is not available.\n", command);
2460 strcat(command, " -n -silent --interpreter=mi2"); //-async //\"%s\"
2462 gdbHandle = DualPipeOpen(PipeOpenMode { output = true, /*error = true, */input = true }, command);
2465 ide.outputView.debugBox.Logf($"Debugger Fatal Error: Couldn't start GDB\n");
2475 gdbProcessId = gdbHandle.GetProcessID();
2478 ide.outputView.debugBox.Logf($"Debugger Fatal Error: Couldn't get GDB process ID\n");
2485 serialSemaphore.Wait();
2488 GdbCommand(0, false, "-gdb-set verbose off");
2489 //GdbCommand(0, false, "-gdb-set exec-done-display on");
2490 GdbCommand(0, false, "-gdb-set step-mode off");
2491 GdbCommand(0, false, "-gdb-set unwindonsignal on");
2492 //GdbCommand(0, false, "-gdb-set shell on");
2493 GdbCommand(0, false, "set print elements 992");
2494 GdbCommand(0, false, "-gdb-set backtrace limit 100000");
2498 //_ChangeState(terminated);
2504 #if defined(__unix__)
2506 CreateTemporaryDir(progFifoDir, "ecereide");
2507 strcpy(progFifoPath, progFifoDir);
2508 PathCat(progFifoPath, "ideprogfifo");
2509 if(!mkfifo(progFifoPath, 0600))
2511 //fileCreated = true;
2516 ide.outputView.debugBox.Logf(createFIFOMsg, progFifoPath);
2523 progThread.terminate = false;
2524 progThread.Create();
2528 #if defined(__WIN32__)
2529 GdbCommand(0, false, "-gdb-set new-console on");
2532 #if defined(__unix__)
2534 GdbCommand(0, false, "-inferior-tty-set %s", progFifoPath);
2538 GdbCommand(0, false, "-gdb-set args %s", ide.workspace.commandLineArgs ? ide.workspace.commandLineArgs : "");
2540 for(e : ide.workspace.environmentVars)
2542 GdbCommand(0, false, "set environment %s=%s", e.name, e.string);
2547 ChangeWorkingDir(oldDirectory);
2551 SetEnvironment(&e, e);
2560 delete targetDirExp;
2566 _dpl2(_dpct, dplchan::debuggerCall, 0, "Debugger::GdbExit()");
2567 if(gdbHandle && gdbProcessId)
2570 GdbCommand(0, false, "-gdb-exit");
2579 vgLogFile.CloseInput();
2580 if(vgLogThread.created)
2590 vgTargetThread.Wait();
2600 _ChangeState(terminated); // this state change seems to be superfluous, is it safety for something?
2605 for(bp : ide.workspace.breakpoints)
2611 bpRunToCursor.Reset();
2613 ide.outputView.debugBox.Logf($"Debugging stopped\n");
2615 #if defined(__unix__)
2616 if(!usingValgrind && FileExists(progFifoPath)) //fileCreated)
2618 progThread.terminate = true;
2621 fifoFile.CloseInput();
2627 DeleteFile(progFifoPath);
2628 progFifoPath[0] = '\0';
2637 bool WatchesLinkCodeEditor()
2639 bool goodFrame = activeFrame && activeFrame.absoluteFile;
2640 //_dpl2(_dpct, dplchan::debuggerCall, 0, "Debugger::WatchesLinkCodeEditor()");
2641 if(codeEditor && (!goodFrame || fstrcmp(codeEditor.fileName, activeFrame.absoluteFile)))
2642 WatchesReleaseCodeEditor();
2644 if(!codeEditor && goodFrame)
2646 codeEditor = (CodeEditor)ide.OpenFile(activeFrame.absoluteFile, false, false, null, no, normal, false);
2649 codeEditor.inUseDebug = true;
2653 return codeEditor != null;
2656 void WatchesReleaseCodeEditor()
2658 //_dpl2(_dpct, dplchan::debuggerCall, 0, "Debugger::WatchesReleaseCodeEditor()");
2661 codeEditor.inUseDebug = false;
2662 if(!codeEditor.visible)
2663 codeEditor.Destroy(0);
2668 bool ResolveWatch(Watch wh)
2670 bool result = false;
2672 _dpl2(_dpct, dplchan::debuggerWatches, 0, "Debugger::ResolveWatch()");
2684 char watchmsg[MAX_F_STRING];
2686 if(state == stopped && !codeEditor)
2687 wh.value = CopyString($"No source file found for selected frame");
2688 //if(codeEditor && state == stopped || state != stopped)
2691 Module backupPrivateModule;
2692 Context backupContext;
2693 Class backupThisClass;
2696 backupPrivateModule = GetPrivateModule();
2697 backupContext = GetCurrentContext();
2698 backupThisClass = GetThisClass();
2701 SetPrivateModule(codeEditor.privateModule);
2702 SetCurrentContext(codeEditor.globalContext);
2703 SetTopContext(codeEditor.globalContext);
2704 SetGlobalContext(codeEditor.globalContext);
2705 SetGlobalData(&codeEditor.globalData);
2708 exp = ParseExpressionString(wh.expression);
2710 if(exp && !GetParseError())
2712 char expString[4096];
2714 PrintExpression(exp, expString);
2717 if(codeEditor && activeFrame)
2718 DebugFindCtxTree(codeEditor.ast, activeFrame.line, 0);
2719 ProcessExpressionType(exp);
2721 wh.type = exp.expType;
2724 DebugComputeExpression(exp);
2725 if(exp.type == instanceExp && exp.instance.data)
2727 Symbol s = exp.instance._class ? exp.instance._class.symbol : null;
2728 Class c = s ? s.registered : null;
2732 bool needClass = false;
2733 char * s = ((char * (*)(void *, void *, void *, void *, void *))(void *)c._vTbl[__ecereVMethodID_class_OnGetString])(c, exp.instance.data, tmp, null, &needClass);
2736 FreeExpContents(exp);
2737 exp.type = constantExp;
2738 exp.isConstant = true;
2739 exp.constant = CopyString(s);
2743 else if(exp.expType && exp.expType.kind == classType && exp.expType._class && exp.expType._class.registered && exp.expType._class.registered.type == bitClass)
2745 Class c = exp.expType._class.registered;
2747 bool needClass = false;
2748 Operand op = GetOperand(exp);
2750 char * (* onGetString)(void *, void *, void *, void *, void *) = (void *)c._vTbl[__ecereVMethodID_class_OnGetString];
2753 if(op.type) op.type.refCount++;
2756 case charType: s = onGetString(c, &op.c, tmp, null, &needClass); break;
2757 case shortType: s = onGetString(c, &op.s, tmp, null, &needClass); break;
2758 case intType: s = onGetString(c, &op.i, tmp, null, &needClass); break;
2759 case int64Type: s = onGetString(c, &op.i64, tmp, null, &needClass); break;
2764 FreeExpContents(exp);
2765 exp.type = constantExp;
2766 exp.isConstant = true;
2767 exp.constant = CopyString(s);
2771 else if(exp.expType && exp.expType.kind == classType && exp.expType._class && exp.expType._class.registered && exp.expType._class.registered.type == structClass && exp.hasAddress)
2773 Class c = exp.expType._class.registered;
2774 byte * data = GdbReadMemory(exp.address, c.structSize);
2778 bool needClass = false;
2779 char * s = ((char * (*)(void *, void *, void *, void *, void *))(void *)c._vTbl[__ecereVMethodID_class_OnGetString])(c, data, tmp, null, &needClass);
2782 FreeExpContents(exp);
2783 exp.type = constantExp;
2784 exp.isConstant = true;
2785 exp.constant = CopyString(s);
2791 if(ExpressionIsError(exp) && exp.type != functionCallErrorExp)
2793 GDBFallBack(exp, expString);
2796 /*if(exp.hasAddress)
2798 char temp[MAX_F_STRING];
2799 sprintf(temp, "0x%x", exp.address);
2800 wh.address = CopyString(temp);
2801 // wh.address = CopyStringf("0x%x", exp.address);
2806 Type dataType = exp.expType;
2809 char temp[MAX_F_STRING];
2810 switch(dataType.kind)
2813 sprintf(temp, "%i", exp.val.c);
2816 sprintf(temp, "%i", exp.val.s);
2821 sprintf(temp, "%i", exp.val.i);
2824 sprintf(temp, "%i", exp.val.i64);
2827 sprintf(temp, "%i", exp.val.p);
2832 long v = (long)exp.val.f;
2833 sprintf(temp, "%i", v);
2838 long v = (long)exp.val.d;
2839 sprintf(temp, "%i", v);
2844 wh.intVal = CopyString(temp);
2845 switch(dataType.kind)
2848 sprintf(temp, "0x%x", exp.val.c);
2851 sprintf(temp, "0x%x", exp.val.s);
2855 sprintf(temp, "0x%x", exp.val.i);
2858 sprintf(temp, "0x%x", exp.val.i64);
2861 sprintf(temp, "0x%x", exp.val.i64);
2864 sprintf(temp, "0x%x", exp.val.p);
2869 long v = (long)exp.val.f;
2870 sprintf(temp, "0x%x", v);
2875 long v = (long)exp.val.d;
2876 sprintf(temp, "0x%x", v);
2881 wh.hexVal = CopyString(temp);
2882 switch(dataType.kind)
2885 sprintf(temp, "0o%o", exp.val.c);
2888 sprintf(temp, "0o%o", exp.val.s);
2892 sprintf(temp, "0o%o", exp.val.i);
2895 sprintf(temp, "0o%o", exp.val.i64);
2898 sprintf(temp, "0o%o", exp.val.i64);
2901 sprintf(temp, "0o%o", exp.val.p);
2906 long v = (long)exp.val.f;
2907 sprintf(temp, "0o%o", v);
2912 long v = (long)exp.val.d;
2913 sprintf(temp, "0o%o", v);
2918 wh.octVal = CopyString(temp);
2921 // WHATS THIS HERE ?
2922 if(exp.type == constantExp && exp.constant)
2923 wh.constant = CopyString(exp.constant);
2929 case symbolErrorExp:
2930 snprintf(watchmsg, sizeof(watchmsg), $"Symbol \"%s\" not found", exp.identifier.string);
2932 case memberSymbolErrorExp:
2935 Expression memberExp = exp.member.exp;
2936 Identifier memberID = exp.member.member;
2937 Type type = memberExp.expType;
2940 if(type.kind == structType || type.kind == unionType)
2942 char string[1024] = "";
2944 PrintTypeNoConst(type, string, false, true);
2945 snprintf(watchmsg, sizeof(watchmsg), $"Member \"%s\" not found in %s \"%s\"",
2946 memberID ? memberID.string : "", type.kind == unionType ? "union" : "struct", type.name ? type.name : string);
2950 Class _class = (memberID && memberID.classSym) ? memberID.classSym.registered : ((type.kind == classType && type._class) ? type._class.registered : null);
2951 char string[1024] = "";
2955 PrintTypeNoConst(type, string, false, true);
2956 classSym = FindClass(string);
2957 _class = classSym ? classSym.registered : null;
2960 snprintf(watchmsg, sizeof(watchmsg), $"Member \"%s\" not found in class \"%s\"", memberID ? memberID.string : "", _class.name);
2962 // NOTE: This should probably never happen, only classes and struct can be dereferenced, a dereferenceErrorExp should be displayed instead
2963 snprintf(watchmsg, sizeof(watchmsg), $"Member \"%s\" not found in type \"%s\"", memberID ? memberID.string : "", string);
2967 // NOTE: This should probably never happen, the error causing the unresolved expression should be displayed instead
2968 snprintf(watchmsg, sizeof(watchmsg), $"Accessing member \"%s\" from unresolved expression", memberID ? memberID.string : "");
2971 case memberPropertyErrorExp:
2973 Expression memberExp = exp.member.exp;
2974 Identifier memberID = exp.member.member;
2975 Type type = memberExp.expType;
2976 Class _class = (type && memberID) ? (memberID && memberID.classSym) ? memberID.classSym.registered : ((type.kind == classType && type._class) ? type._class.registered : null) : null;
2979 if(_class && memberID && memberID.string)
2980 snprintf(watchmsg, sizeof(watchmsg), $"Missing property evaluation for \"%s\" in class \"%s\"", memberID.string, _class.name);
2982 snprintf(watchmsg, sizeof(watchmsg), $"Missing property evaluation for \"%s\"", wh.expression);
2985 case functionCallErrorExp:
2986 if(exp.call.exp && exp.call.exp.type == identifierExp && exp.call.exp.identifier.string)
2987 snprintf(watchmsg, sizeof(watchmsg), $"Missing function evaluation for call to \"%s\"", exp.call.exp.identifier.string);
2989 snprintf(watchmsg, sizeof(watchmsg), $"Missing function evaluation for \"%s\"", wh.expression);
2991 case memoryErrorExp:
2992 // Need to ensure when set to memoryErrorExp, constant is set
2993 snprintf(watchmsg, sizeof(watchmsg), $"Memory can't be read at %s", /*(exp.type == constantExp) ? */exp.constant /*: null*/);
2995 case dereferenceErrorExp:
2996 snprintf(watchmsg, sizeof(watchmsg), $"Dereferencing error evaluating \"%s\"", wh.expression);
2998 case divideBy0ErrorExp:
2999 snprintf(watchmsg, sizeof(watchmsg), $"Integer division by 0");
3001 case noDebuggerErrorExp:
3002 snprintf(watchmsg, sizeof(watchmsg), $"Debugger required for symbol evaluation in \"%s\"", wh.expression);
3004 case unknownErrorExp:
3005 // NOTE: This should never happen
3006 snprintf(watchmsg, sizeof(watchmsg), $"Error evaluating \"%s\"", wh.expression);
3009 // NOTE: This should never happen
3010 snprintf(watchmsg, sizeof(watchmsg), $"Null type for \"%s\"", wh.expression);
3013 wh.value = CopyString(exp.string);
3016 // Temporary Code for displaying Strings
3017 if((exp.expType && ((exp.expType.kind == pointerType ||
3018 exp.expType.kind == arrayType) && exp.expType.type.kind == charType)) ||
3019 (wh.type && wh.type.kind == classType && wh.type._class &&
3020 wh.type._class.registered && wh.type._class.registered.type == normalClass &&
3021 !strcmp(wh.type._class.registered.name, "String")))
3024 if(exp.expType.kind != arrayType || exp.hasAddress)
3028 //char temp[MAX_F_STRING * 32];
3030 ExpressionType evalError = dummyExp;
3031 /*if(exp.expType.kind == arrayType)
3032 sprintf(temp, "(char*)0x%x", exp.address);
3034 sprintf(temp, "(char*)%s", exp.constant);*/
3036 //evaluation = Debugger::EvaluateExpression(temp, &evalError);
3037 // address = strtoul(exp.constant, null, 0);
3038 address = _strtoui64(exp.constant, null, 0);
3039 //_dpl(0, "0x", address);
3040 // snprintf(value, sizeof(value), "0x%08x ", address);
3042 if(address > 0xFFFFFFFFLL)
3043 snprintf(value, sizeof(value), (GetRuntimePlatform() == win32) ? "0x%016I64x " : "0x%016llx ", address);
3045 snprintf(value, sizeof(value), (GetRuntimePlatform() == win32) ? "0x%08I64x " : "0x%08llx ", address);
3046 value[sizeof(value)-1] = 0;
3049 strcat(value, $"Null string");
3052 String string = new char[4097];
3054 bool success = false;
3058 for(start = 0; !done && start + size <= 4096; start += size)
3063 // Try to read 256 bytes at a time, then half if that fails
3064 s = GdbReadMemory(address + start, size);
3068 memcpy(string + start, s, size);
3069 string[start + size] = 0;
3070 if(size == 1 || memchr(s, 0, size))
3084 int len = strlen(value);
3086 if(UTF8Validate(string))
3091 for(c = 0; (ch = string[c]); c++)
3094 value[len++] = '\0';
3099 ISO8859_1toUTF8(string, value + len, strlen(value) - len - 30);
3100 strcat(value, ") (ISO8859-1)");
3104 strcat(value, $"Empty string");
3107 strcat(value, $"Couldn't read memory");
3110 wh.value = CopyString(value);
3113 else if(wh.type && wh.type.kind == classType && wh.type._class &&
3114 wh.type._class.registered && wh.type._class.registered.type == enumClass)
3116 uint64 value = strtoul(exp.constant, null, 0);
3117 Class enumClass = eSystem_FindClass(GetPrivateModule(), wh.type._class.registered.name);
3118 EnumClassData enumeration = (EnumClassData)enumClass.data;
3120 for(item = enumeration.values.first; item; item = item.next)
3121 if((int)item.data == value)
3124 wh.value = CopyString(item.name);
3126 wh.value = PrintString($"Invalid Enum Value (", value, ")");
3129 else if(wh.type && (wh.type.kind == charType || (wh.type.kind == classType && wh.type._class &&
3130 wh.type._class.registered && !strcmp(wh.type._class.registered.fullName, "ecere::com::unichar"))) )
3137 if(exp.constant[0] == '\'')
3139 if((int)((byte *)exp.constant)[1] > 127)
3142 value = UTF8GetChar(exp.constant + 1, &nb);
3143 if(nb < 2) value = exp.constant[1];
3144 signedValue = value;
3148 signedValue = exp.constant[1];
3150 // Precomp Syntax error with boot strap here:
3151 byte b = (byte)(char)signedValue;
3152 value = (unichar) b;
3158 if(wh.type.kind == charType && wh.type.isSigned)
3160 signedValue = (int)(char)strtol(exp.constant, null, 0);
3162 // Precomp Syntax error with boot strap here:
3163 byte b = (byte)(char)signedValue;
3164 value = (unichar) b;
3169 value = (uint)strtoul(exp.constant, null, 0);
3170 signedValue = (int)value;
3174 UTF32toUTF8Len(&value, 1, charString, 5);
3176 snprintf(string, sizeof(string), "\'\\0' (0)");
3177 else if(value == '\t')
3178 snprintf(string, sizeof(string), "\'\\t' (%d)", value);
3179 else if(value == '\n')
3180 snprintf(string, sizeof(string), "\'\\n' (%d)", value);
3181 else if(value == '\r')
3182 snprintf(string, sizeof(string), "\'\\r' (%d)", value);
3183 else if(wh.type.kind == charType && wh.type.isSigned)
3184 snprintf(string, sizeof(string), "\'%s\' (%d)", charString, signedValue);
3185 else if(value > 256 || wh.type.kind != charType)
3187 if(value > 0x10FFFF || !GetCharCategory(value))
3188 snprintf(string, sizeof(string), $"Invalid Unicode Keypoint (0x%08X)", value);
3190 snprintf(string, sizeof(string), "\'%s\' (U+%04X)", charString, value);
3193 snprintf(string, sizeof(string), "\'%s\' (%d)", charString, value);
3194 string[sizeof(string)-1] = 0;
3196 wh.value = CopyString(string);
3201 wh.value = CopyString(exp.constant);
3207 bool genericError = true;
3208 if(exp.expType && exp.hasAddress)
3210 bool showAddress = false;
3211 switch(exp.expType.kind)
3223 Symbol s = exp.expType._class;
3226 Class c = s.registered;
3227 if(c.type == noHeadClass || c.type == normalClass)
3235 wh.value = PrintHexUInt64(exp.address);
3237 genericError = false;
3241 // NOTE: This should ideally be handled with a specific error message
3242 snprintf(watchmsg, sizeof(watchmsg), $"Error evaluating \"%s\"", wh.expression);
3248 snprintf(watchmsg, sizeof(watchmsg), $"Invalid expression: \"%s\"", wh.expression);
3249 if(exp) FreeExpression(exp);
3252 SetPrivateModule(backupPrivateModule);
3253 SetCurrentContext(backupContext);
3254 SetTopContext(backupContext);
3255 SetGlobalContext(backupContext);
3256 SetThisClass(backupThisClass);
3259 // wh.value = CopyString("No source file found for selected frame");
3261 watchmsg[sizeof(watchmsg)-1] = 0;
3263 wh.value = CopyString(watchmsg);
3265 ide.watchesView.UpdateWatch(wh);
3269 void EvaluateWatches()
3271 _dpl2(_dpct, dplchan::debuggerWatches, 0, "Debugger::EvaluateWatches()");
3272 WatchesLinkCodeEditor();
3273 if(state == stopped)
3275 for(wh : ide.workspace.watches)
3280 char * ::GdbEvaluateExpression(char * expression)
3282 _dpl2(_dpct, dplchan::debuggerWatches, 0, "Debugger::GdbEvaluateExpression(", expression, ")");
3285 GdbCommand(0, false, "-data-evaluate-expression \"%s\"", expression);
3287 ide.outputView.debugBox.Logf("Debugger Error: GdbEvaluateExpression\n");
3291 // to be removed... use GdbReadMemory that returns a byte array instead
3292 char * ::GdbReadMemoryString(uint64 address, int size, char format, int rows, int cols)
3294 _dpl2(_dpct, dplchan::debuggerCall, 0, "Debugger::GdbReadMemoryString(", address, ")");
3299 _dpl(0, "GdbReadMemoryString called with size = 0!");
3301 // GdbCommand(0, false, "-data-read-memory 0x%08x %c, %d, %d, %d", address, format, size, rows, cols);
3302 if(GetRuntimePlatform() == win32)
3303 GdbCommand(0, false, "-data-read-memory 0x%016I64x %c, %d, %d, %d", address, format, size, rows, cols);
3305 GdbCommand(0, false, "-data-read-memory 0x%016llx %c, %d, %d, %d", address, format, size, rows, cols);
3307 ide.outputView.debugBox.Logf("Debugger Error: GdbReadMemoryString\n");
3311 byte * ::GdbReadMemory(uint64 address, int bytes)
3313 _dpl2(_dpct, dplchan::debuggerCall, 0, "Debugger::GdbReadMemory(", address, ")");
3316 //GdbCommand(0, false, "-data-read-memory 0x%08x %c, 1, 1, %d", address, 'u', bytes);
3317 if(GetRuntimePlatform() == win32)
3318 GdbCommand(0, false, "-data-read-memory 0x%016I64x %c, 1, 1, %d", address, 'u', bytes);
3320 GdbCommand(0, false, "-data-read-memory 0x%016llx %c, 1, 1, %d", address, 'u', bytes);
3323 _dpl(0, "GdbReadMemory called with bytes = 0!");
3326 ide.outputView.debugBox.Logf("Debugger Error: GdbReadMemory\n");
3327 else if(eval.result && strcmp(eval.result, "N/A"))
3329 byte * result = new byte[bytes];
3330 byte * string = eval.result;
3334 result[c++] = (byte)strtol(string, &string, 10);
3351 bool BreakpointHit(GdbDataStop stopItem, Breakpoint bpInternal, Breakpoint bpUser)
3354 char * s1 = null; char * s2 = null;
3355 _dpl2(_dpct, dplchan::debuggerCall, 0, "Debugger::BreakpointHit(",
3356 "bpInternal(", bpInternal ? s1=bpInternal.CopyLocationString(false) : null, "), ",
3357 "bpUser(", bpUser ? s2=bpUser.CopyLocationString(false) : null, ")) -- ",
3358 "ignoreBreakpoints(", ignoreBreakpoints, "), ",
3359 "hitCursorBreakpoint(", bpUser && bpUser.type == runToCursor, ")");
3360 delete s1; delete s2;
3364 bool conditionMet = true;
3365 if(bpUser.condition)
3367 if(WatchesLinkCodeEditor())
3368 conditionMet = ResolveWatch(bpUser.condition);
3370 conditionMet = false;
3385 if(stopItem.frame.line && bpUser.line != stopItem.frame.line)
3387 // updating user breakpoint on hit location difference
3388 // todo, print something?
3389 bpUser.line = stopItem.frame.line;
3390 ide.breakpointsView.UpdateBreakpoint(bpUser.row);
3391 ide.workspace.Save();
3394 ide.breakpointsView.UpdateBreakpoint(bpUser.row);
3399 if(bpInternal.type == internalModulesLoaded)
3401 if(userAction == stepOver)
3403 if((bpInternal.type == internalEntry && ((intBpMain && intBpMain.inserted) || (intBpWinMain && intBpWinMain.inserted))) ||
3404 (bpInternal.type == internalMain && intBpWinMain && intBpWinMain.inserted))
3407 if(!bpUser && !userAction.breaksOnInternalBreakpoint)
3409 if(userAction == stepOut)
3410 StepOut(ignoreBreakpoints);
3416 if(!bpUser && !bpInternal)
3422 void ValgrindTargetThreadExit()
3424 _dpl2(_dpct, dplchan::debuggerCall, 0, "Debugger::ValgrindTargetThreadExit()");
3427 vgTargetHandle.Wait();
3428 delete vgTargetHandle;
3430 HandleExit(null, null);
3433 void GdbThreadExit()
3435 _dpl2(_dpct, dplchan::debuggerCall, 0, "Debugger::GdbThreadExit()");
3436 if(state != terminated)
3438 _ChangeState(terminated);
3439 targetProcessId = 0;
3440 ClearBreakDisplay();
3445 serialSemaphore.Release();
3450 ide.outputView.debugBox.Logf($"Debugger Fatal Error: GDB lost\n");
3451 ide.outputView.debugBox.Logf($"Debugging stopped\n");
3453 HideDebuggerViews();
3455 //_ChangeState(terminated);
3459 void GdbThreadMain(char * output)
3463 Array<char *> outTokens { minAllocSize = 50 };
3464 Array<char *> subTokens { minAllocSize = 50 };
3465 DebugListItem item { };
3466 DebugListItem item2 { };
3467 bool setWaitingForPID = false;
3469 #if defined(GDB_DEBUG_CONSOLE) || defined(GDB_DEBUG_GUI)
3470 #ifdef GDB_DEBUG_CONSOLE
3471 // _dpl2(_dpct, dplchan::gdbOutput, 0, output);
3474 #ifdef GDB_DEBUG_OUTPUT
3476 int len = strlen(output);
3484 for(c = 0; c < len / 1024; c++)
3486 strncpy(tmp, start, 1024);
3487 ide.outputView.gdbBox.Logf("out: %s\n", tmp);
3490 ide.outputView.gdbBox.Logf("out: %s\n", start);
3494 ide.outputView.gdbBox.Logf("out: %s\n", output);
3498 #ifdef GDB_DEBUG_CONSOLE
3499 strcpy(lastGdbOutput, output);
3501 #ifdef GDB_DEBUG_GUI
3502 if(ide.gdbDialog) ide.gdbDialog.AddOutput(output);
3509 if(strstr(output, "No debugging symbols found") || strstr(output, "(no debugging symbols found)"))
3512 ide.outputView.debugBox.Logf($"Target doesn't contain debug information!\n");
3515 if(!entryPoint && (t = strstr(output, "Entry point:")))
3517 char * addr = t + strlen("Entry point:");
3519 if(*t++ == ' ' && *t++ == '0' && *t == 'x')
3522 while(isxdigit(*++t));
3524 for(bp : sysBPs; bp.type == internalEntry)
3527 bp.enabled = entryPoint = true;
3535 if(TokenizeList(output, ',', outTokens) && !strcmp(outTokens[0], "^done"))
3537 //if(outTokens.count == 1)
3542 _ChangeState(loaded);
3543 targetProcessId = 0;
3544 if(outTokens.count > 1 && TokenizeListItem(outTokens[1], item))
3546 if(!strcmp(item.name, "reason"))
3548 char * reason = item.value;
3549 StripQuotes(reason, reason);
3550 if(!strcmp(reason, "exited-normally") || !strcmp(reason, "exited") || !strcmp(reason, "exited-signalled"))
3553 if(outTokens.count > 2 && TokenizeListItem(outTokens[2], item2))
3555 StripQuotes(item2.value, item2.value);
3556 if(!strcmp(item2.name, "exit-code"))
3557 exitCode = item2.value;
3563 HandleExit(reason, exitCode);
3567 _dpl2(_dpct, dplchan::gdbProtoUnknown, 0, "kill reply (", item.name, "=", item.value, ") is unheard of");
3570 HandleExit(null, null);
3573 if(outTokens.count > 1 && TokenizeListItem(outTokens[1], item))
3575 if(!strcmp(item.name, "bkpt"))
3577 sentBreakInsert = false;
3583 bpItem = ParseBreakpoint(item.value, outTokens);
3584 //breakType = bpValidation;
3586 else if(!strcmp(item.name, "depth"))
3588 StripQuotes(item.value, item.value);
3589 frameCount = atoi(item.value);
3591 stackFrames.Free(Frame::Free);
3593 else if(!strcmp(item.name, "stack"))
3596 if(stackFrames.count)
3597 ide.callStackView.Logf("...\n");
3600 item.value = StripBrackets(item.value);
3601 TokenizeList(item.value, ',', subTokens);
3602 for(i = 0; i < subTokens.count; i++)
3604 if(TokenizeListItem(subTokens[i], item))
3606 if(!strcmp(item.name, "frame"))
3609 stackFrames.Add(frame);
3610 item.value = StripCurlies(item.value);
3611 ParseFrame(frame, item.value);
3612 if(frame.file && frame.from)
3613 _dpl2(_dpct, dplchan::gdbProtoUnknown, 0, "unexpected frame file and from members present");
3617 if(activeFrameLevel == -1)
3619 if(ide.projectView.IsModuleInProject(frame.file));
3621 if(frame.level != 0)
3623 //stopItem.frame = frame;
3624 breakType = selectFrame;
3627 activeFrame = frame;
3628 activeFrameLevel = frame.level;
3631 ide.callStackView.Logf("%3d ", frame.level);
3632 if(!strncmp(frame.func, "__ecereMethod_", strlen("__ecereMethod_")))
3633 ide.callStackView.Logf($"%s Method, %s:%d\n", &frame.func[strlen("__ecereMethod_")], (s = CopySystemPath(frame.file)), frame.line);
3634 else if(!strncmp(frame.func, "__ecereProp_", strlen("__ecereProp_")))
3635 ide.callStackView.Logf($"%s Property, %s:%d\n", &frame.func[strlen("__ecereProp_")], (s = CopySystemPath(frame.file)), frame.line);
3636 else if(!strncmp(frame.func, "__ecereConstructor_", strlen("__ecereConstructor_")))
3637 ide.callStackView.Logf($"%s Constructor, %s:%d\n", &frame.func[strlen("__ecereConstructor_")], (s = CopySystemPath(frame.file)), frame.line);
3638 else if(!strncmp(frame.func, "__ecereDestructor_", strlen("__ecereDestructor_")))
3639 ide.callStackView.Logf($"%s Destructor, %s:%d\n", &frame.func[strlen("__ecereDestructor_")], (s = CopySystemPath(frame.file)), frame.line);
3641 ide.callStackView.Logf($"%s Function, %s:%d\n", frame.func, (s = CopySystemPath(frame.file)), frame.line);
3646 ide.callStackView.Logf("%3d ", frame.level);
3651 ide.callStackView.Logf($"inside %s, %s\n", frame.func, (s = CopySystemPath(frame.from)));
3655 ide.callStackView.Logf("%s\n", frame.func);
3657 ide.callStackView.Logf($"unknown source\n");
3661 _dpl2(_dpct, dplchan::gdbProtoUnknown, 0, "stack content (", item.name, "=", item.value, ") is unheard of");
3664 if(activeFrameLevel == -1)
3666 activeFrameLevel = 0;
3667 activeFrame = stackFrames.first;
3669 ide.callStackView.Home();
3671 subTokens.RemoveAll();
3673 /*else if(!strcmp(item.name, "frame"))
3676 item.value = StripCurlies(item.value);
3677 ParseFrame(&frame, item.value);
3679 else if(!strcmp(item.name, "thread-ids"))
3681 ide.threadsView.Clear();
3682 item.value = StripCurlies(item.value);
3683 TokenizeList(item.value, ',', subTokens);
3684 for(i = subTokens.count - 1; ; i--)
3686 if(TokenizeListItem(subTokens[i], item))
3688 if(!strcmp(item.name, "thread-id"))
3691 StripQuotes(item.value, item.value);
3692 value = atoi(item.value);
3693 ide.threadsView.Logf("%3d \n", value);
3696 _dpl2(_dpct, dplchan::gdbProtoUnknown, 0, "threads content (", item.name, "=", item.value, ") is unheard of");
3701 ide.threadsView.Home();
3703 subTokens.RemoveAll();
3704 //if(!strcmp(outTokens[2], "number-of-threads"))
3706 else if(!strcmp(item.name, "new-thread-id"))
3708 StripQuotes(item.value, item.value);
3709 activeThread = atoi(item.value);
3711 else if(!strcmp(item.name, "value"))
3713 StripQuotes(item.value, item.value);
3714 eval.result = CopyString(item.value);
3715 eval.active = false;
3717 else if(!strcmp(item.name, "addr"))
3719 for(i = 2; i < outTokens.count; i++)
3721 if(TokenizeListItem(outTokens[i], item))
3723 if(!strcmp(item.name, "total-bytes"))
3725 StripQuotes(item.value, item.value);
3726 eval.bytes = atoi(item.value);
3728 else if(!strcmp(item.name, "next-row"))
3730 StripQuotes(item.value, item.value);
3731 eval.nextBlockAddress = _strtoui64(item.value, null, 0);
3733 else if(!strcmp(item.name, "memory"))
3737 //StripQuotes(item.value, item.value);
3738 item.value = StripBrackets(item.value);
3739 // this should be treated as a list...
3740 item.value = StripCurlies(item.value);
3741 TokenizeList(item.value, ',', subTokens);
3742 for(j = 0; j < subTokens.count; j++)
3744 if(TokenizeListItem(subTokens[j], item))
3746 if(!strcmp(item.name, "data"))
3748 item.value = StripBrackets(item.value);
3749 StripQuotes2(item.value, item.value);
3750 eval.result = CopyString(item.value);
3751 eval.active = false;
3755 subTokens.RemoveAll();
3760 else if(!strcmp(item.name, "source-path") || !strcmp(item.name, "BreakpointTable"))
3761 _dpl2(_dpct, dplchan::gdbProtoIgnored, 0, "command reply (", item.name, "=", item.value, ") is ignored");
3763 _dpl2(_dpct, dplchan::gdbProtoUnknown, 0, "command reply (", item.name, "=", item.value, ") is unheard of");
3766 else if(!strcmp(outTokens[0], "^running"))
3768 waitingForPID = true;
3769 setWaitingForPID = true;
3770 ClearBreakDisplay();
3772 else if(!strcmp(outTokens[0], "^exit"))
3774 _ChangeState(terminated);
3775 // ide.outputView.debugBox.Logf("Exit\n");
3776 // ide.Update(null);
3778 serialSemaphore.Release();
3780 else if(!strcmp(outTokens[0], "^error"))
3784 sentBreakInsert = false;
3785 breakpointError = true;
3788 if(outTokens.count > 1 && TokenizeListItem(outTokens[1], item))
3790 if(!strcmp(item.name, "msg"))
3792 StripQuotes(item.value, item.value);
3795 eval.active = false;
3797 if(strstr(item.value, "No symbol") && strstr(item.value, "in current context"))
3798 eval.error = symbolNotFound;
3799 else if(strstr(item.value, "Cannot access memory at address"))
3800 eval.error = memoryCantBeRead;
3802 eval.error = unknown;
3804 else if(!strcmp(item.value, "Previous frame inner to this frame (corrupt stack?)"))
3807 else if(!strncmp(item.value, "Cannot access memory at address", 31))
3810 else if(!strcmp(item.value, "Cannot find bounds of current function"))
3812 _ChangeState(stopped);
3813 gdbHandle.Printf("-exec-continue\n");
3815 else if(!strcmp(item.value, "ptrace: No such process."))
3817 _ChangeState(loaded);
3818 targetProcessId = 0;
3820 else if(!strcmp(item.value, "Function \\\"WinMain\\\" not defined."))
3823 else if(!strcmp(item.value, "You can't do that without a process to debug."))
3825 _ChangeState(loaded);
3826 targetProcessId = 0;
3828 else if(strstr(item.value, "No such file or directory."))
3830 _ChangeState(loaded);
3831 targetProcessId = 0;
3833 else if(strstr(item.value, "During startup program exited with code "))
3835 _ChangeState(loaded);
3836 targetProcessId = 0;
3841 if(strlen(item.value) < MAX_F_STRING)
3844 ide.outputView.debugBox.Logf("GDB: %s\n", (s = CopyUnescapedString(item.value)));
3848 ide.outputView.debugBox.Logf("GDB: %s\n", item.value);
3854 _dpl2(_dpct, dplchan::gdbProtoUnknown, 0, "error content (", item.name, "=", item.value, ") is unheard of");
3857 _dpl2(_dpct, dplchan::gdbProtoUnknown, 0, "result-record: ", outTokens[0]);
3858 outTokens.RemoveAll();
3861 _dpl2(_dpct, dplchan::gdbProtoUnknown, 0, "status-async-output: ", outTokens[0]);
3864 if(TokenizeList(output, ',', outTokens))
3866 if(!strcmp(outTokens[0], "=library-loaded"))
3867 FGODetectLoadedLibraryForAddedProjectIssues(outTokens, false);
3868 else if(!strcmp(outTokens[0], "=shlibs-added"))
3869 FGODetectLoadedLibraryForAddedProjectIssues(outTokens, true);
3870 else if(!strcmp(outTokens[0], "=thread-group-created") || !strcmp(outTokens[0], "=thread-group-added") ||
3871 !strcmp(outTokens[0], "=thread-group-started") || !strcmp(outTokens[0], "=thread-group-exited") ||
3872 !strcmp(outTokens[0], "=thread-created") || !strcmp(outTokens[0], "=thread-exited") ||
3873 !strcmp(outTokens[0], "=cmd-param-changed") || !strcmp(outTokens[0], "=library-unloaded") ||
3874 !strcmp(outTokens[0], "=breakpoint-modified"))
3875 _dpl2(_dpct, dplchan::gdbProtoIgnored, 0, outTokens[0], outTokens.count>1 ? outTokens[1] : "",
3876 outTokens.count>2 ? outTokens[2] : "", outTokens.count>3 ? outTokens[3] : "",
3877 outTokens.count>4 ? outTokens[4] : "", outTokens.count>5 ? outTokens[5] : "",
3878 outTokens.count>6 ? outTokens[6] : "", outTokens.count>7 ? outTokens[7] : "",
3879 outTokens.count>8 ? outTokens[8] : "", outTokens.count>9 ? outTokens[9] : "");
3881 _dpl2(_dpct, dplchan::gdbProtoUnknown, 0, "notify-async-output: ", outTokens[0]);
3883 outTokens.RemoveAll();
3887 if(TokenizeList(output, ',', outTokens))
3889 if(!strcmp(outTokens[0],"*running"))
3891 waitingForPID = true;
3892 setWaitingForPID = true;
3894 else if(!strcmp(outTokens[0], "*stopped"))
3897 _ChangeState(stopped);
3899 for(tk = 1; tk < outTokens.count; tk++)
3901 if(TokenizeListItem(outTokens[tk], item))
3903 if(!strcmp(item.name, "reason"))
3905 char * reason = item.value;
3906 StripQuotes(reason, reason);
3907 if(!strcmp(reason, "exited-normally") || !strcmp(reason, "exited") || !strcmp(reason, "exited-signalled"))
3910 if(outTokens.count > tk+1 && TokenizeListItem(outTokens[tk+1], item2))
3913 StripQuotes(item2.value, item2.value);
3914 if(!strcmp(item2.name, "exit-code"))
3915 exitCode = item2.value;
3921 HandleExit(reason, exitCode);
3924 else if(!strcmp(reason, "breakpoint-hit") ||
3925 !strcmp(reason, "function-finished") ||
3926 !strcmp(reason, "end-stepping-range") ||
3927 !strcmp(reason, "location-reached") ||
3928 !strcmp(reason, "signal-received"))
3932 if(stopItem) _dpl(0, "problem");
3934 stopItem = GdbDataStop { };
3935 stopItem.reason = r == 'b' ? breakpointHit : r == 'f' ? functionFinished : r == 'e' ? endSteppingRange : r == 'l' ? locationReached : signalReceived;
3937 for(i = tk+1; i < outTokens.count; i++)
3939 TokenizeListItem(outTokens[i], item);
3940 StripQuotes(item.value, item.value);
3941 if(!strcmp(item.name, "thread-id"))
3942 stopItem.threadid = atoi(item.value);
3943 else if(!strcmp(item.name, "frame"))
3945 item.value = StripCurlies(item.value);
3946 ParseFrame(stopItem.frame, item.value);
3948 else if(stopItem.reason == breakpointHit && !strcmp(item.name, "bkptno"))
3949 stopItem.bkptno = atoi(item.value);
3950 else if(stopItem.reason == functionFinished && !strcmp(item.name, "gdb-result-var"))
3951 stopItem.gdbResultVar = CopyString(item.value);
3952 else if(stopItem.reason == functionFinished && !strcmp(item.name, "return-value"))
3953 stopItem.returnValue = CopyString(item.value);
3954 else if(stopItem.reason == signalReceived && !strcmp(item.name, "signal-name"))
3955 stopItem.name = CopyString(item.value);
3956 else if(stopItem.reason == signalReceived && !strcmp(item.name, "signal-meaning"))
3957 stopItem.meaning = CopyString(item.value);
3958 else if(!strcmp(item.name, "stopped-threads"))
3959 _dpl2(_dpct, dplchan::gdbProtoIgnored, 0, reason, ": Advanced thread debugging not handled");
3960 else if(!strcmp(item.name, "core"))
3961 _dpl2(_dpct, dplchan::gdbProtoIgnored, 0, reason, ": Information (core) not used");
3962 else if(!strcmp(item.name, "disp"))
3963 _dpl2(_dpct, dplchan::gdbProtoIgnored, 0, reason, ": (", item.name, "=", item.value, ")");
3965 _dpl2(_dpct, dplchan::gdbProtoUnknown, 0, "Unknown ", reason, " item name (", item.name, "=", item.value, ")");
3968 if(stopItem.reason == signalReceived && !strcmp(stopItem.name, "SIGTRAP"))
3984 event = r == 'b' ? hit : r == 'f' ? functionEnd : r == 'e' ? stepEnd : r == 'l' ? locationReached : signal;
3988 else if(!strcmp(reason, "watchpoint-trigger"))
3989 _dpl2(_dpct, dplchan::gdbProtoIgnored, 0, "Reason watchpoint trigger not handled");
3990 else if(!strcmp(reason, "read-watchpoint-trigger"))
3991 _dpl2(_dpct, dplchan::gdbProtoIgnored, 0, "Reason read watchpoint trigger not handled");
3992 else if(!strcmp(reason, "access-watchpoint-trigger"))
3993 _dpl2(_dpct, dplchan::gdbProtoIgnored, 0, "Reason access watchpoint trigger not handled");
3994 else if(!strcmp(reason, "watchpoint-scope"))
3995 _dpl2(_dpct, dplchan::gdbProtoIgnored, 0, "Reason watchpoint scope not handled");
3997 _dpl2(_dpct, dplchan::gdbProtoUnknown, 0, "Unknown reason: ", reason);
4005 if(usingValgrind && event == none && !stopItem)
4006 event = valgrindStartPause;
4011 _dpl2(_dpct, dplchan::gdbProtoUnknown, 0, "Unknown exec-async-output: ", outTokens[0]);
4012 outTokens.RemoveAll();
4015 if(!strcmpi(output, "(gdb) "))
4019 Time startTime = GetTime();
4020 char exeFile[MAX_LOCATION];
4021 int oldProcessID = targetProcessId;
4022 GetLastDirectory(targetFile, exeFile);
4024 while(!targetProcessId/*true*/)
4026 targetProcessId = Process_GetChildExeProcessId(gdbProcessId, exeFile);
4027 if(targetProcessId) break;
4028 // Can't break on Peek(), GDB is giving us =library and other info before the process is listed in /proc
4029 // if(gdbHandle.Peek()) break;
4031 if(gdbHandle.Peek() && GetTime() - startTime > 2.5) // Give the process 2.5 seconds to show up in /proc
4036 _ChangeState(running);
4037 else if(!oldProcessID)
4039 ide.outputView.debugBox.Logf($"Debugger Error: No target process ID\n");
4040 // TO VERIFY: The rest of this block has not been thoroughly tested in this particular location
4041 gdbHandle.Printf("-gdb-exit\n");
4043 _ChangeState(terminated); //loaded;
4048 for(bp : ide.workspace.breakpoints)
4049 bp.inserted = false;
4052 bp.inserted = false;
4054 bpRunToCursor.inserted = false;
4056 ide.outputView.debugBox.Logf($"Debugging stopped\n");
4057 ClearBreakDisplay();
4059 #if defined(__unix__)
4060 if(!usingValgrind && FileExists(progFifoPath)) //fileCreated)
4062 progThread.terminate = true;
4065 fifoFile.CloseInput();
4072 DeleteFile(progFifoPath);
4073 progFifoPath[0] = '\0';
4080 serialSemaphore.Release();
4083 _dpl2(_dpct, dplchan::gdbProtoUnknown, 0, $"Unknown prompt", output);
4087 if(!strncmp(output, "&\"warning:", 10))
4090 content = strstr(output, "\"");
4091 StripQuotes(content, content);
4092 content = strstr(content, ":");
4098 ide.outputView.debugBox.LogRaw((s = CopyUnescapedString(content)));
4105 _dpl2(_dpct, dplchan::gdbProtoUnknown, 0, $"Unknown output: ", output);
4107 if(!setWaitingForPID)
4108 waitingForPID = false;
4109 setWaitingForPID = false;
4117 // From GDB Output functions
4118 void FGODetectLoadedLibraryForAddedProjectIssues(Array<char *> outTokens, bool shlibs)
4120 char path[MAX_LOCATION] = "";
4121 char file[MAX_FILENAME] = "";
4122 bool symbolsLoaded = false;
4123 DebugListItem item { };
4124 //_dpl2(_dpct, dplchan::debuggerCall, 0, "Debugger::FGODetectLoadedLibraryForAddedProjectIssues()");
4125 for(token : outTokens)
4127 if(TokenizeListItem(token, item))
4129 if(!strcmp(item.name, "target-name"))
4131 StripQuotes(item.value, path);
4132 MakeSystemPath(path);
4133 GetLastDirectory(path, file);
4135 else if(!strcmp(item.name, "symbols-loaded"))
4137 symbolsLoaded = (atoi(item.value) == 1);
4139 else if(!strcmp(item.name, "shlib-info"))
4141 DebugListItem subItem { };
4142 Array<char *> tokens { minAllocSize = 50 };
4143 item.value = StripBrackets(item.value);
4144 TokenizeList(item.value, ',', tokens);
4147 if(TokenizeListItem(t, subItem))
4149 if(!strcmp(subItem.name, "path"))
4151 StripQuotes(subItem.value, path);
4152 MakeSystemPath(path);
4153 GetLastDirectory(path, file);
4154 symbolsLoaded = true;
4165 if(path[0] && file[0])
4167 for(prj : ide.workspace.projects; prj != ide.workspace.projects.firstIterator.data)
4171 char prjTargetPath[MAX_LOCATION];
4172 char prjTargetFile[MAX_FILENAME];
4173 DirExpression targetDirExp = prj.GetTargetDir(currentCompiler, prj.config, bitDepth);
4174 strcpy(prjTargetPath, prj.topNode.path);
4175 PathCat(prjTargetPath, targetDirExp.dir);
4176 delete targetDirExp;
4177 prjTargetFile[0] = '\0';
4178 prj.CatTargetFileName(prjTargetFile, currentCompiler, prj.config);
4179 PathCat(prjTargetPath, prjTargetFile);
4180 MakeSystemPath(prjTargetPath);
4182 match = !fstrcmp(prjTargetFile, file);
4183 if(!match && (dot = strstr(prjTargetFile, ".so.")))
4185 char * dot3 = strstr(dot+4, ".");
4189 match = !fstrcmp(prjTargetFile, file);
4194 match = !fstrcmp(prjTargetFile, file);
4199 // TODO: nice visual feedback to better warn user. use some ide notification system or other means.
4200 /* -- this is disabled because we can't trust gdb's symbols-loaded="0" field for =library-loaded (http://sourceware.org/bugzilla/show_bug.cgi?id=10693)
4202 ide.outputView.debugBox.Logf($"Attention! No symbols for loaded library %s matched to the %s added project.\n", path, prj.topNode.name);
4204 match = !fstrcmp(prjTargetPath, path);
4205 if(!match && (dot = strstr(prjTargetPath, ".so.")))
4207 char * dot3 = strstr(dot+4, ".");
4211 match = !fstrcmp(prjTargetPath, path);
4216 match = !fstrcmp(prjTargetPath, path);
4220 projectsLibraryLoaded[prj.name] = true;
4222 ide.outputView.debugBox.Logf($"Loaded library %s doesn't match the %s target of the %s added project.\n", path, prjTargetPath, prj.topNode.name);
4229 void FGOBreakpointModified(Array<char *> outTokens)
4231 _dpl2(_dpct, dplchan::debuggerCall, 0, "Debugger::FGOBreakpointModified() -- TODO only if needed: support breakpoint modified");
4233 DebugListItem item { };
4234 if(outTokens.count > 1 && TokenizeListItem(outTokens[1], item))
4236 if(!strcmp(item.name, "bkpt"))
4238 GdbDataBreakpoint modBp = ParseBreakpoint(item.value, outTokens);
4246 ExpressionType ::DebugEvalExpTypeError(char * result)
4248 _dpl2(_dpct, dplchan::debuggerWatches, 0, "Debugger::DebugEvalExpTypeError()");
4253 case symbolNotFound:
4254 return symbolErrorExp;
4255 case memoryCantBeRead:
4256 return memoryErrorExp;
4258 return unknownErrorExp;
4261 char * ::EvaluateExpression(char * expression, ExpressionType * error)
4264 _dpl2(_dpct, dplchan::debuggerWatches, 0, "Debugger::EvaluateExpression(", expression, ")");
4265 if(ide.projectView && ide.debugger.state == stopped)
4267 result = GdbEvaluateExpression(expression);
4268 *error = DebugEvalExpTypeError(result);
4273 *error = noDebuggerErrorExp;
4278 char * ::ReadMemory(uint64 address, int size, char format, ExpressionType * error)
4281 char * result = GdbReadMemoryString(address, size, format, 1, 1);
4282 _dpl2(_dpct, dplchan::debuggerCall, 0, "Debugger::ReadMemory(", address, ")");
4283 if(!result || !strcmp(result, "N/A"))
4284 *error = memoryErrorExp;
4286 *error = DebugEvalExpTypeError(result);
4291 class ValgrindLogThread : Thread
4297 static char output[4096];
4298 bool lastLineEmpty = true;
4299 Array<char> dynamicBuffer { minAllocSize = 4096 };
4300 File oldValgrindHandle = vgLogFile;
4301 incref oldValgrindHandle;
4304 while(debugger.state != terminated && vgLogFile && vgLogFile.input)
4309 result = vgLogFile.Read(output, 1, sizeof(output));
4311 if(debugger.state == terminated || !vgLogFile) // || vgLogFile.Eof()
4318 for(c = 0; c<result; c++)
4320 if(output[c] == '\n')
4322 int pos = dynamicBuffer.size;
4323 dynamicBuffer.size += c - start;
4324 memcpy(&dynamicBuffer[pos], output + start, c - start);
4325 if(dynamicBuffer.count && dynamicBuffer[dynamicBuffer.count - 1] != '\r')
4326 dynamicBuffer.size++;
4327 dynamicBuffer[dynamicBuffer.count - 1] = '\0';
4329 // printf("%s\n", dynamicBuffer.array);
4331 if(strstr(&dynamicBuffer[0], "vgdb me ..."))
4332 debugger.serialSemaphore.Release();
4334 char * s = strstr(&dynamicBuffer[0], "==");
4336 s = strstr(s+2, "== ");
4340 if(s[0] == '\0' && !lastLineEmpty)
4343 lastLineEmpty = true;
4344 dynamicBuffer[0] = '\0';
4353 if(strstr(s, "vgdb me ..."))
4355 if(strstr(s, "(action on error) vgdb me ..."))
4356 ide.outputView.debugBox.Logf($"...breaked on Valgrind error (F5 to resume)\n");
4363 if(strstr(s, "TO DEBUG THIS PROCESS USING GDB: start GDB like this"))
4369 if(strstr(s, "and then give GDB the following command"))
4375 if(strstr(s, "/path/to/gdb") || strstr(s, "target remote | /usr/lib/valgrind/../../bin/vgdb --pid="))
4381 if(strstr(s, "--pid is optional if only one valgrind process is running"))
4387 if((s = strstr(s, "; rerun with -h for copyright info")))
4399 if(lastLineEmpty && t[0] != '\0')
4400 lastLineEmpty = false;
4403 ide.outputView.debugBox.Logf("%s\n", &dynamicBuffer[0]);
4405 dynamicBuffer.size = 0;
4411 int pos = dynamicBuffer.size;
4412 dynamicBuffer.size += c - start;
4413 memcpy(&dynamicBuffer[pos], output + start, c - start);
4416 else if(debugger.state == stopped)
4419 printf("Got end of file from GDB!\n");
4426 delete dynamicBuffer;
4427 _dpl2(_dpct, dplchan::debuggerCall, 0, "ValgrindLogThreadExit");
4428 //if(oldValgrindHandle == vgLogFile)
4429 debugger.GdbThreadExit/*ValgrindLogThreadExit*/();
4430 delete oldValgrindHandle;
4436 class ValgrindTargetThread : Thread
4442 static char output[4096];
4443 Array<char> dynamicBuffer { minAllocSize = 4096 };
4444 DualPipe oldValgrindHandle = vgTargetHandle;
4445 incref oldValgrindHandle;
4448 while(debugger.state != terminated && vgTargetHandle && !vgTargetHandle.Eof())
4452 result = vgTargetHandle.Read(output, 1, sizeof(output));
4454 if(debugger.state == terminated || !vgTargetHandle || vgTargetHandle.Eof())
4461 for(c = 0; c<result; c++)
4463 if(output[c] == '\n')
4465 int pos = dynamicBuffer.size;
4466 dynamicBuffer.size += c - start;
4467 memcpy(&dynamicBuffer[pos], output + start, c - start);
4468 if(dynamicBuffer.count && dynamicBuffer[dynamicBuffer.count - 1] != '\r')
4469 // COMMENTED OUT DUE TO ISSUE #135, FIXED
4470 //if(dynamicBuffer.array[dynamicBuffer.count - 1] != '\r')
4471 dynamicBuffer.size++;
4472 dynamicBuffer[dynamicBuffer.count - 1] = '\0';
4474 // printf("%s\n", dynamicBuffer.array);
4476 ide.outputView.debugBox.Logf("%s\n", &dynamicBuffer[0]);
4478 dynamicBuffer.size = 0;
4484 int pos = dynamicBuffer.size;
4485 dynamicBuffer.size += c - start;
4486 memcpy(&dynamicBuffer[pos], output + start, c - start);
4492 printf("Got end of file from GDB!\n");
4496 delete dynamicBuffer;
4497 //if(oldValgrindHandle == vgTargetHandle)
4498 debugger.ValgrindTargetThreadExit();
4499 delete oldValgrindHandle;
4505 class GdbThread : Thread
4511 static char output[4096];
4512 Array<char> dynamicBuffer { minAllocSize = 4096 };
4513 DualPipe oldGdbHandle = gdbHandle;
4514 incref oldGdbHandle;
4517 while(debugger.state != terminated && gdbHandle && !gdbHandle.Eof())
4521 result = gdbHandle.Read(output, 1, sizeof(output));
4523 if(debugger.state == terminated || !gdbHandle || gdbHandle.Eof())
4530 for(c = 0; c<result; c++)
4532 if(output[c] == '\n')
4534 int pos = dynamicBuffer.size;
4535 dynamicBuffer.size += c - start;
4536 memcpy(&dynamicBuffer[pos], output + start, c - start);
4537 if(dynamicBuffer.count && dynamicBuffer[dynamicBuffer.count - 1] != '\r')
4538 // COMMENTED OUT DUE TO ISSUE #135, FIXED
4539 //if(dynamicBuffer.array[dynamicBuffer.count - 1] != '\r')
4540 dynamicBuffer.size++;
4541 dynamicBuffer[dynamicBuffer.count - 1] = '\0';
4543 // _dpl(0, dynamicBuffer.array);
4545 debugger.GdbThreadMain(&dynamicBuffer[0]);
4546 dynamicBuffer.size = 0;
4552 int pos = dynamicBuffer.size;
4553 dynamicBuffer.size += c - start;
4554 memcpy(&dynamicBuffer[pos], output + start, c - start);
4560 _dpl(0, "Got end of file from GDB!");
4564 delete dynamicBuffer;
4565 //if(oldGdbHandle == gdbHandle)
4566 debugger.GdbThreadExit();
4567 delete oldGdbHandle;
4573 static define createFIFOMsg = $"err: Unable to create FIFO %s\n";
4574 static define openFIFOMsg = $"err: Unable to open FIFO %s for read\n";
4576 #if defined(__unix__)
4581 #include <sys/types.h>
4586 class ProgramThread : Thread
4592 bool fileCreated = false;
4594 static char output[1000];
4597 /*if(!mkfifo(progFifoPath, mask))
4604 ide.outputView.debugBox.Logf($"err: Unable to create FIFO %s\n", progFifoPath);
4608 if(FileExists(progFifoPath)) //fileCreated)
4610 fifoFile = FileOpen(progFifoPath, read);
4614 ide.outputView.debugBox.Logf(openFIFOMsg, progFifoPath);
4619 fd = fileno((FILE *)fifoFile.input);
4620 //fcntl(fd, F_SETFL, fcntl(fd, F_GETFL) | O_NONBLOCK);
4624 while(!terminate && fifoFile && !fifoFile.Eof())
4627 struct timeval time;
4635 selectResult = select(fd + 1, &rs, null, null, &time);
4636 if(FD_ISSET(fd, &rs))
4638 int result = (int)read(fd, output, sizeof(output)-1);
4639 if(!result || (result < 0 && errno != EAGAIN))
4643 output[result] = '\0';
4644 if(strcmp(output,"&\"warning: GDB: Failed to set controlling terminal: Invalid argument\\n\"\n"))
4647 ide.outputView.debugBox.Log(output);
4656 //fifoFile.CloseInput();
4659 ide.outputView.debugBox.Log("\n");
4663 if(FileExists(progFifoPath)) //fileCreated)
4665 DeleteFile(progFifoPath);
4666 progFifoPath[0] = '\0';
4674 class Argument : struct
4676 Argument prev, next;
4678 property char * name { set { delete name; if(value) name = CopyString(value); } }
4680 property char * val { set { delete val; if(value) val = CopyString(value); } }
4694 class Frame : struct
4699 property char * addr { set { delete addr; if(value) addr = CopyString(value); } }
4701 property char * func { set { delete func; if(value) func = CopyString(value); } }
4705 property char * from { set { delete from; if(value) from = CopyUnescapedUnixPath(value); } }
4707 property char * file { set { delete file; if(value) file = CopyUnescapedUnixPath(value); } }
4708 char * absoluteFile;
4709 property char * absoluteFile { set { delete absoluteFile; if(value) absoluteFile = CopyUnescapedUnixPath(value); } }
4718 delete absoluteFile;
4719 args.Free(Argument::Free);
4728 class GdbDataStop : struct
4730 DebuggerReason reason;
4745 char * gdbResultVar;
4755 if(reason == signalReceived)
4760 else if(reason == functionFinished)
4762 delete gdbResultVar;
4766 if(frame) frame.Free();
4775 class GdbDataBreakpoint : struct
4779 property char * number { set { delete number; if(value) number = CopyString(value); } }
4781 property char * type { set { delete type; if(value) type = CopyString(value); } }
4783 property char * disp { set { delete disp; if(value) disp = CopyString(value); } }
4786 property char * addr { set { delete addr; if(value) addr = CopyString(value); } }
4788 property char * func { set { delete func; if(value) func = CopyString(value); } }
4790 property char * file { set { delete file; if(value) file = CopyUnescapedUnixPath(value); } }
4792 property char * fullname { set { delete fullname; if(value) fullname = CopyUnescapedUnixPath(value); } }
4795 property char * at { set { delete at; if(value) at = CopyString(value); } }
4798 Array<GdbDataBreakpoint> multipleBPs;
4803 PrintLn("{", "#", number, " T", type, " D", disp, " E", enabled, " H", times, " (", func, ") (", file, ":", line, ") (", fullname, ") (", addr, ") (", at, ")", "}");
4814 if(multipleBPs) multipleBPs.Free();
4820 ~GdbDataBreakpoint()
4826 class Breakpoint : struct
4831 property char * function { set { delete function; if(value) function = CopyString(value); } }
4832 char * relativeFilePath;
4833 property char * relativeFilePath { set { delete relativeFilePath; if(value) relativeFilePath = CopyString(value); } }
4834 char * absoluteFilePath;
4835 property char * absoluteFilePath { set { delete absoluteFilePath; if(value) absoluteFilePath = CopyString(value); } }
4837 property char * location { set { delete location; if(value) location = CopyString(value); } }
4846 BreakpointType type;
4848 GdbDataBreakpoint bp;
4851 property char * address { set { delete address; if(value) address = CopyString(value); } }
4853 void ParseLocation()
4855 char * prjName = null;
4856 char * filePath = null;
4859 char fullPath[MAX_LOCATION];
4860 if(location[0] == '(' && location[1] && (file = strchr(location+2, ')')) && file[1])
4862 prjName = new char[file-location];
4863 strncpy(prjName, location+1, file-location-1);
4864 prjName[file-location-1] = '\0';
4869 if((line = strchr(file+1, ':')))
4871 filePath = new char[strlen(file)+1];
4872 strncpy(filePath, file, line-file);
4873 filePath[line-file] = '\0';
4877 filePath = CopyString(file);
4878 property::relativeFilePath = filePath;
4881 for(prj : ide.workspace.projects)
4883 if(!strcmp(prjName, prj.name))
4885 if(ProjectGetAbsoluteFromRelativePath(prj, filePath, fullPath))
4887 property::absoluteFilePath = fullPath;
4894 this.line = atoi(line);
4898 Project prj = ide.project;
4899 if(ProjectGetAbsoluteFromRelativePath(prj, filePath, fullPath))
4901 property::absoluteFilePath = fullPath;
4905 if(!absoluteFilePath)
4906 property::absoluteFilePath = "";
4911 char * CopyLocationString(bool removePath)
4914 char * file = relativeFilePath ? relativeFilePath : absoluteFilePath;
4915 bool removingPath = removePath && file;
4918 char * fileName = new char[MAX_FILENAME];
4919 GetLastDirectory(file, fileName);
4925 location = PrintString(file, ":", function);
4927 location = CopyString(function);
4930 location = PrintString(file, ":", line);
4936 char * CopyUserLocationString()
4939 char * loc = CopyLocationString(false);
4941 if(absoluteFilePath)
4943 for(p : ide.workspace.projects; p != ide.workspace.projects.firstIterator.data)
4945 if(p.topNode.FindByFullPath(absoluteFilePath, false))
4954 location = PrintString("(", prj.name, ")", loc);
4964 if(relativeFilePath && relativeFilePath[0])
4966 char * location = CopyUserLocationString();
4967 f.Printf(" * %d,%d,%d,%d,%s\n", enabled ? 1 : 0, ignore, level, line, location);
4970 f.Printf(" ~ %s\n", condition.expression);
4978 delete relativeFilePath;
4979 delete absoluteFilePath;
4999 class Watch : struct
5010 f.Printf(" ~ %s\n", expression);
5034 class DebugListItem : struct
5040 struct DebugEvaluationData
5045 uint64 nextBlockAddress;
5047 DebuggerEvaluationError error;
5050 class CodeLocation : struct
5053 char * absoluteFile;
5056 CodeLocation ::ParseCodeLocation(char * location)
5060 char * colon = null;
5062 char loc[MAX_LOCATION];
5063 strcpy(loc, location);
5064 for(temp = loc; temp = strstr(temp, ":"); temp++)
5072 int line = atoi(colon);
5075 CodeLocation codloc { line = line };
5076 codloc.file = CopyString(loc);
5077 codloc.absoluteFile = ide.workspace.GetAbsolutePathFromRelative(loc);
5089 delete absoluteFile;
5098 void GDBFallBack(Expression exp, String expString)
5101 ExpressionType evalError = dummyExp;
5102 result = Debugger::EvaluateExpression(expString, &evalError);
5105 FreeExpContents(exp);
5106 exp.constant = result;
5107 exp.type = constantExp;
5111 static Project WorkspaceGetFileOwner(char * absolutePath)
5113 Project owner = null;
5114 for(prj : ide.workspace.projects)
5116 if(prj.topNode.FindByFullPath(absolutePath, false))
5123 WorkspaceGetObjectFileNode(absolutePath, &owner);
5127 static ProjectNode WorkspaceGetObjectFileNode(char * filePath, Project * project)
5129 ProjectNode node = null;
5130 char ext[MAX_EXTENSION];
5131 GetExtension(filePath, ext);
5134 IntermediateFileType type = IntermediateFileType::FromExtension(ext);
5137 char fileName[MAX_FILENAME];
5138 GetLastDirectory(filePath, fileName);
5141 DotMain dotMain = DotMain::FromFileName(fileName);
5142 for(prj : ide.workspace.projects)
5144 if((node = prj.FindNodeByObjectFileName(fileName, type, dotMain, null)))
5157 static ProjectNode ProjectGetObjectFileNode(Project project, char * filePath)
5159 ProjectNode node = null;
5160 char ext[MAX_EXTENSION];
5161 GetExtension(filePath, ext);
5164 IntermediateFileType type = IntermediateFileType::FromExtension(ext);
5167 char fileName[MAX_FILENAME];
5168 GetLastDirectory(filePath, fileName);
5171 DotMain dotMain = DotMain::FromFileName(fileName);
5172 node = project.FindNodeByObjectFileName(fileName, type, dotMain, null);
5179 static void WorkspaceGetRelativePath(char * absolutePath, char * relativePath, Project * owner)
5181 Project prj = WorkspaceGetFileOwner(absolutePath);
5185 prj = ide.workspace.projects.firstIterator.data;
5188 MakePathRelative(absolutePath, prj.topNode.path, relativePath);
5189 MakeSlashPath(relativePath);
5192 relativePath[0] = '\0';
5195 static bool ProjectGetAbsoluteFromRelativePath(Project project, char * relativePath, char * absolutePath)
5197 ProjectNode node = project.topNode.FindWithPath(relativePath, false);
5199 node = ProjectGetObjectFileNode(project, relativePath);
5202 strcpy(absolutePath, node.project.topNode.path);
5203 PathCat(absolutePath, relativePath);
5204 MakeSlashPath(absolutePath);
5206 return node != null;