36 , symbolManager(manager.getReactor().getSymbolManager())
49 for (
const auto& file : symbolManager.
getFiles()) {
50 buf.appendf(
"symbolfile=%s\n", file.filename.c_str());
53 buf.appendf(
"slotsubslot=%d\n", *file.slot);
56 for (
const auto& [file, error, type, slot] : fileError) {
57 buf.appendf(
"symbolfile=%s\n", file.c_str());
60 buf.appendf(
"slotsubslot=%d\n", *slot);
75 }
else if (name ==
"symbolfile") {
76 fileError.emplace_back(std::string{value},
80 }
else if (name ==
"symbolfiletype") {
81 if (!fileError.empty()) {
82 fileError.back().type =
85 }
else if (name ==
"slotsubslot") {
86 if (!fileError.empty()) {
87 if (
auto slot = StringOp::stringTo<int>(value)) {
88 fileError.back().slot = slot;
96 std::vector<FileInfo> tmp;
97 std::swap(tmp, fileError);
98 for (
const auto& info : tmp) {
108 if (!symbolManager.
reloadFile(filename, loadEmpty, type, slot)) {
109 cliComm.printWarning(
"Symbol file \"", filename,
110 "\" doesn't contain any symbols");
112 if (it != fileError.end()) fileError.erase(it);
113 }
catch (MSXException& e) {
114 cliComm.printWarning(
115 "Couldn't load symbol file \"", filename,
"\": ",
e.getMessage());
116 if (it != fileError.end()) {
117 it->error =
e.getMessage();
120 fileError.emplace_back(filename,
e.getMessage(), type, slot);
125static bool isSlotExpanded(MSXMotherBoard* motherBoard,
int ps)
127 return motherBoard !=
nullptr ? motherBoard->getCPUInterface().isExpanded(ps) :
true;
130static std::string formatSlot(std::optional<uint8_t> slot, MSXMotherBoard* motherBoard =
nullptr)
132 if (!slot)
return "-";
133 int ps = (*slot >> 0) & 3;
134 int ss = (*slot >> 2) & 3;
135 return isSlotExpanded(motherBoard, ps) ?
strCat(ps,
'-', ss)
139static void checkSort(
const SymbolManager& manager, std::vector<SymbolRef>& symbols)
141 auto* sortSpecs = ImGui::TableGetSortSpecs();
142 if (!sortSpecs->SpecsDirty)
return;
144 sortSpecs->SpecsDirty =
false;
145 assert(sortSpecs->SpecsCount == 1);
146 assert(sortSpecs->Specs);
147 assert(sortSpecs->Specs->SortOrder == 0);
149 switch (sortSpecs->Specs->ColumnIndex) {
151 sortUpDown_String(symbols, sortSpecs, [&](
const auto& sym) {
return sym.name(manager); });
154 sortUpDown_T(symbols, sortSpecs, [&](
const auto& sym) {
return sym.value(manager); });
157 sortUpDown_String(symbols, sortSpecs, [&](
const auto& sym) {
return formatSlot(sym.slot(manager)); });
160 sortUpDown_T(symbols, sortSpecs, [&](
const auto& sym) {
return sym.segment(manager); });
163 sortUpDown_String(symbols, sortSpecs, [&](
const auto& sym) {
return sym.file(manager); });
170void ImGuiSymbols::drawContext(MSXMotherBoard* motherBoard,
const SymbolRef& sym)
172 if (ImGui::MenuItem(
"Show in Disassembly",
nullptr,
nullptr, motherBoard !=
nullptr)) {
175 if (ImGui::MenuItem(
"Set breakpoint",
nullptr,
nullptr, motherBoard !=
nullptr)) {
176 auto& interp = motherBoard->getReactor().getInterpreter();
178 newBp.setAddress(interp, TclObject(
strCat(
"$sym(", sym.name(symbolManager),
')')));
180 if (
auto slot = sym.slot(symbolManager)) {
181 auto cond =
strCat(
"[pc_in_slot ", *slot & 3,
' ', (*slot >> 2) & 3);
182 if (
auto segment = sym.segment(symbolManager)) {
186 newBp.setCondition(TclObject(cond));
189 auto& cpuInterface = motherBoard->getCPUInterface();
190 cpuInterface.insertBreakPoint(std::move(newBp));
194template<
bool FILTER_FILE>
195void ImGuiSymbols::drawTable(MSXMotherBoard* motherBoard,
const std::string& file)
197 assert(FILTER_FILE == !file.empty());
199 int flags = ImGuiTableFlags_RowBg |
200 ImGuiTableFlags_BordersV |
201 ImGuiTableFlags_BordersOuter |
202 ImGuiTableFlags_ContextMenuInBody |
203 ImGuiTableFlags_Resizable |
204 ImGuiTableFlags_Reorderable |
205 ImGuiTableFlags_Sortable |
206 ImGuiTableFlags_Hideable |
207 ImGuiTableFlags_SizingStretchProp |
208 (FILTER_FILE ? ImGuiTableFlags_ScrollY : 0);
209 im::Table(file.c_str(), (FILTER_FILE ? 4 : 5), flags, {0, 100}, [&]{
210 ImGui::TableSetupScrollFreeze(0, 1);
211 ImGui::TableSetupColumn(
"name", ImGuiTableColumnFlags_NoHide);
212 ImGui::TableSetupColumn(
"value", ImGuiTableColumnFlags_NoHide | ImGuiTableColumnFlags_DefaultSort | ImGuiTableColumnFlags_WidthFixed);
213 ImGui::TableSetupColumn(
"slot", ImGuiTableColumnFlags_DefaultHide);
214 ImGui::TableSetupColumn(
"segment", ImGuiTableColumnFlags_DefaultHide);
216 ImGui::TableSetupColumn(
"file");
218 ImGui::TableHeadersRow();
219 checkSort(symbolManager, symbols);
221 for (
const auto& sym : symbols) {
222 if (FILTER_FILE && (sym.file(symbolManager) != file))
continue;
224 if (ImGui::TableNextColumn()) {
226 const auto& symName = sym.name(symbolManager);
227 ImGui::Selectable(symName.c_str(),
false, ImGuiSelectableFlags_SpanAllColumns | ImGuiSelectableFlags_AllowOverlap);
228 auto symNameMenu =
strCat(
"symbol-manager##", symName);
229 if (ImGui::IsItemClicked(ImGuiMouseButton_Right)) {
230 ImGui::OpenPopup(symNameMenu.c_str());
232 im::Popup(symNameMenu.c_str(), [&]{ drawContext(motherBoard, sym); });
234 if (ImGui::TableNextColumn()) {
238 if (ImGui::TableNextColumn()) {
242 if (ImGui::TableNextColumn()) {
246 if (!FILTER_FILE && ImGui::TableNextColumn()) {
257 const auto& style = ImGui::GetStyle();
258 ImGui::SetNextWindowSize(
gl::vec2{24, 18} * ImGui::GetFontSize(), ImGuiCond_FirstUseEver);
260 if (ImGui::Button(
"Load symbol file...")) {
262 "Select symbol file",
264 [
this](
const std::string& filename) {
270 im::TreeNode(
"Symbols per file", ImGuiTreeNodeFlags_DefaultOpen, [&]{
271 std::optional<FileInfo> reloadAction;
272 std::string removeAction;
273 auto drawFile = [&](
const std::string& filename, std::string_view error,
SymbolFile::Type type, std::optional<int> slot) {
274 bool hasError = !error.empty();
275 auto* file = symbolManager.
findFile(filename);
276 assert((file !=
nullptr) ^ hasError);
278 auto title =
tmpStrCat(
"File: ", filename);
281 ImGui::TextUnformatted(error);
285 auto arrowSize = ImGui::GetFrameHeight();
286 auto extra = arrowSize + 2.0f * style.FramePadding.x;
288 ImGui::AlignTextToFramePadding();
290 auto preview = formatSlot(file->slot, motherBoard);
293 auto setSlot = [&](std::optional<uint8_t> newSlot) {
294 file->slot = newSlot;
295 for (auto& symbol : file->getSymbols()) {
296 symbol.slot = newSlot;
300 if (ImGui::Selectable(
"-", !file->slot)) {
304 for (uint8_t ps = 0; ps < 4; ++ps) {
305 if (isSlotExpanded(motherBoard, ps)) {
306 for (uint8_t ss = 0; ss < 4; ++ss) {
308 int psSs = (ss << 2) + ps;
309 if (ImGui::Selectable(slotInfo.c_str(), psSs == file->slot)) {
314 if (ImGui::Selectable(
tmpStrCat(ps).c_str(), file->slot && ps == file->slot)) {
323 if (ImGui::Button(
"Reload")) {
324 reloadAction.emplace(filename, std::string{}, type, slot);
327 if (ImGui::Button(
"Remove")) {
328 removeAction = filename;
331 drawTable<true>(motherBoard, filename);
338 for (
const auto& file : symbolManager.
getFiles()) {
339 drawFile(file.filename, {}, file.type, file.slot);
341 for (
const auto& info : fileError) {
342 drawFile(info.filename, info.error, info.type, info.slot);
348 reloadAction->type, reloadAction->slot);
350 if (!removeAction.empty()) {
353 it != fileError.end()) {
359 if (ImGui::Button(
"Reload all")) {
361 return FileInfo{file.filename, std::string{}, file.type, file.slot};
363 append(tmp, std::move(fileError));
365 for (
const auto& info : tmp) {
370 if (ImGui::Button(
"Remove all")) {
371 symbolManager.removeAllFiles();
374 drawTable<false>(motherBoard);
379void ImGuiSymbols::notifySymbolsChanged()
382 for (
const auto& [fileIdx, file] :
enumerate(symbolManager.getFiles())) {
383 for (
auto symbolIdx :
xrange(file.symbols.
size())) {
387 symbols.push_back(SymbolRef{narrow<unsigned>(fileIdx),
388 narrow<unsigned>(symbolIdx)});
392 manager.breakPoints->refreshSymbols();
393 manager.watchExpr->refreshSymbols();
const char * c_str() const
std::unique_ptr< ImGuiOpenFile > openFile
std::unique_ptr< ImGuiDebugger > debugger
static std::string getLastFilter()
void loadLine(std::string_view name, zstring_view value) override
void save(ImGuiTextBuffer &buf) override
void loadStart() override
void paint(MSXMotherBoard *motherBoard) override
ImGuiSymbols(ImGuiManager &manager)
const auto & getFiles() const
static std::string getFileFilters()
void removeFile(std::string_view filename)
bool reloadFile(const std::string &filename, LoadEmpty loadEmpty, SymbolFile::Type type, std::optional< uint8_t > slot={})
static SymbolFile::Type getTypeForFilter(std::string_view filter)
SymbolFile * findFile(std::string_view filename)
void setObserver(SymbolObserver *observer_)
Like std::string_view, but with the extra guarantee that it refers to a zero-terminated string.
constexpr const char * c_str() const
constexpr auto enumerate(Iterable &&iterable)
Heavily inspired by Nathan Reed's blog post: Python-Like enumerate() In C++17 http://reedbeta....
auto CalcTextSize(std::string_view str)
void TextUnformatted(const std::string &str)
void Table(const char *str_id, int column, ImGuiTableFlags flags, const ImVec2 &outer_size, float inner_width, std::invocable<> auto next)
void Window(const char *name, bool *p_open, ImGuiWindowFlags flags, std::invocable<> auto next)
bool TreeNode(const char *label, ImGuiTreeNodeFlags flags, std::invocable<> auto next)
void Combo(const char *label, const char *preview_value, ImGuiComboFlags flags, std::invocable<> auto next)
void StyleColor(bool active, Args &&...args)
void Popup(const char *str_id, ImGuiWindowFlags flags, std::invocable<> auto next)
This file implemented 3 utility functions:
bool loadOnePersistent(std::string_view name, zstring_view value, C &c, const std::tuple< Elements... > &tup)
void sortUpDown_String(Range &range, const ImGuiTableSortSpecs *sortSpecs, Projection proj)
void savePersistent(ImGuiTextBuffer &buf, C &c, const std::tuple< Elements... > &tup)
void sortUpDown_T(Range &range, const ImGuiTableSortSpecs *sortSpecs, Projection proj)
ImU32 getColor(imColor col)
auto find(InputRange &&range, const T &value)
size_t size(std::string_view utf8)
constexpr auto transform(Range &&range, UnaryOp op)
auto to_vector(Range &&range) -> std::vector< detail::ToVectorType< T, decltype(std::begin(range))> >
TemporaryString tmpStrCat(Ts &&... ts)
void strAppend(std::string &result, Ts &&...ts)
static std::optional< Type > parseType(std::string_view str)
static zstring_view toString(Type type)
constexpr auto xrange(T e)