diff src/=environ.c @ 115:c7c930b84dbb

entered into RCS
author Jim Blandy <jimb@redhat.com>
date Mon, 12 Nov 1990 20:20:40 +0000
parents
children
line wrap: on
line diff
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/src/=environ.c	Mon Nov 12 20:20:40 1990 +0000
@@ -0,0 +1,316 @@
+/* Environment-hacking for GNU Emacs subprocess
+   Copyright (C) 1986 Free Software Foundation, Inc.
+
+This file is part of GNU Emacs.
+
+GNU Emacs 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 1, or (at your option)
+any later version.
+
+GNU Emacs 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 GNU Emacs; see the file COPYING.  If not, write to
+the Free Software Foundation, 675 Mass Ave, Cambridge, MA 02139, USA.  */
+
+
+#include "config.h"
+#include "lisp.h"
+
+#ifdef MAINTAIN_ENVIRONMENT
+
+#ifdef VMS
+you lose -- this is un*x-only
+#endif
+
+/* alist of (name-string . value-string) */
+Lisp_Object Venvironment_alist;
+extern char **environ;
+
+void
+set_environment_alist (str, val)
+     register Lisp_Object str, val;
+{
+  register Lisp_Object tem;
+
+  tem = Fassoc (str, Venvironment_alist);
+  if (NULL (tem))
+    if (NULL (val))
+      ;
+    else
+      Venvironment_alist = Fcons (Fcons (str, val), Venvironment_alist);
+  else
+    if (NULL (val))
+      Venvironment_alist = Fdelq (tem, Venvironment_alist);
+    else
+      XCONS (tem)->cdr = val;
+}
+
+
+
+static void
+initialize_environment_alist ()
+{
+  register unsigned char **e, *s;
+  extern char *index ();
+
+  for (e = (unsigned char **) environ; *e; e++)
+    {
+      s = (unsigned char *) index (*e, '=');
+      if (s)
+	set_environment_alist (make_string (*e, s - *e),
+			       build_string (s + 1));
+    }
+}
+
+
+unsigned char *
+getenv_1 (str, ephemeral)
+     register unsigned char *str;
+     int ephemeral;		/* if ephmeral, don't need to gc-proof */
+{
+  register Lisp_Object env;
+  int len = strlen (str);
+
+  for (env = Venvironment_alist; CONSP (env); env = XCONS (env)->cdr)
+    {
+      register Lisp_Object car = XCONS (env)->car;
+      register Lisp_Object tem = XCONS (car)->car;
+
+      if ((len == XSTRING (tem)->size) &&
+	  (!bcmp (str, XSTRING (tem)->data, len)))
+	{
+	  /* Found it in the lisp environment */
+	  tem = XCONS (car)->cdr;
+	  if (ephemeral)
+	    /* Caller promises that gc won't make him lose */
+	    return XSTRING (tem)->data;
+	  else
+	    {
+	      register unsigned char **e;
+	      unsigned char *s;
+	      int ll = XSTRING (tem)->size;
+
+	      /* Look for element in the original unix environment */
+	      for (e = (unsigned char **) environ; *e; e++)
+		if (!bcmp (str, *e, len) && *(*e + len) == '=')
+		  {
+		    s = *e + len + 1;
+		    if (strlen (s) >= ll)
+		      /* User hasn't either hasn't munged it or has set it
+			 to something shorter -- we don't have to cons */
+		      goto copy;
+		    else
+		      goto cons;
+		  };
+	    cons:
+	      /* User has setenv'ed it to a diferent value, and our caller
+		 isn't guaranteeing that he won't stash it away somewhere.
+		 We can't just return a pointer to the lisp string, as that
+		 will be corrupted when gc happens.  So, we cons (in such
+		 a way that it can't be freed -- though this isn't such a
+		 problem since the only callers of getenv (as opposed to
+		 those of egetenv) are very early, before the user -could-
+		 have frobbed the environment. */
+	      s = (unsigned char *) xmalloc (ll + 1);
+	    copy:
+	      bcopy (XSTRING (tem)->data, s, ll + 1);
+	      return (s);
+	    }
+	}
+    }
+  return ((unsigned char *) 0);
+}
+
+/* unsigned  -- stupid delcaration in lisp.h */ char *
+getenv (str)
+     register unsigned char *str;
+{
+  return ((char *) getenv_1 (str, 0));
+}
+
+unsigned char *
+egetenv (str)
+     register unsigned char *str;
+{
+  return (getenv_1 (str, 1));
+}
+
+
+#if (1 == 1) /* use caller-alloca versions, rather than callee-malloc */
+int
+size_of_current_environ ()
+{
+  register int size;
+  Lisp_Object tem;
+
+  tem = Flength (Venvironment_alist);
+  
+  size = (XINT (tem) + 1) * sizeof (unsigned char *);
+  /* + 1 for environment-terminating 0 */
+
+  for (tem = Venvironment_alist; !NULL (tem); tem = XCONS (tem)->cdr)
+    {
+      register Lisp_Object str, val;
+
+      str = XCONS (XCONS (tem)->car)->car;
+      val = XCONS (XCONS (tem)->car)->cdr;
+
+      size += (XSTRING (str)->size +
+	       XSTRING (val)->size +
+	       2);	/* 1 for '=', 1 for '\000' */
+    }
+  return size;
+}
+
+void
+get_current_environ (memory_block)
+     unsigned char **memory_block;
+{
+  register unsigned char **e, *s;
+  register int len;
+  register Lisp_Object tem;
+
+  e = memory_block;
+
+  tem = Flength (Venvironment_alist);
+  
+  s = (unsigned char *) memory_block
+		+ (XINT (tem) + 1) * sizeof (unsigned char *);
+
+  for (tem = Venvironment_alist; !NULL (tem); tem = XCONS (tem)->cdr)
+    {
+      register Lisp_Object str, val;
+
+      str = XCONS (XCONS (tem)->car)->car;
+      val = XCONS (XCONS (tem)->car)->cdr;
+
+      *e++ = s;
+      len = XSTRING (str)->size;
+      bcopy (XSTRING (str)->data, s, len);
+      s += len;
+      *s++ = '=';
+      len = XSTRING (val)->size;
+      bcopy (XSTRING (val)->data, s, len);
+      s += len;
+      *s++ = '\000';
+    }
+  *e = 0;
+}
+
+#else
+/* dead code (this function mallocs, caller frees) superseded by above (which allows caller to use alloca) */
+unsigned char **
+current_environ ()
+{
+  unsigned char **env;
+  register unsigned char **e, *s;
+  register int len, env_len;
+  Lisp_Object tem;
+  Lisp_Object str, val;
+
+  tem = Flength (Venvironment_alist);
+
+  env_len = (XINT (tem) + 1) * sizeof (char *);
+  /* + 1 for terminating 0 */
+
+  len = 0;
+  for (tem = Venvironment_alist; !NULL (tem); tem = XCONS (tem)->cdr)
+    {
+      str = XCONS (XCONS (tem)->car)->car;
+      val = XCONS (XCONS (tem)->car)->cdr;
+
+      len += (XSTRING (str)->size +
+	      XSTRING (val)->size +
+	      2);
+    }
+
+  e = env = (unsigned char **) xmalloc (env_len + len);
+  s = (unsigned char *) env + env_len;
+
+  for (tem = Venvironment_alist; !NULL (tem); tem = XCONS (tem)->cdr)
+    {
+      str = XCONS (XCONS (tem)->car)->car;
+      val = XCONS (XCONS (tem)->car)->cdr;
+
+      *e++ = s;
+      len = XSTRING (str)->size;
+      bcopy (XSTRING (str)->data, s, len);
+      s += len;
+      *s++ = '=';
+      len = XSTRING (val)->size;
+      bcopy (XSTRING (val)->data, s, len);
+      s += len;
+      *s++ = '\000';
+    }
+  *e = 0;
+
+  return env;
+}
+
+#endif /* dead code */
+
+
+DEFUN ("getenv", Fgetenv, Sgetenv, 1, 2, "sEnvironment variable: \np",
+  "Return the value of environment variable VAR, as a string.\n\
+When invoked interactively, print the value in the echo area.\n\
+VAR is a string, the name of the variable,\n\
+ or the symbol t, meaning to return an alist representing the\n\
+ current environment.")
+  (str, interactivep)
+     Lisp_Object str, interactivep;
+{
+  Lisp_Object val;
+  
+  if (str == Qt)		/* If arg is t, return whole environment */
+    return (Fcopy_alist (Venvironment_alist));
+
+  CHECK_STRING (str, 0);
+  val = Fcdr (Fassoc (str, Venvironment_alist));
+  if (!NULL (interactivep))
+    {
+      if (NULL (val))
+	message ("%s not defined in environment", XSTRING (str)->data);
+      else
+	message ("\"%s\"", XSTRING (val)->data);
+    }
+  return val;
+}
+
+DEFUN ("setenv", Fsetenv, Ssetenv, 1, 2,
+  "sEnvironment variable: \nsSet %s to value: ",
+  "Set the value of environment variable VAR to VALUE.\n\
+Both args must be strings.  Returns VALUE.")
+  (str, val)
+     Lisp_Object str;
+     Lisp_Object val;
+{
+  Lisp_Object tem;
+
+  CHECK_STRING (str, 0);
+  if (!NULL (val))
+    CHECK_STRING (val, 0);
+
+  set_environment_alist (str, val);
+  return val;
+}
+
+
+syms_of_environ ()
+{
+  staticpro (&Venvironment_alist);
+  defsubr (&Ssetenv);
+  defsubr (&Sgetenv);
+}
+
+init_environ ()
+{
+  Venvironment_alist = Qnil;
+  initialize_environment_alist ();
+}
+
+#endif /* MAINTAIN_ENVIRONMENT */