#include "cwCommon.h"
#include "cwLog.h"
#include "cwCommonImpl.h"
#include "cwMem.h"
#include "cwText.h"
#include "cwFile.h"
#include "cwObject.h"
#include "cwMidi.h"
#include "cwDynRefTbl.h"
#include "cwScoreParse.h"
#include "cwSfScore.h"
#include "cwPerfMeas.h"
#include "cwVectOps.h"

/*

1. if a performed event is part of a set then mark the set complete when all events in it have been performed.
2. when a set is marked as complete it should be evaluated immediately.
3. once a complete set is evaluated it is never evaluated again (although incomplete sets may be evaluated again)

4. if a performed location is marked as a 'calc' location and all sets have been evaluated then the calc. is executed.
5. if a performed location is past the next-calc-loc then the calc is executed.
6. if a calc has already been executed and an event that is part of the calc is recieved then the
associated set is updated and the calc is executed again.

7. if a performed event occurs on a trigger location then the section update is executed.
8. if a performed event occurs on a trigger location that has already been executed then the section update is NOT
performed again.
  
 */

namespace cw
{
  namespace perf_meas
  {
    struct calc_str;
    
    typedef struct section_str
    {      
      const struct section_str* prev_section; // section prior to this section
      const sfscore::section_t* section;      // score section this section_t represents
      struct calc_str*          calc;         // calc record for this section
      bool                      triggeredFl;  // This section has already been triggered
    } section_t;

    typedef struct meas_info_str
    {
      unsigned              missEvtN;      // count of skipped events
      unsigned              missLocN;      // count of missed locations
      unsigned              transposeLocN; // count of transposed locations
      unsigned              interpLocN;    // count of interpolated locations

      unsigned vN;  // vN=locN for even,tempo vN=evtN for dyn
      
      double*               locSecV;       // locSecV[vN]
      unsigned*             locSecNV;      // locSecNV[vN] count of performed and interpolated notes (interpolated notes will have a False in the associated statusV[] location)
      double*               dSecV;         // dSecV[vN]
      bool*                 statusV;       // statusV[vN] - true if this location was performed by at least one note
      struct meas_info_str* link;
    } meas_info_t;

    struct loc_str;
    
    typedef struct set_str
    {
      struct loc_str*       loc;       // end location for this set
      const sfscore::set_t* set;       // score set this set_t represents
      meas_info_t*          meas_info; // 
      unsigned              lastPerfUpdateCnt; 
      double                value;     // The value associated with this set. DBL_MAX on initialization
      struct calc_str*      calc;      // calc record to which this set belongs
      struct set_str*       alink;     // perf_meas_t* links
      struct set_str*       slink;     // loc_t.setL links
      struct set_str*       clink;     // calc_t.setL links
    } set_t;

    // The 'calc_t' record hold pointers to all the sets assigned to a given section.
    // The record is different from a section record because it has a location assignment
    // prior to the section which it is represents. This allows all the sets for
    // a given section to be evaluated prior to the section being triggered.
    // The 'calc' record is assigned to the location of the last event in it's last set. 
    typedef struct calc_str
    {
      set_t*     setL;             // list of sets that this calc is applied to (links via set_t.clink)
      section_t* section;          // section where this calc is applied
      double     value[ kValCnt ]; // Aggregate var values for this section
    } calc_t;
    
    typedef struct loc_str
    {
      unsigned   locId;         // oloc location id and the index of this record into p->locA[]
      section_t* section;       // section that begins on this location
      set_t*     setL;          // sets that end on this location (links via loc_t.slink)
      calc_t*    calc;          // calc that can be completed on this location.
    } loc_t;
    
    typedef struct perf_meas_str
    {
      params_t          params;
      sfscore::handle_t scoreH;

      loc_t*            locA;   // locA[locN] 
      unsigned          locN;   // length of locA[] 
      set_t*            setL;   // sets linked on alink

      // Location index of the next section to be triggered.
      unsigned          last_section_loc_idx;

      // Location of the next calc record to be evaluated.
      unsigned          next_section_loc_idx; 

      unsigned          next_calc_loc_idx;
      
    } perf_meas_t;

    perf_meas_t* _handleToPtr( handle_t h )
    { return handleToPtr<handle_t,perf_meas_t>(h); }

    inline unsigned _loc_id_to_index( perf_meas_t* p, unsigned locId )
    {
      if( locId >= p->locN  )
        return kInvalidIdx;
      
      assert( locId == p->locA[locId].locId );
      
      return locId;
    }

    void _destroy_meas_info_records( set_t* set )
    {
      meas_info_t* m = set->meas_info;
      while( m!=nullptr )
      {
        meas_info_t* m0 = m->link;
        mem::release(m->locSecV);
        mem::release(m->locSecNV);
        mem::release(m->dSecV);
        mem::release(m->statusV);
        mem::release(m);
        m = m0;
      }
      
    }
    
