Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

How should I use FormatMessage() properly in C++?

Without:

  • MFC
  • ATL

How can I use FormatMessage() to get the error text for a HRESULT?

 HRESULT hresult = application.CreateInstance("Excel.Application");   if (FAILED(hresult))  {      // what should i put here to obtain a human-readable      // description of the error?      exit (hresult);  } 
like image 408
Aaron Avatar asked Jan 18 '09 16:01

Aaron


2 Answers

Here's the proper way to get an error message back from the system for an HRESULT (named hresult in this case, or you can replace it with GetLastError()):

LPTSTR errorText = NULL;  FormatMessage(    // use system message tables to retrieve error text    FORMAT_MESSAGE_FROM_SYSTEM    // allocate buffer on local heap for error text    |FORMAT_MESSAGE_ALLOCATE_BUFFER    // Important! will fail otherwise, since we're not     // (and CANNOT) pass insertion parameters    |FORMAT_MESSAGE_IGNORE_INSERTS,      NULL,    // unused with FORMAT_MESSAGE_FROM_SYSTEM    hresult,    MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT),    (LPTSTR)&errorText,  // output     0, // minimum size for output buffer    NULL);   // arguments - see note      if ( NULL != errorText ) {    // ... do something with the string `errorText` - log it, display it to the user, etc.     // release memory allocated by FormatMessage()    LocalFree(errorText);    errorText = NULL; } 

The key difference between this and David Hanak's answer is the use of the FORMAT_MESSAGE_IGNORE_INSERTS flag. MSDN is a bit unclear on how insertions should be used, but Raymond Chen notes that you should never use them when retrieving a system message, as you've no way of knowing which insertions the system expects.

FWIW, if you're using Visual C++ you can make your life a bit easier by using the _com_error class:

{    _com_error error(hresult);    LPCTSTR errorText = error.ErrorMessage();        // do something with the error...     //automatic cleanup when error goes out of scope } 

Not part of MFC or ATL directly as far as I'm aware.

like image 82
Shog9 Avatar answered Sep 23 '22 21:09

Shog9


Keep in mind that you cannot do the following:

{    LPCTSTR errorText = _com_error(hresult).ErrorMessage();     // do something with the error...     //automatic cleanup when error goes out of scope } 

As the class is created and destroyed on the stack leaving errorText to point to an invalid location. In most cases this location will still contain the error string, but that likelihood falls away fast when writing threaded applications.

So always do it as follows as answered by Shog9 above:

{    _com_error error(hresult);    LPCTSTR errorText = error.ErrorMessage();     // do something with the error...     //automatic cleanup when error goes out of scope } 
like image 30
Marius Avatar answered Sep 21 '22 21:09

Marius