Jayesh
Jayesh

Reputation: 3951

Unbalanced Stack!

I have written a VC++ dll. The declaration for one of the methods in the dll is as follows:

extern "C" _declspec(dllexport)
void startIt(int number)
{
     capture = cvCaptureFromCAM(number);
}

I use this dll in a C# code using P/Invoke. I make the declaration as:

[DllImport("Tracking.dll", EntryPoint = "startIt")]
        public extern static void startIt(int number);

and I call the function in the code as:

startIt(0);

Now, when this line is encountered, the compiler is throwing me this error:

A call to PInvoke function 'UsingTracking!UsingTracking.Form1::startIt' has 
unbalanced the stack. This is likely because the managed PInvoke signature does 
not match the unmanaged target signature. Check that the calling convention 
and parameters of the PInvoke signature match the target unmanaged signature.

I cannot understand why is it throwing this error as the signature in both managed and unmanaged code are the same. Moreover, in my another machine, the same code is running perfectly in visual studio. So, this makes me think that the error thrown is mis leading.

Please help.

Thanks

Upvotes: 7

Views: 3553

Answers (3)

John Wigger
John Wigger

Reputation: 904

Constantin and Frederic Hamidi have answered this question correctly as to how to fix this problem. This can help avoid an eventual stack overflow. I have bitten by this several times myself. What is really at play here is that .NET 4 has enabled a managed debugging assistant for debug (not release) builds on 32 bit x86 machines (not 64 bit ) that checks for an incorrectly specified p/invoke call. This MSDN article details this : http://msdn.microsoft.com/en-us/library/0htdy0k3.aspx. Stephen Cleary deserves the credit for identifying this on this post: pinvokestackimbalance -- how can I fix this or turn it off?

Upvotes: 4

Frédéric Hamidi
Frédéric Hamidi

Reputation: 262969

When you p/invoke an external function, the calling convention used defaults to __stdcall. Since your function uses the __cdecl convention, you need to declare it as such:

[DllImport("Tracking.dll", EntryPoint = "startIt",
    CallingConvention = CallingConvention.Cdecl)]
public extern static void startIt(int number);

Upvotes: 14

Constantin
Constantin

Reputation: 28164

Could you be missing CallingConvention=CallingConvention.Cdecl in your DllImport attribute?

Upvotes: 6

Related Questions