    rc_t _destroy( perf_meas_t* p )
    {
      rc_t rc = kOkRC;

      for(unsigned i=0; i<p->locN; ++i)
      {
        set_t* s = p->locA[i].setL;
        while(s!=nullptr)
        {
          set_t* s0 = s->slink;
          
          _destroy_meas_info_records(s);
          
          mem::release(s);
          s = s0;
        }
        
        mem::release(p->locA[i].section);
        mem::release(p->locA[i].calc);
      }

      mem::release(p->locA);
      mem::release(p);
      return rc;
    }

    set_t* _find_set( perf_meas_t* p, unsigned setId )
    {
      for(set_t* s=p->setL; s!=nullptr; s=s->alink)
        if( s->set->id == setId )
          return s;

      cwLogError(kInvalidIdRC,"The setId '%i' is not valid.", setId );
      return nullptr;
    }

    meas_info_t* _create_meas_info_record( set_t* set )
    {
      meas_info_t* m = mem::allocZ<meas_info_t>();
      m->vN          = set->set->varId==score_parse::kDynVarIdx ? set->set->evtCnt :  set->set->locN;
      m->locSecV     = mem::allocZ<double>( m->vN );
      m->locSecNV    = mem::allocZ<unsigned>( m->vN );
      m->dSecV       = mem::allocZ<double>( m->vN );
      m->statusV     = mem::allocZ<bool>( m->vN );
      
      if( set->meas_info == nullptr )
        set->meas_info = m;
      else
      {
        meas_info_t* m0 = set->meas_info;
        while( m0->link != nullptr )
          m0 = m0->link;

        m0->link = m;
      }

      return m;
    }
    
    calc_t* _create_calc_record( perf_meas_t* p, sfscore::section_t* section )
    {
      rc_t rc = kOkRC;

      calc_t* calc = mem::allocZ<calc_t>();
      
      for(unsigned i=0; i<section->setCnt; ++i)
      {
        set_t* s;
        if((s = _find_set(p,section->setArray[i]->id)) == nullptr )
        {
          rc = cwLogError(kInvalidIdRC,"Error forming section set list for the section '%s'.",cwStringNullGuard(section->label));
          goto errLabel;
        }

        s->calc  = calc;
        s->clink = calc->setL;
        calc->setL = s;        
      }

    errLabel:

      if( rc != kOkRC )
        mem::release(calc);
      
      return calc;
    }

    void _advance_next_calc_and_section_indexes( perf_meas_t* p, unsigned cur_section_idx )
    {
      p->last_section_loc_idx = cur_section_idx;
      p->next_calc_loc_idx    = kInvalidIdx;
      p->next_section_loc_idx = kInvalidIdx;
      
      // Set 'last_section_loc_idx' and 'next_calc_loc_idx'
      for(unsigned i=cur_section_idx+1; i<p->locN && (p->next_section_loc_idx==kInvalidIdx || p->next_calc_loc_idx==kInvalidIdx); ++i)
      {
        if( p->next_section_loc_idx == kInvalidIdx && p->locA[i].section != nullptr )
          p->next_section_loc_idx = i;

        if( p->next_calc_loc_idx == kInvalidIdx && p->locA[i].calc != nullptr )
          p->next_calc_loc_idx = i;
      }

      if( p->next_section_loc_idx == kInvalidIdx )
        cwLogInfo("End-of-score reached on section index.");
      else
        if( p->next_calc_loc_idx == kInvalidIdx )
          cwLogWarning("No 'next calc index' was found for the section at location:%i",p->next_section_loc_idx);
    }

    void _reset_loc( loc_t& loc)
    {
      if( loc.section != nullptr )
        loc.section->triggeredFl = false;
      
      if( loc.calc != nullptr )
      {
        for(set_t* set = loc.calc->setL; set!=nullptr; set=set->clink)
        {
          set->value = std::numeric_limits<double>::max();
          set->lastPerfUpdateCnt = 0;
        }
        std::for_each(loc.calc->value,loc.calc->value+kValCnt,[](double& x){ x=std::numeric_limits<double>::max(); });
      }
    }
    
    rc_t _reset( perf_meas_t* p, unsigned init_locId )
    {
      rc_t     rc = kOkRC;
      unsigned i  = 0;

      unsigned init_loc_idx = kInvalidIdx;
      
      for(; i<p->locN; ++i)
      {
        if( p->locA[i].locId == init_locId )
          init_loc_idx = i;

        // reset all locations at and after the init. loc
        if( init_loc_idx != kInvalidIdx )
          _reset_loc(p->locA[i]);
        
      }
      
      if( init_loc_idx == kInvalidIdx )
      {
        rc = cwLogError(kInvalidIdRC,"The initial location id '%i' is not valid.",init_locId);
        goto errLabel;
      }

      _advance_next_calc_and_section_indexes(p,init_loc_idx);
  
    errLabel:
      return rc;
    }

