exec.c 15.1 KB
Newer Older
1 2
/*-------------------------------------------------------------------------
 *
3
 * exec.c
4 5
 *		Functions for finding and validating executable files
 *
6
 *
7
 * Portions Copyright (c) 1996-2007, PostgreSQL Global Development Group
B
Add:  
Bruce Momjian 已提交
8
 * Portions Copyright (c) 1994, Regents of the University of California
9 10 11
 *
 *
 * IDENTIFICATION
12
 *	  $PostgreSQL: pgsql/src/port/exec.c,v 1.45 2007/01/22 18:31:51 momjian Exp $
13 14 15
 *
 *-------------------------------------------------------------------------
 */
16 17

#ifndef FRONTEND
18
#include "postgres.h"
19 20 21
#else
#include "postgres_fe.h"
#endif
22

23 24
#include <grp.h>
#include <pwd.h>
B
Bruce Momjian 已提交
25
#include <sys/stat.h>
26
#include <sys/wait.h>
27
#include <unistd.h>
28

29 30 31 32 33 34 35 36 37 38
#ifndef S_IRUSR					/* XXX [TRH] should be in a header */
#define S_IRUSR		 S_IREAD
#define S_IWUSR		 S_IWRITE
#define S_IXUSR		 S_IEXEC
#define S_IRGRP		 ((S_IRUSR)>>3)
#define S_IWGRP		 ((S_IWUSR)>>3)
#define S_IXGRP		 ((S_IXUSR)>>3)
#define S_IROTH		 ((S_IRUSR)>>6)
#define S_IWOTH		 ((S_IWUSR)>>6)
#define S_IXOTH		 ((S_IXUSR)>>6)
39 40
#endif

41 42
#ifndef FRONTEND
/* We use only 3-parameter elog calls in this file, for simplicity */
43
/* NOTE: caller must provide gettext call around str! */
44
#define log_error(str, param)	elog(LOG, str, param)
45
#else
46
#define log_error(str, param)	(fprintf(stderr, str, param), fputc('\n', stderr))
47 48
#endif

49
#ifdef WIN32_ONLY_COMPILER
50 51 52 53 54 55 56
#define getcwd(cwd,len)  GetCurrentDirectory(len, cwd)
#endif

static int	validate_exec(const char *path);
static int	resolve_symlinks(char *path);
static char *pipe_read_line(char *cmd, char *line, int maxsize);

57

58
/*
59
 * validate_exec -- validate "path" as an executable file
60 61
 *
 * returns 0 if the file is found and no error is encountered.
62 63
 *		  -1 if the regular file "path" does not exist or cannot be executed.
 *		  -2 if the file is otherwise valid but cannot be read.
64
 */
