Parser.st
author claus
Mon, 10 Oct 1994 01:58:23 +0100
changeset 45 e8331ba8ad5d
parent 44 74ddc944c27f
child 47 f861ad42703e
permissions -rw-r--r--
*** empty log message ***
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
     1
"
4
f6fd83437415 *** empty log message ***
claus
parents: 3
diff changeset
     2
 COPYRIGHT (c) 1989 by Claus Gittinger
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
     3
	      All Rights Reserved
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
     4
7ad01559b262 Initial revision
claus
parents:
diff changeset
     5
 This software is furnished under a license and may be used
7ad01559b262 Initial revision
claus
parents:
diff changeset
     6
 only in accordance with the terms of that license and with the
7ad01559b262 Initial revision
claus
parents:
diff changeset
     7
 inclusion of the above copyright notice.   This software may not
7ad01559b262 Initial revision
claus
parents:
diff changeset
     8
 be provided or otherwise made available to, or used by, any
7ad01559b262 Initial revision
claus
parents:
diff changeset
     9
 other person.  No title to or ownership of the software is
7ad01559b262 Initial revision
claus
parents:
diff changeset
    10
 hereby transferred.
7ad01559b262 Initial revision
claus
parents:
diff changeset
    11
"
7ad01559b262 Initial revision
claus
parents:
diff changeset
    12
7ad01559b262 Initial revision
claus
parents:
diff changeset
    13
Scanner subclass:#Parser
7ad01559b262 Initial revision
claus
parents:
diff changeset
    14
       instanceVariableNames:'classToCompileFor selfValue
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
    15
			      contextToEvaluateIn
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
    16
			      selector
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
    17
			      methodArgs methodArgNames 
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
    18
			      methodVars methodVarNames 
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
    19
			      tree
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
    20
			      currentBlock
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
    21
			      usedInstVars usedClassVars usedVars
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
    22
			      modifiedInstVars modifiedClassVars
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
    23
			      usesSuper  
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
    24
			      localVarDefPosition
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
    25
			      evalExitBlock
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
    26
			      selfNode superNode 
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
    27
			      hasPrimitiveCode primitiveNr logged
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
    28
			      warnedUndefVars
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
    29
			      correctedSource'
10
73e97b6175c4 *** empty log message ***
claus
parents: 9
diff changeset
    30
       classVariableNames:'PrevClass PrevInstVarNames 
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
    31
			   PrevClassVarNames PrevClassInstVarNames
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
    32
			   LazyCompilation'
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
    33
       poolDictionaries:''
7ad01559b262 Initial revision
claus
parents:
diff changeset
    34
       category:'System-Compiler'
7ad01559b262 Initial revision
claus
parents:
diff changeset
    35
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
    36
7ad01559b262 Initial revision
claus
parents:
diff changeset
    37
Parser comment:'
4
f6fd83437415 *** empty log message ***
claus
parents: 3
diff changeset
    38
COPYRIGHT (c) 1989 by Claus Gittinger
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
    39
	     All Rights Reserved
23
1f7bcfff8d39 *** empty log message ***
claus
parents: 21
diff changeset
    40
45
e8331ba8ad5d *** empty log message ***
claus
parents: 44
diff changeset
    41
$Header: /cvs/stx/stx/libcomp/Parser.st,v 1.16 1994-10-10 00:56:40 claus Exp $
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
    42
'!
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
    43
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
    44
!Parser class methodsFor:'documentation'!
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
    45
20
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
    46
copyright
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
    47
"
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
    48
 COPYRIGHT (c) 1989 by Claus Gittinger
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
    49
	      All Rights Reserved
20
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
    50
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
    51
 This software is furnished under a license and may be used
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
    52
 only in accordance with the terms of that license and with the
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
    53
 inclusion of the above copyright notice.   This software may not
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
    54
 be provided or otherwise made available to, or used by, any
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
    55
 other person.  No title to or ownership of the software is
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
    56
 hereby transferred.
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
    57
"
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
    58
!
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
    59
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
    60
version
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
    61
"
45
e8331ba8ad5d *** empty log message ***
claus
parents: 44
diff changeset
    62
$Header: /cvs/stx/stx/libcomp/Parser.st,v 1.16 1994-10-10 00:56:40 claus Exp $
20
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
    63
"
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
    64
!
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
    65
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
    66
documentation
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
    67
"
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
    68
    Parser is used for both parsing and evaluating smalltalk expressions;
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
    69
    it first builds a parseTree which is then interpreted (evaluate) or
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
    70
    compiled. Compilation is done in the subclass ByteCodeCompiler and/or
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
    71
    the (planned) MachineCodeCompiler.
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
    72
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
    73
    methods of main interrest are:
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
    74
	Parser evaluateExpression:...
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
    75
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
    76
    and:
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
    77
	Parser parseExpression:...
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
    78
	Parser parseMethod:...
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
    79
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
    80
    there is protocol to parse complete methods, selector specs, body only etc.
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
    81
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
    82
    Parser is also used to find the referenced/modified inst/classvars of
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
    83
    a method - this is done by sending parseXXX message to a parser and asking
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
    84
    the parser for referencedXVars or modifiedXVars (see SystemBrowser).
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
    85
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
    86
    You can also use parsers for all kinds of other things (ChangesBrowser for
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
    87
    example analyzes the expressions in the changelist ...) by looking at the
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
    88
    parsers tree. (Although this is somewhat dangerous, since it exports the
20
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
    89
    compilers internals ... better style is to add specialized query methods here,
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
    90
    which will be done incrementally.)
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
    91
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
    92
    One instance of Parser is created to parse one method or expression - i.e.
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
    93
    its not suggested to reuse parsers.
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
    94
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
    95
    Instance variables:
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
    96
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
    97
	classToCompileFor   <Class>             the class (or nil) we are compiling for
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
    98
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
    99
	selfValue           <any>               value to use as self when interpreting
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   100
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   101
	contextToEvaluateIn <Context>           the context (or nil) when interpreting
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   102
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   103
	selector            <Symbol>            the selector of the parsed method
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   104
						(valid after parseMethodSpecification)
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   105
	methodArgs                              internal
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   106
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   107
	methodArgNames      <Collection>        the names of the arguments
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   108
						(valid after parseMethodSpecification)
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   109
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   110
	methodVars                              internal
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   111
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   112
	methodVarNames      <Collection>        the names of the method locals
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   113
						(valid after parseMethodBodyVarSpec)
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   114
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   115
	tree                <ParseTree>         the parse tree - valid after parsing
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   116
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   117
	currentBlock                            if currently parsing for a block
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   118
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   119
	usedInstVars                            set of all accessed instances variables
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   120
						(valid after parsing)
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   121
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   122
	usedClassVars                           same for classVars
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   123
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   124
	usedVars                                all used variables (inst, class & globals)
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   125
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   126
	modifiedInstVars                        set of all modified instance variables
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   127
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   128
	modifiedClassVars                       same for clasVars
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   129
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   130
	localVarDefPosition <Integer>           the character offset of the local variable
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   131
						def. (i.e. the first '|' if any)
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   132
						Not yet used - prepared for automatic add of
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   133
						undefined variables
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   134
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   135
	evalExitBlock                           internal for interpretation
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   136
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   137
	selfNode            <Node>              cached one-and-only 'self' node
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   138
	superNode           <Node>              cached one-and-only 'super' node
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   139
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   140
	hasPrimitiveCode    <Boolean>           true, if it contains ST/X style primitive code
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   141
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   142
	primitiveNr         <Integer>           the parsed ST-80 type primitive number (or nil)
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   143
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   144
	logged
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   145
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   146
	warnedUndefVars     <Set>               set of all variables which the parser has
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   147
						already output a warning (to avoid multiple
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   148
						warnings about the same variable)
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   149
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   150
    Class variables:
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   151
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   152
	PrevClass           <Class>             class, of which properties are
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   153
						cached in:
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   154
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   155
	PrevInstVarNames      <Collection>      instance variablenames of cached class
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   156
	PrevClassVarNames     <Collection>      class variablenames of cached class
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   157
	PrevClassInstVarNames <Collection>      class instance variablenames of cached class
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   158
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   159
	LazyCompilation     <Boolean>           EXPERIMENTAL: lazy compilation
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   160
"
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   161
! !
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   162
7ad01559b262 Initial revision
claus
parents:
diff changeset
   163
!Parser class methodsFor:'evaluating expressions'!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   164
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   165
evaluate:aStringOrStream
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   166
    "return the result of evaluating an expression in aStringOrStream"
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   167
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   168
    ^ self 
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   169
	evaluate:aStringOrStream 
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   170
	in:nil 
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   171
	receiver:nil 
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   172
	notifying:nil 
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   173
	logged:false
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   174
	ifFail:nil 
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   175
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   176
    "
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   177
     Compiler evaluate:'1 + 2'
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   178
     Compiler evaluate:'''hello world'' asSortedCollection displayString printNL'
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   179
     Compiler evaluate:'''hello world'' asSortedCollection printNL'
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   180
    "
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   181
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   182
20
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   183
evaluate:aStringOrStream ifFail:failBlock
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   184
    "return the result of evaluating an expression in aStringOrStream.
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   185
     In case of any syntax errors, return the value of failBlock."
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   186
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   187
    ^ self 
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   188
	evaluate:aStringOrStream 
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   189
	in:nil 
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   190
	receiver:nil 
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   191
	notifying:nil 
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   192
	logged:false
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   193
	ifFail:failBlock 
20
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   194
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   195
    "
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   196
     Compiler evaluate:'1 +' ifFail:['oops']   
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   197
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   198
    "
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   199
!
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   200
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   201
evaluate:aStringOrStream logged:logged
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   202
    "return the result of evaluating an expression in aStringOrStream"
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   203
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   204
    ^ self 
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   205
	evaluate:aStringOrStream 
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   206
	in:nil 
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   207
	receiver:nil 
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   208
	notifying:nil 
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   209
	logged:logged
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   210
	ifFail:nil 
20
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   211
!
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   212
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   213
evaluate:aStringOrStream notifying:requestor
7ad01559b262 Initial revision
claus
parents:
diff changeset
   214
    "return the result of evaluating aString, 
7ad01559b262 Initial revision
claus
parents:
diff changeset
   215
     errors are reported to requestor"
7ad01559b262 Initial revision
claus
parents:
diff changeset
   216
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   217
    ^ self 
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   218
	evaluate:aStringOrStream 
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   219
	in:nil 
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   220
	receiver:nil 
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   221
	notifying:requestor
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   222
	logged:false
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   223
	ifFail:nil 
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   224
7ad01559b262 Initial revision
claus
parents:
diff changeset
   225
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   226
20
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   227
evaluate:aStringOrStream receiver:anObject notifying:requestor
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   228
    "return the result of evaluating aString, 
7ad01559b262 Initial revision
claus
parents:
diff changeset
   229
     errors are reported to requestor. Allow access to
7ad01559b262 Initial revision
claus
parents:
diff changeset
   230
     anObject as self and to its instVars (used in the inspector)"
7ad01559b262 Initial revision
claus
parents:
diff changeset
   231
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   232
    ^ self 
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   233
	evaluate:aStringOrStream
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   234
	in:nil
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   235
	receiver:anObject
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   236
	notifying:requestor
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   237
	logged:false
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   238
	ifFail:nil
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   239
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   240
7ad01559b262 Initial revision
claus
parents:
diff changeset
   241
evaluate:aStringOrStream in:aContext receiver:anObject 
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   242
				    notifying:requestor
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   243
				       ifFail:failBlock
20
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   244
    ^ self 
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   245
	evaluate:aStringOrStream
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   246
	in:nil
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   247
	receiver:anObject
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   248
	notifying:requestor
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   249
	logged:false
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   250
	ifFail:nil
20
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   251
!
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   252
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   253
evaluate:aStringOrStream in:aContext receiver:anObject notifying:requestor logged:logged ifFail:failBlock
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   254
    "return the result of evaluating aStringOrStream, errors are reported to requestor. 
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   255
     Allow access to anObject as self and to its instVars (used in the inspector).
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   256
     If logged is true, an entry is added to the change-file. If the failBlock argument
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   257
     is non-nil, it is evaluated if an error occurs."
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   258
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   259
    "XXX: logging is not yet implemented"
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   260
35
2884eed75e2a logging doits.
claus
parents: 23
diff changeset
   261
    |parser tree mustBackup loggedString chgStream value|
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   262
7ad01559b262 Initial revision
claus
parents:
diff changeset
   263
    aStringOrStream isNil ifTrue:[^ nil].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   264
    aStringOrStream isStream ifTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   265
	parser := self for:aStringOrStream.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   266
	mustBackup := true
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   267
    ] ifFalse:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   268
	loggedString := aStringOrStream.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   269
	parser := self for:(ReadStream on:aStringOrStream).
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   270
	mustBackup := false
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   271
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   272
    parser setSelf:anObject.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   273
    parser setContext:aContext.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   274
    parser notifying:requestor.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   275
    parser nextToken.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   276
    tree := parser parseMethodBodyOrNil.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   277
7ad01559b262 Initial revision
claus
parents:
diff changeset
   278
    "if reading from a stream, backup for next expression"
7ad01559b262 Initial revision
claus
parents:
diff changeset
   279
    mustBackup ifTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   280
	parser backupPosition
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   281
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   282
7ad01559b262 Initial revision
claus
parents:
diff changeset
   283
    (parser errorFlag or:[tree == #Error]) ifTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   284
	failBlock notNil ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   285
	    ^ failBlock value
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   286
	].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   287
	^ #Error
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   288
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   289
    tree notNil ifTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   290
	(logged
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   291
	and:[loggedString notNil
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   292
	and:[Smalltalk logDoits]]) ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   293
	    chgStream := Class changesStream.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   294
	    chgStream notNil ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   295
		chgStream nextChunkPut:loggedString.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   296
		chgStream cr.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   297
		chgStream close
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   298
	    ].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   299
	].
35
2884eed75e2a logging doits.
claus
parents: 23
diff changeset
   300
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   301
	parser evalExitBlock:[:value | parser release. ^ value].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   302
	value := tree evaluate
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   303
    ].
35
2884eed75e2a logging doits.
claus
parents: 23
diff changeset
   304
    parser release.
2884eed75e2a logging doits.
claus
parents: 23
diff changeset
   305
    ^ value
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   306
! !
7ad01559b262 Initial revision
claus
parents:
diff changeset
   307
7ad01559b262 Initial revision
claus
parents:
diff changeset
   308
!Parser class methodsFor:'instance creation'!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   309
21
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
   310
