2 * CDE - Common Desktop Environment
4 * Copyright (c) 1993-2012, The Open Group. All rights reserved.
6 * These libraries and programs are free software; you can
7 * redistribute them and/or modify them under the terms of the GNU
8 * Lesser General Public License as published by the Free Software
9 * Foundation; either version 2 of the License, or (at your option)
12 * These libraries and programs are distributed in the hope that
13 * they will be useful, but WITHOUT ANY WARRANTY; without even the
14 * implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR
15 * PURPOSE. See the GNU Lesser General Public License for more
18 * You should have received a copy of the GNU Lesser General Public
19 * License along with these librararies and programs; if not, write
20 * to the Free Software Foundation, Inc., 51 Franklin Street, Fifth
21 * Floor, Boston, MA 02110-1301 USA
23 /* $XConsortium: getpreroot.c /main/3 1995/11/01 18:21:07 rswiston $ */
24 /***************************************************************
26 * AT&T - PROPRIETARY *
28 * THIS IS PROPRIETARY SOURCE CODE LICENSED BY *
31 * Copyright (c) 1995 AT&T Corp. *
32 * All Rights Reserved *
34 * This software is licensed by AT&T Corp. *
35 * under the terms and conditions of the license in *
36 * http://www.research.att.com/orgs/ssr/book/reuse *
38 * This software was created by the *
39 * Software Engineering Research Department *
40 * AT&T Bell Laboratories *
42 * For further information contact *
43 * gsf@research.att.com *
45 ***************************************************************/
47 /* : : generated by proto : : */
49 #if !defined(__PROTO__)
50 #if defined(__STDC__) || defined(__cplusplus) || defined(_proto) || defined(c_plusplus)
51 #if defined(__cplusplus)
52 #define __MANGLE__ "C"
57 #define __PROTO__(x) x
59 #define __PARAM__(n,o) n
60 #if !defined(__STDC__) && !defined(__cplusplus)
61 #if !defined(c_plusplus)
72 #define __PROTO__(x) ()
73 #define __OTORP__(x) x
74 #define __PARAM__(n,o) o
82 #if defined(__cplusplus) || defined(c_plusplus)
83 #define __VARARG__ ...
87 #if defined(__STDARG__)
88 #define __VA_START__(p,a) va_start(p,a)
90 #define __VA_START__(p,a) va_start(p)
107 #define ERROR(e) {errno=e;goto error;}
110 getpreroot __PARAM__((char* path, const char* cmd), (path, cmd)) __OTORP__(char* path; const char* cmd;){
116 if (!path) path = buf;
119 sfsprintf(buf, sizeof(buf), "set x `%s= %s - </dev/null 2>&1`\nwhile :\ndo\nshift\ncase $# in\n[012]) break ;;\nesac\ncase \"$1 $2\" in\n\"+ %s\") echo $3; exit ;;\nesac\ndone\necho\n", PR_SILENT, cmd, PR_COMMAND);
120 if (!(fp = popen(buf, "rug"))) return(0);
121 for (p = path; (c = getc(fp)) != EOF && c != '\n'; *p++ = c);
124 if (path == p) return(0);
125 return(path == buf ? strdup(path) : path);
134 struct dirent* entry;
145 if ((ruid = getuid()) != (euid = geteuid())) setuid(ruid);
146 if (stat(PR_REAL, cur) || stat("/", par) || cur->st_dev == par->st_dev && cur->st_ino == par->st_ino) ERROR(ENOTDIR);
149 * like getcwd() but starting at the preroot
154 p = path + PATH_MAX - 1;
161 if ((d - dots) > (PATH_MAX - 4)) ERROR(ERANGE);
165 if (!(dirp = opendir(dots))) ERROR(errno);
166 #if !_dir_ok || _mem_dd_fd_DIR
167 if (fstat(dirp->dd_fd, par)) ERROR(errno);
169 if (stat(dots, par)) ERROR(errno);
172 if (par->st_dev == cur->st_dev)
174 if (par->st_ino == cur->st_ino)
178 if (ruid != euid) setuid(euid);
179 if (path == buf) return(strdup(p));
187 #if _mem_d_fileno_dirent || _mem_d_ino_dirent
188 #if !_mem_d_fileno_dirent
189 #define d_fileno d_ino
191 while (entry = readdir(dirp))
192 if (entry->d_fileno == cur->st_ino)
194 #if _mem_d_namlen_dirent
195 namlen = entry->d_namlen;
197 namlen = strlen(entry->d_name);
203 * this fallthrough handles logical naming
211 if (!(entry = readdir(dirp))) ERROR(ENOENT);
212 #if _mem_d_namlen_dirent
213 namlen = entry->d_namlen;
215 namlen = strlen(entry->d_name);
217 if ((d - dots) > (PATH_MAX - 1 - namlen)) ERROR(ERANGE);
218 memcpy(d, entry->d_name, namlen + 1);
219 if (stat(dots, &tstst)) ERROR(errno);
220 } while (tstst.st_ino != cur->st_ino || tstst.st_dev != cur->st_dev);
223 if ((p -= namlen) <= (path + 1)) ERROR(ERANGE);
224 memcpy(p, entry->d_name, namlen);
229 if (dirp) closedir(dirp);
230 if (ruid != euid) setuid(euid);