/* * Copyright (C) 2023 Barry * * This file is part of Barry's Little Objects in C Library (libBLOC). * * libBLOC is free software: you can redistribute it and/or modify it under the * terms of the GNU General Public License as published by the Free Software * Foundation, either version 3 of the License, or (at your option) any later * version. * * libBLOC is distributed in the hope that it will be useful, but WITHOUT ANY * WARRANTY; without even the implied warranty of MERCHANTIBILITY or FITNESS * FOR A PARTICULAR PURPOSE. See the GNU General Public License for more * details. * * You should have received a copy of the GNU General Public License along with * libBLOC. If not, see . */ #ifdef __linux__ # define _GNU_SOURCE # include #endif #include #include "../assert.h" #include "object.h" /* Get a unique identifier for the current thread */ static unsigned int _obj_get_thread_id(void) { #if defined(__linux__) return (unsigned int) gettid(); #else # error "No _obj_get_thread_id() for target platform" #endif } /* Lock an object */ void obj_lock(void *addr) { struct ObjectHeader *obj = object_header(addr); ASSERT(obj->magic == OBJECT_MAGIC); unsigned int owner = _obj_get_thread_id(); if (obj->owner != owner) { while (__atomic_test_and_set(&(obj->lock), __ATOMIC_ACQUIRE)) __builtin_ia32_pause(); obj->owner = owner; } __atomic_add_fetch(&obj->locks, 1, __ATOMIC_RELAXED); } /* Unlock an object */ void obj_unlock(void *addr) { struct ObjectHeader *obj = object_header(addr); ASSERT(obj->magic == OBJECT_MAGIC); if (__atomic_sub_fetch(&obj->locks, 1, __ATOMIC_RELAXED)) return; __atomic_clear(&(obj->lock), __ATOMIC_RELEASE); obj->owner = 0; }