65
static int
66
validate_exec(const char *path)
67
{
68
	struct stat buf;
B
Bruce Momjian 已提交
69

70
#ifndef WIN32
71 72 73
	uid_t		euid;
	struct group *gp;
	struct passwd *pwp;
74 75
	int			i;
	int			in_grp = 0;
B
Bruce Momjian 已提交
76
#else
77
	char		path_exe[MAXPGPATH + sizeof(".exe") - 1];
78
#endif
79 80
	int			is_r = 0;
	int			is_x = 0;
81

82 83
#ifdef WIN32
	/* Win32 requires a .exe suffix for stat() */
84 85
	if (strlen(path) >= strlen(".exe") &&
		pg_strcasecmp(path + strlen(path) - strlen(".exe"), ".exe") != 0)
86 87 88 89 90 91 92
	{
		strcpy(path_exe, path);
		strcat(path_exe, ".exe");
		path = path_exe;
	}
#endif

93 94 95
	/*
	 * Ensure that the file exists and is a regular file.
	 *
96 97
	 * XXX if you have a broken system where stat() looks at the symlink
	 * instead of the underlying file, you lose.
98 99
	 */
	if (stat(path, &buf) < 0)
100
		return -1;
101 102

	if ((buf.st_mode & S_IFMT) != S_IFREG)
103
		return -1;
104 105

	/*
106
	 * Ensure that we are using an authorized executable.
107 108 109 110 111 112
	 */

	/*
	 * Ensure that the file is both executable and readable (required for
	 * dynamic loading).
	 */
113
#ifdef WIN32
B
Bruce Momjian 已提交
114 115 116
	is_r = buf.st_mode & S_IRUSR;
	is_x = buf.st_mode & S_IXUSR;
	return is_x ? (is_r ? 0 : -2) : -1;
117
#else
118
	euid = geteuid();
119 120

	/* If owned by us, just check owner bits */
121 122 123 124
	if (euid == buf.st_uid)
	{
		is_r = buf.st_mode & S_IRUSR;
		is_x = buf.st_mode & S_IXUSR;
125
		return is_x ? (is_r ? 0 : -2) : -1;
126
	}
127 128

	/* OK, check group bits */
B
Bruce Momjian 已提交
129 130

	pwp = getpwuid(euid);		/* not thread-safe */
131 132
	if (pwp)
	{
133
		if (pwp->pw_gid == buf.st_gid)	/* my primary group? */
134 135
			++in_grp;
		else if (pwp->pw_name &&
136
				 (gp = getgrgid(buf.st_gid)) != NULL && /* not thread-safe */
137
				 gp->gr_mem != NULL)
B
Bruce Momjian 已提交
138
		{						/* try list of member groups */
139 140 141 142 143 144 145 146 147 148 149 150 151
			for (i = 0; gp->gr_mem[i]; ++i)
			{
				if (!strcmp(gp->gr_mem[i], pwp->pw_name))
				{
					++in_grp;
					break;
				}
			}
		}
		if (in_grp)
		{
			is_r = buf.st_mode & S_IRGRP;
			is_x = buf.st_mode & S_IXGRP;
152
			return is_x ? (is_r ? 0 : -2) : -1;
153 154
		}
	}
155 156

	/* Check "other" bits */
157 158
	is_r = buf.st_mode & S_IROTH;
	is_x = buf.st_mode & S_IXOTH;
159
	return is_x ? (is_r ? 0 : -2) : -1;
160
#endif
161 162
}

163

164
/*
165
 * find_my_exec -- find an absolute path to a valid executable
166
 *
167 168 169 170
 *	argv0 is the name passed on the command line
 *	retpath is the output area (must be of size MAXPGPATH)
 *	Returns 0 if OK, -1 if error.
 *
171
 * The reason we have to work so hard to find an absolute path is that
172 173
 * on some platforms we can't do dynamic loading unless we know the
 * executable's location.  Also, we need a full path not a relative
174 175 176
 * path because we will later change working directory.  Finally, we want
 * a true path not a symlink location, so that we can locate other files
 * that are part of our installation relative to the executable.
177
 *
178
 * This function is not thread-safe because it calls validate_exec(),
B
Bruce Momjian 已提交
179
 * which calls getgrgid().	This function should be used only in
180
 * non-threaded binaries, not in library routines.
181 182
 */
