mirror of https://github.com/PCSX2/pcsx2.git
Debugger: Add memory search types: GreaterThan(OrEqual), LesserThan(OrEqual), and Not Equal (#10441)
* Make memory search search type handling more clear with enum Adds an enum class to represent the Search type used in a memory search. Prior, this was just handled with an integer to represent each type, but it was very unclear what corresponded to which type at first glance. Made this easier to follow by using an enum to represent the type. * Debugger : Add support for greater than/less than/not equal search types Adds support for basic greater than/greater than or equal/less than/less than or equal/not equal search types for the debugger's Memory Scan. This adds a new input to allow selecting the search comparison type, which defaults to Equals, and allows switching to the above mentioned comparisons. It's set up to allow for adding more easily. Restructures some of the functions to make having multiple comparisons quite manageable. Adds an enum for search comparison types for easy logic handling. * Debugger: Update Array/String search type error to mention not handling Not Equals Currently array/string searches don't support Not Equals searches, so this needs to be removed. * Debugger: Code cleanup + feedback changes Sets up if expressions to use constexpr for compile time evaluation and makes the is greater/less than logic simpler to read for int. Also removes an unneeded QPushButton cast and simply compares the pointers directly.
This commit is contained in:
parent
9740ebe2a4
commit
ade6a6c3ab
|
@ -39,6 +39,9 @@
|
||||||
using namespace QtUtils;
|
using namespace QtUtils;
|
||||||
using namespace MipsStackWalk;
|
using namespace MipsStackWalk;
|
||||||
|
|
||||||
|
using SearchComparison = CpuWidget::SearchComparison;
|
||||||
|
using SearchType = CpuWidget::SearchType;
|
||||||
|
|
||||||
CpuWidget::CpuWidget(QWidget* parent, DebugInterface& cpu)
|
CpuWidget::CpuWidget(QWidget* parent, DebugInterface& cpu)
|
||||||
: m_cpu(cpu)
|
: m_cpu(cpu)
|
||||||
, m_bpModel(cpu)
|
, m_bpModel(cpu)
|
||||||
|
@ -869,7 +872,7 @@ void CpuWidget::onStackListDoubleClick(const QModelIndex& index)
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename T>
|
template <typename T>
|
||||||
static bool checkAddressValueMatches(DebugInterface* cpu, u32 addr, T value)
|
static T readValueAtAddress(DebugInterface* cpu, u32 addr)
|
||||||
{
|
{
|
||||||
T val = 0;
|
T val = 0;
|
||||||
switch (sizeof(T))
|
switch (sizeof(T))
|
||||||
|
@ -882,42 +885,88 @@ static bool checkAddressValueMatches(DebugInterface* cpu, u32 addr, T value)
|
||||||
break;
|
break;
|
||||||
case sizeof(u32):
|
case sizeof(u32):
|
||||||
{
|
{
|
||||||
if (std::is_same_v<T, float>)
|
|
||||||
{
|
|
||||||
const float fTop = value + 0.00001f;
|
|
||||||
const float fBottom = value - 0.00001f;
|
|
||||||
const float memValue = std::bit_cast<float, u32>(cpu->read32(addr));
|
|
||||||
return (fBottom < memValue && memValue < fTop);
|
|
||||||
}
|
|
||||||
|
|
||||||
val = cpu->read32(addr);
|
val = cpu->read32(addr);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case sizeof(u64):
|
case sizeof(u64):
|
||||||
{
|
{
|
||||||
if (std::is_same_v<T, double>)
|
|
||||||
{
|
|
||||||
const double dTop = value + 0.00001f;
|
|
||||||
const double dBottom = value - 0.00001f;
|
|
||||||
const double memValue = std::bit_cast<double, u64>(cpu->read64(addr));
|
|
||||||
return (dBottom < memValue && memValue < dTop);
|
|
||||||
}
|
|
||||||
|
|
||||||
val = cpu->read64(addr);
|
val = cpu->read64(addr);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
default:
|
|
||||||
Console.Error("Debugger: Unknown type when doing memory search!");
|
|
||||||
return false;
|
|
||||||
break;
|
|
||||||
}
|
}
|
||||||
|
return val;
|
||||||
return val == value;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename T>
|
template <typename T>
|
||||||
static std::vector<u32> searchWorker(DebugInterface* cpu, std::vector<u32> searchAddresses, u32 start, u32 end, T value)
|
static bool memoryValueComparator(SearchComparison searchComparison, T searchValue, T readValue)
|
||||||
|
{
|
||||||
|
const bool isNotOperator = searchComparison == SearchComparison::NotEquals;
|
||||||
|
switch (searchComparison)
|
||||||
|
{
|
||||||
|
case SearchComparison::Equals:
|
||||||
|
case SearchComparison::NotEquals:
|
||||||
|
{
|
||||||
|
bool areValuesEqual = false;
|
||||||
|
if constexpr (std::is_same_v<T, float>)
|
||||||
|
{
|
||||||
|
const T fTop = searchValue + 0.00001f;
|
||||||
|
const T fBottom = searchValue - 0.00001f;
|
||||||
|
const T memValue = std::bit_cast<float, u32>(readValue);
|
||||||
|
areValuesEqual = (fBottom < memValue && memValue < fTop);
|
||||||
|
}
|
||||||
|
else if constexpr (std::is_same_v<T, double>)
|
||||||
|
{
|
||||||
|
const double dTop = searchValue + 0.00001f;
|
||||||
|
const double dBottom = searchValue - 0.00001f;
|
||||||
|
const double memValue = std::bit_cast<double, u64>(readValue);
|
||||||
|
areValuesEqual = (dBottom < memValue && memValue < dTop);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
areValuesEqual = searchValue == readValue;
|
||||||
|
}
|
||||||
|
return isNotOperator ? !areValuesEqual : areValuesEqual;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case SearchComparison::GreaterThan:
|
||||||
|
case SearchComparison::GreaterThanOrEqual:
|
||||||
|
case SearchComparison::LessThan:
|
||||||
|
case SearchComparison::LessThanOrEqual:
|
||||||
|
{
|
||||||
|
const bool hasEqualsCheck = searchComparison == SearchComparison::GreaterThanOrEqual || searchComparison == SearchComparison::LessThanOrEqual;
|
||||||
|
if (hasEqualsCheck && memoryValueComparator(SearchComparison::Equals, searchValue, readValue))
|
||||||
|
return true;
|
||||||
|
|
||||||
|
const bool isGreaterOperator = searchComparison == SearchComparison::GreaterThan || searchComparison == SearchComparison::GreaterThanOrEqual;
|
||||||
|
if (std::is_same_v<T, float>)
|
||||||
|
{
|
||||||
|
const T fTop = searchValue + 0.00001f;
|
||||||
|
const T fBottom = searchValue - 0.00001f;
|
||||||
|
const T memValue = std::bit_cast<float, u32>(readValue);
|
||||||
|
const bool isGreater = memValue > fTop;
|
||||||
|
const bool isLesser = memValue < fBottom;
|
||||||
|
return isGreaterOperator ? isGreater : isLesser;
|
||||||
|
}
|
||||||
|
else if (std::is_same_v<T, double>)
|
||||||
|
{
|
||||||
|
const double dTop = searchValue + 0.00001f;
|
||||||
|
const double dBottom = searchValue - 0.00001f;
|
||||||
|
const double memValue = std::bit_cast<double, u64>(readValue);
|
||||||
|
const bool isGreater = memValue > dTop;
|
||||||
|
const bool isLesser = memValue < dBottom;
|
||||||
|
return isGreaterOperator ? isGreater : isLesser;
|
||||||
|
}
|
||||||
|
|
||||||
|
return isGreaterOperator ? (readValue > searchValue) : (readValue < searchValue);
|
||||||
|
}
|
||||||
|
default:
|
||||||
|
Console.Error("Debugger: Unknown type when doing memory search!");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
template <typename T>
|
||||||
|
std::vector<u32> searchWorker(DebugInterface* cpu, std::vector<u32> searchAddresses, SearchComparison searchComparison, u32 start, u32 end, T searchValue)
|
||||||
{
|
{
|
||||||
std::vector<u32> hitAddresses;
|
std::vector<u32> hitAddresses;
|
||||||
const bool isSearchingRange = searchAddresses.size() <= 0;
|
const bool isSearchingRange = searchAddresses.size() <= 0;
|
||||||
|
@ -925,7 +974,10 @@ static std::vector<u32> searchWorker(DebugInterface* cpu, std::vector<u32> searc
|
||||||
{
|
{
|
||||||
for (u32 addr = start; addr < end; addr += sizeof(T))
|
for (u32 addr = start; addr < end; addr += sizeof(T))
|
||||||
{
|
{
|
||||||
if (checkAddressValueMatches(cpu, addr, value))
|
if (!cpu->isValidAddress(addr))
|
||||||
|
continue;
|
||||||
|
T readValue = readValueAtAddress<T>(cpu, addr);
|
||||||
|
if (memoryValueComparator(searchComparison, searchValue, readValue))
|
||||||
{
|
{
|
||||||
hitAddresses.push_back(addr);
|
hitAddresses.push_back(addr);
|
||||||
}
|
}
|
||||||
|
@ -935,11 +987,13 @@ static std::vector<u32> searchWorker(DebugInterface* cpu, std::vector<u32> searc
|
||||||
{
|
{
|
||||||
for (const u32 addr : searchAddresses)
|
for (const u32 addr : searchAddresses)
|
||||||
{
|
{
|
||||||
if (checkAddressValueMatches(cpu, addr, value))
|
if (!cpu->isValidAddress(addr))
|
||||||
|
continue;
|
||||||
|
T readValue = readValueAtAddress<T>(cpu, addr);
|
||||||
|
if (memoryValueComparator(searchComparison, searchValue, readValue))
|
||||||
{
|
{
|
||||||
hitAddresses.push_back(addr);
|
hitAddresses.push_back(addr);
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return hitAddresses;
|
return hitAddresses;
|
||||||
|
@ -959,7 +1013,6 @@ static bool compareByteArrayAtAddress(DebugInterface* cpu, u32 addr, QByteArray
|
||||||
|
|
||||||
static std::vector<u32> searchWorkerByteArray(DebugInterface* cpu, std::vector<u32> searchAddresses, u32 start, u32 end, QByteArray value)
|
static std::vector<u32> searchWorkerByteArray(DebugInterface* cpu, std::vector<u32> searchAddresses, u32 start, u32 end, QByteArray value)
|
||||||
{
|
{
|
||||||
|
|
||||||
std::vector<u32> hitAddresses;
|
std::vector<u32> hitAddresses;
|
||||||
const bool isSearchingRange = searchAddresses.size() <= 0;
|
const bool isSearchingRange = searchAddresses.size() <= 0;
|
||||||
if (isSearchingRange)
|
if (isSearchingRange)
|
||||||
|
@ -986,27 +1039,26 @@ static std::vector<u32> searchWorkerByteArray(DebugInterface* cpu, std::vector<u
|
||||||
return hitAddresses;
|
return hitAddresses;
|
||||||
}
|
}
|
||||||
|
|
||||||
std::vector<u32> startWorker(DebugInterface* cpu, int type, std::vector<u32> searchAddresses, u32 start, u32 end, QString value, int base)
|
std::vector<u32> startWorker(DebugInterface* cpu, const SearchType type, const SearchComparison searchComparison, std::vector<u32> searchAddresses, u32 start, u32 end, QString value, int base)
|
||||||
{
|
{
|
||||||
|
|
||||||
const bool isSigned = value.startsWith("-");
|
const bool isSigned = value.startsWith("-");
|
||||||
switch (type)
|
switch (type)
|
||||||
{
|
{
|
||||||
case 0:
|
case SearchType::ByteType:
|
||||||
return isSigned ? searchWorker<s8>(cpu, searchAddresses, start, end, value.toShort(nullptr, base)) : searchWorker<u8>(cpu, searchAddresses, start, end, value.toUShort(nullptr, base));
|
return isSigned ? searchWorker<s8>(cpu, searchAddresses, searchComparison, start, end, value.toShort(nullptr, base)) : searchWorker<u8>(cpu, searchAddresses, searchComparison, start, end, value.toUShort(nullptr, base));
|
||||||
case 1:
|
case SearchType::Int16Type:
|
||||||
return isSigned ? searchWorker<s16>(cpu, searchAddresses, start, end, value.toShort(nullptr, base)) : searchWorker<u16>(cpu, searchAddresses, start, end, value.toUShort(nullptr, base));
|
return isSigned ? searchWorker<s16>(cpu, searchAddresses, searchComparison, start, end, value.toShort(nullptr, base)) : searchWorker<u16>(cpu, searchAddresses, searchComparison, start, end, value.toUShort(nullptr, base));
|
||||||
case 2:
|
case SearchType::Int32Type:
|
||||||
return isSigned ? searchWorker<s32>(cpu, searchAddresses, start, end, value.toInt(nullptr, base)) : searchWorker<u32>(cpu, searchAddresses, start, end, value.toUInt(nullptr, base));
|
return isSigned ? searchWorker<s32>(cpu, searchAddresses, searchComparison, start, end, value.toInt(nullptr, base)) : searchWorker<u32>(cpu, searchAddresses, searchComparison, start, end, value.toUInt(nullptr, base));
|
||||||
case 3:
|
case SearchType::Int64Type:
|
||||||
return isSigned ? searchWorker<s64>(cpu, searchAddresses, start, end, value.toLong(nullptr, base)) : searchWorker<s64>(cpu, searchAddresses, start, end, value.toULongLong(nullptr, base));
|
return isSigned ? searchWorker<s64>(cpu, searchAddresses, searchComparison, start, end, value.toLong(nullptr, base)) : searchWorker<s64>(cpu, searchAddresses, searchComparison, start, end, value.toULongLong(nullptr, base));
|
||||||
case 4:
|
case SearchType::FloatType:
|
||||||
return searchWorker<float>(cpu, searchAddresses, start, end, value.toFloat());
|
return searchWorker<float>(cpu, searchAddresses, searchComparison, start, end, value.toFloat());
|
||||||
case 5:
|
case SearchType::DoubleType:
|
||||||
return searchWorker<double>(cpu, searchAddresses, start, end, value.toDouble());
|
return searchWorker<double>(cpu, searchAddresses, searchComparison, start, end, value.toDouble());
|
||||||
case 6:
|
case SearchType::StringType:
|
||||||
return searchWorkerByteArray(cpu, searchAddresses, start, end, value.toUtf8());
|
return searchWorkerByteArray(cpu, searchAddresses, start, end, value.toUtf8());
|
||||||
case 7:
|
case SearchType::ArrayType:
|
||||||
return searchWorkerByteArray(cpu, searchAddresses, start, end, QByteArray::fromHex(value.toUtf8()));
|
return searchWorkerByteArray(cpu, searchAddresses, start, end, QByteArray::fromHex(value.toUtf8()));
|
||||||
default:
|
default:
|
||||||
Console.Error("Debugger: Unknown type when doing memory search!");
|
Console.Error("Debugger: Unknown type when doing memory search!");
|
||||||
|
@ -1020,7 +1072,7 @@ void CpuWidget::onSearchButtonClicked()
|
||||||
if (!m_cpu.isAlive())
|
if (!m_cpu.isAlive())
|
||||||
return;
|
return;
|
||||||
|
|
||||||
const int searchType = m_ui.cmbSearchType->currentIndex();
|
const SearchType searchType = static_cast<SearchType>(m_ui.cmbSearchType->currentIndex());
|
||||||
const bool searchHex = m_ui.chkSearchHex->isChecked();
|
const bool searchHex = m_ui.chkSearchHex->isChecked();
|
||||||
|
|
||||||
bool ok;
|
bool ok;
|
||||||
|
@ -1047,25 +1099,33 @@ void CpuWidget::onSearchButtonClicked()
|
||||||
}
|
}
|
||||||
|
|
||||||
const QString searchValue = m_ui.txtSearchValue->text();
|
const QString searchValue = m_ui.txtSearchValue->text();
|
||||||
|
const SearchComparison searchComparison = static_cast<SearchComparison>(m_ui.cmbSearchComparison->currentIndex());
|
||||||
|
const bool isFilterSearch = sender() == m_ui.btnFilterSearch;
|
||||||
unsigned long long value;
|
unsigned long long value;
|
||||||
|
|
||||||
|
const bool isVariableSize = searchType == SearchType::ArrayType || searchType == SearchType::StringType;
|
||||||
|
if (isVariableSize && searchComparison != SearchComparison::Equals)
|
||||||
|
{
|
||||||
|
QMessageBox::critical(this, tr("Debugger"), tr("Search types Array and String can only be used with Equals search comparisons."));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
switch (searchType)
|
switch (searchType)
|
||||||
{
|
{
|
||||||
case 0:
|
case SearchType::ByteType:
|
||||||
case 1:
|
case SearchType::Int16Type:
|
||||||
case 2:
|
case SearchType::Int32Type:
|
||||||
case 3:
|
case SearchType::Int64Type:
|
||||||
value = searchValue.toULongLong(&ok, searchHex ? 16 : 10);
|
value = searchValue.toULongLong(&ok, searchHex ? 16 : 10);
|
||||||
break;
|
break;
|
||||||
case 4:
|
case SearchType::FloatType:
|
||||||
case 5:
|
case SearchType::DoubleType:
|
||||||
searchValue.toDouble(&ok);
|
searchValue.toDouble(&ok);
|
||||||
break;
|
break;
|
||||||
case 6:
|
case SearchType::StringType:
|
||||||
ok = !searchValue.isEmpty();
|
ok = !searchValue.isEmpty();
|
||||||
break;
|
break;
|
||||||
case 7:
|
case SearchType::ArrayType:
|
||||||
ok = !searchValue.trimmed().isEmpty();
|
ok = !searchValue.trimmed().isEmpty();
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
@ -1078,21 +1138,21 @@ void CpuWidget::onSearchButtonClicked()
|
||||||
|
|
||||||
switch (searchType)
|
switch (searchType)
|
||||||
{
|
{
|
||||||
case 7:
|
case SearchType::ArrayType:
|
||||||
case 6:
|
case SearchType::StringType:
|
||||||
case 5:
|
case SearchType::DoubleType:
|
||||||
case 4:
|
case SearchType::FloatType:
|
||||||
break;
|
break;
|
||||||
case 3:
|
case SearchType::Int64Type:
|
||||||
if (value <= std::numeric_limits<unsigned long long>::max())
|
if (value <= std::numeric_limits<unsigned long long>::max())
|
||||||
break;
|
break;
|
||||||
case 2:
|
case SearchType::Int32Type:
|
||||||
if (value <= std::numeric_limits<unsigned long>::max())
|
if (value <= std::numeric_limits<unsigned long>::max())
|
||||||
break;
|
break;
|
||||||
case 1:
|
case SearchType::Int16Type:
|
||||||
if (value <= std::numeric_limits<unsigned short>::max())
|
if (value <= std::numeric_limits<unsigned short>::max())
|
||||||
break;
|
break;
|
||||||
case 0:
|
case SearchType::ByteType:
|
||||||
if (value <= std::numeric_limits<unsigned char>::max())
|
if (value <= std::numeric_limits<unsigned char>::max())
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
|
@ -1111,19 +1171,16 @@ void CpuWidget::onSearchButtonClicked()
|
||||||
m_searchResults = results;
|
m_searchResults = results;
|
||||||
loadSearchResults();
|
loadSearchResults();
|
||||||
m_ui.btnFilterSearch->setDisabled(m_ui.listSearchResults->count() == 0);
|
m_ui.btnFilterSearch->setDisabled(m_ui.listSearchResults->count() == 0);
|
||||||
|
|
||||||
});
|
});
|
||||||
|
|
||||||
m_ui.btnSearch->setDisabled(true);
|
m_ui.btnSearch->setDisabled(true);
|
||||||
QPushButton* senderButton = qobject_cast<QPushButton*>(sender());
|
|
||||||
bool isFilterSearch = senderButton == m_ui.btnFilterSearch;
|
|
||||||
std::vector<u32> addresses;
|
std::vector<u32> addresses;
|
||||||
if (isFilterSearch)
|
if (isFilterSearch)
|
||||||
{
|
{
|
||||||
addresses = m_searchResults;
|
addresses = m_searchResults;
|
||||||
}
|
}
|
||||||
QFuture<std::vector<u32>> workerFuture =
|
QFuture<std::vector<u32>> workerFuture =
|
||||||
QtConcurrent::run(startWorker, &m_cpu, searchType, addresses, searchStart, searchEnd, searchValue, searchHex ? 16 : 10);
|
QtConcurrent::run(startWorker, &m_cpu, searchType, searchComparison, addresses, searchStart, searchEnd, searchValue, searchHex ? 16 : 10);
|
||||||
workerWatcher->setFuture(workerFuture);
|
workerWatcher->setFuture(workerFuture);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1139,7 +1196,8 @@ void CpuWidget::onSearchResultsListScroll(u32 value)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void CpuWidget::loadSearchResults() {
|
void CpuWidget::loadSearchResults()
|
||||||
|
{
|
||||||
const u32 numLoaded = m_ui.listSearchResults->count();
|
const u32 numLoaded = m_ui.listSearchResults->count();
|
||||||
const u32 amountLeftToLoad = m_searchResults.size() - numLoaded;
|
const u32 amountLeftToLoad = m_searchResults.size() - numLoaded;
|
||||||
if (amountLeftToLoad < 1)
|
if (amountLeftToLoad < 1)
|
||||||
|
|
|
@ -44,6 +44,29 @@ public:
|
||||||
CpuWidget(QWidget* parent, DebugInterface& cpu);
|
CpuWidget(QWidget* parent, DebugInterface& cpu);
|
||||||
~CpuWidget();
|
~CpuWidget();
|
||||||
|
|
||||||
|
enum class SearchType
|
||||||
|
{
|
||||||
|
ByteType,
|
||||||
|
Int16Type,
|
||||||
|
Int32Type,
|
||||||
|
Int64Type,
|
||||||
|
FloatType,
|
||||||
|
DoubleType,
|
||||||
|
StringType,
|
||||||
|
ArrayType
|
||||||
|
};
|
||||||
|
|
||||||
|
// Note: The order of these enum values must reflect the order in thee Search Comparison combobox.
|
||||||
|
enum class SearchComparison
|
||||||
|
{
|
||||||
|
Equals,
|
||||||
|
NotEquals,
|
||||||
|
GreaterThan,
|
||||||
|
GreaterThanOrEqual,
|
||||||
|
LessThan,
|
||||||
|
LessThanOrEqual
|
||||||
|
};
|
||||||
|
|
||||||
public slots:
|
public slots:
|
||||||
void paintEvent(QPaintEvent* event);
|
void paintEvent(QPaintEvent* event);
|
||||||
|
|
||||||
|
|
|
@ -306,6 +306,40 @@
|
||||||
</property>
|
</property>
|
||||||
</widget>
|
</widget>
|
||||||
</item>
|
</item>
|
||||||
|
<item row="1" column="1">
|
||||||
|
<widget class="QComboBox" name="cmbSearchComparison">
|
||||||
|
<item>
|
||||||
|
<property name="text">
|
||||||
|
<string>Equals</string>
|
||||||
|
</property>
|
||||||
|
</item>
|
||||||
|
<item>
|
||||||
|
<property name="text">
|
||||||
|
<string>Not Equals</string>
|
||||||
|
</property>
|
||||||
|
</item>
|
||||||
|
<item>
|
||||||
|
<property name="text">
|
||||||
|
<string>Greater Than</string>
|
||||||
|
</property>
|
||||||
|
</item>
|
||||||
|
<item>
|
||||||
|
<property name="text">
|
||||||
|
<string>Greater Than Or Equal</string>
|
||||||
|
</property>
|
||||||
|
</item>
|
||||||
|
<item>
|
||||||
|
<property name="text">
|
||||||
|
<string>Less Than</string>
|
||||||
|
</property>
|
||||||
|
</item>
|
||||||
|
<item>
|
||||||
|
<property name="text">
|
||||||
|
<string>Less Than Or Equal</string>
|
||||||
|
</property>
|
||||||
|
</item>
|
||||||
|
</widget>
|
||||||
|
</item>
|
||||||
</layout>
|
</layout>
|
||||||
</item>
|
</item>
|
||||||
<item>
|
<item>
|
||||||
|
|
Loading…
Reference in New Issue