    double _calc_set_list_mean( const set_t* setL, unsigned varTypeId )
    {
      double sum = 0.0;
      unsigned n = 0;
      
      for(const set_t* s=setL; s!=nullptr; s=s->clink)
        if( s->set->varId == varTypeId && s->value != std::numeric_limits<double>::max() )          
        {
          sum += s->value;
          n   += 1;
        }

      return n>0 ? sum/n : 0.0;
    }

    void _eval_one_dynamic_set(perf_meas_t* p, set_t* set )
    {
      double sum = 0;
      unsigned n = 0;
      for(unsigned i=0; i<set->set->evtCnt; ++i)
      {
        const sfscore::event_t* e = set->set->evtArray[i];
        if( e->perfFl )
        {
          sum += e->dynLevel>e->perfDynLevel ? e->dynLevel - e->perfDynLevel : e->perfDynLevel - e->dynLevel;
          n += 1;
        }
      }

      set->value = n==0 ? 0 : sum/n;
    }

    // Calc chord onset time as the avg. onset over of all notes in the chord.
    rc_t _calc_loc_sec( set_t* pm_set  )
    {
      rc_t rc = kOkRC;

      const sfscore::set_t* set = pm_set->set;
      meas_info_t*          m   = pm_set->meas_info;
      
      m->missEvtN = 0;

      assert(  set->locN == m->vN );
      
      vop::zero(m->locSecV,  set->locN);
      vop::zero(m->locSecNV, set->locN);
      
      // Get the time for each location by taking the mean
      // of all events on the same location.
      unsigned li       = 0;
      unsigned curLocId = set->evtArray[0]->oLocId;
      
      for(unsigned i=0; i<set->evtCnt; ++i)
      {
        if( !set->evtArray[i]->perfFl )
        {
          ++m->missEvtN;
        }
        else
        {        
          if( curLocId != set->evtArray[i]->oLocId )
          {
            li += 1;
            curLocId = set->evtArray[i]->oLocId;
          }

          if( li >= set->locN )
          {
            rc = cwLogError(kAssertFailRC,"An invalid location id was encountered %i >= %i.",li,set->locN);
            goto errLabel;
          }

          m->locSecV[ li] += set->evtArray[i]->perfSec;
          m->locSecNV[li] += 1;
        }
      }

      // Calc. mean.
      for(unsigned i=0; i<set->locN; ++i)
        if( m->locSecNV[i] > 0 )
          m->locSecV[i] /= m->locSecNV[i];
      
    errLabel:
      return rc;
    }

    void _interpolate_time_of_missing_notes( perf_meas_t* p, set_t* pm_set, unsigned& bi_ref, unsigned& ei_ref )
    {
      bi_ref      = kInvalidIdx;
      ei_ref      = kInvalidIdx;
      
      const sfscore::set_t* set = pm_set->set;
      meas_info_t* m = pm_set->meas_info;
      unsigned missN = 0;
      unsigned ei    = kInvalidIdx;
      unsigned bi    = kInvalidIdx;

      assert(  set->locN == m->vN );
      
      vop::fill(m->statusV, set->locN, false);
   
      // for each location
      for(unsigned i=0; i<set->locN; ++i)
      {
        // if this location was missed 
        if( m->locSecNV[i] == 0   )
        {
          missN += 1;
          m->missLocN += 1;
          continue;
        }

        // if this location was not in time order
        if( ei != kInvalidIdx && (m->locSecV[i]-m->locSecV[ei])<0 )
        {
          missN += 1;
          m->transposeLocN += 1;
          continue;
        }

        
        // if there are unplayed notes between this note
        // and the last played note at 'ei' then 
        // fill in the missing note times by splitting
        // the gap time evenly - note that this will
        // bias the evenness std to be lower than it
        // should be.
        if( missN > 0 && ei!=kInvalidIdx )
        {            
          double dsec = (m->locSecV[i] - m->locSecV[ei])/(missN+1);
          for(unsigned j=ei+1; j<i; ++j)
          {
            m->locSecV[j] = m->locSecV[j-1] + dsec;
            m->interpLocN += 1;
          }
        }

        if( bi == kInvalidIdx )
          bi = i;
          
        m->statusV[i] = true;
        missN = 0;
        ei    = i;  // ei=last valid location in m->locSecV[]
        
      }
      
      bi_ref = bi;
      ei_ref = ei;

      //vop::print( m->locSecV,  set->locN, "%f ", "locSecV:" );
      //vop::print( m->locSecNV, set->locN, "%i ", "locSecN:" );
      //vop::print( m->statusV,  set->locN, "%i ", "ok:");
        
    }

