Added method suport in Array.

This commit is contained in:
yhirose 2015-07-21 06:45:24 -04:00
parent 5079d3406a
commit 111d5d8476
6 changed files with 186 additions and 64 deletions

View File

@ -17,9 +17,10 @@
UNARY_NOT <- UNARY_NOT_OPERATOR? MULTIPLICATIVE UNARY_NOT <- UNARY_NOT_OPERATOR? MULTIPLICATIVE
MULTIPLICATIVE <- CALL (MULTIPLICATIVE_OPERATOR CALL)* MULTIPLICATIVE <- CALL (MULTIPLICATIVE_OPERATOR CALL)*
CALL <- PRIMARY (ARGUMENTS / INDEX)* CALL <- PRIMARY (ARGUMENTS / INDEX / METHOD)*
ARGUMENTS <- '(' _ (EXPRESSION (',' _ EXPRESSION)*)? ')' _ ARGUMENTS <- '(' _ (EXPRESSION (',' _ EXPRESSION)*)? ')' _
INDEX <- '[' _ EXPRESSION ']' _ INDEX <- '[' _ EXPRESSION ']' _
METHOD <- '.' _ IDENTIFIER ARGUMENTS
PRIMARY <- WHILE / IF / FUNCTION / IDENTIFIER / ARRAY / NUMBER / BOOLEAN / STRING / INTERPOLATED_STRING / '(' _ EXPRESSION ')' _ PRIMARY <- WHILE / IF / FUNCTION / IDENTIFIER / ARRAY / NUMBER / BOOLEAN / STRING / INTERPOLATED_STRING / '(' _ EXPRESSION ')' _
@ -38,7 +39,7 @@
IDENTIFIER <- < [a-zA-Z_][a-zA-Z0-9_]* > _ IDENTIFIER <- < [a-zA-Z_][a-zA-Z0-9_]* > _
ARRAY <- '[' _ (EXPRESSION (',' _ EXPRESSION)*) ']' _ ARRAY <- '[' _ (EXPRESSION (',' _ EXPRESSION)*)? ']' _
NUMBER <- < [0-9]+ > _ NUMBER <- < [0-9]+ > _
BOOLEAN <- < ('true' / 'false') > _ BOOLEAN <- < ('true' / 'false') > _
STRING <- ['] < (!['] .)* > ['] _ STRING <- ['] < (!['] .)* > ['] _

View File

@ -81,7 +81,7 @@ private:
} }
static Value eval_function(const Ast& ast, shared_ptr<Environment> env) { static Value eval_function(const Ast& ast, shared_ptr<Environment> env) {
vector<Value::FunctionValue::Parameter> params; std::vector<Value::FunctionValue::Parameter> params;
for (auto node: ast.nodes[0]->nodes) { for (auto node: ast.nodes[0]->nodes) {
auto mut = node->nodes[0]->token == "mut"; auto mut = node->nodes[0]->token == "mut";
const auto& name = node->nodes[1]->token; const auto& name = node->nodes[1]->token;
@ -90,13 +90,15 @@ private:
auto body = ast.nodes[1]; auto body = ast.nodes[1];
return Value(Value::FunctionValue { auto f = Value::FunctionValue(
params, params,
[=](shared_ptr<Environment> callEnv) { [=](shared_ptr<Environment> callEnv) {
callEnv->set_outer(env); callEnv->set_outer(env);
return eval(*body, callEnv); return eval(*body, callEnv);
} }
}); );
return Value(std::move(f));
}; };
static Value eval_call(const Ast& ast, shared_ptr<Environment> env) { static Value eval_call(const Ast& ast, shared_ptr<Environment> env) {
@ -107,14 +109,15 @@ private:
if (n.tag == AstTag::Arguments) { if (n.tag == AstTag::Arguments) {
// Function call // Function call
const auto& f = val.to_function(); const auto& f = val.to_function();
const auto& params = f.data->params;
const auto& args = n.nodes; const auto& args = n.nodes;
if (f.params.size() <= args.size()) { if (params.size() <= args.size()) {
auto callEnv = make_shared<Environment>(); auto callEnv = make_shared<Environment>();
callEnv->initialize("self", val, false); callEnv->initialize("self", val, false);
for (auto iprm = 0u; iprm < f.params.size(); iprm++) { for (auto iprm = 0u; iprm < params.size(); iprm++) {
auto param = f.params[iprm]; auto param = params[iprm];
auto arg = args[iprm]; auto arg = args[iprm];
auto val = eval(*arg, env); auto val = eval(*arg, env);
callEnv->initialize(param.name, val, param.mut); callEnv->initialize(param.name, val, param.mut);
@ -123,17 +126,41 @@ private:
callEnv->initialize("__LINE__", Value((long)ast.line), false); callEnv->initialize("__LINE__", Value((long)ast.line), false);
callEnv->initialize("__COLUMN__", Value((long)ast.column), false); callEnv->initialize("__COLUMN__", Value((long)ast.column), false);
val = f.eval(callEnv); val = f.data->eval(callEnv);
} else { } else {
string msg = "arguments error..."; string msg = "arguments error...";
throw runtime_error(msg); throw runtime_error(msg);
} }
} else { // n.tag == AstTag::Index } else if (n.tag == AstTag::Index) {
// Array reference // Array reference
const auto& a = val.to_array(); const auto& arr = val.to_array();
const auto& idx = eval(*n.nodes[0], env).to_long(); auto idx = eval(*n.nodes[0], env).to_long();
if (0 <= idx && idx < static_cast<long>(a.values.size())) { if (0 <= idx && idx < static_cast<long>(arr.data->values.size())) {
val = a.values[idx]; val = arr.data->values.at(idx);
}
} else { // n.tag == AstTag::Property
// Property
auto name = n.nodes[0]->token;
auto prop = val.get_property(name);
if (prop.get_type() == Value::Function) {
const auto& pf = prop.to_function();
auto f = Value::FunctionValue(
pf.data->params,
[=](shared_ptr<Environment> callEnv) {
auto thisEnv = make_shared<Environment>();
thisEnv->set_outer(env);
thisEnv->initialize("this", val, false);
callEnv->set_outer(thisEnv);
return pf.data->eval(callEnv);
}
);
val = Value(std::move(f));
} else {
val = prop;
} }
} }
} }
@ -251,17 +278,15 @@ private:
}; };
static Value eval_array(const Ast& ast, shared_ptr<Environment> env) { static Value eval_array(const Ast& ast, shared_ptr<Environment> env) {
vector<Value> values; Value::ArrayValue arr;
for (auto i = 0u; i < ast.nodes.size(); i++) { for (auto i = 0u; i < ast.nodes.size(); i++) {
auto expr = ast.nodes[i]; auto expr = ast.nodes[i];
auto val = eval(*expr, env); auto val = eval(*expr, env);
values.push_back(val); arr.data->values.push_back(val);
} }
return Value(Value::ArrayValue { return Value(std::move(arr));
values
});
} }
static Value eval_number(const Ast& ast, shared_ptr<Environment> env) { static Value eval_number(const Ast& ast, shared_ptr<Environment> env) {
@ -282,6 +307,32 @@ private:
}; };
}; };
std::map<std::string, Value> Value::ArrayValue::prototypes = {
{
"size",
Value(FunctionValue(
{},
[](shared_ptr<Environment> callEnv) {
const auto& val = callEnv->get("this");
long n = val.to_array().data->values.size();
return Value(n);
}
))
},
{
"push",
Value(FunctionValue {
{ {"arg", false} },
[](shared_ptr<Environment> callEnv) {
const auto& val = callEnv->get("this");
const auto& arg = callEnv->get("arg");
val.to_array().data->values.push_back(arg);
return Value();
}
})
},
};
bool run( bool run(
const string& path, const string& path,
shared_ptr<Environment> env, shared_ptr<Environment> env,

View File

@ -6,21 +6,59 @@ struct Environment;
struct Value struct Value
{ {
enum Type { Undefined, Bool, Long, String, Array, Function }; enum Type { Undefined, Bool, Long, String, Object, Array, Function };
struct ArrayValue {
std::vector<Value> values;
};
struct FunctionValue { struct FunctionValue {
struct Parameter { struct Parameter {
std::string name; std::string name;
bool mut; bool mut;
}; };
struct Data {
std::vector<Parameter> params; std::vector<Parameter> params;
std::function<Value (std::shared_ptr<Environment> env)> eval; std::function<Value (std::shared_ptr<Environment> env)> eval;
}; };
FunctionValue(
const std::vector<Parameter>& params,
const std::function<Value (std::shared_ptr<Environment> env)>& eval) {
data = std::make_shared<Data>();
data->params = params;
data->eval = eval;
}
std::shared_ptr<Data> data;
};
struct ObjectValue {
Value get_property(const std::string& name) const {
return data->props.at(name);
}
struct Data {
std::map<std::string, Value> props;
};
std::shared_ptr<Data> data = std::make_shared<Data>();
};
struct ArrayValue {
Value get_property(const std::string& name) const {
if (data->props.find(name) == data->props.end()) {
return prototypes.at(name);
}
return data->props.at(name);
}
struct Data {
std::map<std::string, Value> props;
std::vector<Value> values;
};
std::shared_ptr<Data> data = std::make_shared<Data>();
static std::map<std::string, Value> prototypes;
};
Value() : type(Undefined) { Value() : type(Undefined) {
//std::cout << "Val::def ctor: " << std::endl; //std::cout << "Val::def ctor: " << std::endl;
} }
@ -52,56 +90,84 @@ struct Value
explicit Value(bool b) : type(Bool), v(b) {} explicit Value(bool b) : type(Bool), v(b) {}
explicit Value(long l) : type(Long), v(l) {} explicit Value(long l) : type(Long), v(l) {}
explicit Value(std::string&& s) : type(String), v(s) {} explicit Value(std::string&& s) : type(String), v(s) {}
explicit Value(ObjectValue&& o) : type(Object), v(0) {}
explicit Value(ArrayValue&& a) : type(Array), v(a) {} explicit Value(ArrayValue&& a) : type(Array), v(a) {}
explicit Value(FunctionValue&& f) : type(Function), v(f) {} explicit Value(FunctionValue&& f) : type(Function), v(f) {}
long size() const { Type get_type() const {
switch (type) { return type;
case String: return to_string().size();
case Array: return to_array().values.size();
}
throw std::runtime_error("type error.");
} }
bool to_bool() const { bool to_bool() const {
switch (type) { switch (type) {
case Bool: return v.get<bool>(); case Bool: return v.get<bool>();
case Long: return v.get<long>() != 0; case Long: return v.get<long>() != 0;
default: throw std::runtime_error("type error.");
} }
throw std::runtime_error("type error.");
} }
long to_long() const { long to_long() const {
switch (type) { switch (type) {
case Bool: return v.get<bool>(); case Bool: return v.get<bool>();
case Long: return v.get<long>(); case Long: return v.get<long>();
default: throw std::runtime_error("type error.");
} }
throw std::runtime_error("type error.");
} }
std::string to_string() const { std::string to_string() const {
switch (type) { switch (type) {
case String: return v.get<std::string>(); case String: return v.get<std::string>();
default: throw std::runtime_error("type error.");
}
}
ObjectValue to_object() const {
switch (type) {
case Object: return v.get<ObjectValue>();
default: throw std::runtime_error("type error.");
} }
throw std::runtime_error("type error.");
} }
ArrayValue to_array() const { ArrayValue to_array() const {
switch (type) { switch (type) {
case Array: return v.get<ArrayValue>(); case Array: return v.get<ArrayValue>();
default: throw std::runtime_error("type error.");
} }
throw std::runtime_error("type error.");
} }
FunctionValue to_function() const { FunctionValue to_function() const {
switch (type) { switch (type) {
case Function: return v.get<FunctionValue>(); case Function: return v.get<FunctionValue>();
default: throw std::runtime_error("type error.");
} }
throw std::runtime_error("type error."); }
Value get_property(const std::string& name) const {
switch (type) {
case Object: return to_object().get_property(name);
case Array: return to_array().get_property(name);
default: throw std::runtime_error("type error.");
}
}
std::string str_object() const {
const auto& props = to_object().data->props;
std::string s = "{";
auto it = props.begin();
for (; it != props.end(); ++it) {
if (it != props.begin()) {
s += ", ";
}
s += '"' + it->first + '"';
s += ": ";
s += it->second.str();
}
s += "}";
return s;
} }
std::string str_array() const { std::string str_array() const {
const auto& values = to_array().values; const auto& values = to_array().data->values;
std::string s = "["; std::string s = "[";
for (auto i = 0u; i < values.size(); i++) { for (auto i = 0u; i < values.size(); i++) {
if (i != 0) { if (i != 0) {
@ -119,8 +185,9 @@ struct Value
case Bool: return to_bool() ? "true" : "false"; case Bool: return to_bool() ? "true" : "false";
case Long: return std::to_string(to_long()); break; case Long: return std::to_string(to_long()); break;
case String: return to_string(); case String: return to_string();
case Function: return "[function]"; case Object: return str_object();
case Array: return str_array(); case Array: return str_array();
case Function: return "[function]";
default: throw std::logic_error("invalid internal condition."); default: throw std::logic_error("invalid internal condition.");
} }
// NOTREACHED // NOTREACHED
@ -206,7 +273,7 @@ struct Value
private: private:
friend std::ostream& operator<<(std::ostream&, const Value&); friend std::ostream& operator<<(std::ostream&, const Value&);
int type; Type type;
peglib::any v; peglib::any v;
}; };
@ -267,20 +334,19 @@ struct Environment
} }
void setup_built_in_functions() { void setup_built_in_functions() {
initialize( {
"puts", auto f = Value::FunctionValue(
Value(Value::FunctionValue {
{ {"arg", true} }, { {"arg", true} },
[](std::shared_ptr<Environment> env) { [](std::shared_ptr<Environment> env) {
std::cout << env->get("arg").str() << std::endl; std::cout << env->get("arg").str() << std::endl;
return Value(); return Value();
} }
}), );
false); initialize("puts", Value(std::move(f)), false);
}
initialize( {
"assert", auto f = Value::FunctionValue(
Value(Value::FunctionValue {
{ {"arg", true} }, { {"arg", true} },
[](std::shared_ptr<Environment> env) { [](std::shared_ptr<Environment> env) {
auto cond = env->get("arg").to_bool(); auto cond = env->get("arg").to_bool();
@ -292,19 +358,9 @@ struct Environment
} }
return Value(); return Value();
} }
}), );
false); initialize("assert", Value(std::move(f)), false);
initialize(
"size",
Value(Value::FunctionValue {
{ {"arg", true} },
[](std::shared_ptr<Environment> env) {
auto size = env->get("arg").size();
return Value(size);
} }
}),
false);
} }
private: private:

View File

@ -23,9 +23,10 @@ static auto g_grammar = R"(
UNARY_NOT <- UNARY_NOT_OPERATOR? MULTIPLICATIVE UNARY_NOT <- UNARY_NOT_OPERATOR? MULTIPLICATIVE
MULTIPLICATIVE <- CALL (MULTIPLICATIVE_OPERATOR CALL)* MULTIPLICATIVE <- CALL (MULTIPLICATIVE_OPERATOR CALL)*
CALL <- PRIMARY (ARGUMENTS / INDEX)* CALL <- PRIMARY (ARGUMENTS / INDEX / PROPERTY)*
ARGUMENTS <- '(' _ (EXPRESSION (',' _ EXPRESSION)*)? ')' _ ARGUMENTS <- '(' _ (EXPRESSION (',' _ EXPRESSION)*)? ')' _
INDEX <- '[' _ EXPRESSION ']' _ INDEX <- '[' _ EXPRESSION ']' _
PROPERTY <- '.' _ IDENTIFIER
PRIMARY <- WHILE / IF / FUNCTION / IDENTIFIER / ARRAY / NUMBER / BOOLEAN / STRING / INTERPOLATED_STRING / '(' _ EXPRESSION ')' _ PRIMARY <- WHILE / IF / FUNCTION / IDENTIFIER / ARRAY / NUMBER / BOOLEAN / STRING / INTERPOLATED_STRING / '(' _ EXPRESSION ')' _
@ -44,7 +45,7 @@ static auto g_grammar = R"(
IDENTIFIER <- < [a-zA-Z_][a-zA-Z0-9_]* > _ IDENTIFIER <- < [a-zA-Z_][a-zA-Z0-9_]* > _
ARRAY <- '[' _ (EXPRESSION (',' _ EXPRESSION)*) ']' _ ARRAY <- '[' _ (EXPRESSION (',' _ EXPRESSION)*)? ']' _
NUMBER <- < [0-9]+ > _ NUMBER <- < [0-9]+ > _
BOOLEAN <- < ('true' / 'false') > _ BOOLEAN <- < ('true' / 'false') > _
STRING <- ['] < (!['] .)* > ['] _ STRING <- ['] < (!['] .)* > ['] _
@ -91,6 +92,7 @@ peg& get_parser()
{ "CALL", Call, true }, { "CALL", Call, true },
{ "ARGUMENTS", Arguments, false }, { "ARGUMENTS", Arguments, false },
{ "INDEX", Index, false }, { "INDEX", Index, false },
{ "PROPERTY", Property, false },
{ "LOGICAL_OR", LogicalOr, true }, { "LOGICAL_OR", LogicalOr, true },
{ "LOGICAL_AND", LogicalAnd, true }, { "LOGICAL_AND", LogicalAnd, true },
{ "CONDITION", Condition, true }, { "CONDITION", Condition, true },

View File

@ -4,7 +4,7 @@ enum AstTag
{ {
Default = peglib::AstDefaultTag, Default = peglib::AstDefaultTag,
Statements, While, If, Call, Assignment, Statements, While, If, Call, Assignment,
Arguments, Index, Arguments, Index, Property,
LogicalOr, LogicalAnd, Condition, UnaryPlus, UnaryMinus, UnaryNot, BinExpresion, LogicalOr, LogicalAnd, Condition, UnaryPlus, UnaryMinus, UnaryNot, BinExpresion,
Identifier, InterpolatedString, Identifier, InterpolatedString,
Number, Boolean, Function, Array Number, Boolean, Function, Array

View File

@ -32,6 +32,17 @@ test_sum = fn () {
assert(ret == 55) assert(ret == 55)
} }
test_array = fn () {
a = [1,2,3]
assert(a.size() == 3)
a.push(4)
assert(a.size() == 4)
b = []
assert(b.size() == 0)
}
test_fib = fn () { test_fib = fn () {
fib = fn (x) { fib = fn (x) {
if x < 2 { if x < 2 {
@ -48,5 +59,6 @@ test_fib = fn () {
test_call() test_call()
test_closure() test_closure()
test_array()
test_sum() test_sum()
test_fib() test_fib()