268 lines
		
	
	
	
		
			7.3 KiB
		
	
	
	
		
			ArmAsm
		
	
	
	
	
	
			
		
		
	
	
			268 lines
		
	
	
	
		
			7.3 KiB
		
	
	
	
		
			ArmAsm
		
	
	
	
	
	
| /********************************** (C) COPYRIGHT *******************************
 | ||
|  * File Name          : startup_CH59x.s
 | ||
|  * Author             : WCH
 | ||
|  * Version            : V1.0.1
 | ||
|  * Date               : 2023/10/25
 | ||
|  * Description        : CH592 FreeRTOS启动文件
 | ||
|  *********************************************************************************
 | ||
|  * Copyright (c) 2023 Nanjing Qinheng Microelectronics Co., Ltd.
 | ||
|  * Attention: This software (modified or not) and binary are used for
 | ||
|  * microcontroller manufactured by Nanjing Qinheng Microelectronics.
 | ||
|  *******************************************************************************/
 | ||
| 
 | ||
| 	.section	.init,"ax",@progbits
 | ||
| 	.global	_start
 | ||
| 	.align	1
 | ||
| _start:
 | ||
| 	j	handle_reset
 | ||
| 
 | ||
|     .section    .vector,"ax",@progbits
 | ||
|     .global	_BB_IRQHandler_base
 | ||
|     .global	_LLE_IRQHandler_base
 | ||
|     .align  1
 | ||
| _vector_base:
 | ||
|     .option norvc;
 | ||
| 
 | ||
|     .word   0
 | ||
|     .word   0
 | ||
|     .word   NMI_Handler                 			/* NMI Handler */
 | ||
|     .word   HardFault_Handler           			/* Hard Fault Handler */
 | ||
|     .word   0xF5F9BDA9								/* boot option, can't modify */
 | ||
|     .word   Ecall_M_Mode_Handler        			/* 5 */
 | ||
|     .word   0
 | ||
|     .word   0
 | ||
|     .word   Ecall_U_Mode_Handler					/* 8 */
 | ||
|     .word   Break_Point_Handler						/* 9 */
 | ||
|     .word   0
 | ||
|     .word   0
 | ||
|     .word   SysTick_Handler         				/* SysTick Handler */
 | ||
|     .word   0
 | ||
|     .word   SW_Handler                 				/* SW Handler */
 | ||
|     .word   0
 | ||
| 
 | ||
|     /* External Interrupts */
 | ||
|     .word   unified_interrupt_entry           		/* TMR0 */
 | ||
|     .word   unified_interrupt_entry         		/* GPIOA */
 | ||
|     .word   unified_interrupt_entry           		/* GPIOB */
 | ||
|     .word   unified_interrupt_entry            		/* SPI0 */
 | ||
| 
 | ||
| _user_vector_base:
 | ||
| 	.option norvc;
 | ||
| 
 | ||
|     .word   unified_interrupt_entry              	/* BLEB */
 | ||
|     .word   LLE_IRQHandler             				/* BLEL */
 | ||
|     .word   unified_interrupt_entry             	/* USB */
 | ||
|     .word   0
 | ||
|     .word   unified_interrupt_entry          		/* TMR1 */
 | ||
|     .word   unified_interrupt_entry            		/* TMR2 */
 | ||
|     .word   unified_interrupt_entry           		/* UART0 */
 | ||
|     .word   unified_interrupt_entry           		/* UART1 */
 | ||
|     .word   unified_interrupt_entry             	/* RTC */
 | ||
|     .word   unified_interrupt_entry             	/* ADC */
 | ||
|     .word   unified_interrupt_entry 			 	/* I2C */
 | ||
|     .word   unified_interrupt_entry            		/* PWMX */
 | ||
|     .word   unified_interrupt_entry            		/* TMR3 */
 | ||
|     .word   unified_interrupt_entry           		/* UART2 */
 | ||
|     .word   unified_interrupt_entry           		/* UART3 */
 | ||
|     .word   unified_interrupt_entry        			/* WDOG_BAT */
 | ||
| 
 | ||
| _real_user_vector_base:
 | ||
| 	.option norvc;
 | ||
| 
 | ||
|     /* External Interrupts */
 | ||
|     .word   TMR0_IRQHandler            				/* TMR0 */
 | ||
|     .word   GPIOA_IRQHandler           				/* GPIOA */
 | ||
|     .word   GPIOB_IRQHandler           				/* GPIOB */
 | ||
|     .word   SPI0_IRQHandler            				/* SPI0 */
 | ||
| _BB_IRQHandler_base:
 | ||
|     .word   BB_IRQLibHandler      					/* BLEB BB_IRQHandler */
 | ||
