
/*** <<< INCLUDES [T] >>> ***/

#include <INTEGER.h>
#include <OCTET_STRING.h>
#include <REAL.h>
#include <constr_SET.h>
#include <constr_SEQUENCE.h>

/*** <<< DEPS [T] >>> ***/


/*
 * Method of determining the components presence
 */
typedef enum class_PR {
} class_PR;

/*** <<< TYPE-DECLS [T] >>> ***/

typedef struct T {
	INTEGER_t	 Int;
	OCTET_STRING_t	 Char;
	struct Class {
		/*
		 * This type is extensible,
		 * possible extensions are below.
		 */
		
		/* Presence bitmask: ASN_SET_ISPRESENT(pclass, class_PR_x) */
		unsigned int _presence_map
			[((0+(8*sizeof(unsigned int))-1)/(8*sizeof(unsigned int)))];
		
		/* Context for parsing across buffer boundaries */
		asn_struct_ctx_t _asn_ctx;
	} Class;
	REAL_t	 Double;
	
	/* Context for parsing across buffer boundaries */
	asn_struct_ctx_t _asn_ctx;
} T_t;

/*** <<< FUNC-DECLS [T] >>> ***/

extern asn_TYPE_descriptor_t asn_DEF_T;

/*** <<< CODE [T] >>> ***/

static int
memb_char_1_constraint(asn_TYPE_descriptor_t *td, const void *sptr,
			asn_app_consume_bytes_f *app_errlog, void *app_key) {
	const OCTET_STRING_t *st = (const OCTET_STRING_t *)sptr;
	size_t size;
	
	if(!sptr) {
		_ASN_ERRLOG(app_errlog, app_key,
			"%s: value not given (%s:%d)",
			td->name, __FILE__, __LINE__);
		return -1;
	}
	
	size = st->size;
	
	if((size == 1)) {
		/* Constraint check succeeded */
		return 0;
	} else {
		_ASN_ERRLOG(app_errlog, app_key,
			"%s: constraint failed (%s:%d)",
			td->name, __FILE__, __LINE__);
		return -1;
	}
}


/*** <<< STAT-DEFS [T] >>> ***/

static ber_tlv_tag_t asn_DEF_class_4_tags[] = {
	(ASN_TAG_CLASS_UNIVERSAL | (17 << 2))
};
static uint8_t asn_MAP_class_4_mmap[(0 + (8 * sizeof(unsigned int)) - 1) / 8] = {
	0
};
static asn_SET_specifics_t asn_SPC_class_4_specs = {
	sizeof(struct Class),
	offsetof(struct Class, _asn_ctx),
	offsetof(struct Class, _presence_map),
	asn_MAP_class_4_tag2el,
	0,	/* Count of tags in the map */
	asn_MAP_class_4_tag2el,	/* Same as above */
	0,	/* Count of tags in the CXER map */
	1,	/* Whether extensible */
	(unsigned int *)asn_MAP_class_4_mmap	/* Mandatory elements map */
};
static /* Use -fall-defs-global to expose */
asn_TYPE_descriptor_t asn_DEF_class_4 = {
	"class",
	"class",
	SET_free,
	SET_print,
	SET_constraint,
	SET_decode_ber,
	SET_encode_der,
	SET_decode_xer,
	SET_encode_xer,
	0,	/* Use generic outmost tag fetcher */
	asn_DEF_class_4_tags,
	sizeof(asn_DEF_class_4_tags)
		/sizeof(asn_DEF_class_4_tags[0]), /* 1 */
	asn_DEF_class_4_tags,	/* Same as above */
	sizeof(asn_DEF_class_4_tags)
		/sizeof(asn_DEF_class_4_tags[0]), /* 1 */
	0, 0,	/* No members */
	&asn_SPC_class_4_specs	/* Additional specs */
};

static asn_TYPE_member_t asn_MBR_T_1[] = {
	{ ATF_NOFLAGS, 0, offsetof(struct T, Int),
		.tag = (ASN_TAG_CLASS_UNIVERSAL | (2 << 2)),
		.tag_mode = 0,
		.type = (void *)&asn_DEF_INTEGER,
		.memb_constraints = 0,	/* Defer constraints checking to the member type */
		.name = "int"
		},
	{ ATF_NOFLAGS, 0, offsetof(struct T, Char),
		.tag = (ASN_TAG_CLASS_UNIVERSAL | (4 << 2)),
		.tag_mode = 0,
		.type = (void *)&asn_DEF_OCTET_STRING,
		.memb_constraints = memb_char_1_constraint,
		.name = "char"
		},
	{ ATF_NOFLAGS, 0, offsetof(struct T, Class),
		.tag = (ASN_TAG_CLASS_UNIVERSAL | (17 << 2)),
		.tag_mode = 0,
		.type = (void *)&asn_DEF_class_4,
		.memb_constraints = 0,	/* Defer constraints checking to the member type */
		.name = "class"
		},
	{ ATF_NOFLAGS, 0, offsetof(struct T, Double),
		.tag = (ASN_TAG_CLASS_UNIVERSAL | (9 << 2)),
		.tag_mode = 0,
		.type = (void *)&asn_DEF_REAL,
		.memb_constraints = 0,	/* Defer constraints checking to the member type */
		.name = "double"
		},
};
static ber_tlv_tag_t asn_DEF_T_1_tags[] = {
	(ASN_TAG_CLASS_UNIVERSAL | (16 << 2))
};
static asn_TYPE_tag2member_t asn_MAP_T_1_tag2el[] = {
    { (ASN_TAG_CLASS_UNIVERSAL | (2 << 2)), 0, 0, 0 }, /* int at 15 */
    { (ASN_TAG_CLASS_UNIVERSAL | (4 << 2)), 1, 0, 0 }, /* char at 16 */
    { (ASN_TAG_CLASS_UNIVERSAL | (9 << 2)), 3, 0, 0 }, /* double at 18 */
    { (ASN_TAG_CLASS_UNIVERSAL | (17 << 2)), 2, 0, 0 } /* class at 17 */
};
static asn_SEQUENCE_specifics_t asn_SPC_T_1_specs = {
	sizeof(struct T),
	offsetof(struct T, _asn_ctx),
	asn_MAP_T_1_tag2el,
	4,	/* Count of tags in the map */
	-1,	/* Start extensions */
	-1	/* Stop extensions */
};
asn_TYPE_descriptor_t asn_DEF_T = {
	"T",
	"T",
	SEQUENCE_free,
	SEQUENCE_print,
	SEQUENCE_constraint,
	SEQUENCE_decode_ber,
	SEQUENCE_encode_der,
	SEQUENCE_decode_xer,
	SEQUENCE_encode_xer,
	0,	/* Use generic outmost tag fetcher */
	asn_DEF_T_1_tags,
	sizeof(asn_DEF_T_1_tags)
		/sizeof(asn_DEF_T_1_tags[0]), /* 1 */
	asn_DEF_T_1_tags,	/* Same as above */
	sizeof(asn_DEF_T_1_tags)
		/sizeof(asn_DEF_T_1_tags[0]), /* 1 */
	asn_MBR_T_1,
	4,	/* Elements count */
	&asn_SPC_T_1_specs	/* Additional specs */
};

