/* Copyright (C) 2003-2015 LiveCode Ltd. This file is part of LiveCode. LiveCode is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License v3 as published by the Free Software Foundation. LiveCode is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with LiveCode. If not see . */ #include #include #include #include "ffi.h" #include "libscript/script.h" #include "script-private.h" #include "script-bytecode.hpp" #if defined(_MACOSX) || defined(TARGET_SUBPLATFORM_IPHONE) #include #endif //////////////////////////////////////////////////////////////////////////////// // This is the module of themost recent LCB stack frame on the current thread's // stack. It is set before and after a foreign handler call so that the native // code can get some element of context. static MCScriptModuleRef s_current_module = nil; static MCScriptWidgetEnterCallback s_widget_enter_callback = nullptr; static MCScriptWidgetLeaveCallback s_widget_leave_callback = nullptr; void MCScriptSetWidgetBarrierCallbacks(MCScriptWidgetEnterCallback p_entry, MCScriptWidgetLeaveCallback p_leave) { s_widget_enter_callback = p_entry; s_widget_leave_callback = p_leave; } //////////////////////////////////////////////////////////////////////////////// bool MCScriptCreateInstanceOfModule(MCScriptModuleRef p_module, MCScriptInstanceRef& r_instance) { bool t_success; t_success = true; MCScriptInstanceRef t_instance; t_instance = nil; // If the module is not usable, then we cannot create an instance. if (t_success) if (!p_module -> is_usable) return false; // If this is a module which shares a single instance, then return that if we have // one. if (p_module -> module_kind != kMCScriptModuleKindWidget && p_module -> shared_instance != nil) { r_instance = MCScriptRetainInstance(p_module -> shared_instance); return true; } // Attempt to create a script object. if (t_success) t_success = MCScriptCreateObject(kMCScriptObjectKindInstance, t_instance); // Now associate the script object with the module (so the 'slots' field make sense). if (t_success) { t_instance -> module = MCScriptRetainModule(p_module); } // Now allocate the slots field. if (t_success) t_success = MCMemoryNewArray(p_module -> slot_count, t_instance -> slots); if (t_success) { /* Loop over all of the variables of the module, initialising * the corresponding slot to a default value of the * appropriate type, if available. */ for (uindex_t i = 0; i < p_module->definition_count; ++i) { if (p_module->definitions[i]->kind != kMCScriptDefinitionKindVariable) continue; /* Not a variable */ MCScriptVariableDefinition *t_variable = static_cast(p_module->definitions[i]); uindex_t t_slot_index = t_variable->slot_index; /* COMPUTE CHECK */ __MCScriptAssert__(t_slot_index < p_module->slot_count, "computed variable slot out of range"); MCTypeInfoRef t_type = p_module->types[t_variable->type]->typeinfo; if (nil == t_type) continue; /* Variable is untyped */ MCValueRef t_default = MCTypeInfoGetDefault(t_type); if (nil == t_default) continue; /* Type has no default value */ t_instance->slots[t_slot_index] = MCValueRetain(t_default); } // If this is a module which shares its instance, then add a link to it. // (Note this is weak reference - we don't retain, otherwise we would have // a cycle!). if (p_module -> module_kind != kMCScriptModuleKindWidget) p_module -> shared_instance = t_instance; r_instance = t_instance; } else MCScriptDestroyObject(t_instance); return t_success; } void MCScriptDestroyInstance(MCScriptInstanceRef self) { __MCScriptValidateObjectAndKind__(self, kMCScriptObjectKindInstance); // If the object had slots initialized, then free them. if (self -> slots != nil) { for(uindex_t i = 0; i < self -> module -> slot_count; i++) MCValueRelease(self -> slots[i]); MCMemoryDeleteArray(self -> slots); } // If the instance has handler-refs, then free them. for(uindex_t i = 0; i < self -> handler_count; i++) MCValueRelease(self -> handlers[i] . value); MCMemoryDeleteArray(self -> handlers); // If the instance has a module, and this is the shared instance then set // the shared instance field to nil. if (self -> module != nil && self -> module -> shared_instance == self) self -> module -> shared_instance = nil; // If the instance was associated with its module, then release it. if (self -> module != nil) MCScriptReleaseModule(self -> module); // The rest of the deallocation is handled by MCScriptDestroyObject. } //////////////////////////////////////////////////////////////////////////////// MCScriptInstanceRef MCScriptRetainInstance(MCScriptInstanceRef self) { __MCScriptValidateObjectAndKind__(self, kMCScriptObjectKindInstance); return (MCScriptInstanceRef)MCScriptRetainObject(self); } void MCScriptReleaseInstance(MCScriptInstanceRef self) { if (nil == self) return; __MCScriptValidateObjectAndKind__(self, kMCScriptObjectKindInstance); MCScriptReleaseObject(self); } //////////////////////////////////////////////////////////////////////////////// void MCScriptSetInstanceHostPtr(MCScriptInstanceRef self, void *p_host_ptr) { self->host_ptr = p_host_ptr; } void * MCScriptGetInstanceHostPtr(MCScriptInstanceRef self) { return self->host_ptr; } //////////////////////////////////////////////////////////////////////////////// MCScriptModuleRef MCScriptGetModuleOfInstance(MCScriptInstanceRef self) { return self -> module; } //////////////////////////////////////////////////////////////////////////////// static bool __MCHandlerTypeInfoConformsToPropertyGetter(MCTypeInfoRef p_typeinfo) { return MCHandlerTypeInfoGetParameterCount(p_typeinfo) == 0 && MCHandlerTypeInfoGetReturnType(p_typeinfo) != kMCNullTypeInfo; } static bool __MCHandlerTypeInfoConformsToPropertySetter(MCTypeInfoRef p_typeinfo) { return MCHandlerTypeInfoGetParameterCount(p_typeinfo) == 1 && MCHandlerTypeInfoGetParameterMode(p_typeinfo, 0) == kMCHandlerTypeFieldModeIn; } static bool __MCScriptCallHandlerDefinitionInInstance(MCScriptInstanceRef self, MCScriptHandlerDefinition *p_handler_def, MCValueRef *p_arguments, uindex_t p_argument_count, MCValueRef* r_value) { if (!self->module->licensed) return MCErrorThrowGeneric(MCSTR("extension not licensed")); void *t_cookie = nullptr; if (self->module->module_kind == kMCScriptModuleKindWidget) { if (s_widget_enter_callback != nullptr) { t_cookie = s_widget_enter_callback(self, self->host_ptr); } } MCScriptModuleRef t_previous = MCScriptSetCurrentModule(self->module); MCScriptExecuteContext t_execute_ctxt; t_execute_ctxt.Enter(self, p_handler_def, MCMakeSpan(p_arguments, p_argument_count), r_value); while(t_execute_ctxt.Step()) { MCScriptBytecodeExecute(t_execute_ctxt); } if (self->module->module_kind == kMCScriptModuleKindWidget) { if (s_widget_leave_callback != nullptr) { s_widget_leave_callback(self, self->host_ptr, t_cookie); } } MCScriptSetCurrentModule(t_previous); if (!t_execute_ctxt.Leave()) { return false; } return true; } static bool __MCScriptGetVariablePropertyInInstance(MCScriptInstanceRef self, MCScriptVariableDefinition *p_variable_def, MCValueRef& r_value) { MCValueRef t_value = self->slots[p_variable_def->slot_index]; r_value = t_value != nil ? MCValueRetain(t_value) : t_value; return true; } static bool __MCScriptGetHandlerPropertyInInstance(MCScriptInstanceRef self, MCScriptHandlerDefinition *p_handler_def, MCValueRef& r_value) { /* LOAD CHECK */ __MCScriptAssert__(__MCHandlerTypeInfoConformsToPropertyGetter(self->module->types[p_handler_def->type]->typeinfo), "incorrect signature for property getter"); return __MCScriptCallHandlerDefinitionInInstance(self, p_handler_def, nil, 0, &r_value); } static bool __MCScriptSetVariablePropertyInInstance(MCScriptInstanceRef self, MCScriptVariableDefinition *p_variable_def, MCValueRef p_value) { // BRIDGE: This should bridge (convert) the type if necessary. MCValueAssign(self->slots[p_variable_def->slot_index], p_value); return true; } static bool __MCScriptSetHandlerPropertyInInstance(MCScriptInstanceRef self, MCScriptHandlerDefinition *p_handler_def, MCValueRef p_value) { /* LOAD CHECK */ __MCScriptAssert__(__MCHandlerTypeInfoConformsToPropertySetter(self->module->types[p_handler_def->type]->typeinfo), "incorrect signature for property setter"); return __MCScriptCallHandlerDefinitionInInstance(self, p_handler_def, &p_value, 1, nil); } bool MCScriptGetPropertyInInstance(MCScriptInstanceRef self, MCNameRef p_property, MCValueRef& r_value) { __MCScriptValidateObjectAndKind__(self, kMCScriptObjectKindInstance); // Lookup the definition (throws if not found). MCScriptPropertyDefinition *t_property_def; if (!MCScriptLookupPropertyDefinitionInModule(self->module, p_property, t_property_def)) { return MCScriptThrowPropertyNotFoundError(self, p_property); } MCScriptDefinition *t_getter; t_getter = t_property_def->getter != 0 ? self->module->definitions[t_property_def->getter - 1] : nil; /* LOAD CHECK */ __MCScriptAssert__(t_getter != nil, "property has no getter"); switch(t_getter->kind) { case kMCScriptDefinitionKindVariable: { MCValueRef t_value; if (!__MCScriptGetVariablePropertyInInstance(self, static_cast(t_getter), t_value)) { return false; } if (t_value == nil) { return MCScriptThrowPropertyUsedBeforeAssignedError(self, t_property_def); } r_value = t_value; } break; case kMCScriptDefinitionKindHandler: { if (!__MCScriptGetHandlerPropertyInInstance(self, static_cast(t_getter), r_value)) { return false; } } break; default: /* LOAD CHECK */ __MCScriptUnreachable__("property getter is not a variable or handler"); break; } return true; } bool MCScriptSetPropertyInInstance(MCScriptInstanceRef self, MCNameRef p_property, MCValueRef p_value) { __MCScriptValidateObjectAndKind__(self, kMCScriptObjectKindInstance); // Lookup the definition (throws if not found). MCScriptPropertyDefinition *t_property_def; if (!MCScriptLookupPropertyDefinitionInModule(self->module, p_property, t_property_def)) { return MCScriptThrowPropertyNotFoundError(self, p_property); } MCScriptDefinition *t_setter; t_setter = t_property_def->setter != 0 ? self->module->definitions[t_property_def->setter - 1] : nil; // If there is no setter for the property then this is an error. if (t_setter == nil) return MCScriptThrowCannotSetReadOnlyPropertyError(self, p_property); /* LOAD CHECK */ __MCScriptAssert__(t_setter != nil, "property has no setter"); /* LOAD CHECK */ __MCScriptAssert__(t_setter->kind == kMCScriptDefinitionKindVariable || t_setter->kind == kMCScriptDefinitionKindHandler, "property setter is not a variable or handler"); MCTypeInfoRef t_property_type; switch(t_setter->kind) { case kMCScriptDefinitionKindVariable: { t_property_type = self->module->types[static_cast(t_setter)->type]->typeinfo; } break; case kMCScriptDefinitionKindHandler: { MCTypeInfoRef t_setter_signature = self->module->types[static_cast(t_setter)->type]->typeinfo; /* LOAD CHECK */ __MCScriptAssert__(__MCHandlerTypeInfoConformsToPropertySetter(t_setter_signature), "incorrect signature for property setter"); t_property_type = MCHandlerTypeInfoGetParameterType(t_setter_signature, 0); } break; default: { __MCScriptUnreachable__("inconsistency with definition kind in property setting"); return false; } break; } // Check the type passed in conforms to the required type of the // property here so that it generates a more useful error message. if (!MCTypeInfoConforms(MCValueGetTypeInfo(p_value), t_property_type)) { return MCScriptThrowInvalidValueForPropertyError(self, t_property_def, t_property_type, p_value); } switch(t_setter->kind) { case kMCScriptDefinitionKindVariable: { if (!__MCScriptSetVariablePropertyInInstance(self, static_cast(t_setter), p_value)) { return false; } } break; case kMCScriptDefinitionKindHandler: { if (!__MCScriptSetHandlerPropertyInInstance(self, static_cast(t_setter), p_value)) { return false; } } break; default: { __MCScriptUnreachable__("inconsistency with definition kind in property setting"); return false; } break; } return true; } bool MCScriptCallHandlerInInstanceInternal(MCScriptInstanceRef self, MCScriptHandlerDefinition *p_handler_def, MCValueRef *p_arguments, uindex_t p_argument_count, MCValueRef& r_value) { return __MCScriptCallHandlerDefinitionInInstance(self, p_handler_def, p_arguments, p_argument_count, &r_value); } bool MCScriptCallHandlerInInstance(MCScriptInstanceRef self, MCNameRef p_handler, MCValueRef *p_arguments, uindex_t p_argument_count, MCValueRef& r_value) { __MCScriptValidateObjectAndKind__(self, kMCScriptObjectKindInstance); MCScriptHandlerDefinition *t_handler_def; if (!MCScriptLookupHandlerDefinitionInModule(self->module, p_handler, t_handler_def)) { return MCScriptThrowHandlerNotFoundError(self, p_handler); } return MCScriptCallHandlerInInstanceInternal(self, t_handler_def, p_arguments, p_argument_count, r_value); } bool MCScriptCallHandlerInInstanceIfFound(MCScriptInstanceRef self, MCNameRef p_handler, MCValueRef *p_arguments, uindex_t p_argument_count, MCValueRef& r_value) { __MCScriptValidateObjectAndKind__(self, kMCScriptObjectKindInstance); MCScriptHandlerDefinition *t_handler_def; if (!MCScriptLookupHandlerDefinitionInModule(self->module, p_handler, t_handler_def)) { r_value = MCValueRetain(kMCNull); return true; } return MCScriptCallHandlerInInstanceInternal(self, t_handler_def, p_arguments, p_argument_count, r_value); } //////////////////////////////////////////////////////////////////////////////// // This method resolves the binding string in the foreign function. The format is: // [lang:][library>][class.]function[!calling] // // lang - one of c, cpp, objc or java. If not present, it is taken to be c. // library - the library to load the symbol from. If not present, it is taken to be the // main executable. // class - only valid for cpp, objc, or java. If not present, it is taken to be no class. // function - the name of the function or method. // calling - the calling convention, if not present it is taken to be the platform default. // // The calling conventions are: // default (sysv on 32-bit unix intel, unix64 on 64-bit unix intel, win64 on 64-bit windows, cdelc on 32-bit windows) // stdcall (windows 32-bit) // thiscall (windows 32-bit) // fastcall (windows 32-bit) // cdecl (windows 32-bit) // pascal (windows 32-bit) // register (windows 32-bit) // If a windows calling convention is specified, it is taken to be 'default' on // other platforms. // // Split x_string at p_char, returning the portion up to the char, or the end // of the string in r_field. The source string x_string is replaced with the // remainder of the string. // i.e. Split("foo.bar", '.'): // x_string = "bar", r_field = "foo" // static bool __MCScriptSplitForeignBindingString(MCStringRef& x_string, codepoint_t p_char, MCStringRef& r_field) { MCStringRef t_head, t_tail; if (!MCStringDivideAtChar(x_string, p_char, kMCStringOptionCompareExact, t_head, t_tail)) { return false; } MCValueRelease(x_string); if (!MCStringIsEmpty(t_tail)) { r_field = t_head; x_string = t_tail; } else { r_field = t_tail; x_string = t_head; } return true; } static bool __split_function_signature(MCStringRef p_string, MCStringRef& r_function, MCStringRef& r_arguments, MCStringRef& r_return) { MCAutoStringRef t_head, t_args, t_return; uindex_t t_open_bracket_offset, t_close_bracket_offset; if (!MCStringFirstIndexOfChar(p_string, '(', 0, kMCStringOptionCompareExact, t_open_bracket_offset) || !MCStringFirstIndexOfChar(p_string, ')', 0, kMCStringOptionCompareExact, t_close_bracket_offset)) { r_function = MCValueRetain(p_string); r_arguments = MCValueRetain(kMCEmptyString); r_return = MCValueRetain(kMCEmptyString); return true; } if (!MCStringCopySubstring(p_string, MCRangeMakeMinMax(t_open_bracket_offset, t_close_bracket_offset + 1), &t_args)) return false; if (!MCStringCopySubstring(p_string, MCRangeMake(t_close_bracket_offset + 1, UINDEX_MAX), &t_return)) return false; if (!MCStringCopySubstring(p_string, MCRangeMake(0, t_open_bracket_offset), &t_head)) return false; r_arguments = MCValueRetain(*t_args); r_return = MCValueRetain(*t_return); r_function = MCValueRetain(*t_head); return true; } // Resolve the call type static MCJavaCallType __MCScriptGetJavaCallType(MCStringRef p_class, MCStringRef p_function, MCStringRef p_calling) { if (MCStringIsEqualToCString(p_function, "new", kMCStringOptionCompareExact)) return MCJavaCallTypeConstructor; if (MCStringIsEqualToCString(p_function, "interface", kMCStringOptionCompareExact)) return MCJavaCallTypeInterfaceProxy; bool t_is_static = MCStringIsEqualToCString(p_calling, "static", kMCStringOptionCompareCaseless); // If the 'class' is get or set, then the binding is for a field if (p_class != nullptr && MCStringIsEqualToCString(p_class, "get", kMCStringOptionCompareExact)) { if (t_is_static) return MCJavaCallTypeStaticGetter; return MCJavaCallTypeGetter; } if (p_class != nullptr && MCStringIsEqualToCString(p_class, "set", kMCStringOptionCompareExact)) { if (t_is_static) return MCJavaCallTypeStaticSetter; return MCJavaCallTypeSetter; } if (t_is_static) return MCJavaCallTypeStatic; if (MCStringIsEqualToCString(p_calling, "nonvirtual", kMCStringOptionCompareCaseless)) return MCJavaCallTypeNonVirtual; if (MCStringIsEmpty(p_calling) || MCStringIsEqualToCString(p_calling, "instance", kMCStringOptionCompareCaseless)) return MCJavaCallTypeInstance; return MCJavaCallTypeUnknown; } // Resolve the call type static bool __MCScriptGetThreadAffinity(MCStringRef p_thread, MCScriptThreadAffinity &r_thread) { if (MCStringIsEmpty(p_thread)) { r_thread = kMCScriptThreadAffinityDefault; return true; } if (MCStringIsEqualToCString(p_thread, "ui", kMCStringOptionCompareCaseless)) { r_thread = kMCScriptThreadAffinityUI; return true; } return MCScriptThrowUnknownThreadAffinityError(); } static bool __MCScriptResolveForeignFunctionBindingForC(MCScriptInstanceRef p_instance, MCScriptForeignHandlerDefinition *p_handler, MCScriptForeignHandlerInfoRef p_info, bool* r_bound) { MCSLibraryRef t_module; if (!MCScriptLoadModuleLibrary(MCScriptGetModuleOfInstance(p_instance), p_info->c.library, t_module)) { if (r_bound == nil) { return MCScriptThrowUnableToLoadForiegnLibraryError(); } *r_bound = false; return true; } /* Resolve the symbol from the module which we've loaded */ void *t_pointer = MCSLibraryLookupSymbol(t_module, p_info->c.function); /* A nullptr pointer means that the symbol doesn't exist, so we either * throw an error (if in 'must' bind mode - r_bound == nullptr) or * indicate we succeeded, but the binding failed due to not finding the * symbol (r_bound != nullptr). */ if (t_pointer == nullptr) { if (r_bound == nullptr) { return MCScriptThrowUnableToResolveForeignHandlerError(p_instance, p_handler); } *r_bound = false; return true; } /* Now we must compute the libffi CIF for this C language handler. If * this fails, it is an error - not a failure to bind - as it indicates * OOM or something went wrong in libffi. */ if (!MCHandlerTypeInfoGetLayoutType(p_instance->module->types[p_handler->type]->typeinfo, p_info->c.call_type, p_handler->c.function_cif)) { /* The above call already throws an appropriate error, so we can * just return false. */ return false; } p_handler->c.function = t_pointer; if (r_bound != nullptr) { *r_bound = true; } return true; } #if defined(_MACOSX) || defined(TARGET_SUBPLATFORM_IPHONE) /* objc:library>class.method[?thread] */ static bool __MCScriptResolveForeignFunctionBindingForObjC(MCScriptInstanceRef p_instance, MCScriptForeignHandlerDefinition *p_handler, MCScriptForeignHandlerInfoRef p_info, bool* r_bound) { MCSLibraryRef t_module; if (!MCScriptLoadModuleLibrary(MCScriptGetModuleOfInstance(p_instance), p_info->objc.library, t_module)) { if (r_bound == nil) { return MCScriptThrowUnableToLoadForiegnLibraryError(); } *r_bound = false; return true; } MCAutoStringRefAsUTF8String t_selector_name_cstring; if (!t_selector_name_cstring.Lock(p_info->objc.method_name)) { return false; } bool t_valid = true; bool t_is_dynamic = MCStringIsEmpty(p_info->objc.class_name); bool t_is_class = p_info->objc.call_type == kMCScriptForeignHandlerObjcCallTypeClassMethod; /* Lookup the class, to make sure it exists. */ Class t_objc_class = nullptr; if (t_valid) { if (t_is_dynamic) { if (t_is_class) { /* ERROR: should be can't dynamically bind to class methods */ t_valid = false; } } else { MCAutoStringRefAsUTF8String t_class_cstring; if (!t_class_cstring.Lock(p_info->objc.class_name)) { return false; } t_objc_class = objc_getClass(*t_class_cstring); if (t_objc_class == nullptr) { /* ERROR: should be class not found */ t_valid = false; } } } /* Lookup the selector, if this doesn't work then we're not going to get * very far! (OOM, probably). */ SEL t_objc_sel = sel_registerName(*t_selector_name_cstring); if (t_valid && t_objc_sel == nullptr) { /* ERROR: objc runtime failure */ t_valid = false; } uindex_t t_required_args = 0; /* Get the Method - either class or instance - from the class */ if (t_valid && !t_is_dynamic) { if (!t_is_class) { Method t_method_info = class_getInstanceMethod(t_objc_class, t_objc_sel); if (t_method_info != nullptr) { t_required_args = method_getNumberOfArguments(t_method_info); } else { t_valid = false; } /* If a lookup fails, then check to see if it could be a property * binding. Classes such as NSUserNotification declare dynamic props * which means that the methods aren't present until runtime, but * the prop definitions are. */ if (!t_valid && MCStringBeginsWithCString(p_info->objc.method_name, (const char_t *)"set", kMCStringOptionCompareExact) && MCStringEndsWithCString(p_info->objc.method_name, (const char_t *)":", kMCStringOptionCompareExact)) { if (MCStringGetLength(p_info->objc.method_name) < 255) { char t_prop_name[256]; strncpy(t_prop_name, *t_selector_name_cstring + 3, strlen(*t_selector_name_cstring) - 4); t_prop_name[0] = tolower(t_prop_name[0]); t_prop_name[strlen(*t_selector_name_cstring) - 4] = '\0'; objc_property_t t_property = class_getProperty(t_objc_class, t_prop_name); if (t_property != nullptr) { t_valid = strstr(property_getAttributes(t_property), ",R,") == nullptr; } } t_required_args = 3; } else if (!t_valid && MCStringCountChar(p_info->objc.method_name, MCRangeMake(0, MCStringGetLength(p_info->objc.method_name)), ':', kMCStringOptionCompareExact) == 0) { objc_property_t t_property = class_getProperty(t_objc_class, *t_selector_name_cstring); if (t_property != nullptr) { t_valid = true; } t_required_args = 2; } } else { Method t_method_info = class_getClassMethod(t_objc_class, t_objc_sel); if (t_method_info != nullptr) { t_required_args = method_getNumberOfArguments(t_method_info); } else { t_valid = false; } } } /* Next we must compute the cif. */ MCAutoCustomPointer t_layout_cif; ffi_cif *t_cif = nullptr; ffi_type *t_cif_return_type = nullptr; ffi_type **t_cif_arg_types = nullptr; unsigned int t_arg_count = 0; if (t_valid) { MCTypeInfoRef t_signature = p_instance->module->types[p_handler->type]->typeinfo; /* The number of arguments used to call the bound handler */ uindex_t t_user_arg_count = MCHandlerTypeInfoGetParameterCount(t_signature); /* The number of arguments used to pass to objc_msgSend. This is 1 more * if an instance method (the selector); but is 2 more if a class method * (the class object and the selector). */ t_arg_count = t_user_arg_count + (t_is_class ? 2 : 1); MCTypeInfoRef t_return_type = MCHandlerTypeInfoGetReturnType(t_signature); t_valid = t_arg_count == t_required_args || t_is_dynamic; MCResolvedTypeInfo t_resolved_return_type; if (!MCTypeInfoResolve(t_return_type, t_resolved_return_type)) { t_valid = false; } else if (t_valid) { if (t_resolved_return_type.named_type == kMCNullTypeInfo) { t_cif_return_type = &ffi_type_void; } else if (MCTypeInfoIsForeign(t_resolved_return_type.type)) { t_cif_return_type = (ffi_type *)MCForeignTypeInfoGetLayoutType(t_resolved_return_type.type); } else { t_cif_return_type = &ffi_type_pointer; } } unsigned int t_user_arg_index = 0; unsigned int t_arg_index = 0; /* Allocate memory for the handler layout (array of ffi_type*) and * ffi_cif. This is a single block which will contain the cif followed * by the type array. */ if (t_valid) { if (!MCMemoryAllocate(sizeof(ffi_type*) * t_arg_count + sizeof(ffi_cif), &t_layout_cif)) { /* ERROR: OOM */ t_valid = false; } else { t_cif = static_cast(*t_layout_cif); t_cif_arg_types = reinterpret_cast(t_cif + 1); /* The first argument is always of pointer type - the object ptr. This * is only present in the user signature if it is an instance method.*/ t_cif_arg_types[t_arg_index] = &ffi_type_pointer; t_arg_index += 1; if (!t_is_class) { t_user_arg_index += 1; } /* The second argument is always of pointer type - the SEL ptr. This is * never present in the user signature. */ t_cif_arg_types[t_arg_index] = &ffi_type_pointer; t_arg_index += 1; } } while(t_user_arg_index < t_user_arg_count && t_valid) { MCHandlerTypeFieldMode t_mode = MCHandlerTypeInfoGetParameterMode(t_signature, t_user_arg_index); if (t_mode != kMCHandlerTypeFieldModeIn) { /* 'out' and 'inout' parameters are always pointers */ t_cif_arg_types[t_arg_index] = &ffi_type_pointer; } else { MCTypeInfoRef t_type = MCHandlerTypeInfoGetParameterType(t_signature, t_user_arg_index); MCResolvedTypeInfo t_resolved_type; if (!MCTypeInfoResolve(t_type, t_resolved_type)) { t_valid = false; break; } if (MCTypeInfoIsForeign(t_resolved_type.type)) { t_cif_arg_types[t_arg_index] = (ffi_type *)MCForeignTypeInfoGetLayoutType(t_resolved_type.type); } else { t_cif_arg_types[t_arg_index] = &ffi_type_pointer; } } t_user_arg_index += 1; t_arg_index += 1; } } /* Now can prep the cif for the var-args method_invoke* calls we need to * make. */ if (t_valid) { if (ffi_prep_cif(t_cif, FFI_DEFAULT_ABI, t_arg_count, t_cif_return_type, t_cif_arg_types) != FFI_OK) { /* ERROR: libffi failure */ t_valid = false; } } if (!t_valid) { if (r_bound != nullptr) { *r_bound = false; return true; } return MCScriptThrowUnableToResolveForeignHandlerError(p_instance, p_handler); } if (t_is_class) { p_handler->objc.call_type = kMCScriptForeignHandlerObjcCallTypeClassMethod; } else { p_handler->objc.call_type = kMCScriptForeignHandlerObjcCallTypeInstanceMethod; } p_handler->objc.objc_class = t_objc_class; p_handler->objc.objc_selector = t_objc_sel; p_handler->objc.function_cif = static_cast(t_layout_cif.Release()); if (r_bound != nullptr) { *r_bound = true; } return true; } #endif static bool __MCScriptResolveForeignFunctionBindingForJava(MCScriptInstanceRef p_instance, MCScriptForeignHandlerDefinition *p_handler, MCScriptForeignHandlerInfoRef p_info, bool* r_bound) { MCNewAutoNameRef t_class_name; if (!MCNameCreate(p_info->java.class_name, &t_class_name)) return false; p_handler->java.class_name = MCValueRetain(*t_class_name); MCTypeInfoRef t_signature = p_instance -> module -> types[p_handler -> type] -> typeinfo; p_handler->java.call_type = p_info->java.call_type; // Check that the java signature in the binding string is // compatible with the types of the foreign handler if (!MCJavaCheckSignature(t_signature, p_info->java.arguments, p_info->java.return_type, p_info->java.call_type) || !MCJavaVMInitialize()) { if (r_bound == nullptr) { return false; } MCErrorReset(); *r_bound = false; return true; } void *t_method_id = MCJavaGetMethodId(*t_class_name, p_info->java.method_name, p_info->java.arguments, p_info->java.return_type, p_handler -> java . call_type); if (t_method_id != nullptr) { p_handler -> java . method_id = t_method_id; } else { if (r_bound == nullptr) { return MCScriptThrowUnableToResolveForeignHandlerError(p_instance, p_handler); } MCErrorReset(); *r_bound = false; return true; } if (r_bound != nullptr) { *r_bound = true; } return true; } bool MCScriptForeignHandlerInfoParse(MCStringRef p_binding, MCScriptForeignHandlerInfoRef& r_self) { MCAutoCustomPointer t_info; t_info = new (nothrow) MCScriptForeignHandlerInfo; t_info->language = kMCScriptForeignHandlerLanguageUnknown; t_info->thread_affinity = kMCScriptThreadAffinityDefault; if (MCStringIsEqualToCString(p_binding, "", kMCStringOptionCompareExact)) { t_info->language = kMCScriptForeignHandlerLanguageBuiltinC; t_info.ReleaseTo(r_self); return true; } bool t_success = true; MCAutoStringRef t_rest; t_success = MCStringCopy(p_binding, &t_rest); MCAutoStringRef t_language; if (t_success) { t_success = __MCScriptSplitForeignBindingString(InOut(t_rest), ':', &t_language); } if (MCStringIsEmpty(*t_language) || MCStringIsEqualToCString(*t_language, "c", kMCStringOptionCompareExact)) { MCAutoStringRef t_library; MCAutoStringRef t_function; MCAutoStringRef t_calling; MCAutoStringRef t_thread; t_success = __MCScriptSplitForeignBindingString(InOut(t_rest), '>', &t_library) && __MCScriptSplitForeignBindingString(InOut(t_rest), '!', &t_function) && MCStringDivideAtChar(*t_rest, '?', kMCStringOptionCompareExact, &t_calling, &t_thread); /* If the function is empty, but calling is not, then there must have * been no ! part. */ if (t_success && MCStringIsEmpty(*t_function) && !MCStringIsEmpty(*t_calling)) { t_function.Reset(*t_calling); t_calling.Reset(kMCEmptyString); } int t_cc; if (t_success && !MCStringIsEmpty(*t_calling)) { MCLog("%@", *t_calling); static const char *s_callconvs[] = { "default", "sysv", "stdcall", "thiscall", "fastcall", "cdecl", "pascal", "register", nil }; for(t_cc = 0; s_callconvs[t_cc] != nil; t_cc++) { if (MCStringIsEqualToCString(*t_calling, s_callconvs[t_cc], kMCStringOptionCompareCaseless)) break; } if (s_callconvs[t_cc] == nil) { return MCScriptThrowUnknownForeignCallingConventionError(); } } else t_cc = 0; if (t_success && MCStringIsEmpty(*t_function)) { return MCScriptThrowMissingFunctionInForeignBindingError(); } MCScriptThreadAffinity t_thread_affinity; t_success = t_success && __MCScriptGetThreadAffinity(*t_thread, t_thread_affinity); if (t_success) { t_info->thread_affinity = t_thread_affinity; /* The ABI for all platforms is always DEFAULT, except on 32-bit Win32 * where it is specified as part of the signature. */ t_info->c.call_type = (int)FFI_DEFAULT_ABI; #if defined(_WIN32) && defined(__32_BIT__) t_info->c.call_type = t_cc == 0 ? (int)FFI_DEFAULT_ABI : t_cc; #endif t_info->c.function = t_function.Take(); t_info->c.library = t_library.Take(); t_info->language = kMCScriptForeignHandlerLanguageC; } } else if (MCStringIsEqualToCString(*t_language, "objc", kMCStringOptionCompareExact)) { MCAutoStringRef t_library; MCAutoStringRef t_class; MCAutoStringRef t_method; MCAutoStringRef t_thread; if (t_success) { t_success = __MCScriptSplitForeignBindingString(InOut(t_rest), '>', &t_library) && __MCScriptSplitForeignBindingString(InOut(t_rest), '.', &t_class) && __MCScriptSplitForeignBindingString(InOut(t_rest), '?', &t_method) && MCStringCopy(*t_rest, &t_thread); } /* If method is empty, then there must have been no '?' part. */ if (t_success && MCStringIsEmpty(*t_method)) { t_method.Reset(*t_thread); t_thread.Reset(kMCEmptyString); } if (t_success && (*t_method == nullptr || MCStringIsEmpty(*t_method))) { return MCScriptThrowMissingFunctionInForeignBindingError(); } uindex_t t_selector_start = 0; bool t_is_class = false; if (t_success) { if (MCStringGetCharAtIndex(*t_method, 0) == '+') { t_is_class = true; t_selector_start = 1; } else if (MCStringGetCharAtIndex(*t_method, 0) == '-') { t_is_class = false; t_selector_start = 1; } } MCAutoStringRef t_selector_name; t_success = t_success && MCStringCopySubstring(*t_method, MCRangeMake(t_selector_start, UINDEX_MAX), &t_selector_name); MCScriptThreadAffinity t_thread_affinity; t_success = t_success && __MCScriptGetThreadAffinity(*t_thread, t_thread_affinity); if (t_success) { t_info->thread_affinity = t_thread_affinity; if (t_is_class) { t_info->objc.call_type = kMCScriptForeignHandlerObjcCallTypeClassMethod; } else { t_info->objc.call_type = kMCScriptForeignHandlerObjcCallTypeInstanceMethod; } t_info->objc.method_name = t_selector_name.Take(); t_info->objc.class_name = t_class.Take(); t_info->objc.library = t_library.Take(); t_info->language = kMCScriptForeignHandlerLanguageObjC; } } else if (MCStringIsEqualToCString(*t_language, "java", kMCStringOptionCompareExact)) { MCAutoStringRef t_library; MCAutoStringRef t_class; MCAutoStringRef t_function_string; MCAutoStringRef t_calling; MCAutoStringRef t_thread; t_success = __MCScriptSplitForeignBindingString(InOut(t_rest), '>', &t_library) && __MCScriptSplitForeignBindingString(InOut(t_rest), '.', &t_class) && __MCScriptSplitForeignBindingString(InOut(t_rest), '!', &t_function_string); MCAutoStringRef t_to_divide; if (t_success && !MCStringIsEmpty(*t_function_string)) { // Calling convention is not empty t_success = MCStringDivideAtChar(*t_rest, '?', kMCStringOptionCompareExact, &t_calling, &t_thread); } else if (t_success) { MCAutoStringRef t_real_function; // Calling convention is empty t_success = MCStringDivideAtChar(*t_rest, '?', kMCStringOptionCompareExact, &t_real_function, &t_thread); t_function_string.Reset(*t_real_function); t_calling = kMCEmptyString; } MCAutoStringRef t_arguments, t_return, t_function; t_success = t_success && __split_function_signature(*t_function_string, &t_function, &t_arguments, &t_return); MCScriptThreadAffinity t_thread_affinity; t_success = t_success && __MCScriptGetThreadAffinity(*t_thread, t_thread_affinity); if (t_success) { t_info->java.call_type = __MCScriptGetJavaCallType(*t_class, *t_function, *t_calling); if (t_info->java.call_type == MCJavaCallTypeUnknown) { return MCScriptThrowUnknownForeignCallingConventionError(); } t_info->thread_affinity = t_thread_affinity; t_info->java.arguments = t_arguments.Take(); t_info->java.return_type = t_return.Take(); t_info->java.method_name = t_function.Take(); t_info->java.class_name = t_library.Take(); t_info->language = kMCScriptForeignHandlerLanguageJava; } } if (t_success) { t_info.ReleaseTo(r_self); } return t_success; } void MCScriptForeignHandlerInfoRelease(MCScriptForeignHandlerInfoRef self) { if (self == nullptr) { return; } switch (self->language) { case kMCScriptForeignHandlerLanguageC: MCValueRelease(self->c.function); MCValueRelease(self->c.library); break; case kMCScriptForeignHandlerLanguageObjC: MCValueRelease(self->objc.class_name); MCValueRelease(self->objc.method_name); MCValueRelease(self->objc.library); break; case kMCScriptForeignHandlerLanguageJava: MCValueRelease(self->java.class_name); MCValueRelease(self->java.method_name); MCValueRelease(self->java.arguments); MCValueRelease(self->java.return_type); break; case kMCScriptForeignHandlerLanguageBuiltinC: case kMCScriptForeignHandlerLanguageUnknown: break; } delete self; } static bool __MCScriptResolveForeignFunctionBinding(MCScriptInstanceRef p_instance, MCScriptForeignHandlerDefinition *p_handler, bool* r_bound) { /* If the binding string is an integer, then it is a 'builtin-c' handler - * a shim which has been generated by lc-compile in '--outputc' mode and * compiled into the engine. */ integer_t t_ordinal = 0; if (p_instance->module->builtins != nullptr && MCTypeConvertStringToLongInteger(p_handler->binding, t_ordinal)) { p_handler->language = kMCScriptForeignHandlerLanguageBuiltinC; p_handler->builtin_c.function = p_instance->module->builtins[t_ordinal]; if (r_bound != nullptr) { *r_bound = true; } return true; } MCAutoCustomPointer t_info; if (!MCScriptForeignHandlerInfoParse(p_handler->binding, &t_info)) { return false; } bool t_status = false; switch (t_info->language) { case kMCScriptForeignHandlerLanguageC: t_status = __MCScriptResolveForeignFunctionBindingForC(p_instance, p_handler, *t_info, r_bound); break; case kMCScriptForeignHandlerLanguageObjC: #if !defined(_MACOSX) && !defined(TARGET_SUBPLATFORM_IPHONE) if (r_bound == nil) { t_status = MCScriptThrowObjCBindingNotSupported(); } else { *r_bound = false; t_status = true; } #else t_status = __MCScriptResolveForeignFunctionBindingForObjC(p_instance, p_handler, *t_info, r_bound); #endif break; case kMCScriptForeignHandlerLanguageJava: t_status = __MCScriptResolveForeignFunctionBindingForJava(p_instance, p_handler, *t_info, r_bound); break; case kMCScriptForeignHandlerLanguageBuiltinC: // should have been ordinal binding t_status = false; break; case kMCScriptForeignHandlerLanguageUnknown: t_status = MCScriptThrowUnknownForeignLanguageError(); break; } if (t_status) { p_handler->language = t_info->language; p_handler->thread_affinity = t_info->thread_affinity; } return t_status; } // This method attempts to bind a foreign function so it can be used. If // binding is required (as the call is about to be executed), r_bound should be // nil. Otherwise, it is assumed the binding is to be checked afterwards, in // which case r_bound should be non-nil, and whether or not the binding is // present / usable will be indicated via a true return value in that out // parameter. static bool __MCScriptPrepareForeignFunction(MCScriptInstanceRef p_instance, MCScriptForeignHandlerDefinition *p_handler, bool *r_bound) { // If the handler is already bound, then do nothing. if (p_handler->language != kMCScriptForeignHandlerLanguageUnknown) { if (r_bound != nullptr) { *r_bound = true; } return true; } // Attempt to resolve the foreign function binding. If r_bound == nullptr, // then this will fail if binding fails. If r_bound != nullptr, then *r_bound // will indicate whether the function was bound or not. if (!__MCScriptResolveForeignFunctionBinding(p_instance, p_handler, r_bound)) { return false; } // If we are 'try to bind' and the foreign binding failed, then // return the bound status to the caller and return. if (r_bound != nullptr && !*r_bound) { return true; } return true; } bool MCScriptTryToBindForeignHandlerInInstanceInternal(MCScriptInstanceRef p_instance, MCScriptForeignHandlerDefinition *p_handler, bool& r_bound) { return __MCScriptPrepareForeignFunction(p_instance, p_handler, &r_bound); } bool MCScriptBindForeignHandlerInInstanceInternal(MCScriptInstanceRef p_instance, MCScriptForeignHandlerDefinition *p_handler) { return __MCScriptPrepareForeignFunction(p_instance, p_handler, nil); } //////////////////////////////////////////////////////////////////////////////// // This context struct, callbacks and associated functions provide an implementation // of MCHandler for handlers coming from libscript. It enables an MCHandlerRef // created from within LCB to be used 'out-of-frame' - i.e. not just from direct // invocation within the LCB VM. struct __MCScriptInternalHandlerContext { MCScriptInstanceRef instance; MCScriptCommonHandlerDefinition *definition; }; static void __MCScriptInternalHandlerRelease(void *p_context) { __MCScriptInternalHandlerContext *context = (__MCScriptInternalHandlerContext *)p_context; MCScriptReleaseInstance(context->instance); } static bool __MCScriptInternalHandlerInvoke(void *p_context, MCValueRef *p_arguments, uindex_t p_argument_count, MCValueRef& r_return_value) { __MCScriptInternalHandlerContext *context = (__MCScriptInternalHandlerContext *)p_context; if (context->definition->kind != kMCScriptDefinitionKindHandler) return MCErrorThrowGeneric(MCSTR("out-of-frame indirect foreign handler calls not yet supported")); return MCScriptCallHandlerInInstanceInternal(context->instance, static_cast(context->definition), p_arguments, p_argument_count, r_return_value); } static bool __MCScriptInternalHandlerDescribe(void *p_context, MCStringRef& r_description) { __MCScriptInternalHandlerContext *context = (__MCScriptInternalHandlerContext *)p_context; MCScriptModuleRef t_module; t_module = MCScriptGetModuleOfInstance (context->instance); MCNameRef t_module_name; t_module_name = MCScriptGetNameOfModule (t_module); MCNameRef t_handler_name; t_handler_name = MCScriptGetNameOfDefinitionInModule(t_module, context->definition); return MCStringFormat(r_description, "%@.%@()", t_module_name, t_handler_name); } static bool __MCScriptInternalHandlerQuery(void *p_context, MCHandlerQueryType p_query, void *r_info) { if (p_query != kMCHandlerQueryTypeObjcSelector) { return false; } __MCScriptInternalHandlerContext *context = (__MCScriptInternalHandlerContext *)p_context; if (context->definition->kind != kMCScriptDefinitionKindForeignHandler) { return false; } auto t_definition = static_cast(context->definition); if (t_definition->language != kMCScriptForeignHandlerLanguageObjC) { return false; } *(void **)r_info = t_definition->objc.objc_selector; return true; } static MCHandlerCallbacks __kMCScriptInternalHandlerCallbacks = { sizeof(__MCScriptInternalHandlerContext), __MCScriptInternalHandlerRelease, __MCScriptInternalHandlerInvoke, __MCScriptInternalHandlerDescribe, __MCScriptInternalHandlerQuery, }; static bool __MCScriptInternalHandlerCreate(MCScriptInstanceRef p_instance, MCScriptCommonHandlerDefinition *p_handler_def, MCHandlerRef& r_handler) { MCTypeInfoRef t_signature; t_signature = p_instance->module->types[p_handler_def->type]->typeinfo; // The context struct is 'moved' into the handlerref. __MCScriptInternalHandlerContext t_context; t_context . instance = p_instance; t_context . definition = p_handler_def; return MCHandlerCreate(t_signature, &__kMCScriptInternalHandlerCallbacks, &t_context, r_handler); } bool MCScriptHandlerIsInternal(MCHandlerRef p_handler) { return MCHandlerGetCallbacks(p_handler) == &__kMCScriptInternalHandlerCallbacks; } void MCScriptInternalHandlerQuery(MCHandlerRef p_handler, MCScriptInstanceRef& r_instance, MCScriptCommonHandlerDefinition*& r_handler_def) { __MCScriptAssert__(MCScriptHandlerIsInternal(p_handler), "passed handlerref is not an script internal handler ref"); __MCScriptInternalHandlerContext *t_context = static_cast<__MCScriptInternalHandlerContext *>(MCHandlerGetContext(p_handler)); r_instance = t_context->instance; r_handler_def = t_context->definition; } static uindex_t __MCScriptComputeHandlerIndexInInstance(MCScriptInstanceRef p_instance, MCScriptCommonHandlerDefinition *p_handler) { uindex_t t_min, t_max; t_min = 0; t_max = p_instance->handler_count; while(t_min < t_max) { uindex_t t_mid; t_mid = (t_min + t_max) / 2; if (p_instance->handlers[t_mid].definition < p_handler) t_min = t_mid + 1; else t_max = t_mid; } return t_min; } bool MCScriptEvaluateHandlerInInstanceInternal(MCScriptInstanceRef p_instance, MCScriptCommonHandlerDefinition *p_handler_def, MCHandlerRef& r_handler) { // Compute the index in the handler table of p_handler; then, if it is the // definition we are looking for, return its previously computed value. uindex_t t_index; t_index = __MCScriptComputeHandlerIndexInInstance(p_instance, p_handler_def); if (t_index < p_instance->handler_count && p_instance->handlers[t_index].definition == p_handler_def) { r_handler = p_instance->handlers[t_index] . value; return true; } MCTypeInfoRef t_signature; t_signature = p_instance->module->types[p_handler_def->type]->typeinfo; MCAutoValueRefBase t_handler; if (!__MCScriptInternalHandlerCreate(p_instance, p_handler_def, &t_handler)) { return false; } // Now put the handler value into the instance's handler list. First we make // space in the array, then insert the value at the index we computed at the // start. if (!MCMemoryResizeArray(p_instance->handler_count + 1, p_instance->handlers, p_instance->handler_count)) { return false; } MCMemoryMove(p_instance->handlers + t_index + 1, p_instance->handlers + t_index, (p_instance->handler_count - t_index - 1) * sizeof(MCScriptHandlerValue)); p_instance->handlers[t_index].definition = p_handler_def; p_instance->handlers[t_index].value = t_handler.Take(); r_handler = p_instance->handlers[t_index].value; return true; } //////////////////////////////////////////////////////////////////////////////// MCScriptModuleRef MCScriptGetCurrentModule(void) { return s_current_module; } MCScriptModuleRef MCScriptSetCurrentModule(MCScriptModuleRef p_module) { MCScriptModuleRef t_previous = s_current_module; s_current_module = p_module; return t_previous; } //////////////////////////////////////////////////////////////////////////////// extern "C" bool MC_DLLEXPORT_DEF MCScriptBuiltinRepeatCounted(uinteger_t *x_count) { if (*x_count == 0) return false; *x_count -= 1; return true; } extern "C" bool MC_DLLEXPORT_DEF MCScriptBuiltinRepeatUpToCondition(double p_counter, double p_limit) { return p_counter <= p_limit; } extern "C" double MC_DLLEXPORT_DEF MCScriptBuiltinRepeatUpToIterate(double p_counter, double p_step) { return p_counter + p_step; } extern "C" bool MC_DLLEXPORT_DEF MCScriptBuiltinRepeatDownToCondition(double p_counter, double p_limit) { return p_counter >= p_limit; } extern "C" double MC_DLLEXPORT_DEF MCScriptBuiltinRepeatDownToIterate(double p_counter, double p_step) { return p_counter + p_step; } extern "C" void MC_DLLEXPORT_DEF MCScriptBuiltinThrow(MCStringRef p_reason) { MCErrorThrowGeneric(p_reason); } ////////////////////////////////////////////////////////////////////////////////