Freigeben über


Do Not Call IsBadFooPtr, Indeed

Here’s a story that I said a long time ago that I was going to tell you all, and then promptly forgot about it. Raymond Chen’s blog entry today reminded me of it, because this is the story of how I found out the hard way that IsBadFooPtr is bad, bad, bad. Those of you who have been following releases of the script engines incredibly carefully may remember this story, but hopefully we caused and then fixed the problem sufficiently quickly that the vast majority of customers never noticed.

But I’m getting ahead of myself.

One of the most basic and important rules of COM programming is that you never, ever, EVER change an interface without changing its GUID. Or, to put it another way, you simply never change an interface; if you need to, you create an entirely new interface.

You may have wondered why is COM so strict about this rule. It’s because when you write a C++ program that uses an interface, the compiler will generate code that depends upon the particular pattern of stack frame layouts for passing arguments remaining exactly the same even when the user installs a new version of the COM object on their machine. Because COM objects and their “clients” can be versioned independently, the GUID has to be associated with a particular “binary contract” describing how the two pieces of code communicate.

Now, the script engines always talk to scriptable objects via IDispatch or IDispatchEx, so if a scriptable object accidentally breaks its contract, the script engines don’t notice a bit. They let the dispatch layer take care of all those irksome details about stack frame layout, and the dispatch layer figures all this stuff out from the type library data in the object itself.

Perhaps you see where this is going.

The Replace method of the VBScript regular expression object originally took two “in” arguments, both strings, and returned a string. This was the IRegExp interface. Later we realized that we wanted to be able to pass things other than strings for the second argument, so we created a new IRegExp2 interface where the Replace method took a string and a variant, and returned a string. So far, so good.

Then we ported the script engines to the IA64 processor. During the port for some unknown reason one of the developers temporarily changed the second argument from a VARIANT to a VARIANT*, put a comment in the code saying that this needed to be changed back before it was checked in, checked it in, and, uh, we kinda shipped it like that.

We thoroughly violated the rules of COM, but none of our automated tests showed this, because of course, all of the scripting automated tests were written – you guessed it – in script. We had no tests which used the scriptable objects from early-bound code. (We do now!)

Unfortunately, some of our developer customers were using the VBScript regular expression object from C++ and other early bound languages, so as soon as their customers upgraded to the latest version of scripting, all hell broke loose. The caller was fulfilling their end of the contract by putting a sixteen byte variant on the stack, and the callee was suddenly expecting a four byte pointer to a variant, so it was both crashing and misaligning the stack.

We immediately started getting bug reports, of course. About half the bug reports said “please change this back ASAP, our customers are broken”, and the other half said “we have already shipped a patch of our code to our customers using the new stack layout, please do not change it back, ever, we do not want to ship another patch”.

Obviously we realized that we had badly screwed up and were now in a hideous bind. We came up with three options:

First, we could revert to the previous layout, and inconvenience the small number of customers who had already shipped patches to their customers. (And of course, inconvenience all of those people as well, who would have to apply a second patch in as many days.)

Second, we could keep the current layout and tell the rest of our affected customers that they needed to patch our mistake.

Third, we could revert to the previous layout and yet NOT inconvenience anyone.

Obviously the third option is best, but how to do it?

I ended up writing heuristics into the Replace method that examined the stack frame, and then rerouted the call to a special helper method if we detected a VARIANT* on the stack when we were expecting a VARIANT. The helper method did all the necessary magic to adjust the stack pointer, and so on, so that from the caller’s perspective, everything looked perfectly normal.

Now, remember, we expected that in the vast majority of cases, a VARIANT would be passed, not a VARIANT*. My heuristic detected which case we were in by extracting four bytes from the middle of the stack frame and calling IsBadReadPtr on it. If that was a valid pointer then I’d cast it to a VARIANT*, check to see whether the variant type field was a string (by far the most likely argument to Replace), and then call IsBadWritePtr on the bstrVal (because BSTR pointers are almost always writeable). There were a few other details to the heuristic, but the point is that in the vast majority of cases, IsBadReadPtr was expected to fail. Only in the rare cases where we were in an early-bound and third-party patched program should it succeed.

We shipped the fixed version with the heuristics out immediately.