int
B
Bruce Momjian 已提交
183
find_my_exec(const char *argv0, char *retpath)
184
{
B
Bruce Momjian 已提交
185 186 187
	char		cwd[MAXPGPATH],
				test_path[MAXPGPATH];
	char	   *path;
188

189
	if (!getcwd(cwd, MAXPGPATH))
190
	{
191
		log_error(_("could not identify current directory: %s"),
192 193 194
				  strerror(errno));
		return -1;
	}
195

196
	/*
197
	 * If argv0 contains a separator, then PATH wasn't used.
198
	 */
199
	if (first_dir_separator(argv0) != NULL)
200
	{
201
		if (is_absolute_path(argv0))
B
Bruce Momjian 已提交
202
			StrNCpy(retpath, argv0, MAXPGPATH);
203
		else
204
			join_path_components(retpath, cwd, argv0);
B
Bruce Momjian 已提交
205
		canonicalize_path(retpath);
206

B
Bruce Momjian 已提交
207
		if (validate_exec(retpath) == 0)
208
			return resolve_symlinks(retpath);
209

210
		log_error(_("invalid binary \"%s\""), retpath);
211
		return -1;
212
	}
213

214 215
#ifdef WIN32
	/* Win32 checks the current directory first for names without slashes */
216 217
	join_path_components(retpath, cwd, argv0);
	if (validate_exec(retpath) == 0)
218
		return resolve_symlinks(retpath);
219 220
#endif

221
	/*
B
Bruce Momjian 已提交
222 223
	 * Since no explicit path was supplied, the user must have been relying on
	 * PATH.  We'll search the same PATH.
224
	 */
225
	if ((path = getenv("PATH")) && *path)
226
	{
B
Bruce Momjian 已提交
227 228
		char	   *startp = NULL,
				   *endp = NULL;
229 230

		do
231
		{
232 233 234 235
			if (!startp)
				startp = path;
			else
				startp = endp + 1;
236

237
			endp = first_path_separator(startp);
238
			if (!endp)
B
Bruce Momjian 已提交
239
				endp = startp + strlen(startp); /* point to end */
240 241 242 243

			StrNCpy(test_path, startp, Min(endp - startp + 1, MAXPGPATH));

			if (is_absolute_path(test_path))
244
				join_path_components(retpath, test_path, argv0);
245
			else
246 247 248 249
			{
				join_path_components(retpath, cwd, test_path);
				join_path_components(retpath, retpath, argv0);
			}
B
Bruce Momjian 已提交
250
			canonicalize_path(retpath);
251

B
Bruce Momjian 已提交
252
			switch (validate_exec(retpath))
253
			{
B
Bruce Momjian 已提交
254
				case 0: /* found ok */
255
					return resolve_symlinks(retpath);
256
				case -1:		/* wasn't even a candidate, keep looking */
257
					break;
258
				case -2:		/* found but disqualified */
259
					log_error(_("could not read binary \"%s\""),
260
							  retpath);
261
					break;
262
			}
263
		} while (*endp);
264 265
	}

266
	log_error(_("could not find a \"%s\" to execute"), argv0);
267
	return -1;
268
}
269

270 271 272 273

/*
 * resolve_symlinks - resolve symlinks to the underlying file
 *
274
 * Replace "path" by the absolute path to the referenced file.
275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291
 *
 * Returns 0 if OK, -1 if error.
 *
 * Note: we are not particularly tense about producing nice error messages
 * because we are not really expecting error here; we just determined that
 * the symlink does point to a valid executable.
 */
static int
resolve_symlinks(char *path)
{
#ifdef HAVE_READLINK
	struct stat buf;
	char		orig_wd[MAXPGPATH],
				link_buf[MAXPGPATH];
	char	   *fname;

	/*
B
Bruce Momjian 已提交
292 293
	 * To resolve a symlink properly, we have to chdir into its directory and
	 * then chdir to where the symlink points; otherwise we may fail to
294 295 296
	 * resolve relative links correctly (consider cases involving mount
	 * points, for example).  After following the final symlink, we use
	 * getcwd() to figure out where the heck we're at.
297
	 *
298 299 300
	 * One might think we could skip all this if path doesn't point to a
	 * symlink to start with, but that's wrong.  We also want to get rid of
	 * any directory symlinks that are present in the given path. We expect
B
Bruce Momjian 已提交
301
	 * getcwd() to give us an accurate, symlink-free path.
302 303 304
	 */
	if (!getcwd(orig_wd, MAXPGPATH))
	{
305
		log_error(_("could not identify current directory: %s"),
306 307 308 309 310 311
				  strerror(errno));
		return -1;
	}

	for (;;)
	{
B
Bruce Momjian 已提交
312 313
		char	   *lsep;
		int			rllen;
314 315 316 317 318 319 320

		lsep = last_dir_separator(path);
		if (lsep)
		{
			*lsep = '\0';
			if (chdir(path) == -1)
			{
321
				log_error(_("could not change directory to \"%s\""), path);
322 323 324 325 326 327 328 329 330 331 332 333 334 335
				return -1;
			}
			fname = lsep + 1;
		}
		else
			fname = path;

		if (lstat(fname, &buf) < 0 ||
			(buf.st_mode & S_IFMT) != S_IFLNK)
			break;

		rllen = readlink(fname, link_buf, sizeof(link_buf));
		if (rllen < 0 || rllen >= sizeof(link_buf))
		{
336
			log_error(_("could not read symbolic link \"%s\""), fname);
337 338 339 340 341 342 343 344 345 346 347
			return -1;
		}
		link_buf[rllen] = '\0';
		strcpy(path, link_buf);
	}

	/* must copy final component out of 'path' temporarily */
	strcpy(link_buf, fname);

	if (!getcwd(path, MAXPGPATH))
	{
348
		log_error(_("could not identify current directory: %s"),
349 350 351 352 353 354 355 356
				  strerror(errno));
		return -1;
	}
	join_path_components(path, path, link_buf);
	canonicalize_path(path);

	if (chdir(orig_wd) == -1)
	{
357
		log_error(_("could not change directory to \"%s\""), orig_wd);
358 359
		return -1;
	}
B
Bruce Momjian 已提交
360
#endif   /* HAVE_READLINK */
361 362 363 364 365 366 367 368 369 370 371 372 373 374 375 376 377 378 379 380 381 382 383 384 385 386 387 388 389 390 391 392 393 394 395 396 397 398 399 400 401 402

	return 0;
}


