blob: d3e026e2c2064473df8c61d9155642343fb8fa9c [file] [log] [blame]
## -*- c -*-
##
## For documentation on the code generated by this configuration file,
## see the file agent/helpers/table_array.c.
##
######################################################################
## Do the .h file
## @perleval $vars{shortname} =~ s/([A-Z])[a-z]+/$1/g@
######################################################################
@foreach $i table@
@open ${i}.h@
@eval $hack = "Id"
/*
* Note: this file originally auto-generated by mib2c using
* $Id$
*
* $$hack:$
*
* Yes, there is lots of code here that you might not use. But it is much
* easier to remove code than to add it!
*/
#ifndef $i.uc_H
#define $i.uc_H
#ifdef __cplusplus
extern "C" {
#endif
#include <net-snmp/net-snmp-config.h>
#include <net-snmp/library/container.h>
#include <net-snmp/agent/table_array.h>
@eval $ext_index = 0@
@foreach $idx index@
@if "$idx" ne ""@
@eval $found = "external"@
@foreach $c column@
@if "$idx" eq "$c"@
@eval $found = "internal"@
@end@
@end@
/** Index $idx is $found */
@if "$found" eq "external"@
@eval $ext_index = 1@
@end@
@end@
@end@
typedef struct ${i}_context_s {
netsnmp_index index; /** THIS MUST BE FIRST!!! */
/*************************************************************
* You can store data internally in this structure.
*
* TODO: You will probably have to fix a few types here...
*/
@if $ext_index != 0@
/** TODO: add storage for external index(s)! */
@end@
@foreach $c column@
/** $c.syntax = $c.type */
@eval $have_type = 0@
@if "$c.type" eq "ASN_OCTET_STR"@
@eval $have_type = 1@
@eval $o_len = "65535"@
@if "$c.syntax" eq "DisplayString"@
@eval $o_len = "255"@
@end@
@if "$c.syntax" eq "SnmpAdminString"@
@eval $o_len = "255"@
@end@
unsigned char $c[$o_len];
long ${c}_len;
@end@
@if "$c.type" eq "ASN_OBJECT_ID"@
@eval $have_type = 1@
oid $c[MAX_OID_LEN];
long ${c}_len;
@end@
@if "$c.type" eq "ASN_UNSIGNED"@
@eval $have_type = 1@
unsigned long $c;
@end@
@if "$c.type" eq "ASN_TIMETICKS"@
@eval $have_type = 1@
unsigned long $c;
@end@
@if "$c.type" eq "ASN_IPADDRESS"@
@eval $have_type = 1@
unsigned long $c;
@end@
@if "$c.type" eq "ASN_UINTEGER"@
@eval $have_type = 1@
unsigned long $c;
@end@
@if "$c.type" eq "ASN_INTEGER"@
@eval $have_type = 1@
long $c;
@end@
@if "$c.type" eq "ASN_COUNTER"@
@eval $have_type = 1@
unsigned long $c;
@end@
@if $have_type == 0@
/** TODO: Is this type correct? */
long $c;
@end@
@end@
/*
* OR
*
* Keep a pointer to your data
*/
void * data;
/*
*add anything else you want here
*/
} ${i}_context;
/*************************************************************
* function declarations
*/
void init_$i(void);
void initialize_table_$i(void);
const ${i}_context * ${i}_get_by_idx(netsnmp_index *);
const ${i}_context * ${i}_get_by_idx_rs(netsnmp_index *,
int row_status);
int ${i}_get_value(netsnmp_request_info *, netsnmp_index *, netsnmp_table_request_info *);
/*************************************************************
* oid declarations
*/
extern oid ${i}_oid[];
extern size_t ${i}_oid_len;
#define ${i}_TABLE_OID $i.commaoid
/*************************************************************
* column number definitions for table $i
*/
@eval $minv = 0xffffffff@
@eval $maxv = 0@
@foreach $c column@
#define COLUMN_$c.uc $c.subid
@if ! $c.noaccess@
@eval $minv = min($minv, $c.subid)@
@eval $maxv = max($maxv, $c.subid)@
@end@
@if "$c.syntax" eq "RowStatus"@
@eval $rs_name = "$c"@
@end@
@if "$c.syntax" eq "StorageType"@
@eval $st_name = "$c"@
@end@
@end@
#define ${i}_COL_MIN $minv
#define ${i}_COL_MAX $maxv
/* comment out the following line if you don't want a custom sort */
#define ${i}_CUSTOM_SORT
@if "$rs_name" ne ""@
/* uncommend the following line if you allow modifications to an
* active row */
/** define ${i}_CAN_MODIFY_ACTIVE_ROW */
@end@
@if $i.settable@
int ${i}_extract_index( ${i}_context * ctx, netsnmp_index * hdr );
void ${i}_set_reserve1( netsnmp_request_group * );
void ${i}_set_reserve2( netsnmp_request_group * );
void ${i}_set_action( netsnmp_request_group * );
void ${i}_set_commit( netsnmp_request_group * );
void ${i}_set_free( netsnmp_request_group * );
void ${i}_set_undo( netsnmp_request_group * );
${i}_context * ${i}_duplicate_row( ${i}_context* );
netsnmp_index * ${i}_delete_row( ${i}_context* );
@if "$rs_name" ne ""@
int ${i}_can_activate(${i}_context *undo_ctx,
${i}_context *row_ctx,
netsnmp_request_group * rg);
int ${i}_can_deactivate(${i}_context *undo_ctx,
${i}_context *row_ctx,
netsnmp_request_group * rg);
@end@
int ${i}_can_delete(${i}_context *undo_ctx,
${i}_context *row_ctx,
netsnmp_request_group * rg);
@if $i.creatable@
${i}_context * ${i}_create_row( netsnmp_index* );
@end@
@end@
#ifdef ${i}_CUSTOM_SORT
${i}_context * ${i}_get( const char *name, int len );
#endif
#ifdef __cplusplus
}
#endif
#endif /** $i.uc_H */
@end@
######################################################################
## Do the .c file
######################################################################
@foreach $i table@
@open ${i}.c@
@eval $hack = "Id"@
/*
* Note: this file originally auto-generated by mib2c using
* $Id$
*
* $$hack:$
*
*
* For help understanding NET-SNMP in general, please check the
* documentation and FAQ at:
*
* http://www.net-snmp.org/
*
*
* For help understanding this code, the agent and how it processes
* requests, please check the following references.
*
* http://www.net-snmp.org/tutorial-5/
*
*
* You can also join the #net-snmp channel on irc.freenode.net
* and ask for help there.
*
*
* And if all else fails, send a detailed message to the developers
* describing the problem you are having to:
*
* net-snmp-coders@lists.sourceforge.net
*
*
* Yes, there is lots of code here that you might not use. But it is much
* easier to remove code than to add it!
*/
#include <net-snmp/net-snmp-config.h>
#include <net-snmp/net-snmp-includes.h>
#include <net-snmp/agent/net-snmp-agent-includes.h>
#include <net-snmp/library/snmp_assert.h>
#include "${i}.h"
static netsnmp_handler_registration *my_handler = NULL;
static netsnmp_table_array_callbacks cb;
oid ${i}_oid[] = { ${i}_TABLE_OID };
size_t ${i}_oid_len = OID_LENGTH(${i}_oid);
#ifdef ${i}_CUSTOM_SORT
/************************************************************
* keep binary tree to find context by name
*/
static int ${i}_cmp( const void *lhs, const void *rhs );
/************************************************************
* compare two context pointers here. Return -1 if lhs < rhs,
* 0 if lhs == rhs, and 1 if lhs > rhs.
*/
static int
${i}_cmp( const void *lhs, const void *rhs )
{
${i}_context *context_l =
(${i}_context *)lhs;
${i}_context *context_r =
(${i}_context *)rhs;
/*
* check primary key, then secondary. Add your own code if
* there are more than 2 keys
*/
int rc;
/*
* TODO: implement compare. Remove this ifdef code and
* add your own code here.
*/
#ifdef TABLE_CONTAINER_TODO
snmp_log(LOG_ERR,
"${i}_compare not implemented! Container order undefined\n" );
return 0;
#endif
/*
* EXAMPLE (assuming you want to sort on a name):
*
* rc = strcmp( context_l->xxName, context_r->xxName );
*
* if(rc)
* return rc;
*
* TODO: fix secondary keys (or delete if there are none)
*
* if(context_l->yy < context_r->yy)
* return -1;
*
* return (context_l->yy == context_r->yy) ? 0 : 1;
*/
}
/************************************************************
* search tree
*/
/** TODO: set additional indexes as parameters */
${i}_context *
${i}_get( const char *name, int len )
{
${i}_context tmp;
/** we should have a custom container */
netsnmp_assert(cb.container->next != NULL);
/*
* TODO: implement compare. Remove this ifdef code and
* add your own code here.
*/
#ifdef TABLE_CONTAINER_TODO
snmp_log(LOG_ERR, "${i}_get not implemented!\n" );
return NULL;
#endif
/*
* EXAMPLE:
*
* if(len > sizeof(tmp.xxName))
* return NULL;
*
* strncpy( tmp.xxName, name, sizeof(tmp.xxName) );
* tmp.xxName_len = len;
*
* return CONTAINER_FIND(cb.container->next, &tmp);
*/
}
#endif
/************************************************************
* Initializes the $i module
*/
void
init_$i(void)
{
initialize_table_$i();
/*
* TODO: perform any startup stuff here, such as
* populating the table with initial data.
*
* ${i}_context * new_row = create_row(index);
* CONTAINER_INSERT(cb.container,new_row);
*/
}
@if $i.settable@
/************************************************************
* the *_row_copy routine
*/
static int ${i}_row_copy(${i}_context * dst,
${i}_context * src)
{
if(!dst||!src)
return 1;
/*
* copy index, if provided
*/
if(dst->index.oids)
free(dst->index.oids);
if(snmp_clone_mem( (void*)&dst->index.oids, src->index.oids,
src->index.len * sizeof(oid) )) {
dst->index.oids = NULL;
return 1;
}
dst->index.len = src->index.len;
/*
* copy components into the context structure
*/
@if $ext_index != 0@
/** TODO: add code for external index(s)! */
@end@
@foreach $c column@
@eval $have_type = 0@
@if "$c.type" eq "ASN_OCTET_STR"@
@eval $have_type = 1@
memcpy( dst->$c, src->$c, src->${c}_len );
dst->${c}_len = src->${c}_len;
@end@
@if "$c.type" eq "ASN_OBJECT_ID"@
@eval $have_type = 1@
memcpy( dst->$c, src->$c, src->${c}_len );
dst->${c}_len = src->${c}_len;
@end@
@if $have_type == 0@
dst->$c = src->$c;
@end@
@end@
return 0;
}
/**
* the *_extract_index routine
*
* This routine is called when a set request is received for an index
* that was not found in the table container. Here, we parse the oid
* in the the individual index components and copy those indexes to the
* context. Then we make sure the indexes for the new row are valid.
*/
int
${i}_extract_index( ${i}_context * ctx, netsnmp_index * hdr )
{
/*
* temporary local storage for extracting oid index
*
* extract index uses varbinds (netsnmp_variable_list) to parse
* the index OID into the individual components for each index part.
*/
@if $ext_index != 0@
/** TODO: add storage for external index(s)! */
@end@
@eval $first_idx = ""@
@foreach $idx index@
@if "$first_idx" eq ""@
@eval $first_idx = $idx@
@end@
netsnmp_variable_list var_$idx;
@end@
int err;
/*
* copy index, if provided
*/
if(hdr) {
netsnmp_assert(ctx->index.oids == NULL);
if((hdr->len > MAX_OID_LEN) ||
snmp_clone_mem( (void*)&ctx->index.oids, hdr->oids,
hdr->len * sizeof(oid) )) {
return -1;
}
ctx->index.len = hdr->len;
}
/*
* initialize variable that will hold each component of the index.
* If there are multiple indexes for the table, the variable_lists
* need to be linked together, in order.
*/
@if $ext_index != 0@
/** TODO: add code for external index(s)! */
@end@
@foreach $idx index@
memset( &var_$idx, 0x00, sizeof(var_$idx) );
var_${idx}.type = $idx.type; /* type hint for parse_oid_indexes */
/** TODO: link this index to the next, or NULL for the last one */
#ifdef TABLE_CONTAINER_TODO
snmp_log(LOG_ERR, "${i}_extract_index index list not implemented!\n" );
return 0;
#else
var_${idx}.next_variable = &var_XX;
#endif
@end@
/*
* parse the oid into the individual index components
*/
err = parse_oid_indexes( hdr->oids, hdr->len, &var_$first_idx );
if (err == SNMP_ERR_NOERROR) {
/*
* copy index components into the context structure
*/
@foreach $idx index@
@eval $found = "external"@
@foreach $c column@
@if "$idx" eq "$c"@
@eval $found = "internal"@
@end@
@end@
@if "$found" eq "external"@
/** skipping external index $idx */
@end@
@if "$found" eq "internal"@
@eval $have_type = 0@
@if "$idx.type" eq "ASN_OCTET_STR"@
@eval $have_type = 1@
if(var_${idx}.val_len > sizeof(ctx->$idx))
err = -1;
else
memcpy( ctx->$idx, var_${idx}.val.string, var_${idx}.val_len );
ctx->${idx}_len = var_${idx}.val_len;
@end@
@if "$idx.type" eq "ASN_OBJECT_ID"@
@eval $have_type = 1@
memcpy( ctx->$idx, var_${idx}.val.string, var_${idx}.val_len );
ctx->${idx}_len = var_${idx}.val_len;
@end@
@if $have_type == 0@
ctx->$idx = *var_${idx}.val.integer;
@end@
@end@
@end@
@foreach $c index@
/*
* TODO: check index for valid values. For EXAMPLE:
*
@eval $have_check = 0@
@if "$c.type" eq "ASN_IPADDRESS"@
@eval $have_check = 1@
* if ( XXX_check_ip( *var_${c}.val.integer ) ) {
@end@
@if "$c.type" eq "ASN_OBJECT_ID"@
@eval $have_check = 1@
* if ( XXX_check_oid( var_${c}.val.objid, var_${c}.val_len /
sizeof(oid) ) ) {
@end@
@if "$c.type" eq "ASN_OCTET_STR"@
@eval $have_check = 1@
* if ( XXX_check_value( var_${c}.val.string, XXX ) ) {
@end@
@if $have_check != 1@
* if ( *var_${c}.val.integer != XXX ) {
@end@
* err = -1;
* }
*/
@end@
}
/*
* parsing may have allocated memory. free it.
*/
snmp_reset_var_buffers( &var_$first_idx );
return err;
}
@if "$rs_name" ne ""@
/************************************************************
* the *_can_activate routine is called
* when a row is changed to determine if all the values
* set are consistent with the row's rules for a row status
* of ACTIVE.
*
* return 1 if the row could be ACTIVE
* return 0 if the row is not ready for the ACTIVE state
*/
int ${i}_can_activate(${i}_context *undo_ctx,
${i}_context *row_ctx,
netsnmp_request_group * rg)
{
/*
* TODO: check for activation requirements here
*/
/*
* be optimistic.
*/
return 1;
}
/************************************************************
* the *_can_deactivate routine is called when a row that is
* currently ACTIVE is set to a state other than ACTIVE. If
* there are conditions in which a row should not be allowed
* to transition out of the ACTIVE state (such as the row being
* referred to by another row or table), check for them here.
*
* return 1 if the row can be set to a non-ACTIVE state
* return 0 if the row must remain in the ACTIVE state
*/
int ${i}_can_deactivate(${i}_context *undo_ctx,
${i}_context *row_ctx,
netsnmp_request_group * rg)
{
/*
* TODO: check for deactivation requirements here
*/
return 1;
}
@end@
/************************************************************
* the *_can_delete routine is called to determine if a row
* can be deleted.
*
* return 1 if the row can be deleted
* return 0 if the row cannot be deleted
*/
int ${i}_can_delete(${i}_context *undo_ctx,
${i}_context *row_ctx,
netsnmp_request_group * rg)
{
@if "$rs_name" ne ""@
/*
* probably shouldn't delete a row that we can't
* deactivate.
*/
if(${i}_can_deactivate(undo_ctx,row_ctx,rg) != 1)
return 0;
@end@
/*
* TODO: check for other deletion requirements here
*/
return 1;
}
@if $i.creatable@
/************************************************************
* the *_create_row routine is called by the table handler
* to create a new row for a given index. If you need more
* information (such as column values) to make a decision
* on creating rows, you must create an initial row here
* (to hold the column values), and you can examine the
* situation in more detail in the *_set_reserve1 or later
* states of set processing. Simple check for a NULL undo_ctx
* in those states and do detailed creation checking there.
*
* returns a newly allocated ${i}_context
* structure if the specified indexes are not illegal
* returns NULL for errors or illegal index values.
*/
${i}_context *
${i}_create_row( netsnmp_index* hdr)
{
${i}_context * ctx =
SNMP_MALLOC_TYPEDEF(${i}_context);
if(!ctx)
return NULL;
/*
* TODO: check indexes, if necessary.
*/
if(${i}_extract_index( ctx, hdr )) {
if (NULL != ctx->index.oids)
free(ctx->index.oids);
free(ctx);
return NULL;
}
/* netsnmp_mutex_init(ctx->lock);
netsnmp_mutex_lock(ctx->lock); */
/*
* TODO: initialize any default values here. This is also
* first place you really should allocate any memory for
* yourself to use. If you allocated memory earlier,
* make sure you free it for earlier error cases!
*/
/**
@foreach $c column@
@if $c.settable@
ctx->$c = 0;
@end@
@end@
*/
return ctx;
}
@end@
/************************************************************
* the *_duplicate row routine
*/
${i}_context *
${i}_duplicate_row( ${i}_context * row_ctx)
{
${i}_context * dup;
if(!row_ctx)
return NULL;
dup = SNMP_MALLOC_TYPEDEF(${i}_context);
if(!dup)
return NULL;
if(${i}_row_copy(dup,row_ctx)) {
free(dup);
dup = NULL;
}
return dup;
}
/************************************************************
* the *_delete_row method is called to delete a row.
*/
netsnmp_index * ${i}_delete_row( ${i}_context * ctx )
{
/* netsnmp_mutex_destroy(ctx->lock); */
if(ctx->index.oids)
free(ctx->index.oids);
/*
* TODO: release any memory you allocated here...
*/
/*
* release header
*/
free( ctx );
return NULL;
}
/************************************************************
* RESERVE is used to check the syntax of all the variables
* provided, that the values being set are sensible and consistent,
* and to allocate any resources required for performing the SET.
* After this stage, the expectation is that the set ought to
* succeed, though this is not guaranteed. (In fact, with the UCD
* agent, this is done in two passes - RESERVE1, and
* RESERVE2, to allow for dependancies between variables).
*
* BEFORE calling this routine, the agent will call duplicate_row
* to create a copy of the row (unless this is a new row; i.e.
* row_created == 1).
*
* next state -> SET_RESERVE2 || SET_FREE
*/
void ${i}_set_reserve1( netsnmp_request_group *rg )
{
${i}_context *row_ctx =
(${i}_context *)rg->existing_row;
${i}_context *undo_ctx =
(${i}_context *)rg->undo_info;
netsnmp_variable_list *var;
netsnmp_request_group_item *current;
int rc;
@if "$st_name" ne ""@
/*
* Block all attempts to modify a readOnly row
*/
if( row_ctx && (row_ctx->$st_name == SNMP_STORAGE_READONLY) ) {
netsnmp_set_mode_request_error(MODE_SET_BEGIN, rg->list->ri,
SNMP_ERR_NOTWRITABLE);
return;
}
@end@
/*
* TODO: loop through columns, check syntax and lengths. For
* columns which have no dependencies, you could also move
* the value/range checking here to attempt to catch error
* cases as early as possible.
*/
for( current = rg->list; current; current = current->next ) {
var = current->ri->requestvb;
rc = SNMP_ERR_NOERROR;
switch(current->tri->colnum) {
@foreach $c column@
@if $c.settable@
case COLUMN_$c.uc:
/** $c.syntax = $c.type */
@if $c.needlength@
/* or possibly 'netsnmp_check_vb_type_and_size' */
rc = netsnmp_check_vb_type_and_max_size(var, $c.type,
sizeof(row_ctx->$c));
@else@
/* or possibly 'netsnmp_check_vb_int_range' */
rc = netsnmp_check_vb_int( var );
@end@
break;
@end@
@end@
default: /** We shouldn't get here */
rc = SNMP_ERR_GENERR;
snmp_log(LOG_ERR, "unknown column in "
"${i}_set_reserve1\n");
}
if (rc)
netsnmp_set_mode_request_error(MODE_SET_BEGIN, current->ri, rc );
rg->status = SNMP_MAX( rg->status, current->ri->status );
}
/*
* done with all the columns. Could check row related
* requirements here.
*/
}
void ${i}_set_reserve2( netsnmp_request_group *rg )
{
${i}_context *row_ctx = (${i}_context *)rg->existing_row;
${i}_context *undo_ctx = (${i}_context *)rg->undo_info;
netsnmp_request_group_item *current;
netsnmp_variable_list *var;
int rc;
rg->rg_void = rg->list->ri;
/*
* TODO: loop through columns, check for valid
* values and any range constraints.
*/
for( current = rg->list; current; current = current->next ) {
var = current->ri->requestvb;
rc = SNMP_ERR_NOERROR;
switch(current->tri->colnum) {
@foreach $c column@
@if $c.settable@
case COLUMN_$c.uc:
/** $c.syntax = $c.type */
@eval $have_check = 0@
@if "$c" eq "$st_name"@
@eval $have_check = 1@
rc = netsnmp_check_vb_storagetype(current->ri->requestvb,
undo_ctx ?
undo_ctx->$c:0);
@end@
@if "$c" eq "$rs_name"@
@eval $have_check = 1@
rc = netsnmp_check_vb_rowstatus(current->ri->requestvb,
undo_ctx ?
undo_ctx->$c:0);
rg->rg_void = current->ri;
@end@
@if "$c.syntax" eq "TruthValue"@
@eval $have_check = 1@
rc = netsnmp_check_vb_truthvalue(current->ri->requestvb);
@end@
@if $have_check == 0@
/*
* TODO: routine to check valid values
*
* EXAMPLE:
*
@if "$c.type" eq "ASN_IPADDRESS"@
@eval $have_check = 1@
* if ( XXX_check_ip( *var->val.integer ) ) {
@end@
@if "$c.type" eq "ASN_OBJECT_ID"@
@eval $have_check = 1@
* if ( XXX_check_oid( var ) ) {
@end@
@if "$c.type" eq "ASN_OCTET_STR"@
@eval $have_check = 1@
* if ( XXX_check_value( var->val.string, XXX ) ) {
@end@
@if $have_check != 1@
* if ( *var->val.integer != XXX ) {
@end@
* rc = SNMP_ERR_INCONSISTENTVALUE;
* rc = SNMP_ERR_BADVALUE;
* }
*/
@end@
break;
@end@
@end@
default: /** We shouldn't get here */
netsnmp_assert(0); /** why wasn't this caught in reserve1? */
}
if (rc)
netsnmp_set_mode_request_error(MODE_SET_BEGIN, current->ri, rc);
}
/*
* done with all the columns. Could check row related
* requirements here.
*/
}
/************************************************************
* Assuming that the RESERVE phases were successful, the next
* stage is indicated by the action value ACTION. This is used
* to actually implement the set operation. However, this must
* either be done into temporary (persistent) storage, or the
* previous value stored similarly, in case any of the subsequent
* ACTION calls fail.
*
* In your case, changes should be made to row_ctx. A copy of
* the original row is in undo_ctx.
*/
void ${i}_set_action( netsnmp_request_group *rg )
{
netsnmp_variable_list *var;
${i}_context *row_ctx = (${i}_context *)rg->existing_row;
${i}_context *undo_ctx = (${i}_context *)rg->undo_info;
netsnmp_request_group_item *current;
@if "$rs_name" ne ""@
int row_err = 0;
@end@
/*
* TODO: loop through columns, copy varbind values
* to context structure for the row.
*/
for( current = rg->list; current; current = current->next ) {
var = current->ri->requestvb;
switch(current->tri->colnum) {
@foreach $c column@
@if $c.settable@
case COLUMN_$c.uc:
/** $c.syntax = $c.type */
@eval $have_type = 0@
@if "$c.type" eq "ASN_OCTET_STR"@
@eval $have_type = 1@
memcpy(row_ctx->$c,var->val.string,var->val_len);
row_ctx->${c}_len = var->val_len;
@end@
@if "$c.type" eq "ASN_OBJECT_ID"@
@eval $have_type = 1@
memcpy(row_ctx->$c,var->val.objid,var->val_len);
row_ctx->${c}_len = var->val_len;
@end@
@if $have_type == 0@
row_ctx->$c = *var->val.integer;
@end@
break;
@end@
@end@
default: /** We shouldn't get here */
netsnmp_assert(0); /** why wasn't this caught in reserve1? */
}
}
/*
* done with all the columns. Could check row related
* requirements here.
*/
@if "$rs_name" ne ""@
#ifndef ${i}_CAN_MODIFY_ACTIVE_ROW
if( undo_ctx && RS_IS_ACTIVE(undo_ctx->$rs_name) &&
row_ctx && RS_IS_ACTIVE(row_ctx->$rs_name) ) {
row_err = 1;
}
#endif
/*
* check activation/deactivation
*/
row_err = netsnmp_table_array_check_row_status(&cb, rg,
row_ctx ? &row_ctx->$rs_name : NULL,
undo_ctx ? &undo_ctx->$rs_name : NULL);
if(row_err) {
netsnmp_set_mode_request_error(MODE_SET_BEGIN,
(netsnmp_request_info*)rg->rg_void,
row_err);
return;
}
@end@
/*
* TODO: if you have dependencies on other tables, this would be
* a good place to check those, too.
*/
}
/************************************************************
* Only once the ACTION phase has completed successfully, can
* the final COMMIT phase be run. This is used to complete any
* writes that were done into temporary storage, and then release
* any allocated resources. Note that all the code in this phase
* should be "safe" code that cannot possibly fail (cue
* hysterical laughter). The whole intent of the ACTION/COMMIT
* division is that all of the fallible code should be done in
* the ACTION phase, so that it can be backed out if necessary.
*
* BEFORE calling this routine, the agent will update the
* container (inserting a row if row_created == 1, or removing
* the row if row_deleted == 1).
*
* AFTER calling this routine, the agent will delete the
* undo_info.
*/
void ${i}_set_commit( netsnmp_request_group *rg )
{
netsnmp_variable_list *var;
${i}_context *row_ctx = (${i}_context *)rg->existing_row;
${i}_context *undo_ctx = (${i}_context *)rg->undo_info;
netsnmp_request_group_item *current;
/*
* loop through columns
*/
for( current = rg->list; current; current = current->next ) {
var = current->ri->requestvb;
switch(current->tri->colnum) {
@foreach $c column@
@if $c.settable@
case COLUMN_$c.uc:
/** $c.syntax = $c.type */
break;
@end@
@end@
default: /** We shouldn't get here */
netsnmp_assert(0); /** why wasn't this caught in reserve1? */
}
}
/*
* done with all the columns. Could check row related
* requirements here.
*/
}
/************************************************************
* If either of the RESERVE calls fail, the write routines
* are called again with the FREE action, to release any resources
* that have been allocated. The agent will then return a failure
* response to the requesting application.
*
* AFTER calling this routine, the agent will delete undo_info.
*/
void ${i}_set_free( netsnmp_request_group *rg )
{
netsnmp_variable_list *var;
${i}_context *row_ctx = (${i}_context *)rg->existing_row;
${i}_context *undo_ctx = (${i}_context *)rg->undo_info;
netsnmp_request_group_item *current;
/*
* loop through columns
*/
for( current = rg->list; current; current = current->next ) {
var = current->ri->requestvb;
switch(current->tri->colnum) {
@foreach $c column@
@if $c.settable@
case COLUMN_$c.uc:
/** $c.syntax = $c.type */
break;
@end@
@end@
default: /** We shouldn't get here */
/** should have been logged in reserve1 */
}
}
/*
* done with all the columns. Could check row related
* requirements here.
*/
}
/************************************************************
* If the ACTION phase does fail (for example due to an apparently
* valid, but unacceptable value, or an unforeseen problem), then
* the list of write routines are called again, with the UNDO
* action. This requires the routine to reset the value that was
* changed to its previous value (assuming it was actually changed),
* and then to release any resources that had been allocated. As
* with the FREE phase, the agent will then return an indication
* of the error to the requesting application.
*
* BEFORE calling this routine, the agent will update the container
* (remove any newly inserted row, re-insert any removed row).
*
* AFTER calling this routing, the agent will call row_copy
* to restore the data in existing_row from the date in undo_info.
* Then undo_info will be deleted (or existing row, if row_created
* == 1).
*/
void ${i}_set_undo( netsnmp_request_group *rg )
{
netsnmp_variable_list *var;
${i}_context *row_ctx = (${i}_context *)rg->existing_row;
${i}_context *undo_ctx = (${i}_context *)rg->undo_info;
netsnmp_request_group_item *current;
/*
* loop through columns
*/
for( current = rg->list; current; current = current->next ) {
var = current->ri->requestvb;
switch(current->tri->colnum) {
@foreach $c column@
@if $c.settable@
case COLUMN_$c.uc:
/** $c.syntax = $c.type */
break;
@end@
@end@
default: /** We shouldn't get here */
netsnmp_assert(0); /** why wasn't this caught in reserve1? */
}
}
/*
* done with all the columns. Could check row related
* requirements here.
*/
}
@end@
/************************************************************
*
* Initialize the $i table by defining its contents and how it's structured
*/
void
initialize_table_$i(void)
{
netsnmp_table_registration_info *table_info;
if(my_handler) {
snmp_log(LOG_ERR, "initialize_table_${i}_handler called again\n");
return;
}
memset(&cb, 0x00, sizeof(cb));
/** create the table structure itself */
table_info = SNMP_MALLOC_TYPEDEF(netsnmp_table_registration_info);
my_handler = netsnmp_create_handler_registration("$i",
netsnmp_table_array_helper_handler,
${i}_oid,
${i}_oid_len,
@if $i.settable@
HANDLER_CAN_RWRITE
@else@
HANDLER_CAN_RONLY
@end@
);
if (!my_handler || !table_info) {
snmp_log(LOG_ERR, "malloc failed in "
"initialize_table_${i}_handler\n");
return; /** mallocs failed */
}
/***************************************************
* Setting up the table's definition
*/
/*
* TODO: add any external indexes here.
*/
@if $ext_index != 0@
/** TODO: add code for external index(s)! */
@end@
/*
* internal indexes
*/
@foreach $idx index@
/** index: $idx */
netsnmp_table_helper_add_index(table_info, $idx.type);
@end@
table_info->min_column = ${i}_COL_MIN;
table_info->max_column = ${i}_COL_MAX;
/***************************************************
* registering the table with the master agent
*/
cb.get_value = ${i}_get_value;
cb.container = netsnmp_container_find("${i}_primary:"
"${i}:"
"table_container");
#ifdef ${i}_CUSTOM_SORT
netsnmp_container_add_index(cb.container,
netsnmp_container_find("${i}_custom:"
"${i}:"
"table_container"));
cb.container->next->compare = ${i}_cmp;
#endif
@if $i.settable@
cb.can_set = 1;
@if $i.creatable@
cb.create_row = (UserRowMethod*)${i}_create_row;
@end@
cb.duplicate_row = (UserRowMethod*)${i}_duplicate_row;
cb.delete_row = (UserRowMethod*)${i}_delete_row;
cb.row_copy = (Netsnmp_User_Row_Operation *)${i}_row_copy;
@if "$rs_name" ne ""@
cb.can_activate = (Netsnmp_User_Row_Action *)${i}_can_activate;
cb.can_deactivate = (Netsnmp_User_Row_Action *)${i}_can_deactivate;
@end@
cb.can_delete = (Netsnmp_User_Row_Action *)${i}_can_delete;
cb.set_reserve1 = ${i}_set_reserve1;
cb.set_reserve2 = ${i}_set_reserve2;
cb.set_action = ${i}_set_action;
cb.set_commit = ${i}_set_commit;
cb.set_free = ${i}_set_free;
cb.set_undo = ${i}_set_undo;
@end@
DEBUGMSGTL(("initialize_table_$i",
"Registering table $i "
"as a table array\n"));
netsnmp_table_container_register(my_handler, table_info, &cb,
cb.container, 1);
}
/************************************************************
* ${i}_get_value
*
* This routine is called for get requests to copy the data
* from the context to the varbind for the request. If the
* context has been properly maintained, you don't need to
* change in code in this fuction.
*/
int ${i}_get_value(
netsnmp_request_info *request,
netsnmp_index *item,
netsnmp_table_request_info *table_info )
{
netsnmp_variable_list *var = request->requestvb;
${i}_context *context = (${i}_context *)item;
switch(table_info->colnum) {
@foreach $c column@
@if $c.readable@
@eval $have_type = 0@
case COLUMN_$c.uc:
/** $c.syntax = $c.type */
@if "$c.type" eq "ASN_OBJECT_ID"@
@eval $have_type = 1@
snmp_set_var_typed_value(var, $c.type,
(char*)&context->$c,
context->${c}_len );
@end@
@if "$c.type" eq "ASN_OCTET_STR"@
@eval $have_type = 1@
snmp_set_var_typed_value(var, $c.type,
(char*)&context->$c,
context->${c}_len );
@end@
@if $have_type == 0@
snmp_set_var_typed_value(var, $c.type,
(char*)&context->$c,
sizeof(context->$c) );
@end@
break;
@end@
@end@
default: /** We shouldn't get here */
snmp_log(LOG_ERR, "unknown column in "
"${i}_get_value\n");
return SNMP_ERR_GENERR;
}
return SNMP_ERR_NOERROR;
}
/************************************************************
* ${i}_get_by_idx
*/
const ${i}_context *
${i}_get_by_idx(netsnmp_index * hdr)
{
return (const ${i}_context *)
CONTAINER_FIND(cb.container, hdr );
}
@end@