ls-files.c 6.0 KB
Newer Older
1 2 3 4 5 6 7 8
/*
 * This merges the file listing in the directory cache index
 * with the actual working directory list, and shows different
 * combinations of the two.
 *
 * Copyright (C) Linus Torvalds, 2005
 */
#include <dirent.h>
9
#include <fnmatch.h>
10 11 12 13 14 15 16

#include "cache.h"

static int show_deleted = 0;
static int show_cached = 0;
static int show_others = 0;
static int show_ignored = 0;
17
static int show_stage = 0;
18
static int show_unmerged = 0;
19
static int line_terminator = '\n';
20

21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84
static int nr_excludes;
static const char **excludes;
static int excludes_alloc;

static void add_exclude(const char *string)
{
	if (nr_excludes == excludes_alloc) {
		excludes_alloc = alloc_nr(excludes_alloc);
		excludes = realloc(excludes, excludes_alloc*sizeof(char *));
	}
	excludes[nr_excludes++] = string;
}

static void add_excludes_from_file(const char *fname)
{
	int fd, i;
	long size;
	char *buf, *entry;

	fd = open(fname, O_RDONLY);
	if (fd < 0)
		goto err;
	size = lseek(fd, 0, SEEK_END);
	if (size < 0)
		goto err;
	lseek(fd, 0, SEEK_SET);
	if (size == 0) {
		close(fd);
		return;
	}
	buf = xmalloc(size);
	if (read(fd, buf, size) != size)
		goto err;
	close(fd);

	entry = buf;
	for (i = 0; i < size; i++) {
		if (buf[i] == '\n') {
			if (entry != buf + i) {
				buf[i] = 0;
				add_exclude(entry);
			}
			entry = buf + i + 1;
		}
	}
	return;

err:	perror(fname);
	exit(1);
}

static int excluded(const char *pathname)
{
	int i;
	if (nr_excludes) {
		const char *basename = strrchr(pathname, '/');
		basename = (basename) ? basename+1 : pathname;
		for (i = 0; i < nr_excludes; i++)
			if (fnmatch(excludes[i], basename, 0) == 0)
				return 1;
	}
	return 0;
}

85 86 87 88 89 90 91 92 93 94 95 96 97
static const char **dir;
static int nr_dir;
static int dir_alloc;

static void add_name(const char *pathname, int len)
{
	char *name;

	if (cache_name_pos(pathname, len) >= 0)
		return;

	if (nr_dir == dir_alloc) {
		dir_alloc = alloc_nr(dir_alloc);
98
		dir = xrealloc(dir, dir_alloc*sizeof(char *));
99
	}
100
	name = xmalloc(len + 1);
101 102 103 104 105 106 107 108 109 110
	memcpy(name, pathname, len + 1);
	dir[nr_dir++] = name;
}

/*
 * Read a directory tree. We currently ignore anything but
 * directories and regular files. That's because git doesn't
 * handle them at all yet. Maybe that will change some day.
 *
 * Also, we currently ignore all names starting with a dot.
I
Ingo Molnar 已提交
111
 * That likely will not change.
112 113 114 115 116 117 118 119 120 121 122 123 124 125 126
 */
static void read_directory(const char *path, const char *base, int baselen)
{
	DIR *dir = opendir(path);

	if (dir) {
		struct dirent *de;
		char fullname[MAXPATHLEN + 1];
		memcpy(fullname, base, baselen);

		while ((de = readdir(dir)) != NULL) {
			int len;

			if (de->d_name[0] == '.')
				continue;
127 128
			if (excluded(de->d_name) != show_ignored)
				continue;
129 130 131
			len = strlen(de->d_name);
			memcpy(fullname + baselen, de->d_name, len+1);

132
			switch (DTYPE(de)) {
133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170
			struct stat st;
			default:
				continue;
			case DT_UNKNOWN:
				if (lstat(fullname, &st))
					continue;
				if (S_ISREG(st.st_mode))
					break;
				if (!S_ISDIR(st.st_mode))
					continue;
				/* fallthrough */
			case DT_DIR:
				memcpy(fullname + baselen + len, "/", 2);
				read_directory(fullname, fullname, baselen + len + 1);
				continue;
			case DT_REG:
				break;
			}
			add_name(fullname, baselen + len);
		}
		closedir(dir);
	}
}

