mirror of
https://github.com/libuv/libuv
synced 2025-03-28 21:13:16 +00:00
unix,win: fix memory leak in uv_fs_scandir() (#3760)
uv_fs_scandir() leaked an entry when you called it on a directory with a single entry _and_ you didn't run the iterator until UV_EOF. Fixes: https://github.com/libuv/libuv/issues/3748
This commit is contained in:
parent
47effc4bd3
commit
b00d1bd225
@ -120,6 +120,7 @@ endif # WINNT
|
||||
EXTRA_DIST = test/fixtures/empty_file \
|
||||
test/fixtures/load_error.node \
|
||||
test/fixtures/lorem_ipsum.txt \
|
||||
test/fixtures/one_file/one_file \
|
||||
include \
|
||||
docs \
|
||||
img \
|
||||
|
@ -650,14 +650,22 @@ static unsigned int* uv__get_nbufs(uv_fs_t* req) {
|
||||
|
||||
void uv__fs_scandir_cleanup(uv_fs_t* req) {
|
||||
uv__dirent_t** dents;
|
||||
unsigned int* nbufs;
|
||||
unsigned int i;
|
||||
unsigned int n;
|
||||
|
||||
unsigned int* nbufs = uv__get_nbufs(req);
|
||||
if (req->result >= 0) {
|
||||
dents = req->ptr;
|
||||
nbufs = uv__get_nbufs(req);
|
||||
|
||||
dents = req->ptr;
|
||||
if (*nbufs > 0 && *nbufs != (unsigned int) req->result)
|
||||
(*nbufs)--;
|
||||
for (; *nbufs < (unsigned int) req->result; (*nbufs)++)
|
||||
uv__fs_scandir_free(dents[*nbufs]);
|
||||
i = 0;
|
||||
if (*nbufs > 0)
|
||||
i = *nbufs - 1;
|
||||
|
||||
n = (unsigned int) req->result;
|
||||
for (; i < n; i++)
|
||||
uv__fs_scandir_free(dents[i]);
|
||||
}
|
||||
|
||||
uv__fs_scandir_free(req->ptr);
|
||||
req->ptr = NULL;
|
||||
|
0
test/fixtures/one_file/one_file
vendored
Normal file
0
test/fixtures/one_file/one_file
vendored
Normal file
@ -2931,6 +2931,24 @@ TEST_IMPL(fs_scandir_file) {
|
||||
}
|
||||
|
||||
|
||||
/* Run in Valgrind. Should not leak when the iterator isn't exhausted. */
|
||||
TEST_IMPL(fs_scandir_early_exit) {
|
||||
uv_dirent_t d;
|
||||
uv_fs_t req;
|
||||
|
||||
ASSERT_LT(0, uv_fs_scandir(NULL, &req, "test/fixtures/one_file", 0, NULL));
|
||||
ASSERT_NE(UV_EOF, uv_fs_scandir_next(&req, &d));
|
||||
uv_fs_req_cleanup(&req);
|
||||
|
||||
ASSERT_LT(0, uv_fs_scandir(NULL, &req, "test/fixtures", 0, NULL));
|
||||
ASSERT_NE(UV_EOF, uv_fs_scandir_next(&req, &d));
|
||||
uv_fs_req_cleanup(&req);
|
||||
|
||||
MAKE_VALGRIND_HAPPY();
|
||||
return 0;
|
||||
}
|
||||
|
||||
|
||||
TEST_IMPL(fs_open_dir) {
|
||||
const char* path;
|
||||
uv_fs_t req;
|
||||
|
@ -401,6 +401,7 @@ TEST_DECLARE (fs_event_stop_in_cb)
|
||||
TEST_DECLARE (fs_scandir_empty_dir)
|
||||
TEST_DECLARE (fs_scandir_non_existent_dir)
|
||||
TEST_DECLARE (fs_scandir_file)
|
||||
TEST_DECLARE (fs_scandir_early_exit)
|
||||
TEST_DECLARE (fs_open_dir)
|
||||
TEST_DECLARE (fs_readdir_empty_dir)
|
||||
TEST_DECLARE (fs_readdir_file)
|
||||
@ -1071,6 +1072,7 @@ TASK_LIST_START
|
||||
TEST_ENTRY (fs_scandir_empty_dir)
|
||||
TEST_ENTRY (fs_scandir_non_existent_dir)
|
||||
TEST_ENTRY (fs_scandir_file)
|
||||
TEST_ENTRY (fs_scandir_early_exit)
|
||||
TEST_ENTRY (fs_open_dir)
|
||||
TEST_ENTRY (fs_readdir_empty_dir)
|
||||
TEST_ENTRY (fs_readdir_file)
|
||||
|
Loading…
x
Reference in New Issue
Block a user