0

0

C# 操作VeriSign代码签名文件 C#如何解析和验证数字签名文件

幻夢星雲

幻夢星雲

发布时间:2026-03-09 12:47:46

|

430人浏览过

|

来源于php中文网

原创

c# 验证 verisign(digicert)签名的 pe 文件应调用 winverifytrust api,而非 x509certificate2.loadfromsignedfile 或 verify();需正确构造 wintrust_data 结构体,设置 wtd_choice_file、wtd_stateaction_verify 等参数,验证后调用 wintrustfree 释放资源,返回 0 表示签名有效。

c# 操作verisign代码签名文件 c#如何解析和验证数字签名文件

怎么用 C# 验证 VeriSign 签名的 .exe 或 .dll 文件

VeriSign(现为 DigiCert)签发的代码签名证书本身没有特殊解析逻辑,C# 验证的是 Windows 标准的 Authenticode 签名,不是“VeriSign 专属格式”。关键在 WinVerifyTrust API 或 System.Security.Cryptography.X509Certificates + System.Deployment.Internal.CodeSigning 的组合使用。

推荐走 WinVerifyTrust 路径,它直接调用系统验证引擎,能正确识别时间戳、吊销状态、交叉证书链等真实生产环境要素;纯托管方式(如手动加载 X509Certificate2)容易漏掉策略检查或误判吊销。

  • 必须以管理员权限运行?不需要,但文件不能被其他进程独占锁住(比如正在执行的 .exe)
  • 验证失败时常见错误是 TRUST_E_NOSIGNATURE(没签名)、TRUST_E_CERT_SIGNATURE(证书链断)、TRUST_E_TIME_STAMP(时间戳不可信或过期)
  • 不要试图用 X509Certificate2.Verify() 判断代码签名有效性——它只验证书本身,不验 Authenticode 结构和策略

C# 调用 WinVerifyTrust 验证签名的最小可行代码

这是绕不开的底层调用。.NET 没有封装这个 API,必须 P/Invoke。重点不是“能不能调”,而是参数填对、结构体对齐、返回值解读准确。

核心是构造 WINTRUST_DATA 并传入 WINTRUST_FILE_INFO,目标类型设为 WINTRUST_ACTION_GENERIC_VERIFY_V2(常量值 0x00AAC562)。

Replit Ghostwrite
Replit Ghostwrite

一种基于 ML 的工具,可提供代码完成、生成、转换和编辑器内搜索功能。

下载
  • hWVTStateData 必须设为 IntPtr.Zero,否则可能触发未文档化行为
  • dwUIChoice 设为 WTD_UI_NONE,避免弹窗干扰自动化流程
  • 验证后务必调用 WintrustFree 释放内存,否则长期运行会泄漏
  • 返回值为 0 表示成功;非零需查 winerror.h 定义,例如 0x800b0109 对应 CRYPT_E_NO_REVOCATION_CHECK
var data = new WINTRUST_DATA {
    cbStruct = Marshal.SizeOf<WINTRUST_DATA>(),
    pPolicyCallbackData = IntPtr.Zero,
    pSIPClientData = IntPtr.Zero,
    dwUIChoice = WTD_UI_NONE,
    fdwRevocationChecks = WTD_REVOKE_NONE, // 生产环境建议改用 WTD_REVOKE_WHOLE_CHAIN
    dwUnionChoice = WTD_CHOICE_FILE,
    pFile = &fileInfo,
    dwStateAction = WTD_STATEACTION_VERIFY,
    hWVTStateData = IntPtr.Zero,
    pwszURLReference = null,
    dwProvFlags = WTD_SAFER_FLAG | WTD_REVOCATION_CHECK_CHAIN,
    dwUIContext = 0
};

为什么用 X509Certificate2.LoadFromSignedFile 会失败或不准

X509Certificate2.LoadFromSignedFile 只提取嵌入的证书,不验证签名有效性,也不校验哈希是否匹配。它常被误用为“验证签名”的捷径,结果是:证书能加载 ≠ 文件没被篡改 ≠ 签名当前有效。

  • 即使文件被修改过,只要签名块没动,LoadFromSignedFile 仍可能成功返回证书
  • 它不检查时间戳服务器签名,无法确认签名时证书是否有效(比如证书已过期但带有效时间戳)
  • 返回的 X509Certificate2 对象里 Verify() 方法永远只验证书链,不关联文件内容
  • 若文件含多个签名(如 dual-mode x86/x64),它只取第一个,可能跳过主签名

