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

📄 octet_string.c

📁 RSA加密/解密算法源码 asn1c-0.9.12
💻 C
📖 第 1 页 / 共 3 页
字号:
/*- * Copyright (c) 2003, 2004 Lev Walkin <vlm@lionet.info>. All rights reserved. * Redistribution and modifications are permitted subject to BSD license. */#include <asn_internal.h>#include <OCTET_STRING.h>#include <BIT_STRING.h>	/* for .bits_unused member */#include <assert.h>#include <errno.h>/* * OCTET STRING basic type description. */static ber_tlv_tag_t asn_DEF_OCTET_STRING_tags[] = {	(ASN_TAG_CLASS_UNIVERSAL | (4 << 2))};static asn_OCTET_STRING_specifics_t asn_DEF_OCTET_STRING_specs = {	sizeof(OCTET_STRING_t),	offsetof(OCTET_STRING_t, _asn_ctx),	0};asn_TYPE_descriptor_t asn_DEF_OCTET_STRING = {	"OCTET STRING",		/* Canonical name */	"OCTET_STRING",		/* XML tag name */	OCTET_STRING_free,	OCTET_STRING_print,	/* non-ascii stuff, generally */	asn_generic_no_constraint,	OCTET_STRING_decode_ber,	OCTET_STRING_encode_der,	OCTET_STRING_decode_xer_hex,	OCTET_STRING_encode_xer,	0, /* Use generic outmost tag fetcher */	asn_DEF_OCTET_STRING_tags,	sizeof(asn_DEF_OCTET_STRING_tags)	  / sizeof(asn_DEF_OCTET_STRING_tags[0]),	asn_DEF_OCTET_STRING_tags,	/* Same as above */	sizeof(asn_DEF_OCTET_STRING_tags)	  / sizeof(asn_DEF_OCTET_STRING_tags[0]),	0, 0,	/* No members */	&asn_DEF_OCTET_STRING_specs};#undef	_CH_PHASE#undef	NEXT_PHASE#undef	PREV_PHASE#define	_CH_PHASE(ctx, inc) do {	\		if(ctx->phase == 0)	\			ctx->step = 0;	\		ctx->phase += inc;	\	} while(0)#define	NEXT_PHASE(ctx)	_CH_PHASE(ctx, +1)#define	PREV_PHASE(ctx)	_CH_PHASE(ctx, -1)#undef	ADVANCE#define	ADVANCE(num_bytes)	do {			\		size_t num = (num_bytes);		\		buf_ptr = ((char *)buf_ptr) + num;	\		size -= num;				\		consumed_myself += num;			\	} while(0)#undef	RETURN#define	RETURN(_code)	do {			\		rval.code = _code;		\		rval.consumed = consumed_myself;\		return rval;			\	} while(0)#undef	APPEND#define	APPEND(bufptr, bufsize)	do {					\		size_t _bs = (bufsize);					\		size_t _ns = ctx->step;	/* Allocated */			\		if(_ns <= (size_t)(st->size + _bs)) {			\			void *ptr;					\			/* Be nice and round to the memory allocator */	\			do { _ns = _ns ? _ns<<2 : 16; }			\			    while(_ns <= (size_t)(st->size + _bs));	\			ptr = REALLOC(st->buf, _ns);			\			if(ptr) {					\				st->buf = (uint8_t *)ptr;		\				ctx->step = _ns;			\			} else {					\				RETURN(RC_FAIL);			\			}						\		}							\		memcpy(st->buf + st->size, bufptr, _bs);		\		st->size += _bs;					\		if(st->size < 0)					\			/* Why even care?.. JIC */			\			RETURN(RC_FAIL);				\		/* Convenient nul-termination */			\		st->buf[st->size] = '\0';				\	} while(0)/* * Internal variant of the OCTET STRING. */typedef enum OS_type {	_TT_GENERIC	= 0,	/* Just a random OCTET STRING */	_TT_BIT_STRING	= 1,	/* BIT STRING type, a special case */	_TT_ANY		= 2,	/* ANY type, a special case too */} OS_type_e;/* * The main reason why ASN.1 is still alive is that too much time and effort * is necessary for learning it more or less adequately, thus creating a gut * necessity to demonstrate that aquired skill everywhere afterwards. * No, I am not going to explain what the following stuff is. */struct _stack_el {	ber_tlv_len_t	left;	/* What's left to read (or -1) */	ber_tlv_len_t	got;	/* What was actually processed */	int	cont_level;	/* Depth of subcontainment */	int	want_nulls;	/* Want null "end of content" octets? */	int	bits_chopped;	/* Flag in BIT STRING mode */	ber_tlv_tag_t	tag;	/* For debugging purposes */	struct _stack_el *prev;	struct _stack_el *next;};struct _stack {	struct _stack_el *tail;	struct _stack_el *cur_ptr;};static struct _stack_el *OS__add_stack_el(struct _stack *st) {	struct _stack_el *nel;	/*	 * Reuse the old stack frame or allocate a new one.	 */	if(st->cur_ptr && st->cur_ptr->next) {		nel = st->cur_ptr->next;		nel->bits_chopped = 0;		nel->got = 0;		/* Retain the nel->cont_level, it's correct. */	} else {		nel = (struct _stack_el *)CALLOC(1, sizeof(struct _stack_el));		if(nel == NULL)			return NULL;			if(st->tail) {			/* Increase a subcontainment depth */			nel->cont_level = st->tail->cont_level + 1;			st->tail->next = nel;		}		nel->prev = st->tail;		st->tail = nel;	}	st->cur_ptr = nel;	return nel;}static struct _stack *_new_stack() {	return (struct _stack *)CALLOC(1, sizeof(struct _stack));}/* * Decode OCTET STRING type. */asn_dec_rval_tOCTET_STRING_decode_ber(asn_codec_ctx_t *opt_codec_ctx,	asn_TYPE_descriptor_t *td,	void **os_structure, void *buf_ptr, size_t size, int tag_mode) {	asn_OCTET_STRING_specifics_t *specs = td->specifics				? (asn_OCTET_STRING_specifics_t *)td->specifics				: &asn_DEF_OCTET_STRING_specs;	BIT_STRING_t *st = (BIT_STRING_t *)*os_structure;	asn_dec_rval_t rval;	asn_struct_ctx_t *ctx;	ssize_t consumed_myself = 0;	struct _stack *stck;		/* Expectations stack structure */	struct _stack_el *sel = 0;	/* Stack element */	int tlv_constr;	OS_type_e type_variant = (OS_type_e)specs->subvariant;	ASN_DEBUG("Decoding %s as %s (frame %ld)",		td->name,		(type_variant == _TT_GENERIC) ?			"OCTET STRING" : "OS-SpecialCase",		(long)size);	/*	 * Create the string if does not exist.	 */	if(st == NULL) {		*os_structure = CALLOC(1, specs->struct_size);		st = (BIT_STRING_t *)*os_structure;		if(st == NULL)			RETURN(RC_FAIL);	}	/* Restore parsing context */	ctx = (asn_struct_ctx_t *)((char *)st + specs->ctx_offset);	switch(ctx->phase) {	case 0:		/*		 * Check tags.		 */		rval = ber_check_tags(opt_codec_ctx, td, ctx,			buf_ptr, size, tag_mode, -1,			&ctx->left, &tlv_constr);		if(rval.code != RC_OK)			return rval;		if(tlv_constr) {			/*			 * Complex operation, requires stack of expectations.			 */			ctx->ptr = _new_stack();			if(ctx->ptr) {				stck = (struct _stack *)ctx->ptr;			} else {				RETURN(RC_FAIL);			}		} else {			/*			 * Jump into stackless primitive decoding.			 */			_CH_PHASE(ctx, 3);			if(type_variant == _TT_ANY && tag_mode != 1)				APPEND(buf_ptr, rval.consumed);			ADVANCE(rval.consumed);			goto phase3;		}		NEXT_PHASE(ctx);		/* Fall through */	case 1:	phase1:		/*		 * Fill the stack with expectations.		 */		stck = (struct _stack *)ctx->ptr;		sel = stck->cur_ptr;	  do {		ber_tlv_tag_t tlv_tag;		ber_tlv_len_t tlv_len;		ber_tlv_tag_t expected_tag;		ssize_t tl, ll, tlvl;				/* This one works even if (sel->left == -1) */		ssize_t Left = ((!sel||(size_t)sel->left >= size)					?(ssize_t)size:sel->left);		ASN_DEBUG("%p, s->l=%ld, s->wn=%ld, s->g=%ld\n", sel,			(long)(sel?sel->left:0),			(long)(sel?sel->want_nulls:0),			(long)(sel?sel->got:0)		);		if(sel && sel->left <= 0 && sel->want_nulls == 0) {			if(sel->prev) {				struct _stack_el *prev = sel->prev;				if(prev->left != -1) {					if(prev->left < sel->got)						RETURN(RC_FAIL);					prev->left -= sel->got;				}				prev->got += sel->got;				sel = stck->cur_ptr = prev;				if(!sel) break;				tlv_constr = 1;				continue;			} else {				sel = stck->cur_ptr = 0;				break;	/* Nothing to wait */			}		}		tl = ber_fetch_tag(buf_ptr, Left, &tlv_tag);		ASN_DEBUG("fetch tag(size=%ld,L=%ld), %sstack, left=%ld, wn=%ld, tl=%ld",			(long)size, (long)Left, sel?"":"!",			(long)(sel?sel->left:0),			(long)(sel?sel->want_nulls:0),			(long)tl);		switch(tl) {		case -1: RETURN(RC_FAIL);		case 0: RETURN(RC_WMORE);		}		tlv_constr = BER_TLV_CONSTRUCTED(buf_ptr);		ll = ber_fetch_length(tlv_constr,				(char *)buf_ptr + tl, Left - tl, &tlv_len);		ASN_DEBUG("Got tag=%s, tc=%d, left=%ld, tl=%ld, len=%ld, ll=%ld",			ber_tlv_tag_string(tlv_tag), tlv_constr,				(long)Left, (long)tl, (long)tlv_len, (long)ll);		switch(ll) {		case -1: RETURN(RC_FAIL);		case 0: RETURN(RC_WMORE);		}		if(sel && sel->want_nulls			&& ((uint8_t *)buf_ptr)[0] == 0			&& ((uint8_t *)buf_ptr)[1] == 0)		{			ASN_DEBUG("Eat EOC; wn=%d--", sel->want_nulls);			if(type_variant == _TT_ANY			&& (tag_mode != 1 || sel->cont_level))				APPEND("\0\0", 2);			ADVANCE(2);			sel->got += 2;			if(sel->left != -1) {				sel->left -= 2;	/* assert(sel->left >= 2) */			}			sel->want_nulls--;			if(sel->want_nulls == 0) {				/* Move to the next expectation */				sel->left = 0;				tlv_constr = 1;			}			continue;		}		/*		 * Set up expected tags,		 * depending on ASN.1 type being decoded.		 */		switch(type_variant) {		case _TT_BIT_STRING:			/* X.690: 8.6.4.1, NOTE 2 */			/* Fall through */		case _TT_GENERIC:		default:			if(sel) {				int level = sel->cont_level;				if(level < td->all_tags_count) {					expected_tag = td->all_tags[level];					break;				} else if(td->all_tags_count) {					expected_tag = td->all_tags						[td->all_tags_count - 1];					break;				}				/* else, Fall through */			}			/* Fall through */		case _TT_ANY:			expected_tag = tlv_tag;			break;		}		if(tlv_tag != expected_tag) {			char buf[2][32];			ber_tlv_tag_snprint(tlv_tag,				buf[0], sizeof(buf[0]));			ber_tlv_tag_snprint(td->tags[td->tags_count-1],				buf[1], sizeof(buf[1]));			ASN_DEBUG("Tag does not match expectation: %s != %s",				buf[0], buf[1]);			RETURN(RC_FAIL);		}		tlvl = tl + ll;	/* Combined length of T and L encoding */		if((tlv_len + tlvl) < 0) {			/* tlv_len value is too big */			ASN_DEBUG("TLV encoding + length (%ld) is too big",				(long)tlv_len);			RETURN(RC_FAIL);		}		/*		 * Append a new expectation.		 */		sel = OS__add_stack_el(stck);		if(!sel) RETURN(RC_FAIL);		sel->tag = tlv_tag;		sel->want_nulls = (tlv_len==-1);		if(sel->prev && sel->prev->left != -1) {			/* Check that the parent frame is big enough */			if(sel->prev->left < tlvl + (tlv_len==-1?0:tlv_len))				RETURN(RC_FAIL);			if(tlv_len == -1)				sel->left = sel->prev->left - tlvl;			else				sel->left = tlv_len;		} else {			sel->left = tlv_len;		}		if(type_variant == _TT_ANY		&& (tag_mode != 1 || sel->cont_level))			APPEND(buf_ptr, tlvl);		sel->got += tlvl;		ADVANCE(tlvl);		ASN_DEBUG("+EXPECT2 got=%ld left=%ld, wn=%d, clvl=%d",			(long)sel->got, (long)sel->left,			sel->want_nulls, sel->cont_level);	  } while(tlv_constr);		if(sel == NULL) {			/* Finished operation, "phase out" */			ASN_DEBUG("Phase out");			_CH_PHASE(ctx, +3);			break;		}		NEXT_PHASE(ctx);		/* Fall through */	case 2:		stck = (struct _stack *)ctx->ptr;		sel = stck->cur_ptr;		ASN_DEBUG("Phase 2: Need %ld bytes, size=%ld, alrg=%ld, wn=%d",			(long)sel->left, (long)size, (long)sel->got,				sel->want_nulls);	    {		ber_tlv_len_t len;		assert(sel->left >= 0);		len = ((ber_tlv_len_t)size < sel->left)				? (ber_tlv_len_t)size : sel->left;		if(len > 0) {			if(type_variant == _TT_BIT_STRING			&& sel->bits_chopped == 0) {				/* Put the unused-bits-octet away */				st->bits_unused = *(uint8_t *)buf_ptr;				APPEND(((char *)buf_ptr+1), (len - 1));				sel->bits_chopped = 1;			} else {				APPEND(buf_ptr, len);			}			ADVANCE(len);			sel->left -= len;			sel->got += len;		}		if(sel->left) {			ASN_DEBUG("OS left %ld, size = %ld, wn=%d\n",				(long)sel->left, (long)size, sel->want_nulls);			RETURN(RC_WMORE);		}		PREV_PHASE(ctx);		goto phase1;

⌨️ 快捷键说明

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