| /* |
| * QuickThreads -- Threads-building toolkit. |
| * Copyright (c) 1993 by David Keppel |
| * |
| * Permission to use, copy, modify and distribute this software and |
| * its documentation for any purpose and without fee is hereby |
| * granted, provided that the above copyright notice and this notice |
| * appear in all copies. This software is provided as a |
| * proof-of-concept and for demonstration purposes; there is no |
| * representation about the suitability of this software for any |
| * purpose. |
| */ |
| |
| /* axp.s -- assembly support. */ |
| |
| .text |
| .align 4 |
| .file 2 "axp.s" |
| |
| .globl qt_block |
| .globl qt_blocki |
| .globl qt_abort |
| .globl qt_start |
| .globl qt_vstart |
| |
| /* |
| ** $16: ptr to function to call once curr is suspended |
| ** and control is on r19's stack. |
| ** $17: 1'th arg to (*$16)(...). |
| ** $18: 2'th arg to (*$16)(...). |
| ** $19: sp of thread to resume. |
| ** |
| ** The helper routine returns a value that is passed on as the |
| ** return value from the blocking routine. Since we don't |
| ** touch r0 between the helper's return and the end of |
| ** function, we get this behavior for free. |
| */ |
| |
| .ent qt_blocki |
| qt_blocki: |
| subq $30,80, $30 /* Allocate save area. */ |
| stq $26, 0($30) /* Save registers. */ |
| stq $9, 8($30) |
| stq $10,16($30) |
| stq $11,24($30) |
| stq $12,32($30) |
| stq $13,40($30) |
| stq $14,48($30) |
| stq $15,56($30) |
| stq $29,64($30) |
| .end qt_blocki |
| .ent qt_abort |
| qt_abort: |
| addq $16,$31, $27 /* Put argument function in PV. */ |
| addq $30,$31, $16 /* Save stack ptr in outgoing arg. */ |
| addq $19,$31, $30 /* Set new stack pointer. */ |
| jsr $26,($27),0 /* Call helper function. */ |
| |
| ldq $26, 0($30) /* Restore registers. */ |
| ldq $9, 8($30) |
| ldq $10,16($30) |
| ldq $11,24($30) |
| ldq $12,32($30) |
| ldq $13,40($30) |
| ldq $14,48($30) |
| ldq $15,56($30) |
| ldq $29,64($30) |
| |
| addq $30,80, $30 /* Deallocate save area. */ |
| ret $31,($26),1 /* Return, predict===RET. */ |
| .end qt_abort |
| |
| |
| /* |
| ** Non-varargs thread startup. |
| */ |
| .ent qt_start |
| qt_start: |
| addq $9,$31, $16 /* Load up `qu'. */ |
| addq $10,$31, $17 /* ... user function's `pt'. */ |
| addq $11,$31, $18 /* ... user function's `userf'. */ |
| addq $12,$31, $27 /* ... set procedure value to `only'. */ |
| jsr $26,($27),0 /* Call `only'. */ |
| |
| jsr $26,qt_error /* `only' erroniously returned. */ |
| .end qt_start |
| |
| |
| .ent qt_vstart: |
| qt_vstart: |
| /* Call startup function. */ |
| addq $9,$31, $16 /* Arg0 to `startup'. */ |
| addq $12,$31, $27 /* Set procedure value. */ |
| jsr $26,($27),0 /* Call `startup'. */ |
| |
| /* Call user function. */ |
| ldt $f16, 0($30) /* Load fp arg regs. */ |
| ldt $f17, 8($30) |
| ldt $f18,16($30) |
| ldt $f19,24($30) |
| ldt $f20,32($30) |
| ldt $f21,40($30) |
| ldq $16,48($30) /* And integer arg regs. */ |
| ldq $17,56($30) |
| ldq $18,64($30) |
| ldq $19,72($30) |
| ldq $20,80($30) |
| ldq $21,88($30) |
| addq $30,96 $30 /* Pop 6*2*8 saved arg regs. */ |
| addq $11,$31, $27 /* Set procedure value. */ |
| jsr $26,($27),0 /* Call `vuserf'. */ |
| |
| /* Call cleanup. */ |
| addq $9,$31, $16 /* Arg0 to `cleanup'. */ |
| addq $0,$31, $17 /* Users's return value is arg1. */ |
| addq $10,$31, $27 /* Set procedure value. */ |
| jsr $26,($27),0 /* Call `cleanup'. */ |
| |
| jsr $26,qt_error /* Cleanup erroniously returned. */ |
| .end qt_start |
| |
| |
| /* |
| ** Save calle-save floating-point regs $f2..$f9. |
| ** Also save return pc from whomever called us. |
| ** |
| ** Return value from `qt_block' is the same as the return from |
| ** `qt_blocki'. We get that for free since we don't touch $0 |
| ** between the return from `qt_blocki' and the return from |
| ** `qt_block'. |
| */ |
| .ent qt_block |
| qt_block: |
| subq $30,80, $30 /* Allocate a save space. */ |
| stq $26, 0($30) /* Save registers. */ |
| stt $f2, 8($30) |
| stt $f3,16($30) |
| stt $f4,24($30) |
| stt $f5,32($30) |
| stt $f6,40($30) |
| stt $f7,48($30) |
| stt $f8,56($30) |
| stt $f9,64($30) |
| |
| jsr $26,qt_blocki /* Call helper. */ |
| /* .. who will also restore $gp. */ |
| |
| ldq $26, 0($30) /* restore registers. */ |
| ldt $f2, 8($30) |
| ldt $f3,16($30) |
| ldt $f4,24($30) |
| ldt $f5,32($30) |
| ldt $f6,40($30) |
| ldt $f7,48($30) |
| ldt $f8,56($30) |
| ldt $f9,64($30) |
| |
| addq $30,80, $30 /* Deallcate save space. */ |
| ret $31,($26),1 /* Return, predict===RET. */ |
| .end qt_block |