验证通过后,怎么安全提取签名者信息

验证只是第一步。真正要拿“谁签的”,得从签名中解析 SubjectIssuer,但注意:直接读证书 Subject 不可靠,Authenticode 允许签名者用 CounterSigner(时间戳服务)覆盖原始证书上下文。

  • 优先读签名中的 SignerInfo.Certificate,而非 WinVerifyTrust 返回的临时证书句柄
  • X509Certificate2.GetNameInfo(X509NameType.SimpleName, false) 提取可读名称,避免直接拼接 Subject 字符串(含乱序 OID)
  • 如果需要比对签名者白名单,比对 Thumbprint 比比对 CN 更稳妥,因为 CN 可能重复或被伪造
  • 注意:.NET 6+ 中 System.Security.Cryptography.Pkcs.SignedCms 可解析签名结构,但仅限于 .p7s 文件,不支持 PE 内嵌签名

真正的难点从来不在“怎么调 API”,而在于理解 Windows 如何把证书链、时间戳、吊销列表、策略标识揉进一次 WinVerifyTrust 调用里——少设一个标志位,就可能把有效签名判成无效。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
java基础知识汇总
java基础知识汇总

java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。

1564

2023.10.24

js 字符串转数组
js 字符串转数组

js字符串转数组的方法:1、使用“split()”方法;2、使用“Array.from()”方法;3、使用for循环遍历;4、使用“Array.split()”方法。本专题为大家提供js字符串转数组的相关的文章、下载、课程内容,供大家免费下载体验。

739

2023.08.03

js截取字符串的方法
js截取字符串的方法

js截取字符串的方法有substring()方法、substr()方法、slice()方法、split()方法和slice()方法。本专题为大家提供字符串相关的文章、下载、课程内容,供大家免费下载体验。

220

2023.09.04

java基础知识汇总
java基础知识汇总

java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。

1564

2023.10.24

字符串介绍
字符串介绍

字符串是一种数据类型,它可以是任何文本,包括字母、数字、符号等。字符串可以由不同的字符组成,例如空格、标点符号、数字等。在编程中,字符串通常用引号括起来,如单引号、双引号或反引号。想了解更多字符串的相关内容,可以阅读本专题下面的文章。

649

2023.11.24

java读取文件转成字符串的方法
java读取文件转成字符串的方法

Java8引入了新的文件I/O API,使用java.nio.file.Files类读取文件内容更加方便。对于较旧版本的Java,可以使用java.io.FileReader和java.io.BufferedReader来读取文件。在这些方法中,你需要将文件路径替换为你的实际文件路径,并且可能需要处理可能的IOException异常。想了解更多java的相关内容,可以阅读本专题下面的文章。

1188

2024.03.22

php中定义字符串的方式
php中定义字符串的方式

php中定义字符串的方式:单引号;双引号;heredoc语法等等。想了解更多字符串的相关内容,可以阅读本专题下面的文章。

1184

2024.04.29

go语言字符串相关教程
go语言字符串相关教程

本专题整合了go语言字符串相关教程,阅读专题下面的文章了解更多详细内容。

191

2025.07.29

JavaScript浏览器渲染机制与前端性能优化实践
JavaScript浏览器渲染机制与前端性能优化实践

本专题围绕 JavaScript 在浏览器中的执行与渲染机制展开,系统讲解 DOM 构建、CSSOM 解析、重排与重绘原理,以及关键渲染路径优化方法。内容涵盖事件循环机制、异步任务调度、资源加载优化、代码拆分与懒加载等性能优化策略。通过真实前端项目案例,帮助开发者理解浏览器底层工作原理,并掌握提升网页加载速度与交互体验的实用技巧。

59

2026.03.06

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
C# 教程
C# 教程

共94课时 | 11万人学习

C 教程
C 教程

共75课时 | 5.3万人学习

C++教程
C++教程

共115课时 | 21.2万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号