Context.st
author Claus Gittinger <cg@exept.de>
Mon, 23 Oct 1995 17:55:03 +0100
changeset 443 fae13c0f1512
parent 415 e6ef421822d4
child 528 a083413dfbe8
permissions -rw-r--r--
.
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
     1
"
5
67342904af11 *** empty log message ***
claus
parents: 3
diff changeset
     2
 COPYRIGHT (c) 1988 by Claus Gittinger
154
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
     3
	      All Rights Reserved
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
     4
a27a279701f8 Initial revision
claus
parents:
diff changeset
     5
 This software is furnished under a license and may be used
a27a279701f8 Initial revision
claus
parents:
diff changeset
     6
 only in accordance with the terms of that license and with the
a27a279701f8 Initial revision
claus
parents:
diff changeset
     7
 inclusion of the above copyright notice.   This software may not
a27a279701f8 Initial revision
claus
parents:
diff changeset
     8
 be provided or otherwise made available to, or used by, any
a27a279701f8 Initial revision
claus
parents:
diff changeset
     9
 other person.  No title to or ownership of the software is
a27a279701f8 Initial revision
claus
parents:
diff changeset
    10
 hereby transferred.
a27a279701f8 Initial revision
claus
parents:
diff changeset
    11
"
a27a279701f8 Initial revision
claus
parents:
diff changeset
    12
a27a279701f8 Initial revision
claus
parents:
diff changeset
    13
Object variableSubclass:#Context
a27a279701f8 Initial revision
claus
parents:
diff changeset
    14
       instanceVariableNames:'flags sender home receiver selector searchClass
154
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
    15
			      lineNr retvalTemp handle*'
105
7fe3d60db5e1 invalid return now raises a signal
claus
parents: 92
diff changeset
    16
       classVariableNames:'InvalidReturnSignal'
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
    17
       poolDictionaries:''
a27a279701f8 Initial revision
claus
parents:
diff changeset
    18
       category:'Kernel-Methods'
a27a279701f8 Initial revision
claus
parents:
diff changeset
    19
!
a27a279701f8 Initial revision
claus
parents:
diff changeset
    20
a27a279701f8 Initial revision
claus
parents:
diff changeset
    21
Context comment:'
5
67342904af11 *** empty log message ***
claus
parents: 3
diff changeset
    22
COPYRIGHT (c) 1988 by Claus Gittinger
154
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
    23
	      All Rights Reserved
92
0c73b48551ac *** empty log message ***
claus
parents: 88
diff changeset
    24
415
claus
parents: 406
diff changeset
    25
$Header: /cvs/stx/stx/libbasic/Context.st,v 1.40 1995-08-28 15:47:25 claus Exp $
54
06dbdeeed4f9 *** empty log message ***
claus
parents: 48
diff changeset
    26
'!
06dbdeeed4f9 *** empty log message ***
claus
parents: 48
diff changeset
    27
06dbdeeed4f9 *** empty log message ***
claus
parents: 48
diff changeset
    28
!Context class methodsFor:'documentation'!
06dbdeeed4f9 *** empty log message ***
claus
parents: 48
diff changeset
    29
88
81dacba7a63a *** empty log message ***
claus
parents: 77
diff changeset
    30
copyright
81dacba7a63a *** empty log message ***
claus
parents: 77
diff changeset
    31
"
81dacba7a63a *** empty log message ***
claus
parents: 77
diff changeset
    32
 COPYRIGHT (c) 1988 by Claus Gittinger
154
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
    33
	      All Rights Reserved
88
81dacba7a63a *** empty log message ***
claus
parents: 77
diff changeset
    34
81dacba7a63a *** empty log message ***
claus
parents: 77
diff changeset
    35
 This software is furnished under a license and may be used
81dacba7a63a *** empty log message ***
claus
parents: 77
diff changeset
    36
 only in accordance with the terms of that license and with the
81dacba7a63a *** empty log message ***
claus
parents: 77
diff changeset
    37
 inclusion of the above copyright notice.   This software may not
81dacba7a63a *** empty log message ***
claus
parents: 77
diff changeset
    38
 be provided or otherwise made available to, or used by, any
81dacba7a63a *** empty log message ***
claus
parents: 77
diff changeset
    39
 other person.  No title to or ownership of the software is
81dacba7a63a *** empty log message ***
claus
parents: 77
diff changeset
    40
 hereby transferred.
81dacba7a63a *** empty log message ***
claus
parents: 77
diff changeset
    41
"
81dacba7a63a *** empty log message ***
claus
parents: 77
diff changeset
    42
!
81dacba7a63a *** empty log message ***
claus
parents: 77
diff changeset
    43
81dacba7a63a *** empty log message ***
claus
parents: 77
diff changeset
    44
version
81dacba7a63a *** empty log message ***
claus
parents: 77
diff changeset
    45
"
415
claus
parents: 406
diff changeset
    46
$Header: /cvs/stx/stx/libbasic/Context.st,v 1.40 1995-08-28 15:47:25 claus Exp $
88
81dacba7a63a *** empty log message ***
claus
parents: 77
diff changeset
    47
"
81dacba7a63a *** empty log message ***
claus
parents: 77
diff changeset
    48
!
81dacba7a63a *** empty log message ***
claus
parents: 77
diff changeset
    49
54
06dbdeeed4f9 *** empty log message ***
claus
parents: 48
diff changeset
    50
documentation
06dbdeeed4f9 *** empty log message ***
claus
parents: 48
diff changeset
    51
"
402
claus
parents: 384
diff changeset
    52
    Contexts represent the stack frame objects, which keep the processing
claus
parents: 384
diff changeset
    53
    state of a method or block (i.e. its local variables, temporaries etc.)
claus
parents: 384
diff changeset
    54
    Every message send adds a context to a chain, which can be traced back via 
claus
parents: 384
diff changeset
    55
    the sender field. The context of the currently active method is always
claus
parents: 384
diff changeset
    56
    accessable via the pseuodoVariable called 'thisContext'.
141
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
    57
    (The actual implementation uses the machines stack for this, building real 
402
claus
parents: 384
diff changeset
    58
     contexts on demand only).
54
06dbdeeed4f9 *** empty log message ***
claus
parents: 48
diff changeset
    59
69
4564b6328136 *** empty log message ***
claus
parents: 54
diff changeset
    60
    For both method- and block-contexts, the layout is the same. 
402
claus
parents: 384
diff changeset
    61
    For method contexts, the home-field is nil, while for block contexts the home-
141
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
    62
    field is either the context of its surrounding block (i.e. the context of the
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
    63
    block, in which the receiving block was created, if its a nested block) or of
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
    64
    its home method. 
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
    65
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
    66
    Contexts of cheap blocks do not have a home context - their home field is 
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
    67
    also nil.
54
06dbdeeed4f9 *** empty log message ***
claus
parents: 48
diff changeset
    68
141
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
    69
    Currently, contexts do not contain a reference to the method or block which
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
    70
    created it - this is not needed for program execution, but could get the debugger
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
    71
    somewhat into trouble: it has to search the class hierarchy for receiver/selector
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
    72
    combinations to find the method. This usually works, but fails in case of methods
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
    73
    which are not anchored in any class - especially leading to problems with wrapper-
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
    74
    and lazy methods. Also, Method>>valueWithReceiver - type of invocations cannot
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
    75
    be easily debugged.
402
claus
parents: 384
diff changeset
    76
    Therefore, the implementation may be changed in the near future, to include a
141
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
    77
    field for the method/block, and set it in the VM during program execution.
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
    78
    (there may be some small performance penalty for this, though).
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
    79
402
claus
parents: 384
diff changeset
    80
    LineNumbers vs. program counter:
claus
parents: 384
diff changeset
    81
claus
parents: 384
diff changeset
    82
    Due to the compilation to machine code, methods and/or block do not
216
a8abff749575 *** empty log message ***
claus
parents: 212
diff changeset
    83
    always (actually: do seldom) contain bytecodes. Thus, there is no such concept
a8abff749575 *** empty log message ***
claus
parents: 212
diff changeset
    84
    as a bytecode p-counter. To support debugging, the linenumber within the
a8abff749575 *** empty log message ***
claus
parents: 212
diff changeset
    85
    original source is instead remembered when a send or loop entry is performed.
a8abff749575 *** empty log message ***
claus
parents: 212
diff changeset
    86
    Since linenumbers are not always sufficient for debugging (multiple sends in one
402
claus
parents: 384
diff changeset
    87
    line), this may be changed in future versions to a character offset, giving
216
a8abff749575 *** empty log message ***
claus
parents: 212
diff changeset
    88
    the position of the selector in the source.
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
    89
402
claus
parents: 384
diff changeset
    90
    Restartable / Returnable contexts:
claus
parents: 384
diff changeset
    91
claus
parents: 384
diff changeset
    92
    In previous versions (up to ST/X 2.10.5), every method stored enough
claus
parents: 384
diff changeset
    93
    information in the context for that one to be restartable later (for example,
claus
parents: 384
diff changeset
    94
    via the debuggers restart button). With 2.10.6, this is now an stc-compiler
claus
parents: 384
diff changeset
    95
    option, and the system as delivered is compiled to only create restartable
claus
parents: 384
diff changeset
    96
    contexts for those which contain blocks. This resulted in an overall speedup of
claus
parents: 384
diff changeset
    97
    roughly 10-20% percent, depending on the type of CPU. However, it makes most
claus
parents: 384
diff changeset
    98
    methods non-restartable (however, abort, signal handling and unwind blocks work
claus
parents: 384
diff changeset
    99
    as usual).
claus
parents: 384
diff changeset
   100
    In practice, this was reported to be not a severe limitation and all users were happy
claus
parents: 384
diff changeset
   101
    to trade the increased performance for that slight inconvenience.
claus
parents: 384
diff changeset
   102
    (during development, this is seldom a problem, since interpreted methods are always
claus
parents: 384
diff changeset
   103
     returnable and restartable)
claus
parents: 384
diff changeset
   104
    If you do not like this (and you are a happy owner of the full distribution), you
claus
parents: 384
diff changeset
   105
    should recompile all classes with stc's '-optContext' flag.
