Valhalla Legends Archive

Programming => General Programming => Visual Basic Programming => Topic started by: Adron on November 30, 2003, 06:20 AM

Title: VB error handling
Post by: Adron on November 30, 2003, 06:20 AM
I'm copying this post here, to see if I can get any responses from some of the skilled VB programmers here...

Quote from: CupHead on November 26, 2003, 01:00 PM
4) Exception Handling

VB has excellent error handling in my opinion.  You can not only create your own errors, but you can raise them whenever you like, be notified of what error it is, what line caused it, and even return control to the program after your code has adjusted.  We all know about On Error Resume Next, which is essentially a catch-all that keeps your programs from crashing.  Well, you can actually do:

I don't like VB's error handling at all for one big reason: Generic error handling is as far as I know impossible to implement without copy-pasting code.

That is: If I want to handle every error that can possibly happen and do some kind of clean shutdown, I would have to write an On Error statement, a label and handler (or call to a generic handler) into nearly every event handler of every object. (i.e. command1_mousedown, command1_click, command1_mousemove, command1_ ... etc)

I have been able to find no way to create a generic exception handler that is global to the application.

What's worse, VB doesn't even let any errors happening in such functions pass on to a possibly attached debugger, or to the operating system. You get a stupid "Run-time error X: Bla bla bla" box with an OK button instead of invoking Dr.Watson to create a dump file that could be debugged.

Title: Re:VB error handling
Post by: Grok on November 30, 2003, 11:26 AM
The only way I know to do what you ask is to construct your application in specific ways.  However, writing by using control-structure stacking and nesting at the procedure level is sufficient only for your own code.  Objects whose code is written by VB, such as the GUI windows and controls, do not raise their events to your handler.  I don't know a way to do that.
Title: Re:VB error handling
Post by: TheMinistered on November 30, 2003, 11:52 AM
This might be going a bit far, but you could do some patching perhaps?  I know that, in every function, it pushes the address of vbaExceptionHandler on the stack for error handling I believe...


* Reference To: MSVBVM60.__vbaExceptHandler, Ord:0000
:00401096 FF253C104000    Jmp dword ptr [0040103Ch]


Perhaps try patching the jump table to a function in a module?  I'm not sure... it's just a long shot at an answer.
Title: Re:VB error handling
Post by: Banana fanna fo fanna on November 30, 2003, 12:00 PM
Quote from: TheMinistered on November 30, 2003, 11:52 AM
This might be going a bit far, but you could do some patching perhaps?  I know that, in every function, it pushes the address of vbaExceptionHandler on the stack for error handling I believe...


* Reference To: MSVBVM60.__vbaExceptHandler, Ord:0000
:00401096 FF253C104000    Jmp dword ptr [0040103Ch]


Perhaps try patching the jump table to a function in a module?  I'm not sure... it's just a long shot at an answer.

Ew? :)
Title: Re:VB error handling
Post by: Adron on December 01, 2003, 11:46 AM
Quote from: Grok on November 30, 2003, 11:26 AM
The only way I know to do what you ask is to construct your application in specific ways.  However, writing by using control-structure stacking and nesting at the procedure level is sufficient only for your own code.  Objects whose code is written by VB, such as the GUI windows and controls, do not raise their events to your handler.  I don't know a way to do that.

Well, and VB being the language it is, it's very well suited for applications that have a lot of GUI interaction and not so much processing. A very large part of the code ends up GUI code, making most Visual Basic applications I write have a very large number of events called by VB/Windows.

It's rather unfortunate. Like you said, if most of the calls originated in a "main" or "WinMain" function like in C++, I could just add a global error handling there, but as it is now, there has to be handlers spread out all over the place.
Title: Re:VB error handling
Post by: Adron on December 01, 2003, 11:47 AM
Quote from: TheMinistered on November 30, 2003, 11:52 AM
This might be going a bit far, but you could do some patching perhaps?  I know that, in every function, it pushes the address of vbaExceptionHandler on the stack for error handling I believe...


