Win2K 服务程序的源码(框架而已)

发表于:2007-07-01来源:作者:点击数: 标签:
以下为 Win2000 下我所理解的服务程序的源码,可以创建一个服务,并可以随时启动,停止和移除该服务。使用VC6编译,下面是使用方法简介。 需要安装 Service 的,打开 Dos 窗口,然后输入程序名字加上 -install 参数 例如: #Simple_Service -install 如果需

 

  以下为 Win2000 下我所理解的服务程序的源码,可以创建一个服务,并可以随时启动,停止和移除该服务。使用VC6编译,下面是使用方法简介。

  需要安装 Service 的,打开 Dos 窗口,然后输入程序名字加上 -install 参数
 例如:

    #Simple_Service  -install

  如果需要删除 Service 的,在 Dos 窗口,输入程序名字加上 -remove 参数

    #Simple_Service  -Remove


  也可以使用工具 SC.exe 来删除服务,在 Dos 窗口输入  sc delete _Simple_Service_08 即可


  如果要启动或停止服务,请在计算机管理-> 服务那里启动 ,服务名称为 _Simple_Service_08

/////////////////////////////////////////////////////////////////////////////////////

  该服务程序会生成调试用的日志文件,需要包含 RunTimeLog.h 和 RunTimeLog.cpp
  这两个文件在这里


  下面的程序对刚刚入门的伙伴可能有用,"高手"们就省省墨水,免得我又被臭骂了(面对像这样的中国人,我不得不学乖一点)。

  以下是完整的源程序,文件名称是: WinMain.cpp,详细的说明待我组织过后再附上。

 

//********************************************************************************
//***************      FileName: WinMain.cpp      ********************************
//********************************************************************************
// 最小 Link 包含: msvcrt.lib kernel32.lib user32.lib Advapi32.lib Shell32.lib
#define WIN32_LEAN_AND_MEAN   // Say No to MFC !!
#define UNICODE

#include <windows.h>
 //#include <Winsvc.h>
 //#include <Shellapi.h>


#include "RunTimeLog.h"
RunTimeLog log;

LPSTR MsgErr=NULL;

 


//////////////////////////////////////////////////////////////////////////
//////////////////////////////////////////////////////////////////////////
// Define for Service

#define ServiceName "_Simple_Service_08"
#define StrInstall TEXT("install")
#define StrRemove  TEXT("Remove")

void WINAPI service_ctrl(DWORD dwCtrlCode);
void WINAPI service_main(DWORD dwArgc,LPTSTR *lpszArgv);

void  WinMainCleanup(void);

SC_HANDLE                  schService=NULL, schSCManager=NULL;
SERVICE_STATUS             ssStatus;
SERVICE_STATUS_HANDLE      sshStatusHandle=NULL;
HANDLE                     hServerStopEvent = NULL;


//////////////////////////////////////////////////////////////////////////

 

 


// End of Define for Service


  char temp[1024]="Temp77 is Bad!";

 


////////////////////////////////////////////////////////////////
////////////////////////////////////////////////////////////////
////////////////////////////////////////////////////////////////
////////////////////////////////////////////////////////////////
void  WinMainCleanup(void)
{
  if(MsgErr){ LocalFree(MsgErr); MsgErr = NULL; }
  if( schService ) CloseServiceHandle( schService );
  if(schSCManager) CloseServiceHandle(schSCManager);
  log.last(true);
  //log.showResult(1500);
  //log.write("after ShowResult!");
  return;
}
//////////////////////////////////////////////////////////////////////////
//////////////////////////////////////////////////////////////////////////
//////////////////////////////////////////////////////////////////////////
//////////////////////////////////////////////////////////////////////////

