paul | 718e374 | 2002-12-13 20:15:29 +0000 | [diff] [blame] | 1 | /* |
| 2 | * Process id output. |
| 3 | * Copyright (C) 1998, 1999 Kunihiro Ishiguro |
| 4 | * |
| 5 | * This file is part of GNU Zebra. |
| 6 | * |
| 7 | * GNU Zebra is free software; you can redistribute it and/or modify it |
| 8 | * under the terms of the GNU General Public License as published by the |
| 9 | * Free Software Foundation; either version 2, or (at your option) any |
| 10 | * later version. |
| 11 | * |
| 12 | * GNU Zebra is distributed in the hope that it will be useful, but |
| 13 | * WITHOUT ANY WARRANTY; without even the implied warranty of |
| 14 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
| 15 | * General Public License for more details. |
| 16 | * |
| 17 | * You should have received a copy of the GNU General Public License |
| 18 | * along with GNU Zebra; see the file COPYING. If not, write to the Free |
| 19 | * Software Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA |
| 20 | * 02111-1307, USA. |
| 21 | */ |
| 22 | |
| 23 | #include <zebra.h> |
paul | e92fbaf | 2003-10-24 04:10:16 +0000 | [diff] [blame] | 24 | #include <fcntl.h> |
| 25 | #include <log.h> |
ajs | e5879ca | 2004-11-25 16:07:53 +0000 | [diff] [blame] | 26 | #include "version.h" |
paul | 718e374 | 2002-12-13 20:15:29 +0000 | [diff] [blame] | 27 | |
ajs | 202d08c | 2004-12-17 20:50:00 +0000 | [diff] [blame] | 28 | #ifndef HAVE_FCNTL |
| 29 | |
paul | 718e374 | 2002-12-13 20:15:29 +0000 | [diff] [blame] | 30 | pid_t |
hasso | 6ad96ea | 2004-10-07 19:33:46 +0000 | [diff] [blame] | 31 | pid_output (const char *path) |
paul | 718e374 | 2002-12-13 20:15:29 +0000 | [diff] [blame] | 32 | { |
| 33 | FILE *fp; |
| 34 | pid_t pid; |
ajs | 202d08c | 2004-12-17 20:50:00 +0000 | [diff] [blame] | 35 | mode_t oldumask; |
paul | 718e374 | 2002-12-13 20:15:29 +0000 | [diff] [blame] | 36 | |
| 37 | pid = getpid(); |
| 38 | |
gdt | aa593d5 | 2003-12-22 20:15:53 +0000 | [diff] [blame] | 39 | oldumask = umask(0777 & ~LOGFILE_MASK); |
paul | 718e374 | 2002-12-13 20:15:29 +0000 | [diff] [blame] | 40 | fp = fopen (path, "w"); |
| 41 | if (fp != NULL) |
| 42 | { |
| 43 | fprintf (fp, "%d\n", (int) pid); |
| 44 | fclose (fp); |
gdt | aa593d5 | 2003-12-22 20:15:53 +0000 | [diff] [blame] | 45 | umask(oldumask); |
ajs | 202d08c | 2004-12-17 20:50:00 +0000 | [diff] [blame] | 46 | return pid; |
paul | 718e374 | 2002-12-13 20:15:29 +0000 | [diff] [blame] | 47 | } |
ajs | 202d08c | 2004-12-17 20:50:00 +0000 | [diff] [blame] | 48 | /* XXX Why do we continue instead of exiting? This seems incompatible |
| 49 | with the behavior of the fcntl version below. */ |
| 50 | zlog_warn("Can't fopen pid lock file %s (%s), continuing", |
| 51 | path, safe_strerror(errno)); |
gdt | aa593d5 | 2003-12-22 20:15:53 +0000 | [diff] [blame] | 52 | umask(oldumask); |
ajs | 202d08c | 2004-12-17 20:50:00 +0000 | [diff] [blame] | 53 | return -1; |
paul | 718e374 | 2002-12-13 20:15:29 +0000 | [diff] [blame] | 54 | } |
| 55 | |
ajs | 202d08c | 2004-12-17 20:50:00 +0000 | [diff] [blame] | 56 | #else /* HAVE_FCNTL */ |
| 57 | |
| 58 | pid_t |
| 59 | pid_output (const char *path) |
paul | 718e374 | 2002-12-13 20:15:29 +0000 | [diff] [blame] | 60 | { |
| 61 | int tmp; |
| 62 | int fd; |
| 63 | pid_t pid; |
paul | e92fbaf | 2003-10-24 04:10:16 +0000 | [diff] [blame] | 64 | char buf[16]; |
paul | e4eaf1d | 2003-10-30 21:58:06 +0000 | [diff] [blame] | 65 | struct flock lock; |
gdt | aa593d5 | 2003-12-22 20:15:53 +0000 | [diff] [blame] | 66 | mode_t oldumask; |
paul | 718e374 | 2002-12-13 20:15:29 +0000 | [diff] [blame] | 67 | |
| 68 | pid = getpid (); |
| 69 | |
gdt | aa593d5 | 2003-12-22 20:15:53 +0000 | [diff] [blame] | 70 | oldumask = umask(0777 & ~LOGFILE_MASK); |
gdt | aa593d5 | 2003-12-22 20:15:53 +0000 | [diff] [blame] | 71 | fd = open (path, O_RDWR | O_CREAT, LOGFILE_MASK); |
ajs | 202d08c | 2004-12-17 20:50:00 +0000 | [diff] [blame] | 72 | if (fd < 0) |
| 73 | { |
| 74 | zlog_err("Can't create pid lock file %s (%s), exiting", |
| 75 | path, safe_strerror(errno)); |
gdt | aa593d5 | 2003-12-22 20:15:53 +0000 | [diff] [blame] | 76 | umask(oldumask); |
ajs | 202d08c | 2004-12-17 20:50:00 +0000 | [diff] [blame] | 77 | exit(1); |
paul | 718e374 | 2002-12-13 20:15:29 +0000 | [diff] [blame] | 78 | } |
| 79 | else |
| 80 | { |
ajs | e5879ca | 2004-11-25 16:07:53 +0000 | [diff] [blame] | 81 | size_t pidsize; |
| 82 | |
gdt | aa593d5 | 2003-12-22 20:15:53 +0000 | [diff] [blame] | 83 | umask(oldumask); |
paul | e92fbaf | 2003-10-24 04:10:16 +0000 | [diff] [blame] | 84 | memset (&lock, 0, sizeof(lock)); |
| 85 | |
paul | e4eaf1d | 2003-10-30 21:58:06 +0000 | [diff] [blame] | 86 | lock.l_type = F_WRLCK; |
ajs | e5879ca | 2004-11-25 16:07:53 +0000 | [diff] [blame] | 87 | lock.l_whence = SEEK_SET; |
paul | e4eaf1d | 2003-10-30 21:58:06 +0000 | [diff] [blame] | 88 | |
paul | e92fbaf | 2003-10-24 04:10:16 +0000 | [diff] [blame] | 89 | if (fcntl(fd, F_SETLK, &lock) < 0) |
| 90 | { |
ajs | 202d08c | 2004-12-17 20:50:00 +0000 | [diff] [blame] | 91 | zlog_err("Could not lock pid_file %s, exiting", path); |
| 92 | exit(1); |
paul | e92fbaf | 2003-10-24 04:10:16 +0000 | [diff] [blame] | 93 | } |
| 94 | |
paul | 718e374 | 2002-12-13 20:15:29 +0000 | [diff] [blame] | 95 | sprintf (buf, "%d\n", (int) pid); |
ajs | e5879ca | 2004-11-25 16:07:53 +0000 | [diff] [blame] | 96 | pidsize = strlen(buf); |
| 97 | if ((tmp = write (fd, buf, pidsize)) != (int)pidsize) |
| 98 | zlog_err("Could not write pid %d to pid_file %s, rc was %d: %s", |
| 99 | (int)pid,path,tmp,safe_strerror(errno)); |
| 100 | else if (ftruncate(fd, pidsize) < 0) |
| 101 | zlog_err("Could not truncate pid_file %s to %u bytes: %s", |
| 102 | path,(u_int)pidsize,safe_strerror(errno)); |
paul | 718e374 | 2002-12-13 20:15:29 +0000 | [diff] [blame] | 103 | } |
paul | 718e374 | 2002-12-13 20:15:29 +0000 | [diff] [blame] | 104 | return pid; |
| 105 | } |
ajs | 202d08c | 2004-12-17 20:50:00 +0000 | [diff] [blame] | 106 | |
paul | e92fbaf | 2003-10-24 04:10:16 +0000 | [diff] [blame] | 107 | #endif /* HAVE_FCNTL */ |