    rc_t _eval_one_even_set(perf_meas_t* p, set_t* pm_set )
    {
      rc_t                  rc  = kOkRC;      
      const sfscore::set_t* set = pm_set->set;      
      unsigned              bi  = kInvalidIdx;
      unsigned              ei  = kInvalidIdx;
      meas_info_t*          m   = pm_set->meas_info;

      assert(  set->locN == m->vN );
      
      if((rc = _calc_loc_sec(pm_set)) != kOkRC )
        goto errLabel;

      _interpolate_time_of_missing_notes(p, pm_set, bi, ei );
      
      vop::zero(m->dSecV, set->locN);

      // Calc the std. deviation of the note delta times
      // of all notes [bi:ei].  Note that if the notes
      // before bi/after ei  were skipped then they are
      // left out of the calculation.  However skipped notes
      // interal to the range bi:ei were interpolated
      // as perfectly even.
      
      if( (ei - bi)+1 > 2 )
      {
        // calc the delta time for each time in locV[]
        unsigned stdN = 0;
        for(unsigned i=bi+1; i<=ei; ++i)
          m->dSecV[stdN++] = m->locSecV[i] - m->locSecV[i-1];
        
        
        //printf("Even: skipped evt:%i skipped locs:%i interp:%i bi:%i ei:%i final N:%i\n",m->missEvtN,m->missLocN,m->interpLocN,bi,ei,stdN);
        //vop::print( m->dSecV, stdN, "%f ", "std:" );
            
        pm_set->value = vop::std(m->dSecV, stdN );
      }

    errLabel:
      return rc;      
    }

    
    rc_t _eval_one_tempo_set(perf_meas_t* p, set_t* pm_set )
    {
      rc_t                  rc  = kOkRC;      
      const sfscore::set_t* set = pm_set->set;      
      meas_info_t*          m   = pm_set->meas_info;
      unsigned              bi  = kInvalidIdx;
      unsigned              ei  = kInvalidIdx;

      assert(  set->locN == m->vN );
      
      if((rc = _calc_loc_sec(pm_set)) != kOkRC )
        goto errLabel;

      _interpolate_time_of_missing_notes(p, pm_set, bi, ei );
      
      vop::zero(m->dSecV, set->locN);

      if( (ei - bi)+1 > 2 )
      {
        // calc the delta time for each time in locV[]
        unsigned dsecN = 0;
        for(unsigned i=bi+1; i<=ei; ++i)
          m->dSecV[dsecN++] = m->locSecV[i] - m->locSecV[i-1];
                
        //printf("Tempo: skipped evt:%i skipped locs:%i interp:%i bi:%i ei:%i final N:%i\n",m->missEvtN,m->missLocN,m->interpLocN,bi,ei,dsecN);
        //vop::print( m->dSecV, dsecN, "%f ", "dsecV:" );

        double sec_per_beat = vop::mean(m->dSecV, dsecN );
        double bpm = 60.0/sec_per_beat;
        
        
        pm_set->value = bpm;
      }

    errLabel:
      return rc;      
    }

    void _aggregate_dynamic_meas_set_values( perf_meas_t* p, calc_t* calc )
    {
      calc->value[ kDynValIdx ] = _calc_set_list_mean( calc->setL, score_parse::kDynVarIdx );
    }
    
    void _aggregate_even_meas_set_values( perf_meas_t* p, calc_t* calc )
    {
      calc->value[ kEvenValIdx ] = _calc_set_list_mean( calc->setL, score_parse::kEvenVarIdx );
    }
    
    void _aggregate_tempo_meas_set_values( perf_meas_t* p, calc_t* calc )
    {
      calc->value[ kTempoValIdx ] = _calc_set_list_mean( calc->setL, score_parse::kTempoVarIdx );
    }

    void _aggregate_cost_meas_set_values( perf_meas_t* p, calc_t* calc )
    {
      if( calc->section->prev_section != nullptr )
      {
        unsigned              beg_loc_idx = calc->section->prev_section->section->locPtr->index;
        unsigned              end_loc_idx = calc->section->section->locPtr->index;
        const sfscore::loc_t* loc         = loc_base(p->scoreH) + beg_loc_idx;
        const sfscore::loc_t* loc_end     = loc_base(p->scoreH) + end_loc_idx;
        double                sum         = 0;
        unsigned              n           = 0;
        
        for(; loc<loc_end; ++loc)
          for(unsigned i=0; i<loc->evtCnt; ++i)
            if( loc->evtArray[i]->perfFl && loc->evtArray[i]->perfMatchCost != std::numeric_limits<double>::max() )
            {
              sum += loc->evtArray[i]->perfMatchCost;
              n   += 1;
            }
        
        
        calc->value[ kMatchCostValIdx ] = n==0 ? 0 : sum / n;
      }
    }

