mirror of
https://github.com/bsnes-emu/bsnes.git
synced 2025-10-04 13:11:53 +02:00
byuu says: New terminal is in. Much nicer to use now. Command history makes a major difference in usability. The SMP is now fully traceable and debuggable. Basically they act as separate entities, you can trace both at the same time, but for the most part running and stepping is performed on the chip you select. I'm going to put off CPU+SMP interleave support for a while. I don't actually think it'll be too hard. Will get trickier if/when we support coprocessor debugging. Remaining tasks: - aliases - hotkeys - save states - window geometry Basically, the debugger's done. Just have to add the UI fluff. I also removed tracing/memory export from higan. It was always meant to be temporary until the debugger was remade.
158 lines
6.0 KiB
C++
158 lines
6.0 KiB
C++
#ifdef NALL_STRING_INTERNAL_HPP
|
|
|
|
namespace nall {
|
|
namespace Eval {
|
|
|
|
inline string evaluateExpression(Node* node) {
|
|
#define p(n) evaluateExpression(node->link[n])
|
|
switch(node->type) {
|
|
case Node::Type::Null: return "Null";
|
|
case Node::Type::Literal: return {"Literal:", node->literal};
|
|
case Node::Type::Function: return {"Function(0:", p(0), ", 1:", p(1), ")"};
|
|
case Node::Type::Subscript: return {"Subscript(0:", p(0), ", 1:", p(1), ")"};
|
|
case Node::Type::Member: return {"Member(0:", p(0), ", 1:", p(1), ")"};
|
|
case Node::Type::SuffixIncrement: return {"SuffixIncrement(0:", p(0), ")"};
|
|
case Node::Type::SuffixDecrement: return {"SuffixDecrement(0:", p(0), ")"};
|
|
case Node::Type::Reference: return {"Reference(0:", p(0), ")"};
|
|
case Node::Type::Dereference: return {"Dereference(0:", p(0), ")"};
|
|
case Node::Type::BitwiseNot: return {"Complement(0:", p(0), ")"};
|
|
case Node::Type::PrefixIncrement: return {"PrefixIncrement(0:", p(0), ")"};
|
|
case Node::Type::PrefixDecrement: return {"PrefixDecrement(0:", p(0), ")"};
|
|
case Node::Type::Add: return {"Add(0:", p(0), ", 1:", p(1), ")"};
|
|
case Node::Type::Multiply: return {"Multiply(0:", p(0), ", 1:", p(1), ")"};
|
|
case Node::Type::Concatenate: return {"Concatenate(0:", p(0), ", ", p(1), ")"};
|
|
case Node::Type::Coalesce: return {"Coalesce(0:", p(0), ", ", p(1), ")"};
|
|
case Node::Type::Condition: return {"Condition(0:", p(0), ", ", p(1), ", ", p(2), ")"};
|
|
case Node::Type::Assign: return {"Assign(0:", p(0), ", ", p(1), ")"};
|
|
case Node::Type::Separator: {
|
|
string result = "Separator(";
|
|
for(auto& link : node->link) {
|
|
result.append(evaluateExpression(link), ", ");
|
|
}
|
|
return result.rtrim<1>(", ").append(")");
|
|
}
|
|
}
|
|
#undef p
|
|
|
|
throw "invalid operator";
|
|
}
|
|
|
|
inline int64_t evaluateInteger(Node* node) {
|
|
if(node->type == Node::Type::Literal) {
|
|
if(node->literal.beginsWith("0b")) return nall::binary(node->literal);
|
|
if(node->literal.beginsWith("0o")) return nall::octal(node->literal);
|
|
if(node->literal.beginsWith("0x")) return nall::hex(node->literal);
|
|
if(node->literal.beginsWith("%")) return nall::binary(node->literal);
|
|
if(node->literal.beginsWith("$")) return nall::hex(node->literal);
|
|
return nall::integer(node->literal);
|
|
}
|
|
|
|
#define p(n) evaluateInteger(node->link[n])
|
|
switch(node->type) {
|
|
case Node::Type::SuffixIncrement: return p(0);
|
|
case Node::Type::SuffixDecrement: return p(0);
|
|
case Node::Type::LogicalNot: return !p(0);
|
|
case Node::Type::BitwiseNot: return ~p(0);
|
|
case Node::Type::Positive: return +p(0);
|
|
case Node::Type::Negative: return -p(0);
|
|
case Node::Type::PrefixIncrement: return p(0) + 1;
|
|
case Node::Type::PrefixDecrement: return p(0) - 1;
|
|
case Node::Type::Multiply: return p(0) * p(1);
|
|
case Node::Type::Divide: return p(0) / p(1);
|
|
case Node::Type::Modulo: return p(0) % p(1);
|
|
case Node::Type::Add: return p(0) + p(1);
|
|
case Node::Type::Subtract: return p(0) - p(1);
|
|
case Node::Type::ShiftLeft: return p(0) << p(1);
|
|
case Node::Type::ShiftRight: return p(0) >> p(1);
|
|
case Node::Type::BitwiseAnd: return p(0) & p(1);
|
|
case Node::Type::BitwiseOr: return p(0) | p(1);
|
|
case Node::Type::BitwiseXor: return p(0) ^ p(1);
|
|
case Node::Type::Equal: return p(0) == p(1);
|
|
case Node::Type::NotEqual: return p(0) != p(1);
|
|
case Node::Type::LessThanEqual: return p(0) <= p(1);
|
|
case Node::Type::GreaterThanEqual: return p(0) >= p(1);
|
|
case Node::Type::LessThan: return p(0) < p(1);
|
|
case Node::Type::GreaterThan: return p(0) > p(1);
|
|
case Node::Type::LogicalAnd: return p(0) && p(1);
|
|
case Node::Type::LogicalOr: return p(0) || p(1);
|
|
case Node::Type::Condition: return p(0) ? p(1) : p(2);
|
|
case Node::Type::Assign: return p(1);
|
|
case Node::Type::AssignMultiply: return p(0) * p(1);
|
|
case Node::Type::AssignDivide: return p(0) / p(1);
|
|
case Node::Type::AssignModulo: return p(0) % p(1);
|
|
case Node::Type::AssignAdd: return p(0) + p(1);
|
|
case Node::Type::AssignSubtract: return p(0) - p(1);
|
|
case Node::Type::AssignShiftLeft: return p(0) << p(1);
|
|
case Node::Type::AssignShiftRight: return p(0) >> p(1);
|
|
case Node::Type::AssignBitwiseAnd: return p(0) & p(1);
|
|
case Node::Type::AssignBitwiseOr: return p(0) | p(1);
|
|
case Node::Type::AssignBitwiseXor: return p(0) ^ p(1);
|
|
}
|
|
#undef p
|
|
|
|
throw "invalid operator";
|
|
}
|
|
|
|
inline maybe<int64_t> integer(const string& expression) {
|
|
try {
|
|
auto tree = new Node;
|
|
const char* p = expression;
|
|
parse(tree, p, 0);
|
|
auto result = evaluateInteger(tree);
|
|
delete tree;
|
|
return result;
|
|
} catch(const char*) {
|
|
return nothing;
|
|
}
|
|
}
|
|
|
|
inline long double evaluateReal(Node* node) {
|
|
if(node->type == Node::Type::Literal) return nall::real(node->literal);
|
|
|
|
#define p(n) evaluateReal(node->link[n])
|
|
switch(node->type) {
|
|
case Node::Type::LogicalNot: return !p(0);
|
|
case Node::Type::Positive: return +p(0);
|
|
case Node::Type::Negative: return -p(0);
|
|
case Node::Type::Multiply: return p(0) * p(1);
|
|
case Node::Type::Divide: return p(0) / p(1);
|
|
case Node::Type::Add: return p(0) + p(1);
|
|
case Node::Type::Subtract: return p(0) - p(1);
|
|
case Node::Type::Equal: return p(0) == p(1);
|
|
case Node::Type::NotEqual: return p(0) != p(1);
|
|
case Node::Type::LessThanEqual: return p(0) <= p(1);
|
|
case Node::Type::GreaterThanEqual: return p(0) >= p(1);
|
|
case Node::Type::LessThan: return p(0) < p(1);
|
|
case Node::Type::GreaterThan: return p(0) > p(1);
|
|
case Node::Type::LogicalAnd: return p(0) && p(1);
|
|
case Node::Type::LogicalOr: return p(0) || p(1);
|
|
case Node::Type::Condition: return p(0) ? p(1) : p(2);
|
|
case Node::Type::Assign: return p(1);
|
|
case Node::Type::AssignMultiply: return p(0) * p(1);
|
|
case Node::Type::AssignDivide: return p(0) / p(1);
|
|
case Node::Type::AssignAdd: return p(0) + p(1);
|
|
case Node::Type::AssignSubtract: return p(0) - p(1);
|
|
}
|
|
#undef p
|
|
|
|
throw "invalid operator";
|
|
}
|
|
|
|
inline maybe<long double> real(const string& expression) {
|
|
try {
|
|
auto tree = new Node;
|
|
const char* p = expression;
|
|
parse(tree, p, 0);
|
|
auto result = evaluateReal(tree);
|
|
delete tree;
|
|
return result;
|
|
} catch(const char*) {
|
|
return nothing;
|
|
}
|
|
}
|
|
|
|
}
|
|
}
|
|
|
|
#endif
|