/*
 * Find another program in our binary's directory,
 * then make sure it is the proper version.
 */
int
find_other_exec(const char *argv0, const char *target,
				const char *versionstr, char *retpath)
{
	char		cmd[MAXPGPATH];
	char		line[100];

	if (find_my_exec(argv0, retpath) < 0)
		return -1;

	/* Trim off program name and keep just directory */
	*last_dir_separator(retpath) = '\0';
	canonicalize_path(retpath);

	/* Now append the other program's name */
	snprintf(retpath + strlen(retpath), MAXPGPATH - strlen(retpath),
			 "/%s%s", target, EXE);

	if (validate_exec(retpath) != 0)
		return -1;

	snprintf(cmd, sizeof(cmd), "\"%s\" -V 2>%s", retpath, DEVNULL);

	if (!pipe_read_line(cmd, line, sizeof(line)))
		return -1;

	if (strcmp(line, versionstr) != 0)
		return -2;

	return 0;
}


403
/*
404
 * The runtime library's popen() on win32 does not work when being
405 406 407 408 409 410
 * called from a service when running on windows <= 2000, because
 * there is no stdin/stdout/stderr.
 *
 * Executing a command in a pipe and reading the first line from it
 * is all we need.
 */
B
Bruce Momjian 已提交
411 412
static char *
pipe_read_line(char *cmd, char *line, int maxsize)
413 414
{
#ifndef WIN32
B
Bruce Momjian 已提交
415
	FILE	   *pgver;
416 417 418 419 420 421 422

	/* flush output buffers in case popen does not... */
	fflush(stdout);
	fflush(stderr);

	if ((pgver = popen(cmd, "r")) == NULL)
		return NULL;
B
Bruce Momjian 已提交
423

424 425 426 427 428 429 430 431
	if (fgets(line, maxsize, pgver) == NULL)
	{
		perror("fgets failure");
		return NULL;
	}

	if (pclose_check(pgver))
		return NULL;
B
Bruce Momjian 已提交
432

433
	return line;
B
Bruce Momjian 已提交
434
#else							/* WIN32 */
435

436
	SECURITY_ATTRIBUTES sattr;
B
Bruce Momjian 已提交
437 438 439
	HANDLE		childstdoutrd,
				childstdoutwr,
				childstdoutrddup;
440 441
	PROCESS_INFORMATION pi;
	STARTUPINFO si;
B
Bruce Momjian 已提交
442
	char	   *retval = NULL;
443 444 445 446 447 448 449

	sattr.nLength = sizeof(SECURITY_ATTRIBUTES);
	sattr.bInheritHandle = TRUE;
	sattr.lpSecurityDescriptor = NULL;

	if (!CreatePipe(&childstdoutrd, &childstdoutwr, &sattr, 0))
		return NULL;
B
Bruce Momjian 已提交
450

451 452 453 454 455 456 457 458 459 460 461 462 463 464
	if (!DuplicateHandle(GetCurrentProcess(),
						 childstdoutrd,
						 GetCurrentProcess(),
						 &childstdoutrddup,
						 0,
						 FALSE,
						 DUPLICATE_SAME_ACCESS))
	{
		CloseHandle(childstdoutrd);
		CloseHandle(childstdoutwr);
		return NULL;
	}

	CloseHandle(childstdoutrd);
B
Bruce Momjian 已提交
465 466 467

	ZeroMemory(&pi, sizeof(pi));
	ZeroMemory(&si, sizeof(si));
468 469 470 471 472
	si.cb = sizeof(si);
	si.dwFlags = STARTF_USESTDHANDLES;
	si.hStdError = childstdoutwr;
	si.hStdOutput = childstdoutwr;
	si.hStdInput = INVALID_HANDLE_VALUE;
B
Bruce Momjian 已提交
473

474 475 476 477 478 479 480 481 482 483 484 485
	if (CreateProcess(NULL,
					  cmd,
					  NULL,
					  NULL,
					  TRUE,
					  0,
					  NULL,
					  NULL,
					  &si,
					  &pi))
	{
		/* Successfully started the process */
B
Bruce Momjian 已提交
486
		char	   *lineptr;
487

B
Bruce Momjian 已提交
488 489
		ZeroMemory(line, maxsize);

490 491
		/* Try to read at least one line from the pipe */
		/* This may require more than one wait/read attempt */
B
Bruce Momjian 已提交
492
		for (lineptr = line; lineptr < line + maxsize - 1;)
493
		{
494 495 496 497
			DWORD		bytesread = 0;

			/* Let's see if we can read */
			if (WaitForSingleObject(childstdoutrddup, 10000) != WAIT_OBJECT_0)
B
Bruce Momjian 已提交
498
				break;			/* Timeout, but perhaps we got a line already */
499

B
Bruce Momjian 已提交
500
			if (!ReadFile(childstdoutrddup, lineptr, maxsize - (lineptr - line),
501
						  &bytesread, NULL))
B
Bruce Momjian 已提交
502
				break;			/* Error, but perhaps we got a line already */
503 504 505 506

			lineptr += strlen(lineptr);

			if (!bytesread)
B
Bruce Momjian 已提交
507
				break;			/* EOF */
508 509

			if (strchr(line, '\n'))
B
Bruce Momjian 已提交
510
				break;			/* One or more lines read */
511
		}
512

513
		if (lineptr != line)
514
		{
515 516 517 518
			/* OK, we read some data */
			int			len;

			/* If we got more than one line, cut off after the first \n */
B
Bruce Momjian 已提交
519
			lineptr = strchr(line, '\n');
520
			if (lineptr)
B
Bruce Momjian 已提交
521
				*(lineptr + 1) = '\0';
522 523

			len = strlen(line);
524

525
			/*
B
Bruce Momjian 已提交
526 527
			 * If EOL is \r\n, convert to just \n. Because stdout is a
			 * text-mode stream, the \n output by the child process is
B
Bruce Momjian 已提交
528 529 530
			 * received as \r\n, so we convert it to \n.  The server main.c
			 * sets setvbuf(stdout, NULL, _IONBF, 0) which has the effect of
			 * disabling \n to \r\n expansion for stdout.
531
			 */
B
Bruce Momjian 已提交
532
			if (len >= 2 && line[len - 2] == '\r' && line[len - 1] == '\n')
533
			{
B
Bruce Momjian 已提交
534 535
				line[len - 2] = '\n';
				line[len - 1] = '\0';
536
				len--;
537 538
			}

B
Bruce Momjian 已提交
539
			/*
B
Bruce Momjian 已提交
540 541
			 * We emulate fgets() behaviour. So if there is no newline at the
			 * end, we add one...
B
Bruce Momjian 已提交
542
			 */
B
Bruce Momjian 已提交
543 544
			if (len == 0 || line[len - 1] != '\n')
				strcat(line, "\n");
545 546

			retval = line;
547 548 549 550 551
		}

		CloseHandle(pi.hProcess);
		CloseHandle(pi.hThread);
	}
B
Bruce Momjian 已提交
552

553 554 555 556
	CloseHandle(childstdoutwr);
	CloseHandle(childstdoutrddup);

	return retval;
B
Bruce Momjian 已提交
557
#endif   /* WIN32 */
558 559 560
}