* Reference To: MSVBVM60.__vbaExceptHandler, Ord:0000
:00401096 FF253C104000    Jmp dword ptr [0040103Ch]


Perhaps try patching the jump table to a function in a module?  I'm not sure... it's just a long shot at an answer.

I've been working along that path a long time ago. What I ended up having trouble with was determining whether a particular error was handled by the local code or would result in a VB run-time error message box popping up.

I still want to be able to write specific error handling for errors that I foresee and know what to do about; it's just the unknown ones (presumably bugs) that I want a catch-all for to provide me with good information, like a stack-trace.

It seems rather complex though, perhaps requiring a DLL written in C++ to get the calling conventions and register preserving right.
Title: Re:VB error handling
Post by: Banana fanna fo fanna on December 01, 2003, 03:08 PM
I've found an extremely good strategy is to have VB communicate with a backend app, either through DLLs or XML-RPC.
Title: Re:VB error handling
Post by: Skywing on December 01, 2003, 03:17 PM
Quote from: St0rm.iD on December 01, 2003, 03:08 PM
I've found an extremely good strategy is to have VB communicate with a backend app, either through DLLs or XML-RPC.
This isn't an option for an already-written app, and this probably won't be an option for an app with strict design requirements.
Title: Re:VB error handling
Post by: TheMinistered on December 02, 2003, 03:24 PM
Quote
... determining whether a particular error was handled by the local code or would result in a VB run-time error message box popping up.

What do you mean?  The code can be handled by a generic exception handler like so:

// push generic exception handler jump table address on the stack and setup the SEH
004019F6 6896104000 push 00401096


The code for local error handling is a little more complex, but it still uses a generic exception handler.  I would poke around a few of the following procs: vbaOnError, vbaExitProc (this appears to be used in some local error handling), vbaErrorOverflow, vbaFPException
Title: Re:VB error handling
Post by: Adron on December 02, 2003, 05:29 PM
What I mean is that I want to be able to use local error handling inside functions.

Stupid example:

On Error Resume Next
CommitTrans
If Err = 0 Then MsgBox "Error: Missing CommitTrans detected!"
On Error Goto 0/somehandler/whatever to restore behaviour


The global error handling shouldn't interfere with that. And it shouldn't require writing code in every event handler called from VB/Windows. I don't want to replace the coded VB error handling, only the implicit VB error handling of errors that don't have a specific handler written for them.
Title: Re:VB error handling
Post by: Adron on December 02, 2003, 05:30 PM
Quote from: St0rm.iD on December 01, 2003, 03:08 PM
I've found an extremely good strategy is to have VB communicate with a backend app, either through DLLs or XML-RPC.

How does this help? Don't you still have to write error handling code in each and every event called by VB/Windows?
Title: Re:VB error handling
Post by: Banana fanna fo fanna on December 03, 2003, 01:56 PM
Yeah, but you have less subs to write error handlers for.
Title: Re:VB error handling
Post by: Grok on December 11, 2003, 10:01 AM
YAY @ Adron @ vbdump
Title: Re:VB error handling
Post by: CupHead on December 11, 2003, 12:23 PM
Just sort of an afterthought.  VB's error handing can sometimes be useful and other times be painful.  I don't know how VB compilations work entirely (i.e. if they inline all functions), but you have to bear in mind that errors cascade from the caller so if you do something like this:


Private Sub Function1()
 On Error GoTo ErrorHandler

 Call Function2()

 Exit Sub

ErrorHandler:
 Select Case Err.Number
   Case Else
     Debug.Print "Error: (" & Err.Number & ") " & Err.Description
 End Select
End Sub

Private Sub Function2()
 Dim strWhatever As String
 strWhatever = "Test"
 strWhatever = Mid(strWhatever, 0, 1)
End Sub


