LibOpenCM3(五) 基础功能: 系统时钟, GPIO, 定时器

目录

  • LibOpenCM3(一) Linux下命令行开发环境配置
  • LibOpenCM3(二) 项目模板 Makefile分析
  • LibOpenCM3(三) .ld文件(连接器脚本)和startup代码说明
  • LibOpenCM3(四) VSCode IDE 环境配置
  • LibOpenCM3(五) 基础功能: 系统时钟, GPIO, 定时器

LibOpenCM3 时钟, RCC

LibOpenCM3 提供了快捷方法用于初始化系统时钟

旧版本的系统时脉初始化

rcc_clock_setup_in_hse_8mhz_out_72mhz();

方法所在文件lib/stm32/f1/rcc.c, 对应板载晶振为8MHz, 需要设置为72MHz系统频率的应用. 这个文件下还提供了其它的快捷方法

// 使用内部RC产生64MHz
void rcc_clock_setup_in_hsi_out_64mhz(void)
// 使用内部RC产生48MHz
void rcc_clock_setup_in_hsi_out_48mhz(void)
// 使用内部RC产生24MHz
void rcc_clock_setup_in_hsi_out_24mhz(void)
// 以下是使用外部晶振的方法
void rcc_clock_setup_in_hse_8mhz_out_24mhz(void)
void rcc_clock_setup_in_hse_8mhz_out_72mhz(void)
void rcc_clock_setup_in_hse_12mhz_out_72mhz(void)
void rcc_clock_setup_in_hse_16mhz_out_72mhz(void)
void rcc_clock_setup_in_hse_25mhz_out_72mhz(void)

使用时直接调用就可以了, 例如

int main(void)
{
    rcc_clock_setup_in_hse_8mhz_out_72mhz();
    gpio_setup();
    tim_setup();
    while (1)
    {
      //...
    }
    return 0;
}

非常方便

新版本的系统时脉初始化

在最新的版本中, 原来的方法还能调用, 但是已经被标为Deprecated了, 编译会产生warning提示. 需要改为下面的调用方式

rcc_clock_setup_pll(&rcc_hse_configs[RCC_CLOCK_HSE8_72MHZ]);

定义的文件位置没变, 对应的频率放到了参数里

enum rcc_clock_hsi {
	RCC_CLOCK_HSI_24MHZ,
	RCC_CLOCK_HSI_48MHZ,
	RCC_CLOCK_HSI_64MHZ,
	RCC_CLOCK_HSI_END
};

enum rcc_clock_hse {
	RCC_CLOCK_HSE12_72MHZ,
	RCC_CLOCK_HSE16_72MHZ,
	RCC_CLOCK_HSE25_72MHZ,
	RCC_CLOCK_HSE8_24MHZ,
	RCC_CLOCK_HSE8_72MHZ,
	RCC_CLOCK_HSE_END
};

LibOpenCM3 GPIO

GPIO的设置与SPL/HAL流程是一样的, 只是函数名和常量名有些区别.

开启GPIO外设时钟

rcc_periph_clock_enable(RCC_GPIOC);

设置GPIO模式

gpio_mode_setup(GPIOC, GPIO_MODE_OUTPUT, GPIO_PUPD_NONE, GPIO8);
gpio_mode_setup(GPIOC, GPIO_MODE_OUTPUT, GPIO_PUPD_NONE, GPIO9);
// 可以简化为
gpio_mode_setup(GPIOC, GPIO_MODE_OUTPUT, GPIO_PUPD_NONE, GPIO8 | GPIO9);

输入输出模式选项

  • GPIO_MODE_INPUT (default) Digital input
  • GPIO_MODE_OUTPUT Digital output
  • GPIO_MODE_AF Alternate Function (requires defining which alternate function desired)
  • GPIO_MODE_ANALOG Analog (for use with ADC or DAC capable GPIO)

上下拉电阻选项

  • GPIO_PUPD_NONE (default) No internal pull-up or pull-down resistor
  • GPIO_PUPD_PULLUP Internal pull-up resistor
  • GPIO_PUPD_PULLDOWN Internal pull-down resistor

输出模式设置

