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 
19 module dmdscript.darray;
20 
21 //Nonstandard treatment of Infinity as array length in slice/splice functions, supported by majority of browsers
22 //also treats negative starting index in splice wrapping it around just like in slice
23 version =  SliceSpliceExtension;
24 
25 import std..string;
26 import core.stdc.stdlib;
27 import std.math;
28 
29 import dmdscript.script;
30 import dmdscript.value;
31 import dmdscript.dobject;
32 import dmdscript.threadcontext;
33 import dmdscript.identifier;
34 import dmdscript.dfunction;
35 import dmdscript.text;
36 import dmdscript.property;
37 import dmdscript.errmsgs;
38 import dmdscript.dnative;
39 import dmdscript.program;
40 
41 /* ===================== Darray_constructor ==================== */
42 
43 class DarrayConstructor : Dfunction
44 {
45     this()
46     {
47         super(1, Dfunction_prototype);
48         name = "Array";
49     }
50 
51     override void* Construct(CallContext *cc, Value *ret, Value[] arglist)
52     {
53         // ECMA 15.4.2
54         Darray a;
55 
56         a = new Darray();
57         if(arglist.length == 0)
58         {
59             a.ulength = 0;
60             a.length.number = 0;
61         }
62         else if(arglist.length == 1)
63         {
64             Value* v = &arglist[0];
65 
66             if(v.isNumber())
67             {
68                 d_uint32 len;
69 
70                 len = v.toUint32();
71                 if(cast(double)len != v.number)
72                 {
73                     ErrInfo errinfo;
74 
75                     ret.putVundefined();
76                     return RangeError(&errinfo, ERR_ARRAY_LEN_OUT_OF_BOUNDS, v.number);
77                 }
78                 else
79                 {
80                     a.ulength = len;
81                     a.length.number = len;
82                     /+
83                        if (len > 16)
84                        {
85                         //writef("setting %p dimension to %d\n", &a.proptable, len);
86                         if (len > 10000)
87                             len = 10000;		// cap so we don't run out of memory
88                         a.proptable.roots.setDim(len);
89                         a.proptable.roots.zero();
90                        }
91                      +/
92                 }
93             }
94             else
95             {
96                 a.ulength = 1;
97                 a.length.number = 1;
98                 a.Put(cast(d_uint32)0, v, 0);
99             }
100         }
101         else
102         {
103             //if (arglist.length > 10) writef("Array constructor: arglist.length = %d\n", arglist.length);
104             /+
105                if (arglist.length > 16)
106                {
107                 a.proptable.roots.setDim(arglist.length);
108                 a.proptable.roots.zero();
109                }
110              +/
111             a.ulength = cast(uint)arglist.length;
112             a.length.number = arglist.length;
113             for(uint k = 0; k < arglist.length; k++)
114             {
115                 a.Put(k, &arglist[k], 0);
116             }
117         }
118         Value.copy(ret, &a.value);
119         //writef("Darray_constructor.Construct(): length = %g\n", a.length.number);
120         return null;
121     }
122 
123     override void* Call(CallContext *cc, Dobject othis, Value* ret, Value[] arglist)
124     {
125         // ECMA 15.4.1
126         return Construct(cc, ret, arglist);
127     }
128 }
129 
130 
131 /* ===================== Darray_prototype_toString ================= */
132 
133 void *Darray_prototype_toString(Dobject pthis, CallContext *cc, Dobject othis, Value *ret, Value[] arglist)
134 {
135     //writef("Darray_prototype_toString()\n");
136     array_join(othis, ret, null);
137     return null;
138 }
139 
140 /* ===================== Darray_prototype_toLocaleString ================= */
141 
142 void *Darray_prototype_toLocaleString(Dobject pthis, CallContext *cc, Dobject othis, Value *ret, Value[] arglist)
143 {
144     // ECMA v3 15.4.4.3
145     d_string separator;
146     d_string r;
147     d_uint32 len;
148     d_uint32 k;
149     Value* v;
150 
151     //writef("array_join(othis = %p)\n", othis);
152 
153     if(!othis.isClass(TEXT_Array))
154     {
155         ret.putVundefined();
156         ErrInfo errinfo;
157         return Dobject.RuntimeError(&errinfo, ERR_TLS_NOT_TRANSFERRABLE);
158     }
159 
160     v = othis.Get(TEXT_length);
161     len = v ? v.toUint32() : 0;
162 
163     Program prog = cc.prog;
164     if(!prog.slist)
165     {
166         // Determine what list separator is only once per thread
167         //prog.slist = list_separator(prog.lcid);
168         prog.slist = ",";
169     }
170     separator = prog.slist;
171 
172     for(k = 0; k != len; k++)
173     {
174         if(k)
175             r ~= separator;
176         v = othis.Get(k);
177         if(v && !v.isUndefinedOrNull())
178         {
179             Dobject ot;
180 
181             ot = v.toObject();
182             v = ot.Get(TEXT_toLocaleString);
183             if(v && !v.isPrimitive())   // if it's an Object
184             {
185                 void* a;
186                 Dobject o;
187                 Value rt;
188 
189                 o = v.object;
190                 rt.putVundefined();
191                 a = o.Call(cc, ot, &rt, null);
192                 if(a)                   // if exception was thrown
193                     return a;
194                 r ~= rt.toString();
195             }
196         }
197     }
198 
199     ret.putVstring(r);
200     return null;
201 }
202 
203 /* ===================== Darray_prototype_concat ================= */
204 
205 void *Darray_prototype_concat(Dobject pthis, CallContext *cc, Dobject othis, Value *ret, Value[] arglist)
206 {
207     // ECMA v3 15.4.4.4
208     Darray A;
209     Darray E;
210     Value* v;
211     d_uint32 k;
212     d_uint32 n;
213     d_uint32 a;
214 
215     A = new Darray();
216     n = 0;
217     v = &othis.value;
218     for(a = 0;; a++)
219     {
220         if(!v.isPrimitive() && v.object.isDarray())
221         {
222             d_uint32 len;
223 
224             E = cast(Darray)v.object;
225             len = E.ulength;
226             for(k = 0; k != len; k++)
227             {
228                 v = E.Get(k);
229                 if(v)
230                     A.Put(n, v, 0);
231                 n++;
232             }
233         }
234         else
235         {
236             A.Put(n, v, 0);
237             n++;
238         }
239         if(a == arglist.length)
240             break;
241         v = &arglist[a];
242     }
243 
244     A.Put(TEXT_length, n,  DontEnum);
245     Value.copy(ret, &A.value);
246     return null;
247 }
248 
249 /* ===================== Darray_prototype_join ================= */
250 
251 void *Darray_prototype_join(Dobject pthis, CallContext *cc, Dobject othis, Value *ret, Value[] arglist)
252 {
253     array_join(othis, ret, arglist);
254     return null;
255 }
256 
257 void array_join(Dobject othis, Value* ret, Value[] arglist)
258 {
259     // ECMA 15.4.4.3
260     d_string separator;
261     d_string r;
262     d_uint32 len;
263     d_uint32 k;
264     Value* v;
265 
266     //writef("array_join(othis = %p)\n", othis);
267     v = othis.Get(TEXT_length);
268     len = v ? v.toUint32() : 0;
269     if(arglist.length == 0 || arglist[0].isUndefined())
270         separator = TEXT_comma;
271     else
272         separator = arglist[0].toString();
273 
274     for(k = 0; k != len; k++)
275     {
276         if(k)
277             r ~= separator;
278         v = othis.Get(k);
279         if(v && !v.isUndefinedOrNull())
280             r ~= v.toString();
281     }
282 
283     ret.putVstring(r);
284 }
285 
286 /* ===================== Darray_prototype_toSource ================= */
287 
288 void *Darray_prototype_toSource(Dobject pthis, CallContext *cc, Dobject othis, Value *ret, Value[] arglist)
289 {
290     d_string separator;
291     d_string r;
292     d_uint32 len;
293     d_uint32 k;
294     Value* v;
295 
296     v = othis.Get(TEXT_length);
297     len = v ? v.toUint32() : 0;
298     separator = ",";
299 
300     r = "[".idup;
301     for(k = 0; k != len; k++)
302     {
303         if(k)
304             r ~= separator;
305         v = othis.Get(k);
306         if(v && !v.isUndefinedOrNull())
307             r ~= v.toSource();
308     }
309     r ~= "]";
310 
311     ret.putVstring(r);
312     return null;
313 }
314 
315 
316 /* ===================== Darray_prototype_pop ================= */
317 
318 void *Darray_prototype_pop(Dobject pthis, CallContext *cc, Dobject othis, Value *ret, Value[] arglist)
319 {
320     // ECMA v3 15.4.4.6
321     Value* v;
322     d_uint32 u;
323 
324     // If othis is a Darray, then we can optimize this significantly
325     v = othis.Get(TEXT_length);
326     if(!v)
327         v = &vundefined;
328     u = v.toUint32();
329     if(u == 0)
330     {
331         othis.Put(TEXT_length, 0.0,  DontEnum);
332         ret.putVundefined();
333     }
334     else
335     {
336         v = othis.Get(u - 1);
337         if(!v)
338             v = &vundefined;
339         Value.copy(ret, v);
340         othis.Delete(u - 1);
341         othis.Put(TEXT_length, u - 1,  DontEnum);
342     }
343     return null;
344 }
345 
346 /* ===================== Darray_prototype_push ================= */
347 
348 void *Darray_prototype_push(Dobject pthis, CallContext *cc, Dobject othis, Value *ret, Value[] arglist)
349 {
350     // ECMA v3 15.4.4.7
351     Value* v;
352     d_uint32 u;
353     d_uint32 a;
354 
355     // If othis is a Darray, then we can optimize this significantly
356     v = othis.Get(TEXT_length);
357     if(!v)
358         v = &vundefined;
359     u = v.toUint32();
360     for(a = 0; a < arglist.length; a++)
361     {
362         othis.Put(u + a, &arglist[a], 0);
363     }
364     othis.Put(TEXT_length, u + a,  DontEnum);
365     ret.putVnumber(u + a);
366     return null;
367 }
368 
369 /* ===================== Darray_prototype_reverse ================= */
370 
371 void *Darray_prototype_reverse(Dobject pthis, CallContext *cc, Dobject othis, Value *ret, Value[] arglist)
372 {
373     // ECMA 15.4.4.4
374     d_uint32 a;
375     d_uint32 b;
376     Value* va;
377     Value* vb;
378     Value* v;
379     d_uint32 pivot;
380     d_uint32 len;
381     Value tmp;
382 
383     v = othis.Get(TEXT_length);
384     len = v ? v.toUint32() : 0;
385     pivot = len / 2;
386     for(a = 0; a != pivot; a++)
387     {
388         b = len - a - 1;
389         //writef("a = %d, b = %d\n", a, b);
390         va = othis.Get(a);
391         if(va)
392             Value.copy(&tmp, va);
393         vb = othis.Get(b);
394         if(vb)
395             othis.Put(a, vb, 0);
396         else
397             othis.Delete(a);
398 
399         if(va)
400             othis.Put(b, &tmp, 0);
401         else
402             othis.Delete(b);
403     }
404     Value.copy(ret, &othis.value);
405     return null;
406 }
407 
408 /* ===================== Darray_prototype_shift ================= */
409 
410 void *Darray_prototype_shift(Dobject pthis, CallContext *cc, Dobject othis, Value *ret, Value[] arglist)
411 {
412     // ECMA v3 15.4.4.9
413     Value* v;
414     Value* result;
415     d_uint32 len;
416     d_uint32 k;
417 
418     // If othis is a Darray, then we can optimize this significantly
419     //writef("shift(othis = %p)\n", othis);
420     v = othis.Get(TEXT_length);
421     if(!v)
422         v = &vundefined;
423     len = v.toUint32();
424     
425     if(len)
426     {
427         result = othis.Get(0u);
428         Value.copy(ret, result ? result : &vundefined);
429         for(k = 1; k != len; k++)
430         {
431             v = othis.Get(k);
432             if(v)
433             {
434                 othis.Put(k - 1, v, 0);
435             }
436             else
437             {
438                 othis.Delete(k - 1);
439             }
440         }
441         othis.Delete(len - 1);
442         len--;
443     }
444     else
445         Value.copy(ret, &vundefined);
446 
447     othis.Put(TEXT_length, len, DontEnum);
448     return null;
449 }
450 
451 
452 /* ===================== Darray_prototype_slice ================= */
453 
454 void *Darray_prototype_slice(Dobject pthis, CallContext *cc, Dobject othis, Value *ret, Value[] arglist)
455 {
456     // ECMA v3 15.4.4.10
457     d_uint32 len;
458     d_uint32 n;
459     d_uint32 k;
460     d_uint32 r8;
461 
462     Value* v;
463     Darray A;
464 
465     v = othis.Get(TEXT_length);
466     if(!v)
467         v = &vundefined;
468     len = v.toUint32();
469 
470 version(SliceSpliceExtension){
471     d_number start;
472     d_number end;
473     switch(arglist.length)
474     {
475     case 0:
476         start = vundefined.toNumber();
477         end = len;
478         break;
479 
480     case 1:
481         start = arglist[0].toNumber();
482         end = len;
483         break;
484 
485     default:
486         start = arglist[0].toNumber();
487 		if(arglist[1].isUndefined())
488 			end = len;
489 		else{
490 			end = arglist[1].toNumber();
491 		}
492         break;
493     }
494     if(start < 0)
495     {
496         k = cast(uint)(len + start);
497         if(cast(d_int32)k < 0)
498             k = 0;
499     }
500     else if(start == d_number.infinity)
501         k = len;
502     else if(start == -d_number.infinity)
503         k = 0;
504     else
505     {
506         k = cast(uint)start;
507         if(len < k)
508             k = len;
509     }
510 
511     if(end < 0)
512     {
513         r8 = cast(uint)(len + end);
514         if(cast(d_int32)r8 < 0)
515             r8 = 0;
516     }
517     else if(end == d_number.infinity)
518             r8 = len;
519     else if(end == -d_number.infinity)
520             r8 = 0;
521     else
522     {
523         r8 = cast(uint)end;
524         if(len < end)
525             r8 = len;
526     }
527 }
528 else{//Canonical ECMA all kinds of infinity maped to 0
529     int start;
530     int end;
531     switch(arglist.length)
532     {
533     case 0:
534         start = vundefined.toInt32();
535         end = len;
536         break;
537 
538     case 1:
539         start = arglist[0].toInt32();
540         end = len;
541         break;
542 
543     default:
544         start = arglist[0].toInt32();
545 		if(arglist[1].isUndefined())
546 			end = len;
547 		else{
548 			end = arglist[1].toInt32();
549 		}
550         break;
551     }
552     if(start < 0)
553     {
554         k = cast(uint)(len + start);
555         if(cast(d_int32)k < 0)
556             k = 0;
557     }
558     else
559     {
560         k = cast(uint)start;
561         if(len < k)
562             k = len;
563     }
564 
565     if(end < 0)
566     {
567         r8 = cast(uint)(len + end);
568         if(cast(d_int32)r8 < 0)
569             r8 = 0;
570     }
571     else
572     {
573         r8 = cast(uint)end;
574         if(len < end)
575             r8 = len;
576     }
577 }
578     A = new Darray();
579     for(n = 0; k < r8; k++)
580     {
581         v = othis.Get(k);
582         if(v)
583         {
584             A.Put(n, v, 0);
585         }
586         n++;
587     }
588 
589     A.Put(TEXT_length, n, DontEnum);
590     Value.copy(ret, &A.value);
591     return null;
592 }
593 
594 /* ===================== Darray_prototype_sort ================= */
595 
596 
597 void *Darray_prototype_sort(Dobject pthis, CallContext *cc, Dobject othis, Value *ret, Value[] arglist)
598 {
599     // ECMA v3 15.4.4.11
600     Value* v;
601     d_uint32 len;
602     uint u;
603 
604     //writef("Array.prototype.sort()\n");
605     v = othis.Get(TEXT_length);
606     len = v ? v.toUint32() : 0;
607 
608     // This is not optimal, as isArrayIndex is done at least twice
609     // for every array member. Additionally, the qsort() by index
610     // can be avoided if we can deduce it is not a sparse array.
611 
612     Property *p;
613     Value[] pvalues;
614     d_uint32[] pindices;
615     d_uint32 parraydim;
616     d_uint32 nprops;
617 
618     // First, size & alloc our temp array
619     if(len < 100)
620     {   // Probably not too sparse an array
621         parraydim = len;
622     }
623     else
624     {
625         parraydim = 0;
626         foreach(ref Property p; *othis.proptable)
627         {
628             if(p.attributes == 0)       // don't count special properties
629                 parraydim++;
630         }
631         if(parraydim > len)             // could theoretically happen
632             parraydim = len;
633     }
634 
635     Value[] p1 = null;
636     Value* v1;
637     version(Win32)      // eh and alloca() not working under linux
638     {
639         if(parraydim < 128)
640             v1 = cast(Value*)alloca(parraydim * Value.sizeof);
641     }
642     if(v1)
643         pvalues = v1[0 .. parraydim];
644     else
645     {
646         p1 = new Value[parraydim];
647         pvalues = p1;
648     }
649 
650     d_uint32[] p2 = null;
651     d_uint32* p3;
652     version(Win32)
653     {
654         if(parraydim < 128)
655             p3 = cast(d_uint32*)alloca(parraydim * d_uint32.sizeof);
656     }
657     if(p3)
658         pindices = p3[0 .. parraydim];
659     else
660     {
661         p2 = new d_uint32[parraydim];
662         pindices = p2;
663     }
664 
665     // Now fill it with all the Property's that are array indices
666     nprops = 0;
667     foreach(Value key, ref Property p; *othis.proptable)
668     {
669         d_uint32 index;
670 
671         if(p.attributes == 0 && key.isArrayIndex(index))
672         {
673             pindices[nprops] = index;
674             Value.copy(&pvalues[nprops], &p.value);
675             nprops++;
676         }
677     }
678 
679     synchronized
680     {
681         Dobject comparefn;
682 
683         if(arglist.length)
684         {
685             if(!arglist[0].isPrimitive())
686                 comparefn = arglist[0].object;
687         }
688 
689 
690         bool compare_value(ref Value vx, ref Value vy)
691         {
692             d_string sx;
693             d_string sy;
694             int cmp;
695 
696             //writef("compare_value()\n");
697             if(vx.isUndefined())
698             {
699                 cmp = (vy.isUndefined()) ? 0 : 1;
700             }
701             else if(vy.isUndefined())
702                 cmp = -1;
703             else
704             {
705                 if(comparefn)
706                 {
707                     Value[2] arglist;
708                     Value ret;
709                     Value* v;
710                     d_number n;
711 
712                     Value.copy(&arglist[0], &vx);
713                     Value.copy(&arglist[1], &vy);
714                     ret.putVundefined();
715                     comparefn.Call(cc, comparefn, &ret, arglist);
716                     n = ret.toNumber();
717                     if(n < 0)
718                         cmp = -1;
719                     else if(n > 0)
720                         cmp = 1;
721                     else
722                         cmp = 0;
723                 }
724                 else
725                 {
726                     sx = vx.toString();
727                     sy = vy.toString();
728                     cmp = std..string.cmp(sx, sy);
729                     if(cmp < 0)
730                         cmp = -1;
731                     else if(cmp > 0)
732                         cmp = 1;
733                 }
734             }
735             return cmp < 0;
736         }
737 
738         // Sort pvalues[]
739         import std.algorithm.sorting : sort;
740         pvalues[0 .. nprops].sort!compare_value();
741     }
742 
743     // Stuff the sorted value's back into the array
744     for(u = 0; u < nprops; u++)
745     {
746         d_uint32 index;
747 
748         othis.Put(u, &pvalues[u], 0);
749         index = pindices[u];
750         if(index >= nprops)
751         {
752             othis.Delete(index);
753         }
754     }
755 
756     delete p1;
757     delete p2;
758 
759     ret.putVobject(othis);
760     return null;
761 }
762 
763 /* ===================== Darray_prototype_splice ================= */
764 
765 void *Darray_prototype_splice(Dobject pthis, CallContext *cc, Dobject othis, Value *ret, Value[] arglist)
766 {
767     // ECMA v3 15.4.4.12
768     d_uint32 len;
769     d_uint32 k;
770     
771     Value* v;
772     Darray A;
773     d_uint32 a;
774     d_uint32 delcnt;
775     d_uint32 inscnt;
776     d_uint32 startidx;
777     
778     v = othis.Get(TEXT_length);
779     if(!v)
780         v = &vundefined;
781     len = v.toUint32();
782     
783 version(SliceSpliceExtension){
784     d_number start;
785     d_number deleteCount;
786     
787     switch(arglist.length)
788     {
789     case 0:
790         start = vundefined.toNumber();
791         deleteCount = 0;
792         break;
793 
794     case 1:
795         start = arglist[0].toNumber();
796         deleteCount = vundefined.toNumber();
797         break;
798 
799     default:
800         start = arglist[0].toNumber();
801         deleteCount = arglist[1].toNumber();
802 		//checked later
803         break;
804     }
805     if(start == d_number.infinity)
806         startidx = len;
807     else if(start == -d_number.infinity)
808         startidx = 0;
809     else{
810         if(start < 0)
811         {
812             startidx = cast(uint)(len + start);
813             if(cast(d_int32)startidx < 0)
814                 startidx = 0;
815         }
816         else
817             startidx = cast(uint)start;
818     }
819 	startidx = startidx > len ? len : startidx; 
820     if(deleteCount == d_number.infinity)
821         delcnt = len;
822     else if(deleteCount == -d_number.infinity)
823         delcnt = 0;
824     else
825         delcnt = (cast(uint)deleteCount > 0) ? cast(uint) deleteCount : 0;
826     if(delcnt > len - startidx)
827         delcnt = len - startidx;
828 }else{    
829     long start;
830     d_int32 deleteCount;
831     switch(arglist.length)
832     {
833     case 0:
834         start = vundefined.toInt32();
835         deleteCount = 0;
836         break;
837 
838     case 1:
839         start = arglist[0].toInt32();
840         deleteCount = vundefined.toInt32();
841         break;
842 
843     default:
844         start = arglist[0].toInt32();
845         deleteCount = arglist[1].toInt32();
846 		//checked later
847         break;
848     }
849     startidx = cast(uint)start;
850 	startidx = startidx > len ? len : startidx; 
851     delcnt = (deleteCount > 0) ? deleteCount : 0;
852     if(delcnt > len - startidx)
853         delcnt = len - startidx;
854 }
855 	
856     A = new Darray();
857 
858     // If deleteCount is not specified, ECMA implies it should
859     // be 0, while "JavaScript The Definitive Guide" says it should
860     // be delete to end of array. Jscript doesn't implement splice().
861     // We'll do it the Guide way.
862     if(arglist.length < 2)
863         delcnt = len - startidx;
864 
865     //writef("Darray.splice(startidx = %d, delcnt = %d)\n", startidx, delcnt);
866     for(k = 0; k != delcnt; k++)
867     {
868         v = othis.Get(startidx + k);
869         if(v)
870             A.Put(k, v, 0);
871     }
872 
873     A.Put(TEXT_length, delcnt, DontEnum);
874     inscnt = (arglist.length > 2) ? cast(uint)arglist.length - 2 : 0;
875     if(inscnt != delcnt)
876     {
877         if(inscnt <= delcnt)
878         {
879             for(k = startidx; k != (len - delcnt); k++)
880             {
881                 v = othis.Get(k + delcnt);
882                 if(v)
883                     othis.Put(k + inscnt, v, 0);
884                 else
885                     othis.Delete(k + inscnt);
886             }
887 
888             for(k = len; k != (len - delcnt + inscnt); k--)
889                 othis.Delete(k - 1);
890         }
891         else
892         {
893             for(k = len - delcnt; k != startidx; k--)
894             {
895                 v = othis.Get(k + delcnt - 1);
896                 if(v)
897                     othis.Put(k + inscnt - 1, v, 0);
898                 else
899                     othis.Delete(k + inscnt - 1);
900             }
901         }
902     }
903     k = startidx;
904     for(a = 2; a < arglist.length; a++)
905     {
906         v = &arglist[a];
907         othis.Put(k, v, 0);
908         k++;
909     }
910 
911     othis.Put(TEXT_length, len - delcnt + inscnt,  DontEnum);
912     Value.copy(ret, &A.value);
913     return null;
914 }
915 
916 /* ===================== Darray_prototype_unshift ================= */
917 
918 void *Darray_prototype_unshift(Dobject pthis, CallContext *cc, Dobject othis, Value *ret, Value[] arglist)
919 {
920     // ECMA v3 15.4.4.13
921     Value* v;
922     d_uint32 len;
923     d_uint32 k;
924 
925     v = othis.Get(TEXT_length);
926     if(!v)
927         v = &vundefined;
928     len = v.toUint32();
929 
930     for(k = len; k>0; k--)
931     {
932         v = othis.Get(k - 1);
933         if(v)
934             othis.Put(cast(uint)(k + arglist.length - 1), v, 0);
935         else
936             othis.Delete(cast(uint)(k + arglist.length - 1));
937     }
938 
939     for(k = 0; k < arglist.length; k++)
940     {
941         othis.Put(k, &arglist[k], 0);
942     }
943     othis.Put(TEXT_length, len + arglist.length,  DontEnum);
944     ret.putVnumber(len + arglist.length);
945     return null;
946 }
947 
948 /* =========================== Darray_prototype =================== */
949 
950 class DarrayPrototype : Darray
951 {
952     this()
953     {
954         super(Dobject_prototype);
955         Dobject f = Dfunction_prototype;
956 
957         Put(TEXT_constructor, Darray_constructor, DontEnum);
958 
959         static enum NativeFunctionData[] nfd =
960         [
961             { TEXT_toString, &Darray_prototype_toString, 0 },
962             { TEXT_toLocaleString, &Darray_prototype_toLocaleString, 0 },
963             { TEXT_toSource, &Darray_prototype_toSource, 0 },
964             { TEXT_concat, &Darray_prototype_concat, 1 },
965             { TEXT_join, &Darray_prototype_join, 1 },
966             { TEXT_pop, &Darray_prototype_pop, 0 },
967             { TEXT_push, &Darray_prototype_push, 1 },
968             { TEXT_reverse, &Darray_prototype_reverse, 0 },
969             { TEXT_shift, &Darray_prototype_shift, 0, },
970             { TEXT_slice, &Darray_prototype_slice, 2 },
971             { TEXT_sort, &Darray_prototype_sort, 1 },
972             { TEXT_splice, &Darray_prototype_splice, 2 },
973             { TEXT_unshift, &Darray_prototype_unshift, 1 },
974         ];
975 
976         DnativeFunction.initialize(this, nfd, DontEnum);
977     }
978 }
979 
980 
981 /* =========================== Darray =================== */
982 
983 class Darray : Dobject
984 {
985     Value length;               // length property
986     d_uint32 ulength;
987 
988     this()
989     {
990         this(getPrototype());
991     }
992 
993     this(Dobject prototype)
994     {
995         super(prototype);
996         length.putVnumber(0);
997         ulength = 0;
998         classname = TEXT_Array;
999     }
1000 
1001     override  Value* Put(Identifier* key, Value* value, uint attributes)
1002     {
1003         Value* result = proptable.put(&key.value, key.value.hash, value, attributes);
1004         if(!result)
1005             Put(key.value..string, value, attributes);
1006         return null;
1007     }
1008 
1009     override Value* Put(d_string name, Value* v, uint attributes)
1010     {
1011         d_uint32 i;
1012         uint c;
1013         Value* result;
1014 
1015         // ECMA 15.4.5.1
1016         result = proptable.put(name, v, attributes);
1017         if(!result)
1018         {
1019             if(name == TEXT_length)
1020             {
1021                 i = v.toUint32();
1022                 if(i != v.toInteger())
1023                 {
1024                     ErrInfo errinfo;
1025 
1026                     return Dobject.RangeError(&errinfo, ERR_LENGTH_INT);
1027                 }
1028                 if(i < ulength)
1029                 {
1030                     // delete all properties with keys >= i
1031                     d_uint32[] todelete;
1032 
1033                     foreach(Value key, ref Property p; *proptable)
1034                     {
1035                         d_uint32 j;
1036 
1037                         j = key.toUint32();
1038                         if(j >= i)
1039                             todelete ~= j;
1040                     }
1041                     foreach(d_uint32 j; todelete)
1042                     {
1043                         proptable.del(j);
1044                     }
1045                 }
1046                 ulength = i;
1047                 length.number = i;
1048                 proptable.put(name, v, attributes | DontEnum);
1049             }
1050 
1051             // if (name is an array index i)
1052 
1053             i = 0;
1054             for(size_t j = 0; j < name.length; j++)
1055             {
1056                 ulong k;
1057 
1058                 c = name[j];
1059                 if(c == '0' && i == 0 && name.length > 1)
1060                     goto Lret;
1061                 if(c >= '0' && c <= '9')
1062                 {
1063                     k = i * cast(ulong)10 + c - '0';
1064                     i = cast(d_uint32)k;
1065                     if(i != k)
1066                         goto Lret;              // overflow
1067                 }
1068                 else
1069                     goto Lret;
1070             }
1071             if(i >= ulength)
1072             {
1073                 if(i == 0xFFFFFFFF)
1074                     goto Lret;
1075                 ulength = i + 1;
1076                 length.number = ulength;
1077             }
1078         }
1079         Lret:
1080         return null;
1081     }
1082 
1083     override Value* Put(d_string name, Dobject o, uint attributes)
1084     {
1085         return Put(name, &o.value, attributes);
1086     }
1087 
1088     override Value* Put(d_string PropertyName, d_number n, uint attributes)
1089     {
1090         Value v;
1091 
1092         v.putVnumber(n);
1093         return Put(PropertyName, &v, attributes);
1094     }
1095 
1096     override Value* Put(d_string PropertyName, d_string string, uint attributes)
1097     {
1098         Value v;
1099 
1100         v.putVstring(string);
1101         return Put(PropertyName, &v, attributes);
1102     }
1103 
1104     override Value* Put(d_uint32 index, Value* vindex, Value* value, uint attributes)
1105     {
1106         if(index >= ulength)
1107             ulength = index + 1;
1108 
1109         proptable.put(vindex, index ^ 0x55555555 /*Value.calcHash(index)*/, value, attributes);
1110         return null;
1111     }
1112 
1113     override Value* Put(d_uint32 index, Value* value, uint attributes)
1114     {
1115         if(index >= ulength)
1116         {
1117             ulength = index + 1;
1118             length.number = ulength;
1119         }
1120 
1121         proptable.put(index, value, attributes);
1122         return null;
1123     }
1124 
1125     final Value* Put(d_uint32 index, d_string string, uint attributes)
1126     {
1127         if(index >= ulength)
1128         {
1129             ulength = index + 1;
1130             length.number = ulength;
1131         }
1132 
1133         proptable.put(index, string, attributes);
1134         return null;
1135     }
1136 
1137     override Value* Get(Identifier* id)
1138     {
1139         //writef("Darray.Get(%p, '%s')\n", &proptable, PropertyName);
1140         if(id.value..string == TEXT_length)
1141         {
1142             length.number = ulength;
1143             return &length;
1144         }
1145         else
1146             return Dobject.Get(id);
1147     }
1148 
1149     override Value* Get(d_string PropertyName, uint hash)
1150     {
1151         //writef("Darray.Get(%p, '%s')\n", &proptable, PropertyName);
1152         if(PropertyName == TEXT_length)
1153         {
1154             length.number = ulength;
1155             return &length;
1156         }
1157         else
1158             return Dobject.Get(PropertyName, hash);
1159     }
1160 
1161     override Value* Get(d_uint32 index)
1162     {
1163         Value* v;
1164 
1165         //writef("Darray.Get(%p, %d)\n", &proptable, index);
1166         v = proptable.get(index);
1167         return v;
1168     }
1169 
1170     override Value* Get(d_uint32 index, Value* vindex)
1171     {
1172         Value* v;
1173 
1174         //writef("Darray.Get(%p, %d)\n", &proptable, index);
1175         v = proptable.get(vindex, index ^ 0x55555555 /*Value.calcHash(index)*/);
1176         return v;
1177     }
1178 
1179     override int Delete(d_string PropertyName)
1180     {
1181         // ECMA 8.6.2.5
1182         //writef("Darray.Delete('%ls')\n", d_string_ptr(PropertyName));
1183         if(PropertyName == TEXT_length)
1184             return 0;           // can't delete 'length' property
1185         else
1186             return proptable.del(PropertyName);
1187     }
1188 
1189     override int Delete(d_uint32 index)
1190     {
1191         // ECMA 8.6.2.5
1192         return proptable.del(index);
1193     }
1194 
1195 
1196     static Dfunction getConstructor()
1197     {
1198         return Darray_constructor;
1199     }
1200 
1201     static Dobject getPrototype()
1202     {
1203         return Darray_prototype;
1204     }
1205 
1206     static void initialize()
1207     {
1208         Darray_constructor = new DarrayConstructor();
1209         Darray_prototype = new DarrayPrototype();
1210 
1211         Darray_constructor.Put(TEXT_prototype, Darray_prototype, DontEnum |  ReadOnly);
1212     }
1213 }
1214