/* pidfile handling for DBMail, many thanks to Tridge and Samba
 * for making their software available under the GNU GPL.
 *
 * Modified for DBMail by Aaron Stone, July 9, 2004.
 */

/* this code is broken - there is a race condition with the unlink (tridge) */

/* 
   Unix SMB/CIFS implementation.
   pidfile handling
   Copyright (C) Andrew Tridgell 1998
   
   This program is free software; you can redistribute it and/or modify
   it under the terms of the GNU General Public License as published by
   the Free Software Foundation; either version 2 of the License, or
   (at your option) any later version.
   
   This program is distributed in the hope that it will be useful,
   but WITHOUT ANY WARRANTY; without even the implied warranty of
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
   GNU General Public License for more details.
   
   You should have received a copy of the GNU General Public License
   along with this program; if not, write to the Free Software
   Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
*/

#include "dbmail.h"
#define THIS_MODULE "pidfile"

/* These are used by pidfile_remove. */
static FILE *pidfile_to_close;
static char *pidfile_to_remove;

/* Check if a process exists. */
static int process_exists(pid_t pid)
{
	if (pid > 0)
		return (kill(pid, 0) == 0 || errno != ESRCH);
	return 0;
}

/* Return the pid in a pidfile, or return 0 
 * if the process or pidfile does not exist. */
static pid_t pidfile_pid(const char *pidFile)
{
	FILE *f;
	char pidstr[20];
	unsigned ret;

	memset(pidstr, 0, sizeof(pidstr));

	if (!(f = fopen(pidFile, "r"))) {
		return 0;
	}

	if (fread(pidstr, sizeof(char), sizeof(pidstr)-1, f) <= 0) {
		goto noproc;
	}

	ret = atoi(pidstr);
	
	if (!process_exists((pid_t)ret)) {
		goto noproc;
	}

	fclose(f);
	return (pid_t)ret;

 noproc:
	fclose(f);
	unlink(pidFile);
	return 0;
}

static void pidfile_remove(void)
{
	int res;
	extern int isChildProcess;

	if (isChildProcess)
		return;

	if (pidfile_to_close) {
		res = fclose(pidfile_to_close);
		if (res) TRACE(TRACE_ERROR, "Error closing pidfile: [%s].",
			strerror(errno));
		pidfile_to_close = NULL;
	}

	if (pidfile_to_remove) {
		res = unlink(pidfile_to_remove);
		if (res) TRACE(TRACE_ERROR, "Error unlinking pidfile [%s]: [%s].",
			pidfile_to_remove, strerror(errno));
		g_free(pidfile_to_remove);
		pidfile_to_remove = NULL;
	}

}

/* Create a pidfile and leave it open. */
void pidfile_create(const char *pidFile, pid_t pid)
{
	FILE *f;
	pid_t oldpid;

	oldpid = pidfile_pid(pidFile);

	if (oldpid != 0) {
		TRACE(TRACE_FATAL, "File [%s] exists and process id [%d] is running.", 
			pidFile, (int)pid);
	}

	if (!(f = fopen(pidFile, "w"))) {
		TRACE(TRACE_FATAL, "Cannot open pidfile [%s], error was [%s]",
			pidFile, strerror(errno));
	}
	chmod(pidFile, 0644);

	fprintf(f, "%u\n", pid);
	fflush(f);

	/* Leave pid file open & locked for the duration,
	 * but close and remove it upon termination.  */
	atexit(pidfile_remove);

	pidfile_to_close = f;
	pidfile_to_remove = g_strdup(pidFile);

}



syntax highlighted by Code2HTML, v. 0.9.1