unofficial mirror of bug-gnu-emacs@gnu.org 
 help / color / mirror / code / Atom feed
* bug#33255: 27.0.50; expand-file-name: default directory expanded twice if relative
@ 2018-11-04 10:54 immerrr again
  2018-11-04 12:27 ` Noam Postavsky
  2018-11-13 18:26 ` Paul Eggert
  0 siblings, 2 replies; 16+ messages in thread
From: immerrr again @ 2018-11-04 10:54 UTC (permalink / raw)
  To: 33255

I have recently tried to play with a clean .emacs.d directory by setting
HOME=clean/home/dir, but faced an error coming from deep inside
package-install. The error boils down to the fact that
`expand-file-name' for some reason doubles the `default-directory'
prefix when it is relative:


$ emacs -Q -batch --eval '(let ((default-directory "DEFAULT-DIR/"))
(print (expand-file-name "EXPANDED-DIR/")))'

"DEFAULT-DIR/DEFAULT-DIR/EXPANDED-DIR/"


Interestingly, if you run `expand-file-name' twice you end up with four
prefixes:


$ emacs -Q -batch --eval '(let ((default-directory "DEFAULT-DIR/"))
(print (expand-file-name (expand-file-name "EXPANDED-DIR/"))))'

"DEFAULT-DIR/DEFAULT-DIR/DEFAULT-DIR/DEFAULT-DIR/EXPANDED-DIR/"


I couldn't find any requirement for DEFAULT-DIRECTORY parameter or
`default-directory' variable to be absolute, so I would assume this is a
bug in expand-file-name.


In GNU Emacs 27.0.50 (build 1, x86_64-pc-linux-gnu, GTK+ Version 3.18.9)
 of 2018-07-09 built on mmrcomp
Repository revision: 3307353e13a9226d477c9b1a39baae76584b90b9
Windowing system distributor 'The X.Org Foundation', version 11.0.11906000
System Description: Ubuntu 16.04.5 LTS

Recent messages:
Mark set
Saving file /home/immerrr/.cask/cask-cli.el...
Wrote /home/immerrr/.cask/cask-cli.el
Saving file /home/immerrr/.cask/cask-cli.el...
Wrote /home/immerrr/.cask/cask-cli.el
Saving file /home/immerrr/.cask/cask-cli.el...
Wrote /home/immerrr/.cask/cask-cli.el
Saving file /home/immerrr/.cask/cask-cli.el...
Wrote /home/immerrr/.cask/cask-cli.el
Mark set

Configured using:
 'configure --prefix=/home/immerrr/.local'

Configured features:
XPM JPEG TIFF GIF PNG RSVG IMAGEMAGICK SOUND GPM DBUS GSETTINGS NOTIFY
LIBSELINUX GNUTLS LIBXML2 FREETYPE M17N_FLT LIBOTF XFT ZLIB
TOOLKIT_SCROLL_BARS GTK3 X11 THREADS JSON LCMS2

Important settings:
  value of $LC_TIME: en_GB.UTF-8
  value of $LANG: en_US.UTF-8
  value of $XMODIFIERS: @im=ibus
  locale-coding-system: utf-8-unix

Major mode: el

Minor modes in effect:
  magit-auto-revert-mode: t
  auto-revert-mode: t
  global-git-commit-mode: t
  async-bytecomp-package-mode: t
  global-undo-tree-mode: t
  undo-tree-mode: t
  diff-auto-refine-mode: t
  nameless-mode: t
  guide-key-mode: t
  whitespace-mode: t
  flycheck-mode: t
  shell-dirtrack-mode: t
  counsel-projectile-mode: t
  projectile-mode: t
  ivy-mode: t
  global-company-mode: t
  company-mode: t
  pyvenv-mode: t
  paredit-mode: t
  auto-compile-on-save-mode: t
  auto-compile-mode: t
  recentf-mode: t
  yas-global-mode: t
  yas-minor-mode: t
  save-place-mode: t
  show-paren-mode: t
  global-display-line-numbers-mode: t
  display-line-numbers-mode: t
  override-global-mode: t
  bar-cursor-mode: t
  tooltip-mode: t
  global-eldoc-mode: t
  eldoc-mode: t
  electric-indent-mode: t
  mouse-wheel-mode: t
  menu-bar-mode: t
  file-name-shadow-mode: t
  global-font-lock-mode: t
  font-lock-mode: t
  auto-composition-mode: t
  auto-encryption-mode: t
  auto-compression-mode: t
  size-indication-mode: t
  column-number-mode: t
  line-number-mode: t
  transient-mark-mode: t

Load-path shadows:
/home/immerrr/.emacs.d/elpa/ample-regexps-20151023.300/init-tryout
hides /home/immerrr/.emacs.d/elpa/lua-mode-20180104.626/init-tryout

