blob: cd38ecf52bce4db72f2eceac989316236f1c3ed8 [file] [log] [blame]
/*
* mib.c
*
* Update: 1998-07-17 <jhy@gsu.edu>
* Added print_oid_report* functions.
*
*/
/**********************************************************************
Copyright 1988, 1989, 1991, 1992 by Carnegie Mellon University
All Rights Reserved
Permission to use, copy, modify, and distribute this software and its
documentation for any purpose and without fee is hereby granted,
provided that the above copyright notice appear in all copies and that
both that copyright notice and this permission notice appear in
supporting documentation, and that the name of CMU not be
used in advertising or publicity pertaining to distribution of the
software without specific, written prior permission.
CMU DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE, INCLUDING
ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN NO EVENT SHALL
CMU BE LIABLE FOR ANY SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES OR
ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS,
WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION,
ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS
SOFTWARE.
******************************************************************/
#include <config.h>
#include <stdio.h>
#include <ctype.h>
#include <sys/types.h>
#if HAVE_NETINET_IN_H
#include <netinet/in.h>
#endif
#if TIME_WITH_SYS_TIME
# ifdef WIN32
# include <sys/timeb.h>
# else
# include <sys/time.h>
# endif
# include <time.h>
#else
# if HAVE_SYS_TIME_H
# include <sys/time.h>
# else
# include <time.h>
# endif
#endif
#if HAVE_STRING_H
#include <string.h>
#else
#include <strings.h>
#endif
#if HAVE_STDLIB_H
#include <stdlib.h>
#endif
#if HAVE_SYS_SELECT_H
#include <sys/select.h>
#endif
#if HAVE_WINSOCK_H
#include <winsock.h>
#endif
#if HAVE_DMALLOC_H
#include <dmalloc.h>
#endif
#include "asn1.h"
#include "snmp_api.h"
#include "mib.h"
#include "snmp.h"
#include "snmp_impl.h"
#include "parse.h"
#include "int64.h"
#include "tools.h"
#include "system.h"
#include "read_config.h"
#include "snmp_debug.h"
#include "default_store.h"
#include "snmp_logging.h"
static struct tree * _sprint_objid(char *buf, oid *objid, size_t objidlen);
static struct tree * _sprint_realloc_objid(u_char **buf, size_t *buf_len,
size_t *out_len, int allow_realloc,
int *buf_overflow,
oid *objid, size_t objidlen);
static char *uptimeString (u_long, char *);
static struct tree *_get_symbol(oid *objid, size_t objidlen, struct tree *subtree,
char *buf, struct index_list *in_dices, char **end_of_known);
static struct tree *_get_realloc_symbol(oid *objid, size_t objidlen,
struct tree *subtree,
u_char **buf, size_t *buf_len,
size_t *out_len, int allow_realloc,
int *buf_overflow,
struct index_list *in_dices,
size_t *end_of_known);
static void print_tree_node (FILE *, struct tree *, int);
static void handle_mibdirs_conf(const char *token, char *line);
static void handle_mibs_conf(const char *token, char *line);
static void handle_mibfile_conf(const char *token, char *line);
static char *dump_oid_to_string(oid *objid, size_t objidlen,
char *buf, char quotechar);
/* helper functions for get_module_node */
static int node_to_oid(struct tree *, oid *, size_t *);
static int _add_strings_to_oid(struct tree *, char *,
oid *, size_t *, size_t);
extern struct tree *tree_head;
static struct tree *tree_top;
struct tree *Mib; /* Backwards compatibility */
oid RFC1213_MIB[] = { 1, 3, 6, 1, 2, 1 };
static char Standard_Prefix[] = ".1.3.6.1.2.1";
/* Set default here as some uses of read_objid require valid pointer. */
static char *Prefix = &Standard_Prefix[0];
typedef struct _PrefixList {
const char *str;
int len;
} *PrefixListPtr, PrefixList;
/*
* Here are the prefix strings.
* Note that the first one finds the value of Prefix or Standard_Prefix.
* Any of these MAY start with period; all will NOT end with period.
* Period is added where needed. See use of Prefix in this module.
*/
PrefixList mib_prefixes[] = {
{ &Standard_Prefix[0] }, /* placeholder for Prefix data */
{ ".iso.org.dod.internet.mgmt.mib-2" },
{ ".iso.org.dod.internet.experimental" },
{ ".iso.org.dod.internet.private" },
{ ".iso.org.dod.internet.snmpParties" },
{ ".iso.org.dod.internet.snmpSecrets" },
{ NULL, 0 } /* end of list */
};
static char *
uptimeString(u_long timeticks,
char *buf)
{
int centisecs, seconds, minutes, hours, days;
if (ds_get_boolean(DS_LIBRARY_ID, DS_LIB_NUMERIC_TIMETICKS)) {
sprintf(buf,"%lu",timeticks);
return buf;
}
centisecs = timeticks % 100;
timeticks /= 100;
days = timeticks / (60 * 60 * 24);
timeticks %= (60 * 60 * 24);
hours = timeticks / (60 * 60);
timeticks %= (60 * 60);
minutes = timeticks / 60;
seconds = timeticks % 60;
if (ds_get_boolean(DS_LIBRARY_ID, DS_LIB_QUICK_PRINT))
sprintf(buf, "%d:%d:%02d:%02d.%02d",
days, hours, minutes, seconds, centisecs);
else {
if (days == 0){
sprintf(buf, "%d:%02d:%02d.%02d",
hours, minutes, seconds, centisecs);
} else if (days == 1) {
sprintf(buf, "%d day, %d:%02d:%02d.%02d",
days, hours, minutes, seconds, centisecs);
} else {
sprintf(buf, "%d days, %d:%02d:%02d.%02d",
days, hours, minutes, seconds, centisecs);
}
}
return buf;
}
/* prints character pointed to if in human-readable ASCII range,
otherwise prints a blank space */
static void sprint_char(char *buf, const u_char ch)
{
if (isprint(ch)) {
sprintf(buf, "%c", (int)ch);
} else {
sprintf(buf, ".");
}
}
void sprint_hexstring(char *buf,
const u_char *cp,
size_t len)
{
const u_char *tp;
size_t lenleft;
for(; len >= 16; len -= 16){
sprintf(buf, "%02X %02X %02X %02X %02X %02X %02X %02X ", cp[0], cp[1], cp[2], cp[3], cp[4], cp[5], cp[6], cp[7]);
buf += strlen(buf);
cp += 8;
sprintf(buf, "%02X %02X %02X %02X %02X %02X %02X %02X", cp[0], cp[1], cp[2], cp[3], cp[4], cp[5], cp[6], cp[7]);
buf += strlen(buf);
cp += 8;
if (ds_get_boolean(DS_LIBRARY_ID, DS_LIB_PRINT_HEX_TEXT))
{
sprintf(buf, " [");
buf += strlen(buf);
for (tp = cp - 16; tp < cp; tp ++)
{
sprint_char(buf++, *tp);
}
sprintf(buf, "]");
buf += strlen(buf);
}
if (len > 16) { *buf++ = '\n'; *buf = 0; }
}
lenleft = len;
for(; len > 0; len--){
sprintf(buf, "%02X ", *cp++);
buf += strlen(buf);
}
if ((lenleft > 0) && ds_get_boolean(DS_LIBRARY_ID, DS_LIB_PRINT_HEX_TEXT))
{
sprintf(buf, " [");
buf += strlen(buf);
for (tp = cp - lenleft; tp < cp; tp ++)
{
sprint_char(buf++, *tp);
}
sprintf(buf, "]");
buf += strlen(buf);
}
*buf = '\0';
}
int
sprint_realloc_hexstring(u_char **buf, size_t *buf_len, size_t *out_len,
int allow_realloc,
const u_char *cp, size_t len)
{
const u_char *tp;
size_t lenleft;
for(; len >= 16; len -= 16){
while ((*out_len + 50) >= *buf_len) {
if (!(allow_realloc && snmp_realloc(buf, buf_len))) {
return 0;
}
}
sprintf((char*)(*buf + *out_len), "%02X %02X %02X %02X %02X %02X %02X %02X ",
cp[0], cp[1], cp[2], cp[3], cp[4], cp[5], cp[6], cp[7]);
*out_len += strlen((char*)(*buf + *out_len));
cp += 8;
sprintf((char*)(*buf + *out_len), "%02X %02X %02X %02X %02X %02X %02X %02X",
cp[0], cp[1], cp[2], cp[3], cp[4], cp[5], cp[6], cp[7]);
*out_len += strlen((char*)(*buf + *out_len));
cp += 8;
if (ds_get_boolean(DS_LIBRARY_ID, DS_LIB_PRINT_HEX_TEXT)) {
while ((*out_len + 21) >= *buf_len) {
if (!(allow_realloc && snmp_realloc(buf, buf_len))) {
return 0;
}
}
sprintf((char*)(*buf + *out_len), " [");
*out_len += strlen((char*)(*buf + *out_len));
for (tp = cp - 16; tp < cp; tp ++) {
sprint_char((char*)(*buf + *out_len), *tp);
(*out_len)++;
}
sprintf((char*)(*buf + *out_len), "]");
*out_len += strlen((char*)(*buf + *out_len));
}
if (len > 16) {
while ((*out_len + 2) >= *buf_len) {
if (!(allow_realloc && snmp_realloc(buf, buf_len))) {
return 0;
}
}
*(*buf + (*out_len)++) = '\n';
*(*buf + *out_len) = 0;
}
}
lenleft = len;
for(; len > 0; len--) {
while ((*out_len + 4) >= *buf_len) {
if (!(allow_realloc && snmp_realloc(buf, buf_len))) {
return 0;
}
}
sprintf((char*)(*buf + *out_len), "%02X ", *cp++);
*out_len += strlen((char*)(*buf + *out_len));
}
if ((lenleft > 0) && ds_get_boolean(DS_LIBRARY_ID, DS_LIB_PRINT_HEX_TEXT)) {
while ((*out_len + 5 + lenleft) >= *buf_len) {
if (!(allow_realloc && snmp_realloc(buf, buf_len))) {
return 0;
}
}
sprintf((char*)(*buf + *out_len), " [");
*out_len += strlen((char*)(*buf + *out_len));
for (tp = cp - lenleft; tp < cp; tp ++) {
sprint_char((char*)(*buf + *out_len), *tp);
(*out_len)++;
}
sprintf((char*)(*buf + *out_len), "]");
*out_len += strlen((char*)(*buf + *out_len));
}
return 1;
}
void sprint_asciistring(char *buf,
const u_char *cp,
size_t len)
{
int x;
for(x = 0; x < (int)len; x++){
if (isprint(*cp)){
if (*cp == '\\' || *cp == '"') {
if (++x >= len) {
*buf = '\0';
return;
}
*buf++ = '\\';
}
*buf++ = *cp++;
} else {
*buf++ = '.';
cp++;
}
}
*buf = '\0';
}
int
sprint_realloc_asciistring(u_char **buf, size_t *buf_len, size_t *out_len,
int allow_realloc,
const u_char *cp, size_t len)
{
int i;
for(i = 0; i < (int)len; i++) {
if (isprint(*cp)) {
if (*cp == '\\' || *cp == '"') {
if ((*out_len >= *buf_len) &&
!(allow_realloc && snmp_realloc(buf, buf_len))) {
return 0;
}
*(*buf + (*out_len)++) = '\\';
}
if ((*out_len >= *buf_len) &&
!(allow_realloc && snmp_realloc(buf, buf_len))) {
return 0;
}
*(*buf + (*out_len)++) = *cp++;
} else {
if ((*out_len >= *buf_len) &&
!(allow_realloc && snmp_realloc(buf, buf_len))) {
return 0;
}
*(*buf + (*out_len)++) = '.';
cp++;
}
}
if ((*out_len >= *buf_len) &&
!(allow_realloc && snmp_realloc(buf, buf_len))) {
return 0;
}
*(*buf + *out_len) = '\0';
return 1;
}
/*
0
< 4
hex
0 ""
< 4 hex Hex: oo oo oo
< 4 "fgh" Hex: oo oo oo
> 4 hex Hex: oo oo oo oo oo oo oo oo
> 4 "this is a test"
*/
void
sprint_octet_string(char *buf,
struct variable_list *var,
struct enum_list *enums,
const char *hint,
const char *units)
{
int hex, x;
u_char *cp;
const char *saved_hint = hint;
char *saved_buf = buf;
if (var->type != ASN_OCTET_STR){
sprintf(buf, "Wrong Type (should be OCTET STRING): ");
buf += strlen(buf);
sprint_by_type(buf, var, NULL, NULL, NULL);
return;
}
if (hint) {
int repeat, width = 1;
long value;
char code = 'd', separ = 0, term = 0, ch;
u_char *ecp;
*buf = 0;
cp = var->val.string;
ecp = cp + var->val_len;
while (cp < ecp) {
repeat = 1;
if (*hint) {
if (*hint == '*') {
repeat = *cp++;
hint++;
}
width = 0;
while ('0' <= *hint && *hint <= '9')
width = width * 10 + *hint++ - '0';
code = *hint++;
if ((ch = *hint) && ch != '*' && (ch < '0' || ch > '9')
&& (width != 0 || (ch != 'x' && ch != 'd' && ch != 'o')))
separ = *hint++;
else separ = 0;
if ((ch = *hint) && ch != '*' && (ch < '0' || ch > '9')
&& (width != 0 || (ch != 'x' && ch != 'd' && ch != 'o')))
term = *hint++;
else term = 0;
if (width == 0) width = 1;
}
while (repeat && cp < ecp) {
value = 0;
if (code != 'a')
for (x = 0; x < width; x++) value = value * 256 + *cp++;
switch (code) {
case 'x':
sprintf (buf, "%lx", value); break;
case 'd':
sprintf (buf, "%ld", value); break;
case 'o':
sprintf (buf, "%lo", value); break;
case 'a':
for (x = 0; x < width && cp < ecp; x++)
*buf++ = *cp++;
*buf = 0;
break;
default:
sprintf(saved_buf, "(Bad hint ignored: %s) ", saved_hint);
sprint_octet_string(saved_buf+strlen(saved_buf),
var, enums, NULL, NULL);
return;
}
buf += strlen (buf);
if (cp < ecp && separ) *buf++ = separ;
repeat--;
}
if (term && cp < ecp) *buf++ = term;
}
if (units) sprintf (buf, " %s", units);
return;
}
hex = 0;
for(cp = var->val.string, x = 0; x < (int)var->val_len; x++, cp++){
if (!(isprint(*cp) || isspace(*cp)))
hex = 1;
}
if (var->val_len == 0){
strcpy(buf, "\"\"");
return;
}
if (!hex){
*buf++ = '"';
sprint_asciistring(buf, var->val.string, var->val_len);
buf += strlen(buf);
*buf++ = '"';
*buf = '\0';
}
if (hex) {
if (ds_get_boolean(DS_LIBRARY_ID, DS_LIB_QUICK_PRINT)){
*buf++ = '"';
*buf = '\0';
} else {
sprintf(buf, " Hex: ");
buf += strlen(buf);
}
sprint_hexstring(buf, var->val.string, var->val_len);
if (ds_get_boolean(DS_LIBRARY_ID, DS_LIB_QUICK_PRINT)){
buf += strlen(buf);
*buf++ = '"';
*buf = '\0';
}
}
if (units) sprintf (buf, " %s", units);
}
int
sprint_realloc_octet_string(u_char **buf, size_t *buf_len, size_t *out_len,
int allow_realloc,
struct variable_list *var,
struct enum_list *enums,
const char *hint,
const char *units)
{
size_t saved_out_len = *out_len;
const char *saved_hint = hint;
int hex = 0, x = 0;
u_char *cp;
if (var->type != ASN_OCTET_STR) {
const char str[] = "Wrong Type (should be OCTET STRING): ";
if (snmp_strcat(buf, buf_len, out_len, allow_realloc, str)) {
return sprint_realloc_by_type(buf, buf_len, out_len, allow_realloc,
var, NULL, NULL, NULL);
} else {
return 0;
}
}
if (hint) {
int repeat, width = 1;
long value;
char code = 'd', separ = 0, term = 0, ch, intbuf[16];
u_char *ecp;
cp = var->val.string;
ecp = cp + var->val_len;
while (cp < ecp) {
repeat = 1;
if (*hint) {
if (*hint == '*') {
repeat = *cp++;
hint++;
}
width = 0;
while ('0' <= *hint && *hint <= '9')
width = (width * 10) + (*hint++ - '0');
code = *hint++;
if ((ch = *hint) && ch != '*' && (ch < '0' || ch > '9')
&& (width != 0 || (ch != 'x' && ch != 'd' && ch != 'o')))
separ = *hint++;
else separ = 0;
if ((ch = *hint) && ch != '*' && (ch < '0' || ch > '9')
&& (width != 0 || (ch != 'x' && ch != 'd' && ch != 'o')))
term = *hint++;
else term = 0;
if (width == 0) width = 1;
}
while (repeat && cp < ecp) {
value = 0;
if (code != 'a') {
for (x = 0; x < width; x++) {
value = value * 256 + *cp++;
}
}
switch (code) {
case 'x':
sprintf(intbuf, "%lx", value);
if (!snmp_strcat(buf, buf_len, out_len, allow_realloc, intbuf)) {
return 0;
}
break;
case 'd':
sprintf (intbuf, "%ld", value);
if (!snmp_strcat(buf, buf_len, out_len, allow_realloc, intbuf)) {
return 0;
}
break;
case 'o':
sprintf (intbuf, "%lo", value);
if (!snmp_strcat(buf, buf_len, out_len, allow_realloc, intbuf)) {
return 0;
}
break;
case 'a':
while ((*out_len + width + 1) >= *buf_len) {
if (!(allow_realloc && snmp_realloc(buf, buf_len))) {
return 0;
}
}
for (x = 0; x < width && cp < ecp; x++) {
*(*buf + *out_len) = *cp++;
(*out_len)++;
}
*(*buf + *out_len) = '\0';
break;
default:
*out_len = saved_out_len;
if (snmp_strcat(buf, buf_len, out_len, allow_realloc,
"(Bad hint ignored: ") &&
snmp_strcat(buf, buf_len, out_len, allow_realloc,
saved_hint) &&
snmp_strcat(buf, buf_len, out_len, allow_realloc,
") ")) {
return sprint_realloc_octet_string(buf, buf_len, out_len,
allow_realloc, var,
enums, NULL, NULL);
} else {
return 0;
}
}
if (cp < ecp && separ) {
while ((*out_len + 1) >= *buf_len) {
if (!(allow_realloc && snmp_realloc(buf, buf_len))) {
return 0;
}
}
*(*buf + *out_len) = separ;
(*out_len)++;
*(*buf + *out_len) = '\0';
}
repeat--;
}
if (term && cp < ecp) {
while ((*out_len + 1) >= *buf_len) {
if (!(allow_realloc && snmp_realloc(buf, buf_len))) {
return 0;
}
}
*(*buf + *out_len) = term;
(*out_len)++;
*(*buf + *out_len) = '\0';
}
}
if (units) {
return (snmp_strcat(buf, buf_len, out_len, allow_realloc, " ") &&
snmp_strcat(buf, buf_len, out_len, allow_realloc, units));
}
return 1;
}
hex = 0;
for(cp = var->val.string, x = 0; x < (int)var->val_len; x++, cp++) {
if (!(isprint(*cp) || isspace(*cp))) {
hex = 1;
}
}
if (var->val_len == 0) {
return snmp_strcat(buf, buf_len, out_len, allow_realloc, "\"\"");
}
if (hex) {
if (ds_get_boolean(DS_LIBRARY_ID, DS_LIB_QUICK_PRINT)) {
if (!snmp_strcat(buf, buf_len, out_len, allow_realloc, "\"")) {
return 0;
}
} else {
if (!snmp_strcat(buf, buf_len, out_len, allow_realloc, " Hex: ")) {
return 0;
}
}
if (!sprint_realloc_hexstring(buf, buf_len, out_len, allow_realloc,
var->val.string, var->val_len)) {
return 0;
}
if (ds_get_boolean(DS_LIBRARY_ID, DS_LIB_QUICK_PRINT)) {
if (!snmp_strcat(buf, buf_len, out_len, allow_realloc, "\"")) {
return 0;
}
}
} else {
if (!snmp_strcat(buf, buf_len, out_len, allow_realloc, "\"")) {
return 0;
}
if (!sprint_realloc_asciistring(buf, buf_len, out_len, allow_realloc,
var->val.string, var->val_len)) {
return 0;
}
if (!snmp_strcat(buf, buf_len, out_len, allow_realloc, "\"")) {
return 0;
}
}
if (units) {
return (snmp_strcat(buf, buf_len, out_len, allow_realloc, " ") &&
snmp_strcat(buf, buf_len, out_len, allow_realloc, units));
}
return 1;
}
int
sprint_realloc_octet_string(u_char **buf, size_t *buf_len, size_t *out_len,
int allow_realloc,
struct variable_list *var,
struct enum_list *enums,
const char *hint,
const char *units)
{
size_t saved_out_len = *out_len;
const char *saved_hint = hint;
int hex = 0, x = 0;
u_char *cp;
if (var->type != ASN_OCTET_STR) {
const char str[] = "Wrong Type (should be OCTET STRING): ";
if (snmp_strcat(buf, buf_len, out_len, allow_realloc, (const u_char*)str)) {
return sprint_realloc_by_type(buf, buf_len, out_len, allow_realloc,
var, NULL, NULL, NULL);
} else {
return 0;
}
}
if (hint) {
int repeat, width = 1;
long value;
char code = 'd', separ = 0, term = 0, ch, intbuf[16];
u_char *ecp;
cp = var->val.string;
ecp = cp + var->val_len;
while (cp < ecp) {
repeat = 1;
if (*hint) {
if (*hint == '*') {
repeat = *cp++;
hint++;
}
width = 0;
while ('0' <= *hint && *hint <= '9')
width = (width * 10) + (*hint++ - '0');
code = *hint++;
if ((ch = *hint) && ch != '*' && (ch < '0' || ch > '9')
&& (width != 0 || (ch != 'x' && ch != 'd' && ch != 'o')))
separ = *hint++;
else separ = 0;
if ((ch = *hint) && ch != '*' && (ch < '0' || ch > '9')
&& (width != 0 || (ch != 'x' && ch != 'd' && ch != 'o')))
term = *hint++;
else term = 0;
if (width == 0) width = 1;
}
while (repeat && cp < ecp) {
value = 0;
if (code != 'a') {
for (x = 0; x < width; x++) {
value = value * 256 + *cp++;
}
}
switch (code) {
case 'x':
sprintf(intbuf, "%lx", value);
if (!snmp_strcat(buf, buf_len, out_len, allow_realloc,(u_char*)intbuf)) {
return 0;
}
break;
case 'd':
sprintf (intbuf, "%ld", value);
if (!snmp_strcat(buf, buf_len, out_len, allow_realloc, (u_char*)intbuf)) {
return 0;
}
break;
case 'o':
sprintf (intbuf, "%lo", value);
if (!snmp_strcat(buf, buf_len, out_len, allow_realloc, (u_char*)intbuf)) {
return 0;
}
break;
case 'a':
while ((*out_len + width + 1) >= *buf_len) {
if (!(allow_realloc && snmp_realloc(buf, buf_len))) {
return 0;
}
}
for (x = 0; x < width && cp < ecp; x++) {
*(*buf + *out_len) = *cp++;
(*out_len)++;
}
*(*buf + *out_len) = '\0';
break;
default:
*out_len = saved_out_len;
if (snmp_strcat(buf, buf_len, out_len, allow_realloc,
(const u_char*)"(Bad hint ignored: ") &&
snmp_strcat(buf, buf_len, out_len, allow_realloc,
(const u_char*)saved_hint) &&
snmp_strcat(buf, buf_len, out_len, allow_realloc,
(const u_char*)") ")) {
return sprint_realloc_octet_string(buf, buf_len, out_len,
allow_realloc, var,
enums, NULL, NULL);
} else {
return 0;
}
}
if (cp < ecp && separ) {
while ((*out_len + 1) >= *buf_len) {
if (!(allow_realloc && snmp_realloc(buf, buf_len))) {
return 0;
}
}
*(*buf + *out_len) = separ;
(*out_len)++;
*(*buf + *out_len) = '\0';
}
repeat--;
}
if (term && cp < ecp) {
while ((*out_len + 1) >= *buf_len) {
if (!(allow_realloc && snmp_realloc(buf, buf_len))) {
return 0;
}
}
*(*buf + *out_len) = term;
(*out_len)++;
*(*buf + *out_len) = '\0';
}
}
if (units) {
return (snmp_strcat(buf, buf_len, out_len, allow_realloc, (const u_char*)" ") &&
snmp_strcat(buf, buf_len, out_len, allow_realloc, (const u_char*)units));
}
return 1;
}
hex = 0;
for(cp = var->val.string, x = 0; x < (int)var->val_len; x++, cp++) {
if (!(isprint(*cp) || isspace(*cp))) {
hex = 1;
}
}
if (var->val_len == 0) {
return snmp_strcat(buf, buf_len, out_len, allow_realloc, (const u_char*)"\"\"");
}
if (hex) {
if (ds_get_boolean(DS_LIBRARY_ID, DS_LIB_QUICK_PRINT)) {
if (!snmp_strcat(buf, buf_len, out_len, allow_realloc, (const u_char*)"\"")) {
return 0;
}
} else {
if (!snmp_strcat(buf, buf_len, out_len, allow_realloc, (const u_char*)" Hex: ")) {
return 0;
}
}
if (!sprint_realloc_hexstring(buf, buf_len, out_len, allow_realloc,
var->val.string, var->val_len)) {
return 0;
}
if (ds_get_boolean(DS_LIBRARY_ID, DS_LIB_QUICK_PRINT)) {
if (!snmp_strcat(buf, buf_len, out_len, allow_realloc, (const u_char*)"\"")) {
return 0;
}
}
} else {
if (!snmp_strcat(buf, buf_len, out_len, allow_realloc, (const u_char*)"\"")) {
return 0;
}
if (!sprint_realloc_asciistring(buf, buf_len, out_len, allow_realloc,
var->val.string, var->val_len)) {
return 0;
}
if (!snmp_strcat(buf, buf_len, out_len, allow_realloc, (const u_char*)"\"")) {
return 0;
}
}
if (units) {
return (snmp_strcat(buf, buf_len, out_len, allow_realloc, (const u_char*)" ") &&
snmp_strcat(buf, buf_len, out_len, allow_realloc, (const u_char*)units));
}
return 1;
}
#ifdef OPAQUE_SPECIAL_TYPES
void
sprint_float(char *buf,
struct variable_list *var,
struct enum_list *enums,
const char *hint,
const char *units)
{
if (var->type != ASN_OPAQUE_FLOAT) {
sprintf(buf, "Wrong Type (should be Float): ");
buf += strlen(buf);
sprint_by_type(buf, var, NULL, NULL, NULL);
return;
}
if (!ds_get_boolean(DS_LIBRARY_ID, DS_LIB_QUICK_PRINT)){
sprintf(buf, "Opaque: Float:");
buf += strlen(buf);
}
sprintf(buf, " %f", *var->val.floatVal);
buf += strlen (buf);
if (units) sprintf (buf, " %s", units);
}
int
sprint_realloc_float(u_char **buf, size_t *buf_len,
size_t *out_len, int allow_realloc,
struct variable_list *var,
struct enum_list *enums,
const char *hint,
const char *units)
{
if (var->type != ASN_OPAQUE_FLOAT) {
u_char str[] = "Wrong Type (should be Float): ";
if (snmp_strcat(buf, buf_len, out_len, allow_realloc, str)) {
return sprint_realloc_by_type(buf, buf_len, out_len, allow_realloc,
var, NULL, NULL, NULL);
} else {
return 0;
}
}
if (!ds_get_boolean(DS_LIBRARY_ID, DS_LIB_QUICK_PRINT)) {
if (!snmp_strcat(buf, buf_len, out_len, allow_realloc,(const u_char*)"Opaque: Float: ")) {
return 0;
}
}
/* How much space needed for max. length float? 128 is overkill. */
while ((*out_len + 128 + 1) >= *buf_len) {
if (!(allow_realloc && snmp_realloc(buf, buf_len))) {
return 0;
}
}
sprintf((char*)(*buf + *out_len), "%f", *var->val.floatVal);
*out_len += strlen((char*)(*buf + *out_len));
if (units) {
return (snmp_strcat(buf, buf_len, out_len, allow_realloc, (const u_char*)" ") &&
snmp_strcat(buf, buf_len, out_len, allow_realloc, (const u_char*)units));
}
return 1;
}
void
sprint_double(char *buf,
struct variable_list *var,
struct enum_list *enums,
const char *hint,
const char *units)
{
if (var->type != ASN_OPAQUE_DOUBLE) {
sprintf(buf, "Wrong Type (should be Double): ");
buf += strlen(buf);
sprint_by_type(buf, var, NULL, NULL, NULL);
return;
}
if (!ds_get_boolean(DS_LIBRARY_ID, DS_LIB_QUICK_PRINT)){
sprintf(buf, "Opaque: Double:");
buf += strlen(buf);
}
sprintf(buf, " %f", *var->val.doubleVal);
buf += strlen (buf);
if (units) sprintf (buf, " %s", units);
}
int
sprint_realloc_double(u_char **buf, size_t *buf_len,
size_t *out_len, int allow_realloc,
struct variable_list *var,
struct enum_list *enums,
const char *hint,
const char *units)
{
if (var->type != ASN_OPAQUE_DOUBLE) {
const char str[] = "Wrong Type (should be Double): ";
if (snmp_strcat(buf, buf_len, out_len, allow_realloc, (const u_char*)str)) {
return sprint_realloc_by_type(buf, buf_len, out_len, allow_realloc,
var, NULL, NULL, NULL);
} else {
return 0;
}
}
if (!ds_get_boolean(DS_LIBRARY_ID, DS_LIB_QUICK_PRINT)) {
if (!snmp_strcat(buf, buf_len, out_len, allow_realloc,(const u_char*)"Opaque: Float: ")) {
return 0;
}
}
/* How much space needed for max. length double? 128 is overkill. */
while ((*out_len + 128 + 1) >= *buf_len) {
if (!(allow_realloc && snmp_realloc(buf, buf_len))) {
return 0;
}
}
sprintf((char*)(*buf + *out_len), "%f", *var->val.doubleVal);
*out_len += strlen((char*)(*buf + *out_len));
if (units) {
return (snmp_strcat(buf, buf_len, out_len, allow_realloc, (const u_char*)" ") &&
snmp_strcat(buf, buf_len, out_len, allow_realloc, (const u_char*)units));
}
return 1;
}
#endif /* OPAQUE_SPECIAL_TYPES */
void
sprint_opaque(char *buf,
struct variable_list *var,
struct enum_list *enums,
const char *hint,
const char *units)
{
if (var->type != ASN_OPAQUE
#ifdef OPAQUE_SPECIAL_TYPES
&& var->type != ASN_OPAQUE_COUNTER64
&& var->type != ASN_OPAQUE_U64
&& var->type != ASN_OPAQUE_I64
&& var->type != ASN_OPAQUE_FLOAT
&& var->type != ASN_OPAQUE_DOUBLE
#endif /* OPAQUE_SPECIAL_TYPES */
){
sprintf(buf, "Wrong Type (should be Opaque): ");
buf += strlen(buf);
sprint_by_type(buf, var, NULL, NULL, NULL);
return;
}
#ifdef OPAQUE_SPECIAL_TYPES
switch(var->type) {
case ASN_OPAQUE_COUNTER64:
case ASN_OPAQUE_U64:
case ASN_OPAQUE_I64:
sprint_counter64(buf, var, enums, hint, units);
break;
case ASN_OPAQUE_FLOAT:
sprint_float(buf, var, enums, hint, units);
break;
case ASN_OPAQUE_DOUBLE:
sprint_double(buf, var, enums, hint, units);
break;
case ASN_OPAQUE:
#endif
if (!ds_get_boolean(DS_LIBRARY_ID, DS_LIB_QUICK_PRINT)){
sprintf(buf, "OPAQUE: ");
buf += strlen(buf);
}
sprint_hexstring(buf, var->val.string, var->val_len);
buf += strlen (buf);
#ifdef OPAQUE_SPECIAL_TYPES
}
#endif
if (units) sprintf (buf, " %s", units);
}
int
sprint_realloc_counter64(u_char **buf, size_t *buf_len, size_t *out_len,
int allow_realloc,
struct variable_list *var,
struct enum_list *enums,
const char *hint,
const char *units)
{
char a64buf[I64CHARSZ+1];
if (var->type != ASN_COUNTER64
#ifdef OPAQUE_SPECIAL_TYPES
&& var->type != ASN_OPAQUE_COUNTER64
&& var->type != ASN_OPAQUE_I64
&& var->type != ASN_OPAQUE_U64
#endif
) {
u_char str[] = "Wrong Type (should be Counter64): ";
if (snmp_strcat(buf, buf_len, out_len, allow_realloc, str)) {
return sprint_realloc_by_type(buf, buf_len, out_len, allow_realloc,
var, NULL, NULL, NULL);
} else {
return 0;
}
}
if (!ds_get_boolean(DS_LIBRARY_ID, DS_LIB_QUICK_PRINT)) {
#ifdef OPAQUE_SPECIAL_TYPES
if (var->type != ASN_COUNTER64) {
if (!snmp_strcat(buf, buf_len, out_len, allow_realloc, (const u_char*)"Opaque: ")) {
return 0;
}
}
#endif
#ifdef OPAQUE_SPECIAL_TYPES
switch(var->type) {
case ASN_OPAQUE_U64:
if (!snmp_strcat(buf, buf_len, out_len, allow_realloc, (const u_char*)"UInt64: ")) {
return 0;
}
break;
case ASN_OPAQUE_I64:
if (!snmp_strcat(buf, buf_len, out_len, allow_realloc, (const u_char*)"Int64: ")) {
return 0;
}
break;
case ASN_COUNTER64:
case ASN_OPAQUE_COUNTER64:
#endif
if (!snmp_strcat(buf, buf_len, out_len, allow_realloc, (const u_char*)"Counter64: ")) {
return 0;
}
#ifdef OPAQUE_SPECIAL_TYPES
}
#endif
}
#ifdef OPAQUE_SPECIAL_TYPES
if (var->type == ASN_OPAQUE_I64) {
printI64(a64buf, var->val.counter64);
if (!snmp_strcat(buf, buf_len, out_len, allow_realloc, (const u_char*)a64buf)) {
return 0;
}
} else {
#endif
printU64(a64buf, var->val.counter64);
if (!snmp_strcat(buf, buf_len, out_len, allow_realloc, (const u_char*)a64buf)) {
return 0;
}
#ifdef OPAQUE_SPECIAL_TYPES
}
#endif
if (units) {
return (snmp_strcat(buf, buf_len, out_len, allow_realloc, (const u_char*)" ") &&
snmp_strcat(buf, buf_len, out_len, allow_realloc, (const u_char*)units));
}
return 1;
}
int
sprint_realloc_opaque(u_char **buf, size_t *buf_len,
size_t *out_len, int allow_realloc,
struct variable_list *var,
struct enum_list *enums,
const char *hint,
const char *units)
{
if (var->type != ASN_OPAQUE
#ifdef OPAQUE_SPECIAL_TYPES
&& var->type != ASN_OPAQUE_COUNTER64
&& var->type != ASN_OPAQUE_U64
&& var->type != ASN_OPAQUE_I64
&& var->type != ASN_OPAQUE_FLOAT
&& var->type != ASN_OPAQUE_DOUBLE
#endif /* OPAQUE_SPECIAL_TYPES */
) {
u_char str[] = "Wrong Type (should be Opaque): ";
if (snmp_strcat(buf, buf_len, out_len, allow_realloc, str)) {
return sprint_realloc_by_type(buf, buf_len, out_len, allow_realloc,
var, NULL, NULL, NULL);
} else {
return 0;
}
}
#ifdef OPAQUE_SPECIAL_TYPES
switch(var->type) {
case ASN_OPAQUE_COUNTER64:
case ASN_OPAQUE_U64:
case ASN_OPAQUE_I64:
return sprint_realloc_counter64(buf, buf_len, out_len, allow_realloc,
var, enums, hint, units);
break;
case ASN_OPAQUE_FLOAT:
return sprint_realloc_float(buf, buf_len, out_len, allow_realloc,
var, enums, hint, units);
break;
case ASN_OPAQUE_DOUBLE:
return sprint_realloc_double(buf, buf_len, out_len, allow_realloc,
var, enums, hint, units);
break;
case ASN_OPAQUE:
#endif
if (!ds_get_boolean(DS_LIBRARY_ID, DS_LIB_QUICK_PRINT)) {
u_char str[] = "OPAQUE: ";
if (!snmp_strcat(buf, buf_len, out_len, allow_realloc, str)) {
return 0;
}
}
if (!sprint_realloc_hexstring(buf, buf_len, out_len, allow_realloc,
var->val.string, var->val_len)) {
return 0;
}
#ifdef OPAQUE_SPECIAL_TYPES
}
#endif
if (units) {
return (snmp_strcat(buf, buf_len, out_len, allow_realloc, (const u_char*)" ") &&
snmp_strcat(buf, buf_len, out_len, allow_realloc, (const u_char*)units));
}
return 1;
}
void
sprint_object_identifier(char *buf,
struct variable_list *var,
struct enum_list *enums,
const char *hint,
const char *units)
{
if (var->type != ASN_OBJECT_ID){
sprintf(buf, "Wrong Type (should be OBJECT IDENTIFIER): ");
buf += strlen(buf);
sprint_by_type(buf, var, NULL, NULL, NULL);
return;
}
if (!ds_get_boolean(DS_LIBRARY_ID, DS_LIB_QUICK_PRINT)){
sprintf(buf, "OID: ");
buf += strlen(buf);
}
_sprint_objid(buf, (oid *)(var->val.objid), var->val_len / sizeof(oid));
buf += strlen (buf);
if (units) sprintf (buf, " %s", units);
}
int
sprint_realloc_object_identifier(u_char **buf, size_t *buf_len,
size_t *out_len, int allow_realloc,
struct variable_list *var,
struct enum_list *enums,
const char *hint,
const char *units)
{
int buf_overflow = 0;
if (var->type != ASN_OBJECT_ID) {
u_char str[] = "Wrong Type (should be OBJECT IDENTIFIER): ";
if (snmp_strcat(buf, buf_len, out_len, allow_realloc, str)) {
return sprint_realloc_by_type(buf, buf_len, out_len, allow_realloc,
var, NULL, NULL, NULL);
} else {
return 0;
}
}
if (!ds_get_boolean(DS_LIBRARY_ID, DS_LIB_QUICK_PRINT)) {
u_char str[] = "OID: ";
if (!snmp_strcat(buf, buf_len, out_len, allow_realloc, str)) {
return 0;
}
}
_sprint_realloc_objid(buf, buf_len, out_len, allow_realloc, &buf_overflow,
(oid *)(var->val.objid), var->val_len/sizeof(oid));
if (buf_overflow) {
return 0;
}
if (units) {
return (snmp_strcat(buf, buf_len, out_len, allow_realloc, (const u_char*)" ") &&
snmp_strcat(buf, buf_len, out_len, allow_realloc, (const u_char*)units));
}
return 1;
}
void
sprint_timeticks(char *buf,
struct variable_list *var,
struct enum_list *enums,
const char *hint,
const char *units)
{
char timebuf[32];
if (var->type != ASN_TIMETICKS){
sprintf(buf, "Wrong Type (should be Timeticks): ");
buf += strlen(buf);
sprint_by_type(buf, var, NULL, NULL, NULL);
return;
}
if (ds_get_boolean(DS_LIBRARY_ID, DS_LIB_NUMERIC_TIMETICKS)) {
sprintf(buf,"%lu", *(u_long *)(var->val.integer));
return;
}
if (!ds_get_boolean(DS_LIBRARY_ID, DS_LIB_QUICK_PRINT)){
sprintf(buf, "Timeticks: (%lu) ", *(u_long *)(var->val.integer));
buf += strlen(buf);
}
sprintf(buf, "%s", uptimeString(*(u_long *)(var->val.integer), timebuf));
buf += strlen (buf);
if (units) sprintf (buf, " %s", units);
}
int
sprint_realloc_timeticks(u_char **buf, size_t *buf_len, size_t *out_len,
int allow_realloc,
struct variable_list *var,
struct enum_list *enums,
const char *hint,
const char *units)
{
char timebuf[32];
if (var->type != ASN_TIMETICKS) {
u_char str[] = "Wrong Type (should be Timeticks): ";
if (snmp_strcat(buf, buf_len, out_len, allow_realloc, str)) {
return sprint_realloc_by_type(buf, buf_len, out_len, allow_realloc,
var, NULL, NULL, NULL);
} else {
return 0;
}
}
if (ds_get_boolean(DS_LIBRARY_ID, DS_LIB_NUMERIC_TIMETICKS)) {
char str[16];
sprintf(str, "%lu", *(u_long *)var->val.integer);
if (!snmp_strcat(buf, buf_len, out_len, allow_realloc, (const u_char*)str)) {
return 0;
}
}
if (!ds_get_boolean(DS_LIBRARY_ID, DS_LIB_QUICK_PRINT)) {
char str[32];
sprintf(str, "Timeticks: (%lu) ", *(u_long *)var->val.integer);
if (!snmp_strcat(buf, buf_len, out_len, allow_realloc, (const u_char*)str)) {
return 0;
}
}
uptimeString(*(u_long *)(var->val.integer), timebuf);
if (!snmp_strcat(buf, buf_len, out_len, allow_realloc, (const u_char*)timebuf)) {
return 0;
}
if (units) {
return (snmp_strcat(buf, buf_len, out_len, allow_realloc, (const u_char*)" ") &&
snmp_strcat(buf, buf_len, out_len, allow_realloc, (const u_char*)units));
}
return 1;
}
void
sprint_hinted_integer (char *buf,
long val,
const char *hint,
const char *units)
{
char code;
int shift, len;
char tmp[256];
char fmt[10];
code = hint[0];
if (hint [1] == '-') {
shift = atoi (hint+2);
}
else shift = 0;
fmt[0] = '%';
fmt[1] = 'l';
fmt[2] = code;
fmt[3] = 0;
sprintf (tmp, fmt, val);
if (shift != 0) {
len = strlen (tmp);
if (shift <= len) {
tmp[len+1] = 0;
while (shift--) {
tmp[len] = tmp[len-1];
len--;
}
tmp[len] = '.';
}
else {
tmp[shift+1] = 0;
while (shift) {
if (len-- > 0) tmp [shift] = tmp [len];
else tmp[shift] = '0';
shift--;
}
tmp[0] = '.';
}
}
strcpy (buf, tmp);
}
int
sprint_realloc_hinted_integer (u_char **buf, size_t *buf_len, size_t *out_len,
int allow_realloc,
long val, const char decimaltype,
const char *hint,
const char *units)
{
char fmt[10] = "%l@", tmp[256];
int shift, len;
if (hint[1] == '-') {
shift = atoi(hint+2);
} else {
shift = 0;
}
if (hint[0] == 'd') {
/* We might *actually* want a 'u' here. */
fmt[2] = decimaltype;
} else {
/* DISPLAY-HINT character is 'b', 'o', or 'x'. */
fmt[2] = hint[0];
}
sprintf(tmp, fmt, val);
if (shift != 0) {
len = strlen(tmp);
if (shift <= len) {
tmp[len+1] = 0;
while (shift--) {
tmp[len] = tmp[len-1];
len--;
}
tmp[len] = '.';
} else {
tmp[shift+1] = 0;
while (shift) {
if (len-- > 0) {
tmp[shift] = tmp[len];
} else {
tmp[shift] = '0';
}
shift--;
}
tmp[0] = '.';
}
}
return snmp_strcat(buf, buf_len, out_len, allow_realloc, tmp);
}
void
sprint_integer(char *buf,
struct variable_list *var,
struct enum_list *enums,
const char *hint,
const char *units)
{
char *enum_string = NULL;
if (var->type != ASN_INTEGER){
sprintf(buf, "Wrong Type (should be INTEGER): ");
buf += strlen(buf);
sprint_by_type(buf, var, NULL, NULL, NULL);
return;
}
for (; enums; enums = enums->next)
if (enums->value == *var->val.integer){
enum_string = enums->label;
break;
}
if (enum_string == NULL ||
ds_get_boolean(DS_LIBRARY_ID,DS_LIB_PRINT_NUMERIC_ENUM)) {
if (hint) sprint_hinted_integer(buf, *var->val.integer, hint, units);
else sprintf(buf, "%ld", *var->val.integer);
}
else if (ds_get_boolean(DS_LIBRARY_ID, DS_LIB_QUICK_PRINT))
sprintf(buf, "%s", enum_string);
else
sprintf(buf, "%s(%ld)", enum_string, *var->val.integer);
buf += strlen (buf);
if (units) sprintf (buf, " %s", units);
}
void
sprint_uinteger(char *buf,
struct variable_list *var,
struct enum_list *enums,
const char *hint,
const char *units)
{
char *enum_string = NULL;
if (var->type != ASN_UINTEGER){
sprintf(buf, "Wrong Type (should be UInteger32): ");
buf += strlen(buf);
sprint_by_type(buf, var, NULL, NULL, NULL);
return;
}
for (; enums; enums = enums->next)
if (enums->value == *var->val.integer){
enum_string = enums->label;
break;
}
if (enum_string == NULL ||
ds_get_boolean(DS_LIBRARY_ID,DS_LIB_PRINT_NUMERIC_ENUM))
sprintf(buf, "%lu", *var->val.integer);
else if (ds_get_boolean(DS_LIBRARY_ID, DS_LIB_QUICK_PRINT))
sprintf(buf, "%s", enum_string);
else
sprintf(buf, "%s(%lu)", enum_string, *var->val.integer);
buf += strlen (buf);
if (units) sprintf (buf, " %s", units);
}
int
sprint_realloc_uinteger(u_char **buf, size_t *buf_len, size_t *out_len,
int allow_realloc,
struct variable_list *var,
struct enum_list *enums,
const char *hint,
const char *units)
{
char *enum_string = NULL;
if (var->type != ASN_UINTEGER){
u_char str[] = "Wrong Type (should be UInteger32): ";
if (snmp_strcat(buf, buf_len, out_len, allow_realloc, str)) {
return sprint_realloc_by_type(buf, buf_len, out_len, allow_realloc,
var, NULL, NULL, NULL);
} else {
return 0;
}
}
for (; enums; enums = enums->next) {
if (enums->value == *var->val.integer) {
enum_string = enums->label;
break;
}
}
if (enum_string == NULL ||
ds_get_boolean(DS_LIBRARY_ID, DS_LIB_PRINT_NUMERIC_ENUM)) {
if (hint) {
if (!(sprint_realloc_hinted_integer(buf, buf_len, out_len,
allow_realloc, *var->val.integer,
'u', hint, units))) {
return 0;
}
} else {
char str[16];
sprintf(str, "%lu", *var->val.integer);
if (!snmp_strcat(buf, buf_len, out_len, allow_realloc, (const u_char*)str)) {
return 0;
}
}
} else if (ds_get_boolean(DS_LIBRARY_ID, DS_LIB_QUICK_PRINT)) {
if (!snmp_strcat(buf, buf_len, out_len, allow_realloc, (const u_char*)enum_string)) {
return 0;
}
} else {
char str[16];
sprintf(str, "(%lu)", *var->val.integer);
if (!snmp_strcat(buf, buf_len, out_len, allow_realloc, (const u_char*)enum_string)) {
return 0;
}
if (!snmp_strcat(buf, buf_len, out_len, allow_realloc, (const u_char*)str)) {
return 0;
}
}
if (units) {
return (snmp_strcat(buf, buf_len, out_len, allow_realloc, (const u_char*)" ") &&
snmp_strcat(buf, buf_len, out_len, allow_realloc, (const u_char*)units));
}
return 1;
}
void
sprint_gauge(char *buf,
struct variable_list *var,
struct enum_list *enums,
const char *hint,
const char *units)
{
if (var->type != ASN_GAUGE){
sprintf(buf, "Wrong Type (should be Gauge32 or Unsigned32): ");
buf += strlen(buf);
sprint_by_type(buf, var, NULL, NULL, NULL);
return;
}
if (ds_get_boolean(DS_LIBRARY_ID, DS_LIB_QUICK_PRINT))
sprintf(buf, "%lu", *var->val.integer);
else
sprintf(buf, "Gauge32: %lu", *var->val.integer);
buf += strlen (buf);
if (units) sprintf (buf, " %s", units);
}
int
sprint_realloc_gauge(u_char **buf, size_t *buf_len, size_t *out_len,
int allow_realloc,
struct variable_list *var,
struct enum_list *enums,
const char *hint,
const char *units)
{
char tmp[32];
if (var->type != ASN_GAUGE) {
u_char str[] = "Wrong Type (should be Gauge32 or Unsigned32): ";
if (snmp_strcat(buf, buf_len, out_len, allow_realloc, str)) {
return sprint_realloc_by_type(buf, buf_len, out_len, allow_realloc,
var, NULL, NULL, NULL);
} else {
return 0;
}
}
if (!ds_get_boolean(DS_LIBRARY_ID, DS_LIB_QUICK_PRINT)) {
u_char str[] = "Gauge32: ";
if (!snmp_strcat(buf, buf_len, out_len, allow_realloc, str)) {
return 0;
}
}
if (hint) {
if (!sprint_realloc_hinted_integer(buf, buf_len, out_len,
allow_realloc, *var->val.integer,
'u', hint, units)) {
return 0;
}
} else {
sprintf(tmp, "%lu", *var->val.integer);
if (!snmp_strcat(buf, buf_len, out_len, allow_realloc, (const u_char*)tmp)) {
return 0;
}
}
if (units) {
return (snmp_strcat(buf, buf_len, out_len, allow_realloc, (const u_char*)" ") &&
snmp_strcat(buf, buf_len, out_len, allow_realloc, (const u_char*)units));
}
return 1;
}
void
sprint_counter(char *buf,
struct variable_list *var,
struct enum_list *enums,
const char *hint,
const char *units)
{
if (var->type != ASN_COUNTER){
sprintf(buf, "Wrong Type (should be Counter32): ");
buf += strlen(buf);
sprint_by_type(buf, var, NULL, NULL, NULL);
return;
}
if (ds_get_boolean(DS_LIBRARY_ID, DS_LIB_QUICK_PRINT))
sprintf(buf, "%lu", *var->val.integer);
else
sprintf(buf, "Counter32: %lu", *var->val.integer);
buf += strlen (buf);
if (units) sprintf (buf, " %s", units);
}
int
sprint_realloc_counter(u_char **buf, size_t *buf_len, size_t *out_len,
int allow_realloc,
struct variable_list *var,
struct enum_list *enums,
const char *hint,
const char *units)
{
char tmp[32];
if (var->type != ASN_COUNTER) {
u_char str[] = "Wrong Type (should be Counter32): ";
if (snmp_strcat(buf, buf_len, out_len, allow_realloc, str)) {
return sprint_realloc_by_type(buf, buf_len, out_len, allow_realloc,
var, NULL, NULL, NULL);
} else {
return 0;
}
}
if (!ds_get_boolean(DS_LIBRARY_ID, DS_LIB_QUICK_PRINT)) {
u_char str[] = "Counter32: ";
if (!snmp_strcat(buf, buf_len, out_len, allow_realloc, str)) {
return 0;
}
}
sprintf(tmp, "%lu", *var->val.integer);
if (!snmp_strcat(buf, buf_len, out_len, allow_realloc, (const u_char*)tmp)) {
return 0;
}
if (units) {
return (snmp_strcat(buf, buf_len, out_len, allow_realloc, (const u_char*)" ") &&
snmp_strcat(buf, buf_len, out_len, allow_realloc, (const u_char*)units));
}
return 1;
}
void
sprint_networkaddress(char *buf,
struct variable_list *var,
struct enum_list *enums,
const char *hint,
const char *units)
{
int x, len;
u_char *cp;
if (var->type != ASN_IPADDRESS){
sprintf(buf, "Wrong Type (should be NetworkAddress): ");
buf += strlen(buf);
sprint_by_type(buf, var, NULL, NULL, NULL);
return;
}
if (!ds_get_boolean(DS_LIBRARY_ID, DS_LIB_QUICK_PRINT)){
sprintf(buf, "Network Address: ");
buf += strlen(buf);
}
cp = var->val.string;
len = var->val_len;
for(x = 0; x < len; x++){
sprintf(buf, "%02X", *cp++);
buf += strlen(buf);
if (x < (len - 1))
*buf++ = ':';
}
}
int
sprint_realloc_networkaddress(u_char **buf, size_t *buf_len, size_t *out_len,
int allow_realloc,
struct variable_list *var,
struct enum_list *enums,
const char *hint,
const char *units)
{
size_t i;
if (var->type != ASN_IPADDRESS) {
u_char str[] = "Wrong Type (should be NetworkAddress): ";
if (snmp_strcat(buf, buf_len, out_len, allow_realloc, str)) {
return sprint_realloc_by_type(buf, buf_len, out_len, allow_realloc,
var, NULL, NULL, NULL);
} else {
return 0;
}
}
if (!ds_get_boolean(DS_LIBRARY_ID, DS_LIB_QUICK_PRINT)) {
u_char str[] = "Network Address: ";
if (!snmp_strcat(buf, buf_len, out_len, allow_realloc, str)) {
return 0;
}
}
while ((*out_len + (var->val_len * 3) + 2) >= *buf_len) {
if (!(allow_realloc && snmp_realloc(buf, buf_len))) {
return 0;
}
}
for (i = 0; i < var->val_len; i++) {
sprintf((char*)(*buf + *out_len), "%02X", var->val.string[i]);
*out_len += 2;
if (i < var->val_len - 1) {
*(*buf + *out_len) = ':';
(*out_len)++;
}
}
return 1;
}
void
sprint_ipaddress(char *buf,
struct variable_list *var,
struct enum_list *enums,
const char *hint,
const char *units)
{
u_char *ip;
if (var->type != ASN_IPADDRESS){
sprintf(buf, "Wrong Type (should be IpAddress): ");
buf += strlen(buf);
sprint_by_type(buf, var, NULL, NULL, NULL);
return;
}
ip = var->val.string;
if (ds_get_boolean(DS_LIBRARY_ID, DS_LIB_QUICK_PRINT))
sprintf(buf, "%d.%d.%d.%d",ip[0], ip[1], ip[2], ip[3]);
else
sprintf(buf, "IpAddress: %d.%d.%d.%d",ip[0], ip[1], ip[2], ip[3]);
}
int
sprint_realloc_ipaddress(u_char **buf, size_t *buf_len, size_t *out_len,
int allow_realloc,
struct variable_list *var,
struct enum_list *enums,
const char *hint,
const char *units)
{
u_char *ip = var->val.string;
if (var->type != ASN_IPADDRESS) {
u_char str[] = "Wrong Type (should be IpAddress): ";
if (snmp_strcat(buf, buf_len, out_len, allow_realloc, str)) {
return sprint_realloc_by_type(buf, buf_len, out_len, allow_realloc,
var, NULL, NULL, NULL);
} else {
return 0;
}
}
if (!ds_get_boolean(DS_LIBRARY_ID, DS_LIB_QUICK_PRINT)) {
u_char str[] = "IpAddress: ";
if (!snmp_strcat(buf, buf_len, out_len, allow_realloc, str)) {
return 0;
}
}
while ((*out_len + 17) >= *buf_len) {
if (!(allow_realloc && snmp_realloc(buf, buf_len))) {
return 0;
}
}
sprintf((char*)(*buf + *out_len), "%d.%d.%d.%d", ip[0], ip[1], ip[2], ip[3]);
*out_len += strlen((char*)(*buf + *out_len));
return 1;
}
void
sprint_null(char *buf,
struct variable_list *var,
struct enum_list *enums,
const char *hint,
const char *units)
{
if (var->type != ASN_NULL){
sprintf(buf, "Wrong Type (should be NULL): ");
buf += strlen(buf);
sprint_by_type(buf, var, NULL, NULL, NULL);
return;
}
sprintf(buf, "NULL");
}
int
sprint_realloc_null(u_char **buf, size_t *buf_len, size_t *out_len,
int allow_realloc,
struct variable_list *var,
struct enum_list *enums,
const char *hint,
const char *units)
{
if (var->type != ASN_NULL) {
u_char str[] = "Wrong Type (should be NULL): ";
if (snmp_strcat(buf, buf_len, out_len, allow_realloc, str)) {
return sprint_realloc_by_type(buf, buf_len, out_len, allow_realloc,
var, NULL, NULL, NULL);
} else {
return 0;
}
} else {
u_char str[] = "NULL";
return snmp_strcat(buf, buf_len, out_len, allow_realloc, str);
}
}
void
sprint_bitstring(char *buf,
struct variable_list *var,
struct enum_list *enums,
const char *hint,
const char *units)
{
int len, bit;
u_char *cp;
char *enum_string;
if (var->type != ASN_BIT_STR && var->type != ASN_OCTET_STR){
sprintf(buf, "Wrong Type (should be BITS): ");
buf += strlen(buf);
sprint_by_type(buf, var, NULL, NULL, NULL);
return;
}
if (ds_get_boolean(DS_LIBRARY_ID, DS_LIB_QUICK_PRINT)){
*buf++ = '"';
*buf = '\0';
} else {
sprintf(buf, "BITS: ");
buf += strlen(buf);
}
sprint_hexstring(buf, var->val.bitstring, var->val_len);
buf += strlen(buf);
if (ds_get_boolean(DS_LIBRARY_ID, DS_LIB_QUICK_PRINT)){
buf += strlen(buf);
*buf++ = '"';
*buf = '\0';
} else {
cp = var->val.bitstring;
for(len = 0; len < (int)var->val_len; len++){
for(bit = 0; bit < 8; bit++){
if (*cp & (0x80 >> bit)){
enum_string = NULL;
for (; enums; enums = enums->next)
if (enums->value == (len * 8) + bit){
enum_string = enums->label;
break;
}
if (enum_string == NULL ||
ds_get_boolean(DS_LIBRARY_ID,DS_LIB_PRINT_NUMERIC_ENUM))
sprintf(buf, "%d ", (len * 8) + bit);
else
sprintf(buf, "%s(%d) ", enum_string, (len * 8) + bit);
buf += strlen(buf);
}
}
cp ++;
}
}
}
int
sprint_realloc_bitstring(u_char **buf, size_t *buf_len, size_t *out_len,
int allow_realloc,
struct variable_list *var,
struct enum_list *enums,
const char *hint,
const char *units)
{
int len, bit;
u_char *cp;
char *enum_string;
if (var->type != ASN_BIT_STR && var->type != ASN_OCTET_STR) {
u_char str[] = "Wrong Type (should be BITS): ";
if (snmp_strcat(buf, buf_len, out_len, allow_realloc, str)) {
return sprint_realloc_by_type(buf, buf_len, out_len, allow_realloc,
var, NULL, NULL, NULL);
} else {
return 0;
}
}
if (ds_get_boolean(DS_LIBRARY_ID, DS_LIB_QUICK_PRINT)) {
u_char str[] = "\"";
if (!snmp_strcat(buf, buf_len, out_len, allow_realloc, str)) {
return 0;
}
} else {
u_char str[] = "BITS: ";
if (!snmp_strcat(buf, buf_len, out_len, allow_realloc, str)) {
return 0;
}
}
if (!sprint_realloc_hexstring(buf, buf_len, out_len, allow_realloc,
var->val.bitstring, var->val_len)) {
return 0;
}
if (ds_get_boolean(DS_LIBRARY_ID, DS_LIB_QUICK_PRINT)) {
u_char str[] = "\"";
if (!snmp_strcat(buf, buf_len, out_len, allow_realloc, str)) {
return 0;
}
} else {
cp = var->val.bitstring;
for(len = 0; len < (int)var->val_len; len++) {
for(bit = 0; bit < 8; bit++) {
if (*cp & (0x80 >> bit)) {
enum_string = NULL;
for (; enums; enums = enums->next) {
if (enums->value == (len * 8) + bit) {
enum_string = enums->label;
break;
}
}
if (enum_string == NULL ||
ds_get_boolean(DS_LIBRARY_ID, DS_LIB_PRINT_NUMERIC_ENUM)) {
char str[16];
sprintf(str, "%d ", (len * 8) + bit);
if (!snmp_strcat(buf, buf_len, out_len, allow_realloc, (const u_char*)str)) {
return 0;
}
} else {
char str[16];
sprintf(str, "(%d) ", (len * 8) + bit);
if (!snmp_strcat(buf, buf_len,out_len,allow_realloc,(const u_char*)enum_string)) {
return 0;
}
if (!snmp_strcat(buf, buf_len, out_len, allow_realloc, (const u_char*)str)) {
return 0;
}
}
}
}
cp++;
}
}
return 1;
}
void
sprint_nsapaddress(char *buf,
struct variable_list *var,
struct enum_list *enums,
const char *hint,
const char *units)
{
if (var->type != ASN_NSAP){
sprintf(buf, "Wrong Type (should be NsapAddress): ");
buf += strlen(buf);
sprint_by_type(buf, var, NULL, NULL, NULL);
return;
}
if (!ds_get_boolean(DS_LIBRARY_ID, DS_LIB_QUICK_PRINT)){
sprintf(buf, "NsapAddress: ");
buf += strlen(buf);
}
sprint_hexstring(buf, var->val.string, var->val_len);
}
int
sprint_realloc_nsapaddress(u_char **buf, size_t *buf_len, size_t *out_len,
int allow_realloc,
struct variable_list *var,
struct enum_list *enums,
const char *hint,
const char *units)
{
if (var->type != ASN_NSAP) {
u_char str[] = "Wrong Type (should be NsapAddress): ";
if (snmp_strcat(buf, buf_len, out_len, allow_realloc, str)) {
return sprint_realloc_by_type(buf, buf_len, out_len, allow_realloc,
var, NULL, NULL, NULL);
} else {
return 0;
}
}
if (!ds_get_boolean(DS_LIBRARY_ID, DS_LIB_QUICK_PRINT)) {
u_char str[] = "NsapAddress: ";
if (!snmp_strcat(buf, buf_len, out_len, allow_realloc, str)) {
return 0;
}
}
return sprint_realloc_hexstring(buf, buf_len, out_len, allow_realloc,
var->val.string, var->val_len);
}
void
sprint_counter64(char *buf,
struct variable_list *var,
struct enum_list *enums,
const char *hint,
const char *units)
{
char a64buf[I64CHARSZ+1];
if (var->type != ASN_COUNTER64
#ifdef OPAQUE_SPECIAL_TYPES
&& var->type != ASN_OPAQUE_COUNTER64
&& var->type != ASN_OPAQUE_I64
&& var->type != ASN_OPAQUE_U64
#endif
){
sprintf(buf, "Wrong Type (should be Counter64): ");
buf += strlen(buf);
sprint_by_type(buf, var, NULL, NULL, NULL);
return;
}
if (!ds_get_boolean(DS_LIBRARY_ID, DS_LIB_QUICK_PRINT)){
#ifdef OPAQUE_SPECIAL_TYPES
if (var->type != ASN_COUNTER64) {
sprintf(buf, "Opaque: ");
buf += strlen(buf);
}
#endif
#ifdef OPAQUE_SPECIAL_TYPES
switch(var->type) {
case ASN_OPAQUE_U64:
sprintf(buf, "UInt64: ");
break;
case ASN_OPAQUE_I64:
sprintf(buf, "Int64: ");
break;
case ASN_COUNTER64:
case ASN_OPAQUE_COUNTER64:
#endif
sprintf(buf, "Counter64: ");
#ifdef OPAQUE_SPECIAL_TYPES
}
#endif
buf += strlen(buf);
}
#ifdef OPAQUE_SPECIAL_TYPES
if (var->type == ASN_OPAQUE_I64)
{
printI64(a64buf, var->val.counter64);
sprintf(buf, a64buf);
}
else
#endif
{
printU64(a64buf, var->val.counter64);
sprintf(buf, a64buf);
}
buf += strlen (buf);
if (units) sprintf (buf, " %s", units);
}
void
sprint_unknowntype(char *buf,
struct variable_list *var,
struct enum_list *enums,
const char *hint,
const char *units)
{
/* sprintf(buf, "Variable has bad type"); */
sprint_by_type(buf, var, NULL, NULL, NULL);
}
void
sprint_badtype(char *buf,
struct variable_list *var,
struct enum_list *enums,
const char *hint,
const char *units)
{
sprintf(buf, "Variable has bad type");
}
int
sprint_realloc_badtype(u_char **buf, size_t *buf_len, size_t *out_len,
int allow_realloc,
struct variable_list *var,
struct enum_list *enums,
const char *hint,
const char *units)
{
u_char str[] = "Variable has bad type";
return snmp_strcat(buf, buf_len, out_len, allow_realloc, str);
}
void
sprint_by_type(char *buf,
struct variable_list *var,
struct enum_list *enums,
const char *hint,
const char *units)
{
switch (var->type){
case ASN_INTEGER:
sprint_integer(buf, var, enums, hint, units);
break;
case ASN_OCTET_STR:
sprint_octet_string(buf, var, enums, hint, units);
break;
case ASN_BIT_STR:
sprint_bitstring(buf, var, enums, hint, units);
break;
case ASN_OPAQUE:
sprint_opaque(buf, var, enums, hint, units);
break;
case ASN_OBJECT_ID:
sprint_object_identifier(buf, var, enums, hint, units);
break;
case ASN_TIMETICKS:
sprint_timeticks(buf, var, enums, hint, units);
break;
case ASN_GAUGE:
sprint_gauge(buf, var, enums, hint, units);
break;
case ASN_COUNTER:
sprint_counter(buf, var, enums, hint, units);
break;
case ASN_IPADDRESS:
sprint_ipaddress(buf, var, enums, hint, units);
break;
case ASN_NULL:
sprint_null(buf, var, enums, hint, units);
break;
case ASN_UINTEGER:
sprint_uinteger(buf, var, enums, hint, units);
break;
case ASN_COUNTER64:
#ifdef OPAQUE_SPECIAL_TYPES
case ASN_OPAQUE_U64:
case ASN_OPAQUE_I64:
case ASN_OPAQUE_COUNTER64:
#endif /* OPAQUE_SPECIAL_TYPES */
sprint_counter64(buf, var, enums, hint, units);
break;
#ifdef OPAQUE_SPECIAL_TYPES
case ASN_OPAQUE_FLOAT:
sprint_float(buf, var, enums, hint, units);
break;
case ASN_OPAQUE_DOUBLE:
sprint_double(buf, var, enums, hint, units);
break;
#endif /* OPAQUE_SPECIAL_TYPES */
default:
DEBUGMSGTL(("sprint_by_type", "bad type: %d\n", var->type));
sprint_badtype(buf, var, enums, hint, units);
break;
}
}
int
sprint_realloc_by_type(u_char **buf, size_t *buf_len, size_t *out_len,
int allow_realloc,
struct variable_list *var,
struct enum_list *enums,
const char *hint,
const char *units)
{
DEBUGMSGTL(("output", "sprint_by_type, type %d\n", var->type));
switch (var->type) {
case ASN_INTEGER:
return sprint_realloc_integer(buf, buf_len, out_len, allow_realloc,
var, enums, hint, units);
case ASN_OCTET_STR:
return sprint_realloc_octet_string(buf, buf_len, out_len, allow_realloc,
var, enums, hint, units);
case ASN_BIT_STR:
return sprint_realloc_bitstring(buf, buf_len, out_len, allow_realloc,
var, enums, hint, units);
case ASN_OPAQUE:
return sprint_realloc_opaque(buf, buf_len, out_len, allow_realloc,
var, enums, hint, units);
case ASN_OBJECT_ID:
return sprint_realloc_object_identifier(buf, buf_len, out_len,
allow_realloc, var, enums, hint, units);
case ASN_TIMETICKS:
return sprint_realloc_timeticks(buf, buf_len, out_len, allow_realloc,
var, enums, hint, units);
case ASN_GAUGE:
return sprint_realloc_gauge(buf, buf_len, out_len, allow_realloc,
var, enums, hint, units);
case ASN_COUNTER:
return sprint_realloc_counter(buf, buf_len, out_len, allow_realloc,
var, enums, hint, units);
case ASN_IPADDRESS:
return sprint_realloc_ipaddress(buf, buf_len, out_len, allow_realloc,
var, enums, hint, units);
case ASN_NULL:
return sprint_realloc_null(buf, buf_len, out_len, allow_realloc,
var, enums, hint, units);
case ASN_UINTEGER:
return sprint_realloc_uinteger(buf, buf_len, out_len, allow_realloc,
var, enums, hint, units);
case ASN_COUNTER64:
#ifdef OPAQUE_SPECIAL_TYPES
case ASN_OPAQUE_U64:
case ASN_OPAQUE_I64:
case ASN_OPAQUE_COUNTER64:
#endif /* OPAQUE_SPECIAL_TYPES */
return sprint_realloc_counter64(buf, buf_len, out_len, allow_realloc,
var, enums, hint, units);
#ifdef OPAQUE_SPECIAL_TYPES
case ASN_OPAQUE_FLOAT:
return sprint_realloc_float(buf, buf_len, out_len, allow_realloc,
var, enums, hint, units);
case ASN_OPAQUE_DOUBLE:
return sprint_realloc_double(buf, buf_len, out_len, allow_realloc,
var, enums, hint, units);
#endif /* OPAQUE_SPECIAL_TYPES */
default:
DEBUGMSGTL(("sprint_by_type", "bad type: %d\n", var->type));
return sprint_realloc_badtype(buf, buf_len, out_len, allow_realloc,
var, enums, hint, units);
}
}
struct tree *get_tree_head(void)
{
return(tree_head);
}
static char *confmibdir=NULL;
static char *confmibs=NULL;
static void
handle_mibdirs_conf(const char *token,
char *line)
{
char *ctmp;
if (confmibdir) {
ctmp = (char *)malloc(strlen(confmibdir) + strlen(line) + 1);
if (*line == '+')
line++;
sprintf(ctmp,"%s%c%s",confmibdir, ENV_SEPARATOR_CHAR, line);
free(confmibdir);
confmibdir = ctmp;
} else {
confmibdir=strdup(line);
}
DEBUGMSGTL(("read_config:initmib", "using mibdirs: %s\n", confmibdir));
}
static void
handle_mibs_conf(const char *token,
char *line)
{
char *ctmp;
if (confmibs) {
ctmp = (char *)malloc(strlen(confmibs) + strlen(line) + 1);
if (*line == '+')
line++;
sprintf(ctmp,"%s%c%s",confmibs, ENV_SEPARATOR_CHAR, line);
free(confmibs);
confmibs = ctmp;
} else {
confmibs=strdup(line);
}
DEBUGMSGTL(("read_config:initmib", "using mibs: %s\n", confmibs));
}
static void
handle_mibfile_conf(const char *token,
char *line)
{
DEBUGMSGTL(("read_config:initmib", "reading mibfile: %s\n", line));
read_mib(line);
}
char *
snmp_out_toggle_options(char *options)
{
while(*options) {
switch(*options++) {
case 'n':
ds_toggle_boolean(DS_LIBRARY_ID, DS_LIB_PRINT_NUMERIC_OIDS);
break;
case 'e':
ds_toggle_boolean(DS_LIBRARY_ID, DS_LIB_PRINT_NUMERIC_ENUM);
break;
case 'b':
ds_toggle_boolean(DS_LIBRARY_ID, DS_LIB_DONT_BREAKDOWN_OIDS);
break;
case 'E':
ds_toggle_boolean(DS_LIBRARY_ID, DS_LIB_ESCAPE_QUOTES);
break;
case 'X':
ds_toggle_boolean(DS_LIBRARY_ID, DS_LIB_EXTENDED_INDEX);
break;
case 'q':
ds_toggle_boolean(DS_LIBRARY_ID, DS_LIB_QUICK_PRINT);
break;
case 'f':
ds_toggle_boolean(DS_LIBRARY_ID, DS_LIB_PRINT_FULL_OID);
break;
case 't':
ds_toggle_boolean(DS_LIBRARY_ID, DS_LIB_NUMERIC_TIMETICKS);
break;
case 'v':
ds_toggle_boolean(DS_LIBRARY_ID, DS_LIB_PRINT_BARE_VALUE);
break;
case 's':
snmp_set_suffix_only(1);
break;
case 'S':
snmp_set_suffix_only(2);
break;
case 'T':
ds_toggle_boolean(DS_LIBRARY_ID, DS_LIB_PRINT_HEX_TEXT);
break;
default:
return options-1;
}
}
return NULL;
}
void snmp_out_toggle_options_usage(const char *lead, FILE *outf)
{
fprintf(outf, "%sOUTOPTS values:\n", lead);
fprintf(outf, "%s n: Print oids numerically.\n", lead);
fprintf(outf, "%s e: Print enums numerically.\n", lead);
fprintf(outf, "%s E: Escape quotes in string indices.\n", lead);
fprintf(outf, "%s X: Extended index format\n", lead);
fprintf(outf, "%s b: Dont break oid indexes down.\n", lead);
fprintf(outf, "%s q: Quick print for easier parsing.\n", lead);
fprintf(outf, "%s f: Print full oids on output.\n", lead);
fprintf(outf, "%s s: Print only last symbolic element of oid.\n", lead);
fprintf(outf, "%s S: Print MIB module-id plus last element.\n", lead);
fprintf(outf, "%s t: Print timeticks unparsed as numeric integers.\n", lead);
fprintf(outf, "%s v: Print Print values only (not OID = value).\n", lead);
fprintf(outf, "%s T: Print human-readable text along with hex strings.\n", lead);
}
char *
snmp_in_toggle_options(char *options)
{
while(*options) {
switch(*options++) {
case 'R':
ds_toggle_boolean(DS_LIBRARY_ID, DS_LIB_RANDOM_ACCESS);
break;
case 'b':
ds_toggle_boolean(DS_LIBRARY_ID, DS_LIB_REGEX_ACCESS);
break;
case 'r':
ds_toggle_boolean(DS_LIBRARY_ID, DS_LIB_DONT_CHECK_RANGE);
break;
default:
return options-1;
}
}
return NULL;
}
void snmp_in_toggle_options_usage(const char *lead, FILE *outf)
{
fprintf(outf, "%sINOPTS values:\n", lead);
fprintf(outf, "%s R: Do random access to oid labels.\n", lead);
fprintf(outf, "%s r: Don't check values for range/type legality.\n", lead);
fprintf(outf, "%s b: Do best/regex matching to find a MIB node.\n", lead);
}
void
register_mib_handlers (void)
{
register_premib_handler("snmp","mibdirs",
handle_mibdirs_conf, NULL,
"[mib-dirs|+mib-dirs]");
register_premib_handler("snmp","mibs",
handle_mibs_conf,NULL,
"[mib-tokens|+mib-tokens]");
register_config_handler("snmp","mibfile",
handle_mibfile_conf, NULL,
"mibfile-to-read");
/* register the snmp.conf configuration handlers for default
parsing behaviour */
ds_register_premib(ASN_BOOLEAN, "snmp","showMibErrors",
DS_LIBRARY_ID, DS_LIB_MIB_ERRORS);
ds_register_premib(ASN_BOOLEAN, "snmp","strictCommentTerm",
DS_LIBRARY_ID, DS_LIB_MIB_COMMENT_TERM);
ds_register_premib(ASN_BOOLEAN, "snmp","mibAllowUnderline",
DS_LIBRARY_ID, DS_LIB_MIB_PARSE_LABEL);
ds_register_premib(ASN_INTEGER, "snmp","mibWarningLevel",
DS_LIBRARY_ID, DS_LIB_MIB_WARNINGS);
ds_register_premib(ASN_BOOLEAN, "snmp","mibReplaceWithLatest",
DS_LIBRARY_ID, DS_LIB_MIB_REPLACE);
ds_register_premib(ASN_BOOLEAN, "snmp","printNumericEnums",
DS_LIBRARY_ID, DS_LIB_PRINT_NUMERIC_ENUM);
ds_register_premib(ASN_BOOLEAN, "snmp","printNumericOids",
DS_LIBRARY_ID, DS_LIB_PRINT_NUMERIC_OIDS);
ds_register_premib(ASN_BOOLEAN, "snmp","escapeQuotes",
DS_LIBRARY_ID, DS_LIB_ESCAPE_QUOTES);
ds_register_premib(ASN_BOOLEAN, "snmp","dontBreakdownOids",
DS_LIBRARY_ID, DS_LIB_DONT_BREAKDOWN_OIDS);
ds_register_premib(ASN_BOOLEAN, "snmp","quickPrinting",
DS_LIBRARY_ID, DS_LIB_QUICK_PRINT);
ds_register_premib(ASN_BOOLEAN, "snmp","numericTimeticks",
DS_LIBRARY_ID, DS_LIB_NUMERIC_TIMETICKS);
ds_register_premib(ASN_INTEGER, "snmp","suffixPrinting",
DS_LIBRARY_ID, DS_LIB_PRINT_SUFFIX_ONLY);
ds_register_premib(ASN_BOOLEAN, "snmp","extendedIndex",
DS_LIBRARY_ID, DS_LIB_EXTENDED_INDEX);
ds_register_premib(ASN_BOOLEAN, "snmp","printHexText",
DS_LIBRARY_ID, DS_LIB_PRINT_HEX_TEXT);
}
void
init_mib (void)
{
const char *prefix;
char *env_var, *entry;
PrefixListPtr pp = &mib_prefixes[0];
char *new_mibdirs, *homepath, *cp_home;
if (Mib) return;
init_mib_internals();
/* Initialise the MIB directory/ies */
/* we can't use the environment variable directly, because strtok
will modify it. */
env_var = getenv("MIBDIRS");
if ( env_var == NULL ) {
if (confmibdir != NULL)
env_var = strdup(confmibdir);
else
env_var = strdup(DEFAULT_MIBDIRS);
} else {
env_var = strdup(env_var);
}
if (*env_var == '+') {
entry = (char *)malloc(strlen(DEFAULT_MIBDIRS)+strlen(env_var)+2);
sprintf(entry, "%s%c%s", DEFAULT_MIBDIRS, ENV_SEPARATOR_CHAR, env_var+1);
free(env_var);
env_var = entry;
}
/* replace $HOME in the path with the users home directory */
homepath=getenv("HOME");
if (homepath) {
while((cp_home = strstr(env_var, "$HOME"))) {
new_mibdirs = (char *) malloc(strlen(env_var) - strlen("$HOME") +
strlen(homepath)+1);
*cp_home = 0; /* null out the spot where we stop copying */
sprintf(new_mibdirs, "%s%s%s", env_var, homepath,
cp_home + strlen("$HOME"));
/* swap in the new value and repeat */
free(env_var);
env_var = new_mibdirs;
}
}
DEBUGMSGTL(("init_mib","Seen MIBDIRS: Looking in '%s' for mib dirs ...\n",env_var));
entry = strtok( env_var, ENV_SEPARATOR );
while ( entry ) {
add_mibdir(entry);
entry = strtok( NULL, ENV_SEPARATOR);
}
free(env_var);
init_mib_internals();
/* Read in any modules or mibs requested */
env_var = getenv("MIBS");
if ( env_var == NULL ) {
if (confmibs != NULL)
env_var = strdup(confmibs);
else
env_var = strdup(DEFAULT_MIBS);
} else {
env_var = strdup(env_var);
}
if (*env_var == '+') {
entry = (char *)malloc(strlen(DEFAULT_MIBS)+strlen(env_var)+2);
sprintf(entry, "%s%c%s", DEFAULT_MIBS, ENV_SEPARATOR_CHAR, env_var+1);
free(env_var);
env_var = entry;
}
DEBUGMSGTL(("init_mib","Seen MIBS: Looking in '%s' for mib files ...\n",env_var));
entry = strtok( env_var, ENV_SEPARATOR );
while ( entry ) {
if (strcasecmp(entry, DEBUG_ALWAYS_TOKEN) == 0) {
read_all_mibs();
}
else if (strstr (entry, "/") != 0) {
read_mib(entry);
}
else {
read_module(entry);
}
entry = strtok( NULL, ENV_SEPARATOR);
}
adopt_orphans();
free(env_var);
env_var = getenv("MIBFILES");
if ( env_var != NULL ) {
if (*env_var == '+') {
#ifdef DEFAULT_MIBFILES
entry = (char *)malloc(strlen(DEFAULT_MIBFILES)+strlen(env_var)+2);
sprintf(entry, "%s%c%s", DEFAULT_MIBFILES, ENV_SEPARATOR_CHAR,
env_var+1);
free(env_var);
env_var = entry;
#else
env_var = strdup(env_var+1);
#endif
} else {
env_var = strdup(env_var);
}
} else {
#ifdef DEFAULT_MIBFILES
env_var = strdup(DEFAULT_MIBFILES);
#endif
}
if ( env_var != 0 ) {
DEBUGMSGTL(("init_mib","Seen MIBFILES: Looking in '%s' for mib files ...\n",env_var));
entry = strtok( env_var, ENV_SEPARATOR );
while ( entry ) {
read_mib(entry);
entry = strtok( NULL, ENV_SEPARATOR);
}
free(env_var);
}
prefix = getenv("PREFIX");
if (!prefix)
prefix = Standard_Prefix;
Prefix = (char*)malloc(strlen(prefix)+2);
strcpy(Prefix, prefix);
DEBUGMSGTL(("init_mib","Seen PREFIX: Looking in '%s' for prefix ...\n", Prefix));
/* remove trailing dot */
env_var = &Prefix[strlen(Prefix) - 1];
if (*env_var == '.') *env_var = '\0';
pp->str = Prefix; /* fixup first mib_prefix entry */
/* now that the list of prefixes is built, save each string length. */
while (pp->str) {
pp->len = strlen(pp->str);
pp++;
}
if (getenv("SUFFIX"))
ds_set_boolean(DS_LIBRARY_ID, DS_LIB_PRINT_SUFFIX_ONLY, 1);
Mib = tree_head; /* Backwards compatibility */
tree_top = (struct tree *)calloc(1,sizeof(struct tree));
/* XX error check ? */
if (tree_top) {
tree_top->label = strdup("(top)");
tree_top->child_list = tree_head;
}
}
void
shutdown_mib (void)
{
unload_all_mibs();
if (tree_top) {
if (tree_top->label) free(tree_top->label);
free(tree_top); tree_top = NULL;
}
tree_head = NULL;
Mib = NULL;
if (Prefix != NULL && Prefix != &Standard_Prefix[0])
free(Prefix);
if (Prefix)
Prefix = NULL;
}
void
print_mib (FILE *fp)
{
print_subtree(fp, tree_head, 0);
}
void
print_ascii_dump (FILE *fp)
{
fprintf(fp, "dump DEFINITIONS ::= BEGIN\n");
print_ascii_dump_tree(fp, tree_head, 0);
fprintf(fp, "END\n");
}
void
set_function(struct tree *subtree)
{
switch(subtree->type) {
case TYPE_OBJID:
subtree->printer = sprint_object_identifier;
subtree->printomat = sprint_realloc_object_identifier;
break;
case TYPE_OCTETSTR:
subtree->printer = sprint_octet_string;
subtree->printomat = sprint_realloc_octet_string;
break;
case TYPE_INTEGER:
subtree->printer = sprint_integer;
subtree->printomat = sprint_realloc_integer;
break;
case TYPE_INTEGER32:
subtree->printer = sprint_integer;
subtree->printomat = sprint_realloc_integer;
break;
case TYPE_NETADDR:
subtree->printer = sprint_networkaddress;
subtree->printomat = sprint_realloc_networkaddress;
break;
case TYPE_IPADDR:
subtree->printer = sprint_ipaddress;
subtree->printomat = sprint_realloc_ipaddress;
break;
case TYPE_COUNTER:
subtree->printer = sprint_counter;
subtree->printomat = sprint_realloc_counter;
break;
case TYPE_GAUGE:
subtree->printer = sprint_gauge;
subtree->printomat = sprint_realloc_gauge;
break;
case TYPE_TIMETICKS:
subtree->printer = sprint_timeticks;
subtree->printomat = sprint_realloc_timeticks;
break;
case TYPE_OPAQUE:
subtree->printer = sprint_opaque;
subtree->printomat = sprint_realloc_opaque;
break;
case TYPE_NULL:
subtree->printer = sprint_null;
subtree->printomat = sprint_realloc_null;
break;
case TYPE_BITSTRING:
subtree->printer = sprint_bitstring;
subtree->printomat = sprint_realloc_bitstring;
break;
case TYPE_NSAPADDRESS:
subtree->printer = sprint_nsapaddress;
subtree->printomat = sprint_realloc_nsapaddress;
break;
case TYPE_COUNTER64:
subtree->printer = sprint_counter64;
subtree->printomat = sprint_realloc_counter64;
break;
case TYPE_UINTEGER:
subtree->printer = sprint_uinteger;
subtree->printomat = sprint_realloc_uinteger;
break;
case TYPE_UNSIGNED32:
subtree->printer = sprint_gauge;
subtree->printomat = sprint_realloc_gauge;
break;
case TYPE_OTHER:
default:
subtree->printer = sprint_unknowntype;
subtree->printomat = sprint_realloc_by_type;
break;
}
}
/*
* Read an object identifier from input string into internal OID form.
* Returns 1 if successful.
* If an error occurs, this function returns 0 and MAY set snmp_errno.
* snmp_errno is NOT set if SET_SNMP_ERROR evaluates to nothing.
* This can make multi-threaded use a tiny bit more robust.
*/
int read_objid(const char *input,
oid *output,
size_t *out_len) /* number of subid's in "output" */
{
struct tree *root = tree_top;
char buf[SPRINT_MAX_LEN];
int ret, max_out_len;
char *name, ch;
const char *cp;
cp = input;
while ((ch = *cp))
if (('0' <= ch && ch <= '9')
|| ('a' <= ch && ch <= 'z')
|| ('A' <= ch && ch <= 'Z')
|| ch == '-')
cp++;
else
break;
if (ch == ':')
return get_node(input, output, out_len);
if (*input == '.')
input++;
else {
/* get past leading '.', append '.' to Prefix. */
if (*Prefix == '.')
strcpy(buf, Prefix+1);
else
strcpy(buf, Prefix);
strcat(buf, ".");
strcat(buf, input);
input = buf;
}
if (root == NULL){
SET_SNMP_ERROR(SNMPERR_NOMIB);
*out_len = 0;
return 0;
}
name = strdup(input);
max_out_len = *out_len;
*out_len = 0;
if ((ret = _add_strings_to_oid(root, name, output, out_len, max_out_len)) <= 0)
{
if (ret == 0) ret = SNMPERR_UNKNOWN_OBJID;
SET_SNMP_ERROR(ret);
free(name);
return 0;
}
free(name);
return 1;
}
static struct tree *
_sprint_objid(char *buf,
oid *objid,
size_t objidlen) /* number of subidentifiers */
{
char tempbuf[SPRINT_MAX_LEN], *cp;
struct tree *subtree = tree_head;
char *midpoint = 0;
*tempbuf = '.'; /* this is a fully qualified name */
subtree = _get_symbol(objid, objidlen, subtree, tempbuf + 1, 0, &midpoint);
if (ds_get_boolean(DS_LIBRARY_ID,DS_LIB_PRINT_NUMERIC_OIDS)) {
cp = tempbuf;
} else if (ds_get_int(DS_LIBRARY_ID, DS_LIB_PRINT_SUFFIX_ONLY)){
for(cp = tempbuf; *cp; cp++)
;
if (midpoint)
cp = midpoint-2; /* beyond the '.' */
else {
while(cp >= tempbuf){
if (isalpha(*cp))
break;
cp--;
}
}
while(cp >= tempbuf){
if (*cp == '.')
break;
cp--;
}
cp++;
if (ds_get_int(DS_LIBRARY_ID, DS_LIB_PRINT_SUFFIX_ONLY) == 2 && cp > tempbuf) {
char modbuf[256];
char *mod = module_name(subtree->modid, modbuf);
size_t len = strlen(mod);
if ((int)len+1 >= cp-tempbuf) {
memmove(tempbuf+len+2, cp, strlen(cp)+1);
cp = tempbuf+len+2;
}
cp -= len+2;
memcpy(cp, mod, len);
cp[len] = ':';
cp[len+1] = ':';
}
}
else if (!ds_get_boolean(DS_LIBRARY_ID, DS_LIB_PRINT_FULL_OID)) {
PrefixListPtr pp = &mib_prefixes[0];
int ii;
size_t ilen, tlen;
const char *testcp;
cp = tempbuf; tlen = strlen(tempbuf);
ii = 0;
while (pp->str) {
ilen = pp->len; testcp = pp->str;
if ((tlen > ilen) && !memcmp(tempbuf, testcp, ilen)) {
cp += (ilen + 1);
break;
}
pp++;
}
}
else cp = tempbuf;
strcpy(buf, cp);
return subtree;
}
static struct tree *
_sprint_realloc_objid(u_char **buf, size_t *buf_len,
size_t *out_len, int allow_realloc, int *buf_overflow,
oid *objid, size_t objidlen)
{
u_char *tbuf = NULL, *cp = NULL;
size_t tbuf_len = 256, tout_len = 0;
struct tree *subtree = tree_head;
size_t midpoint_offset = 0;
int tbuf_overflow = 0;
if ((tbuf = (u_char *)malloc(tbuf_len)) == NULL) {
tbuf_overflow = 1;
} else {
*tbuf = '.';
tout_len = 1;
}
subtree = _get_realloc_symbol(objid, objidlen, subtree,
&tbuf, &tbuf_len, &tout_len, allow_realloc,
&tbuf_overflow, NULL, &midpoint_offset);
if (tbuf_overflow) {
if (!*buf_overflow) {
snmp_strcat(buf, buf_len, out_len, allow_realloc, tbuf);
*buf_overflow = 1;
}
free(tbuf);
return subtree;
}
if (ds_get_boolean(DS_LIBRARY_ID, DS_LIB_PRINT_NUMERIC_OIDS)) {
cp = tbuf;
} else if (ds_get_int(DS_LIBRARY_ID, DS_LIB_PRINT_SUFFIX_ONLY)) {
for (cp = tbuf; *cp; cp++);
if (midpoint_offset != 0) {
cp = tbuf + midpoint_offset - 2; /* beyond the '.' */
} else {
while (cp >= tbuf) {
if (isalpha(*cp)) {
break;
}
cp--;
}
}
while(cp >= tbuf) {
if (*cp == '.') {
break;
}
cp--;
}
cp++;
if (ds_get_int(DS_LIBRARY_ID, DS_LIB_PRINT_SUFFIX_ONLY) == 2 && cp > tbuf){
char modbuf[256], *mod = module_name(subtree->modid, modbuf);
if (!*buf_overflow) {
if (!snmp_strcat(buf, buf_len, out_len, allow_realloc, (const u_char*)mod) ||
!snmp_strcat(buf, buf_len, out_len, allow_realloc, (const u_char*)"::")) {
*buf_overflow = 1;
}
}
}
} else if (!ds_get_boolean(DS_LIBRARY_ID, DS_LIB_PRINT_FULL_OID)) {
PrefixListPtr pp = &mib_prefixes[0];
size_t ilen, tlen;
const char *testcp;
cp = tbuf;
tlen = strlen((char*)tbuf);
while (pp->str) {
ilen = pp->len;
testcp = pp->str;
if ((tlen > ilen) && memcmp(tbuf, testcp, ilen) == 0) {
cp += (ilen + 1);
break;
}
pp++;
}
} else {
cp = tbuf;
}
if (!*buf_overflow &&
!snmp_strcat(buf, buf_len, out_len, allow_realloc, cp)) {
*buf_overflow = 1;
}
free(tbuf);
return subtree;
}
int
sprint_realloc_objid(u_char **buf, size_t *buf_len,
size_t *out_len, int allow_realloc,
oid *objid, size_t objidlen)
{
int buf_overflow = 0;
_sprint_realloc_objid(buf, buf_len, out_len, allow_realloc, &buf_overflow,
objid, objidlen);
return !buf_overflow;
}
char * sprint_objid(char *buf, oid *objid, size_t objidlen)
{
_sprint_objid(buf,objid,objidlen);
return buf;
}
void
print_objid(oid *objid,
size_t objidlen) /* number of subidentifiers */
{
fprint_objid(stdout, objid, objidlen);
}
void
fprint_objid(FILE *f,
oid *objid,
size_t objidlen) /* number of subidentifiers */
{
u_char *buf = NULL;
size_t buf_len = 256, out_len = 0;
int buf_overflow = 0;
if ((buf = (u_char *)malloc(buf_len)) == NULL) {
fprintf(f, "[TRUNCATED]\n");
return;
} else {
_sprint_realloc_objid(&buf, &buf_len, &out_len, 1, &buf_overflow,
objid, objidlen);
if (buf_overflow) {
fprintf(f, "%s [TRUNCATED]\n", buf);
} else {
fprintf(f, "%s\n", buf);
}
}
free(buf);
}
void
sprint_variable(char *buf,
oid *objid,
size_t objidlen,
struct variable_list *variable)
{
struct tree *subtree;
subtree = _sprint_objid(buf, objid, objidlen);
if (!ds_get_boolean(DS_LIBRARY_ID, DS_LIB_PRINT_BARE_VALUE)) {
buf += strlen(buf);
if (ds_get_boolean(DS_LIBRARY_ID, DS_LIB_QUICK_PRINT))
strcat(buf, " ");
else
strcat(buf, " = ");
buf += strlen(buf);
}
if (variable->type == SNMP_NOSUCHOBJECT)
strcpy(buf, "No Such Object available on this agent");
else if (variable->type == SNMP_NOSUCHINSTANCE)
strcpy(buf, "No Such Instance currently exists");
else if (variable->type == SNMP_ENDOFMIBVIEW)
strcpy(buf, "No more variables left in this MIB View");
else if (subtree) {
if (subtree->printer)
(*subtree->printer)(buf, variable, subtree->enums, subtree->hint, subtree->units);
else {
sprint_by_type(buf, variable, subtree->enums, subtree->hint, subtree->units);
}
}
else { /* handle rare case where tree is empty */
sprint_by_type(buf, variable, 0, 0, 0);
}
}
int
sprint_realloc_variable(u_char **buf, size_t *buf_len,
size_t *out_len, int allow_realloc,
oid *objid, size_t objidlen,
struct variable_list *variable)
{
struct tree *subtree = tree_head;
int buf_overflow = 0;
subtree = _sprint_realloc_objid(buf, buf_len, out_len, allow_realloc,
&buf_overflow, objid, objidlen);
if (buf_overflow) {
return 0;
}
if (!ds_get_boolean(DS_LIBRARY_ID, DS_LIB_PRINT_BARE_VALUE)) {
if (ds_get_boolean(DS_LIBRARY_ID, DS_LIB_QUICK_PRINT)) {
if (!snmp_strcat(buf, buf_len, out_len, allow_realloc, (const u_char*)" ")) {
return 0;
}
} else {
if (!snmp_strcat(buf, buf_len, out_len, allow_realloc, (const u_char*)" = ")) {
return 0;
}
}
} else {
*out_len = 0;
}
if (variable->type == SNMP_NOSUCHOBJECT) {
return snmp_strcat(buf, buf_len, out_len, allow_realloc,
(const u_char*)"No Such Object available on this agent");
} else if (variable->type == SNMP_NOSUCHINSTANCE) {
return snmp_strcat(buf, buf_len, out_len, allow_realloc,
(const u_char*)"No Such Instance currently exists");
} else if (variable->type == SNMP_ENDOFMIBVIEW) {
return snmp_strcat(buf, buf_len, out_len, allow_realloc,
(const u_char*)"No more variables left in this MIB View");
} else if (subtree) {
if (subtree->printomat) {
return (*subtree->printomat)(buf, buf_len, out_len, allow_realloc,
variable, subtree->enums,
subtree->hint, subtree->units);
} else {
return sprint_realloc_by_type(buf, buf_len, out_len, allow_realloc,
variable, subtree->enums,
subtree->hint, subtree->units);
}
} else {
/* Handle rare case where tree is empty. */
return sprint_realloc_by_type(buf, buf_len, out_len, allow_realloc,
variable, 0, 0, 0);
}
}
void
print_variable(oid *objid,
size_t objidlen,
struct variable_list *variable)
{
fprint_variable(stdout, objid, objidlen, variable);
}
void
fprint_variable(FILE *f,
oid *objid,
size_t objidlen,
struct variable_list *variable)
{
u_char *buf = NULL;
size_t buf_len = 256, out_len = 0;
if ((buf = (u_char *)malloc(buf_len)) == NULL) {
fprintf(f, "[TRUNCATED]\n");
return;
} else {
if (sprint_realloc_variable(&buf, &buf_len, &out_len, 1,
objid, objidlen, variable)) {
fprintf(f, "%s\n", buf);
} else {
fprintf(f, "%s [TRUNCATED]\n", buf);
}
}
free(buf);
}
void
sprint_value(char *buf,
oid *objid,
size_t objidlen,
struct variable_list *variable)
{
char tempbuf[SPRINT_MAX_LEN];
struct tree *subtree = tree_head;
if (variable->type == SNMP_NOSUCHOBJECT)
sprintf(buf, "No Such Object available on this agent");
else if (variable->type == SNMP_NOSUCHINSTANCE)
sprintf(buf, "No Such Instance currently exists");
else if (variable->type == SNMP_ENDOFMIBVIEW)
sprintf(buf, "No more variables left in this MIB View");
else {
subtree = get_symbol(objid, objidlen, subtree, tempbuf);
if (subtree->printer)
(*subtree->printer)(buf, variable, subtree->enums, subtree->hint, subtree->units);
else {
sprint_by_type(buf, variable, subtree->enums, subtree->hint, subtree->units);
}
}
}
int
sprint_realloc_value(u_char **buf, size_t *buf_len,
size_t *out_len, int allow_realloc,
oid *objid, size_t objidlen,
struct variable_list *variable)
{
char tempbuf[SPRINT_MAX_LEN];
struct tree *subtree = tree_head;
if (variable->type == SNMP_NOSUCHOBJECT) {
return snmp_strcat(buf, buf_len, out_len, allow_realloc,
(const u_char*)"No Such Object available on this agent");
} else if (variable->type == SNMP_NOSUCHINSTANCE) {
return snmp_strcat(buf, buf_len, out_len, allow_realloc,
(const u_char*)"No Such Instance currently exists");
} else if (variable->type == SNMP_ENDOFMIBVIEW) {
return snmp_strcat(buf, buf_len, out_len, allow_realloc,
(const u_char*)"No more variables left in this MIB View");
} else {
subtree = get_symbol(objid, objidlen, subtree, tempbuf);
if (subtree && subtree->printomat) {
return (*subtree->printomat)(buf, buf_len, out_len, allow_realloc,
variable, subtree->enums,
subtree->hint, subtree->units);
} else {
return sprint_realloc_by_type(buf, buf_len, out_len, allow_realloc,
variable, subtree->enums,
subtree->hint, subtree->units);
}
}
}
void
print_value(oid *objid,
size_t objidlen,
struct variable_list *variable)
{
fprint_value(stdout, objid, objidlen, variable);
}
void
fprint_value(FILE *f,
oid *objid,
size_t objidlen,
struct variable_list *variable)
{
u_char *buf = NULL;
size_t buf_len = 256, out_len = 0;
if ((buf = (u_char *)malloc(buf_len)) == NULL) {
fprintf(f, "[TRUNCATED]\n");
return;
} else {
if (sprint_realloc_value(&buf, &buf_len, &out_len, 1,
objid, objidlen, variable)) {
fprintf(f, "%s\n", buf);
} else {
fprintf(f, "%s [TRUNCATED]\n", buf);
}
}
free(buf);
}
/*
* Append a quoted printable string to buffer "buf"
* that represents a range of sub-identifiers "objid".
*
* Display '.' for all non-printable sub-identifiers.
* If successful, "buf" points past the appended string.
*/
static char *
dump_oid_to_string(oid *objid,
size_t objidlen,
char *buf,
char quotechar)
{
if (buf)
{ int ii, alen;
char *scp;
char *cp = buf + (strlen(buf));
scp = cp;
for (ii= 0, alen = 0; ii < (int)objidlen; ii++)
{
oid tst = objid[ii];
if ((tst > 254) || (!isprint(tst)))
tst = (oid)'.';
if (alen == 0) {
if (ds_get_boolean(DS_LIBRARY_ID,DS_LIB_ESCAPE_QUOTES))
*cp++ = '\\';
*cp++ = quotechar;
}
*cp++ = (char)tst;
alen++;
}
if (alen) {
if (ds_get_boolean(DS_LIBRARY_ID,DS_LIB_ESCAPE_QUOTES))
*cp++ = '\\';
*cp++ = quotechar;
}
*cp = '\0';
buf = cp;
}
return buf;
}
int
dump_realloc_oid_to_string(oid *objid, size_t objidlen,
u_char **buf, size_t *buf_len, size_t *out_len,
int allow_realloc, char quotechar)
{
if (buf) {
int i, alen;
for (i = 0, alen = 0; i < (int)objidlen; i++) {
oid tst = objid[i];
if ((tst > 254) || (!isprint(tst))) {
tst = (oid)'.';
}
if (alen == 0) {
if (ds_get_boolean(DS_LIBRARY_ID, DS_LIB_ESCAPE_QUOTES)) {
while ((*out_len + 2) >= *buf_len) {
if (!(allow_realloc && snmp_realloc(buf, buf_len))) {
return 0;
}
}
*(*buf + *out_len) = '\\';
(*out_len)++;
}
while ((*out_len + 2) >= *buf_len) {
if (!(allow_realloc && snmp_realloc(buf, buf_len))) {
return 0;
}
}
*(*buf + *out_len) = quotechar;
(*out_len)++;
}
while ((*out_len + 2) >= *buf_len) {
if (!(allow_realloc && snmp_realloc(buf, buf_len))) {
return 0;
}
}
*(*buf + *out_len) = (char)tst;
(*out_len)++;
alen++;
}
if (alen) {
if (ds_get_boolean(DS_LIBRARY_ID, DS_LIB_ESCAPE_QUOTES)) {
while ((*out_len + 2) >= *buf_len) {
if (!(allow_realloc && snmp_realloc(buf, buf_len))) {
return 0;
}
}
*(*buf + *out_len) = '\\';
(*out_len)++;
}
while ((*out_len + 2) >= *buf_len) {
if (!(allow_realloc && snmp_realloc(buf, buf_len))) {
return 0;
}
}
*(*buf + *out_len) = quotechar;
(*out_len)++;
}
*(*buf + *out_len) = '\0';
}
return 1;
}
static struct tree *
_get_symbol(oid *objid,
size_t objidlen,
struct tree *subtree,
char *buf,
struct index_list *in_dices,
char **end_of_known)
{
struct tree *return_tree = NULL;
int extended_index = ds_get_boolean(DS_LIBRARY_ID, DS_LIB_EXTENDED_INDEX);
if (!objid || !buf)
return NULL;
for(; subtree; subtree = subtree->next_peer){
if (*objid == subtree->subid){
if (subtree->indexes)
in_dices = subtree->indexes;
else if (subtree->augments) {
struct tree *tp2 = find_tree_node(subtree->augments, -1);
if (tp2) in_dices = tp2->indexes;
}
if (!strncmp( subtree->label, ANON, ANON_LEN) ||
ds_get_boolean(DS_LIBRARY_ID, DS_LIB_PRINT_NUMERIC_OIDS))
sprintf(buf, "%lu", subtree->subid);
else
strcpy(buf, subtree->label);
if (objidlen > 1){
while(*buf)
buf++;
*buf++ = '.';
*buf = '\0';
return_tree = _get_symbol(objid + 1, objidlen - 1, subtree->child_list,
buf, in_dices, end_of_known);
}
if (return_tree != NULL)
return return_tree;
else
return subtree;
}
}
if (end_of_known)
*end_of_known = buf;
/* subtree not found */
while (in_dices && (objidlen > 0) &&
!ds_get_boolean(DS_LIBRARY_ID, DS_LIB_PRINT_NUMERIC_OIDS) &&
!ds_get_boolean(DS_LIBRARY_ID, DS_LIB_DONT_BREAKDOWN_OIDS)) {
size_t numids;
struct tree *tp;
tp = find_tree_node(in_dices->ilabel, -1);
if (!tp) {
/* ack. Can't find an index in the mib tree. bail */
goto finish_it;
}
if (extended_index) {
if (buf[-1] == '.') buf--;
*buf++ = '[';
*buf = 0;
}
switch(tp->type) {
case TYPE_OCTETSTR:
if (extended_index && tp->hint) {
struct variable_list var;
u_char buffer[1024];
int i;
memset(&var, 0, sizeof var);
if (in_dices->isimplied) {
numids = objidlen;
if (numids > objidlen)
goto finish_it;
} else if (tp->ranges && !tp->ranges->next
&& tp->ranges->low == tp->ranges->high) {
numids = tp->ranges->low;
if (numids > objidlen)
goto finish_it;
} else {
numids = *objid;
if (numids >= objidlen)
goto finish_it;
objid++;
objidlen--;
}
if (numids > objidlen)
goto finish_it;
for (i = 0; i < (int)numids; i++) buffer[i] = (u_char)objid[i];
var.type = ASN_OCTET_STR;
var.val.string = buffer;
var.val_len = numids;
sprint_octet_string(buf, &var, NULL, tp->hint, NULL);
while (*buf) buf++;
} else if (in_dices->isimplied) {
numids = objidlen;
if (numids > objidlen)
goto finish_it;
buf = dump_oid_to_string(objid, numids, buf, '\'');
} else if (tp->ranges && !tp->ranges->next
&& tp->ranges->low == tp->ranges->high) {
/* a fixed-length object string */
numids = tp->ranges->low;
if (numids > objidlen)
goto finish_it;
buf = dump_oid_to_string(objid, numids, buf, '\'');
} else {
numids = (size_t)*objid+1;
if (numids > objidlen)
goto finish_it;
if (numids == 1) {
if (ds_get_boolean(DS_LIBRARY_ID,DS_LIB_ESCAPE_QUOTES))
*buf++ = '\\';
*buf++ = '"';
if (ds_get_boolean(DS_LIBRARY_ID,DS_LIB_ESCAPE_QUOTES))
*buf++ = '\\';
*buf++ = '"';
}
else
buf = dump_oid_to_string(objid+1, numids-1, buf, '"');
}
objid += numids;
objidlen -= numids;
break;
case TYPE_INTEGER32:
case TYPE_UINTEGER:
case TYPE_UNSIGNED32:
case TYPE_GAUGE:
case TYPE_INTEGER:
if (tp->enums) {
struct enum_list *ep = tp->enums;
while (ep && ep->value != (int)(*objid)) ep = ep->next;
if (ep) sprintf(buf, "%s", ep->label);
else sprintf(buf, "%lu", *objid);
}
else sprintf(buf, "%lu", *objid);
while(*buf)
buf++;
objid++;
objidlen--;
break;
case TYPE_OBJID:
if (in_dices->isimplied) {
numids = objidlen;
} else {
numids = (size_t)*objid+1;
}
if ( numids > objidlen)
goto finish_it;
if (extended_index)
if (in_dices->isimplied) _sprint_objid(buf, objid, numids);
else _sprint_objid(buf, objid+1, numids-1);
else _get_symbol(objid, numids, NULL, buf, NULL, NULL);
objid += (numids);
objidlen -= (numids);
buf += strlen(buf);
break;
case TYPE_IPADDR:
if (objidlen < 4)
goto finish_it;
sprintf(buf, "%lu.%lu.%lu.%lu",
objid[0], objid[1], objid[2], objid[3]);
objid += 4;
objidlen -= 4;
while (*buf) buf++;
break;
case TYPE_NETADDR:
{ oid ntype;
ntype = *objid++; objidlen--;
sprintf(buf, "%lu.", ntype);
buf += strlen(buf);
switch (ntype) {
case 1:
if (objidlen < 4)
goto finish_it;
sprintf(buf, "%lu.%lu.%lu.%lu",
objid[0], objid[1], objid[2], objid[3]);
objid += 4;
objidlen -= 4;
break;
default:
goto finish_it;
}
}
while (*buf) buf++;
break;
case TYPE_NSAPADDRESS:
default:
goto finish_it;
break;
}
if (extended_index) *buf++ = ']';
else *buf++ = '.';
*buf = 0;
in_dices = in_dices->next;
}
finish_it:
if (buf[-1] != '.') *buf++ = '.';
while(objidlen-- > 0){ /* output rest of name, uninterpreted */
sprintf(buf, "%lu.", *objid++);
while(*buf)
buf++;
}
buf[-1] = '\0'; /* remove trailing dot */
return NULL;
}
static struct tree *
_get_realloc_symbol(oid *objid, size_t objidlen,
struct tree *subtree,
u_char **buf, size_t *buf_len, size_t *out_len,
int allow_realloc, int *buf_overflow,
struct index_list *in_dices,
size_t *end_of_known)
{
struct tree *return_tree = NULL;
int extended_index = ds_get_boolean(DS_LIBRARY_ID, DS_LIB_EXTENDED_INDEX);
char intbuf[64];
if (!objid || !buf) {
return NULL;
}
for(; subtree; subtree = subtree->next_peer) {
if (*objid == subtree->subid) {
if (subtree->indexes) {
in_dices = subtree->indexes;
} else if (subtree->augments) {
struct tree *tp2 = find_tree_node(subtree->augments, -1);
if (tp2) {
in_dices = tp2->indexes;
}
}
if (!strncmp(subtree->label, ANON, ANON_LEN) ||
ds_get_boolean(DS_LIBRARY_ID, DS_LIB_PRINT_NUMERIC_OIDS)) {
sprintf(intbuf, "%lu", subtree->subid);
if (!*buf_overflow && !snmp_strcat(buf, buf_len, out_len,
allow_realloc, (const u_char*)intbuf)) {
*buf_overflow = 1;
}
} else {
if (!*buf_overflow && !snmp_strcat(buf, buf_len, out_len,
allow_realloc, (const u_char*)subtree->label)) {
*buf_overflow = 1;
}
}
if (objidlen > 1) {
if (!*buf_overflow && !snmp_strcat(buf, buf_len, out_len,
allow_realloc, (const u_char*)".")) {
*buf_overflow = 1;
}
return_tree = _get_realloc_symbol(objid + 1, objidlen - 1,
subtree->child_list,
buf, buf_len, out_len,
allow_realloc, buf_overflow,
in_dices, end_of_known);
}
if (return_tree != NULL) {
return return_tree;
} else {
return subtree;
}
}
}
if (end_of_known) {
*end_of_known = *out_len;
}
/* Subtree not found. */
while (in_dices && (objidlen > 0) &&
!ds_get_boolean(DS_LIBRARY_ID, DS_LIB_PRINT_NUMERIC_OIDS) &&
!ds_get_boolean(DS_LIBRARY_ID, DS_LIB_DONT_BREAKDOWN_OIDS)) {
size_t numids;
struct tree *tp;
tp = find_tree_node(in_dices->ilabel, -1);
if (!tp) {
/* Can't find an index in the mib tree. Bail. */
goto finish_it;
}
if (extended_index) {
if (*(*buf + *out_len - 1) == '.') {
(*out_len)--;
}
if (!*buf_overflow && !snmp_strcat(buf, buf_len, out_len,
allow_realloc, (const u_char*)"[")) {
*buf_overflow = 1;
}
}
switch(tp->type) {
case TYPE_OCTETSTR:
if (extended_index && tp->hint) {
struct variable_list var;
u_char buffer[1024];
int i;
memset(&var, 0, sizeof var);
if (in_dices->isimplied) {
numids = objidlen;
if (numids > objidlen)
goto finish_it;
} else if (tp->ranges && !tp->ranges->next
&& tp->ranges->low == tp->ranges->high) {
numids = tp->ranges->low;
if (numids > objidlen)
goto finish_it;
} else {
numids = *objid;
if (numids >= objidlen)
goto finish_it;
objid++;
objidlen--;
}
if (numids > objidlen)
goto finish_it;
for (i = 0; i < (int)numids; i++) buffer[i] = (u_char)objid[i];
var.type = ASN_OCTET_STR;
var.val.string = buffer;
var.val_len = numids;
if (!*buf_overflow) {
if (!sprint_realloc_octet_string(buf, buf_len, out_len,
allow_realloc, &var, NULL, tp->hint, NULL)) {
*buf_overflow = 1;
}
}
} else if (in_dices->isimplied) {
numids = objidlen;
if (numids > objidlen)
goto finish_it;
if (!*buf_overflow) {
if (!dump_realloc_oid_to_string(objid, numids, buf, buf_len,
out_len, allow_realloc, '\'')) {
*buf_overflow = 1;
}
}
} else if (tp->ranges && !tp->ranges->next
&& tp->ranges->low == tp->ranges->high) {
/* a fixed-length object string */
numids = tp->ranges->low;
if (numids > objidlen)
goto finish_it;
if (!*buf_overflow) {
if (!dump_realloc_oid_to_string(objid, numids, buf, buf_len,
out_len, allow_realloc, '\'')) {
*buf_overflow = 1;
}
}
} else {
numids = (size_t)*objid+1;
if (numids > objidlen)
goto finish_it;
if (numids == 1) {
if (ds_get_boolean(DS_LIBRARY_ID, DS_LIB_ESCAPE_QUOTES)) {
if (!*buf_overflow && !snmp_strcat(buf, buf_len, out_len,
allow_realloc, (const u_char*)"\\")) {
*buf_overflow = 1;
}
}
if (!*buf_overflow && !snmp_strcat(buf, buf_len, out_len,
allow_realloc, (const u_char*)"\"")) {
*buf_overflow = 1;
}
if (ds_get_boolean(DS_LIBRARY_ID, DS_LIB_ESCAPE_QUOTES)) {
if (!*buf_overflow && !snmp_strcat(buf, buf_len, out_len,
allow_realloc, (const u_char*)"\\")) {
*buf_overflow = 1;
}
}
if (!*buf_overflow && !snmp_strcat(buf, buf_len, out_len,
allow_realloc, (const u_char*)"\"")) {
*buf_overflow = 1;
}
} else {
if (!*buf_overflow) {
if (!dump_realloc_oid_to_string(objid+1, numids-1, buf,
buf_len, out_len, allow_realloc, '"')) {
*buf_overflow = 1;
}
}
}
}
objid += numids;
objidlen -= numids;
break;
case TYPE_INTEGER32:
case TYPE_UINTEGER:
case TYPE_UNSIGNED32:
case TYPE_GAUGE:
case TYPE_INTEGER:
if (tp->enums) {
struct enum_list *ep = tp->enums;
while (ep && ep->value != (int)(*objid)) {
ep = ep->next;
}
if (ep) {
if (!*buf_overflow && !snmp_strcat(buf, buf_len, out_len,
allow_realloc, (const u_char*)ep->label)) {
*buf_overflow = 1;
}
} else {
sprintf(intbuf, "%lu", *objid);
if (!*buf_overflow && !snmp_strcat(buf, buf_len, out_len,
allow_realloc, (const u_char*)intbuf)) {
*buf_overflow = 1;
}
}
} else {
sprintf(intbuf, "%lu", *objid);
if (!*buf_overflow && !snmp_strcat(buf, buf_len, out_len,
allow_realloc, (const u_char*)intbuf)) {
*buf_overflow = 1;
}
}
objid++;
objidlen--;
break;
case TYPE_OBJID:
if (in_dices->isimplied) {
numids = objidlen;
} else {
numids = (size_t)*objid+1;
}
if (numids > objidlen)
goto finish_it;
if (extended_index) {
if (in_dices->isimplied) {
if (!*buf_overflow && !_sprint_realloc_objid(buf, buf_len,
out_len, allow_realloc, buf_overflow, objid, numids)) {
*buf_overflow = 1;
}
} else {
if (!*buf_overflow && !_sprint_realloc_objid(buf, buf_len,
out_len, allow_realloc, buf_overflow, objid+1, numids-1)) {
*buf_overflow = 1;
}
}
} else {
_get_realloc_symbol(objid, numids, NULL, buf, buf_len, out_len,
allow_realloc, buf_overflow, NULL, NULL);
}
objid += (numids);
objidlen -= (numids);
break;
case TYPE_IPADDR:
if (objidlen < 4)
goto finish_it;
sprintf(intbuf, "%lu.%lu.%lu.%lu",
objid[0], objid[1], objid[2], objid[3]);
objid += 4;
objidlen -= 4;
if (!*buf_overflow && !snmp_strcat(buf, buf_len, out_len,
allow_realloc, (const u_char*)intbuf)) {
*buf_overflow = 1;
}
break;
case TYPE_NETADDR: {
oid ntype = *objid++;
objidlen--;
sprintf(intbuf, "%lu.", ntype);
if (!*buf_overflow && !snmp_strcat(buf, buf_len, out_len,
allow_realloc, (const u_char*)intbuf)) {
*buf_overflow = 1;
}
if (ntype == 1 && objidlen >= 4) {
sprintf(intbuf, "%lu.%lu.%lu.%lu",
objid[0], objid[1], objid[2], objid[3]);
if (!*buf_overflow && !snmp_strcat(buf, buf_len, out_len,
allow_realloc, (const u_char*)intbuf)) {
*buf_overflow = 1;
}
objid += 4;
objidlen -= 4;
} else {
goto finish_it;
}
}
break;
case TYPE_NSAPADDRESS:
default:
goto finish_it;
break;
}
if (extended_index) {
if (!*buf_overflow && !snmp_strcat(buf, buf_len, out_len,
allow_realloc, (const u_char*)"]")) {
*buf_overflow = 1;
}
} else {
if (!*buf_overflow && !snmp_strcat(buf, buf_len, out_len,
allow_realloc, (const u_char*)".")) {
*buf_overflow = 1;
}
}
in_dices = in_dices->next;
}
finish_it:
if (*(*buf + *out_len - 1) != '.') {
if (!*buf_overflow && !snmp_strcat(buf, buf_len, out_len,
allow_realloc, (const u_char*)".")) {
*buf_overflow = 1;
}
}
while(objidlen-- > 0) { /* output rest of name, uninterpreted */
sprintf(intbuf, "%lu.", *objid++);
if (!*buf_overflow && !snmp_strcat(buf, buf_len, out_len,
allow_realloc, (const u_char*)intbuf)) {
*buf_overflow = 1;
}
}
*(*buf + *out_len - 1) = '\0'; /* remove trailing dot */
return NULL;
}
struct tree *
get_symbol(oid *objid,
size_t objidlen,
struct tree *subtree,
char *buf)
{
return _get_symbol(objid, objidlen, subtree, buf, NULL, NULL);
}
/*
* Clone of get_symbol that doesn't take a buffer argument
*/
struct tree *
get_tree(oid *objid,
size_t objidlen,
struct tree *subtree)
{
struct tree *return_tree = NULL;
for(; subtree; subtree = subtree->next_peer){
if (*objid == subtree->subid)
goto found;
}
return NULL;
found:
if (objidlen > 1)
return_tree = get_tree(objid + 1, objidlen - 1, subtree->child_list);
if (return_tree != NULL)
return return_tree;
else
return subtree;
}
void
print_description(oid *objid,
size_t objidlen, /* number of subidentifiers */
int width)
{
fprint_description(stdout, objid, objidlen, width);
}
void
fprint_description(FILE *f,
oid *objid,
size_t objidlen, /* number of subidentifiers */
int width)
{
struct tree *tp = get_tree(objid, objidlen, tree_head);
struct tree *subtree = tree_head;
int pos, len;
char buf[128];
const char *cp;
if (tp->type <= TYPE_SIMPLE_LAST)
cp = "OBJECT-TYPE";
else
switch (tp->type) {
case TYPE_TRAPTYPE: cp = "TRAP-TYPE"; break;
case TYPE_NOTIFTYPE: cp = "NOTIFICATION-TYPE"; break;
case TYPE_OBJGROUP: cp = "OBJECT-GROUP"; break;
case TYPE_AGENTCAP: cp = "AGENT-CAPABILITIES"; break;
case TYPE_MODID: cp = "MODULE-IDENTITY"; break;
case TYPE_MODCOMP: cp = "MODULE-COMPLIANCE"; break;
default: sprintf(buf, "type_%d", tp->type); cp = buf;
}
fprintf(f, "%s %s\n", tp->label, cp);
print_tree_node(f, tp, width);
fprintf(f, "::= {");
pos = 5;
while (objidlen > 1) {
for(; subtree; subtree = subtree->next_peer){
if (*objid == subtree->subid){
if (strncmp( subtree->label, ANON, ANON_LEN))
sprintf(buf, " %s(%lu)", subtree->label, subtree->subid);
else
sprintf(buf, " %lu", subtree->subid);
len = strlen(buf);
if (pos + len + 2 > width) {
fprintf(f, "\n ");
pos = 5;
}
fprintf(f, "%s", buf);
pos += len;
break;
}
}
if (subtree == 0) break;
objid++; objidlen--; subtree = subtree->child_list;
if (subtree == 0) break;
}
fprintf(f, " %lu }\n", *objid);
}
void
print_tree_node(FILE *f,
struct tree *tp, int width)
{
const char *cp;
char str[MAXTOKEN];
int i, prevmod, pos, len;
if (tp) {
module_name(tp->modid, str);
fprintf(f, " -- FROM\t%s", str);
for (i = 1, prevmod = tp->modid; i < tp->number_modules; i++) {
if (prevmod != tp->module_list[i]) {
module_name(tp->module_list[i], str);
fprintf(f, ", %s", str);
}
prevmod = tp->module_list[i];
}
fprintf(f, "\n");
if (tp->tc_index != -1) {
fprintf(f, " -- TEXTUAL CONVENTION %s\n", get_tc_descriptor(tp->tc_index));
}
switch (tp->type) {
case TYPE_OBJID: cp = "OBJECT IDENTIFIER"; break;
case TYPE_OCTETSTR: cp = "OCTET STRING"; break;
case TYPE_INTEGER: cp = "INTEGER"; break;
case TYPE_NETADDR: cp = "NetworkAddress"; break;
case TYPE_IPADDR: cp = "IpAddress"; break;
case TYPE_COUNTER: cp = "Counter32"; break;
case TYPE_GAUGE: cp = "Gauge32"; break;
case TYPE_TIMETICKS: cp = "TimeTicks"; break;
case TYPE_OPAQUE: cp = "Opaque"; break;
case TYPE_NULL: cp = "NULL"; break;
case TYPE_COUNTER64: cp = "Counter64"; break;
case TYPE_BITSTRING: cp = "BITS"; break;
case TYPE_NSAPADDRESS: cp = "NsapAddress"; break;
case TYPE_UINTEGER: cp = "UInteger32"; break;
case TYPE_UNSIGNED32: cp = "Unsigned32"; break;
case TYPE_INTEGER32: cp = "Integer32"; break;
default: cp = NULL; break;
}
#if SNMP_TESTING_CODE
if (!cp && (tp->ranges || tp->enums)) { /* ranges without type ? */
sprintf(str,"?0 with %s %s ?",
tp->ranges ? "Range" : "",
tp->enums ? "Enum" : "");
cp = str;
}
#endif /* SNMP_TESTING_CODE */
if (cp) fprintf(f, " SYNTAX\t%s", cp);
if (tp->ranges) {
struct range_list *rp = tp->ranges;
int first = 1;
fprintf(f, " (");
while (rp) {
if (first) first = 0;
else fprintf(f, " | ");
if (rp->low == rp->high) fprintf(f, "%d", rp->low);
else fprintf(f, "%d..%d", rp->low, rp->high);
rp = rp->next;
}
fprintf(f, ") ");
}
if (tp->enums) {
struct enum_list *ep = tp->enums;
int first = 1;
fprintf(f," { ");
pos = 16+strlen(cp)+2;
while (ep) {
if (first) first = 0;
else fprintf(f, ", ");
sprintf(str, "%s(%d)", ep->label, ep->value);
len = strlen(str);
if (pos+len+2 > width) {
fprintf(f, "\n\t\t ");
pos = 18;
}
fprintf(f, "%s", str);
pos += len+2;
ep = ep->next;
}
fprintf(f," } ");
}
if (cp) fprintf(f, "\n");
if (tp->hint) fprintf(f, " DISPLAY-HINT\t\"%s\"\n", tp->hint);
if (tp->units) fprintf(f, " UNITS\t\"%s\"\n", tp->units);
switch (tp->access) {
case MIB_ACCESS_READONLY: cp = "read-only"; break;
case MIB_ACCESS_READWRITE: cp = "read-write"; break;
case MIB_ACCESS_WRITEONLY: cp = "write-only"; break;
case MIB_ACCESS_NOACCESS: cp = "not-accessible"; break;
case MIB_ACCESS_NOTIFY: cp = "accessible-for-notify"; break;
case MIB_ACCESS_CREATE: cp = "read-create"; break;
case 0: cp = NULL; break;
default: sprintf(str,"access_%d", tp->access); cp = str;
}
if (cp) fprintf(f, " MAX-ACCESS\t%s\n", cp);
switch (tp->status) {
case MIB_STATUS_MANDATORY: cp = "mandatory"; break;
case MIB_STATUS_OPTIONAL: cp = "optional"; break;
case MIB_STATUS_OBSOLETE: cp = "obsolete"; break;
case MIB_STATUS_DEPRECATED: cp = "deprecated"; break;
case MIB_STATUS_CURRENT: cp = "current"; break;
case 0: cp = NULL; break;
default: sprintf(str,"status_%d", tp->status); cp = str;
}
#if SNMP_TESTING_CODE
if (!cp && (tp->indexes)) { /* index without status ? */
sprintf(str,"?0 with %s ?",
tp->indexes ? "Index" : "");
cp = str;
}
#endif /* SNMP_TESTING_CODE */
if (cp) fprintf(f, " STATUS\t%s\n", cp);
if (tp->augments)
fprintf(f, " AUGMENTS\t{ %s }\n", tp->augments);
if (tp->indexes) {
struct index_list *ip = tp->indexes;
int first=1;
fprintf(f, " INDEX\t\t");
fprintf(f,"{ ");
pos = 16+2;
while (ip) {
if (first) first = 0;
else fprintf(f, ", ");
sprintf(str, "%s%s", ip->isimplied ? "IMPLIED " : "",
ip->ilabel);
len = strlen(str);
if (pos+len+2 > width) {
fprintf(f, "\n\t\t ");
pos = 16+2;
}
fprintf(f, "%s", str);
pos += len+2;
ip = ip->next;
}
fprintf(f," }\n");
}
if (tp->varbinds) {
struct varbind_list *vp = tp->varbinds;
int first=1;
fprintf(f, " %s\t", tp->type == TYPE_TRAPTYPE ?
"VARIABLES" : "OBJECTS");
fprintf(f,"{ ");
pos = 16+2;
while (vp) {
if (first) first = 0;
else fprintf(f, ", ");
sprintf(str, "%s", vp->vblabel);
len = strlen(str);
if (pos+len+2 > width) {
fprintf(f, "\n\t\t ");
pos = 16+2;
}
fprintf(f, "%s", str);
pos += len+2;
vp = vp->next;
}
fprintf(f," }\n");
}
if (tp->description) fprintf(f, " DESCRIPTION\t\"%s\"\n", tp->description);
if (tp->defaultValue) fprintf(f, " DEFVAL\t{ %s }\n", tp->defaultValue);
}
else
fprintf(f, "No description\n");
}
int
get_module_node(const char *fname,
const char *module,
oid *objid,
size_t *objidlen)
{
int modid, rc = 0;
struct tree *tp;
char *name, *cp;
if ( !strcmp(module, "ANY") )
modid = -1;
else {
read_module(module);
modid = which_module( module );
if (modid == -1) return 0;
}
/* Isolate the first component of the name ... */
name = strdup(fname);
cp = strchr( name, '.' );
if ( cp != NULL ) {
*cp = '\0';
cp++;
}
/* ... and locate it in the tree. */
tp = find_tree_node(name, modid);
if (tp){
size_t maxlen = *objidlen;
/* Set the first element of the object ID */
if (node_to_oid(tp, objid, objidlen)) {
rc = 1;
/* If the name requested was more than one element,
tag on the rest of the components */
if (cp != NULL)
rc = _add_strings_to_oid(tp, cp, objid, objidlen, maxlen);
}
}
free(name);
return (rc);
}
/*
* Populate object identifier from a node in the MIB hierarchy.
* Build up the object ID, working backwards,
* starting from the end of the objid buffer.
* When the top of the MIB tree is reached, adjust the buffer.
*
* The buffer length is set to the number of subidentifiers
* for the object identifier associated with the MIB node.
* Returns the number of subidentifiers copied.
*
* If 0 is returned, the objid buffer is too small,
* and the buffer contents are indeterminate.
* The buffer length can be used to create a larger buffer.
*/
static int
node_to_oid(struct tree *tp, oid *objid, size_t *objidlen)
{
int numids, lenids;
oid *op;
if (!tp || !objid || !objidlen)
return 0;
lenids = (int)*objidlen;
op = objid + lenids; /* points after the last element */
for(numids = 0; tp; tp = tp->parent, numids++)
{
if (numids >= lenids) continue;
--op;
*op = tp->subid;
}
*objidlen = (size_t)numids;
if (numids > lenids) {
return 0;
}
if (numids < lenids)
memmove(objid, op, numids * sizeof(oid));
return (numids);
}
static int
_add_strings_to_oid(struct tree *tp, char *cp,
oid *objid, size_t *objidlen,
size_t maxlen)
{
oid subid;
int len_index = 1000000;
struct tree *tp2 = NULL;
struct index_list *in_dices = NULL;
char *fcp, *ecp, *cp2 = NULL;
char doingquote;
int len = -1, pos = -1;
int check = !ds_get_boolean(DS_LIBRARY_ID, DS_LIB_DONT_CHECK_RANGE);
while (cp && tp && tp->child_list) {
fcp = cp;
tp2 = tp->child_list;
/* Isolate the next entry */
cp2 = strchr( cp, '.' );
if (cp2) *cp2++ = '\0';
/* Search for the appropriate child */
if ( isdigit( *cp ) ) {
subid = strtoul(cp, &ecp, 0);
if (*ecp) goto bad_id;
while (tp2 && tp2->subid != subid) tp2 = tp2->next_peer;
}
else {
while (tp2 && strcmp(tp2->label, fcp)) tp2 = tp2->next_peer;
if (!tp2) goto bad_id;
subid = tp2->subid;
}
if (*objidlen >= maxlen) goto bad_id;
objid[ *objidlen ] = subid;
(*objidlen)++;
cp = cp2;
if (!tp2) break;
tp = tp2;
}
if (tp && !tp->child_list) {
if ((tp2 = tp->parent)) {
if (tp2->indexes) in_dices = tp2->indexes;
else if (tp2->augments) {
tp2 = find_tree_node(tp2->augments, -1);
if (tp2) in_dices = tp2->indexes;
}
}
tp = NULL;
}
while (cp && in_dices) {
fcp = cp;
tp = find_tree_node(in_dices->ilabel, -1);
if (!tp) break;
switch (tp->type) {
case TYPE_INTEGER:
case TYPE_INTEGER32:
case TYPE_UINTEGER:
case TYPE_UNSIGNED32:
case TYPE_TIMETICKS:
/* Isolate the next entry */
cp2 = strchr( cp, '.' );
if (cp2) *cp2++ = '\0';
if (isdigit(*cp)) {
subid = strtoul(cp, &ecp, 0);
if (*ecp) goto bad_id;
}
else {
if (tp->enums) {
struct enum_list *ep = tp->enums;
while (ep && strcmp(ep->label, cp)) ep = ep->next;
if (!ep) goto bad_id;
subid = ep->value;
}
else goto bad_id;
}
if (check && tp->ranges) {
struct range_list *rp = tp->ranges;
int ok = 0;
while (!ok && rp)
if ((rp->low <= (int)subid) && ((int)subid <= rp->high)) ok = 1;
else rp = rp->next;
if (!ok) goto bad_id;
}
if (*objidlen >= maxlen) goto bad_id;
objid[*objidlen] = subid;
(*objidlen)++;
break;
case TYPE_IPADDR:
if (*objidlen + 4 > maxlen) goto bad_id;
for (subid = 0; cp && subid < 4; subid++) {
fcp = cp; cp2 = strchr(cp, '.'); if (cp2) *cp2++ = 0;
objid[*objidlen] = strtoul(cp, &ecp, 0);
if (*ecp) goto bad_id;
if (check && objid[*objidlen] > 255) goto bad_id;
(*objidlen)++;
cp = cp2;
}
break;
case TYPE_OCTETSTR:
if (tp->ranges && !tp->ranges->next
&& tp->ranges->low == tp->ranges->high)
len = tp->ranges->low;
else
len = -1;
pos = 0;
if (*cp == '"' || *cp == '\'') {
doingquote = *cp++;
/* insert length if requested */
if (!in_dices->isimplied && len == -1) {
if (doingquote == '\'') {
snmp_set_detail("'-quote is for fixed length strings");
return 0;
}
if (*objidlen >= maxlen) goto bad_id;
len_index = *objidlen;
(*objidlen)++;
}
else if (doingquote == '"') {
snmp_set_detail("\"-quote is for variable length strings");
return 0;
}
while(*cp && *cp != doingquote) {
if (*objidlen >= maxlen) goto bad_id;
objid[ *objidlen ] = *cp++;
(*objidlen)++;
pos++;
}
if (!*cp) goto bad_id;
cp2 = cp+1;
if (!*cp2) cp2 = NULL;
else if (*cp2 != '.') goto bad_id;
else cp2++;
if (len == -1) {
struct range_list *rp = tp->ranges;
int ok = 0;
while (rp && !ok)
if (rp->low <= pos && pos <= rp->high) ok = 1;
else rp = rp->next;
if (!ok) goto bad_id;
if (!in_dices->isimplied) objid[len_index] = pos;
}
else if (pos != len) goto bad_id;
}
else {
if (!in_dices->isimplied && len == -1) {
fcp = cp; cp2 = strchr(cp, '.'); if (cp2) *cp2++ = 0;
len = strtoul(cp, &ecp, 0);
if (*ecp) goto bad_id;
if (*objidlen + len + 1 >= maxlen) goto bad_id;
objid[*objidlen] = len;
(*objidlen)++;
cp = cp2;
}
while (len && cp) {
fcp = cp; cp2 = strchr(cp, '.'); if (cp2) *cp2++ = 0;
objid[*objidlen] = strtoul(cp, &ecp, 0);
if (*ecp) goto bad_id;
if (check && objid[*objidlen] > 255)
goto bad_id;
(*objidlen)++;
len--;
cp = cp2;
}
}
break;
case TYPE_OBJID:
in_dices = NULL;
cp2 = cp;
break;
default:
snmp_log(LOG_ERR, "Unexpected index type: %d %s %s\n",
tp->type, in_dices->ilabel, cp);
in_dices = NULL;
cp2 = cp;
break;
}
cp = cp2;
if (in_dices) in_dices = in_dices->next;
}
while (cp) {
fcp = cp;
switch (*cp) {
case '0': case '1': case '2': case '3': case '4':
case '5': case '6': case '7': case '8': case '9':
cp2 = strchr(cp, '.');
if (cp2) *cp2++ = 0;
subid = strtoul(cp, &ecp, 0);
if (*ecp) goto bad_id;
if (*objidlen >= maxlen) goto bad_id;
objid[ *objidlen ] = subid;
(*objidlen)++;
break;
case '"': case '\'':
doingquote = *cp++;
/* insert length if requested */
if (doingquote == '"') {
if (*objidlen >= maxlen) goto bad_id;
objid[ *objidlen ] = len = strchr(cp, doingquote) - cp;
(*objidlen)++;
}
while(*cp && *cp != doingquote) {
if (*objidlen >= maxlen) goto bad_id;
objid[ *objidlen ] = *cp++;
(*objidlen)++;
}
if (!cp) goto bad_id;
cp2 = cp+1;
if (!*cp2) cp2 = NULL;
else if (*cp2 == '.') cp2++;
else goto bad_id;
break;
default:
goto bad_id;
}
cp = cp2;
}
return 1;
bad_id:
{ char buf[256];
if (in_dices) sprintf(buf, "Index out of range: %s (%s)",
fcp, in_dices->ilabel);
else if (tp) sprintf(buf, "Sub-id not found: %s -> %s", tp->label, fcp);
else sprintf(buf, "%s", fcp);
snmp_set_detail(buf);
}
return 0;
}
/*
* see comments on find_best_tree_node for usage after first time.
*/
int
get_wild_node(const char *name,
oid *objid,
size_t *objidlen)
{
struct tree *tp = find_best_tree_node(name, tree_head, NULL);
if (!tp)
return 0;
return get_node(tp->label, objid, objidlen);
}
int
get_node(const char *name,
oid *objid,
size_t *objidlen)
{
const char *cp;
char ch;
int res;
cp = name;
while ((ch = *cp))
if (('0' <= ch && ch <= '9')
|| ('a' <= ch && ch <= 'z')
|| ('A' <= ch && ch <= 'Z')
|| ch == '-')
cp++;
else
break;
if (ch != ':')
if (*name == '.')
res = get_module_node( name+1, "ANY", objid, objidlen );
else
res = get_module_node( name, "ANY", objid, objidlen );
else {
char *module;
/*
* requested name is of the form
* "module:subidentifier"
*/
module = (char *)malloc((size_t)(cp-name+1));
memcpy(module,name,(size_t)(cp-name));
module[cp-name] = 0;
cp++; /* cp now point to the subidentifier */
if (*cp == ':') cp++;
/* 'cp' and 'name' *do* go that way round! */
res = get_module_node( cp, module, objid, objidlen );
free(module);
}
if (res == 0) {
SET_SNMP_ERROR(SNMPERR_UNKNOWN_OBJID);
}
return res;
}
#ifdef testing
main(int argc, char* argv[])
{
oid objid[MAX_OID_LEN];
int objidlen = MAX_OID_LEN;
int count;
struct variable_list variable;
init_mib();
if (argc < 2)
print_subtree(stdout, tree_head, 0);
variable.type = ASN_INTEGER;
variable.val.integer = 3;
variable.val_len = 4;
for (argc--; argc; argc--, argv++) {
objidlen = MAX_OID_LEN;
printf("read_objid(%s) = %d\n",
argv[1], read_objid(argv[1], objid, &objidlen));
for(count = 0; count < objidlen; count++)
printf("%d.", objid[count]);
printf("\n");
print_variable(objid, objidlen, &variable);
}
}
#endif /* testing */
/* initialize: no peers included in the report. */
void clear_tree_flags(register struct tree *tp)
{
for (; tp; tp = tp->next_peer)
{
tp->reported = 0;
if (tp->child_list)
clear_tree_flags(tp->child_list); /*RECURSE*/
}
}
/*
* Update: 1998-07-17 <jhy@gsu.edu>
* Added print_oid_report* functions.
*/
static int print_subtree_oid_report_labeledoid = 0;
static int print_subtree_oid_report_oid = 0;
static int print_subtree_oid_report_symbolic = 0;
static int print_subtree_oid_report_suffix = 0;
/* These methods recurse. */
static void print_parent_labeledoid(FILE *, struct tree *);
static void print_parent_oid(FILE *, struct tree *);
static void print_parent_label(FILE *, struct tree *);
static void print_subtree_oid_report(FILE *, struct tree *, int);
void
print_oid_report (FILE *fp)
{
struct tree *tp;
clear_tree_flags(tree_head);
for (tp = tree_head ; tp ; tp=tp->next_peer)
print_subtree_oid_report (fp, tp, 0);
}
void
print_oid_report_enable_labeledoid (void)
{
print_subtree_oid_report_labeledoid = 1;
}
void
print_oid_report_enable_oid (void)
{
print_subtree_oid_report_oid = 1;
}
void
print_oid_report_enable_suffix (void)
{
print_subtree_oid_report_suffix = 1;
}
void
print_oid_report_enable_symbolic (void)
{
print_subtree_oid_report_symbolic = 1;
}
/*
* helper methods for print_subtree_oid_report()
* each one traverses back up the node tree
* until there is no parent. Then, the label combination
* is output, such that the parent is displayed first.
*
* Warning: these methods are all recursive.
*/
static void
print_parent_labeledoid(FILE *f,
struct tree *tp)
{
if(tp)
{
if(tp->parent)
{
print_parent_labeledoid(f, tp->parent); /*RECURSE*/
}
fprintf(f, ".%s(%lu)", tp->label, tp->subid);
}
}
static void
print_parent_oid(FILE *f,
struct tree *tp)
{
if(tp)
{
if(tp->parent)
{
print_parent_oid(f, tp->parent); /*RECURSE*/
}
fprintf(f, ".%lu", tp->subid);
}
}
static void
print_parent_label(FILE *f,
struct tree *tp)
{
if(tp)
{
if(tp->parent)
{
print_parent_label(f, tp->parent); /*RECURSE*/
}
fprintf(f, ".%s", tp->label);
}
}
/*
* print_subtree_oid_report():
*
* This methods generates variations on the original print_subtree() report.
* Traverse the tree depth first, from least to greatest sub-identifier.
* Warning: this methods recurses and calls methods that recurse.
*/
static void
print_subtree_oid_report(FILE *f,
struct tree *tree,
int count)
{
struct tree *tp;
count++;
/* sanity check */
if(!tree)
{
return;
}
/*
* find the not reported peer with the lowest sub-identifier.
* if no more, break the loop and cleanup.
* set "reported" flag, and create report for this peer.
* recurse using the children of this peer, if any.
*/
while (1)
{
register struct tree *ntp;
tp = 0;
for (ntp = tree->child_list; ntp; ntp = ntp->next_peer)
{
if (ntp->reported) continue;
if (!tp || (tp->subid > ntp->subid))
tp = ntp;
}
if (!tp) break;
tp->reported = 1;
if(print_subtree_oid_report_labeledoid)
{
print_parent_labeledoid(f, tp);
fprintf(f, "\n");
}
if(print_subtree_oid_report_oid)
{
print_parent_oid(f, tp);
fprintf(f, "\n");
}
if(print_subtree_oid_report_symbolic)
{
print_parent_label(f, tp);
fprintf(f, "\n");
}
if(print_subtree_oid_report_suffix)
{
int i;
for(i = 0; i < count; i++)
fprintf(f, " ");
fprintf(f, "%s(%ld) type=%d", tp->label, tp->subid, tp->type);
if (tp->tc_index != -1) fprintf(f, " tc=%d", tp->tc_index);
if (tp->hint) fprintf(f, " hint=%s", tp->hint);
if (tp->units) fprintf(f, " units=%s", tp->units);
fprintf(f, "\n");
}
print_subtree_oid_report(f, tp, count); /*RECURSE*/
}
}
/*
* Convert timeticks to hours, minutes, seconds string.
* CMU compatible does not show centiseconds.
*/
char *uptime_string(u_long timeticks, char *buf)
{
char tbuf[64];
char * cp;
uptimeString(timeticks, tbuf);
cp = strrchr(tbuf, '.');
#ifdef CMU_COMPATIBLE
if (cp) *cp = '\0';
#endif
strcpy(buf, tbuf);
return buf;
}
oid *
snmp_parse_oid(const char *argv,
oid *root,
size_t *rootlen)
{
size_t savlen = *rootlen;
if (snmp_get_random_access() || strchr(argv, ':')) {
if (get_node(argv,root,rootlen)) {
return root;
}
} else if (ds_get_boolean(DS_LIBRARY_ID, DS_LIB_REGEX_ACCESS)) {
if (get_wild_node(argv,root,rootlen)) {
return root;
}
} else {
if (read_objid(argv,root,rootlen)) {
return root;
}
*rootlen = savlen;
if (get_node(argv,root,rootlen)) {
return root;
}
*rootlen = savlen;
DEBUGMSGTL(("parse_oid","wildly parsing\n"));
if (get_wild_node(argv,root,rootlen)) {
return root;
}
}
return NULL;
}
#ifdef CMU_COMPATIBLE
int mib_TxtToOid(char *Buf, oid **OidP, size_t *LenP)
{
return read_objid(Buf, *OidP, LenP);
}
int mib_OidToTxt(oid *O, size_t OidLen, char *Buf, size_t BufLen)
{
_sprint_objid(Buf, O, OidLen);
return 1;
}
#endif /* CMU_COMPATIBLE */