| _LLE_IRQHandler_base:
 | ||
|     .word   LLE_IRQHandler       					/* BLEL LLE_IRQHandler */
 | ||
|     .word   USB_IRQHandler             				/* USB */
 | ||
|     .word   0
 | ||
|     .word   TMR1_IRQHandler            				/* TMR1 */
 | ||
|     .word   TMR2_IRQHandler            				/* TMR2 */
 | ||
|     .word   UART0_IRQHandler           				/* UART0 */
 | ||
|     .word   UART1_IRQHandler           				/* UART1 */
 | ||
|     .word   RTC_IRQHandler             				/* RTC */
 | ||
|     .word   ADC_IRQHandler             				/* ADC */
 | ||
|     .word   I2C_IRQHandler 			   				/* I2C */
 | ||
|     .word   PWMX_IRQHandler            				/* PWMX */
 | ||
|     .word   TMR3_IRQHandler            				/* TMR3 */
 | ||
|     .word   UART2_IRQHandler           				/* UART2 */
 | ||
|     .word   UART3_IRQHandler           				/* UART3 */
 | ||
|     .word   WDOG_BAT_IRQHandler        				/* WDOG_BAT */
 | ||
| 
 | ||
|     .section    .vector_handler, "ax", @progbits
 | ||
|     .weak   NMI_Handler
 | ||
|     .weak   HardFault_Handler
 | ||
|     .weak   Ecall_M_Mode_Handler
 | ||
|     .weak   Ecall_U_Mode_Handler
 | ||
|     .weak   Break_Point_Handler
 | ||
|     .weak   SysTick_Handler
 | ||
|     .weak   SW_Handler
 | ||
|     .weak   TMR0_IRQHandler
 | ||
|     .weak   GPIOA_IRQHandler
 | ||
|     .weak   GPIOB_IRQHandler
 | ||
|     .weak   SPI0_IRQHandler
 | ||
|     .weak   BB_IRQHandler
 | ||
|     .weak   LLE_IRQHandler
 | ||
|   	.weak   BB_IRQLibHandler
 | ||
|     .weak   USB_IRQHandler
 | ||
|     .weak   TMR1_IRQHandler
 | ||
|     .weak   TMR2_IRQHandler
 | ||
|     .weak   UART0_IRQHandler
 | ||
|     .weak   UART1_IRQHandler
 | ||
|     .weak   RTC_IRQHandler
 | ||
|     .weak   ADC_IRQHandler
 | ||
|     .weak   I2C_IRQHandler
 | ||
|     .weak   PWMX_IRQHandler
 | ||
|     .weak   TMR3_IRQHandler
 | ||
|     .weak   UART2_IRQHandler
 | ||
|     .weak   UART3_IRQHandler
 | ||
|     .weak   WDOG_BAT_IRQHandler
 | ||
| 
 | ||
| NMI_Handler:
 | ||
| HardFault_Handler:
 | ||
| Ecall_M_Mode_Handler:
 | ||
| Ecall_U_Mode_Handler:
 | ||
| Break_Point_Handler:
 | ||
| SysTick_Handler:
 | ||
| SW_Handler:
 | ||
| TMR0_IRQHandler:
 | ||
| GPIOA_IRQHandler:
 | ||
| GPIOB_IRQHandler:
 | ||
| SPI0_IRQHandler:
 | ||
| BB_IRQHandler:
 | ||
| LLE_IRQHandler:
 | ||
| BB_IRQLibHandler:
 | ||
| USB_IRQHandler:
 | ||
| TMR1_IRQHandler:
 | ||
| TMR2_IRQHandler:
 | ||
| UART0_IRQHandler:
 | ||
| UART1_IRQHandler:
 | ||
| RTC_IRQHandler:
 | ||
| ADC_IRQHandler:
 | ||
| I2C_IRQHandler:
 | ||
| PWMX_IRQHandler:
 | ||
| TMR3_IRQHandler:
 | ||
| UART2_IRQHandler:
 | ||
| UART3_IRQHandler:
 | ||
| WDOG_BAT_IRQHandler:
 | ||
| 1:
 | ||
| 	j 1b
 | ||
| 
 | ||
| 	.section	.handle_reset,"ax",@progbits
 | ||
| 	.weak	handle_reset
 | ||
| 	.align	1
 | ||
| handle_reset:
 | ||
| .option push
 | ||
| .option	norelax
 | ||
| 	la gp, __global_pointer$
 | ||
| .option	pop
 | ||
| 1:
 | ||
| 	la sp, _eusrstack
 | ||
| 
 | ||
| /* Load highcode code  section from flash to RAM */
 | ||
| 2:
 | ||
|     la a0, _highcode_lma
 | ||