Features:
(shadow sort mail-extr emacsbug sendmail macrostep-c cmacexp macrostep
cc-mode cc-fonts cc-guess cc-menus cc-cmds cc-styles cc-align cc-engine
cc-vars cc-defs ielm tar-mode git-rebase rect bug-reference py-isort
executable vc vc-dispatcher magit-bookmark magit-obsolete magit-blame
magit-stash magit-bisect magit-remote magit-commit magit-sequence
magit-notes magit-worktree magit-branch magit-collab ghub url-auth
magit-files magit-refs magit-status magit dired-x magit-repos
magit-apply magit-wip magit-log magit-diff smerge-mode magit-core
magit-autorevert autorevert filenotify magit-process magit-margin
magit-mode git-commit magit-git magit-section magit-utils ido crm
magit-popup log-edit message rfc822 mml mml-sec epa epg gnus-util rmail
rmail-loaddefs mailabbrev gmm-utils mailheader pcvs-util add-log
with-editor term ehelp esh-var esh-cmd esh-opt esh-io esh-ext esh-proc
esh-arg esh-groups eshell esh-module esh-mode async-bytecomp async
server imenu undo-tree diff edebug network-stream puny nsm rmc
company-jedi jedi-core epc ctable concurrent cap-words superword subword
pipenv f eieio-opt speedbar sb-image ezimage dframe warnings pulse
cl-print debug vc-git diff-mode bookmark pp company-elisp nameless
guide-key popwin face-remap disp-table whitespace flycheck find-func rx
jka-compr let-alist colir mmr-org-settings python tramp-sh tramp
trampver tramp-compat tramp-loaddefs ucs-normalize shell pcomplete
parse-time format-spec counsel-projectile projectile skeleton ibuf-macs
wgrep grep ibuf-ext ibuffer ibuffer-loaddefs counsel dired
dired-loaddefs compile esh-util swiper ivy-rich ivy delsel ivy-overlay
ffap thingatpt traad virtualenvwrapper gud comint ansi-color s
request-deferred request mail-utils url url-proxy url-privacy url-expand
url-methods url-history url-cookie url-domsuf url-util mailcap popup
python-environment deferred subr-x company-oddmuse company-keywords
company-etags etags xref project company-gtags company-dabbrev-code
company-dabbrev company-files company-capf company-cmake company-xcode
company-clang company-semantic company-eclim company-template
company-css company-nxml company-bbdb company pyvenv paredit
auto-compile packed recentf tree-widget wid-edit yasnippet derived
elec-pair saveplace hydra ring lv paren solarized-dark-theme
solarized-theme solarized color dash display-line-numbers
mmr-dir-local-env mmr-bootstrap quelpa-use-package cl-extra advice pcase
quelpa mm-decode mm-bodies mm-encode mail-parse rfc2231 rfc2047 rfc2045
mm-util ietf-drums mail-prsvr help-fns radix-tree help-mode
package-recipe-mode edmacro kmacro package-build-badges package-build
lisp-mnt use-package use-package-ensure use-package-delight
use-package-diminish use-package-bind-key use-package-core bind-key
diminish cl linum-ex bar-cursor easy-mmode mmr-minimal mmr-ediff
windmove time-date mule-util info finder-inf package easymenu epg-config
url-handlers url-parse auth-source cl-seq eieio eieio-core cl-macs
eieio-loaddefs password-cache json map url-vars seq byte-opt gv bytecomp
byte-compile cconv cl-loaddefs cl-lib tooltip eldoc electric uniquify
ediff-hook vc-hooks lisp-float-type mwheel term/x-win x-win
term/common-win x-dnd tool-bar dnd fontset image regexp-opt fringe
tabulated-list replace newcomment text-mode elisp-mode lisp-mode
prog-mode register page menu-bar rfn-eshadow isearch timer select
scroll-bar mouse jit-lock font-lock syntax facemenu font-core
term/tty-colors frame cl-generic cham georgian utf-8-lang misc-lang
vietnamese tibetan thai tai-viet lao korean japanese eucjp-ms cp51932
hebrew greek romanian slovak czech european ethiopic indian cyrillic
chinese composite charscript charprop case-table epa-hook jka-cmpr-hook
help simple abbrev obarray minibuffer cl-preloaded nadvice loaddefs
button faces cus-face macroexp files text-properties overlay sha1 md5
base64 format env code-pages mule custom widget hashtable-print-readable
backquote dbusbind inotify lcms2 dynamic-setting system-font-setting
font-render-setting move-toolbar gtk x-toolkit x multi-tty
make-network-process emacs)

Memory information:
((conses 16 1357726 145867)
 (symbols 48 50837 2)
 (miscs 40 6306 4056)
 (strings 32 121140 10207)
 (string-bytes 1 4278168)
 (vectors 16 68224)
 (vector-slots 8 1809895 106986)
 (floats 8 422 1779)
 (intervals 56 116849 2490)
 (buffers 992 53)
 (heap 1024 76825 5884))





^ permalink raw reply	[flat|nested] 16+ messages in thread

* bug#33255: 27.0.50; expand-file-name: default directory expanded twice if relative
  2018-11-04 10:54 bug#33255: 27.0.50; expand-file-name: default directory expanded twice if relative immerrr again
@ 2018-11-04 12:27 ` Noam Postavsky
  2018-11-05  0:58   ` Glenn Morris
  2018-11-13 18:26 ` Paul Eggert
  1 sibling, 1 reply; 16+ messages in thread
From: Noam Postavsky @ 2018-11-04 12:27 UTC (permalink / raw)
  To: immerrr again; +Cc: 33255

immerrr again <immerrr@gmail.com> writes:

> I have recently tried to play with a clean .emacs.d directory by setting
> HOME=clean/home/dir, but faced an error coming from deep inside
> package-install. The error boils down to the fact that
> `expand-file-name' for some reason doubles the `default-directory'
> prefix when it is relative:
[...]
> I couldn't find any requirement for DEFAULT-DIRECTORY parameter or
> `default-directory' variable to be absolute, so I would assume this is a
> bug in expand-file-name.

(elisp) File Name Expansion says:

 -- Variable: default-directory
     The value of this buffer-local variable is the default directory
     for the current buffer.  It should be an absolute directory name;
     it may start with `~'.[...]

And expand-file-name is supposed to expand file names relative to
default-directory, so if default-directory is itself relative I don't
see how it could possibly work.  So we should just mention the
absoluteness requirement in the docstring as well.

I think it would make sense for Emacs to expand HOME against the current
directory when it starts up, if HOME is relative.  That would let your
motivating example succeed without error.






^ permalink raw reply	[flat|nested] 16+ messages in thread

* bug#33255: 27.0.50; expand-file-name: default directory expanded twice if relative
  2018-11-04 12:27 ` Noam Postavsky
@ 2018-11-05  0:58   ` Glenn Morris
  0 siblings, 0 replies; 16+ messages in thread
From: Glenn Morris @ 2018-11-05  0:58 UTC (permalink / raw)
  To: Noam Postavsky; +Cc: 33255, immerrr again

Noam Postavsky wrote:

> immerrr again <immerrr@gmail.com> writes:
>
>> I have recently tried to play with a clean .emacs.d directory by setting
>> HOME=clean/home/dir,

I think a non-absolute HOME is a user error.
Eg cd $HOME may then not be idempotent.
I would use $PWD/clean/home/dir.

> I think it would make sense for Emacs to expand HOME against the current
> directory when it starts up, if HOME is relative.  

