Line data Source code
1 : /*
2 : * Copyright (c) 1997-2008 Kungliga Tekniska Högskolan
3 : * (Royal Institute of Technology, Stockholm, Sweden).
4 : * All rights reserved.
5 : *
6 : * Redistribution and use in source and binary forms, with or without
7 : * modification, are permitted provided that the following conditions
8 : * are met:
9 : *
10 : * 1. Redistributions of source code must retain the above copyright
11 : * notice, this list of conditions and the following disclaimer.
12 : *
13 : * 2. Redistributions in binary form must reproduce the above copyright
14 : * notice, this list of conditions and the following disclaimer in the
15 : * documentation and/or other materials provided with the distribution.
16 : *
17 : * 3. Neither the name of the Institute nor the names of its contributors
18 : * may be used to endorse or promote products derived from this software
19 : * without specific prior written permission.
20 : *
21 : * THIS SOFTWARE IS PROVIDED BY THE INSTITUTE AND CONTRIBUTORS ``AS IS'' AND
22 : * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
23 : * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
24 : * ARE DISCLAIMED. IN NO EVENT SHALL THE INSTITUTE OR CONTRIBUTORS BE LIABLE
25 : * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
26 : * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
27 : * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
28 : * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
29 : * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
30 : * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
31 : * SUCH DAMAGE.
32 : */
33 :
34 : #include "krb5_locl.h"
35 : #include "store-int.h"
36 :
37 : #define BYTEORDER_IS(SP, V) (((SP)->flags & KRB5_STORAGE_BYTEORDER_MASK) == (V))
38 : #define BYTEORDER_IS_LE(SP) BYTEORDER_IS((SP), KRB5_STORAGE_BYTEORDER_LE)
39 : #define BYTEORDER_IS_BE(SP) BYTEORDER_IS((SP), KRB5_STORAGE_BYTEORDER_BE)
40 : #define BYTEORDER_IS_HOST(SP) (BYTEORDER_IS((SP), KRB5_STORAGE_BYTEORDER_HOST) || \
41 : krb5_storage_is_flags((SP), KRB5_STORAGE_HOST_BYTEORDER))
42 : #define BYTEORDER_IS_PACKED(SP) BYTEORDER_IS((SP), KRB5_STORAGE_BYTEORDER_PACKED)
43 :
44 : /**
45 : * Add the flags on a storage buffer by or-ing in the flags to the buffer.
46 : *
47 : * @param sp the storage buffer to set the flags on
48 : * @param flags the flags to set
49 : *
50 : * @ingroup krb5_storage
51 : */
52 :
53 : KRB5_LIB_FUNCTION void KRB5_LIB_CALL
54 815534 : krb5_storage_set_flags(krb5_storage *sp, krb5_flags flags)
55 : {
56 815534 : sp->flags |= flags;
57 815534 : }
58 :
59 : /**
60 : * Clear the flags on a storage buffer
61 : *
62 : * @param sp the storage buffer to clear the flags on
63 : * @param flags the flags to clear
64 : *
65 : * @ingroup krb5_storage
66 : */
67 :
68 : KRB5_LIB_FUNCTION void KRB5_LIB_CALL
69 0 : krb5_storage_clear_flags(krb5_storage *sp, krb5_flags flags)
70 : {
71 0 : sp->flags &= ~flags;
72 0 : }
73 :
74 : /**
75 : * Return true or false depending on if the storage flags is set or
76 : * not. NB testing for the flag 0 always return true.
77 : *
78 : * @param sp the storage buffer to check flags on
79 : * @param flags The flags to test for
80 : *
81 : * @return true if all the flags are set, false if not.
82 : *
83 : * @ingroup krb5_storage
84 : */
85 :
86 : KRB5_LIB_FUNCTION krb5_boolean KRB5_LIB_CALL
87 12549261 : krb5_storage_is_flags(krb5_storage *sp, krb5_flags flags)
88 : {
89 12549261 : return (sp->flags & flags) == flags;
90 : }
91 :
92 : /**
93 : * Set the new byte order of the storage buffer.
94 : *
95 : * @param sp the storage buffer to set the byte order for.
96 : * @param byteorder the new byte order.
97 : *
98 : * The byte order are: KRB5_STORAGE_BYTEORDER_BE,
99 : * KRB5_STORAGE_BYTEORDER_LE and KRB5_STORAGE_BYTEORDER_HOST.
100 : *
101 : * @ingroup krb5_storage
102 : */
103 :
104 : KRB5_LIB_FUNCTION void KRB5_LIB_CALL
105 0 : krb5_storage_set_byteorder(krb5_storage *sp, krb5_flags byteorder)
106 : {
107 0 : sp->flags &= ~KRB5_STORAGE_BYTEORDER_MASK;
108 0 : sp->flags |= byteorder;
109 0 : }
110 :
111 : /**
112 : * Return the current byteorder for the buffer. See krb5_storage_set_byteorder() for the list or byte order contants.
113 : *
114 : * @ingroup krb5_storage
115 : */
116 :
117 : KRB5_LIB_FUNCTION krb5_flags KRB5_LIB_CALL
118 0 : krb5_storage_get_byteorder(krb5_storage *sp)
119 : {
120 0 : return sp->flags & KRB5_STORAGE_BYTEORDER_MASK;
121 : }
122 :
123 : /**
124 : * Set the max alloc value
125 : *
126 : * @param sp the storage buffer set the max allow for
127 : * @param size maximum size to allocate, use 0 to remove limit
128 : *
129 : * @ingroup krb5_storage
130 : */
131 :
132 : KRB5_LIB_FUNCTION void KRB5_LIB_CALL
133 0 : krb5_storage_set_max_alloc(krb5_storage *sp, size_t size)
134 : {
135 0 : sp->max_alloc = size;
136 0 : }
137 :
138 : /* don't allocate unresonable amount of memory */
139 : static krb5_error_code
140 1073215 : size_too_large(krb5_storage *sp, size_t size)
141 : {
142 1073215 : if (sp->max_alloc && sp->max_alloc < size)
143 0 : return HEIM_ERR_TOO_BIG;
144 1049333 : return 0;
145 : }
146 :
147 : static krb5_error_code
148 126855 : size_too_large_num(krb5_storage *sp, size_t count, size_t size)
149 : {
150 126855 : if (sp->max_alloc == 0 || size == 0)
151 0 : return 0;
152 126855 : size = sp->max_alloc / size;
153 126855 : if (size < count)
154 0 : return HEIM_ERR_TOO_BIG;
155 126855 : return 0;
156 : }
157 :
158 : /**
159 : * Seek to a new offset.
160 : *
161 : * @param sp the storage buffer to seek in.
162 : * @param offset the offset to seek
163 : * @param whence relateive searching, SEEK_CUR from the current
164 : * position, SEEK_END from the end, SEEK_SET absolute from the start.
165 : *
166 : * @return The new current offset
167 : *
168 : * @ingroup krb5_storage
169 : */
170 :
171 : KRB5_LIB_FUNCTION off_t KRB5_LIB_CALL
172 758826 : krb5_storage_seek(krb5_storage *sp, off_t offset, int whence)
173 : {
174 758826 : return (*sp->seek)(sp, offset, whence);
175 : }
176 :
177 : /**
178 : * Truncate the storage buffer in sp to offset.
179 : *
180 : * @param sp the storage buffer to truncate.
181 : * @param offset the offset to truncate too.
182 : *
183 : * @return An Kerberos 5 error code.
184 : *
185 : * @ingroup krb5_storage
186 : */
187 :
188 : KRB5_LIB_FUNCTION int KRB5_LIB_CALL
189 1754 : krb5_storage_truncate(krb5_storage *sp, off_t offset)
190 : {
191 1754 : return (*sp->trunc)(sp, offset);
192 : }
193 :
194 : /**
195 : * Sync the storage buffer to its backing store. If there is no
196 : * backing store this function will return success.
197 : *
198 : * @param sp the storage buffer to sync
199 : *
200 : * @return A Kerberos 5 error code
201 : *
202 : * @ingroup krb5_storage
203 : */
204 :
205 : KRB5_LIB_FUNCTION int KRB5_LIB_CALL
206 2214 : krb5_storage_fsync(krb5_storage *sp)
207 : {
208 2214 : if (sp->fsync != NULL)
209 2214 : return sp->fsync(sp);
210 0 : return 0;
211 : }
212 :
213 : /**
214 : * Read to the storage buffer.
215 : *
216 : * @param sp the storage buffer to read from
217 : * @param buf the buffer to store the data in
218 : * @param len the length to read
219 : *
220 : * @return The length of data read (can be shorter then len), or negative on error.
221 : *
222 : * @ingroup krb5_storage
223 : */
224 :
225 : KRB5_LIB_FUNCTION krb5_ssize_t KRB5_LIB_CALL
226 942892 : krb5_storage_read(krb5_storage *sp, void *buf, size_t len)
227 : {
228 942892 : return sp->fetch(sp, buf, len);
229 : }
230 :
231 : /**
232 : * Write to the storage buffer.
233 : *
234 : * @param sp the storage buffer to write to
235 : * @param buf the buffer to write to the storage buffer
236 : * @param len the length to write
237 : *
238 : * @return The length of data written (can be shorter then len), or negative on error.
239 : *
240 : * @ingroup krb5_storage
241 : */
242 :
243 : KRB5_LIB_FUNCTION krb5_ssize_t KRB5_LIB_CALL
244 1135788 : krb5_storage_write(krb5_storage *sp, const void *buf, size_t len)
245 : {
246 1135788 : return sp->store(sp, buf, len);
247 : }
248 :
249 : /**
250 : * Set the return code that will be used when end of storage is reached.
251 : *
252 : * @param sp the storage
253 : * @param code the error code to return on end of storage
254 : *
255 : * @ingroup krb5_storage
256 : */
257 :
258 : KRB5_LIB_FUNCTION void KRB5_LIB_CALL
259 64467 : krb5_storage_set_eof_code(krb5_storage *sp, int code)
260 : {
261 64467 : sp->eof_code = code;
262 64467 : }
263 :
264 : /**
265 : * Get the return code that will be used when end of storage is reached.
266 : *
267 : * @param sp the storage
268 : *
269 : * @return storage error code
270 : *
271 : * @ingroup krb5_storage
272 : */
273 :
274 : KRB5_LIB_FUNCTION int KRB5_LIB_CALL
275 0 : krb5_storage_get_eof_code(krb5_storage *sp)
276 : {
277 0 : return sp->eof_code;
278 : }
279 :
280 : /**
281 : * Free a krb5 storage.
282 : *
283 : * @param sp the storage to free.
284 : *
285 : * @return An Kerberos 5 error code.
286 : *
287 : * @ingroup krb5_storage
288 : */
289 :
290 : KRB5_LIB_FUNCTION krb5_error_code KRB5_LIB_CALL
291 1242843 : krb5_storage_free(krb5_storage *sp)
292 : {
293 1242843 : if (sp == NULL)
294 0 : return 0;
295 1242843 : if(sp->free)
296 509500 : (*sp->free)(sp);
297 1242843 : free(sp->data);
298 1242843 : free(sp);
299 1242843 : return 0;
300 : }
301 :
302 : /**
303 : * Copy the content of storage to a krb5_data.
304 : *
305 : * @param sp the storage to copy to a data
306 : * @param data the copied data, free with krb5_data_free()
307 : *
308 : * @return 0 for success, or a Kerberos 5 error code on failure.
309 : *
310 : * @ingroup krb5_storage
311 : */
312 :
313 : KRB5_LIB_FUNCTION krb5_error_code KRB5_LIB_CALL
314 446586 : krb5_storage_to_data(krb5_storage *sp, krb5_data *data)
315 : {
316 14678 : off_t pos, size;
317 14678 : krb5_error_code ret;
318 :
319 446586 : pos = sp->seek(sp, 0, SEEK_CUR);
320 446586 : if (pos < 0)
321 0 : return HEIM_ERR_NOT_SEEKABLE;
322 446586 : size = sp->seek(sp, 0, SEEK_END);
323 446586 : ret = size_too_large(sp, size);
324 431908 : if (ret)
325 0 : return ret;
326 446586 : ret = krb5_data_alloc(data, size);
327 446586 : if (ret) {
328 0 : sp->seek(sp, pos, SEEK_SET);
329 0 : return ret;
330 : }
331 446586 : if (size) {
332 14678 : ssize_t bytes;
333 :
334 446586 : sp->seek(sp, 0, SEEK_SET);
335 446586 : bytes = sp->fetch(sp, data->data, data->length);
336 446586 : sp->seek(sp, pos, SEEK_SET);
337 :
338 : /* sp->fetch() really shouldn't fail */
339 446586 : if (bytes < 0)
340 0 : return sp->eof_code;
341 :
342 : /* Maybe the underlying file (or whatever) got truncated? */
343 446586 : data->length = bytes;
344 : }
345 431908 : return 0;
346 : }
347 :
348 : static size_t
349 0 : pack_int(uint8_t *p, uint64_t val)
350 : {
351 0 : size_t l = 0;
352 :
353 0 : if (val < 128) {
354 0 : *p = val;
355 : } else {
356 0 : while (val > 0) {
357 0 : *p-- = val % 256;
358 0 : val /= 256;
359 0 : l++;
360 : }
361 0 : *p = 0x80 | l;
362 : }
363 0 : return l + 1;
364 : }
365 :
366 : static size_t
367 0 : unpack_int_length(uint8_t *v)
368 : {
369 0 : size_t size;
370 :
371 0 : if (*v < 128)
372 0 : size = 0;
373 : else
374 0 : size = *v & 0x7f;
375 :
376 0 : return size + 1;
377 : }
378 :
379 : static int
380 0 : unpack_int(uint8_t *p, size_t len, uint64_t *val, size_t *size)
381 : {
382 0 : size_t v;
383 :
384 0 : if (len == 0)
385 0 : return EINVAL;
386 0 : --len;
387 0 : v = *p++;
388 0 : if (v < 128) {
389 0 : *val = v;
390 0 : *size = 1;
391 : } else {
392 0 : int e;
393 0 : size_t l;
394 0 : uint64_t tmp;
395 :
396 0 : if (v == 0x80) {
397 0 : *size = 1;
398 0 : return EINVAL;
399 : }
400 0 : v &= 0x7F;
401 0 : if (len < v)
402 0 : return ERANGE;
403 0 : e = der_get_unsigned64(p, v, &tmp, &l);
404 0 : if (e)
405 0 : return ERANGE;
406 0 : *val = tmp;
407 0 : *size = l + 1;
408 : }
409 0 : return 0;
410 : }
411 :
412 : static krb5_error_code
413 3008306 : krb5_store_int(krb5_storage *sp,
414 : int64_t value,
415 : size_t len)
416 : {
417 106754 : int ret;
418 3008306 : uint8_t v[9], *p = v;
419 :
420 3008306 : if (len > sizeof(value))
421 0 : return EINVAL;
422 :
423 3008306 : if (BYTEORDER_IS_PACKED(sp)) {
424 0 : uint64_t mask = ~0ULL >> (64 - len * 8);
425 0 : value &= mask;
426 0 : p += sizeof(v) - 1;
427 0 : len = pack_int(p, value);
428 0 : p = v + sizeof(v) - len;
429 : } else
430 3008306 : _krb5_put_int(v, value, len);
431 3008306 : ret = sp->store(sp, p, len);
432 3008306 : if (ret < 0)
433 0 : return errno;
434 3008306 : if ((size_t)ret != len)
435 0 : return sp->eof_code;
436 2901552 : return 0;
437 : }
438 :
439 : /**
440 : * Store a int32 to storage, byte order is controlled by the settings
441 : * on the storage, see krb5_storage_set_byteorder().
442 : *
443 : * @param sp the storage to write too
444 : * @param value the value to store
445 : *
446 : * @return 0 for success, or a Kerberos 5 error code on failure.
447 : *
448 : * @ingroup krb5_storage
449 : */
450 :
451 : KRB5_LIB_FUNCTION krb5_error_code KRB5_LIB_CALL
452 2223175 : krb5_store_int32(krb5_storage *sp,
453 : int32_t value)
454 : {
455 2223175 : if(BYTEORDER_IS_HOST(sp))
456 0 : value = htonl(value);
457 2223175 : else if(BYTEORDER_IS_LE(sp))
458 1736284 : value = bswap32(value);
459 2223175 : return krb5_store_int(sp, value, 4);
460 : }
461 :
462 : /**
463 : * Store a int64 to storage, byte order is controlled by the settings
464 : * on the storage, see krb5_storage_set_byteorder().
465 : *
466 : * @param sp the storage to write too
467 : * @param value the value to store
468 : *
469 : * @return 0 for success, or a Kerberos 5 error code on failure.
470 : *
471 : * @ingroup krb5_storage
472 : */
473 :
474 : KRB5_LIB_FUNCTION krb5_error_code KRB5_LIB_CALL
475 614233 : krb5_store_int64(krb5_storage *sp,
476 : int64_t value)
477 : {
478 614233 : if (BYTEORDER_IS_HOST(sp))
479 : #ifdef WORDS_BIGENDIAN
480 : ;
481 : #else
482 0 : value = bswap64(value); /* There's no ntohll() */
483 : #endif
484 614233 : else if (BYTEORDER_IS_LE(sp))
485 614233 : value = bswap64(value);
486 614233 : return krb5_store_int(sp, value, 8);
487 : }
488 :
489 : /**
490 : * Store a uint32 to storage, byte order is controlled by the settings
491 : * on the storage, see krb5_storage_set_byteorder().
492 : *
493 : * @param sp the storage to write too
494 : * @param value the value to store
495 : *
496 : * @return 0 for success, or a Kerberos 5 error code on failure.
497 : *
498 : * @ingroup krb5_storage
499 : */
500 :
501 : KRB5_LIB_FUNCTION krb5_error_code KRB5_LIB_CALL
502 1737727 : krb5_store_uint32(krb5_storage *sp,
503 : uint32_t value)
504 : {
505 1737727 : return krb5_store_int32(sp, (int32_t)value);
506 : }
507 :
508 : /**
509 : * Store a uint64 to storage, byte order is controlled by the settings
510 : * on the storage, see krb5_storage_set_byteorder().
511 : *
512 : * @param sp the storage to write too
513 : * @param value the value to store
514 : *
515 : * @return 0 for success, or a Kerberos 5 error code on failure.
516 : *
517 : * @ingroup krb5_storage
518 : */
519 :
520 : KRB5_LIB_FUNCTION krb5_error_code KRB5_LIB_CALL
521 614233 : krb5_store_uint64(krb5_storage *sp,
522 : uint64_t value)
523 : {
524 614233 : return krb5_store_int64(sp, (int64_t)value);
525 : }
526 :
527 : static krb5_error_code
528 8842796 : krb5_ret_int(krb5_storage *sp,
529 : int64_t *value,
530 : size_t len)
531 : {
532 174296 : int ret;
533 174296 : unsigned char v[9];
534 8842796 : uint64_t w = 0;
535 8842796 : *value = 0; /* quiets warnings */
536 8842796 : if (BYTEORDER_IS_PACKED(sp)) {
537 0 : ret = sp->fetch(sp, v, 1);
538 0 : if (ret < 0)
539 0 : return errno;
540 0 : if (ret != 1)
541 0 : return sp->eof_code;
542 :
543 0 : len = unpack_int_length(v);
544 0 : if (len < 1)
545 0 : return ERANGE;
546 0 : else if (len > 1) {
547 0 : ret = sp->fetch(sp, v + 1, len - 1);
548 0 : if (ret < 0)
549 0 : return errno;
550 0 : if (ret != len - 1)
551 0 : return sp->eof_code;
552 : }
553 0 : ret = unpack_int(v, len, &w, &len);
554 0 : if (ret)
555 0 : return ret;
556 0 : *value = w;
557 0 : return 0;
558 : }
559 8842796 : ret = sp->fetch(sp, v, len);
560 8842796 : if (ret < 0)
561 0 : return errno;
562 8842796 : if ((size_t)ret != len)
563 28119 : return sp->eof_code;
564 8814677 : _krb5_get_int64(v, &w, len);
565 8814677 : *value = w;
566 8814677 : return 0;
567 : }
568 :
569 : /**
570 : * Read a int64 from storage, byte order is controlled by the settings
571 : * on the storage, see krb5_storage_set_byteorder().
572 : *
573 : * @param sp the storage to write too
574 : * @param value the value read from the buffer
575 : *
576 : * @return 0 for success, or a Kerberos 5 error code on failure.
577 : *
578 : * @ingroup krb5_storage
579 : */
580 :
581 : KRB5_LIB_FUNCTION krb5_error_code KRB5_LIB_CALL
582 1243991 : krb5_ret_int64(krb5_storage *sp,
583 : int64_t *value)
584 : {
585 1243991 : krb5_error_code ret = krb5_ret_int(sp, value, 8);
586 1243991 : if(ret)
587 0 : return ret;
588 1243991 : if(BYTEORDER_IS_HOST(sp))
589 : #ifdef WORDS_BIGENDIAN
590 : ;
591 : #else
592 0 : *value = bswap64(*value); /* There's no ntohll() */
593 : #endif
594 1243991 : else if(BYTEORDER_IS_LE(sp))
595 1243991 : *value = bswap64(*value);
596 1216623 : return 0;
597 : }
598 :
599 : /**
600 : * Read a uint64 from storage, byte order is controlled by the settings
601 : * on the storage, see krb5_storage_set_byteorder().
602 : *
603 : * @param sp the storage to write too
604 : * @param value the value read from the buffer
605 : *
606 : * @return 0 for success, or a Kerberos 5 error code on failure.
607 : *
608 : * @ingroup krb5_storage
609 : */
610 :
611 : KRB5_LIB_FUNCTION krb5_error_code KRB5_LIB_CALL
612 1243991 : krb5_ret_uint64(krb5_storage *sp,
613 : uint64_t *value)
614 : {
615 27368 : krb5_error_code ret;
616 27368 : int64_t v;
617 :
618 1243991 : ret = krb5_ret_int64(sp, &v);
619 1243991 : if (ret == 0)
620 1243991 : *value = (uint64_t)v;
621 :
622 1243991 : return ret;
623 : }
624 :
625 : /**
626 : * Read a int32 from storage, byte order is controlled by the settings
627 : * on the storage, see krb5_storage_set_byteorder().
628 : *
629 : * @param sp the storage to write too
630 : * @param value the value read from the buffer
631 : *
632 : * @return 0 for success, or a Kerberos 5 error code on failure.
633 : *
634 : * @ingroup krb5_storage
635 : */
636 :
637 : KRB5_LIB_FUNCTION krb5_error_code KRB5_LIB_CALL
638 5426119 : krb5_ret_int32(krb5_storage *sp,
639 : int32_t *value)
640 : {
641 105173 : int64_t v;
642 :
643 5426119 : krb5_error_code ret = krb5_ret_int(sp, &v, 4);
644 5426119 : if (ret)
645 27884 : return ret;
646 5398000 : *value = v;
647 5398000 : if (BYTEORDER_IS_HOST(sp))
648 0 : *value = htonl(*value);
649 5398000 : else if (BYTEORDER_IS_LE(sp))
650 3308011 : *value = bswap32(*value);
651 5293062 : return 0;
652 : }
653 :
654 : /**
655 : * Read a uint32 from storage, byte order is controlled by the settings
656 : * on the storage, see krb5_storage_set_byteorder().
657 : *
658 : * @param sp the storage to write too
659 : * @param value the value read from the buffer
660 : *
661 : * @return 0 for success, or a Kerberos 5 error code on failure.
662 : *
663 : * @ingroup krb5_storage
664 : */
665 :
666 : KRB5_LIB_FUNCTION krb5_error_code KRB5_LIB_CALL
667 3732425 : krb5_ret_uint32(krb5_storage *sp, uint32_t *value)
668 : {
669 81382 : krb5_error_code ret;
670 81382 : int32_t v;
671 :
672 3732425 : ret = krb5_ret_int32(sp, &v);
673 3732425 : if (ret == 0)
674 3732425 : *value = (uint32_t)v;
675 :
676 3732425 : return ret;
677 : }
678 :
679 : /**
680 : * Store a int16 to storage, byte order is controlled by the settings
681 : * on the storage, see krb5_storage_set_byteorder().
682 : *
683 : * @param sp the storage to write too
684 : * @param value the value to store
685 : *
686 : * @return 0 for success, or a Kerberos 5 error code on failure.
687 : *
688 : * @ingroup krb5_storage
689 : */
690 :
691 : KRB5_LIB_FUNCTION krb5_error_code KRB5_LIB_CALL
692 170898 : krb5_store_int16(krb5_storage *sp,
693 : int16_t value)
694 : {
695 170898 : if(BYTEORDER_IS_HOST(sp))
696 0 : value = htons(value);
697 170898 : else if(BYTEORDER_IS_LE(sp))
698 83387 : value = bswap16(value);
699 170898 : return krb5_store_int(sp, value, 2);
700 : }
701 :
702 : /**
703 : * Store a uint16 to storage, byte order is controlled by the settings
704 : * on the storage, see krb5_storage_set_byteorder().
705 : *
706 : * @param sp the storage to write too
707 : * @param value the value to store
708 : *
709 : * @return 0 for success, or a Kerberos 5 error code on failure.
710 : *
711 : * @ingroup krb5_storage
712 : */
713 :
714 : KRB5_LIB_FUNCTION krb5_error_code KRB5_LIB_CALL
715 83387 : krb5_store_uint16(krb5_storage *sp,
716 : uint16_t value)
717 : {
718 83387 : return krb5_store_int16(sp, (int16_t)value);
719 : }
720 :
721 : /**
722 : * Read a int16 from storage, byte order is controlled by the settings
723 : * on the storage, see krb5_storage_set_byteorder().
724 : *
725 : * @param sp the storage to write too
726 : * @param value the value read from the buffer
727 : *
728 : * @return 0 for success, or a Kerberos 5 error code on failure.
729 : *
730 : * @ingroup krb5_storage
731 : */
732 :
733 : KRB5_LIB_FUNCTION krb5_error_code KRB5_LIB_CALL
734 2172686 : krb5_ret_int16(krb5_storage *sp,
735 : int16_t *value)
736 : {
737 41755 : int64_t v;
738 41755 : int ret;
739 2172686 : ret = krb5_ret_int(sp, &v, 2);
740 2172686 : if(ret)
741 0 : return ret;
742 2172686 : *value = v;
743 2172686 : if(BYTEORDER_IS_HOST(sp))
744 0 : *value = htons(*value);
745 2172686 : else if(BYTEORDER_IS_LE(sp))
746 922890 : *value = bswap16(*value);
747 2130931 : return 0;
748 : }
749 :
750 : /**
751 : * Read a int16 from storage, byte order is controlled by the settings
752 : * on the storage, see krb5_storage_set_byteorder().
753 : *
754 : * @param sp the storage to write too
755 : * @param value the value read from the buffer
756 : *
757 : * @return 0 for success, or a Kerberos 5 error code on failure.
758 : *
759 : * @ingroup krb5_storage
760 : */
761 :
762 : KRB5_LIB_FUNCTION krb5_error_code KRB5_LIB_CALL
763 922890 : krb5_ret_uint16(krb5_storage *sp,
764 : uint16_t *value)
765 : {
766 22851 : krb5_error_code ret;
767 22851 : int16_t v;
768 :
769 922890 : ret = krb5_ret_int16(sp, &v);
770 922890 : if (ret == 0)
771 922890 : *value = (uint16_t)v;
772 :
773 922890 : return ret;
774 : }
775 :
776 : /**
777 : * Store a int8 to storage.
778 : *
779 : * @param sp the storage to write too
780 : * @param value the value to store
781 : *
782 : * @return 0 for success, or a Kerberos 5 error code on failure.
783 : *
784 : * @ingroup krb5_storage
785 : */
786 :
787 : KRB5_LIB_FUNCTION krb5_error_code KRB5_LIB_CALL
788 5359 : krb5_store_int8(krb5_storage *sp,
789 : int8_t value)
790 : {
791 275 : int ret;
792 :
793 5359 : ret = sp->store(sp, &value, sizeof(value));
794 5359 : if (ret != sizeof(value))
795 0 : return (ret<0)?errno:sp->eof_code;
796 5084 : return 0;
797 : }
798 :
799 : /**
800 : * Store a uint8 to storage.
801 : *
802 : * @param sp the storage to write too
803 : * @param value the value to store
804 : *
805 : * @return 0 for success, or a Kerberos 5 error code on failure.
806 : *
807 : * @ingroup krb5_storage
808 : */
809 :
810 : KRB5_LIB_FUNCTION krb5_error_code KRB5_LIB_CALL
811 0 : krb5_store_uint8(krb5_storage *sp,
812 : uint8_t value)
813 : {
814 0 : return krb5_store_int8(sp, (int8_t)value);
815 : }
816 :
817 : /**
818 : * Read a int8 from storage
819 : *
820 : * @param sp the storage to write too
821 : * @param value the value read from the buffer
822 : *
823 : * @return 0 for success, or a Kerberos 5 error code on failure.
824 : *
825 : * @ingroup krb5_storage
826 : */
827 :
828 : KRB5_LIB_FUNCTION krb5_error_code KRB5_LIB_CALL
829 356032 : krb5_ret_int8(krb5_storage *sp,
830 : int8_t *value)
831 : {
832 5461 : ssize_t ret;
833 :
834 356032 : ret = sp->fetch(sp, value, sizeof(*value));
835 356032 : if (ret < 0 || (size_t)ret != sizeof(*value))
836 0 : return (ret<0)?errno:sp->eof_code;
837 350571 : return 0;
838 : }
839 :
840 : /**
841 : * Read a uint8 from storage
842 : *
843 : * @param sp the storage to write too
844 : * @param value the value read from the buffer
845 : *
846 : * @return 0 for success, or a Kerberos 5 error code on failure.
847 : *
848 : * @ingroup krb5_storage
849 : */
850 :
851 : KRB5_LIB_FUNCTION krb5_error_code KRB5_LIB_CALL
852 0 : krb5_ret_uint8(krb5_storage *sp,
853 : uint8_t *value)
854 : {
855 0 : krb5_error_code ret;
856 0 : int8_t v;
857 :
858 0 : ret = krb5_ret_int8(sp, &v);
859 0 : if (ret == 0)
860 0 : *value = (uint8_t)v;
861 :
862 0 : return ret;
863 : }
864 :
865 : /**
866 : * Store a data to the storage. The data is stored with an int32 as
867 : * lenght plus the data (not padded).
868 : *
869 : * @param sp the storage buffer to write to
870 : * @param data the buffer to store.
871 : *
872 : * @return 0 on success, a Kerberos 5 error code on failure.
873 : *
874 : * @ingroup krb5_storage
875 : */
876 :
877 : KRB5_LIB_FUNCTION krb5_error_code KRB5_LIB_CALL
878 380435 : krb5_store_data(krb5_storage *sp,
879 : krb5_data data)
880 : {
881 10862 : int ret;
882 380435 : ret = krb5_store_int32(sp, data.length);
883 380435 : if(ret < 0)
884 0 : return ret;
885 380435 : ret = sp->store(sp, data.data, data.length);
886 380435 : if(ret < 0)
887 0 : return errno;
888 380435 : if((size_t)ret != data.length)
889 0 : return sp->eof_code;
890 369573 : return 0;
891 : }
892 :
893 : /**
894 : * Store a data blob to the storage. The data is stored with an int32 as
895 : * length plus the data (not padded). This function only differs from
896 : * krb5_store_data() insofar as it takes a void * and a length as parameters.
897 : *
898 : * @param sp the storage buffer to write to
899 : * @param s the string to store.
900 : * @param len length of the string to be stored.
901 : *
902 : * @return 0 on success, a Kerberos 5 error code on failure.
903 : *
904 : * @ingroup krb5_storage
905 : */
906 : KRB5_LIB_FUNCTION krb5_error_code KRB5_LIB_CALL
907 0 : krb5_store_datalen(krb5_storage *sp, const void *d, size_t len)
908 : {
909 0 : krb5_data data;
910 0 : data.length = len;
911 0 : data.data = (void *)d;
912 0 : return krb5_store_data(sp, data);
913 : }
914 :
915 : /**
916 : * Store a data blob to the storage. The data is stored without a length.
917 : *
918 : * @param sp the storage buffer to write to
919 : * @param s the string to store.
920 : * @param len length of the string to be stored.
921 : *
922 : * @return 0 on success, a Kerberos 5 error code on failure.
923 : *
924 : * @ingroup krb5_storage
925 : */
926 : KRB5_LIB_FUNCTION krb5_error_code KRB5_LIB_CALL
927 0 : krb5_store_bytes(krb5_storage *sp, const void *d, size_t len)
928 : {
929 0 : ssize_t ssize;
930 :
931 0 : ssize = krb5_storage_write(sp, d, len);
932 0 : if (ssize != len)
933 0 : return ENOMEM;
934 :
935 0 : return 0;
936 : }
937 :
938 : /**
939 : * Parse a data from the storage.
940 : *
941 : * @param sp the storage buffer to read from
942 : * @param data the parsed data
943 : *
944 : * @return 0 on success, a Kerberos 5 error code on failure.
945 : *
946 : * @ingroup krb5_storage
947 : */
948 :
949 : KRB5_LIB_FUNCTION krb5_error_code KRB5_LIB_CALL
950 650105 : krb5_ret_data(krb5_storage *sp,
951 : krb5_data *data)
952 : {
953 9204 : krb5_error_code ret;
954 9204 : int32_t size;
955 :
956 650105 : ret = krb5_ret_int32(sp, &size);
957 650105 : if(ret)
958 23476 : return ret;
959 626629 : ret = size_too_large(sp, size);
960 617425 : if (ret)
961 0 : return ret;
962 626629 : ret = krb5_data_alloc (data, size);
963 626629 : if (ret)
964 0 : return ret;
965 626629 : if (size) {
966 5877 : ssize_t bytes;
967 :
968 487281 : bytes = sp->fetch(sp, data->data, size);
969 487281 : if (bytes < 0 || bytes != size) {
970 0 : krb5_data_free(data);
971 0 : return (bytes < 0)? errno : sp->eof_code;
972 : }
973 : }
974 617425 : return 0;
975 : }
976 :
977 : /**
978 : * Store a string to the buffer. The data is formated as an len:uint32
979 : * plus the string itself (not padded).
980 : *
981 : * @param sp the storage buffer to write to
982 : * @param s the string to store.
983 : *
984 : * @return 0 on success, a Kerberos 5 error code on failure.
985 : *
986 : * @ingroup krb5_storage
987 : */
988 :
989 : KRB5_LIB_FUNCTION krb5_error_code KRB5_LIB_CALL
990 262838 : krb5_store_string(krb5_storage *sp, const char *s)
991 : {
992 8226 : krb5_data data;
993 :
994 262838 : if (s == NULL)
995 0 : return EINVAL;
996 :
997 262838 : data.length = strlen(s);
998 262838 : data.data = rk_UNCONST(s);
999 262838 : return krb5_store_data(sp, data);
1000 : }
1001 :
1002 : /**
1003 : * Parse a string from the storage.
1004 : *
1005 : * @param sp the storage buffer to read from
1006 : * @param string the parsed string
1007 : *
1008 : * @return 0 on success, a Kerberos 5 error code on failure.
1009 : *
1010 : * @ingroup krb5_storage
1011 : */
1012 :
1013 :
1014 : KRB5_LIB_FUNCTION krb5_error_code KRB5_LIB_CALL
1015 416804 : krb5_ret_string(krb5_storage *sp,
1016 : char **string)
1017 : {
1018 8226 : int ret;
1019 8226 : krb5_data data;
1020 :
1021 416804 : *string = NULL;
1022 416804 : ret = krb5_ret_data(sp, &data);
1023 416804 : if(ret)
1024 0 : return ret;
1025 416804 : *string = realloc(data.data, data.length + 1);
1026 416804 : if(*string == NULL){
1027 0 : free(data.data);
1028 0 : return ENOMEM;
1029 : }
1030 416804 : (*string)[data.length] = 0;
1031 416804 : return 0;
1032 : }
1033 :
1034 : /**
1035 : * Store a zero terminated string to the buffer. The data is stored
1036 : * one character at a time until a NUL is stored.
1037 : *
1038 : * @param sp the storage buffer to write to
1039 : * @param s the string to store.
1040 : *
1041 : * @return 0 on success, a Kerberos 5 error code on failure.
1042 : *
1043 : * @ingroup krb5_storage
1044 : */
1045 :
1046 : KRB5_LIB_FUNCTION krb5_error_code KRB5_LIB_CALL
1047 0 : krb5_store_stringz(krb5_storage *sp, const char *s)
1048 : {
1049 0 : size_t len;
1050 0 : ssize_t ret;
1051 :
1052 0 : if (s == NULL)
1053 0 : return EINVAL;
1054 :
1055 0 : len = strlen(s) + 1;
1056 0 : ret = sp->store(sp, s, len);
1057 0 : if(ret < 0)
1058 0 : return ret;
1059 0 : if((size_t)ret != len)
1060 0 : return sp->eof_code;
1061 0 : return 0;
1062 : }
1063 :
1064 : /**
1065 : * Parse zero terminated string from the storage.
1066 : *
1067 : * @param sp the storage buffer to read from
1068 : * @param string the parsed string
1069 : *
1070 : * @return 0 on success, a Kerberos 5 error code on failure.
1071 : *
1072 : * @ingroup krb5_storage
1073 : */
1074 :
1075 : KRB5_LIB_FUNCTION krb5_error_code KRB5_LIB_CALL
1076 0 : krb5_ret_stringz(krb5_storage *sp,
1077 : char **string)
1078 : {
1079 0 : char c;
1080 0 : char *s = NULL;
1081 0 : size_t len = 0;
1082 0 : ssize_t ret;
1083 :
1084 0 : while((ret = sp->fetch(sp, &c, 1)) == 1){
1085 0 : krb5_error_code eret;
1086 0 : char *tmp;
1087 :
1088 0 : len++;
1089 0 : eret = size_too_large(sp, len);
1090 0 : if (eret) {
1091 0 : free(s);
1092 0 : return eret;
1093 : }
1094 0 : tmp = realloc (s, len);
1095 0 : if (tmp == NULL) {
1096 0 : free (s);
1097 0 : return ENOMEM;
1098 : }
1099 0 : s = tmp;
1100 0 : s[len - 1] = c;
1101 0 : if(c == 0)
1102 0 : break;
1103 : }
1104 0 : if(ret != 1){
1105 0 : free(s);
1106 0 : if(ret == 0)
1107 0 : return sp->eof_code;
1108 0 : return ret;
1109 : }
1110 0 : *string = s;
1111 0 : return 0;
1112 : }
1113 :
1114 : KRB5_LIB_FUNCTION krb5_error_code KRB5_LIB_CALL
1115 0 : krb5_store_stringnl(krb5_storage *sp, const char *s)
1116 : {
1117 0 : size_t len;
1118 0 : ssize_t ret;
1119 :
1120 0 : if (s == NULL)
1121 0 : return EINVAL;
1122 :
1123 0 : len = strlen(s);
1124 0 : ret = sp->store(sp, s, len);
1125 0 : if(ret < 0)
1126 0 : return ret;
1127 0 : if((size_t)ret != len)
1128 0 : return sp->eof_code;
1129 0 : ret = sp->store(sp, "\n", 1);
1130 0 : if(ret != 1) {
1131 0 : if(ret < 0)
1132 0 : return ret;
1133 : else
1134 0 : return sp->eof_code;
1135 : }
1136 :
1137 0 : return 0;
1138 :
1139 : }
1140 :
1141 : KRB5_LIB_FUNCTION krb5_error_code KRB5_LIB_CALL
1142 0 : krb5_ret_stringnl(krb5_storage *sp,
1143 : char **string)
1144 : {
1145 0 : int expect_nl = 0;
1146 0 : char c;
1147 0 : char *s = NULL;
1148 0 : size_t len = 0;
1149 0 : ssize_t ret;
1150 :
1151 0 : while((ret = sp->fetch(sp, &c, 1)) == 1){
1152 0 : krb5_error_code eret;
1153 0 : char *tmp;
1154 :
1155 0 : if (c == '\r') {
1156 0 : expect_nl = 1;
1157 0 : continue;
1158 : }
1159 0 : if (expect_nl && c != '\n') {
1160 0 : free(s);
1161 0 : return KRB5_BADMSGTYPE;
1162 : }
1163 :
1164 0 : len++;
1165 0 : eret = size_too_large(sp, len);
1166 0 : if (eret) {
1167 0 : free(s);
1168 0 : return eret;
1169 : }
1170 0 : tmp = realloc (s, len);
1171 0 : if (tmp == NULL) {
1172 0 : free (s);
1173 0 : return ENOMEM;
1174 : }
1175 0 : s = tmp;
1176 0 : if(c == '\n') {
1177 0 : s[len - 1] = '\0';
1178 0 : break;
1179 : }
1180 0 : s[len - 1] = c;
1181 : }
1182 0 : if(ret != 1){
1183 0 : free(s);
1184 0 : if(ret == 0)
1185 0 : return sp->eof_code;
1186 0 : return ret;
1187 : }
1188 0 : *string = s;
1189 0 : return 0;
1190 : }
1191 :
1192 : /**
1193 : * Write a principal block to storage.
1194 : *
1195 : * @param sp the storage buffer to write to
1196 : * @param p the principal block to write.
1197 : *
1198 : * @return 0 on success, a Kerberos 5 error code on failure.
1199 : *
1200 : * @ingroup krb5_storage
1201 : */
1202 :
1203 : KRB5_LIB_FUNCTION krb5_error_code KRB5_LIB_CALL
1204 4217 : krb5_store_principal(krb5_storage *sp,
1205 : krb5_const_principal p)
1206 : {
1207 0 : size_t i;
1208 0 : int ret;
1209 :
1210 4217 : if(!krb5_storage_is_flags(sp, KRB5_STORAGE_PRINCIPAL_NO_NAME_TYPE)) {
1211 4217 : ret = krb5_store_int32(sp, p->name.name_type);
1212 4217 : if(ret) return ret;
1213 : }
1214 4217 : if(krb5_storage_is_flags(sp, KRB5_STORAGE_PRINCIPAL_WRONG_NUM_COMPONENTS))
1215 0 : ret = krb5_store_int32(sp, p->name.name_string.len + 1);
1216 : else
1217 4217 : ret = krb5_store_int32(sp, p->name.name_string.len);
1218 :
1219 4217 : if(ret) return ret;
1220 4217 : ret = krb5_store_string(sp, p->realm);
1221 4217 : if(ret) return ret;
1222 10610 : for(i = 0; i < p->name.name_string.len; i++){
1223 6393 : ret = krb5_store_string(sp, p->name.name_string.val[i]);
1224 6393 : if(ret) return ret;
1225 : }
1226 4217 : return 0;
1227 : }
1228 :
1229 : /**
1230 : * Parse principal from the storage.
1231 : *
1232 : * @param sp the storage buffer to read from
1233 : * @param princ the parsed principal
1234 : *
1235 : * @return 0 on success, a Kerberos 5 error code on failure.
1236 : *
1237 : * @ingroup krb5_storage
1238 : */
1239 :
1240 : KRB5_LIB_FUNCTION krb5_error_code KRB5_LIB_CALL
1241 68579 : krb5_ret_principal(krb5_storage *sp,
1242 : krb5_principal *princ)
1243 : {
1244 0 : int i;
1245 0 : int ret;
1246 0 : krb5_principal p;
1247 0 : int32_t type;
1248 0 : int32_t ncomp;
1249 :
1250 68579 : p = calloc(1, sizeof(*p));
1251 68579 : if(p == NULL)
1252 0 : return ENOMEM;
1253 :
1254 68579 : if(krb5_storage_is_flags(sp, KRB5_STORAGE_PRINCIPAL_NO_NAME_TYPE))
1255 0 : type = KRB5_NT_UNKNOWN;
1256 68579 : else if((ret = krb5_ret_int32(sp, &type))){
1257 1890 : free(p);
1258 1890 : return ret;
1259 : }
1260 66689 : if((ret = krb5_ret_int32(sp, &ncomp))){
1261 0 : free(p);
1262 0 : return ret;
1263 : }
1264 66689 : if(krb5_storage_is_flags(sp, KRB5_STORAGE_PRINCIPAL_WRONG_NUM_COMPONENTS))
1265 0 : ncomp--;
1266 66689 : if (ncomp < 0) {
1267 0 : free(p);
1268 0 : return EINVAL;
1269 : }
1270 66689 : ret = size_too_large_num(sp, ncomp, sizeof(p->name.name_string.val[0]));
1271 66689 : if (ret) {
1272 0 : free(p);
1273 0 : return ret;
1274 : }
1275 66689 : p->name.name_type = type;
1276 66689 : p->name.name_string.len = ncomp;
1277 66689 : ret = krb5_ret_string(sp, &p->realm);
1278 66689 : if(ret) {
1279 0 : free(p);
1280 0 : return ret;
1281 : }
1282 66689 : p->name.name_string.val = calloc(ncomp, sizeof(p->name.name_string.val[0]));
1283 66689 : if(p->name.name_string.val == NULL && ncomp != 0){
1284 0 : free(p->realm);
1285 0 : free(p);
1286 0 : return ENOMEM;
1287 : }
1288 164576 : for(i = 0; i < ncomp; i++){
1289 97887 : ret = krb5_ret_string(sp, &p->name.name_string.val[i]);
1290 97887 : if(ret) {
1291 0 : while (i >= 0)
1292 0 : free(p->name.name_string.val[i--]);
1293 0 : free(p->realm);
1294 0 : free(p);
1295 0 : return ret;
1296 : }
1297 : }
1298 66689 : *princ = p;
1299 66689 : return 0;
1300 : }
1301 :
1302 : /**
1303 : * Store a keyblock to the storage.
1304 : *
1305 : * @param sp the storage buffer to write to
1306 : * @param p the keyblock to write
1307 : *
1308 : * @return 0 on success, a Kerberos 5 error code on failure.
1309 : *
1310 : * @ingroup krb5_storage
1311 : */
1312 :
1313 : KRB5_LIB_FUNCTION krb5_error_code KRB5_LIB_CALL
1314 74503 : krb5_store_keyblock(krb5_storage *sp, krb5_keyblock p)
1315 : {
1316 978 : int ret;
1317 74503 : ret = krb5_store_int16(sp, p.keytype);
1318 74503 : if(ret) return ret;
1319 :
1320 74503 : if(krb5_storage_is_flags(sp, KRB5_STORAGE_KEYBLOCK_KEYTYPE_TWICE)){
1321 : /* this should really be enctype, but it is the same as
1322 : keytype nowadays */
1323 0 : ret = krb5_store_int16(sp, p.keytype);
1324 0 : if(ret) return ret;
1325 : }
1326 :
1327 74503 : ret = krb5_store_data(sp, p.keyvalue);
1328 74503 : return ret;
1329 : }
1330 :
1331 : /**
1332 : * Read a keyblock from the storage.
1333 : *
1334 : * @param sp the storage buffer to write to
1335 : * @param p the keyblock read from storage, free using krb5_free_keyblock()
1336 : *
1337 : * @return 0 on success, a Kerberos 5 error code on failure.
1338 : *
1339 : * @ingroup krb5_storage
1340 : */
1341 :
1342 : KRB5_LIB_FUNCTION krb5_error_code KRB5_LIB_CALL
1343 102707 : krb5_ret_keyblock(krb5_storage *sp, krb5_keyblock *p)
1344 : {
1345 978 : int ret;
1346 978 : int16_t tmp;
1347 :
1348 102707 : ret = krb5_ret_int16(sp, &tmp);
1349 102707 : if(ret) return ret;
1350 102707 : p->keytype = tmp;
1351 :
1352 102707 : if(krb5_storage_is_flags(sp, KRB5_STORAGE_KEYBLOCK_KEYTYPE_TWICE)){
1353 0 : ret = krb5_ret_int16(sp, &tmp);
1354 0 : if(ret) return ret;
1355 : }
1356 :
1357 102707 : ret = krb5_ret_data(sp, &p->keyvalue);
1358 102707 : return ret;
1359 : }
1360 :
1361 : /**
1362 : * Write a times block to storage.
1363 : *
1364 : * @param sp the storage buffer to write to
1365 : * @param times the times block to write.
1366 : *
1367 : * @return 0 on success, a Kerberos 5 error code on failure.
1368 : *
1369 : * @ingroup krb5_storage
1370 : */
1371 :
1372 : KRB5_LIB_FUNCTION krb5_error_code KRB5_LIB_CALL
1373 1879 : krb5_store_times(krb5_storage *sp, krb5_times times)
1374 : {
1375 0 : int ret;
1376 1879 : ret = krb5_store_int32(sp, times.authtime);
1377 1879 : if(ret) return ret;
1378 1879 : ret = krb5_store_int32(sp, times.starttime);
1379 1879 : if(ret) return ret;
1380 1879 : ret = krb5_store_int32(sp, times.endtime);
1381 1879 : if(ret) return ret;
1382 1879 : ret = krb5_store_int32(sp, times.renew_till);
1383 1879 : return ret;
1384 : }
1385 :
1386 : /**
1387 : * Read a times block from the storage.
1388 : *
1389 : * @param sp the storage buffer to write to
1390 : * @param times the times block read from storage
1391 : *
1392 : * @return 0 on success, a Kerberos 5 error code on failure.
1393 : *
1394 : * @ingroup krb5_storage
1395 : */
1396 :
1397 : KRB5_LIB_FUNCTION krb5_error_code KRB5_LIB_CALL
1398 30083 : krb5_ret_times(krb5_storage *sp, krb5_times *times)
1399 : {
1400 0 : int ret;
1401 0 : int32_t tmp;
1402 :
1403 30083 : ret = krb5_ret_int32(sp, &tmp);
1404 30083 : if (ret) return ret;
1405 30083 : times->authtime = tmp;
1406 30083 : ret = krb5_ret_int32(sp, &tmp);
1407 30083 : if (ret) return ret;
1408 30083 : times->starttime = tmp;
1409 30083 : ret = krb5_ret_int32(sp, &tmp);
1410 30083 : if (ret) return ret;
1411 30083 : times->endtime = tmp;
1412 30083 : ret = krb5_ret_int32(sp, &tmp);
1413 30083 : if (ret) return ret;
1414 30083 : times->renew_till = tmp;
1415 30083 : return ret;
1416 : }
1417 :
1418 : /**
1419 : * Write a address block to storage.
1420 : *
1421 : * @param sp the storage buffer to write to
1422 : * @param p the address block to write.
1423 : *
1424 : * @return 0 on success, a Kerberos 5 error code on failure.
1425 : *
1426 : * @ingroup krb5_storage
1427 : */
1428 :
1429 : KRB5_LIB_FUNCTION krb5_error_code KRB5_LIB_CALL
1430 0 : krb5_store_address(krb5_storage *sp, krb5_address p)
1431 : {
1432 0 : int ret;
1433 0 : ret = krb5_store_int16(sp, p.addr_type);
1434 0 : if(ret) return ret;
1435 0 : ret = krb5_store_data(sp, p.address);
1436 0 : return ret;
1437 : }
1438 :
1439 : /**
1440 : * Read a address block from the storage.
1441 : *
1442 : * @param sp the storage buffer to write to
1443 : * @param adr the address block read from storage
1444 : *
1445 : * @return 0 on success, a Kerberos 5 error code on failure.
1446 : *
1447 : * @ingroup krb5_storage
1448 : */
1449 :
1450 : KRB5_LIB_FUNCTION krb5_error_code KRB5_LIB_CALL
1451 0 : krb5_ret_address(krb5_storage *sp, krb5_address *adr)
1452 : {
1453 0 : int16_t t;
1454 0 : int ret;
1455 0 : ret = krb5_ret_int16(sp, &t);
1456 0 : if(ret) return ret;
1457 0 : adr->addr_type = t;
1458 0 : ret = krb5_ret_data(sp, &adr->address);
1459 0 : return ret;
1460 : }
1461 :
1462 : /**
1463 : * Write a addresses block to storage.
1464 : *
1465 : * @param sp the storage buffer to write to
1466 : * @param p the addresses block to write.
1467 : *
1468 : * @return 0 on success, a Kerberos 5 error code on failure.
1469 : *
1470 : * @ingroup krb5_storage
1471 : */
1472 :
1473 : KRB5_LIB_FUNCTION krb5_error_code KRB5_LIB_CALL
1474 1879 : krb5_store_addrs(krb5_storage *sp, krb5_addresses p)
1475 : {
1476 0 : size_t i;
1477 0 : int ret;
1478 1879 : ret = krb5_store_int32(sp, p.len);
1479 1879 : if(ret) return ret;
1480 1879 : for(i = 0; i<p.len; i++){
1481 0 : ret = krb5_store_address(sp, p.val[i]);
1482 0 : if(ret) break;
1483 : }
1484 1879 : return ret;
1485 : }
1486 :
1487 : /**
1488 : * Read a addresses block from the storage.
1489 : *
1490 : * @param sp the storage buffer to write to
1491 : * @param adr the addresses block read from storage
1492 : *
1493 : * @return 0 on success, a Kerberos 5 error code on failure.
1494 : *
1495 : * @ingroup krb5_storage
1496 : */
1497 :
1498 : KRB5_LIB_FUNCTION krb5_error_code KRB5_LIB_CALL
1499 30083 : krb5_ret_addrs(krb5_storage *sp, krb5_addresses *adr)
1500 : {
1501 0 : size_t i;
1502 0 : int ret;
1503 0 : int32_t tmp;
1504 :
1505 30083 : ret = krb5_ret_int32(sp, &tmp);
1506 30083 : if(ret) return ret;
1507 30083 : ret = size_too_large_num(sp, tmp, sizeof(adr->val[0]));
1508 30083 : if (ret) return ret;
1509 30083 : adr->len = tmp;
1510 30083 : ALLOC(adr->val, adr->len);
1511 30083 : if (adr->val == NULL && adr->len != 0)
1512 0 : return ENOMEM;
1513 30083 : for(i = 0; i < adr->len; i++){
1514 0 : ret = krb5_ret_address(sp, &adr->val[i]);
1515 0 : if(ret) break;
1516 : }
1517 30083 : return ret;
1518 : }
1519 :
1520 : /**
1521 : * Write a auth data block to storage.
1522 : *
1523 : * @param sp the storage buffer to write to
1524 : * @param auth the auth data block to write.
1525 : *
1526 : * @return 0 on success, a Kerberos 5 error code on failure.
1527 : *
1528 : * @ingroup krb5_storage
1529 : */
1530 :
1531 : KRB5_LIB_FUNCTION krb5_error_code KRB5_LIB_CALL
1532 1879 : krb5_store_authdata(krb5_storage *sp, krb5_authdata auth)
1533 : {
1534 0 : krb5_error_code ret;
1535 0 : size_t i;
1536 1879 : ret = krb5_store_int32(sp, auth.len);
1537 1879 : if(ret) return ret;
1538 1879 : for(i = 0; i < auth.len; i++){
1539 0 : ret = krb5_store_int16(sp, auth.val[i].ad_type);
1540 0 : if(ret) break;
1541 0 : ret = krb5_store_data(sp, auth.val[i].ad_data);
1542 0 : if(ret) break;
1543 : }
1544 1879 : return 0;
1545 : }
1546 :
1547 : /**
1548 : * Read a auth data from the storage.
1549 : *
1550 : * @param sp the storage buffer to write to
1551 : * @param auth the auth data block read from storage
1552 : *
1553 : * @return 0 on success, a Kerberos 5 error code on failure.
1554 : *
1555 : * @ingroup krb5_storage
1556 : */
1557 :
1558 : KRB5_LIB_FUNCTION krb5_error_code KRB5_LIB_CALL
1559 30083 : krb5_ret_authdata(krb5_storage *sp, krb5_authdata *auth)
1560 : {
1561 0 : krb5_error_code ret;
1562 0 : int32_t tmp;
1563 0 : int16_t tmp2;
1564 0 : int i;
1565 30083 : ret = krb5_ret_int32(sp, &tmp);
1566 30083 : if(ret) return ret;
1567 30083 : ret = size_too_large_num(sp, tmp, sizeof(auth->val[0]));
1568 30083 : if (ret) return ret;
1569 30083 : ALLOC_SEQ(auth, tmp);
1570 30083 : if (auth->val == NULL && tmp != 0)
1571 0 : return ENOMEM;
1572 30083 : for(i = 0; i < tmp; i++){
1573 0 : ret = krb5_ret_int16(sp, &tmp2);
1574 0 : if(ret) break;
1575 0 : auth->val[i].ad_type = tmp2;
1576 0 : ret = krb5_ret_data(sp, &auth->val[i].ad_data);
1577 0 : if(ret) break;
1578 : }
1579 30083 : return ret;
1580 : }
1581 :
1582 : static int32_t
1583 31962 : bitswap32(int32_t b)
1584 : {
1585 31962 : int32_t r = 0;
1586 0 : int i;
1587 1054746 : for (i = 0; i < 32; i++) {
1588 1022784 : r = r << 1 | (b & 1);
1589 1022784 : b = b >> 1;
1590 : }
1591 31962 : return r;
1592 : }
1593 :
1594 : /**
1595 : * Write a credentials block to storage.
1596 : *
1597 : * @param sp the storage buffer to write to
1598 : * @param creds the creds block to write.
1599 : *
1600 : * @return 0 on success, a Kerberos 5 error code on failure.
1601 : *
1602 : * @ingroup krb5_storage
1603 : */
1604 :
1605 : KRB5_LIB_FUNCTION krb5_error_code KRB5_LIB_CALL
1606 1879 : krb5_store_creds(krb5_storage *sp, krb5_creds *creds)
1607 : {
1608 0 : int ret;
1609 :
1610 1879 : ret = krb5_store_principal(sp, creds->client);
1611 1879 : if(ret)
1612 0 : return ret;
1613 1879 : ret = krb5_store_principal(sp, creds->server);
1614 1879 : if(ret)
1615 0 : return ret;
1616 1879 : ret = krb5_store_keyblock(sp, creds->session);
1617 1879 : if(ret)
1618 0 : return ret;
1619 1879 : ret = krb5_store_times(sp, creds->times);
1620 1879 : if(ret)
1621 0 : return ret;
1622 1879 : ret = krb5_store_int8(sp, creds->second_ticket.length != 0); /* is_skey */
1623 1879 : if(ret)
1624 0 : return ret;
1625 1879 : ret = krb5_store_int32(sp, bitswap32(TicketFlags2int(creds->flags.b)));
1626 1879 : if(ret)
1627 0 : return ret;
1628 1879 : ret = krb5_store_addrs(sp, creds->addresses);
1629 1879 : if(ret)
1630 0 : return ret;
1631 1879 : ret = krb5_store_authdata(sp, creds->authdata);
1632 1879 : if(ret)
1633 0 : return ret;
1634 1879 : ret = krb5_store_data(sp, creds->ticket);
1635 1879 : if(ret)
1636 0 : return ret;
1637 1879 : ret = krb5_store_data(sp, creds->second_ticket);
1638 1879 : return ret;
1639 : }
1640 :
1641 : /**
1642 : * Read a credentials block from the storage.
1643 : *
1644 : * @param sp the storage buffer to write to
1645 : * @param creds the credentials block read from storage
1646 : *
1647 : * @return 0 on success, a Kerberos 5 error code on failure.
1648 : *
1649 : * @ingroup krb5_storage
1650 : */
1651 :
1652 : KRB5_LIB_FUNCTION krb5_error_code KRB5_LIB_CALL
1653 31973 : krb5_ret_creds(krb5_storage *sp, krb5_creds *creds)
1654 : {
1655 0 : krb5_error_code ret;
1656 0 : int8_t dummy8;
1657 0 : int32_t dummy32;
1658 :
1659 31973 : memset(creds, 0, sizeof(*creds));
1660 31973 : ret = krb5_ret_principal (sp, &creds->client);
1661 31973 : if(ret) goto cleanup;
1662 30083 : ret = krb5_ret_principal (sp, &creds->server);
1663 30083 : if(ret) goto cleanup;
1664 30083 : ret = krb5_ret_keyblock (sp, &creds->session);
1665 30083 : if(ret) goto cleanup;
1666 30083 : ret = krb5_ret_times (sp, &creds->times);
1667 30083 : if(ret) goto cleanup;
1668 30083 : ret = krb5_ret_int8 (sp, &dummy8);
1669 30083 : if(ret) goto cleanup;
1670 30083 : ret = krb5_ret_int32 (sp, &dummy32);
1671 30083 : if(ret) goto cleanup;
1672 30083 : creds->flags.b = int2TicketFlags(bitswap32(dummy32));
1673 30083 : ret = krb5_ret_addrs (sp, &creds->addresses);
1674 30083 : if(ret) goto cleanup;
1675 30083 : ret = krb5_ret_authdata (sp, &creds->authdata);
1676 30083 : if(ret) goto cleanup;
1677 30083 : ret = krb5_ret_data (sp, &creds->ticket);
1678 30083 : if(ret) goto cleanup;
1679 30083 : ret = krb5_ret_data (sp, &creds->second_ticket);
1680 31973 : cleanup:
1681 0 : if(ret) {
1682 : #if 0
1683 : krb5_free_cred_contents(context, creds); /* XXX */
1684 : #endif
1685 0 : }
1686 31973 : return ret;
1687 : }
1688 :
1689 : #define SC_CLIENT_PRINCIPAL 0x0001
1690 : #define SC_SERVER_PRINCIPAL 0x0002
1691 : #define SC_SESSION_KEY 0x0004
1692 : #define SC_TICKET 0x0008
1693 : #define SC_SECOND_TICKET 0x0010
1694 : #define SC_AUTHDATA 0x0020
1695 : #define SC_ADDRESSES 0x0040
1696 :
1697 : /**
1698 : * Write a tagged credentials block to storage.
1699 : *
1700 : * @param sp the storage buffer to write to
1701 : * @param creds the creds block to write.
1702 : *
1703 : * @return 0 on success, a Kerberos 5 error code on failure.
1704 : *
1705 : * @ingroup krb5_storage
1706 : */
1707 :
1708 : KRB5_LIB_FUNCTION krb5_error_code KRB5_LIB_CALL
1709 0 : krb5_store_creds_tag(krb5_storage *sp, krb5_creds *creds)
1710 : {
1711 0 : int ret;
1712 0 : int32_t header = 0;
1713 :
1714 0 : if (creds->client)
1715 0 : header |= SC_CLIENT_PRINCIPAL;
1716 0 : if (creds->server)
1717 0 : header |= SC_SERVER_PRINCIPAL;
1718 0 : if (creds->session.keytype != ETYPE_NULL)
1719 0 : header |= SC_SESSION_KEY;
1720 0 : if (creds->ticket.data)
1721 0 : header |= SC_TICKET;
1722 0 : if (creds->second_ticket.length)
1723 0 : header |= SC_SECOND_TICKET;
1724 0 : if (creds->authdata.len)
1725 0 : header |= SC_AUTHDATA;
1726 0 : if (creds->addresses.len)
1727 0 : header |= SC_ADDRESSES;
1728 :
1729 0 : ret = krb5_store_int32(sp, header);
1730 0 : if (ret)
1731 0 : return ret;
1732 :
1733 0 : if (creds->client) {
1734 0 : ret = krb5_store_principal(sp, creds->client);
1735 0 : if(ret)
1736 0 : return ret;
1737 : }
1738 :
1739 0 : if (creds->server) {
1740 0 : ret = krb5_store_principal(sp, creds->server);
1741 0 : if(ret)
1742 0 : return ret;
1743 : }
1744 :
1745 0 : if (creds->session.keytype != ETYPE_NULL) {
1746 0 : ret = krb5_store_keyblock(sp, creds->session);
1747 0 : if(ret)
1748 0 : return ret;
1749 : }
1750 :
1751 0 : ret = krb5_store_times(sp, creds->times);
1752 0 : if(ret)
1753 0 : return ret;
1754 0 : ret = krb5_store_int8(sp, creds->second_ticket.length != 0); /* is_skey */
1755 0 : if(ret)
1756 0 : return ret;
1757 :
1758 0 : ret = krb5_store_int32(sp, bitswap32(TicketFlags2int(creds->flags.b)));
1759 0 : if(ret)
1760 0 : return ret;
1761 :
1762 0 : if (creds->addresses.len) {
1763 0 : ret = krb5_store_addrs(sp, creds->addresses);
1764 0 : if(ret)
1765 0 : return ret;
1766 : }
1767 :
1768 0 : if (creds->authdata.len) {
1769 0 : ret = krb5_store_authdata(sp, creds->authdata);
1770 0 : if(ret)
1771 0 : return ret;
1772 : }
1773 :
1774 0 : if (creds->ticket.data) {
1775 0 : ret = krb5_store_data(sp, creds->ticket);
1776 0 : if(ret)
1777 0 : return ret;
1778 : }
1779 :
1780 0 : if (creds->second_ticket.data) {
1781 0 : ret = krb5_store_data(sp, creds->second_ticket);
1782 0 : if (ret)
1783 0 : return ret;
1784 : }
1785 :
1786 0 : return ret;
1787 : }
1788 :
1789 : /**
1790 : * Read a tagged credentials block from the storage.
1791 : *
1792 : * @param sp the storage buffer to write to
1793 : * @param creds the credentials block read from storage
1794 : *
1795 : * @return 0 on success, a Kerberos 5 error code on failure.
1796 : *
1797 : * @ingroup krb5_storage
1798 : */
1799 :
1800 : KRB5_LIB_FUNCTION krb5_error_code KRB5_LIB_CALL
1801 0 : krb5_ret_creds_tag(krb5_storage *sp,
1802 : krb5_creds *creds)
1803 : {
1804 0 : krb5_error_code ret;
1805 0 : int8_t dummy8;
1806 0 : int32_t dummy32, header;
1807 :
1808 0 : memset(creds, 0, sizeof(*creds));
1809 :
1810 0 : ret = krb5_ret_int32 (sp, &header);
1811 0 : if (ret) goto cleanup;
1812 :
1813 0 : if (header & SC_CLIENT_PRINCIPAL) {
1814 0 : ret = krb5_ret_principal (sp, &creds->client);
1815 0 : if(ret) goto cleanup;
1816 : }
1817 0 : if (header & SC_SERVER_PRINCIPAL) {
1818 0 : ret = krb5_ret_principal (sp, &creds->server);
1819 0 : if(ret) goto cleanup;
1820 : }
1821 0 : if (header & SC_SESSION_KEY) {
1822 0 : ret = krb5_ret_keyblock (sp, &creds->session);
1823 0 : if(ret) goto cleanup;
1824 : }
1825 0 : ret = krb5_ret_times (sp, &creds->times);
1826 0 : if(ret) goto cleanup;
1827 0 : ret = krb5_ret_int8 (sp, &dummy8);
1828 0 : if(ret) goto cleanup;
1829 0 : ret = krb5_ret_int32 (sp, &dummy32);
1830 0 : if(ret) goto cleanup;
1831 0 : creds->flags.b = int2TicketFlags(bitswap32(dummy32));
1832 0 : if (header & SC_ADDRESSES) {
1833 0 : ret = krb5_ret_addrs (sp, &creds->addresses);
1834 0 : if(ret) goto cleanup;
1835 : }
1836 0 : if (header & SC_AUTHDATA) {
1837 0 : ret = krb5_ret_authdata (sp, &creds->authdata);
1838 0 : if(ret) goto cleanup;
1839 : }
1840 0 : if (header & SC_TICKET) {
1841 0 : ret = krb5_ret_data (sp, &creds->ticket);
1842 0 : if(ret) goto cleanup;
1843 : }
1844 0 : if (header & SC_SECOND_TICKET) {
1845 0 : ret = krb5_ret_data (sp, &creds->second_ticket);
1846 0 : if(ret) goto cleanup;
1847 : }
1848 :
1849 0 : cleanup:
1850 0 : if(ret) {
1851 : #if 0
1852 : krb5_free_cred_contents(context, creds); /* XXX */
1853 : #endif
1854 0 : }
1855 0 : return ret;
1856 : }
1857 :
1858 : KRB5_LIB_FUNCTION krb5_error_code KRB5_LIB_CALL
1859 410140 : _krb5_ret_data_at_offset(krb5_storage *sp,
1860 : size_t offset,
1861 : size_t length,
1862 : krb5_data *data)
1863 : {
1864 10156 : krb5_error_code ret;
1865 10156 : off_t cur, size;
1866 :
1867 410140 : krb5_data_zero(data);
1868 :
1869 410140 : cur = sp->seek(sp, 0, SEEK_CUR);
1870 410140 : if (cur < 0)
1871 0 : return HEIM_ERR_NOT_SEEKABLE;
1872 :
1873 410140 : size = sp->seek(sp, 0, SEEK_END);
1874 410140 : if (offset + length > size) {
1875 0 : ret = ERANGE;
1876 0 : goto cleanup;
1877 : }
1878 :
1879 410140 : ret = krb5_data_alloc(data, length);
1880 410140 : if (ret)
1881 0 : goto cleanup;
1882 :
1883 410140 : if (length) {
1884 410140 : sp->seek(sp, offset, SEEK_SET);
1885 :
1886 410140 : size = sp->fetch(sp, data->data, length);
1887 410140 : if (size < 0 || (size_t)size != length)
1888 0 : return sp->eof_code;
1889 : }
1890 :
1891 410140 : cleanup:
1892 410140 : sp->seek(sp, cur, SEEK_SET);
1893 :
1894 410140 : return ret;
1895 : }
1896 :
1897 : KRB5_LIB_FUNCTION krb5_error_code KRB5_LIB_CALL
1898 307605 : _krb5_ret_utf8_from_ucs2le_at_offset(krb5_storage *sp,
1899 : off_t offset,
1900 : size_t length,
1901 : char **utf8)
1902 : {
1903 7617 : krb5_error_code ret;
1904 7617 : krb5_data data;
1905 307605 : size_t ucs2len = length / 2;
1906 307605 : uint16_t *ucs2 = NULL;
1907 7617 : size_t u8len;
1908 307605 : unsigned int flags = WIND_RW_LE;
1909 :
1910 307605 : *utf8 = NULL;
1911 :
1912 307605 : krb5_data_zero(&data);
1913 :
1914 307605 : ret = _krb5_ret_data_at_offset(sp, offset, length, &data);
1915 307605 : if (ret)
1916 0 : goto out;
1917 :
1918 307605 : ucs2 = malloc(sizeof(ucs2[0]) * ucs2len);
1919 307605 : if (ucs2 == NULL) {
1920 0 : ret = ENOMEM;
1921 0 : goto out;
1922 : }
1923 :
1924 307605 : ret = wind_ucs2read(data.data, data.length, &flags, ucs2, &ucs2len);
1925 307605 : if (ret)
1926 0 : goto out;
1927 :
1928 307605 : ret = wind_ucs2utf8_length(ucs2, ucs2len, &u8len);
1929 307605 : if (ret)
1930 0 : goto out;
1931 :
1932 307605 : u8len += 1; /* Add space for NUL */
1933 :
1934 307605 : *utf8 = malloc(u8len);
1935 307605 : if (*utf8 == NULL) {
1936 0 : ret = ENOMEM;
1937 0 : goto out;
1938 : }
1939 :
1940 307605 : ret = wind_ucs2utf8(ucs2, ucs2len, *utf8, &u8len);
1941 307605 : if (ret)
1942 0 : goto out;
1943 :
1944 307605 : out:
1945 307605 : if (ret && *utf8) {
1946 0 : free(*utf8);
1947 0 : *utf8 = NULL;
1948 : }
1949 307605 : free(ucs2);
1950 307605 : krb5_data_free(&data);
1951 :
1952 307605 : return ret;
1953 : }
1954 :
1955 : KRB5_LIB_FUNCTION krb5_error_code KRB5_LIB_CALL
1956 0 : _krb5_store_data_at_offset(krb5_storage *sp,
1957 : size_t offset,
1958 : const krb5_data *data)
1959 : {
1960 0 : krb5_error_code ret;
1961 0 : krb5_ssize_t nbytes;
1962 0 : off_t pos;
1963 :
1964 0 : if (offset == (size_t)-1) {
1965 0 : if (data == NULL || data->data == NULL) {
1966 0 : offset = 0;
1967 : } else {
1968 0 : pos = sp->seek(sp, 0, SEEK_CUR);
1969 0 : offset = sp->seek(sp, 0, SEEK_END);
1970 0 : sp->seek(sp, pos, SEEK_SET);
1971 :
1972 0 : if (offset == (size_t)-1)
1973 0 : return HEIM_ERR_NOT_SEEKABLE;
1974 : }
1975 : }
1976 :
1977 0 : if (offset > 0xFFFF)
1978 0 : return ERANGE;
1979 0 : else if ((offset != 0) != (data && data->data))
1980 0 : return EINVAL;
1981 0 : else if (data && data->length > 0xFFFF)
1982 0 : return ERANGE;
1983 :
1984 0 : ret = krb5_store_uint16(sp, data ? (uint16_t)data->length : 0);
1985 0 : if (ret == 0)
1986 0 : ret = krb5_store_uint16(sp, (uint16_t)offset);
1987 0 : if (ret == 0 && offset) {
1988 0 : pos = sp->seek(sp, 0, SEEK_CUR);
1989 0 : sp->seek(sp, offset, SEEK_SET);
1990 0 : nbytes = krb5_storage_write(sp, data->data, data->length);
1991 0 : if ((size_t)nbytes != data->length)
1992 0 : ret = sp->eof_code;
1993 0 : sp->seek(sp, pos, SEEK_SET);
1994 : }
1995 :
1996 0 : return ret;
1997 : }
1998 :
1999 : KRB5_LIB_FUNCTION krb5_error_code KRB5_LIB_CALL
2000 0 : _krb5_store_utf8_as_ucs2le_at_offset(krb5_storage *sp,
2001 : off_t offset,
2002 : const char *utf8)
2003 : {
2004 0 : krb5_error_code ret;
2005 0 : size_t ucs2_len, ucs2le_size;
2006 0 : uint16_t *ucs2, *ucs2le;
2007 0 : unsigned int flags;
2008 :
2009 0 : if (utf8) {
2010 0 : ret = wind_utf8ucs2_length(utf8, &ucs2_len);
2011 0 : if (ret)
2012 0 : return ret;
2013 :
2014 0 : ucs2 = malloc(sizeof(ucs2[0]) * ucs2_len);
2015 0 : if (ucs2 == NULL)
2016 0 : return ENOMEM;
2017 :
2018 0 : ret = wind_utf8ucs2(utf8, ucs2, &ucs2_len);
2019 0 : if (ret) {
2020 0 : free(ucs2);
2021 0 : return ret;
2022 : }
2023 :
2024 0 : ucs2le_size = (ucs2_len + 1) * 2;
2025 0 : ucs2le = malloc(ucs2le_size);
2026 0 : if (ucs2le == NULL) {
2027 0 : free(ucs2);
2028 0 : return ENOMEM;
2029 : }
2030 :
2031 0 : flags = WIND_RW_LE;
2032 0 : ret = wind_ucs2write(ucs2, ucs2_len, &flags, ucs2le, &ucs2le_size);
2033 0 : free(ucs2);
2034 0 : if (ret) {
2035 0 : free(ucs2le);
2036 0 : return ret;
2037 : }
2038 :
2039 0 : ucs2le_size = ucs2_len * 2;
2040 : } else {
2041 0 : ucs2le = NULL;
2042 0 : ucs2le_size = 0;
2043 0 : offset = 0;
2044 : }
2045 :
2046 : {
2047 0 : krb5_data data;
2048 :
2049 0 : data.data = ucs2le;
2050 0 : data.length = ucs2le_size;
2051 :
2052 0 : ret = _krb5_store_data_at_offset(sp, offset, &data);
2053 : }
2054 :
2055 0 : free(ucs2le);
2056 :
2057 0 : return ret;
2058 : }
|