// Name: WinMain()
// ------ ---------- ----------- ---------
int WINAPI WinMain( HINSTANCE hInstance,
                    HINSTANCE hPrevInstance,
                    LPSTR lpCmdLine, int nCmdShow )
{

  log.nobuff = true;
  log.write(ServiceName" WinMain start");


  // 判断是否为 WinNT
  OSVERSIONINFO  osvi;
  osvi.dwOSVersionInfoSize = sizeof(OSVERSIONINFO);
  if( GetVersionEx(&osvi) )
  {
    if(osvi.dwPlatformId == VER_PLATFORM_WIN32_NT) log.write("WinNT 系统可用!");
    else { log.write("非 WinNT 系统中,清场后退出");  WinMainCleanup(); return 1; }
  }else
  {
    log.write("Failed to GetVersionEx.");
    FormatMessageA( FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_SYSTEM | FORMAT_MESSAGE_IGNORE_INSERTS, NULL,
                    GetLastError(), NULL, (LPSTR)&MsgErr, 0, NULL );
    log.write(MsgErr);
    WinMainCleanup(); return 1;
  }// End of  if( GetVersionEx(&osvi) )
 

 

  // 填充服务入口表
  SERVICE_TABLE_ENTRY  dispatchTable[] =
  {
    { TEXT(ServiceName), LPSERVICE_MAIN_FUNCTION(service_main) },
    { NULL, NULL }
  };


  // 取得命令行以观察
  LPSTR cmdline = GetCommandLineA();
  log.write("CommandLine by following."); log.write(cmdline); log.write("");


  // 取得当前目录以观察
  GetCurrentDirectoryA(1024,temp);
  log.write("CurrentDirectory by following."); log.write(temp); log.write("");

  // 取得模块的文件名以观察
  GetModuleFileNameA(NULL,temp,sizeof(temp));
  log.write("Current Module FileName by following."); log.write(temp); log.write("");

  for(int i1=lstrlenA(temp); i1>0; i1--) { if(temp[i1]==´\\´) { temp[i1]=0;break; } }
  SetCurrentDirectoryA(temp);
  log.write("重新设置了当前目录的位置");

  // 重新取得当前目录以观察
  GetCurrentDirectoryA(1024,temp);
  log.write("CurrentDirectory by following."); log.write(temp); log.write("");


  // 分离出参数组以访问
  int NumArgsCount = NULL;
  LPWSTR * lpArgv = CommandLineToArgvW( GetCommandLineW(),  &NumArgsCount );
  log.numberwrite("NumArgsCount = ", (DWORD)NumArgsCount);

  int i=0;
  for(i=0;i<NumArgsCount;i++)
  {
    log.numberwrite("命令行参数 ",i);
    WideCharToMultiByte(CP_ACP, NULL, lpArgv[i],-1,temp,sizeof(temp),NULL,NULL);
    log.write(temp); log.write("");
  }

  // 判断参数的操作
  if(NumArgsCount>1) // 只有参数个数大于 1 的时候
  {
   if( (lpArgv[1][0]==(WORD)´/´) || (lpArgv[1][0]==(WORD)´-´) )
   {
     log.write("判断参数");
     if(!_wcsicmp(StrInstall, lpArgv[1]+1))
     {
       log.write("Install specify.");
       //CmdInstalService();

      if(!GetModuleFileNameA(NULL,temp,sizeof(temp))) { log.write("GetMouduleFileName failed.");  WinMainCleanup(); return 0; }

      // SC_HANDLE  schService=NULL, schSCManager=NULL;
      schSCManager = OpenSCManager(NULL,NULL,SC_MANAGER_ALL_ACCESS);
     if(schSCManager)
     {
       log.write("OpenSCManager 成功");
       // 打开成功才创建
      schService =
       CreateServiceA( schSCManager, ServiceName, ServiceName"(Dsp)",
                       SERVICE_ALL_ACCESS,SERVICE_WIN32_OWN_PROCESS, SERVICE_AUTO_START,SERVICE_ERROR_NORMAL,
                       temp,NULL,NULL,NULL,NULL,NULL  );
       if(schService)
       {
         log.write("CreateService "ServiceName" 成功");
         CloseServiceHandle(schService); schService = NULL;
       }
       else
       {
         log.write("CreateService "ServiceName" failed.");

         FormatMessageA( FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_SYSTEM | FORMAT_MESSAGE_IGNORE_INSERTS, NULL,
                         GetLastError(), NULL, (LPSTR)&MsgErr, 0, NULL );
         log.write(MsgErr);
        
         WinMainCleanup(); return 0;
       }


       CloseServiceHandle(schSCManager); schSCManager = NULL;
     }
     else { log.write("OpenSCManager failed.");  WinMainCleanup(); return 0; }

     log.write("");
     log.write("完成了服务的安装");
     log.write("");

     }
     else if(!_wcsicmp(StrRemove,  lpArgv[1]+1))
     {
       log.write("Remove  specify.");
       //CmdRemoveService();

  // SC_HANDLE  schService=NULL, schSCManager=NULL;
  schSCManager = OpenSCManager(NULL,NULL,SC_MANAGER_ALL_ACCESS);
  if(schSCManager)
  {
    log.write("OpenSCManager 成功");
    // 打开成功才进行下一步
    schService = OpenServiceA( schSCManager, ServiceName, SERVICE_ALL_ACCESS);
    if(schService)
    {
      log.write("OpenService "ServiceName" 成功");

      QueryServiceStatus( schService, &ssStatus);
      if(ssStatus.dwCurrentState == SERVICE_STOPPED) log.write("Service already Stopped.");
      else
      {
        // try to Stop the Service
        log.write("try to Stop the Service");

        if( ControlService( schService, SERVICE_CONTROL_STOP, &ssStatus ))
        {
          Sleep(50);
          while ( QueryServiceStatus( schService, &ssStatus))
          {
            if( ssStatus.dwCurrentState == SERVICE_STOP_PENDING )
            {
              Sleep(500);
              log.write("仍在停止 Service 中。。。");
            }
            else break;
          }//End of  while ( QueryServiceStatus( schService, &ssStatus))
        }

      }//End of  if(ssStatus.dwCurrentState == SERVICE_STOPPED) else

      // 如果已经停止了 Service ,那么就删除服务
      if(ssStatus.dwCurrentState == SERVICE_STOPPED )
      {
        Sleep(1000);
        if( DeleteService( schService) )  log.write("成功删除 "ServiceName" Service");
        else log.write("删除 "ServiceName" 失败");  log.write("");
      }
      else log.write("failed to Stop Service.");

     CloseServiceHandle(schService); schService = NULL;
    }
   else
   {
     log.write("OpenService "ServiceName" failed.");

     FormatMessageA( FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_SYSTEM | FORMAT_MESSAGE_IGNORE_INSERTS, NULL,
                     GetLastError(), NULL, (LPSTR)&MsgErr, 0, NULL );
     log.write(MsgErr);

     WinMainCleanup(); return 0;
   }
   CloseServiceHandle(schSCManager); schSCManager = NULL;
  }
   else { log.write("OpenSCManager failed.");  WinMainCleanup(); return 0; }


     }
   }
   WinMainCleanup();
   return 0;
  } // End of  if(NumArgsCount>1) // 只有参数个数大于 1 的时候

  log.write("");

  if(StartServiceCtrlDispatcher(dispatchTable)) log.write("StartServiceCtrlDispatcher Complete.");
  else
  {
    log.write("StartServiceCtrlDispatcher failed.");
    FormatMessageA( FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_SYSTEM | FORMAT_MESSAGE_IGNORE_INSERTS, NULL,
                    GetLastError(), NULL, (LPSTR)&MsgErr, 0, NULL );
    log.write(MsgErr);
  }

  
   WinMainCleanup();
   //ExitProcess(0);
   log.last(true);
   return NULL;

}