for:aStringOrStream in:aClass
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
   311
    "return a new parser, reading code for aClass from aStringOrStream"
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   312
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   313
    |parser|
7ad01559b262 Initial revision
claus
parents:
diff changeset
   314
21
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
   315
    parser := self for:aStringOrStream.
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   316
    parser setClassToCompileFor:aClass.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   317
    ^ parser
7ad01559b262 Initial revision
claus
parents:
diff changeset
   318
! !
7ad01559b262 Initial revision
claus
parents:
diff changeset
   319
7ad01559b262 Initial revision
claus
parents:
diff changeset
   320
!Parser class methodsFor:'parsing'!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   321
7
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   322
selectorInExpression:aString
21
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
   323
    "parse an expression - return the selector. 
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
   324
     Used for SystemBrowsers implementors/senders query-box initial text.
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
   325
     Returns nil if unparsable."
7
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   326
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   327
    |tree parser|
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   328
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   329
    (aString isNil or:[aString isEmpty]) ifTrue:[^ nil].
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   330
21
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
   331
    tree := self withSelf:nil 
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   332
		 parseExpression:aString 
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   333
		 notifying:nil 
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   334
		 ignoreErrors:true 
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   335
		 ignoreWarnings:true. 
21
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
   336
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
   337
    "
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
   338
     special: take the expression of the right side, if its an
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
   339
     assignment or return
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
   340
    "
7
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   341
    (tree notNil and:[tree ~~ #Error]) ifTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   342
	tree isAssignment ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   343
	    tree expression isMessage ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   344
		tree := tree expression
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   345
	    ]
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   346
	].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   347
	tree isReturnNode ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   348
	    tree expression isMessage ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   349
		tree := tree expression
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   350
	    ]
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   351
	].
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   352
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   353
	tree isMessage ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   354
	    ^ tree selector
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   355
	].
7
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   356
    ].
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   357
21
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
   358
    "
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
   359
     mhmh, try expression without receiver
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
   360
    "
7
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   361
    parser := self for:(ReadStream on:aString).
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   362
    parser ignoreErrors.
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   363
    parser nextToken.
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   364
    ^ parser degeneratedKeywordExpressionForSelector
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   365
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   366
"
20
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   367
    Parser selectorInExpression:'foo at:1 put:(5 * bar)'     
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   368
    Parser selectorInExpression:'(foo at:1) at:1'           
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   369
    Parser selectorInExpression:'1 + 4'                     
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   370
    Parser selectorInExpression:'1 negated'                 
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   371
    Parser selectorInExpression:'at:1 put:5'            
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   372
    Parser selectorInExpression:'a := foo at:1 put:5'    
7
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   373
"
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   374
!
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   375
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   376
parseExpression:aString
21
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
   377
    "parse aString as an expression; 
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
   378
     Return the parseTree (if ok), nil (for an empty string 
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
   379
     or comment only) or #Error (syntactic error)."
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   380
21
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
   381
    ^ self withSelf:nil 
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   382
	   parseExpression:aString 
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   383
	   notifying:nil 
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   384
	   ignoreErrors:true       "silence on Transcript"
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   385
	   ignoreWarnings:true 
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   386
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   387
7
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   388
withSelf:anObject parseExpression:aString notifying:someOne 
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   389
    "parse aString as an expression with self set to anObject;
21
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
   390
     Return the parseTree (if ok), nil (for an empty string 
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
   391
     or comment only ) or #Error (syntactic error).
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
   392
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
   393
     Errors and warnings are forwarded to someOne (usually some
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
   394
     codeView) which can highlight it and show a popup box."
7
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   395
21
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
   396
    ^ self withSelf:anObject 
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   397
	   parseExpression:aString 
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   398
	   notifying:someOne 
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   399
	   ignoreErrors:false 
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   400
	   ignoreWarnings:false 
7
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   401
!
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   402
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   403
withSelf:anObject parseExpression:aString notifying:someOne ignoreErrors:ignore
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   404
    "parse aString as an expression with self set to anObject;
21
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
   405
     Return the parseTree (if ok), nil (for an empty string 
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
   406
     or comment only ) or #Error (syntactic error).
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
   407
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
   408
     Errors and warnings are forwarded to someOne (usually some
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
   409
     codeView) which can highlight it and show a popup box."
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
   410
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
   411
    ^ self withSelf:anObject
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   412
	   parseExpression:aString 
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   413
	   notifying:someOne 
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   414
	   ignoreErrors:ignore 
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   415
	   ignoreWarnings:ignore 
21
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
   416
!
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
   417
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
   418
withSelf:anObject parseExpression:aString notifying:someOne ignoreErrors:ignoreErrors ignoreWarnings:ignoreWarnings
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
   419
    "parse aString as an expression with self set to anObject;
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
   420
     Return the parseTree (if ok), nil (for an empty string 
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
   421
     or comment only ) or #Error (syntactic error).
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
   422
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
   423
     Errors and warnings are forwarded to someOne (usually some
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
   424
     codeView) which can highlight it and show a popup box,
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
   425
     iff ignoreErrors/ignoreWarnings is true respectively."
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   426
35
2884eed75e2a logging doits.
claus
parents: 23
diff changeset
   427
    |parser tree token|
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   428
7ad01559b262 Initial revision
claus
parents:
diff changeset
   429
    aString isNil ifTrue:[^ nil].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   430
    parser := self for:(ReadStream on:aString).
7ad01559b262 Initial revision
claus
parents:
diff changeset
   431
    parser setSelf:anObject.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   432
    parser notifying:someOne.
21
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
   433
    ignoreErrors ifTrue:[parser ignoreErrors].
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
   434
    ignoreWarnings ifTrue:[parser ignoreWarnings].
35
2884eed75e2a logging doits.
claus
parents: 23
diff changeset
   435
    token := parser nextToken.
2884eed75e2a logging doits.
claus
parents: 23
diff changeset
   436
    (token == $^) ifTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   437
	parser nextToken.
35
2884eed75e2a logging doits.
claus
parents: 23
diff changeset
   438
    ].
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   439
    tree := parser expression.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   440
    (parser errorFlag or:[tree == #Error]) ifTrue:[^ #Error].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   441
    ^ tree
7ad01559b262 Initial revision
claus
parents:
diff changeset
   442
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   443
7ad01559b262 Initial revision
claus
parents:
diff changeset
   444
parseMethodSpecification:aString
7ad01559b262 Initial revision
claus
parents:
diff changeset
   445
    "parse a methods selector & arg specification; 
21
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
   446
     Return a parser (if ok), nil (empty) or #Error (syntax).
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
   447
     The parser can be queried for selector, receiver etc."
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   448
7ad01559b262 Initial revision
claus
parents:
diff changeset
   449
    ^ self parseMethodSpecification:aString in:nil
20
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   450
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   451
    "
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   452
     |p|
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   453
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   454
     p := Parser parseMethodSpecification:'foo:arg1 bar:arg2 baz:arg3'.
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   455
     'nArgs: ' print. p numberOfMethodArgs printNL.
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   456
     'args:  ' print. p methodArgs printNL.
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   457
     'sel:   ' print. p selector printNL
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   458
    "
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   459
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   460
7ad01559b262 Initial revision
claus
parents:
diff changeset
   461
parseMethodSpecification:aString in:aClass
7ad01559b262 Initial revision
claus
parents:
diff changeset
   462
    "parse a methods selector & arg spec for a given class;
21
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
   463
     Return a parser (if ok), nil (empty) or #Error (syntax).
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
   464
     The parser can be queried for selector, receiver etc."
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   465
35
2884eed75e2a logging doits.
claus
parents: 23
diff changeset
   466
    ^ self parseMethodSpecification:aString 
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   467
	   in:aClass 
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   468
	   ignoreErrors:false
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   469
	   ignoreWarnings:false 
35
2884eed75e2a logging doits.
claus
parents: 23
diff changeset
   470
!
2884eed75e2a logging doits.
claus
parents: 23
diff changeset
   471
2884eed75e2a logging doits.
claus
parents: 23
diff changeset
   472
parseMethodSpecification:aString in:aClass ignoreErrors:noErrors ignoreWarnings:noWarnings 
2884eed75e2a logging doits.
claus
parents: 23
diff changeset
   473
    "parse a methods selector & arg spec for a given class;
2884eed75e2a logging doits.
claus
parents: 23
diff changeset
   474
     Return a parser (if ok), nil (empty) or #Error (syntax).
2884eed75e2a logging doits.
claus
parents: 23
diff changeset
   475
     The parser can be queried for selector, receiver etc.
2884eed75e2a logging doits.
claus
parents: 23
diff changeset
   476
     noErrors and noWarnings specify if error- and warningMessages are
2884eed75e2a logging doits.
claus
parents: 23
diff changeset
   477
     to be output onto the Transcript."
2884eed75e2a logging doits.
claus
parents: 23
diff changeset
   478
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   479
    |parser tree|
7ad01559b262 Initial revision
claus
parents:
diff changeset
   480
7ad01559b262 Initial revision
claus
parents:
diff changeset
   481
    aString isNil ifTrue:[^ nil].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   482
    parser := self for:(ReadStream on:aString) in:aClass.
35
2884eed75e2a logging doits.
claus
parents: 23
diff changeset
   483
    noErrors ifTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   484
	parser ignoreErrors
35
2884eed75e2a logging doits.
claus
parents: 23
diff changeset
   485
    ].
2884eed75e2a logging doits.
claus
parents: 23
diff changeset
   486
    noWarnings ifTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   487
	parser ignoreWarnings
35
2884eed75e2a logging doits.
claus
parents: 23
diff changeset
   488
    ].
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   489
    parser nextToken.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   490
    tree := parser parseMethodSpec.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   491
    (parser errorFlag or:[tree == #Error]) ifTrue:[^ #Error].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   492
    ^ parser
7ad01559b262 Initial revision
claus
parents:
diff changeset
   493
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   494
7ad01559b262 Initial revision
claus
parents:
diff changeset
   495
parseMethodArgAndVarSpecification:aString
21
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
   496
    "parse a methods selector, arg and var spec (i.e. locals);
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
   497
     Return a parser (if ok), nil (empty) or #Error (syntax).
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
   498
     The parser can be queried for selector, receiver etc."
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   499
7ad01559b262 Initial revision
claus
parents:
diff changeset
   500
    ^ self parseMethodArgAndVarSpecification:aString in:nil
20
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   501
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   502
    "
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   503
     |p|
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   504
21
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
   505
     p := Parser 
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   506
	     parseMethodArgAndVarSpecification:'
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   507
		      foo:arg1 bar:arg2 baz:arg3 
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   508
		      |l1 l2|'.
21
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
   509
20
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   510
     'nArgs:  ' print. p numberOfMethodArgs printNL.
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   511
     'args:   ' print. p methodArgs printNL.
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   512
     'sel:    ' print. p selector printNL.
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   513
     'nLocal: ' print. p numberOfMethodVars printNL.
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   514
     'locals: ' print. p methodVars printNL.
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   515
    "
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   516
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   517
7ad01559b262 Initial revision
claus
parents:
diff changeset
   518
parseMethodArgAndVarSpecification:aString in:aClass
21
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
   519
    "parse a methods selector, arg and var spec in a given class;
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
   520
     Return a parser (if ok), nil (empty) or #Error (syntax).
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
   521
     The parser can be queried for selector, receiver, args and locals"
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   522
7ad01559b262 Initial revision
claus
parents:
diff changeset
   523
    |parser|
7ad01559b262 Initial revision
claus
parents:
diff changeset
   524
7ad01559b262 Initial revision
claus
parents:
diff changeset
   525
    aString isNil ifTrue:[^ nil].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   526
    parser := self for:(ReadStream on:aString) in:aClass.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   527
    parser nextToken.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   528
    (parser parseMethodSpec == #Error) ifTrue:[^ nil].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   529
    (parser parseMethodBodyVarSpec == #Error) ifTrue:[^ nil].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   530
    parser errorFlag ifTrue:[^ nil].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   531
    ^ parser
7ad01559b262 Initial revision
claus
parents:
diff changeset
   532
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   533
7ad01559b262 Initial revision
claus
parents:
diff changeset
   534
parseMethod:aString
21
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
   535
    "parse a method.
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
   536
     Return a parser (if ok), nil (empty) or #Error (syntax).
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
   537
     The parser can be queried for selector, receiver, args, locals,
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
   538
     used selectors etc."
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   539
7ad01559b262 Initial revision
claus
parents:
diff changeset
   540
    ^ self parseMethod:aString in:nil
20
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   541
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   542
    "
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   543
     |p|
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   544
21
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
   545
     p := Parser 
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   546
	     parseMethod:'
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   547
		 foo:arg1 bar:arg2 baz:arg3 
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   548
		     |l1 l2| 
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   549
		     l1 := 0. 
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   550
		     l2 := arg1. 
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   551
		     ^ self'.
21
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
   552
20
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   553
     'nArgs:  ' print. p numberOfMethodArgs printNL.
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   554
     'args:   ' print. p methodArgs printNL.
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   555
     'sel:    ' print. p selector printNL.
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   556
     'nLocal: ' print. p numberOfMethodVars printNL.
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   557
     'locals: ' print. p methodVars printNL.
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   558
     'tree:   ' printNL. p tree printAllOn:Stdout. Stdout cr.
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   559
    "
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   560
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   561
7ad01559b262 Initial revision
claus
parents:
diff changeset
   562
parseMethod:aString in:aClass
21
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
   563
    "parse a method in a given class.
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
   564
     Return a parser (if ok), nil (empty) or #Error (syntax).
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
   565
     The parser can be queried for selector, receiver, args, locals,
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
   566
     used selectors, modified instvars, referenced classvars etc."
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   567
7ad01559b262 Initial revision
claus
parents:
diff changeset
   568
    |parser tree|
7ad01559b262 Initial revision
claus
parents:
diff changeset
   569
7ad01559b262 Initial revision
claus
parents:
diff changeset
   570
    aString isNil ifTrue:[^ nil].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   571
    parser := self for:(ReadStream on:aString) in:aClass.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   572
    tree := parser parseMethod.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   573
    (parser errorFlag or:[tree == #Error]) ifTrue:[^ nil].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   574
    ^ parser
7ad01559b262 Initial revision
claus
parents:
diff changeset
   575
! !
7ad01559b262 Initial revision
claus
parents:
diff changeset
   576
20
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   577
!Parser class methodsFor:'unparsing'!
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   578
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   579
methodSpecificationForSelector:aSelector
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   580
    "given a selector such as #foo:bar:, return a string that could
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   581
     serve as a methods specification source code.
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   582
     To be used for code generators"
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   583
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   584
    ^ self methodSpecificationForSelector:aSelector 
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   585
				 argNames:#('arg1' 'arg2' 'arg3' 'arg4' 'arg5' 'arg6'
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   586
					    'arg7' 'arg8' 'arg9' 'arg10' 'arg11' 'arg12'
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   587
					    'arg13' 'arg14' 'arg15')
20
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   588
    "
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   589
     Parser methodSpecificationForSelector:#foo:bar:   
21
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
   590
     Parser methodSpecificationForSelector:#+       
20
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   591
     Parser methodSpecificationForSelector:#negated   
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   592
    "
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   593
!
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   594
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   595
methodSpecificationForSelector:aSelector argNames:argNames
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   596
    "given a selector such as #foo:bar:, return a string that could
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   597
     serve as a methods specification source code.
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   598
     To be used for code generators"
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   599
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   600
    |s nargs parts|
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   601
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   602
    s := WriteStream on:String new.
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   603
    nargs := aSelector numArgs.
20
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   604
    nargs == 0 ifTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   605
	s nextPutAll:aSelector
20
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   606
    ] ifFalse:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   607
	parts := aSelector partsIfSelector.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   608
	1 to:nargs do:[:i |
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   609
	    s nextPutAll:(parts at:i); space;
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   610
	      nextPutAll:(argNames at:i); space.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   611
	]
20
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   612
    ].
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   613
    ^ s contents
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   614
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   615
    "
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   616
     Parser methodSpecificationForSelector:#foo:bar: argNames:#('one' 'two' 'three')  
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   617
     Parser methodSpecificationForSelector:#+ argNames:#('one')  
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   618
     Parser methodSpecificationForSelector:#negated   
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   619
    "
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   620
! !
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
   621
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   622
!Parser class methodsFor:'controlling compilation'!
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   623
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   624
compileLazy:aBoolean
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   625
    "turn on/off lazy compilation - return previous setting.
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   626
     Actually this flag belongs into the ByteCodeCompiler subclass,
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   627
     but it also controls the reporting of some errors here; therefore
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   628
     its located here"
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   629
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   630
    |oldLazy|
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   631
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   632
    oldLazy := LazyCompilation.
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   633
    LazyCompilation := aBoolean.
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   634
    ^ oldLazy
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   635
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   636
    "Compiler compileLazy:false"
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   637
! !
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   638
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   639
!Parser methodsFor:'ST-80 compatibility'!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   640
7ad01559b262 Initial revision
claus
parents:
diff changeset
   641
evaluate:aString in:aClass to:to notifying:aRequestor ifFail:failBlock
35
2884eed75e2a logging doits.
claus
parents: 23
diff changeset
   642
    |parseTree value|
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   643
7ad01559b262 Initial revision
claus
parents:
diff changeset
   644
    aString isNil ifTrue:[^ nil].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   645
    self initializeFor:(ReadStream on:aString).
7ad01559b262 Initial revision
claus
parents:
diff changeset
   646
    self setClassToCompileFor:aClass.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   647
    selfValue := nil.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   648
    requestor := aRequestor.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   649
7ad01559b262 Initial revision
claus
parents:
diff changeset
   650
    self nextToken.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   651
    parseTree := self parseMethodBody.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   652
    (errorFlag or:[tree == #Error]) ifTrue:[^ #Error].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   653
    parseTree notNil ifTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   654
	self evalExitBlock:[:value | ^ failBlock value].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   655
	value := parseTree evaluate
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   656
    ].
35
2884eed75e2a logging doits.
claus
parents: 23
diff changeset
   657
    self release.
2884eed75e2a logging doits.
claus
parents: 23
diff changeset
   658
    ^ value
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   659
! !
7ad01559b262 Initial revision
claus
parents:
diff changeset
   660
7ad01559b262 Initial revision
claus
parents:
diff changeset
   661
!Parser class methodsFor:'changes'!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   662
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   663
update:something with:someArgument from:changedObject
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   664
    "aClass has changed its definition - flush name caches if we have to"
7ad01559b262 Initial revision
claus
parents:
diff changeset
   665
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   666
    (changedObject == PrevClass) ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   667
	something == #definition ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   668
	    self flushNameCache
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   669
	]
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   670
    ]
7
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   671
!
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   672
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   673
flushNameCache
7
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   674
    "unconditional flush name caches"
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   675
10
73e97b6175c4 *** empty log message ***
claus
parents: 9
diff changeset
   676
    PrevClass notNil ifTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   677
	PrevClass removeDependent:Parser
7
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   678
    ].
10
73e97b6175c4 *** empty log message ***
claus
parents: 9
diff changeset
   679
    PrevClass := nil.
73e97b6175c4 *** empty log message ***
claus
parents: 9
diff changeset
   680
    PrevInstVarNames := nil.
73e97b6175c4 *** empty log message ***
claus
parents: 9
diff changeset
   681
    PrevClassVarNames := nil.
73e97b6175c4 *** empty log message ***
claus
parents: 9
diff changeset
   682
    PrevClassInstVarNames := nil.
7
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   683
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   684
    "Parser flushNameCache"
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   685
! !
7ad01559b262 Initial revision
claus
parents:
diff changeset
   686
7ad01559b262 Initial revision
claus
parents:
diff changeset
   687
!Parser methodsFor:'setup'!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   688
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   689
initialize
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   690
    super initialize.
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   691
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   692
    hasPrimitiveCode := false
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   693
!
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   694
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   695
initializeFor:aStringOrStream
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   696
    super initializeFor:aStringOrStream.
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   697
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   698
    hasPrimitiveCode := false.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   699
    usesSuper := false
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   700
!
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   701
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   702
setClassToCompileFor:aClass
7ad01559b262 Initial revision
claus
parents:
diff changeset
   703
    "set the class to be used for parsing/evaluating"
7ad01559b262 Initial revision
claus
parents:
diff changeset
   704
7ad01559b262 Initial revision
claus
parents:
diff changeset
   705
    classToCompileFor := aClass.
10
73e97b6175c4 *** empty log message ***
claus
parents: 9
diff changeset
   706
    (classToCompileFor ~~ PrevClass) ifTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   707
	PrevClass notNil ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   708
	    Parser update:PrevClass
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   709
	]
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   710
    ]
