【11】Strongswan processor 详解1

news2025/4/15 15:56:09

processor_t结构体,声明了一些公用方法:
get_total_threads获取总的线程数量;
get_idle_threads获取空闲线程数量;
get_working_threads按指定的优先级获取处理该优先级的job的线程数量;
get_job_load 或取指定优先级job队列中的job数量;
queue_job 添加job到指定job优先级对应的队列中;
execute_job直接调用一个空闲线程处理job,如果没有空闲线程则直接由调用该接口的线程直接处理job。
set_threads 设置线程池中的线程数量,这可能导致线程池的扩大或缩小。
cancel 线程数量设置为0,取消线程池,直到所有job处理完毕。
destroy 销毁processor对象。

private_processor_t 结构体继承processor_t结构体,
processor_create函数创建一个processor_t对象,其中挂载公有函数初始化processor_t对象,除此之外,private_processor_t还有私有的成员:
total_threads 正在运行的工作线程数量,他们可能处于等待job_added条件变量中或正在处理job;
desired_threads 用户指定的线程数量;set_threads 设置的线程数量会保存。
working_threads数组,每个优先级的线程数量;
threads 链表,保存线程池中所有的线程;
jobs数组,每个优先级的job链表;
prio_threads数组,每个优先级指定的线程数量;
mutex互斥锁
job_added条件变量,加入job通知,线程从条件变量处继续执行,从队列中取job执行。
thread_terminated 线程结束条件变量。线程函数退出时发送该信号,上述cancel函数接收该信号。

worker_thread_t结构体描述一个工作线程:
processor 指向实例化的processor对象;
thread 指向工作线程;
job 指向将要被该线程处理的job;
priority job的优先级。

get_idle_threads_nolock函数:
total_threads 总的线程数减去正在执行job的线程数量。

get_job函数:
值越小,优先级越高,从最高优先级的job开始,避让算法,prio_threads[i]大于working_threads[i]说明该i级别的job处理线程不够,reserved保存缺口数量,在for循环执行到下一个优先级的job时,reserved >= idle的化就直接退出不处理,优先高级别的job。跳过这些能get了,就remove_first从该优先级的job链表中取出第一个job。

process_jobs函数:
是线程池多有线程的执行函数。
while (this->desired_threads >= this->total_threads)决定是否跳出循环结束线程,当不成立时说明desired_threads 变小了,线程池缩小了,要把执行完job的线程退出。否则则取出worker线程优先级一致的job,调用process_job处理job,注意互斥锁的加锁与解锁,入时解锁处理job,出时加锁继续下一次的保护。没有job则条件等待。由于条件等待不超时,所以线程池的缩放需要加入job的信号激活。
那么当此线程退出时,发生了线程池缩小,total_threads线程数减少,发送thread_terminated条件信号。在cancel函数中接收判断线程是否都已经释放了:

	while (this->total_threads > 0)
	{
		this->job_added->broadcast(this->job_added);
		this->thread_terminated->wait(this->thread_terminated, this->mutex);
	}

process_job函数处理job:
(1) working_threads++ 增加总的在处理job数
(2) 由于thread_cleanup_pop(FALSE);纯如false,不必理会
(3)调用job->execute的处理job,返回值的几种类型:
若是JOB_REQUEUE_TYPE_NONE,则!= JOB_REQUEUE_TYPE_DIRECT,跳出循环,后续to_destroy = worker->job; 进而to_destroy->destroy(to_destroy);销毁job。
若是JOB_REQUEUE_TYPE_DIRECT,则是再直接执行job->execute,前提是worker->job->cancel不为空,也即可取消,否则while中一直执行不可取消了。
若是JOB_REQUEUE_TYPE_FAIR,后续插入job队列。
若是JOB_REQUEUE_TYPE_SCHEDULE,则…
JOB_REQUEUE_TYPE_DIRECT:
(4)调用job->execute后,working_threads–
(5)如果在job->execute时,worker->job->status 设置为 JOB_STATUS_CANCELED,则销毁job。

queue_job函数:
按不同的优先级插入job,并向线程发送条件变量信号激发处理。

execute_job:
如果有空闲线程,插入和job相同优先级的队列的首部,唤醒等待此条件变量的线程,
如果插入队列没有成功,则由调用此函数的线程直接处理job的execute函数

set_threads:
(1)根据配置预设各优先级job的处理线程数量。
(2)根据要设置的count线程数,扩充或缩小线程池。
(3)缩小,则this->desired_threads = count;,在process_jobs函数中,处理完job后会while条件判断失败,退出循环,退出该线程,达到减少目的。
(4)扩大,则thread_create创建新的线程。

线程局部存储变量private_thread_t :

