blob: c04577885f8fcb5de7c09dc0483afb761aef61df [file] [log] [blame]
vlmfa67ddc2004-06-03 03:38:44 +00001/*-
2 * Copyright (c) 2003, 2004 Lev Walkin <vlm@lionet.info>. All rights reserved.
3 * Redistribution and modifications are permitted subject to BSD license.
4 */
vlm39ba4c42004-09-22 16:06:28 +00005#include <asn_internal.h>
vlmfa67ddc2004-06-03 03:38:44 +00006#include <UTF8String.h>
7
8/*
9 * UTF8String basic type description.
10 */
vlmef6355b2004-09-29 13:26:15 +000011static ber_tlv_tag_t asn_DEF_UTF8String_tags[] = {
vlm72425de2004-09-13 08:31:01 +000012 (ASN_TAG_CLASS_UNIVERSAL | (12 << 2)), /* [UNIVERSAL 12] IMPLICIT ...*/
13 (ASN_TAG_CLASS_UNIVERSAL | (4 << 2)), /* ... OCTET STRING */
vlmfa67ddc2004-06-03 03:38:44 +000014};
vlmef6355b2004-09-29 13:26:15 +000015asn_TYPE_descriptor_t asn_DEF_UTF8String = {
vlmfa67ddc2004-06-03 03:38:44 +000016 "UTF8String",
vlm39ba4c42004-09-22 16:06:28 +000017 OCTET_STRING_free,
18 UTF8String_print,
vlmfa67ddc2004-06-03 03:38:44 +000019 UTF8String_constraint, /* Check for invalid codes, etc. */
20 OCTET_STRING_decode_ber, /* Implemented in terms of OCTET STRING */
21 OCTET_STRING_encode_der, /* Implemented in terms of OCTET STRING */
vlm39ba4c42004-09-22 16:06:28 +000022 0, /* Not implemented yet */
23 OCTET_STRING_encode_xer_ascii, /* Already in UTF-8 format */
vlmfa67ddc2004-06-03 03:38:44 +000024 0, /* Use generic outmost tag fetcher */
vlmef6355b2004-09-29 13:26:15 +000025 asn_DEF_UTF8String_tags,
26 sizeof(asn_DEF_UTF8String_tags)
27 / sizeof(asn_DEF_UTF8String_tags[0]) - 1,
28 asn_DEF_UTF8String_tags,
29 sizeof(asn_DEF_UTF8String_tags)
30 / sizeof(asn_DEF_UTF8String_tags[0]),
vlme413c122004-08-20 13:23:42 +000031 0, 0, /* No members */
vlmb42843a2004-06-05 08:17:50 +000032 0 /* No specifics */
vlmfa67ddc2004-06-03 03:38:44 +000033};
34
vlm51283292004-10-02 11:37:38 +000035/*
36 * This is the table of length expectations.
37 * The second half of this table is only applicable to the long sequentes.
38 */
39static int UTF8String_ht[2][16] = {
40 { /* 0x0 ... 0x7 */
41 /* 0000..0111 */
42 1, 1, 1, 1, 1, 1, 1, 1,
43 /* 1000..1011(0), 1100..1101(2), 1110(3), 1111(-1) */
44 0, 0, 0, 0, 2, 2, 3, -1 },
45 { /* 0xF0 .. 0xF7 */
46 /* 11110000..11110111 */
47 4, 4, 4, 4, 4, 4, 4, 4,
48 5, 5, 5, 5, 6, 6, -1, -1 }
vlmfa67ddc2004-06-03 03:38:44 +000049};
50
51int
vlmef6355b2004-09-29 13:26:15 +000052UTF8String_constraint(asn_TYPE_descriptor_t *td, const void *sptr,
vlmfa67ddc2004-06-03 03:38:44 +000053 asn_app_consume_bytes_f *app_errlog, void *app_key) {
54 ssize_t len;
vlmda674682004-08-11 09:07:36 +000055 len = UTF8String_length((const UTF8String_t *)sptr, td->name,
56 app_errlog, app_key);
vlmfa67ddc2004-06-03 03:38:44 +000057 if(len > 0) len = 0;
58 return len;
59}
60
61ssize_t
62UTF8String_length(const UTF8String_t *st, const char *opt_type_name,
63 asn_app_consume_bytes_f *app_errlog, void *app_key) {
64
65 if(st && st->buf) {
66 size_t length = 0;
67 uint8_t *buf = st->buf;
68 uint8_t *end = buf + st->size;
69 int want; /* Number of bytes wanted */
70
71 for(want = 0; buf < end; buf++) {
72 uint8_t ch = *buf;
vlm51283292004-10-02 11:37:38 +000073 int w = UTF8String_ht[0][ch >> 4];
vlmfa67ddc2004-06-03 03:38:44 +000074 if(want) { /* Continuation expected */
75 if(w) {
vlme3f0f282004-08-11 09:44:13 +000076 _ASN_ERRLOG(app_errlog, app_key,
77 "%s: UTF-8 expectation "
vlm758530a2004-08-22 13:47:59 +000078 "failed at byte %d (%s:%d)",
vlmfa67ddc2004-06-03 03:38:44 +000079 opt_type_name,
vlm758530a2004-08-22 13:47:59 +000080 (buf - st->buf) + 1,
81 __FILE__, __LINE__);
vlmfa67ddc2004-06-03 03:38:44 +000082 return -1;
83 }
84 want--;
85 } else {
86 switch(w) {
87 case -1: /* Long UTF-8 */
vlm51283292004-10-02 11:37:38 +000088 w = UTF8String_ht[1][ch & 0x0F];
vlmfa67ddc2004-06-03 03:38:44 +000089 if(w != -1)
90 break;
91 /* Fall through */
vlm51283292004-10-02 11:37:38 +000092 case 0: /* But we should want something! */
vlme3f0f282004-08-11 09:44:13 +000093 _ASN_ERRLOG(app_errlog, app_key,
vlmfa67ddc2004-06-03 03:38:44 +000094 "%s: UTF-8 expectation"
vlm758530a2004-08-22 13:47:59 +000095 "failed at byte %d (%s:%d)",
vlmfa67ddc2004-06-03 03:38:44 +000096 opt_type_name,
vlm758530a2004-08-22 13:47:59 +000097 (buf - st->buf) + 1,
98 __FILE__, __LINE__);
vlmfa67ddc2004-06-03 03:38:44 +000099 return -1;
100 }
101 want = w - 1; /* Expect this much */
102 }
103 if(!want) length++;
104 }
105
106 /* If still want something, then something is wrong */
107 if(want) {
vlme3f0f282004-08-11 09:44:13 +0000108 _ASN_ERRLOG(app_errlog, app_key,
vlm758530a2004-08-22 13:47:59 +0000109 "%s: truncated UTF-8 sequence (%s:%d)",
110 opt_type_name, __FILE__, __LINE__);
vlmfa67ddc2004-06-03 03:38:44 +0000111 return -1;
112 }
113
114 return length;
115 } else {
vlme3f0f282004-08-11 09:44:13 +0000116 _ASN_ERRLOG(app_errlog, app_key,
117 "%s: value not given", opt_type_name);
vlmfa67ddc2004-06-03 03:38:44 +0000118 return -1;
119 }
120}
121
122int
vlmef6355b2004-09-29 13:26:15 +0000123UTF8String_print(asn_TYPE_descriptor_t *td, const void *sptr, int ilevel,
vlmfa67ddc2004-06-03 03:38:44 +0000124 asn_app_consume_bytes_f *cb, void *app_key) {
vlmda674682004-08-11 09:07:36 +0000125 const UTF8String_t *st = (const UTF8String_t *)sptr;
vlmfa67ddc2004-06-03 03:38:44 +0000126
vlmb42843a2004-06-05 08:17:50 +0000127 (void)td; /* Unused argument */
128 (void)ilevel; /* Unused argument */
129
vlmfa67ddc2004-06-03 03:38:44 +0000130 if(st && st->buf) {
vlm6678cb12004-09-26 13:10:40 +0000131 return (cb(st->buf, st->size, app_key) < 0) ? -1 : 0;
vlmfa67ddc2004-06-03 03:38:44 +0000132 } else {
vlm6678cb12004-09-26 13:10:40 +0000133 return (cb("<absent>", 8, app_key) < 0) ? -1 : 0;
vlmfa67ddc2004-06-03 03:38:44 +0000134 }
135}