/* +----------------------------------------------------------------------+ | PHP Version 5 | +----------------------------------------------------------------------+ | Copyright (c) 1997-2004 The PHP Group | +----------------------------------------------------------------------+ | This source file is subject to version 3.0 of the PHP license, | | that is bundled with this package in the file LICENSE, and is | | available through the world-wide-web at the following url: | | http://www.php.net/license/3_0.txt. | | If you did not receive a copy of the PHP license and are unable to | | obtain it through the world-wide-web, please send a note to | | license@php.net so we can mail you a copy immediately. | +----------------------------------------------------------------------+ | Author: Wez Furlong | +----------------------------------------------------------------------+ */ /* $Id$ */ #ifdef HAVE_CONFIG_H #include "config.h" #endif #include "php.h" #if HAVE_MSCOREE_H # include "php_ini.h" # include "ext/standard/info.h" # include "php_com_dotnet.h" # include "php_com_dotnet_internal.h" # include "Zend/zend_exceptions.h" # include struct dotnet_runtime_stuff { ICorRuntimeHost *dotnet_host; IDispatch *dotnet_domain; DISPID create_instance; }; /* Since there is no official public mscorlib.h header file, and since * generating your own version from the binary .tlb file results in a 3MB * header file (!), we opt for the Dispatch-able approach. This is slightly * slower for creating new objects, but not too bad */ static int dotnet_init(TSRMLS_D) { HRESULT hr; struct dotnet_runtime_stuff *stuff; IUnknown *unk = NULL; OLECHAR *olename; stuff = malloc(sizeof(*stuff)); memset(stuff, 0, sizeof(*stuff)); if (SUCCEEDED(CoCreateInstance(&CLSID_CorRuntimeHost, NULL, CLSCTX_ALL, &IID_ICorRuntimeHost, (LPVOID*)&stuff->dotnet_host))) { /* fire up the host and get the domain object */ if (SUCCEEDED(ICorRuntimeHost_Start(stuff->dotnet_host)) && SUCCEEDED(ICorRuntimeHost_GetDefaultDomain(stuff->dotnet_host, &unk)) && SUCCEEDED(IUnknown_QueryInterface(unk, &IID_IDispatch, (LPVOID*)&stuff->dotnet_domain))) { /* locate the create-instance member */ olename = php_com_string_to_olestring("CreateInstance", sizeof("CreateInstance")-1, CP_ACP TSRMLS_CC); hr = IDispatch_GetIDsOfNames(stuff->dotnet_domain, &IID_NULL, &olename, 1, LOCALE_SYSTEM_DEFAULT, &stuff->create_instance); efree(olename); if (SUCCEEDED(hr)) { COMG(dotnet_runtime_stuff) = stuff; } } if (unk) { IUnknown_Release(unk); } } if (COMG(dotnet_runtime_stuff) == NULL) { /* clean up */ if (stuff->dotnet_domain) { IDispatch_Release(stuff->dotnet_domain); } if (stuff->dotnet_host) { ICorRuntimeHost_Stop(stuff->dotnet_host); ICorRuntimeHost_Release(stuff->dotnet_host); } free(stuff); return FAILURE; } return SUCCESS; } /* {{{ com_dotnet_create_instance - ctor for DOTNET class */ PHP_FUNCTION(com_dotnet_create_instance) { zval *object = getThis(); php_com_dotnet_object *obj; char *assembly_name, *datatype_name; int assembly_name_len, datatype_name_len; struct dotnet_runtime_stuff *stuff; DISPPARAMS params; VARIANT vargs[2]; VARIANT retval; HRESULT hr; int ret = FAILURE; char *where = ""; if (COMG(dotnet_runtime_stuff) == NULL) { if (FAILURE == dotnet_init(TSRMLS_C)) { php_com_throw_exception(E_ERROR, "Failed to initialize .Net runtime" TSRMLS_CC); ZVAL_NULL(object); return; } } stuff = (struct dotnet_runtime_stuff*)COMG(dotnet_runtime_stuff); obj = CDNO_FETCH(object); if (FAILURE == zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "ss|l", &assembly_name, &assembly_name_len, &datatype_name, &datatype_name_len, &obj->code_page)) { php_com_throw_exception(E_INVALIDARG, "Could not create .Net object - invalid arguments!" TSRMLS_CC); ZVAL_NULL(object); return; } params.cArgs = 2; params.cNamedArgs = 0; params.rgdispidNamedArgs = NULL; params.rgvarg = vargs; VariantInit(&vargs[0]); VariantInit(&vargs[1]); VariantInit(&retval); V_VT(&vargs[0]) = VT_BSTR; V_BSTR(&vargs[0]) = php_com_string_to_olestring(datatype_name, datatype_name_len, obj->code_page TSRMLS_CC); V_VT(&vargs[1]) = VT_BSTR; V_BSTR(&vargs[1]) = php_com_string_to_olestring(assembly_name, assembly_name_len, obj->code_page TSRMLS_CC); where = "IDispatch_Invoke"; hr = IDispatch_Invoke(stuff->dotnet_domain, stuff->create_instance, &IID_NULL, LOCALE_SYSTEM_DEFAULT, DISPATCH_METHOD, ¶ms, &retval, NULL, NULL); if (SUCCEEDED(hr)) { /* retval should now be an IUnknown/IDispatch representation of an IObjectHandle interface */ if (V_VT(&retval) == VT_UNKNOWN || V_VT(&retval) == VT_DISPATCH) { VARIANT unwrapped; IObjectHandle *handle; where = "QI: IID_IObjectHandle"; hr = IUnknown_QueryInterface(V_UNKNOWN(&retval), &IID_IObjectHandle, &handle); if (SUCCEEDED(hr)) { where = "IObjectHandle_Unwrap"; hr = IObjectHandle_Unwrap(handle, &unwrapped); if (SUCCEEDED(hr)) { /* unwrapped is now the dispatch pointer we want */ V_DISPATCH(&obj->v) = V_DISPATCH(&unwrapped); V_VT(&obj->v) = VT_DISPATCH; /* get its type-info */ IDispatch_GetTypeInfo(V_DISPATCH(&obj->v), 0, LANG_NEUTRAL, &obj->typeinfo); ret = SUCCESS; } IObjectHandle_Release(handle); } } VariantClear(&retval); } VariantClear(&vargs[0]); VariantClear(&vargs[1]); if (ret == FAILURE) { char buf[1024]; char *err = php_win_err(hr); snprintf(buf, sizeof(buf), "Failed to instantiate .Net object [%s] %s", where, err); if (err) LocalFree(err); php_com_throw_exception(hr, buf TSRMLS_CC); ZVAL_NULL(object); return; } } /* }}} */ void php_com_dotnet_mshutdown(TSRMLS_D) { struct dotnet_runtime_stuff *stuff = COMG(dotnet_runtime_stuff); if (stuff->dotnet_domain) { IDispatch_Release(stuff->dotnet_domain); } if (stuff->dotnet_host) { ICorRuntimeHost_Stop(stuff->dotnet_host); ICorRuntimeHost_Release(stuff->dotnet_host); stuff->dotnet_host = NULL; } free(stuff); COMG(dotnet_runtime_stuff) = NULL; } void php_com_dotnet_rshutdown(TSRMLS_D) { struct dotnet_runtime_stuff *stuff = COMG(dotnet_runtime_stuff); if (stuff->dotnet_domain) { IDispatch_Release(stuff->dotnet_domain); stuff->dotnet_domain = NULL; } } #endif /* HAVE_MSCOREE_H */