//////////////////////////////////////////////////////////////////////////
//////////////////////////////////////////////////////////////////////////
//////////////////////////////////////////////////////////////////////////
//////////////////////////////////////////////////////////////////////////
void WINAPI service_ctrl(DWORD dwCtrlCode)
{
// Handle the requested control code
  log.write("");
  log.write("Go into service_ctrl Function.");
  log.write("");
  int i2=0;
  switch(dwCtrlCode)
  {
    // Stop the Service
  case SERVICE_CONTROL_STOP:
    log.write("recive SERVICE_CONTROL_STOP");
    ssStatus.dwCheckPoint = ssStatus.dwCheckPoint++;
    ssStatus.dwWin32ExitCode = NO_ERROR;
    ssStatus.dwCurrentState = SERVICE_STOP_PENDING;
    SetServiceStatus(sshStatusHandle,&ssStatus);

    SetEvent(hServerStopEvent);
    log.write("已经设置 Service 状态为 SERVICE_STOP_PENDING ,并将同步对象设置为激发状态.");

   for(i2=0; i2<12; i2++)
   {
     QueryServiceStatus( schService, &ssStatus);
     if( ssStatus.dwCurrentState == SERVICE_STOP_PENDING )  Sleep(200);
     else break;
   }
   ResetEvent(hServerStopEvent);
   if(ssStatus.dwCurrentState == SERVICE_STOPPED )
   {
      log.write("The Service is now Stoped");
      ssStatus.dwCheckPoint = 0;
      ssStatus.dwWin32ExitCode = NO_ERROR;
      ssStatus.dwCurrentState = SERVICE_STOP;
      SetServiceStatus(sshStatusHandle,&ssStatus);
   } else  log.write("Stoped Service Failed.");

    return;
   break;
  case SERVICE_CONTROL_INTERROGATE:
    log.write("recive SERVICE_CONTROL_INTERROGATE");
   break;
  default:
    log.numberwrite("recive CtrlCode = ",dwCtrlCode);
    break;
  }

  //ssStatus.dwCurrentState = SERVICE_STOP_PENDING;
  ssStatus.dwCheckPoint = ssStatus.dwCheckPoint++;
  ssStatus.dwWin32ExitCode = NO_ERROR;
  SetServiceStatus(sshStatusHandle,&ssStatus);


  return;
}
//////////////////////////////////////////////////////////////////////////
//////////////////////////////////////////////////////////////////////////
//////////////////////////////////////////////////////////////////////////
//////////////////////////////////////////////////////////////////////////


