diff options
author | Kaz Kylheku <kaz@kylheku.com> | 2015-10-25 11:33:21 -0700 |
---|---|---|
committer | Kaz Kylheku <kaz@kylheku.com> | 2015-10-25 11:33:21 -0700 |
commit | 09f172c98f79f52a6e5c0e4ec87918a6e74aaccf (patch) | |
tree | badd09ef01db2cfd8ce77fd6e86ada747319aa8f /jmp.S | |
parent | 329318ae4f9c180ed68ec484bb5e1acdcb3dec44 (diff) | |
download | txr-09f172c98f79f52a6e5c0e4ec87918a6e74aaccf.tar.gz txr-09f172c98f79f52a6e5c0e4ec87918a6e74aaccf.tar.bz2 txr-09f172c98f79f52a6e5c0e4ec87918a6e74aaccf.zip |
Stop using C library setjmp/longjmp.
TXR is moving to custom assembly-language routines.
This is mainly motivated by a very dubious thing done in the
GNU C Library setjmp and longjmp in the name of security.
Evidently, glibc's setjmp "mangles" certain pointer values
which are stored into the jmp_buf buffer. It's been that way
since 2005, evidently. This means that, firstly, all along,
the use of setjmp in gc.c to get registers into a buffer so
they can be scanned has not actually worked properly. More
importantly, this pointer mangling in setjmp and longjmp is
very hostile to a stack copying implementation of delimited
continuations. The reason is that continuations contain
jmp_buf buffers, which get relocated in the process of
capturing and reviving a continuation. Any pointers in a
jmp_buf which point into the captured stack segment have to be
fixed up to point into the relocated location. Mangled
pointers make this difficult, requiring hacks which are
specific to glibc and the machine architecture. We might as
well implement a clean, well-behaved setjmp and longjmp.
* Makefile (jmp.o): New object file.
(dbg/%.o, opt/%.o): New rules for .S prerequisites.
* args.c, arith.c, cadr.c, combi.c, cadr.c, combi.c, debug.c,
eval.c, filter.c, glob.c, hash.c, lib.c, match.c, parser.c,
rand.c, regex.c, signal.c, stream.c, struct.c, sysif.c,
syslog.c, txr.c, unwind.c, utf8.c: Removed <setjmp.h>
include.
* gc.c: Switch to struct jmp and jmp_save, instead
of jmp_buf and setjmp.
* jmp.S: New source file.
* signal.h (struct jmp): New struct type.
(jmp_save, jmp_restore): New function declarations
denoting assembly language routines in jmp.S.
(extended_jmp_buf): Uses struct jmp instead of
setjmp.
(extended_setjmp): Use jmp_save instead of setjmp.
(extended_longjmp): Use jmp_restore instead of
longjmp.
Diffstat (limited to 'jmp.S')
-rw-r--r-- | jmp.S | 180 |
1 files changed, 180 insertions, 0 deletions
@@ -0,0 +1,180 @@ +/* Copyright 2009-2015 + * Kaz Kylheku <kaz@kylheku.com> + * Vancouver, Canada + * All rights reserved. + * + * Redistribution of this software in source and binary forms, with or without + * modification, is permitted provided that the following two conditions are met. + * + * Use of this software in any manner constitutes agreement with the disclaimer + * which follows the two conditions. + * + * 1. Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * 2. Redistributions in binary form must reproduce the above copyright + * notice, this list of conditions and the following disclaimer in + * the documentation and/or other materials provided with the + * distribution. + * + * THIS SOFTWARE IS PROVIDED ``AS IS'' AND WITHOUT ANY EXPRESS OR IMPLIED + * WARRANTIES, INCLUDING, WITHOUT LIMITATION, THE IMPLIED WARRANTIES OF + * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE. IN NO EVENT SHALL THE + * COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DAMAGES, HOWEVER CAUSED, + * AND UNDER ANY THEORY OF LIABILITY, ARISING IN ANY WAY OUT OF THE USE OF THIS + * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + */ + +#if __MINGW32__ || __CYGWIN32__ +#define DEFUN(NAME) \ +.global _ ## NAME ; \ +_ ## NAME: ; +#elif __APPLE__ +#define DEFUN(NAME) \ +.globl _ ## NAME ; \ +_ ## NAME: ; +#elif __arm__ && !__thumb__ +#define DEFUN(NAME) \ +.text ; \ +.align 4 ; \ +.global NAME ; \ +.type NAME, %function ; \ +NAME: ; +#elif __arm__ && __thumb__ +#define DEFUN(NAME) \ +.text ; \ +.align 4 ; \ +.global NAME ; \ +.thumb ; \ +.thumb_func ; \ +.type NAME, %function ; \ +NAME: ; +#else +#define DEFUN(NAME) \ +.global NAME ; \ +.type NAME, @function ; \ +NAME: ; +#endif + +#if __i386__ + +#define JEIP 0 +#define JESP 4 +#define JEBP 8 +#define JEBX 12 +#define JESI 16 +#define JEDI 20 + +#define RETA 0 +#define ARG1 4 +#define ARG2 8 + +DEFUN(jmp_save) + movl RETA(%esp), %ecx + movl ARG1(%esp), %eax + movl %ecx, JEIP(%eax) + leal ARG1(%esp), %ecx + movl %ecx, JESP(%eax) + movl %ebp, JEBP(%eax) + movl %ebx, JEBX(%eax) + movl %esi, JESI(%eax) + movl %edi, JEDI(%eax) + xorl %eax, %eax + ret + +DEFUN(jmp_restore) + movl ARG1(%esp), %edx + movl ARG2(%esp), %eax + mov JEDI(%edx),%edi + mov JESI(%edx),%esi + mov JEBX(%edx),%ebx + mov JEBP(%edx),%ebp + mov JESP(%edx),%ecx + mov %ecx,%esp + mov JEIP(%edx),%ecx + jmp *%ecx + +#elif __x86_64__ + +#define JRIP 0 +#define JRSP 8 +#define JRBP 16 +#define JRBX 24 +#define JR12 32 +#define JR13 40 +#define JR14 48 +#define JR15 56 + +#define RETA 0 +#define ARG2 8 + +DEFUN(jmp_save) + mov RETA(%rsp), %rdx + mov %rdx, JRIP(%rdi) + lea ARG2(%rsp), %rdx + mov %rdx, JRSP(%rdi) + mov %rbp, JRBP(%rdi) + mov %rbx, JRBX(%rdi) + mov %r12, JR12(%rdi) + mov %r13, JR13(%rdi) + mov %r14, JR14(%rdi) + mov %r15, JR15(%rdi) + xor %rax, %rax + ret + +DEFUN(jmp_restore) + mov JR15(%rdi), %r15 + mov JR14(%rdi), %r14 + mov JR13(%rdi), %r13 + mov JR12(%rdi), %r12 + mov JRBX(%rdi), %rbx + mov JRBP(%rdi), %rbp + mov JRSP(%rdi), %rdx + mov %rdx, %rsp + mov JRIP(%rdi), %rdx + mov %rsi, %rax + jmp *%rdx + +#elif __arm__ && !__thumb__ + +DEFUN(jmp_save) + stmia r0, {r4, r5, r6, r7, r8, r9, r10, fp, sp, lr} + mov r0, #0 + bx lr + +DEFUN(jmp_restore) + ldmia r0, {r4, r5, r6, r7, r8, r9, r10, fp, sp, lr} + mov r0, r1 + bx lr + +#elif __arm__ && __thumb__ + +DEFUN(jmp_save) + mov r2, r0 + mov r3, lr + stmia r0!, {r3, r4, r5, r6, r7} + mov r3, r8 + mov r4, r9 + mov r5, r10 + mov r6, fp + mov r7, sp + stmia r0!, {r3, r4, r5, r6, r7} + ldmia r2!, {r3, r4, r5, r6, r7} + mov r0, #0 + bx lr + +DEFUN(jmp_restore) + mov r2, r0 + add r0, #5*4 + ldmia r0!, {r3, r4, r5, r6, r7} + mov r8, r3 + mov r9, r4 + mov r10, r5 + mov fp, r6 + mov sp, r7 + ldmia r2!, {r3, r4, r5, r6, r7} + mov r0, r1 + bx r3 + +#else +#error port me! +#endif |