Left: | ||
Right: |
OLD | NEW |
---|---|
1 /* | 1 /* |
2 * This file is part of Adblock Plus <http://adblockplus.org/>, | 2 * This file is part of Adblock Plus <http://adblockplus.org/>, |
3 * Copyright (C) 2006-2014 Eyeo GmbH | 3 * Copyright (C) 2006-2014 Eyeo GmbH |
4 * | 4 * |
5 * Adblock Plus is free software: you can redistribute it and/or modify | 5 * Adblock Plus is free software: you can redistribute it and/or modify |
6 * it under the terms of the GNU General Public License version 3 as | 6 * it under the terms of the GNU General Public License version 3 as |
7 * published by the Free Software Foundation. | 7 * published by the Free Software Foundation. |
8 * | 8 * |
9 * Adblock Plus is distributed in the hope that it will be useful, | 9 * Adblock Plus is distributed in the hope that it will be useful, |
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of | 10 * but WITHOUT ANY WARRANTY; without even the implied warranty of |
(...skipping 93 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
104 | 104 |
105 function createSetter(typeId, offset) | 105 function createSetter(typeId, offset) |
106 { | 106 { |
107 typeId = typeId | 0; | 107 typeId = typeId | 0; |
108 offset = offset | 0; | 108 offset = offset | 0; |
109 | 109 |
110 let views = Array.prototype.slice.call(arguments, 2); | 110 let views = Array.prototype.slice.call(arguments, 2); |
111 let reference = new Reference(types, views); | 111 let reference = new Reference(types, views); |
112 return function(value) | 112 return function(value) |
113 { | 113 { |
114 if (value && value.typeId != typeId) | 114 if (value && !isinstance(typeId, value)) |
115 throw new Error("Incompatible type"); | 115 throw new Error("Incompatible type"); |
116 | 116 |
117 reference.bufferIndex = this.bufferIndex | 0; | 117 reference.bufferIndex = this.bufferIndex | 0; |
118 reference.byteOffset = (this.byteOffset | 0) + offset; | 118 reference.byteOffset = (this.byteOffset | 0) + offset; |
119 if (value) | 119 if (value) |
120 { | 120 { |
121 reference.typeId = value.typeId; | 121 reference.typeId = value.typeId; |
122 reference.targetBufferIndex = value.bufferIndex; | 122 reference.targetBufferIndex = value.bufferIndex; |
123 reference.targetByteOffset = value.byteOffset; | 123 reference.targetByteOffset = value.byteOffset; |
124 } | 124 } |
125 else | 125 else |
126 reference.typeId = -1; | 126 reference.typeId = -1; |
127 }; | 127 }; |
128 } | 128 } |
129 | 129 |
130 function ObjectType(properties, meta) | 130 /** |
131 * Overridden methods get the respective method of the superclass as the first | |
132 * parameter. This function will create a wrapper function for the method that | |
133 * forwards all arguments to the actual methods but also injects super as first | |
134 * parameter. | |
135 */ | |
136 function createSubclassMethod(method, super_) | |
137 { | |
138 return function() | |
139 { | |
140 let args = [].slice.apply(arguments); | |
141 args.unshift(() => super_.apply(this, arguments)); | |
142 return method.apply(this, args); | |
143 }; | |
144 } | |
145 | |
146 function extend(parentTypeInfo, typeDescriptor, meta) | |
131 { | 147 { |
132 if (typeof meta != "object" || meta == null) | 148 if (typeof meta != "object" || meta == null) |
133 meta = {}; | 149 meta = {}; |
134 | 150 |
135 let propList = []; | 151 let properties = Object.create(parentTypeInfo && parentTypeInfo.properties); |
136 let proto = {}; | 152 |
153 // Methods have to be actually copied here, prototypes won't work correctly | |
154 // with Object.defineProperties(). | |
155 let methods = Object.create(null); | |
156 if (parentTypeInfo) | |
157 for (let key in parentTypeInfo.methods) | |
158 methods[key] = parentTypeInfo.methods[key]; | |
159 | |
137 let maxReferenceLength = TypedReference.byteLength | 0; | 160 let maxReferenceLength = TypedReference.byteLength | 0; |
138 for (let name in properties) | 161 for (let name in typeDescriptor) |
139 { | 162 { |
140 let type = properties[name]; | 163 let type = typeDescriptor[name]; |
141 if (type && typeof type.referenceLength == "number") | 164 if (type && typeof type.referenceLength == "number") |
142 { | 165 { |
166 if (name in methods) | |
167 throw new Error("Property " + name + " masks a method with the same name "); | |
168 if (name in properties) | |
169 { | |
170 if (properties[name] == type) | |
171 continue; | |
172 else | |
173 throw new Error("Cannot redefine type of property " + name + " in subc lass"); | |
174 } | |
175 | |
143 // Property with type | 176 // Property with type |
144 propList.push([name, type]); | 177 properties[name] = type; |
145 | 178 |
146 let referenceLength = type.referenceLength | 0; | 179 let referenceLength = type.referenceLength | 0; |
147 if (referenceLength > maxReferenceLength) | 180 if (referenceLength > maxReferenceLength) |
148 maxReferenceLength = referenceLength; | 181 maxReferenceLength = referenceLength; |
149 } | 182 } |
150 else if (typeof type == "function") | 183 else if (typeof type == "function") |
151 { | 184 { |
152 // Method | 185 // Method |
153 Object.defineProperty(proto, name, fixedPropertyDescriptor(type)); | 186 if (name in properties) |
187 throw new Error("Method " + name + " masks a property with the same name "); | |
188 | |
189 if (name in methods) | |
190 type = createSubclassMethod(type, methods[name].value); | |
191 methods[name] = fixedPropertyDescriptor(type); | |
154 } | 192 } |
155 else | 193 else |
156 throw new Error("Unrecognized type " + type + " given for property " + nam e); | 194 throw new Error("Unrecognized type " + type + " given for property " + nam e); |
157 } | 195 } |
158 | 196 |
197 let proto = {}; | |
159 let buffers = []; | 198 let buffers = []; |
160 let viewTypes = []; | 199 let viewTypes = []; |
161 let views = []; | 200 let views = []; |
162 let byteLength = defineProperties(proto, propList, viewTypes, views, 0); | 201 let byteLength = defineProperties(proto, properties, viewTypes, views, 0); |
202 Object.defineProperties(proto, methods); | |
163 | 203 |
164 // Round up to be a multiple of the maximal property size | 204 // Round up to be a multiple of the maximal property size |
165 byteLength = ((byteLength - 1) | (maxReferenceLength - 1)) + 1; | 205 byteLength = ((byteLength - 1) | (maxReferenceLength - 1)) + 1; |
166 | 206 |
167 // We need to be able to store a typed reference in the object's buffer | 207 // We need to be able to store a typed reference in the object's buffer |
168 byteLength = Math.max(byteLength, TypedReference.byteLength) | 0; | 208 byteLength = Math.max(byteLength, TypedReference.byteLength) | 0; |
169 let typedReferenceViews = getViewsForType(TypedReference, viewTypes, views); | 209 let typedReferenceViews = getViewsForType(TypedReference, viewTypes, views); |
170 | 210 |
211 // Take constructor from meta parameter, allow calling superclass constructor. | |
212 let constructor = parentTypeInfo && parentTypeInfo.constructor; | |
213 if (meta.hasOwnProperty("constructor") && typeof meta.constructor == "function ") | |
214 { | |
215 if (constructor) | |
216 constructor = createSubclassMethod(meta.constructor, constructor); | |
217 else | |
218 constructor = meta.constructor; | |
219 } | |
220 | |
171 let typeId = types.length | 0; | 221 let typeId = types.length | 0; |
172 let typeInfo = { | 222 let typeInfo = { |
173 byteLength: byteLength, | 223 byteLength: byteLength, |
174 bufferSize: "bufferSize" in meta ? Math.max(meta.bufferSize | 0, 2) : 128, | 224 bufferSize: "bufferSize" in meta ? Math.max(meta.bufferSize | 0, 2) : 128, |
175 firstFree: new TypedReference(typeId, typedReferenceViews), | 225 firstFree: new TypedReference(typeId, typedReferenceViews), |
176 proto: proto, | 226 proto: proto, |
227 properties: properties, | |
228 methods: methods, | |
177 buffers: buffers, | 229 buffers: buffers, |
178 viewTypes: viewTypes, | 230 viewTypes: viewTypes, |
179 views: views, | 231 views: views, |
180 typeId: typeId, | 232 typeId: typeId, |
181 constructor: (meta.hasOwnProperty("constructor") && typeof meta.constructor == "function" ? meta.constructor : null) | 233 parentTypeInfo: parentTypeInfo, |
234 constructor: constructor | |
182 }; | 235 }; |
183 | 236 |
184 let result = create.bind(typeInfo); | 237 let result = create.bind(typeInfo); |
185 Object.defineProperties(result, { | 238 Object.defineProperties(result, { |
186 byteLength: fixedPropertyDescriptor(byteLength), | 239 byteLength: fixedPropertyDescriptor(byteLength), |
187 | 240 |
188 referenceLength: fixedPropertyDescriptor(Reference.byteLength), | 241 referenceLength: fixedPropertyDescriptor(Reference.byteLength), |
189 viewTypes: fixedPropertyDescriptor(Reference.viewTypes), | 242 viewTypes: fixedPropertyDescriptor(Reference.viewTypes), |
190 | 243 |
191 typeId: fixedPropertyDescriptor(typeId), | 244 typeId: fixedPropertyDescriptor(typeId), |
245 extend: fixedPropertyDescriptor(extend.bind(null, typeInfo)), | |
246 isinstance: fixedPropertyDescriptor(isinstance.bind(null, typeId)), | |
René Jeschke
2014/05/02 16:16:37
Just wondering, every other member name you use is
Wladimir Palant
2014/05/02 19:14:31
It's really because the JavaScript operator is cal
| |
192 | 247 |
193 createGetter: fixedPropertyDescriptor(createGetter), | 248 createGetter: fixedPropertyDescriptor(createGetter), |
194 createSetter: fixedPropertyDescriptor(createSetter.bind(null, typeId)) | 249 createSetter: fixedPropertyDescriptor(createSetter.bind(null, typeId)) |
195 }); | 250 }); |
196 types.push(typeInfo); | 251 types.push(typeInfo); |
197 return result; | 252 return result; |
198 } | 253 } |
199 exports.ObjectType = ObjectType; | 254 |
255 function isinstance(typeId, obj) | |
256 { | |
257 typeId = typeId | 0; | |
258 | |
259 // TODO: This could be optimized by compiling the list of all subclasses for | |
260 // each type up front. Question is whether this is worth it. | |
261 let typeInfo = types[obj.typeId | 0]; | |
262 while (typeInfo) | |
263 { | |
264 if ((typeInfo.typeId | 0) == typeId) | |
265 return true; | |
266 typeInfo = typeInfo.parentTypeInfo; | |
267 } | |
268 return false; | |
269 } | |
270 | |
271 let ObjectBase = exports.ObjectBase = extend(null, { | |
272 equals: function(obj) | |
273 { | |
274 if (!obj) | |
275 return false; | |
276 return this.typeId == obj.typeId && this.bufferIndex == obj.bufferIndex && t his.byteOffset == obj.byteOffset; | |
277 } | |
278 }, null); | |
279 | |
280 exports.ObjectType = ObjectBase.extend; | |
OLD | NEW |