/*******************************************************************************
 *  RProxy: Connector implementation between application and R language
 *  Copyright (C) 1999--2006 Thomas Baier
 *
 *  This library is free software; you can redistribute it and/or
 *  modify it under the terms of the GNU Library General Public
 *  License as published by the Free Software Foundation; either
 *  version 2 of the License, or (at your option) any later version.
 * 
 *  This library 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
 *  Library General Public License for more details.
 * 
 *  You should have received a copy of the GNU Library General Public
 *  License along with this library; if not, write to the Free
 *  Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston,
 *  MA 02110-1301, USA.
 *
 *  basic COM utility functions
 *
 ******************************************************************************/

#include "rproxy.h"
#include "com_util.h"

static void _com_object_finalizer(SEXP sexp)
{
  RCOM_OBJHANDLE handle = com_getHandle(sexp);
  if(handle == RCOM_NULLHANDLE) {
    RPROXY_ERR(printf("_com_object_finalizer() called for non-pointer SEXP or non-object pointer\n"));
  } else {
    RPROXY_TRACE(printf("COM object at %p gets garbage collected\n",
			R_ExternalPtrAddr(sexp)));
#ifdef __cplusplus
    com_getObject(handle)->Release();
#else
    com_getObject(handle)->lpVtbl->Release(com_getObject(handle));
#endif
  }
}


RCOM_OBJHANDLE com_getHandle(SEXP handle)
{
  SEXP cls;
  if (TYPEOF (handle) != EXTPTRSXP) {
    RPROXY_TRACE(printf("com_getHandle(): not an external pointer\n"));
    return NULL;
  }
  cls = getAttrib (handle,R_ClassSymbol);
  if (TYPEOF (cls) != STRSXP) {
    RPROXY_TRACE(printf("com_getHandle(): no class symbol (type %d)\n",
			 TYPEOF (cls)));
    return NULL;
  }
  if (strcmp (CHAR(STRING_ELT(cls,0)),RCOM_CLSNAME) != 0) {
    RPROXY_TRACE(printf("com_getHandle(): wrong class symbol \"%s\"\n",
			 CHAR (STRING_ELT (cls,0))));
    return NULL;
  }
  return R_ExternalPtrAddr (handle);
}

/** get the COM object by handle */
LPDISPATCH com_getObject(RCOM_OBJHANDLE handle)
{
  return (LPDISPATCH) handle;
}

/** add the COM object and return the new handle */
RCOM_OBJHANDLE com_addObject(LPDISPATCH object)
{
  return object;
}

SEXP com_createSEXP(RCOM_OBJHANDLE handle)
{
  SEXP sexp = R_NilValue;
  SEXP cls;
  SEXP strsexp;

  if (handle == RCOM_NULLHANDLE) {
    RPROXY_ERR(printf ("com_createSEXP: error, invalid object handle\n"));
    return R_NilValue;
  }
  sexp = R_MakeExternalPtr(handle,R_NilValue,R_NilValue);
  R_RegisterCFinalizerEx(sexp,_com_object_finalizer,(Rboolean) TRUE);
  RPROXY_TRACE(printf("COM object watcher: finalizer for object at %p registered\n",
		    handle));
  cls = allocString(strlen(RCOM_CLSNAME));
  PROTECT (cls);
  strcpy (CHAR(cls),RCOM_CLSNAME);
  strsexp = PROTECT (allocVector (STRSXP,1));
  SET_STRING_ELT(strsexp,0,cls);
  setAttrib (sexp,R_ClassSymbol,strsexp);
  UNPROTECT(2);
  return sexp;
}


syntax highlighted by Code2HTML, v. 0.9.1