Implemented command update0 in YAZ client which is equivalent to
[yaz-moved-to-github.git] / util / oid.c
1 /*
2  * Copyright (c) 1995-2003, Index Data
3  * See the file LICENSE for details.
4  *
5  * $Id: oid.c,v 1.67 2003-09-02 12:12:13 adam Exp $
6  */
7
8 /*
9  * More or less protocol-transparent OID database.
10  * We could (and should?) extend this so that the user app can add new
11  * entries to the list at initialization.
12  */
13 #if HAVE_CONFIG_H
14 #include <config.h>
15 #endif
16
17 #include <stdlib.h>
18 #include <string.h>
19 #include <ctype.h>
20
21 #include <yaz/oid.h>
22 #include <yaz/yaz-util.h>
23
24 static int z3950_prefix[] = { 1, 2, 840, 10003, -1 };
25 static int sr_prefix[]    = { 1, 0, 10163, -1 };
26
27 struct oident_list {
28     struct oident oident;
29     struct oident_list *next;
30 };
31
32 static struct oident_list *oident_table = NULL;
33 static int oid_value_dynamic = VAL_DYNAMIC;
34 static int oid_init_flag = 0;
35 static NMEM_MUTEX oid_mutex = 0;
36 static NMEM oid_nmem = 0;
37
38 /*
39  * OID database
40  */
41 static oident standard_oids[] =
42 {
43     /* General definitions */
44     {PROTO_GENERAL, CLASS_TRANSYN, VAL_BER,          {2,1,1,-1},
45      "BER" },
46     {PROTO_GENERAL, CLASS_TRANSYN, VAL_ISO2709,      {1,0,2709,1,1,-1},
47      "ISO2709"},
48     {PROTO_GENERAL, CLASS_GENERAL, VAL_ISO_ILL_1,    {1,0,10161,2,1,-1},
49      "ISOILL-1"},
50     /* Z39.50v3 definitions */
51     {PROTO_Z3950,   CLASS_ABSYN,   VAL_APDU,         {2,1,-1},
52      "Z-APDU"},    
53     {PROTO_Z3950,   CLASS_APPCTX,  VAL_BASIC_CTX,    {1,1,-1},
54      "Z-BASIC"},
55     {PROTO_Z3950,   CLASS_ATTSET,  VAL_BIB1,         {3,1,-1},
56      "Bib-1"},
57     {PROTO_Z3950,   CLASS_ATTSET,  VAL_EXP1,         {3,2,-1},
58      "Exp-1"},
59     {PROTO_Z3950,   CLASS_ATTSET,  VAL_EXT1,         {3,3,-1},
60      "Ext-1"},
61     {PROTO_Z3950,   CLASS_ATTSET,  VAL_CCL1,         {3,4,-1},
62      "CCL-1"},
63     {PROTO_Z3950,   CLASS_ATTSET,  VAL_GILS,         {3,5,-1},
64      "GILS-attset"},
65     {PROTO_Z3950,   CLASS_ATTSET,  VAL_GILS,         {3,5,-1},
66      "GILS"},
67     {PROTO_Z3950,   CLASS_ATTSET,  VAL_STAS,         {3,6,-1},
68      "STAS-attset"},
69     {PROTO_Z3950,   CLASS_ATTSET,  VAL_COLLECT1,     {3,7,-1},
70      "Collections-attset"},
71     {PROTO_Z3950,   CLASS_ATTSET,  VAL_CIMI1,        {3,8,-1},
72      "CIMI-attset"},
73     {PROTO_Z3950,   CLASS_ATTSET,  VAL_GEO,          {3,9,-1},
74      "Geo-attset"},
75
76     {PROTO_Z3950,   CLASS_ATTSET,  VAL_ZBIG,         {3,10,-1},
77      "ZBIG"},
78     {PROTO_Z3950,   CLASS_ATTSET,  VAL_UTIL,         {3,11,-1},
79      "Util"},
80     {PROTO_Z3950,   CLASS_ATTSET,  VAL_XD1,          {3,12,-1},
81      "XD-1"},
82     {PROTO_Z3950,   CLASS_ATTSET,  VAL_ZTHES,        {3,13,-1},
83      "Zthes"},
84     {PROTO_Z3950,   CLASS_ATTSET,  VAL_FIN1,         {3,14,-1},
85      "Fin-1"},
86     {PROTO_Z3950,   CLASS_ATTSET,  VAL_DAN1,         {3,15,-1},
87      "Dan-1"},
88     {PROTO_Z3950,   CLASS_ATTSET,  VAL_HOLDINGS,     {3,16,-1},
89      "Holdings"},
90     {PROTO_Z3950,   CLASS_ATTSET,  VAL_USMARC,       {3,17,-1},
91      "MARC"},
92     {PROTO_Z3950,   CLASS_ATTSET,  VAL_BIB2,         {3,18,-1},
93      "Bib-2"},
94     {PROTO_Z3950,   CLASS_ATTSET,  VAL_ZEEREX,       {3,19,-1},
95      "ZeeRex"},
96 #if 0  /* This is the spawn of Satan.  Use Zthes-1 instead */
97     {PROTO_Z3950,   CLASS_ATTSET,  VAL_THESAURUS,    {3,1000,81,1,-1},     
98      "Thesaurus-attset"},
99 #endif /*0*/
100     {PROTO_Z3950,   CLASS_ATTSET,  VAL_IDXPATH,      {3,1000,81,2,-1},
101      "IDXPATH"},
102     {PROTO_Z3950,   CLASS_DIAGSET, VAL_BIB1,         {4,1,-1},
103      "Bib-1"},
104     {PROTO_Z3950,   CLASS_DIAGSET, VAL_DIAG1,        {4,2,-1},
105      "Diag-1"},
106     {PROTO_Z3950,   CLASS_DIAGSET, VAL_DIAG_ES,      {4,3,-1},
107      "Diag-ES"},
108     {PROTO_Z3950,   CLASS_DIAGSET, VAL_DIAG_GENERAL, {4,3,-1},
109      "Diag-General"},
110     {PROTO_Z3950,   CLASS_RECSYN,  VAL_UNIMARC,      {5,1,-1},
111      "Unimarc"},
112     {PROTO_Z3950,   CLASS_RECSYN,  VAL_INTERMARC,    {5,2,-1},
113      "Intermarc"},
114     {PROTO_Z3950,   CLASS_RECSYN,  VAL_CCF,          {5,3,-1},
115      "CCF"},
116     {PROTO_Z3950,   CLASS_RECSYN,  VAL_USMARC,       {5,10,-1},
117      "USmarc"},
118     {PROTO_Z3950,   CLASS_RECSYN,  VAL_UKMARC,       {5,11,-1},
119      "UKmarc"},
120     {PROTO_Z3950,   CLASS_RECSYN,  VAL_NORMARC,      {5,12,-1},
121      "Normarc"},
122     {PROTO_Z3950,   CLASS_RECSYN,  VAL_LIBRISMARC,   {5,13,-1},
123      "Librismarc"},
124     {PROTO_Z3950,   CLASS_RECSYN,  VAL_DANMARC,      {5,14,-1},
125      "Danmarc"},
126     {PROTO_Z3950,   CLASS_RECSYN,  VAL_FINMARC,      {5,15,-1},
127      "Finmarc"},
128     {PROTO_Z3950,   CLASS_RECSYN,  VAL_MAB,          {5,16,-1},
129      "MAB"},
130     {PROTO_Z3950,   CLASS_RECSYN,  VAL_CANMARC,      {5,17,-1},
131      "Canmarc"},
132     {PROTO_Z3950,   CLASS_RECSYN,  VAL_SBN,          {5,18,-1},
133      "SBN"},
134     {PROTO_Z3950,   CLASS_RECSYN,  VAL_PICAMARC,     {5,19,-1},
135      "Picamarc"},
136     {PROTO_Z3950,   CLASS_RECSYN,  VAL_AUSMARC,      {5,20,-1},
137      "Ausmarc"},
138     {PROTO_Z3950,   CLASS_RECSYN,  VAL_IBERMARC,     {5,21,-1},
139      "Ibermarc"},
140     {PROTO_Z3950,   CLASS_RECSYN,  VAL_CATMARC,      {5,22,-1},
141      "Carmarc"},
142     {PROTO_Z3950,   CLASS_RECSYN,  VAL_MALMARC,      {5,23,-1},
143      "Malmarc"},
144     {PROTO_Z3950,   CLASS_RECSYN,  VAL_JPMARC,       {5,24,-1},
145      "JPmarc"},
146     {PROTO_Z3950,   CLASS_RECSYN,  VAL_SWEMARC,      {5,25,-1},
147      "SWEmarc"},
148     {PROTO_Z3950,   CLASS_RECSYN,  VAL_SIGLEMARC,    {5,26,-1},
149      "SIGLEmarc"},
150     {PROTO_Z3950,   CLASS_RECSYN,  VAL_ISDSMARC,     {5,27,-1},
151      "ISDSmarc"},
152     {PROTO_Z3950,   CLASS_RECSYN,  VAL_RUSMARC,      {5,28,-1},
153      "RUSmarc"},
154     {PROTO_Z3950,   CLASS_RECSYN,  VAL_HUNMARC,      {5,29,-1},
155      "Hunmarc"},
156     {PROTO_Z3950,   CLASS_RECSYN,  VAL_EXPLAIN,      {5,100,-1},
157      "Explain"},
158     {PROTO_Z3950,   CLASS_RECSYN,  VAL_SUTRS,        {5,101,-1},
159      "SUTRS"},
160     {PROTO_Z3950,   CLASS_RECSYN,  VAL_OPAC,         {5,102,-1},
161      "OPAC"},
162     {PROTO_Z3950,   CLASS_RECSYN,  VAL_SUMMARY,      {5,103,-1},
163      "Summary"},
164     {PROTO_Z3950,   CLASS_RECSYN,  VAL_GRS0,         {5,104,-1},
165      "GRS-0"},
166     {PROTO_Z3950,   CLASS_RECSYN,  VAL_GRS1,         {5,105,-1},
167      "GRS-1"},
168     {PROTO_Z3950,   CLASS_RECSYN,  VAL_EXTENDED,     {5,106,-1},
169      "Extended"},
170     {PROTO_Z3950,   CLASS_RECSYN,  VAL_FRAGMENT,     {5,107,-1},
171      "Fragment"},
172     {PROTO_Z3950,   CLASS_RECSYN,  VAL_PDF,          {5,109,1,-1},
173      "pdf"},
174     {PROTO_Z3950,   CLASS_RECSYN,  VAL_POSTSCRIPT,   {5,109,2,-1},
175      "postscript"},
176     {PROTO_Z3950,   CLASS_RECSYN,  VAL_HTML,         {5,109,3,-1},
177      "html"},
178     {PROTO_Z3950,   CLASS_RECSYN,  VAL_TIFF,         {5,109,4,-1},
179      "tiff"},
180     {PROTO_Z3950,   CLASS_RECSYN,  VAL_GIF,          {5,109,5,-1},
181      "gif"},
182     {PROTO_Z3950,   CLASS_RECSYN,  VAL_JPEG,         {5,109,6,-1},
183      "jpeg"},
184     {PROTO_Z3950,   CLASS_RECSYN,  VAL_PNG,          {5,109,7,-1},
185      "png"},
186     {PROTO_Z3950,   CLASS_RECSYN,  VAL_MPEG,         {5,109,8,-1},
187      "mpeg"},
188     {PROTO_Z3950,   CLASS_RECSYN,  VAL_SGML,         {5,109,9,-1},
189      "sgml"},
190
191     {PROTO_Z3950,   CLASS_RECSYN,  VAL_TIFFB,        {5,110,1,-1},
192      "tiff-b"},
193     {PROTO_Z3950,   CLASS_RECSYN,  VAL_WAV,          {5,110,2,-1},
194      "wav"},
195
196     {PROTO_Z3950,   CLASS_RECSYN,  VAL_SQLRS,        {5,111,-1},
197      "SQL-RS"},
198     {PROTO_Z3950,   CLASS_RECSYN,  VAL_SOIF,         {5,1000,81,2,-1},
199      "SOIF" },
200     {PROTO_Z3950,   CLASS_RECSYN,  VAL_TEXT_XML,     {5,109,10,-1},
201      "text-XML" },
202     {PROTO_Z3950,   CLASS_RECSYN,  VAL_TEXT_XML,     {5,109,10,-1},
203      "XML" },
204     {PROTO_Z3950,   CLASS_RECSYN,  VAL_APPLICATION_XML, {5,109,11,-1},
205      "application-XML" },
206     {PROTO_Z3950,   CLASS_RESFORM, VAL_RESOURCE1,    {7,1,-1},
207      "Resource-1"},
208     {PROTO_Z3950,   CLASS_RESFORM, VAL_RESOURCE2,    {7,2,-1},
209      "Resource-2"},
210     {PROTO_Z3950,   CLASS_RESFORM, VAL_UNIVERSE_REPORT, {7,1000,81,1,-1},
211      "UNIverse-Resource-Report"},
212
213     {PROTO_Z3950,   CLASS_ACCFORM, VAL_PROMPT1,      {8,1,-1},
214      "Prompt-1"},
215     {PROTO_Z3950,   CLASS_ACCFORM, VAL_DES1,         {8,2,-1},
216      "Des-1"},
217     {PROTO_Z3950,   CLASS_ACCFORM, VAL_KRB1,         {8,3,-1},
218      "Krb-1"},
219     {PROTO_Z3950,   CLASS_EXTSERV, VAL_PRESSET,      {9,1,-1},
220      "Pers. set"},
221     {PROTO_Z3950,   CLASS_EXTSERV, VAL_PQUERY,       {9,2,-1},
222      "Pers. query"},
223     {PROTO_Z3950,   CLASS_EXTSERV, VAL_PCQUERY,      {9,3,-1},
224      "Per'd query"},
225     {PROTO_Z3950,   CLASS_EXTSERV, VAL_ITEMORDER,    {9,4,-1},
226      "Item order"},
227     {PROTO_Z3950,   CLASS_EXTSERV, VAL_DBUPDATE0,    {9,5,-1},
228      "DB. Update (first version)"},
229     {PROTO_Z3950,   CLASS_EXTSERV, VAL_DBUPDATE1,    {9,5,1,-1},
230      "DB. Update (second version)"},
231     {PROTO_Z3950,   CLASS_EXTSERV, VAL_DBUPDATE,     {9,5,1,1,-1},
232      "DB. Update"},
233     {PROTO_Z3950,   CLASS_EXTSERV, VAL_EXPORTSPEC,   {9,6,-1},
234      "exp. spec."},
235     {PROTO_Z3950,   CLASS_EXTSERV, VAL_EXPORTINV,    {9,7,-1},
236      "exp. inv."},
237     {PROTO_Z3950,   CLASS_EXTSERV, VAL_ADMINSERVICE, {9,1000,81,1,-1},
238      "Admin"},
239     {PROTO_Z3950,   CLASS_USERINFO,VAL_SEARCHRES1,   {10,1,-1},
240      "searchResult-1"},
241     {PROTO_Z3950,   CLASS_USERINFO,VAL_CHARLANG,     {10,2,-1},
242      "CharSetandLanguageNegotiation"},
243     {PROTO_Z3950,   CLASS_USERINFO,VAL_USERINFO1,    {10,3,-1},
244      "UserInfo-1"},
245     {PROTO_Z3950,   CLASS_USERINFO,VAL_MULTISRCH1,   {10,4,-1},
246      "MultipleSearchTerms-1"},
247     {PROTO_Z3950,   CLASS_USERINFO,VAL_MULTISRCH2,   {10,5,-1},
248      "MultipleSearchTerms-2"},
249     {PROTO_Z3950,   CLASS_USERINFO,VAL_DATETIME,     {10,6,-1},
250      "DateTime"},
251     {PROTO_Z3950,   CLASS_USERINFO,VAL_PROXY,        {10,1000,81,1,-1},
252      "Proxy" },
253     {PROTO_Z3950,   CLASS_USERINFO,VAL_COOKIE,       {10,1000,81,2,-1},
254      "Cookie" },
255     {PROTO_Z3950,   CLASS_USERINFO,VAL_CLIENT_IP,    {10,1000,81,3,-1},
256      "Client-IP" },
257     {PROTO_Z3950,   CLASS_ELEMSPEC,VAL_ESPEC1,       {11,1,-1},
258      "Espec-1"},
259     {PROTO_Z3950,   CLASS_VARSET,  VAL_VAR1,         {12,1,-1},
260      "Variant-1"},
261     {PROTO_Z3950,   CLASS_SCHEMA,  VAL_WAIS,         {13,1,-1},
262      "WAIS-schema"},
263     {PROTO_Z3950,   CLASS_SCHEMA,  VAL_GILS,         {13,2,-1},
264      "GILS-schema"},
265     {PROTO_Z3950,   CLASS_SCHEMA,  VAL_COLLECT1,     {13,3,-1},
266      "Collections-schema"},
267     {PROTO_Z3950,   CLASS_SCHEMA,  VAL_GEO,          {13,4,-1},
268      "Geo-schema"},
269     {PROTO_Z3950,   CLASS_SCHEMA,  VAL_CIMI1,        {13,5,-1},
270      "CIMI-schema"},
271     {PROTO_Z3950,   CLASS_SCHEMA,  VAL_UPDATEES,     {13,6,-1},
272      "Update ES"},
273     {PROTO_Z3950,   CLASS_SCHEMA,  VAL_HOLDINGS,     {13,7,-1},
274      "Holdings"},
275     {PROTO_Z3950,   CLASS_SCHEMA,  VAL_ZTHES,        {13,8,-1},
276      "Zthes"},
277     {PROTO_Z3950,   CLASS_SCHEMA,  VAL_THESAURUS,    {13,1000,81,1,-1},
278      "thesaurus-schema"},
279     {PROTO_Z3950,   CLASS_SCHEMA,  VAL_EXPLAIN,      {13,1000,81,2,-1},
280      "Explain-schema"},
281     {PROTO_Z3950,   CLASS_TAGSET,  VAL_SETM,         {14,1,-1},
282      "TagsetM"},
283     {PROTO_Z3950,   CLASS_TAGSET,  VAL_SETG,         {14,2,-1},
284      "TagsetG"},
285     {PROTO_Z3950,   CLASS_TAGSET,  VAL_STAS,         {14,3,-1},
286      "STAS-tagset"},
287     {PROTO_Z3950,   CLASS_TAGSET,  VAL_GILS,         {14,4,-1},
288      "GILS-tagset"},
289     {PROTO_Z3950,   CLASS_TAGSET,  VAL_COLLECT1,     {14,5,-1},
290      "Collections-tagset"},
291     {PROTO_Z3950,   CLASS_TAGSET,  VAL_CIMI1,        {14,6,-1},
292      "CIMI-tagset"},
293     {PROTO_Z3950,   CLASS_TAGSET,  VAL_THESAURUS,    {14,1000,81,1,-1},
294      "thesaurus-tagset"},       /* What is this Satan-spawn doing here? */
295     {PROTO_Z3950,   CLASS_TAGSET,  VAL_EXPLAIN,      {14,1000,81,2,-1},
296      "Explain-tagset"},
297     {PROTO_Z3950,   CLASS_TAGSET,  VAL_ZTHES,        {14,8,-1},
298      "Zthes-tagset"},
299     {PROTO_Z3950,   CLASS_NEGOT,   VAL_CHARNEG3,     {15,3,-1},
300      "CharSetandLanguageNegotiation-3"},
301     {PROTO_Z3950,   CLASS_NEGOT,   VAL_ID_CHARSET,   {15,1000,81,1,-1},
302      "ID-Charset" },
303     {PROTO_Z3950,   CLASS_USERINFO,VAL_CQL,          {16, 2, -1},
304      "CQL"},
305     {PROTO_GENERAL, CLASS_GENERAL, VAL_UCS2,    {1,0,10646,1,0,2,-1},
306      "UCS-2"},
307     {PROTO_GENERAL, CLASS_GENERAL, VAL_UCS4,    {1,0,10646,1,0,4,-1},
308      "UCS-4"},
309     {PROTO_GENERAL, CLASS_GENERAL, VAL_UTF16,   {1,0,10646,1,0,5,-1},
310      "UTF-16"},
311     {PROTO_GENERAL, CLASS_GENERAL, VAL_UTF8,    {1,0,10646,1,0,8,-1},
312      "UTF-8"},
313     {PROTO_Z3950,   CLASS_USERINFO,VAL_OCLCUI,  {10, 1000, 17, 1, -1},
314      "OCLC-userInfo"},
315     {PROTO_NOP,     CLASS_NOP,     VAL_NOP,       {-1},        0          }
316 };
317
318 /* OID utilities */
319
320 void oid_oidcpy(int *t, int *s)
321 {
322     while ((*(t++) = *(s++)) > -1);
323 }
324
325 void oid_oidcat(int *t, int *s)
326 {
327     while (*t > -1)
328         t++;
329     while ((*(t++) = *(s++)) > -1);
330 }
331
332 int oid_oidcmp(int *o1, int *o2)
333 {
334     while (*o1 == *o2 && *o1 > -1)
335     {
336         o1++;
337         o2++;
338     }
339     if (*o1 == *o2)
340         return 0;
341     else if (*o1 > *o2)
342         return 1;
343     else
344         return -1;
345 }
346
347 int oid_oidlen(int *o)
348 {
349     int len = 0;
350
351     while (*(o++) >= 0)
352         len++;
353     return len;
354 }
355
356
357 static int match_prefix(int *look, int *prefix)
358 {
359     int len;
360
361     for (len = 0; *look == *prefix; look++, prefix++, len++);
362     if (*prefix < 0) /* did we reach the end of the prefix? */
363         return len;
364     return 0;
365 }
366
367 void oid_transfer (struct oident *oidentp)
368 {
369     while (*oidentp->oidsuffix >= 0)
370     {
371         oid_addent (oidentp->oidsuffix, oidentp->proto,
372                     oidentp->oclass,
373                     oidentp->desc, oidentp->value);
374         oidentp++;
375     }
376 }
377
378 void oid_init (void)
379 {
380     if (oid_init_flag == 0)
381     {
382         /* oid_transfer is thread safe, so there's nothing wrong in having
383            two threads calling it simultaniously. On the other hand
384            no thread may exit oid_init before all OID's bave been
385            transferred - which is why checked is set after oid_transfer... 
386         */
387         nmem_mutex_create (&oid_mutex);
388         nmem_mutex_enter (oid_mutex);
389         if (!oid_nmem)
390             oid_nmem = nmem_create ();
391         nmem_mutex_leave (oid_mutex);
392         oid_transfer (standard_oids);
393         oid_init_flag = 1;
394     }
395 }
396
397 void oid_exit (void)
398 {
399     if (oid_init_flag)
400     {
401         oid_init_flag = 0;
402         nmem_mutex_destroy (&oid_mutex);
403         nmem_destroy (oid_nmem);
404         oid_nmem = 0;
405     }
406 }
407
408 static struct oident *oid_getentbyoid_x(int *o)
409 {
410     enum oid_proto proto;
411     int prelen;
412     struct oident_list *ol;
413     
414     /* determine protocol type */
415     if ((prelen = match_prefix(o, z3950_prefix)) != 0)
416         proto = PROTO_Z3950;
417     else if ((prelen = match_prefix(o, sr_prefix)) != 0)
418         proto = PROTO_SR;
419     else
420         proto = PROTO_GENERAL;
421     for (ol = oident_table; ol; ol = ol->next)
422     {
423         struct oident *p = &ol->oident;
424         if (p->proto == proto && !oid_oidcmp(o + prelen, p->oidsuffix))
425             return p;
426         if (p->proto == PROTO_GENERAL && !oid_oidcmp (o, p->oidsuffix))
427             return p;
428     }
429     return 0;
430 }
431
432 /*
433  * To query, fill out proto, class, and value of the ent parameter.
434  */
435 int *oid_ent_to_oid(struct oident *ent, int *ret)
436 {
437     struct oident_list *ol;
438     
439     oid_init ();
440     for (ol = oident_table; ol; ol = ol->next)
441     {
442         struct oident *p = &ol->oident;
443         if (ent->value == p->value &&
444             (p->proto == PROTO_GENERAL || (ent->proto == p->proto &&  
445             (ent->oclass == p->oclass || ent->oclass == CLASS_GENERAL))))
446         {
447             if (p->proto == PROTO_Z3950)
448                 oid_oidcpy(ret, z3950_prefix);
449             else if (p->proto == PROTO_SR)
450                 oid_oidcpy(ret, sr_prefix);
451             else
452                 ret[0] = -1;
453             oid_oidcat(ret, p->oidsuffix);
454             ent->desc = p->desc;
455             return ret;
456         }
457     }
458     ret[0] = -1;
459     return 0;
460 }
461
462 /*
463  * To query, fill out proto, class, and value of the ent parameter.
464  */
465 int *oid_getoidbyent(struct oident *ent)
466 {
467     static int ret[OID_SIZE];
468
469     return oid_ent_to_oid (ent, ret);
470 }
471
472 struct oident *oid_addent (int *oid, enum oid_proto proto,
473                            enum oid_class oclass,
474                            const char *desc, int value)
475 {
476     struct oident *oident = 0;
477
478     nmem_mutex_enter (oid_mutex);
479     if (!oident)
480     {
481         char desc_str[200];
482         struct oident_list *oident_list;
483         oident_list = (struct oident_list *)
484             nmem_malloc (oid_nmem, sizeof(*oident_list));
485         oident = &oident_list->oident;
486         oident->proto = proto;
487         oident->oclass = oclass;
488
489         if (!desc)
490         {
491             int i;
492
493             sprintf (desc_str, "%d", *oid);
494             for (i = 1; i < 12 && oid[i] >= 0; i++)
495                 sprintf (desc_str+strlen(desc_str), ".%d", oid[i]);
496             desc = desc_str;
497         }
498         oident->desc = nmem_strdup (oid_nmem, desc);
499         if (value == VAL_DYNAMIC)
500             oident->value = (enum oid_value) (++oid_value_dynamic);
501         else
502             oident->value = (enum oid_value) value;
503         oid_oidcpy (oident->oidsuffix, oid);
504         oident_list->next = oident_table;
505         oident_table = oident_list;
506     }
507     nmem_mutex_leave (oid_mutex);
508     return oident;
509 }
510
511 struct oident *oid_getentbyoid(int *oid)
512 {
513     struct oident *oident;
514
515     if (!oid)
516         return 0;
517     oid_init ();
518     oident = oid_getentbyoid_x (oid);
519     if (!oident)
520         oident = oid_addent (oid, PROTO_GENERAL, CLASS_GENERAL,
521                              NULL, VAL_DYNAMIC);
522     return oident;
523 }
524
525 static oid_value oid_getval_raw(const char *name)
526 {
527     int val = 0, i = 0, oid[OID_SIZE];
528     struct oident *oident;
529     
530     while (isdigit (*name))
531     {
532         val = val*10 + (*name - '0');
533         name++;
534         if (*name == '.')
535         {
536             if (i < OID_SIZE-1)
537                 oid[i++] = val;
538             val = 0;
539             name++;
540         }
541     }
542     oid[i] = val;
543     oid[i+1] = -1;
544     oident = oid_getentbyoid_x (oid);
545     if (!oident)
546         oident = oid_addent (oid, PROTO_GENERAL, CLASS_GENERAL, NULL,
547                          VAL_DYNAMIC);
548     return oident->value;
549 }
550
551 oid_value oid_getvalbyname(const char *name)
552 {
553     struct oident_list *ol;
554
555     oid_init ();
556     if (isdigit (*name))
557         return oid_getval_raw (name);
558     for (ol = oident_table; ol; ol = ol->next)
559         if (!yaz_matchstr(ol->oident.desc, name))
560         {
561             return ol->oident.value;
562         }
563     return VAL_NONE;
564 }
565
566 void oid_setprivateoids(oident *list)
567 {
568     oid_transfer (list);
569 }
570
571 void oid_trav (void (*func)(struct oident *oidinfo, void *vp), void *vp)
572 {
573     struct oident_list *ol;
574
575     oid_init ();
576     for (ol = oident_table; ol; ol = ol->next)
577         (*func)(&ol->oident, vp);
578 }
579
580 int *oid_name_to_oid(oid_class oclass, const char *name, int *oid) {
581     struct oident ent;
582
583     /* Translate syntax to oid_val */
584     oid_value value = oid_getvalbyname(name);
585
586     /* Build it into an oident */
587     ent.proto = PROTO_Z3950;
588     ent.oclass = oclass;
589     ent.value = value;
590
591     /* Translate to an array of int */
592     return oid_ent_to_oid(&ent, oid);
593 }
594
595 char *oid_to_dotstring(const int *oid, char *oidbuf) {
596     char tmpbuf[20];
597     int i;
598
599     oidbuf[0] = '\0';
600     for (i = 0; oid[i] != -1; i++) {
601         sprintf(tmpbuf, "%d", oid[i]);
602         if (i > 0) strcat(oidbuf, ".");
603         strcat(oidbuf, tmpbuf);
604     }
605
606     return oidbuf;
607 }
608
609 char *oid_name_to_dotstring(oid_class oclass, const char *name, char *oidbuf) {
610     int oid[OID_SIZE];
611
612     (void) oid_name_to_oid(oclass, name, oid);
613     return oid_to_dotstring(oid, oidbuf);
614 }
615