✅作者简介:嵌入式入坑者,与大家一起加油,希望文章能够帮助各位!!!!
📃个人主页:@rivencode的个人主页
🔥系列专栏:玩转STM32
💬推荐一款模拟面试、刷题神器,从基础到大厂面试题👉点击跳转刷题网站进行注册学习
目录
1.什么是时钟对单片机有什么作用
时钟是由电路产生的具有周期性的脉冲信号,相当于单片机的心脏,要想使用单片机的外设必须开启相应的时钟,驱动外设的本质是操作寄存器,而寄存器是由D触发器构成,而触发器需要时钟才能改写值,所以要想操作寄存器必须开启对应外设的时钟。对CPU来说假设CPU在一个时钟周期内执行一条指令(二进制代码),若时钟频率越高,而时钟等于1/f为频率的倒数,则时钟周期更短则在相同的时间CPU能够执行更多的指,CPU的运行速度更快,文章的后面会有STM32超频实验,让灯闪烁的更快。(后面会详细讲解实验原理)
2.为什么要有时钟树
STM32时钟系统主要的目的就是给相对独立的外设模块提供时钟,主要也是为了降低整个芯片的功耗,所有外设时钟默认都是关闭状态(disable)当我们使用某个外设就要开启这个外设的时钟(enable),不同外设需要的时钟频率不同,没必要所有外设都用高速时钟造成浪费,而且有些外设也接受不了这么高的频率,这也是为什么STM32有四个时钟源的原因,就是兼容不同速度的外设,STM32的四个时钟源分别为:HSE、 LSE、HSI、LSI

先粗略看一下时钟图,接下来就是分时钟源一 一详细讲解(重点),这里以原子的STM32f10RCT6mini板子为例



1.作用

2.控制



1.作用

2.控制

1.作用
这个图是重中之重

1.来源:
2.控制

通常的配置是SYSCLK=PLLCLK=72M,配置流程,这里也非常重要后面实验就按照这个过程写代码

1.作用:

2.控制

1.作用

2.控制

1.作用

2.控制

独立看门狗时钟:IWDGCLK,由LSI提供

控制


监视高速外部时钟HSE的工作状态。若HSE失效,会自动切换高速内部时钟HSI作为系统时钟的输入,保证系统的正常运行。一旦CSS被激活,并且HSE时钟出现故障,CSS中断就产生,并且NMI也自动产生。NMI将被不
断执行,直到CSS中断挂起位被清除。
引脚,由PA8复用所得。可以把时钟信号输出供外部使用,也可以用示波器检测时钟信号的参数(峰峰值,频率…)

STM32单片机上电,就会执行启动文件(汇编代码编写)中的复位程序。

执行复位程序
1.调用SystemInit系统初始化函数完成系统时钟的配置(配置成72MHZ)
2.调用_main函数初始化堆栈指针,然后再调用C库函数main函数,去到C语言的世界
所以跳转到C语言的main函数时,已经完成了系统时钟(SYSCLK)的配置。

以芯片型号: STM32F10X_HD为例

这里将系统时钟配置成官方推荐的72MHZ。
步骤:
由外部晶振(HSE)提供的8MHz通过PLLXTPRE分频器不分频,输入到PLLSRC,然后通过PLLMUL锁相环进行9倍频后输出PLLCLK时钟(72MHZ),然后通过系统时钟切换SW 将PLLCLK作为系统时钟(72MHZ)。然后通过AHB预分频器进行1分频得到AHB总线时钟(HCLK),然后再分别经过APB1,APB2预分频器分别2分频,1分频,给APB1(PCLK1)APB2(PCLK2)总线提供时钟,进而给分别挂载在APB1,APB2总线的外设提供时钟
配合图来理解:



