当前位置: 首页 > news >正文

翻译《The Old New Thing》- The importance of the FORMAT_MESSAGE_IGNORE_INSERTS flag

The importance of the FORMAT_MESSAGE_IGNORE_INSERTS flag - The Old New Thing (microsoft.com)icon-default.png?t=N7T8https://devblogs.microsoft.com/oldnewthing/20071128-00/?p=24353

Raymond Chen 2007年11月28日


FORMAT_MESSAGE_IGNORE_INSERTS 标志的重要性

简要

文章讨论了使用FormatMessage函数获取Win32错误代码对应的错误消息时,必须使用FORMAT_MESSAGE_IGNORE_INSERTS标志,以避免因消息中的插入序列导致的潜在错误和安全风险。

 

正文

        你可以使用 FormatMessage 函数,并带上 FORMAT_MESSAGE_FROM_SYSTEM 标志,以指明你传递的消息编号是一个错误代码,并且消息应该在系统消息表中查找。这是一个更具体情况的特例,即你不能控制消息内容,而当你不能控制消息内容时,你最好传递 FORMAT_MESSAGE_IGNORE_INSERTS 标志。

        让我们看看如果你不这么做会发生什么。

#include <windows.h>
#include <stdio.h>
#include <tchar.h>
int __cdecl main(int argc, char **argv)
{TCHAR buffer[1024];DWORD dwError = ERROR_BAD_EXE_FORMAT;DWORD dwFlags = FORMAT_MESSAGE_FROM_SYSTEM;DWORD dwResult = FormatMessage(dwFlags, NULL, dwError,0, buffer, 1024, NULL);if (dwResult) {_tprintf(_T("Message is \"%s\"\n"), buffer);} else {_tprintf(_T("Failed! Error code %d\n"), GetLastError());}return 0;
}

        如果你运行这个程序,你会得到:

Failed! Error code 87

 

        错误 87 是 ERROR_INVALID_PARAMETER(无效参数错误)。出了什么问题呢?

让我们传递 FORMAT_MESSAGE_IGNORE_INSERTS 标志来看看消息是什么。

        将 dwFlags 的值更改为:

DWORD dwFlags = FORMAT_MESSAGE_FROM_SYSTEM |FORMAT_MESSAGE_IGNORE_INSERTS;

        再次运行程序。这次你会得到:

Message is "%1 is not a valid Win32 application."

 

        啊哈,现在我们看到了问题所在。

        对应 ERROR_BAD_EXE_FORMAT 的消息包含一个插入符 %1

        如果你不传递 FORMAT_MESSAGE_IGNORE_INSERTS 标志,FormatMessage 函数将会在参数列表(或参数数组)中插入第一个参数。但我们没有传递参数列表,所以函数失败了。

        实际上,我们很幸运。

        如果我们传递了参数列表或参数数组,函数会插入相应的字符串,即使我们传递的参数列表中第一个位置没有字符串。

        如果你不能控制格式字符串,那么你必须传递 FORMAT_MESSAGE_IGNORE_INSERTS 来防止 %1 造成麻烦。

        如果有人特别恶劣,他们可能会决定给你一个包含 %9 的格式字符串,这几乎可以肯定是你提供的插入符数量的多倍。

        结果是缓冲区溢出,很可能是崩溃。

        这对一些人来说可能是显而易见的,就像你不应该传递你不能控制的字符串作为 printf 函数的格式字符串一样,但我觉得有必要提一下。

        凌晨 2 点

相关文章:

  • 北京网站建设多少钱?
  • 辽宁网页制作哪家好_网站建设
  • 高端品牌网站建设_汉中网站制作
  • pillow学习6
  • LeetCode题练习与总结:从中序与后序遍历序列构造二叉树--106
  • amis中 InputTable Combo InputKV/InputKVS 区别
  • 数据库DCL语句
  • ES基础概念
  • freertos串口DMA队列发送卡死
  • Java的函数式接口和 Lambda 表达式
  • ICML 2024 Mamba 论文总结
  • Vue3判断变量和对象不为null和undefined
  • 为了性能,放弃tft_eSPI,选择arduino_gfx吧
  • Mysql 的 binlog 有几种格式?
  • 用Python Pygame做的一些好玩的小游戏
  • 【大数据】MapReduce JAVA API编程实践及适用场景介绍
  • 自回归模型(二):具有自回归误差的回归
  • 数据库缓存 buffer pool详解
  • [ 一起学React系列 -- 8 ] React中的文件上传
  • 《Javascript高级程序设计 (第三版)》第五章 引用类型
  • 「前端」从UglifyJSPlugin强制开启css压缩探究webpack插件运行机制
  • 【159天】尚学堂高琪Java300集视频精华笔记(128)
  • Intervention/image 图片处理扩展包的安装和使用
  • JavaScript设计模式系列一:工厂模式
  • Java反射-动态类加载和重新加载
  • java小心机(3)| 浅析finalize()
  • jquery ajax学习笔记
  • Python代码面试必读 - Data Structures and Algorithms in Python
  • Rancher-k8s加速安装文档
  • React Native移动开发实战-3-实现页面间的数据传递
  • SpringBoot几种定时任务的实现方式
  • Sublime text 3 3103 注册码
  • Vue2 SSR 的优化之旅
  • Webpack入门之遇到的那些坑,系列示例Demo
  • web标准化(下)
  • 大整数乘法-表格法
  • 分享几个不错的工具
  • 利用jquery编写加法运算验证码
  • 如何实现 font-size 的响应式
  • 什么软件可以剪辑音乐?
  • 腾讯视频格式如何转换成mp4 将下载的qlv文件转换成mp4的方法
  • 网页视频流m3u8/ts视频下载
  • 线上 python http server profile 实践
  • 转载:[译] 内容加速黑科技趣谈
  • # Panda3d 碰撞检测系统介绍
  • #define,static,const,三种常量的区别
  • (3) cmake编译多个cpp文件
  • (Java企业 / 公司项目)点赞业务系统设计-批量查询点赞状态(二)
  • (MATLAB)第五章-矩阵运算
  • (带教程)商业版SEO关键词按天计费系统:关键词排名优化、代理服务、手机自适应及搭建教程
  • (附源码)apringboot计算机专业大学生就业指南 毕业设计061355
  • (附源码)springboot学生选课系统 毕业设计 612555
  • (每日一问)操作系统:常见的 Linux 指令详解
  • (十)T检验-第一部分
  • (转)微软牛津计划介绍——屌爆了的自然数据处理解决方案(人脸/语音识别,计算机视觉与语言理解)...
  • . Flume面试题
  • .MyFile@waifu.club.wis.mkp勒索病毒数据怎么处理|数据解密恢复
  • .NET / MSBuild 扩展编译时什么时候用 BeforeTargets / AfterTargets 什么时候用 DependsOnTargets?