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
25 #include <string.h> // For memchr
33 #include <sys/time.h> // Required on Apple...
40 // use =0 to disable printing of specific channels
41 static enum dplchan { none, gdbProtoIgnored=0/*1*/, gdbProtoUnknown=2, gdbOutput=3/*3*/, gdbCommand=4/*4*/, debuggerCall=0/*5*/, debuggerProblem=6,
42 debuggerUserAction=7,debuggerState=8, debuggerBreakpoints=9, debuggerWatches=0/*10*/, debuggerTemp=0 };
43 static const char * _dpct[] = {
45 "GDB Protocol Ignored",
46 "GDB Protocol ***Unknown***",
50 "Debugger ***Problem***",
51 "Debugger::ChangeUserAction",
52 "Debugger::ChangeState",
55 "-----> Temporary Message",
60 public char * StripQuotes2(char * string, char * output)
64 bool quoted = false, escaped = false;
66 for(c = 0; (ch = string[c]); c++)
70 if(escaped || ch != '\"')
73 escaped = !escaped && ch == '\\';
88 static void strescpy(char * d, char * s)
96 case '\n': d[k] = '\\'; d[++k] = 'n'; break;
97 case '\t': d[k] = '\\'; d[++k] = 't'; break;
98 case '\a': d[k] = '\\'; d[++k] = 'a'; break;
99 case '\b': d[k] = '\\'; d[++k] = 'b'; break;
100 case '\f': d[k] = '\\'; d[++k] = 'f'; break;
101 case '\r': d[k] = '\\'; d[++k] = 'r'; break;
102 case '\v': d[k] = '\\'; d[++k] = 'v'; break;
103 case '\\': d[k] = '\\'; d[++k] = '\\'; break;
104 case '\"': d[k] = '\\'; d[++k] = '\"'; break;
105 default: d[k] = s[j];
112 /*static char * CopyUnescapedSystemPath(char * p)
115 char * d = new char[len + 1];
116 UnescapeString(d, p, len);
117 #if defined(__WIN32__)
118 ChangeCh(d, '/', '\\');
123 static char * CopyUnescapedUnixPath(char * p)
126 char * d = new char[len + 1];
127 UnescapeString(d, p, len);
128 #if defined(__WIN32__)
129 ChangeCh(d, '\\', '/');
134 static char * CopyUnescapedString(char * s)
137 char * d = new char[len + 1];
138 UnescapeString(d, s, len);
142 static char * StripBrackets(char * string)
144 int length = strlen(string);
145 if(length > 1 && *string == '[' && string[length - 1] == ']')
148 string[length - 1] = '\0';
155 static char * StripCurlies(char * string)
157 int length = strlen(string);
158 if(length > 1 && *string == '{' && string[length - 1] == '}')
161 string[length - 1] = '\0';
168 /*static int StringGetInt(char * string, int start)
171 int i, len = strlen(string);
173 for(i = start; i < len && i < start + 8; i++)
175 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')
176 strncat(number, &string[i], 1);
183 static int TokenizeList(char * string, const char seperator, Array<char *> tokens)
187 bool quoted = false, escaped = false;
188 char * start = string, ch;
190 for(; (ch = *string); string++)
197 if(escaped || ch != '\"')
198 escaped = !escaped && ch == '\\';
204 else if(ch == '{' || ch == '[' || ch == '(' || ch == '<')
206 else if(ch == '}' || ch == ']' || ch == ')' || ch == '>')
208 else if(ch == seperator && !level)
217 //tokens[count] = start;
218 //tokens[count++] = start;
225 static bool TokenizeListItem(char * string, DebugListItem item)
227 char * equal = strstr(string, "=");
239 static bool CheckCommandAvailable(const char * command)
241 bool available = false;
243 char * name = new char[MAX_FILENAME];
244 char * pathVar = new char[maxPathLen];
246 GetEnvironment("PATH", pathVar, maxPathLen);
247 count = TokenizeWith(pathVar, sizeof(paths) / sizeof(char *), paths, pathListSep, false);
248 strcpy(name, command);
252 const char * extensions[] = { "exe", "com", "bat", null };
253 for(e=0; extensions[e]; e++)
255 ChangeExtension(name, extensions[e], name);
257 for(c=0; c<count; c++)
259 FileListing fl { paths[c] };
262 if(fl.stats.attribs.isFile && !fstrcmp(fl.name, name))
281 // define GdbGetLineSize = 1638400;
282 define GdbGetLineSize = 5638400;
283 #if defined(__unix__)
284 char progFifoPath[MAX_LOCATION];
285 char progFifoDir[MAX_LOCATION];
288 enum DebuggerState { none, prompt, loaded, running, stopped, terminated, error };
291 none, hit, breakEvent, signal, stepEnd, functionEnd, exit, valgrindStartPause, locationReached;
293 property bool canBeMonitored { get { return (this == hit || this == breakEvent || this == signal || this == stepEnd || this == functionEnd || this == locationReached); } };
295 enum DebuggerAction { none, internal, restart, stop, selectFrame, advance }; //, bpValidation
298 unknown, endSteppingRange, functionFinished, signalReceived, breakpointHit, locationReached
299 //watchpointTrigger, readWatchpointTrigger, accessWatchpointTrigger, watchpointScope,
300 //exited, exitedNormally, exitedSignalled;
304 none, internalMain, internalWinMain, internalModulesLoaded, user, runToCursor, internalModuleLoad, internalEntry;
306 property bool isInternal { get { return (this == internalMain || this == internalWinMain || this == internalModulesLoaded || this == internalModuleLoad || this == internalEntry); } };
307 property bool isUser { get { return (this == user || this == runToCursor); } };
309 enum DebuggerEvaluationError { none, symbolNotFound, memoryCantBeRead, unknown };
310 enum DebuggerUserAction
312 none, start, resume, _break, stop, restart, selectThread, selectFrame, stepInto, stepOver, stepUntil, stepOut, runToCursor;
313 property bool breaksOnInternalBreakpoint { get { return (this == stepInto || this == stepOver || this == stepUntil); } };
317 none, run, _continue, next, until, advance, step, finish;
318 property bool suspendInternalBreakpoints { get { return (this == until || this == advance || this == step || this == finish); } };
321 FileDialog debuggerFileDialog { type = selectDir };
323 static DualPipe vgTargetHandle;
324 static File vgLogFile;
325 static char vgLogPath[MAX_LOCATION];
326 static DualPipe gdbHandle;
327 static DebugEvaluationData eval { };
329 static int targetProcessId;
331 static bool gdbReady;
332 static bool breakpointError;
336 Semaphore serialSemaphore { };
342 bool sentBreakInsert;
343 bool ignoreBreakpoints;
351 int activeFrameLevel;
360 GdbExecution gdbExecution;
361 DebuggerUserAction userAction;
364 DebuggerAction breakType;
366 //DebuggerCommand lastCommand; // THE COMPILER COMPILES STUFF THAT DOES NOT EXIST???
368 GdbDataStop stopItem;
369 GdbDataBreakpoint bpItem;
372 List<Breakpoint> sysBPs { };
373 Breakpoint bpRunToCursor;
374 Breakpoint intBpEntry;
375 Breakpoint intBpMain;
376 Breakpoint intBpWinMain;
380 CompilerConfig currentCompiler;
381 ProjectConfig prjConfig;
384 CodeEditor codeEditor;
386 ValgrindLogThread vgLogThread { debugger = this };
387 ValgrindTargetThread vgTargetThread { debugger = this };
388 GdbThread gdbThread { debugger = this };
391 Map<String, bool> projectsLibraryLoaded { };
395 delay = 0.0, userData = this;
399 bool monitor = false;
400 DebuggerEvent curEvent = event;
401 GdbDataStop stopItem = this.stopItem;
402 Breakpoint bpUser = null;
403 Breakpoint bpInternal = null;
411 this.stopItem = null;
415 DynamicString bpReport { };
417 for(bp : sysBPs; bp.inserted)
419 bpReport.concatx(",", bp.type, "(", s=bp.CopyLocationString(false), ")");
422 if(bpRunToCursor && bpRunToCursor.inserted)
424 Breakpoint bp = bpRunToCursor;
425 bpReport.concatx(",", bp.type, "(", s=bp.CopyLocationString(false), ")");
428 for(bp : ide.workspace.breakpoints; bp.inserted)
430 bpReport.concatx(",", bp.type, "(", s=bp.CopyLocationString(false), ")");
434 _dpcl(_dpct, dplchan::debuggerBreakpoints, 0, "gdbTimer::DelayExpired: ", s+1);
439 Breakpoint bp = GetBreakpointById(stopItem.bkptno, &isInternal);
442 _dpcl(_dpct, dplchan::debuggerBreakpoints, 0, "gdb stopped by a breakpoint: ", bp.type, "(", s=bp.CopyLocationString(false), ")");
453 if(curEvent && curEvent != exit)
455 _dplf(0, "No stop item");
463 Restart(currentCompiler, prjConfig, bitDepth, usingValgrind);
472 GdbCommand(0, false, "-stack-select-frame %d", activeFrameLevel);
473 for(activeFrame = stackFrames.first; activeFrame; activeFrame = activeFrame.next)
474 if(activeFrame.level == activeFrameLevel)
480 // GdbCommand(0, false, "-break-info %s", bpItem.number);
492 Breakpoint bp = stopItem ? GetBreakpointById(stopItem.bkptno, &isInternal) : null;
493 if(bp && bp.inserted && bp.bp.addr)
495 if(bp.type.isInternal)
499 if(stopItem && stopItem.frame)
501 if(bpInternal && bpRunToCursor && bpRunToCursor.inserted && !strcmp(bpRunToCursor.bp.addr, bp.bp.addr))
502 bpUser = bpRunToCursor;
505 for(item : (bpInternal ? ide.workspace.breakpoints : sysBPs); item.inserted)
507 if(item.bp && item.bp.addr && !strcmp(item.bp.addr, bp.bp.addr))
519 _dpcl(_dpct, dplchan::debuggerProblem, 0, "Invalid stopItem!");
520 if(bpUser && stopItem.frame.addr && strcmp(stopItem.frame.addr, bpUser.bp.addr))
521 _dpcl(_dpct, dplchan::debuggerProblem, 0, "Breakpoint bkptno(", stopItem.bkptno, ") address missmatch!");
524 _dpcl(_dpct, dplchan::debuggerProblem, 0, "Breakpoint bkptno(", stopItem.bkptno, ") invalid or not found!");
525 if((bpUser && !ignoreBreakpoints) || (bpInternal && userAction.breaksOnInternalBreakpoint))
527 hitThread = stopItem.threadid;
531 signalThread = stopItem.threadid;
535 case locationReached:
537 ignoreBreakpoints = false;
539 case valgrindStartPause:
540 GdbExecContinue(true);
548 if(curEvent == signal)
552 ide.outputView.debugBox.Logf($"Signal received: %s - %s\n", stopItem.name, stopItem.meaning);
553 ide.outputView.debugBox.Logf(" %s:%d\n", (s = CopySystemPath(stopItem.frame.file)), stopItem.frame.line);
554 ide.outputView.Show();
555 ide.callStackView.Show();
558 else if(curEvent == breakEvent)
560 ide.threadsView.Show();
561 ide.callStackView.Show();
562 ide.callStackView.Activate();
564 else if(curEvent == hit)
566 if(BreakpointHit(stopItem, bpInternal, bpUser))
568 ide.AdjustDebugMenus();
569 if(bpUser && bpUser.type == runToCursor)
571 ignoreBreakpoints = false;
572 UnsetBreakpoint(bpUser);
573 delete bpRunToCursor;
578 if(breakType == advance && bpInternal && (bpInternal.type == internalMain || bpInternal.type == internalEntry))
581 GdbExecAdvance(breakString, 0);
586 GdbExecContinue(false);
592 if(monitor && curEvent.canBeMonitored)
595 activeThread = stopItem.threadid;
596 GdbCommand(0, false, "-thread-list-ids");
597 InternalSelectFrame(activeFrameLevel);
598 GoToStackFrameLine(activeFrameLevel, true, false);
600 ide.ShowCodeEditor();
601 ide.AdjustDebugMenus();
602 ideMainFrame.Activate(); // TOFIX: ide.Activate() is not reliable (app inactive)
615 #ifdef GDB_DEBUG_CONSOLE
616 char lastGdbOutput[GdbGetLineSize];
618 #if defined(__unix__)
619 ProgramThread progThread { };
623 #define _ChangeUserAction(value) ChangeUserAction(__FILE__, __LINE__, value)
624 void ChangeUserAction(const char * file, int line, DebuggerUserAction value)
627 bool same = value == userAction;
628 __dpl(file, line, _dpct, dplchan::debuggerUserAction, 0, userAction, /*same ? " *** == *** " : */" -> ", value);
633 #define _ChangeUserAction(value) userAction = value
637 #define _ChangeState(value) ChangeState(__FILE__, __LINE__, value)
638 void ChangeState(const char * file, int line, DebuggerState value)
640 #define _ChangeState(value) ChangeState(value)
641 void ChangeState(DebuggerState value)
644 bool same = value == state;
646 __dpl(file, line, _dpct, dplchan::debuggerState, 0, state, same ? " *** == *** " : " -> ", value);
649 if(!same) ide.AdjustDebugMenus();
654 _dpcl(_dpct, dplchan::debuggerCall, 0, "Debugger::CleanUp");
656 stackFrames.Free(Frame::Free);
666 waitingForPID = false;
671 sentBreakInsert = false;
672 ignoreBreakpoints = false;
675 activeFrameLevel = 0;
692 bpRunToCursor = null;
694 delete currentCompiler;
697 WatchesReleaseCodeEditor();
700 projectsLibraryLoaded.Free();
702 /*GdbThread gdbThread
708 _dpcl(_dpct, dplchan::debuggerCall, 0, "Debugger::constructor");
709 ideProcessId = Process_GetCurrentProcessId();
711 sysBPs.Add((intBpEntry = Breakpoint { type = internalEntry, enabled = false, level = -1 }));
712 sysBPs.Add((intBpMain = Breakpoint { type = internalMain, function = "main", enabled = true, level = -1 }));
713 #if defined(__WIN32__)
714 sysBPs.Add((intBpWinMain = Breakpoint { type = internalWinMain, function = "WinMain", enabled = true, level = -1 }));
716 sysBPs.Add(Breakpoint { type = internalModulesLoaded, enabled = true, level = -1 });
717 sysBPs.Add(Breakpoint { type = internalModuleLoad, function = "InternalModuleLoadBreakpoint", enabled = true, level = -1 });
722 _dpcl(_dpct, dplchan::debuggerCall, 0, "Debugger::destructor");
730 property bool isActive { get { return state == running || state == stopped; } }
731 property bool isPrepared { get { return state == loaded || state == running || state == stopped; } }
735 _dpcl(_dpct, dplchan::debuggerCall, 0, "Debugger::Resume");
736 _ChangeUserAction(resume);
737 GdbExecContinue(true);
742 _dpcl(_dpct, dplchan::debuggerCall, 0, "Debugger::Break");
743 _ChangeUserAction(_break);
747 GdbDebugBreak(false);
753 _dpcl(_dpct, dplchan::debuggerCall, 0, "Debugger::Stop");
754 _ChangeUserAction(stop);
761 GdbDebugBreak(false);
775 void Restart(CompilerConfig compiler, ProjectConfig config, int bitDepth, bool useValgrind)
777 _dpcl(_dpct, dplchan::debuggerCall, 0, "Debugger::Restart");
778 _ChangeUserAction(restart);
779 if(StartSession(compiler, config, bitDepth, useValgrind, true, false) == loaded)
783 bool GoToCodeLine(char * location)
786 _dpcl(_dpct, dplchan::debuggerCall, 0, "Debugger::GoToCodeLine(", location, ")");
787 codloc = CodeLocation::ParseCodeLocation(location);
790 CodeEditor editor = (CodeEditor)ide.OpenFile(codloc.absoluteFile, false, true, null, no, normal, false);
793 EditBox editBox = editor.editBox;
794 if(editBox.GoToLineNum(codloc.line - 1))
795 editBox.GoToPosition(editBox.line, codloc.line - 1, 0);
802 bool GoToStackFrameLine(int stackLevel, bool askForLocation, bool fromCallStack)
804 _dpcl(_dpct, dplchan::debuggerCall, 0, "Debugger::GoToStackFrameLine(", stackLevel, ", ", askForLocation, ")");
807 char sourceDir[MAX_LOCATION];
809 CodeEditor editor = null;
810 if(stackLevel == -1) // this (the two lines) is part of that fix that I would not put in for some time
812 for(frame = stackFrames.first; frame; frame = frame.next)
813 if(frame.level == stackLevel)
818 ide.callStackView.Show();
820 if(frame.absoluteFile)
821 editor = (CodeEditor)ide.OpenFile(frame.absoluteFile, false, true, null, no, normal, false);
822 if(!editor && frame.file)
823 frame.absoluteFile = ide.workspace.CopyAbsolutePathFromRelative(frame.file);
824 if(!frame.absoluteFile && askForLocation && frame.file)
827 char title[MAX_LOCATION];
828 snprintf(title, sizeof(title), $"Provide source file location for %s", (s = CopySystemPath(frame.file)));
829 title[sizeof(title)-1] = 0;
831 if(SourceDirDialog(title, ide.workspace.projectDir, frame.file, sourceDir))
833 AddSourceDir(sourceDir);
834 frame.absoluteFile = ide.workspace.CopyAbsolutePathFromRelative(frame.file);
837 if(!editor && frame.absoluteFile)
838 editor = (CodeEditor)ide.OpenFile(frame.absoluteFile, false, true, null, no, normal, false);
840 ide.RepositionWindows(false);
842 if(editor && frame.line)
844 EditBox editBox = editor.editBox;
845 editBox.GoToLineNum(frame.line - 1);
846 editBox.GoToPosition(editBox.line, frame.line - 1, 0);
854 void SelectThread(int thread)
856 _dpcl(_dpct, dplchan::debuggerCall, 0, "Debugger::SelectThread(", thread, ")");
857 _ChangeUserAction(selectThread);
860 if(thread != activeThread)
862 activeFrameLevel = -1;
863 ide.callStackView.Clear();
864 GdbCommand(0, false, "-thread-select %d", thread);
866 InternalSelectFrame(activeFrameLevel);
867 GoToStackFrameLine(activeFrameLevel, true, false);
871 ide.callStackView.Show();
875 void SelectFrame(int frame)
877 //_dpcl(_dpct, dplchan::debuggerCall, 0, "Debugger::SelectFrame(", frame, ")");
878 _ChangeUserAction(selectFrame);
881 if(frame != activeFrameLevel)
883 InternalSelectFrame(frame);
890 void InternalSelectFrame(int frame)
892 //_dpcl(_dpct, dplchan::debuggerCall, 0, "Debugger::InternalSelectFrame(", frame, ")");
893 activeFrameLevel = frame; // there is no active frame number in the gdb reply
894 GdbCommand(0, false, "-stack-select-frame %d", activeFrameLevel);
895 for(activeFrame = stackFrames.first; activeFrame; activeFrame = activeFrame.next)
896 if(activeFrame.level == activeFrameLevel)
900 void HandleExit(char * reason, char * code)
902 char verboseExitCode[128];
904 _dpcl(_dpct, dplchan::debuggerCall, 0, "Debugger::HandleExit(", reason, ", ", code, ")");
905 _ChangeState(loaded); // this state change seems to be superfluous, might be in case of gdb crash
910 snprintf(verboseExitCode, sizeof(verboseExitCode), $" with exit code %s", code);
911 verboseExitCode[sizeof(verboseExitCode)-1] = 0;
914 verboseExitCode[0] = '\0';
918 // ClearBreakDisplay();
922 for(wh : ide.workspace.watches)
924 if(wh.type) FreeType(wh.type);
927 ide.watchesView.UpdateWatch(wh);
931 #if defined(__unix__)
934 progThread.terminate = true;
937 fifoFile.CloseInput();
947 char program[MAX_LOCATION];
948 GetSystemPathBuffer(program, targetFile);
950 ide.outputView.debugBox.Logf($"The program %s has exited%s.\n", program, verboseExitCode);
951 else if(!strcmp(reason, "exited-normally"))
952 ide.outputView.debugBox.Logf($"The program %s has exited normally%s.\n", program, verboseExitCode);
953 else if(!strcmp(reason, "exited"))
954 ide.outputView.debugBox.Logf($"The program %s has exited%s.\n", program, verboseExitCode);
955 else if(!strcmp(reason, "exited-signalled"))
956 ide.outputView.debugBox.Logf($"The program %s has exited with a signal%s.\n", program, verboseExitCode);
958 ide.outputView.debugBox.Logf($"The program %s has exited (gdb provided an unknown reason)%s.\n", program, verboseExitCode);
963 DebuggerState StartSession(CompilerConfig compiler, ProjectConfig config, int bitDepth, bool useValgrind, bool restart, bool ignoreBreakpoints)
965 DebuggerState result = none;
966 _dpcl(_dpct, dplchan::debuggerCall, 0, "Debugger::StartSession(restart(", restart, "), ignoreBreakpoints(", ignoreBreakpoints, ")");
967 if(restart && state == running && targetProcessId)
969 breakType = DebuggerAction::restart;
970 GdbDebugBreak(false);
974 if(restart && state == stopped)
976 if(needReset && state == loaded)
977 GdbExit(); // this reset is to get a clean state with all the breakpoints until a better state can be maintained on program exit
979 if(result == none || result == terminated)
981 ide.outputView.ShowClearSelectTab(debug);
982 ide.outputView.debugBox.Logf($"Starting debug mode\n");
989 for(bp : ide.workspace.breakpoints)
995 if(GdbInit(compiler, config, bitDepth, useValgrind))
1000 this.ignoreBreakpoints = ignoreBreakpoints;
1005 void Start(CompilerConfig compiler, ProjectConfig config, int bitDepth, bool useValgrind)
1007 _dpcl(_dpct, dplchan::debuggerCall, 0, "Debugger::Start()");
1008 _ChangeUserAction(start);
1009 if(StartSession(compiler, config, bitDepth, useValgrind, true, false) == loaded)
1013 void StepInto(CompilerConfig compiler, ProjectConfig config, int bitDepth, bool useValgrind)
1015 _dpcl(_dpct, dplchan::debuggerCall, 0, "Debugger::StepInto()");
1016 _ChangeUserAction(stepInto);
1017 switch(StartSession(compiler, config, bitDepth, useValgrind, false, false))
1019 case loaded: GdbExecRun(); break;
1020 case stopped: GdbExecStep(); break;
1024 void StepOver(CompilerConfig compiler, ProjectConfig config, int bitDepth, bool useValgrind, bool ignoreBreakpoints)
1026 _dpcl(_dpct, dplchan::debuggerCall, 0, "Debugger::StepOver()");
1027 _ChangeUserAction(stepOver);
1028 switch(StartSession(compiler, config, bitDepth, useValgrind, false, ignoreBreakpoints))
1030 case loaded: GdbExecRun(); break;
1031 case stopped: GdbExecNext(); break;
1035 void StepUntil(CompilerConfig compiler, ProjectConfig config, int bitDepth, bool useValgrind, bool ignoreBreakpoints)
1037 _dpcl(_dpct, dplchan::debuggerCall, 0, "Debugger::StepUntil()");
1038 _ChangeUserAction(stepUntil);
1039 switch(StartSession(compiler, config, bitDepth, useValgrind, false, ignoreBreakpoints))
1041 case loaded: GdbExecRun(); break;
1042 case stopped: GdbExecUntil(null, 0); break;
1046 void StepOut(bool ignoreBreakpoints)
1048 _dpcl(_dpct, dplchan::debuggerCall, 0, "Debugger::StepOut()");
1049 _ChangeUserAction(stepOut);
1050 if(state == stopped)
1052 this.ignoreBreakpoints = ignoreBreakpoints;
1056 GdbExecContinue(true);
1060 void RunToCursor(CompilerConfig compiler, ProjectConfig config, int bitDepth, bool useValgrind, const char * absoluteFilePath, int lineNumber, bool ignoreBreakpoints, bool atSameLevel, bool oldImplementation)
1062 char relativeFilePath[MAX_LOCATION];
1063 const char * objectFileExt = compiler ? compiler.objectFileExt : objectDefaultFileExt;
1064 _dpcl(_dpct, dplchan::debuggerCall, 0, "Debugger::RunToCursor()");
1065 _ChangeUserAction(runToCursor);
1066 ide.workspace.GetRelativePath(absoluteFilePath, relativeFilePath, null, objectFileExt);
1068 if(bpRunToCursor && bpRunToCursor.inserted && symbols)
1070 UnsetBreakpoint(bpRunToCursor);
1071 delete bpRunToCursor;
1074 StartSession(compiler, config, bitDepth, useValgrind, false, ignoreBreakpoints);
1077 if(oldImplementation)
1079 bpRunToCursor = Breakpoint { };
1080 bpRunToCursor.absoluteFilePath = absoluteFilePath;
1081 bpRunToCursor.relativeFilePath = relativeFilePath;
1082 bpRunToCursor.line = lineNumber;
1083 bpRunToCursor.type = runToCursor;
1084 bpRunToCursor.enabled = true;
1085 bpRunToCursor.level = atSameLevel ? frameCount - activeFrameLevel -1 : -1;
1090 breakType = advance;
1091 breakString = PrintString(relativeFilePath, ":", lineNumber);
1094 else if(state == stopped)
1096 if(oldImplementation)
1097 GdbExecContinue(true);
1101 GdbExecUntil(absoluteFilePath, lineNumber);
1103 GdbExecAdvance(absoluteFilePath, lineNumber);
1108 void GetCallStackCursorLine(bool * error, int * lineCursor, int * lineTopFrame)
1110 //_dpcl(_dpct, dplchan::debuggerCall, 0, "Debugger::GetCallStackCursorLine()");
1111 if(activeFrameLevel == -1)
1119 *error = signalOn && activeThread == signalThread;
1120 *lineCursor = activeFrameLevel - ((frameCount > 192 && activeFrameLevel > 191) ? frameCount - 192 - 1 : 0) + 1;
1121 *lineTopFrame = activeFrameLevel ? 1 : 0;
1125 int GetMarginIconsLineNumbers(const char * fileName, int lines[], bool enabled[], int max, bool * error, int * lineCursor, int * lineTopFrame)
1127 char winFilePath[MAX_LOCATION];
1128 char * absoluteFilePath = GetSlashPathBuffer(winFilePath, fileName);
1130 Iterator<Breakpoint> it { ide.workspace.breakpoints };
1131 //_dpcl(_dpct, dplchan::debuggerCall, 0, "Debugger::GetMarginIconsLineNumbers()");
1132 while(it.Next() && count < max)
1134 Breakpoint bp = it.data;
1137 if(bp.absoluteFilePath && bp.absoluteFilePath[0] && !fstrcmp(bp.absoluteFilePath, absoluteFilePath))
1139 lines[count] = bp.line;
1140 enabled[count] = bp.enabled;
1145 if(activeFrameLevel == -1)
1153 *error = signalOn && activeThread == signalThread;
1154 if(activeFrame && activeFrame.absoluteFile && !fstrcmp(absoluteFilePath, activeFrame.absoluteFile))
1155 *lineCursor = activeFrame.line;
1158 if(activeFrame && stopItem && stopItem.frame && activeFrame.level == stopItem.frame.level)
1160 else if(stopItem && stopItem.frame && stopItem.frame.absoluteFile && !fstrcmp(absoluteFilePath, stopItem.frame.absoluteFile))
1161 *lineTopFrame = stopItem.frame.line;
1165 if(*lineTopFrame == *lineCursor && *lineTopFrame)
1171 void ChangeWatch(DataRow row, char * expression)
1173 Watch wh = (Watch)(intptr)row.tag;
1174 //_dpcl(_dpct, dplchan::debuggerCall, 0, "Debugger::ChangeWatch(", expression, ")");
1177 delete wh.expression;
1179 wh.expression = CopyString(expression);
1182 Iterator<Watch> it { ide.workspace.watches };
1184 ide.workspace.watches.Delete(it.pointer);
1190 row.tag = (int64)(intptr)wh;
1191 ide.workspace.watches.Add(wh);
1193 wh.expression = CopyString(expression);
1195 ide.workspace.Save();
1196 //if(expression && state == stopped)
1201 void MoveIcons(const char * fileName, int lineNumber, int move, bool start)
1203 char winFilePath[MAX_LOCATION];
1204 const char * absoluteFilePath = GetSlashPathBuffer(winFilePath, fileName);
1207 //_dpcl(_dpct, dplchan::debuggerCall, 0, "Debugger::MoveIcons()");
1208 if(ide.workspace.breakpoints)
1210 for(bpLink = ide.workspace.breakpoints.first; bpLink; bpLink = next)
1212 Breakpoint bp = (Breakpoint)(intptr)bpLink.data;
1215 if(bp.type == user && bp.absoluteFilePath && !fstrcmp(bp.absoluteFilePath, absoluteFilePath))
1217 if(bp.line > lineNumber || (bp.line == lineNumber && start))
1219 if(move < 0 && (bp.line < lineNumber - move))
1220 ide.workspace.RemoveBreakpoint(bp);
1224 ide.breakpointsView.UpdateBreakpoint(bp.row);
1225 ide.workspace.Save();
1232 // moving code cursors is futile, on next step, stop, hit, cursors will be offset anyways
1235 bool SourceDirDialog(const char * title, const char * startDir, const char * test, char * sourceDir)
1238 String srcDir = null;
1240 _dpcl(_dpct, dplchan::debuggerCall, 0, "Debugger::SourceDirDialog()");
1241 debuggerFileDialog.text = title;
1242 debuggerFileDialog.currentDirectory = startDir;
1243 debuggerFileDialog.master = ide;
1245 while(debuggerFileDialog.Modal())
1247 strcpy(sourceDir, debuggerFileDialog.filePath);
1248 if(!fstrcmp(ide.workspace.projectDir, sourceDir) &&
1249 MessageBox { type = yesNo, master = ide,
1250 contents = $"This is the project directory.\nWould you like to try again?",
1251 text = $"Invalid Source Directory" }.Modal() == no)
1255 for(dir : ide.workspace.sourceDirs)
1257 if(!fstrcmp(dir, sourceDir))
1265 MessageBox { type = yesNo, master = ide,
1266 contents = $"This source directory is already specified.\nWould you like to try again?",
1267 text = $"Invalid Source Directory" }.Modal() == no)
1273 char file[MAX_LOCATION];
1274 strcpy(file, sourceDir);
1275 PathCat(file, test);
1276 result = FileExists(file);
1278 MessageBox { type = yesNo, master = ide,
1279 contents = $"Unable to locate source file.\nWould you like to try again?",
1280 text = $"Invalid Source Directory" }.Modal() == no)
1294 void AddSourceDir(const char * sourceDir)
1296 _dpcl(_dpct, dplchan::debuggerCall, 0, "Debugger::AddSourceDir(", sourceDir, ")");
1297 ide.workspace.sourceDirs.Add(CopyString(sourceDir));
1298 ide.workspace.Save();
1302 DebuggerState oldState = state;
1307 GdbDebugBreak(true);
1310 GdbCommand(0, false, "-environment-directory \"%s\"", sourceDir);
1313 if(oldState == running)
1314 GdbExecContinue(false);
1318 void ToggleBreakpoint(const char * fileName, int lineNumber)
1320 char absolutePath[MAX_LOCATION];
1321 Breakpoint bp = null;
1323 _dpcl(_dpct, dplchan::debuggerBreakpoints, 0, "Debugger::ToggleBreakpoint(", fileName, ":", lineNumber, ")");
1325 GetSlashPathBuffer(absolutePath, fileName);
1326 for(i : ide.workspace.breakpoints; i.type == user && i.absoluteFilePath && !fstrcmp(i.absoluteFilePath, absolutePath) && i.line == lineNumber)
1335 ide.workspace.RemoveBreakpoint(bp);
1344 char relativePath[MAX_LOCATION];
1345 const char * objectFileExt = currentCompiler ? currentCompiler.objectFileExt : objectDefaultFileExt;
1347 ide.workspace.GetRelativePath(absolutePath, relativePath, &owner, objectFileExt);
1349 if(!owner && !FileExists(absolutePath))
1351 char title[MAX_LOCATION];
1352 char directory[MAX_LOCATION];
1353 char sourceDir[MAX_LOCATION];
1354 StripLastDirectory(absolutePath, directory);
1355 snprintf(title, sizeof(title), $"Provide source files location directory for %s", relativePath);
1356 title[sizeof(title)-1] = 0;
1359 String srcDir = null;
1360 for(dir : ide.workspace.sourceDirs)
1362 if(IsPathInsideOf(absolutePath, dir))
1364 MakePathRelative(absolutePath, dir, relativePath);
1372 if(SourceDirDialog(title, directory, null, sourceDir))
1374 if(IsPathInsideOf(absolutePath, sourceDir))
1376 AddSourceDir(sourceDir);
1377 MakePathRelative(absolutePath, sourceDir, relativePath);
1380 else if(MessageBox { type = yesNo, master = ide,
1381 contents = $"You must provide a valid source directory in order to place a breakpoint in this file.\nWould you like to try again?",
1382 text = $"Invalid Source Directory" }.Modal() == no)
1389 ide.workspace.bpCount++;
1390 bp = { line = lineNumber, type = user, enabled = true, level = -1, project = owner };
1391 ide.workspace.breakpoints.Add(bp);
1392 bp.absoluteFilePath = absolutePath;
1393 bp.relativeFilePath = relativePath;
1394 ide.breakpointsView.AddBreakpoint(bp);
1399 DebuggerState oldState = state;
1404 GdbDebugBreak(true);
1407 if(!SetBreakpoint(bp, false))
1408 SetBreakpoint(bp, true);
1411 if(oldState == running)
1412 GdbExecContinue(false);
1415 ide.workspace.Save();
1418 void UpdateRemovedBreakpoint(Breakpoint bp)
1420 _dpcl(_dpct, dplchan::debuggerCall, 0, "Debugger::UpdateRemovedBreakpoint()");
1421 if(targeted && bp.inserted)
1423 DebuggerState oldState = state;
1428 GdbDebugBreak(true);
1431 UnsetBreakpoint(bp);
1434 if(oldState == running)
1435 GdbExecContinue(false);
1441 void ParseFrame(Frame frame, char * string)
1444 Array<char *> frameTokens { minAllocSize = 50 };
1445 Array<char *> argsTokens { minAllocSize = 50 };
1446 Array<char *> argumentTokens { minAllocSize = 50 };
1447 DebugListItem item { };
1450 //_dpcl(_dpct, dplchan::debuggerCall, 0, "Debugger::ParseFrame()");
1451 TokenizeList(string, ',', frameTokens);
1452 for(i = 0; i < frameTokens.count; i++)
1454 if(TokenizeListItem(frameTokens[i], item))
1456 StripQuotes(item.value, item.value);
1457 if(!strcmp(item.name, "level"))
1458 frame.level = atoi(item.value);
1459 else if(!strcmp(item.name, "addr"))
1460 frame.addr = item.value;
1461 else if(!strcmp(item.name, "func"))
1462 frame.func = item.value;
1463 else if(!strcmp(item.name, "args"))
1465 if(!strcmp(item.value, "[]"))
1466 frame.argsCount = 0;
1469 item.value = StripBrackets(item.value);
1470 TokenizeList(item.value, ',', argsTokens);
1471 for(j = 0; j < argsTokens.count; j++)
1473 argsTokens[j] = StripCurlies(argsTokens[j]);
1474 TokenizeList(argsTokens[j], ',', argumentTokens);
1475 for(k = 0; k < argumentTokens.count; k++)
1478 frame.args.Add(arg);
1479 if(TokenizeListItem(argumentTokens[k], item))
1481 if(!strcmp(item.name, "name"))
1483 StripQuotes(item.value, item.value);
1484 arg.name = item.value;
1486 else if(!strcmp(item.name, "value"))
1488 StripQuotes(item.value, item.value);
1489 arg.val = item.value;
1492 _dpcl(_dpct, dplchan::gdbProtoUnknown, 0, "frame args item (", item.name, "=", item.value, ") is unheard of");
1495 _dplf(0, "Bad frame args item");
1497 argumentTokens.RemoveAll();
1499 frame.argsCount = argsTokens.count;
1500 argsTokens.RemoveAll();
1503 else if(!strcmp(item.name, "from"))
1504 frame.from = item.value;
1505 else if(!strcmp(item.name, "file"))
1506 frame.file = item.value;
1507 else if(!strcmp(item.name, "line"))
1508 frame.line = atoi(item.value);
1509 else if(!strcmp(item.name, "fullname"))
1511 // GDB 6.3 on OS X is giving "fullname" and "dir", all in absolute, but file name only in 'file'
1512 Workspace ws = ide.workspace;
1515 String path = ide.workspace.CopyUnixPathWorkspaceRelativeOrAbsolute(item.value);
1516 if(strcmp(frame.file, path))
1520 frame.absoluteFile = item.value;
1523 _dpcl(_dpct, dplchan::gdbProtoUnknown, 0, "frame member (", item.name, "=", item.value, ") is unheard of");
1526 _dplf(0, "Bad frame");
1531 delete argumentTokens;
1535 Breakpoint GetBreakpointById(int id, bool * isInternal)
1537 Breakpoint bp = null;
1538 //_dpcl(_dpct, dplchan::debuggerBreakpoints, 0, "Debugger::GetBreakpointById(", id, ")");
1540 *isInternal = false;
1543 for(i : sysBPs; i.bp && i.bp.id == id)
1550 if(!bp && bpRunToCursor && bpRunToCursor.bp && bpRunToCursor.bp.id == id)
1554 for(i : ide.workspace.breakpoints; i.bp && i.bp.id == id)
1564 GdbDataBreakpoint ParseBreakpoint(char * string, Array<char *> outTokens)
1567 GdbDataBreakpoint bp { };
1568 DebugListItem item { };
1569 Array<char *> bpTokens { minAllocSize = 16 };
1570 _dpcl(_dpct, dplchan::debuggerCall, 0, "Debugger::ParseBreakpoint()");
1571 string = StripCurlies(string);
1572 TokenizeList(string, ',', bpTokens);
1573 for(i = 0; i < bpTokens.count; i++)
1575 if(TokenizeListItem(bpTokens[i], item))
1577 StripQuotes(item.value, item.value);
1578 if(!strcmp(item.name, "number"))
1580 if(!strchr(item.value, '.'))
1581 bp.id = atoi(item.value);
1582 bp.number = item.value;
1584 else if(!strcmp(item.name, "type"))
1585 bp.type = item.value;
1586 else if(!strcmp(item.name, "disp"))
1587 bp.disp = item.value;
1588 else if(!strcmp(item.name, "enabled"))
1589 bp.enabled = (!strcmpi(item.value, "y"));
1590 else if(!strcmp(item.name, "addr"))
1592 if(outTokens && !strcmp(item.value, "<MULTIPLE>"))
1595 Array<GdbDataBreakpoint> bpArray = bp.multipleBPs = { };
1596 while(outTokens.count > ++c)
1598 GdbDataBreakpoint multBp = ParseBreakpoint(outTokens[c], null);
1599 bpArray.Add(multBp);
1603 bp.addr = item.value;
1605 else if(!strcmp(item.name, "func"))
1606 bp.func = item.value;
1607 else if(!strcmp(item.name, "file"))
1608 bp.file = item.value;
1609 else if(!strcmp(item.name, "fullname"))
1610 bp.fullname = item.value;
1611 else if(!strcmp(item.name, "line"))
1612 bp.line = atoi(item.value);
1613 else if(!strcmp(item.name, "at"))
1615 else if(!strcmp(item.name, "times"))
1616 bp.times = atoi(item.value);
1617 else if(!strcmp(item.name, "original-location") || !strcmp(item.name, "thread-groups"))
1618 _dpcl(_dpct, dplchan::gdbProtoIgnored, 0, "breakpoint member (", item.name, "=", item.value, ") is ignored");
1620 _dpcl(_dpct, dplchan::gdbProtoUnknown, 0, "breakpoint member (", item.name, "=", item.value, ") is unheard of");
1628 void ShowDebuggerViews()
1630 //_dpcl(_dpct, dplchan::debuggerCall, 0, "Debugger::ShowDebuggerViews()");
1631 ide.outputView.Show();
1632 ide.outputView.SelectTab(debug);
1633 ide.threadsView.Show();
1634 ide.callStackView.Show();
1635 ide.watchesView.Show();
1639 void HideDebuggerViews()
1641 //_dpcl(_dpct, dplchan::debuggerCall, 0, "Debugger::HideDebuggerViews()");
1642 ide.RepositionWindows(true);
1645 bool ::GdbCommand(Time timeOut, bool focus, const char * format, ...)
1647 bool result = false;
1651 // TODO: Improve this limit
1652 static char string[MAX_F_STRING*4];
1654 va_start(args, format);
1655 vsnprintf(string, sizeof(string), format, args);
1656 string[sizeof(string)-1] = 0;
1660 ide.debugger.serialSemaphore.TryWait();
1662 #ifdef GDB_DEBUG_CONSOLE
1663 _dpcl(_dpct, dplchan::gdbCommand, 0, string);
1665 #ifdef GDB_DEBUG_OUTPUT
1666 ide.outputView.gdbBox.Logf("cmd: %s\n", string);
1668 #ifdef GDB_DEBUG_GUI
1670 ide.gdbDialog.AddCommand(string);
1673 strcat(string,"\n");
1674 gdbHandle.Puts(string);
1677 Process_ShowWindows(targetProcessId);
1683 startTime = GetTime();
1686 if(ide.debugger.serialSemaphore.TryWait())
1693 if(GetTime() - startTime > timeOut)
1701 ide.debugger.serialSemaphore.Wait();
1710 bool ValidateBreakpoint(Breakpoint bp)
1712 _dpcl(_dpct, dplchan::debuggerCall, 0, "Debugger::ValidateBreakpoint()");
1713 if(modules && bp.line && bp.bp)
1715 if(bp.bp.line != bp.line)
1721 ide.outputView.debugBox.Logf("WOULD HAVE -- Invalid breakpoint disabled: %s:%d\n", bp.relativeFilePath, bp.line);
1723 //UnsetBreakpoint(bp);
1724 //bp.enabled = false;
1730 ide.outputView.debugBox.Logf("Debugger Error: ValidateBreakpoint error\n");
1731 bp.line = bp.bp.line;
1738 void BreakpointsMaintenance()
1740 //_dpcl(_dpct, dplchan::debuggerBreakpoints, 0, "Debugger::BreakpointsMaintenance()");
1743 if(gdbExecution.suspendInternalBreakpoints)
1745 for(bp : sysBPs; bp.inserted)
1746 UnsetBreakpoint(bp);
1750 DirExpression objDir = ide.project.GetObjDir(currentCompiler, prjConfig, bitDepth);
1751 for(bp : sysBPs; !bp.inserted)
1753 bool insert = false;
1754 if(bp.type == internalModulesLoaded)
1756 char path[MAX_LOCATION];
1757 char name[MAX_LOCATION];
1758 char fixedModuleName[MAX_FILENAME];
1761 bool moduleLoadBlock = false;
1763 ReplaceSpaces(fixedModuleName, ide.project.moduleName);
1764 snprintf(name, sizeof(name),"%s.main.ec", fixedModuleName);
1765 name[sizeof(name)-1] = 0;
1766 strcpy(path, ide.workspace.projectDir);
1767 PathCatSlash(path, objDir.dir);
1768 PathCatSlash(path, name);
1769 f = FileOpen(path, read);
1772 for(lineNumber = 1; !f.Eof(); lineNumber++)
1774 if(f.GetLine(line, sizeof(line) - 1))
1776 bool moduleLoadLine;
1777 TrimLSpaces(line, line);
1778 moduleLoadLine = !strncmp(line, "eModule_Load", strlen("eModule_Load"));
1779 if(!moduleLoadBlock && moduleLoadLine)
1780 moduleLoadBlock = true;
1781 else if(moduleLoadBlock && !moduleLoadLine && strlen(line) > 0)
1787 char relative[MAX_LOCATION];
1788 bp.absoluteFilePath = path;
1789 MakePathRelative(path, ide.workspace.projectDir, relative);
1790 bp.relativeFilePath = relative;
1791 bp.line = lineNumber;
1797 else if(bp.type == internalModuleLoad)
1801 for(prj : ide.workspace.projects)
1803 if(!strcmp(prj.moduleName, "ecere"))
1805 ProjectNode node = prj.topNode.Find("instance.c", false);
1808 char path[MAX_LOCATION];
1809 char relative[MAX_LOCATION];
1810 node.GetFullFilePath(path);
1811 bp.absoluteFilePath = path;
1812 MakePathRelative(path, prj.topNode.path, relative);
1813 bp.relativeFilePath = relative;
1825 if(!SetBreakpoint(bp, false))
1826 SetBreakpoint(bp, true);
1832 if(userAction != runToCursor && bpRunToCursor && bpRunToCursor.inserted)
1833 UnsetBreakpoint(bpRunToCursor);
1834 if(bpRunToCursor && !bpRunToCursor.inserted)
1836 if(!SetBreakpoint(bpRunToCursor, false))
1837 SetBreakpoint(bpRunToCursor, true);
1840 if(ignoreBreakpoints)
1842 for(bp : ide.workspace.breakpoints; bp.inserted)
1843 UnsetBreakpoint(bp);
1847 for(bp : ide.workspace.breakpoints; !bp.inserted && bp.type == user)
1851 if(!SetBreakpoint(bp, false))
1852 SetBreakpoint(bp, true);
1858 _dplf(0, "problem");
1861 bp.bp = GdbDataBreakpoint { };
1868 void UnsetBreakpoint(Breakpoint bp)
1870 char * s = null; _dpcl(_dpct, dplchan::debuggerBreakpoints, 0, "Debugger::UnsetBreakpoint(", s=bp.CopyLocationString(false), ") -- ", bp.type); delete s;
1871 if(symbols && bp.inserted)
1873 GdbCommand(0, false, "-break-delete %s", bp.bp.number);
1874 bp.inserted = false;
1880 bool SetBreakpoint(Breakpoint bp, bool removePath)
1882 char * s = null; _dpcl(_dpct, dplchan::debuggerBreakpoints, 0, "Debugger::SetBreakpoint(", s=bp.CopyLocationString(false), ", ", removePath ? "**** removePath(true) ****" : "", ") -- ", bp.type); delete s;
1883 breakpointError = false;
1884 if(symbols && bp.enabled && (!bp.project || bp.project.GetTargetType(bp.project.config) == staticLibrary || bp.project == ide.project || projectsLibraryLoaded[bp.project.name]))
1886 sentBreakInsert = true;
1888 GdbCommand(0, false, "-break-insert *%s", bp.address);
1891 char * location = bp.CopyLocationString(removePath);
1892 GdbCommand(0, false, "-break-insert %s", location);
1895 if(!breakpointError)
1897 char * address = null;
1898 if(bpItem && bpItem.multipleBPs && bpItem.multipleBPs.count)
1901 GdbDataBreakpoint first = null;
1902 for(n : bpItem.multipleBPs)
1904 if(!fstrcmp(n.fullname, bp.absoluteFilePath) && !first)
1914 GdbCommand(0, false, "-break-disable %s", n.number);
1918 _dpcl(_dpct, dplchan::debuggerProblem, 0, "Debugger::SetBreakpoint -- error breakpoint already disabled.");
1923 address = CopyString(first.addr);
1924 bpItem.addr = first.addr;
1925 bpItem.func = first.func;
1926 bpItem.file = first.file;
1927 bpItem.fullname = first.fullname;
1928 bpItem.line = first.line;
1929 //bpItem.thread-groups = first.thread-groups;*/
1932 _dpcl(_dpct, dplchan::debuggerProblem, 0, "Debugger::SetBreakpoint -- error multiple breakpoints all disabled.");
1934 _dpcl(_dpct, dplchan::debuggerProblem, 0, "Debugger::SetBreakpoint -- error multiple breakpoints in exact same file not supported.");
1935 bpItem.multipleBPs.Free();
1936 delete bpItem.multipleBPs;
1941 bp.inserted = (bp.bp && bp.bp.number && strcmp(bp.bp.number, "0"));
1943 ValidateBreakpoint(bp);
1947 UnsetBreakpoint(bp);
1948 bp.address = address;
1950 SetBreakpoint(bp, removePath);
1953 return !breakpointError;
1960 _dpcl(_dpct, dplchan::debuggerCall, 0, "Debugger::GdbGetStack()");
1962 stackFrames.Free(Frame::Free);
1963 GdbCommand(0, false, "-stack-info-depth");
1965 GdbCommand(0, false, "-stack-info-depth 192");
1966 if(frameCount && frameCount <= 192)
1967 GdbCommand(0, false, "-stack-list-frames 0 %d", Min(frameCount-1, 191));
1970 GdbCommand(0, false, "-stack-list-frames 0 %d", Min(frameCount-1, 95));
1971 GdbCommand(0, false, "-stack-list-frames %d %d", Max(frameCount - 96, 96), frameCount - 1);
1973 GdbCommand(0, false, "");
1978 _dpcl(_dpct, dplchan::debuggerCall, 0, "Debugger::GdbTargetSet()");
1981 char escaped[MAX_LOCATION];
1982 strescpy(escaped, targetFile);
1983 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
1990 const char *vgdbCommand = "/usr/bin/vgdb"; // TODO: vgdb command config option
1991 //GdbCommand(0, false, "-target-select remote | %s --pid=%d", "vgdb", targetProcessId);
1992 printf("target remote | %s --pid=%d\n", vgdbCommand, targetProcessId);
1993 GdbCommand(0, false, "target remote | %s --pid=%d", vgdbCommand, targetProcessId); // TODO: vgdb command config option
1996 GdbCommand(0, false, "info target"); //GDB/MI Missing Implementation -file-list-symbol-files and -file-list-exec-sections
1998 /*for(prj : ide.workspace.projects; prj != ide.workspace.projects.firstIterator.data)
1999 GdbCommand(0, false, "-environment-directory \"%s\"", prj.topNode.path);*/
2001 for(dir : ide.workspace.sourceDirs; dir && dir[0])
2003 bool interference = false;
2004 for(prj : ide.workspace.projects)
2006 if(!fstrcmp(prj.topNode.path, dir))
2008 interference = true;
2012 if(!interference && dir[0])
2013 GdbCommand(0, false, "-environment-directory \"%s\"", dir);
2021 /*void GdbTargetRelease()
2025 BreakpointsDeleteAll();
2026 GdbCommand(0, false, "file"); //GDB/MI Missing Implementation -target-detach
2032 void GdbDebugBreak(bool internal)
2034 _dpcl(_dpct, dplchan::debuggerCall, 0, "Debugger::GdbDebugBreak()");
2038 breakType = DebuggerAction::internal;
2040 if(ide) ide.Update(null);
2042 if(Process_Break(targetProcessId)) //GdbCommand(0, false, "-exec-interrupt");
2043 serialSemaphore.Wait();
2046 _ChangeState(loaded);
2047 targetProcessId = 0;
2052 ide.outputView.debugBox.Logf("Debugger Error: GdbDebugBreak with not target id should never happen\n");
2057 _dpcl(_dpct, dplchan::debuggerCall, 0, "Debugger::GdbExecRun()");
2062 ShowDebuggerViews();
2064 GdbExecContinue(true);
2065 else if(!GdbCommand(3, true, "-exec-run"))
2066 gdbExecution = none;
2069 void GdbExecContinue(bool focus)
2071 _dpcl(_dpct, dplchan::debuggerCall, 0, "Debugger::GdbExecContinue()");
2074 GdbCommand(0, focus, "-exec-continue");
2079 _dpcl(_dpct, dplchan::debuggerCall, 0, "Debugger::GdbExecNext()");
2080 gdbExecution = next;
2082 GdbCommand(0, true, "-exec-next");
2085 void ForceUpdateCurrentFrame()
2088 GdbCommand(0, false, "-thread-list-ids");
2089 InternalSelectFrame(activeFrameLevel);
2090 GoToStackFrameLine(activeFrameLevel, true, false);
2092 ide.ShowCodeEditor();
2093 ide.AdjustDebugMenus();
2094 ideMainFrame.Activate(); // TOFIX: ide.Activate() is not reliable (app inactive)
2098 void GdbExecUntil(const char * absoluteFilePath, int lineNumber)
2100 bool forceUpdate = false;
2101 char relativeFilePath[MAX_LOCATION];
2102 _dpcl(_dpct, dplchan::debuggerCall, 0, "Debugger::GdbExecUntil()");
2103 gdbExecution = until;
2105 if(absoluteFilePath)
2107 const char * objectFileExt = currentCompiler ? currentCompiler.objectFileExt : objectDefaultFileExt;
2108 ide.workspace.GetRelativePath(absoluteFilePath, relativeFilePath, null, objectFileExt);
2109 if(!GdbCommand(0.1, true, "-exec-until %s:%d", relativeFilePath, lineNumber))
2111 GetLastDirectory(relativeFilePath, relativeFilePath);
2112 if(GdbCommand(1, true, "-exec-until %s:%d", relativeFilePath, lineNumber))
2117 GdbCommand(0, true, "-exec-until");
2119 // This is to handle GDB 6.3 on OS X not giving us *running then *stopped:
2120 // (It may not be ideal, we may need to wait?)
2122 ForceUpdateCurrentFrame();
2125 void GdbExecAdvance(const char * absoluteFilePathOrLocation, int lineNumber)
2127 bool forceUpdate = false;
2128 char relativeFilePath[MAX_LOCATION];
2129 _dpcl(_dpct, dplchan::debuggerCall, 0, "Debugger::GdbExecAdvance()");
2130 gdbExecution = advance;
2134 const char * objectFileExt = currentCompiler ? currentCompiler.objectFileExt : objectDefaultFileExt;
2135 ide.workspace.GetRelativePath(absoluteFilePathOrLocation, relativeFilePath, null, objectFileExt);
2136 if(!GdbCommand(0.1, true, "advance %s:%d", relativeFilePath, lineNumber)) // should use -exec-advance -- GDB/MI implementation missing
2138 GetLastDirectory(relativeFilePath, relativeFilePath);
2139 if(GdbCommand(1, true, "advance %s:%d", relativeFilePath, lineNumber))
2145 if(!GdbCommand(0.1, true, "advance %s", absoluteFilePathOrLocation)) // should use -exec-advance -- GDB/MI implementation missing
2147 GetLastDirectory(absoluteFilePathOrLocation, relativeFilePath);
2148 if(GdbCommand(1, true, "advance %s", relativeFilePath))
2153 // This is to handle GDB 6.3 on OS X not giving us *running then *stopped:
2154 // (It may not be ideal, we may need to wait?)
2156 ForceUpdateCurrentFrame();
2161 _dpcl(_dpct, dplchan::debuggerCall, 0, "Debugger::GdbExecStep()");
2162 gdbExecution = step;
2164 GdbCommand(0, true, "-exec-step");
2167 void GdbExecFinish()
2169 _dpcl(_dpct, dplchan::debuggerCall, 0, "Debugger::GdbExecFinish()");
2170 gdbExecution = finish;
2172 GdbCommand(0, true, "-exec-finish");
2175 void GdbExecCommon()
2177 //_dpcl(_dpct, dplchan::debuggerCall, 0, "Debugger::GdbExecCommon()");
2178 BreakpointsMaintenance();
2181 #ifdef GDB_DEBUG_GUI
2182 void SendGDBCommand(const char * command)
2184 //_dpcl(_dpct, dplchan::debuggerCall, 0, "Debugger::SendGDBCommand()");
2185 DebuggerState oldState = state;
2190 GdbDebugBreak(true);
2193 GdbCommand(0, false, command);
2196 if(oldState == running)
2197 GdbExecContinue(false);
2201 void ClearBreakDisplay()
2203 //_dpcl(_dpct, dplchan::debuggerCall, 0, "Debugger::ClearBreakDisplay()");
2205 activeFrameLevel = -1;
2215 stackFrames.Free(Frame::Free);
2216 ide.callStackView.Clear();
2217 ide.threadsView.Clear();
2223 _dpcl(_dpct, dplchan::debuggerCall, 0, "Debugger::GdbAbortExec()");
2225 GdbCommand(0, false, "-interpreter-exec console \"kill\""); // should use -exec-abort -- GDB/MI implementation incomplete
2229 bool GdbInit(CompilerConfig compiler, ProjectConfig config, int bitDepth, bool useValgrind)
2232 char oldDirectory[MAX_LOCATION];
2233 char tempPath[MAX_LOCATION];
2234 char command[MAX_F_STRING*4];
2235 Project project = ide.project;
2236 DirExpression targetDirExp = project.GetTargetDir(compiler, config, bitDepth);
2237 PathBackup pathBackup { };
2238 Map<String, String> envBackup { };
2240 _dpcl(_dpct, dplchan::debuggerCall, 0, "Debugger::GdbInit()");
2241 if(currentCompiler != compiler)
2243 delete currentCompiler;
2244 currentCompiler = compiler;
2245 incref currentCompiler;
2248 this.bitDepth = bitDepth;
2249 usingValgrind = useValgrind;
2251 _ChangeState(loaded);
2253 sentBreakInsert = false;
2254 breakpointError = false;
2255 ignoreBreakpoints = false;
2261 projectsLibraryLoaded.Free();
2263 ide.outputView.ShowClearSelectTab(debug);
2264 ide.outputView.debugBox.Logf($"Starting debug mode\n");
2266 #ifdef GDB_DEBUG_OUTPUT
2267 ide.outputView.gdbBox.Logf("run: Starting GDB\n");
2270 strcpy(tempPath, ide.workspace.projectDir);
2271 PathCatSlash(tempPath, targetDirExp.dir);
2273 targetDir = CopyString(tempPath);
2274 project.CatTargetFileName(tempPath, compiler, config);
2276 targetFile = CopyString(tempPath);
2278 GetWorkingDir(oldDirectory, MAX_LOCATION);
2279 if(ide.workspace.debugDir && ide.workspace.debugDir[0])
2281 char temp[MAX_LOCATION];
2282 strcpy(temp, ide.workspace.projectDir);
2283 PathCatSlash(temp, ide.workspace.debugDir);
2284 ChangeWorkingDir(temp);
2287 ChangeWorkingDir(ide.workspace.projectDir);
2289 ide.SetPath(true, compiler, config, bitDepth);
2291 // TODO: This pollutes the environment, but at least it works
2292 // It shouldn't really affect the IDE as the PATH gets restored and other variables set for testing will unlikely cause problems
2293 // What is the proper solution for this? DualPipeOpenEnv?
2294 // gdb set environment commands don't seem to take effect
2295 for(e : ide.workspace.environmentVars)
2297 // Backing up the environment variables until we use DualPipeOpenEnv()
2298 envBackup[e.name] = CopyString(getenv(e.name));
2299 SetEnvironment(e.name, e.string);
2304 char * clArgs = ide.workspace.commandLineArgs;
2305 const char *valgrindCommand = "valgrind"; // TODO: valgrind command config option //TODO: valgrind options
2306 ValgrindLeakCheck vgLeakCheck = ide.workspace.vgLeakCheck;
2307 int vgRedzoneSize = ide.workspace.vgRedzoneSize;
2308 bool vgTrackOrigins = ide.workspace.vgTrackOrigins;
2309 vgLogFile = CreateTemporaryFile(vgLogPath, "ecereidevglog");
2313 vgLogThread.Create();
2317 ide.outputView.debugBox.Logf($"Debugger Fatal Error: Couldn't open temporary log file for Valgrind output\n");
2320 if(result && !CheckCommandAvailable(valgrindCommand))
2322 ide.outputView.debugBox.Logf($"Debugger Fatal Error: Command %s for Valgrind is not available.\n", valgrindCommand);
2327 char * vgRedzoneSizeFlag = PrintString(" --redzone-size=", vgRedzoneSize);
2328 sprintf(command, "%s --vgdb=yes --vgdb-error=0 --log-file=%s --leak-check=%s%s --track-origins=%s %s%s%s",
2329 valgrindCommand, vgLogPath, (char*)vgLeakCheck, vgRedzoneSize > -1 ? vgRedzoneSizeFlag : "", vgTrackOrigins ? "yes" : "no", targetFile, clArgs ? " " : "", clArgs ? clArgs : "");
2330 delete vgRedzoneSizeFlag;
2331 vgTargetHandle = DualPipeOpen(PipeOpenMode { output = true, /*error = true, */input = true }, command);
2334 ide.outputView.debugBox.Logf($"Debugger Fatal Error: Couldn't start Valgrind\n");
2340 incref vgTargetHandle;
2341 vgTargetThread.Create();
2343 targetProcessId = vgTargetHandle.GetProcessID();
2344 waitingForPID = false;
2345 if(!targetProcessId)
2347 ide.outputView.debugBox.Logf($"Debugger Fatal Error: Couldn't get Valgrind process ID\n");
2354 serialSemaphore.Wait();
2361 if(compiler.targetPlatform == win32)
2364 #if !((defined(__WORDSIZE) && __WORDSIZE == 8) || defined(__x86_64__))
2367 bitDepth == 32 ? "i686-w64-mingw32-gdb" : "gdb"); // x86_64-w64-mingw32-gdb
2370 // We really should have a box to select GDB in the compiler/toolchain options
2371 strcpy(command, "gdb");
2372 if(!CheckCommandAvailable(command))
2374 ide.outputView.debugBox.Logf($"Debugger Fatal Error: Command %s for GDB is not available.\n", command);
2379 strcat(command, " -n -silent --interpreter=mi2"); //-async //\"%s\"
2381 gdbHandle = DualPipeOpen(PipeOpenMode { output = true, /*error = true, */input = true }, command);
2384 ide.outputView.debugBox.Logf($"Debugger Fatal Error: Couldn't start GDB\n");
2394 gdbProcessId = gdbHandle.GetProcessID();
2397 ide.outputView.debugBox.Logf($"Debugger Fatal Error: Couldn't get GDB process ID\n");
2404 serialSemaphore.Wait();
2407 GdbCommand(0, false, "-gdb-set verbose off");
2408 //GdbCommand(0, false, "-gdb-set exec-done-display on");
2409 GdbCommand(0, false, "-gdb-set step-mode off");
2410 GdbCommand(0, false, "-gdb-set unwindonsignal on");
2411 //GdbCommand(0, false, "-gdb-set shell on");
2412 GdbCommand(0, false, "set print elements 992");
2413 GdbCommand(0, false, "-gdb-set backtrace limit 100000");
2417 //_ChangeState(terminated);
2423 #if defined(__unix__)
2425 CreateTemporaryDir(progFifoDir, "ecereide");
2426 strcpy(progFifoPath, progFifoDir);
2427 PathCat(progFifoPath, "ideprogfifo");
2428 if(!mkfifo(progFifoPath, 0600))
2430 //fileCreated = true;
2435 ide.outputView.debugBox.Logf(createFIFOMsg, progFifoPath);
2442 progThread.terminate = false;
2443 progThread.Create();
2447 #if defined(__WIN32__)
2448 GdbCommand(0, false, "-gdb-set new-console on");
2451 #if defined(__unix__)
2453 GdbCommand(0, false, "-inferior-tty-set %s", progFifoPath);
2457 GdbCommand(0, false, "-gdb-set args %s", ide.workspace.commandLineArgs ? ide.workspace.commandLineArgs : "");
2459 for(e : ide.workspace.environmentVars)
2461 GdbCommand(0, false, "set environment %s=%s", e.name, e.string);
2466 ChangeWorkingDir(oldDirectory);
2470 SetEnvironment(&e, e);
2479 delete targetDirExp;
2485 _dpcl(_dpct, dplchan::debuggerCall, 0, "Debugger::GdbExit()");
2486 if(gdbHandle && gdbProcessId)
2489 GdbCommand(0, false, "-gdb-exit");
2498 vgLogFile.CloseInput();
2499 if(vgLogThread.created)
2509 vgTargetThread.Wait();
2519 _ChangeState(terminated); // this state change seems to be superfluous, is it safety for something?
2524 for(bp : ide.workspace.breakpoints)
2530 bpRunToCursor.Reset();
2532 ide.outputView.debugBox.Logf($"Debugging stopped\n");
2534 #if defined(__unix__)
2535 if(!usingValgrind && FileExists(progFifoPath)) //fileCreated)
2537 progThread.terminate = true;
2540 fifoFile.CloseInput();
2546 DeleteFile(progFifoPath);
2547 progFifoPath[0] = '\0';
2556 bool WatchesLinkCodeEditor()
2558 bool goodFrame = activeFrame && activeFrame.absoluteFile;
2559 //_dpcl(_dpct, dplchan::debuggerCall, 0, "Debugger::WatchesLinkCodeEditor()");
2560 if(codeEditor && (!goodFrame || fstrcmp(codeEditor.fileName, activeFrame.absoluteFile)))
2561 WatchesReleaseCodeEditor();
2563 if(!codeEditor && goodFrame)
2565 codeEditor = (CodeEditor)ide.OpenFile(activeFrame.absoluteFile, false, false, null, no, normal, false);
2568 codeEditor.inUseDebug = true;
2572 return codeEditor != null;
2575 void WatchesReleaseCodeEditor()
2577 //_dpcl(_dpct, dplchan::debuggerCall, 0, "Debugger::WatchesReleaseCodeEditor()");
2580 codeEditor.inUseDebug = false;
2581 if(!codeEditor.visible)
2582 codeEditor.Destroy(0);
2587 bool ResolveWatch(Watch wh)
2589 bool result = false;
2591 _dpcl(_dpct, dplchan::debuggerWatches, 0, "Debugger::ResolveWatch()");
2603 char watchmsg[MAX_F_STRING];
2605 if(state == stopped && !codeEditor)
2606 wh.value = CopyString($"No source file found for selected frame");
2607 //if(codeEditor && state == stopped || state != stopped)
2610 Module backupPrivateModule;
2611 Context backupContext;
2612 Class backupThisClass;
2615 backupPrivateModule = GetPrivateModule();
2616 backupContext = GetCurrentContext();
2617 backupThisClass = GetThisClass();
2620 SetPrivateModule(codeEditor.privateModule);
2621 SetCurrentContext(codeEditor.globalContext);
2622 SetTopContext(codeEditor.globalContext);
2623 SetGlobalContext(codeEditor.globalContext);
2624 SetGlobalData(&codeEditor.globalData);
2627 exp = ParseExpressionString(wh.expression);
2629 if(exp && !GetParseError())
2631 char expString[4096];
2633 PrintExpression(exp, expString);
2635 SetInDebugger(true);
2638 // NOTE: DebugFindCtxTree() should be called only once for evaluating all watches in the watch window
2639 if(codeEditor && activeFrame)
2640 DebugFindCtxTree(codeEditor.ast, activeFrame.line, 0);
2641 ProcessExpressionType(exp);
2643 wh.type = exp.expType;
2646 DebugComputeExpression(exp);
2647 // e.g. Meters * Degrees has no type set yet for some reason
2648 if(!wh.type && exp.expType)
2650 wh.type = exp.expType;
2651 exp.expType.refCount++;
2654 // This makes Degrees { 45 } work
2655 if(exp.type == constantExp && exp.isConstant && exp.expType && exp.expType.kind == classType &&
2656 exp.expType._class && exp.expType._class.registered && exp.expType._class.registered.type == unitClass && exp.expType._class.registered.base.type == unitClass)
2658 ApplyUnitConverters(exp);
2660 else if(exp.type == instanceExp && exp.instance.data)
2662 Symbol s = exp.instance._class ? exp.instance._class.symbol : null;
2663 Class c = s ? s.registered : null;
2667 bool needClass = false;
2668 char * s = ((char * (*)(void *, void *, void *, void *, void *))(void *)c._vTbl[__ecereVMethodID_class_OnGetString])(c, exp.instance.data, tmp, null, &needClass);
2671 FreeExpContents(exp);
2672 exp.type = constantExp;
2673 exp.isConstant = true;
2674 exp.constant = CopyString(s);
2678 else if(exp.expType && exp.expType.kind == classType && exp.expType._class && exp.expType._class.registered && exp.expType._class.registered.type == bitClass)
2680 Class c = exp.expType._class.registered;
2682 bool needClass = false;
2683 Operand op = GetOperand(exp);
2685 char * (* onGetString)(void *, void *, void *, void *, void *) = (void *)c._vTbl[__ecereVMethodID_class_OnGetString];
2688 if(op.type) op.type.refCount++;
2691 case charType: s = onGetString(c, &op.c, tmp, null, &needClass); break;
2692 case shortType: s = onGetString(c, &op.s, tmp, null, &needClass); break;
2693 case intType: s = onGetString(c, &op.i, tmp, null, &needClass); break;
2694 case int64Type: s = onGetString(c, &op.i64, tmp, null, &needClass); break;
2699 FreeExpContents(exp);
2700 exp.type = constantExp;
2701 exp.isConstant = true;
2702 exp.constant = CopyString(s);
2706 else if(exp.expType && exp.expType.kind == classType && exp.expType._class && exp.expType._class.registered && exp.expType._class.registered.type == structClass && exp.hasAddress)
2708 Class c = exp.expType._class.registered;
2709 char structString[1024];
2710 strcpy(structString, "*(struct ");
2711 FullClassNameCat(structString, c.fullName, false);
2712 strcat(structString, " *)");
2713 strcatf(structString, "0x%p", exp.address);
2714 GDBFallBack(exp, structString);
2716 byte * data = GdbReadMemory(exp.address, c.structSize);
2720 bool needClass = false;
2721 char * s = ((char * (*)(void *, void *, void *, void *, void *))(void *)c._vTbl[__ecereVMethodID_class_OnGetString])(c, data, tmp, null, &needClass);
2724 FreeExpContents(exp);
2725 exp.type = constantExp;
2726 exp.isConstant = true;
2727 exp.constant = CopyString(s);
2734 if(ExpressionIsError(exp) && exp.type != functionCallErrorExp)
2736 GDBFallBack(exp, expString);
2739 /*if(exp.hasAddress)
2741 char temp[MAX_F_STRING];
2742 sprintf(temp, "0x%x", exp.address);
2743 wh.address = CopyString(temp);
2744 // wh.address = CopyStringf("0x%x", exp.address);
2749 Type dataType = exp.expType;
2752 char temp[MAX_F_STRING];
2753 switch(dataType.kind)
2756 sprintf(temp, "%i", exp.val.c);
2759 sprintf(temp, "%i", exp.val.s);
2764 sprintf(temp, "%i", exp.val.i);
2767 sprintf(temp, "%i", exp.val.i64);
2770 sprintf(temp, "%i", exp.val.p);
2775 long v = (long)exp.val.f;
2776 sprintf(temp, "%i", v);
2781 long v = (long)exp.val.d;
2782 sprintf(temp, "%i", v);
2787 wh.intVal = CopyString(temp);
2788 switch(dataType.kind)
2791 sprintf(temp, "0x%x", exp.val.c);
2794 sprintf(temp, "0x%x", exp.val.s);
2798 sprintf(temp, "0x%x", exp.val.i);
2801 sprintf(temp, "0x%x", exp.val.i64);
2804 sprintf(temp, "0x%x", exp.val.i64);
2807 sprintf(temp, "0x%x", exp.val.p);
2812 long v = (long)exp.val.f;
2813 sprintf(temp, "0x%x", v);
2818 long v = (long)exp.val.d;
2819 sprintf(temp, "0x%x", v);
2824 wh.hexVal = CopyString(temp);
2825 switch(dataType.kind)
2828 sprintf(temp, "0o%o", exp.val.c);
2831 sprintf(temp, "0o%o", exp.val.s);
2835 sprintf(temp, "0o%o", exp.val.i);
2838 sprintf(temp, "0o%o", exp.val.i64);
2841 sprintf(temp, "0o%o", exp.val.i64);
2844 sprintf(temp, "0o%o", exp.val.p);
2849 long v = (long)exp.val.f;
2850 sprintf(temp, "0o%o", v);
2855 long v = (long)exp.val.d;
2856 sprintf(temp, "0o%o", v);
2861 wh.octVal = CopyString(temp);
2864 // WHATS THIS HERE ?
2865 if(exp.type == constantExp && exp.constant)
2866 wh.constant = CopyString(exp.constant);
2872 case symbolErrorExp:
2873 snprintf(watchmsg, sizeof(watchmsg), $"Symbol \"%s\" not found", exp.identifier.string);
2875 case memberSymbolErrorExp:
2877 Expression memberExp = exp.member.exp;
2878 Identifier memberID = exp.member.member;
2879 Type type = memberExp.expType;
2882 if(type.kind == structType || type.kind == unionType)
2884 char string[1024] = "";
2886 PrintTypeNoConst(type, string, false, true);
2887 snprintf(watchmsg, sizeof(watchmsg), $"Member \"%s\" not found in %s \"%s\"",
2888 memberID ? memberID.string : "", type.kind == unionType ? "union" : "struct", type.name ? type.name : string);
2892 Class _class = (memberID && memberID.classSym) ? memberID.classSym.registered : ((type.kind == classType && type._class) ? type._class.registered : null);
2893 char string[1024] = "";
2897 PrintTypeNoConst(type, string, false, true);
2898 classSym = FindClass(string);
2899 _class = classSym ? classSym.registered : null;
2902 snprintf(watchmsg, sizeof(watchmsg), $"Member \"%s\" not found in class \"%s\"", memberID ? memberID.string : "", _class.name);
2904 // NOTE: This should probably never happen, only classes and struct can be dereferenced, a dereferenceErrorExp should be displayed instead
2905 snprintf(watchmsg, sizeof(watchmsg), $"Member \"%s\" not found in type \"%s\"", memberID ? memberID.string : "", string);
2909 // NOTE: This should probably never happen, the error causing the unresolved expression should be displayed instead
2910 snprintf(watchmsg, sizeof(watchmsg), $"Accessing member \"%s\" from unresolved expression", memberID ? memberID.string : "");
2913 case memberPropertyErrorExp:
2915 Expression memberExp = exp.member.exp;
2916 Identifier memberID = exp.member.member;
2917 Type type = memberExp.expType;
2918 Class _class = (type && memberID) ? (memberID && memberID.classSym) ? memberID.classSym.registered : ((type.kind == classType && type._class) ? type._class.registered : null) : null;
2919 if(_class && memberID && memberID.string)
2920 snprintf(watchmsg, sizeof(watchmsg), $"Missing property evaluation for \"%s\" in class \"%s\"", memberID.string, _class.name);
2922 snprintf(watchmsg, sizeof(watchmsg), $"Missing property evaluation for \"%s\"", wh.expression);
2925 case functionCallErrorExp:
2926 if(exp.call.exp && exp.call.exp.type == identifierExp && exp.call.exp.identifier.string)
2927 snprintf(watchmsg, sizeof(watchmsg), $"Missing function evaluation for call to \"%s\"", exp.call.exp.identifier.string);
2929 snprintf(watchmsg, sizeof(watchmsg), $"Missing function evaluation for \"%s\"", wh.expression);
2931 case memoryErrorExp:
2932 // Need to ensure when set to memoryErrorExp, constant is set
2933 snprintf(watchmsg, sizeof(watchmsg), $"Memory can't be read at %s", /*(exp.type == constantExp) ? */exp.constant /*: null*/);
2935 case dereferenceErrorExp:
2936 snprintf(watchmsg, sizeof(watchmsg), $"Dereferencing error evaluating \"%s\"", wh.expression);
2938 case divideBy0ErrorExp:
2939 snprintf(watchmsg, sizeof(watchmsg), "%s", $"Integer division by 0");
2941 case noDebuggerErrorExp:
2942 snprintf(watchmsg, sizeof(watchmsg), $"Debugger required for symbol evaluation in \"%s\"", wh.expression);
2944 case unknownErrorExp:
2945 // NOTE: This should never happen
2946 snprintf(watchmsg, sizeof(watchmsg), $"Error evaluating \"%s\"", wh.expression);
2949 // NOTE: This should never happen
2950 snprintf(watchmsg, sizeof(watchmsg), $"Null type for \"%s\"", wh.expression);
2953 wh.value = CopyString(exp.string);
2956 // Temporary Code for displaying Strings
2957 if((exp.expType && ((exp.expType.kind == pointerType ||
2958 exp.expType.kind == arrayType) && exp.expType.type.kind == charType)) ||
2959 (wh.type && wh.type.kind == classType && wh.type._class &&
2960 wh.type._class.registered && wh.type._class.registered.type == normalClass &&
2961 !strcmp(wh.type._class.registered.name, "String")))
2964 if(exp.expType && (exp.expType.kind != arrayType || exp.hasAddress))
2968 //char temp[MAX_F_STRING * 32];
2970 //ExpressionType evalError = dummyExp;
2971 /*if(exp.expType.kind == arrayType)
2972 sprintf(temp, "(char*)0x%x", exp.address);
2974 sprintf(temp, "(char*)%s", exp.constant);*/
2976 //evaluation = Debugger::EvaluateExpression(temp, &evalError);
2977 // address = strtoul(exp.constant, null, 0);
2978 address = _strtoui64(exp.constant, null, 0);
2979 //_dplf(0, "0x", address);
2980 // snprintf(value, sizeof(value), "0x%08x ", address);
2982 if(address > 0xFFFFFFFFLL)
2983 snprintf(value, sizeof(value), (__runtimePlatform == win32) ? "0x%016I64x " : "0x%016llx ", address);
2985 snprintf(value, sizeof(value), (__runtimePlatform == win32) ? "0x%08I64x " : "0x%08llx ", address);
2986 value[sizeof(value)-1] = 0;
2989 strcat(value, $"Null string");
2992 String string = new char[4097];
2994 bool success = false;
2998 for(start = 0; !done && start + size <= 4096; start += size)
3003 // Try to read 256 bytes at a time, then half if that fails
3004 s = GdbReadMemory(address + start, size);
3008 memcpy(string + start, s, size);
3009 string[start + size] = 0;
3010 if(size == 1 || memchr(s, 0, size))
3024 int len = strlen(value);
3026 if(UTF8Validate(string))
3031 for(c = 0; (ch = string[c]); c++)
3034 value[len++] = '\0';
3039 ISO8859_1toUTF8(string, value + len, strlen(value) - len - 30);
3040 strcat(value, ") (ISO8859-1)");
3044 strcat(value, $"Empty string");
3047 strcat(value, $"Couldn't read memory");
3050 wh.value = CopyString(value);
3053 else if(wh.type && wh.type.kind == classType && wh.type._class &&
3054 wh.type._class.registered && wh.type._class.registered.type == enumClass)
3056 Class enumClass = eSystem_FindClass(GetPrivateModule(), wh.type._class.registered.name);
3057 EnumClassData enumeration = (EnumClassData)enumClass.data;
3061 if(!strcmp(enumClass.dataTypeString, "uint64"))
3063 uint64 v = strtoull(exp.constant, null, 0);
3064 value = *(int64*)&v;
3067 value = strtoll(exp.constant, null, 0);
3069 for(item = enumeration.values.first; item; item = item.next)
3070 if(item.data == value)
3073 wh.value = CopyString(item.name);
3075 wh.value = PrintString($"Invalid Enum Value (", value, ")");
3078 else if(wh.type && (wh.type.kind == charType || (wh.type.kind == classType && wh.type._class &&
3079 wh.type._class.registered && !strcmp(wh.type._class.registered.fullName, "unichar"))) )
3086 if(exp.constant[0] == '\'')
3088 if((int)((byte *)exp.constant)[1] > 127)
3091 value = UTF8GetChar(exp.constant + 1, &nb);
3092 if(nb < 2) value = exp.constant[1];
3093 signedValue = value;
3097 signedValue = exp.constant[1];
3099 // Precomp Syntax error with boot strap here:
3100 byte b = (byte)(char)signedValue;
3101 value = (unichar) b;
3107 if(wh.type.kind == charType && wh.type.isSigned)
3109 signedValue = (int)(char)strtol(exp.constant, null, 0);
3111 // Precomp Syntax error with boot strap here:
3112 byte b = (byte)(char)signedValue;
3113 value = (unichar) b;
3118 value = (uint)strtoul(exp.constant, null, 0);
3119 signedValue = (int)value;
3123 UTF32toUTF8Len(&value, 1, charString, 5);
3125 snprintf(string, sizeof(string), "\'\\0' (0)");
3126 else if(value == '\t')
3127 snprintf(string, sizeof(string), "\'\\t' (%d)", value);
3128 else if(value == '\n')
3129 snprintf(string, sizeof(string), "\'\\n' (%d)", value);
3130 else if(value == '\r')
3131 snprintf(string, sizeof(string), "\'\\r' (%d)", value);
3132 else if(wh.type.kind == charType && wh.type.isSigned)
3133 snprintf(string, sizeof(string), "\'%s\' (%d)", charString, signedValue);
3134 else if(value > 256 || wh.type.kind != charType)
3136 if(value > 0x10FFFF || !GetCharCategory(value))
3137 snprintf(string, sizeof(string), $"Invalid Unicode Codepoint (0x%08X)", value);
3139 snprintf(string, sizeof(string), "\'%s\' (U+%04X)", charString, value);
3142 snprintf(string, sizeof(string), "\'%s\' (%d)", charString, value);
3143 string[sizeof(string)-1] = 0;
3145 wh.value = CopyString(string);
3150 wh.value = CopyString(exp.constant);
3156 bool genericError = true;
3157 if(exp.expType && exp.hasAddress)
3159 bool showAddress = false;
3160 switch(exp.expType.kind)
3172 Symbol s = exp.expType._class;
3175 Class c = s.registered;
3176 if(c.type == noHeadClass || c.type == normalClass)
3184 wh.value = PrintHexUInt64(exp.address);
3186 genericError = false;
3190 // NOTE: This should ideally be handled with a specific error message
3191 snprintf(watchmsg, sizeof(watchmsg), $"Error evaluating \"%s\"", wh.expression);
3196 SetInDebugger(false);
3199 snprintf(watchmsg, sizeof(watchmsg), $"Invalid expression: \"%s\"", wh.expression);
3200 if(exp) FreeExpression(exp);
3203 SetPrivateModule(backupPrivateModule);
3204 SetCurrentContext(backupContext);
3205 SetTopContext(backupContext);
3206 SetGlobalContext(backupContext);
3207 SetThisClass(backupThisClass);
3210 // wh.value = CopyString("No source file found for selected frame");
3212 watchmsg[sizeof(watchmsg)-1] = 0;
3214 wh.value = CopyString(watchmsg);
3216 ide.watchesView.UpdateWatch(wh);
3220 void EvaluateWatches()
3222 _dpcl(_dpct, dplchan::debuggerWatches, 0, "Debugger::EvaluateWatches()");
3223 WatchesLinkCodeEditor();
3224 if(state == stopped)
3226 for(wh : ide.workspace.watches)
3231 char * ::GdbEvaluateExpression(char * expression)
3233 _dpcl(_dpct, dplchan::debuggerWatches, 0, "Debugger::GdbEvaluateExpression(", expression, ")");
3236 GdbCommand(0, false, "-data-evaluate-expression \"%s\"", expression);
3238 ide.outputView.debugBox.Logf("Debugger Error: GdbEvaluateExpression\n");
3242 // to be removed... use GdbReadMemory that returns a byte array instead
3243 char * ::GdbReadMemoryString(uint64 address, int size, char format, int rows, int cols)
3245 _dpcl(_dpct, dplchan::debuggerCall, 0, "Debugger::GdbReadMemoryString(", address, ")");
3250 _dplf(0, "GdbReadMemoryString called with size = 0!");
3252 GdbCommand(0, false,
3253 (__runtimePlatform == win32) ? "-data-read-memory 0x%016I64x %c, %d, %d, %d" : "-data-read-memory 0x%016llx %c, %d, %d, %d", address, format, size, rows, cols);
3255 ide.outputView.debugBox.Logf("Debugger Error: GdbReadMemoryString\n");
3259 byte * ::GdbReadMemory(uint64 address, int bytes)
3261 _dpcl(_dpct, dplchan::debuggerCall, 0, "Debugger::GdbReadMemory(", address, ")");
3264 GdbCommand(0, false,
3265 (__runtimePlatform == win32) ? "-data-read-memory 0x%016I64x %c, 1, 1, %d" : "-data-read-memory 0x%016llx %c, 1, 1, %d",
3266 address, 'u', bytes);
3269 _dplf(0, "GdbReadMemory called with bytes = 0!");
3272 ide.outputView.debugBox.Logf("Debugger Error: GdbReadMemory\n");
3273 else if(eval.result && strcmp(eval.result, "N/A"))
3275 byte * result = new byte[bytes];
3276 char * string = eval.result;
3280 result[c++] = (byte)strtol(string, &string, 10);
3297 bool BreakpointHit(GdbDataStop stopItem, Breakpoint bpInternal, Breakpoint bpUser)
3300 char * s1 = null; char * s2 = null;
3301 _dpcl(_dpct, dplchan::debuggerCall, 0, "Debugger::BreakpointHit(",
3302 "bpInternal(", bpInternal ? s1=bpInternal.CopyLocationString(false) : null, "), ",
3303 "bpUser(", bpUser ? s2=bpUser.CopyLocationString(false) : null, ")) -- ",
3304 "ignoreBreakpoints(", ignoreBreakpoints, "), ",
3305 "hitCursorBreakpoint(", bpUser && bpUser.type == runToCursor, ")");
3306 delete s1; delete s2;
3310 bool conditionMet = true;
3311 if(bpUser.condition)
3313 if(WatchesLinkCodeEditor())
3314 conditionMet = ResolveWatch(bpUser.condition);
3316 conditionMet = false;
3331 if(stopItem.frame.line && bpUser.line != stopItem.frame.line)
3333 // updating user breakpoint on hit location difference
3334 // todo, print something?
3335 bpUser.line = stopItem.frame.line;
3336 ide.breakpointsView.UpdateBreakpoint(bpUser.row);
3337 ide.workspace.Save();
3340 ide.breakpointsView.UpdateBreakpoint(bpUser.row);
3345 if(bpInternal.type == internalModulesLoaded)
3347 if(userAction == stepOver)
3349 if((bpInternal.type == internalEntry && ((intBpMain && intBpMain.inserted) || (intBpWinMain && intBpWinMain.inserted))) ||
3350 (bpInternal.type == internalMain && intBpWinMain && intBpWinMain.inserted))
3353 if(!bpUser && !userAction.breaksOnInternalBreakpoint)
3355 if(userAction == stepOut)
3356 StepOut(ignoreBreakpoints);
3362 if(!bpUser && !bpInternal)
3368 void ValgrindTargetThreadExit()
3370 _dpcl(_dpct, dplchan::debuggerCall, 0, "Debugger::ValgrindTargetThreadExit()");
3373 vgTargetHandle.Wait();
3374 delete vgTargetHandle;
3376 HandleExit(null, null);
3379 void GdbThreadExit()
3381 _dpcl(_dpct, dplchan::debuggerCall, 0, "Debugger::GdbThreadExit()");
3382 if(state != terminated)
3384 _ChangeState(terminated);
3385 targetProcessId = 0;
3386 ClearBreakDisplay();
3391 serialSemaphore.Release();
3396 ide.outputView.debugBox.Logf($"Debugger Fatal Error: GDB lost\n");
3397 ide.outputView.debugBox.Logf($"Debugging stopped\n");
3399 HideDebuggerViews();
3401 //_ChangeState(terminated);
3405 void GdbThreadMain(char * output)
3409 Array<char *> outTokens { minAllocSize = 50 };
3410 Array<char *> subTokens { minAllocSize = 50 };
3411 DebugListItem item { };
3412 DebugListItem item2 { };
3413 bool setWaitingForPID = false;
3415 #if defined(GDB_DEBUG_CONSOLE) || defined(GDB_DEBUG_GUI)
3416 #ifdef GDB_DEBUG_CONSOLE
3417 // _dpcl(_dpct, dplchan::gdbOutput, 0, output);
3420 #ifdef GDB_DEBUG_OUTPUT
3422 int len = strlen(output);
3430 for(c = 0; c < len / 1024; c++)
3432 strncpy(tmp, start, 1024);
3433 ide.outputView.gdbBox.Logf("out: %s\n", tmp);
3436 ide.outputView.gdbBox.Logf("out: %s\n", start);
3440 ide.outputView.gdbBox.Logf("out: %s\n", output);
3444 #ifdef GDB_DEBUG_CONSOLE
3445 strcpy(lastGdbOutput, output);
3447 #ifdef GDB_DEBUG_GUI
3448 if(ide.gdbDialog) ide.gdbDialog.AddOutput(output);
3455 if(strstr(output, "No debugging symbols found") || strstr(output, "(no debugging symbols found)"))
3458 ide.outputView.debugBox.Logf($"Target doesn't contain debug information!\n");
3461 if(!entryPoint && (t = strstr(output, "Entry point:")))
3463 char * addr = t + strlen("Entry point:");
3465 if(*t++ == ' ' && *t++ == '0' && *t == 'x')
3468 while(isxdigit(*++t));
3470 for(bp : sysBPs; bp.type == internalEntry)
3473 bp.enabled = entryPoint = true;
3481 if(TokenizeList(output, ',', outTokens) && !strcmp(outTokens[0], "^done"))
3483 //if(outTokens.count == 1)
3488 _ChangeState(loaded);
3489 targetProcessId = 0;
3490 if(outTokens.count > 1 && TokenizeListItem(outTokens[1], item))
3492 if(!strcmp(item.name, "reason"))
3494 char * reason = item.value;
3495 StripQuotes(reason, reason);
3496 if(!strcmp(reason, "exited-normally") || !strcmp(reason, "exited") || !strcmp(reason, "exited-signalled"))
3499 if(outTokens.count > 2 && TokenizeListItem(outTokens[2], item2))
3501 StripQuotes(item2.value, item2.value);
3502 if(!strcmp(item2.name, "exit-code"))
3503 exitCode = item2.value;
3509 HandleExit(reason, exitCode);
3513 _dpcl(_dpct, dplchan::gdbProtoUnknown, 0, "kill reply (", item.name, "=", item.value, ") is unheard of");
3516 HandleExit(null, null);
3519 if(outTokens.count > 1 && TokenizeListItem(outTokens[1], item))
3521 if(!strcmp(item.name, "bkpt"))
3523 sentBreakInsert = false;
3526 _dplf(0, "problem");
3529 bpItem = ParseBreakpoint(item.value, outTokens);
3530 //breakType = bpValidation;
3532 else if(!strcmp(item.name, "depth"))
3534 StripQuotes(item.value, item.value);
3535 frameCount = atoi(item.value);
3537 stackFrames.Free(Frame::Free);
3539 else if(!strcmp(item.name, "stack"))
3542 if(stackFrames.count)
3543 ide.callStackView.Logf("...\n");
3546 item.value = StripBrackets(item.value);
3547 TokenizeList(item.value, ',', subTokens);
3548 for(i = 0; i < subTokens.count; i++)
3550 if(TokenizeListItem(subTokens[i], item))
3552 if(!strcmp(item.name, "frame"))
3555 stackFrames.Add(frame);
3556 item.value = StripCurlies(item.value);
3557 ParseFrame(frame, item.value);
3558 if(frame.file && frame.from)
3559 _dpcl(_dpct, dplchan::gdbProtoUnknown, 0, "unexpected frame file and from members present");
3563 if(activeFrameLevel == -1)
3565 if(ide.projectView.IsModuleInProject(frame.file));
3567 if(frame.level != 0)
3569 //stopItem.frame = frame;
3570 breakType = selectFrame;
3573 activeFrame = frame;
3574 activeFrameLevel = frame.level;
3577 ide.callStackView.Logf("%3d ", frame.level);
3578 if(!strncmp(frame.func, "__ecereMethod_", strlen("__ecereMethod_")))
3579 ide.callStackView.Logf($"%s Method, %s:%d\n", &frame.func[strlen("__ecereMethod_")], (s = CopySystemPath(frame.file)), frame.line);
3580 else if(!strncmp(frame.func, "__ecereProp_", strlen("__ecereProp_")))
3581 ide.callStackView.Logf($"%s Property, %s:%d\n", &frame.func[strlen("__ecereProp_")], (s = CopySystemPath(frame.file)), frame.line);
3582 else if(!strncmp(frame.func, "__ecereConstructor_", strlen("__ecereConstructor_")))
3583 ide.callStackView.Logf($"%s Constructor, %s:%d\n", &frame.func[strlen("__ecereConstructor_")], (s = CopySystemPath(frame.file)), frame.line);
3584 else if(!strncmp(frame.func, "__ecereDestructor_", strlen("__ecereDestructor_")))
3585 ide.callStackView.Logf($"%s Destructor, %s:%d\n", &frame.func[strlen("__ecereDestructor_")], (s = CopySystemPath(frame.file)), frame.line);
3587 ide.callStackView.Logf($"%s Function, %s:%d\n", frame.func, (s = CopySystemPath(frame.file)), frame.line);
3592 ide.callStackView.Logf("%3d ", frame.level);
3597 ide.callStackView.Logf($"inside %s, %s\n", frame.func, (s = CopySystemPath(frame.from)));
3601 ide.callStackView.Logf("%s\n", frame.func);
3603 ide.callStackView.Logf($"unknown source\n");
3607 _dpcl(_dpct, dplchan::gdbProtoUnknown, 0, "stack content (", item.name, "=", item.value, ") is unheard of");
3610 if(activeFrameLevel == -1)
3612 activeFrameLevel = 0;
3613 activeFrame = stackFrames.first;
3615 ide.callStackView.Home();
3617 subTokens.RemoveAll();
3619 /*else if(!strcmp(item.name, "frame"))
3622 item.value = StripCurlies(item.value);
3623 ParseFrame(&frame, item.value);
3625 else if(!strcmp(item.name, "thread-ids"))
3627 ide.threadsView.Clear();
3628 item.value = StripCurlies(item.value);
3629 TokenizeList(item.value, ',', subTokens);
3630 for(i = subTokens.count - 1; ; i--)
3632 if(TokenizeListItem(subTokens[i], item))
3634 if(!strcmp(item.name, "thread-id"))
3637 StripQuotes(item.value, item.value);
3638 value = atoi(item.value);
3639 ide.threadsView.Logf("%3d \n", value);
3642 _dpcl(_dpct, dplchan::gdbProtoUnknown, 0, "threads content (", item.name, "=", item.value, ") is unheard of");
3647 ide.threadsView.Home();
3649 subTokens.RemoveAll();
3650 //if(!strcmp(outTokens[2], "number-of-threads"))
3652 else if(!strcmp(item.name, "new-thread-id"))
3654 StripQuotes(item.value, item.value);
3655 activeThread = atoi(item.value);
3657 else if(!strcmp(item.name, "value"))
3659 StripQuotes(item.value, item.value);
3660 eval.result = CopyString(item.value);
3661 eval.active = false;
3663 else if(!strcmp(item.name, "addr"))
3665 for(i = 2; i < outTokens.count; i++)
3667 if(TokenizeListItem(outTokens[i], item))
3669 if(!strcmp(item.name, "total-bytes"))
3671 StripQuotes(item.value, item.value);
3672 eval.bytes = atoi(item.value);
3674 else if(!strcmp(item.name, "next-row"))
3676 StripQuotes(item.value, item.value);
3677 eval.nextBlockAddress = _strtoui64(item.value, null, 0);
3679 else if(!strcmp(item.name, "memory"))
3683 //StripQuotes(item.value, item.value);
3684 item.value = StripBrackets(item.value);
3685 // this should be treated as a list...
3686 item.value = StripCurlies(item.value);
3687 TokenizeList(item.value, ',', subTokens);
3688 for(j = 0; j < subTokens.count; j++)
3690 if(TokenizeListItem(subTokens[j], item))
3692 if(!strcmp(item.name, "data"))
3694 item.value = StripBrackets(item.value);
3695 StripQuotes2(item.value, item.value);
3696 eval.result = CopyString(item.value);
3697 eval.active = false;
3701 subTokens.RemoveAll();
3706 else if(!strcmp(item.name, "source-path") || !strcmp(item.name, "BreakpointTable"))
3707 _dpcl(_dpct, dplchan::gdbProtoIgnored, 0, "command reply (", item.name, "=", item.value, ") is ignored");
3709 _dpcl(_dpct, dplchan::gdbProtoUnknown, 0, "command reply (", item.name, "=", item.value, ") is unheard of");
3712 else if(!strcmp(outTokens[0], "^running"))
3714 waitingForPID = true;
3715 setWaitingForPID = true;
3716 ClearBreakDisplay();
3718 else if(!strcmp(outTokens[0], "^exit"))
3720 _ChangeState(terminated);
3721 // ide.outputView.debugBox.Logf("Exit\n");
3722 // ide.Update(null);
3724 serialSemaphore.Release();
3726 else if(!strcmp(outTokens[0], "^error"))
3730 sentBreakInsert = false;
3731 breakpointError = true;
3734 if(outTokens.count > 1 && TokenizeListItem(outTokens[1], item))
3736 if(!strcmp(item.name, "msg"))
3738 StripQuotes(item.value, item.value);
3741 eval.active = false;
3743 if(strstr(item.value, "No symbol") && strstr(item.value, "in current context"))
3744 eval.error = symbolNotFound;
3745 else if(strstr(item.value, "Cannot access memory at address"))
3746 eval.error = memoryCantBeRead;
3748 eval.error = unknown;
3750 else if(!strcmp(item.value, "Previous frame inner to this frame (corrupt stack?)"))
3753 else if(!strncmp(item.value, "Cannot access memory at address", 31))
3756 else if(!strcmp(item.value, "Cannot find bounds of current function"))
3758 _ChangeState(stopped);
3759 gdbHandle.Printf("-exec-continue\n");
3761 else if(!strcmp(item.value, "ptrace: No such process."))
3763 _ChangeState(loaded);
3764 targetProcessId = 0;
3766 else if(!strcmp(item.value, "Function \\\"WinMain\\\" not defined."))
3769 else if(!strcmp(item.value, "You can't do that without a process to debug."))
3771 _ChangeState(loaded);
3772 targetProcessId = 0;
3774 else if(strstr(item.value, "No such file or directory."))
3776 _ChangeState(loaded);
3777 targetProcessId = 0;
3779 else if(strstr(item.value, "During startup program exited with code "))
3781 _ChangeState(loaded);
3782 targetProcessId = 0;
3787 if(strlen(item.value) < MAX_F_STRING)
3790 ide.outputView.debugBox.Logf("GDB: %s\n", (s = CopyUnescapedString(item.value)));
3794 ide.outputView.debugBox.Logf("GDB: %s\n", item.value);
3800 _dpcl(_dpct, dplchan::gdbProtoUnknown, 0, "error content (", item.name, "=", item.value, ") is unheard of");
3803 _dpcl(_dpct, dplchan::gdbProtoUnknown, 0, "result-record: ", outTokens[0]);
3804 outTokens.RemoveAll();
3807 _dpcl(_dpct, dplchan::gdbProtoUnknown, 0, "status-async-output: ", outTokens[0]);
3810 if(TokenizeList(output, ',', outTokens))
3812 if(!strcmp(outTokens[0], "=library-loaded"))
3813 FGODetectLoadedLibraryForAddedProjectIssues(outTokens, false);
3814 else if(!strcmp(outTokens[0], "=shlibs-added"))
3815 FGODetectLoadedLibraryForAddedProjectIssues(outTokens, true);
3816 else if(!strcmp(outTokens[0], "=thread-group-created") || !strcmp(outTokens[0], "=thread-group-added") ||
3817 !strcmp(outTokens[0], "=thread-group-started") || !strcmp(outTokens[0], "=thread-group-exited") ||
3818 !strcmp(outTokens[0], "=thread-created") || !strcmp(outTokens[0], "=thread-exited") ||
3819 !strcmp(outTokens[0], "=cmd-param-changed") || !strcmp(outTokens[0], "=library-unloaded") ||
3820 !strcmp(outTokens[0], "=breakpoint-modified"))
3821 _dpcl(_dpct, dplchan::gdbProtoIgnored, 0, outTokens[0], outTokens.count>1 ? outTokens[1] : "",
3822 outTokens.count>2 ? outTokens[2] : "", outTokens.count>3 ? outTokens[3] : "",
3823 outTokens.count>4 ? outTokens[4] : "", outTokens.count>5 ? outTokens[5] : "",
3824 outTokens.count>6 ? outTokens[6] : "", outTokens.count>7 ? outTokens[7] : "",
3825 outTokens.count>8 ? outTokens[8] : "", outTokens.count>9 ? outTokens[9] : "");
3827 _dpcl(_dpct, dplchan::gdbProtoUnknown, 0, "notify-async-output: ", outTokens[0]);
3829 outTokens.RemoveAll();
3833 if(TokenizeList(output, ',', outTokens))
3835 if(!strcmp(outTokens[0],"*running"))
3837 waitingForPID = true;
3838 setWaitingForPID = true;
3840 else if(!strcmp(outTokens[0], "*stopped"))
3843 _ChangeState(stopped);
3845 for(tk = 1; tk < outTokens.count; tk++)
3847 if(TokenizeListItem(outTokens[tk], item))
3849 if(!strcmp(item.name, "reason"))
3851 char * reason = item.value;
3852 StripQuotes(reason, reason);
3853 if(!strcmp(reason, "exited-normally") || !strcmp(reason, "exited") || !strcmp(reason, "exited-signalled"))
3856 if(outTokens.count > tk+1 && TokenizeListItem(outTokens[tk+1], item2))
3859 StripQuotes(item2.value, item2.value);
3860 if(!strcmp(item2.name, "exit-code"))
3861 exitCode = item2.value;
3867 HandleExit(reason, exitCode);
3870 else if(!strcmp(reason, "breakpoint-hit") ||
3871 !strcmp(reason, "function-finished") ||
3872 !strcmp(reason, "end-stepping-range") ||
3873 !strcmp(reason, "location-reached") ||
3874 !strcmp(reason, "signal-received"))
3878 if(stopItem) _dplf(0, "problem");
3880 stopItem = GdbDataStop { };
3881 stopItem.reason = r == 'b' ? breakpointHit : r == 'f' ? functionFinished : r == 'e' ? endSteppingRange : r == 'l' ? locationReached : signalReceived;
3883 for(i = tk+1; i < outTokens.count; i++)
3885 TokenizeListItem(outTokens[i], item);
3886 StripQuotes(item.value, item.value);
3887 if(!strcmp(item.name, "thread-id"))
3888 stopItem.threadid = atoi(item.value);
3889 else if(!strcmp(item.name, "frame"))
3891 item.value = StripCurlies(item.value);
3892 ParseFrame(stopItem.frame, item.value);
3894 else if(stopItem.reason == breakpointHit && !strcmp(item.name, "bkptno"))
3895 stopItem.bkptno = atoi(item.value);
3896 else if(stopItem.reason == functionFinished && !strcmp(item.name, "gdb-result-var"))
3897 stopItem.gdbResultVar = CopyString(item.value);
3898 else if(stopItem.reason == functionFinished && !strcmp(item.name, "return-value"))
3899 stopItem.returnValue = CopyString(item.value);
3900 else if(stopItem.reason == signalReceived && !strcmp(item.name, "signal-name"))
3901 stopItem.name = CopyString(item.value);
3902 else if(stopItem.reason == signalReceived && !strcmp(item.name, "signal-meaning"))
3903 stopItem.meaning = CopyString(item.value);
3904 else if(!strcmp(item.name, "stopped-threads"))
3905 _dpcl(_dpct, dplchan::gdbProtoIgnored, 0, reason, ": Advanced thread debugging not handled");
3906 else if(!strcmp(item.name, "core"))
3907 _dpcl(_dpct, dplchan::gdbProtoIgnored, 0, reason, ": Information (core) not used");
3908 else if(!strcmp(item.name, "disp"))
3909 _dpcl(_dpct, dplchan::gdbProtoIgnored, 0, reason, ": (", item.name, "=", item.value, ")");
3911 _dpcl(_dpct, dplchan::gdbProtoUnknown, 0, "Unknown ", reason, " item name (", item.name, "=", item.value, ")");
3914 if(stopItem.reason == signalReceived && !strcmp(stopItem.name, "SIGTRAP"))
3930 event = r == 'b' ? hit : r == 'f' ? functionEnd : r == 'e' ? stepEnd : r == 'l' ? locationReached : signal;
3934 else if(!strcmp(reason, "watchpoint-trigger"))
3935 _dpcl(_dpct, dplchan::gdbProtoIgnored, 0, "Reason watchpoint trigger not handled");
3936 else if(!strcmp(reason, "read-watchpoint-trigger"))
3937 _dpcl(_dpct, dplchan::gdbProtoIgnored, 0, "Reason read watchpoint trigger not handled");
3938 else if(!strcmp(reason, "access-watchpoint-trigger"))
3939 _dpcl(_dpct, dplchan::gdbProtoIgnored, 0, "Reason access watchpoint trigger not handled");
3940 else if(!strcmp(reason, "watchpoint-scope"))
3941 _dpcl(_dpct, dplchan::gdbProtoIgnored, 0, "Reason watchpoint scope not handled");
3943 _dpcl(_dpct, dplchan::gdbProtoUnknown, 0, "Unknown reason: ", reason);
3951 if(usingValgrind && event == none && !stopItem)
3952 event = valgrindStartPause;
3957 _dpcl(_dpct, dplchan::gdbProtoUnknown, 0, "Unknown exec-async-output: ", outTokens[0]);
3958 outTokens.RemoveAll();
3961 if(!strcmpi(output, "(gdb) "))
3965 Time startTime = GetTime();
3966 char exeFile[MAX_LOCATION];
3967 int oldProcessID = targetProcessId;
3968 GetLastDirectory(targetFile, exeFile);
3970 while(!targetProcessId/*true*/)
3972 targetProcessId = Process_GetChildExeProcessId(gdbProcessId, exeFile);
3973 if(targetProcessId) break;
3974 // Can't break on Peek(), GDB is giving us =library and other info before the process is listed in /proc
3975 // if(gdbHandle.Peek()) break;
3977 if(gdbHandle.Peek() && GetTime() - startTime > 2.5) // Give the process 2.5 seconds to show up in /proc
3982 _ChangeState(running);
3983 else if(!oldProcessID)
3985 ide.outputView.debugBox.Logf($"Debugger Error: No target process ID\n");
3986 // TO VERIFY: The rest of this block has not been thoroughly tested in this particular location
3987 gdbHandle.Printf("-gdb-exit\n");
3989 _ChangeState(terminated); //loaded;
3994 for(bp : ide.workspace.breakpoints)
3995 bp.inserted = false;
3998 bp.inserted = false;
4000 bpRunToCursor.inserted = false;
4002 ide.outputView.debugBox.Logf($"Debugging stopped\n");
4003 ClearBreakDisplay();
4005 #if defined(__unix__)
4006 if(!usingValgrind && FileExists(progFifoPath)) //fileCreated)
4008 progThread.terminate = true;
4011 fifoFile.CloseInput();
4018 DeleteFile(progFifoPath);
4019 progFifoPath[0] = '\0';
4026 serialSemaphore.Release();
4029 _dpcl(_dpct, dplchan::gdbProtoUnknown, 0, $"Unknown prompt", output);
4033 if(!strncmp(output, "&\"warning:", 10))
4036 content = strstr(output, "\"");
4037 StripQuotes(content, content);
4038 content = strstr(content, ":");
4044 ide.outputView.debugBox.LogRaw((s = CopyUnescapedString(content)));
4051 _dpcl(_dpct, dplchan::gdbProtoUnknown, 0, $"Unknown output: ", output);
4053 if(!setWaitingForPID)
4054 waitingForPID = false;
4055 setWaitingForPID = false;
4063 // From GDB Output functions
4064 void FGODetectLoadedLibraryForAddedProjectIssues(Array<char *> outTokens, bool shlibs)
4066 char path[MAX_LOCATION] = "";
4067 char file[MAX_FILENAME] = "";
4068 //bool symbolsLoaded = false;
4069 DebugListItem item { };
4070 //_dpcl(_dpct, dplchan::debuggerCall, 0, "Debugger::FGODetectLoadedLibraryForAddedProjectIssues()");
4071 for(token : outTokens)
4073 if(TokenizeListItem(token, item))
4075 if(!strcmp(item.name, "target-name"))
4077 StripQuotes(item.value, path);
4078 MakeSystemPath(path);
4079 GetLastDirectory(path, file);
4081 else if(!strcmp(item.name, "symbols-loaded"))
4083 //symbolsLoaded = (atoi(item.value) == 1);
4085 else if(!strcmp(item.name, "shlib-info"))
4087 DebugListItem subItem { };
4088 Array<char *> tokens { minAllocSize = 50 };
4089 item.value = StripBrackets(item.value);
4090 TokenizeList(item.value, ',', tokens);
4093 if(TokenizeListItem(t, subItem))
4095 if(!strcmp(subItem.name, "path"))
4097 StripQuotes(subItem.value, path);
4098 MakeSystemPath(path);
4099 GetLastDirectory(path, file);
4100 //symbolsLoaded = true;
4111 if(path[0] && file[0])
4113 for(prj : ide.workspace.projects; prj != ide.workspace.projects.firstIterator.data)
4117 char prjTargetPath[MAX_LOCATION];
4118 char prjTargetFile[MAX_FILENAME];
4119 DirExpression targetDirExp = prj.GetTargetDir(currentCompiler, prj.config, bitDepth);
4120 strcpy(prjTargetPath, prj.topNode.path);
4121 PathCat(prjTargetPath, targetDirExp.dir);
4122 delete targetDirExp;
4123 prjTargetFile[0] = '\0';
4124 prj.CatTargetFileName(prjTargetFile, currentCompiler, prj.config);
4125 PathCat(prjTargetPath, prjTargetFile);
4126 MakeSystemPath(prjTargetPath);
4128 match = !fstrcmp(prjTargetFile, file);
4129 if(!match && (dot = strstr(prjTargetFile, ".so.")))
4131 char * dot3 = strstr(dot+4, ".");
4135 match = !fstrcmp(prjTargetFile, file);
4140 match = !fstrcmp(prjTargetFile, file);
4145 // TODO: nice visual feedback to better warn user. use some ide notification system or other means.
4146 /* -- 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)
4148 ide.outputView.debugBox.Logf($"Attention! No symbols for loaded library %s matched to the %s added project.\n", path, prj.topNode.name);
4150 match = !fstrcmp(prjTargetPath, path);
4151 if(!match && (dot = strstr(prjTargetPath, ".so.")))
4153 char * dot3 = strstr(dot+4, ".");
4157 match = !fstrcmp(prjTargetPath, path);
4162 match = !fstrcmp(prjTargetPath, path);
4166 projectsLibraryLoaded[prj.name] = true;
4168 ide.outputView.debugBox.Logf($"Loaded library %s doesn't match the %s target of the %s added project.\n", path, prjTargetPath, prj.topNode.name);
4175 void FGOBreakpointModified(Array<char *> outTokens)
4177 _dpcl(_dpct, dplchan::debuggerCall, 0, "Debugger::FGOBreakpointModified() -- TODO only if needed: support breakpoint modified");
4179 DebugListItem item { };
4180 if(outTokens.count > 1 && TokenizeListItem(outTokens[1], item))
4182 if(!strcmp(item.name, "bkpt"))
4184 GdbDataBreakpoint modBp = ParseBreakpoint(item.value, outTokens);
4192 ExpressionType ::DebugEvalExpTypeError(char * result)
4194 _dpcl(_dpct, dplchan::debuggerWatches, 0, "Debugger::DebugEvalExpTypeError()");
4199 case symbolNotFound:
4200 return symbolErrorExp;
4201 case memoryCantBeRead:
4202 return memoryErrorExp;
4204 return unknownErrorExp;
4207 char * ::EvaluateExpression(char * expression, ExpressionType * error)
4210 _dpcl(_dpct, dplchan::debuggerWatches, 0, "Debugger::EvaluateExpression(", expression, ")");
4211 if(ide.projectView && ide.debugger.state == stopped)
4213 result = GdbEvaluateExpression(expression);
4214 *error = DebugEvalExpTypeError(result);
4219 *error = noDebuggerErrorExp;
4224 char * ::ReadMemory(uint64 address, int size, char format, ExpressionType * error)
4227 char * result = GdbReadMemoryString(address, size, format, 1, 1);
4228 _dpcl(_dpct, dplchan::debuggerCall, 0, "Debugger::ReadMemory(", address, ")");
4229 if(!result || !strcmp(result, "N/A"))
4230 *error = memoryErrorExp;
4232 *error = DebugEvalExpTypeError(result);
4237 class ValgrindLogThread : Thread
4243 static char output[4096];
4244 bool lastLineEmpty = true;
4245 Array<char> dynamicBuffer { minAllocSize = 4096 };
4246 File oldValgrindHandle = vgLogFile;
4247 incref oldValgrindHandle;
4250 while(debugger.state != terminated && vgLogFile && vgLogFile.input)
4255 result = vgLogFile.Read(output, 1, sizeof(output));
4257 if(debugger.state == terminated || !vgLogFile) // || vgLogFile.Eof()
4264 for(c = 0; c<result; c++)
4266 if(output[c] == '\n')
4268 int pos = dynamicBuffer.size;
4269 dynamicBuffer.size += c - start;
4270 memcpy(&dynamicBuffer[pos], output + start, c - start);
4271 if(dynamicBuffer.count && dynamicBuffer[dynamicBuffer.count - 1] != '\r')
4272 dynamicBuffer.size++;
4273 dynamicBuffer[dynamicBuffer.count - 1] = '\0';
4275 // printf("%s\n", dynamicBuffer.array);
4277 if(strstr(&dynamicBuffer[0], "vgdb me ..."))
4278 debugger.serialSemaphore.Release();
4280 char * s = strstr(&dynamicBuffer[0], "==");
4282 s = strstr(s+2, "== ");
4286 if(s[0] == '\0' && !lastLineEmpty)
4289 lastLineEmpty = true;
4290 dynamicBuffer[0] = '\0';
4299 if(strstr(s, "vgdb me ..."))
4301 if(strstr(s, "(action on error) vgdb me ..."))
4302 ide.outputView.debugBox.Logf($"...breaked on Valgrind error (F5 to resume)\n");
4309 if(strstr(s, "TO DEBUG THIS PROCESS USING GDB: start GDB like this"))
4315 if(strstr(s, "and then give GDB the following command"))
4321 if(strstr(s, "/path/to/gdb") || strstr(s, "target remote | /usr/lib/valgrind/../../bin/vgdb --pid="))
4327 if(strstr(s, "--pid is optional if only one valgrind process is running"))
4333 if((s = strstr(s, "; rerun with -h for copyright info")))
4345 if(lastLineEmpty && t[0] != '\0')
4346 lastLineEmpty = false;
4349 ide.outputView.debugBox.Logf("%s\n", &dynamicBuffer[0]);
4351 dynamicBuffer.size = 0;
4357 int pos = dynamicBuffer.size;
4358 dynamicBuffer.size += c - start;
4359 memcpy(&dynamicBuffer[pos], output + start, c - start);
4362 else if(debugger.state == stopped)
4365 printf("Got end of file from GDB!\n");
4372 delete dynamicBuffer;
4373 _dpcl(_dpct, dplchan::debuggerCall, 0, "ValgrindLogThreadExit");
4374 //if(oldValgrindHandle == vgLogFile)
4375 debugger.GdbThreadExit/*ValgrindLogThreadExit*/();
4376 delete oldValgrindHandle;
4382 class ValgrindTargetThread : Thread
4388 static char output[4096];
4389 Array<char> dynamicBuffer { minAllocSize = 4096 };
4390 DualPipe oldValgrindHandle = vgTargetHandle;
4391 incref oldValgrindHandle;
4394 while(debugger.state != terminated && vgTargetHandle && !vgTargetHandle.Eof())
4398 result = vgTargetHandle.Read(output, 1, sizeof(output));
4400 if(debugger.state == terminated || !vgTargetHandle || vgTargetHandle.Eof())
4407 for(c = 0; c<result; c++)
4409 if(output[c] == '\n')
4411 int pos = dynamicBuffer.size;
4412 dynamicBuffer.size += c - start;
4413 memcpy(&dynamicBuffer[pos], output + start, c - start);
4414 if(dynamicBuffer.count && dynamicBuffer[dynamicBuffer.count - 1] != '\r')
4415 // COMMENTED OUT DUE TO ISSUE #135, FIXED
4416 //if(dynamicBuffer.array[dynamicBuffer.count - 1] != '\r')
4417 dynamicBuffer.size++;
4418 dynamicBuffer[dynamicBuffer.count - 1] = '\0';
4420 // printf("%s\n", dynamicBuffer.array);
4422 ide.outputView.debugBox.Logf("%s\n", &dynamicBuffer[0]);
4424 dynamicBuffer.size = 0;
4430 int pos = dynamicBuffer.size;
4431 dynamicBuffer.size += c - start;
4432 memcpy(&dynamicBuffer[pos], output + start, c - start);
4438 printf("Got end of file from GDB!\n");
4442 delete dynamicBuffer;
4443 //if(oldValgrindHandle == vgTargetHandle)
4444 debugger.ValgrindTargetThreadExit();
4445 delete oldValgrindHandle;
4451 class GdbThread : Thread
4457 static char output[4096];
4458 Array<char> dynamicBuffer { minAllocSize = 4096 };
4459 DualPipe oldGdbHandle = gdbHandle;
4460 incref oldGdbHandle;
4463 while(debugger.state != terminated && gdbHandle && !gdbHandle.Eof())
4467 result = gdbHandle.Read(output, 1, sizeof(output));
4469 if(debugger.state == terminated || !gdbHandle || gdbHandle.Eof())
4476 for(c = 0; c<result; c++)
4478 if(output[c] == '\n')
4480 int pos = dynamicBuffer.size;
4481 dynamicBuffer.size += c - start;
4482 memcpy(&dynamicBuffer[pos], output + start, c - start);
4483 if(dynamicBuffer.count && dynamicBuffer[dynamicBuffer.count - 1] != '\r')
4484 // COMMENTED OUT DUE TO ISSUE #135, FIXED
4485 //if(dynamicBuffer.array[dynamicBuffer.count - 1] != '\r')
4486 dynamicBuffer.size++;
4487 dynamicBuffer[dynamicBuffer.count - 1] = '\0';
4489 // _dplf(0, dynamicBuffer.array);
4491 debugger.GdbThreadMain(&dynamicBuffer[0]);
4492 dynamicBuffer.size = 0;
4498 int pos = dynamicBuffer.size;
4499 dynamicBuffer.size += c - start;
4500 memcpy(&dynamicBuffer[pos], output + start, c - start);
4506 _dplf(0, "Got end of file from GDB!");
4510 delete dynamicBuffer;
4511 //if(oldGdbHandle == gdbHandle)
4512 debugger.GdbThreadExit();
4513 delete oldGdbHandle;
4519 static define createFIFOMsg = $"err: Unable to create FIFO %s\n";
4520 static define openFIFOMsg = $"err: Unable to open FIFO %s for read\n";
4522 #if defined(__unix__)
4527 #include <sys/types.h>
4532 class ProgramThread : Thread
4537 //bool result = true;
4538 //bool fileCreated = false;
4539 //mode_t mask = 0600;
4540 static char output[1000];
4543 /*if(!mkfifo(progFifoPath, mask))
4550 ide.outputView.debugBox.Logf($"err: Unable to create FIFO %s\n", progFifoPath);
4554 if(FileExists(progFifoPath)) //fileCreated)
4556 fifoFile = FileOpen(progFifoPath, read);
4560 ide.outputView.debugBox.Logf(openFIFOMsg, progFifoPath);
4565 fd = fileno((FILE *)fifoFile.input);
4566 //fcntl(fd, F_SETFL, fcntl(fd, F_GETFL) | O_NONBLOCK);
4570 while(!terminate && fifoFile && !fifoFile.Eof())
4573 struct timeval time;
4581 /*selectResult = */select(fd + 1, &rs, null, null, &time);
4582 if(FD_ISSET(fd, &rs))
4584 int result = (int)read(fd, output, sizeof(output)-1);
4585 if(!result || (result < 0 && errno != EAGAIN))
4589 output[result] = '\0';
4590 if(strcmp(output,"&\"warning: GDB: Failed to set controlling terminal: Invalid argument\\n\"\n"))
4593 ide.outputView.debugBox.Log(output);
4602 //fifoFile.CloseInput();
4605 ide.outputView.debugBox.Log("\n");
4609 if(FileExists(progFifoPath)) //fileCreated)
4611 DeleteFile(progFifoPath);
4612 progFifoPath[0] = '\0';
4620 class Argument : struct
4622 Argument prev, next;
4624 property char * name { set { delete name; if(value) name = CopyString(value); } }
4626 property char * val { set { delete val; if(value) val = CopyString(value); } }
4640 class Frame : struct
4645 property char * addr { set { delete addr; if(value) addr = CopyString(value); } }
4647 property char * func { set { delete func; if(value) func = CopyString(value); } }
4651 property char * from { set { delete from; if(value) from = CopyUnescapedUnixPath(value); } }
4653 property char * file { set { delete file; if(value) file = CopyUnescapedUnixPath(value); } }
4654 char * absoluteFile;
4655 property char * absoluteFile { set { delete absoluteFile; if(value) absoluteFile = CopyUnescapedUnixPath(value); } }
4664 delete absoluteFile;
4665 args.Free(Argument::Free);
4674 class GdbDataStop : struct
4676 DebuggerReason reason;
4691 char * gdbResultVar;
4701 if(reason == signalReceived)
4706 else if(reason == functionFinished)
4708 delete gdbResultVar;
4712 if(frame) frame.Free();
4721 class GdbDataBreakpoint : struct
4725 property char * number { set { delete number; if(value) number = CopyString(value); } }
4727 property char * type { set { delete type; if(value) type = CopyString(value); } }
4729 property char * disp { set { delete disp; if(value) disp = CopyString(value); } }
4732 property char * addr { set { delete addr; if(value) addr = CopyString(value); } }
4734 property char * func { set { delete func; if(value) func = CopyString(value); } }
4736 property char * file { set { delete file; if(value) file = CopyUnescapedUnixPath(value); } }
4738 property char * fullname { set { delete fullname; if(value) fullname = CopyUnescapedUnixPath(value); } }
4741 property char * at { set { delete at; if(value) at = CopyString(value); } }
4744 Array<GdbDataBreakpoint> multipleBPs;
4749 PrintLn("{", "#", number, " T", type, " D", disp, " E", enabled, " H", times, " (", func, ") (", file, ":", line, ") (", fullname, ") (", addr, ") (", at, ")", "}");
4760 if(multipleBPs) multipleBPs.Free();
4766 ~GdbDataBreakpoint()
4772 class Breakpoint : struct
4777 property const char * function { set { delete function; if(value) function = CopyString(value); } }
4778 property const char * relativeFilePath { set { delete relativeFilePath; if(value) relativeFilePath = CopyString(value); } }
4779 property const char * absoluteFilePath { set { delete absoluteFilePath; if(value) absoluteFilePath = CopyString(value); } }
4781 property const char * location { set { delete location; if(value) location = CopyString(value); } }
4790 char * relativeFilePath;
4791 char * absoluteFilePath;
4795 BreakpointType type;
4797 GdbDataBreakpoint bp;
4800 property const char * address { set { delete address; if(value) address = CopyString(value); } }
4802 void ParseLocation()
4804 char * prjName = null;
4805 char * filePath = null;
4808 char fullPath[MAX_LOCATION];
4809 if(location[0] == '(' && location[1] && (file = strchr(location+2, ')')) && file[1])
4811 prjName = new char[file-location];
4812 strncpy(prjName, location+1, file-location-1);
4813 prjName[file-location-1] = '\0';
4818 if((line = strchr(file+1, ':')))
4820 filePath = new char[strlen(file)+1];
4821 strncpy(filePath, file, line-file);
4822 filePath[line-file] = '\0';
4826 filePath = CopyString(file);
4827 property::relativeFilePath = filePath;
4830 for(prj : ide.workspace.projects)
4832 if(!strcmp(prjName, prj.name))
4834 if(prj.GetAbsoluteFromRelativePath(filePath, fullPath, null))
4836 property::absoluteFilePath = fullPath;
4843 this.line = atoi(line);
4847 Project prj = ide.project;
4848 if(prj.GetAbsoluteFromRelativePath(filePath, fullPath, null))
4850 property::absoluteFilePath = fullPath;
4854 if(!absoluteFilePath)
4855 property::absoluteFilePath = "";
4860 char * CopyLocationString(bool removePath)
4863 char * file = relativeFilePath ? relativeFilePath : absoluteFilePath;
4864 bool removingPath = removePath && file;
4867 char * fileName = new char[MAX_FILENAME];
4868 GetLastDirectory(file, fileName);
4874 location = PrintString(file, ":", function);
4876 location = CopyString(function);
4879 location = PrintString(file, ":", line);
4885 char * CopyUserLocationString()
4888 char * loc = CopyLocationString(false);
4890 if(absoluteFilePath)
4892 for(p : ide.workspace.projects; p != ide.workspace.projects.firstIterator.data)
4894 if(p.topNode.FindByFullPath(absoluteFilePath, false))
4903 location = PrintString("(", prj.name, ")", loc);
4913 if(relativeFilePath && relativeFilePath[0])
4915 char * location = CopyUserLocationString();
4916 f.Printf(" * %d,%d,%d,%d,%s\n", enabled ? 1 : 0, ignore, level, line, location);
4919 f.Printf(" ~ %s\n", condition.expression);
4927 delete relativeFilePath;
4928 delete absoluteFilePath;
4948 class Watch : struct
4962 f.Printf(" ~ %s\n", expression);
4986 class DebugListItem : struct
4992 struct DebugEvaluationData
4997 uint64 nextBlockAddress;
4999 DebuggerEvaluationError error;
5002 class CodeLocation : struct
5005 char * absoluteFile;
5008 CodeLocation ::ParseCodeLocation(char * location)
5012 char * colon = null;
5014 char loc[MAX_LOCATION];
5015 strcpy(loc, location);
5016 for(temp = loc; (temp = strstr(temp, ":")); temp++)
5024 int line = atoi(colon);
5027 CodeLocation codloc { line = line };
5028 codloc.file = CopyString(loc);
5029 codloc.absoluteFile = ide.workspace.CopyAbsolutePathFromRelative(loc);
5041 delete absoluteFile;
5050 void GDBFallBack(Expression exp, String expString)
5053 ExpressionType evalError = dummyExp;
5054 result = Debugger::EvaluateExpression(expString, &evalError);
5057 FreeExpContents(exp);
5058 exp.constant = result;
5059 exp.type = constantExp;