New nmem utility, nmem_transfer, that transfer blocks from one
[yaz-moved-to-github.git] / util / oid.c
1 /*
2  * Copyright (c) 1995-1998, Index Data
3  * See the file LICENSE for details.
4  * Sebastian Hammer, Adam Dickmeiss
5  *
6  * $Log: oid.c,v $
7  * Revision 1.30  1998-10-18 07:48:56  adam
8  * Fixed oid_getentbyoid so that it returns NULL when parsed oid is NULL.
9  *
10  * Revision 1.29  1998/10/14 13:32:35  adam
11  * Added include of string.h.
12  *
13  * Revision 1.28  1998/10/13 16:01:53  adam
14  * Implemented support for dynamic object identifiers.
15  * Function oid_getvalbyname now accepts raw OID's as well as traditional
16  * names.
17  *
18  * Revision 1.27  1998/05/18 10:10:02  adam
19  * Added Explain-schema and Explain-tagset to OID database.
20  *
21  * Revision 1.26  1998/03/20 14:46:06  adam
22  * Added UNIverse Resource Reports.
23  *
24  * Revision 1.25  1998/02/10 15:32:03  adam
25  * Added new Object Identifiers.
26  *
27  * Revision 1.24  1997/09/29 13:19:00  adam
28  * Added function, oid_ent_to_oid, to replace the function
29  * oid_getoidbyent, which is not thread safe.
30  *
31  * Revision 1.23  1997/09/09 10:10:19  adam
32  * Another MSV5.0 port. Changed projects to include proper
33  * library/include paths.
34  * Server starts server in test-mode when no options are given.
35  *
36  * Revision 1.22  1997/08/29 13:34:58  quinn
37  * Added thesaurus oids
38  *
39  * Revision 1.21  1997/08/19 08:46:05  quinn
40  * Added Thesaurus OID
41  *
42  * Revision 1.20  1997/07/28 12:34:43  adam
43  * Added new OID entries (RVDM).
44  *
45  * Revision 1.19  1997/05/02 08:39:41  quinn
46  * Support for private OID table added. Thanks to Ronald van der Meer
47  *
48  * Revision 1.18  1997/04/30 08:52:12  quinn
49  * Null
50  *
51  * Revision 1.17  1996/10/10  12:35:23  quinn
52  * Added Update extended service.
53  *
54  * Revision 1.16  1996/10/09  15:55:02  quinn
55  * Added SearchInfoReport
56  *
57  * Revision 1.15  1996/10/07  15:29:43  quinn
58  * Added SOIF support
59  *
60  * Revision 1.14  1996/02/20  17:58:28  adam
61  * Added const to oid_getvalbyname.
62  *
63  * Revision 1.13  1996/02/20  16:37:33  quinn
64  * Using yaz_matchstr in oid_getvalbyname
65  *
66  * Revision 1.12  1996/01/02  08:57:53  quinn
67  * Changed enums in the ASN.1 .h files to #defines. Changed oident.class to oclass
68  *
69  * Revision 1.11  1995/12/13  16:03:35  quinn
70  * *** empty log message ***
71  *
72  * Revision 1.10  1995/11/28  09:30:44  quinn
73  * Work.
74  *
75  * Revision 1.9  1995/11/13  09:27:53  quinn
76  * Fiddling with the variant stuff.
77  *
78  * Revision 1.8  1995/10/12  10:34:56  quinn
79  * Added Espec-1.
80  *
81  * Revision 1.7  1995/10/10  16:27:12  quinn
82  * *** empty log message ***
83  *
84  * Revision 1.6  1995/09/29  17:12:35  quinn
85  * Smallish
86  *
87  * Revision 1.5  1995/09/29  17:01:51  quinn
88  * More Windows work
89  *
90  * Revision 1.4  1995/09/27  15:03:03  quinn
91  * Modified function heads & prototypes.
92  *
93  * Revision 1.3  1995/09/12  11:32:06  quinn
94  * Added a looker-upper by name.
95  *
96  * Revision 1.2  1995/08/21  09:11:16  quinn
97  * Smallish fixes to suppport new formats.
98  *
99  * Revision 1.1  1995/05/29  08:17:13  quinn
100  * iMoved oid to util to support comstack.
101  *
102  * Revision 1.5  1995/05/22  11:30:16  quinn
103  * Adding Z39.50-1992 stuff to proto.c. Adding zget.c
104  *
105  * Revision 1.4  1995/05/16  08:50:22  quinn
106  * License, documentation, and memory fixes
107  *
108  * Revision 1.3  1995/04/11  11:52:02  quinn
109  * Fixed possible buf in proto.c
110  *
111  * Revision 1.2  1995/03/29  15:39:38  quinn
112  * Adding some resource control elements, and a null-check to getentbyoid
113  *
114  * Revision 1.1  1995/03/27  08:32:12  quinn
115  * Added OID database
116  *
117  *
118  */
119
120 /*
121  * More or less protocol-transparent OID database.
122  * We could (and should?) extend this so that the user app can add new
123  * entries to the list at initialization.
124  */
125
126 #include <stdlib.h>
127 #include <string.h>
128 #include <ctype.h>
129
130 #include <oid.h>
131 #include <yaz-util.h>
132
133 static int z3950_prefix[] = { 1, 2, 840, 10003, -1 };
134 static int sr_prefix[]    = { 1, 0, 10163, -1 };
135
136 struct oident_list {
137     struct oident oident;
138     struct oident_list *next;
139 };
140
141 static struct oident_list *oident_table = NULL;
142 static int oid_value_dynamic = VAL_DYNAMIC;
143
144 /*
145  * OID database
146  */
147 static oident oids[] =
148 {
149     /* General definitions */
150     {PROTO_GENERAL, CLASS_TRANSYN, VAL_BER,       {2,1,1,-1},  "BER"         },
151     {PROTO_GENERAL, CLASS_TRANSYN, VAL_ISO2709,   {1,0,2709,1,1,-1},"ISO2709"},
152
153     /* Z39.50v3 definitions */
154     {PROTO_Z3950,   CLASS_ABSYN,   VAL_APDU,      {2,1,-1},    "Z-APDU"      },
155
156     {PROTO_Z3950,   CLASS_APPCTX,  VAL_BASIC_CTX, {1,1,-1},    "Z-BASIC"     },
157
158     {PROTO_Z3950,   CLASS_ATTSET,  VAL_BIB1,      {3,1,-1},    "Bib-1"       },
159     {PROTO_Z3950,   CLASS_ATTSET,  VAL_EXP1,      {3,2,-1},    "Exp-1"       },
160     {PROTO_Z3950,   CLASS_ATTSET,  VAL_EXT1,      {3,3,-1},    "Ext-1"       },
161     {PROTO_Z3950,   CLASS_ATTSET,  VAL_CCL1,      {3,4,-1},    "CCL-1"       },
162     {PROTO_Z3950,   CLASS_ATTSET,  VAL_GILS,      {3,5,-1},    "GILS-attset" },
163     {PROTO_Z3950,   CLASS_ATTSET,  VAL_STAS,      {3,6,-1},    "STAS-attset" },
164     {PROTO_Z3950,   CLASS_ATTSET,  VAL_COLLECT1,  {3,7,-1},    "Collections-attset"},
165     {PROTO_Z3950,   CLASS_ATTSET,  VAL_CIMI1,     {3,8,-1},    "CIMI-attset"},
166     {PROTO_Z3950,   CLASS_ATTSET,  VAL_GEO,       {3,9,-1},    "Geo-attset"},
167     {PROTO_Z3950,   CLASS_ATTSET,  VAL_THESAURUS, {3,1000,81,1,-1},"Thesaurus-attset"},
168     {PROTO_Z3950,   CLASS_DIAGSET, VAL_BIB1,      {4,1,-1},    "Bib-1"       },
169     {PROTO_Z3950,   CLASS_DIAGSET, VAL_DIAG1,     {4,2,-1},    "Diag-1"      },
170
171     {PROTO_Z3950,   CLASS_RECSYN,  VAL_UNIMARC,   {5,1,-1},    "Unimarc"     },
172     {PROTO_Z3950,   CLASS_RECSYN,  VAL_INTERMARC, {5,2,-1},    "Intermarc"   },
173     {PROTO_Z3950,   CLASS_RECSYN,  VAL_CCF,       {5,3,-1},    "CCF"         },
174     {PROTO_Z3950,   CLASS_RECSYN,  VAL_USMARC,    {5,10,-1},   "USmarc"      },
175     {PROTO_Z3950,   CLASS_RECSYN,  VAL_UKMARC,    {5,11,-1},   "UKmarc"      },
176     {PROTO_Z3950,   CLASS_RECSYN,  VAL_NORMARC,   {5,12,-1},   "Normarc"     },
177     {PROTO_Z3950,   CLASS_RECSYN,  VAL_LIBRISMARC,{5,13,-1},   "Librismarc"  },
178     {PROTO_Z3950,   CLASS_RECSYN,  VAL_DANMARC,   {5,14,-1},   "Danmarc"     },
179     {PROTO_Z3950,   CLASS_RECSYN,  VAL_FINMARC,   {5,15,-1},   "Finmarc"     },
180     {PROTO_Z3950,   CLASS_RECSYN,  VAL_MAB,       {5,16,-1},   "MAB"         },
181     {PROTO_Z3950,   CLASS_RECSYN,  VAL_CANMARC,   {5,17,-1},   "Canmarc"     },
182     {PROTO_Z3950,   CLASS_RECSYN,  VAL_SBN,       {5,18,-1},   "SBN"         },
183     {PROTO_Z3950,   CLASS_RECSYN,  VAL_PICAMARC,  {5,19,-1},   "Picamarc"    },
184     {PROTO_Z3950,   CLASS_RECSYN,  VAL_AUSMARC,   {5,20,-1},   "Ausmarc"     },
185     {PROTO_Z3950,   CLASS_RECSYN,  VAL_IBERMARC,  {5,21,-1},   "Ibermarc"    },
186     {PROTO_Z3950,   CLASS_RECSYN,  VAL_CATMARC,   {5,22,-1},   "Carmarc"     },
187     {PROTO_Z3950,   CLASS_RECSYN,  VAL_MALMARC,   {5,23,-1},   "Malmarc"     },
188     {PROTO_Z3950,   CLASS_RECSYN,  VAL_EXPLAIN,   {5,100,-1},  "Explain"     },
189     {PROTO_Z3950,   CLASS_RECSYN,  VAL_SUTRS,     {5,101,-1},  "SUTRS"       },
190     {PROTO_Z3950,   CLASS_RECSYN,  VAL_OPAC,      {5,102,-1},  "OPAC"        },
191     {PROTO_Z3950,   CLASS_RECSYN,  VAL_SUMMARY,   {5,103,-1},  "Summary"     },
192     {PROTO_Z3950,   CLASS_RECSYN,  VAL_GRS0,      {5,104,-1},  "GRS-0"       },
193     {PROTO_Z3950,   CLASS_RECSYN,  VAL_GRS1,      {5,105,-1},  "GRS-1"       },
194     {PROTO_Z3950,   CLASS_RECSYN,  VAL_EXTENDED,  {5,106,-1},  "Extended"    },
195     {PROTO_Z3950,   CLASS_RECSYN,  VAL_FRAGMENT,  {5,107,-1},  "Fragment"    },
196
197     {PROTO_Z3950,   CLASS_RECSYN,  VAL_PDF,       {5,109,1,-1},"pdf"         },
198     {PROTO_Z3950,   CLASS_RECSYN,  VAL_POSTSCRIPT,{5,109,2,-1},"postscript"  },
199     {PROTO_Z3950,   CLASS_RECSYN,  VAL_HTML,      {5,109,3,-1},"html"        },
200     {PROTO_Z3950,   CLASS_RECSYN,  VAL_TIFF,      {5,109,4,-1},"tiff"        },
201     {PROTO_Z3950,   CLASS_RECSYN,  VAL_GIF,       {5,109,5,-1},"gif"         },
202     {PROTO_Z3950,   CLASS_RECSYN,  VAL_JPEG,      {5,109,6,-1},"jpeg"        },
203     {PROTO_Z3950,   CLASS_RECSYN,  VAL_PNG,       {5,109,7,-1},"png"         },
204     {PROTO_Z3950,   CLASS_RECSYN,  VAL_MPEG,      {5,109,8,-1},"mpeg"        },
205     {PROTO_Z3950,   CLASS_RECSYN,  VAL_SGML,      {5,109,9,-1},"sgml"        },
206
207     {PROTO_Z3950,   CLASS_RECSYN,  VAL_TIFFB,     {5,110,1,-1},"tiff-b"      },
208     {PROTO_Z3950,   CLASS_RECSYN,  VAL_WAV,       {5,110,2,-1},"wav"         },
209
210     {PROTO_Z3950,   CLASS_RECSYN,  VAL_SQLRS,     {5,111,-1},  "SQL-RS"      },
211 #if 0
212     {PROTO_Z3950,   CLASS_RECSYN,  VAL_ID_SGML,   {5,1000,81,1,-1},"ID-SGML" },
213 #endif
214     {PROTO_Z3950,   CLASS_RECSYN,  VAL_SOIF,      {5,1000,81,2,-1},"SOIF" },
215
216     {PROTO_Z3950,   CLASS_RESFORM, VAL_RESOURCE1, {7,1,-1},    "Resource-1"  },
217     {PROTO_Z3950,   CLASS_RESFORM, VAL_RESOURCE2, {7,2,-1},    "Resource-2"  },
218     {PROTO_Z3950,   CLASS_RESFORM, VAL_UNIVERSE_REPORT,  {7,1000,81,1,-1}, "UNIverse-Resource-Report"},
219
220     {PROTO_Z3950,   CLASS_ACCFORM, VAL_PROMPT1,   {8,1,-1},    "Prompt-1"    },
221     {PROTO_Z3950,   CLASS_ACCFORM, VAL_DES1,      {8,2,-1},    "Des-1"       },
222     {PROTO_Z3950,   CLASS_ACCFORM, VAL_KRB1,      {8,3,-1},    "Krb-1"       },
223     
224     {PROTO_Z3950,   CLASS_EXTSERV, VAL_PRESSET,   {9,1,-1},    "Pers. set"   },
225     {PROTO_Z3950,   CLASS_EXTSERV, VAL_PQUERY,    {9,2,-1},    "Pers. query" },
226     {PROTO_Z3950,   CLASS_EXTSERV, VAL_PCQUERY,   {9,3,-1},    "Per'd query" },
227     {PROTO_Z3950,   CLASS_EXTSERV, VAL_ITEMORDER, {9,4,-1},    "Item order"  },
228     {PROTO_Z3950,   CLASS_EXTSERV, VAL_DBUPDATE,  {9,5,-1},    "DB. Update"  },
229     {PROTO_Z3950,   CLASS_EXTSERV, VAL_EXPORTSPEC,{9,6,-1},    "exp. spec."  },
230     {PROTO_Z3950,   CLASS_EXTSERV, VAL_EXPORTINV, {9,7,-1},    "exp. inv."   },
231
232     {PROTO_Z3950,   CLASS_USERINFO,VAL_SEARCHRES1,{10,1,-1},   "searchResult-1"},
233     {PROTO_Z3950,   CLASS_USERINFO,VAL_CHARLANG,  {10,2,-1},   "CharSetandLanguageNegotiation"},
234     {PROTO_Z3950,   CLASS_USERINFO,VAL_USERINFO1, {10,3,-1},   "UserInfo-1"},
235     {PROTO_Z3950,   CLASS_USERINFO,VAL_MULTISRCH1,{10,4,-1},   "MultipleSearchTerms-1"},
236     {PROTO_Z3950,   CLASS_USERINFO,VAL_MULTISRCH2,{10,5,-1},   "MultipleSearchTerms-2"},
237     {PROTO_Z3950,   CLASS_USERINFO,VAL_DATETIME,  {10,6,-1},   "DateTime"},
238
239     {PROTO_Z3950,   CLASS_ELEMSPEC,VAL_ESPEC1,    {11,1,-1},   "Espec-1"     },
240
241     {PROTO_Z3950,   CLASS_VARSET,  VAL_VAR1,      {12,1,-1},   "Variant-1"   },
242
243     {PROTO_Z3950,   CLASS_SCHEMA,  VAL_WAIS,      {13,1,-1},   "WAIS-schema" },
244     {PROTO_Z3950,   CLASS_SCHEMA,  VAL_GILS,      {13,2,-1},   "GILS-schema" },
245     {PROTO_Z3950,   CLASS_SCHEMA,  VAL_COLLECT1,  {13,3,-1},   "Collections-schema" },
246     {PROTO_Z3950,   CLASS_SCHEMA,  VAL_GEO,       {13,4,-1},   "Geo-schema" },
247     {PROTO_Z3950,   CLASS_SCHEMA,  VAL_CIMI1,     {13,5,-1},   "CIMI-schema" },
248     {PROTO_Z3950,   CLASS_SCHEMA,  VAL_UPDATEES,  {13,6,-1},   "Update ES" },
249     {PROTO_Z3950,   CLASS_SCHEMA,  VAL_THESAURUS, {13,1000,81,1,-1}, "thesaurus-schema"},
250     {PROTO_Z3950,   CLASS_SCHEMA,  VAL_EXPLAIN,   {13,1000,81,2,-1}, "Explain-schema"},
251     {PROTO_Z3950,   CLASS_TAGSET,  VAL_SETM,      {14,1,-1},   "TagsetM"     },
252     {PROTO_Z3950,   CLASS_TAGSET,  VAL_SETG,      {14,2,-1},   "TagsetG"     },
253     {PROTO_Z3950,   CLASS_TAGSET,  VAL_STAS,      {14,3,-1},   "STAS-tagset" },
254     {PROTO_Z3950,   CLASS_TAGSET,  VAL_GILS,      {14,4,-1},   "GILS-tagset" },
255     {PROTO_Z3950,   CLASS_TAGSET,  VAL_COLLECT1,  {14,5,-1},   "Collections-tagset"},
256     {PROTO_Z3950,   CLASS_TAGSET,  VAL_CIMI1,     {14,6,-1},   "CIMI-tagset" },
257     {PROTO_Z3950,   CLASS_TAGSET,  VAL_THESAURUS, {14,1000,81,1,-1}, "thesaurus-tagset"},
258     {PROTO_Z3950,   CLASS_TAGSET,  VAL_EXPLAIN,   {14,1000,81,2,-1}, "Explain-tagset"},
259     
260
261     /* SR definitions. Note that some of them aren't defined by the
262         standard (yet), but are borrowed from Z3950v3 */
263     {PROTO_SR,      CLASS_ABSYN,   VAL_APDU,      {2,1,-1},    "SR-APDU"     },
264
265     {PROTO_SR,      CLASS_APPCTX,  VAL_BASIC_CTX, {1,1,-1},    "SR-BASIC"    },
266
267     {PROTO_SR,      CLASS_ATTSET,  VAL_BIB1,      {3,1,-1},    "Bib-1"       },
268     {PROTO_SR,      CLASS_ATTSET,  VAL_EXP1,      {3,2,-1},    "Exp-1"       },
269     {PROTO_SR,      CLASS_ATTSET,  VAL_EXT1,      {3,3,-1},    "Ext-1"       },
270     {PROTO_SR,      CLASS_ATTSET,  VAL_CCL1,      {3,4,-1},    "CCL-1"       },
271     {PROTO_SR,      CLASS_ATTSET,  VAL_GILS,      {3,5,-1},    "GILS"        },
272     {PROTO_SR,      CLASS_ATTSET,  VAL_STAS,      {3,6,-1},    "STAS",       },
273     {PROTO_SR,      CLASS_ATTSET,  VAL_COLLECT1,  {3,7,-1},    "Collections-attset"},
274     {PROTO_SR,      CLASS_ATTSET,  VAL_CIMI1,     {3,8,-1},    "CIMI-attset"},
275     {PROTO_SR,      CLASS_ATTSET,  VAL_GEO,       {3,9,-1},    "Geo-attset"},
276
277     {PROTO_SR,      CLASS_DIAGSET, VAL_BIB1,      {4,1,-1},    "Bib-1"       },
278     {PROTO_SR,      CLASS_DIAGSET, VAL_DIAG1,     {4,2,-1},    "Diag-1"      },
279
280     {PROTO_SR,      CLASS_RECSYN,  VAL_UNIMARC,   {5,1,-1},    "Unimarc"     },
281     {PROTO_SR,      CLASS_RECSYN,  VAL_INTERMARC, {5,2,-1},    "Intermarc"   },
282     {PROTO_SR,      CLASS_RECSYN,  VAL_CCF,       {5,3,-1},    "CCF"        },
283     {PROTO_SR,      CLASS_RECSYN,  VAL_USMARC,    {5,10,-1},   "USmarc"      },
284     {PROTO_SR,      CLASS_RECSYN,  VAL_UKMARC,    {5,11,-1},   "UKmarc"      },
285     {PROTO_SR,      CLASS_RECSYN,  VAL_NORMARC,   {5,12,-1},   "Normarc"     },
286     {PROTO_SR,      CLASS_RECSYN,  VAL_LIBRISMARC,{5,13,-1},   "Librismarc"  },
287     {PROTO_SR,      CLASS_RECSYN,  VAL_DANMARC,   {5,14,-1},   "Danmarc"     },
288     {PROTO_SR,      CLASS_RECSYN,  VAL_FINMARC,   {5,15,-1},   "Finmarc"     },
289     {PROTO_SR,      CLASS_RECSYN,  VAL_MAB,       {5,16,-1},   "MAB"         },
290     {PROTO_SR,      CLASS_RECSYN,  VAL_CANMARC,   {5,17,-1},   "Canmarc"     },
291     {PROTO_SR,      CLASS_RECSYN,  VAL_MAB,       {5,16,-1},   "MAB"         },
292     {PROTO_SR,      CLASS_RECSYN,  VAL_CANMARC,   {5,17,-1},   "Canmarc"     },
293     {PROTO_SR,      CLASS_RECSYN,  VAL_SBN,       {5,18,-1},   "SBN"         },
294     {PROTO_SR,      CLASS_RECSYN,  VAL_PICAMARC,  {5,19,-1},   "Picamarc"    },
295     {PROTO_SR,      CLASS_RECSYN,  VAL_AUSMARC,   {5,20,-1},   "Ausmarc"     },
296     {PROTO_SR,      CLASS_RECSYN,  VAL_IBERMARC,  {5,21,-1},   "Ibermarc"    },
297     {PROTO_SR,      CLASS_RECSYN,  VAL_CATMARC,   {5,22,-1},   "Catmarc"     },
298     {PROTO_SR,      CLASS_RECSYN,  VAL_MALMARC,   {5,23,-1},   "Malmarc"     },
299     {PROTO_SR,      CLASS_RECSYN,  VAL_EXPLAIN,   {5,100,-1},  "Explain"     },
300     {PROTO_SR,      CLASS_RECSYN,  VAL_SUTRS,     {5,101,-1},  "SUTRS"       },
301     {PROTO_SR,      CLASS_RECSYN,  VAL_OPAC,      {5,102,-1},  "OPAC"        },
302     {PROTO_SR,      CLASS_RECSYN,  VAL_SUMMARY,   {5,103,-1},  "Summary"     },
303     {PROTO_SR,      CLASS_RECSYN,  VAL_GRS0,      {5,104,-1},  "GRS-0"       },
304     {PROTO_SR,      CLASS_RECSYN,  VAL_GRS1,      {5,105,-1},  "GRS-1"       },
305     {PROTO_SR,      CLASS_RECSYN,  VAL_EXTENDED,  {5,106,-1},  "Extended"    },
306     {PROTO_SR,      CLASS_RECSYN,  VAL_FRAGMENT,  {5,107,-1},  "Fragment"    },
307
308     {PROTO_SR,      CLASS_RESFORM, VAL_RESOURCE1, {7,1,-1},    "Resource-1"  },
309     {PROTO_SR,      CLASS_RESFORM, VAL_RESOURCE2, {7,2,-1},    "Resource-2"  },
310
311     {PROTO_SR,      CLASS_ACCFORM, VAL_PROMPT1,   {8,1,-1},    "Prompt-1"    },
312     {PROTO_SR,      CLASS_ACCFORM, VAL_DES1,      {8,2,-1},    "Des-1"       },
313     {PROTO_SR,      CLASS_ACCFORM, VAL_KRB1,      {8,3,-1},    "Krb-1"       },
314
315     {PROTO_SR,      CLASS_EXTSERV, VAL_PRESSET,   {9,1,-1},    "Pers. set"   },
316     {PROTO_SR,      CLASS_EXTSERV, VAL_PQUERY,    {9,2,-1},    "Pers. query" },
317     {PROTO_SR,      CLASS_EXTSERV, VAL_PCQUERY,   {9,3,-1},    "Per'd query" },
318     {PROTO_SR,      CLASS_EXTSERV, VAL_ITEMORDER, {9,4,-1},    "Item order"  },
319     {PROTO_SR,      CLASS_EXTSERV, VAL_DBUPDATE,  {9,5,-1},    "DB. Update"  },
320     {PROTO_SR,      CLASS_EXTSERV, VAL_EXPORTSPEC,{9,6,-1},    "exp. spec."  },
321     {PROTO_SR,      CLASS_EXTSERV, VAL_EXPORTINV, {9,7,-1},    "exp. inv."   },
322
323     {PROTO_SR,      CLASS_ELEMSPEC,VAL_ESPEC1,    {11,1,-1},   "Espec-1"     },
324
325     {PROTO_SR,      CLASS_VARSET,  VAL_VAR1,      {12,1,-1},   "Variant-1"   },
326
327     {PROTO_SR,      CLASS_SCHEMA,  VAL_WAIS,      {13,1,-1},   "WAIS-schema" },
328     {PROTO_SR,      CLASS_SCHEMA,  VAL_GILS,      {13,2,-1},   "GILS-schema" },
329     {PROTO_SR,      CLASS_SCHEMA,  VAL_COLLECT1,  {13,3,-1},   "Collections-schema" },
330     {PROTO_SR,      CLASS_SCHEMA,  VAL_GEO,       {13,4,-1},   "Geo-schema" },
331     {PROTO_SR,      CLASS_SCHEMA,  VAL_CIMI1,     {13,5,-1},   "CIMI-schema" },
332
333     {PROTO_SR,      CLASS_TAGSET,  VAL_SETM,      {14,1,-1},   "TagsetM"     },
334     {PROTO_SR,      CLASS_TAGSET,  VAL_SETG,      {14,2,-1},   "TagsetG"     },
335
336     {PROTO_SR,      CLASS_TAGSET,  VAL_STAS,      {14,3,-1},   "STAS-tagset" },
337     {PROTO_SR,      CLASS_TAGSET,  VAL_GILS,      {14,4,-1},   "GILS-tagset" },
338     {PROTO_SR,      CLASS_TAGSET,  VAL_COLLECT1,  {14,5,-1},   "Collections-tagset"},
339     {PROTO_SR,      CLASS_TAGSET,  VAL_CIMI1,     {14,6,-1},   "CIMI-tagset" },
340
341     {PROTO_NOP,     CLASS_NOP,     VAL_NOP,       {-1},        0          }
342 };
343
344 /* OID utilities */
345
346 void oid_oidcpy(int *t, int *s)
347 {
348     while ((*(t++) = *(s++)) > -1);
349 }
350
351 void oid_oidcat(int *t, int *s)
352 {
353     while (*t > -1)
354         t++;
355     while ((*(t++) = *(s++)) > -1);
356 }
357
358 int oid_oidcmp(int *o1, int *o2)
359 {
360     while (*o1 == *o2 && *o1 > -1)
361     {
362         o1++;
363         o2++;
364     }
365     if (*o1 == *o2)
366         return 0;
367     else if (*o1 > *o2)
368         return 1;
369     else
370         return -1;
371 }
372
373 int oid_oidlen(int *o)
374 {
375     int len = 0;
376
377     while (*(o++) >= 0)
378         len++;
379     return len;
380 }
381
382
383 static int match_prefix(int *look, int *prefix)
384 {
385     int len;
386
387     for (len = 0; *look == *prefix; look++, prefix++, len++);
388     if (*prefix < 0) /* did we reach the end of the prefix? */
389         return len;
390     return 0;
391 }
392
393 void oid_transfer (struct oident *oident)
394 {
395     while (*oident->oidsuffix >= 0)
396     {
397         oid_addent (oident->oidsuffix, oident->proto,
398                     oident->oclass,
399                     oident->desc, oident->value);
400         oident++;
401     }
402 }
403
404 static void oid_init (void)
405 {
406     static int checked = 0;
407     
408     if (checked)
409         return;
410     oid_transfer (oids);
411     checked = 1;
412 }
413
414
415 static struct oident *oid_getentbyoid_x(int *o)
416 {
417     enum oid_proto proto;
418     int prelen;
419     struct oident_list *ol;
420     
421     /* determine protocol type */
422     if ((prelen = match_prefix(o, z3950_prefix)) != 0)
423         proto = PROTO_Z3950;
424     else if ((prelen = match_prefix(o, sr_prefix)) != 0)
425         proto = PROTO_SR;
426     else
427         proto = PROTO_GENERAL;
428     for (ol = oident_table; ol; ol = ol->next)
429     {
430         struct oident *p = &ol->oident;
431         if (p->proto == proto && !oid_oidcmp(o + prelen, p->oidsuffix))
432             return p;
433         if (p->proto == PROTO_GENERAL && !oid_oidcmp (o, p->oidsuffix))
434             return p;
435     }
436     return 0;
437 }
438
439 /*
440  * To query, fill out proto, class, and value of the ent parameter.
441  */
442 int *oid_ent_to_oid(struct oident *ent, int *ret)
443 {
444     struct oident_list *ol;
445     
446     oid_init ();
447     for (ol = oident_table; ol; ol = ol->next)
448     {
449         struct oident *p = &ol->oident;
450         if ((ent->proto == p->proto || p->proto == PROTO_GENERAL) &&
451             (ent->oclass == p->oclass || p->oclass == CLASS_GENERAL) &&
452             ent->value == p->value)
453         {
454             if (p->proto == PROTO_Z3950)
455                 oid_oidcpy(ret, z3950_prefix);
456             else if (p->proto == PROTO_SR)
457                 oid_oidcpy(ret, sr_prefix);
458             else
459                 ret[0] = -1;
460             oid_oidcat(ret, p->oidsuffix);
461             ent->desc = p->desc;
462             return ret;
463         }
464     }
465     ret[0] = -1;
466     return 0;
467 }
468
469 /*
470  * To query, fill out proto, class, and value of the ent parameter.
471  */
472 int *oid_getoidbyent(struct oident *ent)
473 {
474     static int ret[OID_SIZE];
475
476     return oid_ent_to_oid (ent, ret);
477 }
478
479 struct oident *oid_addent (int *oid, int proto, int oclass,
480                            const char *desc, int value)
481 {
482     struct oident *oident;
483
484     nmem_critical_enter ();
485     oident = oid_getentbyoid_x (oid);
486     if (!oident)
487     {
488         char desc_str[200];
489         struct oident_list *oident_list;
490         oident_list = (struct oident_list *) malloc (sizeof(*oident_list));
491         oident = &oident_list->oident;
492         oident->proto = proto;
493         oident->oclass = oclass;
494
495         if (!desc)
496         {
497             int i;
498
499             sprintf (desc_str, "%d", *oid);
500             for (i = 1; oid[i] >= 0; i++)
501                 sprintf (desc_str+strlen(desc_str), ".%d", oid[i]);
502             desc = desc_str;
503         }
504         oident->desc = (char *) malloc (strlen(desc)+1);
505         strcpy (oident->desc, desc);
506         if (value == VAL_DYNAMIC)
507             oident->value = ++oid_value_dynamic;
508         else
509             oident->value = value;
510         oid_oidcpy (oident->oidsuffix, oid);
511         oident_list->next = oident_table;
512         oident_table = oident_list;
513     }
514     nmem_critical_leave ();
515     return oident;
516 }
517
518 struct oident *oid_getentbyoid(int *oid)
519 {
520     struct oident *oident;
521
522     if (!oid)
523         return 0;
524     oid_init ();
525     oident = oid_getentbyoid_x (oid);
526     if (!oident)
527         oident = oid_addent (oid, PROTO_GENERAL, CLASS_GENERAL,
528                              NULL, VAL_DYNAMIC);
529     return oident;
530 }
531
532 static oid_value oid_getval_raw(const char *name)
533 {
534     int val = 0, i = 0, oid[OID_SIZE];
535     struct oident *oident;
536     
537     while (isdigit (*name))
538     {
539         val = val*10 + (*name - '0');
540         name++;
541         if (*name == '.')
542         {
543             if (i < OID_SIZE-1)
544                 oid[i++] = val;
545             val = 0;
546             name++;
547         }
548     }
549     oid[i] = val;
550     oid[i+1] = -1;
551     oident = oid_addent (oid, PROTO_GENERAL, CLASS_GENERAL, NULL,
552                          VAL_DYNAMIC);
553     return oident->value;
554 }
555
556 oid_value oid_getvalbyname(const char *name)
557 {
558     struct oident_list *ol;
559
560     oid_init ();
561     if (isdigit (*name))
562         return oid_getval_raw (name);
563     for (ol = oident_table; ol; ol = ol->next)
564         if (!yaz_matchstr(ol->oident.desc, name))
565         {
566             return ol->oident.value;
567         }
568     return VAL_NONE;
569 }
570
571 void oid_setprivateoids(oident *list)
572 {
573     oid_transfer (list);
574 }