Cleaned up Windows initialization error handling and migrated it to return the error message directly instead of through uiInitError.

This commit is contained in:
Pietro Gagliardi 2015-04-10 17:41:09 -04:00
parent 3069f38a9c
commit f650237614
1 changed files with 43 additions and 27 deletions

View File

@ -13,23 +13,49 @@ struct uiInitError {
char failbuf[256]; char failbuf[256];
}; };
static uiInitError *loadLastError(uiInitError *err, const char *message) #define initErrorFormat L"error %s: %s%sGetLastError() == %I32u%s"
#define initErrorArgs wmessage, sysmsg, beforele, le, afterle
static const char *loadLastError(const char *message)
{ {
WCHAR *sysmsg;
BOOL hassysmsg;
WCHAR *beforele;
WCHAR *afterle;
int n;
WCHAR *wmessage;
WCHAR *wstr;
const char *str;
DWORD le; DWORD le;
le = GetLastError(); le = GetLastError();
// TODO FormatMessageW() it if (FormatMessageW(FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_SYSTEM | FORMAT_MESSAGE_IGNORE_INSERTS, NULL, le, 0, (LPWSTR) (&sysmsg), 0, NULL) != 0) {
// TODO make sure argument is right; _snprintf_s() isn't supported on Windows XP hassysmsg = TRUE;
snprintf(err->failbuf, 256, "error %s (last error %I32u)", message, le); beforele = L" (";
err->msg = err->failbuf; afterle = L")";
return err; } else {
hassysmsg = FALSE;
sysmsg = L"";
beforele = L"";
afterle = L"";
}
wmessage = toUTF16(message);
n = _scwprintf(initErrorFormat, initErrorArgs);
wstr = (WCHAR *) uiAlloc((n + 1) * sizeof (WCHAR), "WCHAR[]");
swprintf_s(wstr, n + 1, initErrorFormat, initErrorArgs);
str = toUTF8(wstr);
uiFree(wstr);
if (hassysmsg)
if (LocalFree(sysmsg) != NULL)
logLastError("error freeing system message in loadLastError()");
uiFree(wmessage);
return str;
} }
uiInitOptions options; uiInitOptions options;
uiInitError *uiInit(uiInitOptions *o) const char *uiInit(uiInitOptions *o)
{ {
uiInitError *err;
STARTUPINFOW si; STARTUPINFOW si;
const char *ce; const char *ce;
HICON hDefaultIcon; HICON hDefaultIcon;
@ -38,11 +64,9 @@ uiInitError *uiInit(uiInitOptions *o)
options = *o; options = *o;
err = uiNew(uiInitError);
hInstance = GetModuleHandle(NULL); hInstance = GetModuleHandle(NULL);
if (hInstance == NULL) if (hInstance == NULL)
return loadLastError(err, "getting program HINSTANCE"); return loadLastError("getting program HINSTANCE");
nCmdShow = SW_SHOWDEFAULT; nCmdShow = SW_SHOWDEFAULT;
GetStartupInfoW(&si); GetStartupInfoW(&si);
@ -51,43 +75,35 @@ uiInitError *uiInit(uiInitOptions *o)
ce = initCommonControls(); ce = initCommonControls();
if (ce != NULL) if (ce != NULL)
return loadLastError(err, ce); return loadLastError(ce);
hDefaultIcon = LoadIconW(NULL, IDI_APPLICATION); hDefaultIcon = LoadIconW(NULL, IDI_APPLICATION);
if (hDefaultIcon == NULL) if (hDefaultIcon == NULL)
return loadLastError(err, "loading default icon for window classes"); return loadLastError("loading default icon for window classes");
hDefaultCursor = LoadCursorW(NULL, IDC_ARROW); hDefaultCursor = LoadCursorW(NULL, IDC_ARROW);
if (hDefaultCursor == NULL) if (hDefaultCursor == NULL)
return loadLastError(err, "loading default cursor for window classes"); return loadLastError("loading default cursor for window classes");
if (registerWindowClass(hDefaultIcon, hDefaultCursor) == 0) if (registerWindowClass(hDefaultIcon, hDefaultCursor) == 0)
return loadLastError(err, "registering uiWindow window class"); return loadLastError("registering uiWindow window class");
ZeroMemory(&ncm, sizeof (NONCLIENTMETRICSW)); ZeroMemory(&ncm, sizeof (NONCLIENTMETRICSW));
ncm.cbSize = sizeof (NONCLIENTMETRICSW); ncm.cbSize = sizeof (NONCLIENTMETRICSW);
if (SystemParametersInfoW(SPI_GETNONCLIENTMETRICS, sizeof (NONCLIENTMETRICSW), &ncm, sizeof (NONCLIENTMETRICSW)) == 0) if (SystemParametersInfoW(SPI_GETNONCLIENTMETRICS, sizeof (NONCLIENTMETRICSW), &ncm, sizeof (NONCLIENTMETRICSW)) == 0)
return loadLastError(err, "getting default fonts"); return loadLastError("getting default fonts");
hMessageFont = CreateFontIndirectW(&(ncm.lfMessageFont)); hMessageFont = CreateFontIndirectW(&(ncm.lfMessageFont));
if (hMessageFont == NULL) if (hMessageFont == NULL)
return loadLastError(err, "loading default messagebox font; this is the default UI font"); return loadLastError("loading default messagebox font; this is the default UI font");
// give each control a reasonable initial parent // give each control a reasonable initial parent
// don't free the initial parent! // don't free the initial parent!
// TODO tune this better; it shouldn't be closed, for instance // TODO tune this better; it shouldn't be closed, for instance
initialParent = (HWND) uiWindowHandle(uiNewWindow("", 0, 0)); initialParent = (HWND) uiWindowHandle(uiNewWindow("", 0, 0));
uiFree(err);
return NULL; return NULL;
} }
const char *uiInitErrorMessage(uiInitError *err) void uiFreeInitError(const char *err)
{ {
return err->msg; uiFree((void *) err);
}
void uiInitErrorFree(uiInitError *err)
{
if (err->msg != err->failbuf)
uiFree(err->msg);
uiFree(err);
} }