信号量的分类

信号量的分类
Mutual Exclusion Semaphores(互斥):一种特殊的二进制信号量,专门针对互斥操作进行了优化。
Binary Semaphores(二进制):完成互斥、同步操作的最佳方式;速度最快,最常用。
Counting Semaphores(计数):类似于二进制信号量,可记录信号量释放的次数,可监视同一资源上的多个实例。

======== Mutual Exclusion Semaphores(互斥信号量)==============================

互斥信号量是一种特殊的二进制信号量,它是针对使用二进制信号量进行互斥操作时存在的一些问题设计的。
互斥信号量主要增加了对优先级倒置、删除安全以及递归访问的处理。
1.互斥信号量只能用于互斥操作。
2.只能由已经获取了互斥信号量的任务去释放它。
3.中断服务程序(ISR)不可以释放(semGive())互斥信号量。
4.互斥信号量不支持semFlush()操作。

A mutual exclusion (mutex) semaphore is a special binary semaphore that supports
ownership, recursive access, task deletion safety, and one or more protocols
for avoiding problems inherent to mutual exclusion.

When a task owns the mutex, it is not possible for any other task to lock or unlock that mutex.
Contrast this concept with the binary semaphore, which can be released by any task,
even a task that did not originally acquire the semaphore.

A mutex is a synchronization object that can have only two states:
Not owned.
Owned.

Two operations are defined for mutexes:
Lock : This operation attempts to take ownership of a mutex,
if the mutex is already owned by another thread then the invoking thread is queued.
Unlock : This operation relinquishes ownership of a mutex.
If there are queued threads then a thread is removed from the queue and resumed,
ownership is implicitly assigned to the thread.

======== Binary Semaphores(二进制信号量)======================================

1.互斥操作:是指不同任务可以利用信号量互斥地访问临界资源。
这种互斥的访问方式比中断禁止(interrupt disable) 与优先级锁定(preemptive locks)
两种互斥方式具有更加精确的粒度。
互斥操作时初始状态设为(SEM_FULL)可用。
并在同一个Task中成对、顺序调用semTake()、semGive()。

2.同步操作:是指一个任务可以利用信号量控制自己的执行进度,
使自己同步于一组外部事件。
同步操作时初始状态设为(SEM_EMPTY)不可用。
在不同Task中分别单独调用semTake()、semGive()。

Because Wait may cause a thread to block (i.e., when the counter is zero),
it has a similar effect of the lock operation of a mutex lock.
Similarly, a Signal may release a waiting threads,
and is similar to the unlock operation.

In fact, semaphores can be used as mutex locks.
Consider a semaphore S with initial value 1.
Then, Wait and Signal correspond to lock and unlock:

A binary semaphore can have a value of either 0 or 1.
When a binary semaphore’s value is 0, the semaphore is considered unavailable (or empty);
when the value is 1, the binary semaphore is considered available (or full ).

Note that when a binary semaphore is first created, it can be initialized to
either available or unavailable (1 or 0, respectively).

However, there is an advantage in using semaphores.
When a mutex lock is created, it is always in the "unlock" position.
If a binary semaphore is used and initialized to 0, it is equivalent to having a mutex lock
that is locked initially. Therefore, the use of binary semaphores is a little more flexible.

A binary semaphore is a synchronization object that can have only two states:
Not taken.
Taken.

Two operations are defined:
Take : Taking a binary semaphore brings it in the “taken” state,
trying to take a semaphore that is already taken enters the invoking thread into a waiting queue.
Release : Releasing a binary semaphore brings it in the “not taken” state
if there are not queued threads. If there are queued threads then a thread is removed
from the queue and resumed, the binary semaphore remains in the “taken” state.
Releasing a semaphore that is already in its “not taken” state has no effect.

======== Counting Semaphores(计数信号量)======================================

计数信号量与二进制信号量都可以用于任务之间的同步与互斥。
其不同点在于,计数信号量可记录信号量释放的次数,可以用来监视某一资源的使用状况。

A counting semaphore is a synchronization object that can have an arbitrarily large number of states.
The internal state is defined by a signed integer variable, the counter.
The counter value (N) has a precise meaning:
Negative, there are exactly -N threads queued on the semaphore.
Zero, no waiting threads, a wait operation would put in queue the invoking thread.
Positive, no waiting threads, a wait operation would not put in queue the invoking thread.

