数学建模社区-数学中国

标题: 如何自动登录linux[转载] [打印本页]

作者: huashi3483    时间: 2004-9-27 14:26
标题: 如何自动登录linux[转载]
<>如何自动登录linux[转载]</P>9 o4 y, I- y) Z/ T2 ]- v6 g7 j! ]4 P
<>本文以redhat 8.0操作系统平台为背景,阐述如何实现启动级别为3时的自动登录,及自动运行相应程序,并简要介绍了如何在redhat 8.0下自动登录X window(系统启动级别为5),并自动运行指定的应用程序。
8 u* H9 ^8 s& c7 C! b: i. [& D9 p[b:804aea7097]一、启动级别为3时自动登录的实现[/b:804aea7097]
* V% V7 J- C( C3 ?# R2 v( d) b9 K' Z" h
启动级别为3时自动登录的实现涉及两个软件包:mingetty-1.00-3.src.rpm软件包及util-linux-2.11r-10.src.rpm软件包。 : O; D$ C8 V7 y  ]! A5 y$ D& q

0 ?0 Q1 s& O9 i6 d6 n2 L8 B* B(1)mingetty-1.00-3.src.rpm软件包  " f8 e5 k, C$ N7 E/ j1 D+ R$ R
对于启动级别为3的自动登录的实现,仍然需要考察/etc/inittab脚本,  
8 y- u7 N6 q+ s( `3:123:respawn:/sbin/mingetty tty3
. P2 s$ ]) p( B+ g4 J
4 A9 F7 H% W* n% }) H3 u
" Q" y. Y! Q. F# i' L$ E  ! E5 n0 N: q4 }

0 `! b  I3 `; y* p) t9 ]! o% c8 k# Q; Y( D  L) V) S
因此,如果想在启动级别3的情况下实现自动登录,必须要了解mingetty的功能,甚至要修改mingetty的代码。用命令rpm -qf /sbin/mingetty 可知redhat 8.0版本的mingetty包含在mingetty-1.00-3.src.rpm软件包中,下载该软件包,安装源代码,缺省情况下,代码会安装在/usr/src/redhat/下,我们关心的只是mingetty.c源文件。mingetty.c约有五百行代码,主要实现如下功能: 3 ?0 _& s( U7 M) I7 v/ r0 X2 o
4 e, Z1 t% E* G9 M7 \0 m* ^

) j# r* }/ v# f% g2 G打开指定的tty(由参数指定);  
# N+ S, u9 f! N+ W& K提示用户登录(login:);  1 V' R/ g1 e- Q$ u! c
获得登录用户名;  8 {2 U( Y8 q; T
把用户登录名作为参数,调用/bin/login。  6 ^; _1 p5 E* z; n) _8 n/ t

% G% b4 v! m+ U5 K) |我们所关心的部分实质上只有以下三行:
. U5 {3 H& N# R
0 |/ x* F7 w# U, V0 ]. _& F0 |7 v' r6 ?
... ... ) z4 D( K6 J( c1 O
438 while ((logname = get_logname ()) == 0); //mingetty.c文件438行 ) Z/ L4 c1 p+ e9 J  w
439 execl (_PATH_LOGIN, _PATH_LOGIN, "--", logname, NULL);
! e# ?+ l* ^8 S7 K8 Z- J% V440 error ("%s: can't exec " _PATH_LOGIN ": %s", tty, sys_errlist[errno]); - X+ u/ S! d0 P, Q" ~9 _
... ... / T6 u& D7 g7 A4 e
- X/ \- S  [9 ~4 w2 Z  s( B- }
  % N& n' p( t# d9 Y; r* m4 [1 }+ R
4 A" |# o8 C1 u, Q8 B6 q; A7 F/ M; W
第一行的功能是输出login提示,并获得用户输入的登录用户名,登录用户名由logname返回。因此,可作如下修改
5 b0 `4 p0 F2 e/ ?2 ^+ q6 B' w1 i
  s/ T: H( Q4 I: \
... ... 2 P  X; u' e% [- v' K2 w0 v
438 // while ((logname = get_logname ()) == 0); //注释掉本行,不再提示login:
; c5 O% ?% K* [2 q: b$ E8 R+ ]439 logname = "root"; //添加本行代码
+ N6 ]# U, \' H8 t440 execl (_PATH_LOGIN, _PATH_LOGIN, "--", logname, NULL);
/ k0 a5 a% T# V# t( d' m9 w441 error ("%s: can't exec " _PATH_LOGIN ": %s", tty, sys_errlist[errno]); 7 A1 S) J/ _2 w8 A5 D( [
... ... 2 w7 ?7 I7 F2 v5 T; {9 w
" C: Z) O$ y* g0 L+ U5 c
  ' }2 O3 y( H8 ~# s0 d+ ]
' m* E# l) e2 V. K7 l. _% R0 Y
注意,这里假定用户以超级用户身份登录。
) q$ j: |& q: z# [' N7 \: y
5 a; K+ l! e7 l; E2 z: P第二行以用户登录名为参数,调用/bin/login程序,进一步实现登录。因此,要想实现自动登录,还应该了解/bin/login的功能,必要时还应修改其源代码。 , D) y6 `" x0 i
0 d6 f8 E. \0 U& Y
第三行为出错处理。 0 J3 Q, l% ~/ V* B! Y, N

! z6 u" j, M; P3 q(2)util-linux-2.11r-10.src.rpm软件包  5 u9 P' j% e$ n+ H- h+ R, g+ }' y
采用同样的方法,查看/bin/login所属软件包(redhad8.0版本的login包含在util-linux-2.11r-10.src.rpm软件包中),下载并安装util-linux-2.11r-10.src.rpm,login可执行文件有几个源文件编译而成,我们最关心的是login.c源文件(大约1500行的代码)。下面简要分析一下login要实现的功能,并对相应部分作必要的修改。   e4 c1 l; K# n- j* X2 u% p1 D* m$ m
+ u9 ], r& Y9 [8 G/ p
Login程序主要可以分为以下几个主要部分:
- u7 s8 X3 `1 h8 W! `- P4 P2 s
% I1 |9 k2 R! R6 B( i9 h1.Login首先检查登录者是否为超级用户,如果不是超级用户,并且存在/etc/nologin文件,则输出该文件内容,并中止登录过程;主要由checknologin()实现;  ; x' r7 B! {$ L& x
2.如果登录用户是超级用户,那么login必须在/etc/securetty/中指定的tty列表中实现登录,否则将导致登录失败。同样可以不指定/etc/securetty文件,此时,超级用户可以在任何tty上登录。  ! _/ ~3 O2 k! K& \9 ^6 l& A" {( D
3.经过前两步测试后,login接下来将提示输入登录密码(由getpass()调用完成,有兴趣的读者可参考其手册页面),并进行验证,如果密码不对,则提示重新登录。  
. u4 A. J6 x* |' b4.顺利经过密码验证后,login还将检查是否存在.hushlogin文件,如果该文件存在,则执行一次"quiet"登录(所谓的quiet登录指的是,登录时不再提示邮件mail,不再显示最后一次登录时间,不输出任何消息。启动级别为3时,正常情况下输出这些信息)  
4 \  ]  T" g* L% |$ E' y5.login接下来设置登录tty的用户ID和组ID,并设置相应的环境变量,包括HOME、PATH、SHELL、TERM、LOGNAME等。对于普通用户来说,PATH缺省被设置成/usr/local/bin: /bin/usr/bin:;对于超级用户来说,PATH被设置成/sbin: /bin: /usr/sbin: /usr/bin:  ! |. t# x8 U4 I" o" M
6.login的最后一步是为用户启动shell。如果在/etc/passwd中没有为用户指定shell,那么将使用/bin/sh,如果在/etc/passwd中没有给出当前工作目录,则使用"/"。  : G. n0 U9 Q9 Q' {4 [
至此,一个完整的登录过程就结束了。 3 p# v& n( P4 ~( o- A( q( J: w
$ v, f0 ^0 ~' ?1 u; L
从以上对login源程序分析过程中可发现,如果要实现自动登录,应该在第三步做文章,设法绕过提示输入密码以及对密码进行验证这一过程。实际上很简单,login源程序对是否要求输入密码设置了一个开关控制passwd_req,缺省情况下,其值为1(passwd_req = 1),即要求输入密码进行身份验证。把该行代码改为(passwd_req = 0)后,问题就解决了。即对源文件作如下修改即可: % B3 O3 k8 P7 Y/ x* a7 O7 P

3 y0 m! G: m" n& z+ o2 @6 r+ U. D/ b" b* _4 N
... ... & V% M, j! \( i) P8 \! }
402 fflag = hflag = pflag = 0; //login.c文件402行 ; H; Y. m- K8 j9 f
403 //passwd_req = 1   //缺省时,要求进行密码验证,注释掉本行
( r% N- S1 r6 E# ]5 s404 passwd_req = 0 //添加本行 * e" d$ k; M- @+ T& _& j( k
... ... & g" `+ N. b  l$ @* D1 y* P

0 n! |( o, A/ X1 {3 z  / m# p& g0 I+ X. L
2 Y) \  @# r/ C# |. f
修改后,可以直接使用util-linux-2.11r-10.src.rpm提供的Makefile进行重新编译,也可以自己对其编译:
' E: a4 P' t- i4 S) H
( V" J6 ~2 f  L6 y2 g
" c% g/ }5 ]( p8 N8 L5 ygcc -o login login.c setproctitle.c checktty.c xstrncpy.c -Wall -lcrypt注意包含后面的编译选项-lcrypt,否则会出问题。 3 c4 Q% q1 w0 W6 u+ S+ `1 A# x

+ M7 T# O7 K/ p
& W! ^" X% V# y# \7 g有了新版的mingetty及login后,拷贝mingetty到/sbin/目录,拷贝login到/bin目录,并将/etc/inittab中的启动级别设置为3,再重新引导系统即可(读者可以自己写一个脚本实现上述过程)。
' s% u1 E. r. ^: d6 g
2 P4 |2 @$ ^& W% m5 q: v* D* V如果读者对mingetty或login代码的其他部分感兴趣,可以反复修改login.c或mingetty.c的源代码,测试一下代码的功能,这里要注意的是,在拷贝新版mingetty和login之前,一定要把原来的mingetty和login备份,同时还要准备系统引导盘(有系统安装盘亦可,这样读者有机会键入linux rescue),在测试新版程序前更应如此,如果对代码修改稍有不当,系统将不能正常启动。
8 \- p) W, [8 j" y1 x
* A# v4 d* L6 w; h( x如果不想再作进一步的代码测试,只是按本文给出的方法进行代码修改,在系统启动上不会出现什么问题。 " h4 h1 r  s  V+ q+ f$ j2 v

) ]8 G5 z2 }+ Z; z5 v7 x3 s[b:804aea7097]二、自动登录后,自动运行特定的应用程序[/b:804aea7097]   E# X8 L2 U/ X4 v" X# B3 t

