Dir* dirstat(char *file) { struct stat lst; struct stat st; int nstr; Dir *d; char *str; #ifdef _WIN32 if(stat(file, &st) < 0) return nil; lst = st; #else if(lstat(file, &lst) < 0) return nil; st = lst; if((lst.st_mode&S_IFMT) == S_IFLNK) stat(file, &st); #endif nstr = _p9dir(&lst, &st, file, nil, nil, nil); d = malloc(sizeof(Dir)+nstr); if(d == nil) return nil; memset(d, 0, sizeof(Dir)+nstr); str = (char*)&d[1]; _p9dir(&lst, &st, file, d, &str, str+nstr); return d; }
Dir* dirfstat(int fd) { struct stat st; int nstr; Dir *d; char *str, tmp[100]; if(fstat(fd, &st) < 0) return nil; snprint(tmp, sizeof tmp, "/dev/fd/%d", fd); nstr = _p9dir(&st, &st, tmp, nil, nil, nil); d = malloc(sizeof(Dir)+(size_t)nstr); if(d == nil) return nil; memset(d, 0, sizeof(Dir)+(size_t)nstr); str = (char*)&d[1]; _p9dir(&st, &st, tmp, d, &str, str+nstr); return d; }
static int dirpackage(int fd, char *buf, int n, Dir **dp) { int oldwd; char *p, *str, *estr; int i, nstr, m; struct dirent *de; struct stat st, lst; Dir *d; n = countde(buf, n); if(n <= 0) return n; if((oldwd = open(".", O_RDONLY)) < 0) return -1; if(fchdir(fd) < 0) return -1; p = buf; nstr = 0; for(i=0; i<n; i++){ de = (struct dirent*)p; memset(&lst, 0, sizeof lst); if(de->d_name[0] == 0) /* nothing */ {} else if(lstat(de->d_name, &lst) < 0) de->d_name[0] = 0; else{ st = lst; if(S_ISLNK(lst.st_mode)) stat(de->d_name, &st); nstr += _p9dir(&lst, &st, de->d_name, nil, nil, nil); } p += d_reclen(de); } d = malloc(sizeof(Dir)*n+nstr); if(d == nil){ fchdir(oldwd); close(oldwd); return -1; } str = (char*)&d[n]; estr = str+nstr; p = buf; m = 0; for(i=0; i<n; i++){ de = (struct dirent*)p; if(de->d_name[0] != 0 && lstat(de->d_name, &lst) >= 0){ st = lst; if((lst.st_mode&S_IFMT) == S_IFLNK) stat(de->d_name, &st); _p9dir(&lst, &st, de->d_name, &d[m++], &str, estr); } p += d_reclen(de); } fchdir(oldwd); close(oldwd); *dp = d; return m; }