Mozilla/mozilla/xpcom/proxy/src/nsProxyObjectManager.cpp
dougt%netscape.com 9d9481b9d6 Bug Fixes to xpcom/proxy. Thanks jband for debuging this with me!
git-svn-id: svn://10.0.0.236/trunk@44177 18797224-902f-48f8-a5cc-f745e15eee43
1999-08-24 00:30:55 +00:00

425 lines
11 KiB
C++

/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*-
*
* The contents of this file are subject to the Netscape Public License
* Version 1.0 (the "NPL"); you may not use this file except in
* compliance with the NPL. You may obtain a copy of the NPL at
* http://www.mozilla.org/NPL/
*
* Software distributed under the NPL is distributed on an "AS IS" basis,
* WITHOUT WARRANTY OF ANY KIND, either express or implied. See the NPL
* for the specific language governing rights and limitations under the
* NPL.
*
* The Initial Developer of this code under the NPL is Netscape
* Communications Corporation. Portions created by Netscape are
* Copyright (C) 1998 Netscape Communications Corporation. All Rights
* Reserved.
*/
#include "nsProxyEvent.h"
#include "nsProxyObjectManager.h"
#include "nsProxyEventPrivate.h"
#include "nsIProxyCreateInstance.h"
#include "nsRepository.h"
#include "nsIServiceManager.h"
#include "nsCOMPtr.h"
#include "nsAppShellCIDs.h"
#include "nsIEventQueueService.h"
#include "nsIThread.h"
static NS_DEFINE_CID(kEventQueueServiceCID, NS_EVENTQUEUESERVICE_CID);
/***************************************************************************/
/* nsProxyCreateInstance */
/* This private class will allow us to create Instances on another thread */
/***************************************************************************/
class nsProxyCreateInstance : public nsIProxyCreateInstance
{
NS_DECL_ISUPPORTS
NS_IMETHOD CreateInstanceByIID(const nsIID & cid, nsISupports *aOuter, const nsIID & iid, void * *result);
NS_IMETHOD CreateInstanceByProgID(const char *aProgID, nsISupports *aOuter, const nsIID & iid, void * *result);
nsProxyCreateInstance();
virtual ~nsProxyCreateInstance();
};
nsProxyCreateInstance::nsProxyCreateInstance()
{
NS_INIT_REFCNT();
NS_ADDREF_THIS();
}
nsProxyCreateInstance::~nsProxyCreateInstance()
{
}
NS_IMPL_ISUPPORTS1(nsProxyCreateInstance, nsIProxyCreateInstance)
NS_IMETHODIMP nsProxyCreateInstance::CreateInstanceByIID(const nsIID & cid, nsISupports *aOuter, const nsIID & iid, void * *result)
{
return nsComponentManager::CreateInstance( cid,
aOuter,
iid,
result);
}
NS_IMETHODIMP nsProxyCreateInstance::CreateInstanceByProgID(const char *aProgID, nsISupports *aOuter, const nsIID & iid, void * *result)
{
return nsComponentManager::CreateInstance( aProgID,
aOuter,
iid,
result);
}
/////////////////////////////////////////////////////////////////////////
// nsProxyObjectManager
/////////////////////////////////////////////////////////////////////////
nsProxyObjectManager* nsProxyObjectManager::mInstance = nsnull;
NS_IMPL_ISUPPORTS1(nsProxyObjectManager, nsIProxyObjectManager)
nsProxyObjectManager::nsProxyObjectManager()
{
NS_INIT_REFCNT();
mProxyClassMap = new nsHashtable(256, PR_TRUE);
mProxyObjectMap = new nsHashtable(256, PR_TRUE);
}
nsProxyObjectManager::~nsProxyObjectManager()
{
delete mProxyClassMap;
delete mProxyObjectMap;
nsProxyObjectManager::mInstance = nsnull;
}
nsProxyObjectManager *
nsProxyObjectManager::GetInstance()
{
if (mInstance == nsnull)
{
mInstance = new nsProxyObjectManager();
}
return mInstance;
}
// Helpers
NS_IMETHODIMP
nsProxyObjectManager::Create(nsISupports* outer, const nsIID& aIID, void* *aInstancePtr)
{
nsProxyObjectManager *proxyObjectManager = GetInstance();
if (proxyObjectManager == nsnull)
return NS_ERROR_OUT_OF_MEMORY;
return proxyObjectManager->QueryInterface(aIID, aInstancePtr);
}
NS_IMETHODIMP
nsProxyObjectManager::GetProxyObject(nsIEventQueue *destQueue, REFNSIID aIID, nsISupports* aObj, PRInt32 proxyType, void** aProxyObject)
{
nsIEventQueue *postQ = destQueue;
*aProxyObject = nsnull;
// post to primordial thread event queue if no queue specified
if (postQ == nsnull)
{
nsresult rv;
nsCOMPtr<nsIThread> mainIThread;
PRThread *mainThread;
// Get the primordial thread
rv = nsIThread::GetMainThread(getter_AddRefs(mainIThread));
if ( NS_FAILED( rv ) )
return NS_ERROR_UNEXPECTED;
rv = mainIThread->GetPRThread(&mainThread);
if ( NS_FAILED( rv ) )
return NS_ERROR_UNEXPECTED;
NS_WITH_SERVICE(nsIEventQueueService, eventQService, kEventQueueServiceCID, &rv);
if ( NS_FAILED( rv ) )
return NS_ERROR_UNEXPECTED;
rv = eventQService->GetThreadEventQueue(mainThread, &postQ);
if ( NS_FAILED( rv ) )
return NS_ERROR_UNEXPECTED;
}
// check to see if the eventQ is on our thread. If so, just return the real object.
if (postQ != nsnull && !(proxyType & PROXY_ASYNC) && !(proxyType & PROXY_ALWAYS))
{
PRBool aResult;
postQ->IsQueueOnCurrentThread(&aResult);
if (aResult)
{
return aObj->QueryInterface(aIID, aProxyObject);
}
}
// check to see if proxy is there or not.
*aProxyObject = nsProxyEventObject::GetNewOrUsedProxy(postQ, proxyType, aObj, aIID);
if (*aProxyObject != nsnull)
{
return NS_OK;
}
return NS_ERROR_NO_INTERFACE; //fix error code?
}
NS_IMETHODIMP
nsProxyObjectManager::GetProxyObject(nsIEventQueue *destQueue,
const nsCID &aClass,
nsISupports *aDelegate,
const nsIID &aIID,
PRInt32 proxyType,
void** aProxyObject)
{
*aProxyObject = nsnull;
// 1. Create a proxy for creating an instance on another thread.
nsIProxyCreateInstance* ciProxy = nsnull;
nsProxyCreateInstance* ciObject = new nsProxyCreateInstance();
if (ciObject == nsnull)
return NS_ERROR_NULL_POINTER;
nsresult rv = GetProxyObject(destQueue, nsIProxyCreateInstance::GetIID(), ciObject, PROXY_SYNC, (void**)&ciProxy);
if (NS_FAILED(rv))
{
delete ciObject;
return rv;
}
// 2. now create a new instance of the request object via our proxy.
nsISupports* aObj;
rv = ciProxy->CreateInstanceByIID(aClass,
aDelegate,
aIID,
(void**)&aObj);
// 3. Delete the create instance proxy and its real object.
NS_RELEASE(ciProxy);
delete ciObject;
ciObject = nsnull;
// 4. Check to see if creating the requested instance failed.
if ( NS_FAILED(rv))
{
return rv;
}
// 5. Now create a proxy object for the requested object.
rv = GetProxyObject(destQueue, aIID, aObj, proxyType, aProxyObject);
// 6. release ownership of aObj so that aProxyObject owns it.
NS_RELEASE(aObj);
// 7. return the error returned from GetProxyObject. Either way, we our out of here.
return rv;
}
nsHashtable*
nsProxyObjectManager::GetRealObjectToProxyObjectMap()
{
NS_VERIFY(mProxyObjectMap, "no hashtable");
return mProxyObjectMap;
}
nsHashtable*
nsProxyObjectManager::GetIIDToProxyClassMap()
{
NS_VERIFY(mProxyClassMap, "no hashtable");
return mProxyClassMap;
}
/////////////////////////////////////////////////////////////////////////
// nsProxyEventFactory
/////////////////////////////////////////////////////////////////////////
nsProxyEventFactory::nsProxyEventFactory(void)
{
NS_INIT_REFCNT();
}
nsProxyEventFactory::~nsProxyEventFactory(void)
{
}
NS_IMPL_ISUPPORTS1(nsProxyEventFactory,nsIFactory)
NS_IMETHODIMP
nsProxyEventFactory::CreateInstance(nsISupports *aOuter, REFNSIID aIID, void **aResult)
{
if (aResult == nsnull)
{
return NS_ERROR_NULL_POINTER;
}
*aResult = nsnull;
nsProxyObjectManager *inst = nsProxyObjectManager::GetInstance();
if (inst == nsnull)
return NS_ERROR_OUT_OF_MEMORY;
nsresult result = inst->QueryInterface(aIID, aResult);
if (NS_FAILED(result))
{
*aResult = nsnull;
}
NS_ADDREF(inst); // Are we sure that we need to addref???
return result;
}
NS_IMETHODIMP
nsProxyEventFactory::LockFactory(PRBool aLock)
{
// not implemented.
return NS_ERROR_NOT_IMPLEMENTED;
}
////////////////////////////////////////////////////////////////////////////////
// DLL Entry Points:
////////////////////////////////////////////////////////////////////////////////
static NS_DEFINE_CID(kProxyObjectManagerCID, NS_PROXYEVENT_MANAGER_CID);
static NS_DEFINE_CID(kComponentManagerCID, NS_COMPONENTMANAGER_CID);
extern "C" NS_EXPORT PRBool
NSCanUnload(nsISupports* aServMgr)
{
return 0;
}
extern "C" NS_EXPORT nsresult
NSRegisterSelf(nsISupports* aServMgr, const char *path)
{
nsresult rv;
nsCOMPtr<nsIServiceManager> servMgr(do_QueryInterface(aServMgr, &rv));
if (NS_FAILED(rv)) return rv;
nsIComponentManager* compMgr;
rv = servMgr->GetService(kComponentManagerCID,
nsIComponentManager::GetIID(),
(nsISupports**)&compMgr);
if (NS_FAILED(rv)) return rv;
#ifdef NS_DEBUG
printf("*** nsProxyObjectManager is being registered. Hold on to your seat...\n");
#endif
rv = compMgr->RegisterComponent(kProxyObjectManagerCID, nsnull, nsnull, path, PR_TRUE, PR_TRUE);
if (NS_FAILED(rv)) goto done;
done:
(void)servMgr->ReleaseService(kComponentManagerCID, compMgr);
return rv;
}
extern "C" NS_EXPORT nsresult
NSUnregisterSelf(nsISupports* aServMgr, const char *path)
{
nsresult rv;
nsCOMPtr<nsIServiceManager> servMgr(do_QueryInterface(aServMgr, &rv));
if (NS_FAILED(rv)) return rv;
nsIComponentManager* compMgr;
rv = servMgr->GetService(kComponentManagerCID,
nsIComponentManager::GetIID(),
(nsISupports**)&compMgr);
if (NS_FAILED(rv)) return rv;
#ifdef NS_DEBUG
printf("*** nsProxyObjectManager is being unregistered. Na na na na hey hey\n");
#endif
rv = compMgr->UnregisterComponent(kProxyObjectManagerCID, path);
if (NS_FAILED(rv)) goto done;
done:
(void)servMgr->ReleaseService(kComponentManagerCID, compMgr);
return rv;
}
extern "C" NS_EXPORT nsresult
NSGetFactory(nsISupports* aServMgr,
const nsCID &aClass,
const char *aClassName,
const char *aProgID,
nsIFactory **aFactory)
{
if (aFactory == nsnull)
{
return NS_ERROR_NULL_POINTER;
}
*aFactory = nsnull;
nsISupports *inst;
if (aClass.Equals(kProxyObjectManagerCID) )
inst = new nsProxyEventFactory();
else
return NS_ERROR_ILLEGAL_VALUE;
if (inst == nsnull)
return NS_ERROR_OUT_OF_MEMORY;
nsresult res = inst->QueryInterface(nsIFactory::GetIID(), (void**) aFactory);
if (NS_FAILED(res))
{
delete inst;
}
return res;
}