; Y5 t& b* d. k9 H$ w- l7 d在实现了启动级别3时的自动登录后,自动运行应用程序非常简单,把应用程序添加在/etc/rc.d/rc.local脚本中既可。(读者可以尝试一下把startx加入脚本中,看一看效果如何。在某种意义上,又增加了一种自动登录X window的方法)
: A! q* {: j, W0 p6 X, P
9 k; B/ R- s2 L* j& }[b:804aea7097]三、对自动登录X window(系统启动级别为5),并自动运行指定的应用程序的补充[/b:804aea7097]
" g, m+ ^+ N- F$ i+ n0 ^/ G5 P
3 e4 K  ?. V+ O! {7 W  r在"如何实现自动登录linux"中,主要以redhat 7.2平台为背景进行阐述的,其中的自动登录部分可以直接用于redhat 8.0中,不需要任何修改。
; C$ v; J2 ^% @3 }9 U) T
/ R7 ]# T" T! |6 x3 Z# y1 U' N但是,登录后自动运行应用程序的接口在redhat 8.0中有所不同,主要是登录gnome后,自动运行应用程序的接口有所改变:首先点击面板上的GNOME帮助(那个红色的小帽子),然后选择/其它/首选项/Sessions,在Session对话框的启动程序属性页中添加要启动的程序即可。 ' b* O5 T) [5 I( n