还有对条件编译不熟的一定要看->预处理指令不然怎么调用的函数都不知道
重点来了
static void SetSysClockTo72(void)
{
__IO uint32_t StartUpCounter = 0, HSEStatus = 0;
/* SYSCLK, HCLK, PCLK2 and PCLK1 configuration ---------------------------*/
/* Enable HSE :使能 HSE */
RCC->CR |= ((uint32_t)RCC_CR_HSEON);
/* Wait till HSE is ready and if Time out is reached exit */
//等待HSE就绪并做超时处理
do
{
HSEStatus = RCC->CR & RCC_CR_HSERDY;
StartUpCounter++;
} while((HSEStatus == 0) && (StartUpCounter != HSE_STARTUP_TIMEOUT));
if ((RCC->CR & RCC_CR_HSERDY) != RESET)
{
HSEStatus = (uint32_t)0x01;
}
else
{
HSEStatus = (uint32_t)0x00;
}
//启动成功执行下一步的代码
if (HSEStatus == (uint32_t)0x01)
{
/*使能预取指 cpu在FLASH取代码*/
FLASH->ACR |= FLASH_ACR_PRFTBE;
/* Flash 2 wait state :设置成两个等待周期 */
FLASH->ACR &= (uint32_t)((uint32_t)~FLASH_ACR_LATENCY);
FLASH->ACR |= (uint32_t)FLASH_ACR_LATENCY_2;
/* HCLK = SYSCLK =72M */
RCC->CFGR |= (uint32_t)RCC_CFGR_HPRE_DIV1;
/* PCLK2 = HCLK =72M */
RCC->CFGR |= (uint32_t)RCC_CFGR_PPRE2_DIV1;
/* PCLK1 = HCLK =36M */
RCC->CFGR |= (uint32_t)RCC_CFGR_PPRE1_DIV2; //二分频
/* 锁相环配置: PLLCLK = HSE * 9 = 72 MHz */
RCC->CFGR &= (uint32_t)((uint32_t)~(RCC_CFGR_PLLSRC | RCC_CFGR_PLLXTPRE |
RCC_CFGR_PLLMULL));
RCC->CFGR |= (uint32_t)(RCC_CFGR_PLLSRC_HSE | RCC_CFGR_PLLMULL9);
/* Enable PLL :使能PLL*/
RCC->CR |= RCC_CR_PLLON;
/* Wait till PLL is ready :等待PLL稳定*/
while((RCC->CR & RCC_CR_PLLRDY) == 0)
{
}
/* Select PLL as system clock source :选择PLL作为系统时钟*/
RCC->CFGR &= (uint32_t)((uint32_t)~(RCC_CFGR_SW));
RCC->CFGR |= (uint32_t)RCC_CFGR_SW_PLL;
/* Wait till PLL is used as system clock source :等待PLLCLK切换为系统时钟*/
while ((RCC->CFGR & (uint32_t)RCC_CFGR_SWS) != (uint32_t)0x08)
{
}
}
else
{ /* If HSE fails to start-up, the application will have wrong clock
configuration. User can add here some code to deal with this error */
//如果HSE 启动失败,用户可以在这里添加处理错误的代码
}
}
老样子把一些if语句折叠看看函数的整体框架:

如果HSE启动成功就执行折叠的if语句的内容

这里由于空间有限实在是不好一个一个去分析寄存器操作,我自己已经一个一个宏跳转然后对应着寄存器看,建议可以一条一条代码对着手册看,看看是不是真的是操作的寄存器的相对应的位
到这里就把系统时钟配置完了,还有总线AHB,APB1,APB2的时钟也配置完成,这些总线上挂载的外设的时钟,要使用哪个外设就开启哪个外设的时钟

这些寄存器的相应位就是对应挂载在总线上的外设时钟使能位。
#include "rccclkconfig.h"
void HSE_System_Config( uint32_t RCC_PLLMul_x)
{
ErrorStatus HSEStatus;
/* 把RCC寄存器配置成复位值 */
RCC_DeInit();
/* 使能SHE */
RCC_HSEConfig(RCC_HSE_ON);
HSEStatus= RCC_WaitForHSEStartUp();
/* 判断SHE启动是否成功 */
if ( HSEStatus ==SUCCESS )
{
/*使能预取指 cpu在FLASH取代码*/
FLASH_PrefetchBufferCmd(FLASH_PrefetchBuffer_Enable);
/* 设置成两个等待周期 */
FLASH_SetLatency(FLASH_Latency_2);
/* HCLK = SYSCLK =72M */
RCC_HCLKConfig(RCC_SYSCLK_Div1);
/* PCLK2 = HCLK =72M */
RCC_PCLK1Config(RCC_HCLK_Div2);
/* PCLK1 = HCLK =36M */
RCC_PCLK2Config(RCC_HCLK_Div1);
/* 锁相环配置: PLLCLK = HSE * x(2-16) = 8*x MHz */
RCC_PLLConfig(RCC_PLLSource_HSE_Div1, RCC_PLLMul_x);
/* 使能PLL*/
RCC_PLLCmd(ENABLE);
/* 等待PLL稳定 */
while( RCC_GetFlagStatus(RCC_FLAG_PLLRDY) == RESET);
/* 选择pLL作为系统时钟 */
RCC_SYSCLKConfig(RCC_SYSCLKSource_PLLCLK);
/* 等待PLLCLK切换为系统时钟 */
while( RCC_GetSYSCLKSource() != 0x08) ;
}
else
{
/* If HSE fails to start-up, the application will have wrong clock
configuration. User can add here some code to deal with this error */
}
}
代码已有详细注释是用库函数编写的,不过基本与前面寄存器版本的一致,只不过把他们封装成了函数,这里要注意的一点是配置前一定要调用一个 RCC_DeInit();函数把RCC寄存器配置成复位值 。
原理:对CPU来说假设CPU在一个时钟周期内执行一条指令(二进制代码),若时钟频率越高,而时钟等于1/f为频率的倒数,则时钟周期更短则在相同的时间CPU能够执行更多的指,CPU的执行代码速度更快,延时的时间就会变短,灯就会闪烁的更快。
#define SOFT_DELAY Delay(0x0FFFFF);
void Delay(__IO u32 nCount);
int main(void)
{
HSE_System_Config( RCC_PLLMul_2);
/* LED 端口初始化 */
LED_GPIO_Config();
while(1)
{
LED_G(NO);
LED_R(OFF);
Delay(0x0FFFFF);
LED_G(OFF);
LED_R(NO);
Delay(0x0FFFFF);
}
}
void Delay(__IO uint32_t nCount) //简单的延时函数
{
for(; nCount != 0; nCount--);
}
这里为了有更明显的实验效果,先让PLLMUL锁相环进行2倍频后输出PLLCLK时钟(16MHZ),然后通过系统时钟切换SW将PLLCLK作为系统时钟(16MHZ)。

