02-单片机入门_点灯

02-单片机入门_点灯

通过汇编代码点亮LED

先看看原理图-看点灯需要操作哪一些GPIO端口

在原理图中有三个LED , 分别是 nLED_1 , nLED_2 , nLED_4

同时nLED_1中的n,表示低电平有效
在这里插入图片描述

这三个LED分配连接的GPIO端口是:GPF4 , GPF5, GPF6
在这里插入图片描述

看用户手册,这3个GPIO端口怎么操作

在这里插入图片描述

配置GPF4 , GPF5 , GPF6 为输出端口

配置GPFDAT 的 4, 5 ,6 位为0(nLED_1低电平有效)

编写代码

led_on.S

.text
.global _start

_start:

/* 配置GPF4,GPF5,GPF6为输出引脚
 * 把=0x1500写到地址0x56000050
 */
	ldr r1, =0x56000050
	ldr r0, =0x1500	/* mov r0, #0x1500 */
	str r0, [r1]


/* 设置GPF4,GPF5,GPF6输出低电平 
 * 把0写到地址0x56000054
 */
	ldr r1, =0x56000054
	ldr r0, =0	/* mov r0, #0 */
	str r0, [r1]

	/* 死循环 */
halt:
	b halt

Makefile

all:
	arm-linux-gcc -c -o led_on.o led_on.S
	arm-linux-ld -Ttext 0 led_on.o -o led_on.elf
	arm-linux-objcopy -O binary -S led_on.elf led_on.bin
	arm-linux-objdump -D led_on.elf > led_on.dis
clean:
	rm *.bin *.o *.elf

进行编译,生成led_on.bin

通过openjtag刷入nand flash,将开发板启动项拨向Nand flash,打开电源,3个LED点亮

在这里插入图片描述

再通过openjtag将led_on.bin刷入nor flash,以nor flash启动,也是正常点亮

在这里插入图片描述

通过C语言代码点亮LED

C语言的函数需要用到栈传递参数(栈就是一片内存空间),这里就需要看看jz2440在单片机模式下的内存布局

先讲讲nor flash 和 nand flash 在非指令模式下的简单区别
nor flashnand flash
可存储代码,像硬盘一样可存储代码,像硬盘一样
可执行代码,但无法修改norFlash的数据(可读,可执行)不能执行代码,需要把代码复制到SRAM(SRAM像内存条一样)执行

在上表中,包含了3个硬件:

  • Nor flash:
    • 1)可以像固态硬盘一样存储数据,
    • 2)从nor flash启动之后,nor flash也像包含代码的内存空间,可以执行代码,
    • 3)由于nor flash本身是存储代码的,为了让其上的数据不被破坏,在其内存空间上无法进行写操作,
    • 4)JZ2440上nor flash的大小为2M
  • Nand flash:
    • 1)可以单纯的理解为PC上的固态硬盘,
    • 2)在JZ440上Nand flash大小为256M
  • SRAM :
    • 1)片内内存(可读,可写,可执行),
    • 2)在JZ2440上SRAM大小为4K

在JZ2440中,也有像PC上内存条一样的硬件,叫做SDRAM,但是需要初始化,暂时就当这个硬件不存在

再说到C语言函数的栈,是需要可被修改的(能被进行写操作)

1)在之前的点亮3个LED的例子中,有nor启动和nand启动

nor启动:

  • Nor Flash的基地址为0,片内RAM地址为0x4000 0000;
  • CPU读出Nor上第1个指令(前4字节),执行;
  • CPU继续读出其它指令执行。

nand启动:

  • 片内4k RAM基地址为0,Nor Flash不可访问;
  • 2440硬件把Nand前4K内容复制到片内的RAM,然后CPU从0地址取出第1条指令执行

在这里插入图片描述

2)当前需要编写C语言的函数,需要栈,而栈是往下增长的,所以nor flash启动和nand flash启动,栈的设置如下

在这里插入图片描述

3)怎么判断是Nor 启动还是Nand Flash启动

从Nor Flash 和 Nand Flash启动的区别是啥:

  • 从Nor Flash启动:0地址在Nor Flash的地址空间上,0地址不可修改
  • 从Nand Flash启动: 0地址在SRAM的地址空间上,0地址可被修改

通过判断0地址是否被修改,来判断是从nor启动还是nand启动

4)关闭看门狗

2440里面有个看门狗定时器,开发板上电后,需要在一定时间内“喂狗”(设置相应的寄存器),否则就会重启开发。这里直接关闭看门狗

编写如下

start.S

.text
.global _start

_start:

	/* 关闭看门狗 */
	ldr r0, =0x53000000
	ldr r1, =0
	str r1, [r0]

	/* 设置内存: sp 栈 */
	/* 分辨是nor/nand启动
	 * 写0到0地址, 再读出来
	 * 如果得到0, 表示0地址上的内容被修改了, 它对应ram, 这就是nand启动
	 * 否则就是nor启动
	 */
	mov r1, #0
	ldr r0, [r1] /* 读出原来的值备份 */
	str r1, [r1] /* 0->[0] */ 
	ldr r2, [r1] /* r2=[0] */
	cmp r1, r2   /* r1==r2? 如果相等表示是NAND启动 */
	ldr sp, =0x40000000+4096 /* 先假设是nor启动 */
	moveq sp, #4096  /* nand启动 */
	streq r0, [r1]   /* 恢复原来的值 */
	

	bl main

halt:
	b halt

led.c

void delay(volatile int d)
{
	while (d--);
}

int main(void)
{
	volatile unsigned int *pGPFCON = (volatile unsigned int *)0x56000050;
	volatile unsigned int *pGPFDAT = (volatile unsigned int *)0x56000054;
	int val = 0;  /* val: 0b000, 0b111 */
	int tmp;

	/* 设置GPFCON让GPF4/5/6配置为输出引脚 */
	*pGPFCON &= ~((3<<8) | (3<<10) | (3<<12));
	*pGPFCON |=  ((1<<8) | (1<<10) | (1<<12));

	/* 循环点亮 */
	while (1)
	{
		tmp = ~val;
		tmp &= 7;
		*pGPFDAT &= ~(7<<4);
		*pGPFDAT |= (tmp<<4);
		delay(100000);
		val++;
		if (val == 8)
			val =0;
		
	}

	return 0;
}

Makefile

all:
	arm-linux-gcc -c -o led.o led.c
	arm-linux-gcc -c -o start.o start.S
	arm-linux-ld -Ttext 0 start.o led.o -o led.elf
	arm-linux-objcopy -O binary -S led.elf led.bin
	arm-linux-objdump -D led.elf > led.dis
clean:
	rm *.bin *.o *.elf *.dis
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值