claus
parents: 384
diff changeset
   106
claus
parents: 384
diff changeset
   107
    Resuming contexts:
claus
parents: 384
diff changeset
   108
claus
parents: 384
diff changeset
   109
    Strictly speaking, ST/X does not support a context to be resumed. However,
claus
parents: 384
diff changeset
   110
    it does support a forced return (i.e. non-local-return) from a context.
claus
parents: 384
diff changeset
   111
    Thus, resume of a context is implemented by forcing a return from the context
claus
parents: 384
diff changeset
   112
    which was created by the method called from the first one. The effect is the same.
claus
parents: 384
diff changeset
   113
claus
parents: 384
diff changeset
   114
    Returning from a dead method:
claus
parents: 384
diff changeset
   115
claus
parents: 384
diff changeset
   116
    Blocksreturn from an outlived context (i.e. its home method has already returned)
claus
parents: 384
diff changeset
   117
    is now rewarded by an invalidReturn exception - it used to be a noop in previous
claus
parents: 384
diff changeset
   118
    releases (The blue book described this to be a noop, but other smalltalk implementations
claus
parents: 384
diff changeset
   119
    changed this to be an invalid operation -  good decision)
claus
parents: 384
diff changeset
   120
claus
parents: 384
diff changeset
   121
69
4564b6328136 *** empty log message ***
claus
parents: 54
diff changeset
   122
    instance variables:
216
a8abff749575 *** empty log message ***
claus
parents: 212
diff changeset
   123
	flags       <SmallInteger>          used by the VM; never touch.
a8abff749575 *** empty log message ***
claus
parents: 212
diff changeset
   124
					    contains info about number of args, 
a8abff749575 *** empty log message ***
claus
parents: 212
diff changeset
   125
					    locals and temporaries.
a8abff749575 *** empty log message ***
claus
parents: 212
diff changeset
   126
a8abff749575 *** empty log message ***
claus
parents: 212
diff changeset
   127
	sender      <Context>               the 'calling / sending' context
a8abff749575 *** empty log message ***
claus
parents: 212
diff changeset
   128
a8abff749575 *** empty log message ***
claus
parents: 212
diff changeset
   129
	home        <Context>               the context, where this block was 
a8abff749575 *** empty log message ***
claus
parents: 212
diff changeset
   130
					    created, or nil if its a method context
a8abff749575 *** empty log message ***
claus
parents: 212
diff changeset
   131
					    There are also cheap blocks, which do
a8abff749575 *** empty log message ***
claus
parents: 212
diff changeset
   132
					    not need a reference to the home context,
a8abff749575 *** empty log message ***
claus
parents: 212
diff changeset
   133
					    for those, its nil too.
a8abff749575 *** empty log message ***
claus
parents: 212
diff changeset
   134
a8abff749575 *** empty log message ***
claus
parents: 212
diff changeset
   135
	receiver    <Object>                the receiver of this message
a8abff749575 *** empty log message ***
claus
parents: 212
diff changeset
   136
a8abff749575 *** empty log message ***
claus
parents: 212
diff changeset
   137
	selector    <Symbol>                the selector of this message
a8abff749575 *** empty log message ***
claus
parents: 212
diff changeset
   138
a8abff749575 *** empty log message ***
claus
parents: 212
diff changeset
   139
	searchClass <Class>                 the class, where the message lookup started
a8abff749575 *** empty log message ***
claus
parents: 212
diff changeset
   140
					    (for super sends) or nil, for regular sends.
141
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   141
216
a8abff749575 *** empty log message ***
claus
parents: 212
diff changeset
   142
	lineNr      <SmallInteger>          the position where the context left off
339
claus
parents: 328
diff changeset
   143
					    (kind of p-counter). Only the low 16bits
claus
parents: 328
diff changeset
   144
					     are valid.
141
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   145
216
a8abff749575 *** empty log message ***
claus
parents: 212
diff changeset
   146
	retValTemp  nil                     temporary - always nil, when you see the context
a8abff749575 *** empty log message ***
claus
parents: 212
diff changeset
   147
					    (used in the VM as temporary)
a8abff749575 *** empty log message ***
claus
parents: 212
diff changeset
   148
a8abff749575 *** empty log message ***
claus
parents: 212
diff changeset
   149
	handle      *noObject*              used by the VM; not accessable, not an object
141
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   150
216
a8abff749575 *** empty log message ***
claus
parents: 212
diff changeset
   151
	<indexed>                           arguments of the send followed by
a8abff749575 *** empty log message ***
claus
parents: 212
diff changeset
   152
					    locals of the method/block followed by
a8abff749575 *** empty log message ***
claus
parents: 212
diff changeset
   153
					    temporaries.
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
   154
216
a8abff749575 *** empty log message ***
claus
parents: 212
diff changeset
   155
    class variables:
a8abff749575 *** empty log message ***
claus
parents: 212
diff changeset
   156
	InvalidReturnSignal                 signal raised when a block tries
a8abff749575 *** empty log message ***
claus
parents: 212
diff changeset
   157
					    to return ('^') from a method context
a8abff749575 *** empty log message ***
claus
parents: 212
diff changeset
   158
					    which itself has already returned
a8abff749575 *** empty log message ***
claus
parents: 212
diff changeset
   159
					    (i.e. there is no place to return to)
a8abff749575 *** empty log message ***
claus
parents: 212
diff changeset
   160
        
69
4564b6328136 *** empty log message ***
claus
parents: 54
diff changeset
   161
    WARNING: layout and size known by the compiler and runtime system - do not change.
54
06dbdeeed4f9 *** empty log message ***
claus
parents: 48
diff changeset
   162
"
06dbdeeed4f9 *** empty log message ***
claus
parents: 48
diff changeset
   163
! !
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
   164
105
7fe3d60db5e1 invalid return now raises a signal
claus
parents: 92
diff changeset
   165
!Context class methodsFor:'initialization'!
7fe3d60db5e1 invalid return now raises a signal
claus
parents: 92
diff changeset
   166
7fe3d60db5e1 invalid return now raises a signal
claus
parents: 92
diff changeset
   167
initialize
7fe3d60db5e1 invalid return now raises a signal
claus
parents: 92
diff changeset
   168
    InvalidReturnSignal isNil ifTrue:[
302
1f76060d58a4 *** empty log message ***
claus
parents: 293
diff changeset
   169
	InvalidReturnSignal := ErrorSignal newSignalMayProceed:true.
154
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   170
	InvalidReturnSignal nameClass:self message:#invalidReturnSignal.
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   171
	InvalidReturnSignal notifierString:'invalid return; method cannot return twice'.
105
7fe3d60db5e1 invalid return now raises a signal
claus
parents: 92
diff changeset
   172
    ]
7fe3d60db5e1 invalid return now raises a signal
claus
parents: 92
diff changeset
   173
! !
7fe3d60db5e1 invalid return now raises a signal
claus
parents: 92
diff changeset
   174
345
claus
parents: 340
diff changeset
   175
!Context class methodsFor:'Signal constants'!
105
7fe3d60db5e1 invalid return now raises a signal
claus
parents: 92
diff changeset
   176
7fe3d60db5e1 invalid return now raises a signal
claus
parents: 92
diff changeset
   177
invalidReturnSignal
141
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   178
    "return the signal used when a method is tried to be returned twice
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   179
     or, when some dead context is unwound or restarted."
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   180
105
7fe3d60db5e1 invalid return now raises a signal
claus
parents: 92
diff changeset
   181
    ^ InvalidReturnSignal
7fe3d60db5e1 invalid return now raises a signal
claus
parents: 92
diff changeset
   182
! !
7fe3d60db5e1 invalid return now raises a signal
claus
parents: 92
diff changeset
   183
3
24d81bf47225 *** empty log message ***
claus
parents: 1
diff changeset
   184
!Context class methodsFor:'queries'!
24d81bf47225 *** empty log message ***
claus
parents: 1
diff changeset
   185
24d81bf47225 *** empty log message ***
claus
parents: 1
diff changeset
   186
isBuiltInClass
54
06dbdeeed4f9 *** empty log message ***
claus
parents: 48
diff changeset
   187
    "this class & subclasses are known by the run-time-system"
3
24d81bf47225 *** empty log message ***
claus
parents: 1
diff changeset
   188
24d81bf47225 *** empty log message ***
claus
parents: 1
diff changeset
   189
    ^ true
24d81bf47225 *** empty log message ***
claus
parents: 1
diff changeset
   190
! !
24d81bf47225 *** empty log message ***
claus
parents: 1
diff changeset
   191
154
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   192
!Context methodsFor:'copying'!
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   193
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   194
deepCopy
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   195
    "raise an error - deepCopy is not allowed for contexts"
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   196
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   197
    ^ self deepCopyError
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   198
! !
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   199
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
   200
!Context methodsFor:'testing'!
a27a279701f8 Initial revision
claus
parents:
diff changeset
   201
a27a279701f8 Initial revision
claus
parents:
diff changeset
   202
isContext
a27a279701f8 Initial revision
claus
parents:
diff changeset
   203
    "return true, iff the receiver is a Context, false otherwise"
a27a279701f8 Initial revision
claus
parents:
diff changeset
   204
a27a279701f8 Initial revision
claus
parents:
diff changeset
   205
    ^ true
a27a279701f8 Initial revision
claus
parents:
diff changeset
   206
!
a27a279701f8 Initial revision
claus
parents:
diff changeset
   207
a27a279701f8 Initial revision
claus
parents:
diff changeset
   208
isBlockContext
a27a279701f8 Initial revision
claus
parents:
diff changeset
   209
    "return true, iff the receiver is a BlockContext, false otherwise"
a27a279701f8 Initial revision
claus
parents:
diff changeset
   210
a27a279701f8 Initial revision
claus
parents:
diff changeset
   211
    ^ false
54
06dbdeeed4f9 *** empty log message ***
claus
parents: 48
diff changeset
   212
! 
25
e34a6267c79b *** empty log message ***
claus
parents: 12
diff changeset
   213
e34a6267c79b *** empty log message ***
claus
parents: 12
diff changeset
   214
isRecursive
e34a6267c79b *** empty log message ***
claus
parents: 12
diff changeset
   215
    "return true, if this context is one of a recursive send of the same
154
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   216
     selector and same arguments to the same receiver before. 
141
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   217
     Used to detect recursive errors or recursive printing - for example."
25
e34a6267c79b *** empty log message ***
claus
parents: 12
diff changeset
   218
170
40ea50c7b9fe *** empty log message ***
claus
parents: 154
diff changeset
   219
    |c rec "numArgs" "{Class: SmallInteger }"|
25
e34a6267c79b *** empty log message ***
claus
parents: 12
diff changeset
   220
69
4564b6328136 *** empty log message ***
claus
parents: 54
diff changeset
   221
    rec := 0.
25
e34a6267c79b *** empty log message ***
claus
parents: 12
diff changeset
   222
    c := self sender.
e34a6267c79b *** empty log message ***
claus
parents: 12
diff changeset
   223
    [c notNil] whileTrue:[
69
4564b6328136 *** empty log message ***
claus
parents: 54
diff changeset
   224
154
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   225
	((c selector == selector) 
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   226
	and:[(c receiver == receiver)]) ifTrue:[
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   227
	    "
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   228
	     stupid: the current ST/X context does not include
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   229
	     the method, but the class, in which the search started ...
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   230
	    "
328
claus
parents: 326
diff changeset
   231
"/            (c searchClass whichClassIncludesSelector:selector) == (self searchClass whichClassIncludesSelector:selector) ifTrue:[
212
3edd10edefaf *** empty log message ***
claus
parents: 202
diff changeset
   232
	      c methodClass == self methodClass ifTrue:[
170
40ea50c7b9fe *** empty log message ***
claus
parents: 154
diff changeset
   233
"/              "
40ea50c7b9fe *** empty log message ***
claus
parents: 154
diff changeset
   234
"/               finally, look for different arguments
40ea50c7b9fe *** empty log message ***
claus
parents: 154
diff changeset
   235
"/              "
40ea50c7b9fe *** empty log message ***
claus
parents: 154
diff changeset
   236
"/              numArgs := self numArgs.
40ea50c7b9fe *** empty log message ***
claus
parents: 154
diff changeset
   237
"/              1 to:numArgs do:[:argIndex |
40ea50c7b9fe *** empty log message ***
claus
parents: 154
diff changeset
   238
"/                  (self argAt:argIndex) == (c argAt:argIndex) ifFalse:[^ false]
40ea50c7b9fe *** empty log message ***
claus
parents: 154
diff changeset
   239
"/              ]. 
154
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   240
		^ true
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   241
	    ]
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   242
	].
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   243
	c := c sender.
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   244
	"
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   245
	 this special test was added to get out after a while
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   246
	 if the sender chain is corrupt - this gives us at least
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   247
	 a chance to find those errors.
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   248
	"
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   249
	rec := rec + 1.
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   250
	rec >= 100000 ifTrue:[
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   251
	    'bad context chain' errorPrintNL.
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   252
	    ^ true
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   253
	]
25
e34a6267c79b *** empty log message ***
claus
parents: 12
diff changeset
   254
    ].
e34a6267c79b *** empty log message ***
claus
parents: 12
diff changeset
   255
    ^ false
e34a6267c79b *** empty log message ***
claus
parents: 12
diff changeset
   256
! !
e34a6267c79b *** empty log message ***
claus
parents: 12
diff changeset
   257
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
   258
!Context methodsFor:'accessing'!
a27a279701f8 Initial revision
claus
parents:
diff changeset
   259
a27a279701f8 Initial revision
claus
parents:
diff changeset
   260
instVarAt:index
69
4564b6328136 *** empty log message ***
claus
parents: 54
diff changeset
   261
    "have to catch instVar access to retVal and handle - they are invalid.
4564b6328136 *** empty log message ***
claus
parents: 54
diff changeset
   262
     Notice, that one of the next ST/X versions will get some syntactic
4564b6328136 *** empty log message ***
claus
parents: 54
diff changeset
   263
     extension to get this automatically)."
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
   264
a27a279701f8 Initial revision
claus
parents:
diff changeset
   265
    (index == 8) ifTrue:[^ nil].
a27a279701f8 Initial revision
claus
parents:
diff changeset
   266
    (index == 9) ifTrue:[^ nil].
a27a279701f8 Initial revision
claus
parents:
diff changeset
   267
    ^ super instVarAt:index
a27a279701f8 Initial revision
claus
parents:
diff changeset
   268
!
a27a279701f8 Initial revision
claus
parents:
diff changeset
   269
a27a279701f8 Initial revision
claus
parents:
diff changeset
   270
instVarAt:index put:value
69
4564b6328136 *** empty log message ***
claus
parents: 54
diff changeset
   271
    "have to catch instVar access to retVal and handle - they are invalid.
4564b6328136 *** empty log message ***
claus
parents: 54
diff changeset
   272
     Notice, that one of the next ST/X versions will get some syntactic
4564b6328136 *** empty log message ***
claus
parents: 54
diff changeset
   273
     extension to get this automatically)."
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
   274
a27a279701f8 Initial revision
claus
parents:
diff changeset
   275
    (index == 8) ifTrue:[^ nil].
a27a279701f8 Initial revision
claus
parents:
diff changeset
   276
    (index == 9) ifTrue:[^ nil].
a27a279701f8 Initial revision
claus
parents:
diff changeset
   277
    ^ super instVarAt:index put:value
a27a279701f8 Initial revision
claus
parents:
diff changeset
   278
!
a27a279701f8 Initial revision
claus
parents:
diff changeset
   279
a27a279701f8 Initial revision
claus
parents:
diff changeset
   280
methodHome
a27a279701f8 Initial revision
claus
parents:
diff changeset
   281
    "return the method-home - for method contexts this is the receiver"
a27a279701f8 Initial revision
claus
parents:
diff changeset
   282
a27a279701f8 Initial revision
claus
parents:
diff changeset
   283
    ^ self
a27a279701f8 Initial revision
claus
parents:
diff changeset
   284
!
a27a279701f8 Initial revision
claus
parents:
diff changeset
   285
a27a279701f8 Initial revision
claus
parents:
diff changeset
   286
home
a27a279701f8 Initial revision
claus
parents:
diff changeset
   287
    "return the immediate home of the receiver.
a27a279701f8 Initial revision
claus
parents:
diff changeset
   288
     for block contexts, this is the methodcontext, where the block was created,
a27a279701f8 Initial revision
claus
parents:
diff changeset
   289
     for nested block contexts, its the surrounding blocks context.
a27a279701f8 Initial revision
claus
parents:
diff changeset
   290
     for method-contexts this is nil."
a27a279701f8 Initial revision
claus
parents:
diff changeset
   291
a27a279701f8 Initial revision
claus
parents:
diff changeset
   292
    ^ nil "home"
a27a279701f8 Initial revision
claus
parents:
diff changeset
   293
!
a27a279701f8 Initial revision
claus
parents:
diff changeset
   294
a27a279701f8 Initial revision
claus
parents:
diff changeset
   295
method
54
06dbdeeed4f9 *** empty log message ***
claus
parents: 48
diff changeset
   296
    "return the method for which the receiver was created.
06dbdeeed4f9 *** empty log message ***
claus
parents: 48
diff changeset
   297
     To save time during normal execution, this information is not held in the
06dbdeeed4f9 *** empty log message ***
claus
parents: 48
diff changeset
   298
     context, but computed here on request."
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
   299
a27a279701f8 Initial revision
claus
parents:
diff changeset
   300
    |c|
a27a279701f8 Initial revision
claus
parents:
diff changeset
   301
325
claus
parents: 308
diff changeset
   302
    c := self searchClass.
claus
parents: 308
diff changeset
   303
    "
claus
parents: 308
diff changeset
   304
     the below cannot happen in normal circumstances
claus
parents: 308
diff changeset
   305
     (added to avoid recursive errors in case of a broken sender chain)
claus
parents: 308
diff changeset
   306
    "
claus
parents: 308
diff changeset
   307
    c isBehavior ifFalse:[
claus
parents: 308
diff changeset
   308
	'OOPS: non class in searchClass' errorPrintNL.
claus
parents: 308
diff changeset
   309
	'      selector: ' errorPrint. selector errorPrint.
claus
parents: 308
diff changeset
   310
	' receiver: ' errorPrint. receiver errorPrintNL.
claus
parents: 308
diff changeset
   311
	^ nil
claus
parents: 308
diff changeset
   312
    ].
claus
parents: 308
diff changeset
   313
328
claus
parents: 326
diff changeset
   314
    c := c whichClassIncludesSelector:selector.
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
   315
    c notNil ifTrue:[
154
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   316
	^ c compiledMethodAt:selector
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
   317
    ].
216
a8abff749575 *** empty log message ***
claus
parents: 212
diff changeset
   318
a8abff749575 *** empty log message ***
claus
parents: 212
diff changeset
   319
    "mhmh - seems to be a context for an unbound method;
a8abff749575 *** empty log message ***
claus
parents: 212
diff changeset
   320
     look in the senders context. Consider this a kludge.
a8abff749575 *** empty log message ***
claus
parents: 212
diff changeset
   321
     (maybe it was not too good of an idea to not keep the current
325
claus
parents: 308
diff changeset
   322
      method in the context ....
claus
parents: 308
diff changeset
   323
      future versions of ST/X's message lookup may store the method in
claus
parents: 308
diff changeset
   324
      the context.)
216
a8abff749575 *** empty log message ***
claus
parents: 212
diff changeset
   325
    "
a8abff749575 *** empty log message ***
claus
parents: 212
diff changeset
   326
    (sender notNil and:[sender selector startsWith:'valueWithReceiver:']) ifTrue:[
a8abff749575 *** empty log message ***
claus
parents: 212
diff changeset
   327
	^ sender receiver
a8abff749575 *** empty log message ***
claus
parents: 212
diff changeset
   328
    ].
a8abff749575 *** empty log message ***
claus
parents: 212
diff changeset
   329
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
   330
    ^ nil
a27a279701f8 Initial revision
claus
parents:
diff changeset
   331
