最新C&C++语言安全编程规范_V1.0.doc
《最新C&C++语言安全编程规范_V1.0.doc》由会员分享,可在线阅读,更多相关《最新C&C++语言安全编程规范_V1.0.doc(62页珍藏版)》请在淘文阁 - 分享文档赚钱的网站上搜索。
1、精品资料C&C+语言安全编程规范_V1.0.华为技术有限公司内部技术规范DKBA 6914-2013.05C&C+语言安全编程规范2013年05月07日发布 2013年05月07日实施华为技术有限公司Huawei Technologies Co., Ltd.版权所有 侵权必究修订声明本规范拟制与解释部门: 网络安全技术能力中心本规范的相关系列规范或文件: Java语言安全编程规范Web应用安全开发规范相关国际规范或文件一致性: 无替代或作废的其它规范或文件: 无相关规范或文件的相互关系:本规范作为C语言编程规范和C+语言编程规范安全性要求的补充和扩展。规范号主要起草部门专家主要评审部门专家修订
2、情况DKBA6914-2013.05网络安全能力中心:罗东 67107、 于鹏 90006799、苗宏 90006736、朱喜红 00210657电信软件与核心网:陈辉军 00190784无线产品线:肖飞龙 00051938网络产品线:魏建雄 00222905IT产品线:熊华梁00106214中央软件院:朱楚毅00217543、林水平 00109837、周强 00048368、辛威 00176185、鞠章蕾 00040951、谢青 00101378中央硬件院:刘永合00222758终端公司:杨棋斌 00060469企业网络:黄凯进 00040281、企业SecoSpace:王瑾 9000382
3、8中央软件院:黄茂青 00057072、卢峰 00210300网络产品线:李强 00203020、罗天 00062283、廖永强 00111217、任志清 00048956、李海蛟 00040826、陈璟 00222879、勾国凯 00048893、范佳甲 00109753中央硬件院:刘崇山 00159994、施文超 00109740企业网络:李有永 90002701IT产品线:李显才 00044635、何昌军 00061280能力中心:郭曙光 00121837网络安全实验室:林结斌 00206214电信软件与核心网:朱刚 00192988无线产品线:李瀛 00130531、王爱成 00223
4、009、杨彬 00065941、于继万 00052142、解然 00234688V1.0目 录 C&C+语言安全编程规范0 规范制定说明0.1 前言随着公司业务发展,越来越多的产品被公众、互联网所熟知,并成为安全研究组织的研究对象、黑客的漏洞挖掘目标,容易引起安全问题。安全问题影响的不只是单个产品,甚至有可能影响到公司整体声誉。产品安全涉及需求、设计、实现、部署多个环节,实现的安全是产品安全的重要一环。为了帮助产品开发团队编写安全的代码,减少甚至规避由于编码错误引入安全风险,特制定本规范。C&C+语言安全编程规范参考业界安全编码的研究成果,并结合产品编码实践的经验总结,针对C/C+语言编程中的
5、字符串操作、整数操作、内存管理、文件操作、STL库使用等方面,描述可能导致安全漏洞或潜在风险的常见错误。以期减少缓冲区溢出、整数溢出、格式化字符串攻击、命令注入攻击、目录遍历等典型安全问题。0.2 使用对象本规范的读者及使用对象主要为使用C和C+语言的开发人员、测试人员等。0.3 适用范围本规范适合于公司基于C或C+语言开发的产品。0.4 术语定义原则: 编程时必须遵守的指导思想。规则:编程时必须遵守的约定。建议:编程时必须加以考虑的约定。说明:对此原则/规则/建议进行必要的解释。错误示例:对此原则/规则/建议从反面给出例子。推荐做法:对此原则/规则/建议从正面给出例子。延伸阅读材料:建议进一
6、步阅读的参考材料。1 通用原则原则1.1:对外部输入进行校验说明:对于外部输入(包括用户输入、外部接口输入、配置文件、网络数据和环境变量等)可能用于以下场景的情况下,需要检验入参的合法性:l 输入会改变系统状态l 输入作为循环条件l 输入作为数组下标l 输入作为内存分配的尺寸参数l 输入作为格式化字符串l 输入作为业务数据(如作为命令执行参数、拼装sql语句、以特定格式持久化)l 输入影响代码逻辑这些情况下如果不对用户数据作合法性验证,很可能导致DoS、内存越界、格式化字符串漏洞、命令注入、SQL注入、缓冲区溢出、数据破坏等问题。对外部输入验证常见有如下几种方式:(1)校验输入数据长度:如果输
7、入数据是字符串,通过校验输入数据的长度可以加大攻击者实施攻击的难度,从而防止缓冲区溢出、恶意代码注入等漏洞。(2)校验输入数据的范围:如果输入数据是数值,必须校验数值的范围是否正确,是否合法、在有效值域内,例如在涉及到内存分配、数组操作、循环条件、计算等安全操作时,若没有进行输入数值有效值域的校验,则可能会造成内存分配失败、数组越界、循环异常、计算错误等问题,这可能会被攻击者利用并进行进一步的攻击。(3)输入验证前,对数据进行归一化处理以防止字符转义绕过校验:通过对输入数据进行归一化处理(规范化,按照常用字符进行编码),彻底去除元字符,可以防止字符转义绕过相应的校验而引起的安全漏洞。(4)输入
8、校验应当采用“白名单”形式:“黑名单”和“白名单”是进行数据净化的两种途径。“黑名单”尝试排斥无效的输入,而“白名单”则通过定义一个可接受的字符列表,并移除任何不接受的字符来仅仅接受有效的输入。有效输入值列表通常是一个可预知的、定义良好的集合,并且其大小易于管理。“白名单”的好处在于,程序员可以确定一个字符串中仅仅包含他认为安全的字符。“白名单”比“黑名单”更受推荐的原因是,程序员不必花力气去捕捉所有不可接受的字符,只需确保识别了可接受的字符就可以了。这样一来,程序员就不用绞尽脑汁去考虑攻击者可能尝试哪些字符来绕过检查。原则1.2:禁止在日志中保存口令、密钥说明:在日志中不能保存口令和密钥,其
9、中的口令包括明文口令和密文口令。对于敏感信息建议采取以下方法,l 不打印在日志中;l 若因为特殊原因必须要打印日志,则用“*”代替。原则1.3:及时清除存储在可复用资源中的敏感信息说明:存储在可复用资源中的敏感信息如果没有正确的清除则很有可能被低权限用户或者攻击者所获取和利用。因此敏感信息在可复用资源中保存应该遵循存储时间最短原则。可复用资源包括以下几个方面:l 堆(heap)l 栈(stack)l 数据段(data segment)l 数据库的映射缓存存储口令、密钥的变量使用完后必须显式覆盖或清空。原则1.4:正确使用经过验证的安全的标准加密算法说明:禁用私有算法或者弱加密算法(如DES,S
10、HA1等),应该使用经过验证的、安全的、公开的加密算法。加密算法分为对称加密算法和非对称加密算法。推荐使用的常用对称加密算法有:l AES推荐使用的常用非对称算法有:l RSAl 数字签名算法(DSA)此外还有验证消息完整性的安全哈希算法(SHA256)等。基于哈希算法的口令安全存储必须加入盐值(salt)。密钥长度符合最低安全要求:l AES: 128位l RSA: 2048位l DSA: 1024位l SHA: 256位原则1.5:遵循最小权限原则说明:程序在运行时可能需要不同的权限,但对于某一种权限不需要始终保留。例如,一个网络程序可能需要超级用户权限来捕获原始网络数据包,但是在执行数据
11、报分析等其它任务时,则可能不需要相同的权限。因此程序在运行时只分配能完成其任务的最小权限。过高的权限可能会被攻击者利用并进行进一步的攻击。(1)撤销权限时应遵循正确的撤销顺序:在涉及到set-user-ID和set-group-ID程序中,当有效的用户ID(user ID)和组ID(group ID)与真实的用户不同时,不但要撤销用户层面(user level)的权限而且要撤销组层面(group level)的权限。在进行这样的操作时,要保证撤销顺序的正确性。权限撤销顺序的不正确操作,可能会被攻击者获得过高的权限而进行进一步的攻击。(2) 完成权限撤销操作后,应确保权限撤销成功:不同平台下所谓
12、的“适当的权限”的意义是不相同的。例如在Solaris中,setuid()的适当的权限指的是PRIV_PROC_SETID权限在进程的有效权限集中。在BSD中意味着有效地用户ID(EUID)为0或者uid=geteuid()。而在Linux中,则是指进程具有CAP_SETUID能力并且当EUID不等于0、真正的用户ID(RUID)或者已保存的set-user ID(SSUID)中任何一个时,setuid(geteuid()是失败的。正是由于权限行为的复杂性,所以所需的权限在撤销时可能会失败。这会被攻击者利用并进行进一步的攻击。例如Kernel版本在2.2.0-2.2.15的Linux就有一个权
13、限撤销漏洞,当权限功能位置为0时,setuid(getuid()没有如预期的那样撤销权限成功。因此在进行权限撤销操作后,应该校验以保证权限撤销成功。原则1.6:删除或修改没有效果的代码说明:删除或修改一些即使执行后、也不会有任何效果的代码。一些存在的代码(声明或表达式),即使它被执行后,也不会对代码的结果或数据的状态产生任何的影响,或者产生不是所预期的效果,这样的代码在可能是由于编码错误引起的,往往隐藏着逻辑上的错误。原则1.7:删除或修改没有使用到的变量或值说明:删除或修改没有使用到的变量或值。一些变量或值存在于代码里,但并没有被使用到,这可能隐含着逻辑上的错误,需要被识别出来,删除这类语句
14、或做相应的修改。2 字符串操作安全规则2.1:确保有足够的空间存储字符串的字符数据和0结束符说明:在分配内存或者在执行字符串复制操作时,除了要保证足够的空间可以容纳字符数据,还要预留0结束符的空间,否则会造成缓冲区溢出。错误示例1:拷贝字符串时,源字符串长度可能大于目标数组空间。void main(int argc, char *argv)char dst128;if ( argc 1 )strcpy(dst, argv1); / 源字符串长度可能大于目标数组空间,造成缓冲区溢出/*/ 推荐做法:根据源字符串长度来为目标字符串分配空间。void main(int argc, char *arg
15、v)char *dst = NULL;if ( argc 1 )dst = (char *)malloc(strlen(argv1) + 1); /* 【修改】确保字符串空间足够容纳argv1 */if( dst != NULL )strncpy(dst, argv1, strlen(argv1);dststrlen(argv1) = 0; /【修改】dst以0结尾/*.dst使用后free.*/ 错误示例2:典型的差一错误,未考虑0结束符写入数组的位置,造成缓冲区溢出和内存改写。void NoCompliant()char dstARRAY_SIZE + 1;char srcARRAY_SI
16、ZE + 1;unsigned int i = 0;memset(src, , sizeof(dst);for(i=0; srci != 0 & (i sizeof(dst); +i )dsti = srci;dsti = 0; /*/ 推荐做法:void Compliant()char dstARRAY_SIZE + 1;char srcARRAY_SIZE + 1;unsigned int i = 0;memset(src, , sizeof(dst);for(i=0; srci!=0 & (i sizeof(dst) - 1 ); +i) /*【修改】考虑0结束符 */dsti = sr
17、ci;dsti = 0; /*/ 规则2.2:字符串操作过程中确保字符串有0结束符说明:字符串结束与否是以0作为标志的。没有正确地使用0结束字符串可能导致字符串操作时发生缓冲区溢出。因此对于字符串或字符数组的定义、设置、复制等操作,要给0预留空间,并保证字符串有0结束符。注意:strncpy、strncat等带n版本的字符串操作函数在源字符串长度超出n标识的长度时,会将包括0结束符在内的超长字符串截断,导致0结束符丢失。这时需要手动为目标字符串设置0结束符。错误示例1:strlen()不会将0结束符算入长度,配合memcpy使用时会丢失0结束符。void Noncompliant()char
18、dst11;char src = 0123456789;char *tmp = NULL;memset(dst, , sizeof(dst); memcpy(dst, src, strlen(src);printf(src: %s rn, src);tmp = dst; /到此,dst还没有以0结尾doputchar(*tmp);while (*tmp+); / 访问越界return; 推荐做法: 为目标字符串设置0结束符void Compliant()char dst11;char src = 0123456789;char *tmp = NULL;memset(dst, , sizeof(
19、dst);memcpy(dst, src, strlen(src);dstsizeof(dst) - 1 = 0; /【修改】dst以0结尾printf(src: %s rn, src);tmp = dst;doputchar(*tmp); while (*tmp+);return; 错误示例2:strncpy()拷贝限长字符串,截断了0结束符。 void Noncompliant()char dst5;char src = 0123456789;strncpy(dst, src, sizeof(dst); printf(dst); /访问越界,dst没有0结束符return;推荐做法: vo
20、id Compliant()char dst5; char src = 0123456789;strncpy(dst, src, sizeof(dst);dstsizeof(dst)-1 = 0; / 【修改】最后字节置为0 printf(dst); return;规则2.3:把数据复制到固定长度的内存前必须检查边界说明:将未知长度的数据复制到固定长度的内存空间可能会造成缓冲区溢出,因此在进行复制之前应首先获取并检查数据长度。典型的如来自gets()、getenv()、scanf()的字符串。错误示例:输入消息长度不可预测,不加检查的复制会造成缓冲区溢出。void Noncompliant()
21、char dst16;char * temp = getInputMsg();if(temp != NULL)strcpy(dst,temp); / temp长度可能超过dst的大小return; 推荐做法: void Compliant()char dst16;char *temp = getInputMsg();if(temp != NULL)strncpy(dst, temp, sizeof(dst); /* 【修改】只复制不超过数组dst大小的数据 */dstsizeof(dst) -1 = 0; /【修改】copy以0结尾return;规则2.4:避免字符串/内存操作函数的源指针和目
- 配套讲稿:
如PPT文件的首页显示word图标,表示该PPT已包含配套word讲稿。双击word图标可打开word文档。
- 特殊限制:
部分文档作品中含有的国旗、国徽等图片,仅作为作品整体效果示例展示,禁止商用。设计者仅对作品中独创性部分享有著作权。
- 关 键 词:
- 最新 C+ 语言 安全 编程 规范 _V1
限制150内