Windows 服務被設計用於須要在後臺運行的應用程序以及實現沒有用戶交互的任務。爲了學習這種控制檯應用程序的基礎知識,C(不是C++)是最佳選擇。本文將創建並 實現一個簡單的服務程序,其功能是查詢系統中可用物理內存數量,而後將結果寫入一個文本文件。最後,你能夠用所學知識編寫本身的 Windows 服務。
當初我寫第一個 NT 服務時,我到 MSDN 上找例子。在那裏我找到了一篇 Nigel Thompson 寫的文章:「Creating a Simple Win32 Servicein C++」, 這篇文章附帶一個 C++ 例子。雖然這篇文章很好地解釋了服務的開發過程,可是,我仍然感受缺乏我須要的重要信息。我想理解經過什麼框架,調用什麼函數,以及什麼時候調用,但 C++ 在這方面沒有讓我輕鬆多少。面向對象的方法當然方便,但因爲用類對底層 Win32 函數調用進行了封裝,它不利於學習服務程序的基本知識。這就是爲何我以爲 C 更加適合於編寫初級服務程序或者實現簡單後臺任務的服務。在你對服務程序有了充分透徹的理解以後,用 C++ 編寫才能遊刃有餘。當我離開原來的工做崗位,不得不向另外一我的轉移個人知識的時候,利用我用 C 所寫的例子就很是容易解釋 NT 服務之因此然。
服務是一個運行在後臺並實現勿需用戶交互的任務的控制檯程序。Windows NT/2000/XP 操做系統提供爲服務程序提供專門的支持。人們能夠用服務控制面板來配置安裝好的服務程序,也就是 Windows 2000/XP 控制面板|管理工具中的「服務」(或在「開始」|「運行」對話框中輸入 services.msc /s——譯者注)。能夠將服務配置成操做系統啓動時自動啓動,這樣你就沒必要每次再重啓系統後還要手動啓動服務。
本文將首先解釋如何建立一個按期查詢可用物理內存並將結果寫入某個文本文件的服務。而後指導你完成生成,安裝和實現服務的整個過程。
第一步:主函數和全局定義
首先,包含所需的頭文件。例子要調用 Win32 函數(windows.h)和磁盤文件寫入(stdio.h):程序員
#include <windows.h>
#include <stdio.h>數據庫
接着,定義兩個常量:編程
#define SLEEP_TIME 5000
#define LOGFILE"C:\\MyServices\\memstatus.txt"windows
SLEEP_TIME 指定兩次連續查詢可用內存之間的毫秒間隔。在第二步中編寫服務工做循環的時候要使用該常量。
LOGFILE 定義日誌文件的路徑,你將會用WriteToLog 函數將內存查詢的結果輸出到該文件,WriteToLog 函數定義以下:數組
int WriteToLog(char* str)
{
FILE* log;
log =fopen(LOGFILE, "a+");
if (log ==NULL)
return -1;
fprintf(log,"%s\n", str);
fclose(log);
return 0;
}服務器
聲明幾個全局變量,以便在程序的多個函數之間共享它們值。此外,作一個函數的前向定義:網絡
SERVICE_STATUS ServiceStatus;
SERVICE_STATUS_HANDLE hStatus;
void ServiceMain(int argc, char** argv);
void ControlHandler(DWORD request);
int InitService();多線程
如今,準備工做已經就緒,你能夠開始編碼了。服務程序控制臺程序的一個子集。所以,開始你能夠定義一個 main 函數,它是程序的入口點。對於服務程序來講,main 的代碼使人驚訝地簡短,由於它只建立分派表並啓動控制分派機。框架
void main()
{
SERVICE_TABLE_ENTRY ServiceTable[2];
ServiceTable[0].lpServiceName = "MemoryStatus";
ServiceTable[0].lpServiceProc = (LPSERVICE_MAIN_FUNCTION)ServiceMain;
ServiceTable[1].lpServiceName = NULL;
ServiceTable[1].lpServiceProc = NULL;
// 啓動服務的控制分派機線程
StartServiceCtrlDispatcher(ServiceTable);
}ssh
一個程序可能包含若干個服務。每個服務都必須列於專門的分派表中(爲此該程序定義了一個 ServiceTable 結構數組)。這個表中的每一項都要在 SERVICE_TABLE_ENTRY 結構之中。它有兩個域:
· lpServiceName: 指向表示服務名稱字符串的指針;當定義了多個服務時,那麼這個域必須指定;
· lpServiceProc: 指向服務主函數的指針(服務入口點);
分派表的最後一項必須是服務名和服務主函數域的 NULL 指針,文本例子程序中只宿主一個服務,因此服務名的定義是可選的。 服務控制管理器(SCM:Services Control Manager)是一個管理系統全部服務的進程。當 SCM 啓動某個服務時,它等待某個進程的主線程來調用 StartServiceCtrlDispatcher 函數。將分派表傳遞給 StartServiceCtrlDispatcher。這將把調用進程的主線程轉換爲控制分派器。該分派器啓動一個新線程,該線程運行分派表中每一個服務的 ServiceMain 函數(本文例子中只有一個服務)分派器還監視程序中全部服務的執行狀況。而後分派器將控制請求從 SCM 傳給服務。 注意:若是 StartServiceCtrlDispatcher 函數30秒沒有被調用,便會報錯,爲了不這種狀況,咱們必須在 ServiceMain 函數中(參見本文例子)或在非主函數的單獨線程中初始化服務分派表。本文所描述的服務不須要防範這樣的狀況。 分派表中全部的服務執行完以後(例如,用戶經過「服務」控制面板程序中止它們),或者發生錯誤時。StartServiceCtrlDispatcher 調用返回。而後主進程終止。第二步:ServiceMain 函數Listing 1
Listing 1: ServiceMain function
void ServiceMain(int argc, char** argv)
{
int error;
ServiceStatus.dwServiceType =
SERVICE_WIN32;
ServiceStatus.dwCurrentState =
SERVICE_START_PENDING;
ServiceStatus.dwControlsAccepted =
SERVICE_ACCEPT_STOP |
SERVICE_ACCEPT_SHUTDOWN;
ServiceStatus.dwWin32ExitCode = 0;
ServiceStatus.dwServiceSpecificExitCode = 0;
ServiceStatus.dwCheckPoint = 0;
ServiceStatus.dwWaitHint = 0;
hStatus = RegisterServiceCtrlHandler(
"MemoryStatus",
(LPHANDLER_FUNCTION)ControlHandler);
if (hStatus == (SERVICE_STATUS_HANDLE)0)
{
// Registering Control Handler failed
return;
}
// Initialize Service
error = InitService();
if (error)
{
// Initialization failed
ServiceStatus.dwCurrentState =
SERVICE_STOPPED;
ServiceStatus.dwWin32ExitCode = -1;
SetServiceStatus(hStatus, &ServiceStatus);
return;
}
// We report the running status to SCM.
ServiceStatus.dwCurrentState =
SERVICE_RUNNING;
SetServiceStatus (hStatus, &ServiceStatus);
MEMORYSTATUS memory;
// The worker loop of a service
while (ServiceStatus.dwCurrentState ==
SERVICE_RUNNING)
{
char buffer[16];
GlobalMemoryStatus(&memory);
sprintf(buffer, "%d", memory.dwAvailPhys);
int result = WriteToLog(buffer);
if (result)
{
ServiceStatus.dwCurrentState =
SERVICE_STOPPED;
ServiceStatus.dwWin32ExitCode = -1;
SetServiceStatus(hStatus,
&ServiceStatus);
return;
}
Sleep(SLEEP_TIME);
}
return;
}
Listing 1 展現了 ServiceMain 的代碼。該函數是服務的入口點。它運行在一個單獨的線程當中,這個線程是由控制分派器建立的。ServiceMain 應該儘量早早爲服務註冊控制處理器。這要經過調用 RegisterServiceCtrlHadler 函數來實現。你要將兩個參數傳遞給此函數:服務名和指向 ControlHandlerfunction 的指針。
它指示控制分派器調用 ControlHandler 函數處理 SCM 控制請求。註冊完控制處理器以後,得到狀態句柄(hStatus)。經過調用 SetServiceStatus 函數,用 hStatus 向 SCM 報告服務的狀態。
Listing 1 展現瞭如何指定服務特徵和其當前狀態來初始化 ServiceStatus 結構,ServiceStatus 結構的每一個域都有其用途:
· dwServiceType:指示服務類型,建立 Win32 服務。賦值 SERVICE_WIN32;
· dwCurrentState:指定服務的當前狀態。由於服務的初始化在這裏沒有完成,因此這裏的狀態爲 SERVICE_START_PENDING;
· dwControlsAccepted:這個域通知 SCM 服務接受哪一個域。本文例子是容許 STOP 和 SHUTDOWN 請求。處理控制請求將在第三步討論;
· dwWin32ExitCode 和 dwServiceSpecificExitCode:這兩個域在你終止服務並報告退出細節時頗有用。初始化服務時並不退出,所以,它們的值爲 0;
· dwCheckPoint 和 dwWaitHint:這兩個域表示初始化某個服務進程時要30秒以上。本文例子服務的初始化過程很短,因此這兩個域的值都爲 0。
調用 SetServiceStatus 函數向 SCM 報告服務的狀態時。要提供 hStatus 句柄和 ServiceStatus 結構。注意 ServiceStatus 一個全局變量,因此你能夠跨多個函數使用它。ServiceMain 函數中,你給結構的幾個域賦值,它們在服務運行的整個過程當中都保持不變,好比:dwServiceType。
在報告了服務狀態以後,你能夠調用 InitService 函數來完成初始化。這個函數只是添加一個說明性字符串到日誌文件。以下面代碼所示:
// 服務初始化
int InitService()
{
int result;
result =WriteToLog("Monitoring started.");
return(result);
}
在 ServiceMain 中,檢查 InitService 函數的返回值。若是初始化有錯(由於有可能寫日誌文件失敗),則將服務狀態置爲終止並退出 ServiceMain:
error = InitService();
if (error)
{
// 初始化失敗,終止服務
ServiceStatus.dwCurrentState = SERVICE_STOPPED;
ServiceStatus.dwWin32ExitCode = -1;
SetServiceStatus(hStatus, &ServiceStatus);
// 退出 ServiceMain
return;
}
若是初始化成功,則向 SCM 報告狀態:
// 向 SCM 報告運行狀態
ServiceStatus.dwCurrentState = SERVICE_RUNNING;
SetServiceStatus (hStatus, &ServiceStatus);
接着,啓動工做循環。每五秒鐘查詢一個可用物理內存並將結果寫入日誌文件。 如 Listing 1 所示,循環一直到服務的狀態爲 SERVICE_RUNNING 或日誌文件寫入出錯爲止。狀態可能在 ControlHandler 函數響應 SCM 控制請求時修改。 第三步:處理控制請求Listing 2
Listing 2: ControlHandler function
void ControlHandler(DWORD request)
{
switch(request)
{
case SERVICE_CONTROL_STOP:
WriteToLog("Monitoring stopped.");
ServiceStatus.dwWin32ExitCode = 0;
ServiceStatus.dwCurrentState = SERVICE_STOPPED;
SetServiceStatus (hStatus, &ServiceStatus);
return;
case SERVICE_CONTROL_SHUTDOWN:
WriteToLog("Monitoring stopped.");
ServiceStatus.dwWin32ExitCode = 0;
ServiceStatus.dwCurrentState = SERVICE_STOPPED;
SetServiceStatus (hStatus, &ServiceStatus);
return;
default:
break;
}
// Report current status
SetServiceStatus (hStatus, &ServiceStatus);
return;
}
在第二步中,你用 ServiceMain 函數註冊了控制處理器函數。控制處理器與處理各類 Windows 消息的窗口回調函數很是相似。它檢查 SCM 發送了什麼請求並採起相應行動。
每次你調用 SetServiceStatus 函數的時候,必須指定服務接收 STOP 和 SHUTDOWN 請求。Listing 2 示範瞭如何在 ControlHandler 函數中處理它們。
STOP 請求是 SCM 終止服務的時候發送的。例如,若是用戶在「服務」控制面板中手動終止服務。SHUTDOWN 請求是關閉機器時,由 SCM 發送給全部運行中服務的請求。兩種狀況的處理方式相同:
· 寫日誌文件,監視中止;
· 向 SCM 報告 SERVICE_STOPPED 狀態;
因爲 ServiceStatus 結構對於整個程序而言爲全局量,ServiceStatus 中的工做循環在當前狀態改變或服務終止後中止。其它的控制請求如:PAUSE 和 CONTINUE 在本文的例子沒有處理。
控制處理器函數必須報告服務狀態,即使 SCM 每次發送控制請求的時候狀態保持相同。所以,無論響應什麼請求,都要調用 SetServiceStatus。
圖一 顯示 MemoryStatus 服務的服務控制面板
第四步:安裝和配置服務
程序編好了,將之編譯成 exe 文件。本文例子建立的文件叫 MemoryStatus.exe,將它拷貝到 C:\MyServices 文件夾。爲了在機器上安裝這個服務,須要用 SC.EXE 可執行文件,它是 Win32 Platform SDK 中附帶的一個工具,通常進入cmd界面就能夠直接使用sc命令。(譯者注:Visaul Studio .NET 2003 IDE 環境中也有這個工具,具體存放位置在:C:\Program Files\Microsoft VisualStudio .NET 2003\Common7\Tools\Bin\winnt)。使用這個實用工具能夠安裝和移除服務。其它控制操做將經過服務控制面板來完成。如下是用命 令行安裝 MemoryStatus 服務的方法:
sc create MemoryStatus binpath=c:\MyServices\MemoryStatus.exe
發出此建立命令。指定服務名和二進制文件的路徑(注意 binpath= 和路徑之間的那個空格)。安裝成功後,即可以用服務控制面板來控制這個服務(參見圖一)。用控制面板的工具欄啓動和終止這個服務。
圖二 MemoryStatus 服務的屬性窗口
MemoryStatus 的啓動類型是手動,也就是說根據須要來啓動這個服務。右鍵單擊該服務,而後選擇上下文菜單中的「屬性」菜單項,此時顯示該服務的屬性窗口。在這裏能夠修改 啓動類型以及其它設置。你還能夠從「常規」標籤中啓動/中止服務。如下是從系統中移除服務的方法:
sc delete MemoryStatus
指定 「delete」 選項和服務名。此服務將被標記爲刪除,下次西通重啓後,該服務將被徹底移除。
建立了服務之後,打開服務管理,檢查這個服務的生成用戶,若是是localsys,右鍵點擊這個服務修改爲administrator用戶,修改的時候須要輸入用戶名和密碼。經過這種方式可使這個服務在administrator用戶下經過關閉和啓動方式來關閉和啓動。
第五步:測試服務
從服務控制面板啓動 MemoryStatus 服務。若是初始化不出錯,表示啓動成功。過一下子將服務中止。檢查一下 C:\MyServices 文件夾中 memstatus.txt 文件的服務輸出。在個人機器上輸出是這樣的:
Monitoring started.
273469440
273379328
273133568
273084416
Monitoring stopped.
爲了測試 MemoryStatus 服務在出錯狀況下的行爲,能夠將 memstatus.txt 文件設置成只讀。這樣一來,服務應該沒法啓動。
去掉只讀屬性,啓動服務,在將文件設成只讀。服務將中止執行,由於此時日誌文件寫入失敗。若是你更新服務控制面板的內容,會發現服務狀態是已經中止。
開發更大更好的服務程序
理解 Win32 服務的基本概念,使你能更好地用 C++ 來設計包裝類。包裝類隱藏了對底層 Win32 函數的調用並提供了一種溫馨的通用接口。修改 MemoryStatus 程序代碼,建立知足本身須要的服務!爲了實現比本文例子所示範的更復雜的任務,你能夠建立多線程的服務,將做業劃分紅幾個工做者線程並從 ServiceMain 函數中監視它們的執行。
做者簡介
Yevgeny Menaker 是一名有着超過5年經驗開發人員,做家和 Linux 顧問。過去的三年,Yevgeny 專一於開發新的高級的 Internet 技術。他牽頭編寫了《Programming Perl in the .NETEnvironment》一書(Prentice-Hall)。此外,做爲 Linux 顧問,他還在 Object Innovations 任職。Yevgeny Menaker 的聯繫方式是:jeka_books@hotmail.com
附錄1:
在2000/XP等基於NT 的操做系統中,有一個服務管理器,它管理的後臺進程被稱爲 service。
服務是一種應用程序類型,它在後臺運行,與 UNIX 後臺應用程序相似。服務應用程序一般能夠
在本地和經過網絡爲用戶提供一些功能,例如客戶端/服務器應用程序、Web 服務器、數據庫服
務器以及其餘基於服務器的應用程序。
後臺服務 程序是在後臺悄悄運行的。咱們經過將本身的程序登記爲服務,可使本身的程序不出現
在任務管理器中,而且隨系統啓動而最早運行,隨系統關閉而最後中止。
服務控制管理器是一個RPC 服務器,它顯露了一組應用編程接口,程序員能夠方便的編寫程序來配置
服務和控制遠程服務器中服務程序。
服務程序一般編寫成控制檯類型的應用程序,總的來講,一個遵照服務控制管理程序接口要求的程序
包含下面三個函數:
1。服務程序主函數(main):調用系統函數 StartServiceCtrlDispatcher 鏈接程序主線程到服務控制管理程序。
2。服務入口點函數(ServiceMain):執行服務初始化任務,同時執行多個服務的服務進程有多個服務入口函數。
3。控制服務處理程序函數(Handler):在服務程序收到控制請求時由控制分發線程引用。(此處是Service_Ctrl)。
另外在系統運行此服務以前須要安裝登記服務程序:installService 函數。刪除服務程序則須要先刪除服務安裝登記:removeService 函數。
服務類型:
類型
說明
SERVICE_FILE_SYSTEM_DRIVER=2
文件系統驅動服務。
SERVICE_KERNEL_DRIVER=1
驅動服務。
SERVICE_WIN32_OWN_PROCESS=16
獨佔一個進程的服務。
SERVICE_WIN32_SHARE_PROCESS=32
與其餘服務共享一個進程的服務。
新建WIN32控制檯程序, 其源文件名爲service.cpp 。我用的開發工具是VC++.NET。
1.服務程序主函數
服務控制管理程序啓動服務程序後,等待服務程序主函數調用系統函 StartServiceCtrlDispatcher。一個SERVICE_WIN32_OWN_PROCESS 類型的服務應該當即調用 StartServiceCtrlDispatcher 函數,能夠在服務啓動後讓服務入口點函數完成初始化工做。對於 SERVICE_WIN32_OWN_PROCESS 類型的服務和程序中全部服務共同的初始化工做能夠在主函數中完成,但不要超過30秒。不然必須創建另外的線程完成這些共同的初始化工做,從而保證服務程序 主函數能及時地調用 StartServiceCtrlDispatcher 函數。
主函數處理了三中命令行參數:- install,- remove,- debug,分別用於安裝,刪除和調試服務程序。若是不帶參數運行,則認爲是服務控制管理出現啓動該服務程序。參數不正確則給出提示信息。
StartServiceCtrlDispatcher 函數負責把程序主線程鏈接到服務控制管理程序。具體描述以下:
BOOL StartServiceCtrlDispatcher(
const LPSERVICE_TABLE_ENTRYlpServiceTable);
lpServiceStartTable 指向 SERVICE_TABLE_ENTRY 結構類型的數組,他包含了調用進程所提供的每一個服務的入口函數和字符串名。表中的最後一個元素必須爲 NULL,指明入口表結束。SERVICE_TABLE_ENTRY 結構具體描述以下:
typedef struct _SERVICE_TABLE_ENTRY{ LPTSTR lpServiceName; LPSERVICE_MAIN_FUNCTION lpServiceProc;
} SERVICE_TABLE_ENTRY,*LPSERVICE_TABLE_ENTRY;
lpServiceName 是一個以 NULL 結尾的字符串,標識服務名。若是是 SERVICE_WIN32_OWN_PROCESS 類型的服務,這個字符串會被忽略。
lpServiceProc 指向服務入口點函數。
//服務程序主函數。
#include "stdafx.h"
#include "Windows.h"
#define SZAPPNAME "serverSample" //服務程序名
#define SZSERVICENAME "serviceSample" //標識服務的內部名
//內部變量
bool bDebugServer=false;
SERVICE_STATUS ssStatus;
SERVICE_STATUS_HANDLE sshStatusHandle;
DWORD dwErr=0;
TCHAR szErr[256];
//下面的函數由程序實現
void WINAPI Service_Main(DWORD dwArgc,LPTSTR *lpszArgv);
void WINAPI Service_Ctrl(DWORDdwCtrlCode);
void installService();
void removeService();
void debugService(int argc,char**argv);
bool ReportStatusToSCMgr(DWORDdwCurrentState,DWORD dwWin32ExitCode,DWORD dwWaitHint);
void AddToMessageLog(LPTSTRlpszMsg);
int _tmain(int argc, _TCHAR* argv[])
{
SERVICE_TABLE_ENTRY dispatchTable[]=
{
{TEXT(SZSERVICENAME),(LPSERVICE_MAIN_FUNCTION)Service_Main},
{ NULL,NULL}
};
if((argc>1)&&((*argv[1]=='-')||(argv[1]=="/")))
{
if(_stricmp("install",argv[1]+1)==0)
{
installService();
}
elseif(_stricmp("remove",argv[1]+1)==0)
{
removeService();
}
elseif(_stricmp("debug",argv[1]+1)==0)
{
bDebugServer=true;
debugService(argc,argv);
}
else
{ //若是未能和上面的如何參數匹配,則多是服務控制管理程序來啓動該程序。當即調用
//StartServiceCtrlDispatcher函數。
printf("%s - install toinstall the service \n",SZAPPNAME);
printf("%s - remove toremove the service \n",SZAPPNAME);
printf("%s - debug to debugthe service \n",SZAPPNAME);
printf("\nStartServiceCtrlDispatcher being called.\n");
printf("This may takeseveral seconds.Please wait.\n");
if(!StartServiceCtrlDispatcher(dispatchTable))
AddToMessageLog(TEXT("StartServiceCtrlDispatcher failed."));
else
AddToMessageLog(TEXT("StartServiceCtrlDispatcher OK."));
}
exit(0);
}
return 0;
}
附錄2
GlobalMemoryStatus,Win32 API函數。
此函數用來得到當前可用的物理和虛擬內存信息,函數定義爲:
VOID GlobalMemoryStatus
(
LPMEMORYSTATUSlpBuffer
);
此函數無返回值,參數是一個指向名爲MEMORYSTATUS的結構的指針。函數的返回信息會被存儲在MEMORYSTATUS結構中。
此函數用來替代用來支持16位應用程序的GetFreeSpace函數。
應用程序應該在申請內存前調用此函數以防止影響到其餘程序運行。
這個函數的返回值是動態的,而且可能返回相同的值。
關於MEMORYSTATUS結構:
機構定義:
typedef struct _MEMORYSTATUS
{ // mst
DWORD dwLength; // sizeof(MEMORYSTATUS)
DWORD dwMemoryLoad; // percent of memory in use
DWORD dwTotalPhys; // bytes of physical memory
DWORD dwAvailPhys; // free physical memory bytes
DWORD dwTotalPageFile; // bytes of paging file
DWORD dwAvailPageFile; // free bytes of paging file
DWORD dwTotalVirtual; // user bytes of address space
DWORD dwAvailVirtual; // free user bytes
} MEMORYSTATUS, *LPMEMORYSTATUS;
結構成員的含義:
dwLength MEMORYSTATUS結構的大小,在調GlobalMemoryStatus函數前用sizeof()函數求得,
用來供函數檢測結構的版本。
dwMemoryLoad 返回一個介於0~100之間的值,用來指示當前系統內存的使用率。
dwTotalPhys 返回總的物理內存大小,以字節(byte)爲單位。
dwAvailPhys 返回可用的物理內存大小,以字節(byte)爲單位。
dwTotalPageFile 顯示能夠存在頁面文件中的字節數。注意這個數值並不表示在頁面文件在磁盤上的真實物理大小。
dwAvailPageFile 返回可用的頁面文件大小,以字節(byte)爲單位。
dwTotalVirtual 返回調用進程的用戶模式部分的所有可用虛擬地址空間,以字節(byte)爲單位。
dwAvailVirtual 返回調用進程的用戶模式部分的實際自由可用的虛擬地址空間,以字節(byte)爲單位。
MEMORYSTATUS結構,
結構的定義以下:
MEMORYSTATUS STRUCT
dwLength DWORD ;本結構的長度
dwMemoryLoad DWORD ;已用內存的百分比
dwTotalPhys DWORD ;物理內存總量
dwAvailPhys DWORD ;可用物理內存
dwTotalPageFile DWORD ;交換文件總的大小
dwAvailPageFile DWORD ;交換文件中空閒部分大小
dwTotalVirtual DWORD ;用戶可用的地址空間
dwAvailVirtual DWORD ;當前空閒的地址空間
MEMORYSTATUS ENDS