Re: Is this the Right way TO release COM interface ?
<DerTopper@web.de> wrote in message
news:842267cf-6294-4cc5-859e-3a4f609933e7@d7g2000prl.googlegroups.com...
Ben Voigt [C++ MVP] wrote:
What function prototype(s) would you use for an API that can be used
to
retrieve a registry value that controls debug logging where
non-existance
of
the value is a normal condition? Non-existance of the value must be
distinguished from every possible value. The function will be called
quite
frequently so that a user can control debug output without restarting
the
program, so using exceptions is contra-indicated.
RegOpenKeyEx will serve just fine. Note that my version of
RegOpenKeyEx as a COM method would look like this:
enum RegOpenKeyExErrorCode
{
SUCCESSFUL,
KEY_DOES_NOT_EXIST
};
HRESULT RegOpenKeyEx
(
[in] HKEY hKey,
[in] BSTR SubKey,
[in]DWORD ulOptions,
[in] REGSAM samDesired,
[out] PHKEY phkResult,
[out, retval] RegOpenKeyExErrorCode** ErrorCode
);
The error code would then contain any error codes that may be of
interest to the user (IOW the programmer), everything else
(communication failure, insufficient rights, etc.) should be set
inside the HRESULT of the function, indicating some kind of
exceptional execution.
Ben Voigt [C++ MVP] wrote:
Well this is rather absurd. Of course a COM version of RegOpenKeyEx
should
yield a COM interface, not an HKEY. Other parameters would need to have
types adjusted as well, to be Automation-friendly.
I just left this out since I only wanted to demonstrate the way of
handling error conditions.
Ok, good enough. I certainly didn't expect you to define the interface in
IDL or anything like that.
But look, you're right back to checking an error code. We need to also
add
InsufficientPermission to your enum to meet the requirement of not
throwing
exceptions to handle non-exceptional conditions. By this time the
HRESULT
is getting nearly useless, so why bother with it at all? Then let's just
rename your RegOpenKeyExErrorCode to HRESULT and...
You also failed to solve the problem. You've broken the request into two
separate steps (open the key, checking for non-existance) and (read the
value within the key). But success in the first is no guarantee of
success
in the second, even under a single tasking system,
Huh? Can Windows delete the registry key by itself? Or do you mean a
driver that gets scheduled inbetween and deletes the registry key?
I meant that the key can exist and have no values inside.
and with Windows there's
a race condition as well.
Right. Probably there are some applications that use the registry as a
kind of playground for messing with asynchronous notificitions, but I
would consider those apps as deeply flawed. In my experience a
Absolutely.
registry key stays where it is, unless, of course, someone installs or
uninstalls the application while executing it. This scenario is so
obstruse that in my eyes we don't have to have an API that can check
for such cases.
You could probably roll it into a generic error message (can't read value
HKCU\Software\whatever\XYZZY) but I wouldn't want to cause an access
violation.
But I think in real life, this is not so uncommon as we would hope. An
application which installs patches during startup (Like Mozilla Seamonkey,
for example) might easily get launched a second time while the patcher is
still running.
And the use case I mentioned, controlling the level of debug output, isn't
likely to be changed by very many users, but the ones who use it are very
likely to try changing it while the process is running. I had even said "a
user can control debug output without restarting the program".
That is probably where we have different views: In my eyes the HRESULT
should only ever return the following values: S_OK, E_UNEXPECTED (an
exception has occured), or one of those error codes that have to do
with COM as a remoting architecture (for example server has died).
Oh, so you have no use for E_OUTOFMEMORY, E_ACCESSDENIED, or any of the
hundreds of other non-remoting errors that Microsoft had defined? I pity
the user that has to debug a failure with absolutely no information on
what
went wrong. Also E_UNEXPECTED isn't the catch all, that's E_FAIL.
E_UNEXPECTED is a run-time assertion failure, the caller messed up by
calling functions out of order or something like that.
I would like to have errors like E_OUTOFMEMORY or E_ACCESSDENIED to be
put into the exception information that will be retrieved through
COM's GetErrorInfo function: just the same as catching std::bad_alloc
when you use operator new under C++. Of course, catching exceptions is
sometimes more cumbersome than just retrieving a error code, and even
requires more knowledge of a programming language.
There are times when exceptions have better performance, and times when
return codes have better performance. Usually the rule of thumb is to use
exceptions to handle system failure, and return codes to handle messed up
user input and/or configuration (whether command line arguments, data files,
environment variables, registry settings, etc.)
Probably you'll say that returning two different error codes (one for
exceptional errors and one for non-exceptional errors) is a waste of
resources. However, I consider this approach as easier to wrap in
other languages (BTW, how does a VB programmer finds out that an
operation returned S_FALSE?).
How does a VB programmer use a RegOpenKeyExErrorCode*?
Well, the same way how the C++ programmer with the automatically
generated wrapper:
If RegOpenKeyEx (...) <> SUCCESSFUL Then
if (RegOpenKeyEx (...))
Type mismatch, can't find an operator<> (or != if you prefer) that takes
(RegOpenKeyExErrorCode*) and (RegOpenKeyExErrorCode) as operands.... I think
you didn't intend to have a double pointer there.
The direct test, with conversion to boolean, would have most unfortunate
results because it's comparing the pointer to NULL, and not looking at the
pointed-to enum value.
I didn't intend to quarrel with you (although I really appreciate your
arguments). It seems as if we have different views about how things
_should_ be, and your view is, sadly (for me ;-), the one that is
shared by MS.
I don't see this as a quarrel. I'm just asking you some of the difficult
edge-case questions so you'll think about which mechanism handles all cases
better, not just the 95%.
Regards,
Stuart