#include "cwCommon.h"
#include "cwLog.h"
#include "cwCommonImpl.h"
#include "cwMem.h"
#include "cwCmInterface.h"

#include "cmGlobal.h"
#include "cmFloatTypes.h"
#include "cmRpt.h"
#include "cmErr.h"
#include "cmCtx.h"
#include "cmMem.h"
#include "cmMallocDebug.h"
#include "cmLinkedHeap.h"
#include "cmTime.h"
#include "cmMidi.h"
#include "cmSymTbl.h"
#include "cmScore.h"
#include "cmText.h"
#include "cmFileSys.h"
#include "cmProcObj.h"

extern "C" {

  
  void _cm_print_info( void* arg, const char* text )
  {
    cw::log::msg( cw::log::globalHandle(), cw::log::kInfo_LogLevel, nullptr, nullptr, 0, 0, cw::kOkRC, "%s", text );
  }

  void _cm_print_error( void* arg, const char* text )
  {
    cw::log::msg( cw::log::globalHandle(), cw::log::kPrint_LogLevel, nullptr, nullptr, 0, 0, cw::kOkRC, "cm-error: %s", text );
  }
}

namespace cw
{
  namespace cm
  {
    
    extern "C" {
      typedef struct cm_str
      {
        cmCtx_t ctx;
        cmSymTblH_t symTblH;
        cmLHeapH_t  lhH;        
        cmProcCtx* procCtx;
        
      } cm_t;
    }
    
    cm_t* _handleToPtr( handle_t h )
    { return handleToPtr<handle_t,cm_t>(h); }

    rc_t _destroy( cm_t* p )
    {
      if( p != nullptr )
      {

        cmLHeapDestroy(&p->lhH);
        cmSymTblDestroy(&p->symTblH);
        cmProcCtxFree(&p->procCtx);
        
        cmTsFinalize();
        cmFsFinalize();
        cmMdReport( kIgnoreNormalMmFl );
        cmMdFinalize();
        
        mem::release(p);
      }
      return kOkRC;
    }

    rc_t _create_proc_ctx( cm_t* p, cmCtx_t* ctx )
    {
      rc_t rc = kOkRC;
      
      // create the linked heap
      if(cmLHeapIsValid( p->lhH = cmLHeapCreate(1024,ctx)) == false)
      {
        rc = cwLogError(kOpFailRC,"The cm linked heap allocation failed.");
        goto errLabel;
      }

      // intialize the symbol table
      if( cmSymTblIsValid( p->symTblH = cmSymTblCreate(cmSymTblNullHandle,1,ctx)) == false )
      {
        rc = cwLogError(kOpFailRC,"The cm linked heap allocation failed.");
        goto errLabel;
      }

      // initialize the proc context
      if( (p->procCtx = cmProcCtxAlloc(NULL,&ctx->rpt,p->lhH,p->symTblH)) == NULL )
      {
        rc = cwLogError(kOpFailRC,"The cm proc context allocation failed.");
        goto errLabel;
      }

    errLabel:
      return rc;
    }
  }
}

cw::rc_t cw::cm::create( handle_t& hRef )
{
  rc_t            rc              = kOkRC;
  cm_t*           p               = nullptr;
  bool            memDebugFl      = 0; //cmDEBUG_FL;
  unsigned        memGuardByteCnt = memDebugFl ? 8 : 0;
  unsigned        memAlignByteCnt = 16;
  unsigned        memFlags        = memDebugFl ? kTrackMmFl | kDeferFreeMmFl | kFillUninitMmFl : 0;  
  const cmChar_t* appTitle        = "cwtest";

  if((rc = destroy(hRef)) != kOkRC )
    return rc;
  
  p = mem::allocZ<cm_t>();
  
  cmCtxSetup(&p->ctx,appTitle,_cm_print_info,_cm_print_error,NULL,memGuardByteCnt,memAlignByteCnt,memFlags);

  cmMdInitialize( memGuardByteCnt, memAlignByteCnt, memFlags, &p->ctx.rpt );

  cmFsInitialize( &p->ctx, appTitle);

  cmTsInitialize( &p->ctx );

  if((rc = _create_proc_ctx(p, &p->ctx )) == kOkRC )
    hRef.set(p);
  else
  {
    _destroy(p);
    rc = cwLogError(rc,"cm Interface context create failed.");
  }
  
  return rc;
}

cw::rc_t cw::cm::destroy( handle_t& hRef )
{
  rc_t rc = kOkRC;
  cm_t* p = nullptr;
  if( !hRef.isValid() )
    return rc;

  p = _handleToPtr(hRef);
  
  if((rc = _destroy(p)) != kOkRC )
    return rc;

  hRef.clear();
  return rc;  
  
}

extern "C" {
  struct cmCtx_str* cw::cm::context( handle_t h )
  {
    cm_t* p = _handleToPtr(h);
    return &p->ctx;
  }
  
  struct cmProcCtx_str* cw::cm::proc_context( handle_t h )
  {
    cm_t* p = _handleToPtr(h);
    return p->procCtx;
  }
  
}