⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 upd_item.cpp

📁 OPCSERVER源代码
💻 CPP
字号:
/**************************************************************************
 *                                                                        *
 * Light OPC Server development library                                   *
 *                                                                        *
 *   Copyright (c) 2000 by Timofei Bondarenko                             *
                                                                          *
 item updating routines
 **************************************************************************/
#include "privopc.h"
#include "nan.h"
#include <math.h>

/*********************************************************************************/

#if 0
#define R8_CLEAR(v) VariantClear(v)
#else
#define R8_CLEAR(v)
#endif

static int lo_compare_lastsent(LightOPCItem *item, loTagEntry *prim, double *deadband)
{
 VARIANT *pv = &prim->primValue;
 VARIANT var;

 if (loTF_NOCOMP & prim->attr.taFlags) return 1;

 if (VT_BSTR == V_VT(pv))
   {
    unsigned lastsent_len = item->lastsent? item->lastsent[0]: 0;
    unsigned len = loBSTR_SIZE(V_BSTR(pv));

    if (deadband &&
        len == lastsent_len &&
        !memcmp(V_BSTR(pv), &item->lastsent[1], len)) return 0;

    if (len <= lastsent_len && lastsent_len ||
        preallocX((void**)&item->lastsent, len + sizeof(item->lastsent[0])))
      {
       memcpy(&item->lastsent[1], V_BSTR(pv), item->lastsent[0] = len);
      }
    return 1;
   }
 VARIANTINIT(&var);

 if (V_ISARRAY(pv)) /* Under Wine the VarChangeType() does not detect arrays. */
   {
    void *ptr;
    HRESULT hr;
    SAFEARRAY *sa = V_ARRAY(pv);

    hr = loSA_AccessData(sa, &ptr);
    if (FAILED(hr))
      {
       UL_ERROR((LOGID, "%!l SafeArrayAccessData() FAILED", hr));
      }
    else
      {
       unsigned len = sa->cbElements * sa->rgsabound[0].cElements;
       unsigned lastsent_len = item->lastsent? item->lastsent[0]: 0;

       if (deadband &&
           len == lastsent_len &&
           !memcmp(ptr, item->lastsent + 1, len))
         {
          SafeArrayUnaccessData(sa); return 0;
         }
       if (len <= lastsent_len && lastsent_len ||
           preallocX((void**)&item->lastsent, len + sizeof(item->lastsent[0])))
         {
          memcpy(item->lastsent + 1, ptr, item->lastsent[0] = len);
         }
       SafeArrayUnaccessData(sa);
      }
   }
 else if (/*VT_EMPTY == (vt = V_VT(pv)) ||*/     
          S_OK == VariantChangeType(&var, pv, 0, VT_R8))
   {
    if (deadband)
      {
       if (!memcmp(&item->numeric, &V_R8(&var), sizeof(item->numeric)))
         {
          R8_CLEAR(&var); return 0;
         }
       if (
#if loTt_ZERORANGE
           !(prim->attr.taFlags & loTt_ZERORANGE) && *deadband &&
#else
           *deadband &&/***/ prim->attr.taRangecent /*OPC_ANALOG*/ &&
#endif
           !lo_IsNANd(V_R8(&var)) && !lo_IsNANd(item->numeric) &&
           fabs(V_R8(&var) - item->numeric) <= *deadband * prim->attr.taRangecent)
         {
          R8_CLEAR(&var); return 0;
         }
/*else UL_ERROR((LOGID, "Dead=%g range/100=%g abs=%g - %g = %g result:%g",
              *deadband,
              prim->attr.rangecent,
              V_R8(&var), item->numeric, fabs(V_R8(&var) - item->numeric),
              fabs(V_R8(&var) - item->numeric)/prim->attr.rangecent));*/
      }
    memcpy(&item->numeric, &V_R8(&var), sizeof(item->numeric));
    R8_CLEAR(&var);
   }
 else UL_INFO((LOGID, "lo_compare_lastent() Unknown VARTYPE 0x%X", V_VT(pv)));

 return 1;
}

static int lo_compare_dummy(LightOPCItem *item, loTagEntry *prim, double *deadband)
{
 return 1; /* Always different */
}