Two operations are defined for counting semaphores:
Wait : This operation decreases the semaphore counter,
if the result is negative then the invoking thread is queued.
Signal : This operation increases the semaphore counter,
if the result is non-negative then a waiting thread is removed from the queue and resumed.

======== Mutexes 互斥信号量 ====================================================

Mutexes are binary semaphores that include a priority inheritance mechanism. <优先级继承>

Whereas binary semaphores are the better choice for implementing synchronisation
(between tasks or between tasks and an interrupt),
mutexes are the better choice for implementing
simple mutual exclusion (hence 'MUT'ual 'EX'clusion).

When used for mutual exclusion the mutex acts
like a token that is used to guard a resource.
When a task wishes to access the resource it must first obtain ('take') the token.
When it has finished with the resource it must 'give' the token back -
allowing other tasks the opportunity to access the same resource.

Priority inheritance does not cure priority inversion!

It just minimises its effect in some situations.
Hard real time applications should be designed such that priority inversion
does not happen in the first place.

======== Recursive Mutexes 递归互斥信号量 ======================================

A mutex used recursively can be 'taken' repeatedly by the owner.
The mutex doesn't become available again until the owner has called
xSemaphoreGiveRecursive() for each successful xSemaphoreTakeRecursive() request.
For example, if a task successfully 'takes' the same mutex 5 times then the mutex
will not be available to any other task until it has also 'given'
the mutex back exactly five times.
This type of semaphore uses a priority inheritance mechanism so a task
'taking' a semaphore MUST ALWAYS 'give' the semaphore back
once the semaphore it is no longer required.

Mutex type semaphores cannot be used from within interrupt service routines.

Task() ----- xSemaphoreTakeRecursive()
|funcA --- xSemaphoreTakeRecursive(), xSemaphoreGiveRecursive()
|funcB --- xSemaphoreTakeRecursive(), xSemaphoreGiveRecursive()
Task() ----- xSemaphoreGiveRecursive()

======== Binary Semaphores(二进制信号量)======================================

Binary semaphores are used for both mutual exclusion and synchronisation purposes.
Binary semaphores and mutexes are very similar but have some subtle differences:
Mutexes include a priority inheritance mechanism, binary semaphores do not. <!优先级继承>
This makes binary semaphores the better choice for implementing synchronisation
(between tasks or between tasks and an interrupt),
and mutexes the better choice for implementing simple mutual exclusion.

Think of a binary semaphore as a queue that can only hold one item.
The queue can therefore only be empty or full (hence binary).
Tasks and interrupts using the queue don't care what the queue holds
- they only want to know if the queue is empty or full.
This mechanism can be exploited to synchronise (for example) a task with an interrupt.
Consider the case where a task is used to service a peripheral.
Polling the peripheral would be wasteful of CPU resources,
and prevent other tasks from executing.
It is therefore preferable that the task spends most of its time
in the Blocked state (allowing other tasks to execute) and
only execute itself when there is actually something for it to do.

This is achieved using a binary semaphore by having the task Block
while attempting to 'take' the semaphore.
An interrupt routine is then written for the peripheral that just 'gives'
the semaphore when the peripheral requires servicing.
The task always 'takes' the semaphore (reads from the queue to make the queue empty),
but never 'gives' it.
The interrupt always 'gives' the semaphore (writes to the queue to make it full)
but never takes it.

Task prioritisation can be used to ensure peripherals get services in a timely manner
- effectively generating a 'differed interrupt' scheme.
An alternative approach is to use a queue in place of the semaphore.
When this is done the interrupt routine can capture the data associated with the peripheral event
and send it on a queue to the task. The task unblocks when data becomes available on the queue,
retrieves the data from the queue, then performs any data processing that is required.
This second scheme permits interrupts to remain as short as possible,
with all post processing instead occurring within a task.

======== Counting Semaphores(计数信号量)======================================

Just as binary semaphores can be thought of as queues of length one,
counting semaphores can be thought of as queues of length greater than one.
Again, users of the semaphore are not interested in the data that is stored in the queue
- just whether or not the queue is empty or not.
Counting semaphores are typically used for two things:

Counting events.

