|
|
最近在搞一个项目,需要程序开机自动运行,可是程序中调用了底层驱动的一些函数,必须以管理员的权限才能运行,否则程序运行不成功,在XP 下 直接写注册表就可以,可是在VISTA 和 WIN7 下写注册表的方式失效,因为必须以管理员的权限运行才可以,迫于无奈,上网查了N多资料,终于找到了一种解决的办法,在此分享出来,以此献给被此问题苦恼的人!
1 `7 N6 d4 w8 D2 f6 C! b* T$ j8 f6 {
首先介绍一下,解决问题的思路,创建一个服务,安装在系统中,在服务里以SYSTEM 的权限创建一个进程,用来启动自己的应用程序,我们想什么时候启动我们的应用程序,只需要我们用应用程序给服务发送一个信号,在服务接收到信号后,就可以启动我们想要运行的应用程序了。
. B! [8 z6 w" |% w. S6 S& h1 M" ?' p5 W0 P% m$ j! v; W& n! J* s% w
实现框架如下:
6 V8 w; P, r+ A0 H: [) x0 f3 C/ h; U7 y4 y6 t
0 s' }' t0 H6 o* b+ b5 [
: h7 K- ]0 o: k0 t& O. ^( n2 W, |所以在这里我们需要创建三个程序:1.Windows 服务程序 2.我们自己的应用程序 3.给服务发送信号的应用程序。. y1 n7 i. V1 [ [' T4 X
8 t9 o( X/ {: u) {- x* _8 W- k当然其中的 1和3 可以合并在一起,而且 2 我们可以做成多个应用程序。这样只要是我们自己写的都可以以SYSTEM的权限运行了,比管理员更实用。这里需要注意的是第一次安装服务的时候必须以管理员的权限运行。( f; G* }* H1 g: \# ]/ L, R* H* x
" ^. |/ Y& d; ?! K好了,闲话少说,下面步入正题,我做了个简单的Demo,是以服务的方式启动系统自带的管理记事本程序(windows 7下面的,用VS2008编辑)。
' W: X" D, K* a1 m# B+ A- q5 |5 t; J9 n- z8 z. l' Y& A2 [
我们以函数CreateProcessAsUser 来开始讲解,我们所要做的动作就是把这个函数的各个参数都填对 就OK,只需把其中的几个关键参数填好,其他设置为NULL 就OK ,* I& {" M, _1 C
. e6 C& t1 F- W8 H& ^3 W6 e! `
首先第一个参数HANDLE hToken ,Token 句柄,怎么得到这个句柄,MSDN 给出了两种办法,一种通过调用 LogonUser函数 ,一种另外一种通过 调用DuplicateTokenEx 函数,在这里我们选择第二种方式创建一个模拟的Token,关键是这个模拟的Token那里来,那我们就得追踪DuplicateTokenEx里面的各个参数,其中DuplicateTokenEx参数如下:! D5 Z" m5 \* V6 R
- Y: y N" H: Y$ I8 e; c
BOOL WINAPI DuplicateTokenEx(( g7 i) Y+ j5 {+ s% n" `% ~
__in HANDLE hExistingToken,
! i+ M7 b0 ]: O' O% E __in DWORD dwDesiredAccess,
5 v4 {% I1 w- Y! y1 v __in_opt LPSECURITY_ATTRIBUTES lpTokenAttributes,* Q$ v" p7 t& y7 n" x! ]! @
__in SECURITY_IMPERSONATION_LEVEL ImpersonationLevel,
; A2 K v8 |5 i/ O' O. y __in TOKEN_TYPE TokenType,+ R3 h, |9 h9 s
__out PHANDLE phNewToken2 \: O6 s1 W, o2 N
);9 ^$ _$ y! N7 z
这个函数的主要功能就是根据现有的Token 来创建一个新的Token,我们需要的就是这个新的Token,如何得到这个新的Token,看来我们还得往下追踪哦,第二个参数是新建Token的权限,这个参数直接设置成MAXIMUM_ALLOWED就可以,第三个参数是 安全描述符的指针,这里直接设置成NULL,分配成一个默认的安全描述符,第四个参数设置成SecurityIdentification 的OK,(安全描述符的级别) 第五个参数 Token类型 直接设置成TokenPrimary OK 因为它就是用于CreateProcessAsUser的函数的。最后一个参数就是我们想要的参数了。我们主要就是得到他,要想得到这个参数 我们必须得到第一个参数。
1 C- Y& Q7 M- N4 Q+ y- T: z5 ]
& F4 S9 S! G7 k- e9 C$ \ o, ?第一个参数是一个已经存在的Token,数要想得到这个Token,必须找到以个进程,把这个进程的Token 拿过来为我们所用,得到这个Token 要运行函数OpenPrcessToken通过这个函数得到Token的句柄,但是要得到这个Token的句柄,必须要通过函数OpenProcess得到进程的句柄,要得到进程的句柄 必须通过函数ProcessIdToSessionId得到进程的ID。因为我们系统每次登入的时候都有一个winlogon.exe,我们就拿这个进程的ID 来得到CreateProcessAsUser函数的第一个参数。也就是把上面的一步步还原回去!
- j4 O$ {* v& \! V- f/ ?4 S
; z* G. L0 d8 @' z/ D3 c, L. u, F 0 l3 a; _: M! Q* h; l
* I$ |% _+ i, N( a4 t
dwSessionId = WTSGetActiveConsoleSessionId(); //得到当前用户的会话ID //////////////////////////////////////////
- a5 `8 z' i; E" s$ V0 V// Find the winlogon process( x1 J3 o- Y' G! b% `8 X+ n
////////////////////////////////////////( A, P+ o9 F# S, M# N# t$ ]! {) Q
PROCESSENTRY32 procEntry; //用来存放快照进程信息的一个结构体+ Y) _, z. K; g8 ^# H/ z8 c; N
//函数为指定的进程、进程使用的堆[HEAP]、模块[MODULE]、线程[THREAD])建立一个快照[snapshot]
. D3 t8 k- d1 h8 `5 s7 t" c2 h HANDLE hSnap = CreateToolhelp32Snapshot(TH32CS_SNAPPROCESS, 0);
$ T6 Y: T6 X' G" n) a4 M- E if (hSnap == INVALID_HANDLE_VALUE)
8 w, a+ v) ^; t" a) ~ }" N" z7 u- A {
# F; `! h2 q' n A+ F return 1 ;5 L) x+ Q3 O! d+ \; @9 s
}) D) ^$ ~) \) I8 V& n" B
procEntry.dwSize = sizeof(PROCESSENTRY32);& a" V# A3 ^1 l- g
if (!Process32First(hSnap, &procEntry)) //获得第一个进程的句柄.
8 }9 F8 B4 o3 H6 ` v {) ?, k% Z! o% E& h& W8 I; y0 S# S
return 1 ;
0 V% Q4 ?; X8 w }9 ^" q, s* O: q; P: M! C# R7 }
do
8 ?3 f6 L8 G/ o- }: S9 | {
2 o& ~6 G7 T9 S+ V# U2 v0 Kif (_wcsicmp(procEntry.szExeFile, _T("winlogon.exe")) == 0) //查找winlogon.exe( C! N. ]3 ?/ l/ I4 e+ ^- g1 _( j) B
{/ j6 L: t2 z$ f7 i4 x
// We found a winlogon process...make sure it's running in the console session" V! M; F6 P' s/ m0 `/ X" A- k
DWORD winlogonSessId = 0;3 }0 P; [) a6 h
if (ProcessIdToSessionId(procEntry.th32ProcessID, &winlogonSessId) && winlogonSessId == dwSessionId)//得到与进程ID对应的终端服务会话ID$ |8 {/ i+ R0 W- G5 i0 e& _
{% M2 Z/ y$ X& k
winlogonPid = procEntry.th32ProcessID; [2 m" [6 J- U K- c
break;
0 V. A6 f3 w2 x' y/ d1 G }
) O7 E$ D+ P3 u1 V1 s }
_/ ~) J9 X6 N4 X7 a1 { } while (Process32Next(hSnap, &procEntry)); //获得下一个进程的句柄$ V# {1 e/ m v( v6 D: q
////////////////////////////////////////////////////////////////////////, ^5 [" l: b! `' @
hProcess = OpenProcess(MAXIMUM_ALLOWED,FALSE,winlogonPid); //获得进程句柄
! a! N8 ~3 s& ?# n( t9 w if(!::OpenProcessToken(hProcess,TOKEN_ADJUST_PRIVILEGES|TOKEN_QUERY
% U: g1 f# U. k1 [ w |TOKEN_DUPLICATE|TOKEN_ASSIGN_PRIMARY|TOKEN_ADJUST_SESSIONID
7 Q2 b; G4 E$ p/ l; z |TOKEN_READ|TOKEN_WRITE,&hPToken)) //获得令牌句柄3 |+ Z" i2 o9 X: b7 R* K
{$ ?4 S1 V; y6 Z1 E e: E
int abcd = GetLastError();
: p' D+ i: Z8 h0 m printf("Process token open Error: %u\n",GetLastError());
- Y; q+ M: s! ^/ F9 l2 g }
- O# ~4 ^+ ?3 c2 l4 h0 s, n) b DuplicateTokenEx(hPToken,MAXIMUM_ALLOWED,NULL,SecurityIdentification,TokenPrimary,&hUserTokenDup);//创建模拟令牌
/ g3 H2 m/ f# A; f int dup = GetLastError();) K# C; G* u1 n; W2 e
通过上面的函数我们得到了这个DuplicateTokenEx函数的最后一个参数后,我们还不能直接用,根据MSDN 的说法要用SetTokenInformation来设置一下
; @, Z8 }+ |# b7 I5 U! T* `" B+ A9 }: J) b% e9 T
BOOL WINAPI SetTokenInformation(
0 l% M3 L6 E5 a6 L% p/ I4 f7 o __in HANDLE TokenHandle,6 k) S# V4 s- Z2 @
__in TOKEN_INFORMATION_CLASS TokenInformationClass,% O+ U# p1 D$ ~6 ^
__in LPVOID TokenInformation,
; Y' B {7 J- T- }, x9 g __in DWORD TokenInformationLength
( Q h. T7 h7 L6 n4 ?);
1 @; d1 j$ I7 b; e& c其中第一个参数就是我们得到Toke 的句柄,第二个参数会话ID ,也就是把会话ID 设置到备份令牌中,代码如下:
( P- G1 q b/ s
. d- v- k% n) m8 l) V* ]7 uSetTokenInformation(hUserTokenDup,TokenSessionId,(void*)dwSessionId,sizeof(DWORD))8 A3 S3 `( L Q" d9 C5 j
设置完成后 最好检验一下,当然不检验也没有关系了,代码如下:# }* @- L& D% G# \
3 u: {" n8 z v5 [- ~0 C9 _) a
if(!AdjustTokenPrivileges(hUserTokenDup,FALSE,&tp,sizeof(TOKEN_PRIVILEGES),(PTOKEN_PRIVILEGES)NULL,NULL)) //这个函数启用或禁止指定访问令牌的特权 _- ?8 h$ v9 Y6 b# X$ s! A
{
5 S6 V; E# Q. w int abc =GetLastError();
/ D% ]' n, k# {* x7 C m4 C printf("Adjust Privilege value Error: %u\n",GetLastError());
( p2 i/ ]* P, p* R }
[# V3 g: N* | if (GetLastError()== ERROR_NOT_ALL_ASSIGNED): j* o/ L- e( n1 e
{2 v( @0 W* q; D* o- G
printf("Token does not have the provilege\n");
. }* ]; i% R$ g; q3 c' t7 m& l- k) Y }% i! h8 K# f) o: q$ d' w: g5 t, x6 E
X2 X8 w1 X8 J# ~; g' F8 X) H这样的话CreateProcessAsUser第一个参数的工作算是正式完成,下面几个参数就简单了,5 l1 }; v4 z( y8 M; j& D2 V. i
# B9 h3 Q- x- P8 D
大家直接看代码吧
. y0 n* E, J" f6 j. g" i; k2 a3 u) [4 L; v0 V1 |
bResult = CreateProcessAsUser(& o3 F0 O i* c
hUserTokenDup, // 这个参数上面已经得到
m: \! ^' t/ ]/ a! l _T("C:\\Windows\\System32\\notepad.exe"), //执行文件的路径* w9 P* E6 T! t& R- d6 p
NULL, // command line
5 } n! a7 ^2 o$ E' J; ?( r1 B, Q NULL, // pointer to process SECURITY_ATTRIBUTES5 k! b/ R4 x. A$ \* I0 V+ k/ E, ^
NULL, // pointer to thread SECURITY_ATTRIBUTES# ~& o7 Q; `: i, j7 l
FALSE, // handles are not inheritable1 t8 ]4 o6 ?+ c$ L8 A' _
dwCreationFlags, // creation flags' J# h0 [6 R6 o% S) B$ o9 R8 A
pEnv, // pointer to new environment block3 A- C" d k8 i7 E! e+ o7 H; H* A) t
NULL, // name of current directory
, n. V( }( |* k, t2 o6 d9 k &si, // pointer to STARTUPINFO structure
3 x( U0 h# Q0 w &pi // receives information about new process9 a9 U4 C2 t7 F# c* B* O
);
" l: ?3 M/ X6 m4 ]3 J+ D$ N n
; |& N$ m0 W( o6 J9 L' U/ _" Z( o+ w# a" X6 {, t& h* K; G' e
最后大家别忘了把打开的句柄关闭。这样看起来是不是就有点简单了,其实也没什么,剩下的就是Winodws 服务的基本框架的一些东西,那些就简单了。 以上都是自己根据MSDN理解得来的,如有什么问题欢迎大家指正!
' Y: X% M1 H( g% O! d
1 q- ]7 C; _1 U附上Demo下载地址:http://download.csdn.net/source/3497793 |
|