( i1 n4 f! b' o  G( R7 F对于登录kde后,自动运行程序的接口没有改变。
! P2 R* }5 W; E& R  |. z  C( K3 S) B
[b:804aea7097]四、结论[/b:804aea7097]
' w, b3 ^8 B4 L' Q+ N5 r8 f7 N* }8 N& M& o# z# F' \
本文同"如何实现自动登录linux"一文,基本上解决了如何实现自动登录Linux,并自动运行相应应用程序的问题。对于两个最常见的启动级别(3、5),都给出了各自的方法。
/ P( S; j9 U( i1 V# B% v% @( H; V! v5 e) l# D
在系统初始化到mingetty及login这一阶段,内核实际上已经完成了引导过程,已经到了系统初始化的最高阶段,与内核没什么关系了。此时,主要是/sbin/init根据/etc/inittab的内容而相机行事。读者可通过(man 8 init)或者(man 5 inittab)了解更多东西。
% }, r4 Q( D+ r! z9 u7 c. ?$ ~5 i. O7 a! ~1 s, J$ n
在对文中提到的软件包修改时,请遵守GNU General Public License(GPL)相关标准,另外,替换login通常被视为黑客行为,应当谨慎行事。 # ]& e# i! y! x0 s. t
' B$ z) C4 p( Q1 N& z7 I' x1 b5 H
: T+ @7 x- ?+ o1 i0 Z
9 g5 J; r0 \) a) B+ q
[b:804aea7097]参考文献[/b:804aea7097]
2 F7 j4 P- ?& e/ c- p8 N  R) X( o3 ^# |5 p

- ]1 Y: W' z4 N1.login手册页面  0 F7 `, i8 {! _* w9 m
2.mingetty-1.00-3.src.rpm,在redhat 8.0的发行版本的源代码中,包含该软件包;  
. i0 a7 k# @6 i2 k) v. t3.util-linux-2.11r-10.src.rpm,  # T9 N, O  _8 y; K/ _

