如何避免用waveformRecord复制数组
这里描述如何使用数组字段内存管理特定。这使得数组数据能够被移入和移出waveform,aai和aao类型的值字段(BPTR)。
使用这种特定包括用另一个(用户分配的)字段替代存储在BPTR字段的指针。基本规则是:
1、BPTR以及它当前指向的内存,只能在这个记录被锁定时才能被访问。
2、NELM不可以被更改。
3、BPTR必须总是指向一段足够大的内存,来装下最大的元素数目(由NELM字段确定)。
规则1意味着从一个设备支持函数或者手动调用dbScanLock()后读,写或者解引用BPTR字段才安全。规则3意味着BPTR不能被设置NULL,并且在替代BPTR时,替换必须为最差情况分配足够大空间。一个外部客户端能够在几乎任何时间写入一个最大NELM个元素的数组。
以下示例演示了为waveform缓存使用了自定义分配。
以下是一个waveform记录设备支持的源代码devwf.c:
#include "errlog.h"
#include "initHooks.h"
#include "ellLib.h"
#include "devSup.h"
#include "dbDefs.h"
#include "dbAccess.h"
#include "cantProceed.h"
#include "epicsTypes.h"
#include "epicsMutex.h"
#include "epicsEvent.h"
#include "epicsThread.h"
#include "menuFtype.h"
#include "dbScan.h"
#include "alarm.h"
#include "recGbl.h"
#include "waveformRecord.h"
#include "epicsExport.h"static ELLLIST allPvt = ELLLIST_INIT;struct devicePvt {ELLNODE node;/* 同步访问这个结构体 */epicsMutexId lock;/* 当需要另一个更新时,唤醒这个worker */epicsEventId wakeup;/* 当另一个更新可用时,通知scanner线程。 */IOSCANPVT scan;/* 下次更新 */void *nextBuffer;epicsUInt32 maxbytes, numbytes;
};static long init(int phase);
static long init_record(dbCommon *pcommon);
static long get_iointr_info(int dir, dbCommon *prec, IOSCANPVT *scan);
static long read_wf(waveformRecord *prec);
static void worker(void*);
static void startWorkers(initHookState);wfdset devWfZeroCopy = {{5, NULL, init, init_record, get_iointr_info},read_wf
};epicsExportAddress(dset, devWfZeroCopy);static long init(int phase)
{if(phase!=0){return 0;}initHookRegister(&startWorkers);return 0;
}static long init_record(dbCommon *pcommon)
{struct devicePvt *priv;waveformRecord * prec = (waveformRecord *)pcommon;if(prec->ftvl!=menuFtypeSHORT) {errlogPrintf("%s.FTVL must be set to SHORT for this example\n", prec->name);return 0;}/* 清理由记录支持分配的数组。
* 由于我们使用lcalloc()/free(),不必要,
* 但用其它方法分配时,不需要。
*/free(prec->bptr);prec->bptr = callocMustSucceed(prec->nelm, dbValueSize(prec->ftvl), "first buf");// struct devicePvtpriv = callocMustSucceed(1, sizeof(*priv), "init_record devWfZeroCopy");priv->lock = epicsMutexMustCreate();priv->wakeup = epicsEventMustCreate(epicsEventFull);scanIoInit(&priv->scan);priv->maxbytes = prec->nelm*dbValueSize(prec->ftvl);ellAdd(&allPvt, &priv->node);prec->dpvt = priv;return 0;
}static void startWorkers(initHookState state)
{ELLNODE *cur;/* 直到调用scanIoRequest安全,才启动worker线程 */if(state!=initHookAfterInterruptAccept){return;}for(cur=ellFirst(&allPvt); cur; cur=ellNext(cur)){struct devicePvt *priv = CONTAINER(cur, struct devicePvt, node);epicsThreadMustCreate("wfworker",epicsThreadPriorityHigh,epicsThreadGetStackSize(epicsThreadStackSmall),&worker, priv);}
}static void worker(void* raw)
{struct devicePvt *priv=raw;void *buf = NULL;epicsUInt32 nbytes = priv->maxbytes;while(1) {epicsThreadSleep(1.0);if(!buf) {/* 为之后(本地)使用分配并且初始化一个新缓存 */size_t i;epicsInt16 *ibuf;buf = callocMustSucceed(1, nbytes, "buffer");ibuf = (epicsInt16*)buf;for(i=0; i<nbytes/2; i++)// 字节数除以2,就是元素数目{ibuf[i] = rand();}}/* 当扫描是'I/O Intr'时,等待事件信号,* 而当记录是周期扫描时,等待超时*/// 等待事件priv-wakeup 1.0秒,如果返回事件出错,挂起正在运行的线程if(epicsEventWaitWithTimeout(priv->wakeup, 1.0)==epicsEventError) {cantProceed("worker encountered an error waiting for wakeup\n");}// 获取互斥锁priv->lockepicsMutexMustLock(priv->lock);if(!priv->nextBuffer) {/* 使得本地缓存对read_wf函数可用 */priv->nextBuffer = buf;buf = NULL;priv->numbytes = priv->maxbytes;scanIoRequest(priv->scan);}epicsMutexUnlock(priv->lock);}
}static long get_iointr_info(int dir, dbCommon *prec, IOSCANPVT *scan)
{struct devicePvt *priv=prec->dpvt;if(!priv){return 0;}*scan = priv->scan;
/* 当这个线程被放入I/O扫描列表时,唤醒这个worker*/if(dir==0){// 发出事件信号epicsEventSignal(priv->wakeup);}return 0;
}static long read_wf(waveformRecord *prec)
{struct devicePvt *priv=prec->dpvt;if(!priv){return 0;}epicsMutexMustLock(priv->lock);if(priv->nextBuffer) {/* 更新可用,所以占用它 */if(prec->bptr){// 指向数据区的指针有效,释放此数据区free(prec->bptr);prec->bptr = priv->nextBuffer; /* 切换指针指向,没有复制内存 */priv->nextBuffer = NULL;prec->nord = priv->numbytes / dbValueSize(prec->ftvl);// 计算实际元素数目epicsEventSignal(priv->wakeup);// 工作线程又可以运行}epicsMutexUnlock(priv->lock);assert(prec->bptr);}return 0;
}/*typedef long (*DEVSUPFUN)(void *);typedef struct dset { //设备支持入口表long number; //支持例程数目DEVSUPFUN report; // 打印报告DEVSUPFUN init; // init支持层DEVSUPFUN init_record; // 特定记录的init设备DEVSUPFUN get_ioint_info; // 获取io中断信息//以下时取决于记录的其它函数
} dset;
*/
编写以上源文件的devwfsup.dbd文件:
device(waveform,CONSTANT,devWfZeroCopy,"Zero Copy Demo")
在Makefile文件中添加以上两个文件:
wfDevSup_DBD += devwfsup.dbd
wfDevSup_SRCS += devwf.c
添加一个只含一个waveform记录的db文件:
record(waveform, "$(TEST):wf") {field(DTYP, "Zero Copy Demo")field(FTVL, "SHORT")field(NELM, "6")field(SCAN, "I/O Intr")
}
更改启动文件st.cmd:
dbLoadRecords("db/wf.db","TEST=TEST")
启动这个IOC程序,并且查看加载的记录实例:
epics> dbl
TEST:wf
用cainfo和camonitor通道访问进行测试:
root@orangepi4-lts:/usr/local/EPICS/base/modules/database/src/std/dev# cainfo TEST:wf
TEST:wfState: connectedHost: 192.168.50.184:5064Access: read, writeNative data type: DBF_SHORTRequest type: DBR_SHORTElement count: 6
root@orangepi4-lts:/usr/local/EPICS/base/modules/database/src/std/dev# camonitor TEST:wf
TEST:wf 2023-08-11 04:38:32.964899 6 11358 -32429 28512 -25174 -11566 -30030
TEST:wf 2023-08-11 04:38:33.965274 6 27856 32389 -2566 -12204 2520 14133
TEST:wf 2023-08-11 04:38:34.965634 6 -19224 -17196 29030 19074 -11420 -20584
TEST:wf 2023-08-11 04:38:35.966003 6 -17959 -21592 -25977 -20875 -3483 10352
TEST:wf 2023-08-11 04:38:36.966368 6 29274 10378 -18625 -19358 -18816 4905
TEST:wf 2023-08-11 04:38:37.966640 6 28740 -7458 -27524 -8283 -32631 26446
TEST:wf记录中数组值每秒钟随机变化一次。
相关文章:
如何避免用waveformRecord复制数组
这里描述如何使用数组字段内存管理特定。这使得数组数据能够被移入和移出waveform,aai和aao类型的值字段(BPTR)。 使用这种特定包括用另一个(用户分配的)字段替代存储在BPTR字段的指针。基本规则是: 1、BPTR以及它当前指向的内存,只能在这个…...

RocketMQ 延迟消息
RocketMQ 延迟消息 RocketMQ 消费者启动流程 什么是延迟消息 RocketMQ 延迟消息是指,生产者发送消息给消费者消息,消费者需要等待一段时间后才能消费到。 使用场景 用户下单之后,15分钟未支付,对支付账单进行提醒或者关单处理…...

Dex文件混淆(一):BlackObfuscator
Dex文件混淆(一):BlackObfuscator 首发地址:http://zhuoyue360.com/crack/105.html 文章目录 Dex文件混淆(一):BlackObfuscator1. 前言2.小试牛刀3. 参考学习1. dex2jar源码简析2. BlackObfuscator简析1. 控制流平坦化1. 控制流平坦化基本介绍 2. Dex解析…...
Linux下编译arm 32 出错(/bin/bash: arm-none-linux-gnueabi-gcc: command not found )
一、arm-none-linux-gnueabi-gcc不能再64位系统下下编译ARM的32位库的问题解决方法如下: sudo apt-get install lib32stdc6 sudo apt-get install lib32ncurses5 sudo apt-get install lib32z1 二、交叉编译工具没有写入环境变量或写错,重新写入环境变量…...
最近遇到的两个小问题总结:git问题和node问题
这两个问题都是我帮别人看问题的解决的,在windows系统上遇到的: 1、git没有配置全局变量 在使用git的时候,报’git‘不是内部或外部命令,也不是可运行的程序。然后再在其他文件下面试一下(git --version)…...
Java # Spring(1)
一、概念 1、核心技术:依赖注入(DI),AOP,事件(events),资源,i18n,验证,数据绑定,类型转换,SpEL。 2、测试:模…...
SCL更换阿里数据源
问题: zabbix安装前端环境报错 yum install zabbix-web-mysql-scl zabbix-apache-conf-scl -y 报错:Could not retrieve mirrorlist http://mirrorlist.centos.org/ 能上网 但是不能ping通http://mirrorlist.centos.org/ 解决: 修改repo数…...

【web逆向】全报文加密流量的去加密测试方案
aHR0cHM6Ly90ZGx6LmNjYi5jb20vIy9sb2dpbg 国密混合 WEB JS逆向篇 先看报文:请求和响应都是全加密,这种情况就不像参数加密可以方便全文搜索定位加密代码,但因为前端必须解密响应的密文,因此万能的方法就是搜索拦截器,…...

Django实现音乐网站 ⑼
使用Python Django框架制作一个音乐网站, 本篇主要是后台对专辑、首页轮播图原有功能的基础上进行部分功能实现和显示优化。 目录 专辑功能优化 新增编辑 专辑语种改为下拉选项 添加单曲优化显示 新增单曲多选 更新歌手专辑数、专辑单曲数 获取歌手专辑数 保…...

【脚踢数据结构】
(꒪ꇴ꒪ ),Hello我是祐言QAQ我的博客主页:C/C语言,Linux基础,ARM开发板,软件配置等领域博主🌍快上🚘,一起学习,让我们成为一个强大的攻城狮!送给自己和读者的一句鸡汤🤔&…...

uni-app使用vue语法进行开发注意事项
目录 uni-app 项目目录结构 生命周期 路由 路由跳转 页面栈 条件编译 文本渲染 样式渲染 条件渲染 遍历渲染 事件处理 事件修饰符 uni-app 项目目录结构 组件/标签 使用(类似)小程序 语法/结构 使用vue 具体项目目录如下: 生命…...

数据结构---B树
目录标题 B-树的由来B-树的规则和原理B-树的插入分析B-树的插入实现准备工作find函数insert中序遍历 B-树的性能测试B-树的删除B树B树的元素插入B*树的介绍 B-树的由来 在前面的学习过程中,我们见过很多搜索结构比比如说顺序查找,二分查找,搜…...
c++11以后c++标准库定义的固定位宽的整数类型(Fixed width integer types)
Fixed width integer types Fixed width integer types (since C11) - cppreference.com 相关定义文件如下: Windows系统MSVC: Microsoft Visual Studio\2022\Community\VC\Tools\MSVC\14.33.31629\include\cstdint Linux系统GCC: gcc\libstdc-v3\include\c_g…...
Object.values()
Object.values() 是ES2017新增的一个对象方法,它可以将一个对象自身的所有可枚举属性值,组成一个数组返回。 基本语法: Object.values(obj)示例: jsCopy codeconst obj {foo: bar,baz: 42 };Object.values(obj); // [bar, 42]Object.values()的特点: 只返回可枚举的属性值…...

Oracle 开发篇+Java调用OJDBC访问Oracle数据库
标签:JAVA语言、Oracle数据库、Java访问Oracle数据库释义:OJDBC是Oracle公司提供的Java数据库连接驱动程序 ★ 实验环境 ※ Oracle 19c ※ OJDBC8 ※ JDK 8 ★ Java代码案例 package PAC_001; import java.sql.Connection; import java.sql.ResultSet…...

linux 查询后台任务及杀掉进程
查看后台任务命令 jobs -l删除后台进程命令 kill -9 28719...

【Vue3 博物馆管理系统】使用Vue3、Element-plus菜单组件构建前台用户菜单
系列文章目录 第一章 定制上中下(顶部菜单、底部区域、中间主区域显示)三层结构首页 第二章 使用Vue3、Element-plus菜单组件构建菜单 [第三章 使用Vue3、Element-plus菜单组件构建轮播图] [第四章 使用Vue3、Element-plus菜单组件构建组图文章] 文章目…...
Windows 11清除无效、回收站、过期、缓存、补丁更新文件
Windows 11与之前的Windows版本类似,也需要定期清理无效、垃圾、过期、缓存文件来保持系统性能和存储空间的优化。以下是在Windows 11中进行这些清理操作的一些建议方法: 磁盘清理工具 Windows 11内置了磁盘清理工具,可以帮助你删除临时文件…...

栈和队列详解(2)
目录 一、什么是队列? 二、创建一个我们自己的队列 1.前置准备 1.1需要的三个文件 1.2结构体的创建和头文件的引用 2.接口的实现 2.1初始化队列 2.2入队 2.3队列元素个数和判空 2.4取队头元素和队尾元素 2.5出队 2.6摧毁队列 2.7测试接口 三、所有代码 1.…...

EMC传导干扰滤波电路设计
1.EMC概念 2.EMC 传导干扰详解 EMC传导滤波电路的设计--传导干扰详解 3.EMC 传导干扰的测量方法 4.EMC 滤波电路设计 5.浪涌抑制电路设计 6.开关电源的安全要求 7.当前开关电源灯的应用...

Debian系统简介
目录 Debian系统介绍 Debian版本介绍 Debian软件源介绍 软件包管理工具dpkg dpkg核心指令详解 安装软件包 卸载软件包 查询软件包状态 验证软件包完整性 手动处理依赖关系 dpkg vs apt Debian系统介绍 Debian 和 Ubuntu 都是基于 Debian内核 的 Linux 发行版ÿ…...

visual studio 2022更改主题为深色
visual studio 2022更改主题为深色 点击visual studio 上方的 工具-> 选项 在选项窗口中,选择 环境 -> 常规 ,将其中的颜色主题改成深色 点击确定,更改完成...
c++ 面试题(1)-----深度优先搜索(DFS)实现
操作系统:ubuntu22.04 IDE:Visual Studio Code 编程语言:C11 题目描述 地上有一个 m 行 n 列的方格,从坐标 [0,0] 起始。一个机器人可以从某一格移动到上下左右四个格子,但不能进入行坐标和列坐标的数位之和大于 k 的格子。 例…...
C++ 基础特性深度解析
目录 引言 一、命名空间(namespace) C 中的命名空间 与 C 语言的对比 二、缺省参数 C 中的缺省参数 与 C 语言的对比 三、引用(reference) C 中的引用 与 C 语言的对比 四、inline(内联函数…...

uniapp微信小程序视频实时流+pc端预览方案
方案类型技术实现是否免费优点缺点适用场景延迟范围开发复杂度WebSocket图片帧定时拍照Base64传输✅ 完全免费无需服务器 纯前端实现高延迟高流量 帧率极低个人demo测试 超低频监控500ms-2s⭐⭐RTMP推流TRTC/即构SDK推流❌ 付费方案 (部分有免费额度&#x…...
拉力测试cuda pytorch 把 4070显卡拉满
import torch import timedef stress_test_gpu(matrix_size16384, duration300):"""对GPU进行压力测试,通过持续的矩阵乘法来最大化GPU利用率参数:matrix_size: 矩阵维度大小,增大可提高计算复杂度duration: 测试持续时间(秒&…...

什么是Ansible Jinja2
理解 Ansible Jinja2 模板 Ansible 是一款功能强大的开源自动化工具,可让您无缝地管理和配置系统。Ansible 的一大亮点是它使用 Jinja2 模板,允许您根据变量数据动态生成文件、配置设置和脚本。本文将向您介绍 Ansible 中的 Jinja2 模板,并通…...
Java线上CPU飙高问题排查全指南
一、引言 在Java应用的线上运行环境中,CPU飙高是一个常见且棘手的性能问题。当系统出现CPU飙高时,通常会导致应用响应缓慢,甚至服务不可用,严重影响用户体验和业务运行。因此,掌握一套科学有效的CPU飙高问题排查方法&…...
Java数值运算常见陷阱与规避方法
整数除法中的舍入问题 问题现象 当开发者预期进行浮点除法却误用整数除法时,会出现小数部分被截断的情况。典型错误模式如下: void process(int value) {double half = value / 2; // 整数除法导致截断// 使用half变量 }此时...

AI+无人机如何守护濒危物种?YOLOv8实现95%精准识别
【导读】 野生动物监测在理解和保护生态系统中发挥着至关重要的作用。然而,传统的野生动物观察方法往往耗时耗力、成本高昂且范围有限。无人机的出现为野生动物监测提供了有前景的替代方案,能够实现大范围覆盖并远程采集数据。尽管具备这些优势…...