From bca673f18339b028833582c6b477f4aa0b80e532 Mon Sep 17 00:00:00 2001
From: "ezekiel.dohmen" <ezekiel.dohmen@ligo.org>
Date: Mon, 22 Apr 2024 16:37:41 -0700
Subject: [PATCH] Cleanup and adding function docs

---
 src/epics/util/lib/Statespace.pm              |   4 -
 src/fe/statespace/stateSpaceCtrl_cdev.c       |   8 +-
 src/fe/statespace/stateSpacePart.c            | 101 ----------
 .../statespace/stateSpaceCtrl_cdev.h          |  21 ++
 .../statespace/stateSpaceCtrl_ioctl.h         |  46 +++++
 .../part_headers/statespace/stateSpacePart.h  | 186 ++++++++++++++++--
 6 files changed, 243 insertions(+), 123 deletions(-)

diff --git a/src/epics/util/lib/Statespace.pm b/src/epics/util/lib/Statespace.pm
index fd8c1107c..d51fa3e11 100644
--- a/src/epics/util/lib/Statespace.pm
+++ b/src/epics/util/lib/Statespace.pm
@@ -68,10 +68,6 @@ sub printFrontEndVars  {
 # Check inputs are connected
 sub checkInputConnect {
     my ($i) = @_;
-
-    #my $ins = $::partInCnt[$::partInNum[$i][0]]; # The number of inputs to the part
-    #my $outs = $::partOutputs[$::partOutNum[$i][0]];
-
     return "";
 }
 
diff --git a/src/fe/statespace/stateSpaceCtrl_cdev.c b/src/fe/statespace/stateSpaceCtrl_cdev.c
index 75570f7d7..9ba652996 100644
--- a/src/fe/statespace/stateSpaceCtrl_cdev.c
+++ b/src/fe/statespace/stateSpaceCtrl_cdev.c
@@ -102,6 +102,7 @@ static long     stateSpaceCtrl_ioctl(struct file *file, unsigned int cmd, unsign
                 ret_val = -EFAULT;
                 goto exit_ioctl;
             }
+            ret_val = 0;
         break;
 
         case STATE_SPACE_GET_PART_NAME:
@@ -134,8 +135,8 @@ static long     stateSpaceCtrl_ioctl(struct file *file, unsigned int cmd, unsign
                 ret_val = -EINVAL;
                 goto exit_ioctl;
             }
+            ret_val = 0;
 
-            return 0;
         break;
 
         case STATE_SPACE_CONFIGURE_PART:
@@ -223,8 +224,7 @@ static long     stateSpaceCtrl_ioctl(struct file *file, unsigned int cmd, unsign
                 ret_val = -ETIMEDOUT;
                 goto exit_ioctl;
             }
-
-            return 0;
+            ret_val = 0;
 
         break;
 
@@ -260,6 +260,7 @@ static long     stateSpaceCtrl_ioctl(struct file *file, unsigned int cmd, unsign
                 goto exit_ioctl;
             }
 
+            ret_val = 0;
         break;
 
         case STATE_SPACE_GET_PART_DATA:
@@ -320,6 +321,7 @@ static long     stateSpaceCtrl_ioctl(struct file *file, unsigned int cmd, unsign
                 ret_val = -EINVAL;
                 goto exit_ioctl;
             }
+            ret_val = 0;
         break;
             
     }
diff --git a/src/fe/statespace/stateSpacePart.c b/src/fe/statespace/stateSpacePart.c
index 7d64a6bb6..3cfd7289a 100644
--- a/src/fe/statespace/stateSpacePart.c
+++ b/src/fe/statespace/stateSpacePart.c
@@ -371,104 +371,3 @@ int getConfigAndDataOfPartStateSpace(unsigned part_index, unsigned data_sz, stru
     return 0;
 }
 
