I.MX6U点灯实验C语言版

笔记性质的文章。

开发流程梳理

想要在CPU上运行C程序,需要做以下的事情:

  1. 编写.S文件(_start.S)设置CPU的工作模式并且指定栈区大小
  2. 写点灯程序
  3. 编译
  4. 烧录

以下一步一步配合代码记录。

.S文件编写

在.S文件里,只需要设置好CPU工作模式,然后指定栈指针(SP)位置即可。其中,栈指针是运行C程序的前提,对应参考是STM32里面的栈区的初始化。以下是代码:


.global _start

_start:
    /* SET CPU TO SVC MODE */
    mrs r0, cpsr // read CPSR to r0
    bic r0, r0, #0x1f// clear CPSR[4:0]
    orr r0, r0, #0x13 // write CPSR[4:0] 0X13
    msr cpsr, r0  //write cpsr by r0

    /* SET STACK POINTER */
    ldr sp, =0x80200000
    b main //GO TO C MAIN FUNCTION

I.MX6U的CPU共有9个运行模型,由CPSR寄存器的值来决定。这里具体解释一下代码

 mrs r0, cpsr

读取CPSR寄存器的值,存到通用寄存器r0中。注意,读取的命令是mrs,不是一般用的ldr。

bic r0, r0, #0x1f

bic是位清除指令。需要清除的位标记位1即可。0x1f = 11111b, 这里实际是:r0 = r0 & ~(11111)所以清除的位置是r0寄存器中的[4:0]。 此时r0寄存器里的值是由cpsr寄存器读来的,而cpsr[4:0]就是配置CPU工作模式的。

orr r0, r0, #0x13 

orr是位或指令,这里是将r0[4:0]写为10011b。

 msr cpsr, r0

这里是把r0的值写回到cpsr寄存器中,注意使用的是msr不是str。

ldr sp, =0x80200000

这句就是指定SP位置了。在I.MX6U的这个开发板中,DDR范围是0x80000000~0xA0000000,一共512MB。堆栈的方向都是向下,将SVC模式栈大小设置为2MB,所以SP指针位置是0x80000000+0x200000 = 0x80200000

点灯程序编写

要点灯,就是配置寄存器,我们需要配置的是LED对应IO的输出,所以C程序里就是配置和初始化。
先看一下头文件的代码:

#ifndef __MAIN_H
#define __MAIN

//DEFINE CCM REGISTERS
#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)

//DEFINE IOMUX REGISTERS
#define SW_MUX_GPIO1_IO03 *((volatile unsigned int*)0X020E0068)
#define SW_PAD_GPIO1_IO03 *((volatile unsigned int*)0X020E02F4)


//DEFINE GPIO REGISTERS
#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)

#endif

点灯程序比较简单,所以头文件里面就只定义了GPIO相关的寄存器,有了这些定义,就可以去写代码了。下面是main的代码:

#include "main.h"
#include <stdio.h>


/*ENABLE PERIPHERAL CLOCKS*/
void clock_enable(void)
{
    CCM_CCGR1 = 0xFFFFFFFF;
    CCM_CCGR2 = 0xFFFFFFFF;
    CCM_CCGR3 = 0xFFFFFFFF;
    CCM_CCGR4 = 0xFFFFFFFF;
    CCM_CCGR5 = 0xFFFFFFFF;
    CCM_CCGR6 = 0xFFFFFFFF;
}

void led_init(void)
{
    SW_MUX_GPIO1_IO03 = 0x5; //MUX: GPIO1
    SW_PAD_GPIO1_IO03 = 0x10B0; //GPIO1 ELEC

    /*GPIO INIT*/
    GPIO1_GDIR = 0x8; //OUTPUT
    GPIO1_DR = 0x0; //LED ON
}
/* SHORT DELAY*/
void delay_short(volatile unsigned int n)
{
    while(n--)
    {
    }
}

/*DELAY FUNCTION
 * n: delay n ms
 * 1 cycle 1ms in 396MHz

*/
void delay(volatile unsigned int n)
{
    while(n--)
    {
        delay_short(0x7ff);
    }
}

/*LED ON*/
void led_on(void)
{
    GPIO1_DR &= ~(1<<3); //BIT 3 = 0
}

/*LED OFF*/
void led_off(void)
{
    GPIO1_DR |= (1<<3); //BIT 3 = 1
}



int main(void)
{
    clock_enable();//ENABLE PERIPHERAL CLOCKS
    led_init();
    //INIT LED

    //SET LED FLASHING
    while(1){
        led_on();
        delay(500);
        
        led_off();
        delay(500);
    }
    return 0;
}

代码没什么好说的,都是比较基础的东西。

编译

编译其实就是写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 : %.c
	arm-linux-gnueabihf-gcc -Wall -nostdlib -c -o $@ $<

%.o : %.S
	arm-linux-gnueabihf-gcc -Wall -nostdlib -c -o $@ $<

clean:
	rm -rf *.o ledc.bin ledc.elf ledc.dis

我们需要的是led.bin。
这里makefile是添加了反汇编的,用于debug。主要是检查看汇编语言和我们的C语言有没有功能上的差异或者地址上有问题。

烧录

得到led.bin之后,就需要通过下载工具下载到介质上(目前用SD卡)。下载工具做的事情可以参考启动流程这篇文章里的相关内容,这里不再重复。

  • 11
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值