123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829 |
- Smalltalk current createPackage: 'Compiler-Interpreter'!
- BlockClosure subclass: #AIBlockClosure
- instanceVariableNames: 'node outerContext'
- package: 'Compiler-Interpreter'!
- !AIBlockClosure commentStamp!
- I am a special `BlockClosure` subclass used by an interpreter to interpret a block node.
- While I am polymorphic with `BlockClosure`, some methods such as `#new` will raise interpretation errors. Unlike a `BlockClosure`, my instance are not JavaScript functions.
- Evaluating an instance will result in interpreting the `node` instance variable (instance of `BlockNode`).!
- !AIBlockClosure methodsFor: 'accessing'!
- compiledSource
- "Unlike blocks, the receiver doesn't represent a JS function"
-
- ^ '[ AST Block closure ]'
- !
- numArgs
- ^ node temps size
- ! !
- !AIBlockClosure methodsFor: 'converting'!
- currySelf
- self interpreterError
- ! !
- !AIBlockClosure methodsFor: 'error handling'!
- interpreterError
- ASTInterpreterError signal: 'Method cannot be interpreted by the interpreter.'
- ! !
- !AIBlockClosure methodsFor: 'evaluating'!
- applyTo: anObject arguments: aCollection
- self interpreterError
- !
- value
- ^ self valueWithPossibleArguments: #()
- !
- value: anArgument
- ^ self valueWithPossibleArguments: {anArgument}
- !
- value: firstArgument value: secondArgument
- ^ self valueWithPossibleArguments: {firstArgument . secondArgument}
- !
- value: firstArgument value: secondArgument value: thirdArgument
- ^ self valueWithPossibleArguments: {firstArgument . secondArgument . thirdArgument}
- !
- valueWithPossibleArguments: aCollection
- | context sequenceNode |
- context := outerContext newBlockContext.
- "Interpret a copy of the sequence node to avoid creating a new AIBlockClosure"
- sequenceNode := node nodes first copy
- parent: nil;
- yourself.
- "Populate the arguments into the context locals"
- node parameters withIndexDo: [ :each :index |
- context localAt: each put: (aCollection at: index ifAbsent: [ nil ]) ].
- "Interpret the first node of the BlockSequenceNode"
- context interpreter
- node: sequenceNode nextChild;
- proceed.
-
- outerContext interpreter
- setNonLocalReturnFromContext: context.
-
- ^ context interpreter pop
- ! !
- !AIBlockClosure methodsFor: 'initialization'!
- initializeWithContext: aContext node: aNode
- node := aNode.
- outerContext := aContext
- ! !
- !AIBlockClosure class methodsFor: 'instance creation'!
- forContext: aContext node: aNode
- ^ self new
- initializeWithContext: aContext node: aNode;
- yourself
- ! !
- MethodContext subclass: #AIContext
- instanceVariableNames: 'outerContext innerContext pc locals selector index sendIndexes evaluatedSelector ast interpreter'
- package: 'Compiler-Interpreter'!
- !AIContext commentStamp!
- I am like a `MethodContext`, used by the `ASTInterpreter`.
- Unlike a `MethodContext`, my instances are not read-only.
- When debugging, my instances are created by copying the current `MethodContext` (thisContext)!
- !AIContext methodsFor: 'accessing'!
- evaluatedSelector
- ^ evaluatedSelector
- !
- evaluatedSelector: aString
- evaluatedSelector := aString
- !
- index
- ^ index ifNil: [ 0 ]
- !
- index: anInteger
- index := anInteger
- !
- innerContext
- ^ innerContext
- !
- innerContext: anAIContext
- innerContext := anAIContext
- !
- localAt: aString
- "Lookup the local value up to the method context"
- ^ self locals at: aString ifAbsent: [
- self outerContext ifNotNil: [ :context |
- context localAt: aString ] ]
- !
- localAt: aString ifAbsent: aBlock
- "Lookup the local value up to the method context"
- ^ self locals at: aString ifAbsent: [
- self outerContext
- ifNotNil: [ :context | context localAt: aString ifAbsent: aBlock ]
- ifNil: [aBlock value] ]
- !
- localAt: aString put: anObject
- self locals at: aString put: anObject
- !
- locals
- locals ifNil: [ self initializeLocals ].
-
- ^ locals
- !
- method
- ^ self methodContext ifNotNil: [
- self methodContext receiver class lookupSelector: self methodContext selector ]
- !
- outerContext
- ^ outerContext
- !
- outerContext: anAIContext
- outerContext := anAIContext.
- outerContext innerContext: self
- !
- selector
- ^ selector
- !
- selector: aString
- selector := aString
- !
- sendIndexAt: aString
- ^ self sendIndexes at: aString ifAbsent: [ 0 ]
- !
- sendIndexes
- ^ sendIndexes ifNil: [ Dictionary new ]
- !
- sendIndexes: aDictionary
- sendIndexes := aDictionary
- ! !
- !AIContext methodsFor: 'factory'!
- newBlockContext
- ^ self class new
- outerContext: self;
- yourself
- ! !
- !AIContext methodsFor: 'initialization'!
- initializeAST
- ast := self method ast.
- (SemanticAnalyzer on: self method methodClass)
- visit: ast
- !
- initializeFromMethodContext: aMethodContext
- self
- evaluatedSelector: aMethodContext evaluatedSelector;
- index: aMethodContext index;
- sendIndexes: aMethodContext sendIndexes;
- receiver: aMethodContext receiver;
- selector: aMethodContext selector.
-
- aMethodContext outerContext ifNotNil: [ :outer |
- "If the method context is nil, the block was defined in JS, so ignore it"
- outer methodContext ifNotNil: [
- self outerContext: (self class fromMethodContext: aMethodContext outerContext) ].
- aMethodContext locals keysAndValuesDo: [ :key :value |
- self locals at: key put: value ] ]
- !
- initializeInterpreter
- interpreter := ASTInterpreter new
- context: self;
- yourself.
-
- self innerContext ifNotNil: [
- self setupInterpreter: interpreter ]
- !
- initializeLocals
- locals := Dictionary new.
- locals at: 'thisContext' put: self.
- ! !
- !AIContext methodsFor: 'interpreting'!
- arguments
- ^ self ast arguments collect: [ :each |
- self localAt: each ]
- !
- ast
- self isBlockContext ifTrue: [
- ^ self outerContext ifNotNil: [ :context | context ast ] ].
- ast ifNil: [ self initializeAST ].
- ^ ast
- !
- interpreter
- interpreter ifNil: [ self initializeInterpreter ].
- ^ interpreter
- !
- interpreter: anInterpreter
- interpreter := anInterpreter
- !
- receiver
- ^ self localAt: 'self'
- !
- receiver: anObject
- self localAt: 'self' put: anObject
- !
- setupInterpreter: anInterpreter
- | currentNode |
-
- "Retrieve the current node"
- currentNode := ASTPCNodeVisitor new
- selector: self evaluatedSelector;
- context: self;
- visit: self ast;
- currentNode.
-
- anInterpreter node: currentNode.
- "Push the send args and receiver to the interpreter stack"
- self innerContext arguments reversed do: [ :each |
- anInterpreter push: each ].
-
- anInterpreter push: (self innerContext receiver)
- ! !
- !AIContext class methodsFor: 'instance creation'!
- fromMethodContext: aMethodContext
- ^ self new
- initializeFromMethodContext: aMethodContext;
- yourself
- ! !
- Object subclass: #ASTDebugger
- instanceVariableNames: 'interpreter context'
- package: 'Compiler-Interpreter'!
- !ASTDebugger commentStamp!
- I am a stepping debugger interface for Amber code.
- I internally use an instance of `ASTInterpreter` to actually step through node and interpret them.
- My instances are created from an `AIContext` with `ASTDebugger class >> context:`.
- They hold an `AIContext` instance internally, recursive copy of the `MethodContext`.
- ## API
- Use the methods of the `'stepping'` protocol to do stepping.!
- !ASTDebugger methodsFor: 'accessing'!
- context
- ^ context
- !
- context: aContext
- context := aContext
- !
- interpreter
- ^ interpreter ifNil: [ interpreter := self defaultInterpreterClass new ]
- !
- interpreter: anInterpreter
- interpreter := anInterpreter
- !
- method
- ^ self context method
- !
- nextNode
- ^ self interpreter nextNode
- ! !
- !ASTDebugger methodsFor: 'defaults'!
- defaultInterpreterClass
- ^ ASTInterpreter
- ! !
- !ASTDebugger methodsFor: 'initialization'!
- buildAST
- "Build the AST tree from the method source code.
- The AST is annotated with a SemanticAnalyzer,
- to know the semantics and bindings of each node needed for later debugging"
-
- | ast |
-
- ast := Smalltalk current parse: self method source.
- (SemanticAnalyzer on: self context receiver class)
- visit: ast.
-
- ^ ast
- !
- initializeInterpreter
- | ast next |
- ast := self buildAST.
- next := ASTPCNodeVisitor new
- context: self context;
- visit: ast;
- currentNode.
- self interpreter node: next
- !
- initializeWithContext: aContext
- "TODO: do we need to handle block contexts?"
-
- self context: aContext.
- self initializeInterpreter
- ! !
- !ASTDebugger methodsFor: 'stepping'!
- proceed
- self shouldBeImplemented
- !
- restart
- self interpreter restart
- !
- skip
- self interpreter skip
- !
- stepInto
- self shouldBeImplemented
- !
- stepOver
- self interpreter stepOver
- ! !
- !ASTDebugger methodsFor: 'testing'!
- atEnd
- ^ self interpreter atEnd
- ! !
- !ASTDebugger class methodsFor: 'instance creation'!
- context: aContext
- ^ self new
- initializeWithContext: aContext;
- yourself
- ! !
- NodeVisitor subclass: #ASTInterpreter
- instanceVariableNames: 'node context stack returnValue returned'
- package: 'Compiler-Interpreter'!
- !ASTInterpreter commentStamp!
- I visit an AST, interpreting (evaluating) nodes one after the other, using a small stack machine.
- ## API
- While my instances should be used from within an `ASTDebugger`, which provides a more high level interface,
- you can use methods from the `interpreting` protocol:
- - `#step` evaluates the current `node` only
- - `#stepOver` evaluates the AST from the current `node` up to the next stepping node (most likely the next send node)
- - `#proceed` evaluates eagerly the AST
- - `#restart` select the first node of the AST
- - `#skip` skips the current node, moving to the next one if any!
- !ASTInterpreter methodsFor: 'accessing'!
- context
- ^ context
- !
- context: aContext
- context := aContext
- !
- node
- "Answer the next node, ie the node to be evaluated in the next step"
-
- ^ node
- !
- node: aNode
- node := aNode
- !
- result
- ^ self hasReturned
- ifTrue: [ self returnValue ]
- ifFalse: [ self context receiver ]
- !
- returnValue
- ^ returnValue
- !
- returnValue: anObject
- returnValue := anObject
- !
- stack
- ^ stack ifNil: [ stack := OrderedCollection new ]
- ! !
- !ASTInterpreter methodsFor: 'interpreting'!
- interpret
- "Interpret the next node to be evaluated"
-
- self visit: self node
- !
- interpret: aNode
- self node: aNode.
- self interpret
- !
- next
- self node: self node nextNode
- !
- proceed
- "Eagerly evaluate the ast"
-
- [ self atEnd ]
- whileFalse: [ self step ]
- !
- restart
- self node: self context ast nextChild
- !
- setNonLocalReturnFromContext: aContext
- aContext interpreter hasReturned ifTrue: [
- returned := true.
- self returnValue: aContext interpreter returnValue ]
- !
- skip
- self next
- !
- step
- self
- interpret;
- next
- !
- stepOver
- self step.
-
- [ self node isSteppingNode ] whileFalse: [
- self step ]
- ! !
- !ASTInterpreter methodsFor: 'private'!
- assign: aNode to: anObject
- aNode binding isInstanceVar
- ifTrue: [ self context receiver instVarAt: aNode value put: anObject ]
- ifFalse: [ self context localAt: aNode value put: anObject ]
- !
- eval: aString
- "Evaluate aString as JS source inside an JS function.
- aString is not sandboxed."
-
- | source function |
-
- source := String streamContents: [ :str |
- str nextPutAll: '(function('.
- self context locals keys
- do: [ :each | str nextPutAll: each ]
- separatedBy: [ str nextPutAll: ',' ].
- str
- nextPutAll: '){ return (function() {';
- nextPutAll: aString;
- nextPutAll: '})() })' ].
-
- function := Compiler new eval: source.
-
- ^ function valueWithPossibleArguments: self context locals values
- !
- messageFromSendNode: aSendNode arguments: aCollection
- ^ Message new
- selector: aSendNode selector;
- arguments: aCollection;
- yourself
- !
- messageNotUnderstood: aMessage receiver: anObject
- MessageNotUnderstood new
- meesage: aMessage;
- receiver: anObject;
- signal
- !
- sendMessage: aMessage to: anObject superSend: aBoolean
- | method |
-
- aBoolean ifFalse: [ ^ aMessage sendTo: anObject ].
- anObject class superclass ifNil: [ ^ self messageNotUnderstood: aMessage receiver: anObject ].
-
- method := anObject class superclass methodDictionary
- at: aMessage selector
- ifAbsent: [ ^ self messageNotUnderstood: aMessage receiver: anObject ].
-
- ^ method sendTo: anObject arguments: aMessage arguments
- ! !
- !ASTInterpreter methodsFor: 'stack'!
- peek
- "Peek the top object of the context stack"
-
- self stack ifEmpty: [ ^ nil ].
-
- ^ self stack last
- !
- pop
- "Pop an object from the context stack"
-
- | peekedValue |
-
- peekedValue := self peek.
- self stack removeLast.
- ^ peekedValue
- !
- push: anObject
- "Push an object to the context stack"
-
- ^ self stack add: anObject
- ! !
- !ASTInterpreter methodsFor: 'testing'!
- atEnd
- ^ self hasReturned or: [ self node isNil ]
- !
- hasReturned
- ^ returned ifNil: [ false ]
- ! !
- !ASTInterpreter methodsFor: 'visiting'!
- visit: aNode
- self hasReturned ifFalse: [ super visit: aNode ]
- !
- visitAssignmentNode: aNode
- | poppedValue |
-
- poppedValue := self pop.
-
- "Pop the left side of the assignment.
- It already has been visited, and we don't need its value."
- self pop.
-
- self push: poppedValue.
- self assign: aNode left to: poppedValue
- !
- visitBlockNode: aNode
- "Do not evaluate the block node.
- Instead, put all instructions into a block that we push to the stack for later evaluation"
-
- | block |
-
- block := AIBlockClosure forContext: self context node: aNode.
-
- self push: block
- !
- visitDynamicArrayNode: aNode
- | array |
-
- array := #().
- aNode nodes do: [ :each |
- array addFirst: self pop ].
-
- self push: array
- !
- visitDynamicDictionaryNode: aNode
- | associations hashedCollection |
-
- associations := OrderedCollection new.
- hashedCollection := HashedCollection new.
-
- aNode nodes do: [ :each |
- associations add: self pop ].
-
- associations reversed do: [ :each |
- hashedCollection add: each ].
-
- self push: hashedCollection
- !
- visitJSStatementNode: aNode
- returned := true.
- self returnValue: (self eval: aNode source)
- !
- visitNode: aNode
- "Do nothing by default. Especially, do not visit children recursively."
- !
- visitReturnNode: aNode
- returned := true.
- self returnValue: self pop
- !
- visitSendNode: aNode
- | receiver args message result |
-
- args := aNode arguments collect: [ :each | self pop ].
- receiver := self pop.
-
- message := self
- messageFromSendNode: aNode
- arguments: args reversed.
-
- result := self sendMessage: message to: receiver superSend: aNode superSend.
-
- "For cascade sends, push the reciever if the send is not the last one"
- (aNode isCascadeSendNode and: [ aNode isLastChild not ])
- ifTrue: [ self push: receiver ]
- ifFalse: [ self push: result ]
- !
- visitValueNode: aNode
- self push: aNode value
- !
- visitVariableNode: aNode
- aNode binding isUnknownVar ifTrue: [
- ^ self push: (PlatformInterface globals at: aNode value ifAbsent: [ self error: 'Unknown variable' ]) ].
-
- self push: (aNode binding isInstanceVar
- ifTrue: [ self context receiver instVarAt: aNode value ]
- ifFalse: [ self context
- localAt: aNode value
- ifAbsent: [
- aNode value isCapitalized
- ifTrue: [
- Smalltalk current
- at: aNode value
- ifAbsent: [ PlatformInterface globals at: aNode value ]]]])
- ! !
- Error subclass: #ASTInterpreterError
- instanceVariableNames: ''
- package: 'Compiler-Interpreter'!
- !ASTInterpreterError commentStamp!
- I get signaled when an AST interpreter is unable to interpret a node.!
- NodeVisitor subclass: #ASTPCNodeVisitor
- instanceVariableNames: 'context index selector currentNode'
- package: 'Compiler-Interpreter'!
- !ASTPCNodeVisitor commentStamp!
- I visit an AST until I get to the current node for the `context` and answer it.
- ## API
- My instances must be filled with a context object using `#context:`.
- After visiting the AST the current node is answered by `#currentNode`!
- !ASTPCNodeVisitor methodsFor: 'accessing'!
- context
- ^ context
- !
- context: aContext
- context := aContext
- !
- currentNode
- ^ currentNode
- !
- increaseIndex
- index := self index + 1
- !
- index
- ^ index ifNil: [ index := 0 ]
- !
- selector
- ^ selector
- !
- selector: aString
- selector := aString
- ! !
- !ASTPCNodeVisitor methodsFor: 'visiting'!
- visitJSStatementNode: aNode
- "If a JSStatementNode is encountered, it always is the current node.
- Stop visiting the AST there"
-
- currentNode := aNode
- !
- visitSendNode: aNode
- | sendIndex |
- sendIndex := self context sendIndexAt: self selector.
-
- super visitSendNode: aNode.
-
- self selector = aNode selector ifTrue: [
- self index < sendIndex ifFalse: [
- self index > sendIndex ifFalse: [ currentNode := aNode ] ].
- self increaseIndex ]
- ! !
- !Node methodsFor: '*Compiler-Interpreter'!
- isSteppingNode
- ^ false
- ! !
- !AssignmentNode methodsFor: '*Compiler-Interpreter'!
- isSteppingNode
- ^ true
- ! !
- !BlockNode methodsFor: '*Compiler-Interpreter'!
- isSteppingNode
- ^ true
- ! !
- !DynamicArrayNode methodsFor: '*Compiler-Interpreter'!
- isSteppingNode
- ^ true
- ! !
- !DynamicDictionaryNode methodsFor: '*Compiler-Interpreter'!
- isSteppingNode
- ^ true
- ! !
- !JSStatementNode methodsFor: '*Compiler-Interpreter'!
- isSteppingNode
- ^ true
- ! !
- !SendNode methodsFor: '*Compiler-Interpreter'!
- isSteppingNode
- ^ true
- ! !
|