1
Fork 0
mirror of https://git.savannah.gnu.org/git/guile.git synced 2025-04-30 11:50:28 +02:00

increase range of relative jumps by aligning blocks to 8-byte boundaries

* libguile/objcodes.c (OBJCODE_COOKIE): Bump again, as our jump offsets
  are now multiplied by 8.

* libguile/vm-i-system.c (BR): Interpret the 16-bit offset as a relative
  jump to the nearest 8-byte-aligned block -- increasing relative jump
  range from +/-32K to +/-240K.
  (mvra): Do the same for the mvra jump.

* libguile/vm.c (really_make_boot_program): Align the mvra.

* module/language/assembly.scm (align-block): New export, for aligning
  blocks.

* module/language/assembly/compile-bytecode.scm (write-bytecode): Emit
  jumps to the nearest 8-byte-aligned block. Effectively our range is 18
  bits in either direction. I would like to do this differently -- have
  long-br and long-br-if, and all the other br instructions go to 8 bits
  only. But the assembler doesn't have an appropriate representation to
  allow me to do this yet, so for now this is what we have.

* module/language/assembly/decompile-bytecode.scm (decode-load-program):
  Decode the 19-bit jumps.
This commit is contained in:
Andy Wingo 2009-07-26 14:01:56 +02:00
parent 28b119ee3d
commit e5dc27b86d
7 changed files with 39 additions and 26 deletions

View file

@ -50,7 +50,7 @@
/* The objcode magic header. */ /* The objcode magic header. */
#define OBJCODE_COOKIE \ #define OBJCODE_COOKIE \
"GOOF-0.8-" OBJCODE_ENDIANNESS "-" OBJCODE_WORD_SIZE "---" "GOOF-0.9-" OBJCODE_ENDIANNESS "-" OBJCODE_WORD_SIZE "---"
/* The length of the header must be a multiple of 8 bytes. */ /* The length of the header must be a multiple of 8 bytes. */
verify (((sizeof (OBJCODE_COOKIE) - 1) & 7) == 0); verify (((sizeof (OBJCODE_COOKIE) - 1) & 7) == 0);

View file

