diff options
| author | 2010-01-31 15:07:41 (JST) | |
|---|---|---|
| committer | 2010-02-01 07:08:49 (JST) | |
| commit | 74ebf82229829bbdbe74a4d9a7b6f29d2889dfc8 (patch) | |
| tree | 0283b46ed39d886ff9665d76f0743de708751986 | |
| parent | 89082346d50ec283a16d3127677f30b427781b6a (diff) | |
| download | cgit-74ebf82229829bbdbe74a4d9a7b6f29d2889dfc8.zip cgit-74ebf82229829bbdbe74a4d9a7b6f29d2889dfc8.tar.gz | |
ui-plain: more efficient walk_tree()
Git's read_tree_recursive() already filters out the objects by pathname,
so we only have to compare baselen to the expected. That is, no string
matching is required.
Additionally, if the requested path is a directory, the old code would
walk through all of its immediate children. This is not necessary, this
so we no longer do that.
Signed-off-by: Mark Lodato <lodatom@gmail.com>
| -rw-r--r-- | ui-plain.c | 20 |
1 files changed, 14 insertions, 6 deletions
| @@ -10,7 +10,7 @@ | |||
| 10 | #include "html.h" | 10 | #include "html.h" |
| 11 | #include "ui-shared.h" | 11 | #include "ui-shared.h" |
| 12 | 12 | ||
| 13 | char *match_path; | 13 | int match_baselen; |
| 14 | int match; | 14 | int match; |
| 15 | 15 | ||
| 16 | static void print_object(const unsigned char *sha1, const char *path) | 16 | static void print_object(const unsigned char *sha1, const char *path) |
| @@ -56,13 +56,21 @@ static int walk_tree(const unsigned char *sha1, const char *base, int baselen, | |||
| 56 | const char *pathname, unsigned mode, int stage, | 56 | const char *pathname, unsigned mode, int stage, |
| 57 | void *cbdata) | 57 | void *cbdata) |
| 58 | { | 58 | { |
| 59 | if (S_ISDIR(mode)) | 59 | if (baselen == match_baselen) { |
| 60 | if (S_ISREG(mode)) | ||
| 61 | print_object(sha1, pathname); | ||
| 62 | } | ||
| 63 | else if (S_ISDIR(mode)) | ||
| 60 | return READ_TREE_RECURSIVE; | 64 | return READ_TREE_RECURSIVE; |
| 61 | 65 | ||
| 62 | if (S_ISREG(mode) && !strncmp(base, match_path, baselen) && | 66 | return 0; |
| 63 | !strcmp(pathname, match_path + baselen)) | 67 | } |
| 64 | print_object(sha1, pathname); | ||
| 65 | 68 | ||
| 69 | static int basedir_len(const char *path) | ||
| 70 | { | ||
| 71 | char *p = strrchr(path, '/'); | ||
| 72 | if (p) | ||
| 73 | return p - path + 1; | ||
| 66 | return 0; | 74 | return 0; |
| 67 | } | 75 | } |
| 68 | 76 | ||
| @@ -85,7 +93,7 @@ void cgit_print_plain(struct cgit_context *ctx) | |||
| 85 | html_status(404, "Not found", 0); | 93 | html_status(404, "Not found", 0); |
| 86 | return; | 94 | return; |
| 87 | } | 95 | } |
| 88 | match_path = ctx->qry.path; | 96 | match_baselen = basedir_len(paths[0]); |
| 89 | read_tree_recursive(commit->tree, "", 0, 0, paths, walk_tree, NULL); | 97 | read_tree_recursive(commit->tree, "", 0, 0, paths, walk_tree, NULL); |
| 90 | if (!match) | 98 | if (!match) |
| 91 | html_status(404, "Not found", 0); | 99 | html_status(404, "Not found", 0); |