2 \; L6 E& t5 L7 a3 T可在http://rpmfind.net/linux/RPM/redhat/8.0/i386/util-linux-2.11r-10.i386.html处下载,注意下载源代码包(..src.rpm)
9 ^, h4 W2 e( R9 b9 I. m% f, V0 q7 U& c3 R+ L; H4 I
[b:804aea7097]关于作者[/b:804aea7097]  / d/ n4 F6 I, \  Y' h
# c3 `/ ~0 c! h5 O9 e; c+ c$ G
郑彦兴,男,现攻读国防科大计算机学院博士学位。您可以通过电子邮件 mlinux@163.com和他联系0 t' K" K/ t. i" O

* E! o) p) U- ^) `5 I; A, I  v</P>
作者: huashi3483    时间: 2004-9-27 14:27
如何实现自动登录Linux (Runlevel 5) , T$ Q& B! X* V* Y7 B) ]1 G2 w
6 x$ B0 L4 L: E
机自动登录linux,并自动运行X window应用程序,有其特殊的应用背景,如基于linux平台的监控系统,linux启动后不需要身份验证,而直接运行监控程序等等。本文以Redhat7.2为平台,结合linux启动过程,介绍了如何避免身份验证自动登录,并直接进入X window自动运行应用程序。
, A0 \) K5 k5 T. X* `一、linux启动的最后阶段的工作 # _! i! x9 L: c$ ~, I" h

