blob: 1554220f87ab7f5ba230b939343b989a33bab0bf [file] [log] [blame]
vlmfa67ddc2004-06-03 03:38:44 +00001/*-
vlm9bfb2d12007-07-23 07:47:04 +00002 * Copyright (c) 2004, 2007 Lev Walkin <vlm@lionet.info>. All rights reserved.
vlmfa67ddc2004-06-03 03:38:44 +00003 * Redistribution and modifications are permitted subject to BSD license.
4 */
5/*
vlmc03306f2006-03-18 05:06:57 +00006 * Read the NativeInteger.h for the explanation wrt. differences between
vlmfa67ddc2004-06-03 03:38:44 +00007 * INTEGER and NativeInteger.
8 * Basically, both are decoders and encoders of ASN.1 INTEGER type, but this
9 * implementation deals with the standard (machine-specific) representation
10 * of them instead of using the platform-independent buffer.
11 */
vlm39ba4c42004-09-22 16:06:28 +000012#include <asn_internal.h>
vlmfa67ddc2004-06-03 03:38:44 +000013#include <NativeEnumerated.h>
14
15/*
16 * NativeEnumerated basic type description.
17 */
vlmef6355b2004-09-29 13:26:15 +000018static ber_tlv_tag_t asn_DEF_NativeEnumerated_tags[] = {
vlmfa67ddc2004-06-03 03:38:44 +000019 (ASN_TAG_CLASS_UNIVERSAL | (10 << 2))
20};
vlmef6355b2004-09-29 13:26:15 +000021asn_TYPE_descriptor_t asn_DEF_NativeEnumerated = {
vlmfa67ddc2004-06-03 03:38:44 +000022 "ENUMERATED", /* The ASN.1 type is still ENUMERATED */
vlm9de248e2004-10-20 15:50:55 +000023 "ENUMERATED",
vlm39ba4c42004-09-22 16:06:28 +000024 NativeInteger_free,
25 NativeInteger_print,
vlmfa67ddc2004-06-03 03:38:44 +000026 asn_generic_no_constraint,
27 NativeInteger_decode_ber,
28 NativeInteger_encode_der,
vlmd52f9b52004-10-21 14:13:48 +000029 NativeInteger_decode_xer,
vlm75fddcc2005-03-29 17:19:53 +000030 NativeEnumerated_encode_xer,
vlm337167e2005-11-26 11:25:14 +000031 NativeEnumerated_decode_uper,
vlm18dd82c2006-08-18 01:34:18 +000032 NativeEnumerated_encode_uper,
vlmfa67ddc2004-06-03 03:38:44 +000033 0, /* Use generic outmost tag fetcher */
vlmef6355b2004-09-29 13:26:15 +000034 asn_DEF_NativeEnumerated_tags,
35 sizeof(asn_DEF_NativeEnumerated_tags) / sizeof(asn_DEF_NativeEnumerated_tags[0]),
36 asn_DEF_NativeEnumerated_tags, /* Same as above */
37 sizeof(asn_DEF_NativeEnumerated_tags) / sizeof(asn_DEF_NativeEnumerated_tags[0]),
vlm337167e2005-11-26 11:25:14 +000038 0, /* No PER visible constraints */
vlme413c122004-08-20 13:23:42 +000039 0, 0, /* No members */
vlmb42843a2004-06-05 08:17:50 +000040 0 /* No specifics */
vlmfa67ddc2004-06-03 03:38:44 +000041};
vlme413c122004-08-20 13:23:42 +000042
vlm75fddcc2005-03-29 17:19:53 +000043asn_enc_rval_t
44NativeEnumerated_encode_xer(asn_TYPE_descriptor_t *td, void *sptr,
45 int ilevel, enum xer_encoder_flags_e flags,
46 asn_app_consume_bytes_f *cb, void *app_key) {
47 asn_INTEGER_specifics_t *specs=(asn_INTEGER_specifics_t *)td->specifics;
48 asn_enc_rval_t er;
49 const long *native = (const long *)sptr;
50 const asn_INTEGER_enum_map_t *el;
51
52 (void)ilevel;
53 (void)flags;
54
55 if(!native) _ASN_ENCODE_FAILED;
56
57 el = INTEGER_map_value2enum(specs, *native);
58 if(el) {
59 size_t srcsize = el->enum_len + 5;
60 char *src = (char *)alloca(srcsize);
61
62 er.encoded = snprintf(src, srcsize, "<%s/>", el->enum_name);
63 assert(er.encoded > 0 && (size_t)er.encoded < srcsize);
64 if(cb(src, er.encoded, app_key) < 0) _ASN_ENCODE_FAILED;
vlm337167e2005-11-26 11:25:14 +000065 _ASN_ENCODED_OK(er);
vlm75fddcc2005-03-29 17:19:53 +000066 } else {
67 ASN_DEBUG("ASN.1 forbids dealing with "
68 "unknown value of ENUMERATED type");
69 _ASN_ENCODE_FAILED;
70 }
vlm337167e2005-11-26 11:25:14 +000071}
vlm75fddcc2005-03-29 17:19:53 +000072
vlm337167e2005-11-26 11:25:14 +000073asn_dec_rval_t
vlm18dd82c2006-08-18 01:34:18 +000074NativeEnumerated_decode_uper(asn_codec_ctx_t *opt_codec_ctx,
75 asn_TYPE_descriptor_t *td, asn_per_constraints_t *constraints,
76 void **sptr, asn_per_data_t *pd) {
vlm337167e2005-11-26 11:25:14 +000077 asn_INTEGER_specifics_t *specs = (asn_INTEGER_specifics_t *)td->specifics;
78 asn_dec_rval_t rval = { RC_OK, 0 };
79 long *native = (long *)*sptr;
80 asn_per_constraint_t *ct;
81 long value;
82
83 (void)opt_codec_ctx;
84
85 if(constraints) ct = &constraints->value;
86 else if(td->per_constraints) ct = &td->per_constraints->value;
87 else _ASN_DECODE_FAILED; /* Mandatory! */
88 if(!specs) _ASN_DECODE_FAILED;
89
90 if(!native) {
91 native = (long *)(*sptr = CALLOC(1, sizeof(*native)));
92 if(!native) _ASN_DECODE_FAILED;
93 }
94
95 ASN_DEBUG("Decoding %s as NativeEnumerated", td->name);
96
97 if(ct->flags & APC_EXTENSIBLE) {
98 int inext = per_get_few_bits(pd, 1);
vlm619448e2006-09-18 20:05:55 +000099 if(inext < 0) _ASN_DECODE_STARVED;
vlm337167e2005-11-26 11:25:14 +0000100 if(inext) ct = 0;
101 }
102
103 if(ct && ct->range_bits >= 0) {
104 value = per_get_few_bits(pd, ct->range_bits);
vlm619448e2006-09-18 20:05:55 +0000105 if(value < 0) _ASN_DECODE_STARVED;
vlm337167e2005-11-26 11:25:14 +0000106 if(value >= (specs->extension
107 ? specs->extension - 1 : specs->map_count))
108 _ASN_DECODE_FAILED;
109 } else {
110 if(!specs->extension)
111 _ASN_DECODE_FAILED;
112 /*
113 * X.691, #10.6: normally small non-negative whole number;
114 */
115 value = uper_get_nsnnwn(pd);
vlm619448e2006-09-18 20:05:55 +0000116 if(value < 0) _ASN_DECODE_STARVED;
vlm337167e2005-11-26 11:25:14 +0000117 value += specs->extension - 1;
118 if(value >= specs->map_count)
119 _ASN_DECODE_FAILED;
120 }
121
122 *native = specs->value2enum[value].nat_value;
123 ASN_DEBUG("Decoded %s = %ld", td->name, *native);
124
125 return rval;
vlm75fddcc2005-03-29 17:19:53 +0000126}
127
vlm18dd82c2006-08-18 01:34:18 +0000128static int
129NativeEnumerated__compar_value2enum(const void *ap, const void *bp) {
130 const asn_INTEGER_enum_map_t *a = ap;
131 const asn_INTEGER_enum_map_t *b = bp;
132 if(a->nat_value == b->nat_value)
133 return 0;
134 if(a->nat_value < b->nat_value)
135 return -1;
136 return 1;
137}
138
139asn_enc_rval_t
140NativeEnumerated_encode_uper(asn_TYPE_descriptor_t *td,
141 asn_per_constraints_t *constraints, void *sptr, asn_per_outp_t *po) {
142 asn_INTEGER_specifics_t *specs = (asn_INTEGER_specifics_t *)td->specifics;
143 asn_enc_rval_t er;
144 long native, value;
145 asn_per_constraint_t *ct;
146 int inext = 0;
147 asn_INTEGER_enum_map_t key;
148 asn_INTEGER_enum_map_t *kf;
149
150 if(!sptr) _ASN_ENCODE_FAILED;
151 if(!specs) _ASN_ENCODE_FAILED;
152
153 if(constraints) ct = &constraints->value;
154 else if(td->per_constraints) ct = &td->per_constraints->value;
155 else _ASN_ENCODE_FAILED; /* Mandatory! */
156
157 ASN_DEBUG("Encoding %s as NativeEnumerated", td->name);
158
159 er.encoded = 0;
160
161 native = *(long *)sptr;
162 if(native < 0) _ASN_ENCODE_FAILED;
163
164 key.nat_value = native;
165 kf = bsearch(&key, specs->value2enum, specs->map_count,
166 sizeof(key), NativeEnumerated__compar_value2enum);
167 if(!kf) {
168 ASN_DEBUG("No element corresponds to %ld", native);
169 _ASN_ENCODE_FAILED;
170 }
171 value = kf - specs->value2enum;
172
173 if(ct->range_bits >= 0) {
174 int cmpWith = specs->extension
175 ? specs->extension - 1 : specs->map_count;
176 if(value >= cmpWith)
177 inext = 1;
178 }
179 if(ct->flags & APC_EXTENSIBLE) {
vlm9bfb2d12007-07-23 07:47:04 +0000180 if(per_put_few_bits(po, inext, 1))
vlm18dd82c2006-08-18 01:34:18 +0000181 _ASN_ENCODE_FAILED;
vlmd024d0b2007-07-23 09:57:00 +0000182 if(inext) ct = 0;
vlm18dd82c2006-08-18 01:34:18 +0000183 } else if(inext) {
184 _ASN_ENCODE_FAILED;
185 }
186
187 if(ct && ct->range_bits >= 0) {
188 if(per_put_few_bits(po, value, ct->range_bits))
189 _ASN_ENCODE_FAILED;
190 _ASN_ENCODED_OK(er);
191 }
192
193 if(!specs->extension)
194 _ASN_ENCODE_FAILED;
195
196 /*
197 * X.691, #10.6: normally small non-negative whole number;
198 */
vlm12164c42007-12-03 13:41:36 +0000199 ASN_DEBUG("value = %ld, ext = %d, inext = %d, res = %ld",
vlm9bfb2d12007-07-23 07:47:04 +0000200 value, specs->extension, inext,
201 value - (inext ? (specs->extension - 1) : 0));
202 if(uper_put_nsnnwn(po, value - (inext ? (specs->extension - 1) : 0)))
vlm18dd82c2006-08-18 01:34:18 +0000203 _ASN_ENCODE_FAILED;
204
205 _ASN_ENCODED_OK(er);
206}
207