!
a27a279701f8 Initial revision
claus
parents:
diff changeset
   332
212
3edd10edefaf *** empty log message ***
claus
parents: 202
diff changeset
   333
methodClass
3edd10edefaf *** empty log message ***
claus
parents: 202
diff changeset
   334
    "return the class in which the method for which the receiver was created is."
3edd10edefaf *** empty log message ***
claus
parents: 202
diff changeset
   335
328
claus
parents: 326
diff changeset
   336
    ^ self searchClass whichClassIncludesSelector:selector.
212
3edd10edefaf *** empty log message ***
claus
parents: 202
diff changeset
   337
!
3edd10edefaf *** empty log message ***
claus
parents: 202
diff changeset
   338
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
   339
sender
a27a279701f8 Initial revision
claus
parents:
diff changeset
   340
    "return the sender of the context"
a27a279701f8 Initial revision
claus
parents:
diff changeset
   341
69
4564b6328136 *** empty log message ***
claus
parents: 54
diff changeset
   342
    "this special test is for the very first context (startup-context);
4564b6328136 *** empty log message ***
claus
parents: 54
diff changeset
   343
     actually, its cosmetics, to avoid a visible nil>>nil context in the debugger."
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
   344
69
4564b6328136 *** empty log message ***
claus
parents: 54
diff changeset
   345
"
4564b6328136 *** empty log message ***
claus
parents: 54
diff changeset
   346
    (sender isNil or:[sender selector isNil and:[sender sender isNil]]) ifTrue:[^ nil].
4564b6328136 *** empty log message ***
claus
parents: 54
diff changeset
   347
"
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
   348
    ^ sender
a27a279701f8 Initial revision
claus
parents:
diff changeset
   349
!
a27a279701f8 Initial revision
claus
parents:
diff changeset
   350
a27a279701f8 Initial revision
claus
parents:
diff changeset
   351
receiver
a27a279701f8 Initial revision
claus
parents:
diff changeset
   352
    "return the receiver of the context"
a27a279701f8 Initial revision
claus
parents:
diff changeset
   353
a27a279701f8 Initial revision
claus
parents:
diff changeset
   354
    ^ receiver
a27a279701f8 Initial revision
claus
parents:
diff changeset
   355
!
a27a279701f8 Initial revision
claus
parents:
diff changeset
   356
a27a279701f8 Initial revision
claus
parents:
diff changeset
   357
searchClass
69
4564b6328136 *** empty log message ***
claus
parents: 54
diff changeset
   358
    "this is the class where the method-lookup started;
4564b6328136 *** empty log message ***
claus
parents: 54
diff changeset
   359
     for normal sends, it is nil (or sometimes the receivers class).
4564b6328136 *** empty log message ***
claus
parents: 54
diff changeset
   360
     For supersends, its the superclass of the one, in which the
4564b6328136 *** empty log message ***
claus
parents: 54
diff changeset
   361
     caller was defined."
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
   362
a27a279701f8 Initial revision
claus
parents:
diff changeset
   363
    searchClass notNil ifTrue:[^ searchClass].
a27a279701f8 Initial revision
claus
parents:
diff changeset
   364
    ^ receiver class
a27a279701f8 Initial revision
claus
parents:
diff changeset
   365
!
a27a279701f8 Initial revision
claus
parents:
diff changeset
   366
a27a279701f8 Initial revision
claus
parents:
diff changeset
   367
selector
54
06dbdeeed4f9 *** empty log message ***
claus
parents: 48
diff changeset
   368
    "return the selector of the method for which the context was created"
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
   369
a27a279701f8 Initial revision
claus
parents:
diff changeset
   370
    ^ selector
a27a279701f8 Initial revision
claus
parents:
diff changeset
   371
!
a27a279701f8 Initial revision
claus
parents:
diff changeset
   372
154
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   373
numArgs
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
   374
    "return the number of arguments to the Block/Method"
a27a279701f8 Initial revision
claus
parents:
diff changeset
   375
a27a279701f8 Initial revision
claus
parents:
diff changeset
   376
%{  /* NOCONTEXT */
a27a279701f8 Initial revision
claus
parents:
diff changeset
   377
a27a279701f8 Initial revision
claus
parents:
diff changeset
   378
    RETURN ( _MKSMALLINT( (_intVal(_INST(flags)) >> __NARG_SHIFT) & __NARG_MASK) );
a27a279701f8 Initial revision
claus
parents:
diff changeset
   379
%}
a27a279701f8 Initial revision
claus
parents:
diff changeset
   380
!
a27a279701f8 Initial revision
claus
parents:
diff changeset
   381
a27a279701f8 Initial revision
claus
parents:
diff changeset
   382
nvars
54
06dbdeeed4f9 *** empty log message ***
claus
parents: 48
diff changeset
   383
    "return the number of local variables of the Block/Method"
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
   384
a27a279701f8 Initial revision
claus
parents:
diff changeset
   385
%{  /* NOCONTEXT */
a27a279701f8 Initial revision
claus
parents:
diff changeset
   386
a27a279701f8 Initial revision
claus
parents:
diff changeset
   387
    RETURN ( _MKSMALLINT( (_intVal(_INST(flags)) >> __NVAR_SHIFT) & __NVAR_MASK) );
a27a279701f8 Initial revision
claus
parents:
diff changeset
   388
%}
a27a279701f8 Initial revision
claus
parents:
diff changeset
   389
!
a27a279701f8 Initial revision
claus
parents:
diff changeset
   390
54
06dbdeeed4f9 *** empty log message ***
claus
parents: 48
diff changeset
   391
ntemp
06dbdeeed4f9 *** empty log message ***
claus
parents: 48
diff changeset
   392
    "return the number of temporary variables of the Block/Method.
06dbdeeed4f9 *** empty log message ***
claus
parents: 48
diff changeset
   393
     (for debugging only)"
06dbdeeed4f9 *** empty log message ***
claus
parents: 48
diff changeset
   394
154
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   395
    ^ self size - self numArgs - self nvars
54
06dbdeeed4f9 *** empty log message ***
claus
parents: 48
diff changeset
   396
!
06dbdeeed4f9 *** empty log message ***
claus
parents: 48
diff changeset
   397
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
   398
args
a27a279701f8 Initial revision
claus
parents:
diff changeset
   399
    "return an array filled with the arguments of this context"
a27a279701f8 Initial revision
claus
parents:
diff changeset
   400
308
f04744ef7b5d *** empty log message ***
claus
parents: 302
diff changeset
   401
    |n|
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
   402
154
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   403
    n := self numArgs.
282
94f5c3a6230d *** empty log message ***
claus
parents: 216
diff changeset
   404
    ^ (Array new:n) replaceFrom:1 to:n with:self.
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
   405
!
a27a279701f8 Initial revision
claus
parents:
diff changeset
   406
a27a279701f8 Initial revision
claus
parents:
diff changeset
   407
argsAndVars
a27a279701f8 Initial revision
claus
parents:
diff changeset
   408
    "return an array filled with the arguments and variables of this context"
a27a279701f8 Initial revision
claus
parents:
diff changeset
   409
308
f04744ef7b5d *** empty log message ***
claus
parents: 302
diff changeset
   410
    |n|
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
   411
282
94f5c3a6230d *** empty log message ***
claus
parents: 216
diff changeset
   412
    n := self numArgs + self nvars.
94f5c3a6230d *** empty log message ***
claus
parents: 216
diff changeset
   413
    ^ (Array new:n) replaceFrom:1 to:n with:self.
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
   414
!
a27a279701f8 Initial revision
claus
parents:
diff changeset
   415
340
claus
parents: 339
diff changeset
   416
vars 
claus
parents: 339
diff changeset
   417
    "return an array filled with the local variables of this context"
claus
parents: 339
diff changeset
   418
claus
parents: 339
diff changeset
   419
    |nonVars mySize|
claus
parents: 339
diff changeset
   420
claus
parents: 339
diff changeset
   421
    nonVars := self numArgs.
claus
parents: 339
diff changeset
   422
    mySize := self nvars.
claus
parents: 339
diff changeset
   423
    ^ (Array new:mySize) replaceFrom:1 to:mySize with:self startingAt:nonVars+1
claus
parents: 339
diff changeset
   424
!
claus
parents: 339
diff changeset
   425
202
40ca7cc6fb9c *** empty log message ***
claus
parents: 170
diff changeset
   426
temporaries 
40ca7cc6fb9c *** empty log message ***
claus
parents: 170
diff changeset
   427
    "return an array filled with the temporaries of this context"
40ca7cc6fb9c *** empty log message ***
claus
parents: 170
diff changeset
   428
308
f04744ef7b5d *** empty log message ***
claus
parents: 302
diff changeset
   429
    |nonTemps mySize|
202
40ca7cc6fb9c *** empty log message ***
claus
parents: 170
diff changeset
   430
40ca7cc6fb9c *** empty log message ***
claus
parents: 170
diff changeset
   431
    nonTemps := self numArgs + self nvars.
40ca7cc6fb9c *** empty log message ***
claus
parents: 170
diff changeset
   432
    mySize := self ntemp.
308
f04744ef7b5d *** empty log message ***
claus
parents: 302
diff changeset
   433
    ^ (Array new:mySize) replaceFrom:1 to:mySize with:self startingAt:nonTemps+1
202
40ca7cc6fb9c *** empty log message ***
claus
parents: 170
diff changeset
   434
!
40ca7cc6fb9c *** empty log message ***
claus
parents: 170
diff changeset
   435
10
claus
parents: 5
diff changeset
   436
argAt:n
claus
parents: 5
diff changeset
   437
    "return the n'th argument"
claus
parents: 5
diff changeset
   438
claus
parents: 5
diff changeset
   439
    ^ self at:n
claus
parents: 5
diff changeset
   440
!
claus
parents: 5
diff changeset
   441
claus
parents: 5
diff changeset
   442
argAt:n put:value
claus
parents: 5
diff changeset
   443
    "set the n'th argument - useful when the receiver should be restarted"
claus
parents: 5
diff changeset
   444
claus
parents: 5
diff changeset
   445
    ^ self at:n put:value