7ad01559b262 Initial revision
claus
parents:
diff changeset
   711
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   712
7ad01559b262 Initial revision
claus
parents:
diff changeset
   713
setSelf:anObject
7ad01559b262 Initial revision
claus
parents:
diff changeset
   714
    "set the value to be used for self while evaluating"
7ad01559b262 Initial revision
claus
parents:
diff changeset
   715
7ad01559b262 Initial revision
claus
parents:
diff changeset
   716
    selfValue := anObject.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   717
    classToCompileFor := anObject class.
10
73e97b6175c4 *** empty log message ***
claus
parents: 9
diff changeset
   718
    (classToCompileFor ~~ PrevClass) ifTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   719
	PrevClass notNil ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   720
	    Parser update:PrevClass
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   721
	]
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   722
    ]
7ad01559b262 Initial revision
claus
parents:
diff changeset
   723
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   724
7ad01559b262 Initial revision
claus
parents:
diff changeset
   725
setContext:aContext
7ad01559b262 Initial revision
claus
parents:
diff changeset
   726
    "set the context used while evaluating"
7ad01559b262 Initial revision
claus
parents:
diff changeset
   727
7ad01559b262 Initial revision
claus
parents:
diff changeset
   728
    contextToEvaluateIn := aContext
7ad01559b262 Initial revision
claus
parents:
diff changeset
   729
! !
7ad01559b262 Initial revision
claus
parents:
diff changeset
   730
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   731
!Parser methodsFor:'queries'!
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   732
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   733
usesSuper
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   734
    "return true if the parsed method uses super (valid after parsing)"
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   735
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   736
    ^ usesSuper
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   737
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   738
7ad01559b262 Initial revision
claus
parents:
diff changeset
   739
selector
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   740
    "return the selector (valid after parsing spec)"
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   741
7ad01559b262 Initial revision
claus
parents:
diff changeset
   742
    ^ selector
7ad01559b262 Initial revision
claus
parents:
diff changeset
   743
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   744
7ad01559b262 Initial revision
claus
parents:
diff changeset
   745
numberOfMethodArgs
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   746
    "return the number of methodargs (valid after parsing spec)"
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   747
7ad01559b262 Initial revision
claus
parents:
diff changeset
   748
    ^ methodArgs size
7ad01559b262 Initial revision
claus
parents:
diff changeset
   749
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   750
7ad01559b262 Initial revision
claus
parents:
diff changeset
   751
methodArgs
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   752
    "return an array with methodarg names (valid after parsing spec)"
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   753
7ad01559b262 Initial revision
claus
parents:
diff changeset
   754
    ^ methodArgNames
7ad01559b262 Initial revision
claus
parents:
diff changeset
   755
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   756
7ad01559b262 Initial revision
claus
parents:
diff changeset
   757
numberOfMethodVars
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   758
    "return the number of method variables (valid after parsing)"
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   759
7ad01559b262 Initial revision
claus
parents:
diff changeset
   760
    ^ methodVars size
7ad01559b262 Initial revision
claus
parents:
diff changeset
   761
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   762
7ad01559b262 Initial revision
claus
parents:
diff changeset
   763
methodVars
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   764
    "return a collection with method variablenames (valid after parsing)"
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   765
7ad01559b262 Initial revision
claus
parents:
diff changeset
   766
    ^ methodVarNames
7ad01559b262 Initial revision
claus
parents:
diff changeset
   767
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   768
3
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
   769
usedVars
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   770
    "return a collection with variablenames refd by method (valid after parsing)"
3
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
   771
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
   772
    ^ usedVars
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
   773
!
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
   774
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   775
usedInstVars
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   776
    "return a collection with instvariablenames refd by method (valid after parsing)"
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   777
7ad01559b262 Initial revision
claus
parents:
diff changeset
   778
    ^ usedInstVars
7ad01559b262 Initial revision
claus
parents:
diff changeset
   779
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   780
7ad01559b262 Initial revision
claus
parents:
diff changeset
   781
usedClassVars
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   782
    "return a collection with classvariablenames refd by method (valid after parsing)"
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   783
7ad01559b262 Initial revision
claus
parents:
diff changeset
   784
    ^ usedClassVars
7ad01559b262 Initial revision
claus
parents:
diff changeset
   785
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   786
7ad01559b262 Initial revision
claus
parents:
diff changeset
   787
modifiedInstVars
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   788
    "return a collection with instvariablenames modified by method (valid after parsing)"
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   789
7ad01559b262 Initial revision
claus
parents:
diff changeset
   790
    ^ modifiedInstVars
7ad01559b262 Initial revision
claus
parents:
diff changeset
   791
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   792
7ad01559b262 Initial revision
claus
parents:
diff changeset
   793
modifiedClassVars
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   794
    "return a collection with classvariablenames modified by method (valid after parsing)"
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   795
7ad01559b262 Initial revision
claus
parents:
diff changeset
   796
    ^ modifiedClassVars
7ad01559b262 Initial revision
claus
parents:
diff changeset
   797
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   798
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   799
hasPrimitiveCode
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   800
    "return true if there was any ST/X style primitive code (valid after parsing)"
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   801
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   802
    ^ hasPrimitiveCode
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   803
! !
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   804
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   805
!Parser methodsFor:'accessing'!
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   806
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   807
tree
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   808
    "return the parsetree"
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   809
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   810
    ^tree
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   811
!
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   812
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   813
tree:aTree
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   814
    "private: set the tree - for internal use only"
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   815
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   816
    tree := aTree
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   817
!
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   818
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   819
correctedSource
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   820
    ^ correctedSource
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   821
!
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   822
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   823
primitiveNumber
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   824
    "return the ST-80 style primitiveNumber or nil (valid after parsing)"
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   825
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   826
    ^ primitiveNr
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   827
!
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   828
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   829
errorFlag
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   830
    "return true if there where any errors (valid after parsing)"
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   831
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   832
    ^ errorFlag
7ad01559b262 Initial revision
claus
parents:
diff changeset
   833
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   834
7ad01559b262 Initial revision
claus
parents:
diff changeset
   835
evalExitBlock:aBlock
7ad01559b262 Initial revision
claus
parents:
diff changeset
   836
    "when evaluating a return expression, this block is evaluated"
7ad01559b262 Initial revision
claus
parents:
diff changeset
   837
7ad01559b262 Initial revision
claus
parents:
diff changeset
   838
    evalExitBlock := aBlock
35
2884eed75e2a logging doits.
claus
parents: 23
diff changeset
   839
!
2884eed75e2a logging doits.
claus
parents: 23
diff changeset
   840
2884eed75e2a logging doits.
claus
parents: 23
diff changeset
   841
release
2884eed75e2a logging doits.
claus
parents: 23
diff changeset
   842
    methodArgs := methodVars := tree := selfNode := superNode := nil.
2884eed75e2a logging doits.
claus
parents: 23
diff changeset
   843
    super release.
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   844
! !
7ad01559b262 Initial revision
claus
parents:
diff changeset
   845
7ad01559b262 Initial revision
claus
parents:
diff changeset
   846
!Parser methodsFor:'error handling'!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   847
7ad01559b262 Initial revision
claus
parents:
diff changeset
   848
showErrorMessage:aMessage position:pos
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   849
    "redefined since parser can give more detailed info about
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   850
     the class & selector where the error occured."
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   851
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   852
    ignoreErrors ifFalse:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   853
	Smalltalk silentLoading == true ifFalse:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   854
	    Transcript show:(pos printString).
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   855
	    Transcript show:' '.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   856
	    selector notNil ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   857
		Transcript show:aMessage.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   858
		classToCompileFor notNil ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   859
		    Transcript showCr:(' in ' , classToCompileFor name , '>>' , selector)
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   860
		] ifFalse:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   861
		    Transcript showCr:(' in ' , selector)
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   862
		]
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   863
	    ] ifFalse:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   864
		classToCompileFor notNil ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   865
		    Transcript showCr:aMessage , ' (' , classToCompileFor name , ')'
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   866
		] ifFalse:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   867
		    Transcript showCr:aMessage
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   868
		]
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   869
	    ]
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   870
	]
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   871
    ]
7ad01559b262 Initial revision
claus
parents:
diff changeset
   872
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   873
7ad01559b262 Initial revision
claus
parents:
diff changeset
   874
parseError:aMessage position:position to:endPos
7ad01559b262 Initial revision
claus
parents:
diff changeset
   875
    "report a syntax error"
