1 /* init.c - init program. 2 * 3 * Copyright 2012 Harvind Singh <harvindsingh1981@gmail.com> 4 * Copyright 2013 Kyungwan Han <asura321@gmail.com> 5 * 6 * No Standard 7 8 USE_INIT(NEWTOY(init, "", TOYFLAG_SBIN)) 9 10 config INIT 11 bool "init" 12 default n 13 help 14 usage: init 15 16 System V style init. 17 18 First program to run (as PID 1) when the system comes up, reading 19 /etc/inittab to determine actions. 20 */ 21 22 #include "toys.h" 23 #include <sys/reboot.h> 24 25 struct action_list_seed { 26 struct action_list_seed *next; 27 pid_t pid; 28 uint8_t action; 29 char *terminal_name; 30 char *command; 31 } *action_list_pointer = NULL; 32 int caught_signal; 33 34 //INITTAB action defination 35 #define SYSINIT 0x01 36 #define WAIT 0x02 37 #define ONCE 0x04 38 #define RESPAWN 0x08 39 #define ASKFIRST 0x10 40 #define CTRLALTDEL 0x20 41 #define SHUTDOWN 0x40 42 #define RESTART 0x80 43 44 static void initialize_console(void) 45 { 46 int fd; 47 char *p = getenv("CONSOLE"); 48 49 if (!p) p = getenv("console"); 50 if (!p) { 51 fd = open("/dev/null", O_RDWR); 52 if (fd >= 0) { 53 while (fd < 2) fd = dup(fd); 54 while (fd > 2) close(fd--); 55 } 56 } else { 57 fd = open(p, O_RDWR | O_NONBLOCK | O_NOCTTY); 58 if (fd < 0) printf("Unable to open console %s\n",p); 59 else { 60 dup2(fd,0); 61 dup2(fd,1); 62 dup2(fd,2); 63 } 64 } 65 66 if (!getenv("TERM")) putenv("TERM=linux"); 67 } 68 69 static void reset_term(int fd) 70 { 71 struct termios terminal; 72 73 tcgetattr(fd, &terminal); 74 terminal.c_cc[VINTR] = 3; //ctrl-c 75 terminal.c_cc[VQUIT] = 28; /*ctrl-\*/ 76 terminal.c_cc[VERASE] = 127; //ctrl-? 77 terminal.c_cc[VKILL] = 21; //ctrl-u 78 terminal.c_cc[VEOF] = 4; //ctrl-d 79 terminal.c_cc[VSTART] = 17; //ctrl-q 80 terminal.c_cc[VSTOP] = 19; //ctrl-s 81 terminal.c_cc[VSUSP] = 26; //ctrl-z 82 83 terminal.c_line = 0; 84 terminal.c_cflag &= CRTSCTS|PARODD|PARENB|CSTOPB|CSIZE|CBAUDEX|CBAUD; 85 terminal.c_cflag |= CLOCAL|HUPCL|CREAD; 86 87 //enable start/stop input and output control + map CR to NL on input 88 terminal.c_iflag = IXON|IXOFF|ICRNL; 89 90 //Map NL to CR-NL on output 91 terminal.c_oflag = ONLCR|OPOST; 92 terminal.c_lflag = IEXTEN|ECHOKE|ECHOCTL|ECHOK|ECHOE|ECHO|ICANON|ISIG; 93 tcsetattr(fd, TCSANOW, &terminal); 94 } 95 96 static void add_new_action(uint8_t action,char *command,char *term) 97 { 98 struct action_list_seed *x,**y; 99 100 y = &action_list_pointer; 101 x = *y; 102 while (x) { 103 if (!(strcmp(x->command, command)) && !(strcmp(x->terminal_name, term))) { 104 *y = x->next; //remove from the list 105 while(*y) y = &(*y)->next; //traverse through list till end 106 x->next = NULL; 107 break; 108 } 109 y = &(x)->next; 110 x = *y; 111 } 112 113 //create a new node 114 if (!x) { 115 x = xzalloc(sizeof(*x)); 116 x->command = xstrdup(command); 117 x->terminal_name = xstrdup(term); 118 } 119 x->action = action; 120 *y = x; 121 } 122 123 static void inittab_parsing(void) 124 { 125 int i, fd, line_number = 0, token_count = 0; 126 char *p, *q, *extracted_token, *tty_name = NULL, *command = NULL, *tmp; 127 uint8_t action = 0; 128 char *act_name = "sysinit\0wait\0once\0respawn\0askfirst\0ctrlaltdel\0" 129 "shutdown\0restart\0"; 130 131 fd = open("/etc/inittab", O_RDONLY); 132 if (fd < 0) { 133 error_msg("Unable to open /etc/inittab. Using Default inittab"); 134 add_new_action(SYSINIT, "/etc/init.d/rcS", ""); 135 add_new_action(RESPAWN, "/sbin/getty -n -l /bin/sh -L 115200 tty1 vt100", ""); 136 } else { 137 while((q = p = get_line(fd))) { //read single line from /etc/inittab 138 char *x; 139 140 if ((x = strchr(p, '#'))) *x = '\0'; 141 line_number++; 142 token_count = 0; 143 action = 0; 144 tty_name = command = NULL; 145 146 while ((extracted_token = strsep(&p,":"))) { 147 token_count++; 148 switch (token_count) { 149 case 1: 150 if (*extracted_token) { 151 if (!strncmp(extracted_token, "/dev/", 5)) 152 tty_name = xmprintf("%s",extracted_token); 153 else tty_name = xmprintf("/dev/%s",extracted_token); 154 } else tty_name = xstrdup(""); 155 break; 156 case 2: 157 break; 158 case 3: 159 for (tmp = act_name, i = 0; *tmp; i++, tmp += strlen(tmp) +1) { 160 if (!strcmp(tmp, extracted_token)) { 161 action = 1 << i; 162 break; 163 } 164 } 165 if (!*tmp) error_msg("Invalid action at line number %d ---- ignoring",line_number); 166 break; 167 case 4: 168 command = xstrdup(extracted_token); 169 break; 170 default: 171 error_msg("Bad inittab entry at line %d", line_number); 172 break; 173 } 174 } //while token 175 176 if (q) free(q); 177 if (token_count != 4) { 178 free(tty_name); 179 free(command); 180 continue; 181 } 182 if (action) add_new_action(action, command, tty_name); 183 free(tty_name); 184 free(command); 185 } //while line 186 187 close(fd); 188 } 189 } 190 191 static void run_command(char *command) 192 { 193 char *final_command[128]; 194 int hyphen = (command[0]=='-'); 195 196 command = command + hyphen; 197 if (!strpbrk(command, "?<>'\";[]{}\\|=()*&^$!`~")) { 198 char *next_command; 199 char *extracted_command; 200 int x = 0; 201 202 next_command = strncpy(toybuf, command - hyphen, sizeof(toybuf)); 203 next_command[sizeof(toybuf) - 1] = toybuf[sizeof(toybuf) - 1 ] = '\0'; 204 command = next_command + hyphen; 205 while ((extracted_command = strsep(&next_command," \t"))) { 206 if (*extracted_command) { 207 final_command[x] = extracted_command; 208 x++; 209 } 210 } 211 final_command[x] = NULL; 212 } else { 213 snprintf(toybuf, sizeof(toybuf), "exec %s", command); 214 command = "-/bin/sh"+1; 215 final_command[0] = ("-/bin/sh"+!hyphen); 216 final_command[1] = "-c"; 217 final_command[2] = toybuf; 218 final_command[3] = NULL; 219 } 220 if (hyphen) ioctl(0, TIOCSCTTY, 0); 221 execvp(command, final_command); 222 error_msg("unable to run %s",command); 223 } 224 225 //runs all same type of actions 226 static pid_t final_run(struct action_list_seed *x) 227 { 228 pid_t pid; 229 int fd; 230 sigset_t signal_set; 231 232 sigfillset(&signal_set); 233 sigprocmask(SIG_BLOCK, &signal_set, NULL); 234 if (x->action & ASKFIRST) pid = fork(); 235 else pid = vfork(); 236 237 if (pid > 0) { 238 //parent process or error 239 //unblock the signals 240 sigfillset(&signal_set); 241 sigprocmask(SIG_UNBLOCK, &signal_set, NULL); 242 243 return pid; 244 } else if (pid < 0) { 245 perror_msg("fork fail"); 246 sleep(1); 247 return 0; 248 } 249 250 //new born child process 251 sigset_t signal_set_c; 252 sigfillset(&signal_set_c); 253 sigprocmask(SIG_UNBLOCK, &signal_set_c, NULL); 254 setsid(); //new session 255 256 if (x->terminal_name[0]) { 257 close(0); 258 fd = open(x->terminal_name, (O_RDWR|O_NONBLOCK),0600); 259 if (fd != 0) { 260 error_msg("Unable to open %s,%s\n", x->terminal_name, strerror(errno)); 261 _exit(EXIT_FAILURE); 262 } else { 263 dup2(0, 1); 264 dup2(0, 2); 265 } 266 } 267 reset_term(0); 268 run_command(x->command); 269 _exit(-1); 270 } 271 272 static struct action_list_seed* mark_as_terminated_process(pid_t pid) 273 { 274 struct action_list_seed *x; 275 276 if (pid > 0) { 277 for (x = action_list_pointer; x; x = x->next) { 278 if (x->pid == pid) { 279 x->pid = 0; 280 return x; 281 } 282 } 283 } 284 285 return NULL; 286 } 287 288 static void waitforpid(pid_t pid) 289 { 290 if (pid <= 0) return; 291 292 for(;;) { 293 pid_t y = wait(NULL); 294 mark_as_terminated_process(y); 295 if (kill(y, 0)) break; 296 } 297 } 298 299 static void run_action_from_list(int action) 300 { 301 pid_t pid; 302 struct action_list_seed *x = action_list_pointer; 303 304 for (; x; x = x->next) { 305 if (!(x->action & action)) continue; 306 if (x->action & (SHUTDOWN|ONCE|SYSINIT|CTRLALTDEL|WAIT)) { 307 pid = final_run(x); 308 if (!pid) return; 309 if (x->action & (SHUTDOWN|SYSINIT|CTRLALTDEL|WAIT)) waitforpid(pid); 310 } 311 if (x->action & (ASKFIRST|RESPAWN)) 312 if (!(x->pid)) x->pid = final_run(x); 313 } 314 } 315 316 static void set_default(void) 317 { 318 sigset_t signal_set_c; 319 320 sigatexit(SIG_DFL); 321 sigfillset(&signal_set_c); 322 sigprocmask(SIG_UNBLOCK,&signal_set_c, NULL); 323 324 run_action_from_list(SHUTDOWN); 325 error_msg("The system is going down NOW!"); 326 kill(-1, SIGTERM); 327 error_msg("Sent SIGTERM to all processes"); 328 sync(); 329 sleep(1); 330 kill(-1,SIGKILL); 331 sync(); 332 } 333 334 static void halt_poweroff_reboot_handler(int sig_no) 335 { 336 unsigned int reboot_magic_no = 0; 337 pid_t pid; 338 339 set_default(); 340 341 switch (sig_no) { 342 case SIGUSR1: 343 error_msg("Requesting system halt"); 344 reboot_magic_no=RB_HALT_SYSTEM; 345 break; 346 case SIGUSR2: 347 error_msg("Requesting system poweroff"); 348 reboot_magic_no=RB_POWER_OFF; 349 break; 350 case SIGTERM: 351 error_msg("Requesting system reboot"); 352 reboot_magic_no=RB_AUTOBOOT; 353 break; 354 default: 355 break; 356 } 357 358 sleep(1); 359 pid = vfork(); 360 361 if (pid == 0) { 362 reboot(reboot_magic_no); 363 _exit(EXIT_SUCCESS); 364 } 365 366 while(1) sleep(1); 367 } 368 369 static void restart_init_handler(int sig_no) 370 { 371 struct action_list_seed *x; 372 pid_t pid; 373 int fd; 374 375 for (x = action_list_pointer; x; x = x->next) { 376 if (!(x->action & RESTART)) continue; 377 378 set_default(); 379 380 if (x->terminal_name[0]) { 381 close(0); 382 fd = open(x->terminal_name, (O_RDWR|O_NONBLOCK),0600); 383 384 if (fd != 0) { 385 error_msg("Unable to open %s,%s\n", x->terminal_name, strerror(errno)); 386 sleep(1); 387 pid = vfork(); 388 389 if (pid == 0) { 390 reboot(RB_HALT_SYSTEM); 391 _exit(EXIT_SUCCESS); 392 } 393 394 while(1) sleep(1); 395 } else { 396 dup2(0, 1); 397 dup2(0, 2); 398 reset_term(0); 399 run_command(x->command); 400 } 401 } 402 } 403 } 404 405 static void catch_signal(int sig_no) 406 { 407 caught_signal = sig_no; 408 error_msg("signal seen"); 409 } 410 411 static void pause_handler(int sig_no) 412 { 413 int signal_backup,errno_backup; 414 pid_t pid; 415 416 errno_backup = errno; 417 signal_backup = caught_signal; 418 xsignal(SIGCONT, catch_signal); 419 420 while(1) { 421 if (caught_signal == SIGCONT) break; 422 do pid = waitpid(-1,NULL,WNOHANG); while((pid==-1) && (errno=EINTR)); 423 mark_as_terminated_process(pid); 424 sleep(1); 425 } 426 427 signal(SIGCONT, SIG_DFL); 428 errno = errno_backup; 429 caught_signal = signal_backup; 430 } 431 432 static int check_if_pending_signals(void) 433 { 434 int signal_caught = 0; 435 436 while(1) { 437 int sig = caught_signal; 438 if (!sig) return signal_caught; 439 caught_signal = 0; 440 signal_caught = 1; 441 if (sig == SIGINT) run_action_from_list(CTRLALTDEL); 442 } 443 } 444 445 void init_main(void) 446 { 447 struct sigaction sig_act; 448 449 if (getpid() != 1) error_exit("Already running"); 450 printf("Started init\n"); 451 initialize_console(); 452 reset_term(0); 453 454 if (chdir("/")) perror_exit("Can't cd to /"); 455 setsid(); 456 457 putenv("HOME=/"); 458 putenv("PATH=/sbin:/usr/sbin:/bin:/usr/bin"); 459 putenv("SHELL=/bin/sh"); 460 putenv("USER=root"); 461 462 inittab_parsing(); 463 xsignal(SIGUSR1, halt_poweroff_reboot_handler);//halt 464 xsignal(SIGUSR2, halt_poweroff_reboot_handler);//poweroff 465 xsignal(SIGTERM, halt_poweroff_reboot_handler);//reboot 466 xsignal(SIGQUIT, restart_init_handler);//restart init 467 memset(&sig_act, 0, sizeof(sig_act)); 468 sigfillset(&sig_act.sa_mask); 469 sigdelset(&sig_act.sa_mask, SIGCONT); 470 sig_act.sa_handler = pause_handler; 471 sigaction(SIGTSTP, &sig_act, NULL); 472 memset(&sig_act, 0, sizeof(sig_act)); 473 sig_act.sa_handler = catch_signal; 474 sigaction(SIGINT, &sig_act, NULL); 475 sigaction(SIGHUP, &sig_act, NULL); 476 run_action_from_list(SYSINIT); 477 check_if_pending_signals(); 478 run_action_from_list(WAIT); 479 check_if_pending_signals(); 480 run_action_from_list(ONCE); 481 while (1) { 482 int suspected_WNOHANG = check_if_pending_signals(); 483 484 run_action_from_list(RESPAWN | ASKFIRST); 485 suspected_WNOHANG = suspected_WNOHANG|check_if_pending_signals(); 486 sleep(1);//let cpu breath 487 suspected_WNOHANG = suspected_WNOHANG|check_if_pending_signals(); 488 if (suspected_WNOHANG) suspected_WNOHANG=WNOHANG; 489 490 while(1) { 491 pid_t pid = waitpid(-1, NULL, suspected_WNOHANG); 492 493 if (pid <= 0) break; 494 mark_as_terminated_process(pid); 495 suspected_WNOHANG = WNOHANG; 496 } 497 } 498 } 499