    set_t* _use_slink(set_t* s){ return s->slink; }
    set_t* _use_clink(set_t* s){ return s->clink; }

    // Set 'force_fl' to true if the sets should be evaluated even if they are not complete.
    void _eval_set_list( perf_meas_t* p, set_t* setL, set_t* (*link_func)(set_t*)  )
    {
      // for each set at this location
      for(set_t* s=setL; s!=nullptr; s=link_func(s))
      {
        // if this set has not been eval'd or has been updated with new perf. information since it was last eval'd
        if( s->lastPerfUpdateCnt==0 || s->set->perfUpdateCnt > s->lastPerfUpdateCnt )
        {
          s->lastPerfUpdateCnt = s->set->perfUpdateCnt;

          if( p->params.print_rt_events_fl )
            cwLogInfo("Set %i eval.",s->set->id);
          
          switch( s->set->varId )
          {
            case score_parse::kDynVarIdx:
              _eval_one_dynamic_set(p,s);
              break;
              
            case score_parse::kEvenVarIdx:
              _eval_one_even_set(p,s);
              break;
              
            case score_parse::kTempoVarIdx:
              _eval_one_tempo_set(p,s);
              break;

            default:
              cwLogError(kInvalidIdRC,"Unknown var type (%i) encountered while evaluating sets.",s->set->varId );
              assert(0);
          }
        }
      }
    }              
    
    rc_t _update_sets( perf_meas_t* p, unsigned loc_idx )
    {
      rc_t rc = kOkRC;
      if( p->locA[loc_idx].setL != nullptr && are_all_loc_set_events_performed( p->scoreH, loc_idx ) )
        _eval_set_list(p,p->locA[loc_idx].setL,_use_slink);
      
      return rc;      
    }

    rc_t _update_calc( perf_meas_t* p, unsigned loc_idx )
    {
      rc_t rc = kOkRC;
      // if the loc is past the next_calc_loc_idx or is equal to next_calc_loc_idx and all the set events have been performed for the location
      if( p->next_calc_loc_idx != kInvalidIdx
          && (loc_idx > p->next_calc_loc_idx
              || (loc_idx == p->next_calc_loc_idx
                  && are_all_loc_set_events_performed( p->scoreH, loc_idx ))))
      {

        if( p->params.print_rt_events_fl )
          cwLogInfo("Calc: Loc:%i  nci:%i",loc_idx,p->next_calc_loc_idx );
        
        calc_t* calc = p->locA[ p->next_calc_loc_idx ].calc;
        assert( calc != nullptr );

        // eval any sets that have not already been evaluated - even if they are not complete
        _eval_set_list(p,calc->setL, _use_clink);

        // aggregate the measurements from each measurement type into a single scalar values.
        _aggregate_dynamic_meas_set_values(p,calc);
        _aggregate_even_meas_set_values(p,calc);
        _aggregate_tempo_meas_set_values(p,calc);
        _aggregate_cost_meas_set_values(p,calc);
      }
      return rc;
    }

    rc_t _update_section( perf_meas_t* p, unsigned loc_idx, result_t& resultRef )
    {
      rc_t rc = kOkRC;

      // if next_section_loc_idx == kInvalidIdx then the end of the score has been encountered
      // (or we are in an invalid state)
      if( p->next_section_loc_idx == kInvalidIdx )
      {
        cwLogWarning("End-of-score or invalid state encountered (next_section_loc_idx==kInvalidIdx).");
        goto errLabel;
      }

      // TODO: Should the following be a loop which iterates
      // until p->next_section_loc_idx > loc_idx
      
      if( p->next_section_loc_idx <= loc_idx )
      {      
        assert( p->locA[p->next_section_loc_idx].section != nullptr );

        section_t* section = p->locA[p->next_section_loc_idx].section;

        if( section->triggeredFl == false )
        {
          assert( section->section != nullptr && section->section->locPtr != nullptr );
          
          resultRef.loc          = loc_idx;
          resultRef.sectionLoc   = section->section->locPtr->index;
          resultRef.sectionLabel = section->section->label;
          if( section->calc == nullptr )
            cwLogWarning("No sets assigned to section %s",cwStringNullGuard(section->section->label));
          else
          {
            resultRef.valueA       = section->calc->value;
            resultRef.valueN       = kValCnt;
          }
        
          section->triggeredFl = true;
          
          _advance_next_calc_and_section_indexes(p,p->next_section_loc_idx);

          //cwLogInfo("Section %s triggered.",cwStringNullGuard(resultRef.sectionLabel));
          
        }
      }
    errLabel:
      return rc;
    }

