| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385 | smalltalk.addPackage('Compiler-Interpreter', {});smalltalk.addClass('ASTInterpreter', smalltalk.NodeVisitor, ['currentNode', 'context', 'shouldReturn'], 'Compiler-Interpreter');smalltalk.addMethod("_blockValue_",smalltalk.method({selector: "blockValue:",category: 'interpreting',fn: function (anASTBlockClosure){var self=this;return smalltalk.withContext(function($ctx) { var $1;$1=_st(self)._interpret_(_st(_st(_st(anASTBlockClosure)._astNode())._nodes())._first());return $1;}, self, "blockValue:", [anASTBlockClosure], smalltalk.ASTInterpreter)},args: ["anASTBlockClosure"],source: "blockValue: anASTBlockClosure\x0a\x09^ self interpret: anASTBlockClosure astNode nodes first",messageSends: ["interpret:", "first", "nodes", "astNode"],referencedClasses: []}),smalltalk.ASTInterpreter);smalltalk.addMethod("_context",smalltalk.method({selector: "context",category: 'accessing',fn: function (){var self=this;return smalltalk.withContext(function($ctx) { return self["@context"];}, self, "context", [], smalltalk.ASTInterpreter)},args: [],source: "context\x0a\x09^ context",messageSends: [],referencedClasses: []}),smalltalk.ASTInterpreter);smalltalk.addMethod("_context_",smalltalk.method({selector: "context:",category: 'accessing',fn: function (aMethodContext){var self=this;return smalltalk.withContext(function($ctx) { self["@context"]=aMethodContext;return self}, self, "context:", [aMethodContext], smalltalk.ASTInterpreter)},args: ["aMethodContext"],source: "context: aMethodContext\x0a\x09context := aMethodContext",messageSends: [],referencedClasses: []}),smalltalk.ASTInterpreter);smalltalk.addMethod("_initialize",smalltalk.method({selector: "initialize",category: 'initialization',fn: function (){var self=this;return smalltalk.withContext(function($ctx) { smalltalk.NodeVisitor.fn.prototype._initialize.apply(_st(self), []);self["@shouldReturn"]=false;return self}, self, "initialize", [], smalltalk.ASTInterpreter)},args: [],source: "initialize\x0a\x09super initialize.\x0a    shouldReturn := false",messageSends: ["initialize"],referencedClasses: []}),smalltalk.ASTInterpreter);smalltalk.addMethod("_interpret_",smalltalk.method({selector: "interpret:",category: 'interpreting',fn: function (aNode){var self=this;return smalltalk.withContext(function($ctx) { var $1;self["@shouldReturn"]=false;$1=_st(self)._interpretNode_(aNode);return $1;}, self, "interpret:", [aNode], smalltalk.ASTInterpreter)},args: ["aNode"],source: "interpret: aNode\x0a\x09shouldReturn := false.\x0a    ^ self interpretNode: aNode",messageSends: ["interpretNode:"],referencedClasses: []}),smalltalk.ASTInterpreter);smalltalk.addMethod("_interpretNode_",smalltalk.method({selector: "interpretNode:",category: 'interpreting',fn: function (aNode){var self=this;return smalltalk.withContext(function($ctx) { var $1;self["@currentNode"]=aNode;$1=_st(self)._visit_(aNode);return $1;}, self, "interpretNode:", [aNode], smalltalk.ASTInterpreter)},args: ["aNode"],source: "interpretNode: aNode\x0a\x09currentNode := aNode.\x0a    ^ self visit: aNode",messageSends: ["visit:"],referencedClasses: []}),smalltalk.ASTInterpreter);smalltalk.addMethod("_send_to_arguments_",smalltalk.method({selector: "send:to:arguments:",category: 'interpreting',fn: function (aSelector,anObject,aCollection){var self=this;return smalltalk.withContext(function($ctx) { var $1;$1=_st(anObject)._perform_withArguments_(aSelector,aCollection);return $1;}, self, "send:to:arguments:", [aSelector,anObject,aCollection], smalltalk.ASTInterpreter)},args: ["aSelector", "anObject", "aCollection"],source: "send: aSelector to: anObject arguments: aCollection\x0a\x09^ anObject perform: aSelector withArguments: aCollection",messageSends: ["perform:withArguments:"],referencedClasses: []}),smalltalk.ASTInterpreter);smalltalk.addMethod("_visitBlockNode_",smalltalk.method({selector: "visitBlockNode:",category: 'visiting',fn: function (aNode){var self=this;return smalltalk.withContext(function($ctx) { var $1;$1=(function(){return _st(self)._interpretNode_(_st(_st(aNode)._nodes())._first());});return $1;}, self, "visitBlockNode:", [aNode], smalltalk.ASTInterpreter)},args: ["aNode"],source: "visitBlockNode: aNode\x0a    ^ [ self interpretNode: aNode nodes first ]",messageSends: ["interpretNode:", "first", "nodes"],referencedClasses: []}),smalltalk.ASTInterpreter);smalltalk.addMethod("_visitCascadeNode_",smalltalk.method({selector: "visitCascadeNode:",category: 'visiting',fn: function (aNode){var self=this;return smalltalk.withContext(function($ctx) { var $1;_st(_st(_st(aNode)._nodes())._allButLast())._do_((function(each){_st(each)._receiver_(_st(aNode)._receiver());return _st(self)._interpretNode_(each);}));_st(_st(_st(aNode)._nodes())._last())._receiver_(_st(aNode)._receiver());$1=_st(self)._interpretNode_(_st(_st(aNode)._nodes())._last());return $1;}, self, "visitCascadeNode:", [aNode], smalltalk.ASTInterpreter)},args: ["aNode"],source: "visitCascadeNode: aNode\x0a\x0a    aNode nodes allButLast\x0a    \x09do: [ :each | \x0a        \x09each receiver: aNode receiver.\x0a\x09\x09\x09self interpretNode: each ].\x0a            \x0a\x09aNode nodes last receiver: aNode receiver.\x0a    ^ self interpretNode: aNode nodes last",messageSends: ["do:", "receiver:", "receiver", "interpretNode:", "allButLast", "nodes", "last"],referencedClasses: []}),smalltalk.ASTInterpreter);smalltalk.addMethod("_visitJSStatementNode_",smalltalk.method({selector: "visitJSStatementNode:",category: 'visiting',fn: function (aNode){var self=this;return smalltalk.withContext(function($ctx) { _st(self)._halt();return self}, self, "visitJSStatementNode:", [aNode], smalltalk.ASTInterpreter)},args: ["aNode"],source: "visitJSStatementNode: aNode\x0a\x09self halt",messageSends: ["halt"],referencedClasses: []}),smalltalk.ASTInterpreter);smalltalk.addMethod("_visitReturnNode_",smalltalk.method({selector: "visitReturnNode:",category: 'visiting',fn: function (aNode){var self=this;return smalltalk.withContext(function($ctx) { var $1;self["@shouldReturn"]=true;$1=_st(self)._interpretNode_(_st(_st(aNode)._nodes())._first());return $1;}, self, "visitReturnNode:", [aNode], smalltalk.ASTInterpreter)},args: ["aNode"],source: "visitReturnNode: aNode\x0a\x09shouldReturn := true.\x0a    ^ self interpretNode: aNode nodes first",messageSends: ["interpretNode:", "first", "nodes"],referencedClasses: []}),smalltalk.ASTInterpreter);smalltalk.addMethod("_visitSendNode_",smalltalk.method({selector: "visitSendNode:",category: 'visiting',fn: function (aNode){var self=this;return smalltalk.withContext(function($ctx) { var $1;var receiver;var arguments;receiver=_st(self)._interpretNode_(_st(aNode)._receiver());arguments=_st(_st(aNode)._arguments())._collect_((function(each){return _st(self)._interpretNode_(each);}));$1=_st(self)._send_to_arguments_(_st(aNode)._selector(),receiver,arguments);return $1;}, self, "visitSendNode:", [aNode], smalltalk.ASTInterpreter)},args: ["aNode"],source: "visitSendNode: aNode\x0a\x09\x22TODO: Handle super sends\x22\x0a\x09| receiver arguments |\x0a    \x0a    receiver := self interpretNode: aNode receiver.\x0a    arguments := aNode arguments collect: [ :each |\x0a\x09\x09self interpretNode: each ].\x0a    \x0a    ^ self send: aNode selector to: receiver arguments: arguments",messageSends: ["interpretNode:", "receiver", "collect:", "arguments", "send:to:arguments:", "selector"],referencedClasses: []}),smalltalk.ASTInterpreter);smalltalk.addMethod("_visitSequenceNode_",smalltalk.method({selector: "visitSequenceNode:",category: 'visiting',fn: function (aNode){var self=this;return smalltalk.withContext(function($ctx) { var $1;var $early={};try {_st(_st(_st(aNode)._nodes())._allButLast())._do_((function(each){var value;value=_st(self)._interpretNode_(each);value;if(smalltalk.assert(self["@shouldReturn"])){throw $early=[value];};}));$1=_st(self)._interpretNode_(_st(_st(aNode)._nodes())._last());return $1;}catch(e) {if(e===$early)return e[0]; throw e}}, self, "visitSequenceNode:", [aNode], smalltalk.ASTInterpreter)},args: ["aNode"],source: "visitSequenceNode: aNode\x0a\x09aNode nodes allButLast do: [ :each | | value |\x0a        value := self interpretNode: each.\x0a\x09\x09shouldReturn ifTrue: [ ^ value ] ].\x0a    ^ self interpretNode: aNode nodes last",messageSends: ["do:", "interpretNode:", "ifTrue:", "allButLast", "nodes", "last"],referencedClasses: []}),smalltalk.ASTInterpreter);smalltalk.addMethod("_visitValueNode_",smalltalk.method({selector: "visitValueNode:",category: 'visiting',fn: function (aNode){var self=this;return smalltalk.withContext(function($ctx) { var $1;$1=_st(aNode)._value();return $1;}, self, "visitValueNode:", [aNode], smalltalk.ASTInterpreter)},args: ["aNode"],source: "visitValueNode: aNode\x0a\x09^ aNode value",messageSends: ["value"],referencedClasses: []}),smalltalk.ASTInterpreter);smalltalk.addClass('ASTInterpreterTest', smalltalk.TestCase, [], 'Compiler-Interpreter');smalltalk.addMethod("_analyze_forClass_",smalltalk.method({selector: "analyze:forClass:",category: 'accessing',fn: function (aNode,aClass){var self=this;return smalltalk.withContext(function($ctx) { _st(_st((smalltalk.SemanticAnalyzer || SemanticAnalyzer))._on_(aClass))._visit_(aNode);return aNode;}, self, "analyze:forClass:", [aNode,aClass], smalltalk.ASTInterpreterTest)},args: ["aNode", "aClass"],source: "analyze: aNode forClass: aClass\x0a\x09(SemanticAnalyzer on: aClass) visit: aNode.\x0a    ^ aNode",messageSends: ["visit:", "on:"],referencedClasses: ["SemanticAnalyzer"]}),smalltalk.ASTInterpreterTest);smalltalk.addMethod("_interpret_",smalltalk.method({selector: "interpret:",category: 'accessing',fn: function (aString){var self=this;return smalltalk.withContext(function($ctx) { var $1;$1=_st(_st((smalltalk.ASTInterpreter || ASTInterpreter))._new())._interpret_(_st(_st(_st(self)._parse_forClass_(aString,(smalltalk.Object || Object)))._nodes())._first());return $1;}, self, "interpret:", [aString], smalltalk.ASTInterpreterTest)},args: ["aString"],source: "interpret: aString\x0a\x09\x22the food is a methodNode. Interpret the sequenceNode only\x22\x0a    ^ ASTInterpreter new\x0a    \x09interpret: (self parse: aString forClass: Object) \x0a        \x09nodes first",messageSends: ["interpret:", "first", "nodes", "parse:forClass:", "new"],referencedClasses: ["Object", "ASTInterpreter"]}),smalltalk.ASTInterpreterTest);smalltalk.addMethod("_parse_",smalltalk.method({selector: "parse:",category: 'accessing',fn: function (aString){var self=this;return smalltalk.withContext(function($ctx) { var $1;$1=_st(_st((smalltalk.Smalltalk || Smalltalk))._current())._parse_(aString);return $1;}, self, "parse:", [aString], smalltalk.ASTInterpreterTest)},args: ["aString"],source: "parse: aString\x0a\x09^ Smalltalk current parse: aString",messageSends: ["parse:", "current"],referencedClasses: ["Smalltalk"]}),smalltalk.ASTInterpreterTest);smalltalk.addMethod("_parse_forClass_",smalltalk.method({selector: "parse:forClass:",category: 'accessing',fn: function (aString,aClass){var self=this;return smalltalk.withContext(function($ctx) { var $1;$1=_st(self)._analyze_forClass_(_st(self)._parse_(aString),aClass);return $1;}, self, "parse:forClass:", [aString,aClass], smalltalk.ASTInterpreterTest)},args: ["aString", "aClass"],source: "parse: aString forClass: aClass\x0a\x09^ self analyze: (self parse: aString) forClass: aClass",messageSends: ["analyze:forClass:", "parse:"],referencedClasses: []}),smalltalk.ASTInterpreterTest);smalltalk.addMethod("_testBinarySend",smalltalk.method({selector: "testBinarySend",category: 'tests',fn: function (){var self=this;return smalltalk.withContext(function($ctx) { _st(self)._assert_equals_(_st(self)._interpret_("foo 2+3+4"),(9));return self}, self, "testBinarySend", [], smalltalk.ASTInterpreterTest)},args: [],source: "testBinarySend\x0a\x09self assert: (self interpret: 'foo 2+3+4') equals: 9",messageSends: ["assert:equals:", "interpret:"],referencedClasses: []}),smalltalk.ASTInterpreterTest);smalltalk.addMethod("_testBlockLiteral",smalltalk.method({selector: "testBlockLiteral",category: 'tests',fn: function (){var self=this;return smalltalk.withContext(function($ctx) { _st(self)._assert_equals_(_st(self)._interpret_("foo ^ true ifTrue: [ 1 ] ifFalse: [ 2 ]"),(1));_st(self)._assert_equals_(_st(self)._interpret_("foo true ifTrue: [ ^ 1 ] ifFalse: [ 2 ]"),(1));_st(self)._assert_equals_(_st(self)._interpret_("foo ^ false ifTrue: [ 1 ] ifFalse: [ 2 ]"),(2));return self}, self, "testBlockLiteral", [], smalltalk.ASTInterpreterTest)},args: [],source: "testBlockLiteral\x0a\x09self assert: (self interpret: 'foo ^ true ifTrue: [ 1 ] ifFalse: [ 2 ]') equals: 1.\x0a    self assert: (self interpret: 'foo true ifTrue: [ ^ 1 ] ifFalse: [ 2 ]') equals: 1.\x0a    self assert: (self interpret: 'foo ^ false ifTrue: [ 1 ] ifFalse: [ 2 ]') equals: 2",messageSends: ["assert:equals:", "interpret:"],referencedClasses: []}),smalltalk.ASTInterpreterTest);
 |