ValueHolderWithWeakDependents.st
author Claus Gittinger <cg@exept.de>
Wed, 19 Mar 2003 11:06:54 +0100
changeset 1718 7da6bd953eb9
parent 1557 e5e78ee63670
child 1803 d69cb54a3b56
permissions -rw-r--r--
comments
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
1557
e5e78ee63670 #valueNowOrOnUnwindDo: -> #ensure:
Claus Gittinger <cg@exept.de>
parents: 1257
diff changeset
     1
"{ Package: 'stx:libview2' }"
e5e78ee63670 #valueNowOrOnUnwindDo: -> #ensure:
Claus Gittinger <cg@exept.de>
parents: 1257
diff changeset
     2
1257
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
     3
ValueHolder subclass:#ValueHolderWithWeakDependents
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
     4
	instanceVariableNames:''
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
     5
	classVariableNames:''
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
     6
	poolDictionaries:''
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
     7
	category:'Interface-Support-Models'
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
     8
!
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
     9
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    10
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    11
!ValueHolderWithWeakDependents methodsFor:'dependents access'!
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    12
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    13
addDependent:anObject
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    14
    "make the argument, anObject be a weak dependent of the receiver"
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    15
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    16
    |wasBlocked|
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    17
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    18
    wasBlocked := OperatingSystem blockInterrupts.
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    19
    [
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    20
        |deps dep|
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    21
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    22
        deps := dependents.
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    23
        "/
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    24
        "/ store the very first dependent directly in
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    25
        "/ the dependents instVar
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    26
        "/
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    27
        (deps isNil and:[anObject isCollection not]) ifTrue:[
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    28
            dependents := WeakArray with:anObject
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    29
        ] ifFalse:[
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    30
            deps class == WeakArray ifTrue:[
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    31
                dep := deps at:1.
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    32
                dep ~~ anObject ifTrue:[
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    33
                    (dep isNil or:[dep == 0]) ifTrue:[
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    34
                        deps at:1 put:anObject
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    35
                    ] ifFalse:[
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    36
                        self dependents:(WeakIdentitySet with:dep with:anObject)
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    37
                    ]
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    38
                ]
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    39
            ] ifFalse:[
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    40
                deps add:anObject
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    41
            ]
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    42
        ]
1557
e5e78ee63670 #valueNowOrOnUnwindDo: -> #ensure:
Claus Gittinger <cg@exept.de>
parents: 1257
diff changeset
    43
    ] ensure:[
1257
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    44
        wasBlocked ifFalse:[
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    45
            OperatingSystem unblockInterrupts
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    46
        ]
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    47
    ]
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    48
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    49
    "Modified: 8.1.1997 / 23:40:30 / cg"
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    50
!
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    51
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    52
dependentsDo:aBlock
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    53
    "evaluate aBlock for all of my dependents"
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    54
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    55
    |deps|
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    56
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    57
    deps := dependents.
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    58
    deps size ~~ 0 ifTrue:[
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    59
        deps do:[:d | 
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    60
                    (d notNil and:[d ~~ 0]) ifTrue:[
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    61
                        aBlock value:d
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    62
                    ]
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    63
                ]
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    64
    ].
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    65
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    66
!
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    67
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    68
removeDependent:anObject
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    69
    "make the argument, anObject be independent of the receiver"
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    70
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    71
    |wasBlocked|
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    72
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    73
    "/ must do this save from interrupts, since the dependents collection
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    74
    "/ is possibly accessed from multiple threads.
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    75
    "/ Used to use #valueUninterruptably here; inlined that code for slightly
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    76
    "/ faster execution.
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    77
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    78
    wasBlocked := OperatingSystem blockInterrupts.
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    79
    [
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    80
        |deps n d|
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    81
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    82
        deps := dependents.
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    83
        deps size ~~ 0 ifTrue:[
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    84
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    85
            "/ to save a fair amount of memory in case of
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    86
            "/ many dependencies, we store a single dependent in
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    87
            "/ a WeakArray, and switch to a WeakSet if more dependents are
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    88
            "/ added. Here we have to do the inverse ...
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    89
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    90
            deps class == WeakArray ifTrue:[
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    91
                ((d := deps at:1) == anObject 
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    92
                or:[d isNil
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    93
                or:[d == 0]]) ifTrue:[
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    94
                    dependents := nil
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    95
                ]
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    96
            ] ifFalse:[
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    97
                deps remove:anObject ifAbsent:[].
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    98
                (n := deps size) == 0 ifTrue:[
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
    99
                    dependents := nil
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
   100
                ] ifFalse:[
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
   101
                    n == 1 ifTrue:[
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
   102
                        d := deps firstIfEmpty:nil.
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
   103
                        d notNil ifTrue:[
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
   104
                            deps := WeakArray with:d
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
   105
                        ] ifFalse:[
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
   106
                            deps := nil
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
   107
                        ].
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
   108
                        dependents := deps.
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
   109
                    ]
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
   110
                ]
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
   111
            ]
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
   112
        ]
1557
e5e78ee63670 #valueNowOrOnUnwindDo: -> #ensure:
Claus Gittinger <cg@exept.de>
parents: 1257
diff changeset
   113
    ] ensure:[
1257
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
   114
        wasBlocked ifFalse:[
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
   115
            OperatingSystem unblockInterrupts
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
   116
        ]
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
   117
    ]
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
   118
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
   119
    "Modified: / 26.1.1998 / 19:51:50 / cg"
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
   120
! !
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
   121
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
   122
!ValueHolderWithWeakDependents class methodsFor:'documentation'!
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
   123
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
   124
version
1557
e5e78ee63670 #valueNowOrOnUnwindDo: -> #ensure:
Claus Gittinger <cg@exept.de>
parents: 1257
diff changeset
   125
    ^ '$Header: /cvs/stx/stx/libview2/ValueHolderWithWeakDependents.st,v 1.2 2002-02-26 12:59:22 cg Exp $'
1257
6c1eab277273 initial checkin
Claus Gittinger <cg@exept.de>
parents:
diff changeset
   126
! !