... the error handling code from Function1 is raised   Anyway, not entirely relevant, but I just thought I'd mention it.
Title: Re:VB error handling
Post by: DarkVirus on December 11, 2003, 04:12 PM
Adron, perhaps switch to VB.Net and try out their Try...Catch error system?  It's really nice for catching those forseeable or unforseeable bugs while using about 1/2 the code as you would on On Error systems used in VB 6.0.
Title: Re:VB error handling
Post by: Adron on December 11, 2003, 06:35 PM
try/catch exists in C++ too, but I'm not switching languages. I have however written a little DLL which will capture unhandled VB errors and write a minidump. If anyone's interested, post, and I might publish.
Title: Re:VB error handling
Post by: Grok on December 11, 2003, 07:22 PM
Quote from: Adron on December 11, 2003, 06:35 PM
try/catch exists in C++ too, but I'm not switching languages. I have however written a little DLL which will capture unhandled VB errors and write a minidump. If anyone's interested, post, and I might publish.

Hey I wrote one too, but mine is copyrighted.  Patent pending.  My lawyer is sending a C&D letter in the morning.  :P
Title: Re:VB error handling
Post by: Skywing on December 12, 2003, 01:27 AM
Quote from: DarkVirus on December 11, 2003, 04:12 PM
Adron, perhaps switch to VB.Net and try out their Try...Catch error system?  It's really nice for catching those forseeable or unforseeable bugs while using about 1/2 the code as you would on On Error systems used in VB 6.0.
You can't just switch a production system like that, especially one worked on by a team of programmers.  Customers won't like the added dependencies suddenly appearing without warning, either.
Title: Re:VB error handling
Post by: DarkVirus on December 12, 2003, 07:14 AM
Quote from: Skywing on December 12, 2003, 01:27 AM
Quote from: DarkVirus on December 11, 2003, 04:12 PM
Adron, perhaps switch to VB.Net and try out their Try...Catch error system?  It's really nice for catching those forseeable or unforseeable bugs while using about 1/2 the code as you would on On Error systems used in VB 6.0.
You can't just switch a production system like that, especially one worked on by a team of programmers.  Customers won't like the added dependencies suddenly appearing without warning, either.

I also wasn't aware of the context of what the program was even about or who was creating it. I thought it was a general topic about error handling. I also wasn't aware that Try...Catch was available in C++.
Title: Re:VB error handling
Post by: Adron on December 12, 2003, 01:44 PM
Well, the question was asked with a particular project in mind, which happens to be coded in VB (non-.NET). I would've preferred if it was coded in C++, which has better error handling in addition to everything else, but that would take too much time. So, I'm stuck with making the best of it!

At least it's not so difficult to write parts (like VBDump.dll) in C++ and call them from the VB parts.
Title: Re:VB error handling
Post by: Adron on December 12, 2003, 03:40 PM
Now, the wonderful vbdump Attach function returns a BOOL.

True - vbdump is attached
False - vbdump failed to attach

or

-1 - vbdump was already attached!
Title: Re:VB error handling
Post by: Skywing on December 12, 2003, 05:34 PM
Quote from: Adron on December 12, 2003, 03:40 PM
Now, the wonderful vbdump Attach function returns a BOOL.

True - vbdump is attached
False - vbdump failed to attach

or

-1 - vbdump was already attached!
Ah, so it's another functions that actually returns a Troolean, like GetMessage.  Just what the world needs...
Title: Re:VB error handling
Post by: Adron on December 13, 2003, 05:04 AM
Yup!

But, it actually returns a "true", non-zero, value whenever it ends up being attached after call. Whether it was already attached or it attached during this call.
Title: Re:VB error handling
Post by: Skywing on December 13, 2003, 12:16 PM
Quote from: Adron on December 13, 2003, 05:04 AM
Yup!

But, it actually returns a "true", non-zero, value whenever it ends up being attached after call. Whether it was already attached or it attached during this call.
I propose that we create a new type, TROOL, for this purpose.
#define TRUE 1
#define FALSE 0
#define MAYBE -1
Title: Re:VB error handling
Post by: CupHead on December 13, 2003, 01:46 PM
That's such a 2-bit data type.  *cough*
Title: Re:VB error handling
Post by: Adron on December 13, 2003, 01:51 PM
I'm currently working on the next version with non-fixed patch offsets for the messagebox in msvbvm*

Will be doing a search through the address space if I can't find a match at the known possible locations.