⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 v3_purp.c

📁 OpenSSL 0.9.8k 最新版OpenSSL
💻 C
📖 第 1 页 / 共 2 页
字号:
/* v3_purp.c *//* Written by Dr Stephen N Henson (steve@openssl.org) for the OpenSSL * project 2001. *//* ==================================================================== * Copyright (c) 1999-2004 The OpenSSL Project.  All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * * 1. Redistributions of source code must retain the above copyright *    notice, this list of conditions and the following disclaimer.  * * 2. Redistributions in binary form must reproduce the above copyright *    notice, this list of conditions and the following disclaimer in *    the documentation and/or other materials provided with the *    distribution. * * 3. All advertising materials mentioning features or use of this *    software must display the following acknowledgment: *    "This product includes software developed by the OpenSSL Project *    for use in the OpenSSL Toolkit. (http://www.OpenSSL.org/)" * * 4. The names "OpenSSL Toolkit" and "OpenSSL Project" must not be used to *    endorse or promote products derived from this software without *    prior written permission. For written permission, please contact *    licensing@OpenSSL.org. * * 5. Products derived from this software may not be called "OpenSSL" *    nor may "OpenSSL" appear in their names without prior written *    permission of the OpenSSL Project. * * 6. Redistributions of any form whatsoever must retain the following *    acknowledgment: *    "This product includes software developed by the OpenSSL Project *    for use in the OpenSSL Toolkit (http://www.OpenSSL.org/)" * * THIS SOFTWARE IS PROVIDED BY THE OpenSSL PROJECT ``AS IS'' AND ANY * EXPRESSED OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR * PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL THE OpenSSL PROJECT OR * ITS CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED * OF THE POSSIBILITY OF SUCH DAMAGE. * ==================================================================== * * This product includes cryptographic software written by Eric Young * (eay@cryptsoft.com).  This product includes software written by Tim * Hudson (tjh@cryptsoft.com). * */#include <stdio.h>#include "cryptlib.h"#include <openssl/x509v3.h>#include <openssl/x509_vfy.h>static void x509v3_cache_extensions(X509 *x);static int check_ssl_ca(const X509 *x);static int check_purpose_ssl_client(const X509_PURPOSE *xp, const X509 *x, int ca);static int check_purpose_ssl_server(const X509_PURPOSE *xp, const X509 *x, int ca);static int check_purpose_ns_ssl_server(const X509_PURPOSE *xp, const X509 *x, int ca);static int purpose_smime(const X509 *x, int ca);static int check_purpose_smime_sign(const X509_PURPOSE *xp, const X509 *x, int ca);static int check_purpose_smime_encrypt(const X509_PURPOSE *xp, const X509 *x, int ca);static int check_purpose_crl_sign(const X509_PURPOSE *xp, const X509 *x, int ca);static int no_check(const X509_PURPOSE *xp, const X509 *x, int ca);static int ocsp_helper(const X509_PURPOSE *xp, const X509 *x, int ca);static int xp_cmp(const X509_PURPOSE * const *a,		const X509_PURPOSE * const *b);static void xptable_free(X509_PURPOSE *p);static X509_PURPOSE xstandard[] = {	{X509_PURPOSE_SSL_CLIENT, X509_TRUST_SSL_CLIENT, 0, check_purpose_ssl_client, "SSL client", "sslclient", NULL},	{X509_PURPOSE_SSL_SERVER, X509_TRUST_SSL_SERVER, 0, check_purpose_ssl_server, "SSL server", "sslserver", NULL},	{X509_PURPOSE_NS_SSL_SERVER, X509_TRUST_SSL_SERVER, 0, check_purpose_ns_ssl_server, "Netscape SSL server", "nssslserver", NULL},	{X509_PURPOSE_SMIME_SIGN, X509_TRUST_EMAIL, 0, check_purpose_smime_sign, "S/MIME signing", "smimesign", NULL},	{X509_PURPOSE_SMIME_ENCRYPT, X509_TRUST_EMAIL, 0, check_purpose_smime_encrypt, "S/MIME encryption", "smimeencrypt", NULL},	{X509_PURPOSE_CRL_SIGN, X509_TRUST_COMPAT, 0, check_purpose_crl_sign, "CRL signing", "crlsign", NULL},	{X509_PURPOSE_ANY, X509_TRUST_DEFAULT, 0, no_check, "Any Purpose", "any", NULL},	{X509_PURPOSE_OCSP_HELPER, X509_TRUST_COMPAT, 0, ocsp_helper, "OCSP helper", "ocsphelper", NULL},};#define X509_PURPOSE_COUNT (sizeof(xstandard)/sizeof(X509_PURPOSE))IMPLEMENT_STACK_OF(X509_PURPOSE)static STACK_OF(X509_PURPOSE) *xptable = NULL;static int xp_cmp(const X509_PURPOSE * const *a,		const X509_PURPOSE * const *b){	return (*a)->purpose - (*b)->purpose;}/* As much as I'd like to make X509_check_purpose use a "const" X509* * I really can't because it does recalculate hashes and do other non-const * things. */int X509_check_purpose(X509 *x, int id, int ca){	int idx;	const X509_PURPOSE *pt;	if(!(x->ex_flags & EXFLAG_SET)) {		CRYPTO_w_lock(CRYPTO_LOCK_X509);		x509v3_cache_extensions(x);		CRYPTO_w_unlock(CRYPTO_LOCK_X509);	}	if(id == -1) return 1;	idx = X509_PURPOSE_get_by_id(id);	if(idx == -1) return -1;	pt = X509_PURPOSE_get0(idx);	return pt->check_purpose(pt, x, ca);}int X509_PURPOSE_set(int *p, int purpose){	if(X509_PURPOSE_get_by_id(purpose) == -1) {		X509V3err(X509V3_F_X509_PURPOSE_SET, X509V3_R_INVALID_PURPOSE);		return 0;	}	*p = purpose;	return 1;}int X509_PURPOSE_get_count(void){	if(!xptable) return X509_PURPOSE_COUNT;	return sk_X509_PURPOSE_num(xptable) + X509_PURPOSE_COUNT;}X509_PURPOSE * X509_PURPOSE_get0(int idx){	if(idx < 0) return NULL;	if(idx < (int)X509_PURPOSE_COUNT) return xstandard + idx;	return sk_X509_PURPOSE_value(xptable, idx - X509_PURPOSE_COUNT);}int X509_PURPOSE_get_by_sname(char *sname){	int i;	X509_PURPOSE *xptmp;	for(i = 0; i < X509_PURPOSE_get_count(); i++) {		xptmp = X509_PURPOSE_get0(i);		if(!strcmp(xptmp->sname, sname)) return i;	}	return -1;}int X509_PURPOSE_get_by_id(int purpose){	X509_PURPOSE tmp;	int idx;	if((purpose >= X509_PURPOSE_MIN) && (purpose <= X509_PURPOSE_MAX))		return purpose - X509_PURPOSE_MIN;	tmp.purpose = purpose;	if(!xptable) return -1;	idx = sk_X509_PURPOSE_find(xptable, &tmp);	if(idx == -1) return -1;	return idx + X509_PURPOSE_COUNT;}int X509_PURPOSE_add(int id, int trust, int flags,			int (*ck)(const X509_PURPOSE *, const X509 *, int),					char *name, char *sname, void *arg){	int idx;	X509_PURPOSE *ptmp;	/* This is set according to what we change: application can't set it */	flags &= ~X509_PURPOSE_DYNAMIC;	/* This will always be set for application modified trust entries */	flags |= X509_PURPOSE_DYNAMIC_NAME;	/* Get existing entry if any */	idx = X509_PURPOSE_get_by_id(id);	/* Need a new entry */	if(idx == -1) {		if(!(ptmp = OPENSSL_malloc(sizeof(X509_PURPOSE)))) {			X509V3err(X509V3_F_X509_PURPOSE_ADD,ERR_R_MALLOC_FAILURE);			return 0;		}		ptmp->flags = X509_PURPOSE_DYNAMIC;	} else ptmp = X509_PURPOSE_get0(idx);	/* OPENSSL_free existing name if dynamic */	if(ptmp->flags & X509_PURPOSE_DYNAMIC_NAME) {		OPENSSL_free(ptmp->name);		OPENSSL_free(ptmp->sname);	}	/* dup supplied name */	ptmp->name = BUF_strdup(name);	ptmp->sname = BUF_strdup(sname);	if(!ptmp->name || !ptmp->sname) {		X509V3err(X509V3_F_X509_PURPOSE_ADD,ERR_R_MALLOC_FAILURE);		return 0;	}	/* Keep the dynamic flag of existing entry */	ptmp->flags &= X509_PURPOSE_DYNAMIC;	/* Set all other flags */	ptmp->flags |= flags;	ptmp->purpose = id;	ptmp->trust = trust;	ptmp->check_purpose = ck;	ptmp->usr_data = arg;	/* If its a new entry manage the dynamic table */	if(idx == -1) {		if(!xptable && !(xptable = sk_X509_PURPOSE_new(xp_cmp))) {			X509V3err(X509V3_F_X509_PURPOSE_ADD,ERR_R_MALLOC_FAILURE);			return 0;		}		if (!sk_X509_PURPOSE_push(xptable, ptmp)) {			X509V3err(X509V3_F_X509_PURPOSE_ADD,ERR_R_MALLOC_FAILURE);			return 0;		}	}	return 1;}static void xptable_free(X509_PURPOSE *p)	{	if(!p) return;	if (p->flags & X509_PURPOSE_DYNAMIC) 		{		if (p->flags & X509_PURPOSE_DYNAMIC_NAME) {			OPENSSL_free(p->name);			OPENSSL_free(p->sname);		}		OPENSSL_free(p);		}	}void X509_PURPOSE_cleanup(void){	unsigned int i;	sk_X509_PURPOSE_pop_free(xptable, xptable_free);	for(i = 0; i < X509_PURPOSE_COUNT; i++) xptable_free(xstandard + i);	xptable = NULL;}int X509_PURPOSE_get_id(X509_PURPOSE *xp){	return xp->purpose;}char *X509_PURPOSE_get0_name(X509_PURPOSE *xp){	return xp->name;}char *X509_PURPOSE_get0_sname(X509_PURPOSE *xp){	return xp->sname;}int X509_PURPOSE_get_trust(X509_PURPOSE *xp){	return xp->trust;}static int nid_cmp(int *a, int *b)	{	return *a - *b;	}int X509_supported_extension(X509_EXTENSION *ex)	{	/* This table is a list of the NIDs of supported extensions:	 * that is those which are used by the verify process. If	 * an extension is critical and doesn't appear in this list	 * then the verify process will normally reject the certificate.	 * The list must be kept in numerical order because it will be	 * searched using bsearch.	 */	static int supported_nids[] = {		NID_netscape_cert_type, /* 71 */        	NID_key_usage,		/* 83 */		NID_subject_alt_name,	/* 85 */		NID_basic_constraints,	/* 87 */		NID_certificate_policies, /* 89 */        	NID_ext_key_usage,	/* 126 */#ifndef OPENSSL_NO_RFC3779		NID_sbgp_ipAddrBlock,	/* 290 */		NID_sbgp_autonomousSysNum, /* 291 */#endif		NID_policy_constraints,	/* 401 */		NID_proxyCertInfo,	/* 661 */		NID_inhibit_any_policy	/* 748 */	};	int ex_nid;	ex_nid = OBJ_obj2nid(X509_EXTENSION_get_object(ex));	if (ex_nid == NID_undef) 		return 0;	if (OBJ_bsearch((char *)&ex_nid, (char *)supported_nids,		sizeof(supported_nids)/sizeof(int), sizeof(int),		(int (*)(const void *, const void *))nid_cmp))		return 1;	return 0;	} static void x509v3_cache_extensions(X509 *x){	BASIC_CONSTRAINTS *bs;	PROXY_CERT_INFO_EXTENSION *pci;	ASN1_BIT_STRING *usage;	ASN1_BIT_STRING *ns;	EXTENDED_KEY_USAGE *extusage;	X509_EXTENSION *ex;		int i;	if(x->ex_flags & EXFLAG_SET) return;#ifndef OPENSSL_NO_SHA	X509_digest(x, EVP_sha1(), x->sha1_hash, NULL);#endif	/* Does subject name match issuer ? */	if(!X509_NAME_cmp(X509_get_subject_name(x), X509_get_issuer_name(x)))			 x->ex_flags |= EXFLAG_SI;	/* V1 should mean no extensions ... */	if(!X509_get_version(x)) x->ex_flags |= EXFLAG_V1;

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -