Compiler-Interpreter.st 24 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185
  1. Smalltalk current createPackage: 'Compiler-Interpreter'!
  2. BlockClosure subclass: #AIBlockClosure
  3. instanceVariableNames: 'interpreter node context'
  4. package: 'Compiler-Interpreter'!
  5. !AIBlockClosure commentStamp!
  6. I am a special `BlockClosure` subclass used by an interpreter to interpret a block node.
  7. While I am polymorphic with `BlockClosure`, some methods such as `#new` will raise interpretation errors. Unlike a `BlockClosure`, my instance are not JavaScript functions.
  8. Evaluating an instance will result in interpreting the `node` instance variable (instance of `BlockNode`).!
  9. !AIBlockClosure methodsFor: 'accessing'!
  10. compiledSource
  11. "Unlike blocks, the receiver doesn't represent a JS function"
  12. ^ '[ AST Block closure ]'
  13. !
  14. numArgs
  15. ^ node temps size
  16. ! !
  17. !AIBlockClosure methodsFor: 'converting'!
  18. currySelf
  19. self interpreterError
  20. ! !
  21. !AIBlockClosure methodsFor: 'error handling'!
  22. interpreterError
  23. AIInterpreterError signal: 'Method cannot be interpreted by the interpreter.'
  24. ! !
  25. !AIBlockClosure methodsFor: 'evaluating'!
  26. applyTo: anObject arguments: aCollection
  27. self interpreterError
  28. !
  29. value
  30. ^ self valueWithPossibleArguments: #()
  31. !
  32. value: anArgument
  33. ^ self valueWithPossibleArguments: #(anArgument)
  34. !
  35. value: firstArgument value: secondArgument
  36. ^ self valueWithPossibleArguments: #(firstArgument secondArgument)
  37. !
  38. value: firstArgument value: secondArgument value: thirdArgument
  39. ^ self valueWithPossibleArguments: #(firstArgument secondArgument thirdArgument)
  40. !
  41. valueWithPossibleArguments: aCollection
  42. "Populate the arguments into the context locals"
  43. node parameters withIndexDo: [ :each :index |
  44. context localAt: each put: (aCollection at: index ifAbsent: [ nil ]) ].
  45. "Interpret the first node of the BlockSequenceNode"
  46. interpreter
  47. context: context;
  48. node: node nodes first nextChild;
  49. step
  50. ! !
  51. !AIBlockClosure methodsFor: 'initialization'!
  52. initializeWithInterpreter: anInterpreter context: aContext node: aNode
  53. interpreter := anInterpreter.
  54. node := aNode.
  55. context := aContext
  56. ! !
  57. !AIBlockClosure class methodsFor: 'instance creation'!
  58. forInterpreter: anInterpreter context: aContext node: aNode
  59. ^ self new
  60. initializeWithInterpreter: anInterpreter context: aContext node: aNode;
  61. yourself
  62. ! !
  63. Object subclass: #AIContext
  64. instanceVariableNames: 'outerContext innerContext pc locals method index ast interpreter methodContext'
  65. package: 'Compiler-Interpreter'!
  66. !AIContext commentStamp!
  67. I am like a `MethodContext`, used by the `ASTInterpreter`.
  68. Unlike a `MethodContext`, my instances are not read-only.
  69. When debugging, my instances are created by copying the current `MethodContext` (thisContext)!
  70. !AIContext methodsFor: 'accessing'!
  71. index
  72. ^ index ifNil: [ 0 ]
  73. !
  74. index: anInteger
  75. index := anInteger
  76. !
  77. innerContext
  78. ^ innerContext
  79. !
  80. innerContext: anAIContext
  81. innerContext := anAIContext
  82. !
  83. localAt: aString
  84. ^ self locals at: aString ifAbsent: [ nil ]
  85. !
  86. localAt: aString put: anObject
  87. self locals at: aString put: anObject
  88. !
  89. locals
  90. locals ifNil: [ self initializeLocals ].
  91. ^ locals
  92. !
  93. method
  94. ^ method
  95. !
  96. method: aCompiledMethod
  97. method := aCompiledMethod
  98. !
  99. outerContext
  100. ^ outerContext
  101. !
  102. outerContext: anAIContext
  103. outerContext := anAIContext.
  104. outerContext innerContext: self
  105. !
  106. selector
  107. ^ self method ifNotNil: [
  108. self method selector ]
  109. ! !
  110. !AIContext methodsFor: 'converting'!
  111. asString
  112. ^ methodContext asString
  113. ! !
  114. !AIContext methodsFor: 'initialization'!
  115. initializeAST
  116. ast := self method ast.
  117. (SemanticAnalyzer on: self method methodClass)
  118. visit: ast
  119. !
  120. initializeFromMethodContext: aMethodContext
  121. methodContext := aMethodContext.
  122. self
  123. pc: aMethodContext pc;
  124. index: aMethodContext index;
  125. receiver: aMethodContext receiver;
  126. method: aMethodContext method.
  127. aMethodContext outerContext ifNotNil: [ :outer |
  128. "If the method context is nil, the block was defined in JS, so ignore it"
  129. outer methodContext ifNotNil: [
  130. self outerContext: (self class fromMethodContext: aMethodContext outerContext) ].
  131. aMethodContext locals keysAndValuesDo: [ :key :value |
  132. self locals at: key put: value ] ]
  133. !
  134. initializeInterpreter
  135. interpreter := Interpreter new
  136. context: self;
  137. node: self retrieveNode;
  138. yourself.
  139. (self innerContext notNil and: [
  140. self innerContext isBlockContext not ]) ifTrue: [
  141. self setupInterpreter: interpreter ]
  142. !
  143. initializeLocals
  144. locals := Dictionary new.
  145. locals at: 'thisContext' put: self.
  146. ! !
  147. !AIContext methodsFor: 'interpreting'!
  148. arguments
  149. ^ self ast arguments collect: [ :each |
  150. self localAt: each ]
  151. !
  152. ast
  153. self isBlockContext ifTrue: [ ^ self outerContext ast ].
  154. ast ifNil: [ self initializeAST ].
  155. ^ ast
  156. !
  157. interpreter
  158. interpreter ifNil: [ self initializeInterpreter ].
  159. ^ interpreter
  160. !
  161. pc
  162. ^ pc ifNil: [ pc := 0 ]
  163. !
  164. pc: anInteger
  165. pc := anInteger
  166. !
  167. receiver
  168. ^ self localAt: 'self'
  169. !
  170. receiver: anObject
  171. self localAt: 'self' put: anObject
  172. !
  173. retrieveNode
  174. ^ ASTPCNodeVisitor new
  175. context: self;
  176. visit: self ast;
  177. currentNode
  178. !
  179. setupInterpreter: anInterpreter
  180. "Push the send args and receiver to the interpreter stack"
  181. self innerContext arguments reversed do: [ :each |
  182. anInterpreter push: each ].
  183. anInterpreter push: (self innerContext receiver)
  184. ! !
  185. !AIContext methodsFor: 'testing'!
  186. isBlockContext
  187. ^ methodContext isBlockContext
  188. ! !
  189. !AIContext class methodsFor: 'instance creation'!
  190. fromMethodContext: aMethodContext
  191. ^ self new
  192. initializeFromMethodContext: aMethodContext;
  193. yourself
  194. ! !
  195. Error subclass: #AIInterpreterError
  196. instanceVariableNames: ''
  197. package: 'Compiler-Interpreter'!
  198. !AIInterpreterError commentStamp!
  199. I get signaled when an AST interpreter is unable to interpret a node.!
  200. Object subclass: #ASTDebugger
  201. instanceVariableNames: 'interpreter context'
  202. package: 'Compiler-Interpreter'!
  203. !ASTDebugger commentStamp!
  204. I am a stepping debugger interface for Amber code.
  205. I internally use an instance of `ASTSteppingInterpreter` to actually step through node and interpret them.
  206. My instances are created from a `MethodContext` with `ASTDebugger class >> context:`.
  207. They hold an `AIContext` instance internally, recursive copy of the `MethodContext`.
  208. ## API
  209. Use the methods of the `'stepping'` protocol to do stepping.!
  210. !ASTDebugger methodsFor: 'accessing'!
  211. context
  212. ^ context
  213. !
  214. context: aContext
  215. context := aContext
  216. !
  217. interpreter
  218. ^ interpreter ifNil: [ interpreter := self defaultInterpreterClass new ]
  219. !
  220. interpreter: anInterpreter
  221. interpreter := anInterpreter
  222. !
  223. method
  224. ^ self context method
  225. !
  226. nextNode
  227. ^ self interpreter nextNode
  228. ! !
  229. !ASTDebugger methodsFor: 'defaults'!
  230. defaultInterpreterClass
  231. ^ ASTSteppingInterpreter
  232. ! !
  233. !ASTDebugger methodsFor: 'initialization'!
  234. buildAST
  235. "Build the AST tree from the method source code.
  236. The AST is annotated with a SemanticAnalyzer,
  237. to know the semantics and bindings of each node needed for later debugging"
  238. | ast |
  239. ast := Smalltalk current parse: self method source.
  240. (SemanticAnalyzer on: self context receiver class)
  241. visit: ast.
  242. ^ ast
  243. !
  244. initializeInterpreter
  245. | ast next |
  246. ast := self buildAST.
  247. next := ASTPCNodeVisitor new
  248. context: self context;
  249. visit: ast;
  250. currentNode.
  251. self interpreter interpret: next
  252. !
  253. initializeWithContext: aContext
  254. "TODO: do we need to handle block contexts?"
  255. self context: aContext.
  256. self initializeInterpreter
  257. ! !
  258. !ASTDebugger methodsFor: 'stepping'!
  259. proceed
  260. self shouldBeImplemented
  261. !
  262. restart
  263. self shouldBeImplemented
  264. !
  265. step
  266. "The ASTSteppingInterpreter stops at each node interpretation.
  267. One step will interpret nodes until:
  268. - we get at the end
  269. - the next node is a stepping node (send, assignment, etc.)"
  270. [ (self interpreter nextNode notNil and: [ self interpreter nextNode stopOnStepping ])
  271. or: [ self interpreter atEnd not ] ]
  272. whileFalse: [
  273. self interpreter step.
  274. self step ]
  275. !
  276. stepInto
  277. self shouldBeImplemented
  278. !
  279. stepOver
  280. self step
  281. ! !
  282. !ASTDebugger methodsFor: 'testing'!
  283. atEnd
  284. ^ self interpreter atEnd
  285. ! !
  286. !ASTDebugger class methodsFor: 'instance creation'!
  287. context: aContext
  288. ^ self new
  289. initializeWithContext: aContext;
  290. yourself
  291. ! !
  292. Object subclass: #ASTInterpreter
  293. instanceVariableNames: 'currentNode nextNode context shouldReturn result'
  294. package: 'Compiler-Interpreter'!
  295. !ASTInterpreter commentStamp!
  296. I am like a `NodeVisitor`, interpreting nodes one after each other.
  297. I am built using Continuation Passing Style for stepping purposes.
  298. ## Usage example:
  299. | ast interpreter |
  300. ast := Smalltalk current parse: 'foo 1+2+4'.
  301. (SemanticAnalyzer on: Object) visit: ast.
  302. ASTInterpreter new
  303. interpret: ast nodes first;
  304. result "Answers 7"!
  305. !ASTInterpreter methodsFor: 'accessing'!
  306. context
  307. ^ context ifNil: [ context := AIContext new ]
  308. !
  309. context: anAIContext
  310. context := anAIContext
  311. !
  312. currentNode
  313. ^ currentNode
  314. !
  315. nextNode
  316. ^ nextNode ifNil: [ self currentNode ]
  317. !
  318. nextNode: aNode
  319. nextNode := aNode
  320. !
  321. result
  322. ^ result
  323. ! !
  324. !ASTInterpreter methodsFor: 'initialization'!
  325. initialize
  326. super initialize.
  327. shouldReturn := false
  328. ! !
  329. !ASTInterpreter methodsFor: 'interpreting'!
  330. interpret: aNode
  331. shouldReturn := false.
  332. self interpret: aNode continue: [ :value |
  333. result := value ]
  334. !
  335. interpret: aNode continue: aBlock
  336. shouldReturn ifTrue: [ ^ self ].
  337. aNode isNode
  338. ifTrue: [
  339. currentNode := aNode.
  340. self interpretNode: aNode continue: [ :value |
  341. self continue: aBlock value: value ] ]
  342. ifFalse: [ self continue: aBlock value: aNode ]
  343. !
  344. interpretAssignmentNode: aNode continue: aBlock
  345. self interpret: aNode right continue: [ :value |
  346. self
  347. continue: aBlock
  348. value: (self assign: aNode left to: value) ]
  349. !
  350. interpretBlockNode: aNode continue: aBlock
  351. self
  352. continue: aBlock
  353. value: [
  354. self withBlockContext: [
  355. self interpret: aNode nodes first; result ] ]
  356. !
  357. interpretBlockSequenceNode: aNode continue: aBlock
  358. self interpretSequenceNode: aNode continue: aBlock
  359. !
  360. interpretCascadeNode: aNode continue: aBlock
  361. "TODO: Handle super sends"
  362. self interpret: aNode receiver continue: [ :receiver |
  363. "Only interpret the receiver once"
  364. aNode nodes do: [ :each | each receiver: receiver ].
  365. self
  366. interpretAll: aNode nodes allButLast
  367. continue: [
  368. self
  369. interpret: aNode nodes last
  370. continue: [ :val | self continue: aBlock value: val ] ] ]
  371. !
  372. interpretClassReferenceNode: aNode continue: aBlock
  373. self continue: aBlock value: (Smalltalk current at: aNode value)
  374. !
  375. interpretDynamicArrayNode: aNode continue: aBlock
  376. self interpretAll: aNode nodes continue: [ :array |
  377. self
  378. continue: aBlock
  379. value: array ]
  380. !
  381. interpretDynamicDictionaryNode: aNode continue: aBlock
  382. self interpretAll: aNode nodes continue: [ :array | | hashedCollection |
  383. hashedCollection := HashedCollection new.
  384. array do: [ :each | hashedCollection add: each ].
  385. self
  386. continue: aBlock
  387. value: hashedCollection ]
  388. !
  389. interpretJSStatementNode: aNode continue: aBlock
  390. shouldReturn := true.
  391. self continue: aBlock value: (self eval: aNode source)
  392. !
  393. interpretMethodNode: aNode continue: aBlock
  394. self interpretAll: aNode nodes continue: [ :array |
  395. self continue: aBlock value: array first ]
  396. !
  397. interpretNode: aNode continue: aBlock
  398. aNode interpreter: self continue: aBlock
  399. !
  400. interpretReturnNode: aNode continue: aBlock
  401. self interpret: aNode nodes first continue: [ :value |
  402. shouldReturn := true.
  403. self continue: aBlock value: value ]
  404. !
  405. interpretSendNode: aNode continue: aBlock
  406. self interpret: aNode receiver continue: [ :receiver |
  407. self interpretAll: aNode arguments continue: [ :args |
  408. self
  409. messageFromSendNode: aNode
  410. arguments: args
  411. do: [ :message |
  412. self context pc: self context pc + 1.
  413. self
  414. continue: aBlock
  415. value: (self sendMessage: message to: receiver superSend: aNode superSend) ] ] ]
  416. !
  417. interpretSequenceNode: aNode continue: aBlock
  418. self interpretAll: aNode nodes continue: [ :array |
  419. self continue: aBlock value: array last ]
  420. !
  421. interpretValueNode: aNode continue: aBlock
  422. self continue: aBlock value: aNode value
  423. !
  424. interpretVariableNode: aNode continue: aBlock
  425. self
  426. continue: aBlock
  427. value: (aNode binding isInstanceVar
  428. ifTrue: [ self context receiver instVarAt: aNode value ]
  429. ifFalse: [ self context localAt: aNode value ])
  430. ! !
  431. !ASTInterpreter methodsFor: 'private'!
  432. assign: aNode to: anObject
  433. ^ aNode binding isInstanceVar
  434. ifTrue: [ self context receiver instVarAt: aNode value put: anObject ]
  435. ifFalse: [ self context localAt: aNode value put: anObject ]
  436. !
  437. continue: aBlock value: anObject
  438. result := anObject.
  439. aBlock value: anObject
  440. !
  441. eval: aString
  442. "Evaluate aString as JS source inside an JS function.
  443. aString is not sandboxed."
  444. | source function |
  445. source := String streamContents: [ :str |
  446. str nextPutAll: '(function('.
  447. self context locals keys
  448. do: [ :each | str nextPutAll: each ]
  449. separatedBy: [ str nextPutAll: ',' ].
  450. str
  451. nextPutAll: '){ return (function() {';
  452. nextPutAll: aString;
  453. nextPutAll: '})() })' ].
  454. function := Compiler new eval: source.
  455. ^ function valueWithPossibleArguments: self context locals values
  456. !
  457. interpretAll: aCollection continue: aBlock
  458. self
  459. interpretAll: aCollection
  460. continue: aBlock
  461. result: OrderedCollection new
  462. !
  463. interpretAll: nodes continue: aBlock result: aCollection
  464. nodes isEmpty
  465. ifTrue: [ self continue: aBlock value: aCollection ]
  466. ifFalse: [
  467. self interpret: nodes first continue: [:value |
  468. self
  469. interpretAll: nodes allButFirst
  470. continue: aBlock
  471. result: aCollection, { value } ] ]
  472. !
  473. messageFromSendNode: aSendNode arguments: aCollection do: aBlock
  474. self
  475. continue: aBlock
  476. value: (Message new
  477. selector: aSendNode selector;
  478. arguments: aCollection;
  479. yourself)
  480. !
  481. sendMessage: aMessage to: anObject superSend: aBoolean
  482. | method |
  483. aBoolean ifFalse: [ ^ aMessage sendTo: anObject ].
  484. anObject class superclass ifNil: [ ^ self messageNotUnderstood: aMessage receiver: anObject ].
  485. method := anObject class superclass methodDictionary
  486. at: aMessage selector
  487. ifAbsent: [ ^ self messageNotUnderstood: aMessage receiver: anObject ].
  488. ^ method fn applyTo: anObject arguments: aMessage arguments
  489. !
  490. withBlockContext: aBlock
  491. "Evaluate aBlock with a BlockContext:
  492. - a context is pushed before aBlock evaluation.
  493. - the context is poped after aBlock evaluation
  494. - the result of aBlock evaluation is answered"
  495. | blockResult |
  496. self context: (AIContext new
  497. outerContext: self context;
  498. yourself).
  499. blockResult := aBlock value.
  500. self context: self context outerContext.
  501. ^ blockResult
  502. ! !
  503. !ASTInterpreter methodsFor: 'testing'!
  504. shouldReturn
  505. ^ shouldReturn ifNil: [ false ]
  506. ! !
  507. ASTInterpreter subclass: #ASTSteppingInterpreter
  508. instanceVariableNames: 'continuation nextNode'
  509. package: 'Compiler-Interpreter'!
  510. !ASTSteppingInterpreter commentStamp!
  511. I am an interpreter with stepping capabilities. The higher level `ASTDebugger` class should be used as a debugger model, as it provides convenience methods for debugging.
  512. ## API
  513. Use `#step` to actually interpret the next node. Interpretation stops at each node evaluation, weither it's a message node or not.
  514. ## Usage example:
  515. | ast interpreter |
  516. ast := Smalltalk current parse: 'foo 1+2+4'.
  517. (SemanticAnalyzer on: Object) visit: ast.
  518. interpreter := ASTSteppingInterpreter new
  519. interpret: ast nodes first;
  520. yourself.
  521. interpreter step; step.
  522. interpreter step; step.
  523. interpreter result."Answers 1"
  524. interpreter step.
  525. interpreter result. "Answers 3"
  526. interpreter step.
  527. interpreter result. "Answers 7"!
  528. !ASTSteppingInterpreter methodsFor: 'accessing'!
  529. nextNode
  530. ^ nextNode
  531. ! !
  532. !ASTSteppingInterpreter methodsFor: 'initialization'!
  533. initialize
  534. super initialize.
  535. continuation := []
  536. ! !
  537. !ASTSteppingInterpreter methodsFor: 'interpreting'!
  538. interpret: aNode continue: aBlock
  539. nextNode := aNode.
  540. continuation := [
  541. super interpret: aNode continue: aBlock ]
  542. ! !
  543. !ASTSteppingInterpreter methodsFor: 'stepping'!
  544. step
  545. continuation value
  546. ! !
  547. !ASTSteppingInterpreter methodsFor: 'testing'!
  548. atEnd
  549. ^ self shouldReturn or: [ self nextNode == self currentNode ]
  550. ! !
  551. NodeVisitor subclass: #ASTPCNodeVisitor
  552. instanceVariableNames: 'useInlinings pc context blockIndex currentNode'
  553. package: 'Compiler-Interpreter'!
  554. !ASTPCNodeVisitor commentStamp!
  555. I visit an AST until I get to the current pc node and answer it.
  556. ## API
  557. My instances must be filled with a context object using `#context:`.
  558. After visiting the AST the current node corresponding to the `pc` is answered by `#currentNode`!
  559. !ASTPCNodeVisitor methodsFor: 'accessing'!
  560. blockIndex
  561. ^ blockIndex ifNil: [ blockIndex := 0 ]
  562. !
  563. context
  564. ^ context
  565. !
  566. context: aContext
  567. context := aContext
  568. !
  569. currentNode
  570. ^ currentNode
  571. !
  572. increaseBlockIndex
  573. blockIndex := self blockIndex + 1
  574. !
  575. pc
  576. ^ pc ifNil: [ 0 ]
  577. !
  578. pc: anInteger
  579. pc := anInteger
  580. !
  581. useInlinings
  582. ^ useInlinings ifNil: [ true ]
  583. !
  584. useInlinings: aBoolean
  585. useInlinings := aBoolean
  586. ! !
  587. !ASTPCNodeVisitor methodsFor: 'visiting'!
  588. visitBlockNode: aNode
  589. self blockIndex >= self context index ifFalse: [
  590. self increaseBlockIndex.
  591. super visitBlockNode: aNode ]
  592. !
  593. visitJSStatementNode: aNode
  594. currentNode := aNode
  595. !
  596. visitSendNode: aNode
  597. super visitSendNode: aNode.
  598. self pc = self context pc ifFalse: [
  599. aNode shouldBeInlined ifFalse: [
  600. self blockIndex = self context index ifTrue: [
  601. self pc: self pc + 1.
  602. currentNode := aNode ] ] ]
  603. ! !
  604. NodeVisitor subclass: #Interpreter
  605. instanceVariableNames: 'node context stack returnValue'
  606. package: 'Compiler-Interpreter'!
  607. !Interpreter methodsFor: 'accessing'!
  608. context
  609. ^ context
  610. !
  611. context: aContext
  612. context := aContext
  613. !
  614. node
  615. "Answer the next node, ie the node to be evaluated in the next step"
  616. ^ node
  617. !
  618. node: aNode
  619. node := aNode
  620. !
  621. result
  622. ^ self returnValue ifNil: [ self context receiver ]
  623. !
  624. returnValue
  625. ^ returnValue
  626. !
  627. returnValue: anObject
  628. returnValue := anObject
  629. !
  630. stack
  631. ^ stack ifNil: [ stack := OrderedCollection new ]
  632. ! !
  633. !Interpreter methodsFor: 'interpreting'!
  634. interpret
  635. "Interpret the next node to be evaluated"
  636. self visit: self node
  637. !
  638. interpret: aNode
  639. self node: aNode.
  640. self interpret
  641. !
  642. next
  643. self node: self node nextNode
  644. !
  645. proceed
  646. "Eagerly evaluate the ast"
  647. [ self atEnd ] whileFalse: [
  648. self step ]
  649. !
  650. restart
  651. self node: self context ast nextChild
  652. !
  653. skip
  654. self next
  655. !
  656. step
  657. self
  658. interpret;
  659. next
  660. !
  661. stepOver
  662. self step.
  663. [ self node isSteppingNode ] whileFalse: [
  664. self step ]
  665. ! !
  666. !Interpreter methodsFor: 'private'!
  667. assign: aNode to: anObject
  668. aNode binding isInstanceVar
  669. ifTrue: [ self context receiver instVarAt: aNode value put: anObject ]
  670. ifFalse: [ self context localAt: aNode value put: anObject ]
  671. !
  672. eval: aString
  673. "Evaluate aString as JS source inside an JS function.
  674. aString is not sandboxed."
  675. | source function |
  676. source := String streamContents: [ :str |
  677. str nextPutAll: '(function('.
  678. self context locals keys
  679. do: [ :each | str nextPutAll: each ]
  680. separatedBy: [ str nextPutAll: ',' ].
  681. str
  682. nextPutAll: '){ return (function() {';
  683. nextPutAll: aString;
  684. nextPutAll: '})() })' ].
  685. function := Compiler new eval: source.
  686. ^ function valueWithPossibleArguments: self context locals values
  687. !
  688. messageFromSendNode: aSendNode arguments: aCollection
  689. ^ Message new
  690. selector: aSendNode selector;
  691. arguments: aCollection;
  692. yourself
  693. !
  694. messageNotUnderstood: aMessage receiver: anObject
  695. MessageNotUnderstood new
  696. meesage: aMessage;
  697. receiver: anObject;
  698. signal
  699. !
  700. sendMessage: aMessage to: anObject superSend: aBoolean
  701. | method |
  702. aBoolean ifFalse: [ ^ aMessage sendTo: anObject ].
  703. anObject class superclass ifNil: [ ^ self messageNotUnderstood: aMessage receiver: anObject ].
  704. method := anObject class superclass methodDictionary
  705. at: aMessage selector
  706. ifAbsent: [ ^ self messageNotUnderstood: aMessage receiver: anObject ].
  707. ^ method sendTo: anObject arguments: aMessage arguments
  708. ! !
  709. !Interpreter methodsFor: 'stack'!
  710. peek
  711. "Peek the top object of the context stack"
  712. ^ self stack last
  713. !
  714. pop
  715. "Pop an object from the context stack"
  716. | value |
  717. value := self peek.
  718. self stack removeLast.
  719. ^ value
  720. !
  721. push: anObject
  722. "Push an object to the context stack"
  723. ^ self stack add: anObject
  724. ! !
  725. !Interpreter methodsFor: 'testing'!
  726. atEnd
  727. ^ self shouldReturn or: [ self node isNil ]
  728. !
  729. shouldReturn
  730. ^ self returnValue notNil
  731. ! !
  732. !Interpreter methodsFor: 'visiting'!
  733. visit: aNode
  734. self shouldReturn ifFalse: [ super visit: aNode ]
  735. !
  736. visitAssignmentNode: aNode
  737. | value |
  738. value := self pop.
  739. "Pop the left side of the assignment.
  740. It already has been visited, and we don't need its value."
  741. self pop.
  742. self push: value.
  743. self assign: aNode left to: value
  744. !
  745. visitBlockNode: aNode
  746. "Do not evaluate the block node.
  747. Instead, put all instructions into a block that we push to the stack for later evaluation"
  748. | blockContext block |
  749. block := AIBlockClosure forInterpreter: self context: self context node: aNode.
  750. self push: block
  751. !
  752. visitClassReferenceNode: aNode
  753. self push: (Smalltalk current at: aNode value)
  754. !
  755. visitDynamicArrayNode: aNode
  756. | array |
  757. array := #().
  758. aNode nodes do: [ :each |
  759. array addFirst: self pop ].
  760. self push: array
  761. !
  762. visitDynamicDictionaryNode: aNode
  763. | hashedCollection |
  764. hashedCollection := HashedCollection new.
  765. aNode nodes do: [ :each |
  766. hashedCollection add: self pop ].
  767. self push: hashedCollection
  768. !
  769. visitJSStatementNode: aNode
  770. self returnValue: (self eval: aNode source)
  771. !
  772. visitNode: aNode
  773. "Do nothing by default. Especially, do not visit children recursively."
  774. !
  775. visitReturnNode: aNode
  776. self returnValue: self pop
  777. !
  778. visitSendNode: aNode
  779. | receiver args message result |
  780. args := aNode arguments collect: [ :each | self pop ].
  781. receiver := self pop.
  782. message := self
  783. messageFromSendNode: aNode
  784. arguments: args reversed.
  785. result := self sendMessage: message to: receiver superSend: aNode superSend.
  786. self context pc: self context pc + 1.
  787. "For cascade sends, push the reciever if the send is not the last one"
  788. (aNode isCascadeSendNode and: [ aNode isLastChild not ])
  789. ifTrue: [ self push: receiver ]
  790. ifFalse: [ self push: result ]
  791. !
  792. visitValueNode: aNode
  793. self push: aNode value
  794. !
  795. visitVariableNode: aNode
  796. aNode binding isUnknownVar ifTrue: [
  797. ^ self push: (window at: aNode value ifAbsent: [ self error: 'Unknown variable' ]) ].
  798. self push: (aNode binding isInstanceVar
  799. ifTrue: [ self context receiver instVarAt: aNode value ]
  800. ifFalse: [ self context localAt: aNode value ])
  801. ! !
  802. !Node methodsFor: '*Compiler-Interpreter'!
  803. interpreter: anInterpreter continue: aBlock
  804. ^ anInterpreter interpretNode: self continue: aBlock
  805. !
  806. isSteppingNode
  807. ^ false
  808. ! !
  809. !AssignmentNode methodsFor: '*Compiler-Interpreter'!
  810. interpreter: anInterpreter continue: aBlock
  811. ^ anInterpreter interpretAssignmentNode: self continue: aBlock
  812. !
  813. isSteppingNode
  814. ^ true
  815. ! !
  816. !BlockNode methodsFor: '*Compiler-Interpreter'!
  817. interpreter: anInterpreter continue: aBlock
  818. ^ anInterpreter interpretBlockNode: self continue: aBlock
  819. !
  820. isSteppingNode
  821. ^ true
  822. ! !
  823. !CascadeNode methodsFor: '*Compiler-Interpreter'!
  824. interpreter: anInterpreter continue: aBlock
  825. ^ anInterpreter interpretCascadeNode: self continue: aBlock
  826. ! !
  827. !DynamicArrayNode methodsFor: '*Compiler-Interpreter'!
  828. interpreter: anInterpreter continue: aBlock
  829. ^ anInterpreter interpretDynamicArrayNode: self continue: aBlock
  830. !
  831. isSteppingNode
  832. ^ true
  833. ! !
  834. !DynamicDictionaryNode methodsFor: '*Compiler-Interpreter'!
  835. interpreter: anInterpreter continue: aBlock
  836. ^ anInterpreter interpretDynamicDictionaryNode: self continue: aBlock
  837. !
  838. isSteppingNode
  839. ^ true
  840. ! !
  841. !JSStatementNode methodsFor: '*Compiler-Interpreter'!
  842. interpreter: anInterpreter continue: aBlock
  843. ^ anInterpreter interpretJSStatementNode: self continue: aBlock
  844. !
  845. isSteppingNode
  846. ^ true
  847. ! !
  848. !MethodNode methodsFor: '*Compiler-Interpreter'!
  849. interpreter: anInterpreter continue: aBlock
  850. ^ anInterpreter interpretMethodNode: self continue: aBlock
  851. ! !
  852. !ReturnNode methodsFor: '*Compiler-Interpreter'!
  853. interpreter: anInterpreter continue: aBlock
  854. ^ anInterpreter interpretReturnNode: self continue: aBlock
  855. ! !
  856. !SendNode methodsFor: '*Compiler-Interpreter'!
  857. interpreter: anInterpreter continue: aBlock
  858. ^ anInterpreter interpretSendNode: self continue: aBlock
  859. !
  860. isSteppingNode
  861. ^ true
  862. ! !
  863. !SequenceNode methodsFor: '*Compiler-Interpreter'!
  864. interpreter: anInterpreter continue: aBlock
  865. ^ anInterpreter interpretSequenceNode: self continue: aBlock
  866. ! !
  867. !BlockSequenceNode methodsFor: '*Compiler-Interpreter'!
  868. interpreter: anInterpreter continue: aBlock
  869. ^ anInterpreter interpretBlockSequenceNode: self continue: aBlock
  870. ! !
  871. !ValueNode methodsFor: '*Compiler-Interpreter'!
  872. interpreter: anInterpreter continue: aBlock
  873. ^ anInterpreter interpretValueNode: self continue: aBlock
  874. ! !
  875. !VariableNode methodsFor: '*Compiler-Interpreter'!
  876. interpreter: anInterpreter continue: aBlock
  877. ^ anInterpreter interpretVariableNode: self continue: aBlock
  878. ! !
  879. !ClassReferenceNode methodsFor: '*Compiler-Interpreter'!
  880. interpreter: anInterpreter continue: aBlock
  881. ^ anInterpreter interpretClassReferenceNode: self continue: aBlock
  882. ! !