: x7 v: y1 x3 d1 ~3 olinux 在启动过程的最后阶段(具体启动步骤略),init 会根据 /etc/inittab文件的最后一行x:5:respawn:/etc/X11/prefdm -nodaemon运行/etc/X11/prefdm脚本,(Red hat 7.2缺省时是这样的)。prefdm脚本的主要任务是完成X window的启动,可以有几种启动X window的方法,都包含在prefdm脚本中,几种主要方法有:
: Z! Q' Y& Y% I& C  Y/ D
* W+ |$ _' h) T运行xdm启动X window;  
9 h- \* \- e& T8 O1 `  O% N运行gdm,进入gnome桌面环境;  
* J+ A1 `3 F! u% }6 b9 V% ^运行kdm进入kde桌面环境;  
: k( Y  u3 C$ z  e' ^8 a自动登录进入linux;  
8 x3 z9 ^+ b' J2 t! h1 H) y7 I$ pprefdm脚本框架大致如下: 5 h' P9 [! L1 y6 D

( i& M. i6 a  X' R: v( p9 j5 j9 T$ H#!/bin/sh 5 Q# J' j/ A" Z! @( J! C5 U6 Q
PATH=/sbin:/usr/sbin:/bin:/usr/bin:/usr/X11R6/bin
/ v0 w7 L* q$ T" d8 P) g! @8 A2 W5 T. /etc/profile.d/lang.sh
, n) v% O* z' h" Q# 第一步:查看是否为自动登录
6 C0 O, ?. G" }' i  u3 ], \" s0 eif [ -f /etc/sysconfig/autologin -a -x /usr/sbin/autologin ]; then
4 {4 m/ g& P, ^+ N* Y! qif /usr/sbin/autologin; then
; R9 u0 K3 v7 m+ S' sexit 0
+ c5 ^3 t1 m- |  _3 u' m: o) I6 pfi
* ?; |8 V# r; Rfi
, v9 k- H! `6 a, ], K0 B
; D2 q3 n) w3 M% K# 第二步:如果不是自动登录方式,就会在/etc/sysconfig/desktop中搜寻用户偏爱的登录方式
. ~& C1 u( |- s; @' a......
* Q( |1 \# }7 c7 i4 E) |; x# 可以是kdm、gdm以及xdm,并运行相应的kdm、gdm以及xdm。 4 T- r+ z, ^# [9 o6 @+ R5 P2 b
...... " k4 M8 I' h  z% A" j

, a- h2 _) ]9 o5 y. G/ q4 [9 [: H! K, m  
5 j! ^$ k4 Z" \* Y& T3 c% E/ i* A9 s

' i$ N! F( R0 f! {. X" m, c' e二、自动登录的实现(autologin的实现)
% j, _% n8 u- e2 D: X- C8 w2 \  `) c( E% E, B
在/etc/X11/prefdm脚本中,是否实现自动登录有一个条件测试开关,事实上,可以在这里注释掉测试开关,直接执行启动X window的操作。
' C; V1 N8 Z2 R* M: S/ Z& J0 b! w6 f- n& l6 o
自动登录实质上就是绕过身份验证,直接启动X window。X window的启动可以由xinit来完成。
, K6 d7 ]. t( {& y9 d: X, q! z" B/ \8 |8 @. g/ }
Xinit用来启动X window系统服务器以及系统上的第一个客户程序,可以通过为xinit传递命令行参数的形式指定要启动的服务器及客户程序。如果不传递参数给xinit,它将在用户的根目录下寻找并运行 .xinitrc脚本来启动客户程序;在用户的根目录下寻找并运行 .xserverrc脚本来启动服务器。如果xinit在用户的根目录下找不到.xinitrc、.xserverrc,xinit将使用缺省的X :0。  
" {( X- g* c; D  I. r! b3 z5 }实际上,用startx来启动X更为方便。对于运行单一会话的X window 系统,startx提供了更为良好的用户接口。同样,startx首先在用户的根目录下寻找 .xinitrc及.xserverrc脚本,如果找不到这两个脚本,startx将使用/etc/X11/xinit/xinitrc以及/etc/X11/xinit/xserverrc脚本。  
" x- j* O% h+ x: f: T1 istartx脚本的最基本框架是:
0 }, X* @! v" h* k: Y/ i; M" A+ Y/ ~9 j1 K" r9 Z6 Q/ V
a、 寻找.xinitrc,如果没有则使用xinitrc;
" M' Y3 z% p, T5 B$ O+ N) wb、 寻找.xserverrc,如果没有则使用xserverrc; / R2 z$ Z) R4 F, N
c、 根据找到的脚本确定xinit的参数; : Z& H2 G# J+ B8 x9 F

# f1 R2 `0 q1 B, t& \, O  i* W# b, S+ K由此可看出,startx在不需要传递任何参数的情况下,可以完成启动X的任务,因此,可以如下修改/etc/X11/prefdm脚本来实现自动登录:
# [9 G  M" O8 R; d$ V( f
: u- }6 I& B/ h0 S7 u' U; A% J/ J/ e  t+ g  i
#!/bin/sh
+ u5 J' b$ L$ z6 r2 w3 hPATH=/sbin:/usr/sbin:/bin:/usr/bin:/usr/X11R6/bin
7 Z! O2 \* D8 m4 M. /etc/profile.d/lang.sh
( E' \$ U5 T" h6 x  A/ R# 第一步:查看是否为自动登录 0 [: \" @( i2 h) K
#if [ -f /etc/sysconfig/autologin -a -x /usr/sbin/autologin ]; then
" I1 l5 h* i( ]8 k, ^7 L#注释掉上边的条件测试,直接运行startx : z% Q) b( _# \2 W3 k4 h9 y
if /usr/X11R6/bin/startx; then
0 K* b& r1 K6 J& j5 rexit 0 3 }* T( B3 Z" F8 a" S$ F* J
fi * z2 r5 n- v0 A
#fi
  l2 [4 |5 b: B) |
" g5 ^8 @8 k* ]8 `. z! I/ z. O" f  9 C. a6 o$ {- Y! F' _5 M* F7 `

. c/ J/ W* Q6 j5 ?$ J. ^! R; P) X+ a: I
/ M8 k2 o( m% K7 I. t; R7 {/ X4 p当然,应确保/etc/inittab中的启动级别为5。
! o3 [  l7 m4 O
3 T" Y; o# F# L' a7 n5 Z重新启动系统,会发现系统不验证用户身份,直接进入X window,此时的用户身份为root。但是,如果原来root有自己的桌面、默认shell时,上述方法启动X不一定保证还能拥有原来的设置。为了在启动X后,在避免验证身份的同时,又不改变用户原来的设置,那么在运行startx之前,还有工作要做。
/ P" X8 q3 @/ g' X5 Z
% T. e9 \8 o" t! O# y5 l5 {" m, j三、自动登录后,保持用户原来的配置(桌面、shell以及其它的一些环境变量)。 , d+ V" D7 W! J3 o
' k) m' R  y, S8 k' M! o3 o
观察原来/etc/X11/prefdm脚本的自动登录部分:
, M$ Z+ _% X, C( X4 R5 B+ Z- l6 }
: ~# d9 V( c4 \' m8 K1 ^; ?# o5 k! ]; o6 m& R
...... ! U2 H4 n3 @4 _6 c$ H
# 第一步:查看是否为自动登录
8 O  g' t0 _; E; d+ A1 d/ Wif [ -f /etc/sysconfig/autologin -a -x /usr/sbin/autologin ]; then
, f. u9 W0 Z3 M5 \) F0 U8 B; Jif /usr/sbin/autologin; then
8 @# L  U* d1 L6 l2 {3 Lexit 0 3 m3 ~. E0 r4 S  j
fi
% ^$ G: F9 I7 m. l6 Tfi ( k6 n$ O+ [; K7 Z1 \
......
2 W5 v) c' u, l9 z3 l, q% k5 d- j3 `: f5 z  Q: b! V
  
6 s' Q6 h: v. |- D2 u2 J! o% j: p1 R( b! j: l7 }5 E2 L

5 L  u& t3 o1 P' X' {" }不难看出,脚本中保留了自动登录的接口:一个可执行文件/usr/sbin/autologin以及一个配置文件/etc/sysconfig/autologin。 8 L, E/ T8 p2 c% V1 o" e

' h4 ]# x; Z. i* S8 a* A: {( }1、/etc/sysconfig/autologin配置文件的实现: 3 K' e+ o' W9 l1 V/ A

# ]5 N) G% B4 B. T# `- T4 `8 B, g* |8 }$ s
#config for autologin
2 _. l- `7 K8 X0 c3 C+ `USER=root
2 _& J0 i6 n- r( WEXEC=/usr/X11R6/bin/startx . [. j6 R! Y9 b6 N" ~3 F( A- a
说明,USER指定自动登录时的用户名;EXEC指定启动X要运行的程序。 - u) `6 I0 s" W5 B8 ]* [7 g2 T: s, Z* x2 y
: T# J% s6 C: \
    d" O: C. n4 {/ K3 n
9 l# G' @$ D, n

; @* ?! G! R: n" o* G  {5 ]2、/usr/sbin/autologin可执行文件的实现:
* l/ r  Z2 O0 @! L0 f, p( L$ z7 d$ T0 f7 J7 K

: o8 v0 `  Y! ?( }% Y; t/********************* 8 l$ v) a+ ]( ?! U$ @
****  autologin.c  **** ) S; G) p9 X! B" T: G9 s) Y, f
*********************/ 0 ?: K$ C' [% h# q' s
#include &lt;stdio.h&gt;
8 \7 F. q7 e) \#include &lt;unistd.h&gt; ' C. W" o1 u4 L# V3 L# D
#include &lt;string.h&gt;
4 @" e/ P+ u& p- ]#include &lt;errno.h&gt; 0 e3 {$ L7 h/ n0 M- a
#include &lt;sys/types.h&gt;
* ]- F' r! Z- @/ p; X8 d#include &lt;sys/stat.h&gt; 7 q# K) [1 w1 K' h
#include &lt;pwd.h&gt;
. }/ ]( {5 q3 k' _% U8 ?
" h7 L- B2 Z4 d- eint main(int argc, char **argv) 0 J5 |0 p7 I6 f! Z, K" ^
{ ( ?2 n$ }7 R$ ^- |6 a3 \: o  j5 K' S3 t
struct stat st;   o) O$ ?$ ], U7 v3 Y1 X
FILE *f;
7 B; I8 z- P. g- Q1 D$ O0 bchar *cfg; $ O6 O# L  E. K8 z/ Q/ W
struct passwd *pw;
) c, Q2 v& z: t8 g+ y) Buid_t uid;
- B7 p0 ?/ ?. q- Qgid_t gid; : j- g* V: @+ ]* b) K+ b+ U
char *dir, *shell; ; `/ q) e3 H+ }( w: x% X
char *user=NULL; 1 V1 \( \' ~4 w; D( `
char *cmd=NULL; , d3 r7 |- `& l, F0 J

* u, r+ R* X4 k& F: ouser="root";
4 {; b" b& c$ E- f/*为了能说明问题又保持程序简洁,这里默认登录用户为root,实际上, $ y0 J8 ?! m% [# u+ F9 O
登陆用户名应该从/etc/sysconfig/autologin中得到,
8 x6 l& i# Q. k' f+ h# ?. h4 t程序实现时要注意过滤掉/etc/sysconfig/autologin中的无效用户名*/ ! @( s- S* o$ z8 M8 a9 y- ~
cmd="/usr/X11R6/bin/startx"; 1 @+ b0 t8 k- ^8 H' ]: P* \
/*同样,这里直接指定启动X window的程序,实际上,该程序应该从/etc/sysconfig/autologin中得到*/ $ f. }2 H: @' j: Q) B7 T9 h) D$ B' I6 Q

8 i# G, K! W+ N0 i2 m& }pw = getpwnam(user); + `: _- U: M+ e4 Y! I
//getpwnam返回包含用户信息的passwd结构(该结构在pwd.h中定义)。
% C" z8 e0 [0 O4 y  o  Yif(pw) { ! {2 @, W/ G- c  u5 r9 d
uid=pw-&gt;pw_uid; $ C- a4 i; j3 z; _5 J0 Q4 \, l
gid=pw-&gt;pw_gid; dir=strdup(pw-&gt;pw_dir);
# G9 q# r8 H' l0 |# h3 E# Q& O* Dshell=strdup(pw-&gt;pw_shell);
* O/ {  y  e; R, W2 B# m. k! Z}
; q; {& A& J) e4 v# m3 c//获得用户相关信息
' m2 \- G# X, U else { + y" R! I5 N& q8 d6 E; O' N4 |
printf("ERROR: No such user %s!\n", user);
& f, u! u# d9 z/ e) X, t0 preturn 1; ) A/ u/ P  ^5 K
}   I( t1 h& K, A7 J% `
) w& y7 D7 |0 ?3 j& U/ j/ {6 I6 d6 @
chown("/dev/console", uid, gid);
7 T- f' M5 I* q& N$ Wchown("/dev/tty", uid, gid);
4 t: q* |, i8 v* a9 k' g/ I//为控制台和终端设置用户ID及组ID ! R, e# {% }, J7 M7 g

9 q* B1 A6 q% d* b+ S+ u5 f# D//下面是设置用户相关ID ; J/ t' s+ W' W: u7 N+ }; e
setregid(gid, gid); 1 o6 `* i3 m9 Z1 `. B% t+ ?# W2 G
setegid(gid); ! V6 L/ g6 b# J$ b8 |. G* l
setgid(gid); $ s# }! x3 U( X! U' d3 o
setreuid(uid, uid);
+ X, V! }; O3 g1 |8 e" |( zseteuid(uid); ( _% Q, K! u/ I& X$ ~$ _
setuid(uid); % y8 F, t' r- N' T/ `# \( x
8 P$ k) X! A  I0 ^3 M
setenv("HOME", dir, 1); 9 g5 h) I) V8 b' P" x' c1 \
setenv("SHELL", shell, 1); 1 K' h7 c( V( p6 B" `. V
setenv("USER", user, 1); * w  H6 B/ G& E0 \
setenv("LOGNAME", user, 1); * J: D3 }* T/ [; d8 m$ F# ^
//设置用户相关环境变量 & \+ \1 N. W: E* _
3 h0 ~. X) U  Y9 {& ]  _( w
chdir(dir); ) T' ?# L* ^& J  v; c* a/ x
//切换到用户根目录 # p/ l: @+ z( c
user=NULL; 7 O. l# J7 h+ t: V  F
) ^  o+ E0 u: H: w
execvp(cmd, argv); 0 X. m0 u: y' [' B- [/ |1 E; g4 F
/*在配置完用户的相关信息后,执行启动X window操作。注意这里默认执行/usr/X11R6/bin/startx */
! f$ G* c$ W& d; `) q: m+ e3 o; ^  e' Hprintf("ERROR: Couldn't exec %s: %s\n", cmd, strerror(errno));
. O, l. {5 _2 c& b( P6 lreturn 2;
/ n3 y! q8 ]0 G) F' M}
4 n; ~% U3 j$ T) }1 f1 S  [; P# ^% Q0 g* b+ _9 P" X! t
  * H6 C7 M8 @8 w5 p) Q1 R% Y

- n9 f6 r' y6 }2 V2 Q
! K4 T) J: I1 h" t运行gcc -o autologin autologin.c,拷贝autologin可执行文件到/usr/sbin/autologin,拷贝autologin配置文件到/etc/sysconfig/autologin。 重新启动系统,会直接进入X window并保留用户原来所有的风格。 & d/ p$ n0 i! q! a

. x5 g: z9 b* X如果不需要自动登录的配置文件/etc/sysconfig/autologin,所有的操作都在/usr/sbin/autologin以默认的方式实现(比如,默认登录身份为root,默认执行操作为/usr/X11R6/bin/startx等),那么,/etc/X11/prefdm脚本的自动登录部分可简化如下: - h; X* A% a( l& Z) H& _  l
" A! D. g0 W; `( v3 b/ e' Z
$ j2 j( R3 K6 q0 r( Q! ?
...... 5 h; W- k- K: X$ E* [4 y; Z' f& T
# 第一步:查看是否为自动登录 8 @4 l! `$ e8 L2 h: Z  y( y
if /usr/sbin/autologin; then
! g* o. h+ F) |& nexit 0
/ n# J) L! G) d/ Y" ^( g' Qfi 9 f6 W+ @% Q1 A& w
//第二步 ...... & N# Q4 B7 y# {9 K- L  [0 V! O; W; e
......
8 b) S" G0 B4 ^) R( A. F5 v/ Q$ N- h6 I% m: J0 O, H
  4 r: W" W. I0 [! ?3 {

! ~0 g; {  x/ n  [/ O, B: U9 |$ R  x7 ~
即在脚本中去掉条件测试开关,直接执行/usr/sbin/autologin,这时,只需要拷贝autologin可执行文件到/usr/sbin/autologin,不再需要拷贝autologin配置文件到/etc/sysconfig/autologin。 : s: q' E- l3 T+ u3 D: ]1 @
, A+ Q4 `0 X6 a+ D- v- E0 p: {1 `
四、选择进入kde或者gnome,并自动启动X window应用程序 0 d: X9 q6 k8 ~- |; L
  p, n7 k6 E* S2 V  x% ?% W
如果重新启动后系统进入了kde,而用户需要进入gnome,只需运行switchdesk gnome在重新启动系统即可,以后每次启动时会自动进入gnome;反之亦然。  
) ]# G  Q1 f4 U7 Q5 K7 g& n3 K一般系统自动登录的目的是启动X window 后自动运行某个X window程序。如果系统默认的启动级别为3,那么如果要在系统启动后自动运行某些应用程序,只需要在某些脚本中加入相应命令即可,不再详述。在X window启动后自动运行应用程序要复杂一些,幸好,kde和gnome都为此留下了自动启动接口。如果在kde桌面环境下自动启动应用程序,只需要把应用程序名字加入/root/.kde/Autostart/目录下即可(这里注意不同用户的根目录可能不同,如用户zyx的根目录可能为/home/zyx)。如果在gnome桌面环境下自动启动应用程序,只需把应用程序的名字加入/主菜单/程序/设置/会话/会话特性及启动程序的startup programs属性页中即可。  
& V7 c$ H' c9 Y3 _3 w0 z: M6 i1 ]7 O. ^
6 \5 E! ]1 e$ Z' J2 H
http://www-900.ibm.com/developerWorks/cn/linux/l-tip-prompt/tip20/index.shtml




欢迎光临 数学建模社区-数学中国 (http://www.madio.net/) Powered by Discuz! X2.5