1151 lines
25 KiB
C
1151 lines
25 KiB
C
|
/* ls 5.4 - List files. Author: Kees J. Bot
|
||
|
* 25 Apr 1989
|
||
|
*
|
||
|
* About the amount of bytes for heap + stack under Minix:
|
||
|
* Ls needs a average amount of 42 bytes per unserviced directory entry, so
|
||
|
* scanning 10 directory levels deep in an ls -R with 100 entries per directory
|
||
|
* takes 42000 bytes of heap. So giving ls 10000 bytes is tight, 20000 is
|
||
|
* usually enough, 40000 is pessimistic.
|
||
|
*/
|
||
|
|
||
|
/* The array l_ifmt[] is used in an 'ls -l' to map the type of a file to a
|
||
|
* letter. This is done so that ls can list any future file or device type
|
||
|
* other than symlinks, without recompilation. (Yes it's dirty.)
|
||
|
*/
|
||
|
char l_ifmt[] = "0pcCd?bB-?l?s???";
|
||
|
|
||
|
#define ifmt(mode) l_ifmt[((mode) >> 12) & 0xF]
|
||
|
|
||
|
#define nil 0
|
||
|
#include <stdio.h>
|
||
|
#include <string.h>
|
||
|
#include <sys/types.h>
|
||
|
#include <sys/stat.h>
|
||
|
#include <stddef.h>
|
||
|
#include <stdlib.h>
|
||
|
#include <unistd.h>
|
||
|
#include <dirent.h>
|
||
|
#include <time.h>
|
||
|
#include <pwd.h>
|
||
|
#include <grp.h>
|
||
|
#include <errno.h>
|
||
|
#include <fcntl.h>
|
||
|
#include <limits.h>
|
||
|
#include <termios.h>
|
||
|
#include <sys/ioctl.h>
|
||
|
|
||
|
#ifndef major
|
||
|
#define major(dev) ((int) (((dev) >> 8) & 0xFF))
|
||
|
#define minor(dev) ((int) (((dev) >> 0) & 0xFF))
|
||
|
#endif
|
||
|
|
||
|
#if !__minix
|
||
|
#define SUPER_ID uid /* Let -A flag be default for SUPER_ID == 0. */
|
||
|
#else
|
||
|
#define SUPER_ID gid
|
||
|
#endif
|
||
|
|
||
|
#ifdef S_IFLNK
|
||
|
int (*status)(const char *file, struct stat *stp);
|
||
|
#else
|
||
|
#define status stat
|
||
|
#endif
|
||
|
|
||
|
/* Basic disk block size is 512 except for one niche O.S. */
|
||
|
#if __minix
|
||
|
#define BLOCK 1024
|
||
|
#else
|
||
|
#define BLOCK 512
|
||
|
#endif
|
||
|
|
||
|
/* Assume other systems have st_blocks. */
|
||
|
#if !__minix
|
||
|
#define ST_BLOCKS 1
|
||
|
#endif
|
||
|
|
||
|
/* Some terminals ignore more than 80 characters on a line. Dumb ones wrap
|
||
|
* when the cursor hits the side. Nice terminals don't wrap until they have
|
||
|
* to print the 81st character. Whether we like it or not, no column 80.
|
||
|
*/
|
||
|
int ncols= 79;
|
||
|
|
||
|
#define NSEP 3 /* # spaces between columns. */
|
||
|
|
||
|
#define MAXCOLS 128 /* Max # of files per line. */
|
||
|
|
||
|
char *arg0; /* Last component of argv[0]. */
|
||
|
int uid, gid; /* callers id. */
|
||
|
int ex= 0; /* Exit status to be. */
|
||
|
int istty; /* Output is on a terminal. */
|
||
|
|
||
|
/* Safer versions of malloc and realloc: */
|
||
|
|
||
|
void heaperr(void)
|
||
|
{
|
||
|
fprintf(stderr, "%s: Out of memory\n", arg0);
|
||
|
exit(-1);
|
||
|
}
|
||
|
|
||
|
void *allocate(size_t n)
|
||
|
/* Deliver or die. */
|
||
|
{
|
||
|
void *a;
|
||
|
|
||
|
if ((a= malloc(n)) == nil) heaperr();
|
||
|
return a;
|
||
|
}
|
||
|
|
||
|
void *reallocate(void *a, size_t n)
|
||
|
{
|
||
|
if ((a= realloc(a, n)) == nil) heaperr();
|
||
|
return a;
|
||
|
}
|
||
|
|
||
|
char allowed[] = "acdfghilnpqrstu1ACDFLMRTX";
|
||
|
char flags[sizeof(allowed)];
|
||
|
|
||
|
char arg0flag[] = "cdfmrtx"; /* These in argv[0] go to upper case. */
|
||
|
|
||
|
void setflags(char *flgs)
|
||
|
{
|
||
|
int c;
|
||
|
|
||
|
while ((c= *flgs++) != 0) {
|
||
|
if (strchr(allowed, c) == nil) {
|
||
|
fprintf(stderr, "Usage: %s [-%s] [file ...]\n",
|
||
|
arg0, allowed);
|
||
|
exit(1);
|
||
|
} else
|
||
|
if (strchr(flags, c) == nil) {
|
||
|
flags[strlen(flags)] = c;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
int present(int f)
|
||
|
{
|
||
|
return f == 0 || strchr(flags, f) != nil;
|
||
|
}
|
||
|
|
||
|
void report(char *f)
|
||
|
/* Like perror(3), but in the style: "ls: junk: No such file or directory. */
|
||
|
{
|
||
|
fprintf(stderr, "%s: %s: %s\n", arg0, f, strerror(errno));
|
||
|
ex= 1;
|
||
|
}
|
||
|
|
||
|
/* Two functions, uidname and gidname, translate id's to readable names.
|
||
|
* All names are remembered to avoid searching the password file.
|
||
|
*/
|
||
|
#define NNAMES (1 << (sizeof(int) + sizeof(char *)))
|
||
|
enum whatmap { PASSWD, GROUP };
|
||
|
|
||
|
struct idname { /* Hash list of names. */
|
||
|
struct idname *next;
|
||
|
char *name;
|
||
|
uid_t id;
|
||
|
} *uids[NNAMES], *gids[NNAMES];
|
||
|
|
||
|
char *idname(unsigned id, enum whatmap map)
|
||
|
/* Return name for a given user/group id. */
|
||
|
{
|
||
|
struct idname *i;
|
||
|
struct idname **ids= &(map == PASSWD ? uids : gids)[id % NNAMES];
|
||
|
|
||
|
while ((i= *ids) != nil && id < i->id) ids= &i->next;
|
||
|
|
||
|
if (i == nil || id != i->id) {
|
||
|
/* Not found, go look in the password or group map. */
|
||
|
char *name= nil;
|
||
|
char noname[3 * sizeof(uid_t)];
|
||
|
|
||
|
if (!present('n')) {
|
||
|
if (map == PASSWD) {
|
||
|
struct passwd *pw= getpwuid(id);
|
||
|
|
||
|
if (pw != nil) name= pw->pw_name;
|
||
|
} else {
|
||
|
struct group *gr= getgrgid(id);
|
||
|
|
||
|
if (gr != nil) name= gr->gr_name;
|
||
|
}
|
||
|
}
|
||
|
if (name == nil) {
|
||
|
/* Can't find it, weird. Use numerical "name." */
|
||
|
sprintf(noname, "%u", id);
|
||
|
name= noname;
|
||
|
}
|
||
|
|
||
|
/* Add a new id-to-name cell. */
|
||
|
i= allocate(sizeof(*i));
|
||
|
i->id= id;
|
||
|
i->name= allocate(strlen(name) + 1);
|
||
|
strcpy(i->name, name);
|
||
|
i->next= *ids;
|
||
|
*ids= i;
|
||
|
}
|
||
|
return i->name;
|
||
|
}
|
||
|
|
||
|
#define uidname(uid) idname((uid), PASSWD)
|
||
|
#define gidname(gid) idname((gid), GROUP)
|
||
|
|
||
|
/* Path name construction, addpath adds a component, delpath removes it.
|
||
|
* The string path is used throughout the program as the file under examination.
|
||
|
*/
|
||
|
|
||
|
char *path; /* Path name constructed in path[]. */
|
||
|
int plen= 0, pidx= 0; /* Lenght/index for path[]. */
|
||
|
|
||
|
void addpath(int *didx, char *name)
|
||
|
/* Add a component to path. (name may also be a full path at the first call)
|
||
|
* The index where the current path ends is stored in *pdi.
|
||
|
*/
|
||
|
{
|
||
|
if (plen == 0) path= (char *) allocate((plen= 32) * sizeof(path[0]));
|
||
|
|
||
|
if (pidx == 1 && path[0] == '.') pidx= 0; /* Remove "." */
|
||
|
|
||
|
*didx= pidx; /* Record point to go back to for delpath. */
|
||
|
|
||
|
if (pidx > 0 && path[pidx-1] != '/') path[pidx++]= '/';
|
||
|
|
||
|
do {
|
||
|
if (*name != '/' || pidx == 0 || path[pidx-1] != '/') {
|
||
|
if (pidx == plen) {
|
||
|
path= (char *) reallocate((void *) path,
|
||
|
(plen*= 2) * sizeof(path[0]));
|
||
|
}
|
||
|
path[pidx++]= *name;
|
||
|
}
|
||
|
} while (*name++ != 0);
|
||
|
|
||
|
--pidx; /* Put pidx back at the null. The path[pidx++]= '/'
|
||
|
* statement will overwrite it at the next call.
|
||
|
*/
|
||
|
}
|
||
|
|
||
|
#define delpath(didx) (path[pidx= didx]= 0) /* Remove component. */
|
||
|
|
||
|
int field = 0; /* (used to be) Fields that must be printed. */
|
||
|
/* (now) Effects triggered by certain flags. */
|
||
|
|
||
|
#define L_INODE 0x0001 /* -i */
|
||
|
#define L_BLOCKS 0x0002 /* -s */
|
||
|
#define L_EXTRA 0x0004 /* -X */
|
||
|
#define L_MODE 0x0008 /* -lMX */
|
||
|
#define L_LONG 0x0010 /* -l */
|
||
|
#define L_GROUP 0x0020 /* -g */
|
||
|
#define L_BYTIME 0x0040 /* -tuc */
|
||
|
#define L_ATIME 0x0080 /* -u */
|
||
|
#define L_CTIME 0x0100 /* -c */
|
||
|
#define L_MARK 0x0200 /* -F */
|
||
|
#define L_MARKDIR 0x0400 /* -p */
|
||
|
#define L_TYPE 0x0800 /* -D */
|
||
|
#define L_LONGTIME 0x1000 /* -T */
|
||
|
#define L_DIR 0x2000 /* -d */
|
||
|
#define L_KMG 0x4000 /* -h */
|
||
|
|
||
|
struct file { /* A file plus stat(2) information. */
|
||
|
struct file *next; /* Lists are made of them. */
|
||
|
char *name; /* Null terminated name. */
|
||
|
ino_t ino;
|
||
|
mode_t mode;
|
||
|
uid_t uid;
|
||
|
gid_t gid;
|
||
|
nlink_t nlink;
|
||
|
dev_t rdev;
|
||
|
off_t size;
|
||
|
time_t mtime;
|
||
|
time_t atime;
|
||
|
time_t ctime;
|
||
|
#if ST_BLOCKS
|
||
|
long blocks;
|
||
|
#endif
|
||
|
};
|
||
|
|
||
|
void setstat(struct file *f, struct stat *stp)
|
||
|
{
|
||
|
f->ino= stp->st_ino;
|
||
|
f->mode= stp->st_mode;
|
||
|
f->nlink= stp->st_nlink;
|
||
|
f->uid= stp->st_uid;
|
||
|
f->gid= stp->st_gid;
|
||
|
f->rdev= stp->st_rdev;
|
||
|
f->size= stp->st_size;
|
||
|
f->mtime= stp->st_mtime;
|
||
|
f->atime= stp->st_atime;
|
||
|
f->ctime= stp->st_ctime;
|
||
|
#if ST_BLOCKS
|
||
|
f->blocks= stp->st_blocks;
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
#define PAST (26*7*24*3600L) /* Half a year ago. */
|
||
|
/* Between PAST and FUTURE from now a time is printed, otherwise a year. */
|
||
|
#define FUTURE ( 1*7*24*3600L) /* One week. */
|
||
|
|
||
|
static char *timestamp(struct file *f)
|
||
|
/* Transform the right time field into something readable. */
|
||
|
{
|
||
|
struct tm *tm;
|
||
|
time_t t;
|
||
|
static time_t now;
|
||
|
static int drift= 0;
|
||
|
static char date[] = "Jan 19 03:14:07 2038";
|
||
|
static char month[] = "JanFebMarAprMayJunJulAugSepOctNovDec";
|
||
|
|
||
|
t= f->mtime;
|
||
|
if (field & L_ATIME) t= f->atime;
|
||
|
if (field & L_CTIME) t= f->ctime;
|
||
|
|
||
|
tm= localtime(&t);
|
||
|
if (--drift < 0) { time(&now); drift= 50; } /* limit time() calls */
|
||
|
|
||
|
if (field & L_LONGTIME) {
|
||
|
sprintf(date, "%.3s %2d %02d:%02d:%02d %d",
|
||
|
month + 3*tm->tm_mon,
|
||
|
tm->tm_mday,
|
||
|
tm->tm_hour, tm->tm_min, tm->tm_sec,
|
||
|
1900 + tm->tm_year);
|
||
|
} else
|
||
|
if (t < now - PAST || t > now + FUTURE) {
|
||
|
sprintf(date, "%.3s %2d %d",
|
||
|
month + 3*tm->tm_mon,
|
||
|
tm->tm_mday,
|
||
|
1900 + tm->tm_year);
|
||
|
} else {
|
||
|
sprintf(date, "%.3s %2d %02d:%02d",
|
||
|
month + 3*tm->tm_mon,
|
||
|
tm->tm_mday,
|
||
|
tm->tm_hour, tm->tm_min);
|
||
|
}
|
||
|
return date;
|
||
|
}
|
||
|
|
||
|
char *permissions(struct file *f)
|
||
|
/* Compute long or short rwx bits. */
|
||
|
{
|
||
|
static char rwx[] = "drwxr-x--x";
|
||
|
|
||
|
rwx[0] = ifmt(f->mode);
|
||
|
/* Note that rwx[0] is a guess for the more alien file types. It is
|
||
|
* correct for BSD4.3 and derived systems. I just don't know how
|
||
|
* "standardized" these numbers are.
|
||
|
*/
|
||
|
|
||
|
if (field & L_EXTRA) { /* Short style */
|
||
|
int mode = f->mode, ucase= 0;
|
||
|
|
||
|
if (uid == f->uid) { /* What group of bits to use. */
|
||
|
/* mode<<= 0, */
|
||
|
ucase= (mode<<3) | (mode<<6);
|
||
|
/* Remember if group or others have permissions. */
|
||
|
} else
|
||
|
if (gid == f->gid) {
|
||
|
mode<<= 3;
|
||
|
} else {
|
||
|
mode<<= 6;
|
||
|
}
|
||
|
rwx[1]= mode&S_IRUSR ? (ucase&S_IRUSR ? 'R' : 'r') : '-';
|
||
|
rwx[2]= mode&S_IWUSR ? (ucase&S_IWUSR ? 'W' : 'w') : '-';
|
||
|
|
||
|
if (mode&S_IXUSR) {
|
||
|
static char sbit[]= { 'x', 'g', 'u', 's' };
|
||
|
|
||
|
rwx[3]= sbit[(f->mode&(S_ISUID|S_ISGID))>>10];
|
||
|
if (ucase&S_IXUSR) rwx[3] += 'A'-'a';
|
||
|
} else {
|
||
|
rwx[3]= f->mode&(S_ISUID|S_ISGID) ? '=' : '-';
|
||
|
}
|
||
|
rwx[4]= 0;
|
||
|
} else { /* Long form. */
|
||
|
char *p= rwx+1;
|
||
|
int mode= f->mode;
|
||
|
|
||
|
do {
|
||
|
p[0] = (mode & S_IRUSR) ? 'r' : '-';
|
||
|
p[1] = (mode & S_IWUSR) ? 'w' : '-';
|
||
|
p[2] = (mode & S_IXUSR) ? 'x' : '-';
|
||
|
mode<<= 3;
|
||
|
} while ((p+=3) <= rwx+7);
|
||
|
|
||
|
if (f->mode&S_ISUID) rwx[3]= f->mode&(S_IXUSR>>0) ? 's' : '=';
|
||
|
if (f->mode&S_ISGID) rwx[6]= f->mode&(S_IXUSR>>3) ? 's' : '=';
|
||
|
if (f->mode&S_ISVTX) rwx[9]= f->mode&(S_IXUSR>>6) ? 't' : '=';
|
||
|
}
|
||
|
return rwx;
|
||
|
}
|
||
|
|
||
|
void numeral(int i, char **pp)
|
||
|
{
|
||
|
char itoa[3*sizeof(int)], *a=itoa;
|
||
|
|
||
|
do *a++ = i%10 + '0'; while ((i/=10) > 0);
|
||
|
|
||
|
do *(*pp)++ = *--a; while (a>itoa);
|
||
|
}
|
||
|
|
||
|
#define K 1024L /* A kilobyte counts in multiples of K */
|
||
|
#define T 1000L /* A megabyte in T*K, a gigabyte in T*T*K */
|
||
|
|
||
|
char *cxsize(struct file *f)
|
||
|
/* Try and fail to turn a 32 bit size into 4 readable characters. */
|
||
|
{
|
||
|
static char siz[] = "1.2m";
|
||
|
char *p= siz;
|
||
|
off_t z;
|
||
|
|
||
|
siz[1]= siz[2]= siz[3]= 0;
|
||
|
|
||
|
if (f->size <= 5*K) { /* <= 5K prints as is. */
|
||
|
numeral((int) f->size, &p);
|
||
|
return siz;
|
||
|
}
|
||
|
z= (f->size + K-1) / K;
|
||
|
|
||
|
if (z <= 999) { /* Print as 123k. */
|
||
|
numeral((int) z, &p);
|
||
|
*p = 'k'; /* Can't use 'K', looks bad */
|
||
|
} else
|
||
|
if (z*10 <= 99*T) { /* 1.2m (Try ls -X /dev/at0) */
|
||
|
z= (z*10 + T-1) / T; /* Force roundup */
|
||
|
numeral((int) z / 10, &p);
|
||
|
*p++ = '.';
|
||
|
numeral((int) z % 10, &p);
|
||
|
*p = 'm';
|
||
|
} else
|
||
|
if (z <= 999*T) { /* 123m */
|
||
|
numeral((int) ((z + T-1) / T), &p);
|
||
|
*p = 'm';
|
||
|
} else { /* 1.2g */
|
||
|
z= (z*10 + T*T-1) / (T*T);
|
||
|
numeral((int) z / 10, &p);
|
||
|
*p++ = '.';
|
||
|
numeral((int) z % 10, &p);
|
||
|
*p = 'g';
|
||
|
}
|
||
|
return siz;
|
||
|
}
|
||
|
|
||
|
/* Transform size of file to number of blocks. This was once a function that
|
||
|
* guessed the number of indirect blocks, but that nonsense has been removed.
|
||
|
*/
|
||
|
#if ST_BLOCKS
|
||
|
#define nblocks(f) ((f)->blocks)
|
||
|
#else
|
||
|
#define nblocks(f) (((f)->size + BLOCK-1) / BLOCK)
|
||
|
#endif
|
||
|
|
||
|
/* From number of blocks to kilobytes. */
|
||
|
#if BLOCK < 1024
|
||
|
#define nblk2k(nb) (((nb) + (1024 / BLOCK - 1)) / (1024 / BLOCK))
|
||
|
#else
|
||
|
#define nblk2k(nb) ((nb) * (BLOCK / 1024))
|
||
|
#endif
|
||
|
|
||
|
static int (*CMP)(struct file *f1, struct file *f2);
|
||
|
static int (*rCMP)(struct file *f1, struct file *f2);
|
||
|
|
||
|
static void mergesort(struct file **al)
|
||
|
/* This is either a stable mergesort, or thermal noise, I'm no longer sure.
|
||
|
* It must be called like this: if (L != nil && L->next != nil) mergesort(&L);
|
||
|
*/
|
||
|
{
|
||
|
/* static */ struct file *l1, **mid; /* Need not be local */
|
||
|
struct file *l2;
|
||
|
|
||
|
l1= *(mid= &(*al)->next);
|
||
|
do {
|
||
|
if ((l1= l1->next) == nil) break;
|
||
|
mid= &(*mid)->next;
|
||
|
} while ((l1= l1->next) != nil);
|
||
|
|
||
|
l2= *mid;
|
||
|
*mid= nil;
|
||
|
|
||
|
if ((*al)->next != nil) mergesort(al);
|
||
|
if (l2->next != nil) mergesort(&l2);
|
||
|
|
||
|
l1= *al;
|
||
|
for (;;) {
|
||
|
if ((*CMP)(l1, l2) <= 0) {
|
||
|
if ((l1= *(al= &l1->next)) == nil) {
|
||
|
*al= l2;
|
||
|
break;
|
||
|
}
|
||
|
} else {
|
||
|
*al= l2;
|
||
|
l2= *(al= &l2->next);
|
||
|
*al= l1;
|
||
|
if (l2 == nil) break;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
int namecmp(struct file *f1, struct file *f2)
|
||
|
{
|
||
|
return strcmp(f1->name, f2->name);
|
||
|
}
|
||
|
|
||
|
int mtimecmp(struct file *f1, struct file *f2)
|
||
|
{
|
||
|
return f1->mtime == f2->mtime ? 0 : f1->mtime > f2->mtime ? -1 : 1;
|
||
|
}
|
||
|
|
||
|
int atimecmp(struct file *f1, struct file *f2)
|
||
|
{
|
||
|
return f1->atime == f2->atime ? 0 : f1->atime > f2->atime ? -1 : 1;
|
||
|
}
|
||
|
|
||
|
int ctimecmp(struct file *f1, struct file *f2)
|
||
|
{
|
||
|
return f1->ctime == f2->ctime ? 0 : f1->ctime > f2->ctime ? -1 : 1;
|
||
|
}
|
||
|
|
||
|
int typecmp(struct file *f1, struct file *f2)
|
||
|
{
|
||
|
return ifmt(f1->mode) - ifmt(f2->mode);
|
||
|
}
|
||
|
|
||
|
int revcmp(struct file *f1, struct file *f2) { return (*rCMP)(f2, f1); }
|
||
|
|
||
|
static void sort(struct file **al)
|
||
|
/* Sort the files according to the flags. */
|
||
|
{
|
||
|
if (!present('f') && *al != nil && (*al)->next != nil) {
|
||
|
CMP= namecmp;
|
||
|
|
||
|
if (!(field & L_BYTIME)) {
|
||
|
/* Sort on name */
|
||
|
|
||
|
if (present('r')) { rCMP= CMP; CMP= revcmp; }
|
||
|
mergesort(al);
|
||
|
} else {
|
||
|
/* Sort on name first, then sort on time. */
|
||
|
|
||
|
mergesort(al);
|
||
|
if (field & L_CTIME) {
|
||
|
CMP= ctimecmp;
|
||
|
} else
|
||
|
if (field & L_ATIME) {
|
||
|
CMP= atimecmp;
|
||
|
} else {
|
||
|
CMP= mtimecmp;
|
||
|
}
|
||
|
|
||
|
if (present('r')) { rCMP= CMP; CMP= revcmp; }
|
||
|
mergesort(al);
|
||
|
}
|
||
|
/* Separate by file type if so desired. */
|
||
|
|
||
|
if (field & L_TYPE) {
|
||
|
CMP= typecmp;
|
||
|
mergesort(al);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
struct file *newfile(char *name)
|
||
|
/* Create file structure for given name. */
|
||
|
{
|
||
|
struct file *new;
|
||
|
|
||
|
new= (struct file *) allocate(sizeof(*new));
|
||
|
new->name= strcpy((char *) allocate(strlen(name)+1), name);
|
||
|
return new;
|
||
|
}
|
||
|
|
||
|
void pushfile(struct file **flist, struct file *new)
|
||
|
/* Add file to the head of a list. */
|
||
|
{
|
||
|
new->next= *flist;
|
||
|
*flist= new;
|
||
|
}
|
||
|
|
||
|
void delfile(struct file *old)
|
||
|
/* Release old file structure. */
|
||
|
{
|
||
|
free((void *) old->name);
|
||
|
free((void *) old);
|
||
|
}
|
||
|
|
||
|
struct file *popfile(struct file **flist)
|
||
|
/* Pop file off top of file list. */
|
||
|
{
|
||
|
struct file *f;
|
||
|
|
||
|
f= *flist;
|
||
|
*flist= f->next;
|
||
|
return f;
|
||
|
}
|
||
|
|
||
|
int dotflag(char *name)
|
||
|
/* Return flag that would make ls list this name: -a or -A. */
|
||
|
{
|
||
|
if (*name++ != '.') return 0;
|
||
|
|
||
|
switch (*name++) {
|
||
|
case 0: return 'a'; /* "." */
|
||
|
case '.': if (*name == 0) return 'a'; /* ".." */
|
||
|
default: return 'A'; /* ".*" */
|
||
|
}
|
||
|
}
|
||
|
|
||
|
int adddir(struct file **aflist, char *name)
|
||
|
/* Add directory entries of directory name to a file list. */
|
||
|
{
|
||
|
DIR *d;
|
||
|
struct dirent *e;
|
||
|
|
||
|
if (access(name, 0) < 0) {
|
||
|
report(name);
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
if ((d= opendir(name)) == nil) {
|
||
|
report(name);
|
||
|
return 0;
|
||
|
}
|
||
|
while ((e= readdir(d)) != nil) {
|
||
|
if (e->d_ino != 0 && present(dotflag(e->d_name))) {
|
||
|
pushfile(aflist, newfile(e->d_name));
|
||
|
aflist= &(*aflist)->next;
|
||
|
}
|
||
|
}
|
||
|
closedir(d);
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
off_t countblocks(struct file *flist)
|
||
|
/* Compute total block count for a list of files. */
|
||
|
{
|
||
|
off_t cb = 0;
|
||
|
|
||
|
while (flist != nil) {
|
||
|
switch (flist->mode & S_IFMT) {
|
||
|
case S_IFDIR:
|
||
|
case S_IFREG:
|
||
|
#ifdef S_IFLNK
|
||
|
case S_IFLNK:
|
||
|
#endif
|
||
|
cb += nblocks(flist);
|
||
|
}
|
||
|
flist= flist->next;
|
||
|
}
|
||
|
return cb;
|
||
|
}
|
||
|
|
||
|
void printname(char *name)
|
||
|
/* Print a name with control characters as '?' (unless -q). The terminal is
|
||
|
* assumed to be eight bit clean.
|
||
|
*/
|
||
|
{
|
||
|
int c, q= present('q');
|
||
|
|
||
|
while ((c= (unsigned char) *name++) != 0) {
|
||
|
if (q && (c < ' ' || c == 0177)) c= '?';
|
||
|
putchar(c);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
int mark(struct file *f, int doit)
|
||
|
{
|
||
|
int c;
|
||
|
|
||
|
c= 0;
|
||
|
|
||
|
if (field & L_MARK) {
|
||
|
switch (f->mode & S_IFMT) {
|
||
|
case S_IFDIR: c= '/'; break;
|
||
|
#ifdef S_IFIFO
|
||
|
case S_IFIFO: c= '|'; break;
|
||
|
#endif
|
||
|
#ifdef S_IFLNK
|
||
|
case S_IFLNK: c= '@'; break;
|
||
|
#endif
|
||
|
#ifdef S_IFSOCK
|
||
|
case S_IFSOCK: c= '='; break;
|
||
|
#endif
|
||
|
case S_IFREG:
|
||
|
if (f->mode & (S_IXUSR | S_IXGRP | S_IXOTH)) c= '*';
|
||
|
break;
|
||
|
}
|
||
|
} else
|
||
|
if (field & L_MARKDIR) {
|
||
|
if (S_ISDIR(f->mode)) c= '/';
|
||
|
}
|
||
|
|
||
|
if (doit && c != 0) putchar(c);
|
||
|
return c;
|
||
|
}
|
||
|
|
||
|
/* Width of entire column, and of several fields. */
|
||
|
enum { W_COL, W_INO, W_BLK, W_NLINK, W_UID, W_GID, W_SIZE, W_NAME, MAXFLDS };
|
||
|
|
||
|
unsigned char fieldwidth[MAXCOLS][MAXFLDS];
|
||
|
|
||
|
void maxise(unsigned char *aw, int w)
|
||
|
/* Set *aw to the larger of it and w. */
|
||
|
{
|
||
|
if (w > *aw) {
|
||
|
if (w > UCHAR_MAX) w= UCHAR_MAX;
|
||
|
*aw= w;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
int numwidth(unsigned long n)
|
||
|
/* Compute width of 'n' when printed. */
|
||
|
{
|
||
|
int width= 0;
|
||
|
|
||
|
do { width++; } while ((n /= 10) > 0);
|
||
|
return width;
|
||
|
}
|
||
|
|
||
|
#if !__minix
|
||
|
int numxwidth(unsigned long n)
|
||
|
/* Compute width of 'n' when printed in hex. */
|
||
|
{
|
||
|
int width= 0;
|
||
|
|
||
|
do { width++; } while ((n /= 16) > 0);
|
||
|
return width;
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
static int nsp= 0; /* This many spaces have not been printed yet. */
|
||
|
#define spaces(n) (nsp= (n))
|
||
|
#define terpri() (nsp= 0, putchar('\n')) /* No trailing spaces */
|
||
|
|
||
|
void print1(struct file *f, int col, int doit)
|
||
|
/* Either compute the number of spaces needed to print file f (doit == 0) or
|
||
|
* really print it (doit == 1).
|
||
|
*/
|
||
|
{
|
||
|
int width= 0, n;
|
||
|
char *p;
|
||
|
unsigned char *f1width = fieldwidth[col];
|
||
|
|
||
|
while (nsp>0) { putchar(' '); nsp--; }/* Fill gap between two columns */
|
||
|
|
||
|
if (field & L_INODE) {
|
||
|
if (doit) {
|
||
|
printf("%*d ", f1width[W_INO], f->ino);
|
||
|
} else {
|
||
|
maxise(&f1width[W_INO], numwidth(f->ino));
|
||
|
width++;
|
||
|
}
|
||
|
}
|
||
|
if (field & L_BLOCKS) {
|
||
|
unsigned long nb= nblk2k(nblocks(f));
|
||
|
if (doit) {
|
||
|
printf("%*lu ", f1width[W_BLK], nb);
|
||
|
} else {
|
||
|
maxise(&f1width[W_BLK], numwidth(nb));
|
||
|
width++;
|
||
|
}
|
||
|
}
|
||
|
if (field & L_MODE) {
|
||
|
if (doit) {
|
||
|
printf("%s ", permissions(f));
|
||
|
} else {
|
||
|
width+= (field & L_EXTRA) ? 5 : 11;
|
||
|
}
|
||
|
}
|
||
|
if (field & L_EXTRA) {
|
||
|
p= cxsize(f);
|
||
|
n= strlen(p)+1;
|
||
|
|
||
|
if (doit) {
|
||
|
n= f1width[W_SIZE] - n;
|
||
|
while (n > 0) { putchar(' '); --n; }
|
||
|
printf("%s ", p);
|
||
|
} else {
|
||
|
maxise(&f1width[W_SIZE], n);
|
||
|
}
|
||
|
}
|
||
|
if (field & L_LONG) {
|
||
|
if (doit) {
|
||
|
printf("%*u ", f1width[W_NLINK], (unsigned) f->nlink);
|
||
|
} else {
|
||
|
maxise(&f1width[W_NLINK], numwidth(f->nlink));
|
||
|
width++;
|
||
|
}
|
||
|
if (!(field & L_GROUP)) {
|
||
|
if (doit) {
|
||
|
printf("%-*s ", f1width[W_UID],
|
||
|
uidname(f->uid));
|
||
|
} else {
|
||
|
maxise(&f1width[W_UID],
|
||
|
strlen(uidname(f->uid)));
|
||
|
width+= 2;
|
||
|
}
|
||
|
}
|
||
|
if (doit) {
|
||
|
printf("%-*s ", f1width[W_GID], gidname(f->gid));
|
||
|
} else {
|
||
|
maxise(&f1width[W_GID], strlen(gidname(f->gid)));
|
||
|
width+= 2;
|
||
|
}
|
||
|
|
||
|
switch (f->mode & S_IFMT) {
|
||
|
case S_IFBLK:
|
||
|
case S_IFCHR:
|
||
|
#ifdef S_IFMPB
|
||
|
case S_IFMPB:
|
||
|
#endif
|
||
|
#ifdef S_IFMPC
|
||
|
case S_IFMPC:
|
||
|
#endif
|
||
|
#if __minix
|
||
|
if (doit) {
|
||
|
printf("%*d, %3d ", f1width[W_SIZE] - 5,
|
||
|
major(f->rdev), minor(f->rdev));
|
||
|
} else {
|
||
|
maxise(&f1width[W_SIZE],
|
||
|
numwidth(major(f->rdev)) + 5);
|
||
|
width++;
|
||
|
}
|
||
|
#else /* !__minix */
|
||
|
if (doit) {
|
||
|
printf("%*lX ", f1width[W_SIZE],
|
||
|
(unsigned long) f->rdev);
|
||
|
} else {
|
||
|
maxise(&f1width[W_SIZE], numwidth(f->rdev));
|
||
|
width++;
|
||
|
}
|
||
|
#endif /* !__minix */
|
||
|
break;
|
||
|
default:
|
||
|
if (field & L_KMG) {
|
||
|
p= cxsize(f);
|
||
|
n= strlen(p)+1;
|
||
|
|
||
|
if (doit) {
|
||
|
n= f1width[W_SIZE] - n;
|
||
|
while (n > 0) { putchar(' '); --n; }
|
||
|
printf("%s ", p);
|
||
|
} else {
|
||
|
maxise(&f1width[W_SIZE], n);
|
||
|
}
|
||
|
} else {
|
||
|
if (doit) {
|
||
|
printf("%*lu ", f1width[W_SIZE],
|
||
|
(unsigned long) f->size);
|
||
|
} else {
|
||
|
maxise(&f1width[W_SIZE],
|
||
|
numwidth(f->size));
|
||
|
width++;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (doit) {
|
||
|
printf("%s ", timestamp(f));
|
||
|
} else {
|
||
|
width+= (field & L_LONGTIME) ? 21 : 13;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
n= strlen(f->name);
|
||
|
if (doit) {
|
||
|
printname(f->name);
|
||
|
if (mark(f, 1) != 0) n++;
|
||
|
#ifdef S_IFLNK
|
||
|
if ((field & L_LONG) && (f->mode & S_IFMT) == S_IFLNK) {
|
||
|
char *buf;
|
||
|
int r, didx;
|
||
|
|
||
|
buf= (char *) allocate(((size_t) f->size + 1)
|
||
|
* sizeof(buf[0]));
|
||
|
addpath(&didx, f->name);
|
||
|
r= readlink(path, buf, (int) f->size);
|
||
|
delpath(didx);
|
||
|
if (r > 0) buf[r] = 0; else r=1, strcpy(buf, "?");
|
||
|
printf(" -> ");
|
||
|
printname(buf);
|
||
|
free((void *) buf);
|
||
|
n+= 4 + r;
|
||
|
}
|
||
|
#endif
|
||
|
spaces(f1width[W_NAME] - n);
|
||
|
} else {
|
||
|
if (mark(f, 0) != 0) n++;
|
||
|
#ifdef S_IFLNK
|
||
|
if ((field & L_LONG) && (f->mode & S_IFMT) == S_IFLNK) {
|
||
|
n+= 4 + (int) f->size;
|
||
|
}
|
||
|
#endif
|
||
|
maxise(&f1width[W_NAME], n + NSEP);
|
||
|
|
||
|
for (n= 1; n < MAXFLDS; n++) width+= f1width[n];
|
||
|
maxise(&f1width[W_COL], width);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
int countfiles(struct file *flist)
|
||
|
/* Return number of files in the list. */
|
||
|
{
|
||
|
int n= 0;
|
||
|
|
||
|
while (flist != nil) { n++; flist= flist->next; }
|
||
|
|
||
|
return n;
|
||
|
}
|
||
|
|
||
|
struct file *filecol[MAXCOLS]; /* filecol[i] is list of files for column i. */
|
||
|
int nfiles, nlines; /* # files to print, # of lines needed. */
|
||
|
|
||
|
void columnise(struct file *flist, int nplin)
|
||
|
/* Chop list of files up in columns. Note that 3 columns are used for 5 files
|
||
|
* even though nplin may be 4, filecol[3] will simply be nil.
|
||
|
*/
|
||
|
{
|
||
|
int i, j;
|
||
|
|
||
|
nlines= (nfiles + nplin - 1) / nplin; /* nlines needed for nfiles */
|
||
|
|
||
|
filecol[0]= flist;
|
||
|
|
||
|
for (i=1; i<nplin; i++) { /* Give nlines files to each column. */
|
||
|
for (j=0; j<nlines && flist != nil; j++) flist= flist->next;
|
||
|
|
||
|
filecol[i]= flist;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
int print(struct file *flist, int nplin, int doit)
|
||
|
/* Try (doit == 0), or really print the list of files over nplin columns.
|
||
|
* Return true if it can be done in nplin columns or if nplin == 1.
|
||
|
*/
|
||
|
{
|
||
|
register struct file *f;
|
||
|
register int col, fld, totlen;
|
||
|
|
||
|
columnise(flist, nplin);
|
||
|
|
||
|
if (!doit) {
|
||
|
for (col= 0; col < nplin; col++) {
|
||
|
for (fld= 0; fld < MAXFLDS; fld++) {
|
||
|
fieldwidth[col][fld]= 0;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
while (--nlines >= 0) {
|
||
|
totlen= 0;
|
||
|
|
||
|
for (col= 0; col < nplin; col++) {
|
||
|
if ((f= filecol[col]) != nil) {
|
||
|
filecol[col]= f->next;
|
||
|
print1(f, col, doit);
|
||
|
}
|
||
|
if (!doit && nplin > 1) {
|
||
|
/* See if this line is not too long. */
|
||
|
if (fieldwidth[col][W_COL] == UCHAR_MAX) {
|
||
|
return 0;
|
||
|
}
|
||
|
totlen+= fieldwidth[col][W_COL];
|
||
|
if (totlen > ncols+NSEP) return 0;
|
||
|
}
|
||
|
}
|
||
|
if (doit) terpri();
|
||
|
}
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
enum depth { SURFACE, SURFACE1, SUBMERGED };
|
||
|
enum state { BOTTOM, SINKING, FLOATING };
|
||
|
|
||
|
void listfiles(struct file *flist, enum depth depth, enum state state)
|
||
|
/* Main workhorse of ls, it sorts and prints the list of files. Flags:
|
||
|
* depth: working with the command line / just one file / listing dir.
|
||
|
* state: How "recursive" do we have to be.
|
||
|
*/
|
||
|
{
|
||
|
struct file *dlist= nil, **afl= &flist, **adl= &dlist;
|
||
|
int nplin;
|
||
|
static int white = 1; /* Nothing printed yet. */
|
||
|
|
||
|
/* Flush everything previously printed, so new error output will
|
||
|
* not intermix with files listed earlier.
|
||
|
*/
|
||
|
fflush(stdout);
|
||
|
|
||
|
if (field != 0 || state != BOTTOM) { /* Need stat(2) info. */
|
||
|
while (*afl != nil) {
|
||
|
static struct stat st;
|
||
|
int r, didx;
|
||
|
|
||
|
addpath(&didx, (*afl)->name);
|
||
|
|
||
|
if ((r= status(path, &st)) < 0
|
||
|
#ifdef S_IFLNK
|
||
|
&& (status == lstat || lstat(path, &st) < 0)
|
||
|
#endif
|
||
|
) {
|
||
|
if (depth != SUBMERGED || errno != ENOENT)
|
||
|
report((*afl)->name);
|
||
|
delfile(popfile(afl));
|
||
|
} else {
|
||
|
setstat(*afl, &st);
|
||
|
afl= &(*afl)->next;
|
||
|
}
|
||
|
delpath(didx);
|
||
|
}
|
||
|
}
|
||
|
sort(&flist);
|
||
|
|
||
|
if (depth == SUBMERGED && (field & (L_BLOCKS | L_LONG))) {
|
||
|
printf("total %ld\n", nblk2k(countblocks(flist)));
|
||
|
}
|
||
|
|
||
|
if (state == SINKING || depth == SURFACE1) {
|
||
|
/* Don't list directories themselves, list their contents later. */
|
||
|
afl= &flist;
|
||
|
while (*afl != nil) {
|
||
|
if (((*afl)->mode & S_IFMT) == S_IFDIR) {
|
||
|
pushfile(adl, popfile(afl));
|
||
|
adl= &(*adl)->next;
|
||
|
} else {
|
||
|
afl= &(*afl)->next;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ((nfiles= countfiles(flist)) > 0) {
|
||
|
/* Print files in how many columns? */
|
||
|
nplin= !present('C') ? 1 : nfiles < MAXCOLS ? nfiles : MAXCOLS;
|
||
|
|
||
|
while (!print(flist, nplin, 0)) nplin--; /* Try first */
|
||
|
|
||
|
print(flist, nplin, 1); /* Then do it! */
|
||
|
white = 0;
|
||
|
}
|
||
|
|
||
|
while (flist != nil) { /* Destroy file list */
|
||
|
if (state == FLOATING && (flist->mode & S_IFMT) == S_IFDIR) {
|
||
|
/* But keep these directories for ls -R. */
|
||
|
pushfile(adl, popfile(&flist));
|
||
|
adl= &(*adl)->next;
|
||
|
} else {
|
||
|
delfile(popfile(&flist));
|
||
|
}
|
||
|
}
|
||
|
|
||
|
while (dlist != nil) { /* List directories */
|
||
|
if (dotflag(dlist->name) != 'a' || depth != SUBMERGED) {
|
||
|
int didx;
|
||
|
|
||
|
addpath(&didx, dlist->name);
|
||
|
|
||
|
flist= nil;
|
||
|
if (adddir(&flist, path)) {
|
||
|
if (depth != SURFACE1) {
|
||
|
if (!white) putchar('\n');
|
||
|
printf("%s:\n", path);
|
||
|
white = 0;
|
||
|
}
|
||
|
listfiles(flist, SUBMERGED,
|
||
|
state == FLOATING ? FLOATING : BOTTOM);
|
||
|
}
|
||
|
delpath(didx);
|
||
|
}
|
||
|
delfile(popfile(&dlist));
|
||
|
}
|
||
|
}
|
||
|
|
||
|
int main(int argc, char **argv)
|
||
|
{
|
||
|
struct file *flist= nil, **aflist= &flist;
|
||
|
enum depth depth;
|
||
|
char *lsflags;
|
||
|
struct winsize ws;
|
||
|
|
||
|
uid= geteuid();
|
||
|
gid= getegid();
|
||
|
|
||
|
if ((arg0= strrchr(argv[0], '/')) == nil) arg0= argv[0]; else arg0++;
|
||
|
argv++;
|
||
|
|
||
|
if (strcmp(arg0, "ls") != 0) {
|
||
|
char *p= arg0+1;
|
||
|
|
||
|
while (*p != 0) {
|
||
|
if (strchr(arg0flag, *p) != nil) *p += 'A' - 'a';
|
||
|
p++;
|
||
|
}
|
||
|
setflags(arg0+1);
|
||
|
}
|
||
|
while (*argv != nil && (*argv)[0] == '-') {
|
||
|
if ((*argv)[1] == '-' && (*argv)[2] == 0) {
|
||
|
argv++;
|
||
|
break;
|
||
|
}
|
||
|
setflags(*argv++ + 1);
|
||
|
}
|
||
|
|
||
|
istty= isatty(1);
|
||
|
|
||
|
if (istty && (lsflags= getenv("LSOPTS")) != nil) {
|
||
|
if (*lsflags == '-') lsflags++;
|
||
|
setflags(lsflags);
|
||
|
}
|
||
|
|
||
|
if (!present('1') && !present('C') && !present('l')
|
||
|
&& (istty || present('M') || present('X') || present('F'))
|
||
|
) setflags("C");
|
||
|
|
||
|
if (istty) setflags("q");
|
||
|
|
||
|
if (SUPER_ID == 0 || present('a')) setflags("A");
|
||
|
|
||
|
if (present('i')) field|= L_INODE;
|
||
|
if (present('s')) field|= L_BLOCKS;
|
||
|
if (present('M')) field|= L_MODE;
|
||
|
if (present('X')) field|= L_EXTRA | L_MODE;
|
||
|
if (present('t')) field|= L_BYTIME;
|
||
|
if (present('u')) field|= L_ATIME;
|
||
|
if (present('c')) field|= L_CTIME;
|
||
|
if (present('l')) field|= L_MODE | L_LONG;
|
||
|
if (present('g')) field|= L_MODE | L_LONG | L_GROUP;
|
||
|
if (present('F')) field|= L_MARK;
|
||
|
if (present('p')) field|= L_MARKDIR;
|
||
|
if (present('D')) field|= L_TYPE;
|
||
|
if (present('T')) field|= L_MODE | L_LONG | L_LONGTIME;
|
||
|
if (present('d')) field|= L_DIR;
|
||
|
if (present('h')) field|= L_KMG;
|
||
|
if (field & L_LONG) field&= ~L_EXTRA;
|
||
|
|
||
|
#ifdef S_IFLNK
|
||
|
status= present('L') ? stat : lstat;
|
||
|
#endif
|
||
|
|
||
|
if (present('C')) {
|
||
|
int t= istty ? 1 : open("/dev/tty", O_WRONLY);
|
||
|
|
||
|
if (t >= 0 && ioctl(t, TIOCGWINSZ, &ws) == 0 && ws.ws_col > 0)
|
||
|
ncols= ws.ws_col - 1;
|
||
|
|
||
|
if (t != 1 && t != -1) close(t);
|
||
|
}
|
||
|
|
||
|
depth= SURFACE;
|
||
|
|
||
|
if (*argv == nil) {
|
||
|
if (!(field & L_DIR)) depth= SURFACE1;
|
||
|
pushfile(aflist, newfile("."));
|
||
|
} else {
|
||
|
if (argv[1] == nil && !(field & L_DIR)) depth= SURFACE1;
|
||
|
|
||
|
do {
|
||
|
pushfile(aflist, newfile(*argv++));
|
||
|
aflist= &(*aflist)->next;
|
||
|
} while (*argv!=nil);
|
||
|
}
|
||
|
listfiles(flist, depth,
|
||
|
(field & L_DIR) ? BOTTOM : present('R') ? FLOATING : SINKING);
|
||
|
return ex;
|
||
|
}
|