Parser.st
author claus
Sat, 11 Dec 1993 02:09:49 +0100
changeset 7 6c2bc76f0b8f
parent 4 f6fd83437415
child 9 a855c33612fb
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
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
     3
              All Rights Reserved
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
7ad01559b262 Initial revision
claus
parents:
diff changeset
    15
                              contextToEvaluateIn
7ad01559b262 Initial revision
claus
parents:
diff changeset
    16
                              selector
7ad01559b262 Initial revision
claus
parents:
diff changeset
    17
                              methodArgs methodArgNames 
7ad01559b262 Initial revision
claus
parents:
diff changeset
    18
                              methodVars methodVarNames 
7ad01559b262 Initial revision
claus
parents:
diff changeset
    19
                              tree
7ad01559b262 Initial revision
claus
parents:
diff changeset
    20
                              currentBlock
3
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
    21
                              usedInstVars usedClassVars usedVars
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
    22
                              modifiedInstVars modifiedClassVars
7ad01559b262 Initial revision
claus
parents:
diff changeset
    23
                              localVarDefPosition
7ad01559b262 Initial revision
claus
parents:
diff changeset
    24
                              evalExitBlock
3
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
    25
                              selfNode superNode primNr logged
7
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
    26
                              warnedUndefVars'
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
    27
       classVariableNames:'prevClass prevInstVarNames 
7ad01559b262 Initial revision
claus
parents:
diff changeset
    28
                           prevClassVarNames prevClassInstVarNames'
7ad01559b262 Initial revision
claus
parents:
diff changeset
    29
       poolDictionaries:''
7ad01559b262 Initial revision
claus
parents:
diff changeset
    30
       category:'System-Compiler'
7ad01559b262 Initial revision
claus
parents:
diff changeset
    31
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
    32
7ad01559b262 Initial revision
claus
parents:
diff changeset
    33
Parser comment:'
7ad01559b262 Initial revision
claus
parents:
diff changeset
    34
4
f6fd83437415 *** empty log message ***
claus
parents: 3
diff changeset
    35
COPYRIGHT (c) 1989 by Claus Gittinger
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
    36
             All Rights Reserved
7ad01559b262 Initial revision
claus
parents:
diff changeset
    37
7ad01559b262 Initial revision
claus
parents:
diff changeset
    38
Parser is used for both evaluating and compiling smalltalk expressions;
7ad01559b262 Initial revision
claus
parents:
diff changeset
    39
it first builds a parseTree which is then interpreted (evaluate) or
7ad01559b262 Initial revision
claus
parents:
diff changeset
    40
compiled. Compilation is done in the subclass BCompiler.
7ad01559b262 Initial revision
claus
parents:
diff changeset
    41
7ad01559b262 Initial revision
claus
parents:
diff changeset
    42
Parser is also used to find the referenced/modified inst/classvars of
7ad01559b262 Initial revision
claus
parents:
diff changeset
    43
a method - this is done by sending parseXXX message to a parser and asking
7ad01559b262 Initial revision
claus
parents:
diff changeset
    44
the parser for referencedXVars or modifiedXVars (see SystemBrowser).
7ad01559b262 Initial revision
claus
parents:
diff changeset
    45
7
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
    46
$Header: /cvs/stx/stx/libcomp/Parser.st,v 1.4 1993-12-11 01:09:03 claus Exp $
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
    47
'!
7ad01559b262 Initial revision
claus
parents:
diff changeset
    48
7ad01559b262 Initial revision
claus
parents:
diff changeset
    49
!Parser class methodsFor:'evaluating expressions'!
7ad01559b262 Initial revision
claus
parents:
diff changeset
    50
7ad01559b262 Initial revision
claus
parents:
diff changeset
    51
evaluate:aString
7ad01559b262 Initial revision
claus
parents:
diff changeset
    52
    "return the result of evaluating aString"
7ad01559b262 Initial revision
claus
parents:
diff changeset
    53
7ad01559b262 Initial revision
claus
parents:
diff changeset
    54
    ^ self evaluate:aString notifying:nil
7ad01559b262 Initial revision
claus
parents:
diff changeset
    55
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
    56
7ad01559b262 Initial revision
claus
parents:
diff changeset
    57
evaluate:aStringOrStream notifying:requestor
7ad01559b262 Initial revision
claus
parents:
diff changeset
    58
    "return the result of evaluating aString, 
7ad01559b262 Initial revision
claus
parents:
diff changeset
    59
     errors are reported to requestor"
7ad01559b262 Initial revision
claus
parents:
diff changeset
    60
7ad01559b262 Initial revision
claus
parents:
diff changeset
    61
    |parser tree mustBackup|
7ad01559b262 Initial revision
claus
parents:
diff changeset
    62
7ad01559b262 Initial revision
claus
parents:
diff changeset
    63
    aStringOrStream isNil ifTrue:[^ nil].
7ad01559b262 Initial revision
claus
parents:
diff changeset
    64
    aStringOrStream isStream ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
    65
        parser := self for:aStringOrStream.
7ad01559b262 Initial revision
claus
parents:
diff changeset
    66
        mustBackup := true
7ad01559b262 Initial revision
claus
parents:
diff changeset
    67
    ] ifFalse:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
    68
        parser := self for:(ReadStream on:aStringOrStream).
7ad01559b262 Initial revision
claus
parents:
diff changeset
    69
        mustBackup := false
7ad01559b262 Initial revision
claus
parents:
diff changeset
    70
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
    71
    parser notifying:requestor.
7ad01559b262 Initial revision
claus
parents:
diff changeset
    72
    parser nextToken.
7ad01559b262 Initial revision
claus
parents:
diff changeset
    73
    tree := parser parseMethodBodyOrNil.
7ad01559b262 Initial revision
claus
parents:
diff changeset
    74
7ad01559b262 Initial revision
claus
parents:
diff changeset
    75
    "if reading from a stream, backup for next expression"
7ad01559b262 Initial revision
claus
parents:
diff changeset
    76
    mustBackup ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
    77
        parser backupPosition
7ad01559b262 Initial revision
claus
parents:
diff changeset
    78
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
    79