-
-//
-// Start Test Code
-//
-
-//TODO: Don't build this into real models
-
-//TODO: Randomize fill order
-int testBufferOverrun(unsigned part_index)
-{
-    unsigned i, j, count = 1, errors=0;
-
-    StateSpaceCtx_t * ctx = &g_model_config.ss_parts_ptr[part_index].context;
-
-
-    //Fill vectors
-    for(i=0; i < ctx->state_len; ++i)
-        ctx->state_vec[i] = count++;
-    for(i=0; i < ctx->state_len; ++i)
-        ctx->state_vec_new[i] = count++;
-    
-    //Fill matrices 
-    for(i=0; i < ctx->state_len; ++i)
-        for(j=0; j < ctx->state_len; ++j)
-            ctx->state_matrix[i * ctx->state_len + j] = count++;
-
-    for(i=0; i < ctx->state_len; ++i)
-        for(j=0; j < ctx->in_len; ++j)
-            ctx->input_matrix[i * ctx->in_len + j] = count++;
-
-    for(i=0; i < ctx->out_len; ++i)
-        for(j=0; j < ctx->state_len; ++j)
-            ctx->output_matrix[i * ctx->state_len + j] = count++;
-
-    for(i=0; i < ctx->out_len; ++i)
-        for(j=0; j < ctx->in_len; ++j)
-            ctx->feedthrough_matrix[i * ctx->in_len + j] = count++;
-
-    //Reset count for checks
-    count = 1;
-
-    //Check vectors
-    for(i=0; i < ctx->state_len; ++i) {
-        if (ctx->state_vec[i] != count++) {
-            RTSLOG_ERROR("state_vec[%u] is %lu expected %lu\n", (uint64_t)ctx->state_vec[i], (uint64_t)count);
-            errors++;
-        }
-    }
-    for(i=0; i < ctx->state_len; ++i) {
-        if (ctx->state_vec_new[i] != count++) {
-            RTSLOG_ERROR("state_vec_new[%u] is %lu expected %lu\n", (uint64_t)ctx->state_vec_new[i], (uint64_t)count);
-            errors++;
-        }
-    }
-    
-    //Check matrices 
-    for(i=0; i < ctx->state_len; ++i) {
-        for(j=0; j < ctx->state_len; ++j) {
-            if ( ctx->state_matrix[i * ctx->state_len + j] != count++) {
-                RTSLOG_ERROR("state_matrix[%u] is %lu expected %lu\n", 
-                (uint64_t)ctx->state_matrix[i * ctx->state_len + j], (uint64_t)count);
-                errors++;
-            }
-        }
-    }
-
-    for(i=0; i < ctx->state_len; ++i) {
-        for(j=0; j < ctx->in_len; ++j) {
-            if ( ctx->input_matrix[i * ctx->in_len + j] != count++) {
-                RTSLOG_ERROR("input_matrix[%u] is %lu expected %lu\n", 
-                (uint64_t)ctx->input_matrix[i * ctx->in_len + j], (uint64_t)count);
-                errors++;
-            }
-        }
-    }
-
-    for(i=0; i < ctx->out_len; ++i) {
-        for(j=0; j < ctx->state_len; ++j) {
-            if ( ctx->output_matrix[i * ctx->state_len + j] != count++) {
-                RTSLOG_ERROR("output_matrix[%u] is %lu expected %lu\n", 
-                (uint64_t)ctx->output_matrix[i * ctx->state_len + j], (uint64_t)count);
-                errors++;
-            }
-        }
-    }
-
-    for(i=0; i < ctx->out_len; ++i) {
-        for(j=0; j < ctx->in_len; ++j) {
-            if( ctx->feedthrough_matrix[i * ctx->in_len + j] != count++)
-            {
-                RTSLOG_ERROR("feedthrough_matrix[%u] is %lu expected %lu\n", 
-                (uint64_t)ctx->feedthrough_matrix[i * ctx->in_len + j], (uint64_t)count);
-                errors++;
-            }
-        }
-    }
-
-
-    return errors;
-
-}
\ No newline at end of file
diff --git a/src/include/part_headers/statespace/stateSpaceCtrl_cdev.h b/src/include/part_headers/statespace/stateSpaceCtrl_cdev.h
index a6be9f9e2..1ed082a60 100644
--- a/src/include/part_headers/statespace/stateSpaceCtrl_cdev.h
+++ b/src/include/part_headers/statespace/stateSpaceCtrl_cdev.h
@@ -5,7 +5,28 @@
 extern "C" {
 #endif
 
+
+/**
+* @brief Initializes the char device for control over the 
+*        statespace parts.
+* 
+* @param model_name The name of the model, that will be used to 
+*                   to create the /dev/<model_name>
+*
+* @return 0 if the char device was created successfully, -1 if 
+*         there was an error.
+*/
 int initStateSpaceCtrl_cdev( const char * model_name);
+
+
+/**
+* @brief Cleans up the char device, should be
+*        called before the model is removed.
+* 
+* @param none
+*
+* @return none
+*/
 void cleanupStateSpaceCtrl_cdev( void );
 
 #ifdef __cplusplus
diff --git a/src/include/part_headers/statespace/stateSpaceCtrl_ioctl.h b/src/include/part_headers/statespace/stateSpaceCtrl_ioctl.h
index 9078b08e4..ffa791332 100644
--- a/src/include/part_headers/statespace/stateSpaceCtrl_ioctl.h
+++ b/src/include/part_headers/statespace/stateSpaceCtrl_ioctl.h
@@ -48,6 +48,16 @@ struct stateSpace_part_config_t {
     //feedthrough_matrix           [output_vec_len][input_vec_len]
 }; 
 
+/**
+* @brief Calculates the total size a stateSpace_part_config_t.data_block 
+*        would be, with the given in/out/state lengths.
+* 
+* @param input_vec_len The number of inputs the part has.
+* @param output_vec_len The number of outputs the part has.
+* @param state_vec_len The number of states the part has.
+*
+* @return The total size (in bytes) of the data_block
+*/
 static inline unsigned calcDataBlockSzStateSpace( uint32_t input_vec_len, uint32_t output_vec_len, uint32_t state_vec_len)
 {
     unsigned num_ele = state_vec_len + 
@@ -64,10 +74,46 @@ static inline unsigned calcDataBlockSzStateSpace( uint32_t input_vec_len, uint32
 //
 #define STATE_SPACE_IOCTL_MAGIC 0xC0
 
+/**
+* @brief Querys the number of statespace parts the model has.
+* 
+* @param A pointer to a stateSpace_info_t structure that will be filled.
+*/
 #define STATE_SPACE_GET_NUM_PARTS   _IOR(STATE_SPACE_IOCTL_MAGIC, 1, struct stateSpace_info_t)
+
+/**
+* @brief Querys the name of the part identified with the given index.
+* 
+* @param A pointer to a stateSpace_get_name_req_t structure that will be filled.
+*/
 #define STATE_SPACE_GET_PART_NAME   _IOWR(STATE_SPACE_IOCTL_MAGIC, 2, struct stateSpace_get_name_req_t)
+
+/**
+* @brief Requests that the given configuration be applied to the part with the given index.
+* 
+* @param A pointer to a stateSpace_part_config_t that contains the requested configuration.
+*/
 #define STATE_SPACE_CONFIGURE_PART  _IOW(STATE_SPACE_IOCTL_MAGIC, 3, struct stateSpace_part_config_t)
+
+/**
+* @brief Reads out the part with the given index's configuration (ioctl_version through 
+*        state_vec_len) in the stateSpace_part_config_t struct.
+*        
+*        stateSpace_part_config_t.part_index must be set to the requested part. 
+* 
+* @param A pointer to a stateSpace_part_config_t that will be filled.
+*/
 #define STATE_SPACE_GET_PART_CONFIG _IOR(STATE_SPACE_IOCTL_MAGIC, 4, struct stateSpace_part_config_t)
+
+/**
+* @brief Reads out the part with the given index's configuration and data
+*        (ioctl_version through data_block) in the stateSpace_part_config_t struct.
+*        
+*        stateSpace_part_config_t.part_index must be set to the requested part. 
+* 
+* @param A pointer to a stateSpace_part_config_t that will be filled. The size of the struct
+*        should be calculated with the calcDataBlockSzStateSpace(...) function.
+*/
 #define STATE_SPACE_GET_PART_DATA   _IOR(STATE_SPACE_IOCTL_MAGIC, 5, struct stateSpace_part_config_t)
 
 
diff --git a/src/include/part_headers/statespace/stateSpacePart.h b/src/include/part_headers/statespace/stateSpacePart.h
index 7c85dfef2..ab8d6198b 100644
--- a/src/include/part_headers/statespace/stateSpacePart.h
+++ b/src/include/part_headers/statespace/stateSpacePart.h
@@ -13,47 +13,203 @@ typedef struct StateSpaceCtx_t
     unsigned out_len;
     unsigned state_len;
 
-    unsigned buffer_sz;
-    //Start of memory block pointers
+    //Stores the size of the data block allocation below, in bytes.
+    unsigned buffer_sz; 
+
+    //Start of memory block pointers, all pointers point to different 
+    //offsets in the same data block.
     double *mem_block_ptr;
     double *state_vec;  //[state_len]
     double *state_vec_new;  //[state_len]
 
-    //TODO: Make sure these dim are ordered for the iteration we do over them
+    //TODO: Make sure these dim are ordered 
+    //for the iteration we do over them
     double *state_matrix; // [state_len][state_len]
     double *input_matrix; // [state_len][in_len]
     double *output_matrix;// [out_len][state_len]
     double *feedthrough_matrix; //Zero or [out_len][in_vec]
 } StateSpaceCtx_t;
 
+//
+// Functions to be called during model setup, before isolation
+//
 
-//Functions to be called during model setup
+/**
+* @brief Initializes the global data structures for statespace parts
+*        in the model.
+* 
+* @param num_parts The number of statespace parts the model has.
+*
+* @return 0 if the data structures were initialized successfully,
+*        -1 if an error occurred.
+*/
 int globStateSpaceInit( unsigned num_parts );
-int globStateSpaceConfigPart( unsigned part_index, const char name [], unsigned num_in, unsigned num_out );
+
+/**
+* @brief Frees all statespace part allocations.
+*
+* This should be called after the RT logic has been stopped,
+* and before model removal, it will free all parts and temp
+* allocations.        
+*/
 void globStateSpaceFree( void );
 
+//
+// Functions called in RT Code
+//
+
+/**
+* @brief Initializes the specific part (by index), 
+*        and gives it the passed name and num ins/outs.
+*
+* AUTO-GENERATED call: This function is generated once for each 
+* statespace part in the model, and is inserted into the feCode() init call.
+* 
+* @param part_index The index of the statespace part to configure.
+* @param name The name the part will have. (Used to configure by name)
+* @param num_in The numer of inputs the part has.
+* @param num_out The number of outputs the part has.
+*
+* @return 0 if the configuration was set successfully,
+*        -1 if an error occurred.
+*/
+int globStateSpaceConfigPart( unsigned part_index, const char name [], unsigned num_in, unsigned num_out );
+
 
-//Functions called in feCode()
+/**
+* @brief Calculates the next timestep of data with the given inputs,
+*        internal state, and producing outputs.
+*
+* AUTO-GENERATED call:
+* 
+* @param part_index The index of the part to process.
+*
+* @param reset If >0 the state vector is cleared (0ed out) 
+*              before the cycle's calculation.
+*
+* @param input_vec A pointer to a double buffer where inputs will be read from.
+*                  The buffer length must match the num_in parameter passed 
+*                  to globStateSpaceConfigPart()
+*
+* @param out_vec   A pointer to a double buffer where outputs will be written to.
+*                  The buffer length must match the num_out parameter passed 
+*                  to globStateSpaceConfigPart()
+* 
+* @return none
+*/
+void calcStateSpace(unsigned part_index, int reset, double * input_vec, double * out_vec);
+
+/**
+* @brief Sets the identified part to the given state vector.
+*
+* @param part_index The index of the statespace part to reset.
+* @param state_vec A pointer to double data to load on reset.
+*
+* @return none
+*/
+void setStateVecStateSpace(unsigned part_index, double * state_vec);
+
+//
+// IOCTL Servicing Functions
+//
+
+/**
+* @brief Gets the name of the part with the given index.
+*
+* @param none
+*
+* @return The number of statespace parts in the model.
+*/
+unsigned getNumPartsStateSpace( void );
+
+/**
+* @brief Gets the current configuration of the part with the given index.
+*
+* @param part_index The index of the part that should be (re)configured. 
+* @param input_vec_len The number of inputs (fixed).
+* @param output_vec_len The number of outputs (fixed).
+* @param state_vec_len The number of states
+* @param initial_state_vec A ptr to a buffer of doubles (state_vec_len long)
+*                          that will be used as the initial state for the part. 
+* @param state_matrix A ptr to a buffer of doubles for the flattened (row-major) 
+*                     state matrix. Length must be state_vec_len * state_vec_len.
+* @param input_matrix A ptr to a buffer of doubles for the flattened (row-major) 
+*                     input matrix. Length must be input_vec_len * state_vec_len.
+* @param output_matrix A ptr to a buffer of doubles for the flattened (row-major) 
+*                     output matrix. Length must be output_vec_len * state_vec_len.
+* @param feedthrough_matrix A ptr to a buffer of doubles for the flattened (row-major) 
+*                     feedthrough matrix. Length is input_vec_len * output_vec_len 
+*                     or NULL if the feedthrough is 0 for this configuration.
+*
+* @return 0 if the new configuration was loaded, -1 if the lengths were wrong etc.
+*/
 int matrixConfigureStateSpace(unsigned part_index, 
                           unsigned input_vec_len, unsigned output_vec_len, unsigned state_vec_len,
                           const double * initial_state_vec,
                           const double * state_matrix, const double * input_matrix,  
                           const double * output_matrix, const double * feedthrough_matrix);
-void calcStateSpace(unsigned part_index, int reset, double * input_vec, double * out_vec);
-void setStateVecStateSpace(unsigned part_index, double * state_vec);
 
-//IOCTL Servicing Functions
-unsigned getNumPartsStateSpace( void );
+
+/**
+* @brief Gets the name of the part with the given index.
+*
+* @param part_index The index of the part we want the name of. 
+* @param name_ptr A pointer to the name of the part. This is for read only. 
+*
+* @return none
+*/
 void getPartNamePtrStateSpace( unsigned part_index, char ** name_ptr );
-void getConfigOfPartStateSpace(unsigned part_index, struct stateSpace_part_config_t * );
+
+/**
+* @brief Gets the current configuration of the part with the given index.
+*
+* @param part_index The index of the part that should be loaded. 
+* @param config A pointer to a buffer to copy the configuration.
+*
+* @return none
+*/
+void getConfigOfPartStateSpace(unsigned part_index, struct stateSpace_part_config_t * config );
+
+/**
+* @brief Copies the part config and data block into the buffer pointed to
+*        by fill_me.
+*
+* @param part_index The index of the statespace part to read.
+* @param data_sz The size of the buffer pointed to by fill_me.
+* @param fill_me A pointer to a buffer where the part config should be copied.
+*
+* @return -1 if data for part is not available
+*/
+int getConfigAndDataOfPartStateSpace(unsigned part_index, unsigned data_sz, struct stateSpace_part_config_t * fill_me);
+
+/**
+* @brief Loads the new configuration (now). 
+*
+* This is used by librts to load swap the pointers and load 
+* the part config immediately.
+*
+* @param part_index The index of the part that should be loaded. 
+*
+* @return none
+*/
 void loadNewMatrixNowStateSpace( unsigned part_index );
+
+/**
+* @brief Checks if a new configuration is queued and waiting for the RT
+*        code to load.
+*
+* Once a new configuration is loaded by matrixConfigureStateSpace(), 
+* the new data is stored and the RT code is responsible to sawp out
+* the pointers before its next cycle. This is used to signal, when 
+* that swap is complete.
+*
+* @param none
+*
+* @return -1 if nothing is queued, the part index we are waiting on otherwise.
+*/
 int check_pending_config( void );
 
-// @return -1 if data for part is not available
-int getConfigAndDataOfPartStateSpace(unsigned part_index, unsigned data_sz, struct stateSpace_part_config_t * fill_me);
 
-//Test Methods
-int testBufferOverrun(unsigned part_index);
 
 #ifdef __cplusplus
 }
-- 
GitLab