blob: ef01718bd11b6769e81bcdcf2f0e54563f96d10d [file] [log] [blame]
Jari Aaltobb706242000-03-17 21:46:59 +00001/* makepath.c - glue PATH and DIR together into a full pathname. */
2
3/* Copyright (C) 1987, 1989, 1991 Free Software Foundation, Inc.
4
5 This file is part of GNU Bash, the Bourne Again SHell.
6
Jari Aalto31859422009-01-12 13:36:28 +00007 Bash is free software: you can redistribute it and/or modify
8 it under the terms of the GNU General Public License as published by
9 the Free Software Foundation, either version 3 of the License, or
10 (at your option) any later version.
Jari Aaltobb706242000-03-17 21:46:59 +000011
Jari Aalto31859422009-01-12 13:36:28 +000012 Bash is distributed in the hope that it will be useful,
13 but WITHOUT ANY WARRANTY; without even the implied warranty of
14 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 GNU General Public License for more details.
Jari Aaltobb706242000-03-17 21:46:59 +000016
Jari Aalto31859422009-01-12 13:36:28 +000017 You should have received a copy of the GNU General Public License
18 along with Bash. If not, see <http://www.gnu.org/licenses/>.
19*/
Jari Aaltobb706242000-03-17 21:46:59 +000020
21#include <config.h>
22
23#if defined (HAVE_UNISTD_H)
24# ifdef _MINIX
25# include <sys/types.h>
26# endif
27# include <unistd.h>
28#endif
29
Jari Aaltof73dda02001-11-13 17:56:06 +000030#include <bashansi.h>
Jari Aaltobb706242000-03-17 21:46:59 +000031#include "shell.h"
32
33#include <tilde/tilde.h>
34
35#ifndef NULL
36# define NULL 0
37#endif
38
39/* MAKE SURE THESE AGREE WITH ../../externs.h. */
40
41#ifndef MP_DOTILDE
42# define MP_DOTILDE 0x01
43# define MP_DOCWD 0x02
44# define MP_RMDOT 0x04
Jari Aalto31859422009-01-12 13:36:28 +000045# define MP_IGNDOT 0x08
Jari Aaltobb706242000-03-17 21:46:59 +000046#endif
47
48extern char *get_working_directory __P((char *));
49
Jari Aalto31859422009-01-12 13:36:28 +000050static char *nullpath = "";
51
Jari Aaltobb706242000-03-17 21:46:59 +000052/* Take PATH, an element from, e.g., $CDPATH, and DIR, a directory name,
53 and paste them together into PATH/DIR. Tilde expansion is performed on
54 PATH if (flags & MP_DOTILDE) is non-zero. If PATH is NULL or the empty
55 string, it is converted to the current directory. A full pathname is
56 used if (flags & MP_DOCWD) is non-zero, otherwise `./' is used. If
57 (flags & MP_RMDOT) is non-zero, any `./' is removed from the beginning
Jari Aalto31859422009-01-12 13:36:28 +000058 of DIR. If (flags & MP_IGNDOT) is non-zero, a PATH that is "." or "./"
59 is ignored. */
Jari Aaltobb706242000-03-17 21:46:59 +000060
61#define MAKEDOT() \
62 do { \
Jari Aaltof73dda02001-11-13 17:56:06 +000063 xpath = (char *)xmalloc (2); \
Jari Aaltobb706242000-03-17 21:46:59 +000064 xpath[0] = '.'; \
65 xpath[1] = '\0'; \
66 pathlen = 1; \
67 } while (0)
68
69char *
70sh_makepath (path, dir, flags)
Jari Aaltof73dda02001-11-13 17:56:06 +000071 const char *path, *dir;
Jari Aaltobb706242000-03-17 21:46:59 +000072 int flags;
73{
74 int dirlen, pathlen;
Jari Aaltof73dda02001-11-13 17:56:06 +000075 char *ret, *xpath, *xdir, *r, *s;
Jari Aaltobb706242000-03-17 21:46:59 +000076
77 if (path == 0 || *path == '\0')
78 {
79 if (flags & MP_DOCWD)
80 {
81 xpath = get_working_directory ("sh_makepath");
82 if (xpath == 0)
83 {
84 ret = get_string_value ("PWD");
85 if (ret)
86 xpath = savestring (ret);
87 }
88 if (xpath == 0)
89 MAKEDOT();
90 else
91 pathlen = strlen (xpath);
92 }
93 else
94 MAKEDOT();
95 }
Jari Aalto31859422009-01-12 13:36:28 +000096 else if ((flags & MP_IGNDOT) && path[0] == '.' && (path[1] == '\0' ||
Chet Rameyac50fba2014-02-26 09:36:43 -050097 (path[1] == '/' && path[2] == '\0')))
Jari Aalto31859422009-01-12 13:36:28 +000098 {
99 xpath = nullpath;
100 pathlen = 0;
101 }
Jari Aaltobb706242000-03-17 21:46:59 +0000102 else
103 {
Jari Aalto7117c2d2002-07-17 14:10:11 +0000104 xpath = ((flags & MP_DOTILDE) && *path == '~') ? bash_tilde_expand (path, 0) : (char *)path;
Jari Aaltobb706242000-03-17 21:46:59 +0000105 pathlen = strlen (xpath);
106 }
107
Jari Aaltof73dda02001-11-13 17:56:06 +0000108 xdir = (char *)dir;
109 dirlen = strlen (xdir);
Jari Aaltobb706242000-03-17 21:46:59 +0000110 if ((flags & MP_RMDOT) && dir[0] == '.' && dir[1] == '/')
111 {
Jari Aaltof73dda02001-11-13 17:56:06 +0000112 xdir += 2;
Jari Aaltobb706242000-03-17 21:46:59 +0000113 dirlen -= 2;
114 }
115
Jari Aaltof73dda02001-11-13 17:56:06 +0000116 r = ret = (char *)xmalloc (2 + dirlen + pathlen);
Jari Aalto28ef6c32001-04-06 19:14:31 +0000117 s = xpath;
118 while (*s)
119 *r++ = *s++;
Jari Aalto31859422009-01-12 13:36:28 +0000120 if (s > xpath && s[-1] != '/')
Jari Aalto28ef6c32001-04-06 19:14:31 +0000121 *r++ = '/';
Jari Aaltof73dda02001-11-13 17:56:06 +0000122 s = xdir;
Jari Aalto28ef6c32001-04-06 19:14:31 +0000123 while (*r++ = *s++)
124 ;
Jari Aalto31859422009-01-12 13:36:28 +0000125 if (xpath != path && xpath != nullpath)
Jari Aaltobb706242000-03-17 21:46:59 +0000126 free (xpath);
127 return (ret);
128}