unsigned lo_refresh_items(loService *se, loUpdList *upl, LightOPCGroup *grp)
{
 int (*compare_item)(LightOPCItem *, loTagEntry *, double *);
 unsigned ii, count = grp->active_count;
 LightOPCItem **items = (LightOPCItem **)grp->itl.gl_list;
 double deadband = grp->Deadband;
 loTagEntry *tags = se->tags;

// UL_DEBUG((LOGID, "refresh_items() {..."));

 if (count > grp->itl.gl_count)
   {
    UL_ERROR((LOGID, "Group::active %u(%d) count %u mismatch",
              count, count, grp->itl.gl_count));
    count = grp->itl.gl_count;
   }

 compare_item = (se->driver.ldFlags & loDF_NOCOMP)
           ? lo_compare_dummy
           : lo_compare_lastsent;

 upl->used = 0;
 upl->master_err = S_OK;
 upl->master_qual = S_OK;
 if (lo_upl_grow(upl, count))
   {
    UL_ERROR((LOGID, "%ls refresh/grow(%u) FAILED",
                      loWnul(grp->name), grp->itl.gl_count));
    upl->master_err = E_OUTOFMEMORY;
    upl->master_qual = S_FALSE;
   }
 else
   {
    unsigned used = 0;
    loUpdList lup = *upl;
// lup.deadband = grp->Deadband;
// lup.tags = se->tags;

    count = grp->itl.gl_count;
    for(ii = 0; ii < count; ii++)
      {
       LightOPCItem *item;
       if ((item = items[ii]) && item->bActive)
         {
          HRESULT hr;
          WORD quality;
          loTagEntry *prim = &tags[item->tid];

#if 0 != LO_CHECK_RIGHTS
          if (!(OPC_READABLE & prim->attr.taRights)) continue;
#endif

          if (item->Quality == (quality = prim->prim.tsQuality))
            {
             if (item->last_changed == prim->primChanged)
               {
                if (0 == lup.trqid) continue;
               }
             else if (!compare_item(item, prim, &deadband))
               {
                item->last_changed = prim->primChanged;
                if (0 == lup.trqid) continue;
               }
            }
          else compare_item(item, prim, 0);
          item->last_changed = prim->primChanged;

          if (OPC_QUALITY_GOOD != (OPC_QUALITY_MASK & quality))
            lup.master_qual = S_FALSE;

#if 1 == LO_EV_TIMESTAMP
          if (!IsFILETIME(prim->prim.tsTime))
            lup.timestamp[used] = loFT_SUBST(prim->prim.tsTime, se->ts_prim);
          else
#endif
          lup.timestamp[used] = prim->prim.tsTime;
          lup.opchandle[used] = item->hClient;

          if (S_OK != (lup.errors[used] = prim->prim.tsError))
            lup.master_err = S_FALSE;
#ifdef VAR_CHANGETYPE
          hr = VAR_CHANGETYPE(&lup.variant[used], &prim->primValue,
                              item->vtRequestedDataType);
#else
          switch(item->convtype)
            {
          case loCONV_CHANGE:
             hr = VARIANTCHANGEt(&lup.variant[used], &prim->primValue,
                                 lup.rctx.cta, item->vtRequestedDataType);
             break;
          case loCONV_COPY:
             hr = VariantCopy(&lup.variant[used],
                              &prim->primValue);
             break;
          case loCONV_ASSIGN:
             lup.variant[used] = prim->primValue;
             goto SkipHR;
             hr = S_OK;
             break;
          case loCONV_CONVERT:
               {
                loTagPair tp;
                tp.tpTi = item->tid;
                tp.tpAP = item->AcPath;
                tp.tpRt = prim->attr.taRt;

                se->driver.ldConvertTags(&lup.rctx.cactx, 1, &tp,
                      &lup.variant[used], &quality, &lup.errors[used],
                      &lup.master_err, &lup.master_qual, &prim->primValue,
                      &item->vtRequestedDataType, grp->grLCID);
                goto SkipHR;
                hr = S_OK;
               }
             break;
            }
#endif
          if (S_OK != hr)
            {
             LO_E_BADTYPE_QUAL(hr, lup.errors[used], quality);
             lup.master_err = S_FALSE;
             lup.master_qual = S_FALSE;
//             item->Quality = lup.quality[used];
             //item->Quality = -1; /* try again next time?? */
//UL_WARNING((LOGID, "%!l upd_item(%ls)::CHNGETYPE",loWnul(prim->attr.taName), hr));
            }
SkipHR:   lup.quality[used] = quality;
          item->Quality = quality;
          used++;
 // if (V_VT(&upl->vars[used]) == VT_EMPTY) UL_ERROR((LOGID, "update_item"));
         }
      } /* end of for(;;) */

    if (used > grp->active_count)
      {
       UL_ERROR((LOGID, "Group::active %u used %u mismatch",
              grp->active_count, lup.used));
      }
//    if (S_OK == lup.master_err && !lup.used) lup.master_err = S_FALSE;
    upl->master_err = lup.master_err;
    upl->master_qual = lup.master_qual;
    upl->used = used;
#if 2 == LO_EV_TIMESTAMP
    if (IsFILETIME(*se->ts_prim))
      lo_check_ev_timestamp(used, upl->timestamp, se->ts_prim);
#endif
   }
// UL_DEBUG((LOGID, "refresh_items()...}"));
 return upl->used;
}

/* end of upd_item.cpp */

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -