diff options
Diffstat (limited to 'src/pkg/runtime/amd64/closure.c')
| -rw-r--r-- | src/pkg/runtime/amd64/closure.c | 123 | 
1 files changed, 0 insertions, 123 deletions
| diff --git a/src/pkg/runtime/amd64/closure.c b/src/pkg/runtime/amd64/closure.c deleted file mode 100644 index 5033468d2..000000000 --- a/src/pkg/runtime/amd64/closure.c +++ /dev/null @@ -1,123 +0,0 @@ -// Copyright 2009 The Go Authors. All rights reserved. -// Use of this source code is governed by a BSD-style -// license that can be found in the LICENSE file. - -#include "runtime.h" - -#pragma textflag 7 -// func closure(siz int32, -//	fn func(arg0, arg1, arg2 *ptr, callerpc uintptr, xxx) yyy, -//	arg0, arg1, arg2 *ptr) (func(xxx) yyy) -void -runtime·closure(int32 siz, byte *fn, byte *arg0) -{ -	byte *p, *q, **ret; -	int32 i, n; -	int64 pcrel; - -	if(siz < 0 || siz%8 != 0) -		runtime·throw("bad closure size"); - -	ret = (byte**)((byte*)&arg0 + siz); - -	if(siz > 100) { -		// TODO(rsc): implement stack growth preamble? -		runtime·throw("closure too big"); -	} - -	// compute size of new fn. -	// must match code laid out below. -	n = 7+10+3;	// SUBQ MOVQ MOVQ -	if(siz <= 4*8) -		n += 2*siz/8;	// MOVSQ MOVSQ... -	else -		n += 7+3;	// MOVQ REP MOVSQ -	n += 12;	// CALL worst case; sometimes only 5 -	n += 7+1;	// ADDQ RET - -	// store args aligned after code, so gc can find them. -	n += siz; -	if(n%8) -		n += 8 - n%8; - -	p = runtime·mal(n); -	*ret = p; -	q = p + n - siz; - -	if(siz > 0) { -		runtime·mcpy(q, (byte*)&arg0, siz); - -		// SUBQ $siz, SP -		*p++ = 0x48; -		*p++ = 0x81; -		*p++ = 0xec; -		*(uint32*)p = siz; -		p += 4; - -		// MOVQ $q, SI -		*p++ = 0x48; -		*p++ = 0xbe; -		*(byte**)p = q; -		p += 8; - -		// MOVQ SP, DI -		*p++ = 0x48; -		*p++ = 0x89; -		*p++ = 0xe7; - -		if(siz <= 4*8) { -			for(i=0; i<siz; i+=8) { -				// MOVSQ -				*p++ = 0x48; -				*p++ = 0xa5; -			} -		} else { -			// MOVQ $(siz/8), CX  [32-bit immediate siz/8] -			*p++ = 0x48; -			*p++ = 0xc7; -			*p++ = 0xc1; -			*(uint32*)p = siz/8; -			p += 4; - -			// REP; MOVSQ -			*p++ = 0xf3; -			*p++ = 0x48; -			*p++ = 0xa5; -		} -	} - -	// call fn -	pcrel = fn - (p+5); -	if((int32)pcrel == pcrel) { -		// can use direct call with pc-relative offset -		// CALL fn -		*p++ = 0xe8; -		*(int32*)p = pcrel; -		p += 4; -	} else { -		// MOVQ $fn, CX  [64-bit immediate fn] -		*p++ = 0x48; -		*p++ = 0xb9; -		*(byte**)p = fn; -		p += 8; - -		// CALL *CX -		*p++ = 0xff; -		*p++ = 0xd1; -	} - -	// ADDQ $siz, SP -	*p++ = 0x48; -	*p++ = 0x81; -	*p++ = 0xc4; -	*(uint32*)p = siz; -	p += 4; - -	// RET -	*p++ = 0xc3; - -	if(p > q) -		runtime·throw("bad math in sys.closure"); -} - - | 
