Text file
    src/runtime/asm_s390x.s
  
  
    Documentation: runtime
  
     1// Copyright 2016 The Go Authors. All rights reserved.
     2// Use of this source code is governed by a BSD-style
     3// license that can be found in the LICENSE file.
     4
     5#include "go_asm.h"
     6#include "go_tls.h"
     7#include "funcdata.h"
     8#include "textflag.h"
     9
    10// _rt0_s390x_lib is common startup code for s390x systems when
    11// using -buildmode=c-archive or -buildmode=c-shared. The linker will
    12// arrange to invoke this function as a global constructor (for
    13// c-archive) or when the shared library is loaded (for c-shared).
    14// We expect argc and argv to be passed in the usual C ABI registers
    15// R2 and R3.
    16TEXT _rt0_s390x_lib(SB), NOSPLIT|NOFRAME, $0
    17	STMG	R6, R15, 48(R15)
    18	MOVD	R2, _rt0_s390x_lib_argc<>(SB)
    19	MOVD	R3, _rt0_s390x_lib_argv<>(SB)
    20
    21	// Save R6-R15 in the register save area of the calling function.
    22	STMG	R6, R15, 48(R15)
    23
    24	// Allocate 80 bytes on the stack.
    25	MOVD	$-80(R15), R15
    26
    27	// Save F8-F15 in our stack frame.
    28	FMOVD	F8, 16(R15)
    29	FMOVD	F9, 24(R15)
    30	FMOVD	F10, 32(R15)
    31	FMOVD	F11, 40(R15)
    32	FMOVD	F12, 48(R15)
    33	FMOVD	F13, 56(R15)
    34	FMOVD	F14, 64(R15)
    35	FMOVD	F15, 72(R15)
    36
    37	// Synchronous initialization.
    38	MOVD	$runtime·libpreinit(SB), R1
    39	BL	R1
    40
    41	// Create a new thread to finish Go runtime initialization.
    42	MOVD	_cgo_sys_thread_create(SB), R1
    43	CMP	R1, $0
    44	BEQ	nocgo
    45	MOVD	$_rt0_s390x_lib_go(SB), R2
    46	MOVD	$0, R3
    47	BL	R1
    48	BR	restore
    49
    50nocgo:
    51	MOVD	$0x800000, R1              // stacksize
    52	MOVD	R1, 0(R15)
    53	MOVD	$_rt0_s390x_lib_go(SB), R1
    54	MOVD	R1, 8(R15)                 // fn
    55	MOVD	$runtime·newosproc(SB), R1
    56	BL	R1
    57
    58restore:
    59	// Restore F8-F15 from our stack frame.
    60	FMOVD	16(R15), F8
    61	FMOVD	24(R15), F9
    62	FMOVD	32(R15), F10
    63	FMOVD	40(R15), F11
    64	FMOVD	48(R15), F12
    65	FMOVD	56(R15), F13
    66	FMOVD	64(R15), F14
    67	FMOVD	72(R15), F15
    68	MOVD	$80(R15), R15
    69
    70	// Restore R6-R15.
    71	LMG	48(R15), R6, R15
    72	RET
    73
    74// _rt0_s390x_lib_go initializes the Go runtime.
    75// This is started in a separate thread by _rt0_s390x_lib.
    76TEXT _rt0_s390x_lib_go(SB), NOSPLIT|NOFRAME, $0
    77	MOVD	_rt0_s390x_lib_argc<>(SB), R2
    78	MOVD	_rt0_s390x_lib_argv<>(SB), R3
    79	MOVD	$runtime·rt0_go(SB), R1
    80	BR	R1
    81
    82DATA _rt0_s390x_lib_argc<>(SB)/8, $0
    83GLOBL _rt0_s390x_lib_argc<>(SB), NOPTR, $8
    84DATA _rt0_s90x_lib_argv<>(SB)/8, $0
    85GLOBL _rt0_s390x_lib_argv<>(SB), NOPTR, $8
    86
    87TEXT runtime·rt0_go(SB),NOSPLIT|TOPFRAME,$0
    88	// R2 = argc; R3 = argv; R11 = temp; R13 = g; R15 = stack pointer
    89	// C TLS base pointer in AR0:AR1
    90
    91	// initialize essential registers
    92	XOR	R0, R0
    93
    94	SUB	$24, R15
    95	MOVW	R2, 8(R15) // argc
    96	MOVD	R3, 16(R15) // argv
    97
    98	// create istack out of the given (operating system) stack.
    99	// _cgo_init may update stackguard.
   100	MOVD	$runtime·g0(SB), g
   101	MOVD	R15, R11
   102	SUB	$(64*1024), R11
   103	MOVD	R11, g_stackguard0(g)
   104	MOVD	R11, g_stackguard1(g)
   105	MOVD	R11, (g_stack+stack_lo)(g)
   106	MOVD	R15, (g_stack+stack_hi)(g)
   107
   108	// if there is a _cgo_init, call it using the gcc ABI.
   109	MOVD	_cgo_init(SB), R11
   110	CMPBEQ	R11, $0, nocgo
   111	MOVW	AR0, R4			// (AR0 << 32 | AR1) is the TLS base pointer; MOVD is translated to EAR
   112	SLD	$32, R4, R4
   113	MOVW	AR1, R4			// arg 2: TLS base pointer
   114	MOVD	$setg_gcc<>(SB), R3 	// arg 1: setg
   115	MOVD	g, R2			// arg 0: G
   116	// C functions expect 160 bytes of space on caller stack frame
   117	// and an 8-byte aligned stack pointer
   118	MOVD	R15, R9			// save current stack (R9 is preserved in the Linux ABI)
   119	SUB	$160, R15		// reserve 160 bytes
   120	MOVD    $~7, R6
   121	AND 	R6, R15			// 8-byte align
   122	BL	R11			// this call clobbers volatile registers according to Linux ABI (R0-R5, R14)
   123	MOVD	R9, R15			// restore stack
   124	XOR	R0, R0			// zero R0
   125
   126nocgo:
   127	// update stackguard after _cgo_init
   128	MOVD	(g_stack+stack_lo)(g), R2
   129	ADD	$const_stackGuard, R2
   130	MOVD	R2, g_stackguard0(g)
   131	MOVD	R2, g_stackguard1(g)
   132
   133	// set the per-goroutine and per-mach "registers"
   134	MOVD	$runtime·m0(SB), R2
   135
   136	// save m->g0 = g0
   137	MOVD	g, m_g0(R2)
   138	// save m0 to g0->m
   139	MOVD	R2, g_m(g)
   140
   141	BL	runtime·check(SB)
   142
   143	// argc/argv are already prepared on stack
   144	BL	runtime·args(SB)
   145	BL	runtime·checkS390xCPU(SB)
   146	BL	runtime·osinit(SB)
   147	BL	runtime·schedinit(SB)
   148
   149	// create a new goroutine to start program
   150	MOVD	$runtime·mainPC(SB), R2		// entry
   151	SUB     $16, R15
   152	MOVD 	R2, 8(R15)
   153	MOVD 	$0, 0(R15)
   154	BL	runtime·newproc(SB)
   155	ADD	$16, R15
   156
   157	// start this M
   158	BL	runtime·mstart(SB)
   159
   160	MOVD	$0, 1(R0)
   161	RET
   162
   163DATA	runtime·mainPC+0(SB)/8,$runtime·main(SB)
   164GLOBL	runtime·mainPC(SB),RODATA,$8
   165
   166TEXT runtime·breakpoint(SB),NOSPLIT|NOFRAME,$0-0
   167	BRRK
   168	RET
   169
   170TEXT runtime·asminit(SB),NOSPLIT|NOFRAME,$0-0
   171	RET
   172
   173TEXT runtime·mstart(SB),NOSPLIT|TOPFRAME,$0
   174	CALL	runtime·mstart0(SB)
   175	RET // not reached
   176
   177/*
   178 *  go-routine
   179 */
   180
   181// void gogo(Gobuf*)
   182// restore state from Gobuf; longjmp
   183TEXT runtime·gogo(SB), NOSPLIT|NOFRAME, $0-8
   184	MOVD	buf+0(FP), R5
   185	MOVD	gobuf_g(R5), R6
   186	MOVD	0(R6), R7	// make sure g != nil
   187	BR	gogo<>(SB)
   188
   189TEXT gogo<>(SB), NOSPLIT|NOFRAME, $0
   190	MOVD	R6, g
   191	BL	runtime·save_g(SB)
   192
   193	MOVD	0(g), R4
   194	MOVD	gobuf_sp(R5), R15
   195	MOVD	gobuf_lr(R5), LR
   196	MOVD	gobuf_ctxt(R5), R12
   197	MOVD	$0, gobuf_sp(R5)
   198	MOVD	$0, gobuf_lr(R5)
   199	MOVD	$0, gobuf_ctxt(R5)
   200	CMP	R0, R0 // set condition codes for == test, needed by stack split
   201	MOVD	gobuf_pc(R5), R6
   202	BR	(R6)
   203
   204// void mcall(fn func(*g))
   205// Switch to m->g0's stack, call fn(g).
   206// Fn must never return.  It should gogo(&g->sched)
   207// to keep running g.
   208TEXT runtime·mcall(SB), NOSPLIT, $-8-8
   209	// Save caller state in g->sched
   210	MOVD	R15, (g_sched+gobuf_sp)(g)
   211	MOVD	LR, (g_sched+gobuf_pc)(g)
   212	MOVD	$0, (g_sched+gobuf_lr)(g)
   213
   214	// Switch to m->g0 & its stack, call fn.
   215	MOVD	g, R3
   216	MOVD	g_m(g), R8
   217	MOVD	m_g0(R8), g
   218	BL	runtime·save_g(SB)
   219	CMP	g, R3
   220	BNE	2(PC)
   221	BR	runtime·badmcall(SB)
   222	MOVD	fn+0(FP), R12			// context
   223	MOVD	0(R12), R4			// code pointer
   224	MOVD	(g_sched+gobuf_sp)(g), R15	// sp = m->g0->sched.sp
   225	SUB	$16, R15
   226	MOVD	R3, 8(R15)
   227	MOVD	$0, 0(R15)
   228	BL	(R4)
   229	BR	runtime·badmcall2(SB)
   230
   231// systemstack_switch is a dummy routine that systemstack leaves at the bottom
   232// of the G stack.  We need to distinguish the routine that
   233// lives at the bottom of the G stack from the one that lives
   234// at the top of the system stack because the one at the top of
   235// the system stack terminates the stack walk (see topofstack()).
   236TEXT runtime·systemstack_switch(SB), NOSPLIT, $0-0
   237	UNDEF
   238	BL	(LR)	// make sure this function is not leaf
   239	RET
   240
   241// func systemstack(fn func())
   242TEXT runtime·systemstack(SB), NOSPLIT, $0-8
   243	MOVD	fn+0(FP), R3	// R3 = fn
   244	MOVD	R3, R12		// context
   245	MOVD	g_m(g), R4	// R4 = m
   246
   247	MOVD	m_gsignal(R4), R5	// R5 = gsignal
   248	CMPBEQ	g, R5, noswitch
   249
   250	MOVD	m_g0(R4), R5	// R5 = g0
   251	CMPBEQ	g, R5, noswitch
   252
   253	MOVD	m_curg(R4), R6
   254	CMPBEQ	g, R6, switch
   255
   256	// Bad: g is not gsignal, not g0, not curg. What is it?
   257	// Hide call from linker nosplit analysis.
   258	MOVD	$runtime·badsystemstack(SB), R3
   259	BL	(R3)
   260	BL	runtime·abort(SB)
   261
   262switch:
   263	// save our state in g->sched.  Pretend to
   264	// be systemstack_switch if the G stack is scanned.
   265	BL	gosave_systemstack_switch<>(SB)
   266
   267	// switch to g0
   268	MOVD	R5, g
   269	BL	runtime·save_g(SB)
   270	MOVD	(g_sched+gobuf_sp)(g), R15
   271
   272	// call target function
   273	MOVD	0(R12), R3	// code pointer
   274	BL	(R3)
   275
   276	// switch back to g
   277	MOVD	g_m(g), R3
   278	MOVD	m_curg(R3), g
   279	BL	runtime·save_g(SB)
   280	MOVD	(g_sched+gobuf_sp)(g), R15
   281	MOVD	$0, (g_sched+gobuf_sp)(g)
   282	RET
   283
   284noswitch:
   285	// already on m stack, just call directly
   286	// Using a tail call here cleans up tracebacks since we won't stop
   287	// at an intermediate systemstack.
   288	MOVD	0(R12), R3	// code pointer
   289	MOVD	0(R15), LR	// restore LR
   290	ADD	$8, R15
   291	BR	(R3)
   292
   293// func switchToCrashStack0(fn func())
   294TEXT runtime·switchToCrashStack0<ABIInternal>(SB), NOSPLIT, $0-8
   295	MOVD	fn+0(FP), R12	// context
   296	MOVD	g_m(g), R4	// curm
   297
   298	// set g to gcrash
   299	MOVD	$runtime·gcrash(SB), g	// g = &gcrash
   300	BL	runtime·save_g(SB)
   301	MOVD	R4, g_m(g)	// g.m = curm
   302	MOVD	g, m_g0(R4)	// curm.g0 = g
   303
   304	// switch to crashstack
   305	MOVD	(g_stack+stack_hi)(g), R4
   306	ADD	$(-4*8), R4, R15
   307
   308	// call target function
   309	MOVD	0(R12), R3	// code pointer
   310	BL	(R3)
   311
   312	// should never return
   313	BL	runtime·abort(SB)
   314	UNDEF
   315
   316/*
   317 * support for morestack
   318 */
   319
   320// Called during function prolog when more stack is needed.
   321// Caller has already loaded:
   322// R3: framesize, R4: argsize, R5: LR
   323//
   324// The traceback routines see morestack on a g0 as being
   325// the top of a stack (for example, morestack calling newstack
   326// calling the scheduler calling newm calling gc), so we must
   327// record an argument size. For that purpose, it has no arguments.
   328TEXT runtime·morestack(SB),NOSPLIT|NOFRAME,$0-0
   329	// Called from f.
   330	// Set g->sched to context in f.
   331	MOVD	R15, (g_sched+gobuf_sp)(g)
   332	MOVD	LR, R8
   333	MOVD	R8, (g_sched+gobuf_pc)(g)
   334	MOVD	R5, (g_sched+gobuf_lr)(g)
   335	MOVD	R12, (g_sched+gobuf_ctxt)(g)
   336
   337	// Cannot grow scheduler stack (m->g0).
   338	MOVD	g_m(g), R7
   339	MOVD	m_g0(R7), R8
   340	CMPBNE	g, R8, 3(PC)
   341	BL	runtime·badmorestackg0(SB)
   342	BL	runtime·abort(SB)
   343
   344	// Cannot grow signal stack (m->gsignal).
   345	MOVD	m_gsignal(R7), R8
   346	CMP	g, R8
   347	BNE	3(PC)
   348	BL	runtime·badmorestackgsignal(SB)
   349	BL	runtime·abort(SB)
   350
   351	// Called from f.
   352	// Set m->morebuf to f's caller.
   353	MOVD	R5, (m_morebuf+gobuf_pc)(R7)	// f's caller's PC
   354	MOVD	R15, (m_morebuf+gobuf_sp)(R7)	// f's caller's SP
   355	MOVD	g, (m_morebuf+gobuf_g)(R7)
   356
   357	// Call newstack on m->g0's stack.
   358	MOVD	m_g0(R7), g
   359	BL	runtime·save_g(SB)
   360	MOVD	(g_sched+gobuf_sp)(g), R15
   361	// Create a stack frame on g0 to call newstack.
   362	MOVD	$0, -8(R15)	// Zero saved LR in frame
   363	SUB	$8, R15
   364	BL	runtime·newstack(SB)
   365
   366	// Not reached, but make sure the return PC from the call to newstack
   367	// is still in this function, and not the beginning of the next.
   368	UNDEF
   369
   370TEXT runtime·morestack_noctxt(SB),NOSPLIT|NOFRAME,$0-0
   371	// Force SPWRITE. This function doesn't actually write SP,
   372	// but it is called with a special calling convention where
   373	// the caller doesn't save LR on stack but passes it as a
   374	// register (R5), and the unwinder currently doesn't understand.
   375	// Make it SPWRITE to stop unwinding. (See issue 54332)
   376	MOVD	R15, R15
   377
   378	MOVD	$0, R12
   379	BR	runtime·morestack(SB)
   380
   381// reflectcall: call a function with the given argument list
   382// func call(stackArgsType *_type, f *FuncVal, stackArgs *byte, stackArgsSize, stackRetOffset, frameSize uint32, regArgs *abi.RegArgs).
   383// we don't have variable-sized frames, so we use a small number
   384// of constant-sized-frame functions to encode a few bits of size in the pc.
   385// Caution: ugly multiline assembly macros in your future!
   386
   387#define DISPATCH(NAME,MAXSIZE)		\
   388	MOVD	$MAXSIZE, R4;		\
   389	CMP	R3, R4;		\
   390	BGT	3(PC);			\
   391	MOVD	$NAME(SB), R5;	\
   392	BR	(R5)
   393// Note: can't just "BR NAME(SB)" - bad inlining results.
   394
   395TEXT ·reflectcall(SB), NOSPLIT, $-8-48
   396	MOVWZ	frameSize+32(FP), R3
   397	DISPATCH(runtime·call16, 16)
   398	DISPATCH(runtime·call32, 32)
   399	DISPATCH(runtime·call64, 64)
   400	DISPATCH(runtime·call128, 128)
   401	DISPATCH(runtime·call256, 256)
   402	DISPATCH(runtime·call512, 512)
   403	DISPATCH(runtime·call1024, 1024)
   404	DISPATCH(runtime·call2048, 2048)
   405	DISPATCH(runtime·call4096, 4096)
   406	DISPATCH(runtime·call8192, 8192)
   407	DISPATCH(runtime·call16384, 16384)
   408	DISPATCH(runtime·call32768, 32768)
   409	DISPATCH(runtime·call65536, 65536)
   410	DISPATCH(runtime·call131072, 131072)
   411	DISPATCH(runtime·call262144, 262144)
   412	DISPATCH(runtime·call524288, 524288)
   413	DISPATCH(runtime·call1048576, 1048576)
   414	DISPATCH(runtime·call2097152, 2097152)
   415	DISPATCH(runtime·call4194304, 4194304)
   416	DISPATCH(runtime·call8388608, 8388608)
   417	DISPATCH(runtime·call16777216, 16777216)
   418	DISPATCH(runtime·call33554432, 33554432)
   419	DISPATCH(runtime·call67108864, 67108864)
   420	DISPATCH(runtime·call134217728, 134217728)
   421	DISPATCH(runtime·call268435456, 268435456)
   422	DISPATCH(runtime·call536870912, 536870912)
   423	DISPATCH(runtime·call1073741824, 1073741824)
   424	MOVD	$runtime·badreflectcall(SB), R5
   425	BR	(R5)
   426
   427#define CALLFN(NAME,MAXSIZE)			\
   428TEXT NAME(SB), WRAPPER, $MAXSIZE-48;		\
   429	NO_LOCAL_POINTERS;			\
   430	/* copy arguments to stack */		\
   431	MOVD	stackArgs+16(FP), R4;			\
   432	MOVWZ	stackArgsSize+24(FP), R5;		\
   433	MOVD	$stack-MAXSIZE(SP), R6;		\
   434loopArgs: /* copy 256 bytes at a time */	\
   435	CMP	R5, $256;			\
   436	BLT	tailArgs;			\
   437	SUB	$256, R5;			\
   438	MVC	$256, 0(R4), 0(R6);		\
   439	MOVD	$256(R4), R4;			\
   440	MOVD	$256(R6), R6;			\
   441	BR	loopArgs;			\
   442tailArgs: /* copy remaining bytes */		\
   443	CMP	R5, $0;				\
   444	BEQ	callFunction;			\
   445	SUB	$1, R5;				\
   446	EXRL	$callfnMVC<>(SB), R5;		\
   447callFunction:					\
   448	MOVD	f+8(FP), R12;			\
   449	MOVD	(R12), R8;			\
   450	PCDATA  $PCDATA_StackMapIndex, $0;	\
   451	BL	(R8);				\
   452	/* copy return values back */		\
   453	MOVD	stackArgsType+0(FP), R7;		\
   454	MOVD	stackArgs+16(FP), R6;			\
   455	MOVWZ	stackArgsSize+24(FP), R5;			\
   456	MOVD	$stack-MAXSIZE(SP), R4;		\
   457	MOVWZ	stackRetOffset+28(FP), R1;		\
   458	ADD	R1, R4;				\
   459	ADD	R1, R6;				\
   460	SUB	R1, R5;				\
   461	BL	callRet<>(SB);			\
   462	RET
   463
   464// callRet copies return values back at the end of call*. This is a
   465// separate function so it can allocate stack space for the arguments
   466// to reflectcallmove. It does not follow the Go ABI; it expects its
   467// arguments in registers.
   468TEXT callRet<>(SB), NOSPLIT, $40-0
   469	MOVD	R7, 8(R15)
   470	MOVD	R6, 16(R15)
   471	MOVD	R4, 24(R15)
   472	MOVD	R5, 32(R15)
   473	MOVD	$0, 40(R15)
   474	BL	runtime·reflectcallmove(SB)
   475	RET
   476
   477CALLFN(·call16, 16)
   478CALLFN(·call32, 32)
   479CALLFN(·call64, 64)
   480CALLFN(·call128, 128)
   481CALLFN(·call256, 256)
   482CALLFN(·call512, 512)
   483CALLFN(·call1024, 1024)
   484CALLFN(·call2048, 2048)
   485CALLFN(·call4096, 4096)
   486CALLFN(·call8192, 8192)
   487CALLFN(·call16384, 16384)
   488CALLFN(·call32768, 32768)
   489CALLFN(·call65536, 65536)
   490CALLFN(·call131072, 131072)
   491CALLFN(·call262144, 262144)
   492CALLFN(·call524288, 524288)
   493CALLFN(·call1048576, 1048576)
   494CALLFN(·call2097152, 2097152)
   495CALLFN(·call4194304, 4194304)
   496CALLFN(·call8388608, 8388608)
   497CALLFN(·call16777216, 16777216)
   498CALLFN(·call33554432, 33554432)
   499CALLFN(·call67108864, 67108864)
   500CALLFN(·call134217728, 134217728)
   501CALLFN(·call268435456, 268435456)
   502CALLFN(·call536870912, 536870912)
   503CALLFN(·call1073741824, 1073741824)
   504
   505// Not a function: target for EXRL (execute relative long) instruction.
   506TEXT callfnMVC<>(SB),NOSPLIT|NOFRAME,$0-0
   507	MVC	$1, 0(R4), 0(R6)
   508
   509TEXT runtime·procyield(SB),NOSPLIT,$0-0
   510	RET
   511
   512// Save state of caller into g->sched,
   513// but using fake PC from systemstack_switch.
   514// Must only be called from functions with no locals ($0)
   515// or else unwinding from systemstack_switch is incorrect.
   516// Smashes R1.
   517TEXT gosave_systemstack_switch<>(SB),NOSPLIT|NOFRAME,$0
   518	MOVD	$runtime·systemstack_switch(SB), R1
   519	ADD	$16, R1	// get past prologue
   520	MOVD	R1, (g_sched+gobuf_pc)(g)
   521	MOVD	R15, (g_sched+gobuf_sp)(g)
   522	MOVD	$0, (g_sched+gobuf_lr)(g)
   523	// Assert ctxt is zero. See func save.
   524	MOVD	(g_sched+gobuf_ctxt)(g), R1
   525	CMPBEQ	R1, $0, 2(PC)
   526	BL	runtime·abort(SB)
   527	RET
   528
   529// func asmcgocall(fn, arg unsafe.Pointer) int32
   530// Call fn(arg) on the scheduler stack,
   531// aligned appropriately for the gcc ABI.
   532// See cgocall.go for more details.
   533TEXT ·asmcgocall(SB),NOSPLIT,$0-20
   534	// R2 = argc; R3 = argv; R11 = temp; R13 = g; R15 = stack pointer
   535	// C TLS base pointer in AR0:AR1
   536	MOVD	fn+0(FP), R3
   537	MOVD	arg+8(FP), R4
   538
   539	MOVD	R15, R2		// save original stack pointer
   540	MOVD	g, R5
   541
   542	// Figure out if we need to switch to m->g0 stack.
   543	// We get called to create new OS threads too, and those
   544	// come in on the m->g0 stack already. Or we might already
   545	// be on the m->gsignal stack.
   546	MOVD	g_m(g), R6
   547	MOVD	m_gsignal(R6), R7
   548	CMPBEQ	R7, g, g0
   549	MOVD	m_g0(R6), R7
   550	CMPBEQ	R7, g, g0
   551	BL	gosave_systemstack_switch<>(SB)
   552	MOVD	R7, g
   553	BL	runtime·save_g(SB)
   554	MOVD	(g_sched+gobuf_sp)(g), R15
   555
   556	// Now on a scheduling stack (a pthread-created stack).
   557g0:
   558	// Save room for two of our pointers, plus 160 bytes of callee
   559	// save area that lives on the caller stack.
   560	SUB	$176, R15
   561	MOVD	$~7, R6
   562	AND	R6, R15                 // 8-byte alignment for gcc ABI
   563	MOVD	R5, 168(R15)             // save old g on stack
   564	MOVD	(g_stack+stack_hi)(R5), R5
   565	SUB	R2, R5
   566	MOVD	R5, 160(R15)             // save depth in old g stack (can't just save SP, as stack might be copied during a callback)
   567	MOVD	$0, 0(R15)              // clear back chain pointer (TODO can we give it real back trace information?)
   568	MOVD	R4, R2                  // arg in R2
   569	BL	R3                      // can clobber: R0-R5, R14, F0-F3, F5, F7-F15
   570
   571	XOR	R0, R0                  // set R0 back to 0.
   572	// Restore g, stack pointer.
   573	MOVD	168(R15), g
   574	BL	runtime·save_g(SB)
   575	MOVD	(g_stack+stack_hi)(g), R5
   576	MOVD	160(R15), R6
   577	SUB	R6, R5
   578	MOVD	R5, R15
   579
   580	MOVW	R2, ret+16(FP)
   581	RET
   582
   583// cgocallback(fn, frame unsafe.Pointer, ctxt uintptr)
   584// See cgocall.go for more details.
   585TEXT ·cgocallback(SB),NOSPLIT,$24-24
   586	NO_LOCAL_POINTERS
   587
   588	// Skip cgocallbackg, just dropm when fn is nil, and frame is the saved g.
   589	// It is used to dropm while thread is exiting.
   590	MOVD	fn+0(FP), R1
   591	CMPBNE	R1, $0, loadg
   592	// Restore the g from frame.
   593	MOVD	frame+8(FP), g
   594	BR	dropm
   595
   596loadg:
   597	// Load m and g from thread-local storage.
   598	MOVB	runtime·iscgo(SB), R3
   599	CMPBEQ	R3, $0, nocgo
   600	BL	runtime·load_g(SB)
   601
   602nocgo:
   603	// If g is nil, Go did not create the current thread,
   604	// or if this thread never called into Go on pthread platforms.
   605	// Call needm to obtain one for temporary use.
   606	// In this case, we're running on the thread stack, so there's
   607	// lots of space, but the linker doesn't know. Hide the call from
   608	// the linker analysis by using an indirect call.
   609	CMPBEQ	g, $0, needm
   610
   611	MOVD	g_m(g), R8
   612	MOVD	R8, savedm-8(SP)
   613	BR	havem
   614
   615needm:
   616	MOVD	g, savedm-8(SP) // g is zero, so is m.
   617	MOVD	$runtime·needAndBindM(SB), R3
   618	BL	(R3)
   619
   620	// Set m->sched.sp = SP, so that if a panic happens
   621	// during the function we are about to execute, it will
   622	// have a valid SP to run on the g0 stack.
   623	// The next few lines (after the havem label)
   624	// will save this SP onto the stack and then write
   625	// the same SP back to m->sched.sp. That seems redundant,
   626	// but if an unrecovered panic happens, unwindm will
   627	// restore the g->sched.sp from the stack location
   628	// and then systemstack will try to use it. If we don't set it here,
   629	// that restored SP will be uninitialized (typically 0) and
   630	// will not be usable.
   631	MOVD	g_m(g), R8
   632	MOVD	m_g0(R8), R3
   633	MOVD	R15, (g_sched+gobuf_sp)(R3)
   634
   635havem:
   636	// Now there's a valid m, and we're running on its m->g0.
   637	// Save current m->g0->sched.sp on stack and then set it to SP.
   638	// Save current sp in m->g0->sched.sp in preparation for
   639	// switch back to m->curg stack.
   640	// NOTE: unwindm knows that the saved g->sched.sp is at 8(R1) aka savedsp-16(SP).
   641	MOVD	m_g0(R8), R3
   642	MOVD	(g_sched+gobuf_sp)(R3), R4
   643	MOVD	R4, savedsp-24(SP)	// must match frame size
   644	MOVD	R15, (g_sched+gobuf_sp)(R3)
   645
   646	// Switch to m->curg stack and call runtime.cgocallbackg.
   647	// Because we are taking over the execution of m->curg
   648	// but *not* resuming what had been running, we need to
   649	// save that information (m->curg->sched) so we can restore it.
   650	// We can restore m->curg->sched.sp easily, because calling
   651	// runtime.cgocallbackg leaves SP unchanged upon return.
   652	// To save m->curg->sched.pc, we push it onto the curg stack and
   653	// open a frame the same size as cgocallback's g0 frame.
   654	// Once we switch to the curg stack, the pushed PC will appear
   655	// to be the return PC of cgocallback, so that the traceback
   656	// will seamlessly trace back into the earlier calls.
   657	MOVD	m_curg(R8), g
   658	BL	runtime·save_g(SB)
   659	MOVD	(g_sched+gobuf_sp)(g), R4 // prepare stack as R4
   660	MOVD	(g_sched+gobuf_pc)(g), R5
   661	MOVD	R5, -(24+8)(R4)	// "saved LR"; must match frame size
   662	// Gather our arguments into registers.
   663	MOVD	fn+0(FP), R1
   664	MOVD	frame+8(FP), R2
   665	MOVD	ctxt+16(FP), R3
   666	MOVD	$-(24+8)(R4), R15	// switch stack; must match frame size
   667	MOVD	R1, 8(R15)
   668	MOVD	R2, 16(R15)
   669	MOVD	R3, 24(R15)
   670	BL	runtime·cgocallbackg(SB)
   671
   672	// Restore g->sched (== m->curg->sched) from saved values.
   673	MOVD	0(R15), R5
   674	MOVD	R5, (g_sched+gobuf_pc)(g)
   675	MOVD	$(24+8)(R15), R4	// must match frame size
   676	MOVD	R4, (g_sched+gobuf_sp)(g)
   677
   678	// Switch back to m->g0's stack and restore m->g0->sched.sp.
   679	// (Unlike m->curg, the g0 goroutine never uses sched.pc,
   680	// so we do not have to restore it.)
   681	MOVD	g_m(g), R8
   682	MOVD	m_g0(R8), g
   683	BL	runtime·save_g(SB)
   684	MOVD	(g_sched+gobuf_sp)(g), R15
   685	MOVD	savedsp-24(SP), R4	// must match frame size
   686	MOVD	R4, (g_sched+gobuf_sp)(g)
   687
   688	// If the m on entry was nil, we called needm above to borrow an m,
   689	// 1. for the duration of the call on non-pthread platforms,
   690	// 2. or the duration of the C thread alive on pthread platforms.
   691	// If the m on entry wasn't nil,
   692	// 1. the thread might be a Go thread,
   693	// 2. or it wasn't the first call from a C thread on pthread platforms,
   694	//    since then we skip dropm to reuse the m in the first call.
   695	MOVD	savedm-8(SP), R6
   696	CMPBNE	R6, $0, droppedm
   697
   698	// Skip dropm to reuse it in the next call, when a pthread key has been created.
   699	MOVD	_cgo_pthread_key_created(SB), R6
   700	// It means cgo is disabled when _cgo_pthread_key_created is a nil pointer, need dropm.
   701	CMPBEQ	R6, $0, dropm
   702	MOVD	(R6), R6
   703	CMPBNE	R6, $0, droppedm
   704
   705dropm:
   706	MOVD	$runtime·dropm(SB), R3
   707	BL	(R3)
   708droppedm:
   709
   710	// Done!
   711	RET
   712
   713// void setg(G*); set g. for use by needm.
   714TEXT runtime·setg(SB), NOSPLIT, $0-8
   715	MOVD	gg+0(FP), g
   716	// This only happens if iscgo, so jump straight to save_g
   717	BL	runtime·save_g(SB)
   718	RET
   719
   720// void setg_gcc(G*); set g in C TLS.
   721// Must obey the gcc calling convention.
   722TEXT setg_gcc<>(SB),NOSPLIT|NOFRAME,$0-0
   723	// The standard prologue clobbers LR (R14), which is callee-save in
   724	// the C ABI, so we have to use NOFRAME and save LR ourselves.
   725	MOVD	LR, R1
   726	// Also save g, R10, and R11 since they're callee-save in C ABI
   727	MOVD	R10, R3
   728	MOVD	g, R4
   729	MOVD	R11, R5
   730
   731	MOVD	R2, g
   732	BL	runtime·save_g(SB)
   733
   734	MOVD	R5, R11
   735	MOVD	R4, g
   736	MOVD	R3, R10
   737	MOVD	R1, LR
   738	RET
   739
   740TEXT runtime·abort(SB),NOSPLIT|NOFRAME,$0-0
   741	MOVW	(R0), R0
   742	UNDEF
   743
   744// int64 runtime·cputicks(void)
   745TEXT runtime·cputicks(SB),NOSPLIT,$0-8
   746	// The TOD clock on s390 counts from the year 1900 in ~250ps intervals.
   747	// This means that since about 1972 the msb has been set, making the
   748	// result of a call to STORE CLOCK (stck) a negative number.
   749	// We clear the msb to make it positive.
   750	STCK	ret+0(FP)      // serialises before and after call
   751	MOVD	ret+0(FP), R3  // R3 will wrap to 0 in the year 2043
   752	SLD	$1, R3
   753	SRD	$1, R3
   754	MOVD	R3, ret+0(FP)
   755	RET
   756
   757// AES hashing not implemented for s390x
   758TEXT runtime·memhash(SB),NOSPLIT|NOFRAME,$0-32
   759	JMP	runtime·memhashFallback(SB)
   760TEXT runtime·strhash(SB),NOSPLIT|NOFRAME,$0-24
   761	JMP	runtime·strhashFallback(SB)
   762TEXT runtime·memhash32(SB),NOSPLIT|NOFRAME,$0-24
   763	JMP	runtime·memhash32Fallback(SB)
   764TEXT runtime·memhash64(SB),NOSPLIT|NOFRAME,$0-24
   765	JMP	runtime·memhash64Fallback(SB)
   766
   767// Called from cgo wrappers, this function returns g->m->curg.stack.hi.
   768// Must obey the gcc calling convention.
   769TEXT _cgo_topofstack(SB),NOSPLIT|NOFRAME,$0
   770	// g (R13), R10, R11 and LR (R14) are callee-save in the C ABI, so save them
   771	MOVD	g, R1
   772	MOVD	R10, R3
   773	MOVD	LR, R4
   774	MOVD	R11, R5
   775
   776	BL	runtime·load_g(SB)	// clobbers g (R13), R10, R11
   777	MOVD	g_m(g), R2
   778	MOVD	m_curg(R2), R2
   779	MOVD	(g_stack+stack_hi)(R2), R2
   780
   781	MOVD	R1, g
   782	MOVD	R3, R10
   783	MOVD	R4, LR
   784	MOVD	R5, R11
   785	RET
   786
   787// The top-most function running on a goroutine
   788// returns to goexit+PCQuantum.
   789TEXT runtime·goexit(SB),NOSPLIT|NOFRAME|TOPFRAME,$0-0
   790	BYTE $0x07; BYTE $0x00; // 2-byte nop
   791	BL	runtime·goexit1(SB)	// does not return
   792	// traceback from goexit1 must hit code range of goexit
   793	BYTE $0x07; BYTE $0x00; // 2-byte nop
   794
   795TEXT ·publicationBarrier(SB),NOSPLIT|NOFRAME,$0-0
   796	// Stores are already ordered on s390x, so this is just a
   797	// compile barrier.
   798	RET
   799
   800// This is called from .init_array and follows the platform, not Go, ABI.
   801// We are overly conservative. We could only save the registers we use.
   802// However, since this function is only called once per loaded module
   803// performance is unimportant.
   804TEXT runtime·addmoduledata(SB),NOSPLIT|NOFRAME,$0-0
   805	// Save R6-R15 in the register save area of the calling function.
   806	// Don't bother saving F8-F15 as we aren't doing any calls.
   807	STMG	R6, R15, 48(R15)
   808
   809	// append the argument (passed in R2, as per the ELF ABI) to the
   810	// moduledata linked list.
   811	MOVD	runtime·lastmoduledatap(SB), R1
   812	MOVD	R2, moduledata_next(R1)
   813	MOVD	R2, runtime·lastmoduledatap(SB)
   814
   815	// Restore R6-R15.
   816	LMG	48(R15), R6, R15
   817	RET
   818
   819TEXT ·checkASM(SB),NOSPLIT,$0-1
   820	MOVB	$1, ret+0(FP)
   821	RET
   822
   823// gcWriteBarrier informs the GC about heap pointer writes.
   824//
   825// gcWriteBarrier does NOT follow the Go ABI. It accepts the
   826// number of bytes of buffer needed in R9, and returns a pointer
   827// to the buffer space in R9.
   828// It clobbers R10 (the temp register) and R1 (used by PLT stub).
   829// It does not clobber any other general-purpose registers,
   830// but may clobber others (e.g., floating point registers).
   831TEXT gcWriteBarrier<>(SB),NOSPLIT,$96
   832	// Save the registers clobbered by the fast path.
   833	MOVD	R4, 96(R15)
   834retry:
   835	MOVD	g_m(g), R1
   836	MOVD	m_p(R1), R1
   837	// Increment wbBuf.next position.
   838	MOVD	R9, R4
   839	ADD	(p_wbBuf+wbBuf_next)(R1), R4
   840	// Is the buffer full?
   841	MOVD	(p_wbBuf+wbBuf_end)(R1), R10
   842	CMPUBGT	R4, R10, flush
   843	// Commit to the larger buffer.
   844	MOVD	R4, (p_wbBuf+wbBuf_next)(R1)
   845	// Make return value (the original next position)
   846	SUB	R9, R4, R9
   847	// Restore registers.
   848	MOVD	96(R15), R4
   849	RET
   850
   851flush:
   852	// Save all general purpose registers since these could be
   853	// clobbered by wbBufFlush and were not saved by the caller.
   854	STMG	R2, R3, 8(R15)
   855	MOVD	R0, 24(R15)
   856	// R1 already saved.
   857	// R4 already saved.
   858	STMG	R5, R12, 32(R15) // save R5 - R12
   859	// R13 is g.
   860	// R14 is LR.
   861	// R15 is SP.
   862
   863	CALL	runtime·wbBufFlush(SB)
   864
   865	LMG	8(R15), R2, R3   // restore R2 - R3
   866	MOVD	24(R15), R0      // restore R0
   867	LMG	32(R15), R5, R12 // restore R5 - R12
   868	JMP	retry
   869
   870TEXT runtime·gcWriteBarrier1<ABIInternal>(SB),NOSPLIT,$0
   871	MOVD	$8, R9
   872	JMP	gcWriteBarrier<>(SB)
   873TEXT runtime·gcWriteBarrier2<ABIInternal>(SB),NOSPLIT,$0
   874	MOVD	$16, R9
   875	JMP	gcWriteBarrier<>(SB)
   876TEXT runtime·gcWriteBarrier3<ABIInternal>(SB),NOSPLIT,$0
   877	MOVD	$24, R9
   878	JMP	gcWriteBarrier<>(SB)
   879TEXT runtime·gcWriteBarrier4<ABIInternal>(SB),NOSPLIT,$0
   880	MOVD	$32, R9
   881	JMP	gcWriteBarrier<>(SB)
   882TEXT runtime·gcWriteBarrier5<ABIInternal>(SB),NOSPLIT,$0
   883	MOVD	$40, R9
   884	JMP	gcWriteBarrier<>(SB)
   885TEXT runtime·gcWriteBarrier6<ABIInternal>(SB),NOSPLIT,$0
   886	MOVD	$48, R9
   887	JMP	gcWriteBarrier<>(SB)
   888TEXT runtime·gcWriteBarrier7<ABIInternal>(SB),NOSPLIT,$0
   889	MOVD	$56, R9
   890	JMP	gcWriteBarrier<>(SB)
   891TEXT runtime·gcWriteBarrier8<ABIInternal>(SB),NOSPLIT,$0
   892	MOVD	$64, R9
   893	JMP	gcWriteBarrier<>(SB)
   894
   895// Note: these functions use a special calling convention to save generated code space.
   896// Arguments are passed in registers, but the space for those arguments are allocated
   897// in the caller's stack frame. These stubs write the args into that stack space and
   898// then tail call to the corresponding runtime handler.
   899// The tail call makes these stubs disappear in backtraces.
   900TEXT runtime·panicIndex(SB),NOSPLIT,$0-16
   901	MOVD	R0, x+0(FP)
   902	MOVD	R1, y+8(FP)
   903	JMP	runtime·goPanicIndex(SB)
   904TEXT runtime·panicIndexU(SB),NOSPLIT,$0-16
   905	MOVD	R0, x+0(FP)
   906	MOVD	R1, y+8(FP)
   907	JMP	runtime·goPanicIndexU(SB)
   908TEXT runtime·panicSliceAlen(SB),NOSPLIT,$0-16
   909	MOVD	R1, x+0(FP)
   910	MOVD	R2, y+8(FP)
   911	JMP	runtime·goPanicSliceAlen(SB)
   912TEXT runtime·panicSliceAlenU(SB),NOSPLIT,$0-16
   913	MOVD	R1, x+0(FP)
   914	MOVD	R2, y+8(FP)
   915	JMP	runtime·goPanicSliceAlenU(SB)
   916TEXT runtime·panicSliceAcap(SB),NOSPLIT,$0-16
   917	MOVD	R1, x+0(FP)
   918	MOVD	R2, y+8(FP)
   919	JMP	runtime·goPanicSliceAcap(SB)
   920TEXT runtime·panicSliceAcapU(SB),NOSPLIT,$0-16
   921	MOVD	R1, x+0(FP)
   922	MOVD	R2, y+8(FP)
   923	JMP	runtime·goPanicSliceAcapU(SB)
   924TEXT runtime·panicSliceB(SB),NOSPLIT,$0-16
   925	MOVD	R0, x+0(FP)
   926	MOVD	R1, y+8(FP)
   927	JMP	runtime·goPanicSliceB(SB)
   928TEXT runtime·panicSliceBU(SB),NOSPLIT,$0-16
   929	MOVD	R0, x+0(FP)
   930	MOVD	R1, y+8(FP)
   931	JMP	runtime·goPanicSliceBU(SB)
   932TEXT runtime·panicSlice3Alen(SB),NOSPLIT,$0-16
   933	MOVD	R2, x+0(FP)
   934	MOVD	R3, y+8(FP)
   935	JMP	runtime·goPanicSlice3Alen(SB)
   936TEXT runtime·panicSlice3AlenU(SB),NOSPLIT,$0-16
   937	MOVD	R2, x+0(FP)
   938	MOVD	R3, y+8(FP)
   939	JMP	runtime·goPanicSlice3AlenU(SB)
   940TEXT runtime·panicSlice3Acap(SB),NOSPLIT,$0-16
   941	MOVD	R2, x+0(FP)
   942	MOVD	R3, y+8(FP)
   943	JMP	runtime·goPanicSlice3Acap(SB)
   944TEXT runtime·panicSlice3AcapU(SB),NOSPLIT,$0-16
   945	MOVD	R2, x+0(FP)
   946	MOVD	R3, y+8(FP)
   947	JMP	runtime·goPanicSlice3AcapU(SB)
   948TEXT runtime·panicSlice3B(SB),NOSPLIT,$0-16
   949	MOVD	R1, x+0(FP)
   950	MOVD	R2, y+8(FP)
   951	JMP	runtime·goPanicSlice3B(SB)
   952TEXT runtime·panicSlice3BU(SB),NOSPLIT,$0-16
   953	MOVD	R1, x+0(FP)
   954	MOVD	R2, y+8(FP)
   955	JMP	runtime·goPanicSlice3BU(SB)
   956TEXT runtime·panicSlice3C(SB),NOSPLIT,$0-16
   957	MOVD	R0, x+0(FP)
   958	MOVD	R1, y+8(FP)
   959	JMP	runtime·goPanicSlice3C(SB)
   960TEXT runtime·panicSlice3CU(SB),NOSPLIT,$0-16
   961	MOVD	R0, x+0(FP)
   962	MOVD	R1, y+8(FP)
   963	JMP	runtime·goPanicSlice3CU(SB)
   964TEXT runtime·panicSliceConvert(SB),NOSPLIT,$0-16
   965	MOVD	R2, x+0(FP)
   966	MOVD	R3, y+8(FP)
   967	JMP	runtime·goPanicSliceConvert(SB)
View as plain text