Reviewed by Oliver Hunt.
[WebKit-https.git] / JavaScriptCore / kjs / Context.cpp
1 // -*- mode: c++; c-basic-offset: 4 -*-
2 /*
3  *  This file is part of the KDE libraries
4  *  Copyright (C) 1999-2001 Harri Porten (porten@kde.org)
5  *  Copyright (C) 2001 Peter Kelly (pmk@post.com)
6  *  Copyright (C) 2003, 2006-2007 Apple Computer, Inc.
7  *
8  *  This library is free software; you can redistribute it and/or
9  *  modify it under the terms of the GNU Library General Public
10  *  License as published by the Free Software Foundation; either
11  *  version 2 of the License, or (at your option) any later version.
12  *
13  *  This library is distributed in the hope that it will be useful,
14  *  but WITHOUT ANY WARRANTY; without even the implied warranty of
15  *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
16  *  Library General Public License for more details.
17  *
18  *  You should have received a copy of the GNU Library General Public License
19  *  along with this library; see the file COPYING.LIB.  If not, write to
20  *  the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
21  *  Boston, MA 02110-1301, USA.
22  *
23  */
24
25 #include "context.h"
26
27 #include "JSGlobalObject.h"
28
29 namespace KJS {
30
31 // ECMA 10.2
32 Context::Context(JSGlobalObject* glob, Interpreter* interpreter, JSObject* thisV, 
33                  FunctionBodyNode* currentBody, CodeType type, Context* callingCon, 
34                  FunctionImp* func, const List* args)
35     : m_interpreter(interpreter)
36     , m_savedContext(interpreter->context())
37     , m_currentBody(currentBody)
38     , m_function(func)
39     , m_arguments(args)
40     , m_iterationDepth(0)
41     , m_switchDepth(0) 
42 {
43     m_codeType = type;
44     m_callingContext = callingCon;
45     
46     // create and initialize activation object (ECMA 10.1.6)
47     if (type == FunctionCode) {
48         m_activation = new ActivationImp(func, *args);
49         m_variable = m_activation;
50     } else {
51         m_activation = 0;
52         m_variable = glob;
53     }
54     
55     // ECMA 10.2
56     switch(type) {
57     case EvalCode:
58         if (m_callingContext) {
59             scope = m_callingContext->scopeChain();
60             m_variable = m_callingContext->variableObject();
61             m_thisVal = m_callingContext->thisValue();
62             break;
63         } // else same as GlobalCode
64     case GlobalCode:
65         scope.clear();
66         scope.push(glob);
67         m_thisVal = static_cast<JSObject*>(glob);
68         break;
69     case FunctionCode:
70         scope = func->scope();
71         scope.push(m_activation);
72         m_variable = m_activation; // TODO: DontDelete ? (ECMA 10.2.3)
73         m_thisVal = thisV;
74         break;
75     }
76
77     m_interpreter->setContext(this);
78 }
79
80 Context::~Context()
81 {
82     m_interpreter->setContext(m_savedContext);
83
84     // The arguments list is only needed to potentially create the  arguments object, 
85     // which isn't accessible from nested scopes so we can discard the list as soon 
86     // as the function is done running.
87     // This prevents lists of Lists from building up, waiting to be garbage collected
88     ActivationImp* activation = static_cast<ActivationImp*>(m_activation);
89     if (activation)
90         activation->releaseArguments();
91 }
92
93 void Context::mark()
94 {
95     for (Context* context = this; context; context = context->m_callingContext)
96         context->scope.mark();
97 }
98
99 } // namespace KJS