The MASM Forum Archive 2004 to 2012

Miscellaneous Forums => Miscellaneous Projects => Topic started by: MichaelW on April 04, 2011, 11:36:10 AM

Title: Self-Expanding Buffer
Post by: MichaelW on April 04, 2011, 11:36:10 AM
The attachment is a first attempt that appears to work OK.

Note that this will have the same problem with overstepping the guard page as the stack does. You must limit the increase in buffer size for any single access to the size of a page (normally 4096 bytes).
Title: Re: Self-Expanding Buffer
Post by: Biterider on April 04, 2011, 06:46:21 PM
Hi Michael
Good approach but I think see a problem. What happens when you have more than 1 such a buffer? How does the exception filter know what buffer raised the exception?

Biterider
Title: Re: Self-Expanding Buffer
Post by: MichaelW on April 05, 2011, 04:49:38 AM
Hi Biterider,

Thanks, I had not considered supporting more than one buffer. I think this would a worthwhile feature, and I think I see a reasonable way to do.
Title: Re: Self-Expanding Buffer
Post by: Biterider on April 08, 2011, 04:31:26 PM
Hi Michael
Did you make any progress on the exception code to identify the buffer that needs more memory?
If you have an idea, we can elaborate it...

Regards,

Biterider
Title: Re: Self-Expanding Buffer
Post by: MichaelW on April 08, 2011, 05:27:16 PM
Hi Biterider,

Yes, I did. I have what I think is a workable plan that will support multiple buffers in multiple threads. Every test I have run indicates that it will work, and while it is somewhat complex it should not be difficult to implement. The current problem is with finding the time to work on it.

Title: Re: Self-Expanding Buffer
Post by: Biterider on April 11, 2011, 06:51:08 AM
Hi Michael
I did some experiments this weekend to test some alternatives and finally found a good way to solve the exception handler problem.

There are additional information passed to the handler when a Page Guard is hit at the end of  the EXCEPTION_RECORD in the ExceptionInformation field. The second dword is a pointer to the memory location that generated the access fault.

The second trick is storing additional information on the guard page before the guard flag is set. This information can be read back by the handler since the OS resets this flag before the handler is executed.

One concern is multithreading. I put in the handler code a synchronisation mechanism to avoid the corruption of the buffer data.

Biterider

PS: new upload with stack checking for regular guard page exceptions.
Title: Re: Self-Expanding Buffer
Post by: MichaelW on April 11, 2011, 04:13:54 PM
Hi Biterider,

Thanks for the code. It made sense to me that the system would send the exception filter additional information on the exception, but when I searched I could not find any reference to this. Your solution is much simpler than mine was. Basically, since the exception filter is called in the context of the thread that caused the fault, I was going to maintain a list of active guard page addresses in TLS and on each exception get the guard page address by scanning the list for the guard page with the PAGE_GUARD modifier cleared.
Title: Re: Self-Expanding Buffer
Post by: Biterider on April 11, 2011, 08:52:30 PM
Hi Michael
http://msdn.microsoft.com/en-us/library/aa363082(v=vs.85).aspx at the end of the page gives a short explanation for the ExceptionInformation field, in particular for the EXCEPTION_ACCESS_VIOLATION. Strange that the EXCEPTION_GUARD_PAGE isn't mentioned, but it behaves in the same way.

Biterider
Title: Re: Self-Expanding Buffer
Post by: MichaelW on April 11, 2011, 11:13:46 PM
Quotehttp://msdn.microsoft.com/en-us/library/aa363082(VS.85).aspx

Thanks, I actually looked at that page, but I was looking at the ExceptionAddress member and not at the details for EXCEPTION_ACCESS_VIOLATION.
Title: Re: Self-Expanding Buffer
Post by: Biterider on April 12, 2011, 08:55:01 AM
Hi
Looking at the UnhandledExceptionHandler procedure, I'm unsure if the classic ebx,edi,esi,etc registers should be preserved or not. I know that callback procedures must do it, but in this particular case, the procedure is not defined as a callback. A simple test doesn't change the behaviour on WinXP, but I don't know what happens on different Windows versions. Any advice?

Biterider
Title: Re: Self-Expanding Buffer
Post by: MichaelW on April 13, 2011, 01:17:53 AM
To do any sort of conclusive test you would need to have access to a wide variety of Windows versions and builds.

It seems to me that the filter procedure is a callback in that the system is calling it. I can't see any reason why the register-preservation conventions would change for a filter procedure, or considering how seldom the procedure is called any significant performance advantage in not preserving the registers. And for what it's worth, Matt Pietrek refers to an exception handler as a "user-defined callback function"  here (http://www.microsoft.com/msj/0197/Exception/Exception.aspx).
Title: Re: Self-Expanding Buffer
Post by: Biterider on April 13, 2011, 05:51:15 AM
Hi Michael
Right, I found it
QuoteMore specifically, when a thread faults, the operating system calls a user-defined callback function. This callback function can do pretty much whatever it wants.
Thanks!  :U

There is another issue a friend of mine (Obivan) told me yesterday. The main goal for this self expanding buffer is safety. What happens when we reach the end of our buffer, we commit the last guard page and the following page was committed before and contains data. We would write beyond the end of the buffer, corrupting other things.
A solution for this problem is to keep the last buffer page as a guard page and raise an exception when this is hit.

I'll modify the code to introduce this last change.

Regards,

Biterider
Title: Re: Self-Expanding Buffer
Post by: Biterider on April 22, 2011, 03:47:37 PM
Hi
I updated the code again with some improvements over the old one.

Biterider