561 562 563 564 565 566 567 568
/*
 * pclose() plus useful error reporting
 * Is this necessary?  bjm 2004-05-11
 * It is better here because pipe.c has win32 backend linkage.
 */
int
pclose_check(FILE *stream)
{
B
Bruce Momjian 已提交
569
	int			exitstatus;
570 571 572 573

	exitstatus = pclose(stream);

	if (exitstatus == 0)
B
Bruce Momjian 已提交
574
		return 0;				/* all is well */
575 576 577 578 579 580 581

	if (exitstatus == -1)
	{
		/* pclose() itself failed, and hopefully set errno */
		perror("pclose failed");
	}
	else if (WIFEXITED(exitstatus))
582
		log_error(_("child process exited with exit code %d"),
B
Bruce Momjian 已提交
583
				  WEXITSTATUS(exitstatus));
584
	else if (WIFSIGNALED(exitstatus))
585
#ifndef WIN32
586
		log_error(_("child process was terminated by signal %d"),
B
Bruce Momjian 已提交
587
				  WTERMSIG(exitstatus));
588 589 590 591
#else
		log_error(_("child process was terminated by exception %X\nSee http://source.winehq.org/source/include/ntstatus.h for a description\nof the hex value."),
				  WTERMSIG(exitstatus));
#endif
592
	else
593
		log_error(_("child process exited with unrecognized status %d"),
B
Bruce Momjian 已提交
594
				  exitstatus);
595 596 597

	return -1;
}
598 599 600 601 602 603 604 605 606 607 608 609 610 611 612 613 614 615 616 617 618 619 620 621 622 623 624 625 626 627 628 629 630 631 632 633 634 635 636 637 638 639 640 641 642 643 644 645 646 647 648 649


