file_lock_posix.cpp 2.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126
  1. // This file is part of Desktop App Toolkit,
  2. // a set of libraries for developing nice desktop applications.
  3. //
  4. // For license and copyright information please follow this link:
  5. // https://github.com/desktop-app/legal/blob/master/LEGAL
  6. //
  7. #include "base/file_lock.h"
  8. #include <variant>
  9. #include <unistd.h>
  10. #include <fcntl.h>
  11. #include <sys/types.h>
  12. #include <signal.h>
  13. namespace base {
  14. namespace {
  15. bool KillProcess(pid_t pid) {
  16. auto signal = SIGTERM;
  17. auto attempts = 0;
  18. while (true) {
  19. const auto result = kill(pid, signal);
  20. if (result < 0) {
  21. return (errno == ESRCH);
  22. }
  23. usleep(10000);
  24. if (++attempts == 50) {
  25. signal = SIGKILL;
  26. }
  27. }
  28. }
  29. } // namespace
  30. struct FileLock::Descriptor {
  31. int value;
  32. };
  33. struct FileLock::LockingPid {
  34. pid_t value;
  35. };
  36. class FileLock::Lock {
  37. public:
  38. using Result = std::variant<Descriptor, LockingPid>;
  39. static Result Acquire(const QFile &file);
  40. explicit Lock(int descriptor);
  41. ~Lock();
  42. private:
  43. int _descriptor = 0;
  44. };
  45. FileLock::Lock::Result FileLock::Lock::Acquire(const QFile &file) {
  46. const auto descriptor = file.handle();
  47. if (!descriptor || !file.isOpen()) {
  48. return Descriptor{ 0 };
  49. }
  50. while (true) {
  51. struct flock lock;
  52. lock.l_type = F_WRLCK;
  53. lock.l_whence = SEEK_SET;
  54. lock.l_start = kLockOffset;
  55. lock.l_len = kLockLimit;
  56. if (fcntl(descriptor, F_SETLK, &lock) == 0) {
  57. return Descriptor{ descriptor };
  58. } else if (fcntl(descriptor, F_GETLK, &lock) < 0) {
  59. return LockingPid{ 0 };
  60. } else if (lock.l_type != F_UNLCK) {
  61. return LockingPid{ lock.l_pid };
  62. }
  63. }
  64. }
  65. FileLock::Lock::Lock(int descriptor) : _descriptor(descriptor) {
  66. }
  67. FileLock::Lock::~Lock() {
  68. struct flock unlock;
  69. unlock.l_type = F_UNLCK;
  70. unlock.l_whence = SEEK_SET;
  71. unlock.l_start = kLockOffset;
  72. unlock.l_len = kLockLimit;
  73. fcntl(_descriptor, F_SETLK, &unlock);
  74. }
  75. FileLock::FileLock() = default;
  76. bool FileLock::lock(QFile &file, QIODevice::OpenMode mode) {
  77. Expects(_lock == nullptr || file.isOpen());
  78. unlock();
  79. file.close();
  80. if (!file.open(mode)) {
  81. return false;
  82. }
  83. while (true) {
  84. const auto result = Lock::Acquire(file);
  85. if (const auto descriptor = std::get_if<Descriptor>(&result)) {
  86. if (descriptor->value > 0) {
  87. _lock = std::make_unique<Lock>(descriptor->value);
  88. return true;
  89. }
  90. break;
  91. } else if (const auto pid = std::get_if<LockingPid>(&result)) {
  92. if (pid->value <= 0 || !KillProcess(pid->value)) {
  93. break;
  94. }
  95. }
  96. }
  97. return false;
  98. }
  99. bool FileLock::locked() const {
  100. return (_lock != nullptr);
  101. }
  102. void FileLock::unlock() {
  103. _lock = nullptr;
  104. }
  105. FileLock::~FileLock() = default;
  106. } // namespace base