Browse Source

Non-compiling version of ccan_depends by default.
Note that this gets a superset of dependencies, eg:

rusty@vivaldi$ tools/ccan_depends ccan/check_type
ccan/build_assert
rusty@vivaldi$ tools/ccan_depends --compile ccan/check_type
rusty@vivaldi$

Rusty Russell 17 years ago
parent
commit
d4ace4a572
3 changed files with 102 additions and 5 deletions
  1. 12 2
      tools/ccan_depends.c
  2. 86 3
      tools/depends.c
  3. 4 0
      tools/tools.h

+ 12 - 2
tools/ccan_depends.c

@@ -9,12 +9,22 @@ int main(int argc, char *argv[])
 {
 	char **deps;
 	unsigned int i;
+	bool compile = false;
 
+	if (argv[1] && streq(argv[1], "--compile")) {
+		argv++;
+		argc--;
+		compile = true;
+	}
 	if (argc != 2)
-		errx(1, "Usage: ccan_depends <dir>\n"
+		errx(1, "Usage: ccan_depends [--compile] <dir>\n"
 		        "Spits out all the ccan dependencies (recursively)");
 
-	deps = get_deps(talloc_autofree_context(), argv[1]);
+	if (compile)
+		deps = get_deps(talloc_autofree_context(), argv[1]);
+	else
+		deps = get_safe_ccan_deps(talloc_autofree_context(), argv[1]);
+
 	for (i = 0; deps[i]; i++)
 		if (strstarts(deps[i], "ccan/"))
 			printf("%s\n", deps[i]);

+ 86 - 3
tools/depends.c

@@ -62,6 +62,76 @@ static char **get_one_deps(const void *ctx, const char *dir, unsigned int *num)
 	return deps;
 }
 
+/* Make copy of src, replacing "from" with "to". */
+static char *replace(const void *ctx, const char *src,
+		     const char *from, const char *to)
+{
+	char *ret = talloc_strdup(ctx, "");
+	unsigned int rlen, len, add;
+
+	rlen = len = 0;
+	for (;;) {
+		const char *next = strstr(src+len, from);
+		if (!next)
+			add = strlen(src+len) + 1;
+		else
+			add = next - (src+len);
+
+		ret = talloc_realloc(ctx, ret, char, rlen + add + strlen(to)+1);
+		memcpy(ret+rlen, src+len, add);
+		if (!next)
+			return ret;
+		len += add;
+		rlen += add;
+		strcpy(ret+rlen, to);
+		rlen += strlen(to);
+		len += strlen(from);
+	}
+}
+
+/* This is a terrible hack.  We scan for ccan/ strings. */
+static char **get_one_safe_deps(const void *ctx,
+				const char *dir, unsigned int *num)
+{
+	char **deps, **lines, *raw, *fname;
+	unsigned int i, n = 0;
+
+	fname = talloc_asprintf(ctx, "%s/_info.c", dir);
+	raw = grab_file(fname, fname, NULL);
+	if (!raw)
+		errx(1, "Could not open %s", fname);
+
+	/* Replace \n by actual line breaks, and split it. */
+	lines = strsplit(raw, replace(raw, raw, "\\n", "\n"), "\n", &n);
+
+	deps = talloc_array(ctx, char *, n+1);
+
+	for (n = i = 0; lines[i]; i++) {
+		char *str;
+		unsigned int len;
+
+		/* Start of line, or after ". */
+		if (strstarts(lines[i], "ccan/"))
+			str = lines[i];
+		else {
+			str = strstr(lines[i], "\"ccan/");
+			if (!str)
+				continue;
+			str++;
+		}
+		
+		len = strspn(str, "/abcdefghijklmnopqrstuvxwyz12345678980_");
+		if (len == 5)
+			continue;
+		deps[n++] = talloc_strndup(deps, str, len);
+	}
+	deps[n] = NULL;
+	talloc_free(fname);
+	if (num)
+		*num = n;
+	return deps;
+}
+
 static bool have_dep(char **deps, unsigned int num, const char *dep)
 {
 	unsigned int i;
@@ -73,12 +143,14 @@ static bool have_dep(char **deps, unsigned int num, const char *dep)
 }
 
 /* Gets all the dependencies, recursively. */
-char **get_deps(const void *ctx, const char *dir)
+static char **
+get_all_deps(const void *ctx, const char *dir,
+	     char **(*get_one)(const void *, const char *, unsigned int *))
 {
 	char **deps;
 	unsigned int i, num;
 
-	deps = get_one_deps(ctx, dir, &num);
+	deps = get_one(ctx, dir, &num);
 	for (i = 0; i < num; i++) {
 		char **newdeps;
 		unsigned int j, newnum;
@@ -86,7 +158,7 @@ char **get_deps(const void *ctx, const char *dir)
 		if (!strstarts(deps[i], "ccan/"))
 			continue;
 
-		newdeps = get_one_deps(ctx, deps[i], &newnum);
+		newdeps = get_one(ctx, deps[i], &newnum);
 
 		/* Should be short, so brute-force out dups. */
 		for (j = 0; j < newnum; j++) {
@@ -100,3 +172,14 @@ char **get_deps(const void *ctx, const char *dir)
 	}
 	return deps;
 }
+
+char **get_deps(const void *ctx, const char *dir)
+{
+	return get_all_deps(ctx, dir, get_one_deps);
+}
+
+char **get_safe_ccan_deps(const void *ctx, const char *dir)
+{
+	return get_all_deps(ctx, dir, get_one_safe_deps);
+}
+	

+ 4 - 0
tools/tools.h

@@ -3,7 +3,11 @@
 
 #define CFLAGS "-O3 -Wall -Wundef -Wstrict-prototypes -Wold-style-definition -Wmissing-prototypes -Wmissing-declarations -Werror -Iccan/ -I."
 
+/* This actually compiles and runs the _info.c file to get dependencies. */
 char **get_deps(const void *ctx, const char *dir);
 
+/* This is safer: just looks for ccan/ strings in _info.c */
+char **get_safe_ccan_deps(const void *ctx, const char *dir);
+
 #endif /* CCAN_TOOLS_H */