然后让PLLMUL锁相环进行16倍频后输出PLLCLK时钟(128MHZ),然后通过系统时钟切换SW将PLLCLK作为系统时钟(128MHZ)。

PLLMUL锁相环二倍频 SYSCLK(16MHZ)实验效果:

PLLMUL锁相环二倍频 SYSCLK(128MHZ)实验效果:

是不是直接起飞,很糊凑合看叭我不会上传视频唉
总而言之时钟树非常重要一定要掌握!!!最好可以自己写一个库函数版的系统时钟初始化函数,好啦本文到这就结束啦,如果本文对你有帮助就赶快收藏点赞叭!!!。
结束语:
最近发现一款刷题神器,如果大家想提升编程水平,玩转C语言指针,还有常见的数据结构(最重要的是链表和队列)后面嵌入式学习操作系统的时如freerots、RT-Thread等操作系统,链表与队列知识大量使用。
大家可以点击下面连接进入牛客网刷题
点击跳转进入网站(C语言方向)
点击跳转进入网站(数据结构算法方向)

电脑0x0000001A蓝屏错误怎么U盘重装系统教学分享。有用户电脑开机之后遇到了系统蓝屏的情况。系统蓝屏问题很多时候都是系统bug,只有通过重装系统来进行解决。那么蓝屏问题如何通过U盘重装新系统来解决呢?来看看以下的详细操作方法教学吧。 准备工作: 1、U盘一个(尽量使用8G以上的U盘)。 2、一台正常联网可使用的电脑。 3、ghost或ISO系统镜像文件(Win10系统下载_Win10专业版_windows10正式版下载-系统之家)。 4、在本页面下载U盘启动盘制作工具:系统之家U盘启动工具。 U盘启动盘制作步骤: 注意:制作期间,U盘会被格式化,因此U盘中的重要文件请注
在应用开发中,有时候我们需要获取系统的设备信息,用于数据上报和行为分析。那在鸿蒙系统中,我们应该怎么去获取设备的系统信息呢,比如说获取手机的系统版本号、手机的制造商、手机型号等数据。1、获取方式这里分为两种情况,一种是设备信息的获取,一种是系统信息的获取。1.1、获取设备信息获取设备信息,鸿蒙的SDK包为我们提供了DeviceInfo类,通过该类的一些静态方法,可以获取设备信息,DeviceInfo类的包路径为:ohos.system.DeviceInfo.具体的方法如下:ModifierandTypeMethodDescriptionstatic StringgetAbiList()Obt
文章目录1.开发板选择*用到的资源2.串口通信(个人理解)3.代码分析(注释比较详细)1.主函数2.串口1配置3.串口2配置以及中断函数4.注意问题5.源码链接1.开发板选择我用的是STM32F103RCT6的板子,不过代码大概在F103系列的板子上都可以运行,我试过在野火103的霸道板上也可以,主要看一下串口对应的引脚一不一样就行了,不一样的就更改一下。*用到的资源keil5软件这里用到了两个串口资源,采集数据一个,串口通信一个,板子对应引脚如下:串口1,TX:PA9,RX:PA10串口2,TX:PA2,RX:PA32.串口通信(个人理解)我就从串口采集传感器数据这个过程说一下我自己的理解,
需求:要创建虚拟机,就需要给他提供一个虚拟的磁盘,我们就在/opt目录下创建一个10G大小的raw格式的虚拟磁盘CentOS-7-x86_64.raw命令格式:qemu-imgcreate-f磁盘格式磁盘名称磁盘大小qemu-imgcreate-f磁盘格式-o?1.创建磁盘qemu-imgcreate-fraw/opt/CentOS-7-x86_64.raw10G执行效果#ls/opt/CentOS-7-x86_64.raw2.安装虚拟机使用virt-install命令,基于我们提供的系统镜像和虚拟磁盘来创建一个虚拟机,另外在创建虚拟机之前,提前打开vnc客户端,在创建虚拟机的时候,通过vnc
因为我现在正在做一些时间测量,我想知道是否可以在不使用Benchmark类或命令行实用程序time的情况下测量用户时间或系统时间。使用Time类只显示挂钟时间,而不显示系统和用户时间,但是我正在寻找具有相同灵active的解决方案,例如time=TimeUtility.now#somecodeuser,system,real=TimeUtility.now-time原因是我有点不喜欢Benchmark,因为它不能只返回数字(编辑:我错了-它可以。请参阅下面的答案。)。当然,我可以解析输出,但感觉不对。*NIX系统的time实用程序也应该可以解决我的问题,但我想知道是否已经在Ruby中实
在Ruby中,以毫秒为单位获取自纪元(1970)以来的当前系统时间的正确方法是什么?我试过了Time.now.to_i,好像不是我想要的结果。我需要结果显示毫秒并且使用long类型,而不是float或double。 最佳答案 (Time.now.to_f*1000).to_iTime.now.to_f显示包含十进制数字的时间。要获得毫秒数,只需将时间乘以1000。 关于ruby-以毫秒为单位获取当前系统时间,我们在StackOverflow上找到一个类似的问题:
关闭。这个问题需要更多focused.它目前不接受答案。想改进这个问题吗?更新问题,使其只关注一个问题editingthispost.关闭8年前。Improvethisquestion我们有以下(以及更多)系统,我们将数据从一个应用推送/拉取到另一个:托管CRM(InsideSales.com)Asterisk电话系统(内部)横幅广告系统(openx,我们托管)潜在客户生成系统(自行开发)电子商务商店(spree,我们托管)工作板(本土)一些工作网站抓取+入站工作提要电子邮件传送系统(如Mailchimp,自主开发)事件管理系统(如eventbrite,自主开发)仪表板系统(大量图表和
我正在尝试找出一种方法来显示来自不在RAILS_ROOT下(在RedHat或Ubuntu环境中)的已安装文件系统的图像。我不想使用符号链接(symboliclink),因为这个应用程序实际上是通过Tomcat部署的,而当我关闭Tomcat时,Tomcat会尝试跟随符号链接(symboliclink)并删除挂载中的所有图像。由于这些文件的数量和大小,将图像放在public/images下也不是一种选择。我查看了send_file,但它只会显示一张图片。我需要在一个格式良好的页面中显示6个请求的图像。由于膨胀,我宁愿不使用Base64编码,但我不知道如何将图像数据与呈现的页面一起传递下去。
当您在Ruby脚本中使用系统调用时,您可以像这样获得该命令的输出:output=`ls`putsoutput这就是thisquestion是关于。但是有没有办法显示系统调用的连续输出?例如,如果您运行此安全复制命令,以通过SSH从服务器获取文件:scpuser@someserver:remoteFile/some/local/folder/...它显示随着下载进度的连续输出。但是这个:output=`scpuser@someserver:remoteFile/some/local/folder/`putsoutput...不捕获该输出。如何从我的Ruby脚本中显示正在进行的下载进度?
下载微PE工具箱进入官网下载微PE工具箱-下载 安装好后,打开微PE工具箱客户端,选择安装PE到U盘 PE壁纸可选择自己喜欢的壁纸,勾选上包含DOS工具箱,个性化盘符图标 下载原版系统进入网站下载镜像NEXT,ITELLYOU如果没有账号,注册一下就好进入选择开始使用选择win10 这里我们选择消费者版,用迅雷把BT种子下载下来 下面的两个盘符,是PE工具箱安装进U盘后,分成的盘符,注意EFI的盘符,这里面不能删东西,也不能添东西,另一个盘符可以当做正常的U盘空间使用,我们现在需要把下载下来的景象文件复制到正常的U盘空间中去 这个时候我们的系统U盘就只做好了 安装系统我们将U盘插入电脑,开机,