//每个线程的id、cleanup_handlers之类在不同的存储空间,互不影响。
thread_t *thread_current()
{
	private_thread_t *this;

	this = (private_thread_t*)current_thread->get(current_thread);
	if (!this)
	{
		this = thread_create_internal();
		this->id = get_thread_id();
		current_thread->set(current_thread, (void*)this);
	}
	return &this->public;
}

u_int thread_current_id()
{
	private_thread_t *this = (private_thread_t*)thread_current();

	return this ? this->id : 0;
}

void thread_cleanup_push(thread_cleanup_t cleanup, void *arg)
{
	private_thread_t *this = (private_thread_t*)thread_current();
	cleanup_handler_t *handler;

	INIT(handler,
		.cleanup = cleanup,
		.arg = arg,
	);

	this->cleanup_handlers->insert_last(this->cleanup_handlers, handler);
}

(1)在多线程编程中,pthread_setspecific 和 pthread_getspecific 是两个用于处理线程局部存储(TLS)的函数,
(2)它们允许线程存储和访问线程特定的数据。这些函数提供了一种在同一线程中不同函数间共享数据的方法,这对于创建线程安全的程序至关重要。
(3)线程局部存储是一种允许数据在多个线程间独立存在的机制。每个线程都有自己的数据副本,因此对这些数据的操作不会影响其他线程。
(4)这种机制在处理不支持原子操作的自定义数据类型时尤其有用,因为它可以在不使用锁的情况下实现线程安全。
(5)要使用这些函数,首先需要通过 pthread_key_create 创建一个 pthread_key_t 类型的键。这个键用于标识线程特定的数据。
(6)然后,可以使用 pthread_setspecific 将数据与键关联,使用 pthread_getspecific 获取与键关联的数据。
(7)在上述代码中,pthread_setspecific 用于设置线程特定的数据,而 pthread_getspecific 用于获取这些数据。
(8)每个线程都有自己的数据副本,因此即使是使用相同的键,不同线程中的数据也是独立的。

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

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

相关文章

Maven和MyBatis学习总结

目录 Maven 1.Maven的概念: 2.在具体的使用中意义: 3.与传统项目引入jar包做对比: 传统方式: 在maven项目当中: 4.在创建maven项目后,想要自定义一些maven配置 5.maven项目的结构 6.maven指令的生…

AndroidTV 当贝播放器-v1.5.2-官方简洁无广告版

AndroidTV 当贝播放器 链接:https://pan.xunlei.com/s/VONXRf0g3cT0ECVt6GEsoODFA1?pwds4qv# AndroidTV 当贝播放器-v1.5.2-官方简洁无广告版

Python生成exe

其中的 -w 参数是 PyInstaller 用于窗口模式(Windowed mode),它会关闭命令行窗口的输出,这通常用于 图形界面程序(GUI),比如使用 PyQt6, Tkinter, PySide6 等。 所以: 如果你在没有…

MySql 自我总结

目录 1. 数据库约束 1.1约束类型 2. 表的设计 2.1 一对一 2.2 一对多 2.3 多对多 3. 新增 4. 查询 4.1 聚合查询 4.2 GROUP BY 4.3 HAVING 4.4 联合查询 4.5 内连接 4.5.1 内连接的核心概念 4.5.2 内连接的语法 4.5.3 ON 与 WHERE 的区别 4.6 自连接 4.6.1 定…

uni-app app 安卓和ios防截屏

