#include "mac.h"
#include <errno.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <limits.h>
#include "defs.h"
#define DOT '.'
#define NULLCHAR '\0'
#define SLASH '/'
#define PARTLY 2
static void rmslash(unsigned char *string);
#ifdef __STDC__
extern const char longpwd[];
#else
extern char longpwd[];
#endif
extern char *getcwd();
unsigned char cwdname[PATH_MAX+1];
static int didpwd = FALSE;
void
cwd(unsigned char *dir)
{
unsigned char *pcwd;
unsigned char *pdir;
rmslash(dir);
pdir = dir;
if (*dir == SLASH)
pdir++;
while (*pdir) {
if ((*pdir == DOT) && (*(pdir+1) == SLASH)) {
movstr(pdir+2, pdir);
continue;
}
pdir++;
while ((*pdir) && (*pdir != SLASH))
pdir++;
if (*pdir)
pdir++;
}
if (*(--pdir) == DOT && pdir > dir && *(--pdir) == SLASH) {
if (pdir > dir) {
*pdir = NULLCHAR;
} else {
*(pdir+1) = NULLCHAR;
}
}
rmslash(dir);
if (*dir == DOT && *(dir+1) == NULLCHAR) {
return;
}
if (*dir == SLASH) {
pcwd = cwdname;
*pcwd++ = *dir++;
didpwd = PARTLY;
}
else
{
if (didpwd == FALSE)
return;
didpwd = PARTLY;
pcwd = cwdname + length(cwdname) - 1;
if (pcwd != cwdname+1)
*pcwd++ = SLASH;
}
while (*dir) {
if (*dir == DOT &&
*(dir+1) == DOT &&
(*(dir+2) == SLASH || *(dir+2) == NULLCHAR)) {
if (pcwd > cwdname+2)
--pcwd;
while (*(--pcwd) != SLASH)
;
pcwd++;
dir += 2;
if (*dir == SLASH) {
dir++;
}
continue;
}
if (pcwd >= &cwdname[PATH_MAX+1]) {
didpwd = FALSE;
return;
}
*pcwd++ = *dir++;
while ((*dir) && (*dir != SLASH)) {
if (pcwd >= &cwdname[PATH_MAX+1]) {
didpwd = FALSE;
return;
}
*pcwd++ = *dir++;
}
if (*dir) {
if (pcwd >= &cwdname[PATH_MAX+1]) {
didpwd = FALSE;
return;
}
*pcwd++ = *dir++;
}
}
if (pcwd >= &cwdname[PATH_MAX+1]) {
didpwd = FALSE;
return;
}
*pcwd = NULLCHAR;
--pcwd;
if (pcwd > cwdname && *pcwd == SLASH) {
*pcwd = NULLCHAR;
}
}
void
cwd2()
{
struct stat stat1, stat2;
unsigned char *pcwd;
if (didpwd == FALSE)
return;
pcwd = cwdname + 1;
if (didpwd == PARTLY) {
while (*pcwd) {
char c;
do {
c = *pcwd++;
} while (c != SLASH && c != NULLCHAR);
*--pcwd = NULLCHAR;
if (lstat((char *)cwdname, &stat1) == -1 ||
(stat1.st_mode & S_IFMT) == S_IFLNK) {
didpwd = FALSE;
*pcwd = c;
return;
}
*pcwd = c;
if (c)
pcwd++;
}
didpwd = TRUE;
} else
if (stat((char *)cwdname, &stat1) == -1) {
didpwd = FALSE;
return;
}
if (stat(".", &stat2) == -1 ||
stat1.st_dev != stat2.st_dev ||
stat1.st_ino != stat2.st_ino)
didpwd = FALSE;
}
unsigned char *
cwdget()
{
cwd2();
if (didpwd == FALSE) {
if (getcwd((char *)cwdname, PATH_MAX+1) == NULL)
*cwdname = NULLCHAR;
didpwd = TRUE;
}
return (cwdname);
}
void
cwdprint(void)
{
unsigned char *cp;
cwd2();
if (didpwd == FALSE) {
if (getcwd((char *)cwdname, PATH_MAX+1) == NULL) {
if (errno && errno != ERANGE)
error(badpwd);
else
error(longpwd);
}
didpwd = TRUE;
}
for (cp = cwdname; *cp; cp++) {
prc_buff(*cp);
}
prc_buff(NL);
}
static void
rmslash(unsigned char *string)
{
unsigned char *pstring;
pstring = string;
while (*pstring) {
if (*pstring == SLASH && *(pstring+1) == SLASH) {
movstr(pstring+1, pstring);
continue;
}
pstring++;
}
--pstring;
if (pstring > string && *pstring == SLASH) {
*pstring = NULLCHAR;
}
}