OpenEdv-开源电子网

 找回密码
 立即注册
正点原子全套STM32/Linux/FPGA开发资料,上千讲STM32视频教程免费下载...
查看: 178|回复: 0

M4内核系统架构的资源分享

[复制链接]

36

主题

36

帖子

0

精华

初级会员

Rank: 2

积分
147
金钱
147
注册时间
2024-12-9
在线时间
8 小时
发表于 5 天前 | 显示全部楼层 |阅读模式
前言
本应用入门指南主要介绍了AT32 M4内核系统架构,并针对M4内核自带的位带、硬件浮点运算单元和滴答时钟中断功能进行基础讲解和案例解析。
备注:本文档仅供有需求的小伙们参考,如需更详细资源,可访问雅特力官网获取。
1                 AT32内核架构概述
AT32F4系列产品是基于CortexTM-M4F处理器架构,该处理器是一款低功耗处理器,具有低门数,低中断延迟和低成本调试的特点。支持包括DSP指令集与浮点运算功能,特别适合用于深度嵌入式应用程序需要快速中断响应功能。CortexTM-M4F处理器是基于ARMv7-M架构,既支持Thumb指令集也支持DSP指令集。

下图为CortexTM-M4F处理器的内部框图,请参阅《ARM®Cortex-M4技术参考手册》了解关于CortexTM-M4F更详尽信息。

本文主要就M4内核自带的位带、硬件浮点运算单元和滴答时钟中断功能进行基础讲解。
2                  案例  位带操作2.1  功能简介
利用位带操作,可以使用普通的加载/存储操作来对单一比特进行读写访问。在Cortex™-M4F中提供了两个位带区:SRAM最低1M字节空间和外设区间的最低1M字节空间。这两个区中的地址除了可以像普通存储器一样访问外,还可以通过它们各自的位带别名区来快捷访问这两个区中任意地址的任意比特位,位带别名区将位带区每个比特膨胀成一个32位的字。当你访问位带别名区的一个地址时,等同于直接访问位带区的一个比特位。

位带区:支持位带操作的地址区
位带别名区:对别名区地址的访问最终作用到位带区的访问上
在位带区中,每个比特都映射到别名地址区的一个字(这是只有LSB有效的字)。当一个位带别名区地址被访问时,会先把该地址变换成位带区地址。对于读操作,读取位带区地址中的一个字,再把需要的位右移到LSB,并把LSB返回。对于写操作,把需要写的位左移到对应的位序号处,然后执行一个比特级的“读-改-写”过程。
支持位带操作的两个内存区的地址范围为:
SRAM区中的最低1M字节:0x2000_0000~0x200F_FFFF
外设区间的最低1M字节:  0x4000_0000~0x400F_FFFF
对于SRAM位带区的某个比特,如果所在字节地址为A,位序号为n(0<=n<=7),则该比特在别名区的地址为:
AliasAddr = 0x2200_0000+(A-0x2000_0000)*32+n*4
对于外设区间位带区的某个比特,如果所在字节地址为A,位序号为n(0<=n<=7),则该比特在别名区的地址为:
AliasAddr = 0x4200_0000+(A-0x4000_0000)*32+n*4
对于SRAM区中,位带区与位带别名区的映射如下表所示:
表1. SRAM区中的位带地址映射
位带区
等效别名区地址
0x2000_0000.0
0x2200_0000.0
0x2000_0000.1
0x2200_0004.0
0x2000_0000.2
0x2200_0008.0
0x2000_0000.31
0x2200_007C.0
0x2000_0004.0
0x2200_0080.0
0x2000_0004.1
0x2200_0084.0
0x2000_0004.2
0x2200_0088.0
0x200F_FFFC.31
0x23FF_FFFC.0
对于外设区中,位带区与位带别名区的映射如下表所示:
表2.外设区中的位带地址映射
     
位带区
   
等效别名区地址
  
   
0x4000_0000.0
  
0x4200_0000.0

   
0x4000_0000.1
  
0x4200_0004.0

   
0x4000_0000.2
  
0x4200_0008.0

   
  

   
0x4000_0000.31
  
0x4200_007C.0

   
0x4000_0004.0
  
