*** empty log message ***
[yaz-moved-to-github.git] / odr / ber_oid.c
1 /*
2  * Copyright (c) 1995, Index Data
3  * See the file LICENSE for details.
4  * Sebastian Hammer, Adam Dickmeiss
5  *
6  * $Log: ber_oid.c,v $
7  * Revision 1.9  1995-09-29 17:12:19  quinn
8  * Smallish
9  *
10  * Revision 1.8  1995/09/27  15:02:56  quinn
11  * Modified function heads & prototypes.
12  *
13  * Revision 1.7  1995/05/16  08:50:47  quinn
14  * License, documentation, and memory fixes
15  *
16  * Revision 1.6  1995/04/18  08:15:18  quinn
17  * Added dynamic memory allocation on encoding (whew). Code is now somewhat
18  * neater. We'll make the same change for decoding one day.
19  *
20  * Revision 1.5  1995/03/20  12:18:22  quinn
21  * Fixed bug in ber_oid
22  *
23  * Revision 1.4  1995/03/08  12:12:11  quinn
24  * Added better error checking.
25  *
26  * Revision 1.3  1995/03/01  08:40:56  quinn
27  * Smallish changes.
28  *
29  * Revision 1.2  1995/02/14  20:39:55  quinn
30  * Fixed bugs in completeBER and (serious one in) ber_oid.
31  *
32  * Revision 1.1  1995/02/03  17:04:36  quinn
33  * Initial revision
34  *
35  */
36
37 #include <odr.h>
38
39 int ber_oidc(ODR o, Odr_oid *p)
40 {
41     int len, lenp, end;
42     int pos, n, res, id;
43     unsigned char octs[8];
44
45     switch (o->direction)
46     {
47         case ODR_DECODE:
48             if ((res = ber_declen(o->bp, &len)) < 1)
49             {
50                 o->error = OPROTO;
51                 return 0;
52             }
53             if (len < 0)
54             {
55                 o->error = OPROTO;
56                 return 0;
57             }
58             o->bp += res;
59             o->left -= res;
60             if (len == 0)
61             {
62                 *p = -1;
63                 return 1;
64             }
65             p[0] = *o->bp / 40;
66             if (p[0] > 2)
67                 p[0] = 2;
68             p[1] = *o->bp - p[0] * 40;
69             o->bp++;
70             o->left--;
71             pos = 2;
72             len--;
73             while (len)
74             {
75                 p[pos] = 0;
76                 do
77                 {
78                     if (!len)
79                     {
80                         o->error = OPROTO;
81                         return 0;
82                     }
83                     p[pos] <<= 7;
84                     p[pos] |= *o->bp & 0X7F;
85                     len--;
86                     o->left--;
87                 }
88                 while (*(o->bp++) & 0X80);
89                 pos++;
90             }
91             p[pos] = -1;
92             return 1;
93         case ODR_ENCODE:
94             /* we'll allow ourselves the quiet luxury of only doing encodings
95                shorter than 127 */
96             lenp = odr_tell(o);
97             if (odr_putc(o, 0) < 0)   /* dummy */
98                 return 0;
99             if (p[0] < 0 && p[1] <= 0)
100             {
101                 o->error = ODATA;
102                 return 0;
103             }
104             for (pos = 1; p[pos] >= 0; pos++)
105             {
106                 id = pos > 1 ? p[pos] : p[0] * 40 + p[1];
107                 n = 0;
108                 do
109                 {
110                     octs[n++] = id & 0X7F;
111                     id >>= 7;
112                 }
113                 while (id);
114                 while (n--)
115                 {
116                     unsigned char p;
117
118                     p = octs[n] | ((n > 0) << 7);
119                     if (odr_putc(o, p) < 0)
120                         return 0;
121                 }
122             }
123             end = odr_tell(o);
124             odr_seek(o, ODR_S_SET, lenp);
125             if (ber_enclen(o, (end - lenp) - 1, 1, 1) != 1)
126             {
127                 o->error = OOTHER;
128                 return 0;
129             }
130             odr_seek(o, ODR_S_END, 0);
131             return 1;
132         default: o->error = OOTHER; return 0;
133     }
134 }