aboutsummaryrefslogtreecommitdiff
path: root/dmenu_path.c
blob: 1575f1d193524e7e582ddcf6cef3e0efd6ce7fdd (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
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
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
/* See LICENSE file for copyright and license details. */
#include <dirent.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <sys/stat.h>

#define CACHE ".dmenu_cache"

static int qstrcmp(const void *a, const void *b);
static void die(const char *s);
static void scan(void);
static int uptodate(void);

static char **items = NULL;
static const char *Home, *Path;
static size_t count = 0;

int
main(void) {
	if(!(Home = getenv("HOME")))
		die("no $HOME");
	if(!(Path = getenv("PATH")))
		die("no $PATH");
	if(chdir(Home) < 0)
		die("chdir failed");
	if(uptodate()) {
		execlp("cat", "cat", CACHE, NULL);
		die("exec failed");
	}
	scan();
	return EXIT_SUCCESS;
}

void
die(const char *s) {
	fprintf(stderr, "dmenu_path: %s\n", s);
	exit(EXIT_FAILURE);
}

int
qstrcmp(const void *a, const void *b) {
	return strcmp(*(const char **)a, *(const char **)b);
}

void
scan(void) {
	char buf[PATH_MAX];
	char *dir, *path;
	size_t i;
	struct dirent *ent;
	DIR *dp;
	FILE *cache;

	if(!(path = strdup(Path)))
		die("strdup failed");
	for(dir = strtok(path, ":"); dir; dir = strtok(NULL, ":")) {
		if(!(dp = opendir(dir)))
			continue;
		while((ent = readdir(dp))) {
			snprintf(buf, sizeof buf, "%s/%s", dir, ent->d_name);
			if(ent->d_name[0] == '.' || access(buf, X_OK) < 0)
				continue;
			if(!(items = realloc(items, ++count * sizeof *items)))
				die("malloc failed");
			if(!(items[count-1] = strdup(ent->d_name)))
				die("strdup failed");
		}
		closedir(dp);
	}
	qsort(items, count, sizeof *items, qstrcmp);
	if(!(cache = fopen(CACHE, "w")))
		die("open failed");
	for(i = 0; i < count; i++) {
		if(i > 0 && !strcmp(items[i], items[i-1]))
			continue;
		fprintf(cache,  "%s\n", items[i]);
		fprintf(stdout, "%s\n", items[i]);
	}
	fclose(cache);
	free(path);
}

int
uptodate(void) {
	char *dir, *path;
	time_t mtime;
	struct stat st;

	if(stat(CACHE, &st) < 0)
		return 0;
	mtime = st.st_mtime;
	if(!(path = strdup(Path)))
		die("strdup failed");
	for(dir = strtok(path, ":"); dir; dir = strtok(NULL, ":"))
		if(!stat(dir, &st) && st.st_mtime > mtime)
			return 0;
	free(path);
	return 1;
}