static int cmp_name(const void *p1, const void *p2)
{
	const char *n1 = *(const char **)p1;
	const char *n2 = *(const char **)p2;
	int l1 = strlen(n1), l2 = strlen(n2);

	return cache_name_compare(n1, l1, n2, l2);
}

static void show_files(void)
{
	int i;

	/* For cached/deleted files we don't need to even do the readdir */
171
	if (show_others) {
172 173 174
		read_directory(".", "", 0);
		qsort(dir, nr_dir, sizeof(char *), cmp_name);
		for (i = 0; i < nr_dir; i++)
175
			printf("%s%c", dir[i], line_terminator);
176
	}
177
	if (show_cached | show_stage) {
178 179
		for (i = 0; i < active_nr; i++) {
			struct cache_entry *ce = active_cache[i];
180 181
			if (excluded(ce->name) != show_ignored)
				continue;
182 183
			if (show_unmerged && !ce_stage(ce))
				continue;
184 185 186 187 188 189 190 191 192 193
			if (!show_stage)
				printf("%s%c", ce->name, line_terminator);
			else
				printf(/* "%06o %s %d %10d %s%c", */
				       "%06o %s %d %s%c",
				       ntohl(ce->ce_mode),
				       sha1_to_hex(ce->sha1),
				       ce_stage(ce),
				       /* ntohl(ce->ce_size), */
				       ce->name, line_terminator); 
194 195 196 197 198 199
		}
	}
	if (show_deleted) {
		for (i = 0; i < active_nr; i++) {
			struct cache_entry *ce = active_cache[i];
			struct stat st;
200 201
			if (excluded(ce->name) != show_ignored)
				continue;
202
			if (!lstat(ce->name, &st))
203
				continue;
204
			printf("%s%c", ce->name, line_terminator);
205 206 207 208
		}
	}
}

209 210
static const char *ls_files_usage =
	"ls-files [-z] (--[cached|deleted|others|stage|unmerged])* "
211 212
	"[ --ignored [--exclude=<pattern>] [--exclude-from=<file>) ]";

213 214 215 216 217 218 219
int main(int argc, char **argv)
{
	int i;

	for (i = 1; i < argc; i++) {
		char *arg = argv[i];

220 221
		if (!strcmp(arg, "-z")) {
			line_terminator = 0;
222
		} else if (!strcmp(arg, "-c") || !strcmp(arg, "--cached")) {
223
			show_cached = 1;
224
		} else if (!strcmp(arg, "-d") || !strcmp(arg, "--deleted")) {
225
			show_deleted = 1;
226
		} else if (!strcmp(arg, "-o") || !strcmp(arg, "--others")) {
227
			show_others = 1;
228
		} else if (!strcmp(arg, "-i") || !strcmp(arg, "--ignored")) {
229
			show_ignored = 1;
230
		} else if (!strcmp(arg, "-s") || !strcmp(arg, "--stage")) {
231
			show_stage = 1;
232
		} else if (!strcmp(arg, "-u") || !strcmp(arg, "--unmerged")) {
233 234 235
			// There's no point in showing unmerged unless you also show the stage information
			show_stage = 1;
			show_unmerged = 1;
236
		} else if (!strcmp(arg, "-x") && i+1 < argc) {
237
			add_exclude(argv[++i]);
238
		} else if (!strncmp(arg, "--exclude=", 10)) {
239
			add_exclude(arg+10);
240
		} else if (!strcmp(arg, "-X") && i+1 < argc) {
241
			add_excludes_from_file(argv[++i]);
242
		} else if (!strncmp(arg, "--exclude-from=", 15)) {
243
			add_excludes_from_file(arg+15);
244
		} else
245
			usage(ls_files_usage);
246 247 248 249 250
	}

	if (show_ignored && !nr_excludes) {
		fprintf(stderr, "%s: --ignored needs some exclude pattern\n", argv[0]);
		exit(1);
251 252 253
	}

	/* With no flags, we default to showing the cached files */
254
	if (!(show_stage | show_deleted | show_others | show_unmerged))
255 256 257 258 259 260
		show_cached = 1;

	read_cache();
	show_files();
	return 0;
}