]> granicus.if.org Git - php/commitdiff
ADW: Fix COM extension bug #35463. Fix in/out argument processing
authorandy wharmby <wharmby@php.net>
Fri, 2 Feb 2007 08:42:32 +0000 (08:42 +0000)
committerandy wharmby <wharmby@php.net>
Fri, 2 Feb 2007 08:42:32 +0000 (08:42 +0000)
ext/com_dotnet/com_misc.c
ext/com_dotnet/com_variant.c
ext/com_dotnet/com_wrapper.c
ext/com_dotnet/php_com_dotnet_internal.h

index a5adaedd36d49ced983b6eec95180d1fab048dfe..bc443e62a468f59acef2f6fb3d6d32c543243524 100644 (file)
@@ -78,6 +78,7 @@ PHPAPI void php_com_wrap_variant(zval *z, VARIANT *v,
 
        VariantInit(&obj->v);
        VariantCopyInd(&obj->v, v);
+       obj->modified = 0;
 
        if ((V_VT(&obj->v) == VT_DISPATCH) && (V_DISPATCH(&obj->v) != NULL)) {
                IDispatch_GetTypeInfo(V_DISPATCH(&obj->v), 0, LANG_NEUTRAL, &obj->typeinfo);
index 8f3d6528f441aec5559181d25d1b4e379d73ef50..8248b1f987fc267c52d59bbb5e3de2120d4ae641 100644 (file)
@@ -264,6 +264,140 @@ PHPAPI int php_com_zval_from_variant(zval *z, VARIANT *v, int codepage TSRMLS_DC
        return ret;
 }
 
+
+PHPAPI int php_com_copy_variant(VARIANT *dstvar, VARIANT *srcvar TSRMLS_DC)
+{
+       int ret = SUCCESS;
+       
+       switch (V_VT(dstvar) & ~VT_BYREF) {
+       case VT_EMPTY:
+       case VT_NULL:
+       case VT_VOID:
+               /* should not be possible */
+               break;
+
+       case VT_UI1:
+               if (V_VT(dstvar) & VT_BYREF) {
+                       *V_UI1REF(dstvar) = V_UI1(srcvar);
+               } else {
+                        V_UI1(dstvar) = V_UI1(srcvar);
+               }
+               break;
+
+       case VT_I1:
+               if (V_VT(dstvar) & VT_BYREF) {
+                       *V_I1REF(dstvar) = V_I1(srcvar);
+               } else {
+                       V_I1(dstvar) = V_I1(srcvar);
+               }
+               break;
+
+       case VT_UI2:
+               if (V_VT(dstvar) & VT_BYREF) {
+                       *V_UI2REF(dstvar) = V_UI2(srcvar);
+               } else {
+                       V_UI2(dstvar) = V_UI2(srcvar); 
+               }
+               break;
+
+       case VT_I2:
+               if (V_VT(dstvar) & VT_BYREF) {
+                       *V_I2REF(dstvar) = V_I2(srcvar);
+               } else {
+                       V_I2(dstvar) = V_I2(srcvar);
+               }
+               break;
+
+       case VT_UI4: 
+               if (V_VT(dstvar) & VT_BYREF) {
+                       *V_UI4REF(dstvar) = V_UI4(srcvar);
+               } else {
+                       V_UI4(dstvar) = V_UI4(srcvar);
+               }
+               break;
+
+       case VT_I4:
+               if (V_VT(dstvar) & VT_BYREF) {
+                       *V_I4REF(dstvar) = V_I4(srcvar);
+               } else {
+                       V_I4(dstvar) = V_I4(srcvar);
+               }
+               break;
+
+       case VT_INT:
+               if (V_VT(dstvar) & VT_BYREF) {
+                       *V_INTREF(dstvar) = V_INT(srcvar);
+               } else {
+                       V_INT(dstvar) = V_INT(srcvar);
+               }
+               break;
+
+       case VT_UINT:
+               if (V_VT(dstvar) & VT_BYREF) {
+                       *V_UINTREF(dstvar) = V_UINT(srcvar);
+               } else {
+                       V_UINT(dstvar) = V_UINT(srcvar);       
+               }
+               break;
+
+       case VT_R4:
+               if (V_VT(dstvar) & VT_BYREF) {
+                       *V_R4REF(dstvar) = V_R4(srcvar);
+               } else {
+                       V_R4(dstvar) = V_R4(srcvar);
+               }
+               break;
+
+       case VT_R8:
+               if (V_VT(dstvar) & VT_BYREF) {
+                       *V_R8REF(dstvar) = V_R8(srcvar);
+               } else {
+                       V_R8(dstvar) = V_R8(srcvar);
+               }
+               break;
+
+       case VT_BOOL:
+               if (V_VT(dstvar) & VT_BYREF) {
+                       *V_BOOLREF(dstvar) = V_BOOL(srcvar);
+               } else {
+                       V_BOOL(dstvar) = V_BOOL(srcvar);
+               }
+        break;
+
+       case VT_BSTR:
+               if (V_VT(dstvar) & VT_BYREF) {
+                       *V_BSTRREF(dstvar) = V_BSTR(srcvar);
+               } else {
+                       V_BSTR(dstvar) = V_BSTR(srcvar);
+        }
+               break;
+
+       case VT_UNKNOWN:
+               if (V_VT(dstvar) & VT_BYREF) {
+                       *V_UNKNOWNREF(dstvar) = V_UNKNOWN(srcvar);
+               } else {
+                       V_UNKNOWN(dstvar) = V_UNKNOWN(srcvar);
+               }
+               break;
+
+       case VT_DISPATCH:
+               if (V_VT(dstvar) & VT_BYREF) {
+                       *V_DISPATCHREF(dstvar) = V_DISPATCH(srcvar);
+               } else {
+                       V_DISPATCH(dstvar) = V_DISPATCH(srcvar);
+               }
+               break;
+
+       case VT_VARIANT:
+               return php_com_copy_variant(V_VARIANTREF(dstvar), srcvar TSRMLS_CC);
+               
+       default:
+               php_error_docref(NULL TSRMLS_CC, E_WARNING, "variant->variant: failed to copy from 0x%x to 0x%x", V_VT(dstvar), V_VT(srcvar));
+               ret = FAILURE;
+       }
+       return ret;
+}
+
 /* {{{ com_variant_create_instance - ctor for new VARIANT() */
 PHP_FUNCTION(com_variant_create_instance)
 {
@@ -364,6 +498,8 @@ PHP_FUNCTION(variant_set)
        VariantClear(&obj->v);
 
        php_com_variant_from_zval(&obj->v, zvalue, obj->code_page TSRMLS_CC);
+       /* remember we modified this variant */
+       obj->modified = 1;
 }
 /* }}} */
 
index af8579c6bd1e80c77d898db585b2cfac6c64d050..4cc81aa04b115a85b251e1da0aee3c204e248f2a 100644 (file)
@@ -298,6 +298,17 @@ static HRESULT STDMETHODCALLTYPE disp_invokeex(
                                                        &retval, pdp->cArgs, params, 1, NULL TSRMLS_CC)) {
                                        ret = S_OK;
                                        trace("function called ok\n");
+
+                                       /* Copy any modified values to callers copy of variant*/
+                                       for (i = 0; i < pdp->cArgs; i++) {
+                                               php_com_dotnet_object *obj = CDNO_FETCH(*params[i]);
+                                               VARIANT *srcvar = &obj->v;
+                                               VARIANT *dstvar = &pdp->rgvarg[ pdp->cArgs - 1 - i];
+                                               if ((V_VT(dstvar) & VT_BYREF) && obj->modified ) {
+                                                       trace("percolate modified value for arg %d VT=%08x\n", i, V_VT(dstvar));
+                                                       php_com_copy_variant(dstvar, srcvar TSRMLS_CC);   
+                                               }
+                                       }
                                } else {
                                        trace("failed to call func\n");
                                        ret = DISP_E_EXCEPTION;
index 4a57cd17209047778c5004fdc4253ccabff25d05..12c632a1cb2f34731b208290f5196625edd4a1ca 100644 (file)
@@ -36,6 +36,7 @@ typedef struct _php_com_dotnet_object {
        zend_object zo;
 
        VARIANT v;
+       int modified;
 
        ITypeInfo *typeinfo;
        long code_page;
@@ -152,6 +153,7 @@ PHP_FUNCTION(variant_cast);
 PHPAPI void php_com_variant_from_zval_with_type(VARIANT *v, zval *z, VARTYPE type, int codepage TSRMLS_DC);
 PHPAPI void php_com_variant_from_zval(VARIANT *v, zval *z, int codepage TSRMLS_DC);
 PHPAPI int php_com_zval_from_variant(zval *z, VARIANT *v, int codepage TSRMLS_DC);
+PHPAPI int php_com_copy_variant(VARIANT *dst, VARIANT *src TSRMLS_DC);
 
 /* com_dotnet.c */
 PHP_FUNCTION(com_dotnet_create_instance);