- 前言
- 一、C语言环境搭建
- 1.设置处理器模式
- 2.设置SP指针
- 3.跳转main
- 二、C语言编写
- 1.寄存器设置
- 2.主程序编写
- 三、编译下载
- 1.编写Makefile
- 2.链接脚本
- 3.程序下载
- 总结
提示:以下是本篇文章正文内容
一、C语言环境搭建阿尔法开发板没有像STM32那样有着已经写好的底层(C语言环境搭建,寄存器初始化…), 我们一般在开始部分用汇编来初始化一下 C 语言环境,比如初始化 DDR、设置堆栈指针 SP ,然后就可以进入C语言运行环境,进入main()函数,执行工作。
1.设置处理器模式设置阿尔法处于SVC模式(Supervisor(SVC) 超级管理员模式,特权模式,供操作系统使用)下
CPSR(程序状态寄存器):该寄存器包含了条件标志位、中断禁止位、当前处理器模式标志等一些状态位以及一些控制位
所以要使用SVC模式,我们要设置CPSR寄存器的bit4-0,也就是M[4:0]为10011=0X13
CPSR不是普通的寄存器读写不能用LDR,读写状态寄存器必须用到MRS和MSR指令(MRS将CPSR寄存器数据读出到通用寄存器里面,MSR指令将通用寄存器的值写入到CPSR寄存器里面去)
@进入SVC模式
mrs r0, cpsr
bic r0, r0, #0x1f @将r0寄存器中的低5位清零,为了不影响其他位
orr r0, r0, #0x13 @r0或上0x13,表示使用SVC模式
msr cpsr, r0 @将r0 的数据写入到cpsr_c中
2.设置SP指针
Sp可以指向内部RAM,也可以指向DDR将其指向DDR
512MB的范围0x80000000~0x9FFFFFFF。栈大小,0x200000=2MB。处理器栈增长方式,对于A7而言是向下增长的。设置sp指向0x80200000
设置指令:
ldr sp, =0X80200000 @设置栈指针
注:这里并没有初始化DDR, 前面我们知道DCD 数据包含了 DDR 配置参数, MX6U 内部的 Boot ROM 会读取 DCD 数据中的 DDR 配置参数然后完成 DDR 初始化的
3.跳转main使用b指令,跳转到main函数
b main @跳转到main函数
二、C语言编写
1.寄存器设置
使用C语言来编程,还要定义寄存器的地址
/*
* CCM相关寄存器地址
*/
#define CCM_CCGR0 *((volatile unsigned int *)0X020C4068)
#define CCM_CCGR1 *((volatile unsigned int *)0X020C406C)
#define CCM_CCGR2 *((volatile unsigned int *)0X020C4070)
#define CCM_CCGR3 *((volatile unsigned int *)0X020C4074)
#define CCM_CCGR4 *((volatile unsigned int *)0X020C4078)
#define CCM_CCGR5 *((volatile unsigned int *)0X020C407C)
#define CCM_CCGR6 *((volatile unsigned int *)0X020C4080)
/*
* IOMUX相关寄存器地址
*/
#define SW_MUX_GPIO1_IO03 *((volatile unsigned int *)0X020E0068)
#define SW_PAD_GPIO1_IO03 *((volatile unsigned int *)0X020E02F4)
/*
* GPIO1相关寄存器地址
*/
#define GPIO1_DR *((volatile unsigned int *)0X0209C000)
#define GPIO1_GDIR *((volatile unsigned int *)0X0209C004)
#define GPIO1_PSR *((volatile unsigned int *)0X0209C008)
#define GPIO1_ICR1 *((volatile unsigned int *)0X0209C00C)
#define GPIO1_ICR2 *((volatile unsigned int *)0X0209C010)
#define GPIO1_IMR *((volatile unsigned int *)0X0209C014)
#define GPIO1_ISR *((volatile unsigned int *)0X0209C018)
#define GPIO1_EDGE_SEL *((volatile unsigned int *)0X0209C01C)
volatile 关键字是一种类型修饰符,声明的类型变量表示可以被某些编译器未知的因素更改
volatile 提醒编译器它后面所修饰的变量随时都有可能改变,因此编译后的程序每次需要存储或读取这个变量的时候,都会直接从变量地址中读取数据
如果没有 volatile 关键字,则编译器可能优化读取和存储,可能暂时使用寄存器中的值,如果这个变量由别的程序更新了的话,将出现不一致的现象
所以volatile关键字声明的变量,编译器对访问该变量的代码就不再进行优化,从而可以提供对特殊地址的稳定访问
(volatile unsigned int *)0X020C4068: 将地址0X020C4068强制转化为int型指针 *((volatile unsigned int *)0X020C4068): 指向该地址的值的变量
以上代码都可以写在main.h里面
2.主程序编写主程序的编写和前面汇编编写的流程一样: 1.使能 GPIO 对应的时钟
void clk_enable(void)
{
CCM_CCGR0 = 0xffffffff;
CCM_CCGR1 = 0xffffffff;
CCM_CCGR2 = 0xffffffff;
CCM_CCGR3 = 0xffffffff;
CCM_CCGR4 = 0xffffffff;
CCM_CCGR5 = 0xffffffff;
CCM_CCGR6 = 0xffffffff;
}
2.设置寄存器 IOMUXC_SW_MUX_CTL_PAD_XX_XX,设置 IO 的复用功能,使其复用为 GPIO 功能
3.设置寄存器 IOMUXC_SW_PAD_CTL_PAD_XX_XX,设置 IO 的上下拉、速度的属性
4.配置 GPIO,设置输入/输出、是否使用中断、默认输出电平
void led_init(void)
{
/* 1、初始化IO复用 */
SW_MUX_GPIO1_IO03 = 0x5; /* 复用为GPIO1_IO03 */
/* 2、、配置GPIO1_IO03的IO属性
*bit 16:0 HYS关闭
*bit [15:14]: 00 默认下拉
*bit [13]: 0 kepper功能
*bit [12]: 1 pull/keeper使能
*bit [11]: 0 关闭开路输出
*bit [7:6]: 10 速度100Mhz
*bit [5:3]: 110 R0/6驱动能力
*bit [0]: 0 低转换率
*/
SW_PAD_GPIO1_IO03 = 0X10B0;
/* 3、初始化GPIO */
GPIO1_GDIR = 0X0000008; /* GPIO1_IO03设置为输出 */
/* 4、设置GPIO1_IO03输出低电平,打开LED0 */
GPIO1_DR = 0X0;
}
三、编译下载
1.编写Makefile
Makefile 里面是由一系列的规则组成的,规则格式
目标…… : 依赖文件集合…… 命令 1 命令 2 (必须有一个Tab键)
Makefile自动化变量
objs := start.o main.o
ledc.bin:$(objs)
arm-linux-gnueabihf-ld -Ttext 0X87800000 -o ledc.elf $^
arm-linux-gnueabihf-objcopy -O binary -S ledc.elf $@
arm-linux-gnueabihf-objdump -D -m arm ledc.elf > ledc.dis
%.o:%.s
arm-linux-gnueabihf-gcc -Wall -nostdlib -c -o $@ $
关注
打赏
最近更新
- 深拷贝和浅拷贝的区别(重点)
- 【Vue】走进Vue框架世界
- 【云服务器】项目部署—搭建网站—vue电商后台管理系统
- 【React介绍】 一文带你深入React
- 【React】React组件实例的三大属性之state,props,refs(你学废了吗)
- 【脚手架VueCLI】从零开始,创建一个VUE项目
- 【React】深入理解React组件生命周期----图文详解(含代码)
- 【React】DOM的Diffing算法是什么?以及DOM中key的作用----经典面试题
- 【React】1_使用React脚手架创建项目步骤--------详解(含项目结构说明)
- 【React】2_如何使用react脚手架写一个简单的页面?