#pragma once #include #include #include "arch/getcycles.h" #include "runtime.h" #include "software_interrupt.h" typedef ck_spinlock_mcs_t lock_t; /** * Initializes a lock of type lock_t * @param lock - the address of the lock */ #define LOCK_INIT(lock) ck_spinlock_mcs_init((lock)) /** * Checks if a lock is locked * @param lock - the address of the lock * @returns bool if lock is locked */ #define LOCK_IS_LOCKED(lock) ck_spinlock_mcs_locked((lock)) /** * Locks a lock, keeping track of overhead * @param lock - the address of the lock * @param unique_variable_name - a unique prefix to hygienically namespace an associated lock/unlock pair */ #define LOCK_LOCK_WITH_BOOKKEEPING(lock, unique_variable_name) \ assert(listener_thread_is_running() || !software_interrupt_is_enabled()); \ struct ck_spinlock_mcs _hygiene_##unique_variable_name##_node; \ uint64_t _hygiene_##unique_variable_name##_pre = __getcycles(); \ ck_spinlock_mcs_lock((lock), &(_hygiene_##unique_variable_name##_node)); \ uint64_t _hygiene_##unique_variable_name##_duration = (__getcycles() - _hygiene_##unique_variable_name##_pre); \ if (_hygiene_##unique_variable_name##_duration > generic_thread_lock_longest) { \ generic_thread_lock_longest = _hygiene_##unique_variable_name##_duration; \ } \ generic_thread_lock_duration += _hygiene_##unique_variable_name##_duration; /** * Unlocks a lock * @param lock - the address of the lock * @param unique_variable_name - a unique prefix to hygienically namespace an associated lock/unlock pair */ #define LOCK_UNLOCK_WITH_BOOKKEEPING(lock, unique_variable_name) \ assert(listener_thread_is_running() || !software_interrupt_is_enabled()); \ ck_spinlock_mcs_unlock(lock, &(_hygiene_##unique_variable_name##_node)); /** * Locks a lock, keeping track of overhead * Assumes the availability of DEFAULT as a hygienic prefix for DEFAULT_node and DEFAULT_pre * * As such, this API can only be used once in a lexical scope. * * Use LOCK_LOCK_WITH_BOOKKEEPING and LOCK_UNLOCK_WITH_BOOKKEEPING if multiple locks are required * @param lock - the address of the lock */ #define LOCK_LOCK(lock) LOCK_LOCK_WITH_BOOKKEEPING(lock, DEFAULT) /** * Unlocks a lock * Uses lock node NODE_DEFAULT and timestamp PRE_DEFAULT, so this assumes use of LOCK_LOCK * This API can only be used once in a lexical scope. If this isn't true, use LOCK_LOCK_WITH_BOOKKEEPING and * LOCK_UNLOCK_WITH_BOOKKEEPING * @param lock - the address of the lock */ #define LOCK_UNLOCK(lock) LOCK_UNLOCK_WITH_BOOKKEEPING(lock, DEFAULT)