gpio_set_output_options(GPIOC, GPIO_OTYPE_PP, GPIO_OSPEED_LOW, GPIO8);
gpio_set_output_options(GPIOC, GPIO_OTYPE_PP, GPIO_OSPEED_LOW, GPIO9);
// 可以简化为
gpio_set_output_options(GPIOC, GPIO_OTYPE_PP, GPIO_OSPEED_LOW, GPIO8 | GPIO9);

输出类型

  • GPIO_OTYPER_PP (default) Push-pull “totem pole” output
  • GPIO_OTYPER_OD Open-drain output

输出速度

  • GPIO_OSPEED_HIGH High output speed
  • GPIO_OSPEED_MED Medium output speed
  • GPIO_OSPEED_LOW (default) Low output speed
  • GPIO_OSPEED_100MHZ Up to 100MHz output speed (equivalent to high)
  • GPIO_OSPEED_50MHZ Up to 50MHz output speed
  • GPIO_OSPEED_25MHZ Up to 25MHz output speed (equivalent to medium)
  • GPIO_OSPEED_2MHZ Up to 2MHz output speed (equivalent to low)

设置状态

GPIO 没有开启或者关闭的操作, 开启时钟后GPIO就处于工作状态, 这时候可以设置输出电平

gpio_set(GPIOC, GPIO8);
gpio_set(GPIOC, GPIO9);
// 可以简化为
gpio_set(GPIOC, GPIO8 | GPIO9);

LibOpenCM3 定时器

不同MCU型号, 能使用的定时器编号不一样, 需要根据手册确定, 下面以TIM2为例说明定时器的设置流程

开启TIM2外设时钟

开启TIM2时钟

/* Enable TIM2 clock. */
rcc_periph_clock_enable(RCC_TIM2);

如果需要初始化(重置)

/* Reset TIM2 peripheral to defaults. */
rcc_periph_reset_pulse(RST_TIM2);

设置定时器工作模式

timer_set_mode(TIM2, TIM_CR1_CKD_CK_INT, TIM_CR1_CMS_EDGE, TIM_CR1_DIR_UP);

参数一: 分频系数

/* CKD[1:0]: Clock division */
#define TIM_CR1_CKD_CK_INT		(0x0 << 8)
#define TIM_CR1_CKD_CK_INT_MUL_2	(0x1 << 8)
#define TIM_CR1_CKD_CK_INT_MUL_4	(0x2 << 8)
#define TIM_CR1_CKD_CK_INT_MASK		(0x3 << 8)

参数二: 对齐方式

/* CMS[1:0]: Center-aligned mode selection */
/****************************************************************************/
/** @defgroup tim_x_cr1_cms TIMx_CR1 CMS[1:0]: Center-aligned Mode Selection
@{*/
#define TIM_CR1_CMS_EDGE		(0x0 << 5)
#define TIM_CR1_CMS_CENTER_1		(0x1 << 5)
#define TIM_CR1_CMS_CENTER_2		(0x2 << 5)
#define TIM_CR1_CMS_CENTER_3		(0x3 << 5)
#define TIM_CR1_CMS_MASK		(0x3 << 5)

参数三: 计数方向

/* DIR: Direction */
/****************************************************************************/
/** @defgroup tim_x_cr1_dir TIMx_CR1 DIR: Direction
@{*/
#define TIM_CR1_DIR_UP			(0 << 4)
#define TIM_CR1_DIR_DOWN		(1 << 4)

设置定时系数

// 禁用 preload
timer_disable_preload(TIM2);
// 循环模式
timer_continuous_mode(TIM2);
// 预分频系数, 根据当前所在总线计算分频后的周期
timer_set_prescaler(TIM2, 36000);
// 计数周期
timer_set_period(TIM2, 1000);

如果要设置中断

/* Enable TIM2 interrupt. */
nvic_enable_irq(NVIC_TIM2_IRQ);
/* Enable Channel 1 compare interrupt to recalculate compare values */
timer_enable_irq(TIM2, TIM_DIER_CC1IE);

启用(使能)TIM2

/* Counter enable. */
timer_enable_counter(TIM2);

完整例子

使用GPIO和定时器外设的例子, 参考 LibOpenCM3(一) Linux下命令行开发环境配置 中的演示示例

你可能感兴趣的:(LibOpenCM3(五) 基础功能: 系统时钟, GPIO, 定时器)