    template< typename T >
    rc_t _write_result_vector(file::handle_t fH, section_t* section, set_t* set, const char* opLabel, const char* fmt, const T* vect, unsigned locN)
    {
      rc_t rc;
      
      if((rc = file::printf(fH,"%s,%i,%s,%s,%i,",
                            section->section->label,
                            set==nullptr ? -1 : set->set->id,
                            set==nullptr ? "" : score_parse::var_index_to_char(set->set->varId),
                            opLabel,
                            locN)) != kOkRC )
      {
        rc = cwLogError(rc,"Perf. measure vector header write failed.");
        goto errLabel;
      }

      for(const T* v=vect; v<vect+locN; ++v)
        if((rc = file::printf(fH,fmt,*v)) != kOkRC )
        {
          rc = cwLogError(rc,"Perf. measure vector write failed.");
          goto errLabel;
        }

      if((rc = file::printf(fH,"\n")) != kOkRC )
      {
          rc = cwLogError(rc,"Perf. measure vector EOL failed.");
          goto errLabel;
      }
      
    errLabel:
      return rc;      
    }
    
    double dbl_max_to_neg_one( double x )
    { return x==std::numeric_limits<double>::max() ? -1.0 : x; }
    
  }
}

cw::rc_t cw::perf_meas::create( handle_t& hRef, sfscore::handle_t scoreH, const params_t& params )
{
  rc_t rc;
  if((rc = destroy(hRef)) != kOkRC )
    return rc;

  perf_meas_t*          p            = mem::allocZ<perf_meas_t>();
  const sfscore::loc_t* locA         = sfscore::loc_base(scoreH);
  const section_t*      prev_section = nullptr;
  
  p->scoreH               = scoreH;
  p->params               = params;
  p->locN                 = loc_count(scoreH);
  p->locA                 = mem::allocZ<loc_t>(p->locN);
  p->next_section_loc_idx = kInvalidIdx;
  p->next_calc_loc_idx    = kInvalidIdx;
  
  // for each score location
  for(unsigned i=0; i<p->locN; ++i)
  {
    // the index of the record into p->locA[]
    // is the same as the locId of the associated location
    assert( i == locA[i].index );
    
    p->locA[i].locId = locA[i].index;

    // if this location is the end of a set (or sets)  ...
    for(sfscore::set_t* s=locA[i].setList; s!=nullptr; s=s->llink)
    {
      // ... link the sets onto this location record
      set_t* set      = mem::allocZ<set_t>();
      set->loc        = p->locA + i;
      set->set        = s;      
      set->slink      = p->locA[i].setL;
      p->locA[i].setL = set;      
      set->alink      = p->setL;
      p->setL         = set;

      _create_meas_info_record(set);
    }
    
    // if this location is the start of a new section
    if( locA[i].begSectPtr != nullptr )
    {
      p->locA[i].section               = mem::allocZ<section_t>();
      p->locA[i].section->prev_section = prev_section;
      p->locA[i].section->section      = locA[i].begSectPtr;
      prev_section                     = p->locA[i].section;
      
      if(locA[i].begSectPtr->measLocPtr != nullptr )
      {
        // get the loc of the last event in the last set that applies to this section
        // (this is where the sets that supply this section will be evaluated)
        unsigned calc_loc_idx = locA[i].begSectPtr->measLocPtr->index;
        assert( calc_loc_idx < p->locN);

        // verify that the 'calc' location is <= the section location
        if( calc_loc_idx > locA[i].index )
        {
          rc = cwLogError(kInvalidStateRC,"The last loc %i of the last set in section '%s' is after the section start at loc: %i.", calc_loc_idx, cwStringNullGuard(locA[i].begSectPtr->label),i);
          goto errLabel;
        }

        // verify that that the calc location is available
        if( p->locA[ calc_loc_idx ].calc != nullptr )
        {
          rc = cwLogError(kInvalidStateRC,"A given location (%i) may only have one 'calc' record.",calc_loc_idx);
          goto errLabel;
        }
        
        // create the 'calc' record preceding this section 
        if(( p->locA[ calc_loc_idx ].calc = _create_calc_record( p, locA[i].begSectPtr )) == nullptr )
        {
          rc = cwLogError(kInvalidIdRC,"The 'calc' object create failed at location %i",calc_loc_idx);
          goto errLabel;
        }

        // set the calc.section pointer and the section.calc pointer
        p->locA[ calc_loc_idx ].calc->section = p->locA[i].section;
        p->locA[i].section->calc              = p->locA[ calc_loc_idx ].calc;
      }      
    }
  }

  if((rc = _reset(p,0)) != kOkRC )
  {
    rc = cwLogError(rc,"Perf. meas initial reset failed.");
    goto errLabel;
  }
  
  hRef.set(p);

 errLabel:
  if(rc != kOkRC )
    _destroy(p);
  
  return rc;
}