7ad01559b262 Initial revision
claus
parents:
diff changeset
   876
7ad01559b262 Initial revision
claus
parents:
diff changeset
   877
    |m|
7ad01559b262 Initial revision
claus
parents:
diff changeset
   878
7ad01559b262 Initial revision
claus
parents:
diff changeset
   879
    errorFlag := true.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   880
    m := ' Error:' , aMessage.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   881
    self notifyError:m position:position to:endPos.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   882
    exitBlock notNil ifTrue:[exitBlock value].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   883
    ^ false
7ad01559b262 Initial revision
claus
parents:
diff changeset
   884
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   885
7ad01559b262 Initial revision
claus
parents:
diff changeset
   886
parseError:aMessage position:position
7ad01559b262 Initial revision
claus
parents:
diff changeset
   887
    "report a syntax error"
7ad01559b262 Initial revision
claus
parents:
diff changeset
   888
7ad01559b262 Initial revision
claus
parents:
diff changeset
   889
    ^ self parseError:aMessage position:position to:nil
7ad01559b262 Initial revision
claus
parents:
diff changeset
   890
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   891
7ad01559b262 Initial revision
claus
parents:
diff changeset
   892
parseError:aMessage
7ad01559b262 Initial revision
claus
parents:
diff changeset
   893
    "report a syntax error"
7ad01559b262 Initial revision
claus
parents:
diff changeset
   894
7ad01559b262 Initial revision
claus
parents:
diff changeset
   895
    ^ self parseError:aMessage position:tokenPosition to:nil
7ad01559b262 Initial revision
claus
parents:
diff changeset
   896
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   897
7ad01559b262 Initial revision
claus
parents:
diff changeset
   898
correctableError:message position:pos1 to:pos2
3
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
   899
    "report an error which can be corrected by compiler -
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
   900
     return true if correction is wanted"
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   901
7ad01559b262 Initial revision
claus
parents:
diff changeset
   902
    |correctIt|
7ad01559b262 Initial revision
claus
parents:
diff changeset
   903
7ad01559b262 Initial revision
claus
parents:
diff changeset
   904
    requestor isNil ifTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   905
	self showErrorMessage:message position:pos1.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   906
	correctIt := false
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   907
    ] ifFalse:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   908
	correctIt := requestor correctableError:message position:pos1 to:pos2
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   909
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   910
    correctIt ifFalse:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   911
	exitBlock notNil ifTrue:[exitBlock value]
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   912
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   913
    ^ correctIt
7ad01559b262 Initial revision
claus
parents:
diff changeset
   914
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   915
7ad01559b262 Initial revision
claus
parents:
diff changeset
   916
undefError:aName position:pos1 to:pos2
3
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
   917
    "report an undefined variable error - return true, if it should be
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
   918
     corrected"
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
   919
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
   920
    requestor isNil ifTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   921
	warnedUndefVars notNil ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   922
	    (warnedUndefVars includes:aName) ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   923
		"already warned about this one"
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   924
		^ false
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   925
	    ].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   926
	].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   927
	self showErrorMessage:('Error: ' , aName , ' is undefined') position:pos1.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   928
	warnedUndefVars isNil ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   929
	    warnedUndefVars := Set new.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   930
	].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   931
	warnedUndefVars add:aName.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   932
	^ false
3
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
   933
    ].
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   934
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   935
    ^ self correctableError:('Error: ' , aName , ' is undefined') position:pos1 to:pos2
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   936
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   937
7ad01559b262 Initial revision
claus
parents:
diff changeset
   938
exitWith:something
7ad01559b262 Initial revision
claus
parents:
diff changeset
   939
    "this is the longjump out of evaluation via a return expression"
7ad01559b262 Initial revision
claus
parents:
diff changeset
   940
7ad01559b262 Initial revision
claus
parents:
diff changeset
   941
    evalExitBlock value:something
7ad01559b262 Initial revision
claus
parents:
diff changeset
   942
! !
7ad01559b262 Initial revision
claus
parents:
diff changeset
   943
7ad01559b262 Initial revision
claus
parents:
diff changeset
   944
!Parser methodsFor:'parsing'!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   945
7ad01559b262 Initial revision
claus
parents:
diff changeset
   946
parseMethod
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   947
    "parse a method.
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   948
     Return the parseTree or #Error.
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   949
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   950
     method ::= methodSpec methodBody
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   951
    "
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   952
7ad01559b262 Initial revision
claus
parents:
diff changeset
   953
    |parseTree|
7ad01559b262 Initial revision
claus
parents:
diff changeset
   954