At the time I did not realize how IsBadReadPtr was implemented. It actually puts a try-except around an attempt to read the pointer! If it gets an exception then it eats the exception and says yeah, it's bad. Essentially what I had done was inserted an almost-always-thrown exception into every call to Replace.

My goal of not inconveniencing anyone turned out to not quite work out as planned. The Active Server Pages team went nuts that afternoon. Their tests assume that all exceptions, whether handled or not, are flaws somewhere in ASP or scripting. Their test machines are all configured to track and report all first-chance exceptions, and suddenly every compatibility and stress test machine they had that happened to do a regular expression replace – that is, approximately all of them – suddenly started reporting hundreds of first-chance exceptions per minute when they upgraded to the latest released version.

So just when I thought that I could breathe easy again for a few minutes I started getting calls from the now highly vexed ASP team. I ended up writing my own versions of IsBadFooPtr which do the right thing – they call VirtualQuery to ask the operating system what the protection state of a given address is, rather than simply trying it out and throwing a first-chance exception on failure.

Long story short: don’t change an interface, don’t call IsBadFooPtr, and run your changes by all your important consumers before you ship a tricky fix!

Comments

  • Anonymous
    September 27, 2006
    Um, what happens if you pass a variant with a value that happens to point to a real address?

  • Anonymous
    September 27, 2006
    As I said, the heuristic is quite a bit more complicated than I indicated here.  We check to see whether all the fields of the purported VARIANT make sense as a VARIANT.  We check to see whether it can also be interpreted as a sensible VARIANT* followed by a BSTR*. If it can, then we check all the contents of the VARIANT* to see if they look like a sensible variant.

    The odds of the first eight bytes of a VARIANT containing two valid pointers, one to something that looks like a VARIANT and one to something that looks like a BSTR are VERY low.  Should a miracle happen, then yeah, it will crash and die horribly due to the stack misalignment.

  • Anonymous
    September 27, 2006
    So if there's a sane version of IsBadFooPtr around, why are you still warning against the bad versions, rather then replacing them with good versions?

  • Anonymous
    September 28, 2006
    That's an excellent question that I do not know the answer to.  A likely guess is that there are existing programs which depend upon the implementation details. You might be surprised at the number of windows programs which depend on implementation details!

  • Anonymous
    September 28, 2006
    What happens in the case of a pointer on 9x that has the virtual page protection xset to PAGE_NOACCESS with an exception handler waiting around for the exception so it can populate the page on demand?  Similar to PAGE_GUARD, which is documented with the suggestion to use PAGE_NOACCESS on 9x.

    I'm assuming on 9x automatic stack checking is implemented with PAGE_NOACCESS...

  • Anonymous
    September 28, 2006
    OK, so the scenario you are describing is:

    * someone is calling a broken version of the IRegExp2 interface
    * the caller has patched around the break the day that the break happened, and shipped the patched version to customers.
    * the caller is using Windows 95/98/ME.
    * the caller is passing a variant* to a string, but the variant* has been allocated on a page with no access. Somewhere, an exception handler lurks to fill in the page with the legal value of the contents of the variant.

    In that INCREDIBLY UNLIKELY and COMPLETELY CONTRIVED scenario, yes, we will probably crash from the stack misalignment, if not earlier.

    It's a heuristic. There are bizarre and unlikely situations which will fool the heuristic.  That's OK.  The heuristic is designed to turn rare but known-to-exist crashes into "incredibly low likelihood of crash", not into "no possible chance of crash". It's pragmatic, not perfect.

  • Anonymous
    September 28, 2006
    The comment has been removed

  • Anonymous
    September 28, 2006
    Welcome to the fifth Community Convergence update. This file is published on my blog, and shortly thereafter...

  • Anonymous
    September 29, 2006
    Thanks Eric - interesting and funny

  • Anonymous
    October 09, 2006
    Learn Videos and Presentations The LINQ Framework: What's New in the May CTP Anders: Chatting about LINQ

  • Anonymous
    October 16, 2006
    Actually, MSDN has an example of a dynamically committed block of memory (http://msdn.microsoft.com/library/default.asp?url=/library/en-us/memory/base/reserving_and_committing_memory.asp) that, I assume, wouldn't dynamically commit with IsBadFooPtr as the virtual page is allocated with PAGE_NOACCES resulting, again assuming, in an E_POINTER or E_INVALIDARG when in fact the pointer would have been valid.