diff options
Diffstat (limited to 'src')
| -rw-r--r-- | src/definition.d | 15 | ||||
| -rw-r--r-- | src/primitives/conditional.d | 77 | ||||
| -rw-r--r-- | src/primitives/core.d (renamed from src/primitives/impl.d) | 70 | ||||
| -rw-r--r-- | src/primitives/eval.d | 98 | ||||
| -rw-r--r-- | src/stack.d | 11 | 
5 files changed, 159 insertions, 112 deletions
| diff --git a/src/definition.d b/src/definition.d index e6a7789..7429f4b 100644 --- a/src/definition.d +++ b/src/definition.d @@ -3,11 +3,12 @@ module src.definition;  import std.string;  import std.variant;  import std.typecons; +  import std.container : DList; -import src.stack : Token; +import src.stack; -alias Words = DList!Token[string]; +alias Words = Stack!Token[string];  Nullable!(DList!Token) definition;  Words                  words; @@ -30,7 +31,7 @@ void end() {  	}  	definition.removeFront; -	words[wordToBeDefined] = definition; +	words[wordToBeDefined] = Stack!Token(definition[]);  	definition.nullify;  } @@ -67,3 +68,11 @@ bool handle(Token token) {  		(string word ) => handle(word)  	);  } + +Stack!Token get(string word) { +	if ( word in words ) { +		return words[word].dup; +	} else { +		return Stack!Token(Token(word)); +	} +} diff --git a/src/primitives/conditional.d b/src/primitives/conditional.d new file mode 100644 index 0000000..ee89ea5 --- /dev/null +++ b/src/primitives/conditional.d @@ -0,0 +1,77 @@ +module src.primitives.conditional; + +import std.variant; +import std.typecons; +import std.container : DList; + +import src.stack; + +Nullable!(DList!Token) buffer; +bool                   concluded = true; +bool                   drop_mode = false; + +void capture(Token token) { +	if ( !drop_mode ) { +		buffer.insertBack(token); +	} +} + +bool drop(Token token) { +	if ( concluded && buffer.isNull ) { +		return false; +	} + +	if ( token.type == typeid(string) ) { +		switch ( *token.peek!string ) { +			case "if"   : eval_if;        break; +			case "then" : eval_then;      break; +			case "else" : eval_else;      break; +			default     : capture(token); break;  +		} +	} else { +		capture(token); +	} + +	return true; +} + +void eval_if() { +	if ( concluded ) { +		buffer    = DList!Token(); +		drop_mode = !stack.pop.get!bool; +		concluded = false; +	} else { +		throw new Exception("conditionals may not be nested directly"); +	} +} + +void eval_then() { +	if ( concluded  ) { +		throw new Exception("`then` without preceding `if`"); +	} else { +		drop_mode = !drop_mode; +	} +} + +void eval_else() { +	if ( concluded ) { +		throw new Exception("`else` without preceding `if`");	 +	} else { +		drop_mode = false; +		concluded = true; +	} +} + +bool dischargeable() { +	return concluded && !buffer.isNull; +} + +Stack!Token discharge() { +	if ( concluded ) { +		Stack!Token result = buffer[]; +		buffer.nullify; +		return result; +	} else { +		throw new Exception("unconcluded conditional may not be discharged");	 +	} +} diff --git a/src/primitives/impl.d b/src/primitives/core.d index 9fddee5..5b3d43d 100644 --- a/src/primitives/impl.d +++ b/src/primitives/core.d @@ -1,4 +1,4 @@ -module src.primitives.impl; +module src.primitives.core;  import std.stdio; @@ -6,65 +6,40 @@ import src.stack;  import src.definition;  Token[string] variables; -bool          drop_mode; -bool definition_start() { +void definition_start() {  	src.definition.start; -	return true;  } -bool binary_op_variable_bind() { -	string name  = stack.pop.get!string; -	Token  value = stack.pop; - +void binary_op_variable_bind() { +	string name     = stack.pop.get!string; +	Token  value    = stack.pop;  	variables[name] = value; -	return true;  } -bool unary_op_variable_resolve() { +void unary_op_variable_resolve() {  	string name = stack.pop.get!string;  	if ( name in variables ) {  		stack.push(variables[name]);  	} - -	return true; -} - -bool unary_conditional_if() { -	drop_mode = !stack.pop.get!bool; -	return true; -} - -bool n_ary_conditional_then() { -	drop_mode = !drop_mode; -	return true;  } -bool n_ary_conditional_else() { -	drop_mode = false; -	return true; -} - -bool binary_op_add() { +void binary_op_add() {  	int b = stack.pop.get!int;  	int a = stack.pop.get!int;  	stack.push(a + b); - -	return true;  } -bool binary_op_multiply() { +void binary_op_multiply() {  	int b = stack.pop.get!int;  	int a = stack.pop.get!int;  	stack.push(a * b); - -	return true;  } -bool binary_op_divide() { +void binary_op_divide() {  	int b = stack.pop.get!int;  	int a = stack.pop.get!int; @@ -73,11 +48,9 @@ bool binary_op_divide() {  	} else {  		stack.push(a / b);  	} - -	return true;  } -bool binary_op_modulo() { +void binary_op_modulo() {  	int b = stack.pop.get!int;  	int a = stack.pop.get!int; @@ -86,52 +59,43 @@ bool binary_op_modulo() {  	} else {  		stack.push(a % b);  	} - -	return true;  } -bool unary_op_io_print() { +void unary_op_io_print() {  	writeln(stack.top); -	return true;  } -bool unary_op_stack_pop() { +void unary_op_stack_pop() {  	stack.pop; -	return true;  } -bool unary_op_stack_dup() { +void unary_op_stack_dup() {  	stack.push(stack.top); -	return true;  } -bool binary_op_stack_swp() { +void binary_op_stack_swp() {  	auto b = stack.pop;  	auto a = stack.pop;  	stack.push(b);  	stack.push(a); -	return true;  } -bool binary_cond_lt() { +void binary_cond_lt() {  	int b = stack.pop.get!int;  	int a = stack.pop.get!int;  	stack.push(a < b); -	return true;  } -bool binary_cond_eq() { +void binary_cond_eq() {  	auto b = stack.pop;  	auto a = stack.pop;  	stack.push(a == b); -	return true;  } -bool integral_value_bool(bool value) { +void integral_value_bool(bool value) {  	stack.push(Token(value)); -	return true;  } diff --git a/src/primitives/eval.d b/src/primitives/eval.d index 8b080e2..08d8fb9 100644 --- a/src/primitives/eval.d +++ b/src/primitives/eval.d @@ -1,65 +1,53 @@  module src.primitives.eval; -import src.primitives.impl; +import std.variant; -bool evaluate(int value) { -	return drop_mode; -} +import src.stack; +import src.primitives.core; +import conditional = src.primitives.conditional; + +bool evaluate_primitive(string word) { +	switch ( word ) { +		case     "§"     : definition_start;           break; +		case     "$"     : binary_op_variable_bind;    break; +		case     "@"     : unary_op_variable_resolve;  break; +		case     "if"    : conditional.eval_if;        break; +		case     "then"  : conditional.eval_then;      break; +		case     "else"  : conditional.eval_else;      break; +		case     "+"     : binary_op_add;              break; +		case     "*"     : binary_op_multiply;         break; +		case     "/"     : binary_op_divide;           break; +		case     "%"     : binary_op_modulo;           break; +		case     "."     : unary_op_io_print;          break; +		case     "pop"   : unary_op_stack_pop;         break; +		case     "dup"   : unary_op_stack_dup;         break; +		case     "swp"   : binary_op_stack_swp;        break; +		case     "true"  : integral_value_bool(true);  break; +		case     "false" : integral_value_bool(false); break; +		case     "<"     : binary_cond_lt;             break; +		case     "="     : binary_cond_eq;             break; +		default          : return false; +	} -bool evaluate(bool value) { -	return drop_mode; +	return true;  } -bool evaluate(string word) { -	if ( drop_mode ) { -		switch ( word ) { -			case "then": -				return n_ary_conditional_then; -			case "else": -				return n_ary_conditional_else; -			default: -				return true; -		} +bool evaluate(Token token) { +	if ( conditional.drop(token) ) { +		return true; +	} else { +		return token.visit!( +			(int    value) => false, +			(bool   value) => false, +			(string word ) => evaluate_primitive(word) +		);  	} +} -	switch ( word ) { -		case "§": -			return definition_start; -		case "$": -			return binary_op_variable_bind; -		case "@": -			return unary_op_variable_resolve; -		case "if": -			return unary_conditional_if; -		case "then": -			return n_ary_conditional_then; -		case "else": -			return n_ary_conditional_else; -		case "+": -			return binary_op_add; -		case "*": -			return binary_op_multiply; -		case "/": -			return binary_op_divide; -		case "%": -			return binary_op_modulo; -		case ".": -			return unary_op_io_print; -		case "pop": -			return unary_op_stack_pop; -		case "dup": -			return unary_op_stack_dup; -		case "swp": -			return binary_op_stack_swp; -		case "true": -			return integral_value_bool(true); -		case "false": -			return integral_value_bool(false); -		case "<": -			return binary_cond_lt; -		case "=": -			return binary_cond_eq; -		default: -			return false; +Stack!Token result() { +	if ( conditional.dischargeable ) { +		return conditional.discharge; +	} else { +		return Stack!Token();  	}  } diff --git a/src/stack.d b/src/stack.d index 2fb602b..678f3e8 100644 --- a/src/stack.d +++ b/src/stack.d @@ -1,7 +1,8 @@  module src.stack; -import std.variant; +import std.conv;  import std.string; +import std.variant;  import std.container : SList;  static import src.definition; @@ -11,6 +12,14 @@ alias Stack = SList;  Stack!Token stack; +Token toToken(string value) { +	if ( value.isNumeric ) { +		return Token(parse!int(value)); +	} else { +		return Token(value); +	} +} +  Token top(ref Stack!Token stack) {  	if ( stack.empty ) {  		throw new Exception("stack is empty"); | 
