2404d,d符号表示dip1045

原文

符号表示

概述

符号是本地语言功能最终表示.在编译器或语言级别,不准确表示它们可能会导致链接失败.这些问题可能是令人沮丧的重要来源,甚至可能导致人们认为无法实现方法.

DIP的目的是解决和纠正各种跨平台和目标的常见共享库链接错误.

理由

对不熟悉链接器的人来说,理解和解决这些错误可能令人生畏,一般会导致寻求帮助.

仅限于静态库和独立可执行文件使用语言时,D中的很大一部分符号的表示问题仍模糊不清.但是,此配置并不能满足所有用户的不同需求和偏好.

需要更灵活的如涉及共享库的二进制配置.

把插件整合D中,为其他语言创建D插件或开发可替换二进制文件等用例,都说明了涉及D的共享库有益的场景.为了确保D一致性和易用性,增强语言的符号表示方法至关重要.

主要焦点应放在模块级别.默认,了解模块是在当前二进制文件内部还是外部,是解决许多链接器问题的关键.这些基本知识为进一步改进和修改奠定了基础.

前期工作

2016年,BenjaminThautDConf上发表了题为"D的导入出业务"的演讲,提出了改进D语言特别是导出和共享库功能的提案.

虽然该DIP并非直接源自Thaut的提议,且具体实现细节不同,但它在导出的作用上得出了类似结论.

即,它认识到,为了在D中有效利用注解,导出不应仅按可见性限定器使用.

Thaut方法的一个显著区别是,他建议设置dllimport开关为"all",来指示符号是在DllImport模式.

然而,根据D社区随后几年中在共享库方面的丰富经验,很明显,该方法经常导致链接器错误.因此,需要一个更细致,更细粒度方法来有效解决这些问题.

描述

变更基石

本节概述未来更新关键更改,以确保无误编译.

1,外部导入路径开关:

引入了新的extI编译器标志.它的功能类似I开关,按当前正在编译二进制文件外部,指定模块.

最好,构建管理器可自动执行此过程,对共享库关联的模块,用依赖关系的知识-extI替换-I.

2,二进制外模块:

(通过-extI开关)按外部标识的模块,对所有非模板化域,有个隐式的extern属性.

3,导入符号:

要按DllImport模式导入符号,需要同时有exportextern注解.此时,是否存在函数体不重要.

以下各节,考虑模板在与共享库,D接口生成器,不同的导出注解内联链接时的可靠性.

模板的可靠性

因为实例化模板的假设,模板可能会导致链接失败.因此,确保从模板继承的符号不会自动符合导出条件或为DllImport模式.

相反,按二进制文件的外部标识这些符号时,应重新实例化,并在设置了适当的重复标志后,放入目标二进制文件中.

这要求在每个模块基础上,应用外部导入路径开关.

为了优化生成代码,编译器可用"固定"策略.如果在其声明的同一模块中,非模板化符号引用了模板实例化(a),则按"固定"对待a.

固定扩展到在模板中未封装变量声明(包括全局变量)及函数参数和返回类型.

固定模板及其关联的符号后,在这些符号二进制文件的外部时,编译器可自行决定,是否省略这些符号的生成代码.此时,编译器应遵循指定的导出DllImport符号模式,以确保高效且无错误的链接.

D接口生成器

D接口生成器是D编译器提供的导出工具,通过省略D文件中的符号体来方便创建.di文件.
当与C解析器(ImportC)一起使用,以生成C库绑定时,此工具特别有用.

但是,当前实现的一个显著局限性是在导出过程中,它无法准确遵循符号模式.

为此,提出了以下修改建议:
1,生成器,不应自动添加extern属性到符号中.
2,仅当按导出特定模块设置可见性覆盖开关时,生成器才可把导出(export)属性添加到所有非模板化域.