In this usage scenario an event handler will 'give' a semaphore each time an event occurs
(incrementing the semaphore count value), and a handler task will 'take' a semaphore each time
it processes an event (decrementing the semaphore count value).
The count value is therefore the difference between the number of events that have occurred
and the number that have been processed.
In this case it is desirable for the count value to be zero when the semaphore is created.

Resource management.

In this usage scenario the count value indicates the number of resources available.
To obtain control of a resource a task must first obtain a semaphore
- decrementing the semaphore count value.
When the count value reaches zero there are no free resources.
When a task finishes with the resource it 'gives' the semaphore back
- incrementing the semaphore count value.

In this case it is desirable for the count value to be equal the maximum count value
when the semaphore is created.

======== Typical Semaphore Use =================================================
Semaphores are useful either for synchronizing execution of multiple tasks
or for coordinating access to a shared resource.

The following examples and general discussions illustrate using different types of semaphores
to address common synchronization design requirements effectively, as listed:
wait-and-signal synchronization
multiple-task wait-and-signal synchronization
credit-tracking synchronization
single shared-resource-access synchronization
multiple shared-resource-access synchronization
recursive shared-resource-access synchronization

死锁(或抱死)(Deadlock (or Deadly Embrace))
死锁也称作抱死,指两个任务无限期地互相等待对方控制着的资源。
设任务T1正独享资源R1,任务T2在独享资源T2,
而此时T1又要独享R2,T2也要独享R1,于是哪个任务都没法继续执行了,
发生了死锁。最简单的防止发生死锁的方法是让每个任务都:
先得到全部需要的资源再做下一步的工作
用同样的顺序去申请多个资源
释放资源时使用相反的顺序
内核大多允许用户在申请信号量时定义等待超时,以此化解死锁。
当等待时间超过了某一确定值,信号量还是无效状态,
就会返回某种形式的出现超时错误的代码,这个出错代码告知该任务,
不是得到了资源使用权,而是系统错误。
死锁一般发生在大型多任务系统中,在嵌入式系统中不易出现。

优先级倒置 : HP_task的优先级降至LP_task的优先级
HP_task等待LP_task的资源,于是处于Pend状态,这时一个中等优先级的MP_task进来,并抢占了LP_task的CPU,
此时的表现是低优先级MP_task在高优先级的HP_task前执行。这种现象就是优先级倒置。

优先级继承 : LP_task的优先级升至HP_task的优先级
HP_task等待LP_task的资源,于是处于Pend状态,这是把LP_task提升到HP_task的优先级
在LP_task semGive()后恢复LP_task的优先级,避免低于HP_task优先级的任务在HP_task
等待期间执行。这种现象就是优先级继承。
LP_task继承了HP_task的优先级。

The rule to go by for the scheduler is:

Activate the task that has the highest priority of all tasks in the READY state.

But what happens if the highest-priority task is blocked because it is waiting for a resource owned by a lower-priority task?
According to the above rule, it would wait until the low-priority-task becomes running again and releases the resource.
Up to this point, everything works as expected.

Problems arise when a task with medium priority becomes ready during the execution of the higher prioritized task.
When the higher priority task is suspended waiting for the resource, the task with the medium priority will run
until it finished its work, because it has higher priority as the low priority task.
In this scenario, a task with medium priority runs before the task with high priority.
This is known as priority inversion.

The low priority task claims the semaphore with OS_Use().
An interrupt activates the high priority task, which also calls OS_Use().
Meanwhile a task with medium got ready and runs when the high priority task is suspended.
After doing some operations, the task with medium priority calls OS_Delay() and is therefore suspended.
The task with lower priority continues now and calls OS_Unuse() to release the resource semaphore.
After the low priority task releases the semaphore, the high priority task is activated and claims the semaphore.

To avoid this kind of situation, the low-priority task that is blocking the highest-priority task gets assigned the highest priority
until it releases the resource, unblocking the task which originally had highest priority. This is known as priority inheritance.

With priority inheritance, the low priority task inherits the priority of the waiting high priority task
as long as it holds the resource semaphore. The lower priority task is activated instead of the medium priority task
when the high priority task tries to claim the semaphore.

mutex — specify the task-waiting order and enable task deletion safety, recursion,
and priority-inversion avoidance protocols, if supported.
binary — specify the initial semaphore state and the task-waiting order.
counting — specify the initial semaphore count and the task-waiting order.

 