claus
parents: 5
diff changeset
   446
!
claus
parents: 5
diff changeset
   447
claus
parents: 5
diff changeset
   448
varAt:n
claus
parents: 5
diff changeset
   449
    "return the n'th local variable"
claus
parents: 5
diff changeset
   450
154
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   451
    ^ self at:(n + self numArgs)
10
claus
parents: 5
diff changeset
   452
!
claus
parents: 5
diff changeset
   453
claus
parents: 5
diff changeset
   454
varAt:n put:value
claus
parents: 5
diff changeset
   455
    "set the n'th local variable - useful when the receiver should be restarted
54
06dbdeeed4f9 *** empty log message ***
claus
parents: 48
diff changeset
   456
     or continued"
10
claus
parents: 5
diff changeset
   457
154
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   458
    self at:(n + self numArgs) put:value
10
claus
parents: 5
diff changeset
   459
!
claus
parents: 5
diff changeset
   460
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
   461
lineNumber
a27a279701f8 Initial revision
claus
parents:
diff changeset
   462
    "this returns the lineNumber within the methods source, where the context was
a27a279701f8 Initial revision
claus
parents:
diff changeset
   463
     interrupted or called another method. (currently, sometimes this information
a27a279701f8 Initial revision
claus
parents:
diff changeset
   464
     is not available - in this case 0 is returned)"
a27a279701f8 Initial revision
claus
parents:
diff changeset
   465
339
claus
parents: 328
diff changeset
   466
    lineNr isNil ifTrue:[^ nil].
claus
parents: 328
diff changeset
   467
    ^ lineNr bitAnd:16rFFFF
360
claus
parents: 345
diff changeset
   468
!
claus
parents: 345
diff changeset
   469
375
claus
parents: 360
diff changeset
   470
setLineNumber:aNumber
claus
parents: 360
diff changeset
   471
    "private entry for uncompiledCodeObject ..."
claus
parents: 360
diff changeset
   472
claus
parents: 360
diff changeset
   473
    lineNr := aNumber
claus
parents: 360
diff changeset
   474
!
claus
parents: 360
diff changeset
   475
360
claus
parents: 345
diff changeset
   476
canReturn
claus
parents: 345
diff changeset
   477
    "return true, if the receiver allows returning through it.
375
claus
parents: 360
diff changeset
   478
     For normal method contexts, this returns true;
360
claus
parents: 345
diff changeset
   479
     for blocks, it (currently) always returns false.
claus
parents: 345
diff changeset
   480
claus
parents: 345
diff changeset
   481
     However, the system can be compiled (for production code), to create
claus
parents: 345
diff changeset
   482
     contexts which cannot be returned or restarted
claus
parents: 345
diff changeset
   483
     (except, if the method contains a returning block). 
claus
parents: 345
diff changeset
   484
     This saves some administrative work in every method
claus
parents: 345
diff changeset
   485
     invocation and makes overall execution faster. However, it limits
claus
parents: 345
diff changeset
   486
     the debugger, in that it cannot return from or restart those contexts.
claus
parents: 345
diff changeset
   487
     (unwinding and termination is not affected by this optimization)
claus
parents: 345
diff changeset
   488
     Currently, the system as delivered has this optimization disabled."
claus
parents: 345
diff changeset
   489
claus
parents: 345
diff changeset
   490
%{  /* NOCONTEXT */
claus
parents: 345
diff changeset
   491
claus
parents: 345
diff changeset
   492
    RETURN ( (_intVal(_INST(flags)) & __CANNOT_RETURN) ? false : true );
claus
parents: 345
diff changeset
   493
%}
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
   494
! !
a27a279701f8 Initial revision
claus
parents:
diff changeset
   495
92
0c73b48551ac *** empty log message ***
claus
parents: 88
diff changeset
   496
!Context methodsFor:'printing & storing'!
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
   497
42
e33491f6f260 *** empty log message ***
claus
parents: 37
diff changeset
   498
argsDisplayString
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
   499
    |fullString n "{ Class: SmallInteger }" |
a27a279701f8 Initial revision
claus
parents:
diff changeset
   500
a27a279701f8 Initial revision
claus
parents:
diff changeset
   501
    fullString := ''.
154
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   502
    n := self numArgs.
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
   503
    1 to:n do:[:index |
154
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   504
	fullString := fullString , (' ' , (self at:index) displayString)
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
   505
    ].
a27a279701f8 Initial revision
claus
parents:
diff changeset
   506
    ^ fullString
a27a279701f8 Initial revision
claus
parents:
diff changeset
   507
!
a27a279701f8 Initial revision
claus
parents:
diff changeset
   508
a27a279701f8 Initial revision
claus
parents:
diff changeset
   509
receiverPrintString
37
d9a302eaa3ef *** empty log message ***
claus
parents: 25
diff changeset
   510
    "return a string describing the receiver of the context" 
d9a302eaa3ef *** empty log message ***
claus
parents: 25
diff changeset
   511
170
40ea50c7b9fe *** empty log message ***
claus
parents: 154
diff changeset
   512
    |receiverClass receiverClassName newString implementorClass|
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
   513
170
40ea50c7b9fe *** empty log message ***
claus
parents: 154
diff changeset
   514
    receiverClass := receiver class.
40ea50c7b9fe *** empty log message ***
claus
parents: 154
diff changeset
   515
    receiverClassName := receiverClass name.
40ea50c7b9fe *** empty log message ***
claus
parents: 154
diff changeset
   516
    (receiverClass == SmallInteger) ifTrue:[
154
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   517
	newString := '(' , receiver printString , ') ' , receiverClassName
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
   518
    ] ifFalse:[
154
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   519
	newString := receiverClassName
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
   520
    ].
a27a279701f8 Initial revision
claus
parents:
diff changeset
   521
a27a279701f8 Initial revision
claus
parents:
diff changeset
   522
    selector notNil ifTrue:[
328
claus
parents: 326
diff changeset
   523
"/        implementorClass := self searchClass whichClassIncludesSelector:selector.
293
31df3850e98c *** empty log message ***
claus
parents: 282
diff changeset
   524
31df3850e98c *** empty log message ***
claus
parents: 282
diff changeset
   525
	"
31df3850e98c *** empty log message ***
claus
parents: 282
diff changeset
   526
	 kludge to avoid slow search for containing class
31df3850e98c *** empty log message ***
claus
parents: 282
diff changeset
   527
	"
31df3850e98c *** empty log message ***
claus
parents: 282
diff changeset
   528
	selector ~~ #doIt ifTrue:[
31df3850e98c *** empty log message ***
claus
parents: 282
diff changeset
   529
	    implementorClass := self methodClass. 
31df3850e98c *** empty log message ***
claus
parents: 282
diff changeset
   530
	].
154
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   531
	implementorClass notNil ifTrue: [
170
40ea50c7b9fe *** empty log message ***
claus
parents: 154
diff changeset
   532
	    (implementorClass ~~ receiverClass) ifTrue: [
154
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   533
		newString := newString , '>>>',
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   534
			     implementorClass name printString
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   535
	    ]
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   536
	] ifFalse:[
170
40ea50c7b9fe *** empty log message ***
claus
parents: 154
diff changeset
   537
	    self searchClass ~~ receiverClass ifTrue:[
154
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   538
		newString := newString , '>>>' , self searchClass name
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   539
	    ].
216
a8abff749575 *** empty log message ***
claus
parents: 212
diff changeset
   540
	    "
a8abff749575 *** empty log message ***
claus
parents: 212
diff changeset
   541
	     kludge for doIt - these unbound methods are not
a8abff749575 *** empty log message ***
claus
parents: 212
diff changeset
   542
	     found in the classes methodDictionary
a8abff749575 *** empty log message ***
claus
parents: 212
diff changeset
   543
	    "
a8abff749575 *** empty log message ***
claus
parents: 212
diff changeset
   544
	    selector ~~ #doIt ifTrue:[
a8abff749575 *** empty log message ***
claus
parents: 212
diff changeset
   545
		newString := newString , '>>>**NONE**'
a8abff749575 *** empty log message ***
claus
parents: 212
diff changeset
   546
	    ]
154
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   547
	]
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
   548
    ].
a27a279701f8 Initial revision
claus
parents:
diff changeset
   549
a27a279701f8 Initial revision
claus
parents:
diff changeset
   550
    ^ newString
a27a279701f8 Initial revision
claus
parents:
diff changeset
   551
!
a27a279701f8 Initial revision
claus
parents:
diff changeset
   552
    
a27a279701f8 Initial revision
claus
parents:
diff changeset
   553
printString
37
d9a302eaa3ef *** empty log message ***
claus
parents: 25
diff changeset
   554
    "return a string describing the context" 
d9a302eaa3ef *** empty log message ***
claus
parents: 25
diff changeset
   555
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
   556
    ^ self receiverPrintString , ' ' , self selector printString
a27a279701f8 Initial revision
claus
parents:
diff changeset
   557
!
a27a279701f8 Initial revision
claus
parents:
diff changeset
   558
12
8e03bd717355 *** empty log message ***
claus
parents: 10
diff changeset
   559
displayString
37
d9a302eaa3ef *** empty log message ***
claus
parents: 25
diff changeset
   560
    "return a string describing the context - for display in Inspector" 
d9a302eaa3ef *** empty log message ***
claus
parents: 25
diff changeset
   561
42
e33491f6f260 *** empty log message ***
claus
parents: 37
diff changeset
   562
    ^ self class name , '(' , self printString , ')'
12
8e03bd717355 *** empty log message ***
claus
parents: 10
diff changeset
   563
!
8e03bd717355 *** empty log message ***
claus
parents: 10
diff changeset
   564
415
claus
parents: 406
diff changeset
   565
fullPrintString
claus
parents: 406
diff changeset
   566
    "return a string describing the context - this includes the linenumber,
claus
parents: 406
diff changeset
   567
     receiver printString and argument printString"
claus
parents: 406
diff changeset
   568
claus
parents: 406
diff changeset
   569
    |s|
claus
parents: 406
diff changeset
   570
claus
parents: 406
diff changeset
   571
    s := WriteStream on:String new.
