|
Right. The DllGetClassObject gets an interface to the class factory for the .dll, not any particular object in the .dll. This will then be used to handle calls to CoCreateInstance by the run time. Client code does not need to be aware of the existence of the class factory.
Here's an excerpt from the documentation for the function. In the generic ATL code generated for you in ATL COM projects, you'll get a class factory, and an implementation of DllGetClassObject. The class factory will tie together the CLSID's for all of the objects that it knows how to create to the actual implementing classes through the OBJECT_MAP.
You could do something similar in raw C++, you just need to tie together the CLSID's with the actual classes for your implementation of a class factory. The DllGetClassObject will just return an interface to the the class factory, which does all of the real work.
STDAPI DllGetClassObject(
REFCLSID rclsid, //CLSID for the class object
REFIID riid, //Reference to the identifier of the interface
// that communicates with the class object
LPVOID * ppv //Address of output variable that receives the
// interface pointer requested in riid
);
Parameters
rclsid
[in] CLSID that will associate the correct data and code.
riid
[in] Reference to the identifier of the interface that the caller is to use to communicate with the class object. Usually, this is IID_IClassFactory (defined in the OLE headers as the interface identifier for IClassFactory).
ppv
[out] Address of pointer variable that receives the interface pointer requested in riid. Upon successful return, *ppv contains the requested interface pointer. If an error occurs, the interface pointer is NULL.
|
|
|
|
|
I've been wondering about marshalling issues in the following senerio
ObjA - Is a single-threaded ATL COM object singleton
ObjA is instantiated in Application A. Sometime later Application B is started and cocreates ObjA. Since it is a singleton they both have the same instance. My questions are whether their are any marshalling issues that have to be address because ObjA was created in a different process. Would it matter if ObjA where multi-threaded.
|
|
|
|
|
Yes, you need to synchronize method and property in your object with Semaphore
and critical section (your choice...). I did the same thing with a Local server
Component and it works really fine.
|
|
|
|
|
Thanks, but my questions were about marshaling not synchronization. I don't see synchronization as an issue since the object is single-threaded. Obviously a multi-threaded object would require it.
|
|
|
|
|
I have an apartmented ATL server program running on W2K at work. Now mangement has decided to run it on a fancy, spaning new machine with 1 gig ram, dual processors, and 4 90-gig hard drives.
Question: is it possible to change to server program to take advantage of the dual-processing capabilities? If it is, how? Is there a book or white paper some place that explains how to do that ?
|
|
|
|
|
just install windows2000 adn make it recognize that the server runs on dual processor. thats it. It will take care. Yes. there are ways in win32 in which u can use the dual processor feature and control them. But then that will defeat the idea on COMponent at all, because it will becomes windows proprietary
Ganesh Ramaswamy
|
|
|
|
|
I've created an ActiveX control with VC++ 6.0, and i try to call CoInitializeEx() at the beginning of a worker thread, but it doesnt recognize the API, and all I get is an undeclared identifier error for that
and COINIT_MULTITHREADED;
I linked to "ole32.lib" and i threw in <objbase.h> in for kicks at the top of the .cpp and in a separate case, the .h
I AM LIKE ----------------| |-------------------
^
THIS CLOSE
to getting this crap... i dont have a clue... i've running the project on a different machine as well and i get the same errors. i've tried to throw in just about every other library as well... am i missing a library? Could someone show me what libraries they link to in a project that works? Any suggestions would be so helpful...
Thanks,
~Tim
SHABBA!!
|
|
|
|
|
CoInitializeEx requires either NT 4.x or greater or Win 98 (Win 95 with DCOM).
Try either #define _WIN32_DCOM or #define WIN32_WINNT 0x0400.
Michael
|
|
|
|
|
If you are using ATL, in your Stdafx.h
make sure sure have before all includes
#define STRICT
#ifndef _WIN32_WINNT
#define _WIN32_WINNT 0x0400
#endif
#define _ATL_FREE_THREADED //_ATL_APARTMENT_THREADED
Make sure your WinMain look like this, you need CLSCTX_LOCAL_SERVER
if you are using a local server.
extern "C" int WINAPI _tWinMain(HINSTANCE hInstance,
HINSTANCE , LPTSTR lpCmdLine, int )
{
lpCmdLine = GetCommandLine();
#if _WIN32_WINNT >= 0x0400 & defined(_ATL_FREE_THREADED)
HRESULT hRes = CoInitializeEx(NULL, COINIT_MULTITHREADED);
#else
HRESULT hRes = CoInitialize(NULL);
#endif
if (bRun)
{
_Module.StartMonitor();
#if _WIN32_WINNT >= 0x0400 & defined(_ATL_FREE_THREADED)
hRes = _Module.RegisterClassObjects(CLSCTX_LOCAL_SERVER,
REGCLS_MULTIPLEUSE | REGCLS_SUSPENDED);
_ASSERTE(SUCCEEDED(hRes));
hRes = CoResumeClassObjects();
#else
hRes = _Module.RegisterClassObjects(CLSCTX_LOCAL_SERVER,
REGCLS_MULTIPLEUSE);
#endif
_ASSERTE(SUCCEEDED(hRes));
MSG msg;
while (GetMessage(&msg, 0, 0, 0))
DispatchMessage(&msg);
_Module.RevokeClassObjects();
Sleep(dwPause);
}
_Module.Term();
CoUninitialize();
return nRet;
}
|
|
|
|
|
Sweet guys! that definitely fixed it.
Now on to our next question... (the next thread)
SHABBA!!
|
|
|
|
|
er... we'll do it right here...
ok... It goes through all of the many events raised, and then when it gets to the last one, it'll crash. hmm... it crashes in COleDispatchDriver::InvokeHelperV 's call to
COleDispatchDriver::Invoke... then it goes into assembly.(grr)
(it's referencing something with nothing in it (0x000005)
Has anyone had an error similar to that?
I'm pretty darn sure i'm not wiping anything out before that last event finishes... i really need this last event too...<sigh>
I hope you guys have had this happen to you as well, and just so you know, the events are sent to a VB app... i'm sure y'all knew that though. Not as much stress on me now, but it's still lingering...
Help!! Thanks, people! Learnin' from the best I guess!
~Tim
SHABBA!!
|
|
|
|
|
Happy to know that you finally made it!
The only reason I can see is that you are calling that function with
invalid arguments (why the other event function crash?)
If you were using an old client with VB, you may need to rebuild your
VB client.
Give a sample of your code, people will help you faster!
|
|
|
|
|
ok here comes the sample:
I have my MFC ActiveX Full Control (according to VC++ 6.0 Wizard).
//Which Calls:
CESpyderCtl::Start() {
//oh, and i call CoInitializeEx(NULL, COINIT_MULTITHREADED)
// in CESpyderApp::InitInstance() also, but i dont know what runs first...
// havent tried to find out. *shrug*
_beginthread(...)
//err checking
}
//Begin Thread Calls a global function called:
void gStart(void *spy) {
//which ultimately calls Parse() that fires an event pretty often
//so it can update a color box to show what it's doing.
CoInitializeEx(NULL, COINIT_MULTITHREADED); //i call this in
//CESpyderApp::InitInstance() as well.
Parse();
CoInitializeEx(NULL, COINIT_MULTITHREADED);
}
//So inside of Parse(); we've got this
int CESpyder::Parse() {
//parse a file
Fire_StatusChange(STAT_SEARCHING); //events are status changes
//connecting, adding, searching, error, and done.
//and on the last one, it's doing the same thing, and i'm firing a done event.
Fire_StatusChange(STAT_DONE);
//i also fire a separate done event, but it never gets to it, and i've also tried to just fire a done event, but it crashes the same way.
Fire_Done();
return 1;
}
I figure this still wont give you enough information, i've tried putting
Sleep()'s all over the place.., but it seems like it doesnt affect it...
<sigh>, i'm using apartment threading, if i remember correctly it wont let me pick apartment or free threaded. i think i even tried doing ATL but free was greyed out... either way i dont know how to Fire events in VC with an ATL object... (you could show me that, but i'll never be able to get rid of this thing that will eat at my soul for eternity... haha!
Thanks,
~Tim
SHABBA!!
|
|
|
|
|
Sorry if I can't help you. All I can tell you is your bug
is maybe related to MFC. I always use ATL to develop ActiveX control.
I know MFC has a macro, I think it's: AFX_MANAGE_STATE( ), where I have
seen developer using: AFX_MANAGE_STATE(AfxGetStaticModuleState( ));
but again I have never use it. Do you always fire event from the same thread?
What's the other thread is doing?
|
|
|
|
|
i tried throwing that in there to no avail, but i went and searched for firing events from worker threads with AFX_MANAGE_STATE in there as well and found some stuff, so i'm working on trying one of those out right at this moment..
do you think you could show me how to raise an event from C++ using an ATL control ? (for kicks i can try it through ATL and see how well it works then)
~Timothy T. Rymer
www.digipen.edu
tim.xpertz.com
|
|
|
|
|
Okay I will send you an example how I did something similar.
|
|
|
|
|
I made a simple demo for you. I sent it to your email at:
heaven_bounty@yahoo.com
Let me know if it helps you!
|
|
|
|
|
That's what baffles me is that it's the exact same idea except that it's in an MFC ActiveX Control...
here's my question:
1. Make a new COM ATL Project.
-I picked FULL CONTROL when i made a new ATL obj... I assume that to be fine.
-Do i want to merge proxy/stub code?
-Support Connection points, of course
-Apartment threading (it wont let you choose Free if you choose to make a
control) so... should i just be making an ATL object and wrap it in a Control?
-Do i need to check free threaded marshalling?(i assumed so)
-Aggregation: (No) i have no clue what this does... i've basically learned COM/ATL/ActiveX from common lingo and trial and error.
2. ok after all of that is done, (the real question is coming, i promise)
- I compile,
- Implement Connection points? I cannot compile to where i can call events in the code and it doesnt show up (like the way you have it in CProxy...).
that's my dilemma. oh and when i try to add a function to the CProxy_... i get: "The implementation file for class CProxy_ITesterEvents<class t=""> cannot be found" ---- what am i not doing, that i should be doing??
well, that's all for now, i'm gonna keep messing with it. RSVP
~Timothy T. Rymer
www.digipen.edu
tim.xpertz.com
|
|
|
|
|
1) Make a new COM ATL Project [ ok ]
- Picked Full .. [ ok ]
- Attributes: Apartment, Dual, Aggregation: Yes [or No] doesn't matter for now
- Connection points [x], Error Info [ ] (no, or I don't mind), Free Thread [ ] (no, I don't
mind, that's not the cause of your problem).
- Select settings for Miscellaneous, Stock Properties [ ok ]
2) Build your project, It should build fine (very important, must work)
3) Add event function to your connection point, right-click "_YourObjectEvents" and select
Add Method ( I am sure you did everything correctly so far...)
4) Verify that you have COM_INTERFACE_ENTRY(IConnectionPointContainer) in your COM_MAP
5) Implement the connection point:
- right-click your "YourObjectClass", chose "Implement Connection Point..."
- check your "_YourObjectEvents" and Click Ok
6) At that point, the proxy stub will be added for you, if you don't see it. You must have
it already on your folder. You may want to delete previous proxy stub class (files) every
time you add a new method (event) to your connection point.
7) Follow the example in the thread function that I sent you.
|
|
|
|
|
cool, ok that's what i've been doing. now after i implement the connection points i get an error:
BEGIN_CONNECTION_POINT_MAP(Ccrap)
CONNECTION_POINT_ENTRY(IID_IPropertyNotifySink)
CONNECTION_POINT_ENTRY(IID__IcrapEvents)
END_CONNECTION_POINT_MAP()
it says that IID_IcrapEvents is undeclared... any clue about that?
also, when it asked me if i wanted a .dll or exe it also had a Checkbox that asked if you wanted to merge proxy/stub code. Is that necessary?
~Timothy T. Rymer
www.digipen.edu
tim.xpertz.com
|
|
|
|
|
MasterWang wrote:
it says that IID_IcrapEvents is undeclared... any clue about that?
You mean DIID_IcrapEvents
MasterWang wrote:
also, when it asked me if i wanted a .dll or exe it also had a Checkbox that asked if you wanted to merge proxy/stub code. Is that necessary?
No!
|
|
|
|
|
oh! wow, there's that error that i told you before, but yes "Fire_Done()" is there though (woohoo!).
~Timothy T. Rymer
www.digipen.edu
tim.xpertz.com
|
|
|
|
|
and it gives it to me as IID_Iklfasjdf, not DIID_afklj;ds
but when i changed it to DIID, we have no errors. i'm sure i'll post something shortly here on my status.
~Timothy T. Rymer
www.digipen.edu
tim.xpertz.com
|
|
|
|
|
that DIID_YOUROBJECTEvent is defined in "YourObject_i.c".
This is an auto-created file.
Glad to see you get there!
|
|
|
|
|
ok ok, good news/bad news... the good news is that it compiles and fires the events, bad news is that it still crashes...
i'm calling CoInitializeEx() and CoUninit
at the beginning/end of the worker thread,
still testin' and messin' with it.
anything that i forgot about that i should for sure have in there?..
~Timothy T. Rymer
www.digipen.edu
tim.xpertz.com
|
|
|
|