这些调整旨在与前面概述变更基石相结合.用来确保:
1,对按export标记的代码基,.di生成器限制引入其他extern属性.
2,对静态库或目标文件,生成的.di文件可同-I标准导入路径开关一起使用.
3,或,在处理共享库依赖项时,可把外部导入路径开关-extI应用至.di文件.

导出符号模式方法

每个符号可有三个不同模式之一:Internal,DllExportDllImport.

1,内部模式:这是符号的默认模式.不管在哪个模块中定义,同一个二进制文件中的其他符号都可访问内部符号.
2,DllExport模式:设置后,此模式指示,不仅自己的二进制文件,且外部二进制文件都可访问符号.这对编译打算跨不同二进制文件使用的符号至关重要.
3,DllImport模式:此模式告诉编译器符号当前二进制文件的外部.因此,编译器生成允许在运行时访问此符号代码.

确定符号适当模式策略如下:
1,使用export正注解:此方法表示D编译器的默认行为,即对导出,用export关键字来显式标记符号.
2,用可见性覆盖开关负注解:默认,不会导出未用导出(export)注解的符号.可用可见性覆盖开关来反转此默认,来强制导出所有符号.对未显式标记导出,但需要导出其符号的特别有用.

3,多步构建边角注解:这是个特定注解,适合需要(具体根据构建步骤)确定符号Internal还是DllImport的方案.
在复杂或冲突的符号模式多步构建过程中非常有用.

导出注解

按参数取标识,来增强导出属性.按版本解释标识,需要以下语法更改:

VisibilityAttribute:
-   export
Attribute:
+   export
+   export ( Identifier )

此参数的功能根据标识是否活动(由version=ident激活).活动时,除非正在编译,符号内部模式.相反,当标识非活动时,类似有外部(extern)注解,符号按DllImport模式.

为了标准化标识使用,在D规范引入了三个新的版本前缀,编译器自动提供了libc,DRuntimePhobos实例:
前缀为Have_,InBinary_Compiling_.这三个后缀将是个逻辑包.

1,Have_前缀:指示在链接过程中,指定逻辑包可用作依赖项.
2,InBinary_前缀:表示当前正在编译二进制文件中有指定逻辑包的符号.
3,Compiling_前缀:表示正在编译指定逻辑包.

这些前缀结合逻辑包后缀,可处理按内部而不是外部的符号的极端场景.

此外,会更新D接口生成器,以支持插入InBinary_version参数.需要此更新才能通过D模块准确表示C文件.此处未介绍确定后缀的具体机制,它可能依赖于C预处理器功能.

正表示法

D中的export属性用于按DllExport来注解符号.要表示已导出符号,应直接应用此属性D符号.

导出注解不能按可见性限定器使用.这样处理它,可能会无意中暴露内部实现细节,可能会在语言级别,导致外部实体代码基不安全的操作.

在(如构,类,联或模块)封装单元中,如果按export标记任一成员,则也还必须导出所有关联生成的符号(如TypeInfo,__initZ,opCmp等,但不包括ModuleInfo).

导出关联符号,却无法导出这些生成符号时,则可能导致链接器错误,则如果不借助链接器脚本,可能无法解决这些错误.

默认,所有符号都是隐藏的.要按隐藏显式标记符号,请用core.attributes中提供的用户定义属性(UDA).该方法比在要导出每个符号注解导出更有效.

相反,它允许你在域级注解,并简单禁止那些不打算导出的符号.

否定符号

设置可见性覆盖开关后,默认导出所有符号.
覆盖默认设置,并按隐藏显式指定符号,应使用core.attributes中提供的UDA.

在适当对DRuntimePhobos库,用导出全面注解并测试前,它们依赖此符号可见性管理方法.

内联

考虑一个二进制外模块:

pragma(inline, true)
export extern void inlineable() {noInline;
}
@hidden void noInline();

在此例中,按隐藏标记noInline函数,因此无法访问内联.如果可跨二进制边界内联inlineable,因为noInline不可用,这会导致链接器错误.

