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