mirror of
https://github.com/bitcoin/bitcoin.git
synced 2025-02-12 11:19:08 -05:00
add unicode compatible file_lock for Windows
boost::interprocess::file_lock cannot open the files that contain characters which cannot be parsed by the user's code page on Windows. This commit add a new class to handle those specific file for Windows.
This commit is contained in:
parent
f180e81d57
commit
1661a472b8
5 changed files with 114 additions and 16 deletions
85
src/fs.cpp
85
src/fs.cpp
|
@ -1,5 +1,12 @@
|
||||||
#include <fs.h>
|
#include <fs.h>
|
||||||
|
|
||||||
|
#ifndef WIN32
|
||||||
|
#include <fcntl.h>
|
||||||
|
#else
|
||||||
|
#include <codecvt>
|
||||||
|
#include <windows.h>
|
||||||
|
#endif
|
||||||
|
|
||||||
namespace fsbridge {
|
namespace fsbridge {
|
||||||
|
|
||||||
FILE *fopen(const fs::path& p, const char *mode)
|
FILE *fopen(const fs::path& p, const char *mode)
|
||||||
|
@ -12,4 +19,82 @@ FILE *freopen(const fs::path& p, const char *mode, FILE *stream)
|
||||||
return ::freopen(p.string().c_str(), mode, stream);
|
return ::freopen(p.string().c_str(), mode, stream);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#ifndef WIN32
|
||||||
|
|
||||||
|
static std::string GetErrorReason() {
|
||||||
|
return std::strerror(errno);
|
||||||
|
}
|
||||||
|
|
||||||
|
FileLock::FileLock(const fs::path& file)
|
||||||
|
{
|
||||||
|
fd = open(file.string().c_str(), O_RDWR);
|
||||||
|
if (fd == -1) {
|
||||||
|
reason = GetErrorReason();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
FileLock::~FileLock()
|
||||||
|
{
|
||||||
|
if (fd != -1) {
|
||||||
|
close(fd);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
bool FileLock::TryLock()
|
||||||
|
{
|
||||||
|
if (fd == -1) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
struct flock lock;
|
||||||
|
lock.l_type = F_WRLCK;
|
||||||
|
lock.l_whence = SEEK_SET;
|
||||||
|
lock.l_start = 0;
|
||||||
|
lock.l_len = 0;
|
||||||
|
if (fcntl(fd, F_SETLK, &lock) == -1) {
|
||||||
|
reason = GetErrorReason();
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
#else
|
||||||
|
|
||||||
|
static std::string GetErrorReason() {
|
||||||
|
wchar_t* err;
|
||||||
|
FormatMessageW(FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_SYSTEM | FORMAT_MESSAGE_IGNORE_INSERTS,
|
||||||
|
nullptr, GetLastError(), MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT), reinterpret_cast<WCHAR*>(&err), 0, nullptr);
|
||||||
|
std::wstring err_str(err);
|
||||||
|
LocalFree(err);
|
||||||
|
return std::wstring_convert<std::codecvt_utf8_utf16<wchar_t>>().to_bytes(err_str);
|
||||||
|
}
|
||||||
|
|
||||||
|
FileLock::FileLock(const fs::path& file)
|
||||||
|
{
|
||||||
|
hFile = CreateFileW(file.wstring().c_str(), GENERIC_READ | GENERIC_WRITE, FILE_SHARE_READ | FILE_SHARE_WRITE | FILE_SHARE_DELETE,
|
||||||
|
nullptr, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, nullptr);
|
||||||
|
if (hFile == INVALID_HANDLE_VALUE) {
|
||||||
|
reason = GetErrorReason();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
FileLock::~FileLock()
|
||||||
|
{
|
||||||
|
if (hFile != INVALID_HANDLE_VALUE) {
|
||||||
|
CloseHandle(hFile);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
bool FileLock::TryLock()
|
||||||
|
{
|
||||||
|
if (hFile == INVALID_HANDLE_VALUE) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
_OVERLAPPED overlapped = {0};
|
||||||
|
if (!LockFileEx(hFile, LOCKFILE_EXCLUSIVE_LOCK | LOCKFILE_FAIL_IMMEDIATELY, 0, 0, 0, &overlapped)) {
|
||||||
|
reason = GetErrorReason();
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
} // fsbridge
|
} // fsbridge
|
||||||
|
|
20
src/fs.h
20
src/fs.h
|
@ -19,6 +19,26 @@ namespace fs = boost::filesystem;
|
||||||
namespace fsbridge {
|
namespace fsbridge {
|
||||||
FILE *fopen(const fs::path& p, const char *mode);
|
FILE *fopen(const fs::path& p, const char *mode);
|
||||||
FILE *freopen(const fs::path& p, const char *mode, FILE *stream);
|
FILE *freopen(const fs::path& p, const char *mode, FILE *stream);
|
||||||
|
|
||||||
|
class FileLock
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
FileLock() = delete;
|
||||||
|
FileLock(const FileLock&) = delete;
|
||||||
|
FileLock(FileLock&&) = delete;
|
||||||
|
explicit FileLock(const fs::path& file);
|
||||||
|
~FileLock();
|
||||||
|
bool TryLock();
|
||||||
|
std::string GetReason() { return reason; }
|
||||||
|
|
||||||
|
private:
|
||||||
|
std::string reason;
|
||||||
|
#ifndef WIN32
|
||||||
|
int fd = -1;
|
||||||
|
#else
|
||||||
|
void* hFile = (void*)-1; // INVALID_HANDLE_VALUE
|
||||||
|
#endif
|
||||||
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif // BITCOIN_FS_H
|
#endif // BITCOIN_FS_H
|
||||||
|
|
|
@ -51,13 +51,13 @@
|
||||||
|
|
||||||
#ifndef WIN32
|
#ifndef WIN32
|
||||||
#include <signal.h>
|
#include <signal.h>
|
||||||
|
#include <sys/stat.h>
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#include <boost/algorithm/string/classification.hpp>
|
#include <boost/algorithm/string/classification.hpp>
|
||||||
#include <boost/algorithm/string/replace.hpp>
|
#include <boost/algorithm/string/replace.hpp>
|
||||||
#include <boost/algorithm/string/split.hpp>
|
#include <boost/algorithm/string/split.hpp>
|
||||||
#include <boost/bind.hpp>
|
#include <boost/bind.hpp>
|
||||||
#include <boost/interprocess/sync/file_lock.hpp>
|
|
||||||
#include <boost/thread.hpp>
|
#include <boost/thread.hpp>
|
||||||
#include <openssl/crypto.h>
|
#include <openssl/crypto.h>
|
||||||
|
|
||||||
|
|
22
src/util.cpp
22
src/util.cpp
|
@ -71,7 +71,6 @@
|
||||||
#include <malloc.h>
|
#include <malloc.h>
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#include <boost/interprocess/sync/file_lock.hpp>
|
|
||||||
#include <boost/thread.hpp>
|
#include <boost/thread.hpp>
|
||||||
#include <openssl/crypto.h>
|
#include <openssl/crypto.h>
|
||||||
#include <openssl/rand.h>
|
#include <openssl/rand.h>
|
||||||
|
@ -139,7 +138,7 @@ instance_of_cinit;
|
||||||
* cleans them up and thus automatically unlocks them, or ReleaseDirectoryLocks
|
* cleans them up and thus automatically unlocks them, or ReleaseDirectoryLocks
|
||||||
* is called.
|
* is called.
|
||||||
*/
|
*/
|
||||||
static std::map<std::string, std::unique_ptr<boost::interprocess::file_lock>> dir_locks;
|
static std::map<std::string, std::unique_ptr<fsbridge::FileLock>> dir_locks;
|
||||||
/** Mutex to protect dir_locks. */
|
/** Mutex to protect dir_locks. */
|
||||||
static std::mutex cs_dir_locks;
|
static std::mutex cs_dir_locks;
|
||||||
|
|
||||||
|
@ -156,18 +155,13 @@ bool LockDirectory(const fs::path& directory, const std::string lockfile_name, b
|
||||||
// Create empty lock file if it doesn't exist.
|
// Create empty lock file if it doesn't exist.
|
||||||
FILE* file = fsbridge::fopen(pathLockFile, "a");
|
FILE* file = fsbridge::fopen(pathLockFile, "a");
|
||||||
if (file) fclose(file);
|
if (file) fclose(file);
|
||||||
|
auto lock = MakeUnique<fsbridge::FileLock>(pathLockFile);
|
||||||
try {
|
if (!lock->TryLock()) {
|
||||||
auto lock = MakeUnique<boost::interprocess::file_lock>(pathLockFile.string().c_str());
|
return error("Error while attempting to lock directory %s: %s", directory.string(), lock->GetReason());
|
||||||
if (!lock->try_lock()) {
|
}
|
||||||
return false;
|
if (!probe_only) {
|
||||||
}
|
// Lock successful and we're not just probing, put it into the map
|
||||||
if (!probe_only) {
|
dir_locks.emplace(pathLockFile.string(), std::move(lock));
|
||||||
// Lock successful and we're not just probing, put it into the map
|
|
||||||
dir_locks.emplace(pathLockFile.string(), std::move(lock));
|
|
||||||
}
|
|
||||||
} catch (const boost::interprocess::interprocess_exception& e) {
|
|
||||||
return error("Error while attempting to lock directory %s: %s", directory.string(), e.what());
|
|
||||||
}
|
}
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
|
@ -57,7 +57,6 @@ EXPECTED_BOOST_INCLUDES=(
|
||||||
boost/filesystem.hpp
|
boost/filesystem.hpp
|
||||||
boost/filesystem/detail/utf8_codecvt_facet.hpp
|
boost/filesystem/detail/utf8_codecvt_facet.hpp
|
||||||
boost/filesystem/fstream.hpp
|
boost/filesystem/fstream.hpp
|
||||||
boost/interprocess/sync/file_lock.hpp
|
|
||||||
boost/multi_index/hashed_index.hpp
|
boost/multi_index/hashed_index.hpp
|
||||||
boost/multi_index/ordered_index.hpp
|
boost/multi_index/ordered_index.hpp
|
||||||
boost/multi_index/sequenced_index.hpp
|
boost/multi_index/sequenced_index.hpp
|
||||||
|
|
Loading…
Add table
Reference in a new issue