241 строка
		
	
	
	
		
			5,2 КиБ
		
	
	
	
		
			ArmAsm
		
	
	
	
	
	
			
		
		
	
	
			241 строка
		
	
	
	
		
			5,2 КиБ
		
	
	
	
		
			ArmAsm
		
	
	
	
	
	
| .section .bss.tinygo_systemStack
 | |
| .global tinygo_systemStack
 | |
| .type   tinygo_systemStack, %object
 | |
| tinygo_systemStack:
 | |
|     .short 0
 | |
| 
 | |
| .section .text.tinygo_startTask
 | |
| .global  tinygo_startTask
 | |
| .type    tinygo_startTask, %function
 | |
| tinygo_startTask:
 | |
|     // Small assembly stub for starting a goroutine. This is already run on the
 | |
|     // new stack, with the callee-saved registers already loaded.
 | |
|     // Most importantly, r2r3 contain the pc of the to-be-started function and
 | |
|     // r4r5 contain the only argument it is given. Multiple arguments are packed
 | |
|     // into one by storing them in a new allocation.
 | |
| 
 | |
|     // Set the first argument of the goroutine start wrapper, which contains all
 | |
|     // the arguments.
 | |
|     movw  r24, r4
 | |
| 
 | |
|     // Branch to the "goroutine start" function. Note that the Z register is
 | |
|     // call-clobbered, so does not need to be restored after use.
 | |
|     movw  Z, r2
 | |
|     icall
 | |
| 
 | |
|     // After return, exit this goroutine. This is a tail call.
 | |
| #if __AVR_ARCH__ == 2 || __AVR_ARCH__ == 25
 | |
|     // Small memory devices (≤8kB flash) that do not have the long call
 | |
|     // instruction availble will need to use rcall instead.
 | |
|     // Note that they will probably not be able to run more than the main
 | |
|     // goroutine anyway, but this file is compiled for all AVRs so it needs to
 | |
|     // compile at least.
 | |
|     rcall  tinygo_pause
 | |
| #else
 | |
|     // Other devices can (and must) use the regular call instruction.
 | |
|     call tinygo_pause
 | |
| #endif
 | |
| 
 | |
| // Get the system stack pointer, independent of whether we're currently on the
 | |
| // system stack or a task stack.
 | |
| .global tinygo_getSystemStackPointer
 | |
| .type tinygo_getSystemStackPointer, %function
 | |
| tinygo_getSystemStackPointer:
 | |
|     // Load system stack pointer.
 | |
|     lds r24, tinygo_systemStack
 | |
|     lds r25, tinygo_systemStack+1
 | |
| 
 | |
|     // Compare against 0.
 | |
|     cp  r24, r1
 | |
|     cpc r25, r1
 | |
| 
 | |
|     // Branch (and then return) if tinygo_systemStack has a non-zero value.
 | |
|     brne 1f
 | |
| 
 | |
|     // tinygo_systemStack is zero, so return the current stack pointer.
 | |
|     in  r24, 0x3d; SPL
 | |
|     in  r25, 0x3e; SPH
 | |
| 
 | |
| 1:
 | |
|     ret
 | |
| 
 | |
| .global tinygo_switchToTask
 | |
| .type tinygo_switchToTask, %function
 | |
| tinygo_switchToTask:
 | |
|     // The sp parameter is the only parameter, so it will take up r24:r25.
 | |
|     // r24:r25 = sp uintptr
 | |
| 
 | |
|     // Save all call-saved registers:
 | |
|     // https://gcc.gnu.org/wiki/avr-gcc#Call-Saved_Registers
 | |
|     push r29 // Y
 | |
|     push r28 // Y
 | |
|     push r17
 | |
|     push r16
 | |
|     push r15
 | |
|     push r14
 | |
|     push r13
 | |
|     push r12
 | |
|     push r11
 | |
|     push r10
 | |
|     push r9
 | |
|     push r8
 | |
|     push r7
 | |
|     push r6
 | |
|     push r5
 | |
|     push r4
 | |
|     push r3
 | |
|     push r2
 | |
| 
 | |
|     // Save the system stack pointer in a global.
 | |
|     in  r2, 0x3d; SPL
 | |
|     in  r3, 0x3e; SPH
 | |
|     sts tinygo_systemStack+0, r2
 | |
|     sts tinygo_systemStack+1, r3
 | |
| 
 | |
|     // Switch to the task stack pointer.
 | |
|     out  0x3d, r24; SPL
 | |
|     out  0x3e, r25; SPH
 | |
| 
 | |
|     // Load saved register from the task stack.
 | |
|     pop r2
 | |
|     pop r3
 | |
|     pop r4
 | |
|     pop r5
 | |
