#include "lint.h"
#include "thr_uberdata.h"
static void
tdb_event_ready(void) {}
static void
tdb_event_sleep(void) {}
static void
tdb_event_switchto(void) {}
static void
tdb_event_switchfrom(void) {}
static void
tdb_event_lock_try(void) {}
static void
tdb_event_catchsig(void) {}
static void
tdb_event_idle(void) {}
static void
tdb_event_create(void) {}
static void
tdb_event_death(void) {}
static void
tdb_event_preempt(void) {}
static void
tdb_event_pri_inherit(void) {}
static void
tdb_event_reap(void) {}
static void
tdb_event_concurrency(void) {}
static void
tdb_event_timeout(void) {}
const tdb_ev_func_t tdb_events[TD_MAX_EVENT_NUM - TD_MIN_EVENT_NUM + 1] = {
tdb_event_ready,
tdb_event_sleep,
tdb_event_switchto,
tdb_event_switchfrom,
tdb_event_lock_try,
tdb_event_catchsig,
tdb_event_idle,
tdb_event_create,
tdb_event_death,
tdb_event_preempt,
tdb_event_pri_inherit,
tdb_event_reap,
tdb_event_concurrency,
tdb_event_timeout
};
#if TDB_HASH_SHIFT != 15
#error "this is all broken because TDB_HASH_SHIFT is not 15"
#endif
static uint_t
tdb_addr_hash(void *addr)
{
#ifdef _LP64
uint64_t value60 = ((uintptr_t)addr >> 4);
uint32_t value30 = (value60 >> 30) ^ (value60 & 0x3fffffff);
#else
uint32_t value30 = ((uintptr_t)addr >> 2);
#endif
return ((value30 >> 15) ^ (value30 & 0x7fff));
}
static tdb_sync_stats_t *
alloc_sync_addr(void *addr)
{
uberdata_t *udp = curthread->ul_uberdata;
tdb_t *tdbp = &udp->tdb;
tdb_sync_stats_t *sap;
ASSERT(MUTEX_OWNED(&udp->tdb_hash_lock, curthread));
if ((sap = tdbp->tdb_sync_addr_free) == NULL) {
void *vaddr;
int i;
if (tdbp->tdb_hash_alloc_failed)
return (NULL);
tdbp->tdb_sync_alloc *= 2;
if ((vaddr = mmap(NULL,
tdbp->tdb_sync_alloc * sizeof (tdb_sync_stats_t),
PROT_READ|PROT_WRITE, MAP_PRIVATE|MAP_ANON,
-1, (off_t)0)) == MAP_FAILED) {
tdbp->tdb_hash_alloc_failed = 1;
return (NULL);
}
sap = tdbp->tdb_sync_addr_free = vaddr;
for (i = 1; i < tdbp->tdb_sync_alloc; sap++, i++)
sap->next = (uintptr_t)(sap + 1);
sap->next = (uintptr_t)0;
tdbp->tdb_sync_addr_last = sap;
sap = tdbp->tdb_sync_addr_free;
}
tdbp->tdb_sync_addr_free = (tdb_sync_stats_t *)(uintptr_t)sap->next;
sap->next = (uintptr_t)0;
sap->sync_addr = (uintptr_t)addr;
(void) memset(&sap->un, 0, sizeof (sap->un));
return (sap);
}
static void
initialize_sync_hash()
{
uberdata_t *udp = curthread->ul_uberdata;
tdb_t *tdbp = &udp->tdb;
uint64_t *addr_hash;
tdb_sync_stats_t *sap;
void *vaddr;
int i;
if (tdbp->tdb_hash_alloc_failed)
return;
lmutex_lock(&udp->tdb_hash_lock);
if (udp->uberflags.uf_tdb_register_sync == REGISTER_SYNC_DISABLE) {
udp->uberflags.uf_tdb_register_sync = REGISTER_SYNC_OFF;
lmutex_unlock(&udp->tdb_hash_lock);
return;
}
if (tdbp->tdb_sync_addr_hash != NULL || tdbp->tdb_hash_alloc_failed) {
lmutex_unlock(&udp->tdb_hash_lock);
return;
}
tdbp->tdb_sync_alloc = 2*1024;
if ((vaddr = mmap(NULL, TDB_HASH_SIZE * sizeof (uint64_t) +
tdbp->tdb_sync_alloc * sizeof (tdb_sync_stats_t),
PROT_READ|PROT_WRITE, MAP_PRIVATE|MAP_ANON,
-1, (off_t)0)) == MAP_FAILED) {
tdbp->tdb_hash_alloc_failed = 1;
return;
}
addr_hash = vaddr;
tdbp->tdb_sync_addr_free = sap =
(tdb_sync_stats_t *)&addr_hash[TDB_HASH_SIZE];
for (i = 1; i < tdbp->tdb_sync_alloc; sap++, i++)
sap->next = (uintptr_t)(sap + 1);
sap->next = (uintptr_t)0;
tdbp->tdb_sync_addr_last = sap;
udp->tdb_hash_lock_stats.next = (uintptr_t)0;
udp->tdb_hash_lock_stats.sync_addr = (uintptr_t)&udp->tdb_hash_lock;
addr_hash[tdb_addr_hash(&udp->tdb_hash_lock)] =
(uintptr_t)&udp->tdb_hash_lock_stats;
tdbp->tdb_register_count = 1;
membar_producer();
tdbp->tdb_sync_addr_hash = addr_hash;
lmutex_unlock(&udp->tdb_hash_lock);
}
tdb_sync_stats_t *
tdb_sync_obj_register(void *addr, int *new)
{
ulwp_t *self = curthread;
uberdata_t *udp = self->ul_uberdata;
tdb_t *tdbp = &udp->tdb;
uint64_t *sapp;
tdb_sync_stats_t *sap = NULL;
int locked = 0;
int i;
if (!self->ul_primarymap)
return (NULL);
if (new)
*new = 0;
if (addr == (void *)&udp->tdb_hash_lock)
return (&udp->tdb_hash_lock_stats);
if (self->ul_sync_obj_reg)
return (NULL);
self->ul_sync_obj_reg = 1;
if (tdbp->tdb_sync_addr_hash == NULL) {
initialize_sync_hash();
if (tdbp->tdb_sync_addr_hash == NULL) {
udp->uberflags.uf_tdb_register_sync = REGISTER_SYNC_OFF;
goto out;
}
}
membar_consumer();
sapp = &tdbp->tdb_sync_addr_hash[tdb_addr_hash(addr)];
if (udp->uberflags.uf_tdb_register_sync == REGISTER_SYNC_ON) {
for (sap = (tdb_sync_stats_t *)(uintptr_t)*sapp; sap != NULL;
sap = (tdb_sync_stats_t *)(uintptr_t)sap->next) {
if (sap->sync_addr == (uintptr_t)addr)
goto out;
}
}
lmutex_lock(&udp->tdb_hash_lock);
locked = 1;
switch (udp->uberflags.uf_tdb_register_sync) {
case REGISTER_SYNC_ON:
break;
case REGISTER_SYNC_OFF:
goto out;
default:
for (i = 0; i < TDB_HASH_SIZE; i++)
for (sap = (tdb_sync_stats_t *)
(uintptr_t)tdbp->tdb_sync_addr_hash[i];
sap != NULL;
sap = (tdb_sync_stats_t *)(uintptr_t)sap->next)
(void) memset(&sap->un, 0, sizeof (sap->un));
switch (udp->uberflags.uf_tdb_register_sync) {
case REGISTER_SYNC_ENABLE:
udp->uberflags.uf_tdb_register_sync = REGISTER_SYNC_ON;
break;
case REGISTER_SYNC_DISABLE:
default:
udp->uberflags.uf_tdb_register_sync = REGISTER_SYNC_OFF;
goto out;
}
break;
}
while ((sap = (tdb_sync_stats_t *)(uintptr_t)*sapp) != NULL) {
if (sap->sync_addr == (uintptr_t)addr)
break;
sapp = &sap->next;
}
if (sap == NULL && (sap = alloc_sync_addr(addr)) != NULL) {
*sapp = (uintptr_t)sap;
tdbp->tdb_register_count++;
if (new)
*new = 1;
}
out:
if (locked)
lmutex_unlock(&udp->tdb_hash_lock);
self->ul_sync_obj_reg = 0;
return (sap);
}
void
tdb_sync_obj_deregister(void *addr)
{
uberdata_t *udp = curthread->ul_uberdata;
tdb_t *tdbp = &udp->tdb;
uint64_t *sapp;
tdb_sync_stats_t *sap;
uint_t hash;
ASSERT(addr != &udp->tdb_hash_lock);
if (tdbp->tdb_sync_addr_hash == NULL ||
tdbp->tdb_sync_addr_hash[hash = tdb_addr_hash(addr)] == 0)
return;
lmutex_lock(&udp->tdb_hash_lock);
sapp = &tdbp->tdb_sync_addr_hash[hash];
while ((sap = (tdb_sync_stats_t *)(uintptr_t)*sapp) != NULL) {
if (sap->sync_addr == (uintptr_t)addr) {
*sapp = sap->next;
tdbp->tdb_register_count--;
sap->next = (uintptr_t)0;
sap->sync_addr = (uintptr_t)0;
if (tdbp->tdb_sync_addr_free == NULL) {
tdbp->tdb_sync_addr_free = sap;
tdbp->tdb_sync_addr_last = sap;
} else {
tdbp->tdb_sync_addr_last->next = (uintptr_t)sap;
tdbp->tdb_sync_addr_last = sap;
}
break;
}
sapp = &sap->next;
}
lmutex_unlock(&udp->tdb_hash_lock);
}
tdb_mutex_stats_t *
tdb_mutex_stats(mutex_t *mp)
{
tdb_sync_stats_t *tssp;
if (mp->mutex_magic != MUTEX_MAGIC)
mp->mutex_magic = MUTEX_MAGIC;
if ((tssp = tdb_sync_obj_register(mp, NULL)) == NULL)
return (NULL);
tssp->un.type = TDB_MUTEX;
return (&tssp->un.mutex);
}
tdb_cond_stats_t *
tdb_cond_stats(cond_t *cvp)
{
tdb_sync_stats_t *tssp;
if (cvp->cond_magic != COND_MAGIC)
cvp->cond_magic = COND_MAGIC;
if ((tssp = tdb_sync_obj_register(cvp, NULL)) == NULL)
return (NULL);
tssp->un.type = TDB_COND;
return (&tssp->un.cond);
}
tdb_rwlock_stats_t *
tdb_rwlock_stats(rwlock_t *rwlp)
{
tdb_sync_stats_t *tssp;
if (rwlp->magic != RWL_MAGIC)
rwlp->magic = RWL_MAGIC;
if ((tssp = tdb_sync_obj_register(rwlp, NULL)) == NULL)
return (NULL);
tssp->un.type = TDB_RWLOCK;
return (&tssp->un.rwlock);
}
tdb_sema_stats_t *
tdb_sema_stats(sema_t *sp)
{
tdb_sync_stats_t *tssp;
int new;
if (sp->magic != SEMA_MAGIC)
sp->magic = SEMA_MAGIC;
if ((tssp = tdb_sync_obj_register(sp, &new)) == NULL)
return (NULL);
tssp->un.type = TDB_SEMA;
if (new) {
tssp->un.sema.sema_max_count = sp->count;
tssp->un.sema.sema_min_count = sp->count;
}
return (&tssp->un.sema);
}