当前位置:首页 » 软件开发
开发技术指南» 文章正文
    引言: 创建SvcHost.exe调用的服务原理与实践tyle="FON
 

 

    摘要:  四月份开始了一个新项目,需要启用多线程。以前也做过这样的项目,但那时刚从学校出来,也不知道什么,瞎写一通反正是搞出来了。今天,又要 “重操旧业”。呵呵,虽今非昔比,却也不敢小视。懂得多了,也就更加清楚其中的复杂性。因为这个项目不能在公司测试,最好能写得严谨、周到,免得到现场还要进行大量的修改。于是,翻出几本“秘集”,重温,准备要大干一场。  翻出......
 ·安装vs 6.0遇到的问题     »显示摘要«
    摘要:今天在安装vs 6.0时,遇到这个问题:setup was unable to create a dcom user account in order to register <path>\valec.exe后来找到原来是vs的一个bug。贴出微软的原文:bug: "setup was unable to create a dcom user account" ......


创建SvcHost.exe调用的服务原理与实践
创建svchost.exe调用的服务原理与实践

   by bingle_at_email.com.cn

【程序编程相关:Using ADO from C++

【推荐阅读:在Visual C++中用ADO进行数据

      www.binglesite.net 【扩展信息:ADO VC++ Extensions使

作者:bingle

 

 

1. 多个服务共享一个svchost.exe进程利与弊

 

windows 系统服务分为独立进程与共享进程两种,在windows nt时只有服务器管理器scm(services.exe)有多个共享服务,随着系统内置服务的增加,在windows 2000中ms又把很多服务做成共享方式,由svchost.exe启动.windows 2000一般有2个svchost进程,一个是rpcss(remote procedure call)服务进程,另外一个则是由很多服务共享的一个svchost.exe.而在windows xp中,则一般有4个以上的svchost.exe服务进程,windows 2003 server中则更多,可以看出把更多的系统内置服务以共享进程方式由svchost启动是ms的一个趋势.这样做在一定程度上减少了系统资源的消耗,不过也带来一定的不稳定因素,因为任何一个共享进程的服务因为错误退出进程就会导致整个进程中的所有服务都退出.另外就是有一点安全隐患,首先要介绍一下svchost.exe的实现机制.

 

 

2. svchost原理

 

svchost本身只是作为服务宿主,并不实现任何服务功能,需要svchost启动的服务以动态链接库形式实现,在安装这些服务时,把服务的可执行程序指向svchost,启动这些服务时由svchost调用相应服务的动态链接库来启动服务.

 

那么svchost如何知道某一服务是由哪个动态链接库负责呢?这不是由服务的可执行程序路径中的参数部分提供的,而是服务在注册表中的参数设置的,注册表中服务下边有一个parameters子键其中的servicedll表明该服务由哪个动态链接库负责.并且所有这些服务动态链接库都必须要导出一个servicemain()函数,用来处理服务任务.

 

例如rpcss(remote procedure call)在注册表中的位置是  hkey_local_machine\system\currentcontrolset\services\rpcss,它的参数子键parameters里有这样一项:

"servicedll"=reg_expand_sz:"%systemroot%\system32\rpcss.dll"

当启动rpcss服务时,svchost就会调用rpcss.dll,并且执行其servicemain()函数执行具体服务.

 

既然这些服务是使用共享进程方式由svchost启动的,为什么系统中会有多个svchost进程呢?ms把这些服务分为几组,同组服务共享一个svchost进程,不同组服务使用多个svchost进程,组的区别是由服务的可执行程序后边的参数决定的.

 

例如rpcss在注册表中 hkey_local_machine\system\currentcontrolset\services\rpcss 有这样一项:

"imagepath"=reg_expand_sz:"%systemroot%\system32\svchost -k rpcss"

因此rpcss就属于rpcss组,这在服务管理控制台也可以看到.

 

svchost的所有组与组内的所有服务都在注册表的如下位置: hkey_local_machine\software\microsoft\windows nt\currentversion\svchost,例如windows 2000共有4组rpcss.netsvcs.wugroup.bitsgroup,其中最多的就是netsvcs=reg_multi_sz:eventsystem.ias.iprip.irmon.netman.nwsapagent.rasauto.rasman.remoteaccess.sens.sharedaccess.tapisrv.ntmssvc.wzcsvc..

 

在启动一个svchost.exe负责的服务时,服务管理器如果遇到可执行程序内容imagepath已经存在于服务管理器的映象库中,就不在启动第2个进程svchost,而是直接启动服务.这样就实现了多个服务共享一个svchost进程.

 

 

3. svchost代码

 

现在我们基本清楚svchost的原理了,但是要自己写一个dll形式的服务,由svchost来启动,仅有上边的信息还有些问题不是很清楚.比如我们在导出的servicemain()函数中接收的参数是ansi还是unicode?我们是否需要调用registerservicectrlhandler与startservicectrldispatcher来注册服务控制及调度函数?

 

这些问题要通过查看svchost代码获得.下边的代码是windows 2000+ service pack 4 的svchost反汇编片段,可以看出svchost程序还是很简单的.

 

主函数首先调用proccommandline()对命令行进行分析,获得要启动的服务组,然后调用svchostoptions()查询该服务组的选项与服务组的所有服务,并使用一个数据结构 svctable 来保存这些服务及其服务的dll,然后调用preparesvctable() 函数创建service_table_entry 结构,把所有处理函数service_main_function 指向自己的一个函数funcservicemain(),最后调用api startservicectrldispatcher() 注册这些服务的调度函数.

 

; =============================== main funcion ===========================================

.text:010010b8                 public start

.text:010010b8 start           proc near

.text:010010b8                 push    esi

.text:010010b9                 push    edi

.text:010010ba                 push    offset sub_1001eba ; lptoplevelexceptionfilter

.text:010010bf                 xor     edi, edi

.text:010010c1                 call    ds:setunhandledexceptionfilter

.text:010010c7                 push    1               ; umode

.text:010010c9                 call    ds:seterrormode

.text:010010cf                 call    ds:getprocessheap

.text:010010d5                 push    eax

.text:010010d6                 call    sub_1001142

.text:010010db                 mov     eax, offset dword_1003018

.text:010010e0                 push    offset unk_1003000 ; lpcriticalsection

.text:010010e5                 mov     dword_100301c, eax

.text:010010ea                 mov     dword_1003018, eax

.text:010010ef                 call    ds:initializecriticalsection

.text:010010f5                 call    ds:getcommandlinew

.text:010010fb                 push    eax             ; lpstring

.text:010010fc                 call    proccommandline

.text:01001101                 mov     esi, eax


...   下一页
 ·ccombobox控件详解     »显示摘要«
    摘要: ccombobox控件又称作组合框控件,其有三种形态可供选择,1.简单组合框(simple)2.下拉组合框(drop-down)3.下拉列表式组合框(drop-down list). ccombobox控件的常用设置属性说明: type属性:里面一共有三个选项.就是其三种形式, 我们常用的是后两种形态,其区别就是dropdown的编辑区为可编辑控件,而droplist为静态控件. data属性:......
» 本期热门文章:

©2000-2007 All Rights Reserved. 最佳浏览:1024X768 MSIE