- 论坛徽章:
- 0
|
本帖最后由 可可火山 于 2011-11-11 15:41 编辑
Read the fucking source code. 
nohup是忽略 SIGHUP信号,这个信号在终端断开等情况下触发,比如当你ssh断开远程连接,就会触发这个信号,默认情况下程序对这个信号的处理就是退出。
不过我想知道 solaris zone环境下的nohup是否有区别,曾有个问题研究了下nohup源代码,不过没找到我生成环境nohup的源代码。- /*
- * Copyright (c) 1989, 1993
- * The Regents of the University of California. All rights reserved.
- * Portions copyright (c) 2007 Apple Inc. All rights reserved.
- *
- * Redistribution and use in source and binary forms, with or without
- * modification, are permitted provided that the following conditions
- * are met:
- * 1. Redistributions of source code must retain the above copyright
- * notice, this list of conditions and the following disclaimer.
- * 2. Redistributions in binary form must reproduce the above copyright
- * notice, this list of conditions and the following disclaimer in the
- * documentation and/or other materials provided with the distribution.
- * 3. Neither the name of the University nor the names of its contributors
- * may be used to endorse or promote products derived from this software
- * without specific prior written permission.
- *
- * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
- * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
- * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
- * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
- * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
- * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
- * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
- * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
- * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
- * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
- * SUCH DAMAGE.
- */
- #if 0
- #ifndef lint
- static const char copyright[] =
- "@(#) Copyright (c) 1989, 1993\n\
- The Regents of the University of California. All rights reserved.\n";
- #endif /* not lint */
- #ifndef lint
- static char sccsid[] = "@(#)nohup.c 8.1 (Berkeley) 6/6/93";
- #endif /* not lint */
- #endif
- #include <sys/cdefs.h>
- #ifndef __APPLE__
- __FBSDID("$FreeBSD: src/usr.bin/nohup/nohup.c,v 1.10 2003/05/03 19:44:46 obrien Exp $");
- #endif
- #include <sys/param.h>
- #include <sys/stat.h>
- #include <err.h>
- #include <errno.h>
- #include <fcntl.h>
- #include <signal.h>
- #include <stdio.h>
- #include <stdlib.h>
- #include <string.h>
- #include <unistd.h>
- #ifdef __APPLE__
- #include <vproc.h>
- #include <vproc_priv.h>
- #endif
- static void dofile(void);
- static void usage(void);
- #define FILENAME "nohup.out"
- /*
- * POSIX mandates that we exit with:
- * 126 - If the utility was found, but failed to execute.
- * 127 - If any other error occurred.
- */
- #define EXIT_NOEXEC 126
- #define EXIT_NOTFOUND 127
- #define EXIT_MISC 127
- int
- main(int argc, char *argv[])
- {
- int exit_status;
- while (getopt(argc, argv, "") != -1)
- usage();
- argc -= optind;
- argv += optind;
- if (argc < 1)
- usage();
- if (isatty(STDOUT_FILENO))
- dofile();
- if (isatty(STDERR_FILENO) && dup2(STDOUT_FILENO, STDERR_FILENO) == -1)
- /* may have just closed stderr */
- err(EXIT_MISC, "%s", argv[0]);
- (void)signal(SIGHUP, SIG_IGN);
- #ifdef __APPLE__
- if (_vprocmgr_move_subset_to_user(geteuid(), "Background") != NULL)
- err(EXIT_MISC, "can't migrate to background session");
- #endif
- execvp(*argv, argv);
- exit_status = (errno == ENOENT) ? EXIT_NOTFOUND : EXIT_NOEXEC;
- err(exit_status, "%s", argv[0]);
- }
- static void
- dofile(void)
- {
- int fd;
- char path[MAXPATHLEN];
- const char *p;
- /*
- * POSIX mandates if the standard output is a terminal, the standard
- * output is appended to nohup.out in the working directory. Failing
- * that, it will be appended to nohup.out in the directory obtained
- * from the HOME environment variable. If file creation is required,
- * the mode_t is set to S_IRUSR | S_IWUSR.
- */
- p = FILENAME;
- fd = open(p, O_RDWR | O_CREAT | O_APPEND, S_IRUSR | S_IWUSR);
- if (fd != -1)
- goto dupit;
- if ((p = getenv("HOME")) != NULL && *p != '\0' &&
- (size_t)snprintf(path, sizeof(path), "%s/%s", p, FILENAME) <
- sizeof(path)) {
- fd = open(p = path, O_RDWR | O_CREAT | O_APPEND,
- S_IRUSR | S_IWUSR);
- if (fd != -1)
- goto dupit;
- }
- errx(EXIT_MISC, "can't open a nohup.out file");
- dupit:
- #ifdef __APPLE__
- (void)lseek(fd, 0L, SEEK_END);
- #endif
- if (dup2(fd, STDOUT_FILENO) == -1)
- err(EXIT_MISC, NULL);
- (void)fprintf(stderr, "appending output to %s\n", p);
- }
- static void
- usage(void)
- {
- (void)fprintf(stderr, "usage: nohup [--] utility [arguments]\n");
- exit(EXIT_MISC);
- }
复制代码FreeBSD源码学习——nohup命令源程序分析
(2005-10-31 16:04:0
nohup命令是最长用到的几个命令之一,看了一下他的源码,真没想到内部执行过程原来是如此的简单 ho...ho......
step 1:
if (argc < 1)
usage();
判断参数个数如果小于1,则显示帮助并退出。
step 2:
if (isatty(STDOUT_FILENO))
dofile();
判断标准输出是否为终端机,如果是则执行dofile自定义函数。对于使用nohup命令后跟被执行的命令后没有提供将信息输出指向到其他设备或文件时,isatty(STDOUT_FILENO)将返回1,并执行dofile()。
step 3:
static void dofile(void)
这个自定义函数完成的功能是使用getenv("HOME" 获取路径并以 #define FILENAME "nohup.out" 中定义的FILENAME为文件名,创建程序执行输出信息存放的文件句柄,并使用dup2(fd, STDOUT_FILENO)将标准输出句柄复制到此文件句柄上,这样被执行程序本应该默认将信息输出到标准输出设备上的内容就转到默认的nohup.out文件里了。
step 4:
if (isatty(STDERR_FILENO) && dup2(STDOUT_FILENO, STDERR_FILENO) == -1)
err(EXIT_MISC, "%s", argv[0]);
判断标准错误输出是否为终端机,如果是,并且将标准错误输出句柄复制到标准输出句柄上,为了将错误信息直接输出到标准输出终端,这两步有一步操作失败则打印错误信息。
step 5:
(void)signal(SIGHUP, SIG_IGN);
屏蔽连接中断信号,以防止由于客户端连接中断造成nohup执行的程序无法完成。因为此语句的存在才让nohup命令具有实际意义。
step 6:
execvp(*argv, argv);
会从PATH环境变量所指的目录中查找符合nohup参数所指定的执行程序,找到后便执行该文件,然后将第二个参数argv传给该欲执行的文件。执行失败则直接返回-1,失败原因存于errno中。
step 7:
exit_status = (errno == ENOENT) ? EXIT_NOTFOUND : EXIT_NOEXEC;
err(exit_status, "%s", argv[0]);
execvp执行后如果失败,只有两种可能,一种是被执行的程序不存在,另一种是不能执行,因此程序只判断了这两种状态,并当在发生的时候显示错误信息。 |
|