背景
服务应用程序由Windows Service Control Manager执行。启动后,它会对StartServiceCtrlDispatcher进行阻塞调用,该函数提供用作服务主函数的回调函数。
服务的主函数应注册第二个回调以处理传入的命令,如开始、停止、继续等。它通过调用RegisterServiceCtrlHandler来实现此目的。
问题
我能够编写一个程序来注册服务主函数。然后,我可以将该程序安装为Windows服务,并从服务管理控制台启动它。服务能够启动,报告自己正在运行,然后等待传入请求。
问题在于我无法调用服务处理函数。查询服务状态显示它正在运行,但是一旦我发送一个“停止”命令,Windows弹出一个消息:
Windows could not stop the Test service on Local Computer.
Error 1061: The service cannot accept control messages at this time.
根据MSDN文档,StartServiceCtrlDispatcher函数会阻塞直到所有服务报告它们已停止。在调用服务主函数后,调度程序线程应等待服务控制管理器发送命令,然后该线程应调用处理程序函数。
细节
以下是我尝试做的非常简化版本,但它展示了我的处理程序函数未被调用的问题。
首先,一些名称和导入:
module Main where
import Control.Applicative
import Foreign
import System.Win32
wIN32_OWN_PROCESS :: DWORD
wIN32_OWN_PROCESS = 0x00000010
sTART_PENDING, rUNNING :: DWORD
sTART_PENDING = 0x00000002
rUNNING = 0x00000004
aCCEPT_STOP, aCCEPT_NONE :: DWORD
aCCEPT_STOP = 0x00000001
aCCEPT_NONE = 0x00000000
nO_ERROR :: DWORD
nO_ERROR = 0x00000000
type HANDLER_FUNCTION = DWORD -> IO ()
type MAIN_FUNCTION = DWORD -> Ptr LPTSTR -> IO ()
我需要为数据编组定义一些特殊的数据类型,并创建 Storable 实例。
data TABLE_ENTRY = TABLE_ENTRY LPTSTR (FunPtr MAIN_FUNCTION)
instance Storable TABLE_ENTRY where
sizeOf _ = 8
alignment _ = 4
peek ptr = TABLE_ENTRY <$> peek (castPtr ptr) <*> peek (castPtr ptr `plusPtr` 4)
poke ptr (TABLE_ENTRY name proc) = do
poke (castPtr ptr) name
poke (castPtr ptr `plusPtr` 4) proc
data STATUS = STATUS DWORD DWORD DWORD DWORD DWORD DWORD DWORD
instance Storable STATUS where
sizeOf _ = 28
alignment _ = 4
peek ptr = STATUS
<$> peek (castPtr ptr)
<*> peek (castPtr ptr `plusPtr` 4)
<*> peek (castPtr ptr `plusPtr` 8)
<*> peek (castPtr ptr `plusPtr` 12)
<*> peek (castPtr ptr `plusPtr` 16)
<*> peek (castPtr ptr `plusPtr` 20)
<*> peek (castPtr ptr `plusPtr` 24)
poke ptr (STATUS a b c d e f g) = do
poke (castPtr ptr) a
poke (castPtr ptr `plusPtr` 4) b
poke (castPtr ptr `plusPtr` 8) c
poke (castPtr ptr `plusPtr` 12) d
poke (castPtr ptr `plusPtr` 16) e
poke (castPtr ptr `plusPtr` 20) f
poke (castPtr ptr `plusPtr` 24) g
只需要进行三个外部导入。对于我将提供给Win32的两个回调函数,还需要一个“包装器”导入:
foreign import stdcall "wrapper"
smfToFunPtr :: MAIN_FUNCTION -> IO (FunPtr MAIN_FUNCTION)
foreign import stdcall "wrapper"
handlerToFunPtr :: HANDLER_FUNCTION -> IO (FunPtr HANDLER_FUNCTION)
foreign import stdcall "windows.h RegisterServiceCtrlHandlerW"
c_RegisterServiceCtrlHandler
:: LPCTSTR -> FunPtr HANDLER_FUNCTION -> IO HANDLE
foreign import stdcall "windows.h SetServiceStatus"
c_SetServiceStatus :: HANDLE -> Ptr STATUS -> IO BOOL
foreign import stdcall "windows.h StartServiceCtrlDispatcherW"
c_StartServiceCtrlDispatcher :: Ptr TABLE_ENTRY -> IO BOOL
主程序
最后,这是主服务应用程序:
main :: IO ()
main =
withTString "Test" $ \name ->
smfToFunPtr svcMain >>= \fpMain ->
withArray [TABLE_ENTRY name fpMain, TABLE_ENTRY nullPtr nullFunPtr] $ \ste ->
c_StartServiceCtrlDispatcher ste >> return ()
svcMain :: MAIN_FUNCTION
svcMain argc argv = do
appendFile "c:\\log.txt" "svcMain: svcMain here!\n"
args <- peekArray (fromIntegral argc) argv
fpHandler <- handlerToFunPtr svcHandler
h <- c_RegisterServiceCtrlHandler (head args) fpHandler
_ <- setServiceStatus h running
appendFile "c:\\log.txt" "svcMain: exiting\n"
svcHandler :: DWORD -> IO ()
svcHandler _ = appendFile "c:\\log.txt" "svcCtrlHandler: received.\n"
setServiceStatus :: HANDLE -> STATUS -> IO BOOL
setServiceStatus h status = with status $ c_SetServiceStatus h
running :: STATUS
running = STATUS wIN32_OWN_PROCESS rUNNING aCCEPT_STOP nO_ERROR 0 0 3000
输出
我之前使用sc create Test binPath= c:\Main.exe
安装了该服务。
这是编译程序的输出:
C:\path>ghc -threaded --make Main.hs
[1 of 1] Compiling Main ( Main.hs, Main.o )
Linking Main.exe ...
C:\path>
然后我从服务控制监视器启动服务。这里有证据表明我的调用SetServiceStatus被接受了:
C:\Path>sc query Test
SERVICE_NAME: Test
TYPE : 10 WIN32_OWN_PROCESS
STATE : 4 RUNNING
(STOPPABLE, NOT_PAUSABLE, IGNORES_SHUTDOWN)
WIN32_EXIT_CODE : 0 (0x0)
SERVICE_EXIT_CODE : 0 (0x0)
CHECKPOINT : 0x0
WAIT_HINT : 0x0
C:\Path>
以下是
log.txt
文件的内容,证明了我的第一个回调函数svcMain
已被调用:svcMain: svcMain here!
svcMain: exiting
一旦我使用服务控制管理器发送停止命令,就会收到错误消息。我的处理程序应该向日志文件添加一行,但这并没有发生。然后我的服务显示为已停止状态:
C:\Path>sc query Test
SERVICE_NAME: Test
TYPE : 10 WIN32_OWN_PROCESS
STATE : 1 STOPPED
WIN32_EXIT_CODE : 0 (0x0)
SERVICE_EXIT_CODE : 0 (0x0)
CHECKPOINT : 0x0
WAIT_HINT : 0x0
C:\Path>
问题
有没有人有想法,可以让我的处理程序被调用?
更新20130306
我在Windows 7 64位上遇到了这个问题,但在Windows XP上没有。其他版本的Windows尚未测试。当我将编译后的可执行文件复制到多台机器上并执行相同的步骤时,结果不同。
svcHandler
实现调用SetServiceStatus
来报告服务的SERVICE_STOPPED
状态。另一个问题:您在日志文件中是否看到消息“svcCtrlHandler:received.\n”?顺便说一下,我建议您使用sc interrogate Test
来验证svcHandler
是否正确注册。如果svcHandler
接收到输入为SERVICE_CONTROL_INTERROGATE
(4),它应该像svcMain
一样调用SetServiceStatus
以报告状态rUNNING
(SERVICE_RUNNING
)和其接受的控制(aCCEPT_STOP
)。 - Olegsc interrogate Test
抛出了 _"The service cannot accept control messages at this time"_,这是一个很大的线索。根据微软提供的这个实例,在SERVICE_RUNNING
之前,你需要将初始状态设置为SERVICE_START_PENDING
。我相信当你纠正了状态转换后,代码将开始按预期工作--其他所有内容都已检查完毕。 - MrGomez