blob: 25b3bfac8d7be655e87d06cdfac6b7ad093f71ab [file] [log] [blame]
Lev Walkina9cc46e2004-09-22 16:06:28 +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 */
5#include <asn_internal.h>
Lev Walkina9cc46e2004-09-22 16:06:28 +00006#include <stdio.h>
Lev Walkina9cc46e2004-09-22 16:06:28 +00007#include <errno.h>
8
9/*
10 * The XER encoder of any type. May be invoked by the application.
11 */
12asn_enc_rval_t
Lev Walkin5e033762004-09-29 13:26:15 +000013xer_encode(asn_TYPE_descriptor_t *td, void *sptr,
Lev Walkina9cc46e2004-09-22 16:06:28 +000014 enum xer_encoder_flags_e xer_flags,
15 asn_app_consume_bytes_f *cb, void *app_key) {
16 asn_enc_rval_t er, tmper;
17 const char *mname;
18 size_t mlen;
19 int xcan = (xer_flags & XER_F_CANONICAL) ? 1 : 2;
20
Lev Walkin942fd082004-10-03 09:13:02 +000021 if(!td || !sptr) goto cb_failed;
Lev Walkina9cc46e2004-09-22 16:06:28 +000022
Lev Walkindc06f6b2004-10-20 15:50:55 +000023 mname = td->xml_tag;
Lev Walkina9cc46e2004-09-22 16:06:28 +000024 mlen = strlen(mname);
25
Lev Walkin7c1dc052016-03-14 03:08:15 -070026 ASN__CALLBACK3("<", 1, mname, mlen, ">", 1);
Lev Walkina9cc46e2004-09-22 16:06:28 +000027
Bi-Ruei, Chiu1f87ac02017-08-20 01:25:45 +080028 tmper = td->op->xer_encoder(td, sptr, 1, xer_flags, cb, app_key);
Lev Walkina9cc46e2004-09-22 16:06:28 +000029 if(tmper.encoded == -1) return tmper;
30
Lev Walkin7c1dc052016-03-14 03:08:15 -070031 ASN__CALLBACK3("</", 2, mname, mlen, ">\n", xcan);
Lev Walkina9cc46e2004-09-22 16:06:28 +000032
Lev Walkin942fd082004-10-03 09:13:02 +000033 er.encoded = 4 + xcan + (2 * mlen) + tmper.encoded;
Lev Walkina9cc46e2004-09-22 16:06:28 +000034
Lev Walkin7c1dc052016-03-14 03:08:15 -070035 ASN__ENCODED_OK(er);
Lev Walkin942fd082004-10-03 09:13:02 +000036cb_failed:
Lev Walkin7c1dc052016-03-14 03:08:15 -070037 ASN__ENCODE_FAILED;
Lev Walkina9cc46e2004-09-22 16:06:28 +000038}
39
Lev Walkincc6a9102004-09-23 22:06:26 +000040/*
Lev Walkin11c3e172004-09-24 21:00:50 +000041 * This is a helper function for xer_fprint, which directs all incoming data
Lev Walkincc6a9102004-09-23 22:06:26 +000042 * into the provided file descriptor.
43 */
44static int
45xer__print2fp(const void *buffer, size_t size, void *app_key) {
46 FILE *stream = (FILE *)app_key;
47
48 if(fwrite(buffer, 1, size, stream) != size)
49 return -1;
50
51 return 0;
52}
Lev Walkina9cc46e2004-09-22 16:06:28 +000053
54int
Lev Walkin5e033762004-09-29 13:26:15 +000055xer_fprint(FILE *stream, asn_TYPE_descriptor_t *td, void *sptr) {
Lev Walkina9cc46e2004-09-22 16:06:28 +000056 asn_enc_rval_t er;
57
58 if(!stream) stream = stdout;
59 if(!td || !sptr)
60 return -1;
61
Lev Walkincc6a9102004-09-23 22:06:26 +000062 er = xer_encode(td, sptr, XER_F_BASIC, xer__print2fp, stream);
Lev Walkina9cc46e2004-09-22 16:06:28 +000063 if(er.encoded == -1)
64 return -1;
65
66 return fflush(stream);
67}
Lev Walkine4589f92017-07-25 09:31:33 -070068
69struct xer_buffer {
70 char *buffer;
71 size_t buffer_size;
72 size_t allocated_size;
73};
74
75static int
76xer__buffer_append(const void *buffer, size_t size, void *app_key) {
77 struct xer_buffer *xb = app_key;
78
79 while(xb->buffer_size + size + 1 > xb->allocated_size) {
80 size_t new_size = 2 * (xb->allocated_size ? xb->allocated_size : 64);
81 char *new_buf = MALLOC(new_size);
82 if(!new_buf) return -1;
83 memcpy(new_buf, xb->buffer, xb->buffer_size);
84 FREEMEM(xb->buffer);
85 xb->buffer = new_buf;
86 xb->allocated_size = new_size;
87 }
88
89 memcpy(xb->buffer + xb->buffer_size, buffer, size);
90 xb->buffer_size += size;
91 xb->buffer[xb->buffer_size] = '\0';
92 return 0;
93}
94
95enum xer_equivalence_e
96xer_equivalent(struct asn_TYPE_descriptor_s *td, void *struct1,
97 void *struct2, FILE *opt_debug_stream) {
98 struct xer_buffer xb1 = {0, 0, 0};
99 struct xer_buffer xb2 = {0, 0, 0};
100 asn_enc_rval_t e1, e2;
101 asn_dec_rval_t rval;
102 void *sptr = NULL;
103
104 if(!td || !struct1 || !struct2) {
105 if(opt_debug_stream) {
106 if(!td) fprintf(opt_debug_stream, "Type descriptor missing\n");
107 if(!struct1) fprintf(opt_debug_stream, "Structure 1 missing\n");
108 if(!struct2) fprintf(opt_debug_stream, "Structure 2 missing\n");
109 }
110 return XEQ_FAILURE;
111 }
112
113 e1 = xer_encode(td, struct1, XER_F_BASIC, xer__buffer_append, &xb1);
114 if(e1.encoded == -1) {
115 if(opt_debug_stream) {
116 fprintf(stderr, "XER Encoding of %s failed\n", td->name);
117 }
118 FREEMEM(xb1.buffer);
119 return XEQ_ENCODE1_FAILED;
120 }
121
122 e2 = xer_encode(td, struct2, XER_F_BASIC, xer__buffer_append, &xb2);
123 if(e2.encoded == -1) {
124 if(opt_debug_stream) {
125 fprintf(stderr, "XER Encoding of %s failed\n", td->name);
126 }
127 FREEMEM(xb1.buffer);
128 FREEMEM(xb2.buffer);
129 return XEQ_ENCODE1_FAILED;
130 }
131
132 if(xb1.buffer_size != xb2.buffer_size
133 || memcmp(xb1.buffer, xb2.buffer, xb1.buffer_size) != 0) {
134 if(opt_debug_stream) {
135 fprintf(opt_debug_stream,
136 "Structures XER-encoded into different byte streams:\n=== "
137 "Structure 1 ===\n%s\n=== Structure 2 ===\n%s\n",
138 xb1.buffer, xb2.buffer);
139 }
140 FREEMEM(xb1.buffer);
141 FREEMEM(xb2.buffer);
142 return XEQ_DIFFERENT;
143 } else {
144 if(opt_debug_stream) {
145 fprintf(opt_debug_stream,
146 "Both structures encoded into the same XER byte stream "
147 "of size %zu:\n%s",
148 xb1.buffer_size, xb1.buffer);
149 }
150 }
151
152 rval = xer_decode(NULL, td, (void **)&sptr, xb1.buffer,
153 xb1.buffer_size);
154 switch(rval.code) {
155 case RC_OK:
156 break;
157 case RC_WMORE:
158 if(opt_debug_stream) {
159 fprintf(opt_debug_stream,
160 "Structure %s XER decode unexpectedly requires "
161 "more data:\n%s\n",
162 td->name, xb1.buffer);
163 }
164 /* Fall through */
165 case RC_FAIL:
166 default:
167 if(opt_debug_stream) {
168 fprintf(opt_debug_stream,
169 "Structure %s XER decoding resulted in failure.\n",
170 td->name);
171 }
172 ASN_STRUCT_FREE(*td, sptr);
173 FREEMEM(xb1.buffer);
174 FREEMEM(xb2.buffer);
175 return XEQ_DECODE_FAILED;
176 }
177
178 if(rval.consumed != xb1.buffer_size
179 && ((rval.consumed > xb1.buffer_size)
180 || xer_whitespace_span(xb1.buffer + rval.consumed,
181 xb1.buffer_size - rval.consumed)
182 != (xb1.buffer_size - rval.consumed))) {
183 if(opt_debug_stream) {
184 fprintf(opt_debug_stream,
185 "Round-trip decode of %s required less bytes (%zu) than "
186 "encoded (%zu)\n",
187 td->name, rval.consumed, xb1.buffer_size);
188 }
189 ASN_STRUCT_FREE(*td, sptr);
190 FREEMEM(xb1.buffer);
191 FREEMEM(xb2.buffer);
192 return XEQ_ROUND_TRIP_FAILED;
193 }
194
195 /*
196 * Reuse xb2 to encode newly decoded structure.
197 */
198 FREEMEM(xb2.buffer);
199 memset(&xb2, 0, sizeof(xb2));
200
201 e2 = xer_encode(td, sptr, XER_F_BASIC, xer__buffer_append, &xb2);
202 if(e2.encoded == -1) {
203 if(opt_debug_stream) {
204 fprintf(stderr, "XER Encoding of round-trip decode of %s failed\n",
205 td->name);
206 }
207 ASN_STRUCT_FREE(*td, sptr);
208 FREEMEM(xb1.buffer);
209 FREEMEM(xb2.buffer);
210 return XEQ_ROUND_TRIP_FAILED;
211 }
212
213 ASN_STRUCT_FREE(*td, sptr);
214 sptr = 0;
215
216 if(xb1.buffer_size != xb2.buffer_size
217 || memcmp(xb1.buffer, xb2.buffer, xb1.buffer_size) != 0) {
218 if(opt_debug_stream) {
219 fprintf(opt_debug_stream,
220 "XER Encoding of round-trip decode of %s resulted in "
221 "different byte stream:\n"
222 "=== Original ===\n%s\n"
223 "=== Round-tripped ===\n%s\n",
224 xb1.buffer, xb2.buffer, td->name);
225 }
226 FREEMEM(xb1.buffer);
227 FREEMEM(xb2.buffer);
228 return XEQ_ROUND_TRIP_FAILED;
229 }
230
231 return XEQ_SUCCESS;
232}
233