mfd-access-unsorted-external-defines.m2i
上传用户:wxp200602
上传日期:2007-10-30
资源大小:4028k
文件大小:40k
源码类别:

SNMP编程

开发平台:

Unix_Linux

  1. #############################################################  -*- c -*-
  2. ## generic include for XXX. Do not use directly.
  3. ##
  4. ## $Id: mfd-access-unsorted-external-defines.m2i,v 1.12 2004/10/08 23:39:17 rstory Exp $
  5. ########################################################################
  6. ##
  7. @eval $mfd_aue_wrap_param = "wrap_ctx"@
  8. @eval $mfd_aue_wrap_param_type = "${context}_interface_ctx *"@
  9. @eval $mfd_aue_wrap_param_decl = "$mfd_aue_wrap_param_type $mfd_aue_wrap_param"@
  10. ##
  11. @eval $mfd_aue_param = "${context}_reg"@
  12. @eval $mfd_aue_param_type = "${context}_registration_ptr"@
  13. @eval $mfd_aue_param_decl = "$mfd_aue_param_type $mfd_aue_param"@
  14. @eval $mfd_aue_param_cmt = "$mfd_aue_param Pointer to a $mfd_aue_param_type"
  15. ##
  16. @if $m2c_mark_boundary == 1@
  17. /** START code generated by $RCSfile: mfd-access-unsorted-external-defines.m2i,v $ $Revision: 1.12 $ */
  18. @end@
  19. ##//####################################################################
  20. ##//@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
  21. ##//@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
  22. @if $m2c_processing_type eq 'h'@
  23. ##
  24. @if $m2c_include_examples == 1@
  25. $example_start
  26. /* *********************************************************************
  27.  * Since we have no idea how you really access your data, we'll go with
  28.  * a worst case example: a flat text file.
  29.  @   if $m2c_data_transient != 2@
  30.  @      print Example code is for fully transient data. Either turn off@
  31.  @      print m2c_include_examples or set m2c_data_transient to 2.@
  32.  @      exit@
  33.  @   end@
  34.  */
  35. #define MAX_LINE_SIZE 256
  36. $example_end
  37. @end@
  38. /**
  39.  * loop context
  40.  *
  41.  * TODO:
  42.  * define loop context structure
  43.  *
  44.  *  Since the actual loop is in the MFD handler, a loop contex parameter
  45.  *  is provided to help you keep track of where you are in between calls
  46.  *  to functions that you wrote and the master MFD handler calls. The
  47.  *  structure of this context is user defineable, and is defined in the
  48.  *  file ${table}_data_access.h.
  49.  *
  50.  *  E.G., if your data is stored in a linked list, the obvious thing you
  51.  *  want to know from one function call to the next is your current
  52.  *  position in the linked list.  Thus the easiest context to use is a
  53.  *  pointer within the linked list.  For an array, the current index to
  54.  *  that array would be easiest.
  55.  *
  56.  *  The funtion calls are actually passed a reference to the loop
  57.  *  context, to allow the loop context to be allocated memory. Here are
  58.  *  some simple examples definitions for various data formats. These
  59.  *  definitions are used in examples later on.
  60.  *
  61.  */
  62. typedef struct ${context}_loop_context_s {
  63.     /*
  64.      * temporary context used during iteration
  65.      */
  66.     ${context}_rowreq_ctx *rowreq_ctx;
  67. @if $m2c_include_examples == 1@
  68.     
  69.     /*
  70.      * this example code is based on a data source that is a
  71.      * text file to be read and parsed.
  72.      */
  73.     FILE *filep;
  74.     char line[MAX_LINE_SIZE];
  75. @end@
  76. } ${context}_loop_context;
  77. /*
  78.  * define a reference to the loop context
  79.  *
  80.  * NOTE: DO NOT ADD ITEMS TO THIS STRUCTURE!
  81.  */
  82. typedef struct ${context}_ref_loop_ctx_s {
  83.     ${context}_loop_context *loop_ctx;
  84. } ${context}_ref_loop_ctx;
  85. int ${context}_loop_get_first( $mfd_aue_param_decl, ${context}_ref_loop_ctx *loop_ctx_ref,
  86.                     ${context}_ref_rowreq_ctx *rowreq_ctx_ref);
  87. int ${context}_loop_get_next( $mfd_aue_param_decl, ${context}_ref_loop_ctx *loop_ctx_ref,
  88.                         ${context}_ref_rowreq_ctx *rowreq_ctx_ref);
  89. int ${context}_loop_get_data( $mfd_aue_param_decl, ${context}_ref_loop_ctx *loop_ctx_ref,
  90.                         ${context}_ref_rowreq_ctx *rowreq_ctx_ref);
  91. int ${context}_loop_save_position($mfd_aue_param_decl,
  92.                        ${context}_ref_loop_ctx *loop_ctx_ref,
  93.                        ${context}_ref_loop_ctx *save_loop_ctx_ref, int reuse);
  94. int ${context}_loop_cleanup_context( $mfd_aue_param_decl, ${context}_ref_loop_ctx *ref);
  95. ##
  96. @end@ // m2c_processing_type eq 'h'
  97. ########################################################################
  98. ##//####################################################################
  99. ##//@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
  100. ##//@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
  101. @if $m2c_processing_type eq 'i'@
  102. /**
  103.  * @internal
  104.  * wrapper around clean up a loop reference
  105.  */
  106. static int
  107. _${context}_loop_cleanup_context( $mfd_aue_wrap_param_decl,
  108.                             ${context}_ref_loop_ctx *ref)
  109. {
  110.    DEBUGMSGTL(("internal:${context}:_${context}_loop_cleanup_context","calledn"));
  111.    return ${context}_loop_cleanup_context($mfd_aue_wrap_param->user_ctx, ref);
  112. } /* _${context}_loop_cleanup_context */
  113. /**
  114.  * @internal
  115.  * wrapper around save position
  116.  */
  117. static int
  118. _${context}_loop_save_position( $mfd_aue_wrap_param_decl, ${context}_ref_loop_ctx *ref,
  119.                          ${context}_ref_loop_ctx *ref_copy, int reuse)
  120. {
  121.    DEBUGMSGTL(("internal:${context}:_${context}_loop_save_position","calledn"));
  122.    return ${context}_loop_save_position($mfd_aue_wrap_param->user_ctx, ref,
  123.                                  ref_copy, reuse);
  124. } /* _${context}_loop_save_position */
  125. /**
  126.  * @internal
  127.  * wrapper around user get_first to setup the index oid
  128.  */
  129. static int
  130. _${context}_loop_get_first_wrapper($mfd_aue_wrap_param_decl,
  131.                            ${context}_ref_loop_ctx * loop_ctx_ref,
  132.                            ${context}_ref_rowreq_ctx * rowreq_ctx_ref)
  133. {
  134.     int rc;
  135.     DEBUGMSGTL(("internal:${context}:_${context}_loop_get_first_wrapper","calledn"));
  136.     rc = ${context}_loop_get_first($mfd_aue_wrap_param->user_ctx, loop_ctx_ref,
  137.                                  rowreq_ctx_ref);
  138.     /*
  139.      * convert index to OID
  140.      */
  141.     if(SNMPERR_SUCCESS == rc ) {
  142.         netsnmp_assert((NULL != rowreq_ctx_ref) &&
  143.                        (rowreq_ctx_ref->rowreq_ctx->oid_idx.oids == rowreq_ctx_ref->rowreq_ctx->oid_tmp));
  144.         rowreq_ctx_ref->rowreq_ctx->oid_idx.len = sizeof(rowreq_ctx_ref->rowreq_ctx->oid_tmp);
  145.         rc = ${context}_index_to_oid(&rowreq_ctx_ref->rowreq_ctx->oid_idx,
  146.                                    &rowreq_ctx_ref->rowreq_ctx->tbl_idx);
  147.         netsnmp_assert(rowreq_ctx_ref->rowreq_ctx->oid_idx.len !=
  148.                        sizeof(rowreq_ctx_ref->rowreq_ctx->oid_tmp));
  149.     }
  150.     return rc;
  151. } /* _${context}_loop_get_first_wrapper */
  152. /**
  153.  * @internal
  154.  * wrapper around user get_next to setup the index oid
  155.  */
  156. static int
  157. _${context}_loop_get_next_wrapper($mfd_aue_wrap_param_decl,
  158.                           ${context}_ref_loop_ctx * loop_ctx_ref,
  159.                           ${context}_ref_rowreq_ctx * rowreq_ctx_ref)
  160. {
  161.     int rc;
  162.     DEBUGMSGTL(("internal:${context}:_${context}_loop_get_next_wrapper","calledn"));
  163.     rc = ${context}_loop_get_next($mfd_aue_wrap_param->user_ctx, loop_ctx_ref,
  164.                                 rowreq_ctx_ref);
  165.     /*
  166.      * convert index to OID
  167.      */
  168.     if(SNMPERR_SUCCESS == rc ) {
  169.         netsnmp_assert((NULL != rowreq_ctx_ref) &&
  170.                        (rowreq_ctx_ref->rowreq_ctx->oid_idx.oids == rowreq_ctx_ref->rowreq_ctx->oid_tmp));
  171.         rowreq_ctx_ref->rowreq_ctx->oid_idx.len = sizeof(rowreq_ctx_ref->rowreq_ctx->oid_tmp);
  172.         rc = ${context}_index_to_oid(&rowreq_ctx_ref->rowreq_ctx->oid_idx,
  173.                                    &rowreq_ctx_ref->rowreq_ctx->tbl_idx);
  174.         netsnmp_assert(rowreq_ctx_ref->rowreq_ctx->oid_idx.len !=
  175.                        sizeof(rowreq_ctx_ref->rowreq_ctx->oid_tmp));
  176.     }
  177.     return rc;
  178. } /* _${context}_loop_get_next_wrapper */
  179. @if $m2c_data_transient != 0@ # 
  180. /**
  181.  * @internal
  182.  * get data wrapper to allocate context for the user
  183.  */
  184. static int
  185. _${context}_loop_get_data_wrapper($mfd_aue_wrap_param_decl,
  186.                            ${context}_ref_loop_ctx * loop_ctx_ref,
  187.                            ${context}_ref_rowreq_ctx * rowreq_ctx_ref)
  188. {
  189. //    ${context}_rowreq_ctx *orig_ctx = rowreq_ctx_ref->rowreq_ctx;
  190.     DEBUGMSGTL(("internal:${context}:_${context}_loop_get_data_wrapper","calledn"));
  191.     return ${context}_loop_get_data($mfd_aue_wrap_param->user_ctx, loop_ctx_ref, rowreq_ctx_ref);
  192. } /* _${context}_loop_get_data_wrapper */
  193. @end@ // transient != 0
  194. /**
  195.  * @internal
  196.  * initialize the iterator container with functions or wrappers
  197.  */
  198. void
  199. _${context}_container_init(${context}_interface_ctx *if_ctx)
  200. {
  201.     DEBUGMSGTL(("internal:${context}:_${context}_container_init","calledn"));
  202.     
  203.     if_ctx->container = netsnmp_container_iterator_get(/** registration */
  204.         if_ctx,
  205.         /** compare */
  206.         NULL,
  207.         /** get_first */
  208.         (Netsnmp_Iterator_Loop_Key*)_${context}_loop_get_first_wrapper,
  209.         /** get_next */
  210.         (Netsnmp_Iterator_Loop_Key*)_${context}_loop_get_next_wrapper,
  211.         /** get_data */
  212. @if $m2c_data_transient != 0@ # 
  213.         (Netsnmp_Iterator_Loop_Data*)_${context}_loop_get_data_wrapper,
  214. @else@
  215.         NULL,
  216. @end@
  217.         /** save_pos */
  218.         (Netsnmp_Iterator_Ctx_Dup*)_${context}_loop_save_position,
  219.         /** init_context */
  220.         (Netsnmp_Iterator_Ctx*)NULL,
  221.         /** cleanup_context */
  222.         (Netsnmp_Iterator_Ctx*)_${context}_loop_cleanup_context,
  223.         /** free_user_ctx */
  224.         NULL,
  225.         /** sorted */
  226.         0);
  227. } /* _${context}_container_init */
  228. ##
  229. @end@ // m2c_processing_type eq 'i'
  230. ########################################################################
  231. ##//####################################################################
  232. ##//@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
  233. ##//@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
  234. @if $m2c_processing_type eq 'c'@
  235. /**
  236.  * unsorted-external overview
  237.  *
  238.  * The unsorted external data access code works by calling a few simple
  239.  * functions to get the index value for each row. Once the agent determines
  240.  * which row is needed to process an incoming request, another function
  241.  * is called to retrieve the data for that row.
  242.  *
  243.  * A simplified version of the pseudo-code looks like this:
  244.  *
  245.  *    ${context}_loop_get_first(loop,data)
  246.  *    while( no_error ) {
  247.  *       if( best_match(data, key)
  248.  *          ${context}_loop_save_position(loop,pos);
  249.  *       ${context}_loop_get_next(loop,data)
  250.  *    }
  251.  *    ${context}_loop_get_data(pos,data)
  252.  *    ${context}_loop_cleanup_context(loop)
  253.  */
  254. /***********************************************************************
  255.  *
  256.  * ITERATION
  257.  *
  258.  ***********************************************************************/
  259. /**
  260.  * get the first data index
  261.  *
  262.  * Summary
  263.  * -------
  264.  *  This function is called to initialize the iterator loop context for a
  265.  *  new iteration loop and return the index(es) for the first
  266.  *  ${context}_data in the data set.
  267.  *
  268.  *  Note that during the loop, the only important thing is the indexes.
  269.  *  If access to your data is cheap/fast (e.g. you have a pointer to a
  270.  *  structure in memory), it would make sense to update the data here.
  271.  *  If, however, the accessing the data invovles more work (e.g. parsing
  272.  *  some other existing data, or peforming calculations to derive the data),
  273.  *  then you should limit yourself to setting the indexes. Extracting the
  274.  *  can be put off until the desired row is found. See the notes on
  275.  *  ${context}_loop_get_data().
  276.  *
  277.  *  Note that this function does not correspond to a SNMP GET pdu, and
  278.  *  you should return data items in whatever order they are already in.
  279.  *  (In fact, if your data is already ordered in the same order as the
  280.  *  SNMP indexes, you shouldn't be using the unsorted-access code).
  281.  *
  282.  *  This function should update the table index (rowreq_ctx_ref->rowreq_ctx->tbl_idx)
  283.  *  values for the raw data (rowreq_ctx_ref->rowreq_ctx->data).
  284.  *
  285.  * More Details
  286.  * ------------
  287.  *  If there is currently no data available, return MFD_END_OF_DATA.
  288.  *  Otherwise, you should set rowreq_ctx_ref->rowreq_ctx and its indexes.
  289.  *
  290.  *  rowreq_ctx_ref->rowreq_ctx will be NULL. You should allocate a new context
  291.  *  for this loop. [Alternatively, you could allocate one in
  292.  *  ${context}_loop_init_context, save it in your
  293.  *  ${context}_ref_loop_ctx, and use it here.]
  294.  *
  295.  *  Once you have your context pointer, you should set the index (or indexes)
  296.  *  in rowreq_ctx_ref->rowreq_ctx->tbl_idx to the appropriate value for this row. [If you
  297.  *  use your loop_ctx_ref cleverly, you might be able to put this work in
  298.  *  ${context}_loop_get_next, and simply call that function.]
  299.  *
  300.  * @param $mfd_aue_param_cmt
  301.  * @param loop_ctx_ref  Pointer to your loop reference.
  302.  * @param rowreq_ctx_ref  Pointer to a context reference.
  303.  *
  304.  * @retval MFD_SUCCESS     : success.
  305.  * @retval MFD_END_OF_DATA : no data available
  306.  * @retval MFD_ERROR       : error.
  307.  */
  308. int
  309. ${context}_loop_get_first( $mfd_aue_param_decl, ${context}_ref_loop_ctx *loop_ctx_ref,
  310.                     ${context}_ref_rowreq_ctx *rowreq_ctx_ref)
  311. {
  312.     DEBUGMSGTL(("verbose:${context}:${context}_loop_get_first","calledn"));
  313.     netsnmp_assert(rowreq_ctx_ref);
  314.     netsnmp_assert(loop_ctx_ref);
  315.     
  316.     /*
  317.      * allocate memory for new structure
  318.      */
  319.     loop_ctx_ref->loop_ctx = SNMP_MALLOC_TYPEDEF(${context}_loop_context);
  320.     if(NULL == loop_ctx_ref->loop_ctx)
  321.         return MFD_ERROR;
  322.     /*
  323.      * allocate a temporary context to use during iteration
  324.      */
  325. @   eval $m2c_tmp = ""@
  326. @   if ($m2c_data_allocate == 1) || ($m2c_data_init == 1)@
  327. @      eval $m2c_tmp = "NULL"@
  328. @      if ($m2c_data_allocate == 1) && ($m2c_data_init == 1)@
  329. @         eval $m2c_tmp = "$m2c_tmp, NULL"@
  330. @      @end@
  331. @   end@
  332.     loop_ctx_ref->loop_ctx->rowreq_ctx = ${context}_allocate_rowreq_ctx($m2c_tmp);
  333.     if(NULL == loop_ctx_ref->loop_ctx->rowreq_ctx) {
  334.         SNMP_FREE(loop_ctx_ref->loop_ctx);
  335.         return MFD_RESOURCE_UNAVAILABLE;
  336.     }
  337.     /*
  338.      * TODO:
  339.      * set up loop context
  340.      */
  341. @if $m2c_include_examples == 1@
  342. $example_start
  343.     /*
  344.      * open our data file.
  345.      */
  346.     loop_ctx_ref->loop_ctx->filep = fopen("/etc/dummy.conf", "r");
  347.     if(NULL ==  loop_ctx_ref->loop_ctx->filep) {
  348.         return MFD_RESOURCE_UNAVAILABLE;
  349.     }
  350. $example_end
  351. @end@
  352. @ifconf ${table}_update_idx.m2i@
  353. @   include ${table}_update_idx.m2i@
  354. @else@
  355. @   if $m2c_include_examples == 1@
  356. $example_start
  357.     /*
  358.      * in this example, after opening the file, get next does the same thing
  359.      * as get first, we let's just call get next...
  360.      */
  361.     return ${context}_loop_get_next($mfd_aue_param, loop_ctx_ref, rowreq_ctx_ref);
  362. $example_end
  363. @   else@
  364.     /*
  365.      * we just need the index for now. Reuse the one in the loop context's
  366.      * temporary row request context. (rowreq_ctx_ref->rowreq_ctx->tbl_idx)
  367.      */
  368.     rowreq_ctx_ref->rowreq_ctx = loop_ctx_ref->loop_ctx->rowreq_ctx;
  369.     /*
  370.      * TODO:
  371.      * set local vars for index from loop_ctx_ref->loop_ctx
  372.      *  this can be done in one of two ways:
  373.      */
  374.     
  375.     /*
  376.      * 1) individually
  377.      */
  378. @   foreach $node index@
  379. @      include m2c_setup_node.m2i@
  380.         /*
  381.          * TODO:
  382.          * set rowreq_ctx_ref->rowreq_ctx->tbl_idx->$node
  383. @      if $m2c_node_needlength == 1@
  384.          *     and rowreq_ctx_ref->tbl_idx->${node}_len
  385. @      end@
  386.          */
  387. @   end@ #foreach
  388.     /*
  389.      * OR
  390.      */
  391.         /*
  392.          * 2) by calling ${context}_indexes_set()
  393.          * ${context}_indexes_set(rowreq_ctx_ref->tbl_idx,
  394. @   foreach $node index@
  395. @      include m2c_setup_node.m2i@
  396. @        if $m2c_node_needlength == 1@
  397.     *       ${node}_ptr, ${node}_len
  398. @        else@
  399.     *       $node
  400. @        end@
  401. @   end@ # foreach index
  402.     *      );
  403.     */
  404. @   end@ # example
  405. @end@ #ifconf
  406.     return MFD_SUCCESS;
  407. } /* ${context}_loop_get_first */
  408. /**
  409.  * get the next data index
  410.  *
  411.  * Summary
  412.  * -------
  413.  *  This function returns the next data item in the data set. The same
  414.  *  caveat applies here as did above. The indexes are the important parts
  415.  *  during loop processing.
  416.  *
  417.  *  Note that this function does not correspond to a SNMP GET-NEXT pdu, and
  418.  *  you should return data items in whatever order they are already in.
  419.  *  (In fact, if your data is already ordered in the same order as the
  420.  *  SNMP indexes, you shouldn't be using the unsorted-access code).
  421.  *
  422.  * More Details
  423.  * ------------
  424.  * rowreq_ctx_ref->rowreq_ctx will have been set in ${context}_loop_get_first.
  425.  *
  426.  * If there is currently no data available, return MFD_END_OF_DATA.
  427.  * Otherwise, you should set the indexes in rowreq_ctx_ref->rowreq_ctx->tbl_idx.
  428.  *
  429.  * You should set the index (or indexes) in rowreq_ctx_ref->rowreq_ctx->tbl_idx to the
  430.  * appropriate value for this row.
  431.  *
  432.  * @param $mfd_aue_param_cmt
  433.  * @param loop_ctx_ref  Pointer to your loop reference.
  434.  * @param rowreq_ctx_ref  Pointer to a context reference.
  435.  *
  436.  * @retval MFD_SUCCESS     : success.
  437.  * @retval MFD_END_OF_DATA : no more data available
  438.  * @retval MFD_ERROR       : error.
  439.  */
  440. int
  441. ${context}_loop_get_next( $mfd_aue_param_decl, ${context}_ref_loop_ctx *loop_ctx_ref,
  442.                         ${context}_ref_rowreq_ctx *rowreq_ctx_ref)
  443. {
  444.     DEBUGMSGTL(("verbose:${context}:${context}_loop_get_next","calledn"));
  445.     netsnmp_assert(loop_ctx_ref && loop_ctx_ref->loop_ctx);
  446.     netsnmp_assert(rowreq_ctx_ref);
  447.     /*
  448.      * we just need the index for now. Reuse the one in the loop context's
  449.      * temporary row request context. (rowreq_ctx_ref->rowreq_ctx->tbl_idx)
  450.      */
  451.     rowreq_ctx_ref->rowreq_ctx = loop_ctx_ref->loop_ctx->rowreq_ctx;
  452.    
  453. @   if $m2c_include_examples == 1@
  454. $example_start
  455.     /*
  456.      * get a line (skip blank lines)
  457.      */
  458.     do {
  459.         if (!fgets(loop_ctx_ref->loop_ctx->line, sizeof(loop_ctx_ref->loop_ctx->line),
  460.                    loop_ctx_ref->loop_ctx->filep)) {
  461.             /* we're done */
  462.             fclose(loop_ctx_ref->loop_ctx->filep);
  463.             loop_ctx_ref->loop_ctx->filep = NULL;
  464.         }
  465.     } while (loop_ctx_ref->loop_ctx->filep && (loop_ctx_ref->loop_ctx->line[0] == 'n'));
  466.     /*
  467.      * check for end of data
  468.      */
  469.     if(NULL == loop_ctx_ref->loop_ctx->filep)
  470.         return MFD_END_OF_DATA;
  471.     /*
  472.      * TODO:
  473.      * set local vars for index from loop_ctx_ref->loop_ctx
  474.      *  this can be done in one of two ways:
  475.      */
  476.     
  477.     /*
  478.      * 1) individually
  479.      */
  480. @   foreach $node index@
  481. @      include m2c_setup_node.m2i@
  482.         /*
  483.          * TODO:
  484.          * set rowreq_ctx_ref->rowreq_ctx->tbl_idx->$node
  485. @      if $m2c_node_needlength == 1@
  486.          *     and rowreq_ctx_ref->tbl_idx->${node}_len
  487. @      end@
  488.          */
  489. @   end@ #foreach
  490.     /*
  491.      * OR
  492.      */
  493.         /*
  494.          * 2) by calling ${context}_indexes_set()
  495.          * ${context}_indexes_set(rowreq_ctx_ref->tbl_idx,
  496. @   foreach $node index@
  497. @      include m2c_setup_node.m2i@
  498. @        if $m2c_node_needlength == 1@
  499.     *       ${node}_ptr, ${node}_len
  500. @        else@
  501.     *       $node
  502. @        end@
  503. @   end@ # foreach index
  504.     *      );
  505.     */
  506. $example_end
  507. @    end@ # example
  508.     return MFD_SUCCESS;
  509. } /* ${context}_loop_get_next */
  510. /**
  511.  * duplicate the current loop reference
  512.  *
  513.  * Summary
  514.  *  -------
  515.  *  During loop iteration, the iterator keeps track of the row that
  516.  *  is the current best match. This function is called when the
  517.  *  current row is a better match than any previous row.
  518.  *
  519.  *  You should save any information you need to be able to locate this row
  520.  *  again from the current loop context to a new loop context.
  521.  *
  522.  *  At the end of the loop, when the best match has been found, the saved
  523.  *  loop context will be used to get the data for the row by calling
  524.  *  ${context}_loop_get_data().
  525. @if $m2c_data_transient != 0@ # persistent
  526.  *
  527.  *  Since your data is transient, you need to make a copy of it before
  528.  *  the iterator moves on to the next row.
  529. @end@
  530.  *
  531. @if $m2c_data_transient != 0@ # persistent
  532.  * More Details
  533.  * ------------
  534. @   if $m2c_data_transient == 1@ # short term
  535.  *  Since your data is semi-TRANSIENT data, you could just keep a pointer
  536.  *  to the data in the loop reference. The data should then be copied in
  537.  *  ${context}_loop_get_data().
  538. @   else@ # $m2c_data_transient == 2@ # copy immediately
  539.  *  One idea would be to copy it space allocated in the loop reference
  540.  *  structure. Another would be to simply have a pointer in the loop
  541.  *  reference structure, and allocate memory here.
  542.  *
  543. @   end@
  544. @end@
  545.  * @param $mfd_aue_param_cmt
  546.  * @param loop_ctx_ref  Reference to current loop context.
  547.  * @param save_loop_ctx_ref Reference to a loop context for saving the current
  548.  *                  position. If reuse is not set or
  549.  *                  save_loop_ctx_ref->loop_ctx is NULL, allocate
  550.  *                  a new one. If reuse is set, you may reuse  the existing
  551.  *                  loop_ctx.
  552.  * @param reuse     Indicates if an existing save_loop_ctx_ref->loop_ctx
  553.  *                  may be reused.
  554.  *
  555.  * @retval MFD_SUCCESS : success.
  556.  * @retval MFD_ERROR   : error.
  557.  */
  558. int
  559. ${context}_loop_save_position($mfd_aue_param_decl,
  560.                        ${context}_ref_loop_ctx *loop_ctx_ref,
  561.                        ${context}_ref_loop_ctx *save_loop_ctx_ref,
  562.                        int reuse)
  563. {
  564.     DEBUGMSGTL(("verbose:${context}:${context}_loop_save_position","calledn"));
  565.     netsnmp_assert(loop_ctx_ref && save_loop_ctx_ref);
  566.     /*
  567.      * TODO:
  568.      * 1) allocate new loop context, unless you can reuse a previous pointer.
  569.      * 2) save information for the position of loop_ctx_ref in save_loop_ctx_ref.
  570.      */
  571.     if((0 == reuse) || (NULL == save_loop_ctx_ref->loop_ctx))
  572.         save_loop_ctx_ref->loop_ctx = SNMP_MALLOC_TYPEDEF(${context}_loop_context);
  573.     if(NULL == save_loop_ctx_ref->loop_ctx) {
  574.         snmp_log(LOG_ERR, "could not allocate memoryn");
  575.         return MFD_ERROR;
  576.     }
  577.     /*
  578.      * if you can reuse a previously saved contex, just swap
  579.      * it out with the loop iterator
  580.      */
  581.     if(reuse && save_loop_ctx_ref->loop_ctx->rowreq_ctx) {
  582.         ${context}_rowreq_ctx * tmp_rowreq_ctx = save_loop_ctx_ref->loop_ctx->rowreq_ctx;
  583.         save_loop_ctx_ref->loop_ctx->rowreq_ctx = loop_ctx_ref->loop_ctx->rowreq_ctx;
  584.         loop_ctx_ref->loop_ctx->rowreq_ctx = tmp_rowreq_ctx;
  585.     }
  586.     else {
  587.         /*
  588.          * take the current pointer
  589.          */
  590.         save_loop_ctx_ref->loop_ctx->rowreq_ctx = loop_ctx_ref->loop_ctx->rowreq_ctx;
  591.         
  592.         /*
  593.          * allocate a new context to replace the one you just took.
  594.          */
  595. @   eval $m2c_tmp = ""@
  596. @   if ($m2c_data_allocate == 1) || ($m2c_data_init == 1)@
  597. @      eval $m2c_tmp = "NULL"@
  598. @      if ($m2c_data_allocate == 1) && ($m2c_data_init == 1)@
  599. @         eval $m2c_tmp = "$m2c_tmp, NULL"@
  600. @      @end@
  601. @   end@
  602.         loop_ctx_ref->loop_ctx->rowreq_ctx = ${context}_allocate_rowreq_ctx($m2c_tmp);
  603.         if(NULL == loop_ctx_ref->loop_ctx->rowreq_ctx) {
  604.             SNMP_FREE(loop_ctx_ref->loop_ctx);
  605.             return MFD_ERROR;
  606.         }
  607.     }
  608.  
  609. @if $m2c_data_transient == 0@ # persistent
  610.     /** non-TRANSIENT data: no need to copy */
  611. @elsif $m2c_data_transient == 1@ # short term
  612.     /** semi-TRANSIENT data: will copy data when index found */
  613.     /** only need to copy pertinent data from loop context */
  614. @elsif $m2c_data_transient == 2@ # copy immediately
  615.     /*
  616.      * TRANSIENT data: copy all the data.
  617.      */
  618. @end@
  619. @if $m2c_include_examples == 1@
  620. $example_start
  621. @  if $m2c_data_transient == 1@ # short term
  622.     /** save line to do that */
  623.     memcpy(save_loop_ctx_ref->loop_ctx->line, loop_ctx_ref->loop_ctx->line,
  624.            sizeof(loop_ctx_ref->loop_ctx->line));
  625. @  elsif $m2c_data_transient == 2@ # copy immediately
  626. @    foreach $node nonindex@
  627. @      include m2c_setup_node.m2i@
  628.     /*
  629.      * TODO:
  630.      * set rowreq_ctx_ref->${m2c_data_item}$node
  631.      *     from the loop context
  632.      */
  633. @    end@
  634. @  end@
  635. $example_end
  636. @end@ # example
  637.     
  638.     return MFD_SUCCESS;
  639. } /* ${context}_loop_save_position */
  640. @if $m2c_data_transient != 0@ # semi-transient
  641. /**
  642.  * set ${context}_data from a data context
  643.  *
  644.  * Summary
  645.  * -------
  646.  *  At the end of the loop, when the best match has been found, the saved
  647.  *  loop context will be used to get the data for the row by calling
  648.  *  ${context}_loop_get_data().
  649.  *
  650.  *  You should return a fully populated row request context in
  651.  *  rowreq_ctx_ref->rowreq_ctx.
  652.  *
  653.  * More Details
  654.  * ------------
  655.  * @param $mfd_aue_param_cmt
  656.  * @param loop_ctx_ref pointer to your loop reference.
  657.  * @param rowreq_ctx_ref pointer to a context reference.
  658.  */
  659. int
  660. ${context}_loop_get_data( $mfd_aue_param_decl, ${context}_ref_loop_ctx *loop_ctx_ref,
  661.                            ${context}_ref_rowreq_ctx *rowreq_ctx_ref)
  662. {
  663.     DEBUGMSGTL(("verbose:${context}:${context}_loop_get_data","calledn"));
  664.     netsnmp_assert((NULL != loop_ctx_ref) && (NULL != loop_ctx_ref->loop_ctx));
  665.     netsnmp_assert(NULL != rowreq_ctx_ref);
  666.     netsnmp_assert(NULL != rowreq_ctx_ref->rowreq_ctx);
  667.     /*
  668.      * take temporary row request context from loop context
  669.      */
  670.     rowreq_ctx_ref->rowreq_ctx = loop_ctx_ref->loop_ctx->rowreq_ctx;
  671.     loop_ctx_ref->loop_ctx->rowreq_ctx = NULL;
  672.     
  673.     /*
  674.      * copy data to the data context (rowreq_ctx_ref->${m2c_data_item})
  675. @   if $m2c_include_examples == 1@
  676.      * in loop_save_position, we saved line to do that
  677. @   end@
  678.      */
  679. @   foreach $node nonindex@
  680. @      include m2c_setup_node.m2i@
  681.     /*
  682.      * $m2c_node_summary
  683.      */
  684. @   eval $m2c_ctx_lh = "rowreq_ctx_ref->$m2c_ctx_rh"@
  685. @   eval $m2c_ctx_lhs = "rowreq_ctx_ref->$m2c_ctx_rhs"@
  686. @   eval $m2c_ctx_rh = "loop_ctx_ref->loop_ctx->$node"@
  687. @   eval $m2c_ctx_rhs = "loop_ctx_ref->loop_ctx->${node}_len"@
  688. @   include generic-value-map.m2i@
  689.     
  690. @   end@
  691.    return MFD_SUCCESS;
  692. } /* ${context}_loop_get_data */
  693. @end@ // if $m2c_data_transient != 0
  694. /**
  695.  * clean up a loop reference
  696.  *
  697.  * Summary
  698.  * -------
  699.  *  This function will be called once the loop iteration has completed
  700.  *  to release any memory or resources allocated for the loop context.
  701.  *
  702.  * More Details
  703.  * ------------
  704.  * @param $mfd_aue_param_cmt
  705.  * @param loop_ctx_ref  Pointer to your loop reference.
  706.  *
  707.  * @retval MFD_SUCCESS : success.
  708.  * @retval MFD_ERROR   : error.
  709.  */
  710. int
  711. ${context}_loop_cleanup_context( $mfd_aue_param_decl, ${context}_ref_loop_ctx *loop_ctx_ref)
  712. {
  713.     DEBUGMSGTL(("verbose:${context}:${context}_loop_cleanup_context","calledn"));
  714.     
  715.     netsnmp_assert(loop_ctx_ref);
  716.     
  717.     if(!loop_ctx_ref->loop_ctx)
  718.         return MFD_ERROR;
  719.     /*
  720.      * release the row request context, if it wasn't taken
  721.      */
  722.     if(loop_ctx_ref->loop_ctx->rowreq_ctx)
  723.         ${context}_release_rowreq_ctx(loop_ctx_ref->loop_ctx->rowreq_ctx);
  724.     /*
  725.      * TODO:
  726.      * release resources
  727.      */
  728. @if $m2c_include_examples == 1@
  729. $example_start
  730.     /*
  731.      * close file
  732.      */
  733.     if(loop_ctx_ref->loop_ctx->filep)
  734.         fclose(loop_ctx_ref->loop_ctx->filep);
  735. $example_end
  736.     
  737. @end@
  738.     /*
  739.      * free loop context
  740.      */
  741.     free(loop_ctx_ref->loop_ctx);
  742.     
  743.     return MFD_SUCCESS;
  744. } /* ${context}_loop_cleanup_context */
  745. @end@ // m2c_processing_type eq 'c'
  746. ########################################################################
  747. ##//####################################################################
  748. ##//@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
  749. ##//@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
  750. @if $m2c_processing_type eq 'r'@
  751. ##
  752.   unsorted-external summary
  753.   -------------------------
  754.     The unsorted-external data access code is for cases when you data is
  755.     kept UNSORTED and EXTERNAL to the agent/sub-agent.
  756.     This code was generated based on the following assumptions or settings:
  757.     1) The raw data for this table is UNSORTED.
  758.     @if $mfd_readme_verbose != 0@
  759.        UNSORTED data is data that is not kept in the same order as the way
  760.        SNMP expects the index(es) for the table to be kept. [It could very
  761.        well be sorted in some other order, but for the purpose of SNMP, the
  762.        order is incorrect.]  If you're not sure if your data is sorted
  763.        in an SNMP compliant way, its likely not.
  764.        Because the raw data is unsorted, to satisfy a particular request, the
  765.        entire data set must be examined to find the apropriate index. This
  766.        is done via a simple loop. The MFD handler will call your get_first
  767.        function and the call the get_next function repeatedly, until it
  768.        returns SNMPERR_NO_VARS.
  769.     @end@
  770.     2) The raw data for this table is EXTERNAL.
  771.     @if $mfd_readme_verbose != 0@
  772.        EXTERNAL data is data that is owned by some other process,
  773.        device, file or mechanism.  The agent must use some interface to
  774.        read or modify the data.  An external process may modify the data
  775.        without the agent's knowledge. For example, the Net-SNMP agent
  776.        implements the interface table (ifTable), which reports on
  777.        network interfaces. The host operating system owns this data, and
  778.        Net-SNMP must use system calls to report or manipulate the data.
  779.        Examples of external data include data stored in kernel space, in
  780.        files, in another non-memory shared process, and data stored in
  781.        devices.
  782.     @end@
  783.     3) The raw data for this table is TRANSIENT.
  784.     @if $mfd_readme_verbose != 0@
  785.        TRANSIENT data is data that may be overwritten by another funtion
  786.        or process. For example, many OS functions return data in a
  787.        static buffer that will be reused the next time the function is
  788.        called.  Because of this, we will assume that you will copy the
  789.        raw data retrieved from these other sources to a generated
  790.        structure for use within the Net-SNMP agent.  (Don't worry, we'll
  791.        help you)
  792.     @end@
  793. ##
  794. ## this should be syncronized with master version of comments in
  795. ## mfd-access-unsorted-external-body.m2i You should be able to copy
  796. ## the comments here and replace " * " with "   ".
  797. ##
  798.   The unsorted external data access code works by calling a few simple
  799.   functions to get the index value for each row. Once the agent determines
  800.   which row is needed to process an incoming request, another function
  801.   is called to retrieve the data for that row.
  802.   A simplified version of the pseudo-code looks like this:
  803.      ${context}_loop_init_context(loop)
  804.      ${context}_loop_get_first(loop,data)
  805.      while( no_error ) {
  806.         if( best_match(data, key)
  807.            ${context}_loop_save_position(loop,pos);
  808.         ${context}_loop_get_next(loop,data)
  809.      }
  810.      ${context}_loop_get_data(pos,data)
  811.      ${context}_loop_cleanup_context(loop)
  812. ##
  813. ## end sync
  814. ##
  815.   We will talk about each individual step below.
  816. ########################################################################
  817.   Defining context for the loop
  818.   -----------------------------
  819.     TODO : typedef ${context}_loop_context
  820.     WHERE: ${table}_data_access.h
  821.     @if $mfd_readme_verbose != 0@
  822. ##
  823. ## this should be syncronized with master version of comments in
  824. ## mfd-access-unsorted-external-body.m2i You should be able to copy
  825. ## the comments here and replace " * " with "   ".
  826. ##
  827.     Since the actual loop is in the MFD handler, a loop contex parameter
  828.     is provided to help you keep track of where you are in between calls
  829.     to functions that you wrote and the master MFD handler calls. The
  830.     structure of this context is user defineable, and is defined in the
  831.     file ${table}_data_access.h.
  832.     E.G., if your data is stored in a linked list, the obvious thing you
  833.     want to know from one function call to the next is your current
  834.     position in the linked list.  Thus the easiest context to use is a
  835.     pointer within the linked list.  For an array, the current index to
  836.     that array would be easiest.
  837.     The funtion calls are actually passed a reference to the loop
  838.     context, to allow the loop context to be allocated memory. Here are
  839.     some simple examples definitions for various data formats. These
  840.     definitions are used in examples later on.
  841. ##
  842. ## end sync
  843. ##
  844.       Linked list
  845.       -----------
  846.           typedef list_node ${context}_loop_context;
  847.       Array
  848.       -----
  849.           typedef integer ${context}_loop_context;
  850.       File
  851.       ----
  852.           typedef struct ${context}_loop_context_s {
  853.                char *      file_name;
  854.                FILE *      f;
  855.                char        line[128];
  856.           } ${context}_loop_context;
  857.   @end@
  858. ########################################################################
  859.   Initialization
  860.   --------------
  861.     TODO : Initialization
  862.     FUNC : ${context}_loop_init_data
  863.     WHERE: ${table}_data_access.c
  864.     @if $mfd_readme_verbose != 0@
  865.     The ${context}_loop_init_data function will be called during startup to
  866.     allow for any initialization needed for the data access routines.
  867.     @end@
  868. ########################################################################
  869.   Preparing for the loop
  870.   ----------------------
  871.     TODO : initialize loop context
  872.     FUNC : ${context}_loop_init_context
  873.     WHERE: ${table}_data_access.c
  874.     @if $mfd_readme_verbose != 0@
  875. ##
  876. ## this should be syncronized with master version of comments in
  877. ## mfd-access-unsorted-external-body.m2i You should be able to copy
  878. ## the comments here and replace " * " with "   ".
  879. ##
  880.     This function will be called before the start of a new itertion over
  881.     the data. The loop context that is initialized here will be passed to
  882.     ${context}_loop_get_first and ${context}_loop_get_next.
  883.   
  884.     Set the loop context variable ref->loop_ctx so that the iteration
  885.     functions (get_first and get_next) can locate the apropriate data
  886.     context.
  887. ##
  888. ## end sync
  889. ##
  890.     The primary purpose of the loop_init_context call  is to initialize
  891.     the loop context data (ref). Here are some simple examples, based on the
  892.     earlier example loop contexts.
  893.       Linked list
  894.       -----------
  895.           ref->loop_ctx = my_table_head_ptr;
  896.       Array
  897.       -----
  898.           /* instead of actually allocating memory, just use the pointer */
  899.           /* as an integer */
  900.           (integer)(ref->loop_ctx) = 0;
  901.       File
  902.       ----
  903.           ref->loop_ctx = SNMP_MALLOC_TYPEDEF(${context}_loop_context);
  904.           /* error checking here */
  905.           ref->loop_ctx->file_name = (char*) reg->mfd_user_ctx;
  906.           ref->loop_ctx->f = fopen( ref->loop_ctx->file_name, "r+" );
  907.   @end@
  908. ########################################################################
  909.   The Loop
  910.   --------
  911.     TODO : return raw data
  912.     FUNC : ${context}_loop_get_first
  913.     WHERE: ${table}_data_access.c
  914.     @if $mfd_readme_verbose != 0@
  915. ##
  916. ## this should be syncronized with master version of comments in
  917. ## mfd-access-unsorted-external-body.m2i You should be able to copy
  918. ## the comments here and replace " * " with "   ".
  919. ##
  920.     This function is called to return set the index(es) for the first
  921.     ${context}_data in the data set.
  922.     Note that during the loop, the only important thing is the indexes.
  923.     If access to your data is cheap/fast (e.g. you have a pointer to a
  924.     structure in memory), it would make sense to update the data here.
  925.     If, however, the accessing the data invovles more work (e.g. parsing
  926.     some other existing data, or peforming calculations to derive the data),
  927.     then you should limit yourslef to setting the indexes. Extracting the
  928.     can be put off until the desired row is found See the notes on
  929.     ${context}_loop_get_data().
  930.     Note that this function does not correspond to a SNMP GET pdu, and
  931.     you should return data items in whatever order they are already in.
  932.     (In fact, if your data is already ordered in the same order as the
  933.     SNMP indexes, you shouldn't be using the unsorted-access code).
  934.   
  935.     This function should update the table index (rowreq_ctx_ref->rowreq_ctx->tbl_idx)
  936.     values for the raw data (rowreq_ctx_ref->rowreq_ctx->data).
  937. ##
  938. ## end sync
  939. ##
  940.       Linked list
  941.       -----------
  942.           rowreq_ctx_ref->rowreq_ctx->data = loop_ctx_ref->loop_ctx;
  943.       Array
  944.       -----
  945.           /* assuming registration has array of pointers */
  946.           rowreq_ctx_ref->rowreq_ctx->data = reg->mfd_user_ctx[(integer)(ref->loop_ctx)];
  947.       File
  948.       ----
  949.           fgets(loop_ctx_ref->loop_ctx->line, sizeof(loop_ctx_ref->loop_ctx->line),
  950.                 loop_ctx_ref->loop_ctx->f);
  951.           rowreq_ctx_ref->rowreq_ctx->data = loop_ctx_ref->loop_ctx->line;
  952.     @end@
  953.     TODO : return raw data
  954.     FUNC : ${context}_loop_get_next
  955.     WHERE: ${table}_data_access.c
  956.     @if $mfd_readme_verbose != 0@
  957. ##
  958. ## this should be syncronized with master version of comments in
  959. ## mfd-access-unsorted-external-body.m2i You should be able to copy
  960. ## the comments here and replace " * " with "   ".
  961. ##
  962.     This function returns the next data item in the data set. The same
  963.     caveat applies here as did above. The indexes are the important parts
  964.     during loop processing.
  965.     Note that this function does not correspond to a SNMP GET-NEXT pdu, and
  966.     you should return data items in whatever order they are already in.
  967.     (In fact, if your data is already ordered in the same order as the
  968.     SNMP indexes, you shouldn't be using the unsorted-access code).
  969. ##
  970. ## end sync
  971. ##
  972.       Linked list
  973.       -----------
  974.           loop_ctx_ref->loop_ctx = loop_ctx_ref->loop_ctx->next;
  975.           rowreq_ctx_ref->rowreq_ctx->data = loop_ctx_ref->loop_ctx;
  976.       Array
  977.       -----
  978.           ++((integer)(ref->loop_ctx));
  979.           /* assuming registration has array of pointers */
  980.           rowreq_ctx_ref->rowreq_ctx->data = reg->mfd_user_ctx[(integer)(ref->loop_ctx)];
  981.       File
  982.       ----
  983.           fgets(loop_ctx_ref->loop_ctx->line, sizeof(loop_ctx_ref->loop_ctx->line),
  984.                 loop_ctx_ref->loop_ctx->f);
  985.           rowreq_ctx_ref->rowreq_ctx->data = loop_ctx_ref->loop_ctx->line;
  986.     @end@
  987. ########################################################################
  988.   Updating the Index
  989.   ------------------
  990.     TODO : update index for the raw data
  991.     FUNC : ${context}_indexes_set
  992.     WHERE: ${table}_data_access.c
  993.     This is a convenience function for setting the index context from
  994.     the native C data. Where necessary, value mapping should be done.
  995.     @if $mfd_readme_verbose == 1@
  996.     This function should update the table index values (found in
  997.     tbl_idx) for the given raw data.
  998.     @end@
  999. ########################################################################
  1000.   Saving a position in the loop
  1001.   -----------------------------
  1002.     TODO : Saving a position in the loop
  1003.     FUNC : ${context}_loop_save_position
  1004.     WHERE: ${table}_data_access.c
  1005.     @if $mfd_readme_verbose != 0@
  1006. ##
  1007. ## this should be syncronized with master version of comments in
  1008. ## mfd-access-unsorted-external-body.m2i You should be able to copy
  1009. ## the comments here and replace " * " with "   ".
  1010. ##
  1011.     During loop iteration, the iterator keeps track of the row that
  1012.     is the current best match. This function is called when the
  1013.     current row is a better match than any previous row.
  1014.   
  1015.     You should save any information you need to be able to locate this row
  1016.     again from the current loop context to a new loop context.
  1017.   
  1018.     At the end of the loop, when the best match has been found, the saved
  1019.     loop context will be used to get the data for the row by calling
  1020.     ${context}_loop_get_data().
  1021. @if $m2c_data_transient != 0@ # persistent
  1022.   
  1023.     Since your data is transient, you need to make a copy of it before
  1024.     the iterator moves on to the next row.
  1025. @end@
  1026. ##
  1027. ## end sync
  1028. ##
  1029.     @end@
  1030. ########################################################################
  1031.   Returning Data For an Index
  1032.   ---------------------------
  1033.     TODO : copy transient raw data to generated structure
  1034.     FUNC : ${context}_loop_get_data
  1035.     WHERE: ${table}_data_access.c
  1036.     @if $mfd_readme_verbose != 0@
  1037. ##
  1038. ## this should be syncronized with master version of comments in
  1039. ## mfd-access-unsorted-external-body.m2i You should be able to copy
  1040. ## the comments here and replace " * " with "   ".
  1041. ##
  1042.     At the end of the loop, when the best match has been found, the saved
  1043.     loop context will be used to get the data for the row by calling
  1044.     ${context}_loop_get_data().
  1045. ##
  1046. ## end sync
  1047. ##
  1048.     @end@
  1049. ########################################################################
  1050.   Cleaning up after the loop
  1051.   --------------------------
  1052.     TODO : release any allocated memory
  1053.     FUNC : ${context}_loop_cleanup_context
  1054.     WHERE: ${table}_data_access.c
  1055.     @if $mfd_readme_verbose != 0@
  1056. ##
  1057. ## this should be syncronized with master version of comments in
  1058. ## mfd-access-unsorted-external-body.m2i You should be able to copy
  1059. ## the comments here and replace " * " with "   ".
  1060. ##
  1061.     This function will be called once the loop iteration has completed
  1062.     to release any memory allocated for loop reference.
  1063. ##
  1064. ## end sync
  1065. ##
  1066.     The purpose of the loop_cleanup_context call is to release any memory
  1067.     allocated for the loop context data. Here are some simple examples, based
  1068.     on the earlier example loop contexts.
  1069.       Linked list
  1070.       -----------
  1071.           /* nothing to do */
  1072.       Array
  1073.       -----
  1074.           /* nothing to do */
  1075.       File
  1076.       ----
  1077.           free(ref->loop_ctx);
  1078.     @end@
  1079. ##
  1080. @end@ // m2c_processing_type eq 'r
  1081. ########################################################################
  1082. @if $m2c_mark_boundary == 1@
  1083. /** END code generated by $RCSfile: mfd-access-unsorted-external-defines.m2i,v $ $Revision: 1.12 $ */
  1084. @end@