1 /* Digital Mars DMDScript source code.
2  * Copyright (c) 2000-2002 by Chromium Communications
3  * D version Copyright (c) 2004-2010 by Digital Mars
4  * Distributed under the Boost Software License, Version 1.0.
5  * (See accompanying file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
6  * written by Walter Bright
7  * http://www.digitalmars.com
8  *
9  * D2 port by Dmitry Olshansky 
10  *
11  * DMDScript is implemented in the D Programming Language,
12  * http://www.digitalmars.com/d/
13  *
14  * For a C++ implementation of DMDScript, including COM support, see
15  * http://www.digitalmars.com/dscript/cppscript.html
16  */
17 
18 module dmdscript.ddeclaredfunction;
19 
20 import std.stdio;
21 import core.stdc.stdlib;
22 import std.exception;
23 import dmdscript.script;
24 import dmdscript.dobject;
25 import dmdscript.dfunction;
26 import dmdscript.darguments;
27 import dmdscript.opcodes;
28 import dmdscript.ir;
29 import dmdscript.identifier;
30 import dmdscript.value;
31 import dmdscript.functiondefinition;
32 import dmdscript.text;
33 import dmdscript.property;
34 
35 /* ========================== DdeclaredFunction ================== */
36 
37 class DdeclaredFunction : Dfunction
38 {
39     FunctionDefinition fd;
40 
41     this(FunctionDefinition fd)
42     {
43         super(cast(uint)fd.parameters.length, Dfunction.getPrototype());
44         assert(Dfunction.getPrototype());
45         assert(internal_prototype);
46         this.fd = fd;
47 
48         Dobject o;
49 
50         // ECMA 3 13.2
51         o = new Dobject(Dobject.getPrototype());        // step 9
52         Put(TEXT_prototype, o, DontEnum);               // step 11
53         o.Put(TEXT_constructor, this, DontEnum);        // step 10
54     }
55 
56     override void *Call(CallContext *cc, Dobject othis, Value* ret, Value[] arglist)
57     {
58         // 1. Create activation object per ECMA 10.1.6
59         // 2. Instantiate function variables as properties of
60         //    activation object
61         // 3. The 'this' value is the activation object
62 
63         Dobject actobj;         // activation object
64         Darguments args;
65         Value[] locals;
66         uint i;
67         void *result;
68 
69         //writefln("DdeclaredFunction.Call() '%s'", toString());
70         //writefln("this.scopex.length = %d", this.scopex.length);
71         //writefln("\tinstantiate(this = %x, fd = %x)", cast(uint)cast(void*)this, cast(uint)cast(void*)fd);
72 
73         // if it's an empty function, just return
74         if(fd.code[0].opcode == IRret)
75         {
76             return null;
77         }
78 
79         // Generate the activation object
80         // ECMA v3 10.1.6
81         actobj = new Dobject(null);
82         
83         Value vtmp;//should not be referenced by the end of func
84         if(fd.name){ 
85            vtmp.putVobject(this);
86            actobj.Put(fd.name,&vtmp,DontDelete);
87         }
88         // Instantiate the parameters
89         {
90             uint a = 0;
91             foreach(Identifier* p; fd.parameters)
92             {
93                 Value* v = (a < arglist.length) ? &arglist[a++] : &vundefined;
94                 actobj.Put(p.toString(), v, DontDelete);
95             }
96         }
97 
98         // Generate the Arguments Object
99         // ECMA v3 10.1.8
100         args = new Darguments(cc.caller, this, actobj, fd.parameters, arglist);
101 
102         actobj.Put(TEXT_arguments, args, DontDelete);
103 
104         // The following is not specified by ECMA, but seems to be supported
105         // by jscript. The url www.grannymail.com has the following code
106         // which looks broken to me but works in jscript:
107         //
108         //	    function MakeArray() {
109         //	      this.length = MakeArray.arguments.length
110         //	      for (var i = 0; i < this.length; i++)
111         //		  this[i+1] = arguments[i]
112         //	    }
113         //	    var cardpic = new MakeArray("LL","AP","BA","MB","FH","AW","CW","CV","DZ");
114         Put(TEXT_arguments, args, DontDelete);          // make grannymail bug work
115 
116         
117         
118 
119         Dobject[] newScopex;
120         newScopex = this.scopex.dup;//copy this function object scope chain
121         assert(newScopex.length != 0);
122         newScopex ~= actobj;//and put activation object on top of it
123         
124         fd.instantiate(newScopex, actobj, DontDelete);
125 
126         Dobject[] scopesave = cc.scopex;
127         cc.scopex = newScopex; 
128         auto scoperootsave = cc.scoperoot;
129         cc.scoperoot++;//to accaunt extra activation object on scopex chain
130         Dobject variablesave = cc.variable;
131         cc.variable = actobj;
132         auto callersave = cc.caller;
133         cc.caller = this;
134         auto callerfsave = cc.callerf;
135         cc.callerf = fd;
136 
137         Value[] p1;
138         Value* v;
139         if(fd.nlocals < 128)
140             v = cast(Value*)alloca(fd.nlocals * Value.sizeof);
141         if(v)
142             locals = v[0 .. fd.nlocals];
143         else
144         {
145             p1 = new Value[fd.nlocals];
146             locals = p1;
147         }
148 
149         result = IR.call(cc, othis, fd.code, ret, locals.ptr);
150 
151         delete p1;
152 
153         cc.callerf = callerfsave;
154         cc.caller = callersave;
155         cc.variable = variablesave;
156         cc.scopex = scopesave;
157         cc.scoperoot = scoperootsave;
158 
159         // Remove the arguments object
160         //Value* v;
161         //v=Get(TEXT_arguments);
162         //writef("1v = %x, %s, v.object = %x\n", v, v.getType(), v.object);
163         Put(TEXT_arguments, &vundefined, 0);
164         //actobj.Put(TEXT_arguments, &vundefined, 0);
165 
166         version(none)
167         {
168             writef("args = %x, actobj = %x\n", args, actobj);
169             v = Get(TEXT_arguments);
170             writef("2v = %x, %s, v.object = %x\n", v, v.getType(), v.object);
171             v.object = null;
172 
173             {
174                 uint *p = cast(uint *)0x40a49a80;
175                 uint i;
176                 for(i = 0; i < 16; i++)
177                 {
178                     writef("p[%x] = %x\n", &p[i], p[i]);
179                 }
180             }
181         }
182 
183         return result;
184     }
185 
186     override void *Construct(CallContext *cc, Value *ret, Value[] arglist)
187     {
188         // ECMA 3 13.2.2
189         Dobject othis;
190         Dobject proto;
191         Value* v;
192         void *result;
193 
194         v = Get(TEXT_prototype);
195         if(v.isPrimitive())
196             proto = Dobject.getPrototype();
197         else
198             proto = v.toObject();
199         othis = new Dobject(proto);
200         result = Call(cc, othis, ret, arglist);
201         if(!result)
202         {
203             if(ret.isPrimitive())
204                 ret.putVobject(othis);
205         }
206         return result;
207     }
208 
209     override string toString()
210     {
211         char[] s;
212 
213         //writef("DdeclaredFunction.toString()\n");
214         fd.toBuffer(s);
215         return assumeUnique(s);
216     }
217 }
218 
219