home *** CD-ROM | disk | FTP | other *** search
- /*
- * MALLOC2.CPP
- * C++ Malloc Demonstration Chapter 2
- *
- * Copyright (c)1993-1995 Microsoft Corporation, All Rights Reserved
- *
- * Kraig Brockschmidt, Microsoft
- * Internet : kraigb@microsoft.com
- * Compuserve: >INTERNET:kraigb@microsoft.com
- */
-
-
- #include "malloc2.h"
-
-
- /*
- * WinMain
- *
- * Purpose:
- * Main entry point of application.
- */
-
- int PASCAL WinMain(HINSTANCE hInst, HINSTANCE hInstPrev
- , LPSTR pszCmdLine, int nCmdShow)
- {
- MSG msg;
- PAPP pApp;
- DWORD dwVer;
- HRESULT hr;
-
- SETMESSAGEQUEUE;
-
- //Make sure COM is the right version
- dwVer=CoBuildVersion();
-
- if (rmm!=HIWORD(dwVer))
- return 0;
-
- //Initialize OLE
- hr=CoInitialize(NULL);
-
- if (FAILED(hr))
- return 0;
-
- pApp=new CApp(hInst, hInstPrev, nCmdShow);
-
- if (NULL!=pApp)
- {
- if (pApp->Init())
- {
- while (GetMessage(&msg, NULL, 0,0 ))
- {
- TranslateMessage(&msg);
- DispatchMessage(&msg);
- }
- }
-
- delete pApp;
- }
-
- //Only call this if CoInitialize worked
- CoUninitialize();
- return 0;
- }
-
-
-
-
-
- /*
- * MallocWndProc
- *
- * Purpose:
- * Standard window class procedure.
- */
-
- LRESULT APIENTRY MallocWndProc(HWND hWnd, UINT iMsg
- , WPARAM wParam, LPARAM lParam)
- {
- PAPP pApp;
- ULONG cb;
- UINT i;
- BOOL fResult;
-
- pApp=(PAPP)GetWindowLong(hWnd, MALLOCWL_STRUCTURE);
-
- switch (iMsg)
- {
- case WM_NCCREATE:
- pApp=(PAPP)(((LPCREATESTRUCT)lParam)->lpCreateParams);
- SetWindowLong(hWnd, MALLOCWL_STRUCTURE, (LONG)pApp);
- return (DefWindowProc(hWnd, iMsg, wParam, lParam));
-
- case WM_DESTROY:
- PostQuitMessage(0);
- break;
-
- case WM_COMMAND:
- switch (LOWORD(wParam))
- {
- case IDM_COGETMALLOC:
- pApp->GetAllocator();
- break;
-
-
- case IDM_RELEASE:
- if (!pApp->HaveAllocator())
- break;
-
- pApp->FreeAllocations(TRUE);
- pApp->Message(TEXT("Release complete"));
- break;
-
-
- case IDM_ALLOC:
- if (!pApp->HaveAllocator())
- break;
-
- fResult=pApp->DoAllocations(FALSE);
-
- pApp->Message(fResult ? TEXT("Alloc succeeded")
- : TEXT("Alloc failed"));
-
- break;
-
-
- case IDM_FREE:
- if (!pApp->HaveAllocations())
- break;
-
- pApp->FreeAllocations(FALSE);
- pApp->Message(TEXT("Free complete"));
- break;
-
-
- case IDM_REALLOC:
- if (!pApp->HaveAllocator())
- break;
-
- fResult=pApp->DoAllocations(TRUE);
-
- pApp->Message(fResult ? TEXT("Realloc succeeded")
- : TEXT("Realloc failed"));
-
- break;
-
-
- case IDM_GETSIZE:
- if (!pApp->HaveAllocations())
- break;
-
- fResult=TRUE;
-
- for (i=0; i < CALLOCS; i++)
- {
- cb=pApp->m_pIMalloc->GetSize(pApp->m_rgpv[i]);
-
- /*
- * We test that the size is *at least*
- * what we wanted.
- */
- fResult &= (pApp->m_rgcb[i] <= cb);
- }
-
- pApp->Message(fResult ? TEXT("Sizes matched")
- : TEXT("Sizes mismatch"));
-
- break;
-
-
- case IDM_DIDALLOC:
- if (!pApp->HaveAllocations())
- break;
-
- /*
- * DidAlloc may return -1 if it does not know
- * whether or not it actually allocated
- * something. In that case we just blindly
- * & in a -1 with no affect.
- */
-
- fResult=(BOOL)-1;
-
- for (i=0; i < CALLOCS; i++)
- {
- fResult &= pApp->m_pIMalloc
- ->DidAlloc(pApp->m_rgpv[i]);
- }
-
- if (0==fResult)
- pApp->Message(TEXT("DidAlloc is FALSE"));
-
- if (1==fResult)
- pApp->Message(TEXT("DidAlloc is TRUE"));
-
- if (-1==fResult)
- pApp->Message(TEXT("DidAlloc is unknown"));
-
- break;
-
-
- case IDM_HEAPMINIMIZE:
- if (!pApp->HaveAllocator())
- break;
-
- pApp->m_pIMalloc->HeapMinimize();
- pApp->Message(TEXT("HeapMinimize finished"));
- break;
-
-
- case IDM_EXIT:
- PostMessage(hWnd, WM_CLOSE, 0, 0L);
- break;
- }
- break;
-
- default:
- return (DefWindowProc(hWnd, iMsg, wParam, lParam));
- }
-
- return 0L;
- }
-
-
-
-
-
- /*
- * CApp::CApp
- * CApp::~CApp
- *
- * Constructor Parameters: (from WinMain)
- * hInst HINSTANCE of the application.
- * hInstPrev HINSTANCE of a previous instance.
- * nCmdShow UINT specifying how to show the app window.
- */
-
- CApp::CApp(HINSTANCE hInst, HINSTANCE hInstPrev
- , UINT nCmdShow)
- {
- UINT i;
- ULONG cb;
-
- m_hInst =hInst;
- m_hInstPrev =hInstPrev;
- m_nCmdShow =nCmdShow;
-
- m_hWnd =NULL;
- m_pIMalloc =NULL;
- m_fAllocated =FALSE;
-
- //100 is arbitrary. IMalloc can handle larger.
- cb=100;
-
- for (i=0; i < CALLOCS; i++)
- {
- m_rgcb[i]=cb;
- m_rgpv[i]=NULL;
-
- cb*=2;
- }
-
- return;
- }
-
-
-
- CApp::~CApp(void)
- {
- FreeAllocations(TRUE);
- return;
- }
-
-
-
-
-
- /*
- * CApp::Init
- *
- * Purpose:
- * Initializes an CApp object by registering window classes,
- * creating the main window, and doing anything else prone to
- * failure. If this function fails the caller should insure
- * that the destructor is called.
- *
- * Parameters:
- * None
- *
- * Return Value:
- * BOOL TRUE if successful, FALSE otherwise.
- */
-
- BOOL CApp::Init(void)
- {
- WNDCLASS wc;
-
- if (!m_hInstPrev)
- {
- wc.style = CS_HREDRAW | CS_VREDRAW;
- wc.lpfnWndProc = MallocWndProc;
- wc.cbClsExtra = 0;
- wc.cbWndExtra = CBWNDEXTRA;
- wc.hInstance = m_hInst;
- wc.hIcon = LoadIcon(m_hInst, TEXT("Icon"));
- wc.hCursor = LoadCursor(NULL, IDC_ARROW);
- wc.hbrBackground = (HBRUSH)(COLOR_WINDOW + 1);
- wc.lpszMenuName = MAKEINTRESOURCE(IDR_MENU);
- wc.lpszClassName = TEXT("MALLOC2");
-
- if (!RegisterClass(&wc))
- return FALSE;
- }
-
- m_hWnd=CreateWindow(TEXT("MALLOC2"), TEXT("C++ Malloc Demo")
- , WS_OVERLAPPEDWINDOW, 35, 35, 350, 250, NULL, NULL
- , m_hInst, this);
-
- if (NULL==m_hWnd)
- return FALSE;
-
- ShowWindow(m_hWnd, m_nCmdShow);
- UpdateWindow(m_hWnd);
-
- return TRUE;
- }
-
-
-
-
- /*
- * CApp::GetAllocator
- *
- * Purpose:
- * Retrieves the current allocator.
- *
- * Parameters:
- * None
- *
- * Return Value:
- * None
- */
-
- void CApp::GetAllocator(void)
- {
- HRESULT hr;
- TCHAR szTemp[80];
-
- //This also releases m_pIMalloc
- FreeAllocations(TRUE);
-
- hr=CoGetMalloc(MEMCTX_TASK, &m_pIMalloc);
-
- wsprintf(szTemp, TEXT("CoGetMalloc %s")
- , SUCCEEDED(hr) ? TEXT("succeeded") : TEXT("failed"));
-
- Message(szTemp);
- return;
- }
-
-
-
- /*
- * CApp::HaveAllocator
- *
- * Purpose:
- * Checks if there's a valid allocator and displays a
- * message if not.
- *
- * Parameters:
- * None
- *
- * Return Value:
- * BOOL TRUE if there is an allocator, FALSE otherwise.
- */
-
- BOOL CApp::HaveAllocator(void)
- {
- if (NULL==m_pIMalloc)
- {
- Message(TEXT("Create the allocator first"));
- return FALSE;
- }
-
- return TRUE;
- }
-
-
-
-
- /*
- * CApp::DoAllocations
- *
- * Purpose:
- * Centralized place to clean up allocations made on
- * the current IMalloc.
- *
- * Parameters:
- * fRelease BOOL indicating if we're to
- * IMalloc::Release as well.
- *
- * Return Value:
- * BOOL TRUE if all allocations succeeded.
- */
-
- BOOL CApp::DoAllocations(BOOL fRealloc)
- {
- UINT i;
- ULONG iByte;
- BOOL fResult=TRUE;
- ULONG cb;
- LPVOID pv;
-
- if (!fRealloc)
- FreeAllocations(FALSE);
-
- for (i=0; i < CALLOCS; i++)
- {
- //cb is set in the code below for later initialization
- if (fRealloc)
- {
- m_rgcb[i]+=128;
- cb=m_rgcb[i];
-
- //Old memory is not freed if Realloc fails
- pv=m_pIMalloc->Realloc(m_rgpv[i], cb);
- }
- else
- {
- cb=m_rgcb[i];
- pv=m_pIMalloc->Alloc(cb);
- }
-
- m_rgpv[i]=pv;
-
- //Fill the memory with letters.
- if (NULL!=pv)
- {
- LPBYTE pb=(LPBYTE)pv;
-
- for (iByte=0; iByte < cb; iByte++)
- *pb++=('a'+i);
- }
-
- fResult &= (NULL!=pv);
- }
-
- m_fAllocated=fResult;
-
- //Clean up whatever we might have allocated
- if (!fResult)
- FreeAllocations(FALSE);
-
- return fResult;
- }
-
-
-
-
- /*
- * CApp::HaveAllocations
- *
- * Purpose:
- * Checks if we have allocated memory from the current allocator,
- * displaying a message if not.
- *
- * Parameters:
- * None
- *
- * Return Value:
- * BOOL TRUE if there are allocations, FALSE otherwise.
- */
-
- BOOL CApp::HaveAllocations(void)
- {
- if (!HaveAllocator())
- return FALSE;
-
- if (!m_fAllocated)
- {
- Message(TEXT("Choose Alloc first"));
- return FALSE;
- }
-
- return TRUE;
- }
-
-
-
-
-
-
- /*
- * CApp::FreeAllocations
- *
- * Purpose:
- * Centralized place to clean up allocations made on
- * the current IMalloc.
- *
- * Parameters:
- * fRelease BOOL indicating if we're to
- * IMalloc::Release as well.
- *
- * Return Value:
- * None
- */
-
- void CApp::FreeAllocations(BOOL fRelease)
- {
- UINT i;
-
- if (NULL==m_pIMalloc)
- return;
-
- if (m_fAllocated)
- {
- for (i=0; i < CALLOCS; i++)
- {
- if (NULL!=m_rgpv[i])
- m_pIMalloc->Free(m_rgpv[i]);
-
- m_rgpv[i]=NULL;
- }
-
- m_fAllocated=FALSE;
- }
-
- if (fRelease)
- {
- m_pIMalloc->Release();
- m_pIMalloc=NULL;
- }
-
- return;
- }
-
-
-
-
-
- /*
- * CApp::Message
- *
- * Purpose:
- * Displays a message in the client area of the window. This is
- * just to centralize the call to simpify other code.
- *
- * Parameters:
- * psz LPTSTR to the string to display.
- *
- * Return Value:
- * None
- */
-
- void CApp::Message(LPTSTR psz)
- {
- HDC hDC;
- RECT rc;
-
- hDC=GetDC(m_hWnd);
- GetClientRect(m_hWnd, &rc);
-
- SetBkColor(hDC, GetSysColor(COLOR_WINDOW));
- SetTextColor(hDC, GetSysColor(COLOR_WINDOWTEXT));
-
- /*
- * We'll just be sloppy and clear the whole window as
- * well as write the string with one ExtTextOut call.
- * No word wrapping here...
- */
-
- ExtTextOut(hDC, 0, 0, ETO_OPAQUE, &rc, psz, lstrlen(psz), NULL);
- ReleaseDC(m_hWnd, hDC);
- return;
- }
-