32单片机从入门到精通之软件编程——任务调度(十)
无论是在学习、工作还是生活中,我们都会遇到各种困难和挑战。然而,成功的人并不是没有困难和挑战,而是他们能够勇敢面对困难和挑战,并从中汲取力量和经验。他们相信自己的能力,从不轻易放弃。他们知道,只有克服困难,才能取得更好的成绩和更大的成就。所以,无论何时何地,我们都要坚持自己的梦想和目标,努力奋斗,永不言败。只要我们有信心和勇气,就能够克服一切困难,迎接成功的曙光。让我们一起努力吧,相信自己,坚持不懈,成就辉煌!
目录
上一张试卷
一、选择题(每题2分,共10分)
二、简答题(每题10分,共30分)
三、编程题(每题20分,共40分)
四、分析题(每题15分,共30分)
五、应用设计题(每题15分,共15分)
知识点
1. 任务与进程的区别
2. 调度器类型
3. 调度算法
4. 实时调度理论
5. 任务间通信与同步
6. 中断处理
7. 内存管理
8. 资源分配与管理
9. 能耗管理
案例
1. FreeRTOS简介
2. 创建与管理任务
3. 同步与通信机制
4. 计时器服务
试卷
一、选择题(每题2分,共20分)
二、简答题(每题5分,共20分)
三、编程题(每题15分,共30分)
四、设计题(共30分)
上一张试卷
一、选择题(每题2分,共10分)
-
关于STM32的嵌套向量中断控制器(NVIC),下列哪项描述是正确的?
- B) NVIC允许配置每个中断的主优先级和子优先级。
- 正确。NVIC支持多个级别的优先级设置,包括抢占优先级(主优先级)和响应优先级(子优先级),这使得可以精细地控制中断处理顺序。
- B) NVIC允许配置每个中断的主优先级和子优先级。
-
以下哪个函数用于设置STM32的中断优先级?
- A) HAL_NVIC_SetPriority()
- 正确。
HAL_NVIC_SetPriority()
函数用于设置指定中断线的优先级。
- 正确。
- A) HAL_NVIC_SetPriority()
-
在STM32中,使用SWD接口相比于JTAG接口的主要优点是什么?
- B) SWD接口占用较少引脚资源。
- 正确。SWD只需要两个引脚(SWDIO和SWCLK),而JTAG需要五个或更多(TMS, TCK, TDI, TDO, nTRST)。
- B) SWD接口占用较少引脚资源。
-
如果需要同时调试多个核心或设备链路,应该优先选择哪种接口?
- B) JTAG
- 正确。虽然SWD对于大多数应用来说足够了,但JTAG提供了更多的功能,比如支持多核调试和更复杂的边界扫描操作。
- B) JTAG
-
在STM32CubeMX中生成初始化代码时,默认使用的库是哪一个?
- B) HAL库
- 正确。STM32CubeMX默认情况下会为新项目配置HAL库以简化硬件抽象层的使用。
- B) HAL库
二、简答题(每题10分,共30分)
-
解释为什么对于大多数STM32应用来说,ST-Link V3搭配SWD接口是一个理想的选择,并列出至少两个理由。
ST-Link V3搭配SWD接口对大多数STM32应用来说是理想选择的原因如下:
- 效率高:SWD接口只需要两个引脚即可完成编程和调试任务,减少了连接复杂度并提高了效率。
- 成本效益:ST-Link V3通常集成在开发板上,无需额外购买昂贵的调试工具,降低了开发成本。
- 性能优越:V3版本的ST-Link具有更快的数据传输速率,能够加速下载和调试过程。
-
描述如何在STM32CubeMX中配置项目以确保使用SWD接口进行调试。
要确保使用SWD接口进行调试,请按照以下步骤操作:
- 打开STM32CubeMX并创建或加载一个现有项目。
- 进入“Pinout & Configuration”标签页,找到“System Core”下的“DEBUG”选项。
- 将“DEBUG”模式从“JTAG/SWD”改为仅“SWD”,这样就禁用了JTAG信号线,只保留SWD通信。
- 确认其他设置无误后,点击“Project Manager”标签页,然后生成代码。
-
举例说明一个场景,在这个场景中你会选择JTAG而不是SWD接口来进行调试,并解释原因。
假设我们正在开发一个多处理器系统,其中包含两个或更多Cortex-M内核,如某些STM32H7系列MCU。在这种情况下,可能会选择JTAG接口,因为:
- 多核调试:JTAG支持对所有内核的同时访问,这对于调试多核系统中的同步问题至关重要。
- 复杂性管理:JTAG提供了一种更加全面的方式来管理和诊断硬件问题,尤其是在涉及到边界扫描测试时。
三、编程题(每题20分,共40分)
- 编写一段C代码,使用HAL库配置并启动一个定时器,使其每隔1秒触发一次中断。请包括必要的初始化步骤,并添加适当的注释。
#include "stm32f4xx_hal.h" // 根据你的具体MCU型号调整 // 定义全局变量或声明外部函数等... void SystemClock_Config(void); // 系统时钟配置函数声明 void Error_Handler(void); // 错误处理函数声明 // 初始化定时器TIM2,配置为1Hz频率中断 void MX_TIM2_Init(void) { TIM_HandleTypeDef htim2; __HAL_RCC_TIM2_CLK_ENABLE(); // 使能定时器时钟 htim2.Instance = TIM2; htim2.Init.Prescaler = (SystemCoreClock / 1000) - 1; // 预分频值设定为1ms周期 htim2.Init.CounterMode = TIM_COUNTERMODE_UP; htim2.Init.Period = 999; // 自动重装载值设定为999,即1000个计数周期 htim2.Init.ClockDivision = TIM_CLOCKDIVISION_DIV1; htim2.Init.AutoReloadPreload = TIM_AUTORELOAD_PRELOAD_DISABLE; if (HAL_TIM_Base_Init(&htim2) != HAL_OK) { Error_Handler(); } // 启用定时器更新中断 if (HAL_TIM_Base_Start_IT(&htim2) != HAL_OK) { Error_Handler(); } } // 中断服务例程 void TIM2_IRQHandler(void) { /* USER CODE BEGIN TIM2_IRQn 0 */ /* USER CODE END TIM2_IRQn 0 */ HAL_TIM_IRQHandler(&htim2); /* USER CODE BEGIN TIM2_IRQn 1 */ /* USER CODE END TIM2_IRQn 1 */ }
- 编写一段C代码,使用LL库实现GPIO引脚的配置为推挽输出模式,并使LED连接到该引脚上闪烁。要求代码适用于STM32系列微控制器,并附上详细注释。
#include "stm32f4xx_ll_gpio.h" // 根据你的具体MCU型号调整 #include "stm32f4xx_ll_bus.h" #include "stm32f4xx_ll_rcc.h" #define LED_PIN LL_GPIO_PIN_5 #define LED_PORT GPIOA void SystemClock_Config(void); // 系统时钟配置函数声明 void Error_Handler(void); // 错误处理函数声明 int main(void) { // 初始化系统时钟 SystemClock_Config(); // 使能GPIOA的时钟 LL_AHB1_GRP1_EnableClock(LL_AHB1_GRP1_PERIPH_GPIOA); // 配置GPIOA的PIN5为推挽输出模式 LL_GPIO_SetPinMode(LED_PORT, LED_PIN, LL_GPIO_MODE_OUTPUT); LL_GPIO_SetPinOutputType(LED_PORT, LED_PIN, LL_GPIO_OUTPUT_PUSHPULL); LL_GPIO_SetPinSpeed(LED_PORT, LED_PIN, LL_GPIO_SPEED_FREQ_LOW); LL_GPIO_SetPinPull(LED_PORT, LED_PIN, LL_GPIO_PULL_NO); while (1) { // 切换LED状态 LL_GPIO_TogglePin(LED_PORT, LED_PIN); // 延迟一段时间(这里可以用软件延迟或者硬件定时器) for (volatile uint32_t i = 0; i < 500000; i++); } }
四、分析题(每题15分,共30分)
-
比较JTAG和SWD接口在实际应用中的优缺点,并说明在什么情况下应该选择其中一个而不是另一个。
- SWD的优点:
- 更少的引脚需求,适合引脚紧张的应用。
- 成本较低,因为SWD调试器通常比JTAG便宜。
- 对于单核系统或不需要复杂边界扫描的情况,SWD已经足够强大。
- SWD的缺点:
- 不支持多核调试和复杂的边界扫描测试。
- JTAG的优点:
- 支持多核调试和复杂的边界扫描测试。
- 更适合用于生产测试和故障诊断。
- JTAG的缺点:
- 需要更多的引脚资源,增加了电路板设计的复杂度。
- 相较于SWD,JTAG调试器可能更昂贵。
选择建议:
- 如果项目涉及多核处理器或者需要进行复杂的硬件诊断,则应选择JTAG。
- 对于大多数普通应用,尤其是那些注重成本和引脚资源的应用,SWD将是更好的选择。
- SWD的优点:
-
深入探讨在项目初期选择ST-Link V2还是V3时应考虑的因素,并给出针对不同项目类型的推荐。
在选择ST-Link V2或V3时,主要考虑以下几个因素:
- 速度与性能:V3提供了更高的数据传输速率,有助于加快程序下载和调试过程,特别适合大型项目的快速迭代。
- 兼容性和扩展性:V3拥有更好的兼容性和扩展性,支持更多的目标板和特性,例如SWO跟踪。
- 成本:V2的成本通常更低,但如果考虑到长期维护和支持,V3可能是更经济的选择。
- 特殊需求:如果项目有特定的需求,如远程调试、高级调试特性等,那么V3可能是必需的。
推荐:
- 对于小型项目或预算有限的情况下,可以选择V2,因为它能满足基本的编程和调试需求。
- 对于大型项目、多核系统或者有高级调试需求的项目,建议选择V3,以充分利用其性能优势和额外的功能。
五、应用设计题(每题15分,共15分)
设计一个简单的温度监控系统,该系统使用STM32微控制器、温度传感器以及LCD显示器。请详细描述你将如何根据系统的性能需求选择合适的调试接口(JTAG或SWD),并说明选择的理由。此外,请概述如何利用所选接口的功能来优化开发流程和系统性能。
对于这个温度监控系统,考虑到它是一个相对简单的单核应用,且主要关注的是温度测量和显示,我倾向于选择SWD接口。理由如下:
- 减少引脚占用:SWD只需要两根线,这样可以节省宝贵的PCB空间,特别是当我们在设计紧凑型产品时。
- 成本控制:SWD调试器通常价格更为亲民,有助于控制整体开发成本。
- 足够性能:对于这样一个应用,SWD提供的调试功能完全足够,包括设置断点、读取寄存器值、查看变量等。
为了优化开发流程和系统性能,我们可以这样做:
- 使用SWD进行快速迭代:SWD接口可以帮助我们迅速上传固件并实时监控运行状态,从而加快开发进度。
- 通过SWO端口进行跟踪:尽管不是必须的,但如果有需要,还可以利用SWD的SWO(串行线观察)功能来进行实时日志记录或性能分析,进一步提升开发效率。
- 保持简洁的设计:由于不需要额外的JTAG引脚,整个电路设计可以更加简洁,减少了潜在的布线错误和其他问题。
综上所述,对于这个温度监控系统,SWD接口是一个合理且高效的选择,它能够在满足开发需求的同时,最大限度地简化设计并降低成本。
知识点
任务调度是嵌入式系统和实时应用开发中的核心概念之一,它决定了多个并发任务如何被安排执行,确保系统的响应性和效率。以下是关于任务调度的详细讲解:
1. 任务与进程的区别
在讨论任务调度之前,了解任务(Task)和进程(Process)之间的区别是很重要的。在操作系统中,进程是程序的一次执行实例,拥有独立的地址空间。而任务,尤其是在RTOS环境中,通常指的是一个轻量级的线程,它们共享相同的地址空间但有各自的堆栈。
2. 调度器类型
- 非抢占式调度器(Cooperative Scheduling):也称为轮询或协作式调度,每个任务必须显式地交出CPU控制权给下一个任务。这种方式简单但可能导致较差的实时性能。
- 抢占式调度器(Preemptive Scheduling):调度器可以根据设定的优先级或其他条件中断当前正在运行的任务,并切换到更高优先级的任务。这提供了更好的实时响应性。
3. 调度算法
不同的调度算法适用于不同类型的系统需求:
- 先来先服务(FCFS, First-Come, First-Served):按照任务到达的顺序执行。
- 最短作业优先(SJF, Shortest Job First):优先处理预计执行时间最短的任务。
- 优先级调度(Priority Scheduling):根据预先分配的优先级来决定哪个任务应该首先执行。
- 时间片轮转(Round Robin):为每个任务分配固定的时间片,然后轮流执行。
- 多级反馈队列(Multilevel Feedback Queue):结合了时间片轮转和其他策略的优点,通过调整任务在不同优先级队列间的移动来优化调度。
4. 实时调度理论
对于实时系统,需要考虑更复杂的调度理论,如速率单调调度(Rate Monotonic Scheduling, RMS)、最早截止时间优先(Earliest Deadline First, EDF)等,这些理论旨在保证所有任务都能在其期限内完成,从而满足系统的实时性要求。
5. 任务间通信与同步
为了协调任务间的操作,必须使用适当的机制进行通信和同步,比如信号量、互斥锁、事件标志、消息队列等。这些工具帮助防止竞争条件,并确保数据一致性和任务间的有序交互。
6. 中断处理
在RTOS中,中断处理是非常重要的一部分。中断可以用来触发任务的创建或改变现有任务的状态。正确的中断管理能够提高系统的响应速度和可靠性。
7. 内存管理
任务可能需要动态分配内存。RTOS通常提供专门的内存池和服务来管理动态内存分配,以避免碎片化并确保高效利用资源。
8. 资源分配与管理
除了CPU时间,RTOS还需要管理其他有限的资源,如I/O端口、网络连接、文件句柄等。有效的资源管理可以帮助减少冲突并提高系统的整体性能。
9. 能耗管理
对于电池供电的设备,节能是一个关键问题。现代RTOS支持各种低功耗模式,并可以通过智能调度减少不必要的唤醒,延长设备的工作时间。
综上所述,任务调度不仅仅是选择一个合适的算法,而是涉及到了解整个系统的架构、需求以及限制。一个好的调度策略能够在满足实时性要求的同时,最大化系统的吞吐量和响应速度,同时最小化能耗。
案例
关于任务调度和RTOS的使用,特别是FreeRTOS,我将提供一个较为详细的讲解,并附带代码示例和注释。由于这方面的内容非常广泛,我们将专注于以下几个方面:
- FreeRTOS简介
- 创建与管理任务
- 同步与通信机制
- 计时器服务
1. FreeRTOS简介
FreeRTOS是一个轻量级的操作系统,专为微控制器设计,适用于实时应用。它提供了预占式或协作式的多任务调度器,以及一系列用于任务间通信和同步的服务。
2. 创建与管理任务
在FreeRTOS中,可以通过xTaskCreate()
函数来创建一个新的任务。每个任务都是在一个无限循环中运行的函数,该函数执行特定的任务逻辑。
#include "FreeRTOS.h" #include "task.h" // 定义一个简单的任务函数,这个任务会打印一条消息然后挂起自身。 void vSimpleTask(void *pvParameters) { // 将参数转换为字符串(这里假设传递的是一个字符串) const char *pcTaskName = (char *)pvParameters; while (1) { // 任务主体是一个无限循环 // 打印任务名 printf("Hello from %s\n", pcTaskName); // 挂起任务一段时间(例如1秒),让其他任务有机会运行 vTaskDelay(pdMS_TO_TICKS(1000)); } } int main(void) { // 初始化硬件等... // 创建两个任务,分别传递不同的名称作为参数 xTaskCreate(vSimpleTask, "Task1", configMINIMAL_STACK_SIZE, "Task 1", tskIDLE_PRIORITY + 1, NULL); xTaskCreate(vSimpleTask, "Task2", configMINIMAL_STACK_SIZE, "Task 2", tskIDLE_PRIORITY + 1, NULL); // 启动调度器 vTaskStartScheduler(); // 如果调度器启动失败,程序不应该到达这里。 for (;;); }
3. 同步与通信机制
FreeRTOS提供了多种同步和通信机制,如信号量、互斥锁、队列等。下面是一个使用队列的例子,其中两个任务通过队列交换数据。
#include "FreeRTOS.h" #include "task.h" #include "queue.h" QueueHandle_t xQueue; // 生产者任务 void vProducerTask(void *pvParameters) { uint32_t ulVarToSend = 0; while (1) { ulVarToSend++; // 发送数据到队列 if (xQueueSend(xQueue, &ulVarToSend, portMAX_DELAY) != pdPASS) { // 发送失败处理... } vTaskDelay(pdMS_TO_TICKS(1000)); // 等待1秒后再次发送 } } // 消费者任务 void vConsumerTask(void *pvParameters) { uint32_t ulReceivedValue; while (1) { // 从队列接收数据 if (xQueueReceive(xQueue, &ulReceivedValue, portMAX_DELAY) == pdPASS) { printf("Received value: %lu\n", ulReceivedValue); } } } int main(void) { // 创建队列,可以存储10个uint32_t类型的变量 xQueue = xQueueCreate(10, sizeof(uint32_t)); if (xQueue != NULL) { xTaskCreate(vProducerTask, "Producer", configMINIMAL_STACK_SIZE, NULL, tskIDLE_PRIORITY + 1, NULL); xTaskCreate(vConsumerTask, "Consumer", configMINIMAL_STACK_SIZE, NULL, tskIDLE_PRIORITY + 1, NULL); vTaskStartScheduler(); } // 如果创建队列失败或调度器启动失败,程序不应该到达这里。 for (;;); }
4. 计时器服务
FreeRTOS还支持软件定时器,这些定时器可以在指定的时间间隔后触发回调函数。下面是创建一个一次性定时器的例子。
#include "FreeRTOS.h" #include "timers.h" // 定时器回调函数 void vTimerCallback(TimerHandle_t xTimer) { // 这里是定时器到期后的动作 printf("Timer expired!\n"); } int main(void) { TimerHandle_t xTimer; // 创建一个一次性定时器,5秒后触发 xTimer = xTimerCreate("OneShotTimer", // 名字 pdMS_TO_TICKS(5000), // 5秒超时 pdFALSE, // 不重复 (void *)0, // 定时器ID vTimerCallback); // 回调函数 if (xTimer != NULL) { // 启动定时器 if (xTimerStart(xTimer, 0) != pdPASS) { // 启动定时器失败处理... } } vTaskStartScheduler(); // 如果调度器启动失败,程序不应该到达这里。 for (;;); }
以上代码片段展示了如何使用FreeRTOS进行基本的任务管理和资源分配。需要注意的是,实际项目中的实现可能会根据具体的硬件平台和需求有所不同。此外,上述代码假设你已经有了FreeRTOS环境设置,并且包含了所有必要的头文件。
试卷
一、选择题(每题2分,共20分)
-
在RTOS中,什么是“任务”?
- A) 一个进程
- B) 一个轻量级线程
- C) 系统调用
- D) 内核模块
-
哪种调度器类型允许高优先级的任务中断低优先级的任务执行?
- A) 协作式调度器
- B) 抢占式调度器
- C) 非抢占式调度器
- D) 循环调度器
-
下列哪一项不是常见的调度算法?
- A) 先来先服务(FCFS)
- B) 最短作业优先(SJF)
- C) 时间片轮转(Round Robin)
- D) 最长作业优先(LJF)
-
实时调度理论中,哪种方法是基于任务周期频率来分配优先级?
- A) 最早截止时间优先(EDF)
- B) 最短剩余时间优先(SRTF)
- C) 速率单调调度(RMS)
- D) 多级反馈队列(MFQ)
-
信号量主要用于什么目的?
- A) 提供网络连接
- B) 控制CPU温度
- C) 同步任务间的操作
- D) 分配内存块
-
在FreeRTOS中,哪个函数用于创建新任务?
- A)
vTaskStartScheduler
- B)
xQueueCreate
- C)
xTimerCreate
- D)
xTaskCreate
- A)
-
FreeRTOS中的
vTaskDelay
函数的作用是什么?- A) 永久挂起当前任务
- B) 设置任务的优先级
- C) 暂停当前任务一段时间
- D) 创建新的任务
-
中断处理在RTOS中通常用来做什么?
- A) 触发任务的创建或改变现有任务的状态
- B) 执行长时间运行的操作
- C) 进行文件读写
- D) 显示图形用户界面
-
动态内存分配可能会导致的问题是什么?
- A) 能耗增加
- B) 数据丢失
- C) 内存碎片化
- D) 任务超时
-
为了节能,现代RTOS支持哪些模式?
- A) 只有活动模式
- B) 活动模式和低功耗模式
- C) 仅低功耗模式
- D) 高性能模式
二、简答题(每题5分,共20分)
- 解释什么是抢占式调度器,并说明它与协作式调度器的区别。
- 描述一下多级反馈队列调度算法的工作原理。
- 为什么在实时系统中使用最早的截止时间优先(EDF)调度策略?
- 在RTOS中,如何通过任务间通信机制确保数据的一致性和同步?
三、编程题(每题15分,共30分)
- 编写一段C代码,创建两个任务:一个作为生产者,另一个作为消费者。生产者生成随机数并将其发送到队列中;消费者从队列中接收数据并打印出来。请包括必要的头文件导入和错误检查。
- 使用FreeRTOS实现一个简单的定时器功能,该定时器每隔5秒触发一次回调函数,在回调函数中打印一条消息。请包含完整的代码实现,包括定时器的创建、启动和回调函数的定义。
四、设计题(共30分)
设计一个小型嵌入式系统,该系统需要同时处理多个传感器输入(如温度、湿度等),并将数据发送到远程服务器。要求:
- 系统必须保证所有传感器数据都能在规定时间内被处理。
- 设计应考虑能效问题,即尽可能减少系统的能耗。
- 详细描述你的设计方案,包括任务划分、调度策略选择、任务间通信方式以及任何其他相关的设计决策。