xref: /openbsd-src/lib/libutil/pidfile.c (revision 80e53f5bf656e1c4a7e523023dc34564a1eae491)
1*80e53f5bSschwarze /*	$OpenBSD: pidfile.c,v 1.14 2019/06/28 14:20:40 schwarze Exp $	*/
27ed310cbSjakob /*	$NetBSD: pidfile.c,v 1.4 2001/02/19 22:43:42 cgd Exp $	*/
37ed310cbSjakob 
47ed310cbSjakob /*-
57ed310cbSjakob  * Copyright (c) 1999 The NetBSD Foundation, Inc.
67ed310cbSjakob  * All rights reserved.
77ed310cbSjakob  *
87ed310cbSjakob  * This code is derived from software contributed to The NetBSD Foundation
97ed310cbSjakob  * by Jason R. Thorpe.
107ed310cbSjakob  *
117ed310cbSjakob  * Redistribution and use in source and binary forms, with or without
127ed310cbSjakob  * modification, are permitted provided that the following conditions
137ed310cbSjakob  * are met:
147ed310cbSjakob  * 1. Redistributions of source code must retain the above copyright
157ed310cbSjakob  *    notice, this list of conditions and the following disclaimer.
167ed310cbSjakob  * 2. Redistributions in binary form must reproduce the above copyright
177ed310cbSjakob  *    notice, this list of conditions and the following disclaimer in the
187ed310cbSjakob  *    documentation and/or other materials provided with the distribution.
197ed310cbSjakob  *
207ed310cbSjakob  * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
217ed310cbSjakob  * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
227ed310cbSjakob  * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
237ed310cbSjakob  * PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
247ed310cbSjakob  * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
257ed310cbSjakob  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
267ed310cbSjakob  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
277ed310cbSjakob  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
287ed310cbSjakob  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
297ed310cbSjakob  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
307ed310cbSjakob  * POSSIBILITY OF SUCH DAMAGE.
317ed310cbSjakob  */
327ed310cbSjakob 
33429b3653Sderaadt #include <sys/types.h>
347ed310cbSjakob #include <errno.h>
357ed310cbSjakob #include <paths.h>
367ed310cbSjakob #include <stdio.h>
377ed310cbSjakob #include <stdlib.h>
387ed310cbSjakob #include <unistd.h>
397ed310cbSjakob #include <util.h>
407ed310cbSjakob 
417ed310cbSjakob static char *pidfile_path;
42921948f2Smpech static pid_t pidfile_pid;
437ed310cbSjakob 
447ed310cbSjakob static void pidfile_cleanup(void);
457ed310cbSjakob 
467ed310cbSjakob extern char *__progname;
477ed310cbSjakob 
487ed310cbSjakob int
pidfile(const char * basename)497ed310cbSjakob pidfile(const char *basename)
507ed310cbSjakob {
517ed310cbSjakob 	int save_errno;
52cfa4d1b7Sderaadt 	pid_t pid;
531477552aSderaadt 	FILE *f;
547ed310cbSjakob 
557ed310cbSjakob 	if (basename == NULL)
567ed310cbSjakob 		basename = __progname;
577ed310cbSjakob 
587ed310cbSjakob 	free(pidfile_path);
597ed310cbSjakob 	pidfile_path = NULL;
607ed310cbSjakob 
617ed310cbSjakob 	/* _PATH_VARRUN includes trailing / */
62aa48e8d1Smillert 	if (asprintf(&pidfile_path, "%s%s.pid", _PATH_VARRUN, basename) == -1)
637ed310cbSjakob 		return (-1);
647ed310cbSjakob 
657ed310cbSjakob 	if ((f = fopen(pidfile_path, "w")) == NULL) {
667ed310cbSjakob 		save_errno = errno;
677ed310cbSjakob 		free(pidfile_path);
687ed310cbSjakob 		pidfile_path = NULL;
697ed310cbSjakob 		errno = save_errno;
707ed310cbSjakob 		return (-1);
717ed310cbSjakob 	}
727ed310cbSjakob 
73cfa4d1b7Sderaadt 	pid = getpid();
7452d6904dSmillert 	if (fprintf(f, "%ld\n", (long)pid) <= 0 || fflush(f) != 0) {
757ed310cbSjakob 		save_errno = errno;
7652d6904dSmillert 		(void) fclose(f);
777ed310cbSjakob 		(void) unlink(pidfile_path);
787ed310cbSjakob 		free(pidfile_path);
797ed310cbSjakob 		pidfile_path = NULL;
807ed310cbSjakob 		errno = save_errno;
817ed310cbSjakob 		return (-1);
827ed310cbSjakob 	}
8352d6904dSmillert 	(void) fclose(f);
847ed310cbSjakob 
85cfa4d1b7Sderaadt 	pidfile_pid = pid;
86*80e53f5bSschwarze 	if (atexit(pidfile_cleanup) != 0) {
871c511475Sitojun 		save_errno = errno;
881c511475Sitojun 		(void) unlink(pidfile_path);
891c511475Sitojun 		free(pidfile_path);
901c511475Sitojun 		pidfile_path = NULL;
911c511475Sitojun 		pidfile_pid = 0;
921c511475Sitojun 		errno = save_errno;
931c511475Sitojun 		return (-1);
941c511475Sitojun 	}
957ed310cbSjakob 
967ed310cbSjakob 	return (0);
977ed310cbSjakob }
987ed310cbSjakob 
997ed310cbSjakob static void
pidfile_cleanup(void)1007ed310cbSjakob pidfile_cleanup(void)
1017ed310cbSjakob {
1027ed310cbSjakob 
103cfa4d1b7Sderaadt 	if (pidfile_path != NULL && pidfile_pid == getpid())
1047ed310cbSjakob 		(void) unlink(pidfile_path);
1057ed310cbSjakob }
106