/*
 *	set_pglocale_pgservice
 *
 *	Set application-specific locale and service directory
 *
 *	This function takes the value of argv[0] rather than a full path.
 *
 * (You may be wondering why this is in exec.c.  It requires this module's
 * services and doesn't introduce any new dependencies, so this seems as
 * good as anyplace.)
 */
void
set_pglocale_pgservice(const char *argv0, const char *app)
{
	char		path[MAXPGPATH];
	char		my_exec_path[MAXPGPATH];
	char		env_path[MAXPGPATH + sizeof("PGSYSCONFDIR=")];	/* longer than
																 * PGLOCALEDIR */

	/* don't set LC_ALL in the backend */
	if (strcmp(app, "postgres") != 0)
		setlocale(LC_ALL, "");

	if (find_my_exec(argv0, my_exec_path) < 0)
		return;

#ifdef ENABLE_NLS
	get_locale_path(my_exec_path, path);
	bindtextdomain(app, path);
	textdomain(app);

	if (getenv("PGLOCALEDIR") == NULL)
	{
		/* set for libpq to use */
		snprintf(env_path, sizeof(env_path), "PGLOCALEDIR=%s", path);
		canonicalize_path(env_path + 12);
		putenv(strdup(env_path));
	}
#endif

	if (getenv("PGSYSCONFDIR") == NULL)
	{
		get_etc_path(my_exec_path, path);

		/* set for libpq to use */
		snprintf(env_path, sizeof(env_path), "PGSYSCONFDIR=%s", path);
		canonicalize_path(env_path + 13);
		putenv(strdup(env_path));
	}
}