aboutsummaryrefslogtreecommitdiff
path: root/lib/daemon.c
diff options
context:
space:
mode:
authorBen Pfaff <blp@nicira.com>2010-09-23 09:39:47 -0700
committerBen Pfaff <blp@nicira.com>2010-09-23 11:45:34 -0700
commite4bd5e2a6c4223fd9cfb2dd1ad82a4eeb1b1fbe6 (patch)
treea65109dc63a3d51db42f5a1cd8be4ef7275106c4 /lib/daemon.c
parentcbbdf81cf8bfcc87e141f66b93bf3bcf1220bff8 (diff)
daemon: Fix behavior of read_pidfile() for our own pidfile.
Opening a file descriptor and then closing it always discards any locks held on the underlying file, even if the file is still open as another file descriptor. This meant that calling read_pidfile() on the process's own pidfile would discard the lock and make other OVS processes think that the process had died. This commit fixes the problem.
Diffstat (limited to 'lib/daemon.c')
-rw-r--r--lib/daemon.c25
1 files changed, 25 insertions, 0 deletions
diff --git a/lib/daemon.c b/lib/daemon.c
index 6b61879d..bbcfe6af 100644
--- a/lib/daemon.c
+++ b/lib/daemon.c
@@ -42,6 +42,10 @@ static bool detach;
/* --pidfile: Name of pidfile (null if none). */
static char *pidfile;
+/* Device and inode of pidfile, so we can avoid reopening it. */
+static dev_t pidfile_dev;
+static ino_t pidfile_ino;
+
/* --overwrite-pidfile: Create pidfile even if one already exists and is
locked? */
static bool overwrite_pidfile;
@@ -208,6 +212,15 @@ make_pidfile(void)
close(fd);
} else {
/* Keep 'fd' open to retain the lock. */
+ struct stat s;
+
+ if (!fstat(fd, &s)) {
+ pidfile_dev = s.st_dev;
+ pidfile_ino = s.st_ino;
+ } else {
+ VLOG_ERR("%s: fstat failed: %s",
+ pidfile, strerror(errno));
+ }
}
free(text);
} else {
@@ -494,9 +507,21 @@ read_pidfile(const char *pidfile)
{
char line[128];
struct flock lck;
+ struct stat s;
FILE *file;
int error;
+ if ((pidfile_ino || pidfile_dev)
+ && !stat(pidfile, &s)
+ && s.st_ino == pidfile_ino && s.st_dev == pidfile_dev) {
+ /* It's our own pidfile. We can't afford to open it, because closing
+ * *any* fd for a file that a process has locked also releases all the
+ * locks on that file.
+ *
+ * Fortunately, we know the associated pid anyhow: */
+ return getpid();
+ }
+
file = fopen(pidfile, "r");
if (!file) {
error = errno;