]>
Commit | Line | Data |
---|---|---|
93a37866 A |
1 | /* |
2 | * Copyright (C) 2013 Apple Inc. All rights reserved. | |
3 | * | |
4 | * Redistribution and use in source and binary forms, with or without | |
5 | * modification, are permitted provided that the following conditions | |
6 | * are met: | |
7 | * 1. Redistributions of source code must retain the above copyright | |
8 | * notice, this list of conditions and the following disclaimer. | |
9 | * 2. Redistributions in binary form must reproduce the above copyright | |
10 | * notice, this list of conditions and the following disclaimer in the | |
11 | * documentation and/or other materials provided with the distribution. | |
12 | * | |
13 | * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY | |
14 | * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE | |
15 | * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR | |
16 | * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR | |
17 | * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, | |
18 | * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, | |
19 | * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR | |
20 | * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY | |
21 | * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | |
22 | * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE | |
23 | * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | |
24 | */ | |
25 | ||
26 | #include "config.h" | |
27 | #import "JavaScriptCore.h" | |
28 | ||
29 | #if JSC_OBJC_API_ENABLED | |
30 | ||
12899fa2 | 31 | #import "APICallbackFunction.h" |
93a37866 | 32 | #import "APICast.h" |
81345200 | 33 | #import "DelayedReleaseScope.h" |
93a37866 A |
34 | #import "Error.h" |
35 | #import "JSCJSValueInlines.h" | |
36 | #import "JSCell.h" | |
37 | #import "JSCellInlines.h" | |
38 | #import "JSContextInternal.h" | |
39 | #import "JSWrapperMap.h" | |
40 | #import "JSValueInternal.h" | |
41 | #import "ObjCCallbackFunction.h" | |
42 | #import "ObjcRuntimeExtras.h" | |
43 | #import <objc/runtime.h> | |
44 | #import <wtf/RetainPtr.h> | |
45 | ||
46 | class CallbackArgument { | |
47 | public: | |
48 | virtual ~CallbackArgument(); | |
49 | virtual void set(NSInvocation *, NSInteger, JSContext *, JSValueRef, JSValueRef*) = 0; | |
50 | ||
51 | OwnPtr<CallbackArgument> m_next; | |
52 | }; | |
53 | ||
54 | CallbackArgument::~CallbackArgument() | |
55 | { | |
56 | } | |
57 | ||
58 | class CallbackArgumentBoolean : public CallbackArgument { | |
59 | virtual void set(NSInvocation *invocation, NSInteger argumentNumber, JSContext *context, JSValueRef argument, JSValueRef*) override | |
60 | { | |
61 | bool value = JSValueToBoolean([context JSGlobalContextRef], argument); | |
62 | [invocation setArgument:&value atIndex:argumentNumber]; | |
63 | } | |
64 | }; | |
65 | ||
66 | template<typename T> | |
67 | class CallbackArgumentInteger : public CallbackArgument { | |
68 | virtual void set(NSInvocation *invocation, NSInteger argumentNumber, JSContext *context, JSValueRef argument, JSValueRef* exception) override | |
69 | { | |
70 | T value = (T)JSC::toInt32(JSValueToNumber([context JSGlobalContextRef], argument, exception)); | |
71 | [invocation setArgument:&value atIndex:argumentNumber]; | |
72 | } | |
73 | }; | |
74 | ||
75 | template<typename T> | |
76 | class CallbackArgumentDouble : public CallbackArgument { | |
77 | virtual void set(NSInvocation *invocation, NSInteger argumentNumber, JSContext *context, JSValueRef argument, JSValueRef* exception) override | |
78 | { | |
79 | T value = (T)JSValueToNumber([context JSGlobalContextRef], argument, exception); | |
80 | [invocation setArgument:&value atIndex:argumentNumber]; | |
81 | } | |
82 | }; | |
83 | ||
84 | class CallbackArgumentJSValue : public CallbackArgument { | |
85 | virtual void set(NSInvocation *invocation, NSInteger argumentNumber, JSContext *context, JSValueRef argument, JSValueRef*) override | |
86 | { | |
87 | JSValue *value = [JSValue valueWithJSValueRef:argument inContext:context]; | |
88 | [invocation setArgument:&value atIndex:argumentNumber]; | |
89 | } | |
90 | }; | |
91 | ||
92 | class CallbackArgumentId : public CallbackArgument { | |
93 | virtual void set(NSInvocation *invocation, NSInteger argumentNumber, JSContext *context, JSValueRef argument, JSValueRef*) override | |
94 | { | |
95 | id value = valueToObject(context, argument); | |
96 | [invocation setArgument:&value atIndex:argumentNumber]; | |
97 | } | |
98 | }; | |
99 | ||
100 | class CallbackArgumentOfClass : public CallbackArgument { | |
101 | public: | |
102 | CallbackArgumentOfClass(Class cls) | |
103 | : CallbackArgument() | |
104 | , m_class(cls) | |
105 | { | |
106 | [m_class retain]; | |
107 | } | |
108 | ||
109 | private: | |
110 | virtual ~CallbackArgumentOfClass() | |
111 | { | |
112 | [m_class release]; | |
113 | } | |
114 | ||
115 | virtual void set(NSInvocation *invocation, NSInteger argumentNumber, JSContext *context, JSValueRef argument, JSValueRef* exception) override | |
116 | { | |
117 | JSGlobalContextRef contextRef = [context JSGlobalContextRef]; | |
118 | ||
119 | id object = tryUnwrapObjcObject(contextRef, argument); | |
120 | if (object && [object isKindOfClass:m_class]) { | |
121 | [invocation setArgument:&object atIndex:argumentNumber]; | |
122 | return; | |
123 | } | |
124 | ||
125 | if (JSValueIsNull(contextRef, argument) || JSValueIsUndefined(contextRef, argument)) { | |
126 | object = nil; | |
127 | [invocation setArgument:&object atIndex:argumentNumber]; | |
128 | return; | |
129 | } | |
130 | ||
81345200 | 131 | *exception = toRef(JSC::createTypeError(toJS(contextRef), ASCIILiteral("Argument does not match Objective-C Class"))); |
93a37866 A |
132 | } |
133 | ||
134 | Class m_class; | |
135 | }; | |
136 | ||
137 | class CallbackArgumentNSNumber : public CallbackArgument { | |
138 | virtual void set(NSInvocation *invocation, NSInteger argumentNumber, JSContext *context, JSValueRef argument, JSValueRef* exception) override | |
139 | { | |
140 | id value = valueToNumber([context JSGlobalContextRef], argument, exception); | |
141 | [invocation setArgument:&value atIndex:argumentNumber]; | |
142 | } | |
143 | }; | |
144 | ||
145 | class CallbackArgumentNSString : public CallbackArgument { | |
146 | virtual void set(NSInvocation *invocation, NSInteger argumentNumber, JSContext *context, JSValueRef argument, JSValueRef* exception) override | |
147 | { | |
148 | id value = valueToString([context JSGlobalContextRef], argument, exception); | |
149 | [invocation setArgument:&value atIndex:argumentNumber]; | |
150 | } | |
151 | }; | |
152 | ||
153 | class CallbackArgumentNSDate : public CallbackArgument { | |
154 | virtual void set(NSInvocation *invocation, NSInteger argumentNumber, JSContext *context, JSValueRef argument, JSValueRef* exception) override | |
155 | { | |
156 | id value = valueToDate([context JSGlobalContextRef], argument, exception); | |
157 | [invocation setArgument:&value atIndex:argumentNumber]; | |
158 | } | |
159 | }; | |
160 | ||
161 | class CallbackArgumentNSArray : public CallbackArgument { | |
162 | virtual void set(NSInvocation *invocation, NSInteger argumentNumber, JSContext *context, JSValueRef argument, JSValueRef* exception) override | |
163 | { | |
164 | id value = valueToArray([context JSGlobalContextRef], argument, exception); | |
165 | [invocation setArgument:&value atIndex:argumentNumber]; | |
166 | } | |
167 | }; | |
168 | ||
169 | class CallbackArgumentNSDictionary : public CallbackArgument { | |
170 | virtual void set(NSInvocation *invocation, NSInteger argumentNumber, JSContext *context, JSValueRef argument, JSValueRef* exception) override | |
171 | { | |
172 | id value = valueToDictionary([context JSGlobalContextRef], argument, exception); | |
173 | [invocation setArgument:&value atIndex:argumentNumber]; | |
174 | } | |
175 | }; | |
176 | ||
177 | class CallbackArgumentStruct : public CallbackArgument { | |
178 | public: | |
179 | CallbackArgumentStruct(NSInvocation *conversionInvocation, const char* encodedType) | |
180 | : m_conversionInvocation(conversionInvocation) | |
181 | , m_buffer(encodedType) | |
182 | { | |
183 | } | |
184 | ||
185 | private: | |
186 | virtual void set(NSInvocation *invocation, NSInteger argumentNumber, JSContext *context, JSValueRef argument, JSValueRef*) override | |
187 | { | |
188 | JSValue *value = [JSValue valueWithJSValueRef:argument inContext:context]; | |
189 | [m_conversionInvocation invokeWithTarget:value]; | |
190 | [m_conversionInvocation getReturnValue:m_buffer]; | |
191 | [invocation setArgument:m_buffer atIndex:argumentNumber]; | |
192 | } | |
193 | ||
194 | RetainPtr<NSInvocation> m_conversionInvocation; | |
195 | StructBuffer m_buffer; | |
196 | }; | |
197 | ||
198 | class ArgumentTypeDelegate { | |
199 | public: | |
200 | typedef CallbackArgument* ResultType; | |
201 | ||
202 | template<typename T> | |
203 | static ResultType typeInteger() | |
204 | { | |
205 | return new CallbackArgumentInteger<T>; | |
206 | } | |
207 | ||
208 | template<typename T> | |
209 | static ResultType typeDouble() | |
210 | { | |
211 | return new CallbackArgumentDouble<T>; | |
212 | } | |
213 | ||
214 | static ResultType typeBool() | |
215 | { | |
216 | return new CallbackArgumentBoolean; | |
217 | } | |
218 | ||
219 | static ResultType typeVoid() | |
220 | { | |
221 | RELEASE_ASSERT_NOT_REACHED(); | |
222 | return 0; | |
223 | } | |
224 | ||
225 | static ResultType typeId() | |
226 | { | |
227 | return new CallbackArgumentId; | |
228 | } | |
229 | ||
230 | static ResultType typeOfClass(const char* begin, const char* end) | |
231 | { | |
232 | StringRange copy(begin, end); | |
233 | Class cls = objc_getClass(copy); | |
234 | if (!cls) | |
235 | return 0; | |
236 | ||
237 | if (cls == [JSValue class]) | |
238 | return new CallbackArgumentJSValue; | |
239 | if (cls == [NSString class]) | |
240 | return new CallbackArgumentNSString; | |
241 | if (cls == [NSNumber class]) | |
242 | return new CallbackArgumentNSNumber; | |
243 | if (cls == [NSDate class]) | |
244 | return new CallbackArgumentNSDate; | |
245 | if (cls == [NSArray class]) | |
246 | return new CallbackArgumentNSArray; | |
247 | if (cls == [NSDictionary class]) | |
248 | return new CallbackArgumentNSDictionary; | |
249 | ||
250 | return new CallbackArgumentOfClass(cls); | |
251 | } | |
252 | ||
253 | static ResultType typeBlock(const char*, const char*) | |
254 | { | |
255 | return nil; | |
256 | } | |
257 | ||
258 | static ResultType typeStruct(const char* begin, const char* end) | |
259 | { | |
260 | StringRange copy(begin, end); | |
261 | if (NSInvocation *invocation = valueToTypeInvocationFor(copy)) | |
262 | return new CallbackArgumentStruct(invocation, copy); | |
263 | return 0; | |
264 | } | |
265 | }; | |
266 | ||
267 | class CallbackResult { | |
268 | public: | |
269 | virtual ~CallbackResult() | |
270 | { | |
271 | } | |
272 | ||
273 | virtual JSValueRef get(NSInvocation *, JSContext *, JSValueRef*) = 0; | |
274 | }; | |
275 | ||
276 | class CallbackResultVoid : public CallbackResult { | |
277 | virtual JSValueRef get(NSInvocation *, JSContext *context, JSValueRef*) override | |
278 | { | |
279 | return JSValueMakeUndefined([context JSGlobalContextRef]); | |
280 | } | |
281 | }; | |
282 | ||
283 | class CallbackResultId : public CallbackResult { | |
284 | virtual JSValueRef get(NSInvocation *invocation, JSContext *context, JSValueRef*) override | |
285 | { | |
286 | id value; | |
287 | [invocation getReturnValue:&value]; | |
288 | return objectToValue(context, value); | |
289 | } | |
290 | }; | |
291 | ||
292 | template<typename T> | |
293 | class CallbackResultNumeric : public CallbackResult { | |
294 | virtual JSValueRef get(NSInvocation *invocation, JSContext *context, JSValueRef*) override | |
295 | { | |
296 | T value; | |
297 | [invocation getReturnValue:&value]; | |
298 | return JSValueMakeNumber([context JSGlobalContextRef], value); | |
299 | } | |
300 | }; | |
301 | ||
302 | class CallbackResultBoolean : public CallbackResult { | |
303 | virtual JSValueRef get(NSInvocation *invocation, JSContext *context, JSValueRef*) override | |
304 | { | |
305 | bool value; | |
306 | [invocation getReturnValue:&value]; | |
307 | return JSValueMakeBoolean([context JSGlobalContextRef], value); | |
308 | } | |
309 | }; | |
310 | ||
311 | class CallbackResultStruct : public CallbackResult { | |
312 | public: | |
313 | CallbackResultStruct(NSInvocation *conversionInvocation, const char* encodedType) | |
314 | : m_conversionInvocation(conversionInvocation) | |
315 | , m_buffer(encodedType) | |
316 | { | |
317 | } | |
318 | ||
319 | private: | |
320 | virtual JSValueRef get(NSInvocation *invocation, JSContext *context, JSValueRef*) override | |
321 | { | |
322 | [invocation getReturnValue:m_buffer]; | |
323 | ||
324 | [m_conversionInvocation setArgument:m_buffer atIndex:2]; | |
325 | [m_conversionInvocation setArgument:&context atIndex:3]; | |
326 | [m_conversionInvocation invokeWithTarget:[JSValue class]]; | |
327 | ||
328 | JSValue *value; | |
329 | [m_conversionInvocation getReturnValue:&value]; | |
330 | return valueInternalValue(value); | |
331 | } | |
332 | ||
333 | RetainPtr<NSInvocation> m_conversionInvocation; | |
334 | StructBuffer m_buffer; | |
335 | }; | |
336 | ||
337 | class ResultTypeDelegate { | |
338 | public: | |
339 | typedef CallbackResult* ResultType; | |
340 | ||
341 | template<typename T> | |
342 | static ResultType typeInteger() | |
343 | { | |
344 | return new CallbackResultNumeric<T>; | |
345 | } | |
346 | ||
347 | template<typename T> | |
348 | static ResultType typeDouble() | |
349 | { | |
350 | return new CallbackResultNumeric<T>; | |
351 | } | |
352 | ||
353 | static ResultType typeBool() | |
354 | { | |
355 | return new CallbackResultBoolean; | |
356 | } | |
357 | ||
358 | static ResultType typeVoid() | |
359 | { | |
360 | return new CallbackResultVoid; | |
361 | } | |
362 | ||
363 | static ResultType typeId() | |
364 | { | |
365 | return new CallbackResultId(); | |
366 | } | |
367 | ||
368 | static ResultType typeOfClass(const char*, const char*) | |
369 | { | |
370 | return new CallbackResultId(); | |
371 | } | |
372 | ||
373 | static ResultType typeBlock(const char*, const char*) | |
374 | { | |
375 | return new CallbackResultId(); | |
376 | } | |
377 | ||
378 | static ResultType typeStruct(const char* begin, const char* end) | |
379 | { | |
380 | StringRange copy(begin, end); | |
381 | if (NSInvocation *invocation = typeToValueInvocationFor(copy)) | |
382 | return new CallbackResultStruct(invocation, copy); | |
383 | return 0; | |
384 | } | |
385 | }; | |
386 | ||
387 | enum CallbackType { | |
12899fa2 | 388 | CallbackInitMethod, |
93a37866 A |
389 | CallbackInstanceMethod, |
390 | CallbackClassMethod, | |
391 | CallbackBlock | |
392 | }; | |
393 | ||
394 | namespace JSC { | |
395 | ||
396 | class ObjCCallbackFunctionImpl { | |
397 | public: | |
81345200 A |
398 | ObjCCallbackFunctionImpl(NSInvocation *invocation, CallbackType type, Class instanceClass, PassOwnPtr<CallbackArgument> arguments, PassOwnPtr<CallbackResult> result) |
399 | : m_type(type) | |
93a37866 A |
400 | , m_instanceClass([instanceClass retain]) |
401 | , m_invocation(invocation) | |
402 | , m_arguments(arguments) | |
403 | , m_result(result) | |
404 | { | |
12899fa2 | 405 | ASSERT((type != CallbackInstanceMethod && type != CallbackInitMethod) || instanceClass); |
93a37866 A |
406 | } |
407 | ||
81345200 | 408 | void destroy(Heap& heap) |
93a37866 | 409 | { |
81345200 | 410 | // We need to explicitly release the target since we didn't call |
12899fa2 A |
411 | // -retainArguments on m_invocation (and we don't want to do so). |
412 | if (m_type == CallbackBlock || m_type == CallbackClassMethod) | |
81345200 | 413 | heap.releaseSoon(adoptNS([m_invocation.get() target])); |
93a37866 A |
414 | [m_instanceClass release]; |
415 | } | |
416 | ||
417 | JSValueRef call(JSContext *context, JSObjectRef thisObject, size_t argumentCount, const JSValueRef arguments[], JSValueRef* exception); | |
418 | ||
93a37866 A |
419 | id wrappedBlock() |
420 | { | |
421 | return m_type == CallbackBlock ? [m_invocation target] : nil; | |
422 | } | |
423 | ||
12899fa2 A |
424 | id wrappedConstructor() |
425 | { | |
426 | switch (m_type) { | |
427 | case CallbackBlock: | |
428 | return [m_invocation target]; | |
429 | case CallbackInitMethod: | |
430 | return m_instanceClass; | |
431 | default: | |
432 | return nil; | |
433 | } | |
434 | } | |
435 | ||
436 | bool isConstructible() | |
437 | { | |
438 | return !!wrappedBlock() || m_type == CallbackInitMethod; | |
439 | } | |
440 | ||
441 | String name(); | |
442 | ||
93a37866 | 443 | private: |
93a37866 A |
444 | CallbackType m_type; |
445 | Class m_instanceClass; | |
446 | RetainPtr<NSInvocation> m_invocation; | |
447 | OwnPtr<CallbackArgument> m_arguments; | |
448 | OwnPtr<CallbackResult> m_result; | |
449 | }; | |
450 | ||
451 | static JSValueRef objCCallbackFunctionCallAsFunction(JSContextRef callerContext, JSObjectRef function, JSObjectRef thisObject, size_t argumentCount, const JSValueRef arguments[], JSValueRef* exception) | |
452 | { | |
453 | // Retake the API lock - we need this for a few reasons: | |
454 | // (1) We don't want to support the C-API's confusing drops-locks-once policy - should only drop locks if we can do so recursively. | |
455 | // (2) We're calling some JSC internals that require us to be on the 'inside' - e.g. createTypeError. | |
456 | // (3) We need to be locked (per context would be fine) against conflicting usage of the ObjCCallbackFunction's NSInvocation. | |
81345200 | 457 | JSC::JSLockHolder locker(toJS(callerContext)); |
93a37866 A |
458 | |
459 | ObjCCallbackFunction* callback = static_cast<ObjCCallbackFunction*>(toJS(function)); | |
460 | ObjCCallbackFunctionImpl* impl = callback->impl(); | |
81345200 | 461 | JSContext *context = [JSContext contextWithJSGlobalContextRef:toGlobalRef(callback->globalObject()->globalExec())]; |
93a37866 A |
462 | |
463 | CallbackData callbackData; | |
464 | JSValueRef result; | |
465 | @autoreleasepool { | |
81345200 | 466 | [context beginCallbackWithData:&callbackData calleeValue:function thisValue:thisObject argumentCount:argumentCount arguments:arguments]; |
93a37866 A |
467 | result = impl->call(context, thisObject, argumentCount, arguments, exception); |
468 | if (context.exception) | |
469 | *exception = valueInternalValue(context.exception); | |
470 | [context endCallbackWithData:&callbackData]; | |
471 | } | |
472 | return result; | |
473 | } | |
474 | ||
12899fa2 A |
475 | static JSObjectRef objCCallbackFunctionCallAsConstructor(JSContextRef callerContext, JSObjectRef constructor, size_t argumentCount, const JSValueRef arguments[], JSValueRef* exception) |
476 | { | |
81345200 | 477 | JSC::JSLockHolder locker(toJS(callerContext)); |
12899fa2 A |
478 | |
479 | ObjCCallbackFunction* callback = static_cast<ObjCCallbackFunction*>(toJS(constructor)); | |
480 | ObjCCallbackFunctionImpl* impl = callback->impl(); | |
481 | JSContext *context = [JSContext contextWithJSGlobalContextRef:toGlobalRef(toJS(callerContext)->lexicalGlobalObject()->globalExec())]; | |
482 | ||
483 | CallbackData callbackData; | |
484 | JSValueRef result; | |
485 | @autoreleasepool { | |
81345200 | 486 | [context beginCallbackWithData:&callbackData calleeValue:constructor thisValue:nil argumentCount:argumentCount arguments:arguments]; |
12899fa2 A |
487 | result = impl->call(context, NULL, argumentCount, arguments, exception); |
488 | if (context.exception) | |
489 | *exception = valueInternalValue(context.exception); | |
490 | [context endCallbackWithData:&callbackData]; | |
491 | } | |
492 | ||
493 | JSGlobalContextRef contextRef = [context JSGlobalContextRef]; | |
494 | if (*exception) | |
495 | return 0; | |
496 | ||
497 | if (!JSValueIsObject(contextRef, result)) { | |
81345200 | 498 | *exception = toRef(JSC::createTypeError(toJS(contextRef), ASCIILiteral("Objective-C blocks called as constructors must return an object."))); |
12899fa2 A |
499 | return 0; |
500 | } | |
501 | return (JSObjectRef)result; | |
502 | } | |
503 | ||
93a37866 A |
504 | const JSC::ClassInfo ObjCCallbackFunction::s_info = { "CallbackFunction", &Base::s_info, 0, 0, CREATE_METHOD_TABLE(ObjCCallbackFunction) }; |
505 | ||
81345200 A |
506 | ObjCCallbackFunction::ObjCCallbackFunction(JSC::VM& vm, JSC::JSGlobalObject* globalObject, JSObjectCallAsFunctionCallback functionCallback, JSObjectCallAsConstructorCallback constructCallback, PassOwnPtr<ObjCCallbackFunctionImpl> impl) |
507 | : Base(vm, globalObject->objcCallbackFunctionStructure()) | |
12899fa2 A |
508 | , m_functionCallback(functionCallback) |
509 | , m_constructCallback(constructCallback) | |
93a37866 A |
510 | , m_impl(impl) |
511 | { | |
512 | } | |
513 | ||
12899fa2 | 514 | ObjCCallbackFunction* ObjCCallbackFunction::create(JSC::VM& vm, JSC::JSGlobalObject* globalObject, const String& name, PassOwnPtr<ObjCCallbackFunctionImpl> impl) |
93a37866 | 515 | { |
12899fa2 A |
516 | ObjCCallbackFunction* function = new (NotNull, allocateCell<ObjCCallbackFunction>(vm.heap)) ObjCCallbackFunction(vm, globalObject, objCCallbackFunctionCallAsFunction, objCCallbackFunctionCallAsConstructor, impl); |
517 | function->finishCreation(vm, name); | |
93a37866 A |
518 | return function; |
519 | } | |
520 | ||
521 | void ObjCCallbackFunction::destroy(JSCell* cell) | |
522 | { | |
81345200 A |
523 | ObjCCallbackFunction& function = *jsCast<ObjCCallbackFunction*>(cell); |
524 | function.impl()->destroy(*Heap::heap(cell)); | |
525 | function.~ObjCCallbackFunction(); | |
93a37866 A |
526 | } |
527 | ||
12899fa2 A |
528 | |
529 | CallType ObjCCallbackFunction::getCallData(JSCell*, CallData& callData) | |
530 | { | |
531 | callData.native.function = APICallbackFunction::call<ObjCCallbackFunction>; | |
532 | return CallTypeHost; | |
533 | } | |
534 | ||
535 | ConstructType ObjCCallbackFunction::getConstructData(JSCell* cell, ConstructData& constructData) | |
536 | { | |
537 | ObjCCallbackFunction* callback = jsCast<ObjCCallbackFunction*>(cell); | |
538 | if (!callback->impl()->isConstructible()) | |
539 | return Base::getConstructData(cell, constructData); | |
540 | constructData.native.function = APICallbackFunction::construct<ObjCCallbackFunction>; | |
541 | return ConstructTypeHost; | |
542 | } | |
543 | ||
544 | String ObjCCallbackFunctionImpl::name() | |
545 | { | |
546 | if (m_type == CallbackInitMethod) | |
547 | return class_getName(m_instanceClass); | |
548 | // FIXME: Maybe we could support having the selector as the name of the non-init | |
549 | // functions to make it a bit more user-friendly from the JS side? | |
550 | return ""; | |
551 | } | |
552 | ||
93a37866 A |
553 | JSValueRef ObjCCallbackFunctionImpl::call(JSContext *context, JSObjectRef thisObject, size_t argumentCount, const JSValueRef arguments[], JSValueRef* exception) |
554 | { | |
555 | JSGlobalContextRef contextRef = [context JSGlobalContextRef]; | |
556 | ||
12899fa2 | 557 | id target; |
93a37866 A |
558 | size_t firstArgument; |
559 | switch (m_type) { | |
12899fa2 A |
560 | case CallbackInitMethod: { |
561 | RELEASE_ASSERT(!thisObject); | |
562 | target = [m_instanceClass alloc]; | |
563 | if (!target || ![target isKindOfClass:m_instanceClass]) { | |
81345200 | 564 | *exception = toRef(JSC::createTypeError(toJS(contextRef), ASCIILiteral("self type check failed for Objective-C instance method"))); |
12899fa2 A |
565 | return JSValueMakeUndefined(contextRef); |
566 | } | |
567 | [m_invocation setTarget:target]; | |
568 | firstArgument = 2; | |
569 | break; | |
570 | } | |
93a37866 | 571 | case CallbackInstanceMethod: { |
12899fa2 | 572 | target = tryUnwrapObjcObject(contextRef, thisObject); |
93a37866 | 573 | if (!target || ![target isKindOfClass:m_instanceClass]) { |
81345200 | 574 | *exception = toRef(JSC::createTypeError(toJS(contextRef), ASCIILiteral("self type check failed for Objective-C instance method"))); |
93a37866 A |
575 | return JSValueMakeUndefined(contextRef); |
576 | } | |
577 | [m_invocation setTarget:target]; | |
12899fa2 A |
578 | firstArgument = 2; |
579 | break; | |
93a37866 | 580 | } |
93a37866 A |
581 | case CallbackClassMethod: |
582 | firstArgument = 2; | |
583 | break; | |
584 | case CallbackBlock: | |
585 | firstArgument = 1; | |
586 | } | |
587 | ||
588 | size_t argumentNumber = 0; | |
589 | for (CallbackArgument* argument = m_arguments.get(); argument; argument = argument->m_next.get()) { | |
590 | JSValueRef value = argumentNumber < argumentCount ? arguments[argumentNumber] : JSValueMakeUndefined(contextRef); | |
591 | argument->set(m_invocation.get(), argumentNumber + firstArgument, context, value, exception); | |
592 | if (*exception) | |
593 | return JSValueMakeUndefined(contextRef); | |
594 | ++argumentNumber; | |
595 | } | |
596 | ||
597 | [m_invocation invoke]; | |
598 | ||
12899fa2 A |
599 | JSValueRef result = m_result->get(m_invocation.get(), context, exception); |
600 | ||
601 | // Balance our call to -alloc with a call to -autorelease. We have to do this after calling -init | |
602 | // because init family methods are allowed to release the allocated object and return something | |
603 | // else in its place. | |
604 | if (m_type == CallbackInitMethod) { | |
605 | id objcResult = tryUnwrapObjcObject(contextRef, result); | |
606 | if (objcResult) | |
607 | [objcResult autorelease]; | |
608 | } | |
609 | ||
610 | return result; | |
93a37866 A |
611 | } |
612 | ||
613 | } // namespace JSC | |
614 | ||
615 | static bool blockSignatureContainsClass() | |
616 | { | |
617 | static bool containsClass = ^{ | |
618 | id block = ^(NSString *string){ return string; }; | |
619 | return _Block_has_signature(block) && strstr(_Block_signature(block), "NSString"); | |
620 | }(); | |
621 | return containsClass; | |
622 | } | |
623 | ||
624 | inline bool skipNumber(const char*& position) | |
625 | { | |
626 | if (!isASCIIDigit(*position)) | |
627 | return false; | |
628 | while (isASCIIDigit(*++position)) { } | |
629 | return true; | |
630 | } | |
631 | ||
632 | static JSObjectRef objCCallbackFunctionForInvocation(JSContext *context, NSInvocation *invocation, CallbackType type, Class instanceClass, const char* signatureWithObjcClasses) | |
633 | { | |
81345200 A |
634 | if (!signatureWithObjcClasses) |
635 | return nil; | |
636 | ||
93a37866 A |
637 | const char* position = signatureWithObjcClasses; |
638 | ||
639 | OwnPtr<CallbackResult> result = adoptPtr(parseObjCType<ResultTypeDelegate>(position)); | |
640 | if (!result || !skipNumber(position)) | |
641 | return nil; | |
642 | ||
643 | switch (type) { | |
12899fa2 | 644 | case CallbackInitMethod: |
93a37866 A |
645 | case CallbackInstanceMethod: |
646 | case CallbackClassMethod: | |
647 | // Methods are passed two implicit arguments - (id)self, and the selector. | |
648 | if ('@' != *position++ || !skipNumber(position) || ':' != *position++ || !skipNumber(position)) | |
649 | return nil; | |
650 | break; | |
651 | case CallbackBlock: | |
652 | // Blocks are passed one implicit argument - the block, of type "@?". | |
653 | if (('@' != *position++) || ('?' != *position++) || !skipNumber(position)) | |
654 | return nil; | |
655 | // Only allow arguments of type 'id' if the block signature contains the NS type information. | |
656 | if ((!blockSignatureContainsClass() && strchr(position, '@'))) | |
657 | return nil; | |
658 | break; | |
659 | } | |
660 | ||
661 | OwnPtr<CallbackArgument> arguments = 0; | |
662 | OwnPtr<CallbackArgument>* nextArgument = &arguments; | |
663 | unsigned argumentCount = 0; | |
664 | while (*position) { | |
665 | OwnPtr<CallbackArgument> argument = adoptPtr(parseObjCType<ArgumentTypeDelegate>(position)); | |
666 | if (!argument || !skipNumber(position)) | |
667 | return nil; | |
668 | ||
669 | *nextArgument = argument.release(); | |
670 | nextArgument = &(*nextArgument)->m_next; | |
671 | ++argumentCount; | |
672 | } | |
673 | ||
674 | JSC::ExecState* exec = toJS([context JSGlobalContextRef]); | |
81345200 A |
675 | JSC::JSLockHolder locker(exec); |
676 | OwnPtr<JSC::ObjCCallbackFunctionImpl> impl = adoptPtr(new JSC::ObjCCallbackFunctionImpl(invocation, type, instanceClass, arguments.release(), result.release())); | |
12899fa2 A |
677 | return toRef(JSC::ObjCCallbackFunction::create(exec->vm(), exec->lexicalGlobalObject(), impl->name(), impl.release())); |
678 | } | |
679 | ||
680 | JSObjectRef objCCallbackFunctionForInit(JSContext *context, Class cls, Protocol *protocol, SEL sel, const char* types) | |
681 | { | |
682 | NSInvocation *invocation = [NSInvocation invocationWithMethodSignature:[NSMethodSignature signatureWithObjCTypes:types]]; | |
683 | [invocation setSelector:sel]; | |
684 | return objCCallbackFunctionForInvocation(context, invocation, CallbackInitMethod, cls, _protocol_getMethodTypeEncoding(protocol, sel, YES, YES)); | |
93a37866 A |
685 | } |
686 | ||
687 | JSObjectRef objCCallbackFunctionForMethod(JSContext *context, Class cls, Protocol *protocol, BOOL isInstanceMethod, SEL sel, const char* types) | |
688 | { | |
689 | NSInvocation *invocation = [NSInvocation invocationWithMethodSignature:[NSMethodSignature signatureWithObjCTypes:types]]; | |
690 | [invocation setSelector:sel]; | |
12899fa2 A |
691 | // We need to retain the target Class because m_invocation doesn't retain it |
692 | // by default (and we don't want it to). | |
93a37866 | 693 | if (!isInstanceMethod) |
12899fa2 | 694 | [invocation setTarget:[cls retain]]; |
93a37866 A |
695 | return objCCallbackFunctionForInvocation(context, invocation, isInstanceMethod ? CallbackInstanceMethod : CallbackClassMethod, isInstanceMethod ? cls : nil, _protocol_getMethodTypeEncoding(protocol, sel, YES, isInstanceMethod)); |
696 | } | |
697 | ||
698 | JSObjectRef objCCallbackFunctionForBlock(JSContext *context, id target) | |
699 | { | |
700 | if (!_Block_has_signature(target)) | |
701 | return 0; | |
702 | const char* signature = _Block_signature(target); | |
703 | NSInvocation *invocation = [NSInvocation invocationWithMethodSignature:[NSMethodSignature signatureWithObjCTypes:signature]]; | |
12899fa2 A |
704 | |
705 | // We don't want to use -retainArguments because that leaks memory. Arguments | |
706 | // would be retained indefinitely between invocations of the callback. | |
707 | // Additionally, we copy the target because we want the block to stick around | |
708 | // until the ObjCCallbackFunctionImpl is destroyed. | |
93a37866 | 709 | [invocation setTarget:[target copy]]; |
12899fa2 | 710 | |
93a37866 A |
711 | return objCCallbackFunctionForInvocation(context, invocation, CallbackBlock, nil, signature); |
712 | } | |
713 | ||
12899fa2 | 714 | id tryUnwrapConstructor(JSObjectRef object) |
93a37866 | 715 | { |
81345200 | 716 | if (!toJS(object)->inherits(JSC::ObjCCallbackFunction::info())) |
93a37866 | 717 | return nil; |
12899fa2 A |
718 | JSC::ObjCCallbackFunctionImpl* impl = static_cast<JSC::ObjCCallbackFunction*>(toJS(object))->impl(); |
719 | if (!impl->isConstructible()) | |
720 | return nil; | |
721 | return impl->wrappedConstructor(); | |
93a37866 A |
722 | } |
723 | ||
724 | #endif |