cw::rc_t cw::perf_meas::destroy( handle_t& hRef )
{
  rc_t rc = kOkRC;
  if( !hRef.isValid() )
    return rc;

  perf_meas_t* p = _handleToPtr(hRef);

  if((rc = _destroy(p)) != kOkRC )
  {
    rc = cwLogError(rc,"Destroy failed.");
    return rc;
  }

  hRef.clear();

  return rc;
}

cw::rc_t cw::perf_meas::reset( handle_t h, unsigned init_locId )
{
  perf_meas_t* p = _handleToPtr(h);
  return _reset(p,init_locId);
}


cw::rc_t cw::perf_meas::exec( handle_t h, const sfscore::event_t* event, result_t& resultRef )
{
  rc_t         rc  = kOkRC;
  perf_meas_t* p   = _handleToPtr(h);
  unsigned     loc_idx;

  resultRef = {};
  resultRef.loc = kInvalidIdx;
  resultRef.sectionLoc = kInvalidIdx;
  
  // get the index of the location record associated with this event
  if((loc_idx = _loc_id_to_index(p,event->oLocId )) == kInvalidIdx )
  {
    cwLogError(kInvalidIdRC,"The event loc %i is not valid.",event->oLocId);
    goto errLabel;
  }

  // if this event is prior to the last triggered section then ignore it
  if( p->last_section_loc_idx != kInvalidIdx && loc_idx < p->last_section_loc_idx )
  {
    cwLogWarning("Backtrack before last triggered section loc:%i < last section:%i.",loc_idx,p->last_section_loc_idx);
    goto errLabel;
  }


  // if this location is attached to a set then eval the set if it is complete
  if((rc = _update_sets(p,loc_idx)) != kOkRC )
  {
    rc = cwLogError(rc,"Set update failed.");
    goto errLabel;
  }

  // if this location is at or after 'next_calc_loc_idx' then evaluate the calc record
  if((rc = _update_calc(p,loc_idx)) != kOkRC )
  {
    rc = cwLogError(rc,"Calc. update failed.");
    goto errLabel;
  }

  // if this location is at or after the next section then update the section from the calc.
  if((rc = _update_section(p,loc_idx,resultRef)) != kOkRC )
  {
    rc = cwLogError(rc,"Section update failed.");
    goto errLabel;
  }
  
 errLabel:

  if( rc != kOkRC )
    rc = cwLogError(rc,"Perf-meas exec failed.");
  return rc;
}

void cw::perf_meas::report( handle_t h )
{
  perf_meas_t* p   = _handleToPtr(h);
  
  for(unsigned i=0; i<p->locN; ++i)
  {
    loc_t* loc = p->locA + i;
    bool fl = loc->section || loc->setL || loc->calc;
    
    if( fl )
      printf("%i : ",loc->locId );
    
    if( loc->section )
    {
      printf("section:%s ", loc->section->section->label );
    }
    
    set_t* s;
    if( loc->setL != nullptr )
    {
      printf("set: ");
      for(s=loc->setL; s!=nullptr; s=s->slink)
        printf("%i ", s->set->id);
    }
    
    if( loc->calc != nullptr )
    {
      printf("calc: ");
      for(s=loc->calc->setL; s!=nullptr; s=s->clink)
        printf("%i ", s->set->id);
    }

    if(fl)
      printf("\n");
      
  }
}


