Mercurial > projects > dmdscript-tango
comparison dmdscript_tango/ddeclaredfunction.d @ 0:55c2951c07be
initial, files origin, premoved tree
author | saaadel |
---|---|
date | Sun, 24 Jan 2010 12:34:47 +0200 |
parents | |
children | 8363a4bf6a8f |
comparison
equal
deleted
inserted
replaced
-1:000000000000 | 0:55c2951c07be |
---|---|
1 | |
2 /* Digital Mars DMDScript source code. | |
3 * Copyright (c) 2000-2002 by Chromium Communications | |
4 * D version Copyright (c) 2004-2006 by Digital Mars | |
5 * All Rights Reserved | |
6 * written by Walter Bright | |
7 * www.digitalmars.com | |
8 * Use at your own risk. There is no warranty, express or implied. | |
9 * License for redistribution is by the GNU General Public License in gpl.txt. | |
10 * | |
11 * A binary, non-exclusive license for commercial use can be | |
12 * purchased from www.digitalmars.com/dscript/buy.html. | |
13 * | |
14 * DMDScript is implemented in the D Programming Language, | |
15 * www.digitalmars.com/d/ | |
16 * | |
17 * For a C++ implementation of DMDScript, including COM support, | |
18 * see www.digitalmars.com/dscript/cppscript.html. | |
19 */ | |
20 | |
21 | |
22 module dmdscript.ddeclaredfunction; | |
23 | |
24 import std.stdio; | |
25 import std.c.stdlib; | |
26 | |
27 import dmdscript.script; | |
28 import dmdscript.dobject; | |
29 import dmdscript.dfunction; | |
30 import dmdscript.darguments; | |
31 import dmdscript.opcodes; | |
32 import dmdscript.ir; | |
33 import dmdscript.identifier; | |
34 import dmdscript.value; | |
35 import dmdscript.functiondefinition; | |
36 import dmdscript.text; | |
37 import dmdscript.property; | |
38 | |
39 /* ========================== DdeclaredFunction ================== */ | |
40 | |
41 class DdeclaredFunction : Dfunction | |
42 { | |
43 FunctionDefinition fd; | |
44 | |
45 this(FunctionDefinition fd) | |
46 { | |
47 super(fd.parameters.length, Dfunction.getPrototype()); | |
48 assert(Dfunction.getPrototype()); | |
49 assert(internal_prototype); | |
50 this.fd = fd; | |
51 | |
52 Dobject o; | |
53 | |
54 // ECMA 3 13.2 | |
55 o = new Dobject(Dobject.getPrototype()); // step 9 | |
56 Put(TEXT_prototype, o, DontEnum); // step 11 | |
57 o.Put(TEXT_constructor, this, DontEnum); // step 10 | |
58 } | |
59 | |
60 void *Call(CallContext *cc, Dobject othis, Value* ret, Value[] arglist) | |
61 { | |
62 // 1. Create activation object per ECMA 10.1.6 | |
63 // 2. Instantiate function variables as properties of | |
64 // activation object | |
65 // 3. The 'this' value is the activation object | |
66 | |
67 Dobject actobj; // activation object | |
68 Darguments args; | |
69 Value[] locals; | |
70 uint i; | |
71 void *result; | |
72 | |
73 //writefln("DdeclaredFunction.Call() '%s'", toString()); | |
74 //writefln("this.scopex.length = %d", this.scopex.length); | |
75 //writefln("\tinstantiate(this = %x, fd = %x)", cast(uint)cast(void*)this, cast(uint)cast(void*)fd); | |
76 | |
77 // if it's an empty function, just return | |
78 if (fd.code[0].opcode == IRret) | |
79 { | |
80 return null; | |
81 } | |
82 | |
83 // Generate the activation object | |
84 // ECMA v3 10.1.6 | |
85 actobj = new Dobject(null); | |
86 | |
87 // Instantiate the parameters | |
88 { | |
89 uint a = 0; | |
90 foreach (Identifier* p; fd.parameters) | |
91 { | |
92 Value* v = (a < arglist.length) ? &arglist[a++] : &vundefined; | |
93 actobj.Put(p.toString(), v, DontDelete); | |
94 } | |
95 } | |
96 | |
97 // Generate the Arguments Object | |
98 // ECMA v3 10.1.8 | |
99 args = new Darguments(cc.caller, this, actobj, fd.parameters, arglist); | |
100 | |
101 actobj.Put(TEXT_arguments, args, DontDelete); | |
102 | |
103 // The following is not specified by ECMA, but seems to be supported | |
104 // by jscript. The url www.grannymail.com has the following code | |
105 // which looks broken to me but works in jscript: | |
106 // | |
107 // function MakeArray() { | |
108 // this.length = MakeArray.arguments.length | |
109 // for (var i = 0; i < this.length; i++) | |
110 // this[i+1] = arguments[i] | |
111 // } | |
112 // var cardpic = new MakeArray("LL","AP","BA","MB","FH","AW","CW","CV","DZ"); | |
113 Put(TEXT_arguments, args, DontDelete); // make grannymail bug work | |
114 | |
115 auto scoperootsave = cc.scoperoot; | |
116 | |
117 /* If calling a nested function, need to increase scoperoot | |
118 */ | |
119 if (fd.enclosingFunction == cc.callerf) | |
120 { assert(cc.scoperoot < cc.scopex.length); | |
121 cc.scoperoot++; | |
122 } | |
123 else | |
124 { | |
125 auto df = cast(DdeclaredFunction)cc.caller; | |
126 if (df && !fd.isanonymous) | |
127 { | |
128 version (none) | |
129 { | |
130 writefln("current nestDepth = %d, calling %d, cc.scopex.length = %d", | |
131 df.fd.nestDepth, | |
132 fd.nestDepth, | |
133 cc.scopex.length); | |
134 } | |
135 int diff = df.fd.nestDepth - fd.nestDepth; | |
136 if (diff > 0) | |
137 { if (diff >= cc.scoperoot) | |
138 writefln("diff %s cc.scoperoot %s", diff, cc.scoperoot); | |
139 else | |
140 cc.scoperoot -= diff; | |
141 assert(cc.scoperoot >= 1); | |
142 } | |
143 } | |
144 } | |
145 | |
146 Dobject[] scopesave = cc.scopex; | |
147 | |
148 Dobject[] scopex; | |
149 //scopex = cc.scopex[0 .. cc.scoperoot].dup; | |
150 scopex = this.scopex.dup; | |
151 if (scopex.length == 0) | |
152 { this.scopex = cc.scopex[0 .. cc.scoperoot]; | |
153 scopex = this.scopex.dup; | |
154 } | |
155 scopex ~= actobj; | |
156 | |
157 fd.instantiate(scopex, actobj, DontDelete); | |
158 | |
159 cc.scopex = scopex; | |
160 Dobject variablesave = cc.variable; | |
161 cc.variable = actobj; | |
162 auto callersave = cc.caller; | |
163 cc.caller = this; | |
164 auto callerfsave = cc.callerf; | |
165 cc.callerf = fd; | |
166 | |
167 Value[] p1; | |
168 Value* v; | |
169 if (fd.nlocals < 128) | |
170 v = cast(Value*) alloca(fd.nlocals * Value.sizeof); | |
171 if (v) | |
172 locals = v[0 .. fd.nlocals]; | |
173 else | |
174 { | |
175 p1 = new Value[fd.nlocals]; | |
176 locals = p1; | |
177 } | |
178 | |
179 result = IR.call(cc, othis, fd.code, ret, locals.ptr); | |
180 | |
181 delete p1; | |
182 | |
183 cc.callerf = callerfsave; | |
184 cc.caller = callersave; | |
185 cc.variable = variablesave; | |
186 cc.scopex = scopesave; | |
187 cc.scoperoot = scoperootsave; | |
188 | |
189 // Remove the arguments object | |
190 //Value* v; | |
191 //v=Get(TEXT_arguments); | |
192 //writef("1v = %x, %s, v.object = %x\n", v, v.getType(), v.object); | |
193 Put(TEXT_arguments, &vundefined, 0); | |
194 //actobj.Put(TEXT_arguments, &vundefined, 0); | |
195 | |
196 version (none) | |
197 { | |
198 writef("args = %x, actobj = %x\n", args, actobj); | |
199 v=Get(TEXT_arguments); | |
200 writef("2v = %x, %s, v.object = %x\n", v, v.getType(), v.object); | |
201 v.object = null; | |
202 | |
203 { | |
204 uint *p = cast(uint *)0x40a49a80; | |
205 uint i; | |
206 for (i = 0; i < 16; i++) | |
207 { | |
208 writef("p[%x] = %x\n", &p[i], p[i]); | |
209 } | |
210 } | |
211 } | |
212 | |
213 return result; | |
214 } | |
215 | |
216 void *Construct(CallContext *cc, Value *ret, Value[] arglist) | |
217 { | |
218 // ECMA 3 13.2.2 | |
219 Dobject othis; | |
220 Dobject proto; | |
221 Value* v; | |
222 void *result; | |
223 | |
224 v = Get(TEXT_prototype); | |
225 if (v.isPrimitive()) | |
226 proto = Dobject.getPrototype(); | |
227 else | |
228 proto = v.toObject(); | |
229 othis = new Dobject(proto); | |
230 result = Call(cc, othis, ret, arglist); | |
231 if (!result) | |
232 { | |
233 if (ret.isPrimitive()) | |
234 ret.putVobject(othis); | |
235 } | |
236 return result; | |
237 } | |
238 | |
239 d_string toString() | |
240 { | |
241 d_string s; | |
242 | |
243 //writef("DdeclaredFunction.toString()\n"); | |
244 fd.toBuffer(s); | |
245 return s; | |
246 } | |
247 } | |
248 | |
249 |