1
0
mirror of git://git.suckless.org/sbase synced 2025-01-26 17:34:40 +00:00
sbase/ls.c

302 lines
5.7 KiB
C
Raw Normal View History

2011-05-26 03:01:20 +00:00
/* See LICENSE file for copyright and license details. */
#include <dirent.h>
#include <errno.h>
#include <grp.h>
#include <pwd.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/stat.h>
2011-05-26 03:01:20 +00:00
#include <time.h>
#include <unistd.h>
2011-05-26 03:01:20 +00:00
#include "util.h"
typedef struct {
char *name;
mode_t mode;
nlink_t nlink;
uid_t uid;
gid_t gid;
off_t size;
time_t mtime;
2013-09-27 14:32:50 +00:00
ino_t ino;
2011-05-26 03:01:20 +00:00
} Entry;
2013-07-18 19:14:53 +00:00
static int entcmp(const void *, const void *);
static void ls(Entry *);
2011-05-26 03:01:20 +00:00
static void lsdir(const char *);
static void mkent(Entry *, char *, int);
2011-05-26 03:01:20 +00:00
static void output(Entry *);
static int aflag = 0;
static int dflag = 0;
static int Fflag = 0;
static int hflag = 0;
static int iflag = 0;
static int lflag = 0;
static int rflag = 0;
static int tflag = 0;
static int Uflag = 0;
static int first = 1;
static int many;
2011-05-26 03:01:20 +00:00
2013-06-14 18:20:47 +00:00
static void
usage(void)
{
eprintf("usage: %s [-1adFhilrtU] [FILE...]\n", argv0);
2013-06-14 18:20:47 +00:00
}
2011-05-26 03:01:20 +00:00
int
main(int argc, char *argv[])
{
2013-07-18 19:14:53 +00:00
int i;
2011-06-16 00:13:46 +00:00
Entry *ents;
2011-05-26 03:01:20 +00:00
2013-06-14 18:20:47 +00:00
ARGBEGIN {
2014-07-05 14:35:34 +00:00
case '1':
/* ignore */
break;
2013-06-14 18:20:47 +00:00
case 'a':
aflag = 1;
2013-06-14 18:20:47 +00:00
break;
case 'd':
dflag = 1;
2013-06-14 18:20:47 +00:00
break;
2014-02-17 14:41:09 +00:00
case 'F':
Fflag = 1;
2014-02-17 14:41:09 +00:00
break;
case 'h':
hflag = 1;
break;
2013-09-27 14:32:50 +00:00
case 'i':
iflag = 1;
2013-09-27 14:32:50 +00:00
break;
2013-06-14 18:20:47 +00:00
case 'l':
lflag = 1;
2013-06-14 18:20:47 +00:00
break;
case 'r':
rflag = 1;
break;
2013-06-14 18:20:47 +00:00
case 't':
tflag = 1;
2013-06-14 18:20:47 +00:00
break;
2013-07-18 19:14:53 +00:00
case 'U':
Uflag = 1;
2013-07-18 19:14:53 +00:00
break;
2013-06-14 18:20:47 +00:00
default:
usage();
} ARGEND;
many = (argc > 1);
if (argc == 0)
2013-07-18 19:14:53 +00:00
*--argv = ".", argc++;
2013-06-14 18:20:47 +00:00
ents = emalloc(argc * sizeof(*ents));
for (i = 0; i < argc; i++)
mkent(&ents[i], argv[i], 1);
2013-07-18 19:14:53 +00:00
qsort(ents, argc, sizeof *ents, entcmp);
for (i = 0; i < argc; i++)
ls(&ents[rflag ? argc-i-1 : i]);
2013-06-14 18:20:47 +00:00
2014-10-02 22:46:04 +00:00
return 0;
2011-05-26 03:01:20 +00:00
}
static int
2013-07-18 19:14:53 +00:00
entcmp(const void *va, const void *vb)
2011-05-26 03:01:20 +00:00
{
2013-07-18 19:14:53 +00:00
const Entry *a = va, *b = vb;
if (tflag)
return b->mtime - a->mtime;
2011-05-26 03:01:20 +00:00
else
return strcmp(a->name, b->name);
2011-05-26 03:01:20 +00:00
}
static void
2013-07-18 19:14:53 +00:00
ls(Entry *ent)
2011-05-26 03:01:20 +00:00
{
if (S_ISDIR(ent->mode) && !dflag) {
2013-07-18 19:14:53 +00:00
lsdir(ent->name);
} else {
output(ent);
}
2011-05-26 03:01:20 +00:00
}
static void
2011-05-26 03:01:20 +00:00
lsdir(const char *path)
{
2011-05-26 03:17:06 +00:00
char *cwd, *p;
2011-05-26 03:01:20 +00:00
long i, n = 0;
struct dirent *d;
DIR *dp;
2013-07-18 19:14:53 +00:00
Entry ent, *ents = NULL;
size_t sz;
2011-05-26 03:01:20 +00:00
2011-05-26 03:17:06 +00:00
cwd = agetcwd();
if (!(dp = opendir(path)))
2011-05-26 03:01:20 +00:00
eprintf("opendir %s:", path);
2014-11-19 19:59:37 +00:00
if (chdir(path) < 0)
2011-05-26 03:01:20 +00:00
eprintf("chdir %s:", path);
if (many) {
if (!first)
2011-05-26 04:03:37 +00:00
putchar('\n');
2011-05-26 03:01:20 +00:00
printf("%s:\n", path);
first = 0;
2011-05-26 04:03:37 +00:00
}
2013-07-18 19:14:53 +00:00
while ((d = readdir(dp))) {
if (d->d_name[0] == '.' && !aflag)
2013-07-18 19:14:53 +00:00
continue;
if (Uflag){
2014-02-17 14:41:09 +00:00
mkent(&ent, d->d_name, Fflag || lflag || iflag);
2013-07-18 19:14:53 +00:00
output(&ent);
} else {
ents = erealloc(ents, ++n * sizeof *ents);
p = emalloc((sz = strlen(d->d_name)+1));
memcpy(p, d->d_name, sz);
2014-02-17 14:41:09 +00:00
mkent(&ents[n-1], p, tflag || Fflag || lflag || iflag);
2013-07-18 19:14:53 +00:00
}
}
closedir(dp);
if (!Uflag){
2013-07-18 19:14:53 +00:00
qsort(ents, n, sizeof *ents, entcmp);
for (i = 0; i < n; i++) {
output(&ents[rflag ? n-i-1 : i]);
free(ents[rflag ? n-i-1 : i].name);
2013-07-18 19:14:53 +00:00
}
2011-05-26 03:01:20 +00:00
}
2014-11-19 19:59:37 +00:00
if (chdir(cwd) < 0)
2011-05-26 03:17:06 +00:00
eprintf("chdir %s:", cwd);
2011-05-26 03:01:20 +00:00
free(ents);
2011-05-26 03:17:06 +00:00
free(cwd);
2011-05-26 03:01:20 +00:00
}
static void
mkent(Entry *ent, char *path, int dostat)
2011-05-26 03:01:20 +00:00
{
struct stat st;
2013-07-18 19:14:53 +00:00
ent->name = path;
if (!dostat)
2013-07-18 19:14:53 +00:00
return;
2014-11-19 19:59:37 +00:00
if (lstat(path, &st) < 0)
2011-05-26 03:01:20 +00:00
eprintf("lstat %s:", path);
ent->mode = st.st_mode;
ent->nlink = st.st_nlink;
ent->uid = st.st_uid;
ent->gid = st.st_gid;
ent->size = st.st_size;
ent->mtime = st.st_mtime;
2013-09-27 14:32:50 +00:00
ent->ino = st.st_ino;
2011-05-26 03:01:20 +00:00
}
static char *
2014-02-17 14:41:09 +00:00
indicator(mode_t mode)
{
if (!Fflag)
2014-02-17 14:41:09 +00:00
return "";
if (S_ISLNK(mode))
2014-02-17 14:41:09 +00:00
return "@";
else if (S_ISDIR(mode))
2014-02-17 14:41:09 +00:00
return "/";
else if (S_ISFIFO(mode))
2014-02-17 14:41:09 +00:00
return "|";
else if (S_ISSOCK(mode))
2014-02-17 14:41:09 +00:00
return "=";
else if (mode & S_IXUSR ||
mode & S_IXGRP ||
mode & S_IXOTH)
2014-02-17 14:41:09 +00:00
return "*";
else
return "";
}
static void
2011-05-26 03:01:20 +00:00
output(Entry *ent)
{
2011-05-27 22:56:43 +00:00
char buf[BUFSIZ], *fmt;
char mode[] = "----------";
2011-06-04 11:40:05 +00:00
ssize_t len;
2011-05-26 03:01:20 +00:00
struct group *gr;
struct passwd *pw;
char pwname[_SC_LOGIN_NAME_MAX];
char grname[_SC_LOGIN_NAME_MAX];
2014-02-17 14:41:09 +00:00
Entry entlnk;
2011-05-26 03:01:20 +00:00
2013-09-27 14:32:50 +00:00
if (iflag)
printf("%lu ", (unsigned long)ent->ino);
if (!lflag) {
2014-02-17 14:41:09 +00:00
printf("%s%s\n", ent->name, indicator(ent->mode));
2011-05-26 03:01:20 +00:00
return;
}
if (S_ISREG(ent->mode))
2011-05-26 03:01:20 +00:00
mode[0] = '-';
else if (S_ISBLK(ent->mode))
2011-05-26 03:01:20 +00:00
mode[0] = 'b';
else if (S_ISCHR(ent->mode))
2011-05-26 03:01:20 +00:00
mode[0] = 'c';
else if (S_ISDIR(ent->mode))
2011-05-26 03:01:20 +00:00
mode[0] = 'd';
else if (S_ISFIFO(ent->mode))
2011-05-26 03:01:20 +00:00
mode[0] = 'p';
else if (S_ISLNK(ent->mode))
2011-05-26 03:01:20 +00:00
mode[0] = 'l';
else if (S_ISSOCK(ent->mode))
2011-05-26 03:01:20 +00:00
mode[0] = 's';
else
mode[0] = '?';
if (ent->mode & S_IRUSR) mode[1] = 'r';
if (ent->mode & S_IWUSR) mode[2] = 'w';
if (ent->mode & S_IXUSR) mode[3] = 'x';
if (ent->mode & S_IRGRP) mode[4] = 'r';
if (ent->mode & S_IWGRP) mode[5] = 'w';
if (ent->mode & S_IXGRP) mode[6] = 'x';
if (ent->mode & S_IROTH) mode[7] = 'r';
if (ent->mode & S_IWOTH) mode[8] = 'w';
if (ent->mode & S_IXOTH) mode[9] = 'x';
2011-05-27 22:56:43 +00:00
if (ent->mode & S_ISUID) mode[3] = (mode[3] == 'x') ? 's' : 'S';
if (ent->mode & S_ISGID) mode[6] = (mode[6] == 'x') ? 's' : 'S';
if (ent->mode & S_ISVTX) mode[9] = (mode[9] == 'x') ? 't' : 'T';
2011-05-26 03:01:20 +00:00
errno = 0;
pw = getpwuid(ent->uid);
if (errno || !pw)
snprintf(pwname, sizeof(pwname), "%d", ent->uid);
else
snprintf(pwname, sizeof(pwname), "%s", pw->pw_name);
2011-05-26 03:01:20 +00:00
errno = 0;
gr = getgrgid(ent->gid);
if (errno || !gr)
snprintf(grname, sizeof(grname), "%d", ent->gid);
else
snprintf(grname, sizeof(grname), "%s", gr->gr_name);
2011-05-26 03:01:20 +00:00
if (time(NULL) > ent->mtime + (180*24*60*60)) /* 6 months ago? */
2011-06-04 11:40:05 +00:00
fmt = "%b %d %Y";
2011-05-26 03:01:20 +00:00
else
fmt = "%b %d %H:%M";
strftime(buf, sizeof buf, fmt, localtime(&ent->mtime));
printf("%s %4ld %-8.8s %-8.8s ", mode, (long)ent->nlink, pwname, grname);
if (hflag)
printf("%10s ", humansize((unsigned long)ent->size));
else
printf("%10lu ", (unsigned long)ent->size);
printf("%s %s%s", buf, ent->name, indicator(ent->mode));
if (S_ISLNK(ent->mode)) {
2014-11-19 19:59:37 +00:00
if ((len = readlink(ent->name, buf, sizeof buf)) < 0)
2011-06-04 11:40:05 +00:00
eprintf("readlink %s:", ent->name);
buf[len] = '\0';
2014-02-17 14:41:09 +00:00
mkent(&entlnk, buf, Fflag);
printf(" -> %s%s", buf, indicator(entlnk.mode));
2011-06-04 11:40:05 +00:00
}
putchar('\n');
2011-05-26 03:01:20 +00:00
}