14887Schin /*********************************************************************** 24887Schin * * 34887Schin * This software is part of the ast package * 4*8462SApril.Chin@Sun.COM * Copyright (c) 1982-2008 AT&T Intellectual Property * 54887Schin * and is licensed under the * 64887Schin * Common Public License, Version 1.0 * 7*8462SApril.Chin@Sun.COM * by AT&T Intellectual Property * 84887Schin * * 94887Schin * A copy of the License is available at * 104887Schin * http://www.opensource.org/licenses/cpl1.0.txt * 114887Schin * (with md5 checksum 059e8cd6165cb4c31e351f2b69388fd9) * 124887Schin * * 134887Schin * Information and Software Systems Research * 144887Schin * AT&T Research * 154887Schin * Florham Park NJ * 164887Schin * * 174887Schin * David Korn <dgk@research.att.com> * 184887Schin * * 194887Schin ***********************************************************************/ 204887Schin #pragma prototyped 214887Schin /* 224887Schin * David Korn 234887Schin * AT&T Labs 244887Schin * 254887Schin * Shell interface private definitions 264887Schin * 274887Schin */ 28*8462SApril.Chin@Sun.COM #ifndef defs_h_defined 29*8462SApril.Chin@Sun.COM #define defs_h_defined 304887Schin 314887Schin #include <ast.h> 324887Schin #include <sfio.h> 334887Schin #include <error.h> 344887Schin #include "FEATURE/options" 354887Schin #include <cdt.h> 364887Schin #include <history.h> 374887Schin #include "fault.h" 384887Schin #include "argnod.h" 39*8462SApril.Chin@Sun.COM #include "name.h" 40*8462SApril.Chin@Sun.COM #define _SH_PRIVATE 41*8462SApril.Chin@Sun.COM #include <shcmd.h> 42*8462SApril.Chin@Sun.COM #undef _SH_PRIVATE 434887Schin 444887Schin #ifndef pointerof 454887Schin #define pointerof(x) ((void*)((char*)0+(x))) 464887Schin #endif 474887Schin 48*8462SApril.Chin@Sun.COM #define Empty ((char*)(e_sptbnl+3)) 49*8462SApril.Chin@Sun.COM 504887Schin #define env_change() (++ast.env_serial) 514887Schin #if SHOPT_ENV 524887Schin # include <env.h> 534887Schin #else 544887Schin # define Env_t void 554887Schin # define sh_envput(e,p) env_change() 564887Schin # define env_delete(e,p) env_change() 574887Schin #endif 584887Schin 594887Schin /* 604887Schin * note that the first few fields have to be the same as for 614887Schin * Shscoped_t in <shell.h> 624887Schin */ 63*8462SApril.Chin@Sun.COM 64*8462SApril.Chin@Sun.COM 654887Schin struct sh_scoped 664887Schin { 674887Schin struct sh_scoped *prevst; /* pointer to previous state */ 684887Schin int dolc; 694887Schin char **dolv; 704887Schin char *cmdname; 714887Schin char *filename; 72*8462SApril.Chin@Sun.COM char *funname; 734887Schin int lineno; 744887Schin Dt_t *save_tree; /* var_tree for calling function */ 754887Schin struct sh_scoped *self; /* pointer to copy of this scope*/ 764887Schin Dt_t *var_local; /* local level variables for name() */ 774887Schin struct slnod *staklist; /* link list of function stacks */ 784887Schin int states; 794887Schin int breakcnt; 804887Schin int execbrk; 814887Schin int loopcnt; 824887Schin int firstline; 834887Schin int32_t optindex; 844887Schin int32_t optnum; 854887Schin int32_t tmout; /* value for TMOUT */ 864887Schin short optchar; 874887Schin short opterror; 884887Schin int ioset; 894887Schin unsigned short trapmax; 904887Schin char *trap[SH_DEBUGTRAP+1]; 914887Schin char **trapcom; 924887Schin char **otrapcom; 934887Schin void *timetrap; 94*8462SApril.Chin@Sun.COM struct Ufunction *real_fun; /* current 'function name' function */ 954887Schin }; 964887Schin 974887Schin struct limits 984887Schin { 994887Schin long arg_max; /* max arg+env exec() size */ 1004887Schin int open_max; /* maximum number of file descriptors */ 1014887Schin int clk_tck; /* number of ticks per second */ 1024887Schin int child_max; /* maxumum number of children */ 1034887Schin int ngroups_max; /* maximum number of process groups */ 1044887Schin unsigned char posix_version; /* posix version number */ 1054887Schin unsigned char posix_jobcontrol;/* non-zero for job control systems */ 1064887Schin unsigned char fs3d; /* non-zero for 3-d file system */ 1074887Schin }; 1084887Schin 1094887Schin #define _SH_PRIVATE \ 1104887Schin struct sh_scoped st; /* scoped information */ \ 1114887Schin struct limits lim; /* run time limits */ \ 112*8462SApril.Chin@Sun.COM Stk_t *stk; /* stack poiter */ \ 1134887Schin Sfio_t *heredocs; /* current here-doc temp file */ \ 1144887Schin Sfio_t *funlog; /* for logging function definitions */ \ 1154887Schin int **fdptrs; /* pointer to file numbers */ \ 1164887Schin int savexit; \ 1174887Schin char *lastarg; \ 1184887Schin char *lastpath; /* last alsolute path found */ \ 1194887Schin int path_err; /* last error on path search */ \ 1204887Schin Dt_t *track_tree; /* for tracked aliases*/ \ 1214887Schin Dt_t *var_base; /* global level variables */ \ 1224887Schin Namval_t *namespace; /* current active namespace*/ \ 1234887Schin Namval_t *last_table; /* last table used in last nv_open */ \ 124*8462SApril.Chin@Sun.COM Namval_t *prev_table; /* previous table used in nv_open */ \ 1254887Schin Sfio_t *outpool; /* ouput stream pool */ \ 1264887Schin long timeout; /* read timeout */ \ 1274887Schin short curenv; /* current subshell number */ \ 1284887Schin short jobenv; /* subshell number for jobs */ \ 129*8462SApril.Chin@Sun.COM int infd; /* input file descriptor */ \ 1304887Schin int nextprompt; /* next prompt is PS<nextprompt> */ \ 131*8462SApril.Chin@Sun.COM int bltin_nnodes; /* number of bltins nodes */ \ 132*8462SApril.Chin@Sun.COM Namval_t *bltin_nodes; /* pointer to built-in variables */ \ 1334887Schin Namval_t *bltin_cmds; /* pointer to built-in commands */ \ 1344887Schin Namval_t *posix_fun; /* points to last name() function */ \ 1354887Schin char *outbuff; /* pointer to output buffer */ \ 1364887Schin char *errbuff; /* pointer to stderr buffer */ \ 1374887Schin char *prompt; /* pointer to prompt string */ \ 1384887Schin char *shname; /* shell name */ \ 1394887Schin char *shpath; /* path name of shell */ \ 1404887Schin char *user; /* name of real user for pfsh */ \ 1414887Schin char *comdiv; /* points to sh -c argument */ \ 1424887Schin char *prefix; /* prefix for compound assignment */ \ 1434887Schin sigjmp_buf *jmplist; /* longjmp return stack */ \ 1444887Schin char **sigmsg; /* points to signal messages */ \ 1454887Schin int oldexit; \ 1464887Schin uid_t userid,euserid; /* real and effective user id */ \ 1474887Schin gid_t groupid,egroupid;/* real and effective group id */ \ 1484887Schin pid_t pid; /* process id of shell */ \ 1494887Schin pid_t bckpid; /* background process id */ \ 1504887Schin pid_t cpid; \ 151*8462SApril.Chin@Sun.COM pid_t spid; /* subshell process id */ \ 1524887Schin int32_t ppid; /* parent process id of shell */ \ 1534887Schin int topfd; \ 1544887Schin int sigmax; /* maximum number of signals */ \ 1554887Schin int savesig; \ 1564887Schin unsigned char *sigflag; /* pointer to signal states */ \ 1574887Schin char intrap; \ 1584887Schin char login_sh; \ 1594887Schin char lastbase; \ 1604887Schin char forked; \ 1614887Schin char binscript; \ 1624887Schin char deftype; \ 163*8462SApril.Chin@Sun.COM char funload; \ 1644887Schin char used_pos; /* used postional parameter */\ 165*8462SApril.Chin@Sun.COM char universe; \ 166*8462SApril.Chin@Sun.COM char winch; \ 167*8462SApril.Chin@Sun.COM char indebug; /* set when in debug trap */ \ 1684887Schin unsigned char lastsig; /* last signal received */ \ 1694887Schin char *readscript; /* set before reading a script */ \ 1704887Schin int *inpipe; /* input pipe pointer */ \ 1714887Schin int *outpipe; /* output pipe pointer */ \ 1724887Schin int cpipe[2]; \ 1734887Schin int coutpipe; \ 1744887Schin int inuse_bits; \ 1754887Schin struct argnod *envlist; \ 1764887Schin struct dolnod *arglist; \ 1774887Schin int fn_depth; \ 178*8462SApril.Chin@Sun.COM int fn_reset; \ 1794887Schin int dot_depth; \ 1804887Schin int hist_depth; \ 1814887Schin int xargmin; \ 1824887Schin int xargmax; \ 1834887Schin int xargexit; \ 184*8462SApril.Chin@Sun.COM int nenv; \ 1854887Schin mode_t mask; \ 1864887Schin long nforks; \ 1874887Schin Env_t *env; \ 1884887Schin void *init_context; \ 1894887Schin void *mac_context; \ 1904887Schin void *lex_context; \ 1914887Schin void *arg_context; \ 1924887Schin void *ed_context; \ 1934887Schin void *job_context; \ 1944887Schin void *pathlist; \ 1954887Schin void *defpathlist; \ 1964887Schin void *cdpathlist; \ 1974887Schin char **argaddr; \ 1984887Schin void *optlist; \ 1994887Schin struct sh_scoped global; \ 2004887Schin struct checkpt checkbase; \ 2014887Schin Shinit_f userinit; \ 2024887Schin Shbltin_f bltinfun; \ 203*8462SApril.Chin@Sun.COM Shbltin_t bltindata; \ 2044887Schin Shwait_f waitevent; \ 2054887Schin char *cur_line; \ 2064887Schin char *rcfile; \ 2074887Schin char **login_files; \ 208*8462SApril.Chin@Sun.COM int offsets[10]; \ 2094887Schin Sfio_t **sftable; \ 2104887Schin unsigned char *fdstatus; \ 2114887Schin const char *pwd; \ 2124887Schin History_t *hist_ptr; \ 2134887Schin void *jmpbuffer; \ 2144887Schin void *mktype; \ 2154887Schin Sfio_t *strbuf; \ 216*8462SApril.Chin@Sun.COM Sfio_t *strbuf2; \ 2174887Schin Dt_t *last_root; \ 218*8462SApril.Chin@Sun.COM Dt_t *prev_root; \ 219*8462SApril.Chin@Sun.COM Dt_t *fpathdict; \ 220*8462SApril.Chin@Sun.COM Dt_t *typedict; \ 2214887Schin char ifstable[256]; \ 222*8462SApril.Chin@Sun.COM unsigned char sigruntime[2]; \ 223*8462SApril.Chin@Sun.COM unsigned long test; \ 224*8462SApril.Chin@Sun.COM Shopt_t offoptions; \ 225*8462SApril.Chin@Sun.COM Shopt_t glob_options; \ 226*8462SApril.Chin@Sun.COM Namval_t *typeinit; \ 227*8462SApril.Chin@Sun.COM int *stats; \ 228*8462SApril.Chin@Sun.COM Namfun_t nvfun; 2294887Schin 2304887Schin #include <shell.h> 2314887Schin 2324887Schin 2334887Schin /* error exits from various parts of shell */ 2344887Schin #define NIL(type) ((type)0) 2354887Schin 2364887Schin #define new_of(type,x) ((type*)malloc((unsigned)sizeof(type)+(x))) 2374887Schin 2384887Schin #define exitset() (sh.savexit=sh.exitval) 2394887Schin 2404887Schin #ifndef SH_DICT 2414887Schin #define SH_DICT (void*)e_dict 2424887Schin #endif 2434887Schin 2444887Schin #ifndef SH_CMDLIB_DIR 2454887Schin #define SH_CMDLIB_DIR "/opt/ast/bin" 2464887Schin #endif 2474887Schin 2484887Schin /* states */ 2494887Schin /* low numbered states are same as options */ 2504887Schin #define SH_NOFORK 0 /* set when fork not necessary, not a state */ 2514887Schin #define SH_COMPLETE 0 /* set for command completion */ 2524887Schin #define SH_FORKED 7 /* set when process has been forked */ 2534887Schin #define SH_PROFILE 8 /* set when processing profiles */ 2544887Schin #define SH_NOALIAS 9 /* do not expand non-exported aliases */ 2554887Schin #define SH_NOTRACK 10 /* set to disable sftrack() function */ 2564887Schin #define SH_STOPOK 11 /* set for stopable builtins */ 2574887Schin #define SH_GRACE 12 /* set for timeout grace period */ 2584887Schin #define SH_TIMING 13 /* set while timing pipelines */ 2594887Schin #define SH_DEFPATH 14 /* set when using default path */ 2604887Schin #define SH_INIT 15 /* set when initializing the shell */ 2614887Schin #define SH_TTYWAIT 16 /* waiting for keyboard input */ 2624887Schin #define SH_FCOMPLETE 17 /* set for filename completion */ 2634887Schin #define SH_PREINIT 18 /* set with SH_INIT before parsing options */ 2644887Schin 2654887Schin #define SH_BASH 41 2664887Schin #define SH_BRACEEXPAND 42 2674887Schin #define SH_POSIX 46 2684887Schin #define SH_MULTILINE 47 2694887Schin 2704887Schin #define SH_NOPROFILE 78 2714887Schin #define SH_NOUSRPROFILE 79 2724887Schin #define SH_LOGIN_SHELL 67 2734887Schin #define SH_COMMANDLINE 0x100 2744887Schin #define SH_BASHEXTRA 0x200 2754887Schin #define SH_BASHOPT 0x400 2764887Schin 2774887Schin #define SH_ID "ksh" /* ksh id */ 2784887Schin #define SH_STD "sh" /* standard sh id */ 2794887Schin 2804887Schin /* defines for sh_type() */ 2814887Schin 2824887Schin #define SH_TYPE_SH 001 2834887Schin #define SH_TYPE_KSH 002 2844887Schin #define SH_TYPE_BASH 004 2854887Schin #define SH_TYPE_LOGIN 010 2864887Schin #define SH_TYPE_PROFILE 020 2874887Schin #define SH_TYPE_RESTRICTED 040 2884887Schin 2894887Schin #if SHOPT_BASH 2904887Schin # ifndef SHOPT_HISTEXPAND 2914887Schin # define SHOPT_HISTEXPAND 1 2924887Schin # endif 2934887Schin /* 2944887Schin * define for all the bash options 2954887Schin */ 2964887Schin # define SH_CDABLE_VARS 51 2974887Schin # define SH_CDSPELL 52 2984887Schin # define SH_CHECKHASH 53 2994887Schin # define SH_CHECKWINSIZE 54 3004887Schin # define SH_CMDHIST 55 3014887Schin # define SH_DOTGLOB 56 3024887Schin # define SH_EXECFAIL 57 3034887Schin # define SH_EXPAND_ALIASES 58 3044887Schin # define SH_EXTGLOB 59 3054887Schin # define SH_HOSTCOMPLETE 63 3064887Schin # define SH_HUPONEXIT 64 3074887Schin # define SH_INTERACTIVE_COMM 65 3084887Schin # define SH_LITHIST 66 3094887Schin # define SH_MAILWARN 68 3104887Schin # define SH_NOEMPTYCMDCOMPL 69 3114887Schin # define SH_NOCASEGLOB 70 3124887Schin # define SH_NULLGLOB 71 3134887Schin # define SH_PHYSICAL 45 3144887Schin # define SH_PROGCOMP 72 3154887Schin # define SH_PROMPTVARS 73 3164887Schin # define SH_RESTRICTED2 74 3174887Schin # define SH_SHIFT_VERBOSE 75 3184887Schin # define SH_SOURCEPATH 76 3194887Schin # define SH_XPG_ECHO 77 3204887Schin #endif 3214887Schin 3224887Schin #if SHOPT_HISTEXPAND 3234887Schin # define SH_HISTAPPEND 60 3244887Schin # define SH_HISTEXPAND 43 3254887Schin # define SH_HISTORY2 44 3264887Schin # define SH_HISTREEDIT 61 3274887Schin # define SH_HISTVERIFY 62 3284887Schin #endif 3294887Schin 3304887Schin #ifndef PIPE_BUF 3314887Schin # define PIPE_BUF 512 3324887Schin #endif 3334887Schin 3344887Schin #define MATCH_MAX 64 3354887Schin 336*8462SApril.Chin@Sun.COM #define SH_READEVAL 0x4000 /* for sh_eval */ 337*8462SApril.Chin@Sun.COM 338*8462SApril.Chin@Sun.COM extern Shell_t *nv_shell(Namval_t*); 3394887Schin extern int sh_addlib(void*); 340*8462SApril.Chin@Sun.COM extern void sh_applyopts(Shell_t*,Shopt_t); 341*8462SApril.Chin@Sun.COM extern char **sh_argbuild(Shell_t*,int*,const struct comnod*,int); 342*8462SApril.Chin@Sun.COM extern struct dolnod *sh_argfree(Shell_t *, struct dolnod*,int); 343*8462SApril.Chin@Sun.COM extern struct dolnod *sh_argnew(Shell_t*,char*[],struct dolnod**); 3444887Schin extern void *sh_argopen(Shell_t*); 345*8462SApril.Chin@Sun.COM extern void sh_argreset(Shell_t*,struct dolnod*,struct dolnod*); 3464887Schin extern Namval_t *sh_assignok(Namval_t*,int); 347*8462SApril.Chin@Sun.COM extern struct dolnod *sh_arguse(Shell_t*); 3484887Schin extern char *sh_checkid(char*,char*); 349*8462SApril.Chin@Sun.COM extern int sh_debug(Shell_t *shp,const char*,const char*,const char*,char *const[],int); 3504887Schin extern int sh_echolist(Sfio_t*, int, char**); 351*8462SApril.Chin@Sun.COM extern struct argnod *sh_endword(Shell_t*,int); 3524887Schin extern char **sh_envgen(void); 3534887Schin #if SHOPT_ENV 3544887Schin extern void sh_envput(Env_t*, Namval_t*); 3554887Schin #endif 3564887Schin extern void sh_envnolocal(Namval_t*,void*); 3574887Schin extern Sfdouble_t sh_arith(const char*); 3584887Schin extern void *sh_arithcomp(char*); 3594887Schin extern pid_t sh_fork(int,int*); 3604887Schin extern pid_t _sh_fork(pid_t, int ,int*); 361*8462SApril.Chin@Sun.COM extern char *sh_mactrim(Shell_t*,char*,int); 362*8462SApril.Chin@Sun.COM extern int sh_macexpand(Shell_t*,struct argnod*,struct argnod**,int); 363*8462SApril.Chin@Sun.COM extern int sh_macfun(Shell_t*,const char*,int); 364*8462SApril.Chin@Sun.COM extern void sh_machere(Shell_t*,Sfio_t*, Sfio_t*, char*); 3654887Schin extern void *sh_macopen(Shell_t*); 366*8462SApril.Chin@Sun.COM extern char *sh_macpat(Shell_t*,struct argnod*,int); 367*8462SApril.Chin@Sun.COM extern char *sh_mactry(Shell_t*,char*); 3684887Schin extern void sh_printopts(Shopt_t,int,Shopt_t*); 3694887Schin extern int sh_readline(Shell_t*,char**,int,int,long); 3704887Schin extern Sfio_t *sh_sfeval(char*[]); 3714887Schin extern void sh_setmatch(const char*,int,int,int[]); 3724887Schin extern Dt_t *sh_subaliastree(int); 373*8462SApril.Chin@Sun.COM extern void sh_scope(Shell_t*, struct argnod*, int); 374*8462SApril.Chin@Sun.COM extern Namval_t *sh_scoped(Shell_t*, Namval_t*); 3754887Schin extern Dt_t *sh_subfuntree(int); 376*8462SApril.Chin@Sun.COM extern void sh_subjobcheck(pid_t); 3774887Schin extern int sh_subsavefd(int); 378*8462SApril.Chin@Sun.COM extern void sh_subtmpfile(int); 3794887Schin extern char *sh_substitute(const char*,const char*,char*); 3804887Schin extern const char *_sh_translate(const char*); 3814887Schin extern int sh_trace(char*[],int); 3824887Schin extern void sh_trim(char*); 3834887Schin extern int sh_type(const char*); 384*8462SApril.Chin@Sun.COM extern void sh_unscope(Shell_t*); 3854887Schin extern void sh_utol(const char*, char*); 3864887Schin extern int sh_whence(char**,int); 3874887Schin 3884887Schin #ifndef ERROR_dictionary 3894887Schin # define ERROR_dictionary(s) (s) 3904887Schin #endif 3914887Schin #define sh_translate(s) _sh_translate(ERROR_dictionary(s)) 3924887Schin 3934887Schin #define WBITS (sizeof(long)*8) 3944887Schin #define WMASK (0xff) 3954887Schin 3964887Schin #define is_option(s,x) ((s)->v[((x)&WMASK)/WBITS] & (1L << ((x) % WBITS))) 3974887Schin #define on_option(s,x) ((s)->v[((x)&WMASK)/WBITS] |= (1L << ((x) % WBITS))) 3984887Schin #define off_option(s,x) ((s)->v[((x)&WMASK)/WBITS] &= ~(1L << ((x) % WBITS))) 3994887Schin #define sh_isoption(x) is_option(&sh.options,x) 4004887Schin #define sh_onoption(x) on_option(&sh.options,x) 4014887Schin #define sh_offoption(x) off_option(&sh.options,x) 4024887Schin 4034887Schin 4044887Schin #define sh_state(x) ( 1<<(x)) 4054887Schin #define sh_isstate(x) (sh.st.states&sh_state(x)) 4064887Schin #define sh_onstate(x) (sh.st.states |= sh_state(x)) 4074887Schin #define sh_offstate(x) (sh.st.states &= ~sh_state(x)) 4084887Schin #define sh_getstate() (sh.st.states) 4094887Schin #define sh_setstate(x) (sh.st.states = (x)) 4104887Schin 4114887Schin #define sh_sigcheck() do{if(sh.trapnote&SH_SIGSET)sh_exit(SH_EXITSIG);} while(0) 4124887Schin 4134887Schin extern int32_t sh_mailchk; 4144887Schin extern const char e_dict[]; 4154887Schin 4164887Schin /* sh_printopts() mode flags -- set --[no]option by default */ 4174887Schin 4184887Schin #define PRINT_VERBOSE 0x01 /* option on|off list */ 419*8462SApril.Chin@Sun.COM #define PRINT_ALL 0x02 /* list unset options too */ 4204887Schin #define PRINT_NO_HEADER 0x04 /* omit listing header */ 4214887Schin #define PRINT_SHOPT 0x08 /* shopt -s|-u */ 4224887Schin #define PRINT_TABLE 0x10 /* table of all options */ 423*8462SApril.Chin@Sun.COM 424*8462SApril.Chin@Sun.COM #ifdef SHOPT_STATS 425*8462SApril.Chin@Sun.COM /* performance statistics */ 426*8462SApril.Chin@Sun.COM # define STAT_ARGHITS 0 427*8462SApril.Chin@Sun.COM # define STAT_ARGEXPAND 1 428*8462SApril.Chin@Sun.COM # define STAT_COMSUB 2 429*8462SApril.Chin@Sun.COM # define STAT_FORKS 3 430*8462SApril.Chin@Sun.COM # define STAT_FUNCT 4 431*8462SApril.Chin@Sun.COM # define STAT_GLOBS 5 432*8462SApril.Chin@Sun.COM # define STAT_READS 6 433*8462SApril.Chin@Sun.COM # define STAT_NVHITS 7 434*8462SApril.Chin@Sun.COM # define STAT_NVOPEN 8 435*8462SApril.Chin@Sun.COM # define STAT_PATHS 9 436*8462SApril.Chin@Sun.COM # define STAT_SVFUNCT 10 437*8462SApril.Chin@Sun.COM # define STAT_SCMDS 11 438*8462SApril.Chin@Sun.COM # define STAT_SPAWN 12 439*8462SApril.Chin@Sun.COM # define STAT_SUBSHELL 13 440*8462SApril.Chin@Sun.COM extern const Shtable_t shtab_stats[]; 441*8462SApril.Chin@Sun.COM # define sh_stats(x) (sh.stats[(x)]++) 442*8462SApril.Chin@Sun.COM #else 443*8462SApril.Chin@Sun.COM # define sh_stats(x) 444*8462SApril.Chin@Sun.COM #endif /* SHOPT_STATS */ 445*8462SApril.Chin@Sun.COM 446*8462SApril.Chin@Sun.COM 447*8462SApril.Chin@Sun.COM #endif 448