/* * %CopyrightBegin% * * Copyright Ericsson AB 2007-2009. All Rights Reserved. * * The contents of this file are subject to the Erlang Public License, * Version 1.1, (the "License"); you may not use this file except in * compliance with the License. You should have received a copy of the * Erlang Public License along with this software. If not, it can be * retrieved online at http://www.erlang.org/. * * Software distributed under the License is distributed on an "AS IS" * basis, WITHOUT WARRANTY OF ANY KIND, either express or implied. See * the License for the specific language governing rights and limitations * under the License. * * %CopyrightEnd% */ #ifdef HAVE_CONFIG_H # include "config.h" #endif #include "global.h" #include <string.h> #define ERL_DRV_THR_OPTS_SIZE(LAST_FIELD) \ (((size_t) &((ErlDrvThreadOpts *) 0)->LAST_FIELD) \ + sizeof(((ErlDrvThreadOpts *) 0)->LAST_FIELD)) static void fatal_error(int err, char *func) { char *estr = strerror(err); if (!estr) { if (err == ENOTSUP) estr = "Not supported"; else estr = "Unknown error"; } erl_exit(ERTS_ABORT_EXIT, "Fatal error in %s: %s [%d]\n", func, estr, err); } #define ERL_DRV_TSD_KEYS_INC 10 #define ERL_DRV_TSD_EXTRA 10 #define ERL_DRV_INVALID_TSD_KEY INT_MAX #ifdef USE_THREADS struct ErlDrvMutex_ { ethr_mutex mtx; char *name; }; struct ErlDrvCond_ { ethr_cond cnd; char *name; }; struct ErlDrvRWLock_ { ethr_rwmutex rwmtx; char *name; }; struct ErlDrvTid_ { ethr_tid tid; void* (*func)(void*); void* arg; int drv_thr; Uint tsd_len; void **tsd; char *name; }; static ethr_tsd_key tid_key; static ethr_thr_opts def_ethr_opts = ETHR_THR_OPTS_DEFAULT_INITER; #else /* USE_THREADS */ static Uint tsd_len; static void **tsd; #endif static ErlDrvTSDKey next_tsd_key; static ErlDrvTSDKey max_used_tsd_key; static ErlDrvTSDKey used_tsd_keys_len; static char **used_tsd_keys; static erts_mtx_t tsd_mtx; static char *no_name; #ifdef USE_THREADS static void thread_exit_handler(void) { struct ErlDrvTid_ *dtid = ethr_tsd_get(tid_key); if (dtid) { if (dtid->tsd) erts_free(ERTS_ALC_T_DRV_TSD, dtid->tsd); if (!dtid->drv_thr) erts_free(ERTS_ALC_T_DRV_TID, dtid); } } static void * erl_drv_thread_wrapper(void *vdtid) { int res; struct ErlDrvTid_ *dtid = (struct ErlDrvTid_ *) vdtid; res = ethr_tsd_set(tid_key, vdtid); if (res != 0) fatal_error(res, "erl_drv_thread_wrapper()"); return (*dtid->func)(dtid->arg); } #endif void erl_drv_thr_init(void) { int i; #ifdef USE_THREADS int res = ethr_tsd_key_create(&tid_key); if (res == 0) res = ethr_install_exit_handler(thread_exit_handler); if (res != 0) fatal_error(res, "erl_drv_thr_init()"); #else tsd_len = 0; tsd = NULL; #endif no_name = "unknown"; next_tsd_key = 0; max_used_tsd_key = -1; used_tsd_keys_len = ERL_DRV_TSD_KEYS_INC; used_tsd_keys = erts_alloc(ERTS_ALC_T_DRV_TSD, sizeof(char *)*ERL_DRV_TSD_KEYS_INC); for (i = 0; i < ERL_DRV_TSD_KEYS_INC; i++) used_tsd_keys[i] = NULL; erts_mtx_init(&tsd_mtx, "drv_tsd"); } /* * These functions implement the driver thread interface in erl_driver.h. * NOTE: Only use this interface from drivers. From within the emulator use * either the erl_threads.h, the erl_smp.h or the ethread.h interface. */ ErlDrvMutex * erl_drv_mutex_create(char *name) { #ifdef USE_THREADS ErlDrvMutex *dmtx = erts_alloc_fnf(ERTS_ALC_T_DRV_MTX, (sizeof(ErlDrvMutex) + (name ? sys_strlen(name) + 1 : 0))); if (dmtx) { if (ethr_mutex_init(&dmtx->mtx) != 0) { erts_free(ERTS_ALC_T_DRV_MTX, (void *) dmtx); dmtx = NULL; } else if (!name) dmtx->name = no_name; else { dmtx->name = ((char *) dmtx) + sizeof(ErlDrvMutex); sys_strcpy(dmtx->name, name); } } return dmtx; #else return (ErlDrvMutex *) NULL; #endif } void erl_drv_mutex_destroy(ErlDrvMutex *dmtx) { #ifdef USE_THREADS int res = dmtx ? ethr_mutex_destroy(&dmtx->mtx) : EINVAL; if (res != 0) fatal_error(res, "erl_drv_mutex_destroy()"); erts_free(ERTS_ALC_T_DRV_MTX, (void *) dmtx); #endif } int erl_drv_mutex_trylock(ErlDrvMutex *dmtx) { #ifdef USE_THREADS int res = dmtx ? ethr_mutex_trylock(&dmtx->mtx) : EINVAL; if (res != 0 && res != EBUSY) fatal_error(res, "erl_drv_mutex_trylock()"); return res; #else return 0; #endif } void erl_drv_mutex_lock(ErlDrvMutex *dmtx) { #ifdef USE_THREADS int res = dmtx ? ethr_mutex_lock(&dmtx->mtx) : EINVAL; if (res != 0) fatal_error(res, "erl_drv_mutex_lock()"); #endif } void erl_drv_mutex_unlock(ErlDrvMutex *dmtx) { #ifdef USE_THREADS int res = dmtx ? ethr_mutex_unlock(&dmtx->mtx) : EINVAL; if (res != 0) fatal_error(res, "erl_drv_mutex_unlock()"); #endif } ErlDrvCond * erl_drv_cond_create(char *name) { #ifdef USE_THREADS ErlDrvCond *dcnd = erts_alloc_fnf(ERTS_ALC_T_DRV_CND, (sizeof(ErlDrvCond) + (name ? sys_strlen(name) + 1 : 0))); if (dcnd) { if (ethr_cond_init(&dcnd->cnd) != 0) { erts_free(ERTS_ALC_T_DRV_CND, (void *) dcnd); dcnd = NULL; } else if (!name) dcnd->name = no_name; else { dcnd->name = ((char *) dcnd) + sizeof(ErlDrvCond); sys_strcpy(dcnd->name, name); } } return dcnd; #else return (ErlDrvCond *) NULL; #endif } void erl_drv_cond_destroy(ErlDrvCond *dcnd) { #ifdef USE_THREADS int res = dcnd ? ethr_cond_destroy(&dcnd->cnd) : EINVAL; if (res != 0) fatal_error(res, "erl_drv_cond_destroy()"); erts_free(ERTS_ALC_T_DRV_CND, (void *) dcnd); #endif } void erl_drv_cond_signal(ErlDrvCond *dcnd) { #ifdef USE_THREADS int res = dcnd ? ethr_cond_signal(&dcnd->cnd) : EINVAL; if (res != 0) fatal_error(res, "erl_drv_cond_signal()"); #endif } void erl_drv_cond_broadcast(ErlDrvCond *dcnd) { #ifdef USE_THREADS int res = dcnd ? ethr_cond_broadcast(&dcnd->cnd) : EINVAL; if (res != 0) fatal_error(res, "erl_drv_cond_broadcast()"); #endif } void erl_drv_cond_wait(ErlDrvCond *dcnd, ErlDrvMutex *dmtx) { #ifdef USE_THREADS int res; if (!dcnd || !dmtx) { res = EINVAL; error: fatal_error(res, "erl_drv_cond_wait()"); } while (1) { res = ethr_cond_wait(&dcnd->cnd, &dmtx->mtx); if (res == 0) break; if (res != EINTR) goto error; } #endif } ErlDrvRWLock * erl_drv_rwlock_create(char *name) { #ifdef USE_THREADS ErlDrvRWLock *drwlck = erts_alloc_fnf(ERTS_ALC_T_DRV_RWLCK, (sizeof(ErlDrvRWLock) + (name ? sys_strlen(name) + 1 : 0))); if (drwlck) { if (ethr_rwmutex_init(&drwlck->rwmtx) != 0) { erts_free(ERTS_ALC_T_DRV_RWLCK, (void *) drwlck); drwlck = NULL; } else if (!name) drwlck->name = no_name; else { drwlck->name = ((char *) drwlck) + sizeof(ErlDrvRWLock); sys_strcpy(drwlck->name, name); } } return drwlck; #else return (ErlDrvRWLock *) NULL; #endif } void erl_drv_rwlock_destroy(ErlDrvRWLock *drwlck) { #ifdef USE_THREADS int res = drwlck ? ethr_rwmutex_destroy(&drwlck->rwmtx) : EINVAL; if (res != 0) fatal_error(res, "erl_drv_rwlock_destroy()"); erts_free(ERTS_ALC_T_DRV_RWLCK, (void *) drwlck); #endif } int erl_drv_rwlock_tryrlock(ErlDrvRWLock *drwlck) { #ifdef USE_THREADS int res = drwlck ? ethr_rwmutex_tryrlock(&drwlck->rwmtx) : EINVAL; if (res != 0 && res != EBUSY) fatal_error(res, "erl_drv_rwlock_tryrlock()"); return res; #else return 0; #endif } void erl_drv_rwlock_rlock(ErlDrvRWLock *drwlck) { #ifdef USE_THREADS int res = drwlck ? ethr_rwmutex_rlock(&drwlck->rwmtx) : EINVAL; if (res != 0) fatal_error(res, "erl_drv_rwlock_rlock()"); #endif } void erl_drv_rwlock_runlock(ErlDrvRWLock *drwlck) { #ifdef USE_THREADS int res = drwlck ? ethr_rwmutex_runlock(&drwlck->rwmtx) : EINVAL; if (res != 0) fatal_error(res, "erl_drv_rwlock_runlock()"); #endif } int erl_drv_rwlock_tryrwlock(ErlDrvRWLock *drwlck) { #ifdef USE_THREADS int res = drwlck ? ethr_rwmutex_tryrwlock(&drwlck->rwmtx) : EINVAL; if (res != 0 && res != EBUSY) fatal_error(res, "erl_drv_rwlock_tryrwlock()"); return res; #else return 0; #endif } void erl_drv_rwlock_rwlock(ErlDrvRWLock *drwlck) { #ifdef USE_THREADS int res = drwlck ? ethr_rwmutex_rwlock(&drwlck->rwmtx) : EINVAL; if (res != 0) fatal_error(res, "erl_drv_rwlock_rwlock()"); #endif } void erl_drv_rwlock_rwunlock(ErlDrvRWLock *drwlck) { #ifdef USE_THREADS int res = drwlck ? ethr_rwmutex_rwunlock(&drwlck->rwmtx) : EINVAL; if (res != 0) fatal_error(res, "erl_drv_rwlock_rwunlock()"); #endif } int erl_drv_tsd_key_create(char *name, ErlDrvTSDKey *key) { char *name_copy; Uint old_used_tsd_keys_len; ErlDrvTSDKey res; if (!key) fatal_error(EINVAL, "erl_drv_tsd_key_create()"); if (!name) name_copy = no_name; else { name_copy = erts_alloc_fnf(ERTS_ALC_T_DRV_TSD, sizeof(char)*(strlen(name) + 1)); if (!name_copy) { *key = -1; return ENOMEM; } sys_strcpy(name_copy, name); } erts_mtx_lock(&tsd_mtx); *key = next_tsd_key; if (next_tsd_key < 0) res = ENOMEM; else { res = 0; ASSERT(!used_tsd_keys[next_tsd_key]); used_tsd_keys[next_tsd_key] = name_copy; if (max_used_tsd_key < next_tsd_key) max_used_tsd_key = next_tsd_key; if (max_used_tsd_key + 1 >= used_tsd_keys_len) { int i; old_used_tsd_keys_len = used_tsd_keys_len; if (used_tsd_keys_len + ERL_DRV_TSD_KEYS_INC >= INT_MAX) next_tsd_key = -1; else { char **new_used_tsd_keys; used_tsd_keys_len += ERL_DRV_TSD_KEYS_INC; new_used_tsd_keys = erts_realloc_fnf(ERTS_ALC_T_DRV_TSD, used_tsd_keys, (sizeof(char *) * used_tsd_keys_len)); if (!new_used_tsd_keys) next_tsd_key = -1; else { used_tsd_keys = new_used_tsd_keys; for (i = old_used_tsd_keys_len; i < used_tsd_keys_len; i++) used_tsd_keys[i] = NULL; } } } if (next_tsd_key >= 0) { do { next_tsd_key++; } while (used_tsd_keys[next_tsd_key]); } ASSERT(next_tsd_key < used_tsd_keys_len); } erts_mtx_unlock(&tsd_mtx); return res; } void erl_drv_tsd_key_destroy(ErlDrvTSDKey key) { erts_mtx_lock(&tsd_mtx); if (key < 0 || max_used_tsd_key < key || !used_tsd_keys[key]) fatal_error(EINVAL, "erl_drv_tsd_key_destroy()"); if (used_tsd_keys[key] != no_name) erts_free(ERTS_ALC_T_DRV_TSD, used_tsd_keys[key]); used_tsd_keys[key] = NULL; if (next_tsd_key < 0 || key < next_tsd_key) next_tsd_key = key; erts_mtx_unlock(&tsd_mtx); } #ifdef USE_THREADS #define ERL_DRV_TSD__ (dtid->tsd) #define ERL_DRV_TSD_LEN__ (dtid->tsd_len) #else #define ERL_DRV_TSD__ (tsd) #define ERL_DRV_TSD_LEN__ (tsd_len) #endif void erl_drv_tsd_set(ErlDrvTSDKey key, void *data) { #ifdef USE_THREADS struct ErlDrvTid_ *dtid = (struct ErlDrvTid_ *) erl_drv_thread_self(); #endif if (key < 0 || max_used_tsd_key < key || !used_tsd_keys[key]) fatal_error(EINVAL, "erl_drv_tsd_set()"); if (!ERL_DRV_TSD__) { ErlDrvTSDKey i; ERL_DRV_TSD_LEN__ = key + ERL_DRV_TSD_EXTRA; ERL_DRV_TSD__ = erts_alloc(ERTS_ALC_T_DRV_TSD, sizeof(void *)*ERL_DRV_TSD_LEN__); for (i = 0; i < ERL_DRV_TSD_LEN__; i++) ERL_DRV_TSD__[i] = NULL; } else if (ERL_DRV_TSD_LEN__ <= key) { ErlDrvTSDKey i = ERL_DRV_TSD_LEN__; ERL_DRV_TSD_LEN__ = key + ERL_DRV_TSD_EXTRA; ERL_DRV_TSD__ = erts_realloc(ERTS_ALC_T_DRV_TSD, ERL_DRV_TSD__, sizeof(void *)*ERL_DRV_TSD_LEN__); for (; i < ERL_DRV_TSD_LEN__; i++) ERL_DRV_TSD__[i] = NULL; } ERL_DRV_TSD__[key] = data; } void * erl_drv_tsd_get(ErlDrvTSDKey key) { #ifdef USE_THREADS struct ErlDrvTid_ *dtid = ethr_tsd_get(tid_key); #endif if (key < 0 || max_used_tsd_key < key || !used_tsd_keys[key]) fatal_error(EINVAL, "erl_drv_tsd_get()"); #ifdef USE_THREADS if (!dtid) return NULL; #endif if (ERL_DRV_TSD_LEN__ < key) return NULL; return ERL_DRV_TSD__[key]; } #undef ERL_DRV_TSD_LEN__ #undef ERL_DRV_TSD__ ErlDrvThreadOpts * erl_drv_thread_opts_create(char *name) { ErlDrvThreadOpts *opts = erts_alloc_fnf(ERTS_ALC_T_DRV_THR_OPTS, sizeof(ErlDrvThreadOpts)); if (!opts) return NULL; opts->suggested_stack_size = -1; return opts; } void erl_drv_thread_opts_destroy(ErlDrvThreadOpts *opts) { if (!opts) fatal_error(EINVAL, "erl_drv_thread_opts_destroy()"); erts_free(ERTS_ALC_T_DRV_THR_OPTS, opts); } int erl_drv_thread_create(char *name, ErlDrvTid *tid, void* (*func)(void*), void* arg, ErlDrvThreadOpts *opts) { #ifdef USE_THREADS int res; struct ErlDrvTid_ *dtid; ethr_thr_opts ethr_opts; ethr_thr_opts *use_opts; if (!opts) use_opts = NULL; else { sys_memcpy((void *) ðr_opts, (void *) &def_ethr_opts, sizeof(ethr_thr_opts)); ethr_opts.suggested_stack_size = opts->suggested_stack_size; use_opts = ðr_opts; } dtid = erts_alloc_fnf(ERTS_ALC_T_DRV_TID, (sizeof(struct ErlDrvTid_) + (name ? sys_strlen(name) + 1 : 0))); if (!dtid) return ENOMEM; dtid->drv_thr = 1; dtid->func = func; dtid->arg = arg; dtid->tsd = NULL; dtid->tsd_len = 0; if (!name) dtid->name = no_name; else { dtid->name = ((char *) dtid) + sizeof(struct ErlDrvTid_); sys_strcpy(dtid->name, name); } #ifdef ERTS_ENABLE_LOCK_COUNT res = erts_lcnt_thr_create(&dtid->tid, erl_drv_thread_wrapper, dtid, use_opts); #else res = ethr_thr_create(&dtid->tid, erl_drv_thread_wrapper, dtid, use_opts); #endif if (res != 0) { erts_free(ERTS_ALC_T_DRV_TID, dtid); return res; } *tid = (ErlDrvTid) dtid; return 0; #else return ENOTSUP; #endif } ErlDrvTid erl_drv_thread_self(void) { #ifdef USE_THREADS struct ErlDrvTid_ *dtid = ethr_tsd_get(tid_key); if (!dtid) { int res; /* This is a thread not spawned by this interface. thread_exit_handler() will clean it up when it terminates. */ dtid = erts_alloc(ERTS_ALC_T_DRV_TID, sizeof(struct ErlDrvTid_)); dtid->drv_thr = 0; /* Not a driver thread */ dtid->tid = ethr_self(); dtid->func = NULL; dtid->arg = NULL; dtid->tsd = NULL; dtid->tsd_len = 0; dtid->name = no_name; res = ethr_tsd_set(tid_key, (void *) dtid); if (res != 0) fatal_error(res, "erl_drv_thread_self()"); } return (ErlDrvTid) dtid; #else return (ErlDrvTid) NULL; #endif } int erl_drv_equal_tids(ErlDrvTid tid1, ErlDrvTid tid2) { #ifdef USE_THREADS int res; struct ErlDrvTid_ *dtid1 = (struct ErlDrvTid_ *) tid1; struct ErlDrvTid_ *dtid2 = (struct ErlDrvTid_ *) tid2; if (!dtid1 || !dtid2) fatal_error(EINVAL, "erl_drv_equal_tids()"); res = dtid1 == dtid2; ASSERT(res ? ethr_equal_tids(dtid1->tid, dtid2->tid) : !ethr_equal_tids(dtid1->tid, dtid2->tid)); return res; #else return 1; #endif } void erl_drv_thread_exit(void *res) { #ifdef USE_THREADS struct ErlDrvTid_ *dtid = ethr_tsd_get(tid_key); if (dtid && dtid->drv_thr) { ethr_thr_exit(res); fatal_error(0, "erl_drv_thread_exit()"); } #endif fatal_error(EACCES, "erl_drv_thread_exit()"); } int erl_drv_thread_join(ErlDrvTid tid, void **respp) { #ifdef USE_THREADS int res; struct ErlDrvTid_ *dtid = (struct ErlDrvTid_ *) tid; ASSERT(dtid); if (!dtid->drv_thr) return EINVAL; res = ethr_thr_join(dtid->tid, respp); if (res == 0) erts_free(ERTS_ALC_T_DRV_TID, dtid); return res; #else return ENOTSUP; #endif }