7ad01559b262 Initial revision
claus
parents:
diff changeset
    80
    (parser errorFlag or:[tree == #Error]) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
    81
        ^ #Error
7ad01559b262 Initial revision
claus
parents:
diff changeset
    82
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
    83
    tree notNil ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
    84
        parser evalExitBlock:[:value | ^ value].
7ad01559b262 Initial revision
claus
parents:
diff changeset
    85
        ^ tree evaluate
7ad01559b262 Initial revision
claus
parents:
diff changeset
    86
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
    87
    ^ nil
7ad01559b262 Initial revision
claus
parents:
diff changeset
    88
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
    89
7ad01559b262 Initial revision
claus
parents:
diff changeset
    90
evaluate:aString receiver:anObject notifying:requestor
7ad01559b262 Initial revision
claus
parents:
diff changeset
    91
    "return the result of evaluating aString, 
7ad01559b262 Initial revision
claus
parents:
diff changeset
    92
     errors are reported to requestor. Allow access to
7ad01559b262 Initial revision
claus
parents:
diff changeset
    93
     anObject as self and to its instVars (used in the inspector)"
7ad01559b262 Initial revision
claus
parents:
diff changeset
    94
7ad01559b262 Initial revision
claus
parents:
diff changeset
    95
    ^ self evaluate:aString
7ad01559b262 Initial revision
claus
parents:
diff changeset
    96
                 in:nil
7ad01559b262 Initial revision
claus
parents:
diff changeset
    97
           receiver:anObject
7ad01559b262 Initial revision
claus
parents:
diff changeset
    98
          notifying:requestor
7ad01559b262 Initial revision
claus
parents:
diff changeset
    99
             ifFail:nil
7ad01559b262 Initial revision
claus
parents:
diff changeset
   100
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   101
7ad01559b262 Initial revision
claus
parents:
diff changeset
   102
evaluate:aStringOrStream in:aContext receiver:anObject 
7ad01559b262 Initial revision
claus
parents:
diff changeset
   103
                                    notifying:requestor
7ad01559b262 Initial revision
claus
parents:
diff changeset
   104
                                       ifFail:failBlock
7ad01559b262 Initial revision
claus
parents:
diff changeset
   105
    |parser tree mustBackup|
7ad01559b262 Initial revision
claus
parents:
diff changeset
   106
7ad01559b262 Initial revision
claus
parents:
diff changeset
   107
    aStringOrStream isNil ifTrue:[^ nil].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   108
    aStringOrStream isStream ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   109
        parser := self for:aStringOrStream.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   110
        mustBackup := true
7ad01559b262 Initial revision
claus
parents:
diff changeset
   111
    ] ifFalse:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   112
        parser := self for:(ReadStream on:aStringOrStream).
7ad01559b262 Initial revision
claus
parents:
diff changeset
   113
        mustBackup := false
7ad01559b262 Initial revision
claus
parents:
diff changeset
   114
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   115
    parser setSelf:anObject.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   116
    parser setContext:aContext.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   117
    parser notifying:requestor.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   118
    parser nextToken.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   119
    tree := parser parseMethodBodyOrNil.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   120
7ad01559b262 Initial revision
claus
parents:
diff changeset
   121
    "if reading from a stream, backup for next expression"
7ad01559b262 Initial revision
claus
parents:
diff changeset
   122
    mustBackup ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   123
        parser backupPosition
7ad01559b262 Initial revision
claus
parents:
diff changeset
   124
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   125
7ad01559b262 Initial revision
claus
parents:
diff changeset
   126
    (parser errorFlag or:[tree == #Error]) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   127
        failBlock notNil ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   128
            ^ failBlock value
7ad01559b262 Initial revision
claus
parents:
diff changeset
   129
        ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   130
        ^ #Error
7ad01559b262 Initial revision
claus
parents:
diff changeset
   131
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   132
    tree notNil ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   133
        parser evalExitBlock:[:value | ^ value].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   134
        ^ tree evaluate
7ad01559b262 Initial revision
claus
parents:
diff changeset
   135
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   136
    ^ nil
7ad01559b262 Initial revision
claus
parents:
diff changeset
   137
! !
7ad01559b262 Initial revision
claus
parents:
diff changeset
   138
7ad01559b262 Initial revision
claus
parents:
diff changeset
   139
!Parser class methodsFor:'instance creation'!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   140
7ad01559b262 Initial revision
claus
parents:
diff changeset
   141
for:aStream in:aClass
7ad01559b262 Initial revision
claus
parents:
diff changeset
   142
    |parser|
7ad01559b262 Initial revision
claus
parents:
diff changeset
   143
7ad01559b262 Initial revision
claus
parents:
diff changeset
   144
    parser := self for:aStream.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   145
    parser setClassToCompileFor:aClass.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   146
    ^ parser
7ad01559b262 Initial revision
claus
parents:
diff changeset
   147
! !
7ad01559b262 Initial revision
claus
parents:
diff changeset
   148
7ad01559b262 Initial revision
claus
parents:
diff changeset
   149
!Parser class methodsFor:'parsing'!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   150
7
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   151
selectorInExpression:aString
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   152
    "parse an expression - return the selector. Used for
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   153
     SystemBrowsers implementors/senders query-box initial text"
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   154
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   155
    |tree parser|
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   156
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   157
    (aString isNil or:[aString isEmpty]) ifTrue:[^ nil].
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   158
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   159
    tree := self withSelf:nil parseExpression:aString notifying:nil ignoreErrors:true. 
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   160
    (tree notNil and:[tree ~~ #Error]) ifTrue:[
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   161
	tree isMessage ifTrue:[
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   162
            ^ tree selector
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   163
	].
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   164
    ].
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   165
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   166
    "mhmh, try expression without receiver"
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   167
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   168
    parser := self for:(ReadStream on:aString).
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   169
    parser ignoreErrors.
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   170
    parser nextToken.
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   171
    ^ parser degeneratedKeywordExpressionForSelector
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   172
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   173
"
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   174
    Parser selectorInExpression:'foo at:1 put:(5 * bar)'
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   175
    Parser selectorInExpression:'(foo at:1) at:1'
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   176
    Parser selectorInExpression:'1 + 4'
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   177
    Parser selectorInExpression:'1 negated'
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   178
    Parser selectorInExpression:'at:1 put:5'
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   179
"
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   180
!
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   181
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   182
parseExpression:aString
7ad01559b262 Initial revision
claus
parents:
diff changeset
   183
    "parse aString as an expression; return the parseTree"
7ad01559b262 Initial revision
claus
parents:
diff changeset
   184
7
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   185
    ^ self withSelf:nil parseExpression:aString notifying:nil ignoreErrors:false 
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   186
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   187
7
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   188
withSelf:anObject parseExpression:aString notifying:someOne 
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   189
    "parse aString as an expression with self set to anObject;
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   190
     return the parseTree"
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   191
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   192
    ^ self withSelf:anObject parseExpression:aString notifying:someOne ignoreErrors:false 
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   193
!
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   194
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   195
withSelf:anObject parseExpression:aString notifying:someOne ignoreErrors:ignore
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   196
    "parse aString as an expression with self set to anObject;
7ad01559b262 Initial revision
claus
parents:
diff changeset
   197
     return the parseTree"
7ad01559b262 Initial revision
claus
parents:
diff changeset
   198
7ad01559b262 Initial revision
claus
parents:
diff changeset
   199
    |parser tree|
7ad01559b262 Initial revision
claus
parents:
diff changeset
   200
7ad01559b262 Initial revision
claus
parents:
diff changeset
   201
    aString isNil ifTrue:[^ nil].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   202
    parser := self for:(ReadStream on:aString).
7ad01559b262 Initial revision
claus
parents:
diff changeset
   203
    parser setSelf:anObject.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   204
    parser notifying:someOne.
7
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   205
    ignore ifTrue:[parser ignoreErrors].
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   206
    parser nextToken.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   207
    tree := parser expression.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   208
    (parser errorFlag or:[tree == #Error]) ifTrue:[^ #Error].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   209
    ^ tree
7ad01559b262 Initial revision
claus
parents:
diff changeset
   210
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   211
7ad01559b262 Initial revision
claus
parents:
diff changeset
   212
parseMethodSpecification:aString
7ad01559b262 Initial revision
claus
parents:
diff changeset
   213
    "parse a methods selector & arg specification; 
7ad01559b262 Initial revision
claus
parents:
diff changeset
   214
     return the parser or nil on error"
7ad01559b262 Initial revision
claus
parents:
diff changeset
   215
7ad01559b262 Initial revision
claus
parents:
diff changeset
   216
    ^ self parseMethodSpecification:aString in:nil
7ad01559b262 Initial revision
claus
parents:
diff changeset
   217
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   218
7ad01559b262 Initial revision
claus
parents:
diff changeset
   219
parseMethodSpecification:aString in:aClass
7ad01559b262 Initial revision
claus
parents:
diff changeset
   220
    "parse a methods selector & arg spec for a given class;
7ad01559b262 Initial revision
claus
parents:
diff changeset
   221
     return the parser or nil on error"
7ad01559b262 Initial revision
claus
parents:
diff changeset
   222
7ad01559b262 Initial revision
claus
parents:
diff changeset
   223
    |parser tree|
7ad01559b262 Initial revision
claus
parents:
diff changeset
   224
7ad01559b262 Initial revision
claus
parents:
diff changeset
   225
    aString isNil ifTrue:[^ nil].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   226
    parser := self for:(ReadStream on:aString) in:aClass.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   227
    parser nextToken.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   228
    tree := parser parseMethodSpec.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   229
    (parser errorFlag or:[tree == #Error]) ifTrue:[^ #Error].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   230
    ^ parser
7ad01559b262 Initial revision
claus
parents:
diff changeset
   231
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   232
7ad01559b262 Initial revision
claus
parents:
diff changeset
   233
parseMethodArgAndVarSpecification:aString
7ad01559b262 Initial revision
claus
parents:
diff changeset
   234
    "parse a methods selector, arg and var spec;
7ad01559b262 Initial revision
claus
parents:
diff changeset
   235
     return the parser or nil on error"
7ad01559b262 Initial revision
claus
parents:
diff changeset
   236
7ad01559b262 Initial revision
claus
parents:
diff changeset
   237
    ^ self parseMethodArgAndVarSpecification:aString in:nil
7ad01559b262 Initial revision
claus
parents:
diff changeset
   238
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   239
7ad01559b262 Initial revision
claus
parents:
diff changeset
   240
parseMethodArgAndVarSpecification:aString in:aClass
7ad01559b262 Initial revision
claus
parents:
diff changeset
   241
    "parse a methods selector, arg and var spec for a given class;
7ad01559b262 Initial revision
claus
parents:
diff changeset
   242
     return the parser or nil on error"
7ad01559b262 Initial revision
claus
parents:
diff changeset
   243
7ad01559b262 Initial revision
claus
parents:
diff changeset
   244
    |parser|
7ad01559b262 Initial revision
claus
parents:
diff changeset
   245
7ad01559b262 Initial revision
claus
parents:
diff changeset
   246
    aString isNil ifTrue:[^ nil].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   247
    parser := self for:(ReadStream on:aString) in:aClass.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   248
    parser nextToken.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   249
    (parser parseMethodSpec == #Error) ifTrue:[^ nil].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   250
    (parser parseMethodBodyVarSpec == #Error) ifTrue:[^ nil].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   251
    parser errorFlag ifTrue:[^ nil].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   252
    ^ parser
7ad01559b262 Initial revision
claus
parents:
diff changeset
   253
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   254
7ad01559b262 Initial revision
claus
parents:
diff changeset
   255
parseMethod:aString
7ad01559b262 Initial revision
claus
parents:
diff changeset
   256
    "parse a method; return parseTree"
7ad01559b262 Initial revision
claus
parents:
diff changeset
   257
7ad01559b262 Initial revision
claus
parents:
diff changeset
   258
    ^ self parseMethod:aString in:nil
7ad01559b262 Initial revision
claus
parents:
diff changeset
   259
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   260
7ad01559b262 Initial revision
claus
parents:
diff changeset
   261
parseMethod:aString in:aClass
7ad01559b262 Initial revision
claus
parents:
diff changeset
   262
    "parse a method for a given class; return parser or nil on error"
7ad01559b262 Initial revision
claus
parents:
diff changeset
   263
7ad01559b262 Initial revision
claus
parents:
diff changeset
   264
    |parser tree|
7ad01559b262 Initial revision
claus
parents:
diff changeset
   265
7ad01559b262 Initial revision
claus
parents:
diff changeset
   266
    aString isNil ifTrue:[^ nil].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   267
    parser := self for:(ReadStream on:aString) in:aClass.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   268
    tree := parser parseMethod.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   269
    (parser errorFlag or:[tree == #Error]) ifTrue:[^ nil].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   270
    ^ parser
7ad01559b262 Initial revision
claus
parents:
diff changeset
   271
! !
7ad01559b262 Initial revision
claus
parents:
diff changeset
   272
7ad01559b262 Initial revision
claus
parents:
diff changeset
   273
!Parser methodsFor:'ST-80 compatibility'!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   274
7ad01559b262 Initial revision
claus
parents:
diff changeset
   275
evaluate:aString in:aClass to:to notifying:aRequestor ifFail:failBlock
7ad01559b262 Initial revision
claus
parents:
diff changeset
   276
    |parseTree|
7ad01559b262 Initial revision
claus
parents:
diff changeset
   277
7ad01559b262 Initial revision
claus
parents:
diff changeset
   278
    aString isNil ifTrue:[^ nil].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   279
    self initializeFor:(ReadStream on:aString).
7ad01559b262 Initial revision
claus
parents:
diff changeset
   280
    self setClassToCompileFor:aClass.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   281
    selfValue := nil.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   282
    requestor := aRequestor.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   283
7ad01559b262 Initial revision
claus
parents:
diff changeset
   284
    self nextToken.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   285
    parseTree := self parseMethodBody.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   286
    (errorFlag or:[tree == #Error]) ifTrue:[^ #Error].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   287
    parseTree notNil ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   288
        self evalExitBlock:[:value | ^ failBlock value].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   289
        ^ parseTree evaluate
7ad01559b262 Initial revision
claus
parents:
diff changeset
   290
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   291
    ^ nil
7ad01559b262 Initial revision
claus
parents:
diff changeset
   292
! !
7ad01559b262 Initial revision
claus
parents:
diff changeset
   293
7ad01559b262 Initial revision
claus
parents:
diff changeset
   294
!Parser class methodsFor:'changes'!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   295
7ad01559b262 Initial revision
claus
parents:
diff changeset
   296
update:aClass
7ad01559b262 Initial revision
claus
parents:
diff changeset
   297
    "aClass has changed its definition - flush name caches if we have to"
7ad01559b262 Initial revision
claus
parents:
diff changeset
   298
7ad01559b262 Initial revision
claus
parents:
diff changeset
   299
    (aClass == prevClass) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   300
        prevClass := nil.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   301
        prevInstVarNames := nil.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   302
        prevClassVarNames := nil.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   303
        prevClassInstVarNames := nil.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   304
        aClass removeDependent:Parser
7ad01559b262 Initial revision
claus
parents:
diff changeset
   305
    ]
7
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   306
!
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   307
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   308
flush
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   309
    "unconditional flush name caches"
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   310
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   311
    prevClass notNil ifTrue:[
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   312
        prevClass removeDependent:Parser
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   313
    ].
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   314
    prevClass := nil.
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   315
    prevInstVarNames := nil.
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   316
    prevClassVarNames := nil.
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   317
    prevClassInstVarNames := nil.
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   318
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   319
    "Parser flush"
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   320
! !
7ad01559b262 Initial revision
claus
parents:
diff changeset
   321
7ad01559b262 Initial revision
claus
parents:
diff changeset
   322
!Parser methodsFor:'setup'!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   323
7ad01559b262 Initial revision
claus
parents:
diff changeset
   324
setClassToCompileFor:aClass
7ad01559b262 Initial revision
claus
parents:
diff changeset
   325
    "set the class to be used for parsing/evaluating"
7ad01559b262 Initial revision
claus
parents:
diff changeset
   326
7ad01559b262 Initial revision
claus
parents:
diff changeset
   327
    classToCompileFor := aClass.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   328
    (classToCompileFor ~~ prevClass) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   329
        prevClass notNil ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   330
            Parser update:prevClass
7ad01559b262 Initial revision
claus
parents:
diff changeset
   331
        ]
7ad01559b262 Initial revision
claus
parents:
diff changeset
   332
    ]
7ad01559b262 Initial revision
claus
parents:
diff changeset
   333
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   334
7ad01559b262 Initial revision
claus
parents:
diff changeset
   335
setSelf:anObject
7ad01559b262 Initial revision
claus
parents:
diff changeset
   336
    "set the value to be used for self while evaluating"
7ad01559b262 Initial revision
claus
parents:
diff changeset
   337
7ad01559b262 Initial revision
claus
parents:
diff changeset
   338
    selfValue := anObject.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   339
    classToCompileFor := anObject class.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   340
    (classToCompileFor ~~ prevClass) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   341
        prevClass notNil ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   342
            Parser update:prevClass
7ad01559b262 Initial revision
claus
parents:
diff changeset
   343
        ]
7ad01559b262 Initial revision
claus
parents:
diff changeset
   344
    ]
7ad01559b262 Initial revision
claus
parents:
diff changeset
   345
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   346
7ad01559b262 Initial revision
claus
parents:
diff changeset
   347
setContext:aContext
7ad01559b262 Initial revision
claus
parents:
diff changeset
   348
    "set the context used while evaluating"
7ad01559b262 Initial revision
claus
parents:
diff changeset
   349
7ad01559b262 Initial revision
claus
parents:
diff changeset
   350
    contextToEvaluateIn := aContext
7ad01559b262 Initial revision
claus
parents:
diff changeset
   351
! !
7ad01559b262 Initial revision
claus
parents:
diff changeset
   352
7ad01559b262 Initial revision
claus
parents:
diff changeset
   353
!Parser methodsFor:'accessing'!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   354
7ad01559b262 Initial revision
claus
parents:
diff changeset
   355
tree
7ad01559b262 Initial revision
claus
parents:
diff changeset
   356
    "return the parsetree"
7ad01559b262 Initial revision
claus
parents:
diff changeset
   357
7ad01559b262 Initial revision
claus
parents:
diff changeset
   358
    ^tree
7ad01559b262 Initial revision
claus
parents:
diff changeset
   359
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   360
7ad01559b262 Initial revision
claus
parents:
diff changeset
   361
tree:aTree
7ad01559b262 Initial revision
claus
parents:
diff changeset
   362
    tree := aTree
7ad01559b262 Initial revision
claus
parents:
diff changeset
   363
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   364
7ad01559b262 Initial revision
claus
parents:
diff changeset
   365
selector
7ad01559b262 Initial revision
claus
parents:
diff changeset
   366
    "return the selector"
7ad01559b262 Initial revision
claus
parents:
diff changeset
   367
7ad01559b262 Initial revision
claus
parents:
diff changeset
   368
    ^ selector
7ad01559b262 Initial revision
claus
parents:
diff changeset
   369
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   370
7ad01559b262 Initial revision
claus
parents:
diff changeset
   371
primitiveNumber
7ad01559b262 Initial revision
claus
parents:
diff changeset
   372
    "return the primitiveNumber"
7ad01559b262 Initial revision
claus
parents:
diff changeset
   373
7ad01559b262 Initial revision
claus
parents:
diff changeset
   374
    ^ primNr
7ad01559b262 Initial revision
claus
parents:
diff changeset
   375
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   376
7ad01559b262 Initial revision
claus
parents:
diff changeset
   377
numberOfMethodArgs
7ad01559b262 Initial revision
claus
parents:
diff changeset
   378
    "return the number of methodargs"
7ad01559b262 Initial revision
claus
parents:
diff changeset
   379
7ad01559b262 Initial revision
claus
parents:
diff changeset
   380
    ^ methodArgs size
7ad01559b262 Initial revision
claus
parents:
diff changeset
   381
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   382
7ad01559b262 Initial revision
claus
parents:
diff changeset
   383
methodArgs
7ad01559b262 Initial revision
claus
parents:
diff changeset
   384
    "return an array with methodarg names"
7ad01559b262 Initial revision
claus
parents:
diff changeset
   385
7ad01559b262 Initial revision
claus
parents:
diff changeset
   386
    ^ methodArgNames
7ad01559b262 Initial revision
claus
parents:
diff changeset
   387
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   388
7ad01559b262 Initial revision
claus
parents:
diff changeset
   389
numberOfMethodVars
7ad01559b262 Initial revision
claus
parents:
diff changeset
   390
    "return the number of method variables"
7ad01559b262 Initial revision
claus
parents:
diff changeset
   391
7ad01559b262 Initial revision
claus
parents:
diff changeset
   392
    ^ methodVars size
7ad01559b262 Initial revision
claus
parents:
diff changeset
   393
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   394
7ad01559b262 Initial revision
claus
parents:
diff changeset
   395
methodVars
7ad01559b262 Initial revision
claus
parents:
diff changeset
   396
    "return a collection with method variablenames"
7ad01559b262 Initial revision
claus
parents:
diff changeset
   397
7ad01559b262 Initial revision
claus
parents:
diff changeset
   398
    ^ methodVarNames
7ad01559b262 Initial revision
claus
parents:
diff changeset
   399
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   400
3
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
   401
usedVars
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
   402
    "return a collection with variablenames refd by method"
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
   403
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
   404
    ^ usedVars
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
   405
!
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
   406
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   407
usedInstVars
7ad01559b262 Initial revision
claus
parents:
diff changeset
   408
    "return a collection with instvariablenames refd by method"
7ad01559b262 Initial revision
claus
parents:
diff changeset
   409
7ad01559b262 Initial revision
claus
parents:
diff changeset
   410
    ^ usedInstVars
7ad01559b262 Initial revision
claus
parents:
diff changeset
   411
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   412
7ad01559b262 Initial revision
claus
parents:
diff changeset
   413
usedClassVars
7ad01559b262 Initial revision
claus
parents:
diff changeset
   414
    "return a collection with classvariablenames refd by method"
7ad01559b262 Initial revision
claus
parents:
diff changeset
   415
7ad01559b262 Initial revision
claus
parents:
diff changeset
   416
    ^ usedClassVars
7ad01559b262 Initial revision
claus
parents:
diff changeset
   417
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   418
7ad01559b262 Initial revision
claus
parents:
diff changeset
   419
modifiedInstVars
7ad01559b262 Initial revision
claus
parents:
diff changeset
   420
    "return a collection with instvariablenames modified by method"
7ad01559b262 Initial revision
claus
parents:
diff changeset
   421
7ad01559b262 Initial revision
claus
parents:
diff changeset
   422
    ^ modifiedInstVars
7ad01559b262 Initial revision
claus
parents:
diff changeset
   423
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   424
7ad01559b262 Initial revision
claus
parents:
diff changeset
   425
modifiedClassVars
7ad01559b262 Initial revision
claus
parents:
diff changeset
   426
    "return a collection with classvariablenames modified by method"
7ad01559b262 Initial revision
claus
parents:
diff changeset
   427
7ad01559b262 Initial revision
claus
parents:
diff changeset
   428
    ^ modifiedClassVars
7ad01559b262 Initial revision
claus
parents:
diff changeset
   429
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   430
7ad01559b262 Initial revision
claus
parents:
diff changeset
   431
errorFlag
7ad01559b262 Initial revision
claus
parents:
diff changeset
   432
    ^ errorFlag
7ad01559b262 Initial revision
claus
parents:
diff changeset
   433
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   434
7ad01559b262 Initial revision
claus
parents:
diff changeset
   435
evalExitBlock:aBlock
7ad01559b262 Initial revision
claus
parents:
diff changeset
   436
    "when evaluating a return expression, this block is evaluated"
7ad01559b262 Initial revision
claus
parents:
diff changeset
   437
7ad01559b262 Initial revision
claus
parents:
diff changeset
   438
    evalExitBlock := aBlock
7ad01559b262 Initial revision
claus
parents:
diff changeset
   439
! !
7ad01559b262 Initial revision
claus
parents:
diff changeset
   440
7ad01559b262 Initial revision
claus
parents:
diff changeset
   441
!Parser methodsFor:'error handling'!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   442
7ad01559b262 Initial revision
claus
parents:
diff changeset
   443
showErrorMessage:aMessage position:pos
7ad01559b262 Initial revision
claus
parents:
diff changeset
   444
    Transcript show:(pos printString).
7ad01559b262 Initial revision
claus
parents:
diff changeset
   445
    Transcript show:' '.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   446
    selector notNil ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   447
        Transcript show:aMessage.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   448
        Transcript showCr:(' in ' , selector)
7ad01559b262 Initial revision
claus
parents:
diff changeset
   449
    ] ifFalse:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   450
        Transcript showCr:aMessage
7ad01559b262 Initial revision
claus
parents:
diff changeset
   451
    ]
7ad01559b262 Initial revision
claus
parents:
diff changeset
   452
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   453
7ad01559b262 Initial revision
claus
parents:
diff changeset
   454
parseError:aMessage position:position to:endPos
7ad01559b262 Initial revision
claus
parents:
diff changeset
   455
    "report a syntax error"
7ad01559b262 Initial revision
claus
parents:
diff changeset
   456
7ad01559b262 Initial revision
claus
parents:
diff changeset
   457
    |m|
7ad01559b262 Initial revision
claus
parents:
diff changeset
   458
7ad01559b262 Initial revision
claus
parents:
diff changeset
   459
    errorFlag := true.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   460
    m := ' Error:' , aMessage.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   461
    self notifyError:m position:position to:endPos.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   462
    exitBlock notNil ifTrue:[exitBlock value].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   463
    ^ false
7ad01559b262 Initial revision
claus
parents:
diff changeset
   464
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   465
7ad01559b262 Initial revision
claus
parents:
diff changeset
   466
parseError:aMessage position:position
7ad01559b262 Initial revision
claus
parents:
diff changeset
   467
    "report a syntax error"
7ad01559b262 Initial revision
claus
parents:
diff changeset
   468
7ad01559b262 Initial revision
claus
parents:
diff changeset
   469
    ^ self parseError:aMessage position:position to:nil
7ad01559b262 Initial revision
claus
parents:
diff changeset
   470
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   471
7ad01559b262 Initial revision
claus
parents:
diff changeset
   472
parseError:aMessage
7ad01559b262 Initial revision
claus
parents:
diff changeset
   473
    "report a syntax error"
7ad01559b262 Initial revision
claus
parents:
diff changeset
   474
7ad01559b262 Initial revision
claus
parents:
diff changeset
   475
    ^ self parseError:aMessage position:tokenPosition to:nil
7ad01559b262 Initial revision
claus
parents:
diff changeset
   476
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   477
7ad01559b262 Initial revision
claus
parents:
diff changeset
   478
selectorCheck:aSelectorString position:pos to:pos2
7ad01559b262 Initial revision
claus
parents:
diff changeset
   479
    aSelectorString knownAsSymbol ifFalse:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   480
        self warning:(aSelectorString , ' is currently nowhere implemented') 
7ad01559b262 Initial revision
claus
parents:
diff changeset
   481
            position:pos to:pos2
7ad01559b262 Initial revision
claus
parents:
diff changeset
   482
    ]
7ad01559b262 Initial revision
claus
parents:
diff changeset
   483
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   484
7ad01559b262 Initial revision
claus
parents:
diff changeset
   485
correctableError:message position:pos1 to:pos2
3
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
   486
    "report an error which can be corrected by compiler -
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
   487
     return true if correction is wanted"
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   488
7ad01559b262 Initial revision
claus
parents:
diff changeset
   489
    |correctIt|
7ad01559b262 Initial revision
claus
parents:
diff changeset
   490
7ad01559b262 Initial revision
claus
parents:
diff changeset
   491
    requestor isNil ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   492
        self showErrorMessage:message position:pos1.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   493
        correctIt := false
7ad01559b262 Initial revision
claus
parents:
diff changeset
   494
    ] ifFalse:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   495
        correctIt := requestor correctableError:message position:pos1 to:pos2
7ad01559b262 Initial revision
claus
parents:
diff changeset
   496
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   497
    correctIt ifFalse:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   498
        exitBlock notNil ifTrue:[exitBlock value]
7ad01559b262 Initial revision
claus
parents:
diff changeset
   499
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   500
    ^ correctIt
7ad01559b262 Initial revision
claus
parents:
diff changeset
   501
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   502
7ad01559b262 Initial revision
claus
parents:
diff changeset
   503
undefError:aName position:pos1 to:pos2
3
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
   504
    "report an undefined variable error - return true, if it should be
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
   505
     corrected"
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
   506
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
   507
    requestor isNil ifTrue:[
7
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   508
        warnedUndefVars notNil ifTrue:[
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   509
            (warnedUndefVars includes:aName) ifTrue:[
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   510
                "already warned about this one"
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   511
                ^ false
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   512
            ].
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   513
        ].
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   514
        self showErrorMessage:('Error: ' , aName , ' is undefined') position:pos1.
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   515
        warnedUndefVars isNil ifTrue:[
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   516
            warnedUndefVars := Set new.
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   517
        ].
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   518
        warnedUndefVars add:aName.
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   519
        ^ false
3
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
   520
    ].
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   521
7ad01559b262 Initial revision
claus
parents:
diff changeset
   522
    ^ self correctableError:('Error: ' , aName , ' is undefined') 
7ad01559b262 Initial revision
claus
parents:
diff changeset
   523
                   position:pos1 to:pos2
7ad01559b262 Initial revision
claus
parents:
diff changeset
   524
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   525
7ad01559b262 Initial revision
claus
parents:
diff changeset
   526
exitWith:something
7ad01559b262 Initial revision
claus
parents:
diff changeset
   527
    "this is the longjump out of evaluation via a return expression"
7ad01559b262 Initial revision
claus
parents:
diff changeset
   528
7ad01559b262 Initial revision
claus
parents:
diff changeset
   529
    evalExitBlock value:something
7ad01559b262 Initial revision
claus
parents:
diff changeset
   530
! !
7ad01559b262 Initial revision
claus
parents:
diff changeset
   531
7ad01559b262 Initial revision
claus
parents:
diff changeset
   532
!Parser methodsFor:'parsing'!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   533
7ad01559b262 Initial revision
claus
parents:
diff changeset
   534
parseMethod
7ad01559b262 Initial revision
claus
parents:
diff changeset
   535
    "parse a method"
7ad01559b262 Initial revision
claus
parents:
diff changeset
   536
7ad01559b262 Initial revision
claus
parents:
diff changeset
   537
    |parseTree|
7ad01559b262 Initial revision
claus
parents:
diff changeset
   538
7ad01559b262 Initial revision
claus
parents:
diff changeset
   539
    self nextToken.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   540
    (self parseMethodSpec == #Error) ifTrue:[^ #Error].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   541
    parseTree := self parseMethodBody.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   542
    (parseTree == #Error) ifFalse:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   543
        self tree:parseTree
7ad01559b262 Initial revision
claus
parents:
diff changeset
   544
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   545
    ^ parseTree
7ad01559b262 Initial revision
claus
parents:
diff changeset
   546
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   547
7ad01559b262 Initial revision
claus
parents:
diff changeset
   548
parseMethodSpec
7ad01559b262 Initial revision
claus
parents:
diff changeset
   549
    "parse a methods selector & arg specification;
7ad01559b262 Initial revision
claus
parents:
diff changeset
   550
     set selector and methodArgs as a side effect"
7ad01559b262 Initial revision
claus
parents:
diff changeset
   551
7ad01559b262 Initial revision
claus
parents:
diff changeset
   552
    |var|
7ad01559b262 Initial revision
claus
parents:
diff changeset
   553
7ad01559b262 Initial revision
claus
parents:
diff changeset
   554
    (tokenType == #Keyword) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   555
        selector := ''.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   556
        [tokenType == #Keyword] whileTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   557
            selector := selector , tokenName.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   558
            self nextToken.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   559
            (tokenType ~~ #Identifier) ifTrue:[^ #Error].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   560
            var := Variable name:tokenName.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   561
            methodArgs isNil ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   562
                methodArgs := Array with:var.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   563
                methodArgNames := Array with:tokenName
7ad01559b262 Initial revision
claus
parents:
diff changeset
   564
            ] ifFalse:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   565
                methodArgs := methodArgs copyWith:var.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   566
                methodArgNames := methodArgNames copyWith:tokenName
7ad01559b262 Initial revision
claus
parents:
diff changeset
   567
            ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   568
            self nextToken
7ad01559b262 Initial revision
claus
parents:
diff changeset
   569
        ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   570
        selector := selector asSymbol.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   571
        ^ self
7ad01559b262 Initial revision
claus
parents:
diff changeset
   572
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   573
    (tokenType == #Identifier) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   574
        selector := tokenName asSymbol.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   575
        self nextToken.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   576
        ^ self
7ad01559b262 Initial revision
claus
parents:
diff changeset
   577
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   578
    (tokenType == #BinaryOperator) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   579
        selector := tokenName asSymbol.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   580
        self nextToken.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   581
        (tokenType ~~ #Identifier) ifTrue:[^ #Error].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   582
        var := Variable name:tokenName.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   583
        methodArgs isNil ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   584
            methodArgs := Array with:var.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   585
            methodArgNames := Array with:tokenName
7ad01559b262 Initial revision
claus
parents:
diff changeset
   586
        ] ifFalse:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   587
            methodArgs := methodArgs copyWith:var.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   588
            methodArgNames := methodArgNames copyWith:tokenName
7ad01559b262 Initial revision
claus
parents:
diff changeset
   589
        ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   590
        self nextToken.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   591
        ^ self
7ad01559b262 Initial revision
claus
parents:
diff changeset
   592
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   593
    ^ #Error
7ad01559b262 Initial revision
claus
parents:
diff changeset
   594
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   595
7ad01559b262 Initial revision
claus
parents:
diff changeset
   596
parseMethodBodyOrNil
7ad01559b262 Initial revision
claus
parents:
diff changeset
   597
    "parse a methods body (locals & statements);
7ad01559b262 Initial revision
claus
parents:
diff changeset
   598
     return  a node-tree, nil or #Error. empty (or comment only) input
7ad01559b262 Initial revision
claus
parents:
diff changeset
   599
     is accepted and returns nil"
7ad01559b262 Initial revision
claus
parents:
diff changeset
   600
7ad01559b262 Initial revision
claus
parents:
diff changeset
   601
    |stats|
7ad01559b262 Initial revision
claus
parents:
diff changeset
   602
7ad01559b262 Initial revision
claus
parents:
diff changeset
   603
    ((tokenType == #BinaryOperator) and:[tokenName = '<']) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   604
        "an ST-80 primitive - parsed but ignored"
7ad01559b262 Initial revision
claus
parents:
diff changeset
   605
        self nextToken.
7
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   606
        primNr := self parseST80Primitive.
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   607
        (primNr == #Error) ifTrue:[^ #Error].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   608
        self warning:'ST-80 primitives not supported - ignored'
7ad01559b262 Initial revision
claus
parents:
diff changeset
   609
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   610
7ad01559b262 Initial revision
claus
parents:
diff changeset
   611
    (self parseMethodBodyVarSpec == #Error) ifTrue:[^ #Error].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   612
7ad01559b262 Initial revision
claus
parents:
diff changeset
   613
    (tokenType ~~ #EOF) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   614
        stats := self statementList
7ad01559b262 Initial revision
claus
parents:
diff changeset
   615
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   616
    ^ stats
7ad01559b262 Initial revision
claus
parents:
diff changeset
   617
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   618
7ad01559b262 Initial revision
claus
parents:
diff changeset
   619
parseMethodBody
7ad01559b262 Initial revision
claus
parents:
diff changeset
   620
    "parse a methods body (locals & statements); no more token may follow
7ad01559b262 Initial revision
claus
parents:
diff changeset
   621
    return  a node-tree, nil or #Error"
7ad01559b262 Initial revision
claus
parents:
diff changeset
   622
7ad01559b262 Initial revision
claus
parents:
diff changeset
   623
    |stats|
7ad01559b262 Initial revision
claus
parents:
diff changeset
   624
7ad01559b262 Initial revision
claus
parents:
diff changeset
   625
    stats := self parseMethodBodyOrNil.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   626
    (stats == #Error) ifFalse:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   627
        (tokenType ~~ #EOF) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   628
            self parseError:(tokenType printString , ' unexpected').
7ad01559b262 Initial revision
claus
parents:
diff changeset
   629
            ^#Error
7ad01559b262 Initial revision
claus
parents:
diff changeset
   630
        ]
7ad01559b262 Initial revision
claus
parents:
diff changeset
   631
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   632
    ^ stats
7ad01559b262 Initial revision
claus
parents:
diff changeset
   633
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   634
    
7ad01559b262 Initial revision
claus
parents:
diff changeset
   635
parseMethodBodyVarSpec
7ad01559b262 Initial revision
claus
parents:
diff changeset
   636
    "parse a methods local variable specification"
7ad01559b262 Initial revision
claus
parents:
diff changeset
   637
7ad01559b262 Initial revision
claus
parents:
diff changeset
   638
    |var|
7ad01559b262 Initial revision
claus
parents:
diff changeset
   639
7ad01559b262 Initial revision
claus
parents:
diff changeset
   640
    (tokenType == $|) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   641
        "memorize position for declaration in correction"
7ad01559b262 Initial revision
claus
parents:
diff changeset
   642
        localVarDefPosition := tokenPosition.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   643
        self nextToken.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   644
        [tokenType == #Identifier] whileTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   645
            var := Variable name:tokenName.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   646
            methodVars isNil ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   647
                methodVars := Array with:var.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   648
                methodVarNames := Array with:tokenName
7ad01559b262 Initial revision
claus
parents:
diff changeset
   649
            ] ifFalse:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   650
                methodVars := methodVars copyWith:var.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   651
                methodVarNames := methodVarNames copyWith:tokenName
7ad01559b262 Initial revision
claus
parents:
diff changeset
   652
            ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   653
            self nextToken
7ad01559b262 Initial revision
claus
parents:
diff changeset
   654
        ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   655
        (tokenType ~~ $|) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   656
            self syntaxError:'error in local var specification; | expected.'.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   657
            ^ #Error
7ad01559b262 Initial revision
claus
parents:
diff changeset
   658
        ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   659
        self nextToken
7ad01559b262 Initial revision
claus
parents:
diff changeset
   660
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   661
    ^ nil
7ad01559b262 Initial revision
claus
parents:
diff changeset
   662
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   663
7
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   664
parseST80Primitive
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   665
    "parse an ST-80 type primitive as '< primitive: nr >';
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   666
    return primitive number or #Error"
7ad01559b262 Initial revision
claus
parents:
diff changeset
   667
7ad01559b262 Initial revision
claus
parents:
diff changeset
   668
    |primNumber|
7ad01559b262 Initial revision
claus
parents:
diff changeset
   669
7ad01559b262 Initial revision
claus
parents:
diff changeset
   670
    ((tokenType == #Keyword) and:[tokenName = 'primitive:']) ifFalse:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   671
        self parseError:'bad primitive definition (primitive: expected)'.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   672
        ^ #Error
7ad01559b262 Initial revision
claus
parents:
diff changeset
   673
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   674
    self nextToken.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   675
    (tokenType == #Integer) ifFalse:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   676
        self parseError:'primitive number expected'.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   677
        ^ #Error
7ad01559b262 Initial revision
claus
parents:
diff changeset
   678
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   679
    primNumber := tokenValue.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   680
    self nextToken.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   681
    ((tokenType == #BinaryOperator) and:[tokenName = '>']) ifFalse:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   682
        self parseError:'bad primitive definition (> expected)'.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   683
        ^ #Error
7ad01559b262 Initial revision
claus
parents:
diff changeset
   684
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   685
    self nextToken.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   686
    ^ primNumber
7ad01559b262 Initial revision
claus
parents:
diff changeset
   687
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   688
7ad01559b262 Initial revision
claus
parents:
diff changeset
   689
statementList
7ad01559b262 Initial revision
claus
parents:
diff changeset
   690
    "parse a statementlist; return a node-tree, nil or #Error.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   691
     Statements must be separated by periods."
7ad01559b262 Initial revision
claus
parents:
diff changeset
   692
7ad01559b262 Initial revision
claus
parents:
diff changeset
   693
    |thisStatement prevStatement firstStatement correctIt periodPos|
7ad01559b262 Initial revision
claus
parents:
diff changeset
   694
7ad01559b262 Initial revision
claus
parents:
diff changeset
   695
    thisStatement := self statement.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   696
    (thisStatement == #Error) ifTrue:[^ #Error].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   697
    firstStatement := thisStatement.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   698
    [tokenType == $.] whileTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   699
        periodPos := tokenPosition.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   700
        self nextToken.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   701
        (tokenType == $]) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   702
            currentBlock isNil ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   703
                self parseError:'block nesting error'.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   704
                errorFlag := true
7ad01559b262 Initial revision
claus
parents:
diff changeset
   705
"
7ad01559b262 Initial revision
claus
parents:
diff changeset
   706
            *** I had a warning here (since it was not defined
7ad01559b262 Initial revision
claus
parents:
diff changeset
   707
            *** in the blue-book; but PD-code contains a lot of
7ad01559b262 Initial revision
claus
parents:
diff changeset
   708
            *** code with periods at the end so that the warnings
7ad01559b262 Initial revision
claus
parents:
diff changeset
   709
            *** became annoying
7ad01559b262 Initial revision
claus
parents:
diff changeset
   710
7ad01559b262 Initial revision
claus
parents:
diff changeset
   711
            ] ifFalse:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   712
                self warning:'period after last statement' position:periodPos
7ad01559b262 Initial revision
claus
parents:
diff changeset
   713
"
7ad01559b262 Initial revision
claus
parents:
diff changeset
   714
            ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   715
            ^ firstStatement
7ad01559b262 Initial revision
claus
parents:
diff changeset
   716
        ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   717
        (tokenType == #EOF) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   718
            currentBlock notNil ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   719
                self parseError:'block nesting error'.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   720
                errorFlag := true
7ad01559b262 Initial revision
claus
parents:
diff changeset
   721
"
7ad01559b262 Initial revision
claus
parents:
diff changeset
   722
            *** I had a warning here (since it was not defined
7ad01559b262 Initial revision
claus
parents:
diff changeset
   723
            *** in the blue-book; but PD-code contains a lot of
7ad01559b262 Initial revision
claus
parents:
diff changeset
   724
            *** code with periods at the end so that the warnings
7ad01559b262 Initial revision
claus
parents:
diff changeset
   725
            *** became annoying
7ad01559b262 Initial revision
claus
parents:
diff changeset
   726
7ad01559b262 Initial revision
claus
parents:
diff changeset
   727
            ] ifFalse:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   728
                self warning:'period after last statement' position:periodPos
7ad01559b262 Initial revision
claus
parents:
diff changeset
   729
"
7ad01559b262 Initial revision
claus
parents:
diff changeset
   730
            ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   731
            ^ firstStatement
7ad01559b262 Initial revision
claus
parents:
diff changeset
   732
        ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   733
7ad01559b262 Initial revision
claus
parents:
diff changeset
   734
        prevStatement := thisStatement.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   735
        (prevStatement isKindOf:ReturnNode) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   736
            self warning:'statements after return' position:tokenPosition
7ad01559b262 Initial revision
claus
parents:
diff changeset
   737
        ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   738
"
7ad01559b262 Initial revision
claus
parents:
diff changeset
   739
        periodPos := tokenPosition.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   740
        self nextToken.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   741
"
7ad01559b262 Initial revision
claus
parents:
diff changeset
   742
7ad01559b262 Initial revision
claus
parents:
diff changeset
   743
        ((tokenType == $]) or:[tokenType == #EOF]) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   744
            (currentBlock isNil and:[tokenType == $]]) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   745
                self parseError:'block nesting error'.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   746
                errorFlag := true
7ad01559b262 Initial revision
claus
parents:
diff changeset
   747
            ] ifFalse:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   748
                correctIt := self correctableError:'period after last statement in block'
7ad01559b262 Initial revision
claus
parents:
diff changeset
   749
                                          position:periodPos to:(periodPos + 1).
7ad01559b262 Initial revision
claus
parents:
diff changeset
   750
                correctIt ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   751
                    (self correctByDeleting == #Error) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   752
                        errorFlag := true
7ad01559b262 Initial revision
claus
parents:
diff changeset
   753
                    ]
7ad01559b262 Initial revision
claus
parents:
diff changeset
   754
                ]
7ad01559b262 Initial revision
claus
parents:
diff changeset
   755
            ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   756
            ^ firstStatement
7ad01559b262 Initial revision
claus
parents:
diff changeset
   757
        ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   758
        thisStatement := self statement.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   759
        (thisStatement == #Error) ifTrue:[^ #Error].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   760
        prevStatement nextStatement:thisStatement
7ad01559b262 Initial revision
claus
parents:
diff changeset
   761
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   762
    ^ firstStatement
7ad01559b262 Initial revision
claus
parents:
diff changeset
   763
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   764
7ad01559b262 Initial revision
claus
parents:
diff changeset
   765
statement
7ad01559b262 Initial revision
claus
parents:
diff changeset
   766
    "parse a statement; return a node-tree, nil or #Error"
7ad01559b262 Initial revision
claus
parents:
diff changeset
   767
7ad01559b262 Initial revision
claus
parents:
diff changeset
   768
    |expr node|
7ad01559b262 Initial revision
claus
parents:
diff changeset
   769
7ad01559b262 Initial revision
claus
parents:
diff changeset
   770
    (tokenType == $^) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   771
        self nextToken.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   772
        expr := self expression.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   773
        (expr == #Error) ifTrue:[^ #Error].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   774
        node := ReturnNode expression:expr.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   775
        node home:self blockHome:currentBlock.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   776
        ^ node
7ad01559b262 Initial revision
claus
parents:
diff changeset
   777
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   778
    (tokenType == #Primitive) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   779
        self parseError:'cannot compile primitives (yet)'.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   780
        self nextToken.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   781
        ^ PrimitiveNode code:''
7ad01559b262 Initial revision
claus
parents:
diff changeset
   782
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   783
    (tokenType == #EOF) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   784
        self syntaxError:'period after last statement'.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   785
        ^ #Error
7ad01559b262 Initial revision
claus
parents:
diff changeset
   786
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   787
    expr := self expression.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   788
"
7ad01559b262 Initial revision
claus
parents:
diff changeset
   789
    classToCompileFor notNil ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   790
        currentBlock isNil ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   791
            (expr isKindOf:PrimaryNode) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   792
                self warning:'useless computation - missing ^ ?'
7ad01559b262 Initial revision
claus
parents:
diff changeset
   793
            ]
7ad01559b262 Initial revision
claus
parents:
diff changeset
   794
        ]
7ad01559b262 Initial revision
claus
parents:
diff changeset
   795
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   796
"
7ad01559b262 Initial revision
claus
parents:
diff changeset
   797
    (expr == #Error) ifTrue:[^ #Error].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   798
    ^ StatementNode expression:expr
7ad01559b262 Initial revision
claus
parents:
diff changeset
   799
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   800
7ad01559b262 Initial revision
claus
parents:
diff changeset
   801
expression
7ad01559b262 Initial revision
claus
parents:
diff changeset
   802
    "parse a cascade-expression; return a node-tree, nil or #Error"
7ad01559b262 Initial revision
claus
parents:
diff changeset
   803
7ad01559b262 Initial revision
claus
parents:
diff changeset
   804
    |receiver arg sel args pos pos2|
7ad01559b262 Initial revision
claus
parents:
diff changeset
   805
3
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
   806
    pos := tokenPosition.
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   807
    receiver := self keywordExpression.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   808
    (receiver == #Error) ifTrue:[^ #Error].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   809
    [tokenType == $;] whileTrue:[
3
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
   810
        receiver isMessage ifFalse:[
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
   811
            self syntaxError:'left side of cascade must be a message expression'
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
   812
                    position:pos to:tokenPosition
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
   813
        ].
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   814
        self nextToken.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   815
        (tokenType == #Identifier) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   816
            sel := tokenName.
3
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
   817
            self selectorCheck:sel position:tokenPosition to:(tokenPosition + sel size - 1).
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
   818
            receiver := CascadeNode receiver:receiver selector:sel.
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   819
            self nextToken
7ad01559b262 Initial revision
claus
parents:
diff changeset
   820
        ] ifFalse:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   821
            (tokenType == #BinaryOperator) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   822
                sel := tokenName.
3
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
   823
                self selectorCheck:sel position:tokenPosition to:(tokenPosition + sel size - 1).
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   824
                self nextToken.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   825
                arg := self unaryExpression.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   826
                (arg == #Error) ifTrue:[^ #Error].
3
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
   827
                receiver := CascadeNode receiver:receiver selector:sel arg:arg
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   828
            ] ifFalse:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   829
                (tokenType == #Keyword) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   830
                    pos := tokenPosition.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   831
                    sel := tokenName.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   832
                    self nextToken.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   833
                    arg := self binaryExpression.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   834
                    (arg == #Error) ifTrue:[^ #Error].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   835
                    args := Array with:arg.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   836
                    [tokenType == #Keyword] whileTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   837
                        sel := sel , tokenName.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   838
                        self nextToken.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   839
                        arg := self binaryExpression.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   840
                        (arg == #Error) ifTrue:[^ #Error].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   841
                        args := args copyWith:arg.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   842
                        pos2 := tokenPosition
7ad01559b262 Initial revision
claus
parents:
diff changeset
   843
                    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   844
                    self selectorCheck:sel position:pos to:pos2.
3
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
   845
                    receiver := CascadeNode receiver:receiver selector:sel args:args
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   846
                ] ifFalse:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   847
                    (tokenType == #Error) ifTrue:[^ #Error].
3
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
   848
                    self syntaxError:('invalid cascade; ' , tokenType printString , ' unexpected')
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
   849
                            position:tokenPosition to:source position - 1.
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   850
                    ^ #Error
7ad01559b262 Initial revision
claus
parents:
diff changeset
   851
                ]
7ad01559b262 Initial revision
claus
parents:
diff changeset
   852
            ]
7ad01559b262 Initial revision
claus
parents:
diff changeset
   853
        ]
7ad01559b262 Initial revision
claus
parents:
diff changeset
   854
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   855
    ^ receiver
7ad01559b262 Initial revision
claus
parents:
diff changeset
   856
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   857
7ad01559b262 Initial revision
claus
parents:
diff changeset
   858
keywordExpression
7ad01559b262 Initial revision
claus
parents:
diff changeset
   859
    "parse a keyword-expression; return a node-tree, nil or #Error"
7ad01559b262 Initial revision
claus
parents:
diff changeset
   860
3
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
   861
    |receiver sel arg args pos1 pos2 try lno note|
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   862
7ad01559b262 Initial revision
claus
parents:
diff changeset
   863
    receiver := self binaryExpression.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   864
    (receiver == #Error) ifTrue:[^ #Error].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   865
    (tokenType == #Keyword) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   866
        pos1 := tokenPosition.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   867
        sel := tokenName.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   868
        lno := tokenLineNr.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   869
        self nextToken.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   870
        arg := self binaryExpression.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   871
        (arg == #Error) ifTrue:[^ #Error].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   872
        args := Array with:arg.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   873
        [tokenType == #Keyword] whileTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   874
            sel := sel , tokenName.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   875
            self nextToken.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   876
            arg := self binaryExpression.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   877
            (arg == #Error) ifTrue:[^ #Error].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   878
            args := args copyWith:arg.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   879
            pos2 := tokenPosition
7ad01559b262 Initial revision
claus
parents:
diff changeset
   880
        ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   881
        self selectorCheck:sel position:pos1 to:pos2.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   882
        try := MessageNode receiver:receiver selector:sel args:args.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   883
        (try isMemberOf:String) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   884
            self parseError:try position:pos1 to:pos2.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   885
            receiver := MessageNode receiver:receiver selector:sel args:args
7ad01559b262 Initial revision
claus
parents:
diff changeset
   886
                                        fold:false
7ad01559b262 Initial revision
claus
parents:
diff changeset
   887
        ] ifFalse:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   888
            receiver := try
7ad01559b262 Initial revision
claus
parents:
diff changeset
   889
        ].
3
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
   890
        note := receiver plausibilityCheck.
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
   891
        note notNil ifTrue:[
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
   892
            self warning:note position:pos1 to:pos2
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
   893
        ].
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   894
        receiver lineNumber:lno
7ad01559b262 Initial revision
claus
parents:
diff changeset
   895
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   896
    ^ receiver
7ad01559b262 Initial revision
claus
parents:
diff changeset
   897
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   898
7
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   899
degeneratedKeywordExpressionForSelector
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   900
    "parse a keyword-expression without receiver - for the selector
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   901
     only. return the selector or nil"
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   902
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   903
    |receiver sel arg args pos1 pos2 try lno note|
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   904
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   905
    (tokenType == #Keyword) ifTrue:[
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   906
        sel := tokenName.
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   907
        self nextToken.
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   908
        arg := self binaryExpression.
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   909
        (arg == #Error) ifTrue:[^ sel].
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   910
        [tokenType == #Keyword] whileTrue:[
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   911
            sel := sel , tokenName.
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   912
            self nextToken.
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   913
            arg := self binaryExpression.
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   914
            (arg == #Error) ifTrue:[^ sel].
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   915
        ].
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   916
	^ sel
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   917
    ].
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   918
    ^ nil
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   919
!
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
   920
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
   921
binaryExpression
7ad01559b262 Initial revision
claus
parents:
diff changeset
   922
    "parse a binary-expression; return a node-tree, nil or #Error"
7ad01559b262 Initial revision
claus
parents:
diff changeset
   923
7ad01559b262 Initial revision
claus
parents:
diff changeset
   924
    |receiver arg sel pos try lno note|
7ad01559b262 Initial revision
claus
parents:
diff changeset
   925
7ad01559b262 Initial revision
claus
parents:
diff changeset
   926
    receiver := self unaryExpression.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   927
    (receiver == #Error) ifTrue:[^ #Error].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   928
7ad01559b262 Initial revision
claus
parents:
diff changeset
   929
    "special kludge: since Scanner cannot know if -digit is a binary
7ad01559b262 Initial revision
claus
parents:
diff changeset
   930
     expression or a negative constant, handle cases here"
7ad01559b262 Initial revision
claus
parents:
diff changeset
   931
7ad01559b262 Initial revision
claus
parents:
diff changeset
   932
    [(tokenType == #BinaryOperator) or:[(tokenType == $|)
7ad01559b262 Initial revision
claus
parents:
diff changeset
   933
     or:[(tokenType == #Integer) and:[tokenValue < 0]]]] whileTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   934
        pos := tokenPosition.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   935
7ad01559b262 Initial revision
claus
parents:
diff changeset
   936
        lno := tokenLineNr.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   937
7ad01559b262 Initial revision
claus
parents:
diff changeset
   938
        "kludge here: bar and minus are not scanned as binop "
7ad01559b262 Initial revision
claus
parents:
diff changeset
   939
        (tokenType == $|) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   940
            sel := '|'.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   941
            self nextToken
7ad01559b262 Initial revision
claus
parents:
diff changeset
   942
        ] ifFalse:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   943
            (tokenType == #BinaryOperator) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   944
                sel := tokenName.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   945
                self selectorCheck:sel position:tokenPosition
7ad01559b262 Initial revision
claus
parents:
diff changeset
   946
                                             to:(tokenPosition + sel size - 1).
7ad01559b262 Initial revision
claus
parents:
diff changeset
   947
                self nextToken
7ad01559b262 Initial revision
claus
parents:
diff changeset
   948
            ] ifFalse:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   949
                sel := '-'.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   950
                tokenValue := tokenValue negated
7ad01559b262 Initial revision
claus
parents:
diff changeset
   951
            ]
7ad01559b262 Initial revision
claus
parents:
diff changeset
   952
        ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   953
        arg := self unaryExpression.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   954
        (arg == #Error) ifTrue:[^ #Error].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   955
        try := BinaryNode receiver:receiver selector:sel arg:arg.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   956
        (try isMemberOf:String) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   957
            self parseError:try position:pos to:tokenPosition.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   958
            receiver := BinaryNode receiver:receiver selector:sel arg:arg
7ad01559b262 Initial revision
claus
parents:
diff changeset
   959
                                       fold:false
7ad01559b262 Initial revision
claus
parents:
diff changeset
   960
        ] ifFalse:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   961
            receiver := try
7ad01559b262 Initial revision
claus
parents:
diff changeset
   962
        ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   963
        note := receiver plausibilityCheck.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   964
        note notNil ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   965
            self warning:note position:pos to:tokenPosition
7ad01559b262 Initial revision
claus
parents:
diff changeset
   966
        ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   967
        receiver lineNumber:lno
7ad01559b262 Initial revision
claus
parents:
diff changeset
   968
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   969
    ^ receiver
7ad01559b262 Initial revision
claus
parents:
diff changeset
   970
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   971
7ad01559b262 Initial revision
claus
parents:
diff changeset
   972
unaryExpression
7ad01559b262 Initial revision
claus
parents:
diff changeset
   973
    "parse a unary-expression; return a node-tree, nil or #Error"
7ad01559b262 Initial revision
claus
parents:
diff changeset
   974
7ad01559b262 Initial revision
claus
parents:
diff changeset
   975
    |receiver sel pos try|
7ad01559b262 Initial revision
claus
parents:
diff changeset
   976
7ad01559b262 Initial revision
claus
parents:
diff changeset
   977
    receiver := self primary.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   978
    (receiver == #Error) ifTrue:[^ #Error].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   979
    [tokenType == #Identifier] whileTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   980
        pos := tokenPosition.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   981
        sel := tokenName.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   982
        self selectorCheck:sel position:tokenPosition
7ad01559b262 Initial revision
claus
parents:
diff changeset
   983
                                     to:(tokenPosition + sel size - 1).
7ad01559b262 Initial revision
claus
parents:
diff changeset
   984
        try := UnaryNode receiver:receiver selector:sel.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   985
        (try isMemberOf:String) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   986
            self warning:try position:pos to:(tokenPosition + sel size - 1).
7ad01559b262 Initial revision
claus
parents:
diff changeset
   987
            receiver := UnaryNode receiver:receiver selector:sel fold:false
7ad01559b262 Initial revision
claus
parents:
diff changeset
   988
        ] ifFalse:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
   989
            receiver := try
7ad01559b262 Initial revision
claus
parents:
diff changeset
   990
        ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   991
        receiver lineNumber:tokenLineNr.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   992
        self nextToken.
7ad01559b262 Initial revision
claus
parents:
diff changeset
   993
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
   994
    ^ receiver
7ad01559b262 Initial revision
claus
parents:
diff changeset
   995
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
   996
7ad01559b262 Initial revision
claus
parents:
diff changeset
   997
primary
7ad01559b262 Initial revision
claus
parents:
diff changeset
   998
    "parse a primary-expression; return a node-tree, nil or #Error"
7ad01559b262 Initial revision
claus
parents:
diff changeset
   999
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1000
    |val var expr pos name|
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1001
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1002
    pos := tokenPosition.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1003
    (tokenType == #Identifier) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1004
        var := self variable.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1005
        (var == #Error) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1006
            errorFlag := true
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1007
        ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1008
        self nextToken.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1009
        (tokenType == $_) ifFalse:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1010
            ^ var
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1011
        ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1012
        (var ~~ #Error) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1013
            (var type == #MethodArg) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1014
                self parseError:'assignment to method argument' position:pos to:tokenPosition.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1015
                errorFlag := true
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1016
            ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1017
            (var type == #BlockArg) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1018
                self parseError:'assignment to block argument' position:pos to:tokenPosition.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1019
                errorFlag := true
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1020
            ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1021
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1022
            (var type == #InstanceVariable) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1023
                modifiedInstVars isNil ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1024
                    modifiedInstVars := OrderedCollection new
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1025
                ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1026
                name := prevInstVarNames at:(var index).
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1027
                (modifiedInstVars includes:name) ifFalse:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1028
                    modifiedInstVars add:name
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1029
                ]
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1030
            ] ifFalse:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1031
                (var type == #ClassVariable) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1032
                    modifiedClassVars isNil ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1033
                        modifiedClassVars := OrderedCollection new
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1034
                    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1035
                    name := var name.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1036
                    name := name copyFrom:((name indexOf:$:) + 1).
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1037
                    (modifiedClassVars includes:name) ifFalse:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1038
                        modifiedClassVars add:name
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1039
                    ]
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1040
                ]
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1041
            ]
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1042
        ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1043
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1044
        self nextToken.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1045
        expr := self expression.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1046
        (errorFlag or:[expr == #Error]) ifTrue:[^ #Error].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1047
        ^ AssignmentNode variable:var expression:expr
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1048
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1049
    ((tokenType == #Integer) or:
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1050
     [(tokenType == #Character) or:
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1051
      [tokenType == #Float]]) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1052
        val := ConstantNode type:tokenType value:tokenValue.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1053
        self nextToken.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1054
        (tokenType == $_) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1055
            self parseError:'assignment to a constant' position:pos to:tokenPosition.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1056
            ^ #Error
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1057
        ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1058
        ^ val
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1059
    ].
3
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
  1060
    (tokenType == #Self) ifTrue:[
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
  1061
        self nextToken.
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
  1062
        (tokenType == $_) ifTrue:[
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
  1063
            self parseError:'assignment to self' position:pos to:tokenPosition.
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
  1064
            ^ #Error
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
  1065
        ].
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
  1066
        selfNode isNil ifTrue:[
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
  1067
            selfNode := PrimaryNode type:#Self value:selfValue
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
  1068
        ].
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
  1069
        ^ selfNode
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
  1070
    ].
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1071
    (tokenType == #String) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1072
        val := ConstantNode type:tokenType value:tokenValue.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1073
        self nextToken.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1074
        (tokenType == $_) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1075
            self parseError:'assignment to a constant' position:pos to:tokenPosition.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1076
            ^ #Error
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1077
        ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1078
        ^ val
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1079
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1080
    (tokenType == #Symbol) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1081
        val := ConstantNode type:tokenType value:tokenValue.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1082
        self nextToken.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1083
        (tokenType == $_) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1084
            self parseError:'assignment to a constant' position:pos to:tokenPosition.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1085
            ^ #Error
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1086
        ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1087
        ^ val
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1088
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1089
    (tokenType == #Nil) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1090
        self nextToken.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1091
        (tokenType == $_) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1092
            self parseError:'assignment to nil' position:pos to:tokenPosition.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1093
            ^ #Error
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1094
        ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1095
        ^ ConstantNode type:#Nil value:nil
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1096
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1097
    (tokenType == #True) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1098
        self nextToken.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1099
        (tokenType == $_) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1100
            self parseError:'assignment to true' position:pos to:tokenPosition.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1101
            ^ #Error
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1102
        ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1103
        ^ ConstantNode type:#True value:true
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1104
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1105
    (tokenType == #False) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1106
        self nextToken.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1107
        (tokenType == $_) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1108
            self parseError:'assignment to false' position:pos to:tokenPosition.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1109
            ^ #Error
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1110
        ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1111
        ^ ConstantNode type:#False value:false
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1112
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1113
    (tokenType  == #Super) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1114
        self nextToken.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1115
        (tokenType == $_) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1116
            self parseError:'assignment to super' position:pos to:tokenPosition.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1117
            ^ #Error
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1118
        ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1119
        superNode isNil ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1120
            superNode := PrimaryNode type:#Super value:selfValue
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1121
        ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1122
        ^ superNode
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1123
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1124
    (tokenType == #ThisContext) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1125
        self nextToken.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1126
        (tokenType == $_) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1127
            self parseError:'assignment to thisContext' position:pos to:tokenPosition.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1128
            ^ #Error
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1129
        ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1130
        ^ PrimaryNode type:#ThisContext value:nil
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1131
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1132
    (tokenType == #HashLeftParen) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1133
        self nextToken.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1134
        val := self array.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1135
        self nextToken.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1136
        ^ ConstantNode type:#Array value:val
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1137
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1138
    (tokenType == #HashLeftBrack) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1139
        self nextToken.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1140
        val := self byteArray.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1141
        self nextToken.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1142
        ^ ConstantNode type:#Array value:val
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1143
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1144
    (tokenType == $() ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1145
        self nextToken.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1146
        val := self expression.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1147
        (val == #Error) ifTrue:[^ #Error].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1148
        (tokenType ~~ $) ) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1149
            (tokenType isMemberOf:Character) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1150
                self syntaxError:'missing '')'' (i.e. ''' , tokenType asString , ''' unexpected)' withCRs position:pos to:tokenPosition.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1151
            ] ifFalse:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1152
                self syntaxError:'missing '')''' position:pos to:tokenPosition.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1153
            ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1154
            ^ #Error
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1155
        ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1156
        self nextToken.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1157
        ^ val
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1158
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1159
    (tokenType == $[ ) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1160
        val := self block.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1161
        self nextToken.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1162
        ^ val
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1163
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1164
    (tokenType == #Error) ifTrue:[^ #Error].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1165
    (tokenType isKindOf:Character) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1166
        self syntaxError:('error in primary; ' 
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1167
                           , tokenType printString ,
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1168
                           ' unexpected') position:tokenPosition to:tokenPosition
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1169
    ] ifFalse:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1170
        self syntaxError:('error in primary; ' 
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1171
                           , tokenType printString ,
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1172
                           ' unexpected') 
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1173
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1174
    ^ #Error
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1175
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1176
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1177
variableOrError
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1178
    "parse a variable; return a node-tree, nil or #Error"
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1179
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1180
    |tokenFound var instIndex aClass searchBlock args vars
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1181
     varName tokenSymbol theBlock className
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1182
     runIndex "{ Class: SmallInteger }" |
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1183
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1184
    varName := tokenName.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1185
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1186
    "is it a block-arg or block-var ?"
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1187
    searchBlock := currentBlock.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1188
    [searchBlock notNil] whileTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1189
        runIndex := 1.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1190
        args := searchBlock arguments.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1191
        args notNil ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1192
            args do:[:aBlockArg |
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1193
                (aBlockArg name = varName) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1194
                    tokenFound := aBlockArg.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1195
                    instIndex := runIndex.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1196
                    theBlock := searchBlock
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1197
                ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1198
                runIndex := runIndex + 1
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1199
            ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1200
            tokenFound notNil ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1201
                ^ PrimaryNode type:#BlockArg
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1202
                              name:varName
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1203
                             token:tokenFound
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1204
                             index:instIndex
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1205
                             block:theBlock
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1206
            ]
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1207
        ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1208
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1209
        runIndex := 1.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1210
        vars := searchBlock variables.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1211
        vars notNil ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1212
            vars do:[:aBlockVar |
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1213
                (aBlockVar name = varName) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1214
                    tokenFound := aBlockVar.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1215
                    instIndex := runIndex.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1216
                    theBlock := searchBlock
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1217
                ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1218
                runIndex := runIndex + 1
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1219
            ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1220
            tokenFound notNil ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1221
                ^ PrimaryNode type:#BlockVariable
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1222
                              name:varName
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1223
                             token:tokenFound
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1224
                             index:instIndex
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1225
                             block:theBlock
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1226
            ]
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1227
        ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1228
        searchBlock := searchBlock home
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1229
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1230
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1231
    "is it a method-variable ?"
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1232
    methodVars notNil ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1233
        instIndex := methodVarNames indexOf:varName.
7
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
  1234
        instIndex ~~ 0 ifTrue:[
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1235
            var := methodVars at:instIndex.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1236
            var used:true.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1237
            ^ PrimaryNode type:#MethodVariable
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1238
                          name:varName
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1239
                         token:var
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1240
                         index:instIndex
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1241
        ]
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1242
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1243
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1244
    "is it a method-argument ?"
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1245
    methodArgs notNil ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1246
        instIndex := methodArgNames indexOf:varName.
7
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
  1247
        instIndex ~~ 0 ifTrue:[
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1248
            ^ PrimaryNode type:#MethodArg
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1249
                          name:varName
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1250
                         token:(methodArgs at:instIndex)
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1251
                         index:instIndex
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1252
        ]
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1253
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1254
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1255
    "is it an instance-variable ?"
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1256
    classToCompileFor notNil ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1257
        "caching allInstVarNames for next compilation saves time ..."
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1258
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1259
        (prevInstVarNames isNil or:[prevClass ~~ classToCompileFor]) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1260
            prevClass notNil ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1261
                prevClass removeDependent:Parser
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1262
            ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1263
            prevClass := classToCompileFor.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1264
            prevInstVarNames := classToCompileFor allInstVarNames.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1265
            prevClassInstVarNames := nil.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1266
            prevClassVarNames := nil.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1267
            prevClass addDependent:Parser
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1268
        ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1269
7
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
  1270
        instIndex := prevInstVarNames indexOf:varName startingAt:1.
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
  1271
        instIndex ~~ 0 ifTrue:[
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1272
            usedInstVars isNil ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1273
                usedInstVars := OrderedCollection new
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1274
            ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1275
            (usedInstVars includes:varName) ifFalse:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1276
                usedInstVars add:varName
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1277
            ].
3
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
  1278
            usedVars isNil ifTrue:[
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
  1279
                usedVars := OrderedCollection new
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
  1280
            ].
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
  1281
            (usedVars includes:varName) ifFalse:[
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
  1282
                usedVars add:varName
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
  1283
            ].
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1284
            ^ PrimaryNode type:#InstanceVariable 
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1285
                          name:varName
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1286
                         index:instIndex
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1287
                     selfValue:selfValue
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1288
        ]
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1289
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1290
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1291
    "is it a class-instance-variable ?"
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1292
    classToCompileFor notNil ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1293
        prevClassInstVarNames isNil ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1294
            prevClassInstVarNames := classToCompileFor class allInstVarNames
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1295
        ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1296
7
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
  1297
        instIndex := prevClassInstVarNames indexOf:varName startingAt:1.
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
  1298
        instIndex ~~ 0 ifTrue:[
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1299
            aClass := self inWhichClassIsClassInstVar:varName.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1300
            aClass notNil ifTrue:[
3
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
  1301
                usedVars isNil ifTrue:[
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
  1302
                    usedVars := OrderedCollection new
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
  1303
                ].
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
  1304
                (usedVars includes:varName) ifFalse:[
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
  1305
                    usedVars add:varName
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
  1306
                ].
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1307
                ^ PrimaryNode type:#ClassInstanceVariable
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1308
                              name:varName
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1309
                             index:instIndex
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1310
                         selfValue:selfValue
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1311
            ]
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1312
        ]
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1313
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1314
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1315
    "is it a class-variable ?"
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1316
    classToCompileFor notNil ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1317
        prevClassVarNames isNil ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1318
            aClass := classToCompileFor.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1319
            classToCompileFor isMeta ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1320
                className := aClass name.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1321
                className := className copyFrom:1 to:(className size - 5).
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1322
                aClass := Smalltalk at:(className asSymbol).
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1323
                aClass isNil ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1324
                    aClass := classToCompileFor
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1325
                ]
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1326
            ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1327
            prevClassVarNames := aClass allClassVarNames
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1328
        ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1329
7
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
  1330
        instIndex := prevClassVarNames indexOf:varName startingAt:1.
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
  1331
        instIndex ~~ 0 ifTrue:[
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1332
            aClass := self inWhichClassIsClassVar:varName.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1333
            aClass notNil ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1334
                usedClassVars isNil ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1335
                    usedClassVars := OrderedCollection new
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1336
                ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1337
                (usedClassVars includes:varName) ifFalse:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1338
                    usedClassVars add:varName
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1339
                ].
3
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
  1340
                usedVars isNil ifTrue:[
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
  1341
                    usedVars := OrderedCollection new
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
  1342
                ].
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
  1343
                (usedVars includes:varName) ifFalse:[
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
  1344
                    usedVars add:varName
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
  1345
                ].
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1346
                ^ PrimaryNode type:#ClassVariable 
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1347
                              name:(aClass name , ':' , varName) asSymbol
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1348
            ]
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1349
        ]
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1350
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1351
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1352
    "is it a global-variable ?"
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1353
    tokenSymbol := varName asSymbol.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1354
    (Smalltalk includesKey:tokenSymbol) ifTrue:[
3
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
  1355
        usedVars isNil ifTrue:[
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
  1356
            usedVars := OrderedCollection new
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
  1357
        ].
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
  1358
        (usedVars includes:varName) ifFalse:[
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
  1359
            usedVars add:varName
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
  1360
        ].
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1361
        ^ PrimaryNode type:#GlobalVariable 
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1362
                      name:tokenSymbol
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1363
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1364
    ^ #Error
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1365
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1366
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1367
variable
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1368
    "parse a variable; if undefined, notify error and correct if user wants to"
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1369
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1370
    |v|
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1371
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1372
    v := self variableOrError.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1373
    (v == #Error) ifFalse:[^ v].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1374
    v := self correctVariable.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1375
    (v == #Error) ifFalse:[^ v].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1376
    ^ PrimaryNode type:#GlobalVariable
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1377
                  name:tokenName asSymbol
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1378
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1379
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1380
inWhichClassIsClassVar:aString
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1381
    "search class-chain for the classvariable named aString
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1382
     - return the class or nil if not found"
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1383
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1384
    |aClass className baseClass|
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1385
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1386
    aClass := classToCompileFor.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1387
    aClass isMeta ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1388
        className := aClass name.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1389
        className := className copyFrom:1 to:(className size - 5).
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1390
        baseClass := Smalltalk at:(className asSymbol).
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1391
        baseClass notNil ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1392
            aClass := baseClass
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1393
        ]
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1394
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1395
    [aClass notNil] whileTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1396
        (aClass classVarNames includes:aString) ifTrue:[ ^ aClass].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1397
        aClass := aClass superclass
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1398
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1399
    ^ nil
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1400
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1401
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1402
inWhichClassIsClassInstVar:aString
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1403
    "search class-chain for the class-instance variable named aString
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1404
     - return the class or nil if not found"
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1405
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1406
    |aClass|
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1407
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1408
    aClass := classToCompileFor.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1409
    [aClass notNil] whileTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1410
        (aClass class instVarNames includes:aString) ifTrue:[ ^ aClass].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1411
        aClass := aClass superclass
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1412
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1413
    ^ nil
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1414
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1415
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1416
block
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1417
    "parse a block; return a node-tree, nil or #Error"
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1418
7
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
  1419
    |stats node args var vars pos lno b|
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1420
7
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
  1421
    lno := tokenLineNr.
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1422
    self nextToken.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1423
    (tokenType == $: ) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1424
        [tokenType == $:] whileTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1425
            pos := tokenPosition.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1426
            self nextToken.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1427
            (tokenType == #Identifier) ifFalse:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1428
                self syntaxError:'Identifier expected in block-arg declaration'
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1429
                        position:pos to:tokenPosition-1.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1430
                ^ #Error
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1431
            ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1432
            var := Variable name:tokenName.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1433
            args isNil ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1434
                args := Array with:var
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1435
            ] ifFalse:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1436
                args := args copyWith:var
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1437
            ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1438
            self nextToken
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1439
        ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1440
        (tokenType ~~ $| ) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1441
            "ST-80 allows [:arg ]"
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1442
            (tokenType == $] ) ifTrue:[
7
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
  1443
                b := BlockNode arguments:args home:currentBlock variables:nil.
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
  1444
                b lineNumber:lno.
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
  1445
                ^ b
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1446
            ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1447
            self syntaxError:'| expected after block-arg declaration'.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1448
            ^ #Error
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1449
        ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1450
        self nextToken
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1451
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1452
    (tokenType == $| ) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1453
        self nextToken.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1454
        pos := tokenPosition.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1455
        [tokenType == $|] whileFalse:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1456
            (tokenType == #Identifier) ifFalse:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1457
                self syntaxError:'Identifier expected in block-var declaration' position:pos.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1458
                ^ #Error
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1459
            ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1460
            var := Variable name:tokenName.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1461
            vars isNil ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1462
                vars := Array with:var
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1463
            ] ifFalse:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1464
                vars := vars copyWith:var
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1465
            ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1466
            self nextToken
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1467
        ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1468
        self nextToken
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1469
    ].
3
b63b8a6b71fb *** empty log message ***
claus
parents: 0
diff changeset
  1470
    node := BlockNode arguments:args home:currentBlock variables:vars.
7
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
  1471
    node lineNumber:lno.
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1472
    currentBlock := node.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1473
    stats := self blockStatementList.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1474
    node statements:stats.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1475
    currentBlock := node home.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1476
    (stats == #Error) ifTrue:[^ #Error].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1477
    ^ node
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1478
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1479
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1480
blockStatementList
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1481
    "parse a blocks statementlist; return a node-tree, nil or #Error"
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1482
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1483
    |thisStatement prevStatement firstStatement|
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1484
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1485
    (tokenType == $] ) ifTrue:[^ nil].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1486
    thisStatement := self statement.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1487
    (thisStatement == #Error) ifTrue:[^ #Error].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1488
    firstStatement := thisStatement.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1489
    [tokenType == $] ] whileFalse:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1490
        (tokenType == $.) ifFalse:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1491
            (tokenType == #EOF) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1492
                self syntaxError:'missing '']'' in block'
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1493
            ] ifFalse:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1494
                self syntaxError:'missing ''.'' in block'
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1495
            ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1496
            ^ #Error
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1497
        ] ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1498
            prevStatement := thisStatement.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1499
            self nextToken.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1500
            tokenType == $] ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1501
"
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1502
                *** I had a warning here (since it was not defined
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1503
                *** in the blue-book; but PD-code contains a lot of
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1504
                *** code with periods at the end so that the warnings
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1505
                *** became annoying
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1506
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1507
                self warning:'period after last statement in block'.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1508
"
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1509
                ^ firstStatement
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1510
            ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1511
            thisStatement := self statement.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1512
            (thisStatement == #Error) ifTrue:[^ #Error].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1513
            prevStatement nextStatement:thisStatement
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1514
        ]
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1515
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1516
    ^ firstStatement
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1517
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1518
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1519
array
7
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
  1520
    |arr elem pos1|
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1521
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1522
    pos1 := tokenPosition.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1523
    arr := OrderedCollection new:200.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1524
    [tokenType ~~ $) ] whileTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1525
        elem := self arrayConstant.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1526
        (elem == #Error) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1527
            (tokenType == #EOF) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1528
                self syntaxError:'unterminated array-constant; '')'' expected' 
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1529
                        position:pos1 to:tokenPosition
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1530
            ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1531
            ^ #Error
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1532
        ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1533
        arr add:elem.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1534
        self nextToken
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1535
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1536
    ^ Array withAll:arr
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1537
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1538
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1539
byteArray
7
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
  1540
    "started with ST-80 R4 - allow byteArray constants as #[ ... ]"
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
  1541
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1542
    |arr elem pos1 pos2|
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1543
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1544
    pos1 := tokenPosition.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1545
    arr := OrderedCollection new.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1546
    [tokenType ~~ $] ] whileTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1547
        pos2 := tokenPosition.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1548
        elem := self arrayConstant.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1549
        (elem == #Error) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1550
            (tokenType == #EOF) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1551
                self syntaxError:'unterminated bytearray-constant; '']'' expected' 
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1552
                        position:pos1 to:tokenPosition
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1553
            ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1554
            ^ #Error
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1555
        ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1556
        ((elem isMemberOf:SmallInteger) and:
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1557
        [(elem >= 0) and:[elem <= 255]]) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1558
            arr add:elem
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1559
        ] ifFalse:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1560
            self parseError:'invalid ByteArray element' position:pos2 to:tokenPosition - 1
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1561
        ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1562
        self nextToken
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1563
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1564
    ^ ByteArray withAll:arr
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1565
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1566
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1567
arrayConstant
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1568
    (tokenType == #String) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1569
        ^ tokenValue
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1570
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1571
    (tokenType == #Nil) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1572
        ^ nil
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1573
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1574
    (tokenType == #Integer) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1575
        ^ tokenValue
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1576
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1577
    (tokenType == #Character) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1578
        ^ tokenValue
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1579
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1580
    (tokenType == #Float) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1581
        ^ tokenValue
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1582
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1583
    (tokenType == #True) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1584
        ^ true
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1585
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1586
    (tokenType == #False) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1587
        ^ false
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1588
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1589
    (tokenType == #Error) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1590
        ^ #Error
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1591
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1592
    (tokenType == #BinaryOperator) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1593
        ^ tokenName asSymbol
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1594
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1595
    (tokenType == #Keyword) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1596
        ^ tokenName asSymbol
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1597
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1598
    (tokenType == #Identifier) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1599
        ^ tokenName asSymbol
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1600
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1601
    (tokenType == $() ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1602
        self nextToken.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1603
        ^ self array
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1604
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1605
    (tokenType == $[) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1606
        self nextToken.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1607
        ^ self byteArray
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1608
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1609
    (tokenType == #Symbol) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1610
"
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1611
        self warning:'no # for symbols within array-constants'.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1612
"
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1613
        ^ tokenValue
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1614
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1615
    (tokenType == #HashLeftParen) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1616
"
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1617
        self warning:'no # for arrays within array-constants'.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1618
"
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1619
        self nextToken.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1620
        ^ self array
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1621
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1622
    (tokenType == #HashLeftBrack) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1623
"
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1624
        self warning:'no # for arrays within array-constants'.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1625
"
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1626
        self nextToken.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1627
        ^ self byteArray
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1628
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1629
    (tokenType == #EOF) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1630
        "just for the better error-hilight; let caller handle error"
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1631
        ^ #Error
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1632
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1633
    self syntaxError:('error in array-constant; ' 
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1634
                      , tokenType printString 
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1635
                      , ' unexpected').
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1636
    ^ #Error
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1637
! !
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1638
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1639
!Parser methodsFor:'error correction'!
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1640
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1641
correctByDeleting
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1642
    "correct (by deleting token) if user wants to;
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1643
     return #Error if there was no correction or nil" 
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1644
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1645
    (self confirm:'confirm deleting') ifFalse:[^ #Error].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1646
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1647
    "tell requestor about the change"
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1648
    requestor deleteSelection.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1649
    ^ nil
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1650
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1651
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1652
findBestVariableFor:aString
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1653
    "collect known variables with their levenshtein distances to aString;
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1654
     return the 10 best suggestions"
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1655
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1656
    |names dists searchBlock args vars globalVarName aClass className baseClass n|
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1657
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1658
    names := VariableArray new.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1659
    dists := VariableArray new.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1660
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1661
    "block arguments"
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1662
    searchBlock := currentBlock.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1663
    [searchBlock notNil] whileTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1664
        args := searchBlock arguments.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1665
        args notNil ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1666
            args do:[:aBlockArg |
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1667
                names add:(aBlockArg name).
7
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
  1668
                dists add:(aString spellAgainst: "levenshteinTo:"(aBlockArg name))
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1669
            ]
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1670
        ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1671
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1672
        vars := searchBlock variables.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1673
        vars notNil ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1674
            vars do:[:aBlockVar |
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1675
                names add:(aBlockVar name).
7
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
  1676
                dists add:(aString spellAgainst: "levenshteinTo:"(aBlockVar name))
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1677
            ]
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1678
        ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1679
        searchBlock := searchBlock home
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1680
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1681
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1682
    "method-variables"
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1683
    methodVars notNil ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1684
        methodVarNames do:[:methodVarName |
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1685
            names add:methodVarName.
7
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
  1686
            dists add:(aString spellAgainst: "levenshteinTo:"methodVarName)
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1687
        ]
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1688
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1689
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1690
    "method-arguments"
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1691
    methodArgs notNil ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1692
        methodArgNames do:[:methodArgName |
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1693
            names add:methodArgName.
7
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
  1694
            dists add:(aString spellAgainst: "levenshteinTo:"methodArgName)
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1695
        ]
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1696
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1697
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1698
    "instance-variables"
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1699
    classToCompileFor notNil ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1700
        prevInstVarNames do:[:instVarName |
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1701
            names add:instVarName.
7
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
  1702
            dists add:(aString spellAgainst: "levenshteinTo:"instVarName)
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1703
        ]
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1704
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1705
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1706
    "class-variables"
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1707
    classToCompileFor notNil ifTrue:[
7
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
  1708
        prevClassVarNames do:[:classVarName |
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
  1709
            names add:classVarName.
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
  1710
            dists add:(aString spellAgainst: "levenshteinTo:"classVarName)
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
  1711
        ].
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
  1712
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
  1713
false ifTrue:[
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1714
        aClass := classToCompileFor.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1715
        aClass isMeta ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1716
            className := aClass name.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1717
            className := className copyFrom:1 to:(className size - 5).
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1718
            baseClass := Smalltalk at:(className asSymbol).
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1719
            baseClass notNil ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1720
                aClass := baseClass
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1721
            ]
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1722
        ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1723
        [aClass notNil] whileTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1724
            (aClass classVarNames) do:[:classVarName |
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1725
                names add:classVarName.
7
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
  1726
                dists add:(aString spellAgainst: "levenshteinTo:"classVarName)
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1727
            ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1728
            aClass := aClass superclass
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1729
        ]
7
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
  1730
].
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1731
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1732
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1733
    "globals"
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1734
    Smalltalk allKeysDo:[:aKey |
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1735
        globalVarName := aKey asString.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1736
        "only compare strings where length is about right"
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1737
        ((globalVarName size - aString size) abs < 3) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1738
            names add:globalVarName.
7
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
  1739
            dists add:(aString spellAgainst: "levenshteinTo:"globalVarName)
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1740
        ]
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1741
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1742
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1743
    "misc"
7
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
  1744
    #('self' 'super' 'nil' 'thisContext') do:[:name |
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1745
        "only compare strings where length is about right"
7
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
  1746
        names add:name.
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
  1747
        dists add:(aString spellAgainst: "levenshteinTo:"name)
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1748
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1749
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1750
    (dists size ~~ 0) ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1751
        dists sortWith:names.
7
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
  1752
        dists := dists reverse.             
6c2bc76f0b8f *** empty log message ***
claus
parents: 4
diff changeset
  1753
        names := names reverse.
0
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1754
        n := names size min:10.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1755
        ^ names copyFrom:1 to:n
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1756
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1757
    ^ nil
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1758
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1759
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1760
correctVariable
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1761
    "notify error and correct if user wants to;
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1762
     return #Error if there was no correction 
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1763
     or a ParseNode as returned by variable"
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1764
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1765
    |correctIt varName suggestedNames newName pos1 pos2|
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1766
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1767
    pos1 := tokenPosition.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1768
    varName := tokenName.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1769
    pos2 := pos1 + varName size - 1.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1770
    (varName at:1) isLowercase ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1771
        correctIt := self undefError:varName position:pos1 to:pos2.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1772
        correctIt ifFalse:[^ #Error]
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1773
    ] ifFalse:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1774
        correctIt := self warning:(varName , ' is undefined') position:pos1 to:pos2.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1775
        correctIt ifFalse:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1776
            ^ PrimaryNode type:#GlobalVariable
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1777
                          name:(varName asSymbol)
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1778
        ]
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1779
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1780
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1781
    suggestedNames := self findBestVariableFor:varName.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1782
    suggestedNames notNil ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1783
        newName := self askForVariable:'correct variable to: ' fromList:suggestedNames.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1784
        newName isNil ifTrue:[^ #Error].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1785
"
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1786
        newName := suggestedNames at:1.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1787
        (self confirm:('confirm correction to: ' , newName)) ifFalse:[^ #Error]
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1788
"
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1789
    ] ifFalse:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1790
        self notify:'no good correction found'.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1791
        ^ #Error
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1792
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1793
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1794
    "tell requestor about the change"
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1795
    requestor replaceSelectionBy:newName.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1796
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1797
    "redo parse with new value"
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1798
    tokenName := newName.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1799
    ^ self variableOrError
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1800
!
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1801
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1802
askForVariable:aString fromList:aList
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1803
    "launch a selection box, which allows user to enter correction.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1804
     return true for yes, false for no"
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1805
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1806
    |box|
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1807
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1808
    ListSelectionBox isNil ifTrue:[
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1809
        ^ self confirm:aString
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1810
    ].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1811
    box := ListSelectionBox new.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1812
    box title:aString.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1813
    box initialText:(aList at:1).
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1814
    box list:aList.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1815
    box okText:'replace'.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1816
    box abortText:'abort'.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1817
    box action:[:aString | ^ aString].
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1818
    box showAtPointer.
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1819
    ^ nil
7ad01559b262 Initial revision
claus
parents:
diff changeset
  1820
! !