| /***************************************************************************** |
| # * |
| # Copyright 2019 AT&T Intellectual Property * |
| # * |
| # Licensed under the Apache License, Version 2.0 (the "License"); * |
| # you may not use this file except in compliance with the License. * |
| # You may obtain a copy of the License at * |
| # * |
| # http://www.apache.org/licenses/LICENSE-2.0 * |
| # * |
| # Unless required by applicable law or agreed to in writing, software * |
| # distributed under the License is distributed on an "AS IS" BASIS, * |
| # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * |
| # See the License for the specific language governing permissions and * |
| # limitations under the License. * |
| # * |
| ******************************************************************************/ |
| |
| /* |
| * Copyright (c) 2017 Lev Walkin <vlm@lionet.info>. All rights reserved. |
| * Redistribution and modifications are permitted subject to BSD license. |
| */ |
| #include <asn_internal.h> |
| #include <asn_application.h> |
| #include <errno.h> |
| |
| static asn_enc_rval_t asn_encode_internal(const asn_codec_ctx_t *opt_codec_ctx, |
| enum asn_transfer_syntax syntax, |
| const asn_TYPE_descriptor_t *td, |
| const void *sptr, |
| asn_app_consume_bytes_f *callback, |
| void *callback_key); |
| |
| |
| struct callback_count_bytes_key { |
| asn_app_consume_bytes_f *callback; |
| void *callback_key; |
| size_t computed_size; |
| }; |
| |
| /* |
| * Encoder which just counts bytes that come through it. |
| */ |
| static int |
| callback_count_bytes_cb(const void *data, size_t size, void *keyp) { |
| struct callback_count_bytes_key *key = keyp; |
| int ret; |
| |
| ret = key->callback(data, size, key->callback_key); |
| if(ret >= 0) { |
| key->computed_size += size; |
| } |
| |
| return ret; |
| } |
| |
| struct overrun_encoder_key { |
| void *buffer; |
| size_t buffer_size; |
| size_t computed_size; |
| }; |
| |
| struct dynamic_encoder_key { |
| void *buffer; |
| size_t buffer_size; |
| size_t computed_size; |
| }; |
| |
| struct callback_failure_catch_key { |
| asn_app_consume_bytes_f *callback; |
| void *callback_key; |
| int callback_failed; |
| }; |
| |
| /* |
| * Encoder which doesn't stop counting bytes |
| * even if it reaches the end of the buffer. |
| */ |
| static int |
| overrun_encoder_cb(const void *data, size_t size, void *keyp) { |
| struct overrun_encoder_key *key = keyp; |
| |
| if(key->computed_size + size > key->buffer_size) { |
| /* |
| * Avoid accident on the next call: |
| * stop adding bytes to the buffer. |
| */ |
| key->buffer_size = 0; |
| } else { |
| memcpy((char *)key->buffer + key->computed_size, data, size); |
| } |
| key->computed_size += size; |
| |
| return 0; |
| } |
| |
| /* |
| * Encoder which dynamically allocates output, and continues |
| * to count even if allocation failed. |
| */ |
| static int |
| dynamic_encoder_cb(const void *data, size_t size, void *keyp) { |
| struct dynamic_encoder_key *key = keyp; |
| |
| if(key->buffer) { |
| if(key->computed_size + size >= key->buffer_size) { |
| void *p; |
| size_t new_size = key->buffer_size; |
| |
| do { |
| new_size *= 2; |
| } while(new_size <= key->computed_size + size); |
| |
| p = REALLOC(key->buffer, new_size); |
| if(p) { |
| key->buffer = p; |
| key->buffer_size = new_size; |
| } else { |
| FREEMEM(key->buffer); |
| key->buffer = 0; |
| key->buffer_size = 0; |
| key->computed_size += size; |
| return 0; |
| } |
| } |
| memcpy((char *)key->buffer + key->computed_size, data, size); |
| } |
| |
| key->computed_size += size; |
| |
| return 0; |
| } |
| |
| /* |
| * Encoder which help convert the application level encoder failure into EIO. |
| */ |
| static int |
| callback_failure_catch_cb(const void *data, size_t size, void *keyp) { |
| struct callback_failure_catch_key *key = keyp; |
| int ret; |
| |
| ret = key->callback(data, size, key->callback_key); |
| if(ret < 0) { |
| key->callback_failed = 1; |
| } |
| |
| return ret; |
| } |
| |
| asn_enc_rval_t |
| asn_encode(const asn_codec_ctx_t *opt_codec_ctx, |
| enum asn_transfer_syntax syntax, const asn_TYPE_descriptor_t *td, |
| const void *sptr, asn_app_consume_bytes_f *callback, void *callback_key) { |
| struct callback_failure_catch_key cb_key; |
| asn_enc_rval_t er = {0,0,0}; |
| |
| if(!callback) { |
| errno = EINVAL; |
| ASN__ENCODE_FAILED; |
| } |
| |
| cb_key.callback = callback; |
| cb_key.callback_key = callback_key; |
| cb_key.callback_failed = 0; |
| |
| er = asn_encode_internal(opt_codec_ctx, syntax, td, sptr, |
| callback_failure_catch_cb, &cb_key); |
| if(cb_key.callback_failed) { |
| assert(er.encoded == -1); |
| assert(errno == EBADF); |
| errno = EIO; |
| } |
| |
| return er; |
| } |
| |
| asn_enc_rval_t |
| asn_encode_to_buffer(const asn_codec_ctx_t *opt_codec_ctx, |
| enum asn_transfer_syntax syntax, |
| const asn_TYPE_descriptor_t *td, const void *sptr, |
| void *buffer, size_t buffer_size) { |
| struct overrun_encoder_key buf_key; |
| asn_enc_rval_t er = {0,0,0}; |
| |
| if(buffer_size > 0 && !buffer) { |
| errno = EINVAL; |
| ASN__ENCODE_FAILED; |
| } |
| |
| buf_key.buffer = buffer; |
| buf_key.buffer_size = buffer_size; |
| buf_key.computed_size = 0; |
| |
| er = asn_encode_internal(opt_codec_ctx, syntax, td, sptr, |
| overrun_encoder_cb, &buf_key); |
| |
| if(er.encoded >= 0 && (size_t)er.encoded != buf_key.computed_size) { |
| ASN_DEBUG("asn_encode() returned %" ASN_PRI_SSIZE |
| " yet produced %" ASN_PRI_SIZE " bytes", |
| er.encoded, buf_key.computed_size); |
| assert(er.encoded < 0 || (size_t)er.encoded == buf_key.computed_size); |
| } |
| |
| return er; |
| } |
| |
| asn_encode_to_new_buffer_result_t |
| asn_encode_to_new_buffer(const asn_codec_ctx_t *opt_codec_ctx, |
| enum asn_transfer_syntax syntax, |
| const asn_TYPE_descriptor_t *td, const void *sptr) { |
| struct dynamic_encoder_key buf_key; |
| asn_encode_to_new_buffer_result_t res; |
| |
| buf_key.buffer_size = 16; |
| buf_key.buffer = MALLOC(buf_key.buffer_size); |
| buf_key.computed_size = 0; |
| |
| res.result = asn_encode_internal(opt_codec_ctx, syntax, td, sptr, |
| dynamic_encoder_cb, &buf_key); |
| |
| if(res.result.encoded >= 0 |
| && (size_t)res.result.encoded != buf_key.computed_size) { |
| ASN_DEBUG("asn_encode() returned %" ASN_PRI_SSIZE |
| " yet produced %" ASN_PRI_SIZE " bytes", |
| res.result.encoded, buf_key.computed_size); |
| assert(res.result.encoded < 0 |
| || (size_t)res.result.encoded == buf_key.computed_size); |
| } |
| |
| res.buffer = buf_key.buffer; |
| |
| /* 0-terminate just in case. */ |
| if(res.buffer) { |
| assert(buf_key.computed_size < buf_key.buffer_size); |
| ((char *)res.buffer)[buf_key.computed_size] = '\0'; |
| } |
| |
| return res; |
| } |
| |
| static asn_enc_rval_t |
| asn_encode_internal(const asn_codec_ctx_t *opt_codec_ctx, |
| enum asn_transfer_syntax syntax, |
| const asn_TYPE_descriptor_t *td, const void *sptr, |
| asn_app_consume_bytes_f *callback, void *callback_key) { |
| asn_enc_rval_t er = {0,0,0}; |
| enum xer_encoder_flags_e xer_flags = XER_F_CANONICAL; |
| |
| (void)opt_codec_ctx; /* Parameters are not checked on encode yet. */ |
| |
| if(!td || !sptr) { |
| errno = EINVAL; |
| ASN__ENCODE_FAILED; |
| } |
| |
| switch(syntax) { |
| case ATS_NONSTANDARD_PLAINTEXT: |
| if(td->op->print_struct) { |
| struct callback_count_bytes_key cb_key; |
| cb_key.callback = callback; |
| cb_key.callback_key = callback_key; |
| cb_key.computed_size = 0; |
| if(td->op->print_struct(td, sptr, 1, callback_count_bytes_cb, |
| &cb_key) |
| < 0 |
| || callback_count_bytes_cb("\n", 1, &cb_key) < 0) { |
| errno = EBADF; /* Structure has incorrect form. */ |
| er.encoded = -1; |
| er.failed_type = td; |
| er.structure_ptr = sptr; |
| } else { |
| er.encoded = cb_key.computed_size; |
| er.failed_type = 0; |
| er.structure_ptr = 0; |
| } |
| } else { |
| errno = ENOENT; /* Transfer syntax is not defined for this type. */ |
| ASN__ENCODE_FAILED; |
| } |
| break; |
| |
| case ATS_RANDOM: |
| errno = ENOENT; /* Randomization doesn't make sense on output. */ |
| ASN__ENCODE_FAILED; |
| |
| case ATS_BER: |
| /* BER is a superset of DER. */ |
| /* Fall through. */ |
| case ATS_DER: |
| if(td->op->der_encoder) { |
| er = der_encode(td, sptr, callback, callback_key); |
| if(er.encoded == -1) { |
| if(er.failed_type && er.failed_type->op->der_encoder) { |
| errno = EBADF; /* Structure has incorrect form. */ |
| } else { |
| errno = ENOENT; /* DER is not defined for this type. */ |
| } |
| } |
| } else { |
| errno = ENOENT; /* Transfer syntax is not defined for this type. */ |
| ASN__ENCODE_FAILED; |
| } |
| break; |
| case ATS_CER: |
| errno = ENOENT; /* Transfer syntax is not defined for any type. */ |
| ASN__ENCODE_FAILED; |
| |
| #ifdef ASN_DISABLE_OER_SUPPORT |
| case ATS_BASIC_OER: |
| case ATS_CANONICAL_OER: |
| errno = ENOENT; /* PER is not defined. */ |
| ASN__ENCODE_FAILED; |
| break; |
| #else /* ASN_DISABLE_OER_SUPPORT */ |
| case ATS_BASIC_OER: |
| /* CANONICAL-OER is a superset of BASIC-OER. */ |
| /* Fall through. */ |
| case ATS_CANONICAL_OER: |
| if(td->op->oer_encoder) { |
| er = oer_encode(td, sptr, callback, callback_key); |
| if(er.encoded == -1) { |
| if(er.failed_type && er.failed_type->op->oer_encoder) { |
| errno = EBADF; /* Structure has incorrect form. */ |
| } else { |
| errno = ENOENT; /* OER is not defined for this type. */ |
| } |
| } |
| } else { |
| errno = ENOENT; /* Transfer syntax is not defined for this type. */ |
| ASN__ENCODE_FAILED; |
| } |
| break; |
| #endif /* ASN_DISABLE_OER_SUPPORT */ |
| |
| #ifdef ASN_DISABLE_PER_SUPPORT |
| case ATS_UNALIGNED_BASIC_PER: |
| case ATS_UNALIGNED_CANONICAL_PER: |
| case ATS_ALIGNED_BASIC_PER: |
| case ATS_ALIGNED_CANONICAL_PER: |
| errno = ENOENT; /* PER is not defined. */ |
| ASN__ENCODE_FAILED; |
| break; |
| #else /* ASN_DISABLE_PER_SUPPORT */ |
| case ATS_UNALIGNED_BASIC_PER: |
| /* CANONICAL-UPER is a superset of BASIC-UPER. */ |
| /* Fall through. */ |
| case ATS_UNALIGNED_CANONICAL_PER: |
| if(td->op->uper_encoder) { |
| er = uper_encode(td, 0, sptr, callback, callback_key); |
| if(er.encoded == -1) { |
| if(er.failed_type && er.failed_type->op->uper_encoder) { |
| errno = EBADF; /* Structure has incorrect form. */ |
| } else { |
| errno = ENOENT; /* UPER is not defined for this type. */ |
| } |
| } else { |
| ASN_DEBUG("Complete encoded in %ld bits", (long)er.encoded); |
| if(er.encoded == 0) { |
| /* Enforce "Complete Encoding" of X.691 #11.1 */ |
| if(callback("\0", 1, callback_key) < 0) { |
| errno = EBADF; |
| ASN__ENCODE_FAILED; |
| } |
| er.encoded = 8; /* Exactly 8 zero bits is added. */ |
| } |
| /* Convert bits into bytes */ |
| er.encoded = (er.encoded + 7) >> 3; |
| } |
| } else { |
| errno = ENOENT; /* Transfer syntax is not defined for this type. */ |
| ASN__ENCODE_FAILED; |
| } |
| break; |
| case ATS_ALIGNED_BASIC_PER: |
| /* CANONICAL-APER is a superset of BASIC-APER. */ |
| /* Fall through. */ |
| case ATS_ALIGNED_CANONICAL_PER: |
| if(td->op->aper_encoder) { |
| er = aper_encode(td, 0, sptr, callback, callback_key); |
| if(er.encoded == -1) { |
| if(er.failed_type && er.failed_type->op->aper_encoder) { |
| errno = EBADF; /* Structure has incorrect form. */ |
| } else { |
| errno = ENOENT; /* APER is not defined for this type. */ |
| } |
| } else { |
| ASN_DEBUG("Complete encoded in %ld bits", (long)er.encoded); |
| if(er.encoded == 0) { |
| /* Enforce "Complete Encoding" of X.691 #11.1 */ |
| if(callback("\0", 1, callback_key) < 0) { |
| errno = EBADF; |
| ASN__ENCODE_FAILED; |
| } |
| er.encoded = 8; /* Exactly 8 zero bits is added. */ |
| } |
| /* Convert bits into bytes */ |
| er.encoded = (er.encoded + 7) >> 3; |
| } |
| } else { |
| errno = ENOENT; /* Transfer syntax is not defined for this type. */ |
| ASN__ENCODE_FAILED; |
| } |
| break; |
| #endif /* ASN_DISABLE_PER_SUPPORT */ |
| |
| case ATS_BASIC_XER: |
| /* CANONICAL-XER is a superset of BASIC-XER. */ |
| xer_flags &= ~XER_F_CANONICAL; |
| xer_flags |= XER_F_BASIC; |
| /* Fall through. */ |
| case ATS_CANONICAL_XER: |
| if(td->op->xer_encoder) { |
| er = xer_encode(td, sptr, xer_flags, callback, callback_key); |
| if(er.encoded == -1) { |
| if(er.failed_type && er.failed_type->op->xer_encoder) { |
| errno = EBADF; /* Structure has incorrect form. */ |
| } else { |
| errno = ENOENT; /* XER is not defined for this type. */ |
| } |
| } |
| } else { |
| errno = ENOENT; /* Transfer syntax is not defined for this type. */ |
| ASN__ENCODE_FAILED; |
| } |
| break; |
| |
| default: |
| errno = ENOENT; |
| ASN__ENCODE_FAILED; |
| } |
| |
| return er; |
| } |
| |
| asn_dec_rval_t |
| asn_decode(const asn_codec_ctx_t *opt_codec_ctx, |
| enum asn_transfer_syntax syntax, const asn_TYPE_descriptor_t *td, |
| void **sptr, const void *buffer, size_t size) { |
| if(!td || !td->op || !sptr || (size && !buffer)) { |
| ASN__DECODE_FAILED; |
| } |
| |
| switch(syntax) { |
| case ATS_CER: |
| case ATS_NONSTANDARD_PLAINTEXT: |
| default: |
| errno = ENOENT; |
| ASN__DECODE_FAILED; |
| |
| case ATS_RANDOM: |
| if(!td->op->random_fill) { |
| ASN__DECODE_FAILED; |
| } else { |
| if(asn_random_fill(td, sptr, 16000) == 0) { |
| asn_dec_rval_t ret = {RC_OK, 0}; |
| return ret; |
| } else { |
| ASN__DECODE_FAILED; |
| } |
| } |
| break; |
| |
| case ATS_DER: |
| case ATS_BER: |
| return ber_decode(opt_codec_ctx, td, sptr, buffer, size); |
| |
| case ATS_BASIC_OER: |
| case ATS_CANONICAL_OER: |
| #ifdef ASN_DISABLE_OER_SUPPORT |
| errno = ENOENT; |
| ASN__DECODE_FAILED; |
| #else |
| return oer_decode(opt_codec_ctx, td, sptr, buffer, size); |
| #endif |
| |
| case ATS_UNALIGNED_BASIC_PER: |
| case ATS_UNALIGNED_CANONICAL_PER: |
| #ifdef ASN_DISABLE_PER_SUPPORT |
| errno = ENOENT; |
| ASN__DECODE_FAILED; |
| #else |
| return uper_decode_complete(opt_codec_ctx, td, sptr, buffer, size); |
| #endif |
| |
| case ATS_ALIGNED_BASIC_PER: |
| case ATS_ALIGNED_CANONICAL_PER: |
| #ifdef ASN_DISABLE_PER_SUPPORT |
| errno = ENOENT; |
| ASN__DECODE_FAILED; |
| #else |
| return aper_decode_complete(opt_codec_ctx, td, sptr, buffer, size); |
| #endif |
| |
| case ATS_BASIC_XER: |
| case ATS_CANONICAL_XER: |
| return xer_decode(opt_codec_ctx, td, sptr, buffer, size); |
| } |
| } |
| |