Reviewed by John.
[WebKit-https.git] / JavaScriptCore / kjs / scope_chain.cpp
1 /*
2  *  This file is part of the KDE libraries
3  *  Copyright (C) 2003 Apple Computer, Inc.
4  *
5  *  This library is free software; you can redistribute it and/or
6  *  modify it under the terms of the GNU Library General Public
7  *  License as published by the Free Software Foundation; either
8  *  version 2 of the License, or (at your option) any later version.
9  *
10  *  This library is distributed in the hope that it will be useful,
11  *  but WITHOUT ANY WARRANTY; without even the implied warranty of
12  *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
13  *  Library General Public License for more details.
14  *
15  *  You should have received a copy of the GNU Library General Public License
16  *  along with this library; see the file COPYING.LIB.  If not, write to
17  *  the Free Software Foundation, Inc., 59 Temple Place - Suite 330,
18  *  Boston, MA 02111-1307, USA.
19  *
20  */
21
22 #include "scope_chain.h"
23
24 #include "object.h"
25
26 namespace KJS {
27
28 inline void ScopeChain::ref() const
29 {
30     for (ScopeChainNode *n = _node; n; n = n->next) {
31         if (n->refCount++ != 0)
32             break;
33     }
34 }
35
36 ScopeChain &ScopeChain::operator=(const ScopeChain &c)
37 {
38     c.ref();
39     deref();
40     _node = c._node;
41     return *this;
42 }
43
44 void ScopeChain::push(ObjectImp *o)
45 {
46     assert(o);
47     _node = new ScopeChainNode(_node, o);
48 }
49
50 void ScopeChain::pop()
51 {
52     ScopeChainNode *oldNode = _node;
53     assert(oldNode);
54     ScopeChainNode *newNode = oldNode->next;
55     _node = newNode;
56     
57     if (--oldNode->refCount != 0) {
58         if (newNode)
59             ++newNode->refCount;
60     } else {
61         delete oldNode;
62     }
63 }
64
65 void ScopeChain::release()
66 {
67     // This function is only called by deref(),
68     // Deref ensures these conditions are true.
69     assert(_node && _node->refCount == 0);
70     ScopeChainNode *n = _node;
71     do {
72         ScopeChainNode *next = n->next;
73         delete n;
74         n = next;
75     } while (n && --n->refCount == 0);
76 }
77
78 void ScopeChain::mark()
79 {
80     for (ScopeChainNode *n = _node; n; n = n->next) {
81         ObjectImp *o = n->object;
82         if (!o->marked())
83             o->mark();
84     }
85 }
86
87 ObjectImp *ScopeChain::bottom() const
88 {
89     ScopeChainNode *last = 0;
90     for (ScopeChainNode *n = _node; n; n = n->next) {
91         if (!n->next) {
92             last = n;
93         }
94     }
95     if (!last) {
96         return 0;
97     }
98
99     return last->object;
100 }
101
102
103
104 } // namespace KJS