On 12/8/2019 1:04 PM, otalpster@xxxxxxxxx wrote: > From: Plato <otalpster@xxxxxxxxx> > > Replace usage of opendir/readdir/closedir API to traverse directories > recursively, at remove_subtree() function, by the dir-iterator API. This > simplifies the code and avoids recursive calls to remove_subtree(). > > Signed-off-by: Plato <otalpster@xxxxxxxxx> > --- > Hello, > > This is my first patch. > I hope I cc'd the correct people and didn't mess up. Welcome, Plato! > The changes pass the test suite t/ and Travis CI. > Please point out any mistakes. > > Thanks for your time! :) > > entry.c | 32 +++++++++++++++----------------- > 1 file changed, 15 insertions(+), 17 deletions(-) > > diff --git a/entry.c b/entry.c > index 53380bb614..e7f4881d3b 100644 > --- a/entry.c > +++ b/entry.c > @@ -2,6 +2,8 @@ > #include "blob.h" > #include "object-store.h" > #include "dir.h" > +#include "iterator.h" > +#include "dir-iterator.h" > #include "streaming.h" > #include "submodule.h" > #include "progress.h" > @@ -50,29 +52,25 @@ static void create_directories(const char *path, int path_len, > > static void remove_subtree(struct strbuf *path) > { > - DIR *dir = opendir(path->buf); > - struct dirent *de; > - int origlen = path->len; > + int ok; > + unsigned int flags = DIR_ITERATOR_PEDANTIC; > + struct dir_iterator *iter = dir_iterator_begin(path->buf, flags); > > - if (!dir) > + if (!iter) > die_errno("cannot opendir '%s'", path->buf); > - while ((de = readdir(dir)) != NULL) { > - struct stat st; > > - if (is_dot_or_dotdot(de->d_name)) > + while ((ok = dir_iterator_advance(iter)) == ITER_OK) { > + if (is_dot_or_dotdot(iter->path.buf)) > continue; The documentation in dir-iterator.h seems to skip . and .. for you. This 'if' shouldn't be needed. > - strbuf_addch(path, '/'); > - strbuf_addstr(path, de->d_name); > - if (lstat(path->buf, &st)) > - die_errno("cannot lstat '%s'", path->buf); > - if (S_ISDIR(st.st_mode)) > - remove_subtree(path); It took me reading dir-iterator.h to realize that the iterator is recursive, because I'm new to this API. This is a nice cleanup! > - else if (unlink(path->buf)) > - die_errno("cannot unlink '%s'", path->buf); > - strbuf_setlen(path, origlen); > + if (unlink(iter->path.buf)) { > + die_errno("cannot unlink '%s'", iter->path.buf); > + } nit: don't use braces for a single-line block. > } > - closedir(dir); > + > + if (ok != ITER_DONE) > + die(_("failed to iterate over '%s'"), path->buf); > + > if (rmdir(path->buf)) > die_errno("cannot rmdir '%s'", path->buf); > } Thanks, -Stolee