summaryrefslogtreecommitdiff
path: root/lib-src
diff options
context:
space:
mode:
authorGerd Moellmann <gerd@gnu.org>2000-01-28 15:02:20 +0000
committerGerd Moellmann <gerd@gnu.org>2000-01-28 15:02:20 +0000
commit9f637eea618b3b6b764b92405a480c254756f94a (patch)
tree3aa571fe716fe83937d20ee45abec22f0e61d1c0 /lib-src
parentb02786f9b40a1f1f268550a20ce8a6ac1d828969 (diff)
downloademacs-9f637eea618b3b6b764b92405a480c254756f94a.tar.gz
(socket_status): New function.
(main): if $LOGNAME or $USER exist and differ from our euid, look for a socket based on the UID associated with the name.
Diffstat (limited to 'lib-src')
-rw-r--r--lib-src/emacsclient.c91
1 files changed, 76 insertions, 15 deletions
diff --git a/lib-src/emacsclient.c b/lib-src/emacsclient.c
index 01ae0356797..c4e01b616ff 100644
--- a/lib-src/emacsclient.c
+++ b/lib-src/emacsclient.c
@@ -1,5 +1,5 @@
/* Client process that communicates with GNU Emacs acting as server.
- Copyright (C) 1986, 1987, 1994, 1999 Free Software Foundation, Inc.
+ Copyright (C) 1986, 1987, 1994, 1999, 2000 Free Software Foundation, Inc.
This file is part of GNU Emacs.
@@ -29,6 +29,12 @@ Boston, MA 02111-1307, USA. */
#include <unistd.h>
#endif
+#ifdef VMS
+# include "vms-pwd.h"
+#else
+# include <pwd.h>
+#endif /* not VMS */
+
char *getenv (), *getwd ();
char *getcwd ();
@@ -217,6 +223,26 @@ main (argc, argv)
extern char *strerror ();
extern int errno;
+/* Three possibilities:
+ 2 - can't be `stat'ed (sets errno)
+ 1 - isn't owned by us
+ 0 - success: none of the above */
+
+static int
+socket_status (socket_name)
+ char *socket_name;
+{
+ struct stat statbfr;
+
+ if (stat (socket_name, &statbfr) == -1)
+ return 2;
+
+ if (statbfr.st_uid != geteuid ())
+ return 1;
+
+ return 0;
+}
+
int
main (argc, argv)
int argc;
@@ -272,25 +298,60 @@ main (argc, argv)
#ifndef SERVER_HOME_DIR
{
struct stat statbfr;
+ int sock_status = 0;
sprintf (server.sun_path, "/tmp/esrv%d-%s", geteuid (), system_name);
- if (stat (server.sun_path, &statbfr) == -1)
+ /* See if the socket exists, and if it's owned by us. */
+ sock_status = socket_status (server.sun_path);
+ if (sock_status)
{
- if (errno == ENOENT)
- fprintf (stderr,
- "%s: can't find socket; have you started the server?\n",
- argv[0]);
- else
- fprintf (stderr, "%s: can't stat %s: %s\n",
- argv[0], server.sun_path, strerror (errno));
- fail (argc, argv);
- }
- if (statbfr.st_uid != geteuid ())
- {
- fprintf (stderr, "%s: Invalid socket owner\n", argv[0]);
- fail (argc, argv);
+ /* Failing that, see if LOGNAME or USER exist and differ from
+ our euid. If so, look for a socket based on the UID
+ associated with the name. This is reminiscent of the logic
+ that init_editfns uses to set the global Vuser_full_name. */
+
+ char *user_name = (char *) getenv ("LOGNAME");
+ if (!user_name)
+ user_name = (char *) getenv ("USER");
+
+ if (user_name)
+ {
+ struct passwd *pw = getpwnam (user_name);
+ if (pw && (pw->pw_uid != geteuid ()))
+ {
+ /* We're running under su, apparently. */
+ sprintf (server.sun_path, "/tmp/esrv%d-%s",
+ pw->pw_uid, system_name);
+ sock_status = socket_status (server.sun_path);
+ }
+ }
}
+
+ switch (sock_status)
+ {
+ case 1:
+ /* There's a socket, but it isn't owned by us. This is OK if
+ we are root. */
+ if (0 != geteuid ())
+ {
+ fprintf (stderr, "%s: Invalid socket owner\n", argv[0]);
+ fail (argc, argv);
+ }
+ break;
+
+ case 2:
+ /* `stat' failed */
+ if (errno == ENOENT)
+ fprintf (stderr,
+ "%s: can't find socket; have you started the server?\n",
+ argv[0]);
+ else
+ fprintf (stderr, "%s: can't stat %s: %s\n",
+ argv[0], server.sun_path, strerror (errno));
+ fail (argc, argv);
+ break;
+ }
}
#else
if ((homedir = getenv ("HOME")) == NULL)