void service_mainCleanup(void)
{
  log.write("");
  log.write("service_mainCleanup!"); log.write("");

  if(hServerStopEvent){ CloseHandle(hServerStopEvent); hServerStopEvent = NULL; }

  ssStatus.dwWin32ExitCode = 0;
  ssStatus.dwCurrentState = SERVICE_STOPPED;
  ssStatus.dwWaitHint = 1000;
  if(SetServiceStatus(sshStatusHandle,&ssStatus))
  {
    log.write("SetServiceStatus to SERVICE_STOPPED 成功!");
  }
  else
  {
   log.write("SetServiceStatus to SERVICE_STOPPED 失败!");
   FormatMessageA( FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_SYSTEM | FORMAT_MESSAGE_IGNORE_INSERTS, NULL,
                   GetLastError(), NULL, (LPSTR)&MsgErr, 0, NULL );
   log.write(MsgErr);
  }
 
  //log.showResult(false);
  return;
}
////////////////////////////////////////////////////////////////
////////////////////////////////////////////////////////////////
////////////////////////////////////////////////////////////////
////////////////////////////////////////////////////////////////
void WINAPI service_main(DWORD dwArgc,LPTSTR *lpszArgv)
{

 // 注册我们的 Service 控制处理函数
 sshStatusHandle = RegisterServiceCtrlHandlerA(ServiceName,service_ctrl);
 if(!sshStatusHandle)
 {
   FormatMessageA( FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_SYSTEM | FORMAT_MESSAGE_IGNORE_INSERTS, NULL,
                   GetLastError(), NULL, (LPSTR)&MsgErr, 0, NULL );
   log.write(MsgErr);
   log.write("Failed to RegisterServiceCtrlHandler");
   service_mainCleanup();
 }

 log.write("");
 log.write("We now Running in service_main.");
 log.write("Suclearcase/" target="_blank" >ccess to RegisterServiceCtrlHandler");


 hServerStopEvent = CreateEvent(NULL,TRUE,false,NULL);
 if(hServerStopEvent) log.write("创建了同步对象");
 else  log.write("创建同步对象失败");

 

 // 以下是测试同步信号的代码
 DWORD dwWait = NULL;

 dwWait = WaitForSingleObject(hServerStopEvent,200);
 {
   if(dwWait==WAIT_TIMEOUT)  log.write("等待同步信号超时");
   if(dwWait==WAIT_OBJECT_0) log.write("同步对象被触发了");
 }


 // 继续报告现在 Service 启动的进度
  ssStatus.dwServiceType = SERVICE_WIN32_OWN_PROCESS;
  ssStatus.dwServiceSpecificExitCode = 0;
  ssStatus.dwControlsAccepted = SERVICE_ACCEPT_STOP ;//| SERVICE_ACCEPT_PAUSE_CONTINUE;
  ssStatus.dwWaitHint = 1000;

  // 设置以后放送
  ssStatus.dwCheckPoint = 0;
  ssStatus.dwWin32ExitCode = NO_ERROR;
  ssStatus.dwCurrentState = SERVICE_START_PENDING;

  if(SetServiceStatus(sshStatusHandle,&ssStatus))
  {
    log.write("SetServiceStatus 成功!");
  }
  else
  {
   log.write("SetServiceStatus 失败!");
   FormatMessageA( FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_SYSTEM | FORMAT_MESSAGE_IGNORE_INSERTS, NULL,
                   GetLastError(), NULL, (LPSTR)&MsgErr, 0, NULL );
   log.write(MsgErr);
  }


   Sleep(300);

  // 设置以后放送
  ssStatus.dwCheckPoint++;
  log.numberwrite("Progress = ",ssStatus.dwCheckPoint);
  ssStatus.dwWin32ExitCode = NO_ERROR;
  ssStatus.dwCurrentState = SERVICE_START_PENDING;

  SetServiceStatus(sshStatusHandle,&ssStatus);

 


  for(int i1=0; i1<6;i1++)
  {

    Sleep(300);
    // 设置以后放送
    //ssStatus.dwCheckPoint++;
    ssStatus.dwCheckPoint = ssStatus.dwCheckPoint + 20;
    log.numberwrite("Progress = ",ssStatus.dwCheckPoint);
    ssStatus.dwWin32ExitCode = NO_ERROR;
    ssStatus.dwCurrentState = SERVICE_START_PENDING;
    SetServiceStatus(sshStatusHandle,&ssStatus);

  }

   Sleep(200);

   // 设置以后放送
   ssStatus.dwCheckPoint = 0;
   ssStatus.dwWin32ExitCode = NO_ERROR;
   ssStatus.dwCurrentState = SERVICE_RUNNING;
   SetServiceStatus(sshStatusHandle,&ssStatus);
   log.write("SetServiceStatus to SERVICE_RUNNING !");

   Sleep(500);

 

// 放入代码运行服务

 

   
   while (true)
   {
      dwWait = WaitForSingleObject(hServerStopEvent,1);
      {
        //if(dwWait==WAIT_TIMEOUT)  log.write("等待同步信号超时");
        if(dwWait==WAIT_OBJECT_0) { log.write("同步对象被触发了"); ResetEvent(hServerStopEvent); break; }
      }
//////////////////////////////////////////////////////////////////////////

 

       log.write("Doing Something...");
       Sleep(100); // 假设要运行100毫秒的操作

 

 

 

 

 

//////////////////////////////////////////////////////////////////////////

   }// End of while (true)


service_mainCleanup();

  return;
}


 


原文转自:http://www.ltesting.net