Merged JK's version from Monticello
Name: PetitParser-JanKurs.260
Author: JanKurs
Time: 17-11-2014, 12:09:05.490 PM
UUID: 07411cef-ef69-40db-9d93-d4018a9b34ef
Name: PetitTests-JanKurs.65
Author: JanKurs
Time: 17-11-2014, 12:09:04.530 PM
UUID: f98d613f-f4ce-4e0e-a7e9-310ee7c7e7a6
Name: PetitSmalltalk-JanKurs.78
Author: JanKurs
Time: 14-11-2014, 05:05:07.765 PM
UUID: 3d68330d-44d5-46c3-9705-97f627b3edbc
Name: PetitCompiler-JanKurs.71
Author: JanKurs
Time: 18-11-2014, 09:48:35.425 AM
UUID: 06352c33-3c76-4382-8536-0cc48e225117
Name: PetitCompiler-Tests-JanKurs.21
Author: JanKurs
Time: 17-11-2014, 05:51:53.134 PM
UUID: 8d6c0799-14e7-4871-8d91-8b0f9886db83
Name: PetitCompiler-Benchmarks-JanKurs.2
Author: JanKurs
Time: 17-11-2014, 05:51:07.887 PM
UUID: d5e3a980-7871-487a-a232-e3ca93fc2483
"{ Package: 'stx:goodies/petitparser/compiler' }"
PPCListNode subclass:#PPCSequenceNode
instanceVariableNames:''
classVariableNames:''
poolDictionaries:''
category:'PetitCompiler-Nodes'
!
!PPCSequenceNode methodsFor:'accessing'!
prefix
^ #seq
! !
!PPCSequenceNode methodsFor:'analysis'!
acceptsEpsilon
^ self acceptsEpsilonOpenSet: IdentitySet new.
!
acceptsEpsilonOpenSet: set
set add: self.
^ self children allSatisfy: [:e | e acceptsEpsilonOpenSet: set ]
!
firstSetSuchThat: block into: aCollection openSet: aSet
(aSet includes: self) ifTrue: [ ^ aCollection ].
aSet add: self.
(block value: self) ifTrue: [ aCollection add: self. ^ aCollection ].
self children do: [ :child |
child firstSetSuchThat: block into: aCollection openSet: aSet.
child acceptsEpsilon ifFalse: [ ^ aCollection ]
].
^ aCollection
!
firstSets: aFirstDictionary into: aSet suchThat: aBlock
| nullable |
"TODO JK: aBlock is ignored by now"
children do: [ :node |
nullable := false.
(aFirstDictionary at: node) do: [ :each |
each isNullable
ifTrue: [ nullable := true ]
ifFalse: [ aSet add: each ] ].
nullable
ifFalse: [ ^ self ] ].
aSet add: PPCSentinelNode instance
! !
!PPCSequenceNode methodsFor:'compiling'!
addGuard: compiler id: id
| guard firsts |
(compiler guards not or: [(guard := PPCGuard on: self) makesSense not]) ifTrue: [ ^ self].
firsts := (self firstSetSuchThat: [ :e | (e isKindOf: PPCTrimmingTokenNode) or: [ e isTerminal ] ]).
(firsts allSatisfy: [ :e | e isKindOf: PPCTrimmingTokenNode ]) ifTrue: [
"If we start with trimming, we should invoke the whitespace parser"
firsts anyOne compileWhitespace: compiler.
compiler add: 'context atEnd ifTrue: [ ^ self error ].'.
guard id: id, '_guard'.
guard compileGuard: compiler.
compiler addOnLine: 'ifFalse: [ ^ self error ].'
].
(firsts allSatisfy: [ :e | e isTerminal ]) ifTrue: [
compiler add: 'context atEnd ifTrue: [ ^ self error ].'.
guard id: id, '_guard'.
guard compileGuard: compiler.
compiler addOnLine: 'ifFalse: [ ^ self error ].'
].
!
compileWith: compiler effect: effect id: id
compiler startMethod: id.
compiler addVariable: 'retval'.
compiler addVariable: 'element'.
compiler addVariable: 'memento'.
compiler add: (compiler smartRemember: self).
compiler add: 'retval := Array new: ', children size asString, '.'.
self addGuard: compiler id: id.
" Halt if: [ self name = #qualifiedName ]."
(1 to: children size) do: [ :idx | |child|
child := children at: idx.
compiler add: 'element := '.
compiler callOnLine: (child compileWith: compiler).
compiler add: 'error ifTrue: [ ', (compiler smartRestore: self) ,' ^ failure ].'.
compiler add: 'retval at: ', idx asString, ' put: element.'.
].
compiler add: '^ retval'.
^ compiler stopMethod.
! !
!PPCSequenceNode methodsFor:'first follow next'!
followSets: aFollowDictionary firstSets: aFirstDictionary into: aSet suchThat: aBlock
children keysAndValuesDo: [ :index :node |
| follow first |
follow := aFollowDictionary at: node.
index = children size
ifTrue: [ follow addAll: aSet ]
ifFalse: [
(self class withAll: (children
copyFrom: index + 1 to: children size))
firstSets: aFirstDictionary
into: (first := IdentitySet new)
suchThat: aBlock.
(first anySatisfy: [ :each | each isNullable ])
ifTrue: [ follow addAll: aSet ].
follow addAll: (first
reject: [ :each | each isNullable ]) ] ]
! !
!PPCSequenceNode methodsFor:'optimizing'!
asFast
^ PPCTokenSequenceNode new
children: children;
name: self name;
properties: properties;
yourself
! !