7ad01559b262 Initial revision
claus
parents:
diff changeset
   955
    self nextToken.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   956
    (self parseMethodSpec == #Error) ifTrue:[^ #Error].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   957
    parseTree := self parseMethodBody.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   958
    (parseTree == #Error) ifFalse:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   959
	self tree:parseTree
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   960
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   961
    ^ parseTree
7ad01559b262 Initial revision
claus
parents:
diff changeset
   962
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   963
7ad01559b262 Initial revision
claus
parents:
diff changeset
   964
parseMethodSpec
7ad01559b262 Initial revision
claus
parents:
diff changeset
   965
    "parse a methods selector & arg specification;
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   966
     Set selector and methodArgs in the receiver as a side effect.
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   967
     Return the receiver or #Error.
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   968
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   969
     methodSpec ::= { KEYWORD IDENTIFIER }
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   970
		    | binaryOperator IDENTIFIER
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   971
		    | IDENTIFIER
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
   972
    "
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   973
7ad01559b262 Initial revision
claus
parents:
diff changeset
   974
    |var|
7ad01559b262 Initial revision
claus
parents:
diff changeset
   975
7ad01559b262 Initial revision
claus
parents:
diff changeset
   976
    (tokenType == #Keyword) ifTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   977
	selector := ''.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   978
	[tokenType == #Keyword] whileTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   979
	    selector := selector , tokenName.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   980
	    self nextToken.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   981
	    (tokenType ~~ #Identifier) ifTrue:[^ #Error].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   982
	    var := Variable name:tokenName.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   983
	    methodArgs isNil ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   984
		methodArgs := Array with:var.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   985
		methodArgNames := Array with:tokenName
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   986
	    ] ifFalse:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   987
		(methodArgNames includes:tokenName) ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   988
		    self syntaxError:'redefinition of ''' , tokenName , ''' in argument list.'
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   989
			    position:tokenPosition 
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   990
				  to:(tokenPosition + tokenName size - 1)
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   991
		].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   992
		methodArgs := methodArgs copyWith:var.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   993
		methodArgNames := methodArgNames copyWith:tokenName
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   994
	    ].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   995
	    self nextToken
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   996
	].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   997
	selector := selector asSymbol.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
   998
	^ self
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   999
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1000
    (tokenType == #Identifier) ifTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1001
	selector := tokenName asSymbol.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1002
	self nextToken.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1003
	^ self
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1004
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1005
    (tokenType == #BinaryOperator) ifTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1006
	selector := tokenName asSymbol.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1007
	self nextToken.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1008
	(tokenType ~~ #Identifier) ifTrue:[^ #Error].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1009
	var := Variable name:tokenName.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1010
	methodArgs isNil ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1011
	    methodArgs := Array with:var.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1012
	    methodArgNames := Array with:tokenName
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1013
	] ifFalse:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1014
	    methodArgs := methodArgs copyWith:var.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1015
	    methodArgNames := methodArgNames copyWith:tokenName
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1016
	].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1017
	self nextToken.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1018
	^ self
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1019
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1020
    ^ #Error
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1021
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1022
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  1023
parseMethodBody
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  1024
    "parse a methods body (locals & statements). 
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  1025
     No more tokens may follow.
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  1026
     Return a node-tree, or #Error
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  1027
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  1028
     methodBody ::= '<' st80Primitive '>' #EOF
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1029
		    | '<' st80Primitive '>' methodBodyVarSpec statementList #EOF
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  1030
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  1031
    "
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  1032
    |stats|
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  1033
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  1034
    stats := self parseMethodBodyOrNil.
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  1035
    (stats == #Error) ifFalse:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1036
	(tokenType ~~ #EOF) ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1037
	    "/ just for the nicer error message
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1038
	    (#(Self Nil True False Super) includes:tokenType) ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1039
		self parseError:tokenName , ' unexpected (missing ''.'' before ' , tokenName , ' ?)' 
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1040
		       position:tokenPosition to:(tokenPosition + tokenName size - 1)
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1041
	    ] ifFalse:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1042
		self parseError:(tokenType printString , ' unexpected').
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1043
	    ].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1044
	    ^#Error
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1045
	]
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  1046
    ].
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  1047
    ^ stats
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  1048
!
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  1049
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1050
parseMethodBodyOrNil
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1051
    "parse a methods body (locals & statements);
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  1052
     return  a node-tree, nil or #Error. 
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  1053
     empty (or comment only) input is accepted and returns nil.
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  1054
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  1055
     methodBodyOrNil ::= '<' st80Primitive '>'
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1056
			 | '<' st80Primitive '>' methodBodyVarSpec statementList
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1057
			 | <empty>
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  1058
    "
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1059
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1060
    |stats|
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1061
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1062
    ((tokenType == #BinaryOperator) and:[tokenName = '<']) ifTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1063
	"an ST-80 primitive - parsed but ignored"
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1064
	self nextToken.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1065
	primitiveNr := self parseST80Primitive.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1066
	(primitiveNr == #Error) ifTrue:[^ #Error].
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  1067
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1068
	self warning:'ST-80 primitives not supported - ignored'
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1069
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1070
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1071
    (self parseMethodBodyVarSpec == #Error) ifTrue:[^ #Error].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1072
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1073
    (tokenType ~~ #EOF) ifTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1074
	stats := self statementList
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1075
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1076
    ^ stats
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1077
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1078
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  1079
parseMethodBodyVarSpec
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  1080
    "parse a methods local variable specification. 
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  1081
     Leave spec of locals in methodLocals as a side effect.
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  1082
     Return #Error or nil.
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1083
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  1084
     methodBodyVarSpec ::= '|' { IDENTIFIER } '|'
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1085
			    | <empty>
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  1086
    "
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1087
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1088
    |var|
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1089
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1090
    (tokenType == $|) ifTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1091
	"memorize position for declaration in correction"
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1092
	localVarDefPosition := tokenPosition.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1093
	self nextToken.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1094
	[tokenType == #Identifier] whileTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1095
	    var := Variable name:tokenName.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1096
	    methodVars isNil ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1097
		methodVars := Array with:var.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1098
		methodVarNames := Array with:tokenName
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1099
	    ] ifFalse:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1100
		methodVars := methodVars copyWith:var.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1101
		methodVarNames := methodVarNames copyWith:tokenName
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1102
	    ].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1103
	    self nextToken
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1104
	].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1105
	(tokenType ~~ $|) ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1106
	    self syntaxError:'error in local var specification; | expected.'.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1107
	    ^ #Error
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1108
	].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1109
	self nextToken
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1110
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1111
    ^ nil
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1112
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1113
7
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
  1114
parseST80Primitive
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
  1115
    "parse an ST-80 type primitive as '< primitive: nr >';
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  1116
     return primitive number or #Error.
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  1117
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  1118
     st80Primitive ::= 'primitive:' INTEGER
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  1119
    "
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1120
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1121
    |primNumber|
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1122
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1123
    ((tokenType == #Keyword) and:[tokenName = 'primitive:']) ifFalse:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1124
	self parseError:'bad primitive definition (primitive: expected)'.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1125
	^ #Error
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1126
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1127
    self nextToken.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1128
    (tokenType == #Integer) ifFalse:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1129
	self parseError:'primitive number expected'.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1130
	^ #Error
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1131
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1132
    primNumber := tokenValue.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1133
    self nextToken.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1134
    ((tokenType == #BinaryOperator) and:[tokenName = '>']) ifFalse:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1135
	self parseError:'bad primitive definition (> expected)'.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1136
	^ #Error
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1137
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1138
    self nextToken.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1139
    ^ primNumber
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1140
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1141
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1142
statementList
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1143
    "parse a statementlist; return a node-tree, nil or #Error.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1144
     Statements must be separated by periods."
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1145
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1146
    |thisStatement prevStatement firstStatement correctIt periodPos|
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1147
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1148
    thisStatement := self statement.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1149
    (thisStatement == #Error) ifTrue:[^ #Error].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1150
    firstStatement := thisStatement.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1151
    [tokenType == $.] whileTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1152
	periodPos := tokenPosition.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1153
	self nextToken.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1154
	(tokenType == $]) ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1155
	    currentBlock isNil ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1156
		self parseError:'block nesting error'.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1157
		errorFlag := true
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1158
"
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1159
	    *** I had a warning here (since it was not defined
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1160
	    *** in the blue-book; but PD-code contains a lot of
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1161
	    *** code with periods at the end so that the warnings
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1162
	    *** became annoying
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1163
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1164
	    ] ifFalse:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1165
		self warning:'period after last statement' position:periodPos
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1166
"
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1167
	    ].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1168
	    ^ firstStatement
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1169
	].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1170
	(tokenType == #EOF) ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1171
	    currentBlock notNil ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1172
		self parseError:'block nesting error (expected '']'')'.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1173
		errorFlag := true
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1174
"
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1175
	    *** I had a warning here (since it was not defined
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1176
	    *** in the blue-book; but PD-code contains a lot of
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1177
	    *** code with periods at the end so that the warnings
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1178
	    *** became annoying
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1179
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1180
	    ] ifFalse:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1181
		self warning:'period after last statement' position:periodPos
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1182
"
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1183
	    ].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1184
	    ^ firstStatement
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1185
	].
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1186
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1187
	prevStatement := thisStatement.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1188
	(prevStatement isKindOf:ReturnNode) ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1189
	    self warning:'statements after return' position:tokenPosition
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1190
	].
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1191
"
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1192
	periodPos := tokenPosition.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1193
	self nextToken.
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1194
"
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1195
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1196
	((tokenType == $]) or:[tokenType == #EOF]) ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1197
	    (currentBlock isNil and:[tokenType == $]]) ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1198
		self parseError:'block nesting error'.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1199
		errorFlag := true
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1200
	    ] ifFalse:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1201
		correctIt := self correctableError:'period after last statement in block'
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1202
					  position:periodPos to:(periodPos + 1).
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1203
		correctIt ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1204
		    (self correctByDeleting == #Error) ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1205
			errorFlag := true
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1206
		    ]
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1207
		]
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1208
	    ].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1209
	    ^ firstStatement
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1210
	].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1211
	thisStatement := self statement.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1212
	(thisStatement == #Error) ifTrue:[^ #Error].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1213
	prevStatement nextStatement:thisStatement
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1214
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1215
    ^ firstStatement
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1216
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1217
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1218
statement
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  1219
    "parse a statement; return a node-tree or #Error.
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  1220
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  1221
     statement ::= '^' expression
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1222
		   | PRIMITIVECODE
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1223
		   | expression
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  1224
    "
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1225
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1226
    |expr node|
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1227
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1228
    (tokenType == $^) ifTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1229
	self nextToken.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1230
	expr := self expression.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1231
	(expr == #Error) ifTrue:[^ #Error].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1232
	node := ReturnNode expression:expr.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1233
	node home:self blockHome:currentBlock.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1234
	^ node
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1235
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1236
    (tokenType == #Primitive) ifTrue:[
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  1237
"
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1238
	self parseError:'cannot compile primitives (yet)'.
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  1239
"
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1240
	self nextToken.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1241
	hasPrimitiveCode := true.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1242
	^ PrimitiveNode code:''
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1243
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1244
    (tokenType == #EOF) ifTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1245
	self syntaxError:'period after last statement'.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1246
	^ #Error
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1247
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1248
    expr := self expression.
13
30e69e21d1d1 *** empty log message ***
claus
parents: 11
diff changeset
  1249
    (expr == #Error) ifTrue:[^ #Error].
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1250
"
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1251
    classToCompileFor notNil ifTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1252
	currentBlock isNil ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1253
	    (expr isKindOf:PrimaryNode) ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1254
		self warning:'useless computation - missing ^ ?'
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1255
	    ]
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1256
	]
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1257
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1258
"
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1259
    ^ StatementNode expression:expr
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1260
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1261
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1262
expression
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  1263
    "parse a cascade-expression; return a node-tree, nil or #Error.
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  1264
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  1265
     expression ::= keywordExpression
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1266
		    | keywordExpression cascade
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  1267
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  1268
     cascade ::= ';' expressionSendPart
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1269
		 | cascade ';' expressionSendPart
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  1270
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  1271
     expressionSendPart ::= { KEYWORD binaryExpression }
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1272
			    | BINARYOPERATOR unaryExpression
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1273
			    | IDENTIFIER
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  1274
    "
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1275
21
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
  1276
    |receiver arg sel args pos pos2 lno|
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1277
3
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
  1278
    pos := tokenPosition.
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1279
    receiver := self keywordExpression.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1280
    (receiver == #Error) ifTrue:[^ #Error].
15
992c3d87edbf *** empty log message ***
claus
parents: 13
diff changeset
  1281
    (tokenType == $;) ifTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1282
	[tokenType == $;] whileTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1283
	    receiver isMessage ifFalse:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1284
		self syntaxError:'left side of cascade must be a message expression'
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1285
			position:pos to:tokenPosition
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1286
	    ].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1287
	    self nextToken.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1288
	    (tokenType == #Identifier) ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1289
		sel := self selectorCheck:tokenName for:receiver position:tokenPosition to:(tokenPosition + tokenName size - 1).
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1290
		receiver := CascadeNode receiver:receiver selector:sel.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1291
		receiver lineNumber:tokenLineNr.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1292
		self nextToken.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1293
	    ] ifFalse:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1294
		(tokenType == #BinaryOperator) ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1295
		    sel := self selectorCheck:tokenName for:receiver position:tokenPosition to:(tokenPosition + tokenName size - 1).
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1296
		    lno := tokenLineNr. 
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1297
		    self nextToken.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1298
		    arg := self unaryExpression.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1299
		    (arg == #Error) ifTrue:[^ #Error].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1300
		    receiver := CascadeNode receiver:receiver selector:sel arg:arg.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1301
		    receiver lineNumber:lno.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1302
		] ifFalse:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1303
		    (tokenType == #Keyword) ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1304
			pos := tokenPosition. 
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1305
			lno := tokenLineNr. 
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1306
			sel := tokenName.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1307
			self nextToken.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1308
			arg := self binaryExpression.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1309
			(arg == #Error) ifTrue:[^ #Error].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1310
			args := Array with:arg.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1311
			[tokenType == #Keyword] whileTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1312
			    sel := sel , tokenName.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1313
			    self nextToken.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1314
			    arg := self binaryExpression.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1315
			    (arg == #Error) ifTrue:[^ #Error].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1316
			    args := args copyWith:arg.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1317
			    pos2 := tokenPosition
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1318
			].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1319
			sel := self selectorCheck:sel for:receiver position:pos to:pos2.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1320
			receiver := CascadeNode receiver:receiver selector:sel args:args.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1321
			receiver lineNumber:lno.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1322
		    ] ifFalse:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1323
			(tokenType == #Error) ifTrue:[^ #Error].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1324
			self syntaxError:('invalid cascade; ' , tokenType printString , ' unexpected')
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1325
				position:tokenPosition to:source position - 1.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1326
			^ #Error
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1327
		    ]
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1328
		]
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1329
	    ]
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1330
	].
15
992c3d87edbf *** empty log message ***
claus
parents: 13
diff changeset
  1331
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1332
	"obscure (uspecified ?) if selector follows; Question:
15
992c3d87edbf *** empty log message ***
claus
parents: 13
diff changeset
  1333
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1334
	is
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1335
		'expr sel1; sel2 sel3'
15
992c3d87edbf *** empty log message ***
claus
parents: 13
diff changeset
  1336
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1337
	to be parsed as: 
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1338
		(t := expr.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1339
		 t sel1.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1340
		 t sel2) sel3
15
992c3d87edbf *** empty log message ***
claus
parents: 13
diff changeset
  1341
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1342
	 or:
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1343
		(t := expr.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1344
		 t sel1.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1345
		 t sel2 sel3)
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1346
	"
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1347
	((tokenType == #Identifier) 
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1348
	 or:[(tokenType == #BinaryOperator)
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1349
	     or:[tokenType == #Keyword]]) ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1350
	    self syntaxError:'ambigous cascade - please group using ( ...)'
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1351
		    position:tokenPosition to:source position - 1.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1352
	    ^ #Error
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1353
	]
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1354
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1355
    ^ receiver
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1356
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1357
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1358
keywordExpression
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1359
    "parse a keyword-expression; return a node-tree, nil or #Error"
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1360
3
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
  1361
    |receiver sel arg args pos1 pos2 try lno note|
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1362
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1363
    receiver := self binaryExpression.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1364
    (receiver == #Error) ifTrue:[^ #Error].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1365
    (tokenType == #Keyword) ifTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1366
	pos1 := tokenPosition.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1367
	sel := tokenName.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1368
	lno := tokenLineNr.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1369
	self nextToken.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1370
	arg := self binaryExpression.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1371
	(arg == #Error) ifTrue:[^ #Error].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1372
	args := Array with:arg.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1373
	[tokenType == #Keyword] whileTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1374
	    sel := sel , tokenName.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1375
	    self nextToken.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1376
	    arg := self binaryExpression.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1377
	    (arg == #Error) ifTrue:[^ #Error].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1378
	    args := args copyWith:arg.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1379
	    pos2 := tokenPosition
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1380
	].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1381
	sel := self selectorCheck:sel for:receiver position:pos1 to:pos2.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1382
	try := MessageNode receiver:receiver selector:sel args:args.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1383
	(try isMemberOf:String) ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1384
	    self parseError:try position:pos1 to:pos2.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1385
	    receiver := MessageNode receiver:receiver selector:sel args:args fold:false.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1386
	    note := receiver plausibilityCheck.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1387
	    note notNil ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1388
		self warning:note position:pos1 to:pos2
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1389
	    ].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1390
	] ifFalse:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1391
	    receiver := try
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1392
	].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1393
	receiver lineNumber:lno
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1394
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1395
    ^ receiver
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1396
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1397
7
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
  1398
degeneratedKeywordExpressionForSelector
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
  1399
    "parse a keyword-expression without receiver - for the selector
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
  1400
     only. return the selector or nil"
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
  1401
10
73e97b6175c4 *** empty log message ***
claus
parents: 9
diff changeset
  1402
    |sel arg|
7
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
  1403
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
  1404
    (tokenType == #Keyword) ifTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1405
	sel := tokenName.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1406
	self nextToken.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1407
	arg := self binaryExpression.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1408
	(arg == #Error) ifTrue:[^ sel].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1409
	[tokenType == #Keyword] whileTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1410
	    sel := sel , tokenName.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1411
	    self nextToken.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1412
	    arg := self binaryExpression.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1413
	    (arg == #Error) ifTrue:[^ sel].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1414
	].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1415
	^ sel
7
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
  1416
    ].
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
  1417
    ^ nil
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
  1418
!
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
  1419
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1420
binaryExpression
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1421
    "parse a binary-expression; return a node-tree, nil or #Error"
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1422
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1423
    |receiver arg sel pos try lno note|
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1424
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1425
    receiver := self unaryExpression.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1426
    (receiver == #Error) ifTrue:[^ #Error].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1427
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1428
    "special kludge: since Scanner cannot know if -digit is a binary
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1429
     expression or a negative constant, handle cases here"
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1430
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1431
    [(tokenType == #BinaryOperator) or:[(tokenType == $|)
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1432
     or:[(tokenType == #Integer) and:[tokenValue < 0]]]] whileTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1433
	pos := tokenPosition.
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1434
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1435
	lno := tokenLineNr.
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1436
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1437
	"kludge here: bar and minus are not scanned as binop "
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1438
	(tokenType == $|) ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1439
	    sel := '|'.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1440
	    self nextToken
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1441
	] ifFalse:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1442
	    (tokenType == #BinaryOperator) ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1443
		sel := self selectorCheck:tokenName for:receiver position:tokenPosition to:(tokenPosition + tokenName size - 1).
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1444
		self nextToken
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1445
	    ] ifFalse:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1446
		sel := '-'.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1447
		tokenValue := tokenValue negated
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1448
	    ]
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1449
	].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1450
	arg := self unaryExpression.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1451
	(arg == #Error) ifTrue:[^ #Error].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1452
	try := BinaryNode receiver:receiver selector:sel arg:arg.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1453
	(try isMemberOf:String) ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1454
	    self parseError:try position:pos to:tokenPosition.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1455
	    receiver := BinaryNode receiver:receiver selector:sel arg:arg fold:false.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1456
	    note := receiver plausibilityCheck.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1457
	    note notNil ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1458
		self warning:note position:pos to:tokenPosition
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1459
	    ].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1460
	] ifFalse:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1461
	    receiver := try
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1462
	].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1463
	receiver lineNumber:lno.
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1464
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1465
    ^ receiver
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1466
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1467
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1468
unaryExpression
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1469
    "parse a unary-expression; return a node-tree, nil or #Error"
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1470
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  1471
    |receiver sel pos pos2 try|
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1472
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1473
    receiver := self primary.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1474
    (receiver == #Error) ifTrue:[^ #Error].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1475
    [tokenType == #Identifier] whileTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1476
	pos := tokenPosition.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1477
	pos2 := pos + tokenName size - 1.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1478
	sel := self selectorCheck:tokenName for:receiver position:pos to:pos2.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1479
	try := UnaryNode receiver:receiver selector:sel.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1480
	(try isMemberOf:String) ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1481
	    self warning:try position:pos to:pos2.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1482
	    receiver := UnaryNode receiver:receiver selector:sel fold:false.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1483
	] ifFalse:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1484
	    receiver := try
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1485
	].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1486
	receiver lineNumber:tokenLineNr.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1487
	self nextToken.
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1488
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1489
    ^ receiver
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1490
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1491
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1492
primary
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1493
    "parse a primary-expression; return a node-tree, nil or #Error"
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1494
45
e8331ba8ad5d *** empty log message ***
claus
parents: 44
diff changeset
  1495
    |val var expr pos name t|
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1496
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1497
    pos := tokenPosition.
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1498
    (tokenType == #Self) ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1499
	self nextToken.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1500
	(tokenType == $_) ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1501
	    self parseError:'assignment to self' position:pos to:tokenPosition.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1502
	    ^ #Error
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1503
	].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1504
	selfNode isNil ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1505
	    selfNode := SelfNode value:selfValue
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1506
	].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1507
	^ selfNode
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1508
    ].
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1509
    (tokenType == #Identifier) ifTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1510
	var := self variable.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1511
	(var == #Error) ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1512
	    errorFlag := true
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1513
	].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1514
	self nextToken.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1515
	(tokenType == $_) ifFalse:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1516
	    ^ var
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1517
	].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1518
	(var ~~ #Error) ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1519
	    t := var type.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1520
	    (t == #MethodArg) ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1521
		self parseError:'assignment to method argument' position:pos to:tokenPosition.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1522
		errorFlag := true
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1523
	    ] ifFalse:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1524
		(t == #BlockArg) ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1525
		    self parseError:'assignment to block argument' position:pos to:tokenPosition.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1526
		    errorFlag := true
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1527
		] ifFalse:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1528
		    (t == #InstanceVariable) ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1529
			modifiedInstVars isNil ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1530
			    modifiedInstVars := OrderedCollection new
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1531
			].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1532
			name := PrevInstVarNames at:(var index).
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1533
			(modifiedInstVars includes:name) ifFalse:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1534
			    modifiedInstVars add:name
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1535
			]
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1536
		    ] ifFalse:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1537
			(t == #ClassVariable) ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1538
			    modifiedClassVars isNil ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1539
				modifiedClassVars := OrderedCollection new
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1540
			    ].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1541
			    name := var name.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1542
			    name := name copyFrom:((name indexOf:$:) + 1).
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1543
			    (modifiedClassVars includes:name) ifFalse:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1544
				modifiedClassVars add:name
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1545
			    ]
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1546
			] ifFalse:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1547
			    (t == #GlobalVariable) ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1548
				(Smalltalk classNamed:var name) notNil ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1549
				    self warning:'assignment to global which contains class' position:pos to:tokenPosition.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1550
				]
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1551
			    ]
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1552
			]
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1553
		    ]
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1554
		]
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1555
	    ]
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1556
	].
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1557
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1558
	self nextToken.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1559
	expr := self expression.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1560
	(errorFlag or:[expr == #Error]) ifTrue:[^ #Error].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1561
	^ AssignmentNode variable:var expression:expr
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1562
    ].
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1563
    ((tokenType == #Integer) 
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1564
     or:[(tokenType == #Character) 
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1565
     or:[(tokenType == #Float)
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1566
     or:[(tokenType == #String)
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1567
     or:[(tokenType == #Symbol)]]]]) ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1568
	val := ConstantNode type:tokenType value:tokenValue.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1569
	self nextToken.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1570
	(tokenType == $_) ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1571
	    self parseError:'assignment to a constant' position:pos to:tokenPosition.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1572
	    ^ #Error
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1573
	].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1574
	^ val
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1575
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1576
    (tokenType == #Nil) ifTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1577
	self nextToken.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1578
	(tokenType == $_) ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1579
	    self parseError:'assignment to nil' position:pos to:tokenPosition.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1580
	    ^ #Error
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1581
	].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1582
	^ ConstantNode type:#Nil value:nil
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1583
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1584
    (tokenType == #True) ifTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1585
	self nextToken.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1586
	(tokenType == $_) ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1587
	    self parseError:'assignment to true' position:pos to:tokenPosition.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1588
	    ^ #Error
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1589
	].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1590
	^ ConstantNode type:#True value:true
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1591
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1592
    (tokenType == #False) ifTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1593
	self nextToken.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1594
	(tokenType == $_) ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1595
	    self parseError:'assignment to false' position:pos to:tokenPosition.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1596
	    ^ #Error
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1597
	].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1598
	^ ConstantNode type:#False value:false
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1599
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1600
    (tokenType  == #Super) ifTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1601
	usesSuper := true.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1602
	self nextToken.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1603
	(tokenType == $_) ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1604
	    self parseError:'assignment to super' position:pos to:tokenPosition.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1605
	    ^ #Error
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1606
	].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1607
	(classToCompileFor isNil or:[classToCompileFor superclass isNil]) ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1608
	    self warning:'superclass is (currently ?) nil' 
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1609
		 position:pos to:(pos + 4).
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1610
	].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1611
	superNode isNil ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1612
	    superNode := SuperNode value:selfValue inClass:classToCompileFor
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1613
	].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1614
	^ superNode
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1615
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1616
    (tokenType == #ThisContext) ifTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1617
	self nextToken.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1618
	(tokenType == $_) ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1619
	    self parseError:'assignment to thisContext' position:pos to:tokenPosition.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1620
	    ^ #Error
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1621
	].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1622
	^ VariableNode type:#ThisContext
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1623
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1624
    (tokenType == #HashLeftParen) ifTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1625
	self nextToken.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1626
	val := self array.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1627
	self nextToken.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1628
	(tokenType == $_) ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1629
	    self parseError:'assignment to a constant' position:pos to:tokenPosition.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1630
	    ^ #Error
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1631
	].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1632
	^ ConstantNode type:#Array value:val
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1633
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1634
    (tokenType == #HashLeftBrack) ifTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1635
	self nextToken.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1636
	val := self byteArray.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1637
	self nextToken.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1638
	(tokenType == $_) ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1639
	    self parseError:'assignment to a constant' position:pos to:tokenPosition.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1640
	    ^ #Error
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1641
	].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1642
	^ ConstantNode type:#Array value:val
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1643
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1644
    (tokenType == $() ifTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1645
	self nextToken.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1646
	val := self expression.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1647
	(val == #Error) ifTrue:[^ #Error].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1648
	(tokenType ~~ $) ) ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1649
	    tokenType isCharacter ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1650
		self syntaxError:'missing '')'' (i.e. ''' , tokenType asString , ''' unexpected)' withCRs position:pos to:tokenPosition.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1651
	    ] ifFalse:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1652
		self syntaxError:'missing '')''' position:pos to:tokenPosition.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1653
	    ].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1654
	    ^ #Error
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1655
	].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1656
	self nextToken.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1657
	(tokenType == $_) ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1658
	    self parseError:'invalid assignment' position:pos to:tokenPosition.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1659
	    ^ #Error
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1660
	].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1661
	val parenthized:true.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1662
	^ val
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1663
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1664
    (tokenType == $[ ) ifTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1665
	val := self block.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1666
	self nextToken.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1667
	(tokenType == $_) ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1668
	    self parseError:'invalid assignment' position:pos to:tokenPosition.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1669
	    ^ #Error
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1670
	].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1671
	^ val
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1672
    ].
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  1673
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1674
    (tokenType == #Error) ifTrue:[^ #Error].
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1675
    tokenType isCharacter ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1676
	self syntaxError:('error in primary; ' 
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1677
			   , tokenType printString , 
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1678
			   ' unexpected') position:tokenPosition to:tokenPosition
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1679
    ] ifFalse:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1680
	(#(BinaryOperator Keyword) includes:tokenType) ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1681
	    self syntaxError:('error in primary; ' 
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1682
			       , tokenType printString , '(' , tokenName , ') ' ,
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1683
			       ' unexpected')
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1684
	] ifFalse:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1685
	    self syntaxError:('error in primary; ' 
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1686
			       , tokenType printString ,
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1687
			       ' unexpected') 
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1688
	]
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1689
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1690
    ^ #Error
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1691
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1692
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1693
variableOrError
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1694
    "parse a variable; return a node-tree, nil or #Error"
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1695
21
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
  1696
    ^ self variableOrError:tokenName
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
  1697
