#ifndef _THR_UBERDATA_H
#define _THR_UBERDATA_H
#include <stdlib.h>
#include <unistd.h>
#include <sys/types.h>
#include <fcntl.h>
#include <string.h>
#include <signal.h>
#include <ucontext.h>
#include <thread.h>
#include <pthread.h>
#include <atomic.h>
#include <link.h>
#include <sys/resource.h>
#include <sys/lwp.h>
#include <errno.h>
#include <sys/asm_linkage.h>
#include <sys/regset.h>
#include <sys/fcntl.h>
#include <sys/mman.h>
#include <synch.h>
#include <door.h>
#include <limits.h>
#include <sys/synch32.h>
#include <schedctl.h>
#include <sys/priocntl.h>
#include <thread_db.h>
#include <setjmp.h>
#include <sys/thread.h>
#include <sys/debug.h>
#include "libc_int.h"
#include "tdb_agent.h"
#undef errno
#define errno (*curthread->ul_errnop)
#if defined(__sparc)
#define LOCKMASK 0xff000000
#define WAITERMASK 0x000000ff
#define SPINNERMASK 0x00ff0000
#define SPINNERSHIFT 16
#define WAITER 0x00000001
#define LOCKSET 0xff
#define LOCKCLEAR 0
#define PIDSHIFT 32
#define LOCKMASK64 0xffffffffff000000ULL
#define LOCKBYTE64 0x00000000ff000000ULL
#define WAITERMASK64 0x00000000000000ffULL
#define SPINNERMASK64 0x0000000000ff0000ULL
#elif defined(__x86)
#define LOCKMASK 0xff000000
#define WAITERMASK 0x00ff0000
#define SPINNERMASK 0x0000ff00
#define SPINNERSHIFT 8
#define WAITER 0x00010000
#define LOCKSET 0x01
#define LOCKCLEAR 0
#define PIDSHIFT 0
#define LOCKMASK64 0xff000000ffffffffULL
#define LOCKBYTE64 0x0100000000000000ULL
#define WAITERMASK64 0x00ff000000000000ULL
#define SPINNERMASK64 0x0000ff0000000000ULL
#else
#error "neither __sparc nor __x86 is defined"
#endif
#define MUTEX_OWNER(mp) ((ulwp_t *)(uintptr_t)(mp)->mutex_owner)
#define MUTEX_OWNED(mp, thrp) \
((mp)->mutex_lockw != 0 && MUTEX_OWNER(mp) == thrp)
typedef union {
int uf_all;
struct {
short h_pad;
short h_trs_ted;
} uf_h;
struct {
char x_mt;
char x_pad;
char x_tdb_register_sync;
char x_thread_error_detection;
} uf_x;
} uberflags_t;
#define uf_mt uf_x.x_mt
#define uf_tdb_register_sync uf_x.x_tdb_register_sync
#define uf_thread_error_detection uf_x.x_thread_error_detection
#define uf_trs_ted uf_h.h_trs_ted
#ifdef _LP64
#define MINSTACK (8 * 1024)
#define DEFAULTSTACK (2 * 1024 * 1024)
#else
#define MINSTACK (4 * 1024)
#define DEFAULTSTACK (1024 * 1024)
#endif
#define MUTEX_TRY 0
#define MUTEX_LOCK 1
#define MUTEX_NOCEIL 0x40
#if defined(__x86)
typedef struct {
int fctrl;
int fstat;
int ftag;
int misc[4];
} fpuenv_t;
#ifdef _SYSCALL32
typedef fpuenv_t fpuenv32_t;
#endif
#elif defined(__sparc)
typedef struct {
greg_t fsr;
greg_t fpu_en;
} fpuenv_t;
#ifdef _SYSCALL32
typedef struct {
greg32_t fsr;
greg32_t fpu_en;
} fpuenv32_t;
#endif
#endif
#if defined(__x86)
extern void ht_pause(void);
#define SMT_PAUSE() ht_pause()
#elif defined(SMT_PAUSE_FUNCTION)
extern void SMT_PAUSE_FUNCTION(void);
#define SMT_PAUSE() SMT_PAUSE_FUNCTION()
#else
#define SMT_PAUSE() smt_pause()
#endif
typedef struct __cleanup {
struct __cleanup *next;
caddr_t fp;
void (*func)(void *);
void *arg;
} __cleanup_t;
typedef void (*PFrV)(void *);
#define TSD_UNALLOCATED ((PFrV)1)
#define TSD_NFAST 9
typedef union tsd {
uint_t tsd_nalloc;
void *tsd_pad[TSD_NFAST];
void *tsd_data[1];
} tsd_t;
typedef struct {
mutex_t tsdm_lock;
uint_t tsdm_nkeys;
uint_t tsdm_nused;
PFrV *tsdm_destro;
char tsdm_pad[64 -
(sizeof (mutex_t) + 2 * sizeof (uint_t) + sizeof (PFrV *))];
} tsd_metadata_t;
#ifdef _SYSCALL32
typedef union tsd32 {
uint_t tsd_nalloc;
caddr32_t tsd_pad[TSD_NFAST];
caddr32_t tsd_data[1];
} tsd32_t;
typedef struct {
mutex_t tsdm_lock;
uint_t tsdm_nkeys;
uint_t tsdm_nused;
caddr32_t tsdm_destro;
char tsdm_pad[64 -
(sizeof (mutex_t) + 2 * sizeof (uint_t) + sizeof (caddr32_t))];
} tsd_metadata32_t;
#endif
typedef struct {
void *tls_data;
size_t tls_size;
} tls_t;
typedef struct {
mutex_t tls_lock;
tls_t tls_modinfo;
tls_t static_tls;
char tls_pad[64 -
(sizeof (mutex_t) + 2 * sizeof (tls_t))];
} tls_metadata_t;
#ifdef _SYSCALL32
typedef struct {
caddr32_t tls_data;
size32_t tls_size;
} tls32_t;
typedef struct {
mutex_t tls_lock;
tls32_t tls_modinfo;
tls32_t static_tls;
char tls_pad[64 -
(sizeof (mutex_t) + 2 * sizeof (tls32_t))];
} tls_metadata32_t;
#endif
typedef struct queue_root {
struct queue_root *qr_next;
struct queue_root *qr_prev;
struct ulwp *qr_head;
struct ulwp *qr_tail;
void *qr_wchan;
uint32_t qr_rtcount;
uint32_t qr_qlen;
uint32_t qr_qmax;
} queue_root_t;
#ifdef _SYSCALL32
typedef struct queue_root32 {
caddr32_t qr_next;
caddr32_t qr_prev;
caddr32_t qr_head;
caddr32_t qr_tail;
caddr32_t qr_wchan;
uint32_t qr_rtcount;
uint32_t qr_qlen;
uint32_t qr_qmax;
} queue_root32_t;
#endif
typedef union {
uint64_t qh_64[16];
struct {
mutex_t q_lock;
uint8_t q_qcnt;
uint8_t q_type;
uint8_t q_pad1[2];
uint32_t q_lockcount;
uint32_t q_qlen;
uint32_t q_qmax;
void *q_wchan;
struct queue_root *q_root;
struct queue_root *q_hlist;
#if !defined(_LP64)
caddr_t q_pad2[3];
#endif
queue_root_t q_def_root;
uint32_t q_hlen;
uint32_t q_hmax;
} qh_qh;
} queue_head_t;
#define qh_lock qh_qh.q_lock
#define qh_qcnt qh_qh.q_qcnt
#define qh_type qh_qh.q_type
#if defined(DEBUG)
#define qh_lockcount qh_qh.q_lockcount
#define qh_qlen qh_qh.q_qlen
#define qh_qmax qh_qh.q_qmax
#endif
#define qh_wchan qh_qh.q_wchan
#define qh_root qh_qh.q_root
#define qh_hlist qh_qh.q_hlist
#define qh_def_root qh_qh.q_def_root
#define qh_hlen qh_qh.q_hlen
#define qh_hmax qh_qh.q_hmax
#define MX 0
#define CV 1
#define QHASHSHIFT 9
#define QHASHSIZE (1 << QHASHSHIFT)
#define QUEUE_HASH(wchan, type) ((uint_t) \
((((uintptr_t)(wchan) >> 3) \
^ ((uintptr_t)(wchan) >> (QHASHSHIFT + 3))) \
& (QHASHSIZE - 1)) + (((type) == MX)? 0 : QHASHSIZE))
extern queue_head_t *queue_lock(void *, int);
extern void queue_unlock(queue_head_t *);
extern void enqueue(queue_head_t *, struct ulwp *, int);
extern struct ulwp *dequeue(queue_head_t *, int *);
extern struct ulwp **queue_slot(queue_head_t *, struct ulwp **, int *);
extern struct ulwp *queue_waiter(queue_head_t *);
extern int dequeue_self(queue_head_t *);
extern void queue_unlink(queue_head_t *,
struct ulwp **, struct ulwp *);
extern void unsleep_self(void);
extern void spin_lock_set(mutex_t *);
extern void spin_lock_clear(mutex_t *);
typedef struct {
short pcc_state;
short pcc_policy;
pri_t pcc_primin;
pri_t pcc_primax;
pcinfo_t pcc_info;
} pcclass_t;
typedef struct mxchain {
struct mxchain *mxchain_next;
mutex_t *mxchain_mx;
} mxchain_t;
typedef struct {
void *rd_rwlock;
size_t rd_count;
} readlock_t;
#ifdef _SYSCALL32
typedef struct {
caddr32_t rd_rwlock;
size32_t rd_count;
} readlock32_t;
#endif
#define NTMEMBASE 16
typedef struct {
size_t tm_size;
void *tm_roots[NTMEMBASE];
} tumem_t;
#ifdef _SYSCALL32
typedef struct {
uint32_t tm_size;
caddr32_t tm_roots[NTMEMBASE];
} tumem32_t;
#endif
typedef void (*tmem_func_t)(void *, int);
#define READ_LOCK_MAX 100000
#define ul_tlsent ul_tls.tls_data
#define ul_ntlsent ul_tls.tls_size
#define roundup64(x) (-(-(x) & -64))
typedef struct ulwp {
#if defined(__sparc)
uint32_t ul_dinstr;
uint32_t ul_padsparc0[15];
uint32_t ul_dsave;
uint32_t ul_drestore;
uint32_t ul_dftret;
uint32_t ul_dreturn;
#endif
struct ulwp *ul_self;
#if defined(__i386)
uint8_t ul_dinstr[40];
#elif defined(__amd64)
uint8_t ul_dinstr[56];
#endif
struct uberdata *ul_uberdata;
tls_t ul_tls;
struct ulwp *ul_forw;
struct ulwp *ul_back;
struct ulwp *ul_next;
struct ulwp *ul_hash;
void *ul_rval;
caddr_t ul_stk;
size_t ul_mapsiz;
size_t ul_guardsize;
uintptr_t ul_stktop;
size_t ul_stksiz;
stack_t ul_ustack;
int ul_ix;
lwpid_t ul_lwpid;
pri_t ul_pri;
pri_t ul_epri;
char ul_policy;
char ul_cid;
union {
struct {
char cursig;
char pleasestop;
} s;
short curplease;
} ul_cp;
char ul_stop;
char ul_signalled;
char ul_dead;
char ul_unwind;
char ul_detached;
char ul_writer;
char ul_stopping;
char ul_cancel_prologue;
short ul_preempt;
short ul_savpreempt;
char ul_sigsuspend;
char ul_main;
char ul_fork;
char ul_primarymap;
uint8_t ul_max_spinners;
char ul_door_noreserve;
char ul_queue_fifo;
char ul_cond_wait_defer;
char ul_error_detection;
char ul_async_safe;
char ul_rt;
char ul_rtqueued;
char ul_misaligned;
char ul_pad[3];
int ul_adaptive_spin;
int ul_queue_spin;
volatile int ul_critical;
int ul_sigdefer;
int ul_vfork;
int ul_cancelable;
char ul_cancel_pending;
char ul_cancel_disabled;
char ul_cancel_async;
char ul_save_async;
char ul_mutator;
char ul_created;
char ul_replace;
uchar_t ul_nocancel;
int ul_errno;
int *ul_errnop;
__cleanup_t *ul_clnup_hdr;
uberflags_t *ul_schedctl_called;
volatile sc_shared_t *ul_schedctl;
int ul_bindflags;
uint_t ul_libc_locks;
tsd_t *ul_stsd;
void *ul_ftsd[TSD_NFAST];
td_evbuf_t ul_td_evbuf;
char ul_td_events_enable;
char ul_sync_obj_reg;
char ul_qtype;
char ul_cv_wake;
int ul_rtld;
int ul_usropts;
void *(*ul_startpc)(void *);
void *ul_startarg;
void *ul_wchan;
struct ulwp *ul_link;
queue_head_t *ul_sleepq;
mutex_t *ul_cvmutex;
mxchain_t *ul_mxchain;
int ul_save_state;
uint_t ul_rdlockcnt;
union {
readlock_t single;
readlock_t *array;
} ul_readlock;
uint_t ul_heldlockcnt;
union {
mutex_t *single;
mutex_t **array;
} ul_heldlocks;
void *ul_tpdp;
ucontext_t *ul_siglink;
uint_t ul_spin_lock_spin;
uint_t ul_spin_lock_spin2;
uint_t ul_spin_lock_sleep;
uint_t ul_spin_lock_wakeup;
queue_root_t ul_queue_root;
id_t ul_rtclassid;
uint_t ul_pilocks;
sigset_t ul_sigmask;
sigset_t ul_tmpmask;
siginfo_t ul_siginfo;
mutex_t ul_spinlock;
fpuenv_t ul_fpuenv;
uintptr_t ul_sp;
void *ul_ex_unwind;
#if defined(sparc)
void *ul_unwind_ret;
#endif
tumem_t ul_tmem;
uint_t ul_ptinherit;
char ul_ntoabuf[18];
} ulwp_t;
#define ul_cursig ul_cp.s.cursig
#define ul_pleasestop ul_cp.s.pleasestop
#define ul_curplease ul_cp.curplease
#define REPLACEMENT_SIZE ((size_t)&((ulwp_t *)NULL)->ul_sigmask)
#define SIGMASK(sig) ((uint32_t)1 << (((sig) - 1) & (32 - 1)))
#if (MAXSIG > (2 * 32) && MAXSIG <= (3 * 32))
#define FILLSET0 0xffffffffu
#define FILLSET1 0xffffffffu
#define FILLSET2 ((1u << (MAXSIG - 64)) - 1)
#define FILLSET3 0
#else
#error "fix me: MAXSIG out of bounds"
#endif
#define CANTMASK0 (SIGMASK(SIGKILL) | SIGMASK(SIGSTOP))
#define CANTMASK1 0
#define CANTMASK2 0
#define CANTMASK3 0
#define MASKSET0 (FILLSET0 & ~CANTMASK0)
#define MASKSET1 (FILLSET1 & ~CANTMASK1)
#define MASKSET2 (FILLSET2 & ~CANTMASK2)
#define MASKSET3 (FILLSET3 & ~CANTMASK3)
extern const sigset_t maskset;
extern int thread_adaptive_spin;
extern uint_t thread_max_spinners;
extern int thread_queue_spin;
extern int thread_queue_fifo;
extern int thread_queue_dump;
extern int thread_cond_wait_defer;
extern int thread_async_safe;
extern int thread_queue_verify;
typedef struct atfork {
struct atfork *forw;
struct atfork *back;
void (*prepare)(void);
void (*parent)(void);
void (*child)(void);
} atfork_t;
typedef struct robust {
struct robust *robust_next;
struct robust *robust_list;
mutex_t *robust_lock;
} robust_t;
#define INVALID_ADDR ((void *)(uintptr_t)(-1L))
#define LOCKSHIFT 15
#define LOCKHASHSZ (1 << LOCKSHIFT)
#define LOCK_HASH(addr) (uint_t) \
((((uintptr_t)(addr) >> 3) \
^ ((uintptr_t)(addr) >> (LOCKSHIFT + 3))) \
& (LOCKHASHSZ - 1))
typedef struct {
mutex_t pad_lock;
char pad_pad[64 - sizeof (mutex_t)];
} pad_lock_t;
typedef struct {
mutex_t pad_lock;
char pad_pad[32 - sizeof (mutex_t)];
} pad32_lock_t;
typedef struct {
mutex_t hash_lock;
cond_t hash_cond;
ulwp_t *hash_bucket;
char hash_pad[64 -
(sizeof (mutex_t) + sizeof (cond_t) + sizeof (ulwp_t *))];
} thr_hash_table_t;
#ifdef _SYSCALL32
typedef struct {
mutex_t hash_lock;
cond_t hash_cond;
caddr32_t hash_bucket;
char hash_pad[64 -
(sizeof (mutex_t) + sizeof (cond_t) + sizeof (caddr32_t))];
} thr_hash_table32_t;
#endif
typedef struct {
rwlock_t sig_lock;
struct sigaction sig_uaction;
char sig_pad[128 - sizeof (rwlock_t) - sizeof (struct sigaction)];
} siguaction_t;
#ifdef _SYSCALL32
typedef struct {
rwlock_t sig_lock;
struct sigaction32 sig_uaction;
char sig_pad[128 - sizeof (rwlock_t) - sizeof (struct sigaction32)];
} siguaction32_t;
#endif
typedef struct {
mutex_t bucket_lock;
void *free_list;
size_t chunks;
char pad64[64 -
(sizeof (mutex_t) + sizeof (void *) + sizeof (size_t))];
} bucket_t;
#ifdef _SYSCALL32
typedef struct {
mutex_t bucket_lock;
caddr32_t free_list;
size32_t chunks;
char pad64[64 -
(sizeof (mutex_t) + sizeof (caddr32_t) + sizeof (size32_t))];
} bucket32_t;
#endif
#define NBUCKETS 10
typedef void (*_exithdlr_func_t) (void*);
typedef struct _exthdlr {
struct _exthdlr *next;
_exithdlr_func_t hdlr;
void *arg;
void *dso;
} _exthdlr_t;
typedef struct {
mutex_t exitfns_lock;
_exthdlr_t *head;
void *exit_frame_monitor;
char exit_pad[64 -
(sizeof (mutex_t) + sizeof (_exthdlr_t *) + sizeof (void *))];
} atexit_root_t;
#ifdef _SYSCALL32
typedef struct {
mutex_t exitfns_lock;
caddr32_t head;
caddr32_t exit_frame_monitor;
char exit_pad[64 -
(sizeof (mutex_t) + sizeof (caddr32_t) + sizeof (caddr32_t))];
} atexit_root32_t;
#endif
typedef void (*_quick_exithdlr_func_t)(void);
typedef struct _qexthdlr {
struct _qexthdlr *next;
_quick_exithdlr_func_t hdlr;
} _qexthdlr_t;
typedef struct {
mutex_t exitfns_lock;
_qexthdlr_t *head;
#if !defined(_LP64)
uint32_t pad;
#endif
} quickexit_root_t;
#ifdef _SYSCALL32
typedef struct {
mutex_t exitfns_lock;
caddr32_t head;
uint32_t pad;
} quickexit_root32_t;
#endif
typedef struct uberdata {
pad_lock_t _link_lock;
pad_lock_t _ld_lock;
pad_lock_t _fork_lock;
pad_lock_t _atfork_lock;
pad32_lock_t _callout_lock;
pad32_lock_t _tdb_hash_lock;
tdb_sync_stats_t tdb_hash_lock_stats;
siguaction_t siguaction[NSIG];
bucket_t bucket[NBUCKETS];
atexit_root_t atexit_root;
quickexit_root_t quickexit_root;
tsd_metadata_t tsd_metadata;
tls_metadata_t tls_metadata;
char primary_map;
char bucket_init;
char pad[2];
uberflags_t uberflags;
queue_head_t *queue_head;
thr_hash_table_t *thr_hash_table;
uint_t hash_size;
uint_t hash_mask;
ulwp_t *ulwp_one;
ulwp_t *all_lwps;
ulwp_t *all_zombies;
int nthreads;
int nzombies;
int ndaemons;
pid_t pid;
void (*sigacthandler)(int, siginfo_t *, void *);
ulwp_t *lwp_stacks;
ulwp_t *lwp_laststack;
int nfreestack;
int thread_stack_cache;
ulwp_t *ulwp_freelist;
ulwp_t *ulwp_lastfree;
ulwp_t *ulwp_replace_free;
ulwp_t *ulwp_replace_last;
atfork_t *atforklist;
robust_t **robustlocks;
robust_t *robustlist;
char *progname;
void *ub_comm_page;
struct uberdata **tdb_bootstrap;
tdb_t tdb;
} uberdata_t;
#define link_lock _link_lock.pad_lock
#define ld_lock _ld_lock.pad_lock
#define fork_lock _fork_lock.pad_lock
#define atfork_lock _atfork_lock.pad_lock
#define callout_lock _callout_lock.pad_lock
#define tdb_hash_lock _tdb_hash_lock.pad_lock
#pragma align 64(__uberdata)
extern uberdata_t __uberdata;
extern uberdata_t **__tdb_bootstrap;
extern int primary_link_map;
#define ulwp_mutex(ulwp, udp) \
(&(udp)->thr_hash_table[(ulwp)->ul_ix].hash_lock)
#define ulwp_condvar(ulwp, udp) \
(&(udp)->thr_hash_table[(ulwp)->ul_ix].hash_cond)
#define ulwp_lock(ulwp, udp) lmutex_lock(ulwp_mutex(ulwp, udp))
#define ulwp_unlock(ulwp, udp) lmutex_unlock(ulwp_mutex(ulwp, udp))
#ifdef _SYSCALL32
typedef struct ulwp32 {
#if defined(__sparc)
uint32_t ul_dinstr;
uint32_t ul_padsparc0[15];
uint32_t ul_dsave;
uint32_t ul_drestore;
uint32_t ul_dftret;
uint32_t ul_dreturn;
#endif
caddr32_t ul_self;
#if defined(__x86)
uint8_t ul_dinstr[40];
#endif
caddr32_t ul_uberdata;
tls32_t ul_tls;
caddr32_t ul_forw;
caddr32_t ul_back;
caddr32_t ul_next;
caddr32_t ul_hash;
caddr32_t ul_rval;
caddr32_t ul_stk;
size32_t ul_mapsiz;
size32_t ul_guardsize;
caddr32_t ul_stktop;
size32_t ul_stksiz;
stack32_t ul_ustack;
int ul_ix;
lwpid_t ul_lwpid;
pri_t ul_pri;
pri_t ul_epri;
char ul_policy;
char ul_cid;
union {
struct {
char cursig;
char pleasestop;
} s;
short curplease;
} ul_cp;
char ul_stop;
char ul_signalled;
char ul_dead;
char ul_unwind;
char ul_detached;
char ul_writer;
char ul_stopping;
char ul_cancel_prologue;
short ul_preempt;
short ul_savpreempt;
char ul_sigsuspend;
char ul_main;
char ul_fork;
char ul_primarymap;
uint8_t ul_max_spinners;
char ul_door_noreserve;
char ul_queue_fifo;
char ul_cond_wait_defer;
char ul_error_detection;
char ul_async_safe;
char ul_rt;
char ul_rtqueued;
char ul_misaligned;
char ul_pad[3];
int ul_adaptive_spin;
int ul_queue_spin;
int ul_critical;
int ul_sigdefer;
int ul_vfork;
int ul_cancelable;
char ul_cancel_pending;
char ul_cancel_disabled;
char ul_cancel_async;
char ul_save_async;
char ul_mutator;
char ul_created;
char ul_replace;
uchar_t ul_nocancel;
int ul_errno;
caddr32_t ul_errnop;
caddr32_t ul_clnup_hdr;
caddr32_t ul_schedctl_called;
caddr32_t ul_schedctl;
int ul_bindflags;
uint_t ul_libc_locks;
caddr32_t ul_stsd;
caddr32_t ul_ftsd[TSD_NFAST];
td_evbuf32_t ul_td_evbuf;
char ul_td_events_enable;
char ul_sync_obj_reg;
char ul_qtype;
char ul_cv_wake;
int ul_rtld;
int ul_usropts;
caddr32_t ul_startpc;
caddr32_t ul_startarg;
caddr32_t ul_wchan;
caddr32_t ul_link;
caddr32_t ul_sleepq;
caddr32_t ul_cvmutex;
caddr32_t ul_mxchain;
int ul_save_state;
uint_t ul_rdlockcnt;
union {
readlock32_t single;
caddr32_t array;
} ul_readlock;
uint_t ul_heldlockcnt;
union {
caddr32_t single;
caddr32_t array;
} ul_heldlocks;
caddr32_t ul_tpdp;
caddr32_t ul_siglink;
uint_t ul_spin_lock_spin;
uint_t ul_spin_lock_spin2;
uint_t ul_spin_lock_sleep;
uint_t ul_spin_lock_wakeup;
queue_root32_t ul_queue_root;
id_t ul_rtclassid;
uint_t ul_pilocks;
sigset_t ul_sigmask;
sigset_t ul_tmpmask;
siginfo32_t ul_siginfo;
mutex_t ul_spinlock;
fpuenv32_t ul_fpuenv;
caddr32_t ul_sp;
#if defined(sparc)
caddr32_t ul_unwind_ret;
#endif
tumem32_t ul_tmem;
uint_t ul_ptinherit;
char ul_ntoabuf[18];
} ulwp32_t;
#define REPLACEMENT_SIZE32 ((size_t)&((ulwp32_t *)NULL)->ul_sigmask)
typedef struct uberdata32 {
pad_lock_t _link_lock;
pad_lock_t _ld_lock;
pad_lock_t _fork_lock;
pad_lock_t _atfork_lock;
pad32_lock_t _callout_lock;
pad32_lock_t _tdb_hash_lock;
tdb_sync_stats_t tdb_hash_lock_stats;
siguaction32_t siguaction[NSIG];
bucket32_t bucket[NBUCKETS];
atexit_root32_t atexit_root;
quickexit_root32_t quickexit_root;
tsd_metadata32_t tsd_metadata;
tls_metadata32_t tls_metadata;
char primary_map;
char bucket_init;
char pad[2];
uberflags_t uberflags;
caddr32_t queue_head;
caddr32_t thr_hash_table;
uint_t hash_size;
uint_t hash_mask;
caddr32_t ulwp_one;
caddr32_t all_lwps;
caddr32_t all_zombies;
int nthreads;
int nzombies;
int ndaemons;
int pid;
caddr32_t sigacthandler;
caddr32_t lwp_stacks;
caddr32_t lwp_laststack;
int nfreestack;
int thread_stack_cache;
caddr32_t ulwp_freelist;
caddr32_t ulwp_lastfree;
caddr32_t ulwp_replace_free;
caddr32_t ulwp_replace_last;
caddr32_t atforklist;
caddr32_t robustlocks;
caddr32_t robustlist;
caddr32_t progname;
caddr32_t ub_comm_page;
caddr32_t tdb_bootstrap;
tdb32_t tdb;
} uberdata32_t;
#endif
#define TSTP_REGULAR 0x01
#define TSTP_MUTATOR 0x08
#define TSTP_FORK 0x20
typedef struct _cvattr {
int pshared;
clockid_t clockid;
} cvattr_t;
typedef struct _mattr {
int pshared;
int protocol;
int prioceiling;
int type;
int robustness;
} mattr_t;
typedef struct _thrattr {
size_t stksize;
void *stkaddr;
int detachstate;
int daemonstate;
int scope;
int prio;
int policy;
int inherit;
size_t guardsize;
char name[THREAD_NAME_MAX];
} thrattr_t;
typedef struct _rwlattr {
int pshared;
} rwlattr_t;
extern ulwp_t *_curthread(void);
#define curthread (_curthread())
extern ulwp_t *__curthread(void);
extern greg_t stkptr(void);
#if !defined(__GNUC__)
#define __attribute__(string)
#endif
#define real_priority(ulwp) \
((ulwp)->ul_schedctl? (ulwp)->ul_schedctl->sc_priority : 0)
extern int __nanosleep(const timespec_t *, timespec_t *);
extern void getgregs(ulwp_t *, gregset_t);
extern void setgregs(ulwp_t *, gregset_t);
extern void thr_panic(const char *) __NORETURN;
extern void mutex_panic(mutex_t *, const char *) __NORETURN;
extern ulwp_t *find_lwp(thread_t);
extern void finish_init(void);
extern void update_sched(ulwp_t *);
extern void queue_alloc(void);
extern void tmem_exit(void);
extern void tsd_exit(void);
extern void tsd_free(ulwp_t *);
extern void tls_setup(void);
extern void tls_exit(void);
extern void tls_free(ulwp_t *);
extern void rwl_free(ulwp_t *);
extern void heldlock_exit(void);
extern void heldlock_free(ulwp_t *);
extern void sigacthandler(int, siginfo_t *, void *);
extern void signal_init(void);
extern int sigequalset(const sigset_t *, const sigset_t *);
extern void mutex_setup(void);
extern void take_deferred_signal(int);
extern void *setup_top_frame(void *, size_t, ulwp_t *);
extern int setup_context(ucontext_t *, void *(*func)(ulwp_t *),
ulwp_t *ulwp, caddr_t stk, size_t stksize);
extern volatile sc_shared_t *setup_schedctl(void);
extern void *lmalloc(size_t);
extern void lfree(void *, size_t);
extern void *libc_malloc(size_t);
extern void *libc_realloc(void *, size_t);
extern void libc_free(void *);
extern char *libc_strdup(const char *);
extern void ultos(uint64_t, int, char *);
extern void lock_error(const mutex_t *, const char *, void *, const char *);
extern void rwlock_error(const rwlock_t *, const char *, const char *);
extern void thread_error(const char *);
extern void grab_assert_lock(void);
extern void dump_queue_statistics(void);
extern void collect_queue_statistics(void);
extern void record_spin_locks(ulwp_t *);
extern void remember_lock(mutex_t *);
extern void forget_lock(mutex_t *);
extern void register_lock(mutex_t *);
extern void unregister_locks(void);
#if defined(__sparc)
extern void _flush_windows(void);
#else
#define _flush_windows()
#endif
extern void set_curthread(void *);
extern void ssp_init(void);
extern int sema_relclockwait(sema_t *, clockid_t, const timespec_t *);
extern int sema_clockwait(sema_t *, clockid_t, const timespec_t *);
#define MAXLWPS 128
#define NEWLWPS 2048
extern lwpid_t *alloc_lwpids(lwpid_t *, int *, int *);
#define enter_critical(self) (self->ul_critical++)
extern void do_exit_critical(void);
#define exit_critical(self) \
(void) (self->ul_critical--, \
((self->ul_curplease && self->ul_critical == 0)? \
(do_exit_critical(), 0) : 0))
#define sigoff(self) (self->ul_sigdefer++)
#define sigon(self) \
(void) ((--self->ul_sigdefer == 0 && \
self->ul_curplease && self->ul_critical == 0)? \
(do_exit_critical(), 0) : 0)
extern void _sigoff(void);
extern void _sigon(void);
#define sigorset(s1, s2) \
(((s1)->__sigbits[0] |= (s2)->__sigbits[0]), \
((s1)->__sigbits[1] |= (s2)->__sigbits[1]), \
((s1)->__sigbits[2] |= (s2)->__sigbits[2]), \
((s1)->__sigbits[3] |= (s2)->__sigbits[3]))
#define sigandset(s1, s2) \
(((s1)->__sigbits[0] &= (s2)->__sigbits[0]), \
((s1)->__sigbits[1] &= (s2)->__sigbits[1]), \
((s1)->__sigbits[2] &= (s2)->__sigbits[2]), \
((s1)->__sigbits[3] &= (s2)->__sigbits[3]))
#define sigdiffset(s1, s2) \
(((s1)->__sigbits[0] &= ~(s2)->__sigbits[0]), \
((s1)->__sigbits[1] &= ~(s2)->__sigbits[1]), \
((s1)->__sigbits[2] &= ~(s2)->__sigbits[2]), \
((s1)->__sigbits[3] &= ~(s2)->__sigbits[3]))
#define delete_reserved_signals(s) \
(((s)->__sigbits[0] &= MASKSET0), \
((s)->__sigbits[1] &= (MASKSET1 & ~SIGMASK(SIGCANCEL))),\
((s)->__sigbits[2] &= MASKSET2), \
((s)->__sigbits[3] &= MASKSET3))
extern void block_all_signals(ulwp_t *self);
#define restore_signals(self) \
((void) ((self)->ul_cursig? \
(ASSERT((self)->ul_critical + (self)->ul_sigdefer != 0), 0) : \
__lwp_sigmask(SIG_SETMASK, &(self)->ul_sigmask)))
extern void set_cancel_pending_flag(ulwp_t *, int);
extern void set_cancel_eintr_flag(ulwp_t *);
extern void set_parking_flag(ulwp_t *, int);
extern int cancel_active(void);
extern void *_thrp_setup(ulwp_t *);
extern void _fpinherit(ulwp_t *);
extern void _lwp_start(void);
extern void _lwp_terminate(void);
extern void lmutex_lock(mutex_t *);
extern void lmutex_unlock(mutex_t *);
extern void lrw_rdlock(rwlock_t *);
extern void lrw_wrlock(rwlock_t *);
extern void lrw_unlock(rwlock_t *);
extern void sig_mutex_lock(mutex_t *);
extern void sig_mutex_unlock(mutex_t *);
extern int sig_mutex_trylock(mutex_t *);
extern int sig_cond_wait(cond_t *, mutex_t *);
extern int sig_cond_reltimedwait(cond_t *, mutex_t *, const timespec_t *);
extern void cancel_safe_mutex_lock(mutex_t *);
extern void cancel_safe_mutex_unlock(mutex_t *);
extern int cancel_safe_mutex_trylock(mutex_t *);
extern void _prefork_handler(void);
extern void _postfork_parent_handler(void);
extern void _postfork_child_handler(void);
extern void postfork1_child(void);
extern void postfork1_child_aio(void);
extern void postfork1_child_sigev_aio(void);
extern void postfork1_child_sigev_mq(void);
extern void postfork1_child_sigev_timer(void);
extern void postfork1_child_tpool(void);
extern void fork_lock_enter(void);
extern void fork_lock_exit(void);
extern void suspend_fork(void);
extern void continue_fork(int);
extern void do_sigcancel(void);
extern void setup_cancelsig(int);
extern void init_sigev_thread(void);
extern void init_aio(void);
extern void init_progname(void);
extern void _cancelon(void);
extern void _canceloff(void);
extern void _canceloff_nocancel(void);
extern void _cancel_prologue(void);
extern void _cancel_epilogue(void);
extern void no_preempt(ulwp_t *);
extern void preempt(ulwp_t *);
extern void _thrp_unwind(void *);
extern pid_t __forkx(int);
extern pid_t __forkallx(int);
extern int __open(const char *, int, mode_t);
extern int __open64(const char *, int, mode_t);
extern int __openat(int, const char *, int, mode_t);
extern int __openat64(int, const char *, int, mode_t);
extern int __close(int);
extern ssize_t __read(int, void *, size_t);
extern ssize_t __write(int, const void *, size_t);
extern int __fcntl(int, int, ...);
extern int __lwp_continue(lwpid_t);
extern int __lwp_create(ucontext_t *, uint_t, lwpid_t *);
extern int ___lwp_suspend(lwpid_t);
extern int lwp_wait(lwpid_t, lwpid_t *);
extern int __lwp_wait(lwpid_t, lwpid_t *);
extern int __lwp_detach(lwpid_t);
extern sc_shared_t *__schedctl(void);
extern int __setcontext(const ucontext_t *);
extern int __getcontext(ucontext_t *);
extern int __clock_gettime(clockid_t, timespec_t *);
extern void abstime_to_reltime(clockid_t, const timespec_t *, timespec_t *);
extern void hrt2ts(hrtime_t, timespec_t *);
extern int __sigaction(int, const struct sigaction *, struct sigaction *);
extern int __sigprocmask(int, const sigset_t *, sigset_t *);
extern int __lwp_sigmask(int, const sigset_t *);
extern void __sighndlr(int, siginfo_t *, ucontext_t *, void (*)());
extern caddr_t __sighndlrend;
#define PTHREAD_CREATE_DAEMON_NP 0x100
#define PTHREAD_CREATE_NONDAEMON_NP 0
extern int pthread_attr_setdaemonstate_np(pthread_attr_t *, int);
extern int pthread_attr_getdaemonstate_np(const pthread_attr_t *, int *);
extern int mutex_held(mutex_t *);
extern int mutex_lock_internal(mutex_t *, timespec_t *, int);
extern int mutex_unlock_internal(mutex_t *, int);
extern int __cond_wait(cond_t *, mutex_t *);
extern int __cond_timedwait(cond_t *, mutex_t *, clockid_t,
const timespec_t *);
extern int __cond_reltimedwait(cond_t *, mutex_t *, const timespec_t *);
extern int rw_read_held(rwlock_t *);
extern int rw_write_held(rwlock_t *);
extern int _thrp_create(void *, size_t, void *(*)(void *), void *, long,
thread_t *, size_t, const char *);
extern int _thrp_suspend(thread_t, uchar_t);
extern int _thrp_continue(thread_t, uchar_t);
extern void _thrp_terminate(void *);
extern void _thrp_exit(void);
extern const pcclass_t *get_info_by_class(id_t);
extern const pcclass_t *get_info_by_policy(int);
extern const thrattr_t *def_thrattr(void);
extern id_t setparam(idtype_t, id_t, int, int);
extern id_t setprio(idtype_t, id_t, int, int *);
extern id_t getparam(idtype_t, id_t, int *, struct sched_param *);
extern int ___lwp_mutex_register(mutex_t *, mutex_t **);
extern int ___lwp_mutex_trylock(mutex_t *, ulwp_t *);
extern int ___lwp_mutex_timedlock(mutex_t *, timespec_t *, ulwp_t *);
extern int ___lwp_mutex_unlock(mutex_t *);
extern int ___lwp_mutex_wakeup(mutex_t *, int);
extern int ___lwp_cond_wait(cond_t *, mutex_t *, timespec_t *, int);
extern int ___lwp_sema_timedwait(lwp_sema_t *, timespec_t *, int);
extern int __lwp_rwlock_rdlock(rwlock_t *, timespec_t *);
extern int __lwp_rwlock_wrlock(rwlock_t *, timespec_t *);
extern int __lwp_rwlock_tryrdlock(rwlock_t *);
extern int __lwp_rwlock_trywrlock(rwlock_t *);
extern int __lwp_rwlock_unlock(rwlock_t *);
extern int __lwp_park(timespec_t *, lwpid_t);
extern int __lwp_unpark(lwpid_t);
extern int __lwp_unpark_all(lwpid_t *, int);
#if defined(__x86)
extern int ___lwp_private(int, int, void *);
#endif
extern int set_lock_byte(volatile uint8_t *);
extern uint32_t atomic_swap_32(volatile uint32_t *, uint32_t);
extern uint32_t atomic_cas_32(volatile uint32_t *, uint32_t, uint32_t);
extern void atomic_inc_32(volatile uint32_t *);
extern void atomic_dec_32(volatile uint32_t *);
extern void atomic_and_32(volatile uint32_t *, uint32_t);
extern void atomic_or_32(volatile uint32_t *, uint32_t);
#if defined(__sparc)
extern ulong_t caller(void);
extern ulong_t getfp(void);
#endif
#include "thr_inlines.h"
#endif