2 * Implementation of SNMPAPI.DLL
4 * Copyright 2002 Patrik Stridvall
5 * Copyright 2007 Hans Leidekker
7 * This library is free software; you can redistribute it and/or
8 * modify it under the terms of the GNU Lesser General Public
9 * License as published by the Free Software Foundation; either
10 * version 2.1 of the License, or (at your option) any later version.
12 * This library is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
15 * Lesser General Public License for more details.
17 * You should have received a copy of the GNU Lesser General Public
18 * License along with this library; if not, write to the Free Software
19 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
31 #include "wine/debug.h"
33 WINE_DEFAULT_DEBUG_CHANNEL(snmpapi);
35 static INT asn_any_copy(AsnAny *dst, AsnAny *src)
37 memset(dst, 0, sizeof(AsnAny));
40 case ASN_INTEGER32: dst->asnValue.number = src->asnValue.number; break;
41 case ASN_UNSIGNED32: dst->asnValue.unsigned32 = src->asnValue.unsigned32; break;
42 case ASN_COUNTER64: dst->asnValue.counter64 = src->asnValue.counter64; break;
43 case ASN_COUNTER32: dst->asnValue.counter = src->asnValue.counter; break;
44 case ASN_GAUGE32: dst->asnValue.gauge = src->asnValue.gauge; break;
45 case ASN_TIMETICKS: dst->asnValue.ticks = src->asnValue.ticks; break;
54 UINT length = src->asnValue.string.length;
56 if (!(stream = HeapAlloc(GetProcessHeap(), 0, length))) return SNMPAPI_ERROR;
58 dst->asnValue.string.stream = stream;
59 dst->asnValue.string.length = length;
60 dst->asnValue.string.dynamic = TRUE;
63 case ASN_OBJECTIDENTIFIER:
65 UINT *ids, i, size = src->asnValue.object.idLength * sizeof(UINT);
67 if (!(ids = HeapAlloc(GetProcessHeap(), 0, size))) return SNMPAPI_ERROR;
69 dst->asnValue.object.ids = ids;
70 dst->asnValue.object.idLength = src->asnValue.object.idLength;
72 for (i = 0; i < dst->asnValue.object.idLength; i++)
73 dst->asnValue.object.ids[i] = src->asnValue.object.ids[i];
78 WARN("unknown ASN type: %d\n", src->asnType);
82 dst->asnType = src->asnType;
83 return SNMPAPI_NOERROR;
86 static void asn_any_free(AsnAny *any)
96 if (any->asnValue.string.dynamic)
98 HeapFree(GetProcessHeap(), 0, any->asnValue.string.stream);
99 any->asnValue.string.stream = NULL;
103 case ASN_OBJECTIDENTIFIER:
105 HeapFree(GetProcessHeap(), 0, any->asnValue.object.ids);
106 any->asnValue.object.ids = NULL;
111 any->asnType = ASN_NULL;
114 /***********************************************************************
115 * DllMain for SNMPAPI
122 TRACE("(%p,%d,%p)\n", hInstDLL, fdwReason, lpvReserved);
125 case DLL_WINE_PREATTACH:
126 return FALSE; /* prefer native version */
127 case DLL_PROCESS_ATTACH:
128 DisableThreadLibraryCalls(hInstDLL);
130 case DLL_PROCESS_DETACH:
137 /***********************************************************************
138 * SnmpUtilDbgPrint (SNMPAPI.@)
141 * The Microsoft headers claim this function uses the stdcall calling
142 * convention. But stdcall functions cannot take a variable number of
143 * arguments so this does not make sense. The stdcall specification is
144 * probably ignored by Microsoft's compiler in this case. So declare it
145 * correctly in Wine so it works with all compilers.
147 VOID WINAPIV SnmpUtilDbgPrint(INT loglevel, LPSTR format, ...)
149 FIXME("(%d, %s)\n", loglevel, debugstr_a(format));
152 /***********************************************************************
153 * SnmpUtilMemAlloc (SNMPAPI.@)
155 LPVOID WINAPI SnmpUtilMemAlloc(UINT nbytes)
157 TRACE("(%d)\n", nbytes);
158 return HeapAlloc(GetProcessHeap(), 0, nbytes);
161 /***********************************************************************
162 * SnmpUtilMemReAlloc (SNMPAPI.@)
164 LPVOID WINAPI SnmpUtilMemReAlloc(LPVOID mem, UINT nbytes)
166 TRACE("(%p, %d)\n", mem, nbytes);
167 return HeapReAlloc(GetProcessHeap(), 0, mem, nbytes);
170 /***********************************************************************
171 * SnmpUtilMemFree (SNMPAPI.@)
173 VOID WINAPI SnmpUtilMemFree(LPVOID mem)
175 TRACE("(%p)\n", mem);
176 HeapFree(GetProcessHeap(), 0, mem);
179 /***********************************************************************
180 * SnmpUtilAsnAnyCpy (SNMPAPI.@)
182 INT WINAPI SnmpUtilAsnAnyCpy(AsnAny *dst, AsnAny *src)
184 TRACE("(%p, %p)\n", dst, src);
185 return asn_any_copy(dst, src);
188 /***********************************************************************
189 * SnmpUtilAsnAnyFree (SNMPAPI.@)
191 VOID WINAPI SnmpUtilAsnAnyFree(AsnAny *any)
193 TRACE("(%p)\n", any);
197 /***********************************************************************
198 * SnmpUtilOctetsCpy (SNMPAPI.@)
200 INT WINAPI SnmpUtilOctetsCpy(AsnOctetString *dst, AsnOctetString *src)
202 TRACE("(%p, %p)\n", dst, src);
204 if (!dst) return SNMPAPI_ERROR;
207 dst->dynamic = FALSE;
210 return SNMPAPI_NOERROR;
212 if ((dst->stream = HeapAlloc(GetProcessHeap(), 0, src->length)))
217 dst->length = src->length;
218 for (i = 0; i < dst->length; i++) dst->stream[i] = src->stream[i];
219 return SNMPAPI_NOERROR;
221 return SNMPAPI_ERROR;
224 /***********************************************************************
225 * SnmpUtilOctetsFree (SNMPAPI.@)
227 VOID WINAPI SnmpUtilOctetsFree(AsnOctetString *octets)
229 TRACE("(%p)\n", octets);
234 if (octets->dynamic) HeapFree(GetProcessHeap(), 0, octets->stream);
235 octets->stream = NULL;
236 octets->dynamic = FALSE;
240 /***********************************************************************
241 * SnmpUtilOctetsNCmp (SNMPAPI.@)
243 INT WINAPI SnmpUtilOctetsNCmp(AsnOctetString *octets1, AsnOctetString *octets2, UINT count)
248 TRACE("(%p, %p, %d)\n", octets1, octets2, count);
250 if (!octets1 || !octets2) return 0;
252 for (i = 0; i < count; i++)
253 if ((ret = octets1->stream[i] - octets2->stream[i])) return ret;
258 /***********************************************************************
259 * SnmpUtilOctetsCmp (SNMPAPI.@)
261 INT WINAPI SnmpUtilOctetsCmp(AsnOctetString *octets1, AsnOctetString *octets2)
263 TRACE("(%p, %p)\n", octets1, octets2);
265 if (octets1->length < octets2->length) return -1;
266 if (octets1->length > octets2->length) return 1;
268 return SnmpUtilOctetsNCmp(octets1, octets2, octets1->length);
271 /***********************************************************************
272 * SnmpUtilOidAppend (SNMPAPI.@)
274 INT WINAPI SnmpUtilOidAppend(AsnObjectIdentifier *dst, AsnObjectIdentifier *src)
278 TRACE("(%p, %p)\n", dst, src);
280 if (!dst) return SNMPAPI_ERROR;
281 if (!src) return SNMPAPI_NOERROR;
283 size = (src->idLength + dst->idLength) * sizeof(UINT);
284 if (!(ids = HeapReAlloc(GetProcessHeap(), 0, dst->ids, size)))
286 if (!(ids = HeapAlloc(GetProcessHeap(), 0, size)))
288 SetLastError(SNMP_MEM_ALLOC_ERROR);
289 return SNMPAPI_ERROR;
291 else memcpy(ids, dst->ids, dst->idLength * sizeof(UINT));
294 for (i = 0; i < src->idLength; i++) ids[i + dst->idLength] = src->ids[i];
295 dst->idLength = dst->idLength + src->idLength;
298 return SNMPAPI_NOERROR;
301 /***********************************************************************
302 * SnmpUtilOidCpy (SNMPAPI.@)
304 INT WINAPI SnmpUtilOidCpy(AsnObjectIdentifier *dst, AsnObjectIdentifier *src)
306 TRACE("(%p, %p)\n", dst, src);
308 if (!dst) return SNMPAPI_ERROR;
313 return SNMPAPI_NOERROR;
315 if ((dst->ids = HeapAlloc(GetProcessHeap(), 0, src->idLength * sizeof(UINT))))
319 dst->idLength = src->idLength;
320 for (i = 0; i < dst->idLength; i++) dst->ids[i] = src->ids[i];
321 return SNMPAPI_NOERROR;
323 return SNMPAPI_ERROR;
326 /***********************************************************************
327 * SnmpUtilOidFree (SNMPAPI.@)
329 VOID WINAPI SnmpUtilOidFree(AsnObjectIdentifier *oid)
331 TRACE("(%p)\n", oid);
336 HeapFree(GetProcessHeap(), 0, oid->ids);
340 /***********************************************************************
341 * SnmpUtilOidNCmp (SNMPAPI.@)
343 INT WINAPI SnmpUtilOidNCmp(AsnObjectIdentifier *oid1, AsnObjectIdentifier *oid2, UINT count)
347 TRACE("(%p, %p, %d)\n", oid1, oid2, count);
349 if (!oid1 || !oid2) return 0;
351 for (i = 0; i < count; i++)
353 if (oid1->ids[i] > oid2->ids[i]) return 1;
354 if (oid1->ids[i] < oid2->ids[i]) return -1;
359 /***********************************************************************
360 * SnmpUtilOidCmp (SNMPAPI.@)
362 INT WINAPI SnmpUtilOidCmp(AsnObjectIdentifier *oid1, AsnObjectIdentifier *oid2)
364 TRACE("(%p, %p)\n", oid1, oid2);
366 if (oid1->idLength < oid2->idLength) return -1;
367 if (oid1->idLength > oid2->idLength) return 1;
369 return SnmpUtilOidNCmp(oid1, oid2, oid1->idLength);
372 /***********************************************************************
373 * SnmpUtilVarBindCpy (SNMPAPI.@)
375 INT WINAPI SnmpUtilVarBindCpy(SnmpVarBind *dst, SnmpVarBind *src)
377 unsigned int i, size;
379 TRACE("(%p, %p)\n", dst, src);
381 if (!dst) return SNMPAPI_ERROR;
384 dst->value.asnType = ASN_NULL;
385 return SNMPAPI_NOERROR;
388 size = src->name.idLength * sizeof(UINT);
389 if (!(dst->name.ids = HeapAlloc(GetProcessHeap(), 0, size))) return SNMPAPI_ERROR;
391 for (i = 0; i < src->name.idLength; i++) dst->name.ids[i] = src->name.ids[i];
392 dst->name.idLength = src->name.idLength;
394 if (!asn_any_copy(&dst->value, &src->value))
396 HeapFree(GetProcessHeap(), 0, dst->name.ids);
397 return SNMPAPI_ERROR;
399 return SNMPAPI_NOERROR;
402 /***********************************************************************
403 * SnmpUtilVarBindFree (SNMPAPI.@)
405 VOID WINAPI SnmpUtilVarBindFree(SnmpVarBind *vb)
411 asn_any_free(&vb->value);
412 HeapFree(GetProcessHeap(), 0, vb->name.ids);
413 vb->name.idLength = 0;
417 /***********************************************************************
418 * SnmpUtilVarBindListCpy (SNMPAPI.@)
420 INT WINAPI SnmpUtilVarBindListCpy(SnmpVarBindList *dst, SnmpVarBindList *src)
422 unsigned int i, size;
423 SnmpVarBind *src_entry, *dst_entry;
425 TRACE("(%p, %p)\n", dst, src);
431 return SNMPAPI_NOERROR;
433 size = src->len * sizeof(SnmpVarBind *);
434 if (!(dst->list = HeapAlloc(GetProcessHeap(), 0, size)))
436 HeapFree(GetProcessHeap(), 0, dst);
437 return SNMPAPI_ERROR;
439 src_entry = src->list;
440 dst_entry = dst->list;
441 for (i = 0; i < src->len; i++)
443 if (SnmpUtilVarBindCpy(dst_entry, src_entry))
450 for (--i; i > 0; i--) SnmpUtilVarBindFree(--dst_entry);
451 HeapFree(GetProcessHeap(), 0, dst->list);
452 return SNMPAPI_ERROR;
456 return SNMPAPI_NOERROR;
459 /***********************************************************************
460 * SnmpUtilVarBindListFree (SNMPAPI.@)
462 VOID WINAPI SnmpUtilVarBindListFree(SnmpVarBindList *vb)
470 for (i = 0; i < vb->len; i++) SnmpUtilVarBindFree(entry++);
471 HeapFree(GetProcessHeap(), 0, vb->list);
476 /***********************************************************************
477 * SnmpUtilIdsToA (SNMPAPI.@)
479 LPSTR WINAPI SnmpUtilIdsToA(UINT *ids, UINT length)
481 static char one[10], oid[514], null_oid[] = "<null oid>";
482 unsigned int i, len, left = sizeof(oid) - 1;
484 TRACE("(%p, %d)\n", ids, length);
486 if (!ids || !length) return null_oid;
489 for (i = 0; i < length; i++)
491 sprintf(one, "%d", ids[i]);
513 /***********************************************************************
514 * SnmpUtilOidToA (SNMPAPI.@)
516 LPSTR WINAPI SnmpUtilOidToA(AsnObjectIdentifier *oid)
518 static char null_oid[] = "<null oid>";
520 TRACE("(%p)\n", oid);
523 return SnmpUtilIdsToA(oid->ids, oid->idLength);
528 /***********************************************************************
529 * SnmpUtilPrintOid (SNMPAPI.@)
531 VOID WINAPI SnmpUtilPrintOid(AsnObjectIdentifier *oid)
535 TRACE("(%p)\n", oid);
539 for (i = 0; i < oid->idLength; i++)
541 TRACE("%u", oid->ids[i]);
542 if (i < oid->idLength - 1) TRACE(".");
547 /***********************************************************************
548 * SnmpUtilPrintAsnAny (SNMPAPI.@)
550 VOID WINAPI SnmpUtilPrintAsnAny(AsnAny *any)
554 TRACE("(%p)\n", any);
556 switch (any->asnType)
558 case ASN_NULL: TRACE("Null value\n"); return;
559 case ASN_INTEGER32: TRACE("Integer32 %d\n", any->asnValue.number); return;
560 case ASN_UNSIGNED32: TRACE("Unsigned32 %u\n", any->asnValue.unsigned32); return;
561 case ASN_COUNTER32: TRACE("Counter32 %u\n", any->asnValue.counter); return;
562 case ASN_GAUGE32: TRACE("Gauge32 %u\n", any->asnValue.gauge); return;
563 case ASN_TIMETICKS: TRACE("Timeticks %u\n", any->asnValue.ticks); return;
566 TRACE("Counter64 %llu\n", any->asnValue.counter64.QuadPart);
569 case ASN_OCTETSTRING:
572 for (i = 0; i < any->asnValue.string.length; i++)
573 TRACE("%c", any->asnValue.string.stream[i]);
580 if (any->asnValue.string.length < 4)
585 for (i = 0; i < 4; i++)
587 TRACE("%u", any->asnValue.string.stream[i]);
588 if (i < 3) TRACE(".");
596 for (i = 0; i < any->asnValue.string.length; i++)
598 TRACE("0x%02x", any->asnValue.string.stream[i]);
599 if (i < any->asnValue.object.idLength - 1) TRACE(" ");
607 for (i = 0; i < any->asnValue.string.length; i++)
609 TRACE("0x%02x", any->asnValue.string.stream[i]);
610 if (i < any->asnValue.object.idLength - 1) TRACE(" ");
615 case ASN_OBJECTIDENTIFIER:
618 for (i = 0; i < any->asnValue.object.idLength; i++)
620 TRACE("%u", any->asnValue.object.ids[i]);
621 if (i < any->asnValue.object.idLength - 1) TRACE(".");
628 TRACE("Invalid type %d\n", any->asnType);