首先可参考文档 uni.setUserCaptureScreen 这里需要在项目中引入这个插件 uni-usercapturescreen - DCloud 插件市场 否则会报错,在需要防止截屏录屏的页面中,加入 uni.setUserCaptureScreen({enable: false,success() {console.log(全局截屏录屏功能已禁用);},fail(err)…

【Go】windows下的Go安装与配置,并运行第一个Go程序

【Go】windows下的Go安装与配置,并运行第一个Go程序 安装环境:windows10 64位 安装版本:go1.16 windows/amd64 一、安装配置步骤 1.到官方网址下载安装包 https://golang.google.cn/dl/ 默认情况下 .msi 文件会安装在 c:\Go 目录下。可自行配…

vue3腾讯云直播 前端拉流(前端页面展示直播)

1、引入文件&#xff0c;在index.html <link href"https://tcsdk.com/player/tcplayer/release/v5.3.2/tcplayer.min.css" rel"stylesheet" /><!--播放器脚本文件--><script src"https://tcsdk.com/player/tcplayer/release/v5.3.2/t…

【MYSQL从入门到精通】数据库基础操作、数据类型

目录 一些基础操作语句 创建库名 选择要操作的数据库 删除数据库 磁盘中删除文件的原理 数据库安全的各种措置 查看MYSQL的帮助 数值类型 字符串类型 日期类型 一些基础操作语句 1.使用客户端工具连接数据库服务器&#xff1a;mysql -uroot -p 2.查看所有数据库&am…

论文阅读笔记——Multi-Token Attention

MTA 论文 在 Transformer 中计算注意力权重时&#xff0c;仅依赖单个 Q 和 K 的相似度&#xff0c;无法有效捕捉多标记组合信息。&#xff08;对于 A、B 两个词&#xff0c;单标记注意力需要分别计算两个词的注意力分数&#xff0c;再通过后处理定位共同出现的位置或通过多层隐…

vue3 antdesign table表格特定单元格背景变色

效果&#xff1a; <a-table :columns"columnsAll" :data-source"tableAllData"bordered size"middle" :scroll"{ x: 100,y: 600 }" :pagination"false"style"margin: 0 10px 10px 10px;" ><template #…

【C语言】--- 编译和链接

编译和链接 1. 翻译环境和运行环境2. 翻译环境2.1 预处理2.2 编译2.2.1 词法分析2.2.2 语法分析2.2.3 语义分析 2.3 汇编2.4 链接 3. 运行环境 1. 翻译环境和运行环境 计算机只能运行二进制指令&#xff0c;所以我们的.c的文本程序需要先翻译为二进制程序才能被计算机执行。在…

深入解析Python爬虫技术:从基础到实战的功能工具开发指南

一、引言:Python 爬虫技术的核心价值 在数据驱动的时代,网络爬虫作为获取公开数据的重要工具,正发挥着越来越关键的作用。Python 凭借其简洁的语法、丰富的生态工具以及强大的扩展性,成为爬虫开发的首选语言。根据 Stack Overflow 2024 年开发者调查,68% 的专业爬虫开发者…

前端 Vue: Cannot find module XX or its corresponding type declarations.

记一个常见错误&#xff0c;每次创建完新的vuetsvite项目&#xff0c;在配置路由的时候总会找不到vue文件&#xff0c;我用的是Webstorm&#xff0c;在设置里面修改以下设置&#xff0c;即可消除警告。

数字内容体验案例解析与行业应用

数字内容案例深度解析 在零售行业头部品牌的实践中&#xff0c;数字内容体验的革新直接推动了用户行为模式的转变。某国际美妆集团通过搭建智能内容中台&#xff0c;将产品信息库与消费者行为数据实时对接&#xff0c;实现不同渠道的动态内容生成。其电商平台首页的交互式AR试…

HBuilderX中uni-app打包Android(apk)全流程超详细打包

一、Android生成打包证书 1、Android平台签名证书(.keystore)生成指南_android 签名生成-CSDN博客&#xff08;如果不上架应用商店可以跳过&#xff0c;可以使用云端证书&#xff09; 二、打开manifest.json配置基础设置 三、配置安卓应用图标 四、配置安卓启动页图片 五、…

多模态大模型重塑自动驾驶:技术融合与实践路径全解析

目录 1、 引言&#xff1a;AI与自动驾驶的革命性融合 2、五大领先多模态模型解析 2.1 Qwen2.5-Omni&#xff1a;全模态集大成者 2.2. LLaVA&#xff1a;视觉语言理解专家 2.3. Qwen2-VL&#xff1a;长视频理解能手 2.4. X-InstructBLIP&#xff1a;跨模态理解框架 2.5. …

vue2 el-element中el-select选中值,数据已经改变但选择框中不显示值,需要其他输入框输入值才显示这个选择框才会显示刚才选中的值。

项目场景&#xff1a; <el-table-column label"税率" prop"TaxRate" width"180" align"center" show-overflow-tooltip><template slot-scope"{row, $index}"><el-form-item :prop"InquiryItemList. …

OFDM CP 对解码影响

OFDM符号间会存在ISI&#xff0c;为了解决该问题在符号间插入了循环前缀&#xff0c;可以说这个发明是OFDM能够实用的关键&#xff0c;在多径信道中CP可以有效的解决符号间干扰。3GPP中对于不同SCS 定义了不同的CP长度&#xff1a; 5G Cyclic Prefix (CP) Design -5G Physical …

Vue3.5 企业级管理系统实战(十四):动态主题切换

动态主题切换是针对用户体验的常见的功能之一&#xff0c;我们可以自己实现如暗黑模式、明亮模式的切换&#xff0c;也可以利用 Element Plus 默认支持的强大动态主题方案实现。这里我们探讨的是后者通过 CSS 变量设置的方案。 1 组件准备 1.1 修改 Navbar 组件 在 src/layo…

解决Ubuntu20.04安装ROS2的问题(操作记录)

一、ROS 系统安装版本选择 每版的Ubuntu系统版本都有与之对应ROS版本&#xff0c;每一版ROS都有其对应版本的Ubuntu版本&#xff0c;切记不可随便装。ROS 和Ubuntu之间的版本对应关系如下&#xff1a;&#xff08; 可以从这个网站查看ROS2的各个发行版本的介绍信息。&#xff…