转载于:https://www.cnblogs.com/shangdawei/p/3939376.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/473084.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

问题总结:一个 list 使用 for 遍历,边循环边删除的问题

一、需求&#xff1a; 对一个 list 数据类型写一个循环删除的程序 二、问题 来&#xff0c;我们来看看代码跟效果&#xff1a; # 初始化一个 list 列表&#xff0c;为了下边的方便比较&#xff0c;我就使用跟 list 索引来做 list 的元素 datas [0,1,2,3,4]# 打印元素组&am…

cpu使用率_漫话性能:CPU使用率

序言CPU 使用率是最直观和最常用的系统性能指标&#xff0c;更是我们在排查性能问题时&#xff0c;通常会关注的第一个指标。节拍率为了维护 CPU 时间&#xff0c;Linux 通过事先定义的节拍率&#xff08;内核中表示为 HZ&#xff09;&#xff0c;触发时间中断&#xff0c;并使…

TensorFlow 2.0 - 自定义模型、训练过程

文章目录1. 自定义模型2. 学习流程学习于&#xff1a;简单粗暴 TensorFlow 2 1. 自定义模型 重载 call() 方法&#xff0c;pytorch 是重载 forward() 方法 import tensorflow as tf X tf.constant([[1.0, 2.0, 3.0],[4.0, 5.0, 6.0]]) y tf.constant([[10.0],[20.0]])clas…

谁动了我的产品

2014年3月中旬离开了自己奋斗三年的公司&#xff0c;这是一家海关政府公司&#xff0c;三年里无论是做项目需求分析、项目开发、项目测试、项目上线实施、项目上线跟踪、收集反馈、做项目版本修改&#xff0c;我和我的团队都在一个有非常明确目标、有非常明确思路的过程中&…

LeetCode 352. 将数据流变为多个不相交区间(map二分查找)

文章目录1. 题目2. 解题1. 题目 给定一个非负整数的数据流输入 a1&#xff0c;a2&#xff0c;…&#xff0c;an&#xff0c;…&#xff0c;将到目前为止看到的数字总结为不相交的区间列表。 例如&#xff0c;假设数据流中的整数为 1&#xff0c;3&#xff0c;7&#xff0c;2&…

windows键按了没反应_windows快捷键使用 - 小怜

1、总的参考图&#xff1a;2、ctrl的组合使用&#xff1a;1与shift键结合&#xff1a;2 ctrl shift del # 快速清除浏览器缓存记录3 ctrl shift N # 浏览器当中&#xff0c;快速打开无痕新窗口。chrome内核的应该都可以&#xff0c;chrome和新…

Python倒计时自动发微信(电脑版微信)

一、前言&#xff1a; Python倒计时自动发微信&#xff08;电脑版微信登录状态&#xff09; 二、主要思路及步骤&#xff1a; 1、先启动微信 2、定位到搜索框 3、搜索微信 4、进入聊天窗口 5、粘贴文本内容 6、发送 7、关闭微信窗口 三、代码&#xff1a; import …

Javascript日期函数使用需要注意地方

当我们需要获取未来某个时间的毫秒数时&#xff0c;可能第一时间想到的方法是 (new Date("2014-12-08 12:00:00")).getTime() 这个在方法在chrome下是会返回一个数字的。 但是在IE下返回的是一个NaN&#xff0c;是因为IE下认为 YYYY-mm-dd HH:ii:ss不是一个合理的日期…

python判断是工作日还是休息日

一、概述 最近在做数据分析&#xff0c;需要判断一个日期是否为工作日&#xff0c;节假日。 找到一个现成的插件&#xff0c;蛮好用的。 1.1、插件介绍 chinesecalendar PyPI 判断某年某月某一天是不是工作日/节假日。 1.2、安装 pip install chinese_calendar该模块常…

win10主题更换_还不升级? win10精简版不到10G,运行比win7还快,旧电脑的福音

即使现在win7系统已经停止了服务&#xff0c;但是还有许多人宁愿面对随时有可能出现问题的win7&#xff0c;还是不愿升级win10系统。至于原因&#xff0c;五花八门&#xff0c;比如win7兼容性和稳定性更好&#xff0c;比如win10经常更新&#xff0c;还有许多软件无法在win10环境…