0x4200_0080.0

   
0x4000_0004.1
  
0x4200_0084.0

   
0x4000_0004.2
  
0x4200_0088.0

   
  

   
0x400F_FFFC.31
  
0x43FF_FFFC.0
位带操作的优越性最容易想到的是通过GPIO的管脚来单独控制每盏LED的点亮与熄灭。另一方面,也对操作串行接口提供很大的方便。总之,位带操作对于硬件I/O密集型的底层程序最有用处。
位带操作还能简化跳转的判断。当跳转依据是某个位时,以前必须这样做:
读取整个寄存器
屏蔽不需要的位
比较并跳转
现在只需要:
从位带别名区读取该位的状态
比较并跳转

使代码更简洁,这只是位带操作优越性的初步体现,位带操作还有一个重要的好处是在多任务以及多任务环境中,将以前的读-改-写需要的三条指令,做成了一个硬件级别支持的原子操作,消除了以前读-改-写可能被中断,导致出现紊乱的情况。
2.2            注意事项
1)     因各系列的外设IP地址排布的不同,AT32F421xx与AT32F425xx系列的GPIO外设基地址不在位带映射地址范围内。
2.3            资源准备
1)     硬件环境:
对应产品型号的AT-START BOARD
2)     软件环境
project\at_start_f4xx\examples\cortex_m4\bit_band
2.4            软件设计
1)     配置流程
SRAM位带操作
n 定义全局变量variables = 0xA5A5A5A5,
n 对variables  bit0的位带地址写0
n 检查variables 是否修改为0xA5A5A5A4,如果是则表示操作成功
n 对variables  bit0的位带地址写1
n 检查variables 是否修改为0xA5A5A5A5,如果是则表示操作成功
n 对variables  bit16的位带地址写0
n 检查variables 是否修改为0xA5A4A5A5,如果是则表示操作成功
n 对variables  bit16的位带地址写1
n 检查variables 是否修改为0xA5A5A5A5,如果是则表示操作成功
n 对variables  bit31的位带地址写0
n 检查variables 是否修改为0x25A5A5A5,如果是则表示操作成功
n 对variables  bit31的位带地址写1
n 检查variables 是否修改为0xA5A5A5A5,如果是则表示操作成功
外设位带操作
n 对LED2对应GPIO ODT寄存器bit位的位带地址写0,
n 对LED2对应GPIO ODT寄存器bit位的位带地址写1
n 循环执行上述操作,实现LED toggle功能
2)     代码介绍
main函数代码描述
  
int main(void)
  
{
  
  /* 系统时钟配置 */
  
   system_clock_config();
  
  /*  LED延时函数等初始化 */
  
   at32_board_init();
  
/*  初始化变量 */
  
   variables = 0xA5A5A5A5;
  
  /* 获取变量地址 */
  
   variables_addr = (uint32_t)&variables;
  
  
  
  /* 变量variables  bit0 位带写0并检查操作结果 */
  
   VARIABLES_RESET_BIT(variables_addr, 0);
  
   if((variables != 0xA5A5A5A4) || (VARIABLES_GET_BIT(variables_addr, 0)  != 0))
  
  {
  
     result_error();
  
  }
  
  
  
  /* 变量variables bit0 位带写1并检查操作结果 */
  
   VARIABLES_SET_BIT(variables_addr, 0);
  
   if((variables != 0xA5A5A5A5) || (VARIABLES_GET_BIT(variables_addr, 0)  != 1))
  
  {
  
     result_error();
  
  }
  
  
  
  /* 变量variables  bit16 位带写0并检查操作结果 */
  
   VARIABLES_RESET_BIT(variables_addr, 16);
  
   if((variables != 0xA5A4A5A5) || (VARIABLES_GET_BIT(variables_addr, 16)  != 0))
  
  {
  
     result_error();
  
  }
  
  
  /* 变量variables  bit16 位带写1并检查操作结果 */
  
   VARIABLES_SET_BIT(variables_addr, 16);
  
   if((variables != 0xA5A5A5A5) || (VARIABLES_GET_BIT(variables_addr, 16)  != 1))
  
  {
  
     result_error();
  
  }
  
  
  
  /* 变量variables  bit31 位带写0并检查操作结果 */
  
   VARIABLES_RESET_BIT(variables_addr, 31);
  
   if((variables != 0x25A5A5A5) || (VARIABLES_GET_BIT(variables_addr, 31)  != 0))
  
  {
  
     result_error();
  
  }
  
  
  /* 变量variables  bit31 位带写1并检查操作结果 */
  
   VARIABLES_SET_BIT(variables_addr, 31);
  
   if((variables != 0xA5A5A5A5) || (VARIABLES_GET_BIT(variables_addr, 31)  != 1))
  
  {
  
     result_error();
  
  }
  
  
  
  
  
   for(;;)
  
  {
  
    /* 外设地址的位带操作,实现LED2 toggle */
  
     PERIPHERAL_RESET_BIT((uint32_t)&LED2_GPIO->odt, 13);
  
     delay_ms(500);
  
     PERIPHERAL_SET_BIT((uint32_t)&LED2_GPIO->odt, 13);
  
     delay_ms(500);   
  
  }
  
}
  