I would rather see Emacs abort with an error.
Otherwise you have to eg fix up process-environment too, else child
processes may behave oddly.





^ permalink raw reply	[flat|nested] 16+ messages in thread

* bug#33255: 27.0.50; expand-file-name: default directory expanded twice if relative
  2018-11-04 10:54 bug#33255: 27.0.50; expand-file-name: default directory expanded twice if relative immerrr again
  2018-11-04 12:27 ` Noam Postavsky
@ 2018-11-13 18:26 ` Paul Eggert
  2018-11-13 20:12   ` Eli Zaretskii
  2018-11-14 18:10   ` Glenn Morris
  1 sibling, 2 replies; 16+ messages in thread
From: Paul Eggert @ 2018-11-13 18:26 UTC (permalink / raw)
  To: Glenn Morris; +Cc: 33255-done, immerrr again, Noam Postavsky

[-- Attachment #1: Type: text/plain, Size: 691 bytes --]

> I think a non-absolute HOME is a user error.
> Eg cd $HOME may then not be idempotent.
That's true. However, POSIX says that sh treats ~/foo like $HOME/foo 
even when HOME is not absolute, and it's better if Emacs is consistent 
with POSIX as much as possible within the Emacs constraint that 
expand-file-name must expand to an absolute file name. So I implemented 
something along the line of Noam's suggestion by installing the attached 
patch into master; this should fix the bug originally reported.

Unlike Noam's suggestion, this patch causes Emacs to look at the current 
value of HOME, not the value HOME had when Emacs started up, as that 
corresponds more closely to POSIX sh.


[-- Attachment #2: 0001-Act-like-POSIX-sh-if-HOME-is-relative.patch --]
[-- Type: text/x-patch, Size: 14613 bytes --]

From a2df04a05a68d5a4afa39cfe463f72108b31bc25 Mon Sep 17 00:00:00 2001
From: Paul Eggert <eggert@cs.ucla.edu>
Date: Tue, 13 Nov 2018 09:29:14 -0800
Subject: [PATCH] Act like POSIX sh if $HOME is relative

POSIX says sh ~/foo should act like $HOME/foo even if $HOME is
relative, so be consistent with that (Bug#33255).
* admin/merge-gnulib (GNULIB_MODULES): Add dosname.
* src/buffer.c (init_buffer): Use emacs_wd to get
initial working directory with slash appended if needed.
(default-directory): Say it must be absolute.
* src/emacs.c (emacs_wd): New global variable.
(init_cmdargs): Dir arg is now char const *.
(main): Set emacs_wd.
* src/emacs.c (main) [NS_IMPL_COCOA]:
* src/fileio.c (Fexpand_file_name):
Use get_homedir instead of egetenv ("HOME").
* src/fileio.c: Include dosname.h, for IS_ABSOLUTE_FILE_NAME.
(splice_dir_file, get_homedir): New functions.
* src/xrdb.c (gethomedir): Remove.  All callers changed
to use get_homedir and splice_dir_file.
* test/src/fileio-tests.el (fileio-tests--relative-HOME): New test.
---
 admin/merge-gnulib       |  2 +-
 src/buffer.c             | 28 ++++++------------
 src/emacs.c              | 20 ++++++++-----
 src/fileio.c             | 62 +++++++++++++++++++++++++++++++++++++---
 src/lisp.h               |  3 ++
 src/xrdb.c               | 54 +++++++---------------------------
 test/src/fileio-tests.el |  8 ++++++
 7 files changed, 103 insertions(+), 74 deletions(-)

diff --git a/admin/merge-gnulib b/admin/merge-gnulib
index 575e3fa74a..84dcb0b875 100755
--- a/admin/merge-gnulib
+++ b/admin/merge-gnulib
@@ -30,7 +30,7 @@ GNULIB_MODULES=
   careadlinkat close-stream
   count-leading-zeros count-one-bits count-trailing-zeros
   crypto/md5-buffer crypto/sha1-buffer crypto/sha256-buffer crypto/sha512-buffer
-  d-type diffseq dtoastr dtotimespec dup2
+  d-type diffseq dosname dtoastr dtotimespec dup2
   environ execinfo explicit_bzero faccessat
   fcntl fcntl-h fdatasync fdopendir
   filemode filevercmp flexmember fpieee fstatat fsusage fsync
diff --git a/src/buffer.c b/src/buffer.c
index ac2de7d19f..90ef886b22 100644
--- a/src/buffer.c
+++ b/src/buffer.c
@@ -5268,9 +5268,7 @@ init_buffer_once (void)
 void
 init_buffer (int initialized)
 {
-  char *pwd;
   Lisp_Object temp;
-  ptrdiff_t len;
 
 #ifdef USE_MMAP_FOR_BUFFERS
   if (initialized)
@@ -5324,7 +5322,7 @@ init_buffer (int initialized)
   if (NILP (BVAR (&buffer_defaults, enable_multibyte_characters)))
     Fset_buffer_multibyte (Qnil);
 
-  pwd = emacs_get_current_dir_name ();
+  char const *pwd = emacs_wd;
 
   if (!pwd)
     {
@@ -5336,22 +5334,16 @@ init_buffer (int initialized)
     {
       /* Maybe this should really use some standard subroutine
          whose definition is filename syntax dependent.  */
-      len = strlen (pwd);
-      if (!(IS_DIRECTORY_SEP (pwd[len - 1])))
-        {
-          /* Grow buffer to add directory separator and '\0'.  */
-          pwd = realloc (pwd, len + 2);
-          if (!pwd)
-            fatal ("get_current_dir_name: %s\n", strerror (errno));
-          pwd[len] = DIRECTORY_SEP;
-          pwd[len + 1] = '\0';
-          len++;
-        }
+      ptrdiff_t len = strlen (pwd);
+      bool add_slash = ! IS_DIRECTORY_SEP (pwd[len - 1]);
 
       /* At this moment, we still don't know how to decode the directory
          name.  So, we keep the bytes in unibyte form so that file I/O
          routines correctly get the original bytes.  */
-      bset_directory (current_buffer, make_unibyte_string (pwd, len));
+      Lisp_Object dirname = make_unibyte_string (pwd, len + add_slash);
+      if (add_slash)
+	SSET (dirname, len, DIRECTORY_SEP);
+      bset_directory (current_buffer, dirname);
 
       /* Add /: to the front of the name
          if it would otherwise be treated as magic.  */
@@ -5372,8 +5364,6 @@ init_buffer (int initialized)
 
   temp = get_minibuffer (0);
   bset_directory (XBUFFER (temp), BVAR (current_buffer, directory));
-
-  free (pwd);
 }
 
 /* Similar to defvar_lisp but define a variable whose value is the
@@ -5706,8 +5696,8 @@ visual lines rather than logical lines.  See the documentation of
   DEFVAR_PER_BUFFER ("default-directory", &BVAR (current_buffer, directory),
 		     Qstringp,
 		     doc: /* Name of default directory of current buffer.
-It should be a directory name (as opposed to a directory file-name).
-On GNU and Unix systems, directory names end in a slash `/'.
+It should be an absolute directory name; on GNU and Unix systems,
+these names start with `/' or `~' and end with `/'.
 To interactively change the default directory, use command `cd'. */);
 
   DEFVAR_PER_BUFFER ("auto-fill-function", &BVAR (current_buffer, auto_fill_function),
diff --git a/src/emacs.c b/src/emacs.c
index 512174d562..acb4959bfe 100644
--- a/src/emacs.c
+++ b/src/emacs.c
@@ -204,6 +204,9 @@ HANDLE w32_daemon_event;
 char **initial_argv;
 int initial_argc;
 
+/* The name of the working directory, or NULL if this info is unavailable.  */
+char const *emacs_wd;
+
 static void sort_args (int argc, char **argv);
 static void syms_of_emacs (void);
 
@@ -406,7 +409,7 @@ terminate_due_to_signal (int sig, int backtrace_limit)
 /* Code for dealing with Lisp access to the Unix command line.  */
 
 static void
-init_cmdargs (int argc, char **argv, int skip_args, char *original_pwd)
+init_cmdargs (int argc, char **argv, int skip_args, char const *original_pwd)
 {
   int i;
   Lisp_Object name, dir, handler;
@@ -694,7 +697,7 @@ main (int argc, char **argv)
   char *ch_to_dir = 0;
 
   /* If we use --chdir, this records the original directory.  */
-  char *original_pwd = 0;
+  char const *original_pwd = 0;
 
   /* Record (approximately) where the stack begins.  */
   stack_bottom = (char *) &stack_bottom_variable;
@@ -794,6 +797,8 @@ main (int argc, char **argv)
       exit (0);
     }
 
+  emacs_wd = emacs_get_current_dir_name ();
+
   if (argmatch (argv, argc, "-chdir", "--chdir", 4, &ch_to_dir, &skip_args))
     {
 #ifdef WINDOWSNT
@@ -804,13 +809,14 @@ main (int argc, char **argv)
       filename_from_ansi (ch_to_dir, newdir);
       ch_to_dir = newdir;
 #endif
-      original_pwd = emacs_get_current_dir_name ();
       if (chdir (ch_to_dir) != 0)
         {
           fprintf (stderr, "%s: Can't chdir to %s: %s\n",
                    argv[0], ch_to_dir, strerror (errno));
           exit (1);
         }
+      original_pwd = emacs_wd;
+      emacs_wd = emacs_get_current_dir_name ();
     }
 
 #if defined (HAVE_SETRLIMIT) && defined (RLIMIT_STACK) && !defined (CYGWIN)
@@ -1289,21 +1295,21 @@ Using an Emacs configured with --with-x-toolkit=lucid does not have this problem
     {
 #ifdef NS_IMPL_COCOA
       /* Started from GUI? */
-      /* FIXME: Do the right thing if getenv returns NULL, or if
+      /* FIXME: Do the right thing if get_homedir returns "", or if
          chdir fails.  */
       if (! inhibit_window_system && ! isatty (STDIN_FILENO) && ! ch_to_dir)
-        chdir (getenv ("HOME"));
+        chdir (get_homedir ());
       if (skip_args < argc)
         {
           if (!strncmp (argv[skip_args], "-psn", 4))
             {
               skip_args += 1;
-              if (! ch_to_dir) chdir (getenv ("HOME"));
+              if (! ch_to_dir) chdir (get_homedir ());
             }
           else if (skip_args+1 < argc && !strncmp (argv[skip_args+1], "-psn", 4))
             {
               skip_args += 2;
-              if (! ch_to_dir) chdir (getenv ("HOME"));
+              if (! ch_to_dir) chdir (get_homedir ());
             }
         }
 #endif  /* COCOA */
diff --git a/src/fileio.c b/src/fileio.c
index 7fb865809f..e178c39fc1 100644
--- a/src/fileio.c
+++ b/src/fileio.c
@@ -96,6 +96,7 @@ along with GNU Emacs.  If not, see <https://www.gnu.org/licenses/>.  */
 #include <acl.h>
 #include <allocator.h>
 #include <careadlinkat.h>
+#include <dosname.h>
 #include <fsusage.h>
 #include <stat-time.h>
 #include <tempname.h>
@@ -1093,8 +1094,7 @@ the root directory.  */)
 	{
 	  Lisp_Object tem;
 
-	  if (!(newdir = egetenv ("HOME")))
-	    newdir = newdirlim = "";
+	  newdir = get_homedir ();
 	  nm++;
 #ifdef WINDOWSNT
 	  if (newdir[0])
@@ -1109,7 +1109,7 @@ the root directory.  */)
 #endif
 	    tem = build_string (newdir);
 	  newdirlim = newdir + SBYTES (tem);
-	  /* `egetenv' may return a unibyte string, which will bite us
+	  /* get_homedir may return a unibyte string, which will bite us
 	     if we expect the directory to be multibyte.  */
 	  if (multibyte && !STRING_MULTIBYTE (tem))
 	    {
@@ -1637,7 +1637,6 @@ See also the function `substitute-in-file-name'.")
 }
 #endif
 \f
-/* If /~ or // appears, discard everything through first slash.  */
 static bool
 file_name_absolute_p (const char *filename)
 {
@@ -1650,6 +1649,61 @@ file_name_absolute_p (const char *filename)
      );
 }
 
+/* Put into BUF the concatenation of DIR and FILE, with an intervening
+   directory separator if needed.  Return a pointer to the null byte
+   at the end of the concatenated string.  */
+char *
+splice_dir_file (char *buf, char const *dir, char const *file)
+{
+  char *e = stpcpy (buf, dir);
+  *e = DIRECTORY_SEP;
+  e += ! (buf < e && IS_DIRECTORY_SEP (e[-1]));
+  return stpcpy (e, file);
+}
+
+/* Get the home directory, an absolute file name.  Return the empty
+   string on failure.  The returned value does not survive garbage
+   collection, calls to this function, or calls to the getpwnam class
+   of functions.  */
+char const *
+get_homedir (void)
+{
+  char const *home = egetenv ("HOME");
+  if (!home)
+    {
+      static char const *userenv[] = {"LOGNAME", "USER"};
+      struct passwd *pw = NULL;
+      for (int i = 0; i < ARRAYELTS (userenv); i++)
+	{
+	  char *user = egetenv (userenv[i]);
+	  if (user)
+	    {
+	      pw = getpwnam (user);
+	      if (pw)
+		break;
+	    }
+	}
+      if (!pw)
+	pw = getpwuid (getuid ());
+      if (pw)
+	home = pw->pw_dir;
+      if (!home)
+	return "";
+    }
+  if (IS_ABSOLUTE_FILE_NAME (home))
+    return home;
+  if (!emacs_wd)
+    error ("$HOME is relative to unknown directory");
+  static char *ahome;
+  static ptrdiff_t ahomesize;
+  ptrdiff_t ahomelenbound = strlen (emacs_wd) + 1 + strlen (home) + 1;
+  if (ahomesize <= ahomelenbound)
+    ahome = xpalloc (ahome, &ahomesize, ahomelenbound + 1 - ahomesize, -1, 1);
+  splice_dir_file (ahome, emacs_wd, home);
+  return ahome;
+}
+
+/* If /~ or // appears, discard everything through first slash.  */
 static char *
 search_embedded_absfilename (char *nm, char *endp)
 {
diff --git a/src/lisp.h b/src/lisp.h
index f8ffb33a64..7e7dba631f 100644
--- a/src/lisp.h
+++ b/src/lisp.h
@@ -4061,6 +4061,8 @@ extern void syms_of_marker (void);
 
 /* Defined in fileio.c.  */
 
+extern char *splice_dir_file (char *, char const *, char const *);
+extern char const *get_homedir (void);
 extern Lisp_Object expand_and_dir_to_file (Lisp_Object);
 extern Lisp_Object write_region (Lisp_Object, Lisp_Object, Lisp_Object,
 				 Lisp_Object, Lisp_Object, Lisp_Object,
@@ -4185,6 +4187,7 @@ extern void syms_of_frame (void);
 /* Defined in emacs.c.  */
 extern char **initial_argv;
 extern int initial_argc;
+extern char const *emacs_wd;
 #if defined (HAVE_X_WINDOWS) || defined (HAVE_NS)
 extern bool display_arg;
 #endif
diff --git a/src/xrdb.c b/src/xrdb.c
index 4abf1ad84e..87c2faf659 100644
--- a/src/xrdb.c
+++ b/src/xrdb.c
@@ -202,35 +202,6 @@ magic_db (const char *string, ptrdiff_t string_len, const char *class,
 }
 
 
-static char *
-gethomedir (void)
-{
-  struct passwd *pw;
-  char *ptr;
-  char *copy;
-
-  if ((ptr = getenv ("HOME")) == NULL)
-    {
-      if ((ptr = getenv ("LOGNAME")) != NULL
-	  || (ptr = getenv ("USER")) != NULL)
-	pw = getpwnam (ptr);
-      else
-	pw = getpwuid (getuid ());
-
-      if (pw)
-	ptr = pw->pw_dir;
-    }
-
-  if (ptr == NULL)
-    return xstrdup ("/");
-
-  ptrdiff_t len = strlen (ptr);
-  copy = xmalloc (len + 2);
-  strcpy (copy + len, "/");
-  return memcpy (copy, ptr, len);
-}
-
-
 /* Find the first element of SEARCH_PATH which exists and is readable,
    after expanding the %-escapes.  Return 0 if we didn't find any, and
    the path name of the one we found otherwise.  */
@@ -316,12 +287,11 @@ get_user_app (const char *class)
   if (! db)
     {
       /* Check in the home directory.  This is a bit of a hack; let's
-	 hope one's home directory doesn't contain any %-escapes.  */
-      char *home = gethomedir ();
+	 hope one's home directory doesn't contain ':' or '%'.  */
+      char const *home = get_homedir ();
       db = search_magic_path (home, class, "%L/%N");
       if (! db)
 	db = search_magic_path (home, class, "%N");
-      xfree (home);
     }
 
   return db;
@@ -346,10 +316,9 @@ get_user_db (Display *display)
   else
     {
       /* Use ~/.Xdefaults.  */
-      char *home = gethomedir ();
-      ptrdiff_t homelen = strlen (home);
-      char *filename = xrealloc (home, homelen + sizeof xdefaults);
-      strcpy (filename + homelen, xdefaults);
+      char const *home = get_homedir ();
+      char *filename = xmalloc (strlen (home) + 1 + sizeof xdefaults);
+      splice_dir_file (filename, home, xdefaults);
       db = XrmGetFileDatabase (filename);
       xfree (filename);
     }
@@ -380,13 +349,12 @@ get_environ_db (void)
       if (STRINGP (system_name))
 	{
 	  /* Use ~/.Xdefaults-HOSTNAME.  */
-	  char *home = gethomedir ();
-	  ptrdiff_t homelen = strlen (home);
-	  ptrdiff_t filenamesize = (homelen + sizeof xdefaults
-				    + 1 + SBYTES (system_name));
-	  p = filename = xrealloc (home, filenamesize);
-	  lispstpcpy (stpcpy (stpcpy (filename + homelen, xdefaults), "-"),
-		      system_name);
+	  char const *home = get_homedir ();
+	  p = filename = xmalloc (strlen (home) + 1 + sizeof xdefaults
+				  + 1 + SBYTES (system_name));
+	  char *e = splice_dir_file (p, home, xdefaults);
+	  *e++ = '/';
+	  lispstpcpy (e, system_name);
 	}
     }
 
diff --git a/test/src/fileio-tests.el b/test/src/fileio-tests.el
index 5d12685fa1..b7b78bbda0 100644
--- a/test/src/fileio-tests.el
+++ b/test/src/fileio-tests.el
@@ -95,3 +95,11 @@ fileio-tests--symlink-failure
   (should (equal (file-name-as-directory "d:/abc/") "d:/abc/"))
   (should (equal (file-name-as-directory "D:\\abc/") "d:/abc/"))
   (should (equal (file-name-as-directory "D:/abc//") "d:/abc//")))
+
+(ert-deftest fileio-tests--relative-HOME ()
+  "Test that expand-file-name works even when HOME is relative."
+  (let ((old-home (getenv "HOME")))
+    (setenv "HOME" "a/b/c")
+    (should (equal (expand-file-name "~/foo")
+                   (expand-file-name "a/b/c/foo")))
+    (setenv "HOME" old-home)))
-- 
2.19.1


^ permalink raw reply related	[flat|nested] 16+ messages in thread

* bug#33255: 27.0.50; expand-file-name: default directory expanded twice if relative
  2018-11-13 18:26 ` Paul Eggert
@ 2018-11-13 20:12   ` Eli Zaretskii
  2018-11-14 18:10   ` Glenn Morris
  1 sibling, 0 replies; 16+ messages in thread
From: Eli Zaretskii @ 2018-11-13 20:12 UTC (permalink / raw)
  To: Paul Eggert; +Cc: 33255, immerrr

> From: Paul Eggert <eggert@cs.ucla.edu>
> Date: Tue, 13 Nov 2018 10:26:43 -0800
> Cc: 33255-done@debbugs.gnu.org, immerrr again <immerrr@gmail.com>,
> 	Noam Postavsky <npostavs@gmail.com>
> 
> > I think a non-absolute HOME is a user error.
> > Eg cd $HOME may then not be idempotent.
> That's true. However, POSIX says that sh treats ~/foo like $HOME/foo 
> even when HOME is not absolute, and it's better if Emacs is consistent 
> with POSIX as much as possible within the Emacs constraint that 
> expand-file-name must expand to an absolute file name. So I implemented 
> something along the line of Noam's suggestion by installing the attached 
> patch into master; this should fix the bug originally reported.
> 
> Unlike Noam's suggestion, this patch causes Emacs to look at the current 
> value of HOME, not the value HOME had when Emacs started up, as that 
> corresponds more closely to POSIX sh.

Thanks.  I think this needs to be called out in NEWS as an
incompatible change, and probably also documented in the manuals.





^ permalink raw reply	[flat|nested] 16+ messages in thread

* bug#33255: 27.0.50; expand-file-name: default directory expanded twice if relative
  2018-11-13 18:26 ` Paul Eggert
  2018-11-13 20:12   ` Eli Zaretskii
@ 2018-11-14 18:10   ` Glenn Morris
  2018-11-14 18:17     ` Paul Eggert
  1 sibling, 1 reply; 16+ messages in thread
From: Glenn Morris @ 2018-11-14 18:10 UTC (permalink / raw)
  To: Paul Eggert; +Cc: 33255, immerrr again, Noam Postavsky

Paul Eggert wrote:

> However, POSIX says that sh treats ~/foo like $HOME/foo even when HOME
> is not absolute [...]

Can you point me to the citation for that please (I'm not doubting you,
just interested to read the wording, which I could not find).

I'm a bit disappointed to see that a relative HOME is now documented
in the Emacs manuals, since IMO it lends legitimacy to a questionable
usage. I don't see it mentioned in eg the bash manual.
I also don't see why it is in NEWS since it is described as
a bug fix, and those aren't normally mentioned.





^ permalink raw reply	[flat|nested] 16+ messages in thread

* bug#33255: 27.0.50; expand-file-name: default directory expanded twice if relative
  2018-11-14 18:10   ` Glenn Morris
@ 2018-11-14 18:17     ` Paul Eggert
  2018-11-14 19:52       ` Eli Zaretskii
  2018-11-20 19:08       ` Glenn Morris
  0 siblings, 2 replies; 16+ messages in thread
From: Paul Eggert @ 2018-11-14 18:17 UTC (permalink / raw)
  To: Glenn Morris; +Cc: 33255, immerrr again, Noam Postavsky

On 11/14/18 10:10 AM, Glenn Morris wrote:
> Can you point me to the citation for that please

http://pubs.opengroup.org/onlinepubs/9699919799/utilities/V3_chap02.html#tag_18_06_01

> I'm a bit disappointed to see that a relative HOME is now documented
> in the Emacs manuals, since IMO it lends legitimacy to a questionable
> usage. I don't see it mentioned in eg the bash manual.
> I also don't see why it is in NEWS since it is described as
> a bug fix, and those aren't normally mentioned.

My initial reaction was the same as yours, which is why my original fix 
didn't document the change in the manual or in NEWS. Eli felt otherwise, 
though.






^ permalink raw reply	[flat|nested] 16+ messages in thread

* bug#33255: 27.0.50; expand-file-name: default directory expanded twice if relative
  2018-11-14 18:17     ` Paul Eggert
@ 2018-11-14 19:52       ` Eli Zaretskii
  2018-11-20 19:11         ` Glenn Morris
  2018-11-20 19:08       ` Glenn Morris
  1 sibling, 1 reply; 16+ messages in thread
From: Eli Zaretskii @ 2018-11-14 19:52 UTC (permalink / raw)
  To: Paul Eggert; +Cc: immerrr, npostavs, 33255

> From: Paul Eggert <eggert@cs.ucla.edu>
> Date: Wed, 14 Nov 2018 10:17:59 -0800
> Cc: 33255@debbugs.gnu.org, immerrr again <immerrr@gmail.com>,
> 	Noam Postavsky <npostavs@gmail.com>
> 
> > I'm a bit disappointed to see that a relative HOME is now documented
> > in the Emacs manuals, since IMO it lends legitimacy to a questionable
> > usage. I don't see it mentioned in eg the bash manual.
> > I also don't see why it is in NEWS since it is described as
> > a bug fix, and those aren't normally mentioned.
> 
> My initial reaction was the same as yours, which is why my original fix 
> didn't document the change in the manual or in NEWS. Eli felt otherwise, 
> though.

I agree that we should discourage relative $HOME, so I added text to
the documentation to that effect.





^ permalink raw reply	[flat|nested] 16+ messages in thread

* bug#33255: 27.0.50; expand-file-name: default directory expanded twice if relative
  2018-11-14 18:17     ` Paul Eggert
  2018-11-14 19:52       ` Eli Zaretskii
@ 2018-11-20 19:08       ` Glenn Morris
  2018-11-20 20:44         ` Paul Eggert
  1 sibling, 1 reply; 16+ messages in thread
From: Glenn Morris @ 2018-11-20 19:08 UTC (permalink / raw)
  To: Paul Eggert; +Cc: 33255, immerrr again, Noam Postavsky

Paul Eggert wrote:

> On 11/14/18 10:10 AM, Glenn Morris wrote:
>> Can you point me to the citation for that please
>
> http://pubs.opengroup.org/onlinepubs/9699919799/utilities/V3_chap02.html#tag_18_06_01

I had looked at that page before. Looking again, I still don't see
anything as clear-cut as "POSIX says that sh treats ~/foo like $HOME/foo
even when HOME is not absolute".

There are no matches for "relative" on that page.
There are three for "absolute". One is for ENV, and says that the
results are unspecified if it is not absolute. The other two are for
PWD, and say that it must be absolute.

What am I missing?
Just that the section on ~ expansion says nothing one way or the other?
I was really looking for a reference about HOME.






^ permalink raw reply	[flat|nested] 16+ messages in thread

* bug#33255: 27.0.50; expand-file-name: default directory expanded twice if relative
  2018-11-14 19:52       ` Eli Zaretskii
@ 2018-11-20 19:11         ` Glenn Morris
  2018-11-20 19:26           ` Eli Zaretskii
  0 siblings, 1 reply; 16+ messages in thread
From: Glenn Morris @ 2018-11-20 19:11 UTC (permalink / raw)
  To: Eli Zaretskii; +Cc: 33255, Paul Eggert, immerrr, npostavs

Eli Zaretskii wrote:

> I agree that we should discourage relative $HOME, so I added text to
> the documentation to that effect.

I really think it would be better not to mention it at all.
I have never heard of it in many years of using Unix (I know this is an
awful kind of statement), and see no need for it. Again, the GNU Bash
manual says nothing about this.





^ permalink raw reply	[flat|nested] 16+ messages in thread

* bug#33255: 27.0.50; expand-file-name: default directory expanded twice if relative
  2018-11-20 19:11         ` Glenn Morris
@ 2018-11-20 19:26           ` Eli Zaretskii
  0 siblings, 0 replies; 16+ messages in thread
From: Eli Zaretskii @ 2018-11-20 19:26 UTC (permalink / raw)
  To: Glenn Morris; +Cc: 33255, eggert, immerrr, npostavs

> From: Glenn Morris <rgm@gnu.org>
> Cc: Paul Eggert <eggert@cs.ucla.edu>,  33255@debbugs.gnu.org,  immerrr@gmail.com,  npostavs@gmail.com
> Date: Tue, 20 Nov 2018 14:11:00 -0500
> 
> Eli Zaretskii wrote:
> 
> > I agree that we should discourage relative $HOME, so I added text to
> > the documentation to that effect.
> 
> I really think it would be better not to mention it at all.

Sorry, I disagree.  We should not hide from the users what Emacs does
in such corner cases.  Hiding won't work anyway, because someone
determined enough will find out by reading the code.  We just punish
those who aren't determined enough.





^ permalink raw reply	[flat|nested] 16+ messages in thread

* bug#33255: 27.0.50; expand-file-name: default directory expanded twice if relative
  2018-11-20 19:08       ` Glenn Morris
@ 2018-11-20 20:44         ` Paul Eggert
  2018-11-22 18:25           ` Glenn Morris
  0 siblings, 1 reply; 16+ messages in thread
From: Paul Eggert @ 2018-11-20 20:44 UTC (permalink / raw)
  To: Glenn Morris; +Cc: 33255, immerrr again, Noam Postavsky

On 11/20/18 11:08 AM, Glenn Morris wrote:
>> http://pubs.opengroup.org/onlinepubs/9699919799/utilities/V3_chap02.html#tag_18_06_01
> I had looked at that page before. Looking again, I still don't see
> anything as clear-cut as "POSIX says that sh treats ~/foo like $HOME/foo
> even when HOME is not absolute".

It needs to be read in context. Section 2.6.1 is about tilde expansion 
in the shell, e.g., how to treat commands like this:

cd ~eggert/xxx
cd ~/yyy

2.6.1 says "If the login name is null (that is, the tilde-prefix 
contains only the tilde), the tilde-prefix is replaced by the value of 
the variable /HOME."/ This is talking about the second "echo" example 
which uses a null login name, and it means that the second example is 
treated like this:

cd "$HOME"/yyy/
/

(The quotes are because of the last sentence in that section of the 
spec.) This occurs regardless of whether $HOME starts with /"/".//
/






^ permalink raw reply	[flat|nested] 16+ messages in thread

* bug#33255: 27.0.50; expand-file-name: default directory expanded twice if relative
  2018-11-20 20:44         ` Paul Eggert
@ 2018-11-22 18:25           ` Glenn Morris
  2018-11-23 20:22             ` Paul Eggert
  0 siblings, 1 reply; 16+ messages in thread
From: Glenn Morris @ 2018-11-22 18:25 UTC (permalink / raw)
  To: Paul Eggert; +Cc: 33255, immerrr again, Noam Postavsky

Paul Eggert wrote:

> It needs to be read in context. Section 2.6.1 is about tilde expansion
> in the shell, e.g., how to treat commands like this:
>
> cd ~eggert/xxx
> cd ~/yyy
>
> 2.6.1 says "If the login name is null (that is, the tilde-prefix
> contains only the tilde), the tilde-prefix is replaced by the value of
> the variable /HOME."/ This is talking about the second "echo" example
> which uses a null login name, and it means that the second example is
> treated like this:
>
> cd "$HOME"/yyy/

This seems to be telling me what tilde expansion is.

I am at this point looking for any documentation (not even from POSIX,
any shell or frankly any Unix utility will do) that says "HOME need not
be absolute, if not, here's how that is handled". So far all I see from
POSIX is that it doesn't say anything about whether HOME is absolute or
not.






^ permalink raw reply	[flat|nested] 16+ messages in thread

* bug#33255: 27.0.50; expand-file-name: default directory expanded twice if relative
  2018-11-22 18:25           ` Glenn Morris
@ 2018-11-23 20:22             ` Paul Eggert
  2018-11-27  5:42               ` Glenn Morris
  0 siblings, 1 reply; 16+ messages in thread
From: Paul Eggert @ 2018-11-23 20:22 UTC (permalink / raw)
  To: Glenn Morris; +Cc: 33255, immerrr again, Noam Postavsky

Glenn Morris wrote:
> So far all I see from POSIX is that it doesn't say anything about whether HOME is absolute or not.

By not saying anything POSIX gives permission to the application to set HOME to 
a relative name. When POSIX intends the requirement that an environment 
variable's value must be absolute (e.g., PWD), it says so. When it doesn't 
intend such a requirement (e.g., HOME, PATH, SHELL, TMPDIR) it says nothing.





^ permalink raw reply	[flat|nested] 16+ messages in thread

* bug#33255: 27.0.50; expand-file-name: default directory expanded twice if relative
  2018-11-23 20:22             ` Paul Eggert
@ 2018-11-27  5:42               ` Glenn Morris
  2018-11-27 18:11                 ` Paul Eggert
  0 siblings, 1 reply; 16+ messages in thread
From: Glenn Morris @ 2018-11-27  5:42 UTC (permalink / raw)
  To: Paul Eggert; +Cc: 33255, immerrr again, Noam Postavsky

Paul Eggert wrote:

> By not saying anything POSIX gives permission to the application to
> set HOME to a relative name. When POSIX intends the requirement that
> an environment variable's value must be absolute (e.g., PWD), it says
> so. When it doesn't intend such a requirement (e.g., HOME, PATH,
> SHELL, TMPDIR) it says nothing.

So the justification for implementing this is indeed "it's not
explicitly forbidden".

Here's an example of how this can be confusing:

cd /tmp
mkdir foo
echo hi > foo/bar
HOME=foo emacs
C-x C-f ~/bar   ; works
M-: (shell-command "ls ~/bar")  ; fails

So, Emacs and external processes it spawns interpret ~ differently;
ie external processes are likely to fail in odd ways. All this would be
avoided if the user had just said HOME=$PWD/foo to start with.

>> I am at this point looking for any documentation (not even from POSIX,
>> any shell or frankly any Unix utility will do) that says "HOME need
>> not be absolute, if not, here's how that is handled".

AFAICS Emacs is the only thing documenting this scenario.





^ permalink raw reply	[flat|nested] 16+ messages in thread

* bug#33255: 27.0.50; expand-file-name: default directory expanded twice if relative
  2018-11-27  5:42               ` Glenn Morris
@ 2018-11-27 18:11                 ` Paul Eggert
  0 siblings, 0 replies; 16+ messages in thread
From: Paul Eggert @ 2018-11-27 18:11 UTC (permalink / raw)
  To: Glenn Morris; +Cc: 33255, immerrr again, Noam Postavsky

On 11/26/18 9:42 PM, Glenn Morris wrote:
> Here's an example of how this can be confusing:
>
> cd /tmp
> mkdir foo
> echo hi > foo/bar
> HOME=foo emacs
> C-x C-f ~/bar   ; works
> M-: (shell-command "ls ~/bar")  ; fails

Yes, and there's a similar confusion in ordinary POSIX shells:

$ cd /tmp
$ mkdir foo
$ echo hi > foo/bar
$ HOME=foo sh
$ cat ~/bar
hi
$ cd
$ cat ~/bar
cat: foo/bar: No such file or directory

The moral of this story in POSIX is "Don't set HOME to a relative file 
name, as it's trouble for any application that chdirs." In your example 
Emacs chdirs; in mine, sh chdirs.

The question is whether Emacs should attempt to insulate users from this 
trouble, presumably by replacing HOME with an absolute directory name 
when HOME's value is relative (or is absent or empty, for that matter). 
The POSIX shell doesn't do that, which is an argument for Emacs not 
doing it either.






^ permalink raw reply	[flat|nested] 16+ messages in thread

end of thread, other threads:[~2018-11-27 18:11 UTC | newest]

Thread overview: 16+ messages (download: mbox.gz follow: Atom feed
-- links below jump to the message on this page --
2018-11-04 10:54 bug#33255: 27.0.50; expand-file-name: default directory expanded twice if relative immerrr again
2018-11-04 12:27 ` Noam Postavsky
2018-11-05  0:58   ` Glenn Morris
2018-11-13 18:26 ` Paul Eggert
2018-11-13 20:12   ` Eli Zaretskii
2018-11-14 18:10   ` Glenn Morris
2018-11-14 18:17     ` Paul Eggert
2018-11-14 19:52       ` Eli Zaretskii
2018-11-20 19:11         ` Glenn Morris
2018-11-20 19:26           ` Eli Zaretskii
2018-11-20 19:08       ` Glenn Morris
2018-11-20 20:44         ` Paul Eggert
2018-11-22 18:25           ` Glenn Morris
2018-11-23 20:22             ` Paul Eggert
2018-11-27  5:42               ` Glenn Morris
2018-11-27 18:11                 ` Paul Eggert

Code repositories for project(s) associated with this public inbox

	https://git.savannah.gnu.org/cgit/emacs.git

This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for read-only IMAP folder(s) and NNTP newsgroup(s).