From d8df82a9f9d859fef892b6fdddadb89bda5f05ff Mon Sep 17 00:00:00 2001 From: Takeshi Nakatani Date: Mon, 15 Jul 2024 10:23:53 +0000 Subject: [PATCH] Checked and fixed the internal data lock of PseudoFdInfo class --- src/fdcache_fdinfo.cpp | 99 ++++++++++++++++++++++++++---------------- src/fdcache_fdinfo.h | 5 ++- 2 files changed, 64 insertions(+), 40 deletions(-) diff --git a/src/fdcache_fdinfo.cpp b/src/fdcache_fdinfo.cpp index 500ed0011d..c958abda4a 100644 --- a/src/fdcache_fdinfo.cpp +++ b/src/fdcache_fdinfo.cpp @@ -146,6 +146,17 @@ bool PseudoFdInfo::Clear() return true; } +bool PseudoFdInfo::IsUploadingHasLock() const +{ + return !upload_id.empty(); +} + +bool PseudoFdInfo::IsUploading() const +{ + const std::lock_guard lock(upload_list_lock); + return IsUploadingHasLock(); +} + void PseudoFdInfo::CloseUploadFd() { const std::lock_guard lock(upload_list_lock); @@ -157,6 +168,8 @@ void PseudoFdInfo::CloseUploadFd() bool PseudoFdInfo::OpenUploadFd() { + const std::lock_guard lock(upload_list_lock); + if(-1 != upload_fd){ // already initialized return true; @@ -287,7 +300,9 @@ bool PseudoFdInfo::CompleteInstruction(int result) bool PseudoFdInfo::GetUploadId(std::string& id) const { - if(!IsUploading()){ + const std::lock_guard lock(upload_list_lock); + + if(!IsUploadingHasLock()){ S3FS_PRN_ERR("Multipart Upload has not started yet."); return false; } @@ -297,13 +312,13 @@ bool PseudoFdInfo::GetUploadId(std::string& id) const bool PseudoFdInfo::GetEtaglist(etaglist_t& list) const { - if(!IsUploading()){ + const std::lock_guard lock(upload_list_lock); + + if(!IsUploadingHasLock()){ S3FS_PRN_ERR("Multipart Upload has not started yet."); return false; } - const std::lock_guard lock(upload_list_lock); - list.clear(); for(filepart_list_t::const_iterator iter = upload_list.begin(); iter != upload_list.end(); ++iter){ if(iter->petag){ @@ -325,12 +340,13 @@ bool PseudoFdInfo::GetEtaglist(etaglist_t& list) const // bool PseudoFdInfo::AppendUploadPart(off_t start, off_t size, bool is_copy, etagpair** ppetag) { - if(!IsUploading()){ + const std::lock_guard lock(upload_list_lock); + + if(!IsUploadingHasLock()){ S3FS_PRN_ERR("Multipart Upload has not started yet."); return false; } - const std::lock_guard lock(upload_list_lock); off_t next_start_pos = 0; if(!upload_list.empty()){ next_start_pos = upload_list.back().startpos + upload_list.back().size; @@ -365,9 +381,11 @@ static bool filepart_partnum_compare(const filepart& src1, const filepart& src2) bool PseudoFdInfo::InsertUploadPart(off_t start, off_t size, int part_num, bool is_copy, etagpair** ppetag) { + const std::lock_guard lock(upload_list_lock); + //S3FS_PRN_DBG("[start=%lld][size=%lld][part_num=%d][is_copy=%s]", static_cast(start), static_cast(size), part_num, (is_copy ? "true" : "false")); - if(!IsUploading()){ + if(!IsUploadingHasLock()){ S3FS_PRN_ERR("Multipart Upload has not started yet."); return false; } @@ -405,6 +423,11 @@ bool PseudoFdInfo::ParallelMultipartUpload(const char* path, const mp_part_list_ return false; } + std::string tmp_upload_id; + if(!GetUploadId(tmp_upload_id)){ + return false; + } + for(mp_part_list_t::const_iterator iter = mplist.begin(); iter != mplist.end(); ++iter){ // Insert upload part etagpair* petag = nullptr; @@ -417,7 +440,7 @@ bool PseudoFdInfo::ParallelMultipartUpload(const char* path, const mp_part_list_ pseudofdinfo_thparam* thargs = new pseudofdinfo_thparam; thargs->ppseudofdinfo = this; thargs->path = SAFESTRPTR(path); - thargs->upload_id = upload_id; + thargs->upload_id = tmp_upload_id; thargs->upload_fd = upload_fd; thargs->start = iter->start; thargs->size = iter->size; @@ -448,16 +471,12 @@ bool PseudoFdInfo::ParallelMultipartUploadAll(const char* path, const mp_part_li result = 0; - { - const std::lock_guard lock(upload_list_lock); - if(!OpenUploadFd()){ - return false; - } - - if(!ParallelMultipartUpload(path, to_upload_list, false) || !ParallelMultipartUpload(path, copy_list, true)){ - S3FS_PRN_ERR("Failed setup instruction for uploading(path=%s, to_upload_list=%zu, copy_list=%zu).", SAFESTRPTR(path), to_upload_list.size(), copy_list.size()); - return false; - } + if(!OpenUploadFd()){ + return false; + } + if(!ParallelMultipartUpload(path, to_upload_list, false) || !ParallelMultipartUpload(path, copy_list, true)){ + S3FS_PRN_ERR("Failed setup instruction for uploading(path=%s, to_upload_list=%zu, copy_list=%zu).", SAFESTRPTR(path), to_upload_list.size(), copy_list.size()); + return false; } // Wait for all thread exiting @@ -695,28 +714,32 @@ bool PseudoFdInfo::ExtractUploadPartsFromUntreatedArea(const off_t& untreated_st // Also, it is assumed that it must not be a copy area. // So if the areas overlap, include uploaded area as an untreated area. // - for(filepart_list_t::iterator cur_iter = upload_list.begin(); cur_iter != upload_list.end(); /* ++cur_iter */){ - // Check overlap - if((cur_iter->startpos + cur_iter->size - 1) < aligned_start || (aligned_start + aligned_size - 1) < cur_iter->startpos){ - // Areas do not overlap - ++cur_iter; + { + const std::lock_guard lock(upload_list_lock); - }else{ - // The areas overlap - // - // Since the start position of the uploaded area is aligned with the boundary, - // it is not necessary to check the start position. - // If the uploaded area exceeds the untreated area, expand the untreated area. - // - if((aligned_start + aligned_size - 1) < (cur_iter->startpos + cur_iter->size - 1)){ - aligned_size += (cur_iter->startpos + cur_iter->size) - (aligned_start + aligned_size); - } + for(filepart_list_t::iterator cur_iter = upload_list.begin(); cur_iter != upload_list.end(); /* ++cur_iter */){ + // Check overlap + if((cur_iter->startpos + cur_iter->size - 1) < aligned_start || (aligned_start + aligned_size - 1) < cur_iter->startpos){ + // Areas do not overlap + ++cur_iter; - // - // Add this to cancel list - // - cancel_upload_list.push_back(*cur_iter); // Copy and Push to cancel list - cur_iter = upload_list.erase(cur_iter); + }else{ + // The areas overlap + // + // Since the start position of the uploaded area is aligned with the boundary, + // it is not necessary to check the start position. + // If the uploaded area exceeds the untreated area, expand the untreated area. + // + if((aligned_start + aligned_size - 1) < (cur_iter->startpos + cur_iter->size - 1)){ + aligned_size += (cur_iter->startpos + cur_iter->size) - (aligned_start + aligned_size); + } + + // + // Add this to cancel list + // + cancel_upload_list.push_back(*cur_iter); // Copy and Push to cancel list + cur_iter = upload_list.erase(cur_iter); + } } } diff --git a/src/fdcache_fdinfo.h b/src/fdcache_fdinfo.h index d7b3a2e6e5..d008b7cb03 100644 --- a/src/fdcache_fdinfo.h +++ b/src/fdcache_fdinfo.h @@ -81,10 +81,11 @@ class PseudoFdInfo bool ResetUploadInfo() REQUIRES(upload_list_lock); bool RowInitialUploadInfo(const std::string& id, bool is_cancel_mp); bool CompleteInstruction(int result) REQUIRES(upload_list_lock); - bool ParallelMultipartUpload(const char* path, const mp_part_list_t& mplist, bool is_copy) REQUIRES(upload_list_lock); + bool ParallelMultipartUpload(const char* path, const mp_part_list_t& mplist, bool is_copy); bool InsertUploadPart(off_t start, off_t size, int part_num, bool is_copy, etagpair** ppetag) REQUIRES(upload_list_lock); bool CancelAllThreads(); bool ExtractUploadPartsFromUntreatedArea(const off_t& untreated_start, const off_t& untreated_size, mp_part_list_t& to_upload_list, filepart_list_t& cancel_upload_list, off_t max_mp_size); + bool IsUploadingHasLock() const REQUIRES(upload_list_lock); public: explicit PseudoFdInfo(int fd = -1, int open_flags = 0); @@ -104,7 +105,7 @@ class PseudoFdInfo bool ClearUploadInfo(bool is_cancel_mp = false); bool InitialUploadInfo(const std::string& id){ return RowInitialUploadInfo(id, true); } - bool IsUploading() const { return !upload_id.empty(); } + bool IsUploading() const; bool GetUploadId(std::string& id) const; bool GetEtaglist(etaglist_t& list) const;