#include #include #include #include #include typedef struct NDir NDir; struct NDir { Dir; char *prefix; }; int errs = 0; int dflag; int lflag; int nflag; int pflag; int qflag; int rflag; int sflag; int tflag; int uflag; int Fflag; int ndirbuf; int ndir; NDir* dirbuf; int ls(char*, int); int compar(NDir*, NDir*); char* asciitime(long); char* darwx(long); void rwx(long, char*); void growto(long); void dowidths(Dir*); void format(Dir*, char*); void output(void); ulong clk; int swidth; /* max width of -s size */ int qwidth; /* max width of -q version */ int vwidth; /* max width of dev */ int uwidth; /* max width of userid */ int glwidth; /* max width of groupid and length */ Biobuf bin; #define HUNK 50 void main(int argc, char *argv[]) { int i, fd; char buf[64]; Binit(&bin, 1, OWRITE); ARGBEGIN{ case 'F': Fflag++; break; case 'd': dflag++; break; case 'l': lflag++; break; case 'n': nflag++; break; case 'p': pflag++; break; case 'q': qflag++; break; case 'r': rflag++; break; case 's': sflag++; break; case 't': tflag++; break; case 'u': uflag++; break; default: fprint(2, "usage: ls [-dlnpqrstuF] [file ...]\n"); exits("usage"); }ARGEND fmtinstall('M', dirmodeconv); if(lflag){ fd = open("/dev/time", OREAD); if(fd<0 || read(fd, buf, sizeof buf-1)<=0) fprint(2, "ls: can't open /dev/time\n"); close(fd); clk = strtoul(buf, 0, 0); } if(argc == 0) errs = ls(".", 0); else for(i=0; i 0){ n /= sizeof(Dir); growto(ndir+n); for(i=0; ilength+1023)/1024); if(n > swidth) swidth = n; } if(qflag) { n = sprint(buf, "%lud", db->qid.vers); if(n > qwidth) qwidth = n; } if(lflag) { n = sprint(buf, "%ud", db->dev); if(n > vwidth) vwidth = n; n = strlen(db->uid); if(n > uwidth) uwidth = n; n = sprint(buf, "%llud", db->length); n += strlen(db->gid); if(n > glwidth) glwidth = n; } } char* fileflag(Dir *db) { if(Fflag == 0) return ""; if(CHDIR & db->qid.path) return "/"; if(0111 & db->mode) return "*"; return ""; } void format(Dir *db, char *name) { if(sflag) Bprint(&bin, "%*llud ", swidth, (db->length+1023)/1024); if(qflag) Bprint(&bin, "%.8lux %*lud ", db->qid.path, qwidth, db->qid.vers); if(lflag) Bprint(&bin, "%M %C %*ud %*s %s %*llud %s %s\n", db->mode, db->type, vwidth, db->dev, -uwidth, db->uid, db->gid, (int)(glwidth-strlen(db->gid)), db->length, asciitime(uflag? db->atime : db->mtime), name); else Bprint(&bin, "%s%s\n", name, fileflag(db)); } void growto(long n) { if(n <= ndirbuf) return; ndirbuf = n; dirbuf=(NDir *)realloc(dirbuf, ndirbuf*sizeof(NDir)); if(dirbuf == 0){ fprint(2, "ls: malloc fail\n"); exits("malloc fail"); } } int compar(NDir *a, NDir *b) { long i; if(tflag){ if(uflag) i = b->atime-a->atime; else i = b->mtime-a->mtime; }else{ if(a->prefix && b->prefix){ i = strcmp(a->prefix, b->prefix); if(i == 0) i = strcmp(a->name, b->name); }else if(a->prefix){ i = strcmp(a->prefix, b->name); if(i == 0) i = 1; /* a is longer than b */ }else if(b->prefix){ i = strcmp(a->name, b->prefix); if(i == 0) i = -1; /* b is longer than a */ }else i = strcmp(a->name, b->name); } if(i == 0) i = (a