宏定义内容描述
  
/*  定义SRAM及其位带地址 */
  
#define RAM_BASE            0x20000000
  
#define RAM_BITBAND_BASE    0x22000000
  
  
/*  定义SRAM位带写0函数 */
  
#define  VARIABLES_RESET_BIT(variables_addr, bit_number)    \
  
         (*(uint32_t *)(RAM_BITBAND_BASE + ((variables_addr - RAM_BASE) * 32) +  ((bit_number) * 4)) = 0)
  
  
/*  定义SRAM位带写1函数 */
  
#define VARIABLES_SET_BIT(variables_addr,  bit_number)       \
  
         (*(uint32_t *)(RAM_BITBAND_BASE + ((variables_addr - RAM_BASE) * 32) +  ((bit_number) * 4)) = 1)
  
  
/*  定义SRAM位带值获取函数 */
  
#define VARIABLES_GET_BIT(variables_addr,  bit_number)       \
  
         (*(uint32_t *)(RAM_BITBAND_BASE + ((variables_addr - RAM_BASE) * 32) +  ((bit_number) * 4)))
  
  
/*  定义外设及其位带地址 */
  
#define PERIPHERAL_BASE            0x40000000
  
#define PERIPHERAL_BITBAND_BASE    0x42000000
  
  
/*  定义外设位带写0函数 */
  
#define  PERIPHERAL_RESET_BIT(peripheral_addr, bit_number)    \
  
         (*(uint32_t *)(PERIPHERAL_BITBAND_BASE + ((peripheral_addr -  PERIPHERAL_BASE) * 32) + ((bit_number) * 4)) = 0)
  
   
  
/*  定义外设位带写1函数 */
  
#define  PERIPHERAL_SET_BIT(peripheral_addr, bit_number)       \
  
         (*(uint32_t *)(PERIPHERAL_BITBAND_BASE + ((peripheral_addr -  PERIPHERAL_BASE) * 32) + ((bit_number) * 4)) = 1)
  
2.5   实验效果
n  SRAM位带操作:如果不满足预期,LED4翻转。

n  外设位带操作:如果满足预期,LED2翻转。

3                  案例  硬件浮点运算单元3.1  功能简介
FPU即浮点运算单元(Float Point Unit)。浮点运算,对于定点CPU(没有FPU的CPU)来说必须要按照IEEE-754标准的算法来完成运算,是相当耗费时间的。而对于有FPU的CPU来说,浮点运算则只是几条指令的事情,速度相当快。
AT32F4属于Cortex M4F架构,带有32位单精度硬件FPU,支持浮点指令集,相对于Cortex M0和Cortex M3等,高出数十倍甚至上百倍的运算性能
3.2           注意事项
1)     由各系列应用方向及成本的综合考虑,AT32F415xx、AT32F421xx和AT32F425xx系列不支持硬件浮点运算单元。
3.3           资源准备
1)     硬件环境:
对应产品型号的AT-START BOARD

2)     软件环境
project\at_start_f4xx\examples\cortex_m4\fpu
3.4           软件设计
1)     配置流程
FPU功能的开启必须要编译器和代码都开启才可以。若只开启编译器FPU,程序会进入hardfault;若只开启代码中FPU,编译器不会编译出FPU的代码指令。
n 编译器上开启FPU功能