|     pop r6
 | |
|     pop r7
 | |
|     pop r8
 | |
|     pop r9
 | |
|     pop r10
 | |
|     pop r11
 | |
|     pop r12
 | |
|     pop r13
 | |
|     pop r14
 | |
|     pop r15
 | |
|     pop r16
 | |
|     pop r17
 | |
|     pop r28 // Y
 | |
|     pop r29 // Y
 | |
| 
 | |
|     // Return into the new task, as if tinygo_switchToScheduler was a regular
 | |
|     // call.
 | |
|     ret
 | |
| 
 | |
| .global tinygo_switchToScheduler
 | |
| .type tinygo_switchToScheduler, %function
 | |
| tinygo_switchToScheduler:
 | |
|     // The sp parameter is the only parameter, so it will take up r24:r25.
 | |
|     // r24:r25 = sp *uintptr
 | |
| 
 | |
|     // Save all call-saved registers on the task stack:
 | |
|     // https://gcc.gnu.org/wiki/avr-gcc#Call-Saved_Registers
 | |
|     push r29 // Y
 | |
|     push r28 // Y
 | |
|     push r17
 | |
|     push r16
 | |
|     push r15
 | |
|     push r14
 | |
|     push r13
 | |
|     push r12
 | |
|     push r11
 | |
|     push r10
 | |
|     push r9
 | |
|     push r8
 | |
|     push r7
 | |
|     push r6
 | |
|     push r5
 | |
|     push r4
 | |
|     push r3
 | |
|     push r2
 | |
| 
 | |
|     // Save the task stack.
 | |
|     in  r2, 0x3d; SPL
 | |
|     in  r3, 0x3e; SPH
 | |
|     movw Y, r24
 | |
|     std Y+0, r2
 | |
|     std Y+1, r3
 | |
| 
 | |
|     // Switch to the system stack.
 | |
|     lds r2, tinygo_systemStack
 | |
|     lds r3, tinygo_systemStack+1
 | |
|     out  0x3d, r2; SPL
 | |
|     out  0x3e, r3; SPH
 | |
| 
 | |
|     // Clear tinygo_systemStack to make sure tinygo_getSystemStackPointer knows
 | |
|     // which pointer to return.
 | |
|     sts tinygo_systemStack+0, r1
 | |
|     sts tinygo_systemStack+1, r1
 | |
| 
 | |
|     // Load saved register from the system stack.
 | |
|     pop r2
 | |
|     pop r3
 | |
|     pop r4
 | |
|     pop r5
 | |
|     pop r6
 | |
|     pop r7
 | |
|     pop r8
 | |
|     pop r9
 | |
|     pop r10
 | |
|     pop r11
 | |
|     pop r12
 | |
|     pop r13
 | |
|     pop r14
 | |
|     pop r15
 | |
|     pop r16
 | |
|     pop r17
 | |
|     pop r28 // Y
 | |
|     pop r29 // Y
 | |
| 
 | |
|     // Return into the scheduler, as if tinygo_switchToTask was a regular call.
 | |
|     ret
 | |
| 
 | |
| .global tinygo_scanCurrentStack
 | |
| .type tinygo_scanCurrentStack, %function
 | |
| tinygo_scanCurrentStack:
 | |
|     // Save callee-saved registers.
 | |
|     push r29 // Y
 | |
|     push r28 // Y
 | |
|     push r17
 | |
|     push r16
 | |
|     push r15
 | |
|     push r14
 | |
|     push r13
 | |
|     push r12
 | |
|     push r11
 | |
|     push r10
 | |
|     push r9
 | |
|     push r8
 | |
|     push r7
 | |
|     push r6
 | |
|     push r5
 | |
|     push r4
 | |
|     push r3
 | |
|     push r2
 | |
| 
 | |
|     // Scan the stack.
 | |
|     in  r24, 0x3d; SPL
 | |
|     in  r25, 0x3e; SPH
 | |
| #if __AVR_ARCH__ == 2 || __AVR_ARCH__ == 25
 | |
|     rcall tinygo_scanstack
 | |
| #else
 | |
|     call tinygo_scanstack
 | |
| #endif
 | |
| 
 | |
|     // Restore callee-saved registers.
 | |
|     pop r2
 | |
|     pop r3
 | |
|     pop r4
 | |
|     pop r5
 | |
|     pop r6
 | |
|     pop r7
 | |
|     pop r8
 | |
|     pop r9
 | |
|     pop r10
 | |
|     pop r11
 | |
|     pop r12
 | |
|     pop r13
 | |
|     pop r14
 | |
|     pop r15
 | |
|     pop r16
 | |
|     pop r17
 | |
|     pop r28 // Y
 | |
|     pop r29 // Y
 | 