@ -426,7 +426,7 @@ VM_DEFINE_INSTRUCTION (34, long_toplevel_set, "long-toplevel-set", 2, 1, 0)
* branch and jump * branch and jump
*/ */
/* offset must be a signed short!!! */ /* offset must be a signed 16 bit int!!! */
#define FETCH_OFFSET(offset) \ #define FETCH_OFFSET(offset) \
{ \ { \
int h = FETCH (); \ int h = FETCH (); \
@ -436,10 +436,10 @@ VM_DEFINE_INSTRUCTION (34, long_toplevel_set, "long-toplevel-set", 2, 1, 0)
#define BR(p) \ #define BR(p) \
{ \ { \
signed short offset; \ scm_t_int16 offset; \
FETCH_OFFSET (offset); \ FETCH_OFFSET (offset); \
if (p) \ if (p) \
ip += offset; \ ip += ((scm_t_ptrdiff)offset) * 8 - (((unsigned long)ip) % 8); \
NULLSTACK (1); \ NULLSTACK (1); \
DROP (); \ DROP (); \
NEXT; \ NEXT; \
@ -447,9 +447,9 @@ VM_DEFINE_INSTRUCTION (34, long_toplevel_set, "long-toplevel-set", 2, 1, 0)
VM_DEFINE_INSTRUCTION (35, br, "br", 2, 0, 0) VM_DEFINE_INSTRUCTION (35, br, "br", 2, 0, 0)
{ {
int h = FETCH (); scm_t_int16 offset;
int l = FETCH (); FETCH_OFFSET (offset);
ip += (signed short) (h << 8) + l; ip += ((scm_t_ptrdiff)offset) * 8 - (((unsigned long)ip) % 8);
NEXT; NEXT;
} }
@ -812,10 +812,12 @@ VM_DEFINE_INSTRUCTION (46, call_nargs, "call/nargs", 0, 0, 1)
VM_DEFINE_INSTRUCTION (47, mv_call, "mv-call", 3, -1, 1) VM_DEFINE_INSTRUCTION (47, mv_call, "mv-call", 3, -1, 1)
{ {
SCM x; SCM x;
signed short offset; scm_t_int16 offset;
scm_t_uint8 *mvra;
nargs = FETCH (); nargs = FETCH ();
FETCH_OFFSET (offset); FETCH_OFFSET (offset);
mvra = ip + ((scm_t_ptrdiff)offset) * 8 - ((unsigned long)ip) % 8;
x = sp[-nargs]; x = sp[-nargs];
@ -828,7 +830,7 @@ VM_DEFINE_INSTRUCTION (47, mv_call, "mv-call", 3, -1, 1)
CACHE_PROGRAM (); CACHE_PROGRAM ();
INIT_ARGS (); INIT_ARGS ();
NEW_FRAME (); NEW_FRAME ();
SCM_FRAME_DATA_ADDRESS (fp)[1] = (SCM)(SCM_FRAME_RETURN_ADDRESS (fp) + offset); SCM_FRAME_DATA_ADDRESS (fp)[1] = (SCM)mvra;
ENTER_HOOK (); ENTER_HOOK ();
APPLY_HOOK (); APPLY_HOOK ();
NEXT; NEXT;
@ -853,7 +855,7 @@ VM_DEFINE_INSTRUCTION (47, mv_call, "mv-call", 3, -1, 1)
len = scm_length (values); len = scm_length (values);
PUSH_LIST (values, SCM_NULLP); PUSH_LIST (values, SCM_NULLP);
PUSH (len); PUSH (len);
ip += offset; ip = mvra;
} }
NEXT; NEXT;
} }

View file

@ -226,7 +226,7 @@ really_make_boot_program (long nargs)
SCM u8vec; SCM u8vec;
/* Make sure "bytes" is 64-bit aligned. */ /* Make sure "bytes" is 64-bit aligned. */
scm_t_uint8 text[] = { scm_op_mv_call, 0, 0, 1, scm_t_uint8 text[] = { scm_op_mv_call, 0, 0, 1,
scm_op_make_int8_1, scm_op_make_int8_1, scm_op_nop, scm_op_nop, scm_op_nop,
scm_op_halt }; scm_op_halt };
struct scm_objcode *bp; struct scm_objcode *bp;
SCM ret; SCM ret;

View file

@ -24,7 +24,7 @@
#:use-module (system vm instruction) #:use-module (system vm instruction)
#:use-module ((srfi srfi-1) #:select (fold)) #:use-module ((srfi srfi-1) #:select (fold))
#:export (byte-length #:export (byte-length
addr+ align-program align-code addr+ align-program align-code align-block
assembly-pack assembly-unpack assembly-pack assembly-unpack
object->assembly assembly->object)) object->assembly assembly->object))
@ -63,17 +63,24 @@
(define *program-alignment* 8) (define *program-alignment* 8)
(define *block-alignment* 8)
(define (addr+ addr code) (define (addr+ addr code)
(fold (lambda (x len) (+ (byte-length x) len)) (fold (lambda (x len) (+ (byte-length x) len))
addr addr
code)) code))
(define (code-alignment addr alignment header-len)
(make-list (modulo (- alignment
(modulo (+ addr header-len) alignment))
alignment)
'(nop)))
(define (align-block addr)
(code-alignment addr *block-alignment* 0))
(define (align-code code addr alignment header-len) (define (align-code code addr alignment header-len)
`(,@(make-list (modulo (- alignment `(,@(code-alignment addr alignment header-len)
(modulo (+ addr header-len) alignment))
alignment)
'(nop))
,code)) ,code))
(define (align-program prog addr) (define (align-program prog addr)

View file

@ -77,10 +77,12 @@
;; Ew! ;; Ew!
(for-each write-byte (bytevector->u8-list bv))) (for-each write-byte (bytevector->u8-list bv)))
(define (write-break label) (define (write-break label)
(let ((offset (- (assq-ref labels label) (+ (get-addr) 2)))) (let ((offset (- (assq-ref labels label)
(cond ((>= offset (ash 1 15)) (error "jump too big" offset)) (logand (+ (get-addr) 2) (lognot #x7)))))
((< offset (- (ash 1 15))) (error "reverse jump too big" offset)) (cond ((not (= 0 (modulo offset 8))) (error "unaligned jump" offset))
(else (write-uint16-be offset))))) ((>= offset (ash 1 18)) (error "jump too far forward" offset))
((< offset (- (ash 1 18))) (error "jump too far backwards" offset))
(else (write-uint16-be (ash offset -3))))))
(let ((inst (car asm)) (let ((inst (car asm))
(args (cdr asm)) (args (cdr asm))

View file

@ -61,7 +61,8 @@
(labels '()) (labels '())
(i 0)) (i 0))
(define (ensure-label rel1 rel2) (define (ensure-label rel1 rel2)
(let ((where (+ i (bytes->s16 rel1 rel2)))) (let ((where (+ (logand i (lognot #x7))
(* (bytes->s16 rel1 rel2) 8))))
(or (assv-ref labels where) (or (assv-ref labels where)
(begin (begin
(let ((l (gensym ":L"))) (let ((l (gensym ":L")))

View file

@ -340,11 +340,12 @@
(error "unknown module var kind" op key))))) (error "unknown module var kind" op key)))))
((<glil-label> label) ((<glil-label> label)
(values '() (let ((code (align-block addr)))
bindings (values code
source-alist bindings
(acons label addr label-alist) source-alist
object-alist)) (acons label (addr+ addr code) label-alist)
object-alist)))
((<glil-branch> inst label) ((<glil-branch> inst label)
(emit-code `((,inst ,label)))) (emit-code `((,inst ,label))))