aboutsummaryrefslogtreecommitdiff
path: root/gitstatus/src
diff options
context:
space:
mode:
authorRoman Perepelitsa <roman.perepelitsa@gmail.com>2021-08-18 18:05:04 +0300
committerRoman Perepelitsa <roman.perepelitsa@gmail.com>2021-08-18 18:05:04 +0300
commitec44300155e99268cb2b23deb7e789d0bb4b1723 (patch)
tree766bee13af37a2cab40fe5ceadbf7cb5b9d7b6b4 /gitstatus/src
parent25e5f5985f0b977461c0d14663a910363684014e (diff)
parent80ec734a953d930838ea6839923c97c3da880a0d (diff)
Merge commit '80ec734a953d930838ea6839923c97c3da880a0d'
Diffstat (limited to 'gitstatus/src')
-rw-r--r--gitstatus/src/dir.cc14
-rw-r--r--gitstatus/src/dir.h8
-rw-r--r--gitstatus/src/index.cc1
-rw-r--r--gitstatus/src/tag_db.cc2
4 files changed, 14 insertions, 11 deletions
diff --git a/gitstatus/src/dir.cc b/gitstatus/src/dir.cc
index e7ce7141..39cf1c2c 100644
--- a/gitstatus/src/dir.cc
+++ b/gitstatus/src/dir.cc
@@ -106,14 +106,14 @@ bool ListDir(int dir_fd, Arena& arena, std::vector<char*>& entries, bool precomp
};
constexpr size_t kBufSize = 8 << 10;
- entries.clear();
+ const size_t orig_size = entries.size();
while (true) {
char* buf = static_cast<char*>(arena.Allocate(kBufSize, alignof(linux_dirent64)));
// Save 256 bytes for the rainy day.
int n = syscall(SYS_getdents64, dir_fd, buf, kBufSize - 256);
if (n < 0) {
- entries.clear();
+ entries.resize(orig_size);
return false;
}
for (int pos = 0; pos < n;) {
@@ -131,9 +131,9 @@ bool ListDir(int dir_fd, Arena& arena, std::vector<char*>& entries, bool precomp
}
if (case_sensitive) {
- SortEntries<true>(entries.data(), entries.data() + entries.size());
+ SortEntries<true>(entries.data() + orig_size, entries.data() + entries.size());
} else {
- SortEntries<false>(entries.data(), entries.data() + entries.size());
+ SortEntries<false>(entries.data() + orig_size, entries.data() + entries.size());
}
return true;
@@ -211,7 +211,7 @@ char* DirenvConvert(Arena& arena, struct dirent& ent, bool do_convert) {
bool ListDir(int dir_fd, Arena& arena, std::vector<char*>& entries, bool precompose_unicode,
bool case_sensitive) {
- entries.clear();
+ const size_t orig_size = entries.size();
dir_fd = dup(dir_fd);
if (dir_fd < 0) return false;
DIR* dir = fdopendir(dir_fd);
@@ -225,10 +225,10 @@ bool ListDir(int dir_fd, Arena& arena, std::vector<char*>& entries, bool precomp
entries.push_back(DirenvConvert(arena, *ent, precompose_unicode));
}
if (errno) {
- entries.clear();
+ entries.resize(orig_size);
return false;
}
- StrSort(entries.data(), entries.data() + entries.size(), case_sensitive);
+ StrSort(entries.data() + orig_size, entries.data() + entries.size(), case_sensitive);
return true;
}
diff --git a/gitstatus/src/dir.h b/gitstatus/src/dir.h
index 42ab29bb..4d4cf3da 100644
--- a/gitstatus/src/dir.h
+++ b/gitstatus/src/dir.h
@@ -25,11 +25,11 @@
namespace gitstatus {
-// On error, clears entries and returns false. Does not throw.
+// On error, leaves entries unchaged and returns false. Does not throw.
//
-// On success, fills entries with the names of files from the specified directory and returns true.
-// Every entry is a null-terminated string. At -1 offset is its d_type. All elements point into the
-// arena. They are sorted either by strcmp or strcasecmp depending on case_sensitive.
+// On success, appends names of files from the specified directory to entries and returns true.
+// Every appended entry is a null-terminated string. At -1 offset is its d_type. All elements
+// point into the arena. They are sorted either by strcmp or strcasecmp depending on case_sensitive.
//
// Does not close dir_fd.
//
diff --git a/gitstatus/src/index.cc b/gitstatus/src/index.cc
index ae8ca54c..4d66876b 100644
--- a/gitstatus/src/index.cc
+++ b/gitstatus/src/index.cc
@@ -242,6 +242,7 @@ std::vector<const char*> ScanDirs(git_index* index, int root_fd, IndexDir* const
dir.st = st;
}
+ entries.clear();
arena.Reuse();
if (!ListDir(*dir_fd, arena, entries, caps.precompose_unicode, caps.case_sensitive)) {
AddUnmached("");
diff --git a/gitstatus/src/tag_db.cc b/gitstatus/src/tag_db.cc
index 31b150bd..52cbaede 100644
--- a/gitstatus/src/tag_db.cc
+++ b/gitstatus/src/tag_db.cc
@@ -155,6 +155,8 @@ void TagDb::ReadLooseTags() {
int dir_fd = open(dirname.c_str(), O_RDONLY | O_DIRECTORY | O_CLOEXEC);
if (dir_fd < 0) return;
ON_SCOPE_EXIT(&) { CHECK(!close(dir_fd)) << Errno(); };
+ // TODO: recursively traverse directories so that the file refs/tags/foo/bar gets interpreted
+ // as the tag foo/bar. See https://github.com/romkatv/gitstatus/issues/254.
(void)ListDir(dir_fd, loose_arena_, loose_tags_, /* precompose_unicode = */ false,
/* case_sensitive = */ true);
}