LeetCode 1732. 找到最高海拔

文章目录1. 题目2. 解题1. 题目 有一个自行车手打算进行一场公路骑行&#xff0c;这条路线总共由 n 1 个不同海拔的点组成。 自行车手从海拔为 0 的点 0 开始骑行。 给你一个长度为 n 的整数数组 gain &#xff0c;其中 gain[i] 是点 i 和点 i 1 的 净海拔高度差&#xff0…

小案例:利用python估算最外轮廓区域面积

一、需求&#xff1a; 给出一张图片&#xff0c;估算最外轮廓区域面积 二、步骤&#xff1a; 1、读取图片信息 2、利用open-cv&#xff0c;自适应分割图片 3、提取最外轮廓像素值 4、利用像素值标记轮廓 5、计算轮廓面积 三、代码&#xff1a; import cv2# 读取图片信息…

ApplicationContext容器的设计原理

1.在ApplicationContext容器中&#xff0c;我们以常用的FileSystemXmlApplicationContext的实现为例来说明ApplicationContext容器的设计原理。 2.在FileSystemXmlApplicationContext的设计中&#xff0c;我们看到ApplicationContext应用上下文的主要功能已经在FileSystemXmlAp…

使用c++查看linux服务器某个进程正在使用的内存_Linux 系统管理

1、进程管理介绍什么是进程程序是人使用计算机语言编写的&#xff0c;可以实现一定功能&#xff0c;并且可以执行的代码集合进程是正在执行当中的程序。程序在执行时&#xff0c;执行人的权限和属性、以及程序的代码都会被加载进内存&#xff0c;操作系统给这个进程分配一个 ID…

小案例:利用Python写个教师常用的点名软件

一、需求&#xff1a; 教师上课常用的点名软件 二、python库安装&#xff1a; openpyxl是Python中用于读写excel文件tkinter是Python中GUI编程非常好用的库&#xff0c;而且是标准库&#xff0c;不需要安装&#xff0c;导入即可使用random库是Python中用于实现随机功能的库&…

如何用DNS+GeoIP+Nginx+Varnish做世界级的CDN

如何用DNSGeoIPNginxVarnish做世界级的CDN 如何用BIND, GeoIP, Nginx, Varnish来创建你自己的高效的CDN网络&#xff1f;CDN&#xff0c;意思是Content Distrubtion Network&#xff0c;意思是内容分发网络&#xff0c;简单的说&#xff0c;就是全地域范围内的负载均衡&#xf…

python contains类似函数_01--实际工作中,python基础理念和数据处理

1.工作中遇到的python坑1.1 合并文件问题&#xff1a;正常将文件依次读取并append时&#xff0c;莫名出现很多空行。解决&#xff1a;在append前删除空行&#xff1a;data_tmp 1.2 重复数据行问题&#xff1a; append多日文件时&#xff0c;由于人工误操作&#xff0c;容易存在…

小案例:利用Python实现图片上下、左右翻转

一、前言需求&#xff1a; 对图片进行操作&#xff0c;使图片上下、左右翻转 二、函数库&#xff1a; 使用Pillow模块提供的transpose()方法可以让图像翻转&#xff0c;上下翻转&#xff0c;或者左右翻转 三、操作说明&#xff1a; 原图如下&#xff1a; 图片上下翻转代码…

LeetCode 1736. 替换隐藏数字得到的最晚时间

文章目录1. 题目2. 解题1. 题目 给你一个字符串 time &#xff0c;格式为 hh:mm&#xff08;小时&#xff1a;分钟&#xff09;&#xff0c;其中某几位数字被隐藏&#xff08;用 ? 表示&#xff09;。 有效的时间为 00:00 到 23:59 之间的所有时间&#xff0c;包括 00:00 和…

【SSH进阶之路】一步步重构MVC实现Struts框架——封装业务逻辑和跳转路径(四)...

目录&#xff1a; 【SSH进阶之路】Struts基本原理 实现简单登录&#xff08;二&#xff09; 【SSH进阶之路】一步步重构MVC实现Struts框架——从一个简单MVC开始&#xff08;三&#xff09; 【SSH进阶之路】一步步重构MVC实现Struts框架——封装业务逻辑和跳转路径&#xff08…