1 /* Client process that communicates with GNU Emacs acting as server.
2 Copyright (C) 1986, 1987, 1994 Free Software Foundation, Inc.
4 This file is part of GNU Emacs.
6 GNU Emacs is free software; you can redistribute it and/or modify
7 it under the terms of the GNU General Public License as published by
8 the Free Software Foundation; either version 2, or (at your option)
11 GNU Emacs is distributed in the hope that it will be useful,
12 but WITHOUT ANY WARRANTY; without even the implied warranty of
13 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 GNU General Public License for more details.
16 You should have received a copy of the GNU General Public License
17 along with GNU Emacs; see the file COPYING. If not, write to
18 the Free Software Foundation, 675 Mass Ave, Cambridge, MA 02139, USA. */
22 #include <../src/config.h>
30 #if !defined (HAVE_SOCKETS) && !defined (HAVE_SYSVIPC)
37 fprintf (stderr
, "%s: Sorry, the Emacs server is supported only\n",
39 fprintf (stderr
, "on systems with Berkeley sockets or System V IPC.\n");
43 #else /* HAVE_SOCKETS or HAVE_SYSVIPC */
45 #if defined (HAVE_SOCKETS) && ! defined (NO_SOCKETS_IN_FILE_SYSTEM)
46 /* BSD code is very different from SYSV IPC code */
48 #include <sys/types.h>
49 #include <sys/socket.h>
55 extern char *strerror ();
66 struct sockaddr_un server
;
67 char *homedir
, *cwd
, *str
;
70 char *getenv (), *getwd ();
76 fprintf (stderr
, "Usage: %s [+linenumber] filename\n", argv
[0]);
81 * Open up an AF_UNIX socket in this person's home directory
84 if ((s
= socket (AF_UNIX
, SOCK_STREAM
, 0)) < 0)
86 fprintf (stderr
, "%s: ", argv
[0]);
90 server
.sun_family
= AF_UNIX
;
91 #ifndef SERVER_HOME_DIR
95 gethostname (system_name
, sizeof (system_name
));
96 sprintf (server
.sun_path
, "/tmp/esrv%d-%s", geteuid (), system_name
);
98 if (stat (server
.sun_path
, &statbfr
) == -1)
102 "%s: can't find socket; have you started the server?\n",
105 fprintf (stderr
, "%s: can't stat %s: %s\n",
106 argv
[0], server
.sun_path
, strerror (errno
));
109 if (statbfr
.st_uid
!= geteuid ())
111 fprintf (stderr
, "%s: Invalid socket owner\n", argv
[0]);
116 if ((homedir
= getenv ("HOME")) == NULL
)
118 fprintf (stderr
, "%s: No home directory\n", argv
[0]);
121 strcpy (server
.sun_path
, homedir
);
122 strcat (server
.sun_path
, "/.emacs-server-");
123 gethostname (system_name
, sizeof (system_name
));
124 strcat (server
.sun_path
, system_name
);
127 if (connect (s
, (struct sockaddr
*) &server
, strlen (server
.sun_path
) + 2)
130 fprintf (stderr
, "%s: ", argv
[0]);
134 if ((out
= fdopen (s
, "r+")) == NULL
)
136 fprintf (stderr
, "%s: ", argv
[0]);
142 cwd
= getwd (string
);
144 cwd
= getcwd (string
, sizeof string
);
148 /* getwd puts message in STRING if it fails. */
149 fprintf (stderr
, "%s: %s (%s)\n", argv
[0], string
, strerror (errno
));
153 for (i
= 1; i
< argc
; i
++)
157 char *p
= argv
[i
] + 1;
158 while (*p
>= '0' && *p
<= '9') p
++;
160 fprintf (out
, "%s/", cwd
);
162 else if (*argv
[i
] != '/')
163 fprintf (out
, "%s/", cwd
);
164 fprintf (out
, "%s ", argv
[i
]);
169 printf ("Waiting for Emacs...");
172 rewind (out
); /* re-read the output */
173 str
= fgets (string
, BUFSIZ
, out
);
175 /* Now, wait for an answer and print any messages. */
177 while (str
= fgets (string
, BUFSIZ
, out
))
183 #else /* This is the SYSV IPC section */
185 #include <sys/types.h>
188 #include <sys/utsname.h>
191 char *getwd (), *getcwd (), *getenv ();
192 struct utsname system_name
;
200 /* Size of text allocated in MSGP. */
201 int size_allocated
= BUFSIZ
;
202 /* Amount of text used in MSGP. */
205 = (struct msgbuf
*) malloc (sizeof (struct msgbuf
) + size_allocated
);
206 struct msqid_ds
* msg_st
;
207 char *homedir
, buf
[BUFSIZ
];
211 char *progname
= argv
[0];
215 fprintf (stderr
, "Usage: %s [+linenumber] filename\n", argv
[0]);
220 * Create a message queue using ~/.emacs-server as the path for ftok
222 if ((homedir
= getenv ("HOME")) == NULL
)
224 fprintf (stderr
, "%s: No home directory\n", argv
[0]);
227 strcpy (buf
, homedir
);
228 #ifndef HAVE_LONG_FILE_NAMES
229 /* If file names are short, we can't fit the host name. */
230 strcat (buf
, "/.emacs-server");
232 strcat (buf
, "/.emacs-server-");
233 uname (&system_name
);
234 strcat (buf
, system_name
.nodename
);
237 key
= ftok (buf
, 1); /* unlikely to be anyone else using it */
238 s
= msgget (key
, 0600 | IPC_CREAT
);
241 fprintf (stderr
, "%s: ", argv
[0]);
246 /* Determine working dir, so we can prefix it to all the arguments. */
248 temp
= getwd (gwdirb
);
250 temp
= getcwd (gwdirb
, sizeof gwdirb
);
256 /* On some systems, cwd can look like `@machine/...';
257 ignore everything before the first slash in such a case. */
258 while (*cwd
&& *cwd
!= '/')
264 fprintf (stderr
, "%s: %s\n", argv
[0], cwd
);
274 char *modified_arg
= argv
[0];
275 if (*modified_arg
== '+')
277 char *p
= modified_arg
+ 1;
278 while (*p
>= '0' && *p
<= '9') p
++;
282 else if (*modified_arg
!= '/')
286 used
+= strlen (cwd
);
287 used
+= strlen (modified_arg
) + 1;
288 while (used
+ 2 > size_allocated
)
291 msgp
= (struct msgbuf
*) realloc (msgp
,
292 (sizeof (struct msgbuf
)
297 strcat (msgp
->mtext
, cwd
);
299 strcat (msgp
->mtext
, modified_arg
);
300 strcat (msgp
->mtext
, " ");
303 strcat (msgp
->mtext
, "\n");
304 #ifdef HPUX /* HPUX has a bug. */
305 if (strlen (msgp
->mtext
) >= 512)
307 fprintf (stderr
, "%s: args too long for msgsnd\n", progname
);
312 if (msgsnd (s
, msgp
, strlen (msgp
->mtext
)+1, 0) < 0)
314 fprintf (stderr
, "%s: ", progname
);
319 * Now, wait for an answer
321 printf ("Waiting for Emacs...");
324 msgrcv (s
, msgp
, BUFSIZ
, getpid (), 0); /* wait for anything back */
325 strcpy (buf
, msgp
->mtext
);
327 printf ("\n%s\n", buf
);
331 #endif /* HAVE_SYSVIPC */
333 #endif /* HAVE_SOCKETS or HAVE_SYSVIPC */
335 #ifndef HAVE_STRERROR
340 extern char *sys_errlist
[];
343 if (errnum
>= 0 && errnum
< sys_nerr
)
344 return sys_errlist
[errnum
];
345 return (char *) "Unknown error";
348 #endif /* ! HAVE_STRERROR */