claus
parents: 406
diff changeset
   572
    s nextPutAll:self receiverPrintString; space; nextPutAll:selector.
claus
parents: 406
diff changeset
   573
    self size ~~ 0 ifTrue: [
claus
parents: 406
diff changeset
   574
	s space.
claus
parents: 406
diff changeset
   575
	s nextPutAll:self argsDisplayString
claus
parents: 406
diff changeset
   576
    ].
claus
parents: 406
diff changeset
   577
    s nextPutAll:' ['; nextPutAll:self lineNumber printString; nextPutAll:']' .
claus
parents: 406
diff changeset
   578
    ^ s contents
claus
parents: 406
diff changeset
   579
claus
parents: 406
diff changeset
   580
    "
claus
parents: 406
diff changeset
   581
     thisContext fullPrintString
claus
parents: 406
diff changeset
   582
    "
claus
parents: 406
diff changeset
   583
!
claus
parents: 406
diff changeset
   584
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
   585
printOn:aStream
37
d9a302eaa3ef *** empty log message ***
claus
parents: 25
diff changeset
   586
    "append a printed description of the receiver onto aStream"
d9a302eaa3ef *** empty log message ***
claus
parents: 25
diff changeset
   587
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
   588
    aStream nextPutAll:(self receiverPrintString).
a27a279701f8 Initial revision
claus
parents:
diff changeset
   589
    aStream space.
a27a279701f8 Initial revision
claus
parents:
diff changeset
   590
    self selector printOn:aStream
282
94f5c3a6230d *** empty log message ***
claus
parents: 216
diff changeset
   591
! !
94f5c3a6230d *** empty log message ***
claus
parents: 216
diff changeset
   592
94f5c3a6230d *** empty log message ***
claus
parents: 216
diff changeset
   593
!Context methodsFor:'minidebugger printing'!
94f5c3a6230d *** empty log message ***
claus
parents: 216
diff changeset
   594
94f5c3a6230d *** empty log message ***
claus
parents: 216
diff changeset
   595
fullPrint
94f5c3a6230d *** empty log message ***
claus
parents: 216
diff changeset
   596
    "print the receiver, selector and args of the context 
94f5c3a6230d *** empty log message ***
claus
parents: 216
diff changeset
   597
     - used only for MiniDebuggers walkback print"
94f5c3a6230d *** empty log message ***
claus
parents: 216
diff changeset
   598
415
claus
parents: 406
diff changeset
   599
    self receiverPrintString print. ' ' print. selector print.
282
94f5c3a6230d *** empty log message ***
claus
parents: 216
diff changeset
   600
    self size ~~ 0 ifTrue: [
415
claus
parents: 406
diff changeset
   601
	' ' print. self argsDisplayString print
282
94f5c3a6230d *** empty log message ***
claus
parents: 216
diff changeset
   602
    ].
339
claus
parents: 328
diff changeset
   603
    ' [' print. self lineNumber print. ']' printNewline
282
94f5c3a6230d *** empty log message ***
claus
parents: 216
diff changeset
   604
94f5c3a6230d *** empty log message ***
claus
parents: 216
diff changeset
   605
    "
94f5c3a6230d *** empty log message ***
claus
parents: 216
diff changeset
   606
     thisContext fullPrint
94f5c3a6230d *** empty log message ***
claus
parents: 216
diff changeset
   607
    "
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
   608
!
a27a279701f8 Initial revision
claus
parents:
diff changeset
   609
a27a279701f8 Initial revision
claus
parents:
diff changeset
   610
fullPrintAll
37
d9a302eaa3ef *** empty log message ***
claus
parents: 25
diff changeset
   611
    "print a full walkback starting at the receiver
282
94f5c3a6230d *** empty log message ***
claus
parents: 216
diff changeset
   612
     - used only for MiniDebuggers walkback print"
37
d9a302eaa3ef *** empty log message ***
claus
parents: 25
diff changeset
   613
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
   614
    |context|
282
94f5c3a6230d *** empty log message ***
claus
parents: 216
diff changeset
   615
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
   616
    context := self.
a27a279701f8 Initial revision
claus
parents:
diff changeset
   617
    [context notNil] whileTrue: [
154
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   618
	context fullPrint.
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   619
	context := context sender
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
   620
    ]
282
94f5c3a6230d *** empty log message ***
claus
parents: 216
diff changeset
   621
94f5c3a6230d *** empty log message ***
claus
parents: 216
diff changeset
   622
    "
94f5c3a6230d *** empty log message ***
claus
parents: 216
diff changeset
   623
     thisContext fullPrintAll
94f5c3a6230d *** empty log message ***
claus
parents: 216
diff changeset
   624
    "
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
   625
! !
a27a279701f8 Initial revision
claus
parents:
diff changeset
   626
77
6c38ca59927f *** empty log message ***
claus
parents: 69
diff changeset
   627
!Context methodsFor:'error handling'!
6c38ca59927f *** empty log message ***
claus
parents: 69
diff changeset
   628
6c38ca59927f *** empty log message ***
claus
parents: 69
diff changeset
   629
invalidReturn:returnValue
6c38ca59927f *** empty log message ***
claus
parents: 69
diff changeset
   630
    "this message is sent by the VM, when a methods context
6c38ca59927f *** empty log message ***
claus
parents: 69
diff changeset
   631
     which has already returned is about to return again.
6c38ca59927f *** empty log message ***
claus
parents: 69
diff changeset
   632
     (i.e. about to execute a return from an already returned
105
7fe3d60db5e1 invalid return now raises a signal
claus
parents: 92
diff changeset
   633
      method in a block).
141
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   634
    We raise a signal here, to allow catching of that situation."
77
6c38ca59927f *** empty log message ***
claus
parents: 69
diff changeset
   635
406
claus
parents: 402
diff changeset
   636
    "
claus
parents: 402
diff changeset
   637
     in previous versions of ST/X and ST-80, this was no error;
claus
parents: 402
diff changeset
   638
     (instead, a normal blockreturn was performed to the value-sender).
claus
parents: 402
diff changeset
   639
     From a note in comp.lang.smalltalk, I conclude that new ST-80 versions
claus
parents: 402
diff changeset
   640
     now raise an error if this happens.
claus
parents: 402
diff changeset
   641
     Comment out the raise below to get that (old) behavior
claus
parents: 402
diff changeset
   642
     BETTER REWRITE YOUR APPLICATION
claus
parents: 402
diff changeset
   643
    " 
282
94f5c3a6230d *** empty log message ***
claus
parents: 216
diff changeset
   644
"/ new behavior:
94f5c3a6230d *** empty log message ***
claus
parents: 216
diff changeset
   645
406
claus
parents: 402
diff changeset
   646
    ^ InvalidReturnSignal 
claus
parents: 402
diff changeset
   647
	raiseRequestWith:returnValue.
282
94f5c3a6230d *** empty log message ***
claus
parents: 216
diff changeset
   648
94f5c3a6230d *** empty log message ***
claus
parents: 216
diff changeset
   649
"/ old behavior:
406
claus
parents: 402
diff changeset
   650
"/  ^ returnValue
claus
parents: 402
diff changeset
   651
!
claus
parents: 402
diff changeset
   652
claus
parents: 402
diff changeset
   653
invalidReturnOrRestart:returnValue
claus
parents: 402
diff changeset
   654
    "this message is sent by the VM, when a methods context
claus
parents: 402
diff changeset
   655
     which was compiled non-returnable is about to return again.
claus
parents: 402
diff changeset
   656
     We raise a signal here, to allow catching of that situation."
claus
parents: 402
diff changeset
   657
360
claus
parents: 345
diff changeset
   658
    ^ InvalidReturnSignal
claus
parents: 345
diff changeset
   659
	raiseRequestWith:returnValue
claus
parents: 345
diff changeset
   660
	errorString:'method was compiled non-resumable'
claus
parents: 345
diff changeset
   661
!
claus
parents: 345
diff changeset
   662
claus
parents: 345
diff changeset
   663
invalidReturnOrRestartError:how with:value
claus
parents: 345
diff changeset
   664
    "common error reporter for restart/return errors"
claus
parents: 345
diff changeset
   665
    
claus
parents: 345
diff changeset
   666
    self canReturn ifTrue:[
379
5b5a130ccd09 revision added
claus
parents: 375
diff changeset
   667
	"
5b5a130ccd09 revision added
claus
parents: 375
diff changeset
   668
	 tried to return from/restart a context which is already dead
5b5a130ccd09 revision added
claus
parents: 375
diff changeset
   669
	 (i.e. the method/block has already executed a return)
5b5a130ccd09 revision added
claus
parents: 375
diff changeset
   670
	"
5b5a130ccd09 revision added
claus
parents: 375
diff changeset
   671
	^ InvalidReturnSignal 
360
claus
parents: 345
diff changeset
   672
	      raiseRequestWith:value
claus
parents: 345
diff changeset
   673
	      errorString:(how , ': context not on calling chain')
claus
parents: 345
diff changeset
   674
    ].
claus
parents: 345
diff changeset
   675
    "
claus
parents: 345
diff changeset
   676
     tried to return from/restart a context of a method which was compiled
claus
parents: 345
diff changeset
   677
     unrestartable or of a block (which is never restartable)
claus
parents: 345
diff changeset
   678
    "
claus
parents: 345
diff changeset
   679
    ^ InvalidReturnSignal 
379
5b5a130ccd09 revision added
claus
parents: 375
diff changeset
   680
	  raiseRequestWith:value
360
claus
parents: 345
diff changeset
   681
	  errorString:(how , ': context cannot be restarted/returned from')
77
6c38ca59927f *** empty log message ***
claus
parents: 69
diff changeset
   682
! !
6c38ca59927f *** empty log message ***
claus
parents: 69
diff changeset
   683
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
   684
!Context methodsFor:'non local control flow'!
a27a279701f8 Initial revision
claus
parents:
diff changeset
   685
a27a279701f8 Initial revision
claus
parents:
diff changeset
   686
restart
a27a279701f8 Initial revision
claus
parents:
diff changeset
   687
    "restart the receiver - i.e. the method is evaluated again.
402
claus
parents: 384
diff changeset
   688
     if the context to restart already died, trigger an error.
