NetBSD/sys/compat/common/compat_sigaltstack.h
2011-06-05 09:37:10 +00:00

57 lines
1.7 KiB
C

/* $NetBSD: compat_sigaltstack.h,v 1.3 2011/06/05 09:37:10 dsl Exp $ */
/* Wrapper for calling sigaltstack1() from compat (or other) code */
/* Maybe these definitions could be global. */
#ifdef SCARG_P32
/* compat32 */
#define SCARG_COMPAT_PTR(uap,p) SCARG_P32(uap, p)
#define COMPAT_GET_PTR(p) NETBSD32PTR64(p)
#define COMPAT_SET_PTR(p, v) NETBSD32PTR32(p, v)
#else
/* not a size change */
#define SCARG_COMPAT_PTR(uap,p) SCARG(uap, p)
#define COMPAT_GET_PTR(p) (p)
#define COMPAT_SET_PTR(p, v) ((p) = (v))
#endif
#define compat_sigaltstack(uap, compat_ss, ss_onstack, ss_disable) do { \
struct compat_ss css; \
struct sigaltstack nss, oss; \
int error; \
\
if (SCARG_COMPAT_PTR(uap, nss)) { \
error = copyin(SCARG_COMPAT_PTR(uap, nss), &css, sizeof css); \
if (error) \
return error; \
nss.ss_sp = COMPAT_GET_PTR(css.ss_sp); \
nss.ss_size = css.ss_size; \
if (ss_onstack == SS_ONSTACK && ss_disable == SS_DISABLE) \
nss.ss_flags = css.ss_flags; \
else \
nss.ss_flags = \
(css.ss_flags & ss_onstack ? SS_ONSTACK : 0) \
| (css.ss_flags & ss_disable ? SS_DISABLE : 0); \
} \
\
error = sigaltstack1(curlwp, SCARG_COMPAT_PTR(uap, nss) ? &nss : 0, \
SCARG_COMPAT_PTR(uap, oss) ? &oss : 0); \
if (error) \
return (error); \
\
if (SCARG_COMPAT_PTR(uap, oss)) { \
COMPAT_SET_PTR(css.ss_sp, oss.ss_sp); \
css.ss_size = oss.ss_size; \
if (ss_onstack == SS_ONSTACK && ss_disable == SS_DISABLE) \
css.ss_flags = oss.ss_flags; \
else \
css.ss_flags = \
(oss.ss_flags & SS_ONSTACK ? ss_onstack : 0) \
| (oss.ss_flags & SS_DISABLE ? ss_disable : 0); \
error = copyout(&css, SCARG_COMPAT_PTR(uap, oss), sizeof(css));\
if (error) \
return (error); \
} \
return (0); \
} while (0)