diff options
author | Johannes Schindelin <Johannes.Schindelin@gmx.de> | 2007-09-28 16:28:54 +0100 |
---|---|---|
committer | Junio C Hamano <gitster@pobox.com> | 2007-09-30 00:04:39 -0700 |
commit | 7155b727c9baae9ef6d7829370aefc09c4ab64e2 (patch) | |
tree | bf3a04f7d1081f740b472b48654a3594c59d3bb5 /dir.c | |
parent | 90446a0009d9c9c0a06c512f0836e0d30f78d2d0 (diff) | |
download | git-7155b727c9baae9ef6d7829370aefc09c4ab64e2.tar.gz git-7155b727c9baae9ef6d7829370aefc09c4ab64e2.tar.xz |
Introduce remove_dir_recursively()
There was a function called remove_empty_dir_recursive() buried
in refs.c. Expose a slightly enhanced version in dir.h: it can now
optionally remove a non-empty directory.
Signed-off-by: Johannes Schindelin <johannes.schindelin@gmx.de>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
Diffstat (limited to 'dir.c')
-rw-r--r-- | dir.c | 41 |
1 files changed, 41 insertions, 0 deletions
@@ -685,3 +685,44 @@ int is_inside_dir(const char *dir) char buffer[PATH_MAX]; return get_relative_cwd(buffer, sizeof(buffer), dir) != NULL; } + +int remove_dir_recursively(struct strbuf *path, int only_empty) +{ + DIR *dir = opendir(path->buf); + struct dirent *e; + int ret = 0, original_len = path->len, len; + + if (!dir) + return -1; + if (path->buf[original_len - 1] != '/') + strbuf_addch(path, '/'); + + len = path->len; + while ((e = readdir(dir)) != NULL) { + struct stat st; + if ((e->d_name[0] == '.') && + ((e->d_name[1] == 0) || + ((e->d_name[1] == '.') && e->d_name[2] == 0))) + continue; /* "." and ".." */ + + strbuf_setlen(path, len); + strbuf_addstr(path, e->d_name); + if (lstat(path->buf, &st)) + ; /* fall thru */ + else if (S_ISDIR(st.st_mode)) { + if (!remove_dir_recursively(path, only_empty)) + continue; /* happy */ + } else if (!only_empty && !unlink(path->buf)) + continue; /* happy, too */ + + /* path too long, stat fails, or non-directory still exists */ + ret = -1; + break; + } + closedir(dir); + + strbuf_setlen(path, original_len); + if (!ret) + ret = rmdir(path->buf); + return ret; +} |