141
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   689
     This is a low level helper for unwindAndRestart.
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   690
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   691
     NOTICE: 
154
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   692
	 NO unwind actions are performed - this is usually not
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   693
	 what you want (see Context>>unwindAndRestart).
141
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   694
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   695
     LIMITATION: 
154
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   696
	 currently a context can only be restarted by
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   697
	 the owning process - not from outside."
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
   698
a27a279701f8 Initial revision
claus
parents:
diff changeset
   699
    sender isNil ifTrue:[^ nil].
a27a279701f8 Initial revision
claus
parents:
diff changeset
   700
%{
141
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   701
    __RESUMECONTEXT__(SND_COMMA self, RESTART_VALUE, 0);
360
claus
parents: 345
diff changeset
   702
%}.
69
4564b6328136 *** empty log message ***
claus
parents: 54
diff changeset
   703
    "
360
claus
parents: 345
diff changeset
   704
     when we arrive here, something went wrong.
69
4564b6328136 *** empty log message ***
claus
parents: 54
diff changeset
   705
     debugging ...
4564b6328136 *** empty log message ***
claus
parents: 54
diff changeset
   706
    "
360
claus
parents: 345
diff changeset
   707
    ^ self invalidReturnOrRestartError:#restart with:nil
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
   708
!
a27a279701f8 Initial revision
claus
parents:
diff changeset
   709
69
4564b6328136 *** empty log message ***
claus
parents: 54
diff changeset
   710
return
141
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   711
    "return from this context with nil. I.e. as if it did a ^ nil.
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   712
     NOTICE:
154
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   713
	 NO unwind actions are performed - this is usually not
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   714
	 what you want (See Context>>unwind).
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   715
	 This is a low level method - a helper for unwind.
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
   716
141
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   717
     LIMITATION: 
154
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   718
	 currently a context can only be returned by
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   719
	 the owning process - not from outside."
141
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   720
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   721
    ^ self return:nil
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
   722
!
a27a279701f8 Initial revision
claus
parents:
diff changeset
   723
69
4564b6328136 *** empty log message ***
claus
parents: 54
diff changeset
   724
return:value
4564b6328136 *** empty log message ***
claus
parents: 54
diff changeset
   725
    "return from this context as if it did a '^ value'.
141
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   726
     NOTICE:
402
claus
parents: 384
diff changeset
   727
	 NO unwind actions are performed - this is usually not
154
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   728
	 what you want (See Context>>unwind:).
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   729
	 This is a low level method - a helper for unwind.
141
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   730
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   731
     LIMITATION: 
154
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   732
	 currently a context can only be returned by
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   733
	 the owning process - not from outside."
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
   734
a27a279701f8 Initial revision
claus
parents:
diff changeset
   735
    sender isNil ifTrue:[^ nil].
a27a279701f8 Initial revision
claus
parents:
diff changeset
   736
%{
141
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   737
    __RESUMECONTEXT__(SND_COMMA self, value, 0);
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
   738
360
claus
parents: 345
diff changeset
   739
%}.
69
4564b6328136 *** empty log message ***
claus
parents: 54
diff changeset
   740
    "
360
claus
parents: 345
diff changeset
   741
     when we arrive here, something went wrong.
69
4564b6328136 *** empty log message ***
claus
parents: 54
diff changeset
   742
     debugging ...
4564b6328136 *** empty log message ***
claus
parents: 54
diff changeset
   743
    "
360
claus
parents: 345
diff changeset
   744
    ^ self invalidReturnOrRestartError:#return with:value
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
   745
!
a27a279701f8 Initial revision
claus
parents:
diff changeset
   746
328
claus
parents: 326
diff changeset
   747
returnDoing:aBlock
claus
parents: 326
diff changeset
   748
    "return from this context as if it did a '^ aBlock value'.
claus
parents: 326
diff changeset
   749
     The block is evaluated as if called by the receiver context;
claus
parents: 326
diff changeset
   750
     NOT the true executing context.
claus
parents: 326
diff changeset
   751
     NOTICE:
402
claus
parents: 384
diff changeset
   752
	 NO unwind actions are performed - this is usually not
claus
parents: 384
diff changeset
   753
	 what you want (See Context>>unwindThenDo:).
328
claus
parents: 326
diff changeset
   754
	 This is a low level method - a helper for unwind.
claus
parents: 326
diff changeset
   755
claus
parents: 326
diff changeset
   756
     LIMITATION:
claus
parents: 326
diff changeset
   757
	 currently a context can only be returned by
claus
parents: 326
diff changeset
   758
	 the owning process - not from outside."
claus
parents: 326
diff changeset
   759
claus
parents: 326
diff changeset
   760
    sender isNil ifTrue:[^ nil].
claus
parents: 326
diff changeset
   761
%{
claus
parents: 326
diff changeset
   762
    __RESUMECONTEXT__(SND_COMMA self, aBlock, 2);
360
claus
parents: 345
diff changeset
   763
%}.
328
claus
parents: 326
diff changeset
   764
    "
360
claus
parents: 345
diff changeset
   765
     when we arrive here, something went wrong.
328
claus
parents: 326
diff changeset
   766
     debugging ...
claus
parents: 326
diff changeset
   767
    "
360
claus
parents: 345
diff changeset
   768
    ^ self invalidReturnOrRestartError:#return with:aBlock
328
claus
parents: 326
diff changeset
   769
!
claus
parents: 326
diff changeset
   770
69
4564b6328136 *** empty log message ***
claus
parents: 54
diff changeset
   771
resume
141
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   772
    "resume execution in this context. I.e. as if the method called
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   773
     last by the receiver did a ^ nil.
360
claus
parents: 345
diff changeset
   774
     If the context has already returned, report an error.
141
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   775
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   776
     NOTICE:
154
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   777
	 NO unwind actions are performed (see Context>>unwind).
69
4564b6328136 *** empty log message ***
claus
parents: 54
diff changeset
   778
141
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   779
     LIMITATION: 
154
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   780
	 currently a context can only be resumed by
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   781
	 the owning process - not from outside."
141
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   782
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   783
    ^ self resume:nil
69
4564b6328136 *** empty log message ***
claus
parents: 54
diff changeset
   784
!
4564b6328136 *** empty log message ***
claus
parents: 54
diff changeset
   785
4564b6328136 *** empty log message ***
claus
parents: 54
diff changeset
   786
resume:value
4564b6328136 *** empty log message ***
claus
parents: 54
diff changeset
   787
    "resume the receiver - as if it got 'value' from whatever
360
claus
parents: 345
diff changeset
   788
     it called. This continues execution in the receivers method
claus
parents: 345
diff changeset
   789
     after the point where it did its last send.
claus
parents: 345
diff changeset
   790
     If the context has already returned - report an error. 
141
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   791
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   792
     NOTICE:
154
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   793
	 NO unwind actions are performed (see Context>>unwind:).
141
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   794
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   795
     LIMITATION: 
154
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   796
	 currently a context can only be resumed by
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   797
	 the owning process - not from outside."
69
4564b6328136 *** empty log message ***
claus
parents: 54
diff changeset
   798
4564b6328136 *** empty log message ***
claus
parents: 54
diff changeset
   799
    |con|
4564b6328136 *** empty log message ***
claus
parents: 54
diff changeset
   800
141
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   801
    "
360
claus
parents: 345
diff changeset
   802
     starting with this context, find the one below (i.e. the one that I
claus
parents: 345
diff changeset
   803
     have called) and return from it.
141
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   804
    "
69
4564b6328136 *** empty log message ***
claus
parents: 54
diff changeset
   805
    con := thisContext.
216
a8abff749575 *** empty log message ***
claus
parents: 212
diff changeset
   806
%{
a8abff749575 *** empty log message ***
claus
parents: 212
diff changeset
   807
    while ((con != nil) && (_ContextInstPtr(con)->c_sender != self)) {
a8abff749575 *** empty log message ***
claus
parents: 212
diff changeset
   808
	con = _ContextInstPtr(con)->c_sender;
a8abff749575 *** empty log message ***
claus
parents: 212
diff changeset
   809
    }
a8abff749575 *** empty log message ***
claus
parents: 212
diff changeset
   810
%}.
a8abff749575 *** empty log message ***
claus
parents: 212
diff changeset
   811
69
4564b6328136 *** empty log message ***
claus
parents: 54
diff changeset
   812
    con isNil ifTrue:[
154
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   813
	"
360
claus
parents: 345
diff changeset
   814
	 tried to resume in context which is already dead
154
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   815
	 (i.e. the method/block has already executed a return)
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   816
	"
379
5b5a130ccd09 revision added
claus
parents: 375
diff changeset
   817
	^ con invalidReturnOrRestartError:#resume with:value
69
4564b6328136 *** empty log message ***
claus
parents: 54
diff changeset
   818
    ].
141
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   819
    ^ con return:value
69
4564b6328136 *** empty log message ***
claus
parents: 54
diff changeset
   820
!
4564b6328136 *** empty log message ***
claus
parents: 54
diff changeset
   821
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
   822
unwind
69
4564b6328136 *** empty log message ***
claus
parents: 54
diff changeset
   823
    "return nil from the receiver - i.e. simulate a '^ nil'.
360
claus
parents: 345
diff changeset
   824
     If the context has already retruned, report an error.
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
   825
     Evaluate all unwind-blocks as specified in Block>>valueNowOrOnUnwind:
69
4564b6328136 *** empty log message ***
claus
parents: 54
diff changeset
   826
     and Block>>valueOnUnwindDo: on the way.
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
   827
141
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   828
     LIMITATION: 
154
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   829
	 currently a context can only be unwound by
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   830
	 the owning process - not from outside."
141
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   831
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   832
    ^ self unwind:nil
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
   833
!
a27a279701f8 Initial revision
claus
parents:
diff changeset
   834
a27a279701f8 Initial revision
claus
parents:
diff changeset
   835
unwind:value
69
4564b6328136 *** empty log message ***
claus
parents: 54
diff changeset
   836
    "return value from the receiver - i.e. simulate a '^ value'.
