From cd54134a22a55a0ad84f169aa96075703d64dd95 Mon Sep 17 00:00:00 2001 From: Peter Lohrmann Date: Thu, 13 Mar 2014 11:45:49 -0700 Subject: [PATCH] UI: Initial support for saving and loading a debug session * This saves a json document which links the initial trace file to a set of snapshots * Edited / outdated / valid properties are stored with each snapshot so that the snapshot indicator in the UI can be restored properly. * All 'rel_path' properties are relative to the location of the session file. This allows the trace file, session file, and session data folder be moved to a new location and still be loaded. Only relative paths are currently supported. Future goals: * Ideally, any state of the UI should be included in the session so that active tabs / drop-down selections / texture zoom settings / etc are restored when the session is loaded. * Indications of which shaders are edited should also be included (after the UI itself supports indicators of these changes) * The original trace file should NOT be changed when the initial snapshot is edited or if/when we support the insertion and removal of API calls, so any changes to the API calls should be stored in this session file as well. (cherry picked from commit 8ede72592ca94d754442e3f097ba03b546047ab1) --- src/vogleditor/vogleditor.cpp | 445 +++++++++++++++++- src/vogleditor/vogleditor.h | 10 + src/vogleditor/vogleditor.ui | 13 + .../vogleditor_qapicalltreemodel.cpp | 56 ++- src/vogleditor/vogleditor_qapicalltreemodel.h | 3 + 5 files changed, 522 insertions(+), 5 deletions(-) diff --git a/src/vogleditor/vogleditor.cpp b/src/vogleditor/vogleditor.cpp index e67967e6..0fcca776 100644 --- a/src/vogleditor/vogleditor.cpp +++ b/src/vogleditor/vogleditor.cpp @@ -45,6 +45,7 @@ #include "vogl_texture_format.h" #include "vogl_trace_file_reader.h" +#include "vogl_trace_file_writer.h" #include "vogleditor_qstatetreemodel.h" #include "vogleditor_statetreetextureitem.h" #include "vogleditor_statetreeprogramitem.h" @@ -118,6 +119,7 @@ VoglEditor::VoglEditor(QWidget *parent) : m_pTrimButton(NULL), m_pStopButton(NULL), m_pTraceReader(NULL), + m_pTraceWriter(NULL), m_pApicallTreeModel(NULL) { ui->setupUi(this); @@ -339,8 +341,16 @@ void VoglEditor::close_trace_file() if (m_pTraceReader != NULL) { m_pTraceReader->close(); + vogl_delete(m_pTraceReader); m_pTraceReader = NULL; + if (m_pTraceWriter != NULL) + { + m_pTraceWriter->close(); + vogl_delete(m_pTraceWriter); + m_pTraceWriter = NULL; + } + setWindowTitle(g_PROJECT_NAME); m_openFilename.clear(); @@ -387,8 +397,7 @@ void VoglEditor::on_actionExport_API_Calls_triggered() } suggestedName += "-ApiCalls.txt"; - QString fileName = QFileDialog::getSaveFileName(this, tr("Export API Calls"), suggestedName, - tr("Text (*.txt)")); + QString fileName = QFileDialog::getSaveFileName(this, tr("Export API Calls"), suggestedName, tr("Text (*.txt)")); if (!fileName.isEmpty()) { @@ -406,6 +415,394 @@ void VoglEditor::on_actionExport_API_Calls_triggered() } } +static const unsigned int VOGLEDITOR_SESSION_FILE_FORMAT_VERSION_1 = 1; +static const unsigned int VOGLEDITOR_SESSION_FILE_FORMAT_VERSION = VOGLEDITOR_SESSION_FILE_FORMAT_VERSION_1; + +bool VoglEditor::load_session_from_disk(QString sessionFile) +{ + // open the json doc + json_document sessionDoc; + if (!sessionDoc.deserialize_file(sessionFile.toStdString().c_str())) + { + return false; + } + + // look for expected metadata + json_node* pMetadata = sessionDoc.get_root()->find_child_object("metadata"); + if (pMetadata == NULL) + { + return false; + } + + const json_value& rFormatVersion = pMetadata->find_value("session_file_format_version"); + if (!rFormatVersion.is_valid()) + { + return false; + } + + if (rFormatVersion.as_uint32() != VOGLEDITOR_SESSION_FILE_FORMAT_VERSION_1) + { + return false; + } + + // load base trace file + json_node* pBaseTraceFile = sessionDoc.get_root()->find_child_object("base_trace_file"); + if (pBaseTraceFile == NULL) + { + return false; + } + + const json_value& rBaseTraceFilePath = pBaseTraceFile->find_value("rel_path"); + const json_value& rBaseTraceFileUuid = pBaseTraceFile->find_value("uuid"); + + if (!rBaseTraceFilePath.is_valid() || !rBaseTraceFileUuid.is_valid()) + { + return false; + } + + dynamic_string sessionPathName; + dynamic_string sessionFileName; + file_utils::split_path(sessionFile.toStdString().c_str(), sessionPathName, sessionFileName); + + dynamic_string traceFilePath = sessionPathName; + traceFilePath.append(rBaseTraceFilePath.as_string()); + + if (!open_trace_file(traceFilePath)) + { + return false; + } + + // TODO: verify UUID of the loaded trace file + + // load session data if it is available + json_node* pSessionData = sessionDoc.get_root()->find_child_object("session_data"); + if (pSessionData != NULL) + { + const json_value& rSessionPath = pSessionData->find_value("rel_path"); + if (!rSessionPath.is_valid()) + { + return false; + } + + dynamic_string sessionDataPath = sessionPathName; + sessionDataPath.append(rSessionPath.as_string()); + + vogl_loose_file_blob_manager file_blob_manager; + file_blob_manager.init(cBMFReadWrite, sessionDataPath.c_str()); + vogl_blob_manager* pBlob_manager = static_cast(&file_blob_manager); + + // load snapshots + const json_node* pSnapshots = pSessionData->find_child_array("snapshots"); + for (unsigned int i = 0; i < pSnapshots->size(); i++) + { + const json_node* pSnapshotNode = pSnapshots->get_value_as_object(i); + + const json_value& uuid = pSnapshotNode->find_value("uuid"); + const json_value& isValid = pSnapshotNode->find_value("is_valid"); + const json_value& isEdited = pSnapshotNode->find_value("is_edited"); + const json_value& isOutdated = pSnapshotNode->find_value("is_outdated"); + const json_value& frameNumber = pSnapshotNode->find_value("frame_number"); + const json_value& callIndex = pSnapshotNode->find_value("call_index"); + const json_value& path = pSnapshotNode->find_value("rel_path"); + + // make sure expected nodes are valid + if (!isValid.is_valid() || !isEdited.is_valid() || !isOutdated.is_valid()) + { + return false; + } + + vogl_gl_state_snapshot* pSnapshot = NULL; + + if (path.is_valid() && isValid.as_bool() && uuid.is_valid()) + { + dynamic_string snapshotPath = sessionDataPath; + snapshotPath.append(path.as_string()); + + // load the snapshot + json_document snapshotDoc; + if (!snapshotDoc.deserialize_file(snapshotPath.c_str())) + { + return false; + } + + // attempt to verify the snapshot file + json_node* pSnapshotRoot = snapshotDoc.get_root(); + if (pSnapshotRoot == NULL) + { + vogl_warning_printf("Invalid snapshot file at %s.", path.as_string_ptr()); + continue; + } + + const json_value& snapshotUuid = pSnapshotRoot->find_value("uuid"); + if (!snapshotUuid.is_valid()) + { + vogl_warning_printf("Invalid 'uuid' in snapshot file at %s.", path.as_string_ptr()); + continue; + } + + if (snapshotUuid.as_string() != uuid.as_string()) + { + vogl_warning_printf("Mismatching 'uuid' between snapshot file at %s and that stored in the session file at %s.", path.as_string_ptr(), sessionFile.toStdString().c_str()); + continue; + } + + vogl_ctypes trace_gl_ctypes(m_pTraceReader->get_sof_packet().m_pointer_sizes); + pSnapshot = vogl_new(vogl_gl_state_snapshot); + if (!pSnapshot->deserialize(*snapshotDoc.get_root(), *pBlob_manager, &trace_gl_ctypes)) + { + vogl_delete(pSnapshot); + pSnapshot = NULL; + vogl_warning_printf("Unable to deserialize the snapshot with uuid %s.", uuid.as_string_ptr()); + continue; + } + } + + vogleditor_gl_state_snapshot* pContainer = vogl_new(vogleditor_gl_state_snapshot, pSnapshot); + pContainer->set_edited(isEdited.as_bool()); + pContainer->set_outdated(isOutdated.as_bool()); + + if (callIndex.is_valid()) + { + // the snapshot is associated with an api call + vogleditor_apiCallTreeItem* pItem = m_pApicallTreeModel->find_call_number(callIndex.as_uint64()); + if (pItem != NULL) + { + pItem->set_snapshot(pContainer); + } + else + { + vogl_warning_printf("Unable to find API call index %" PRIu64 " to load the snapshot into.", callIndex.as_uint64()); + if (pSnapshot != NULL) { vogl_delete(pSnapshot); pSnapshot = NULL; } + if (pContainer != NULL) { vogl_delete(pContainer); pContainer = NULL; } + } + } + else if (frameNumber.is_valid()) + { + // the snapshot is associated with a frame. + // frame snapshots have the additional requirement that the snapshot itself MUST exist since + // we only save a frame snapshot if it is the inital frame and it has been edited. + // If we allow NULL snapshots, that we could accidently remove the initial snapshot that was loaded with the trace file. + if (pSnapshot != NULL) + { + vogleditor_apiCallTreeItem* pItem = m_pApicallTreeModel->find_frame_number(frameNumber.as_uint64()); + if (pItem != NULL) + { + pItem->set_snapshot(pContainer); + } + else + { + vogl_warning_printf("Unable to find frame number %" PRIu64 " to load the snapshot into.", frameNumber.as_uint64()); + if (pSnapshot != NULL) { vogl_delete(pSnapshot); pSnapshot = NULL; } + if (pContainer != NULL) { vogl_delete(pContainer); pContainer = NULL; } + } + } + } + else + { + vogl_warning_printf("Session file contains invalid call or frame number for snapshot with uuid %s", uuid.as_string_ptr()); + if (pSnapshot != NULL) { vogl_delete(pSnapshot); pSnapshot = NULL; } + if (pContainer != NULL) { vogl_delete(pContainer); pContainer = NULL; } + } + } + } + + return true; +} + +/* + * Below is a summary of the information that needs to be saved out in a session's json file so that we can reload the session and be fully-featured. + * Note that not all of this information is currently supported (either by VoglEditor or the save/load functionality). + * + * sample data structure for version 1: +{ + "metadata" : { + "session_file_format_version" : "0x1" <- would need to be updated when organization of existing data is changed + }, + "base_trace_file" : { + "path" : "../traces/trimmed4.bin", + "uuid" : [ 2761638124, 1361789091, 2623121922, 1789156619 ] + }, + "session_data" : { + "path" : "/home/peterl/voglproj/vogl_build/traces/trimmed4-vogleditor-sessiondata/", + "snapshots" : [ + { + "uuid" : "B346B680801ED2F5144E421DEA5EFDCC", + "is_valid" : true, + "is_edited" : false, + "is_outdated" : false, + "frame_number" : 0 + }, + { + "uuid" : "BC261B884088DBEADF376A03A489F2B9", + "is_valid" : true, + "is_edited" : false, + "is_outdated" : false, + "call_index" : 881069, + "path" : "/home/peterl/voglproj/vogl_build/traces/trimmed4-vogleditor-sessiondata/snapshot_call_881069.json" + }, + { + "uuid" : "176DE3DEAA437B871FE122C84D5432E3", + "is_valid" : true, + "is_edited" : true, + "is_outdated" : false, + "call_index" : 881075, + "path" : "/home/peterl/voglproj/vogl_build/traces/trimmed4-vogleditor-sessiondata/snapshot_call_881075.json" + }, + { + "is_valid" : false, + "is_edited" : false, + "is_outdated" : true, + "call_index" : 881080 + } + ] + } +} +*/ +bool VoglEditor::save_session_to_disk(QString sessionFile) +{ + dynamic_string sessionPathName; + dynamic_string sessionFileName; + file_utils::split_path(sessionFile.toStdString().c_str(), sessionPathName, sessionFileName); + + // modify the session file name to make a sessiondata folder + QString sessionDataFolder(sessionFileName.c_str()); + int lastIndex = sessionDataFolder.lastIndexOf('.'); + if (lastIndex != -1) + { + sessionDataFolder = sessionDataFolder.remove(lastIndex, sessionDataFolder.size() - lastIndex); + } + sessionDataFolder += "-sessiondata/"; + + dynamic_string sessionDataPath = sessionPathName; + sessionDataPath.append(sessionDataFolder.toStdString().c_str()); + file_utils::create_directories(sessionDataPath, false); + + vogl_loose_file_blob_manager file_blob_manager; + file_blob_manager.init(cBMFReadWrite, sessionDataPath.c_str()); + vogl_blob_manager* pBlob_manager = static_cast(&file_blob_manager); + + QCursor origCursor = this->cursor(); + setCursor(Qt::WaitCursor); + + json_document sessionDoc; + json_node& metadata = sessionDoc.get_root()->add_object("metadata"); + metadata.add_key_value("session_file_format_version", to_hex_string(VOGLEDITOR_SESSION_FILE_FORMAT_VERSION)); + + // find relative path from session file to trace file + QDir relativeAppDir; + QString absoluteTracePath = relativeAppDir.absoluteFilePath(m_openFilename.toStdString().c_str()); + QDir absoluteSessionFileDir(sessionPathName.c_str()); + QString tracePathRelativeToSessionFile = absoluteSessionFileDir.relativeFilePath(absoluteTracePath); + + json_node& baseTraceFile = sessionDoc.get_root()->add_object("base_trace_file"); + baseTraceFile.add_key_value("rel_path", tracePathRelativeToSessionFile.toStdString().c_str()); + json_node &uuid_array = baseTraceFile.add_array("uuid"); + for (uint i = 0; i < VOGL_ARRAY_SIZE(m_pTraceReader->get_sof_packet().m_uuid); i++) + { + uuid_array.add_value(m_pTraceReader->get_sof_packet().m_uuid[i]); + } + + json_node& sessionDataNode = sessionDoc.get_root()->add_object("session_data"); + sessionDataNode.add_key_value("rel_path", sessionDataFolder.toStdString().c_str()); + json_node& snapshotArray = sessionDataNode.add_array("snapshots"); + + vogleditor_apiCallTreeItem* pItem = m_pApicallTreeModel->find_next_snapshot(NULL); + vogleditor_apiCallTreeItem* pLastItem = NULL; + bool bSavedSuccessfully = true; + while (pItem != pLastItem && pItem != NULL) + { + dynamic_string filename; + + json_node& snapshotNode = snapshotArray.add_object(); + if (pItem->get_snapshot()->get_snapshot() != NULL) + { + dynamic_string strUUID; + snapshotNode.add_key_value("uuid", pItem->get_snapshot()->get_snapshot()->get_uuid().get_string(strUUID)); + } + snapshotNode.add_key_value("is_valid", pItem->get_snapshot()->is_valid()); + snapshotNode.add_key_value("is_edited", pItem->get_snapshot()->is_edited()); + snapshotNode.add_key_value("is_outdated", pItem->get_snapshot()->is_outdated()); + + if (pItem->apiCallItem() != NULL) + { + uint64_t callIndex = pItem->apiCallItem()->globalCallIndex(); + snapshotNode.add_key_value("call_index", callIndex); + if (pItem->get_snapshot()->get_snapshot() != NULL) + { + filename = filename.format("snapshot_call_%" PRIu64 ".json", callIndex); + snapshotNode.add_key_value("rel_path", filename); + dynamic_string filepath = sessionDataPath; + filepath.append(filename); + if (!save_snapshot_to_disk(pItem->get_snapshot()->get_snapshot(), filepath, pBlob_manager)) + { + bSavedSuccessfully = false; + break; + } + } + } + else if (pItem->frameItem() != NULL) + { + // the first frame of a trim will have a snapshot. + // this should only be saved out if the snapshot has been edited + uint64_t frameNumber = pItem->frameItem()->frameNumber(); + snapshotNode.add_key_value("frame_number", frameNumber); + if (pItem->get_snapshot()->is_edited()) + { + filename = filename.format("snapshot_frame_%" PRIu64 ".json", frameNumber); + snapshotNode.add_key_value("rel_path", filename); + dynamic_string filepath = sessionDataPath; + filepath.append(filename); + if (!save_snapshot_to_disk(pItem->get_snapshot()->get_snapshot(), filepath, pBlob_manager)) + { + bSavedSuccessfully = false; + break; + } + } + } + + pLastItem = pItem; + pItem = m_pApicallTreeModel->find_next_snapshot(pLastItem); + } + + if (bSavedSuccessfully) + { + bSavedSuccessfully = sessionDoc.serialize_to_file(sessionFile.toStdString().c_str()); + } + + setCursor(origCursor); + + return bSavedSuccessfully; +} + +bool VoglEditor::save_snapshot_to_disk(vogl_gl_state_snapshot *pSnapshot, dynamic_string filename, vogl_blob_manager *pBlob_manager) +{ + if (pSnapshot == NULL) + { + return false; + } + + json_document doc; + + vogl_ctypes trace_gl_ctypes(m_pTraceReader->get_sof_packet().m_pointer_sizes); + + if (!pSnapshot->serialize(*doc.get_root(), *pBlob_manager, &trace_gl_ctypes)) + { + vogl_error_printf("Failed serializing state snapshot document!\n"); + return false; + } + else if (!doc.serialize_to_file(filename.get_ptr(), true)) + { + vogl_error_printf("Failed writing state snapshot to file \"%s\"!\n", filename.get_ptr()); + return false; + } + else + { + vogl_printf("Successfully wrote JSON snapshot to file \"%s\"\n", filename.get_ptr()); + } + + return true; +} + //---------------------------------------------------------------------------------------------------------------------- // read_state_snapshot_from_trace //---------------------------------------------------------------------------------------------------------------------- @@ -553,6 +950,13 @@ bool VoglEditor::open_trace_file(dynamic_string filename) close_trace_file(); m_pTraceReader = tmpReader; + vogl_ctypes trace_ctypes; + trace_ctypes.init(m_pTraceReader->get_sof_packet().m_pointer_sizes); + m_pTraceWriter = vogl_new(vogl_trace_file_writer, &trace_ctypes); + + dynamic_string traceSessionFilename = "vogleditor_session.bin"; + m_pTraceWriter->open(traceSessionFilename.c_str()); + m_pApicallTreeModel = new vogleditor_QApiCallTreeModel(m_pTraceReader); ui->treeView->setModel(m_pApicallTreeModel); @@ -583,6 +987,7 @@ bool VoglEditor::open_trace_file(dynamic_string filename) ui->searchNextButton->setEnabled(true); ui->action_Close->setEnabled(true); + ui->actionSave_Session->setEnabled(true); ui->actionExport_API_Calls->setEnabled(true); ui->prevSnapshotButton->setEnabled(true); @@ -799,6 +1204,7 @@ void VoglEditor::reset_tracefile_ui() { ui->action_Close->setEnabled(false); ui->actionExport_API_Calls->setEnabled(false); + ui->actionSave_Session->setEnabled(false); ui->prevSnapshotButton->setEnabled(false); ui->nextSnapshotButton->setEnabled(false); @@ -1413,3 +1819,38 @@ void VoglEditor::recursive_update_snapshot_flags(vogleditor_apiCallTreeItem* pIt #undef VOGLEDITOR_DISABLE_TAB #undef VOGLEDITOR_ENABLE_TAB + +void VoglEditor::on_actionSave_Session_triggered() +{ + QString baseName = m_openFilename; + + int lastIndex = baseName.lastIndexOf('.'); + if (lastIndex != -1) + { + baseName = baseName.remove(lastIndex, baseName.size() - lastIndex); + } + + QString suggestedName = baseName + "-vogleditor.json"; + + QString sessionFilename = QFileDialog::getSaveFileName(this, tr("Save Debug Session"), suggestedName, tr("JSON (*.json)")); + + if (!save_session_to_disk(sessionFilename)) + { + m_statusLabel->setText("ERROR: Failed to save session"); + } +} + +void VoglEditor::on_actionOpen_Session_triggered() +{ + QString sessionFilename = QFileDialog::getOpenFileName(this, tr("Load Debug Session"), QString(), tr("JSON (*.json)")); + + QCursor origCursor = this->cursor(); + setCursor(Qt::WaitCursor); + + if (!load_session_from_disk(sessionFilename)) + { + m_statusLabel->setText("ERROR: Failed to load session"); + } + + setCursor(origCursor); +} diff --git a/src/vogleditor/vogleditor.h b/src/vogleditor/vogleditor.h index 53247d6d..ed91623d 100644 --- a/src/vogleditor/vogleditor.h +++ b/src/vogleditor/vogleditor.h @@ -60,6 +60,7 @@ class vogl_replay_window; class vogl_shader_state; class vogl_texture_state; class vogl_trace_file_reader; +class vogl_trace_file_writer; class vogl_trace_packet; class vogl_gl_state_snapshot; class vogleditor_apiCallTimelineModel; @@ -105,6 +106,10 @@ private slots: void on_program_edited(vogl_program_state* pNewProgramState); + void on_actionSave_Session_triggered(); + + void on_actionOpen_Session_triggered(); + private: Ui::VoglEditor* ui; @@ -132,6 +137,10 @@ private slots: void write_child_api_calls(vogleditor_apiCallTreeItem* pItem, FILE* pFile); + bool load_session_from_disk(QString sessionFile); + bool save_session_to_disk(QString sessionFile); + bool save_snapshot_to_disk(vogl_gl_state_snapshot* pSnapshot, dynamic_string filename, vogl_blob_manager *pBlob_manager); + QString m_openFilename; QLabel* m_statusLabel; vogleditor_QFramebufferExplorer* m_framebufferExplorer; @@ -151,6 +160,7 @@ private slots: vogleditor_traceReplayer m_traceReplayer; vogl_trace_file_reader* m_pTraceReader; + vogl_trace_file_writer* m_pTraceWriter; vogl::json_document m_backtraceDoc; vogl::hash_map m_backtraceToJsonMap; diff --git a/src/vogleditor/vogleditor.ui b/src/vogleditor/vogleditor.ui index f0d3a1af..278d2ba8 100644 --- a/src/vogleditor/vogleditor.ui +++ b/src/vogleditor/vogleditor.ui @@ -334,6 +334,9 @@ + + + @@ -372,6 +375,16 @@ Export API Calls... + + + Save Session... + + + + + Open Session... + + diff --git a/src/vogleditor/vogleditor_qapicalltreemodel.cpp b/src/vogleditor/vogleditor_qapicalltreemodel.cpp index c89e1589..28e7e380 100644 --- a/src/vogleditor/vogleditor_qapicalltreemodel.cpp +++ b/src/vogleditor/vogleditor_qapicalltreemodel.cpp @@ -454,10 +454,14 @@ vogleditor_apiCallTreeItem* vogleditor_QApiCallTreeModel::find_next_snapshot(vog { QLinkedListIterator iter(m_itemList); - if (iter.findNext(start) == false) + // if start is NULL, then search will begin from top, otherwise it will begin from the start item and search onwards + if (start != NULL) { - // the object wasn't found in the list, so just return the same item - return start; + if (iter.findNext(start) == false) + { + // the object wasn't found in the list, so just return the same item + return start; + } } // now the iterator is pointing to the desired start object in the list, @@ -546,3 +550,49 @@ vogleditor_apiCallTreeItem *vogleditor_QApiCallTreeModel::find_next_drawcall(vog return pFound; } + +vogleditor_apiCallTreeItem* vogleditor_QApiCallTreeModel::find_call_number(uint64_t callNumber) +{ + QLinkedListIterator iter(m_itemList); + + vogleditor_apiCallTreeItem* pFound = NULL; + while (iter.hasNext()) + { + vogleditor_apiCallTreeItem* pItem = iter.peekNext(); + if (pItem->apiCallItem() != NULL) + { + if (pItem->apiCallItem()->globalCallIndex() == callNumber) + { + pFound = iter.peekNext(); + break; + } + } + + iter.next(); + } + + return pFound; +} + +vogleditor_apiCallTreeItem* vogleditor_QApiCallTreeModel::find_frame_number(uint64_t frameNumber) +{ + QLinkedListIterator iter(m_itemList); + + vogleditor_apiCallTreeItem* pFound = NULL; + while (iter.hasNext()) + { + vogleditor_apiCallTreeItem* pItem = iter.peekNext(); + if (pItem->frameItem() != NULL) + { + if (pItem->frameItem()->frameNumber() == frameNumber) + { + pFound = iter.peekNext(); + break; + } + } + + iter.next(); + } + + return pFound; +} diff --git a/src/vogleditor/vogleditor_qapicalltreemodel.h b/src/vogleditor/vogleditor_qapicalltreemodel.h index 4fc0114f..91305c86 100644 --- a/src/vogleditor/vogleditor_qapicalltreemodel.h +++ b/src/vogleditor/vogleditor_qapicalltreemodel.h @@ -71,6 +71,9 @@ class vogleditor_QApiCallTreeModel : public QAbstractItemModel vogleditor_apiCallTreeItem* find_prev_drawcall(vogleditor_apiCallTreeItem* start); vogleditor_apiCallTreeItem* find_next_drawcall(vogleditor_apiCallTreeItem* start); + vogleditor_apiCallTreeItem* find_call_number(uint64_t callNumber); + vogleditor_apiCallTreeItem* find_frame_number(uint64_t frameNumber); + signals: public slots: