Home | History | Annotate | Download | only in src
      1 // Copyright 2006-2008 the V8 project authors. All rights reserved.
      2 // Redistribution and use in source and binary forms, with or without
      3 // modification, are permitted provided that the following conditions are
      4 // met:
      5 //
      6 //     * Redistributions of source code must retain the above copyright
      7 //       notice, this list of conditions and the following disclaimer.
      8 //     * Redistributions in binary form must reproduce the above
      9 //       copyright notice, this list of conditions and the following
     10 //       disclaimer in the documentation and/or other materials provided
     11 //       with the distribution.
     12 //     * Neither the name of Google Inc. nor the names of its
     13 //       contributors may be used to endorse or promote products derived
     14 //       from this software without specific prior written permission.
     15 //
     16 // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
     17 // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
     18 // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
     19 // A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
     20 // OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
     21 // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
     22 // LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
     23 // DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
     24 // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
     25 // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
     26 // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
     27 
     28 // This file contains infrastructure used by the API.  See
     29 // v8natives.js for an explanation of these files are processed and
     30 // loaded.
     31 
     32 
     33 function CreateDate(time) {
     34   var date = new $Date();
     35   date.setTime(time);
     36   return date;
     37 }
     38 
     39 
     40 var kApiFunctionCache = new InternalArray();
     41 var functionCache = kApiFunctionCache;
     42 
     43 
     44 function Instantiate(data, name) {
     45   if (!%IsTemplate(data)) return data;
     46   var tag = %GetTemplateField(data, kApiTagOffset);
     47   switch (tag) {
     48     case kFunctionTag:
     49       return InstantiateFunction(data, name);
     50     case kNewObjectTag:
     51       var Constructor = %GetTemplateField(data, kApiConstructorOffset);
     52       // Note: Do not directly use a function template as a condition, our
     53       // internal ToBoolean doesn't handle that!
     54       var result = typeof Constructor === 'undefined' ?
     55           {} : new (Instantiate(Constructor))();
     56       ConfigureTemplateInstance(result, data);
     57       result = %ToFastProperties(result);
     58       return result;
     59     default:
     60       throw 'Unknown API tag <' + tag + '>';
     61   }
     62 }
     63 
     64 
     65 function InstantiateFunction(data, name) {
     66   // We need a reference to kApiFunctionCache in the stack frame
     67   // if we need to bail out from a stack overflow.
     68   var cache = kApiFunctionCache;
     69   var serialNumber = %GetTemplateField(data, kApiSerialNumberOffset);
     70   var isFunctionCached =
     71    (serialNumber in cache) && (cache[serialNumber] != kUninitialized);
     72   if (!isFunctionCached) {
     73     try {
     74       var fun = %CreateApiFunction(data);
     75       if (name) %FunctionSetName(fun, name);
     76       var flags = %GetTemplateField(data, kApiFlagOffset);
     77       var doNotCache = flags & (1 << kDoNotCacheBit);
     78       if (!doNotCache) cache[serialNumber] = fun;
     79       if (flags & (1 << kRemovePrototypeBit)) {
     80         %FunctionRemovePrototype(fun);
     81       } else {
     82         var prototype = %GetTemplateField(data, kApiPrototypeTemplateOffset);
     83         // Note: Do not directly use an object template as a condition, our
     84         // internal ToBoolean doesn't handle that!
     85         fun.prototype = typeof prototype === 'undefined' ?
     86             {} : Instantiate(prototype);
     87         if (flags & (1 << kReadOnlyPrototypeBit)) {
     88           %FunctionSetReadOnlyPrototype(fun);
     89         }
     90         %SetProperty(fun.prototype, "constructor", fun, DONT_ENUM);
     91         var parent = %GetTemplateField(data, kApiParentTemplateOffset);
     92         // Note: Do not directly use a function template as a condition, our
     93         // internal ToBoolean doesn't handle that!
     94         if (!(typeof parent === 'undefined')) {
     95           var parent_fun = Instantiate(parent);
     96           %SetPrototype(fun.prototype, parent_fun.prototype);
     97         }
     98       }
     99       ConfigureTemplateInstance(fun, data);
    100       if (doNotCache) return fun;
    101     } catch (e) {
    102       cache[serialNumber] = kUninitialized;
    103       throw e;
    104     }
    105   }
    106   return cache[serialNumber];
    107 }
    108 
    109 
    110 function ConfigureTemplateInstance(obj, data) {
    111   var properties = %GetTemplateField(data, kApiPropertyListOffset);
    112   if (!properties) return;
    113   // Disable access checks while instantiating the object.
    114   var requires_access_checks = %DisableAccessChecks(obj);
    115   try {
    116     for (var i = 1; i < properties[0];) {
    117       var length = properties[i];
    118       if (length == 3) {
    119         var name = properties[i + 1];
    120         var prop_data = properties[i + 2];
    121         var attributes = properties[i + 3];
    122         var value = Instantiate(prop_data, name);
    123         %SetProperty(obj, name, value, attributes);
    124       } else if (length == 5) {
    125         var name = properties[i + 1];
    126         var getter = properties[i + 2];
    127         var setter = properties[i + 3];
    128         var attribute = properties[i + 4];
    129         var access_control = properties[i + 5];
    130         %SetAccessorProperty(
    131             obj, name, getter, setter, attribute, access_control);
    132       } else {
    133         throw "Bad properties array";
    134       }
    135       i += length + 1;
    136     }
    137   } finally {
    138     if (requires_access_checks) %EnableAccessChecks(obj);
    139   }
    140 }
    141