storage/smbclient: add Mutex attribute

This per-object Mutex replaces the global `smbclient_mutex`.
This commit is contained in:
Max Kellermann 2020-07-20 22:37:23 +02:00
parent f5a85a816c
commit a74140842c

View File

@ -23,7 +23,6 @@
#include "storage/FileInfo.hxx" #include "storage/FileInfo.hxx"
#include "lib/smbclient/Init.hxx" #include "lib/smbclient/Init.hxx"
#include "lib/smbclient/Context.hxx" #include "lib/smbclient/Context.hxx"
#include "lib/smbclient/Mutex.hxx"
#include "fs/Traits.hxx" #include "fs/Traits.hxx"
#include "thread/Mutex.hxx" #include "thread/Mutex.hxx"
#include "system/Error.hxx" #include "system/Error.hxx"
@ -60,6 +59,12 @@ class SmbclientStorage final : public Storage {
const std::string base; const std::string base;
/**
* This mutex protects all calls into the #SmbclientContext,
* which is not thread-safe.
*/
Mutex mutex;
SmbclientContext ctx = SmbclientContext::New(); SmbclientContext ctx = SmbclientContext::New();
public: public:
@ -92,12 +97,12 @@ SmbclientStorage::MapToRelativeUTF8(std::string_view uri_utf8) const noexcept
} }
static StorageFileInfo static StorageFileInfo
GetInfo(SmbclientContext &ctx, const char *path) GetInfo(SmbclientContext &ctx, Mutex &mutex, const char *path)
{ {
struct stat st; struct stat st;
{ {
const std::lock_guard<Mutex> protect(smbclient_mutex); const std::lock_guard<Mutex> protect(mutex);
if (ctx.Stat(path, st) != 0) if (ctx.Stat(path, st) != 0)
throw MakeErrno("Failed to access file"); throw MakeErrno("Failed to access file");
} }
@ -121,7 +126,7 @@ StorageFileInfo
SmbclientStorage::GetInfo(std::string_view uri_utf8, [[maybe_unused]] bool follow) SmbclientStorage::GetInfo(std::string_view uri_utf8, [[maybe_unused]] bool follow)
{ {
const std::string mapped = MapUTF8(uri_utf8); const std::string mapped = MapUTF8(uri_utf8);
return ::GetInfo(ctx, mapped.c_str()); return ::GetInfo(ctx, mutex, mapped.c_str());
} }
std::unique_ptr<StorageDirectoryReader> std::unique_ptr<StorageDirectoryReader>
@ -132,11 +137,12 @@ SmbclientStorage::OpenDirectory(std::string_view uri_utf8)
SMBCFILE *handle; SMBCFILE *handle;
{ {
const std::lock_guard<Mutex> protect(smbclient_mutex); const std::lock_guard<Mutex> protect(mutex);
handle = ctx.OpenDirectory(mapped.c_str()); handle = ctx.OpenDirectory(mapped.c_str());
}
if (handle == nullptr) if (handle == nullptr)
throw MakeErrno("Failed to open directory"); throw MakeErrno("Failed to open directory");
}
return std::make_unique<SmbclientDirectoryReader>(*this, return std::make_unique<SmbclientDirectoryReader>(*this,
std::move(mapped), std::move(mapped),
@ -154,14 +160,14 @@ SkipNameFS(const char *name) noexcept
SmbclientDirectoryReader::~SmbclientDirectoryReader() SmbclientDirectoryReader::~SmbclientDirectoryReader()
{ {
const std::lock_guard<Mutex> lock(smbclient_mutex); const std::lock_guard<Mutex> lock(storage.mutex);
storage.ctx.CloseDirectory(handle); storage.ctx.CloseDirectory(handle);
} }
const char * const char *
SmbclientDirectoryReader::Read() noexcept SmbclientDirectoryReader::Read() noexcept
{ {
const std::lock_guard<Mutex> protect(smbclient_mutex); const std::lock_guard<Mutex> protect(storage.mutex);
while (auto e = storage.ctx.ReadDirectory(handle)) { while (auto e = storage.ctx.ReadDirectory(handle)) {
name = e->name; name = e->name;
@ -176,7 +182,7 @@ StorageFileInfo
SmbclientDirectoryReader::GetInfo([[maybe_unused]] bool follow) SmbclientDirectoryReader::GetInfo([[maybe_unused]] bool follow)
{ {
const std::string path = PathTraitsUTF8::Build(base, name); const std::string path = PathTraitsUTF8::Build(base, name);
return ::GetInfo(storage.ctx, path.c_str()); return ::GetInfo(storage.ctx, storage.mutex, path.c_str());
} }
static std::unique_ptr<Storage> static std::unique_ptr<Storage>