blob: 6646e3994d87bb725371c6f95586f5558fa680f7 [file]
// Copyright 2026 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#ifndef COMPONENTS_SQLITE_VFS_SHARED_LOCKS_H_
#define COMPONENTS_SQLITE_VFS_SHARED_LOCKS_H_
#include <stdint.h>
#include <array>
#include <optional>
#include "base/component_export.h"
#include "base/memory/shared_memory_mapping.h"
#include "base/memory/shared_memory_safety_checker.h"
#include "base/memory/unsafe_shared_memory_region.h"
#include "components/sqlite_vfs/lock_state.h"
namespace sqlite_vfs {
// Manages the set of cross-connection (within or across processes) locks for
// a SQLite database.
//
// Each database that supports multiple connections requires that each
// connection use the same set of shared locks to coordinate access. A database
// using a rollback journal requires only the primary database lock. This is
// the only supported scenario at present.
class COMPONENT_EXPORT(SQLITE_VFS) SharedLocks {
public:
// Returns a new unmapped shared memory region sized appropriately to hold
// the shared locks. If `wal_mode` is true, the region is sized to hold the
// locks for a wal-mode database.
static base::UnsafeSharedMemoryRegion CreateRegion(bool wal_mode);
// Maps `region` and returns a SharedLocks object backed by it. Returns no
// value in case of failure to map `region` into the process's address space.
// If `wal_mode` is true, the region must have been created to hold the locks
// for a wal-mode database.
static std::optional<SharedLocks> Create(
const base::UnsafeSharedMemoryRegion& region,
bool wal_mode);
SharedLocks(const SharedLocks&) = delete;
SharedLocks& operator=(const SharedLocks&) = delete;
SharedLocks(SharedLocks&&) = default;
SharedLocks& operator=(SharedLocks&&) = default;
~SharedLocks();
// Lock operations on the primary database lock corresponding to SQLite file
// operations; see https://sqlite.org/c3ref/io_methods.html.
// Attempts to raise the lock from `current_mode` to `mode`, writing the new
// mode into `current_mode` and returning `SQLITE_OK` on success. Returns a
// SQLite error code on failure, in which case `current_mode` may have been
// modified (e.g., an attempt to acquire an exclusive lock may acquire a
// pending lock before failing with SQLITE_BUSY). Returns `SQLITE_IOERR_LOCK`
// if the locks have been abandoned.
int Lock(int mode, int& current_mode);
// Lowers the lock from a higher level down to `mode`, which must be either
// `SQLITE_LOCK_SHARED` or `SQLITE_LOCK_NONE`. Sets `current_mode` to `mode`
// and returns `SQLITE_OK`.
int Unlock(int mode, int& current_mode);
// Returns true if any connection holds the reserved lock.
bool IsReserved();
// Marks the locks as abandoned, causing all subsequent attempts to raise the
// primary database lock to a higher level by any party to fail with
// `SQLITE_IOERR_LOCK`. Returns the state of the primary database lock at the
// time of abandonment. Determination of this state is made based on a
// snapshot of the lock at the moment that the lock is abandoned. This is the
// only point where it is possible to know the state of the lock owing to the
// nature of atomic bitwise operations on the lock itself; the lock level may
// be raised to pending or reserved after abandonment, although the callers
// requesting such will properly detect that the lock has been abandoned.
LockState Abandon();
// Returns true if the database lock has been marked as abandoned.
bool IsAbandoned() const;
// Acquires or releases `num_locks` WAL locks beginning at index `lock_index`.
// Shared locks are acquired and released individually (`num_locks` ==
// 1). Exclusive locks are acquired in a range (`num_locks` >= 1).
enum class LockOperation { kAcquire, kRelease };
enum class LockType { kShared, kExclusive };
int ShmLock(int lock_index,
int num_locks,
LockOperation operation,
LockType type);
// Enforces a memory barrier.
void ShmBarrier();
private:
// The primary database lock.
using DatabaseLock = base::subtle::SharedAtomic<uint32_t>;
// A WAL lock.
using WalLock = base::subtle::SharedAtomic<uint32_t>;
// The number of WAL locks.
static constexpr int kNumWalLocks = 8;
// The primary database lock and the WAL-index locks.
struct DatabaseAndWalLocks {
DatabaseLock primary_lock;
// The eight WAL locks:
// 0: WAL_WRITE_LOCK. Held exclusively while a read/write connection is
// appending to the WAL or is recovering the WAL-index.
// 1: WAL_CKPT_LOCK. Held exclusively while a read/write connection is
// performing a checkpoint or is recovering the WAL-index.
// 2: WAL_RECOVER_LOCK. Held exclusively while a read/write connection is
// recovering the WAL-index.
// 3-7: WAL_READ_LOCK(I). Read locks corresponding to the read-marks in the
// WAL-index. One of the read locks is held shared by any connection
// that is within a transaction. One read lock is held exclusively
// while updating the corresponding read-mark in the WAL-index. All but
// the first read locks are held exclusively when resetting the WAL
// after a complete checkpoint or while recovering the WAL-index.
std::array<WalLock, kNumWalLocks> wal_locks;
};
SharedLocks(base::WritableSharedMemoryMapping mapping, bool wal_mode);
// Returns the atomic for the primary database lock.
DatabaseLock& GetDatabaseLock();
// Returns the specified WAL lock.
WalLock& GetWalLock(int index);
base::WritableSharedMemoryMapping mapping_;
bool wal_mode_;
};
} // namespace sqlite_vfs
#endif // COMPONENTS_SQLITE_VFS_SHARED_LOCKS_H_