|     la a1, _highcode_vma_start
 | ||
|     la a2, _highcode_vma_end
 | ||
|     bgeu a1, a2, 2f
 | ||
| 1:
 | ||
|     lw t0, (a0)
 | ||
|     sw t0, (a1)
 | ||
|     addi a0, a0, 4
 | ||
|     addi a1, a1, 4
 | ||
|     bltu a1, a2, 1b
 | ||
| 
 | ||
| /* Load data section from flash to RAM */
 | ||
| 2:
 | ||
| 	la a0, _data_lma
 | ||
| 	la a1, _data_vma
 | ||
| 	la a2, _edata
 | ||
| 	bgeu a1, a2, 2f
 | ||
| 1:
 | ||
| 	lw t0, (a0)
 | ||
| 	sw t0, (a1)
 | ||
| 	addi a0, a0, 4
 | ||
| 	addi a1, a1, 4
 | ||
| 	bltu a1, a2, 1b
 | ||
| 2:
 | ||
| 	/* clear bss section */
 | ||
| 	la a0, _sbss
 | ||
| 	la a1, _ebss
 | ||
| 	bgeu a0, a1, 2f
 | ||
| 1:
 | ||
| 	sw zero, (a0)
 | ||
| 	addi a0, a0, 4
 | ||
| 	bltu a0, a1, 1b
 | ||
| 2:
 | ||
| 	/* 流水线控制位 & 动态预测控制位 */
 | ||
| 	li t0, 0x1f
 | ||
| 	csrw 0xbc0, t0
 | ||
| 
 | ||
| 	/* 打开硬件压栈,关闭中断嵌套,在中断函数入口处再打开中断嵌套 */
 | ||
| 	li t0, 0x01
 | ||
| 	csrw 0x804, t0
 | ||
| 
 | ||
| 	/* 使用机器模式,在freertos中打开中断 */
 | ||
|     li t0, 0x1800
 | ||
|     csrw mstatus, t0
 | ||
| 
 | ||
| 	/* 配置系统异常及中断向量表,用户真正使用的向量表在统一入口 unified_interrupt_entry 中处理 */
 | ||
| 	la t0, _vector_base
 | ||
|     ori t0, t0, 3
 | ||
| 	csrw mtvec, t0
 | ||
| 
 | ||
| 	la t0, main
 | ||
| 	csrw mepc, t0
 | ||
| 
 | ||
| 	mret
 | ||
| 
 | ||
| .extern xISRStackTop
 | ||
| 
 | ||
| .section	.highcode.unified_interrupt_entry,"ax",@progbits
 | ||
| .align 2
 | ||
| .func
 | ||
| unified_interrupt_entry:
 | ||
| 	csrr a1, mcause
 | ||
| 	slli a1, a1, 2
 | ||
| 	la a0, _user_vector_base
 | ||
| 	add a0, a0, a1
 | ||
| 	lw a0, 0(a0)						/* read isr functions addr from _vector_base. */
 | ||
| 
 | ||
| #if ENABLE_INTERRUPT_NEST
 | ||
| 	csrr t0, 0x804						/* justice whether it's in interrupt nesting. */
 | ||
| 	andi t1, t0, 0x02
 | ||
| 	bnez t1, interrupt_nesting			/* if it's not zero, it's in interrupt nesting. */
 | ||
| #endif
 | ||
| 
 | ||
| 	csrw mscratch, sp
 | ||
| 	lw sp, xISRStackTop					/* Switch to ISR stack before function call. */
 | ||
| 
 | ||
| #if ENABLE_INTERRUPT_NEST
 | ||
| 	csrs 0x804, 0x02					/* enable interrupt nests. */
 | ||
| #endif
 | ||
| 
 | ||
| 	jalr x1, 0(a0)
 | ||
| 
 | ||
| #if ENABLE_INTERRUPT_NEST
 | ||
| 	csrc mstatus, 8						/* disable interrupt in machine mode. */
 | ||
| 	nop									/* wait for 3 grade pipeline. */
 | ||
| 	nop
 | ||
| 	nop
 | ||
| 	csrc 0x804, 0x02					/* disable interrupt nests. */
 | ||
| 	nop
 | ||
| #endif
 | ||
| 
 | ||
| 	csrr sp, mscratch					/* resume sp from mscratch. */
 | ||
| 	mret
 | ||
| 
 | ||
| #if ENABLE_INTERRUPT_NEST
 | ||
| interrupt_nesting:
 | ||
| 	jalr x1, 0(a0)						/* if it's in interrupt nesting, don't need to change sp. */
 | ||
| 	mret
 | ||
| #endif
 | ||
| 
 | ||
| 	.endfunc
 |