From 2becf7922364be08a79e6e0b921b6d07146a05b3 Mon Sep 17 00:00:00 2001 From: Ilya ilyxa Tyshchenko Date: Thu, 21 Dec 2017 16:55:24 +0000 Subject: [PATCH 1/4] correct action for compile on Solaris 11.3 X86 --- NEWS | 1 + src/util/WStringAPI.hxx | 6 ++++++ 2 files changed, 7 insertions(+) diff --git a/NEWS b/NEWS index 7ae86d430..75889f7ed 100644 --- a/NEWS +++ b/NEWS @@ -1,6 +1,7 @@ ver 0.20.14 (not yet released) * database - simple: fix file corruption in the presence of mount points +* fix Solaris build failure ver 0.20.13 (2017/12/18) * output diff --git a/src/util/WStringAPI.hxx b/src/util/WStringAPI.hxx index a71802dbb..7f5b6c57b 100644 --- a/src/util/WStringAPI.hxx +++ b/src/util/WStringAPI.hxx @@ -106,6 +106,8 @@ UnsafeCopyStringP(wchar_t *dest, const wchar_t *src) noexcept /* emulate wcpcpy() */ UnsafeCopyString(dest, src); return dest + StringLength(dest); +#elif defined(__sun) && defined (__SVR4) + return std::wcpcpy(dest, src); #else return wcpcpy(dest, src); #endif @@ -140,7 +142,11 @@ gcc_malloc gcc_nonnull_all static inline wchar_t * DuplicateString(const wchar_t *p) { +#if defined(__sun) && defined (__SVR4) + return std::wcsdup(p); +#else return wcsdup(p); +#endif } #endif From 412c0a965cd3e7eba8f1649ada4da802cddccfda Mon Sep 17 00:00:00 2001 From: Max Kellermann Date: Thu, 21 Dec 2017 18:45:16 +0100 Subject: [PATCH 2/4] util/WStringAPI: fix indent --- src/util/WStringAPI.hxx | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/src/util/WStringAPI.hxx b/src/util/WStringAPI.hxx index 7f5b6c57b..38357c3bb 100644 --- a/src/util/WStringAPI.hxx +++ b/src/util/WStringAPI.hxx @@ -103,13 +103,13 @@ UnsafeCopyStringP(wchar_t *dest, const wchar_t *src) noexcept { #if defined(_WIN32) || defined(__BIONIC__) || defined(__OpenBSD__) || \ defined(__NetBSD__) - /* emulate wcpcpy() */ - UnsafeCopyString(dest, src); - return dest + StringLength(dest); + /* emulate wcpcpy() */ + UnsafeCopyString(dest, src); + return dest + StringLength(dest); #elif defined(__sun) && defined (__SVR4) return std::wcpcpy(dest, src); #else - return wcpcpy(dest, src); + return wcpcpy(dest, src); #endif } From 4b18460bc6d69332c8d4da64bf6fcaceadbb4864 Mon Sep 17 00:00:00 2001 From: Max Kellermann Date: Fri, 22 Dec 2017 16:02:23 +0100 Subject: [PATCH 3/4] archive/bz2: unlock the archive mutex and lock the file mutex Fixes deadlock because FileInputStream::Read() unlocks the mutex (which was not locked) and then locks it, keeping it locked. This can result in a deadlock. This happens because the archive and the file mutex are different. --- NEWS | 2 ++ src/archive/plugins/Bzip2ArchivePlugin.cxx | 5 +++-- 2 files changed, 5 insertions(+), 2 deletions(-) diff --git a/NEWS b/NEWS index 75889f7ed..394860cfc 100644 --- a/NEWS +++ b/NEWS @@ -1,6 +1,8 @@ ver 0.20.14 (not yet released) * database - simple: fix file corruption in the presence of mount points +* archive + - bz2: fix deadlock * fix Solaris build failure ver 0.20.13 (2017/12/18) diff --git a/src/archive/plugins/Bzip2ArchivePlugin.cxx b/src/archive/plugins/Bzip2ArchivePlugin.cxx index 3dc0b51c3..04597e2a2 100644 --- a/src/archive/plugins/Bzip2ArchivePlugin.cxx +++ b/src/archive/plugins/Bzip2ArchivePlugin.cxx @@ -162,7 +162,7 @@ Bzip2InputStream::FillBuffer() if (bzstream.avail_in > 0) return true; - size_t count = archive->istream->Read(buffer, sizeof(buffer)); + size_t count = archive->istream->LockRead(buffer, sizeof(buffer)); if (count == 0) return false; @@ -174,6 +174,8 @@ Bzip2InputStream::FillBuffer() size_t Bzip2InputStream::Read(void *ptr, size_t length) { + const ScopeUnlock unlock(mutex); + int bz_result; size_t nbytes = 0; @@ -224,4 +226,3 @@ const ArchivePlugin bz2_archive_plugin = { bz2_open, bz2_extensions, }; - From d094c168aa0d9752ca2ffaf3d9569445c251ccc0 Mon Sep 17 00:00:00 2001 From: Max Kellermann Date: Fri, 22 Dec 2017 16:06:07 +0100 Subject: [PATCH 4/4] archive/{iso9660,zzip}: unlock the mutex during I/O Similar to commit 31ab78ae8e10af948ec95496df0d2abf1ea631a4 --- NEWS | 1 + src/archive/plugins/Iso9660ArchivePlugin.cxx | 2 ++ src/archive/plugins/ZzipArchivePlugin.cxx | 4 ++++ 3 files changed, 7 insertions(+) diff --git a/NEWS b/NEWS index 394860cfc..9690c3202 100644 --- a/NEWS +++ b/NEWS @@ -3,6 +3,7 @@ ver 0.20.14 (not yet released) - simple: fix file corruption in the presence of mount points * archive - bz2: fix deadlock + - reduce lock contention, fixing lots of xrun problems * fix Solaris build failure ver 0.20.13 (2017/12/18) diff --git a/src/archive/plugins/Iso9660ArchivePlugin.cxx b/src/archive/plugins/Iso9660ArchivePlugin.cxx index 9e4c4f16a..536745d85 100644 --- a/src/archive/plugins/Iso9660ArchivePlugin.cxx +++ b/src/archive/plugins/Iso9660ArchivePlugin.cxx @@ -182,6 +182,8 @@ Iso9660ArchiveFile::OpenStream(const char *pathname, size_t Iso9660InputStream::Read(void *ptr, size_t read_size) { + const ScopeUnlock unlock(mutex); + int readed = 0; int no_blocks, cur_block; size_t left_bytes = statbuf->size - offset; diff --git a/src/archive/plugins/ZzipArchivePlugin.cxx b/src/archive/plugins/ZzipArchivePlugin.cxx index 5ba087f48..fc1e50b16 100644 --- a/src/archive/plugins/ZzipArchivePlugin.cxx +++ b/src/archive/plugins/ZzipArchivePlugin.cxx @@ -138,6 +138,8 @@ ZzipArchiveFile::OpenStream(const char *pathname, size_t ZzipInputStream::Read(void *ptr, size_t read_size) { + const ScopeUnlock unlock(mutex); + int ret = zzip_file_read(file, ptr, read_size); if (ret < 0) throw std::runtime_error("zzip_file_read() has failed"); @@ -155,6 +157,8 @@ ZzipInputStream::IsEOF() noexcept void ZzipInputStream::Seek(offset_type new_offset) { + const ScopeUnlock unlock(mutex); + zzip_off_t ofs = zzip_seek(file, new_offset, SEEK_SET); if (ofs < 0) throw std::runtime_error("zzip_seek() has failed");