cw::rc_t cw::perf_meas::write_result_json( handle_t h,
                                           const char* player_name,
                                           const char* perf_date,
                                           unsigned perf_take_numb,
                                           const char* out_fname )
{
  rc_t         rc       = kOkRC;
  perf_meas_t* p        = _handleToPtr(h);
  object_t*    root     = newDictObject();
  object_t*    sectionL = root->insert_pair("sectionL",newListObject());
  
  for(loc_t* loc=p->locA; loc<p->locA+p->locN; ++loc)    
    if( loc->section != nullptr && loc->section->calc != nullptr && loc->section->triggeredFl )
    {
      object_t*     sect   = newDictObject(sectionL);
      const double* valueV = loc->section->calc->value;
      object_t*     setL   = newListObject();
       
      sect->putv("label",loc->section->section->label,
                 "player_name",player_name,
                 "perf_date",perf_date,
                 "perf_take_numb",perf_take_numb,
                 "locId",loc->locId,
                 "dyn",dbl_max_to_neg_one(valueV[kDynValIdx]),
                 "even",dbl_max_to_neg_one(valueV[kEvenValIdx]),
                 "tempo",dbl_max_to_neg_one(valueV[kTempoValIdx]),
                 "cost",dbl_max_to_neg_one(valueV[kMatchCostValIdx]),
                 "setL",setL);

      for(set_t* set=loc->section->calc->setL; set!=nullptr; set=set->clink)
      {
        object_t* set_o = newDictObject(setL);
        
        set_o->putv("type",score_parse::var_index_to_char(set->set->varId),
                    "setId",set->set->id,
                    "locId",set->loc->locId,
                    "value",dbl_max_to_neg_one(set->value),
                    "missEvtN",set->meas_info->missEvtN,
                    "missLocN",set->meas_info->missLocN,
                    "transposeLocN",set->meas_info->transposeLocN,
                    "interpLocN",set->meas_info->interpLocN);

        set_o->put_numeric_list("locSecV",  set->meas_info->locSecV,  set->meas_info->vN );
        set_o->put_numeric_list("locSecNV", set->meas_info->locSecNV, set->meas_info->vN );
        set_o->put_numeric_list("dSecV",    set->meas_info->dSecV,    set->meas_info->vN );
        set_o->put_numeric_list("statusV",  set->meas_info->statusV,  set->meas_info->vN );

      }
    }
  
  if((rc = objectToFile(out_fname,root)) != kOkRC )
  {
    rc = cwLogError(rc,"Perf. measurement JSON file write failed.");
    goto errLabel;
  }
      
 errLabel:
  root->free();

  return rc;          
}

cw::rc_t cw::perf_meas::write_result_csv( handle_t h, const char* out_fname )
{
  rc_t rc = kOkRC;
  perf_meas_t* p = _handleToPtr(h);
  file::handle_t fH;

  if((rc = file::open(fH,out_fname, file::kWriteFl)) != kOkRC )
  {
    rc = cwLogError(rc,"Perf. measure result file create failed on '%s'.",cwStringNullGuard(out_fname));
    goto errLabel;
  }

  if((rc = file::printf(fH,"section,set,type,op,value,missEvtN,missLocN,transposeLocN,interpLocN\n")) != kOkRC )
  {
    rc = cwLogError(rc,"Perf. measure title write failed.");
    goto errLabel;
  }
  
  for(loc_t* loc=p->locA; loc<p->locA+p->locN; ++loc)    
    if( loc->section != nullptr && loc->section->calc != nullptr )
    {
      if((rc = _write_result_vector(fH,loc->section,nullptr,"section","%f,",loc->section->calc->value,kValCnt)) != kOkRC )
      {
          rc = cwLogError(rc,"Perf. measure file write result section header failed.");
          goto errLabel;          
      }

      for(set_t* set=loc->section->calc->setL; set!=nullptr; set=set->clink)
      {

        if((rc = file::printf(fH,"%s,%i,%s,summary,%f,%i,%i,%i,%i\n",
                              loc->section->section->label,
                              set->set->id,
                              score_parse::var_index_to_char(set->set->varId),
                              set->value != std::numeric_limits<double>::max() ? set->value : 0,
                              set->meas_info->missEvtN,
                              set->meas_info->missLocN,
                              set->meas_info->transposeLocN,
                              set->meas_info->interpLocN)) != kOkRC )
        {
          rc = cwLogError(rc,"Perf. measure file write result failed.");
          goto errLabel;          
        }

        if((rc = _write_result_vector(fH,loc->section,set,"locSecV","%f,",set->meas_info->locSecV,set->set->locN)) != kOkRC )
        {
          rc = cwLogError(rc,"Perf.measure locSecV[] write result failed.");
          goto errLabel;
        }

        if((rc = _write_result_vector(fH,loc->section,set,"locSecNV","%i,",set->meas_info->locSecNV,set->set->locN)) != kOkRC )
        {
          rc = cwLogError(rc,"Perf.measure locSecNV[] write result failed.");
          goto errLabel;
        }

        if((rc = _write_result_vector(fH,loc->section,set,"dSecV","%f,", set->meas_info->dSecV,set->set->locN)) != kOkRC )
        {
          rc = cwLogError(rc,"Perf.measure dSecV[] write result failed.");
          goto errLabel;
        }

        if((rc = _write_result_vector(fH,loc->section,set,"statusV","%i,",set->meas_info->statusV,set->set->locN)) != kOkRC )
        {
          rc = cwLogError(rc,"Perf.measure status[] write result failed.");
          goto errLabel;
        }
        
      }
    }

  if((rc = file::close(fH)) != kOkRC )
  {
    rc = cwLogError(rc,"Perf. meas. file close failed.");
    goto errLabel;
  }
  
 errLabel:
  return rc;
    
}