【FreeRTOS进阶】优先级翻转现象详解及解决方案
【FreeRTOS进阶】优先级翻转现象详解及解决方案
接下来我们聊聊优先级翻转这个经典问题。这个问题在实时系统中经常出现,尤其是在任务较多的场景下,而且问题定位起来比较麻烦。
什么是优先级翻转?
优先级翻转的核心定义很简单:较低优先级的任务阻塞了较高优先级任务的执行。
有同学可能会觉得奇怪:"这不对啊,不是应该高优先级任务抢占低优先级任务吗?怎么会反过来呢?“没错,这确实与我们对抢占式调度的认知相反,所以它被称为"优先级翻转”。
优先级翻转的发生过程
我们通过一个具体场景来详细分析这个问题:
假设我们现在有三个任务,分别是Task A(高优先级)、Task C(中优先级)和Task B(低优先级),以及一个信号量S。
事件顺序如下:
- 低优先级的Task B先运行,并获取了信号量S
- 高优先级的Task A变为就绪状态,抢占了Task B开始执行
- Task A也需要获取信号量S,但此时信号量已被Task B持有
- Task A进入阻塞状态,等待信号量释放
- 控制权回到Task B,Task B继续执行
- 中优先级的Task C变为就绪状态,抢占了低优先级的Task B
- 由于Task B未释放信号量,Task A继续阻塞
- Task C可以不受干扰地执行完毕(类似"山中无老虎,猴子称大王")
- Task C执行完后,控制权回到Task B
- Task B最终执行到释放信号量S的代码
- 信号量释放后,Task A获得信号量并恢复执行
这整个过程中,我们可以看到高优先级任务A被低优先级任务B阻塞了,而且这个阻塞时间不仅取决于B,还包括所有优先级介于A和B之间的任务(如Task C)的执行时间。
优先级翻转的危害
这个问题的危害在于:
- 实时性丧失:高优先级任务无法在预期时间内执行完成
- 系统行为不可预测:高优先级任务的阻塞时间取决于中低优先级任务的执行时间
- 可能导致严重后果:在关键实时系统中(如航空、医疗设备),这可能导致灾难性后果
解决优先级翻转的方法
FreeRTOS提供了三种主要解决优先级翻转问题的机制:
1. 优先级继承(Priority Inheritance)
这是最常用的解决方案:当低优先级任务持有高优先级任务需要的资源时,低优先级任务临时"继承"高优先级任务的优先级,直到释放资源。
在FreeRTOS中,互斥量(Mutex)默认启用优先级继承机制:
/* 创建带有优先级继承的互斥量 */
SemaphoreHandle_t xMutex = xSemaphoreCreateMutex();/* 使用互斥量进行资源保护 */
if (xSemaphoreTake(xMutex, portMAX_DELAY) == pdTRUE) {/* 临界区代码 */xSemaphoreGive(xMutex);
}
2. 优先级天花板协议(Priority Ceiling Protocol)
为每个共享资源设置一个"天花板优先级",任何访问该资源的任务都会临时提升到该优先级,直到释放资源。
/* 定义天花板优先级并创建互斥量 */
#define CEILING_PRIORITY (configMAX_PRIORITIES - 1)
SemaphoreHandle_t xMutex;/* 创建普通互斥量 */
xMutex = xSemaphoreCreateMutex();/* 使用前先将优先级提升到天花板优先级 */
UBaseType_t uxOriginalPriority = uxTaskPriorityGet(NULL);
vTaskPrioritySet(NULL, CEILING_PRIORITY);/* 使用互斥量进行资源保护 */
if (xSemaphoreTake(xMutex, portMAX_DELAY) == pdTRUE) {/* 临界区代码 */xSemaphoreGive(xMutex);
}/* 恢复原始优先级 */
vTaskPrioritySet(NULL, uxOriginalPriority);
3. 关闭中断或调度器
对于短小的临界区,可以通过临时关闭中断或调度器来避免优先级翻转:
/* 方法1:关闭中断 */
taskENTER_CRITICAL();
/* 临界区代码(必须非常短) */
taskEXIT_CRITICAL();/* 方法2:挂起调度器 */
vTaskSuspendAll();
/* 临界区代码 */
xTaskResumeAll();
实际案例:Mars Pathfinder任务重启事件
NASA的火星探路者任务在1997年就遇到了由于优先级翻转导致的系统重启问题。系统检测到了"长时间任务"而触发保护性重启。调查发现,这是由于优先级翻转导致高优先级任务被阻塞太久。幸运的是,VxWorks操作系统支持优先级继承,工程师远程启用了这一功能,问题得到解决。
这个经典案例说明了优先级翻转在实际中的危害及其解决方案的重要性。
如何避免优先级翻转?
- 尽量减少共享资源:降低任务间的资源依赖
- 合理设计优先级:相互有资源依赖的任务尽量使用接近的优先级
- 使用优先级继承的互斥量:替代普通信号量进行资源保护
- 缩小临界区范围:尽量减少持有互斥量的时间
- 避免嵌套锁:防止死锁和复杂的优先级问题
实验验证
我们可以通过一个简单实验来模拟优先级翻转现象:
/* 创建不带优先级继承的二值信号量 */
SemaphoreHandle_t xSemaphore = xSemaphoreCreateBinary();
xSemaphoreGive(xSemaphore); /* 初始状态为可用 *//* 低优先级任务 */
void vLowPriorityTask(void *pvParameters) {for(;;) {/* 获取信号量 */if(xSemaphoreTake(xSemaphore, portMAX_DELAY) == pdTRUE) {printf("低优先级任务获取信号量\r\n");/* 模拟长时间处理 */vTaskDelay(pdMS_TO_TICKS(2000));printf("低优先级任务释放信号量\r\n");xSemaphoreGive(xSemaphore);}vTaskDelay(pdMS_TO_TICKS(1000));}
}/* 中优先级任务 */
void vMediumPriorityTask(void *pvParameters) {for(;;) {printf("中优先级任务运行\r\n");/* 模拟占用CPU时间 */vTaskDelay(pdMS_TO_TICKS(3000));}
}/* 高优先级任务 */
void vHighPriorityTask(void *pvParameters) {for(;;) {printf("高优先级任务尝试获取信号量\r\n");TickType_t xStartTime = xTaskGetTickCount();if(xSemaphoreTake(xSemaphore, portMAX_DELAY) == pdTRUE) {TickType_t xEndTime = xTaskGetTickCount();printf("高优先级任务获取信号量,等待时间: %lu ms\r\n", (xEndTime - xStartTime) * portTICK_PERIOD_MS);vTaskDelay(pdMS_TO_TICKS(500));xSemaphoreGive(xSemaphore);}vTaskDelay(pdMS_TO_TICKS(5000));}
}/* 创建任务 */
void vStartTasks(void) {xTaskCreate(vLowPriorityTask, "LowTask", configMINIMAL_STACK_SIZE, NULL, 1, NULL);xTaskCreate(vMediumPriorityTask, "MedTask", configMINIMAL_STACK_SIZE, NULL, 2, NULL);xTaskCreate(vHighPriorityTask, "HighTask", configMINIMAL_STACK_SIZE, NULL, 3, NULL);
}
通过这个实验,我们能观察到高优先级任务需要等待低优先级任务释放信号量,而这个过程又被中优先级任务"插队",导致高优先级任务的等待时间大大延长。
总结
优先级翻转是实时系统设计中容易被忽视但又非常重要的问题。合理使用互斥量和优先级继承机制是解决此问题的关键。在设计多任务实时系统时,我必须谨慎思考任务间的资源共享和优先级分配。
如果你想深入学习FreeRTOS的更多高级特性及实战案例,欢迎访问我的GitHub仓库:Despacito0o/FreeRTOS,那里有从入门到精通的完整教程和示例代码,包括本文讨论的优先级翻转问题的实战解决方案!
📚 FreeRTOS + STM32学习资源库 - 从入门到精通的完整学习路径
相关推荐阅读:
- FreeRTOS二值信号量详解与实战教程
- FreeRTOS计数型信号量详解与实战教程
相关文章:
【FreeRTOS进阶】优先级翻转现象详解及解决方案
【FreeRTOS进阶】优先级翻转现象详解及解决方案 接下来我们聊聊优先级翻转这个经典问题。这个问题在实时系统中经常出现,尤其是在任务较多的场景下,而且问题定位起来比较麻烦。 什么是优先级翻转? 优先级翻转的核心定义很简单:…...
解决 IntelliJ IDEA 项目启动时端口冲突问题
1.问题 Description: The Tomcat connector configured to listen on port 8082 failed to start. The port may already be in use or the connector may be misconfigured. Action: Verify the connectors configuration, identify and stop any process thats listening…...
笔试专题(十一)
文章目录 添加字符(暴力枚举)题解代码 城市群数量(dfs)题解代码 判断是不是平衡二叉树(递归)题解代码 最大子矩阵(二维前缀和)题解代码 小葱的01串 (固定区间大小的滑动窗…...
C++11新增语法:列表初始化
前言: 接下来我们将要讲解,相较于c98,c11中新增的语法以及如何使用~。我们首先来讲解:列表初始化。 下文预告:右值引用和移动语义 C98中传统的{} 在c98中的{},仅能初始化数组和结构体 #include<iostrea…...
Linux:基础IO---动静态库
文章目录 1. 动静态库前置知识1.1 动静态库知识回顾1.2 什么是动静态库 2. 动静态库2.1 站在库的制作者的角度2.2 站在库的使用者的角度2.3 动态库是怎么被加载的(原理) 序:上一篇文章我们从认识到理解,从理解到实现场景ÿ…...
从裸仓库到GitLab全解析
Git服务器搭建与使用指南:从裸仓库到GitLab全解析 前言 在团队协作开发中,版本控制系统是必不可少的工具。虽然GitHub提供了优秀的代码托管服务,但企业级项目往往需要更安全的私有化部署方案。本文将手把手教你两种主流的Git服务器搭建方式…...
OzGIS:地理信息分析与处理软件
大家好,今天为大家介绍的软件是OzGIS:一款地理信息分析与处理软件。下面,我们将从软件的主要功能、支持的系统、软件官网等方面对其进行简单的介绍。 OzGIS官网网址为:https://ozgis.sourceforge.io/。 OzGIS是一款开源软件&#…...
PHP异常处理__Throwable
在 PHP 里,Throwable 是一个极为关键的接口,自 PHP 7 起被引入。它为错误和异常处理构建了一个统一的框架。下面会详细介绍 Throwable 的相关内容。 1. 基本概念 Throwable 是 Exception 和 Error 的父接口。在 PHP 7 之前,异常(…...
PHP异常处理__Exception类
以下是对 PHP 中 Exception 类的详细解释: 一、Exception 类概述 Exception 是 PHP 中所有异常类的基类。它提供了一个通用的异常处理机制,用于处理程序执行过程中可能出现的错误情况。当程序中出现异常时,可以创建 Exception 的实例并将其…...
C++中动态多态类别浅析
非抽象类继承和虚函数 #include <iostream> using namespace std;class Base { public:virtual void func() { // 虚函数,支持动态绑定cout << "Base::func()" << endl;} };class Derived : public Base { public:void func() overrid…...
游戏引擎学习第234天:实现基数排序
回顾并为今天的内容设定背景 我们今天继续进行排序的相关,虽然基本已经完成了,但还是想收尾一下,让整个流程更完整。其实这次排序只是个借口,主要是想顺便聊一聊一些计算机科学的知识点,这些内容在我们项目中平时不会…...
系分架构论文《论高并发场景的架构设计和开发方法》
系统分析师论文范文系列 【摘要】 2022年8月,我司承接了某知名电商平台“秒杀系统架构优化”项目,我作为系统分析师主导了整体架构设计与技术选型工作。该平台在促销活动中面临瞬时流量超过50万QPS的挑战,原有架构存在数据库崩溃、服务响应延…...
最新得物小程序sign签名加密,请求参数解密,响应数据解密逆向分析
点击精选,出现https://app.dewu.com/api/v1/h5/index/fire/index 这个请求 直接搜索sign的话不容易定位 直接搜newAdvForH5就一个,进去再搜sign,打上断点 可以看到t.params就是没有sign的请求参数, 经过Object(a.default)该函数…...
jangow靶机笔记(Vulnhub)
环境准备: 靶机下载地址: https://download.vulnhub.com/jangow/jangow-01-1.0.1.ova kali地址:192.168.144.128 靶机(jangow)地址:192.168.144.180 一.信息收集 1.主机探测 使用arp-scan进行主机探…...
Spring Boot + Caffeine:打造高性能缓存解决方案
1. 引言 1.1 缓存的重要性 缓存是提升系统性能的关键技术之一,通过将频繁访问的数据存储在内存中,减少对数据库或其他外部系统的访问次数,从而降低延迟并提高吞吐量。 缓存的基本概念:缓存是一种临时存储机制,用于快速访问常用数据。缓存在提升系统性能中的作用:减少数…...
C++入门小馆: 深入string类
嘿,各位技术潮人!好久不见甚是想念。生活就像一场奇妙冒险,而编程就是那把超酷的万能钥匙。此刻,阳光洒在键盘上,灵感在指尖跳跃,让我们抛开一切束缚,给平淡日子加点料,注入满满的pa…...
命令行基础
学习目标 掌握VRP命令行的基础知识 利用VRP命令行进行基本的配置 VRP命令行的基础知识 一、VRP 命令行基本架构 1. 用户视图(User View) 进入方式:设备启动后默认进入,提示符为 <HUAWEI>。功能&#…...
10-DevOps-Jenkins参数化构建实现多版本发布
在之前的Jenkins配置中,固定写死了程序的版本号,实际情况是随着版本的不断迭代,版本号也是不断变化的,版本号由代码仓库(GitLab)设置。 当前Jenkins配置是固定写的1.0,本节我们要把它改成动态的…...
C++游戏服务器开发之⑦redis的使用
目录 1.当前进度 2.守护进程 3.进程监控 4.玩家姓名添加文件 5.文件删除玩家姓名 6.redis安装 7.redis存取命令 8.redis链表存取 9.redis程序结构 10.hiredisAPI使用 11.基于redis查找玩家姓名 12.MAKEFILE编写 13.游戏业务实现总结 1.当前进度 2.守护进程 3.进程监…...
二进制裁剪命令mips-linux-gnu-strip 命令的使用
-s 或者--strip-all:移除所有符号和调试信息 -g 或者--strip-debug:仅移除调试信息 -d 或者--strip-unneeded:移除不需要的符号 默认不传任何参数 也是移除所有符号和调试 应用:把文件系统所有二进制镜像使用一遍,缩小文件系统大小 79K Apr 19 15:47 fat.ko //使用前 mips-l…...
【Bluedroid】蓝牙存储模块配置管理:启动、读写、加密与保存流程解析
本文围绕蓝牙存储模块展开,主要解析了蓝牙存储模块(StorageModule)的初始化流程,重点围绕配置文件校验、读取、设备类型修复及加密处理展开。通过工厂重置检测、校验和验证、多源配置加载、设备类型推断修正等步骤,确保…...
SpringBoot启动后初始化的几种方式
目录 一、静态代码块 二、构造方法 三、PostConstruct 四、InitializingBean 接口 五、 Bean 注解中的 initMethod 六、 CommandLineRunner 接口 七、ApplicationRunner 接口 八、EventListener事件 九、SmartInitializingSingleton接口 十、ApplicationListener接口…...
asp.net core webapi+efcore
简洁的restfull风格 目前c#提供了多种风格的web编程,因为微软有自己的前端,所以集成了很多内容,不过基于现在编程前后端分离的模式,webapi是合适的。 webapi 目前网络上有很多介绍,不反复说这个了。在建立控制器时&…...
java怎么完善注册,如果邮箱中途更换,能否判断
解析在下面 附赠代码 private static class CodeInfo {String code;long timestamp;CodeInfo(String code, long timestamp) {this.code code;this.timestamp timestamp;}}// 存储验证码(邮箱 -> 验证码信息)(保证线程安全) 以免中途更改邮箱pri…...
实战设计模式之备忘录模式
概述 与解释器模式、迭代器模式一样,备忘录模式也是一种行为设计模式。备忘录模式允许我们保存一个对象的状态,并在稍后恢复到这个状态。该模式非常适合于需要回滚、撤销或历史记录等功能的应用场景。通过使用备忘录模式,开发者可以轻松添加诸…...
数据库表设计
一对一关系 共享主键 两个表的主键是相同的 唯一外键 从表中记录主表的id 一对多关系 从表(多的表)存储主表的id 多对多关系 设计一个中间表(关联表),它有两列分别记录两个主表(A 和 B)…...
Linux 桌面环境 LXQt 2.2 发布
Linux 桌面环境 LXQt 2.2 于 2025 年 4 月 17 日正式发布。这是该轻量级开源 Qt 桌面环境的最新稳定版本,带来了诸多改进,特别是在 Wayland 支持方面。以下是一些主要的更新内容: Wayland 支持增强: 提升了多屏支持,使…...
多人五子棋联机对战平台 测试报告
目录 项目介绍 测试用例设计 部分功能测试示例 自动化测试 测试范围 排除范围 自动化测试目录编辑 执行全部自动化测试用例 性能说明 总结 性能测试 结果分析 测试总结 项目介绍 该项目基于WebSocket实现实时通信,采用SSM框架构建在线五子棋多人联机…...
探索 .bat 文件:自动化任务的利器
在现代计算机操作中,批处理文件(.bat 文件)是一种简单而强大的工具,它可以帮助我们自动化重复性任务,工作效率提高。尽管随着编程语言和脚本工具的发展,.bat 文件的使用频率有所下降,但它依然是…...
240419 leetcode exercises
240419 leetcode exercises jarringslee 文章目录 240419 leetcode exercises[19. 删除链表的倒数第 N 个结点](https://leetcode.cn/problems/remove-nth-node-from-end-of-list/)🔁 经典方法:两次遍历暴力求解🔁 双指针法 :快慢…...
开源Midjourney替代方案:企业级AI绘画+PPT生成系统+AI源码
「AI取代设计师?」开源Midjourney替代方案:企业级AI绘画PPT生成系统 ——零代码私有化部署,5倍速出图100%版权合规 设计师行业的危机与机遇 1. 传统设计流程的致命短板 痛点 人工设计 AI系统 单张海报耗时 3小时(含反复修改…...
学习笔记十七——Rust 支持面向对象编程吗?
🧠 Rust 支持面向对象编程吗? Rust 是一门多范式语言,主要以 安全、并发、函数式、系统级编程为核心目标,但它同时也支持面向对象的一些关键特性,比如: 特性传统 OOP(如 Java/C)Ru…...
图灵奖得主LeCun:DeepSeek开源在产品层是一种竞争,但在基础方法层更像是一种合作;新一代AI将情感化
图片来源:This is World 来源 | Z Potential Z Highlights: 新型的AI系统是以深度学习为基础,能够理解物理世界并且拥有记忆、推理和规划能力的。一旦成功构建这样的系统,它们可能会有类似情感的反应,但这些情感是基…...
Flink框架十大应用场景
Flink框架适合应用的场景 1. 流式数据处理 Flink框架最常用的应用场景是流式数据处理。流式数据处理是指对实时数据进行处理,以便及时地做出决策。例如,一个电商网站需要对用户的行为进行实时分析,以便根据用户的兴趣和行为推荐商品。Flink框架可以帮助电商网站实时地处理数…...
C++镌刻数据密码的树之铭文:二叉搜索树
文章目录 1.二叉搜索树的概念2.二叉搜索树的实现2.1 二叉搜索树的结构2.2 二叉搜索树的节点寻找2.2.1 非递归2.2.2 递归 2.3 二叉搜索树的插入2.3.1 非递归2.3.2 递归 2.4 二叉搜索树的删除2.4.1 非递归2.4.2 递归 2.5 二叉搜索树的拷贝 3.二叉树的应用希望读者们多多三连支持小…...
CAN与CANFD协议说明
在 CAN(Controller Area Network,控制器局域网)协议里,仲裁域波特率和数据域比特率有着不同的含义和作用,下面为你详细介绍并举例说明。 概念解释 仲裁域波特率 含义:仲裁域是 CAN 数据帧中的一部分&…...
【C++ Qt】信号和槽(内配思维导图 图文并茂 通俗易懂)
每日激励:“不设限和自我肯定的心态:I can do all things。 — Stephen Curry” 绪论: 本章是Qt中的第三章,也是我们理解Qt中必备的点 信号槽,它本质由信号和槽两个来实现,其中将细致的讲述如何自定义信号…...
【实战】在 Linux 上使用 Nginx 部署 Python Flask 应用
在 Linux 上使用 Nginx 部署 Python Flask 应用 步骤一:准备 Flask 应用 创建 Flask 应用 确保你有一个可以运行的 Flask 应用。例如,创建一个简单的 app.py 文件: from flask import Flask app Flask(__name__)app.route(/) def hello_wor…...
java ai 图像处理
Java AI 图像处理 图像处理是人工智能(AI)领域中非常重要的一个应用方向。通过使用Java编程语言和相应的库,我们可以实现各种图像处理任务,如图像识别、图像分类、图像分割等。本文将介绍一些常见的图像处理算法,并通过…...
【绘制图像轮廓】图像处理(OpenCV) -part7
15 绘制图像轮廓 15.1 什么是轮廓 轮廓是一系列相连的点组成的曲线,代表了物体的基本外形。相对于边缘,轮廓是连续的,边缘不一定连续,如下图所示。轮廓是一个闭合的、封闭的形状。 轮廓的作用: 形状分析 目标识别 …...
Mesh模型孔洞修补算法总汇
关于Mesh 孔洞修补算法(Hole Filling in Meshes),这是计算几何和图形学中的一个重要话题,常用于重建、3D 扫描、建模等领域。下面我会系统总结主流和经典的孔洞修补方法,并按技术路线分类说明每种的原理、优缺点&#…...
ARINC818协议(六)
上图中,红色虚线上面为我们常用的simple mode简单模式,下面和上面的结合在一起,就形成了extended mode扩展模式。 ARINC818协议 container header容器头 ancillary data辅助数据 视频流 ADVB帧映射 FHCP传输协议 R_CTRL:路由控制routing ctr…...
RTMP握手流程
RTMP(Real-Time Messaging Protocol) 不支持除H.264/AAC之外的标准。 使用TCP,当到达网络拥塞、宽带上限时,传输质量受到影响。 URL格式: rtmp://host:port/app(名称)/stream(流IDÿ…...
【解决】torch引入过程中的ImportError: __nvJitLinkAddData_12_1, version libnvJitLink.so.12
大纲 本文记录在环境配置好后,在 import torch 过程中报了 异常 ImportError: /home/Coding/Envs/envs/only_test/lib/python3.10/site-packages/torch/lib/../../nvidia/cusparse/lib/libcusparse.so.12: undefined symbol: __nvJitLinkAddData_12_1, version lib…...
面试招聘:新能源汽车研发测试人员需求内部研讨会纪要(2025年4月19日草稿流出)
标题:XX汽车技术中心技术管理岗闭门会议纪要完整版:双非招聘策略、面试话术与风控方案(附待定事项) 【内部密级文件】 时间:2025年4月15日 14:00-17:30 地点:某主机厂研发中心会议室(305&#…...
从零开始学习 Lucene.Net:.NET Core 中的全文搜索与索引管理
Lucene.Net 是一个开源的全文搜索引擎库,它是 Apache Lucene 项目的 .NET 移植版本。Lucene.Net 提供了强大的搜索功能,广泛应用于文档搜索、日志分析、数据检索等场景。随着大数据的爆发,开发者越来越依赖高效的搜索引擎来实现复杂的搜索需求…...
opencv--图像处理
图像处理技术 图像处理技术是利用计算机对图像进行计算,分析和处理的技术,包括数字图像处理和计算机视觉两大领域。 对图像的处理包括滤波,缩放,分割,识别(两种信息对比)等。 链接 数字图像处理 1. 数字图像处理(Digital Image Processing) 数字图像处理主要关注图…...
JCST 2025年 区块链论文 录用汇总
Conference:Journal of Computer Science and Technology (JCST) CCF level:CCF B Categories:交叉/综合/新兴 Year:2025(截止4.19) JCST 2024年 区块链论文 录用汇总 1 Title: An Understandable Cro…...
股指期货跨期套利是如何赚取价差利润的?
股指期货跨期套利,简单来说,就是在同一交易所内,针对同一股指期货品种的不同交割月份合约进行的套利交易。投资者会同时买入某一月份的股指期货合约,并卖出另一月份的股指期货合约,待未来某个时间点,再将这…...
【java实现+4种变体完整例子】排序算法中【冒泡排序】的详细解析,包含基础实现、常见变体的完整代码示例,以及各变体的对比表格
以下是冒泡排序的详细解析,包含基础实现、常见变体的完整代码示例,以及各变体的对比表格: 一、冒泡排序基础实现 原理 通过重复遍历数组,比较相邻元素并交换逆序对,逐步将最大值“冒泡”到数组末尾。 代码示例 pu…...