2005-04-18 20:39:48 +02:00
|
|
|
#include "tree.h"
|
|
|
|
#include "blob.h"
|
2005-09-05 08:03:51 +02:00
|
|
|
#include "commit.h"
|
|
|
|
#include "tag.h"
|
2005-04-18 20:39:48 +02:00
|
|
|
#include "cache.h"
|
|
|
|
#include <stdlib.h>
|
|
|
|
|
|
|
|
const char *tree_type = "tree";
|
|
|
|
|
2005-04-23 01:42:37 +02:00
|
|
|
static int read_one_entry(unsigned char *sha1, const char *base, int baselen, const char *pathname, unsigned mode, int stage)
|
|
|
|
{
|
|
|
|
int len = strlen(pathname);
|
|
|
|
unsigned int size = cache_entry_size(baselen + len);
|
2005-04-26 21:00:58 +02:00
|
|
|
struct cache_entry *ce = xmalloc(size);
|
2005-04-23 01:42:37 +02:00
|
|
|
|
|
|
|
memset(ce, 0, size);
|
|
|
|
|
|
|
|
ce->ce_mode = create_ce_mode(mode);
|
|
|
|
ce->ce_flags = create_ce_flags(baselen + len, stage);
|
|
|
|
memcpy(ce->name, base, baselen);
|
|
|
|
memcpy(ce->name + baselen, pathname, len+1);
|
|
|
|
memcpy(ce->sha1, sha1, 20);
|
2005-06-25 11:25:29 +02:00
|
|
|
return add_cache_entry(ce, ADD_CACHE_OK_TO_ADD|ADD_CACHE_SKIP_DFCHECK);
|
2005-04-23 01:42:37 +02:00
|
|
|
}
|
|
|
|
|
2005-07-14 20:39:27 +02:00
|
|
|
static int match_tree_entry(const char *base, int baselen, const char *path, unsigned int mode, const char **paths)
|
2005-07-14 20:26:31 +02:00
|
|
|
{
|
2005-07-14 20:39:27 +02:00
|
|
|
const char *match;
|
2005-07-14 20:26:31 +02:00
|
|
|
int pathlen;
|
|
|
|
|
|
|
|
if (!paths)
|
|
|
|
return 1;
|
|
|
|
pathlen = strlen(path);
|
|
|
|
while ((match = *paths++) != NULL) {
|
|
|
|
int matchlen = strlen(match);
|
|
|
|
|
|
|
|
if (baselen >= matchlen) {
|
|
|
|
/* If it doesn't match, move along... */
|
|
|
|
if (strncmp(base, match, matchlen))
|
|
|
|
continue;
|
|
|
|
/* The base is a subdirectory of a path which was specified. */
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Does the base match? */
|
|
|
|
if (strncmp(base, match, baselen))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
match += baselen;
|
|
|
|
matchlen -= baselen;
|
|
|
|
|
|
|
|
if (pathlen > matchlen)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (matchlen > pathlen) {
|
|
|
|
if (match[pathlen] != '/')
|
|
|
|
continue;
|
|
|
|
if (!S_ISDIR(mode))
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (strncmp(path, match, pathlen))
|
|
|
|
continue;
|
2005-07-14 20:39:27 +02:00
|
|
|
|
|
|
|
return 1;
|
2005-07-14 20:26:31 +02:00
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2005-04-23 01:42:37 +02:00
|
|
|
static int read_tree_recursive(void *buffer, unsigned long size,
|
2005-07-14 20:26:31 +02:00
|
|
|
const char *base, int baselen,
|
2005-07-14 20:39:27 +02:00
|
|
|
int stage, const char **match)
|
2005-04-23 01:42:37 +02:00
|
|
|
{
|
|
|
|
while (size) {
|
|
|
|
int len = strlen(buffer)+1;
|
|
|
|
unsigned char *sha1 = buffer + len;
|
|
|
|
char *path = strchr(buffer, ' ')+1;
|
|
|
|
unsigned int mode;
|
|
|
|
|
|
|
|
if (size < len + 20 || sscanf(buffer, "%o", &mode) != 1)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
buffer = sha1 + 20;
|
|
|
|
size -= len + 20;
|
|
|
|
|
2005-07-14 20:26:31 +02:00
|
|
|
if (!match_tree_entry(base, baselen, path, mode, match))
|
|
|
|
continue;
|
|
|
|
|
2005-04-23 01:42:37 +02:00
|
|
|
if (S_ISDIR(mode)) {
|
|
|
|
int retval;
|
|
|
|
int pathlen = strlen(path);
|
2005-04-27 00:00:01 +02:00
|
|
|
char *newbase;
|
2005-04-23 01:42:37 +02:00
|
|
|
void *eltbuf;
|
|
|
|
char elttype[20];
|
|
|
|
unsigned long eltsize;
|
|
|
|
|
|
|
|
eltbuf = read_sha1_file(sha1, elttype, &eltsize);
|
2005-04-27 00:00:01 +02:00
|
|
|
if (!eltbuf || strcmp(elttype, "tree")) {
|
|
|
|
if (eltbuf) free(eltbuf);
|
2005-04-23 01:42:37 +02:00
|
|
|
return -1;
|
2005-04-27 00:00:01 +02:00
|
|
|
}
|
|
|
|
newbase = xmalloc(baselen + 1 + pathlen);
|
2005-04-23 01:42:37 +02:00
|
|
|
memcpy(newbase, base, baselen);
|
|
|
|
memcpy(newbase + baselen, path, pathlen);
|
|
|
|
newbase[baselen + pathlen] = '/';
|
|
|
|
retval = read_tree_recursive(eltbuf, eltsize,
|
|
|
|
newbase,
|
2005-07-14 20:26:31 +02:00
|
|
|
baselen + pathlen + 1,
|
|
|
|
stage, match);
|
2005-04-23 01:42:37 +02:00
|
|
|
free(eltbuf);
|
|
|
|
free(newbase);
|
|
|
|
if (retval)
|
|
|
|
return -1;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (read_one_entry(sha1, base, baselen, path, mode, stage) < 0)
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2005-07-14 20:39:27 +02:00
|
|
|
int read_tree(void *buffer, unsigned long size, int stage, const char **match)
|
2005-04-23 01:42:37 +02:00
|
|
|
{
|
2005-07-14 20:26:31 +02:00
|
|
|
return read_tree_recursive(buffer, size, "", 0, stage, match);
|
2005-04-23 01:42:37 +02:00
|
|
|
}
|
|
|
|
|
2005-06-03 17:05:39 +02:00
|
|
|
struct tree *lookup_tree(const unsigned char *sha1)
|
2005-04-18 20:39:48 +02:00
|
|
|
{
|
|
|
|
struct object *obj = lookup_object(sha1);
|
|
|
|
if (!obj) {
|
2005-04-26 21:00:58 +02:00
|
|
|
struct tree *ret = xmalloc(sizeof(struct tree));
|
2005-04-18 20:39:48 +02:00
|
|
|
memset(ret, 0, sizeof(struct tree));
|
|
|
|
created_object(sha1, &ret->object);
|
2005-04-24 23:17:13 +02:00
|
|
|
ret->object.type = tree_type;
|
2005-04-18 20:39:48 +02:00
|
|
|
return ret;
|
|
|
|
}
|
2005-05-20 22:59:17 +02:00
|
|
|
if (!obj->type)
|
|
|
|
obj->type = tree_type;
|
2005-04-24 23:22:09 +02:00
|
|
|
if (obj->type != tree_type) {
|
2005-04-18 20:39:48 +02:00
|
|
|
error("Object %s is a %s, not a tree",
|
|
|
|
sha1_to_hex(sha1), obj->type);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
return (struct tree *) obj;
|
|
|
|
}
|
|
|
|
|
2005-05-06 19:48:34 +02:00
|
|
|
int parse_tree_buffer(struct tree *item, void *buffer, unsigned long size)
|
2005-04-18 20:39:48 +02:00
|
|
|
{
|
2005-05-06 19:48:34 +02:00
|
|
|
void *bufptr = buffer;
|
2005-04-24 03:47:23 +02:00
|
|
|
struct tree_entry_list **list_p;
|
2005-05-06 19:48:34 +02:00
|
|
|
|
2005-04-18 20:39:48 +02:00
|
|
|
if (item->object.parsed)
|
|
|
|
return 0;
|
|
|
|
item->object.parsed = 1;
|
2005-04-24 03:47:23 +02:00
|
|
|
list_p = &item->entries;
|
2005-04-18 20:39:48 +02:00
|
|
|
while (size) {
|
|
|
|
struct object *obj;
|
2005-04-24 03:47:23 +02:00
|
|
|
struct tree_entry_list *entry;
|
2005-04-18 20:39:48 +02:00
|
|
|
int len = 1+strlen(bufptr);
|
|
|
|
unsigned char *file_sha1 = bufptr + len;
|
|
|
|
char *path = strchr(bufptr, ' ');
|
|
|
|
unsigned int mode;
|
|
|
|
if (size < len + 20 || !path ||
|
2005-05-06 19:48:34 +02:00
|
|
|
sscanf(bufptr, "%o", &mode) != 1)
|
2005-04-18 20:39:48 +02:00
|
|
|
return -1;
|
|
|
|
|
2005-04-26 21:00:58 +02:00
|
|
|
entry = xmalloc(sizeof(struct tree_entry_list));
|
2005-04-24 03:47:23 +02:00
|
|
|
entry->name = strdup(path + 1);
|
2005-05-06 01:18:48 +02:00
|
|
|
entry->directory = S_ISDIR(mode) != 0;
|
|
|
|
entry->executable = (mode & S_IXUSR) != 0;
|
|
|
|
entry->symlink = S_ISLNK(mode) != 0;
|
2005-07-28 01:08:43 +02:00
|
|
|
entry->zeropad = *(char *)bufptr == '0';
|
2005-05-06 01:18:48 +02:00
|
|
|
entry->mode = mode;
|
2005-04-24 03:47:23 +02:00
|
|
|
entry->next = NULL;
|
|
|
|
|
2005-04-18 20:39:48 +02:00
|
|
|
bufptr += len + 20;
|
|
|
|
size -= len + 20;
|
|
|
|
|
2005-04-24 03:47:23 +02:00
|
|
|
if (entry->directory) {
|
|
|
|
entry->item.tree = lookup_tree(file_sha1);
|
|
|
|
obj = &entry->item.tree->object;
|
2005-04-18 20:39:48 +02:00
|
|
|
} else {
|
2005-04-24 03:47:23 +02:00
|
|
|
entry->item.blob = lookup_blob(file_sha1);
|
|
|
|
obj = &entry->item.blob->object;
|
2005-04-18 20:39:48 +02:00
|
|
|
}
|
2005-04-24 23:31:57 +02:00
|
|
|
if (obj)
|
|
|
|
add_ref(&item->object, obj);
|
[PATCH] Rewrite ls-tree to behave more like "/bin/ls -a"
This is a complete rewrite of ls-tree to make it behave more
like what "/bin/ls -a" does in the current working directory.
Namely, the changes are:
- Unlike the old ls-tree behaviour that used paths arguments to
restrict output (not that it worked as intended---as pointed
out in the mailing list discussion, it was quite incoherent),
this rewrite uses paths arguments to specify what to show.
- Without arguments, it implicitly uses the root level as its
sole argument ("/bin/ls -a" behaves as if "." is given
without argument).
- Without -r (recursive) flag, it shows the named blob (either
file or symlink), or the named tree and its immediate
children.
- With -r flag, it shows the named path, and recursively
descends into it if it is a tree.
- With -d flag, it shows the named path and does not show its
children even if the path is a tree, nor descends into it
recursively.
This is still request-for-comments patch. There is no mailing
list consensus that this proposed new behaviour is a good one.
The patch to t/t3100-ls-tree-restrict.sh illustrates
user-visible behaviour changes. Namely:
* "git-ls-tree $tree path1 path0" lists path1 first and then
path0. It used to use paths as an output restrictor and
showed output in cache entry order (i.e. path0 first and then
path1) regardless of the order of paths arguments.
* "git-ls-tree $tree path2" lists path2 and its immediate
children but having explicit paths argument does not imply
recursive behaviour anymore, hence paths/baz is shown but not
paths/baz/b.
Signed-off-by: Junio C Hamano <junkio@cox.net>
Signed-off-by: Linus Torvalds <torvalds@osdl.org>
2005-05-28 09:05:38 +02:00
|
|
|
entry->parent = NULL; /* needs to be filled by the user */
|
2005-04-24 03:47:23 +02:00
|
|
|
*list_p = entry;
|
|
|
|
list_p = &entry->next;
|
2005-04-18 20:39:48 +02:00
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
2005-05-06 19:48:34 +02:00
|
|
|
|
|
|
|
int parse_tree(struct tree *item)
|
|
|
|
{
|
|
|
|
char type[20];
|
|
|
|
void *buffer;
|
|
|
|
unsigned long size;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
if (item->object.parsed)
|
|
|
|
return 0;
|
|
|
|
buffer = read_sha1_file(item->object.sha1, type, &size);
|
|
|
|
if (!buffer)
|
|
|
|
return error("Could not read %s",
|
|
|
|
sha1_to_hex(item->object.sha1));
|
|
|
|
if (strcmp(type, tree_type)) {
|
|
|
|
free(buffer);
|
|
|
|
return error("Object %s not a tree",
|
|
|
|
sha1_to_hex(item->object.sha1));
|
|
|
|
}
|
|
|
|
ret = parse_tree_buffer(item, buffer, size);
|
|
|
|
free(buffer);
|
|
|
|
return ret;
|
|
|
|
}
|
2005-09-05 08:03:51 +02:00
|
|
|
|
|
|
|
struct tree *parse_tree_indirect(const unsigned char *sha1)
|
|
|
|
{
|
|
|
|
struct object *obj = parse_object(sha1);
|
|
|
|
do {
|
|
|
|
if (!obj)
|
|
|
|
return NULL;
|
|
|
|
if (obj->type == tree_type)
|
|
|
|
return (struct tree *) obj;
|
|
|
|
else if (obj->type == commit_type)
|
|
|
|
obj = &(((struct commit *) obj)->tree->object);
|
|
|
|
else if (obj->type == tag_type)
|
|
|
|
obj = ((struct tag *) obj)->tagged;
|
|
|
|
else
|
|
|
|
return NULL;
|
|
|
|
if (!obj->parsed)
|
|
|
|
parse_object(obj->sha1);
|
|
|
|
} while (1);
|
|
|
|
}
|