为了避免此类错误,当这些函数引用未导出的符号时,必须指示编译器不要内联二进制外模块的函数.

用例

本节介绍场景说明了本DIP中提议的修改实际影响和应用.
第一个情况与需要细致控制导出符号用户有关.
第二个重点是,无需调整符号的DllImport状态,方便整合DRuntime二进制文件.

正注解

用例概述了用导出正注解和可选使用.di生成器的过程.它使用窗口文件命名约定展示.
目录布局:

dependency/source/library.d
dependency/imports/library.di
dependency/library.dll
dependency/library.lib
dependency/library.exp
executable/source/app.d
executable/app.exe
executable/library.dll

dependency/source/library.d的源:

module library;
export void myLibraryFunction() {import std.stdio;writeln("Hello from my libraries function!");
}

生成的dependency/source/library.di:

module library;
export void myLibraryFunction();

executable/source/app.d:

module app;
void main() {import library;myLibraryFunction();
}

使用共享库:

dmd of=dependency/library.dll shared Hd=dependency/imports dependency/source/library.d
cp dependency/library.dll executable/library.dll
dmd of=executable/app.exe extI=dependency/imports executable/source/app.d dependency/library.lib

使用静态库:

dmd of=dependency/library.lib Hd=dependency/imports dependency/source/library.d
dmd of=executable/app.exe I=dependency/imports executable/source/app.d dependency/library.lib

使用共享库和静态库的主要区别在于,编译前者时包含-shared,而在链接后者时用-I替换-extI.

二进制中的DRuntime

用例侧重于整合DRuntime二进制文件中的过程.不考虑Phobos等其他库,只是个说明性示例,因为,在静态或共享DRuntime间选择的开关是编译器相关的.

对此用例,考虑把DRuntime放入生成的二进制文件中会怎样.不考虑其他库(如Phobos),且选择Druntime静态的还是共享的开关是编译器相关的,因此它只是演示了它的流程.

目录布局:

dependency/source/dependency.d
dependency/dependency.lib
mydll/source/api.d
mydll/mydll.dll
mydll/mydll.lib
mydll/mydll.exp

dependency/source/dependency.d的源:

