Files
epics-base/src/rec/erRecord.c
T
2003-04-01 21:06:49 +00:00

260 lines
6.7 KiB
C

/*************************************************************************\
* Copyright (c) 2002 The University of Chicago, as Operator of Argonne
* National Laboratory.
* Copyright (c) 2002 The Regents of the University of California, as
* Operator of Los Alamos National Laboratory.
* EPICS BASE Versions 3.13.7
* and higher are distributed subject to a Software License Agreement found
* in file LICENSE that is included with this distribution.
\*************************************************************************/
/*
*
* Author: John Winans
* Date: 9/27/93
*/
#include <stddef.h>
#include <stdlib.h>
#include <stdarg.h>
#include <stdio.h>
#include <string.h>
#include "dbDefs.h"
#include "epicsPrint.h"
#include "alarm.h"
#include "dbAccess.h"
#include "dbEvent.h"
#include "dbFldTypes.h"
#include "devSup.h"
#include "errMdef.h"
#include "recSup.h"
#include "recGbl.h"
#include "erDefs.h"
#define GEN_SIZE_OFFSET
#include "erRecord.h"
#undef GEN_SIZE_OFFSET
#include "epicsExport.h"
#define STATIC static
STATIC void ErMonitor(struct erRecord *pRec);
/* Create RSET - Record Support Entry Table*/
#define report NULL
#define initialize NULL
STATIC long ErInitRec(struct erRecord *, int);
STATIC long ErProc(struct erRecord *);
#define ErSpecial NULL
#define get_value NULL
#define cvt_dbaddr NULL
#define get_array_info NULL
#define put_array_info NULL
#define get_units NULL
#define get_precision NULL
#define get_enum_str NULL
#define get_enum_strs NULL
#define put_enum_str NULL
static long get_graphic_double(struct dbAddr *paddr, struct dbr_grDouble *pgd);
static long get_control_double(struct dbAddr *paddr, struct dbr_ctrlDouble *pcd);
static long get_alarm_double(struct dbAddr *paddr, struct dbr_alDouble *pad);
rset erRSET={
RSETNUMBER,
report,
initialize,
ErInitRec,
ErProc,
ErSpecial,
get_value,
cvt_dbaddr,
get_array_info,
put_array_info,
get_units,
get_precision,
get_enum_str,
get_enum_strs,
put_enum_str,
get_graphic_double,
get_control_double,
get_alarm_double
};
epicsExportAddress(rset,erRSET);
/******************************************************************************
*
* Init an ER record.
*
******************************************************************************/
STATIC long ErInitRec(struct erRecord *pRec, int pass)
{
ErDsetStruct *pDset = (ErDsetStruct *) pRec->dset;
if (pass == 1)
{
/* Init the card via driver calls */
/* Make sure we have a usable device support module */
if (pDset == NULL)
{
recGblRecordError(S_dev_noDSET,(void *)pRec, "er: ErInitRec");
return(S_dev_noDSET);
}
if (pDset->number < 5)
{
recGblRecordError(S_dev_missingSup,(void *)pRec, "er: ErInitRec");
return(S_dev_missingSup);
}
if( pDset->init_record != NULL)
return(*pDset->init_record)(pRec);
}
pRec->taxi = 0;
return(0);
}
/******************************************************************************
*
* Process an ER record.
*
******************************************************************************/
STATIC long ErProc(struct erRecord *pRec)
{
ErDsetStruct *pDset = (ErDsetStruct *) pRec->dset;
pRec->pact=TRUE;
pRec->ltax = pRec->taxi; /* The only monitorable field we can change */
if (pRec->tpro > 10)
printf("recEr::ErProc(%s) entered\n", pRec->name);
if (pDset->proc != NULL)
(*pDset->proc)(pRec);
/* Take care of time stamps and such */
pRec->udf=FALSE;
/* tsLocalTime(&pRec->time);*/
recGblGetTimeStamp(pRec);
/* Deal with monitor stuff */
ErMonitor(pRec);
/* process the forward scan link record */
recGblFwdLink(pRec);
pRec->pact=FALSE;
return(0);
}
#if 0 /* this is not needed */
/******************************************************************************
*
* Used to check for changed field values
*
******************************************************************************/
STATIC long ErSpecial(struct dbAddr *paddr, int after)
{
struct erRecord *pRec = (struct erRecord *)(paddr->precord);
void *p
if(!after)
return(0);
/* Make sure we have the proper special flag type spec'd */
if (pdbAddr->special != SPC_MOD)
{
recGblDbaddrError(S_db_badChoice,pdbAddr,"bp: special");
return(S_db_badChoice);
}
p = (void *)(pdbAddr->pfield);
/* Figure out which field has been changed */
if (p == &pRec->dg0d)
pRec->dgcm |= 1;
else ...
return(0);
}
#endif
/******************************************************************************
*
* Post any events for fields that might have changed while processing.
*
******************************************************************************/
STATIC void ErMonitor(struct erRecord *pRec)
{
unsigned short monitor_mask;
monitor_mask = recGblResetAlarms(pRec);
monitor_mask |= (DBE_VALUE | DBE_LOG);
db_post_events(pRec, &pRec->val, monitor_mask);
if (pRec->taxi != pRec->ltax)
db_post_events(pRec, &pRec->taxi, monitor_mask);
return;
}
static long get_graphic_double(struct dbAddr *paddr, struct dbr_grDouble *pgd)
{
struct erRecord *pRec=(struct erRecord *)paddr->precord;
if(paddr->pfield==(void *)&pRec->val)
{
pgd->upper_disp_limit = 0;
pgd->lower_disp_limit = 0;
}
else if(paddr->pfield==(void *)&pRec->dg0d
|| paddr->pfield==(void *)&pRec->dg0w
|| paddr->pfield==(void *)&pRec->dg1d
|| paddr->pfield==(void *)&pRec->dg1w
|| paddr->pfield==(void *)&pRec->dg2d
|| paddr->pfield==(void *)&pRec->dg2w
|| paddr->pfield==(void *)&pRec->dg3d
|| paddr->pfield==(void *)&pRec->dg3w)
{
pgd->upper_disp_limit = 64*1024;
pgd->lower_disp_limit = 0;
}
else
recGblGetGraphicDouble(paddr,pgd);
return(0);
}
static long get_control_double(struct dbAddr *paddr, struct dbr_ctrlDouble *pcd)
{
struct erRecord *pRec=(struct erRecord *)paddr->precord;
if(paddr->pfield==(void *)&pRec->val)
{
pcd->upper_ctrl_limit = 0;
pcd->lower_ctrl_limit = 1;
}
else if(paddr->pfield==(void *)&pRec->dg0d
|| paddr->pfield==(void *)&pRec->dg0w
|| paddr->pfield==(void *)&pRec->dg1d
|| paddr->pfield==(void *)&pRec->dg1w
|| paddr->pfield==(void *)&pRec->dg2d
|| paddr->pfield==(void *)&pRec->dg2w
|| paddr->pfield==(void *)&pRec->dg3d
|| paddr->pfield==(void *)&pRec->dg3w)
{
pcd->upper_ctrl_limit = 64*1024;
pcd->lower_ctrl_limit = 0;
}
else
recGblGetControlDouble(paddr,pcd);
return(0);
}
static long get_alarm_double(struct dbAddr *paddr, struct dbr_alDouble *pad)
{
#if 0
{
pad->upper_alarm_limit = 2;
pad->upper_warning_limit = 2;
pad->lower_warning_limit = -1;
pad->lower_alarm_limit = -1;
} else
#endif
recGblGetAlarmDouble(paddr,pad);
return(0);
}