!
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
  1698
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
  1699
variableOrError:varName
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
  1700
    "parse a variable; return a node-tree, nil or #Error"
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
  1701
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  1702
    |var instIndex aClass searchBlock args vars
21
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
  1703
     tokenSymbol className|
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1704
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1705
    "is it a block-arg or block-var ?"
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1706
    searchBlock := currentBlock.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1707
    [searchBlock notNil] whileTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1708
	args := searchBlock arguments.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1709
	args notNil ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1710
	    instIndex := args findFirst:[:aBlockArg | aBlockArg name = varName].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1711
	    instIndex ~~ 0 ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1712
		^ VariableNode type:#BlockArg
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1713
			       name:varName
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1714
			      token:(args at:instIndex)
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1715
			      index:instIndex
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1716
			      block:searchBlock
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1717
	    ].
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  1718
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1719
	].
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1720
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1721
	vars := searchBlock variables.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1722
	vars notNil ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1723
	    instIndex := vars findFirst:[:aBlockVar | aBlockVar name = varName].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1724
	    instIndex ~~ 0 ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1725
		^ VariableNode type:#BlockVariable
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1726
			       name:varName
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1727
			      token:(vars at:instIndex)
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1728
			      index:instIndex
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1729
			      block:searchBlock
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1730
	    ].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1731
	].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1732
	searchBlock := searchBlock home
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1733
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1734
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1735
    "is it a method-variable ?"
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1736
    methodVars notNil ifTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1737
	instIndex := methodVarNames indexOf:varName.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1738
	instIndex ~~ 0 ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1739
	    var := methodVars at:instIndex.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1740
	    var used:true.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1741
	    ^ VariableNode type:#MethodVariable
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1742
			   name:varName
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1743
			  token:var
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1744
			  index:instIndex
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1745
	]
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1746
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1747
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1748
    "is it a method-argument ?"
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1749
    methodArgs notNil ifTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1750
	instIndex := methodArgNames indexOf:varName.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1751
	instIndex ~~ 0 ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1752
	    ^ VariableNode type:#MethodArg
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1753
			   name:varName
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1754
			  token:(methodArgs at:instIndex)
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1755
			  index:instIndex
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1756
	]
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1757
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1758
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1759
    "is it an instance-variable ?"
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1760
    classToCompileFor notNil ifTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1761
	"caching allInstVarNames for next compilation saves time ..."
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1762
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1763
	(PrevInstVarNames isNil or:[PrevClass ~~ classToCompileFor]) ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1764
	    PrevClass notNil ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1765
		PrevClass removeDependent:Parser
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1766
	    ].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1767
	    PrevClass := classToCompileFor.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1768
	    PrevInstVarNames := classToCompileFor allInstVarNames.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1769
	    PrevClassInstVarNames := nil.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1770
	    PrevClassVarNames := nil.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1771
	    PrevClass addDependent:Parser
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1772
	].
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1773
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1774
	instIndex := PrevInstVarNames indexOf:varName startingAt:1.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1775
	instIndex ~~ 0 ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1776
	    usedInstVars isNil ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1777
		usedInstVars := OrderedCollection with:varName
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1778
	    ] ifFalse:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1779
		(usedInstVars includes:varName) ifFalse:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1780
		    usedInstVars add:varName
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1781
		]
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1782
	    ].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1783
	    usedVars isNil ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1784
		usedVars := OrderedCollection with:varName
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1785
	    ] ifFalse:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1786
		(usedVars includes:varName) ifFalse:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1787
		    usedVars add:varName
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1788
		]
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1789
	    ].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1790
	    ^ VariableNode type:#InstanceVariable 
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1791
			   name:varName
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1792
			  index:instIndex
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1793
		      selfValue:selfValue
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1794
	]
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1795
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1796
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1797
    "is it a class-instance-variable ?"
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1798
    classToCompileFor notNil ifTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1799
	PrevClassInstVarNames isNil ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1800
	    PrevClassInstVarNames := classToCompileFor class allInstVarNames
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1801
	].
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1802
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1803
	instIndex := PrevClassInstVarNames indexOf:varName startingAt:1.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1804
	instIndex ~~ 0 ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1805
	    aClass := self inWhichClassIsClassInstVar:varName.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1806
	    aClass notNil ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1807
		usedVars isNil ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1808
		    usedVars := OrderedCollection with:varName
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1809
		] ifFalse:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1810
		    (usedVars includes:varName) ifFalse:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1811
			usedVars add:varName
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1812
		    ]
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1813
		].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1814
		^ VariableNode type:#ClassInstanceVariable
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1815
			       name:varName
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1816
			      index:instIndex
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1817
			  selfClass:aClass
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1818
	    ]
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1819
	]
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1820
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1821
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1822
    "is it a class-variable ?"
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1823
    classToCompileFor notNil ifTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1824
	PrevClassVarNames isNil ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1825
	    aClass := classToCompileFor.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1826
	    classToCompileFor isMeta ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1827
		className := aClass name.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1828
		className := className copyTo:(className size - 5).
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1829
		aClass := Smalltalk at:(className asSymbol).
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1830
		aClass isNil ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1831
		    aClass := classToCompileFor
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1832
		]
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1833
	    ].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1834
	    PrevClassVarNames := aClass allClassVarNames
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1835
	].
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1836
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1837
	instIndex := PrevClassVarNames indexOf:varName startingAt:1.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1838
	instIndex ~~ 0 ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1839
	    aClass := self inWhichClassIsClassVar:varName.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1840
	    aClass notNil ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1841
		usedClassVars isNil ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1842
		    usedClassVars := OrderedCollection with:varName
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1843
		] ifFalse:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1844
		    (usedClassVars includes:varName) ifFalse:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1845
			usedClassVars add:varName
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1846
		    ].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1847
		].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1848
		usedVars isNil ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1849
		    usedVars := OrderedCollection with:varName
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1850
		] ifFalse:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1851
		    (usedVars includes:varName) ifFalse:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1852
			usedVars add:varName
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1853
		    ].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1854
		].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1855
		^ VariableNode type:#ClassVariable 
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1856
			       name:(aClass name , ':' , varName) asSymbol
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1857
	    ]
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1858
	]
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1859
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1860
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1861
    "is it a global-variable ?"
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1862
    tokenSymbol := varName asSymbol.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1863
    (Smalltalk includesKey:tokenSymbol) ifTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1864
	usedVars isNil ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1865
	    usedVars := OrderedCollection with:varName
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1866
	] ifFalse:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1867
	    (usedVars includes:varName) ifFalse:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1868
		usedVars add:varName
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1869
	    ]
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1870
	].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1871
	^ VariableNode type:#GlobalVariable name:tokenSymbol
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1872
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1873
    ^ #Error
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1874
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1875
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1876
variable
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1877
    "parse a variable; if undefined, notify error and correct if user wants to"
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1878
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1879
    |v|
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1880
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1881
    v := self variableOrError.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1882
    (v == #Error) ifFalse:[^ v].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1883
    v := self correctVariable.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1884
    (v == #Error) ifFalse:[^ v].
13
30e69e21d1d1 *** empty log message ***
claus
parents: 11
diff changeset
  1885
    ^ VariableNode type:#GlobalVariable name:tokenName asSymbol
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1886
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1887
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1888
inWhichClassIsClassVar:aString
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1889
    "search class-chain for the classvariable named aString
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1890
     - return the class or nil if not found"
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1891
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1892
    |aClass className baseClass|
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1893
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1894
    aClass := classToCompileFor.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1895
    aClass isMeta ifTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1896
	className := aClass name.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1897
	className := className copyTo:(className size - 5).
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1898
	baseClass := Smalltalk at:(className asSymbol).
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1899
	baseClass notNil ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1900
	    aClass := baseClass
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1901
	]
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1902
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1903
    [aClass notNil] whileTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1904
	(aClass classVarNames includes:aString) ifTrue:[ ^ aClass].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1905
	aClass := aClass superclass
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1906
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1907
    ^ nil
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1908
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1909
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1910
inWhichClassIsClassInstVar:aString
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1911
    "search class-chain for the class-instance variable named aString
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1912
     - return the class or nil if not found"
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1913
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1914
    |aClass|
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1915
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1916
    aClass := classToCompileFor.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1917
    [aClass notNil] whileTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1918
	(aClass class instVarNames includes:aString) ifTrue:[ ^ aClass].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1919
	aClass := aClass superclass
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1920
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1921
    ^ nil
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1922
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1923
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1924
block
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1925
    "parse a block; return a node-tree, nil or #Error"
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1926
23
1f7bcfff8d39 *** empty log message ***
claus
parents: 21
diff changeset
  1927
    |node args argNames arg pos lno|
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1928
7
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
  1929
    lno := tokenLineNr.
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1930
    self nextToken.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1931
    (tokenType == $: ) ifTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1932
	[tokenType == $:] whileTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1933
	    pos := tokenPosition.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1934
	    self nextToken.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1935
	    (tokenType == #Identifier) ifFalse:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1936
		self syntaxError:'Identifier expected in block-arg declaration'
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1937
			position:pos to:tokenPosition-1.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1938
		^ #Error
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1939
	    ].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1940
	    arg := Variable name:tokenName.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1941
	    args isNil ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1942
		args := Array with:arg.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1943
		argNames := Array with:tokenName.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1944
	    ] ifFalse:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1945
		(argNames includes:tokenName) ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1946
		    self syntaxError:'redefinition of ''' , tokenName , ''' in argument list.'
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1947
			    position:tokenPosition 
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1948
				   to:(tokenPosition + tokenName size - 1)
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1949
		].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1950
		args := args copyWith:arg.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1951
		argNames := argNames copyWith:tokenName.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1952
	    ].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1953
	    self nextToken
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1954
	].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1955
	(tokenType ~~ $| ) ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1956
	    "ST-80 allows [:arg ]"
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1957
	    (tokenType == $] ) ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1958
		node := BlockNode arguments:args home:currentBlock variables:nil.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1959
		node lineNumber:lno.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1960
		^ node
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1961
	    ].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1962
	    self syntaxError:'| expected after block-arg declaration'.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1963
	    ^ #Error
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1964
	].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1965
	self nextToken
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1966
    ].
23
1f7bcfff8d39 *** empty log message ***
claus
parents: 21
diff changeset
  1967
    node := self blockBody:args.
1f7bcfff8d39 *** empty log message ***
claus
parents: 21
diff changeset
  1968
    (node notNil and:[node ~~ #Error]) ifTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1969
	node lineNumber:lno.
23
1f7bcfff8d39 *** empty log message ***
claus
parents: 21
diff changeset
  1970
    ].
1f7bcfff8d39 *** empty log message ***
claus
parents: 21
diff changeset
  1971
    ^ node
1f7bcfff8d39 *** empty log message ***
claus
parents: 21
diff changeset
  1972
!
1f7bcfff8d39 *** empty log message ***
claus
parents: 21
diff changeset
  1973
1f7bcfff8d39 *** empty log message ***
claus
parents: 21
diff changeset
  1974
blockBody:args
1f7bcfff8d39 *** empty log message ***
claus
parents: 21
diff changeset
  1975
    "parse a blocks body; return a node-tree, nil or #Error"
1f7bcfff8d39 *** empty log message ***
claus
parents: 21
diff changeset
  1976
1f7bcfff8d39 *** empty log message ***
claus
parents: 21
diff changeset
  1977
    |stats node var vars pos lno|
1f7bcfff8d39 *** empty log message ***
claus
parents: 21
diff changeset
  1978
1f7bcfff8d39 *** empty log message ***
claus
parents: 21
diff changeset
  1979
    lno := tokenLineNr.
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1980
    (tokenType == $| ) ifTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1981
	self nextToken.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1982
	pos := tokenPosition.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1983
	[tokenType == $|] whileFalse:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1984
	    (tokenType == #Identifier) ifFalse:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1985
		self syntaxError:'Identifier expected in block-var declaration' position:pos.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1986
		^ #Error
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1987
	    ].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1988
	    var := Variable name:tokenName.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1989
	    vars isNil ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1990
		vars := Array with:var
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1991
	    ] ifFalse:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1992
		vars := vars copyWith:var
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1993
	    ].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1994
	    self nextToken
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1995
	].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  1996
	self nextToken
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1997
    ].
3
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
  1998
    node := BlockNode arguments:args home:currentBlock variables:vars.
7
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
  1999
    node lineNumber:lno.
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2000
    currentBlock := node.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2001
    stats := self blockStatementList.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2002
    node statements:stats.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2003
    currentBlock := node home.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2004
    (stats == #Error) ifTrue:[^ #Error].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2005
    ^ node
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2006
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2007
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2008
blockStatementList
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2009
    "parse a blocks statementlist; return a node-tree, nil or #Error"
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2010
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2011
    |thisStatement prevStatement firstStatement|
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2012
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2013
    (tokenType == $] ) ifTrue:[^ nil].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2014
    thisStatement := self statement.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2015
    (thisStatement == #Error) ifTrue:[^ #Error].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2016
    firstStatement := thisStatement.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2017
    [tokenType == $] ] whileFalse:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2018
	(tokenType == $.) ifFalse:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2019
	    (tokenType == #EOF) ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2020
		self syntaxError:'missing '']'' in block'
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2021
	    ] ifFalse:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2022
		self syntaxError:'missing ''.'' in block'
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2023
	    ].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2024
	    ^ #Error
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2025
	] ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2026
	    prevStatement := thisStatement.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2027
	    self nextToken.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2028
	    tokenType == $] ifTrue:[
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2029
"
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2030
		*** I had a warning here (since it was not defined
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2031
		*** in the blue-book; but PD-code contains a lot of
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2032
		*** code with periods at the end so that the warnings
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2033
		*** became annoying
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2034
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2035
		self warning:'period after last statement in block'.
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2036
"
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2037
		^ firstStatement
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2038
	    ].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2039
	    thisStatement := self statement.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2040
	    (thisStatement == #Error) ifTrue:[^ #Error].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2041
	    prevStatement nextStatement:thisStatement
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2042
	]
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2043
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2044
    ^ firstStatement
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2045
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2046
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2047
array
7
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
  2048
    |arr elem pos1|
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2049
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2050
    pos1 := tokenPosition.
20
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
  2051
    arr := OrderedCollection new:20.
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2052
    [tokenType ~~ $) ] whileTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2053
	elem := self arrayConstant.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2054
	(elem == #Error) ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2055
	    (tokenType == #EOF) ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2056
		self syntaxError:'unterminated array-constant; '')'' expected' 
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2057
			position:pos1 to:tokenPosition
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2058
	    ].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2059
	    ^ #Error
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2060
	].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2061
	arr add:elem.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2062
	self nextToken
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2063
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2064
    ^ Array withAll:arr
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2065
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2066
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2067
byteArray
7
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
  2068
    "started with ST-80 R4 - allow byteArray constants as #[ ... ]"
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
  2069
45
e8331ba8ad5d *** empty log message ***
claus
parents: 44
diff changeset
  2070
    |bytes index limit newArray elem pos1 pos2|
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2071
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2072
    pos1 := tokenPosition.
45
e8331ba8ad5d *** empty log message ***
claus
parents: 44
diff changeset
  2073
    bytes := ByteArray new:5000.
e8331ba8ad5d *** empty log message ***
claus
parents: 44
diff changeset
  2074
    index := 0. limit := 5000.
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2075
    [tokenType ~~ $] ] whileTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2076
	pos2 := tokenPosition.
45
e8331ba8ad5d *** empty log message ***
claus
parents: 44
diff changeset
  2077
	"
e8331ba8ad5d *** empty log message ***
claus
parents: 44
diff changeset
  2078
	 this is not good programming style, but speeds up
e8331ba8ad5d *** empty log message ***
claus
parents: 44
diff changeset
  2079
	 reading of huge byte arrays (i.e. stored Images ...)
e8331ba8ad5d *** empty log message ***
claus
parents: 44
diff changeset
  2080
	"
e8331ba8ad5d *** empty log message ***
claus
parents: 44
diff changeset
  2081
	(tokenType == #Integer) ifTrue:[
e8331ba8ad5d *** empty log message ***
claus
parents: 44
diff changeset
  2082
	    elem := tokenValue
e8331ba8ad5d *** empty log message ***
claus
parents: 44
diff changeset
  2083
	] ifFalse:[
e8331ba8ad5d *** empty log message ***
claus
parents: 44
diff changeset
  2084
	    elem := self arrayConstant.
e8331ba8ad5d *** empty log message ***
claus
parents: 44
diff changeset
  2085
	].
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2086
	(elem == #Error) ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2087
	    (tokenType == #EOF) ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2088
		self syntaxError:'unterminated bytearray-constant; '']'' expected' 
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2089
			position:pos1 to:tokenPosition
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2090
	    ].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2091
	    ^ #Error
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2092
	].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2093
	((elem isMemberOf:SmallInteger) and:
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2094
	[(elem >= 0) and:[elem <= 255]]) ifTrue:[
45
e8331ba8ad5d *** empty log message ***
claus
parents: 44
diff changeset
  2095
	    index := index + 1.
e8331ba8ad5d *** empty log message ***
claus
parents: 44
diff changeset
  2096
	    bytes at:index put:elem.
e8331ba8ad5d *** empty log message ***
claus
parents: 44
diff changeset
  2097
	    index == limit ifTrue:[
e8331ba8ad5d *** empty log message ***
claus
parents: 44
diff changeset
  2098
		newArray := ByteArray new:(limit * 2).
e8331ba8ad5d *** empty log message ***
claus
parents: 44
diff changeset
  2099
		newArray replaceFrom:1 to:limit with:bytes startingAt:1.
e8331ba8ad5d *** empty log message ***
claus
parents: 44
diff changeset
  2100
		limit := limit * 2.
e8331ba8ad5d *** empty log message ***
claus
parents: 44
diff changeset
  2101
		bytes := newArray
e8331ba8ad5d *** empty log message ***
claus
parents: 44
diff changeset
  2102
	    ].
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2103
	] ifFalse:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2104
	    self parseError:'invalid ByteArray element' position:pos2 to:tokenPosition - 1
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2105
	].
45
e8331ba8ad5d *** empty log message ***
claus
parents: 44
diff changeset
  2106
	self nextToken.
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2107
    ].
45
e8331ba8ad5d *** empty log message ***
claus
parents: 44
diff changeset
  2108
    newArray := ByteArray new:index.
e8331ba8ad5d *** empty log message ***
claus
parents: 44
diff changeset
  2109
    newArray replaceFrom:1 to:index with:bytes startingAt:1.
e8331ba8ad5d *** empty log message ***
claus
parents: 44
diff changeset
  2110
    ^ newArray
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2111
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2112
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2113
arrayConstant
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2114
    (tokenType == #String) ifTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2115
	^ tokenValue
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2116
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2117
    (tokenType == #Nil) ifTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2118
	^ nil
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2119
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2120
    (tokenType == #Integer) ifTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2121
	^ tokenValue
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2122
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2123
    (tokenType == #Character) ifTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2124
	^ tokenValue
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2125
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2126
    (tokenType == #Float) ifTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2127
	^ tokenValue
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2128
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2129
    (tokenType == #True) ifTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2130
	^ true
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2131
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2132
    (tokenType == #False) ifTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2133
	^ false
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2134
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2135
    (tokenType == #Error) ifTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2136
	^ #Error
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2137
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2138
    (tokenType == #BinaryOperator) ifTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2139
	^ tokenName asSymbol
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2140
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2141
    (tokenType == #Keyword) ifTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2142
	^ tokenName asSymbol
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2143
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2144
    (tokenType == #Identifier) ifTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2145
	^ tokenName asSymbol
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2146
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2147
    (tokenType == $() ifTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2148
	self nextToken.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2149
	^ self array
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2150
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2151
    (tokenType == $[) ifTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2152
	self nextToken.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2153
	^ self byteArray
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2154
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2155
    (tokenType == #Symbol) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2156
"
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2157
	self warning:'no # for symbols within array-constants'.
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2158
"
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2159
	^ tokenValue
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2160
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2161
    (tokenType == #HashLeftParen) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2162
"
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2163
	self warning:'no # for arrays within array-constants'.
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2164
"
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2165
	self nextToken.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2166
	^ self array
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2167
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2168
    (tokenType == #HashLeftBrack) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2169
"
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2170
	self warning:'no # for arrays within array-constants'.
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2171
"
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2172
	self nextToken.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2173
	^ self byteArray
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2174
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2175
    (tokenType == #EOF) ifTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2176
	"just for the better error-hilight; let caller handle error"
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2177
	^ #Error
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2178
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2179
    self syntaxError:('error in array-constant; ' 
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2180
		      , tokenType printString 
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2181
		      , ' unexpected').
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2182
    ^ #Error
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2183
! !
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2184
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2185
!Parser methodsFor:'error correction'!
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2186
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2187
correctByDeleting
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2188
    "correct (by deleting token) if user wants to;
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2189
     return #Error if there was no correction;
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2190
     nil if there was one." 
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2191
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2192
    (self confirm:'confirm deleting') ifFalse:[^ #Error].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2193
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2194
    "
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2195
     tell requestor (i.e. CodeView) about the change
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2196
     this will update what the requestor shows.
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2197
    "
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2198
    requestor deleteSelection.
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2199
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2200
    "
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2201
     get the updated source-string 
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2202
     which is needed, when we eventually install the new method
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2203
    "
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2204
    correctedSource := requestor currentSourceCode.
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2205
    ^ nil
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2206
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2207
15
992c3d87edbf *** empty log message ***
claus
parents: 13
diff changeset
  2208
findBestVariablesFor:aString
992c3d87edbf *** empty log message ***
claus
parents: 13
diff changeset
  2209
    "collect known variables with their spelling distances to aString;
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2210
     return the 10 best suggestions"
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2211
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2212
    |names dists searchBlock args vars globalVarName aClass className baseClass n|
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2213
15
992c3d87edbf *** empty log message ***
claus
parents: 13
diff changeset
  2214
    names := OrderedCollection new.
992c3d87edbf *** empty log message ***
claus
parents: 13
diff changeset
  2215
    dists := OrderedCollection new.
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2216
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2217
    "block arguments"
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2218
    searchBlock := currentBlock.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2219
    [searchBlock notNil] whileTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2220
	args := searchBlock arguments.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2221
	args notNil ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2222
	    args do:[:aBlockArg |
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2223
		names add:(aBlockArg name).
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2224
		dists add:(aString spellAgainst: "levenshteinTo:"(aBlockArg name))
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2225
	    ]
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2226
	].
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2227
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2228
	vars := searchBlock variables.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2229
	vars notNil ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2230
	    vars do:[:aBlockVar |
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2231
		names add:(aBlockVar name).
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2232
		dists add:(aString spellAgainst: "levenshteinTo:"(aBlockVar name))
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2233
	    ]
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2234
	].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2235
	searchBlock := searchBlock home
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2236
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2237
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2238
    "method-variables"
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2239
    methodVars notNil ifTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2240
	methodVarNames do:[:methodVarName |
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2241
	    names add:methodVarName.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2242
	    dists add:(aString spellAgainst: "levenshteinTo:"methodVarName)
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2243
	]
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2244
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2245
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2246
    "method-arguments"
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2247
    methodArgs notNil ifTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2248
	methodArgNames do:[:methodArgName |
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2249
	    names add:methodArgName.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2250
	    dists add:(aString spellAgainst: "levenshteinTo:"methodArgName)
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2251
	]
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2252
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2253
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2254
    "instance-variables"
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2255
    classToCompileFor notNil ifTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2256
	PrevInstVarNames do:[:instVarName |
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2257
	    names add:instVarName.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2258
	    dists add:(aString spellAgainst: "levenshteinTo:"instVarName)
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2259
	]
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2260
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2261
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2262
    "class-variables"
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2263
    classToCompileFor notNil ifTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2264
	PrevClassVarNames do:[:classVarName |
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2265
	    names add:classVarName.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2266
	    dists add:(aString spellAgainst: "levenshteinTo:"classVarName)
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2267
	].
7
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
  2268
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
  2269
false ifTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2270
	aClass := classToCompileFor.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2271
	aClass isMeta ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2272
	    className := aClass name.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2273
	    className := className copyTo:(className size - 5).
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2274
	    baseClass := Smalltalk at:(className asSymbol).
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2275
	    baseClass notNil ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2276
		aClass := baseClass
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2277
	    ]
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2278
	].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2279
	[aClass notNil] whileTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2280
	    (aClass classVarNames) do:[:classVarName |
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2281
		names add:classVarName.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2282
		dists add:(aString spellAgainst: "levenshteinTo:"classVarName)
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2283
	    ].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2284
	    aClass := aClass superclass
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2285
	]
7
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
  2286
].
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2287
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2288
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2289
    "globals"
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2290
    Smalltalk allKeysDo:[:aKey |
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2291
	globalVarName := aKey asString.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2292
	"only compare strings where length is about right"
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2293
	((globalVarName size - aString size) abs < 3) ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2294
	    names add:globalVarName.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2295
	    dists add:(aString spellAgainst: "levenshteinTo:"globalVarName)
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2296
	]
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2297
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2298
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2299
    "misc"
7
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
  2300
    #('self' 'super' 'nil' 'thisContext') do:[:name |
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2301
	names add:name.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2302
	dists add:(aString spellAgainst: "levenshteinTo:"name)
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2303
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2304
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2305
    (dists size ~~ 0) ifTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2306
	dists sortWith:names.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2307
	dists := dists reverse.             
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2308
	names := names reverse.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2309
	n := names size min:10.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2310
	names := names copyTo:n.
20
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
  2311
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2312
	"if it starts with a lower case character, add all local & instvar names"
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2313
	(aString at:1) isLowercase ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2314
	    methodVarNames size > 0 ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2315
		names add:'---- method locals ----'.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2316
		methodVarNames do:[:methodVarName |
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2317
		    names add:methodVarName.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2318
		].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2319
	    ].
20
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
  2320
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
  2321
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2322
	    methodArgs notNil ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2323
		names add:'---- method arguments ----'.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2324
		methodArgNames do:[:methodArgName |
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2325
		    names add:methodArgName.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2326
		]
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2327
	    ].
20
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
  2328
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2329
	    names add:'---- instance variables ----'.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2330
	    PrevInstVarNames do:[:instVarName |
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2331
		(names includes:instVarName) ifFalse:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2332
		    names add:instVarName.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2333
		]
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2334
	    ]
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2335
	].
20
f8dd8ba75205 *** empty log message ***
claus
parents: 19
diff changeset
  2336
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2337
	^ names
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2338
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2339
    ^ nil
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2340
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2341
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2342
correctVariable
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2343
    "notify error and correct if user wants to;
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2344
     return #Error if there was no correction 
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2345
     or a ParseNode as returned by variable"
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2346
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2347
    |correctIt varName suggestedNames newName pos1 pos2|
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2348
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2349
    pos1 := tokenPosition.
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2350
    varName := tokenName.
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2351
    pos2 := pos1 + varName size - 1.
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2352
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2353
"OLD:
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2354
    (varName at:1) isLowercase ifTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2355
	correctIt := self undefError:varName position:pos1 to:pos2.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2356
	correctIt ifFalse:[^ #Error]
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2357
    ] ifFalse:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2358
	correctIt := self warning:(varName , ' is undefined') position:pos1 to:pos2.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2359
	correctIt ifFalse:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2360
	    ^ VariableNode type:#GlobalVariable name:(varName asSymbol)
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2361
	]
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2362
    ].
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2363
"
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2364
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2365
    correctIt := self undefError:varName position:pos1 to:pos2.
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2366
    correctIt ifFalse:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2367
	(varName at:1) isLowercase ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2368
	    ^ #Error
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2369
	] ifFalse:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2370
	    ^ VariableNode type:#GlobalVariable name:(varName asSymbol)
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2371
	]
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2372
    ].
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2373
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2374
    suggestedNames := self findBestVariablesFor:varName.
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2375
    suggestedNames notNil ifTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2376
	newName := self askForCorrection:'correct variable to: ' fromList:suggestedNames.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2377
	newName isNil ifTrue:[^ #Error].
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2378
"
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2379
	newName := suggestedNames at:1.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2380
	(self confirm:('confirm correction to: ' , newName)) ifFalse:[^ #Error]
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2381
"
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2382
    ] ifFalse:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2383
	self notify:'no good correction found'.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2384
	^ #Error
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2385
    ].
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2386
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2387
    "
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2388
     tell requestor (i.e. CodeView) about the change
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2389
     this will update what the requestor shows.
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2390
    "
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2391
    requestor replaceSelectionBy:newName.
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2392
    "
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2393
     get the updated source-string 
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2394
     which is needed, when we eventually install the new method
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2395
    "
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2396
    correctedSource := requestor currentSourceCode.
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2397
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2398
    "redo parse with new value"
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2399
    tokenName := newName.
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2400
    ^ self variableOrError
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2401
!
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2402
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2403
askForCorrection:aString fromList:aList
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2404
    "launch a selection box, which allows user to enter correction.
23
1f7bcfff8d39 *** empty log message ***
claus
parents: 21
diff changeset
  2405
     return newString or nil (for abort)"
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2406
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2407
    |box|
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2408
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2409
    ListSelectionBox isNil ifTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2410
	^ self confirm:aString
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2411
    ].
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2412
    box := ListSelectionBox new.
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2413
    box title:aString.
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2414
    box initialText:(aList at:1).
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2415
    box list:aList.
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2416
    box okText:'correct'.
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2417
    box action:[:aString | ^ aString].
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2418
    box showAtPointer.
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2419
    ^ nil
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2420
!
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2421
15
992c3d87edbf *** empty log message ***
claus
parents: 13
diff changeset
  2422
findBestSelectorsFor:aString
992c3d87edbf *** empty log message ***
claus
parents: 13
diff changeset
  2423
    "collect known selectors with their spelling distances to aString;
992c3d87edbf *** empty log message ***
claus
parents: 13
diff changeset
  2424
     return the 10 best suggestions"
992c3d87edbf *** empty log message ***
claus
parents: 13
diff changeset
  2425
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2426
    |info n|
15
992c3d87edbf *** empty log message ***
claus
parents: 13
diff changeset
  2427
992c3d87edbf *** empty log message ***
claus
parents: 13
diff changeset
  2428
    info := SortedCollection new.
992c3d87edbf *** empty log message ***
claus
parents: 13
diff changeset
  2429
    info sortBlock:[:a :b | a value > b value].
992c3d87edbf *** empty log message ***
claus
parents: 13
diff changeset
  2430
992c3d87edbf *** empty log message ***
claus
parents: 13
diff changeset
  2431
    n := 0.
992c3d87edbf *** empty log message ***
claus
parents: 13
diff changeset
  2432
992c3d87edbf *** empty log message ***
claus
parents: 13
diff changeset
  2433
    Symbol allInstancesDo:[:sym |
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2434
	|dist|
15
992c3d87edbf *** empty log message ***
claus
parents: 13
diff changeset
  2435
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2436
	dist := aString spellAgainst:sym.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2437
	dist > 20 ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2438
	    info add:(sym -> dist).
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2439
	    n := n + 1.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2440
	    n > 10 ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2441
		info removeLast.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2442
	    ]
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2443
	]
15
992c3d87edbf *** empty log message ***
claus
parents: 13
diff changeset
  2444
    ].
992c3d87edbf *** empty log message ***
claus
parents: 13
diff changeset
  2445
992c3d87edbf *** empty log message ***
claus
parents: 13
diff changeset
  2446
    ^ info asOrderedCollection collect:[:a | a key]
992c3d87edbf *** empty log message ***
claus
parents: 13
diff changeset
  2447
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2448
    "Time millisecondsToRun:[Parser new findBestSelectorsFor:'foo']"
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2449
    "Parser new findBestSelectorsFor:'findBestSel'"
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2450
    "Parser new findBestSelectorsFor:'fildBestSelectrFr'"
15
992c3d87edbf *** empty log message ***
claus
parents: 13
diff changeset
  2451
!
992c3d87edbf *** empty log message ***
claus
parents: 13
diff changeset
  2452
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2453
correctSelector:aSelectorString message:msg position:pos1 to:pos2
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2454
    "notify error and correct if user wants to;
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2455
     return #Error if there was no correction 
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2456
     or a ParseNode as returned by variable"
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2457
23
1f7bcfff8d39 *** empty log message ***
claus
parents: 21
diff changeset
  2458
    |correctIt suggestedNames newSelector|
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2459
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2460
    "
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2461
     sorry, but I cannot handle keywords with more than one-part
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2462
     currently (too much work - maybe Ill do it later when everything else works :-)
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2463
    "
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2464
    (aSelectorString occurrencesOf:$:) > 1 ifTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2465
	self warning:msg position:pos1 to:pos2.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2466
	^ aSelectorString
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2467
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2468
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2469
    correctIt := self correctableError:msg position:pos1 to:pos2.
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2470
    correctIt ifFalse:[^ aSelectorString].
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2471
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2472
    suggestedNames := self findBestSelectorsFor:aSelectorString.
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2473
    suggestedNames notNil ifTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2474
	newSelector := self askForCorrection:'correct selector to: ' fromList:suggestedNames.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2475
	newSelector isNil ifTrue:[^ aSelectorString].
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2476
    ] ifFalse:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2477
	self notify:'no good correction found'.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2478
	^ aSelectorString
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2479
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2480
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2481
    "
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2482
     tell requestor (i.e. CodeView) about the change
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2483
     this will update what the requestor shows.
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2484
    "
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2485
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2486
    requestor replaceSelectionBy:newSelector keepCursor:false.
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2487
    "
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2488
     get the updated source-string 
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2489
     which is needed, when we eventually install the new method
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2490
    "
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2491
    correctedSource := requestor currentSourceCode.
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2492
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2493
    ^ newSelector
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2494
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2495
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2496
selectorCheck:aSelectorString for:receiver position:pos1 to:pos2
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2497
    "just a quick check: if the selector is totally unknown
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2498
     as a symbol, it cannot possibly be understood.
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2499
     Simple, but catches many typos"
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2500
21
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
  2501
    |ok err sym node|
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
  2502
45
e8331ba8ad5d *** empty log message ***
claus
parents: 44
diff changeset
  2503
    "
e8331ba8ad5d *** empty log message ***
claus
parents: 44
diff changeset
  2504
     if compiling lazy, or errors are to be ignored, or there
e8331ba8ad5d *** empty log message ***
claus
parents: 44
diff changeset
  2505
     is no requestor, do not check
e8331ba8ad5d *** empty log message ***
claus
parents: 44
diff changeset
  2506
    "
21
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
  2507
    (LazyCompilation == true) ifTrue:[^ aSelectorString].
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
  2508
    (ignoreErrors or:[ignoreWarnings]) ifTrue:[^ aSelectorString].
45
e8331ba8ad5d *** empty log message ***
claus
parents: 44
diff changeset
  2509
    requestor isNil ifTrue:[^ aSelectorString].
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2510
21
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
  2511
    err := ' is currently nowhere implemented'.
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
  2512
    "
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
  2513
     check if the selector is known at all
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
  2514
     - if not, it cannot be understood
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
  2515
    "
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
  2516
    ok := aSelectorString knownAsSymbol.
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
  2517
    ok ifTrue:[
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2518
	sym := aSelectorString asSymbol.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2519
	receiver notNil ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2520
	    "
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2521
	     if the receiver is a constant, we can check if it responds
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2522
	     to this selector
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2523
	    "
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2524
	    receiver isConstant ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2525
		ok := receiver evaluate respondsTo:sym.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2526
		err := ' will not be understood here'.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2527
	    ] ifFalse:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2528
		"
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2529
		 if the receiver is a global, we check it too ...
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2530
		"
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2531
		receiver type == #GlobalVariable ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2532
		    ok := receiver evaluate respondsTo:sym.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2533
		    err := ' may not be understood here'.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2534
		] ifFalse:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2535
		    aSelectorString numArgs == 0 ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2536
			"
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2537
			 if the (unary) selector is the name of a variable,
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2538
			 check more (usually, there is a missing '.' somewhere)
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2539
			"
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2540
			err := ' is currently nowhere implemented (''.'' missing ?)'.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2541
			node := self variableOrError:aSelectorString.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2542
			node ~~ #Error ifTrue:[
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2543
			    "
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2544
			     ok, its known as variable too ...
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2545
			    "
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2546
			    ok := false.
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2547
			    Smalltalk allClassesDo:[:aClass |
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2548
				ok := ok or:[aClass implements:sym]
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2549
			    ].
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2550
			]
45
e8331ba8ad5d *** empty log message ***
claus
parents: 44
diff changeset
  2551
		    ].
e8331ba8ad5d *** empty log message ***
claus
parents: 44
diff changeset
  2552
		    ok ifTrue:[
e8331ba8ad5d *** empty log message ***
claus
parents: 44
diff changeset
  2553
			"if its a super send, we can do more checking"
e8331ba8ad5d *** empty log message ***
claus
parents: 44
diff changeset
  2554
			receiver type == #Super ifTrue:[
e8331ba8ad5d *** empty log message ***
claus
parents: 44
diff changeset
  2555
			    (classToCompileFor superclass whichClassImplements:sym) isNil ifTrue:[
e8331ba8ad5d *** empty log message ***
claus
parents: 44
diff changeset
  2556
				err := ' is currently not implemented in any superclass'.
e8331ba8ad5d *** empty log message ***
claus
parents: 44
diff changeset
  2557
				ok := false
e8331ba8ad5d *** empty log message ***
claus
parents: 44
diff changeset
  2558
			    ]
e8331ba8ad5d *** empty log message ***
claus
parents: 44
diff changeset
  2559
			]
e8331ba8ad5d *** empty log message ***
claus
parents: 44
diff changeset
  2560
		    ].
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2561
		]
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2562
	    ]
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2563
	]
21
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
  2564
    ].
338c3cfeffbf *** empty log message ***
claus
parents: 20
diff changeset
  2565
    ok ifFalse:[
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2566
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2567
"OLD: "
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2568
	self warning:('#' , aSelectorString , err) position:pos1 to:pos2
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2569
" "   
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2570
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2571
"NEW:    - not finished - need more interfaces
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2572
   (currently produces warning output on Transcript while filing in
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2573
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2574
44
74ddc944c27f *** empty log message ***
claus
parents: 35
diff changeset
  2575
	^ self correctSelector:aSelectorString message:('#' , aSelectorString , err) position:pos1 to:pos2
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2576
"
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2577
    ].
19
84a1ddf215a5 *** empty log message ***
claus
parents: 15
diff changeset
  2578
    ^ aSelectorString
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  2579
! !