360
claus
parents: 345
diff changeset
   837
     If the context has already returned , report an error.
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
   838
     Evaluate all unwind-blocks as specified in Block>>valueNowOrOnUnwind:
69
4564b6328136 *** empty log message ***
claus
parents: 54
diff changeset
   839
     and Block>>valueOnUnwindDo: on the way.
141
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   840
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   841
     LIMITATION: 
154
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   842
	 currently a context can only be unwound by
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   843
	 the owning process - not from outside."
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
   844
216
a8abff749575 *** empty log message ***
claus
parents: 212
diff changeset
   845
    |con sel unwindBlock|
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
   846
141
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   847
    sender isNil ifTrue:[
154
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   848
	"
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   849
	 tried to return to a context which is already dead
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   850
	 (i.e. the method/block has already executed a return)
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   851
	"
379
5b5a130ccd09 revision added
claus
parents: 375
diff changeset
   852
	^ self invalidReturnOrRestartError:#unwind with:value
141
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   853
    ].
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
   854
141
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   855
    "
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   856
     start with this context, moving up, looking for unwind actions
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   857
    "
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
   858
    con := thisContext.
69
4564b6328136 *** empty log message ***
claus
parents: 54
diff changeset
   859
    [con notNil and:[con ~~ self]] whileTrue:[
154
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   860
	con isBlockContext ifFalse:[
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   861
	    "
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   862
	     the way we find those unwind contexts seems kludgy ...
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   863
	    "
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   864
	    sel := con selector.
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   865
	    ((sel == #valueNowOrOnUnwindDo:) or:[sel == #valueOnUnwindDo:]) ifTrue:[
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   866
		"
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   867
		 ... the way we evaluate the unwind blocks too
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   868
		"
216
a8abff749575 *** empty log message ***
claus
parents: 212
diff changeset
   869
		unwindBlock := con argAt:1.
a8abff749575 *** empty log message ***
claus
parents: 212
diff changeset
   870
		unwindBlock value
154
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   871
	    ]
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   872
	].
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   873
	con := con sender
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
   874
    ].
69
4564b6328136 *** empty log message ***
claus
parents: 54
diff changeset
   875
141
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   876
    "oops, I am not on the calling chain
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   877
     (should we check for this situation first and NOT evaluate
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   878
      the unwind actions in this case ?)
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   879
    "
69
4564b6328136 *** empty log message ***
claus
parents: 54
diff changeset
   880
    con isNil ifTrue:[
154
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   881
	"
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   882
	 tried to return to a context which is already dead
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   883
	 (i.e. the method/block has already executed a return)
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   884
	"
379
5b5a130ccd09 revision added
claus
parents: 375
diff changeset
   885
	^ self invalidReturnOrRestartError:#unwind with:value
69
4564b6328136 *** empty log message ***
claus
parents: 54
diff changeset
   886
    ].
141
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   887
    "
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   888
     now, that all unwind-actions are done, I can use the
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   889
     low-level return ...
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   890
    "
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   891
    ^ self return:value
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   892
!
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   893
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   894
unwindAndRestart
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   895
    "restart the receiver - i.e. the method is evaluated again.
402
claus
parents: 384
diff changeset
   896
     if the context to restart already died report an error.
141
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   897
     Evaluate all unwind-blocks as specified in Block>>valueNowOrOnUnwind:
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   898
     and Block>>valueOnUnwindDo: before restarting.
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   899
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   900
     LIMITATION: 
154
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   901
	 currently a context can only be restarted by
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   902
	 the owning process - not from outside."
141
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   903
216
a8abff749575 *** empty log message ***
claus
parents: 212
diff changeset
   904
    |con sel unwindBlock|
141
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   905
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   906
    "
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   907
     start with this context, moving up, looking for unwind actions
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   908
    "
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   909
    con := thisContext.
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   910
    [con notNil and:[con ~~ self]] whileTrue:[
154
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   911
	con isBlockContext ifFalse:[
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   912
	    "
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   913
	     the way we find those unwind contexts seems kludgy ...
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   914
	    "
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   915
	    sel := con selector.
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   916
	    ((sel == #valueNowOrOnUnwindDo:) or:[sel == #valueOnUnwindDo:]) ifTrue:[
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   917
		"
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   918
		 ... the way we evaluate the unwind blocks too
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   919
		"
216
a8abff749575 *** empty log message ***
claus
parents: 212
diff changeset
   920
		unwindBlock := con argAt:1.
a8abff749575 *** empty log message ***
claus
parents: 212
diff changeset
   921
		unwindBlock value
154
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   922
	    ]
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   923
	].
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   924
	con := con sender
141
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   925
    ].
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   926
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   927
    "oops, I am not on the calling chain
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   928
     (should we check for this situation first and NOT evaluate
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   929
      the unwind actions in this case ?)
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   930
    "
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   931
    con isNil ifTrue:[
154
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   932
	"
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   933
	 tried to return to a context which is already dead
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   934
	 (i.e. the method/block has already executed a return)
d4236ec280a6 *** empty log message ***
claus
parents: 141
diff changeset
   935
	"
379
5b5a130ccd09 revision added
claus
parents: 375
diff changeset
   936
	^ self invalidReturnOrRestartError:#unwindAndRestart with:nil
141
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   937
    ].
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   938
    "
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   939
     now, that all unwind-actions are done, I can use the
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   940
     low-level restart ...
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   941
    "
b530e69052e4 better unwind & new unwindAndRestart
claus
parents: 105
diff changeset
   942
    ^ self restart
328
claus
parents: 326
diff changeset
   943
!
claus
parents: 326
diff changeset
   944
claus
parents: 326
diff changeset
   945
unwindThenDo:aBlock 
claus
parents: 326
diff changeset
   946
    "return the value of aBlock from the receiver - i.e. simulate a '^ aBlock value'.
360
claus
parents: 345
diff changeset
   947
     If the context has already returned , report an error.
328
claus
parents: 326
diff changeset
   948
     Evaluate all unwind-blocks as specified in Block>>valueNowOrOnUnwind:
claus
parents: 326
diff changeset
   949
     and Block>>valueOnUnwindDo: on the way.
claus
parents: 326
diff changeset
   950
     The block is evaluated AFTER all unwind actions are performed
claus
parents: 326
diff changeset
   951
     (i.e. the blocks sender will be the receiving context, not the
402
claus
parents: 384
diff changeset
   952
      currently executing context)
328
claus
parents: 326
diff changeset
   953
claus
parents: 326
diff changeset
   954
     LIMITATION: 
claus
parents: 326
diff changeset
   955
	 currently a context can only be unwound by
claus
parents: 326
diff changeset
   956
	 the owning process - not from outside."
claus
parents: 326
diff changeset
   957
claus
parents: 326
diff changeset
   958
    |con sel unwindBlock|
claus
parents: 326
diff changeset
   959
claus
parents: 326
diff changeset
   960
    sender isNil ifTrue:[
claus
parents: 326
diff changeset
   961
	"
claus
parents: 326
diff changeset
   962
	 tried to return to a context which is already dead
claus
parents: 326
diff changeset
   963
	 (i.e. the method/block has already executed a return)
claus
parents: 326
diff changeset
   964
	"
379
5b5a130ccd09 revision added
claus
parents: 375
diff changeset
   965
	^ self invalidReturnOrRestartError:#unwind with:aBlock
328
claus
parents: 326
diff changeset
   966
    ].
claus
parents: 326
diff changeset
   967
claus
parents: 326
diff changeset
   968
    "
claus
parents: 326
diff changeset
   969
     start with this context, moving up, looking for unwind actions
claus
parents: 326
diff changeset
   970
    "
claus
parents: 326
diff changeset
   971
    con := thisContext.
claus
parents: 326
diff changeset
   972
    [con notNil and:[con ~~ self]] whileTrue:[
claus
parents: 326
diff changeset
   973
	con isBlockContext ifFalse:[
claus
parents: 326
diff changeset
   974
	    "
claus
parents: 326
diff changeset
   975
	     the way we find those unwind contexts seems kludgy ...
claus
parents: 326
diff changeset
   976
	    "
claus
parents: 326
diff changeset
   977
	    sel := con selector.
claus
parents: 326
diff changeset
   978
	    ((sel == #valueNowOrOnUnwindDo:) or:[sel == #valueOnUnwindDo:]) ifTrue:[
claus
parents: 326
diff changeset
   979
		"
claus
parents: 326
diff changeset
   980
		 ... the way we evaluate the unwind blocks too
claus
parents: 326
diff changeset
   981
		"
claus
parents: 326
diff changeset
   982
		unwindBlock := con argAt:1.
claus
parents: 326
diff changeset
   983
		unwindBlock value
claus
parents: 326
diff changeset
   984
	    ]
claus
parents: 326
diff changeset
   985
	].
claus
parents: 326
diff changeset
   986
	con := con sender
claus
parents: 326
diff changeset
   987
    ].
claus
parents: 326
diff changeset
   988
claus
parents: 326
diff changeset
   989
    "oops, I am not on the calling chain
claus
parents: 326
diff changeset
   990
     (should we check for this situation first and NOT evaluate
claus
parents: 326
diff changeset
   991
      the unwind actions in this case ?)
claus
parents: 326
diff changeset
   992
    "
claus
parents: 326
diff changeset
   993
    con isNil ifTrue:[
claus
parents: 326
diff changeset
   994
	"
claus
parents: 326
diff changeset
   995
	 tried to return to a context which is already dead
claus
parents: 326
diff changeset
   996
	 (i.e. the method/block has already executed a return)
claus
parents: 326
diff changeset
   997
	"
379
5b5a130ccd09 revision added
claus
parents: 375
diff changeset
   998
	^ self invalidReturnOrRestartError:#unwind with:aBlock
328
claus
parents: 326
diff changeset
   999
    ].
claus
parents: 326
diff changeset
  1000
    "
claus
parents: 326
diff changeset
  1001
     now, that all unwind-actions are done, I can use the
claus
parents: 326
diff changeset
  1002
     low-level return ...
claus
parents: 326
diff changeset
  1003
    "
claus
parents: 326
diff changeset
  1004
    ^ self returnDoing:aBlock 
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
  1005
! !