module dependency;
void myLibraryFunction() {foreach(m; ModuleInfo) {//非模板化的符号都适合此例!}
}

mydll/source/api.d的源:

module api;
export void api() {import dependency;myLibraryFunction();
}

编译命令:

dmd of=dependency/dependency.lib lib libdruntime=static dependency/source/dependency.d
dmd of=mydll/mydll.dll shared I=dependency/source libdruntime=static mydll/source/api.d dependency/dependency.lib

特别令人感兴趣的是-lib-druntime=static的行为.
在典型的D编译中,会自动添加导入路径静态/导入库.使用此DIP,要指定共享DRuntime版本,则用-extI替换-I,用导入库替换静态DRuntime库,并按externalOnly设置覆盖dllimport.

该方法大大简化了共享和静态DRuntime之间的区别,可能允许编译器配置文件掩盖这些差异.

目前,未使用export注解DRuntime.如果是,则不必添加-dllimport=externalOnly,从而降低链接器试访问未导出符号的风险.

DIP旨在消除在当前环境中使用有共享DRuntime共享库时很常见的(如LNK4217)链接器警告.

重大更改和弃用

导出不再表示"超级公开"的可见性.可能会影响现有代码中符号的可见性和可访问性.

为了缓解此潜在问题,可在使用导出前的行中添加public:调整代码基.此方法向后兼容,确保它与当前和未来编译器一起正常运行.

对那些喜欢保留传统导出行为的人,建议这样.

DIP中建议的所有其他修改基本上都是通过-extI编译器开关选入的.

参考

在C和C++中,一般通过宏预处理器交换的属性来选择DllExportDllImport.它虽然实用,但很麻烦,需要配置每个库.

Rust使用一个包括库名,可通过命令行参数调整的link属性,以在编译过程中切换默认符号模式.此DIP引入了版本命名约定InBinary_,来指定包是在二进制文件内部还是外部.

与配音使用的现有Have_约定一致,并利用了D的现有机制.

二进制外:不在当前编译二进制文件中的符号.如果按DllExport模式设置,并通过DllImport访问,则当前正在编译的二进制文件可在运行时访问它.

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

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

相关文章

数位DP模型

目录 度的数量思路代码实现暴力解法数位DP 数字游戏代码实现 Windy数代码实现 数字游戏 II代码实现 不要62代码实现 恨7不成妻代码实现 度的数量 题目描述: 求给定区间 [ X , Y ] [X,Y] [X,Y] 中满足下列条件的整数个数:这个数恰好等于 K K K 个互不…

安装Schedule库的方法最终解答!_Python第三方库

安装Python第三方库Schedule 我的环境:Window10,Python3.7,Anaconda3,Pycharm2023.1.3 Schedule库 Schedule 是一个轻量级、功能强大而灵活的任务调度工具库,用于在指定的时间间隔内执行任务。为用户提供了简单易用的…

kali使用msf+apkhook520+cploar实现安卓手的攻击

学习网络安全的过程中,突发奇想怎么才能控制或者说是攻击手机 边找工作边实验 话不多说启动kali 一、使用msfapktool生成简单的木马程序 首先使用kali自带的msfvenom写上这样一段代码 选择安卓 kali的ip 一个空闲的端口 要输出的文件名 msfvenom -p android/met…

kafka(五)——消费者流程分析(c++)

概念 ​ 消费者组(Consumer Group):由多个consumer组成。消费者组内每个消费者负责消费不同分区的数据,一个分区只能由一个组内消费者消费;消费者组之间互不影响。所有的消费者都属于某个消费者组,即消费者…

计算机是怎么工作的7

内存分配——内存管理(Memory Mangae) 进程如何管理内存,其实是一个非常复杂的事情 核心结论:每个进程的内存,是彼此独立,互不干扰的 通常情况下,进程A不能直接访问进程B的内存; 为了系统的稳定性 如果…

今日arXiv最热大模型论文:Dataverse,针对大模型的开源ETL工具,数据清洗不再难!

引言:大数据时代下的ETL挑战 随着大数据时代的到来,数据处理的规模和复杂性不断增加,尤其是在大语言模型(LLMs)的开发中,对海量数据的需求呈指数级增长。这种所谓的“规模化法则”表明,LLM的性…

HTML - 请你说一下如何阻止a标签跳转

难度级别:初级及以上 提问概率:55% a标签的默认语义化功能就是超链接,HTML给它的定位就是与外部页面进行交流,不过也可以通过锚点功能,定位到本页面的固定id区域去。但在开发场景中,又避免不了禁用a标签的需求,那么都有哪些方式可以禁用…

【就近接入,智能DNS-Geo DNS ,大揭秘!】

做过后端服务或者网络加速的小伙伴,可能或多或少都听说过,智能DNS或者Geo DNS,就是根据用户的位置,返回离用户最近的服务节点,就近接入,以达到服务提速的效果。 那么大家有没想过,这个背后的原理…

成绩分析 蓝桥杯 java

成绩分析 小蓝给学生们组织了一场考试,卷面总分为 100 分,每个学生的得分都是一个 0 到 100 的整数。 请计算这次考试的最高分、最低分和平均分。 输入格式 输入的第一行包含一个整数 n,表示考试人数。 接下来 n 行,每行包含一…

C++:日期类的实现 const修饰 取地址及const取地址操作符重载(类的6个默认成员函数完结篇)

一、日期类的实现 根据之前赋值运算符重载逻辑&#xff0c;我们现在来实现完整的日期类。 1.1 判断小于 上篇博客已经实现: bool operator<(const Date& d) {if (_year < d._year){return true;}else if (_year d._year){if (_month < d._month){return true…

总结C/C++中程序内存区域划分

C/C程序内存分配的几个区域&#xff1a; 1. 栈区&#xff08;stack&#xff09;&#xff1a;在执行函数时&#xff0c;函数内局部变量的存储单元都可以在栈上创建&#xff0c;函数执行结束时这些存储单元自动被释放。栈内存分配运算内置于处理器的指令集中&#xff0c;效率很⾼…

KISS 原则和 YAGNI原则

设计模式专栏&#xff1a;http://t.csdnimg.cn/3a25S 目录 1.引言 2.代码并非行数越少越简单 3.代码复杂不一定违反 KISS 原则 4.如何写出满足 KISS 原则的代码 5.YAGNI原则和 KISS 原则的区别 1.引言 KISS原则的英文描述有3种版本:Keep It Simple and Supid、 keep It Sh…

OpenHarmony开发技术:【国际化】实例

国际化 如今越来的越多的应用都走向了海外&#xff0c;应用走向海外需要支持不同国家的语言&#xff0c;这就意味着应用资源文件需要支持不同语言环境下的显示。本节就介绍一下设备语言环境变更后&#xff0c;如何让应用支持多语言。 应用支持多语言 ArkUI开发框架对多语言的…

Autowired和Resource的关系?

相同点 对于下面的代码来说&#xff0c;如果是Spring容器的话&#xff0c;两个注解的功能基本是等价的&#xff0c;他们都可以将bean注入到对应的field中 Autowired private Bean beanA; Resource private Bean beanB; 不同点 byName和byType匹配顺序不同 Autowired在获取…

TypeScript—详解、小案例(配合源代码)

简介&#xff1a;TypeScript是微软开发的 JavaScript 的超集&#xff0c;TypeScript兼容JavaScript&#xff0c;可以载入JavaScript代码然后运行。TypeScript与JavaScript相比进步的地方 包括&#xff1a;加入注释&#xff0c;让编译器理解所支持的对象和函数&#xff0c;编译器…

Web路径专题

文章目录 1.资源定位1.前置条件上下文路径设置 2.上下文路径介绍重点说明 3.资源定位方式资源路径 上下文路径 资源位置a.html定位C.java定位 4.浏览器和服务器解析的区别1.浏览器解析/&#xff08;地址变化&#xff09;2.服务器解析/&#xff08;地址不变&#xff09; 5.带/…

备份工具--kopia

kopia 备份测试 安装minio wget https://dl.min.io/server/minio/release/linux-amd64/minio 创建目录 mkdir -p /data/myapp/minio mkdir -p /data/minidata 配置启动 cat start_minio.sh nohup /data/myapp/minio/minio server --console-address :“9090” /data/minidata…

数据流重定向

数据流重定向 标准输入 0 标准输出 1 标准错误输出 2 输出 > 标准输出 1>标准错误输出 2>标准输出标准错误输出 2>&1 或 &> 输入 < 标准输入 0<eof输入 cat > catfile <<“eof” 双向重定向 tee [command] | tee -a [file] [comma…

Spring和Spring MVC和MyBatis面试题

面试题1&#xff1a;请简述Spring、Spring MVC和MyBatis在整合开发中的作用&#xff1f; 答案&#xff1a; Spring&#xff1a;是一个轻量级的控制反转&#xff08;IoC&#xff09;和面向切面&#xff08;AOP&#xff09;的容器框架。它提供了强大的依赖注入功能&#xff0c;…

git学习 1

打开自己想要存放git仓库的文件夹&#xff0c;右键打开git bush&#xff0c;用git init命令建立仓库 用 ls -a(表示全都要看&#xff0c;包括隐藏的)可以看到git仓库 也可以用 git clone 接github链接&#xff08;点code选项里面会给链接&#xff0c;结尾是git的那个&#xf…