IAR开启FPU方式如下图
MDK开启FPU方式如下图
n 代码中开启FPU功能
在system_at32f4xx.c文件中void SystemInit (void)函数确保有如下粗斜体代码
  
void SystemInit (void)
  
{
  
#if defined  (__FPU_USED) && (__FPU_USED == 1U)
  
  SCB->CPACR |= ((3U << 10U * 2U)  |         /* set cp10 full access */
  
                 (3U << 11U * 2U)  );        /* set cp11 full access */
  
#endif
  
  
  /* reset the crm clock configuration to the  default reset state(for debug purpose) */
  
  /* set hicken bit */
  
  CRM->ctrl_bit.hicken = TRUE;
  
  …
  
}
  
n  执行Julia算法函数
比较开启和不开启FPU功能的Julia运算速度。
2)     代码介绍
main函数代码描述
  
int main(void)
  
{
  
/* 系统时钟配置 */
  
   system_clock_config();
  
  /*  LED延时函数等初始化 */
  
   at32_board_init();
  
   for(;;)
  
  {
  
    /*  LED4翻转  */
  
at32_LED_toggle(LED4);
  
    /* 执行julia算法函数 */
  
     generate_julia_fpu(SCREEN_X_SIZE, SCREEN_Y_SIZE, SCREEN_X_SIZE / 2,  SCREEN_Y_SIZE / 2, ZOOM, buffer);   
  
  }
  
}  
  

3.5   实验效果
n  编译器上开启FPU功能,观察LED4翻转速度。
n  编译器上关闭FPU功能,观察LED4翻转速度。
n  对比以上两种情形LED4翻转速度区别
4                  案例  系统滴答时钟中断4.1  功能简介
系统嘀嗒定时器是一个24位递减计数器,递减至零可自动重载计数初值。可产生周期性异常,用作嵌入式操作系统的多任务调度计数器,或对于无嵌入式操作系统,可用于调用需周期性执行的任务。系统嘀嗒定时器校准值固定值9000,当系统嘀嗒时钟设定为9MHz,产生1ms时间基准。
4.2           资源准备
1)     硬件环境:
对应产品型号的AT-START BOARD
2)     软件环境

project\at_start_f4xx\examples\cortex_m4\systick_interrupt
4.3            软件设计
1)     配置流程
n 配置systick 时钟源
n 配置 systick重载值并开启systick中断
n 在void SysTick_Handler(void) 函数中添加应用代码
2)     代码介绍
main函数代码描述
  
int main(void)
  
{
  
  /* 系统时钟配置 */
  
   system_clock_config();
  
  
/* 配置systick 时钟源
  
SYSTICK_CLOCK_SOURCE_AHBCLK_NODIV  表示systick时钟来源于AHB不分频
  
SYSTICK_CLOCK_SOURCE_AHBCLK_DIV8  表示systick时钟来源于AHB 8分频  */
  
   systick_clock_source_config(SYSTICK_CLOCK_SOURCE_AHBCLK_NODIV);
  
  
  /* 配置  systick重载值并开启systick中断 */
  
   SysTick_Config(MS_TICK);
  
  
  /* 初始化LED2  */
  
   at32_led_init(LED2);
  
  
   for(;;)
  
  {
  
  }
  
}
  
4.4   实验效果
n  本应配置的是1 ms systick中断,每进200次systick中断LED2翻转一次,因此应该观察到的现象是LED2以200 ms一次的频率进行翻转。


回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则



关闭

原子哥极力推荐上一条 /1 下一条

正点原子公众号

如发现本坛存在违规或侵权内容, 请点击这里发送邮件举报 (或致电020-38271790)。请提供侵权说明和联系方式。我们将及时审核依法处理,感谢配合。

QQ|手机版|OpenEdv-开源电子网 ( 粤ICP备12000418号-1 )

GMT+8, 2026-4-4 02:45

Powered by OpenEdv-开源电子网

© 2001-2030 OpenEdv-开源电子网

快速回复 返回顶部 返回列表