blob: 04513857e7b5488ec0470c53fabc336d97b4b5e8 [file] [log] [blame]
/*
* 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.
*/
#ifndef QUICKTHREADS_386_H
#define QUICKTHREADS_386_H
typedef unsigned long qt_word_t;
/* Thread's initial stack layout on the i386:
non-varargs:
+---
| arg[2] === `userf' on startup
| arg[1] === `pt' on startup
| arg[0] === `pu' on startup
+---
| ret pc === qt_error
+---
| ret pc === `only' on startup
+---
| %ebp
| %esi
| %edi
| %ebx <--- qt_t.sp
+---
When a non-varargs thread is started, it ``returns'' directly to
the client's `only' function.
varargs:
+---
| arg[n-1]
| ..
| arg[0]
+---
| ret pc === `qt_vstart'
+---
| %ebp === `startup'
| %esi === `cleanup'
| %edi === `pt'
| %ebx === `vuserf' <--- qt_t.sp
+---
When a varargs thread is started, it ``returns'' to the `qt_vstart'
startup code. The startup code calls the appropriate functions. */
/* What to do to start a varargs thread running. */
extern void qt_vstart (void);
/* Hold 4 saved regs plus two return pcs (qt_error, qt_start) plus
three args. */
#define QUICKTHREADS_STKBASE (13 * 4)
/* Hold 4 saved regs plus one return pc (qt_vstart). */
#define QUICKTHREADS_VSTKBASE (5 * 4)
/* Stack must be 16-byte aligned at function call instr. (SSE data support) */
#define QUICKTHREADS_STKALIGN (16)
/* Where to place various arguments. */
#define QUICKTHREADS_ONLY_INDEX (QUICKTHREADS_PC)
#define QUICKTHREADS_USER_INDEX (QUICKTHREADS_ARG2)
#define QUICKTHREADS_ARGT_INDEX (QUICKTHREADS_ARG1)
#define QUICKTHREADS_ARGU_INDEX (QUICKTHREADS_ARG0)
#define QUICKTHREADS_VSTARTUP_INDEX (QUICKTHREADS_EBP)
#define QUICKTHREADS_VUSERF_INDEX (QUICKTHREADS_EBX)
#define QUICKTHREADS_VCLEANUP_INDEX (QUICKTHREADS_ESI)
#define QUICKTHREADS_VARGT_INDEX (QUICKTHREADS_EDI)
#define QUICKTHREADS_EBX 0
#define QUICKTHREADS_EDI 1
#define QUICKTHREADS_ESI 2
#define QUICKTHREADS_EBP 3
#define QUICKTHREADS_POP0 4
#define QUICKTHREADS_POP1 5
#define QUICKTHREADS_POP2 6
#define QUICKTHREADS_PC 7
/* The following are defined only for non-varargs. */
#define QUICKTHREADS_POPE 8
#define QUICKTHREADS_ARG0 9
#define QUICKTHREADS_ARG1 10
#define QUICKTHREADS_ARG2 11
#define QUICKTHREADS_RPC 12
/* Stack grows down. The top of the stack is the first thing to
pop off (preincrement, postdecrement). */
#define QUICKTHREADS_GROW_DOWN
extern void qt_error (void);
/* For correct 16-byte stack alignment (auto-relocatable functions) */
extern void qt_tramp (void);
extern void qt_align (void);
/* Push on the error return address and the alignment/trampoline functions. */
#define QUICKTHREADS_ARGS_MD(sto) \
(QUICKTHREADS_SPUT (sto, QUICKTHREADS_POP0, qt_align), \
QUICKTHREADS_SPUT (sto, QUICKTHREADS_POP1, qt_align), \
QUICKTHREADS_SPUT (sto, QUICKTHREADS_POP2, qt_align), \
QUICKTHREADS_SPUT (sto, QUICKTHREADS_POPE, qt_tramp), \
QUICKTHREADS_SPUT (sto, QUICKTHREADS_RPC, qt_error))
/* When varargs are pushed, allocate space for all the args. */
#define QUICKTHREADS_VARGS_MD0(sto, nbytes) \
((qt_t *)(((char *)(sto)) - QUICKTHREADS_STKROUNDUP(nbytes)))
#define QUICKTHREADS_VARGS_MD1(sto) \
(QUICKTHREADS_SPUT (sto, QUICKTHREADS_PC, qt_vstart))
#define QUICKTHREADS_VARGS_DEFAULT
#endif /* QUICKTHREADS_386_H */