533 lines
14 KiB
C
533 lines
14 KiB
C
/*************************************************************************\
|
|
* Copyright (c) 2009 UChicago Argonne LLC, 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 is distributed subject to a Software License Agreement found
|
|
* in file LICENSE that is included with this distribution.
|
|
\*************************************************************************/
|
|
/* $Id$ */
|
|
/*
|
|
* Original Author: Marty Kraimer
|
|
* Date: 06-01-91
|
|
*/
|
|
|
|
|
|
#include <stddef.h>
|
|
#include <stdlib.h>
|
|
#include <stdarg.h>
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
#include <errno.h>
|
|
#include <limits.h>
|
|
|
|
#include "dbDefs.h"
|
|
#include "epicsThread.h"
|
|
#include "epicsPrint.h"
|
|
#include "ellLib.h"
|
|
#include "dbDefs.h"
|
|
#include "dbBase.h"
|
|
#include "caeventmask.h"
|
|
#include "dbAddr.h"
|
|
#include "dbBkpt.h"
|
|
#include "dbFldTypes.h"
|
|
#include "link.h"
|
|
#include "dbLock.h"
|
|
#include "dbAccess.h"
|
|
#include "recGbl.h"
|
|
#include "dbNotify.h"
|
|
#include "dbCa.h"
|
|
#include "dbScan.h"
|
|
#include "taskwd.h"
|
|
#include "callback.h"
|
|
#include "dbCommon.h"
|
|
#include "dbLock.h"
|
|
#include "devSup.h"
|
|
#include "drvSup.h"
|
|
#include "menuPini.h"
|
|
#include "registryRecordType.h"
|
|
#include "registryDeviceSupport.h"
|
|
#include "registryDriverSupport.h"
|
|
#include "errMdef.h"
|
|
#include "recSup.h"
|
|
#include "envDefs.h"
|
|
#include "rsrv.h"
|
|
#include "asDbLib.h"
|
|
#include "dbStaticLib.h"
|
|
#include "db_access_routines.h"
|
|
#include "initHooks.h"
|
|
#include "epicsExit.h"
|
|
#include "epicsSignal.h"
|
|
|
|
#define epicsExportSharedSymbols
|
|
#include "epicsRelease.h"
|
|
#include "iocInit.h"
|
|
|
|
static enum {
|
|
iocVirgin, iocBuilding, iocBuilt, iocRunning, iocPaused, iocStopped
|
|
} iocState = iocVirgin;
|
|
|
|
/* define forward references*/
|
|
static void initDrvSup(void);
|
|
static void initRecSup(void);
|
|
static void initDevSup(void);
|
|
static void finishDevSup(void);
|
|
static void initDatabase(void);
|
|
static void initialProcess(void);
|
|
static void exitDatabase(void *dummy);
|
|
|
|
|
|
/*
|
|
* Initialize EPICS on the IOC.
|
|
*/
|
|
int iocInit(void)
|
|
{
|
|
return iocBuild() || iocRun();
|
|
}
|
|
|
|
int iocBuild(void)
|
|
{
|
|
if (iocState != iocVirgin) {
|
|
errlogPrintf("iocBuild: IOC can only be initialized once\n");
|
|
return -1;
|
|
}
|
|
initHookAnnounce(initHookAtIocBuild);
|
|
|
|
if (!epicsThreadIsOkToBlock()) {
|
|
epicsThreadSetOkToBlock(1);
|
|
}
|
|
|
|
errlogPrintf("Starting iocInit\n");
|
|
if (!pdbbase) {
|
|
errlogPrintf("iocBuild: Aborting, no database loaded!\n");
|
|
return -1;
|
|
}
|
|
epicsSignalInstallSigHupIgnore();
|
|
initHookAnnounce(initHookAtBeginning);
|
|
|
|
coreRelease();
|
|
/* After this point, further calls to iocInit() are disallowed. */
|
|
iocState = iocBuilding;
|
|
|
|
taskwdInit();
|
|
callbackInit();
|
|
initHookAnnounce(initHookAfterCallbackInit);
|
|
|
|
dbCaLinkInit();
|
|
initHookAnnounce(initHookAfterCaLinkInit);
|
|
|
|
initDrvSup();
|
|
initHookAnnounce(initHookAfterInitDrvSup);
|
|
|
|
initRecSup();
|
|
initHookAnnounce(initHookAfterInitRecSup);
|
|
|
|
initDevSup();
|
|
initHookAnnounce(initHookAfterInitDevSup);
|
|
|
|
initDatabase();
|
|
dbLockInitRecords(pdbbase);
|
|
dbBkptInit();
|
|
initHookAnnounce(initHookAfterInitDatabase);
|
|
|
|
finishDevSup();
|
|
initHookAnnounce(initHookAfterFinishDevSup);
|
|
|
|
scanInit();
|
|
if (asInit()) {
|
|
errlogPrintf("iocBuild: asInit Failed.\n");
|
|
return -1;
|
|
}
|
|
dbPutNotifyInit();
|
|
epicsThreadSleep(.5);
|
|
initHookAnnounce(initHookAfterScanInit);
|
|
|
|
initialProcess();
|
|
initHookAnnounce(initHookAfterInitialProcess);
|
|
|
|
/* Start CA server threads */
|
|
rsrv_init();
|
|
initHookAnnounce(initHookAfterCaServerInit);
|
|
|
|
iocState = iocBuilt;
|
|
initHookAnnounce(initHookAfterIocBuilt);
|
|
return 0;
|
|
}
|
|
|
|
int iocRun(void)
|
|
{
|
|
if (iocState != iocPaused && iocState != iocBuilt) {
|
|
errlogPrintf("iocRun: IOC not paused\n");
|
|
return -1;
|
|
}
|
|
initHookAnnounce(initHookAtIocRun);
|
|
|
|
/* Enable scan tasks and some driver support functions. */
|
|
scanRun();
|
|
dbCaRun();
|
|
initHookAnnounce(initHookAfterDatabaseRunning);
|
|
if (iocState == iocBuilt)
|
|
initHookAnnounce(initHookAfterInterruptAccept);
|
|
|
|
rsrv_run();
|
|
initHookAnnounce(initHookAfterCaServerRunning);
|
|
if (iocState == iocBuilt)
|
|
initHookAnnounce(initHookAtEnd);
|
|
|
|
errlogPrintf("iocRun: %s\n", iocState == iocBuilt ?
|
|
"All initialization complete" :
|
|
"IOC restarted");
|
|
iocState = iocRunning;
|
|
initHookAnnounce(initHookAfterIocRunning);
|
|
return 0;
|
|
}
|
|
|
|
int iocPause(void)
|
|
{
|
|
if (iocState != iocRunning) {
|
|
errlogPrintf("iocPause: IOC not running\n");
|
|
return -1;
|
|
}
|
|
initHookAnnounce(initHookAtIocPause);
|
|
|
|
rsrv_pause();
|
|
initHookAnnounce(initHookAfterCaServerPaused);
|
|
|
|
dbCaPause();
|
|
scanPause();
|
|
initHookAnnounce(initHookAfterDatabasePaused);
|
|
|
|
iocState = iocPaused;
|
|
errlogPrintf("iocPause: IOC suspended\n");
|
|
initHookAnnounce(initHookAfterIocPaused);
|
|
return 0;
|
|
}
|
|
|
|
|
|
static void initDrvSup(void) /* Locate all driver support entry tables */
|
|
{
|
|
drvSup *pdrvSup;
|
|
|
|
for (pdrvSup = (drvSup *)ellFirst(&pdbbase->drvList); pdrvSup;
|
|
pdrvSup = (drvSup *)ellNext(&pdrvSup->node)) {
|
|
struct drvet *pdrvet = registryDriverSupportFind(pdrvSup->name);
|
|
|
|
if (!pdrvet) {
|
|
errlogPrintf("iocInit: driver %s not found\n", pdrvSup->name);
|
|
continue;
|
|
}
|
|
pdrvSup->pdrvet = pdrvet;
|
|
|
|
if (pdrvet->init)
|
|
pdrvet->init();
|
|
}
|
|
}
|
|
|
|
static void initRecSup(void)
|
|
{
|
|
dbRecordType *pdbRecordType;
|
|
|
|
for (pdbRecordType = (dbRecordType *)ellFirst(&pdbbase->recordTypeList);
|
|
pdbRecordType;
|
|
pdbRecordType = (dbRecordType *)ellNext(&pdbRecordType->node)) {
|
|
recordTypeLocation *precordTypeLocation =
|
|
registryRecordTypeFind(pdbRecordType->name);
|
|
struct rset *prset;
|
|
|
|
if (!precordTypeLocation) {
|
|
errlogPrintf("iocInit record support for %s not found\n",
|
|
pdbRecordType->name);
|
|
continue;
|
|
}
|
|
prset = precordTypeLocation->prset;
|
|
pdbRecordType->prset = prset;
|
|
if (prset->init) {
|
|
prset->init();
|
|
}
|
|
}
|
|
}
|
|
|
|
static void initDevSup(void)
|
|
{
|
|
dbRecordType *pdbRecordType;
|
|
|
|
for (pdbRecordType = (dbRecordType *)ellFirst(&pdbbase->recordTypeList);
|
|
pdbRecordType;
|
|
pdbRecordType = (dbRecordType *)ellNext(&pdbRecordType->node)) {
|
|
devSup *pdevSup;
|
|
|
|
for (pdevSup = (devSup *)ellFirst(&pdbRecordType->devList);
|
|
pdevSup;
|
|
pdevSup = (devSup *)ellNext(&pdevSup->node)) {
|
|
struct dset *pdset = registryDeviceSupportFind(pdevSup->name);
|
|
|
|
if (!pdset) {
|
|
errlogPrintf("device support %s not found\n",pdevSup->name);
|
|
continue;
|
|
}
|
|
dbInitDevSup(pdevSup, pdset); /* Calls pdset->init(0) */
|
|
}
|
|
}
|
|
}
|
|
|
|
static void finishDevSup(void)
|
|
{
|
|
dbRecordType *pdbRecordType;
|
|
|
|
for (pdbRecordType = (dbRecordType *)ellFirst(&pdbbase->recordTypeList);
|
|
pdbRecordType;
|
|
pdbRecordType = (dbRecordType *)ellNext(&pdbRecordType->node)) {
|
|
devSup *pdevSup;
|
|
|
|
for (pdevSup = (devSup *)ellFirst(&pdbRecordType->devList);
|
|
pdevSup;
|
|
pdevSup = (devSup *)ellNext(&pdevSup->node)) {
|
|
struct dset *pdset = pdevSup->pdset;
|
|
|
|
if (pdset && pdset->init)
|
|
pdset->init(1);
|
|
}
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Iterate through all record instances (but not aliases),
|
|
* calling a function for each one.
|
|
*/
|
|
typedef void (*recIterFunc)(dbRecordType *rtyp, dbCommon *prec, void *user);
|
|
|
|
static void iterateRecords(recIterFunc func, void *user)
|
|
{
|
|
dbRecordType *pdbRecordType;
|
|
|
|
for (pdbRecordType = (dbRecordType *)ellFirst(&pdbbase->recordTypeList);
|
|
pdbRecordType;
|
|
pdbRecordType = (dbRecordType *)ellNext(&pdbRecordType->node)) {
|
|
dbRecordNode *pdbRecordNode;
|
|
|
|
for (pdbRecordNode = (dbRecordNode *)ellFirst(&pdbRecordType->recList);
|
|
pdbRecordNode;
|
|
pdbRecordNode = (dbRecordNode *)ellNext(&pdbRecordNode->node)) {
|
|
dbCommon *precord = pdbRecordNode->precord;
|
|
|
|
if (!precord->name[0] ||
|
|
pdbRecordNode->flags & DBRN_FLAGS_ISALIAS)
|
|
continue;
|
|
|
|
func(pdbRecordType, precord, user);
|
|
}
|
|
}
|
|
return;
|
|
}
|
|
|
|
static void doInitRecord0(dbRecordType *pdbRecordType, dbCommon *precord,
|
|
void *user)
|
|
{
|
|
struct rset *prset = pdbRecordType->prset;
|
|
devSup *pdevSup;
|
|
|
|
if (!prset) return; /* unlikely */
|
|
|
|
precord->rset = prset;
|
|
precord->rdes = pdbRecordType;
|
|
precord->mlok = epicsMutexMustCreate();
|
|
ellInit(&precord->mlis);
|
|
|
|
/* Reset the process active field */
|
|
precord->pact = FALSE;
|
|
|
|
/* Init DSET NOTE that result may be NULL */
|
|
pdevSup = dbDTYPtoDevSup(pdbRecordType, precord->dtyp);
|
|
precord->dset = pdevSup ? pdevSup->pdset : NULL;
|
|
|
|
if (prset->init_record)
|
|
prset->init_record(precord, 0);
|
|
}
|
|
|
|
static void doResolveLinks(dbRecordType *pdbRecordType, dbCommon *precord,
|
|
void *user)
|
|
{
|
|
devSup *pdevSup;
|
|
int j;
|
|
|
|
/* Convert all PV_LINKs to DB_LINKs or CA_LINKs */
|
|
/* For all the links in the record type... */
|
|
for (j = 0; j < pdbRecordType->no_links; j++) {
|
|
dbFldDes *pdbFldDes =
|
|
pdbRecordType->papFldDes[pdbRecordType->link_ind[j]];
|
|
DBLINK *plink = (DBLINK *)((char *)precord + pdbFldDes->offset);
|
|
|
|
if (plink->type == PV_LINK) {
|
|
DBADDR dbaddr;
|
|
|
|
if (plink == &precord->tsel) recGblTSELwasModified(plink);
|
|
if (!(plink->value.pv_link.pvlMask&(pvlOptCA|pvlOptCP|pvlOptCPP))
|
|
&& (dbNameToAddr(plink->value.pv_link.pvname,&dbaddr)==0)) {
|
|
DBADDR *pdbAddr;
|
|
|
|
plink->type = DB_LINK;
|
|
pdbAddr = dbCalloc(1,sizeof(struct dbAddr));
|
|
*pdbAddr = dbaddr; /*structure copy*/;
|
|
plink->value.pv_link.pvt = pdbAddr;
|
|
} else {/*It is a CA link*/
|
|
|
|
if (pdbFldDes->field_type == DBF_INLINK) {
|
|
plink->value.pv_link.pvlMask |= pvlOptInpNative;
|
|
}
|
|
dbCaAddLink(plink);
|
|
if (pdbFldDes->field_type == DBF_FWDLINK) {
|
|
char *pperiod =
|
|
strrchr(plink->value.pv_link.pvname,'.');
|
|
|
|
if (pperiod && strstr(pperiod,"PROC")) {
|
|
plink->value.pv_link.pvlMask |= pvlOptFWD;
|
|
} else {
|
|
errlogPrintf("%s.FLNK is a Channel Access Link "
|
|
" but does not link to a PROC field\n",
|
|
precord->name);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
pdevSup = dbDTYPtoDevSup(pdbRecordType, precord->dtyp);
|
|
if (pdevSup) {
|
|
struct dsxt *pdsxt = pdevSup->pdsxt;
|
|
if (pdsxt && pdsxt->add_record) {
|
|
pdsxt->add_record(precord);
|
|
}
|
|
}
|
|
}
|
|
|
|
static void doInitRecord1(dbRecordType *pdbRecordType, dbCommon *precord,
|
|
void *user)
|
|
{
|
|
struct rset *prset = pdbRecordType->prset;
|
|
|
|
if (!prset) return; /* unlikely */
|
|
|
|
if (prset->init_record)
|
|
prset->init_record(precord, 1);
|
|
}
|
|
|
|
static void initDatabase(void)
|
|
{
|
|
iterateRecords(doInitRecord0, NULL);
|
|
iterateRecords(doResolveLinks, NULL);
|
|
iterateRecords(doInitRecord1, NULL);
|
|
|
|
epicsAtExit(exitDatabase, NULL);
|
|
return;
|
|
}
|
|
|
|
/*
|
|
* Process database records at initialization ordered by phase
|
|
* if their pini (process at init) field is set.
|
|
*/
|
|
typedef struct {
|
|
int this;
|
|
int next;
|
|
epicsEnum16 pini;
|
|
} phaseData_t;
|
|
|
|
static void doRecordPini(dbRecordType *rtype, dbCommon *precord, void *user)
|
|
{
|
|
phaseData_t *pphase = (phaseData_t *)user;
|
|
int phas;
|
|
|
|
if (precord->pini != pphase->pini) return;
|
|
|
|
phas = precord->phas;
|
|
if (phas == pphase->this) {
|
|
dbScanLock(precord);
|
|
dbProcess(precord);
|
|
dbScanUnlock(precord);
|
|
} else if (phas > pphase->this && phas < pphase->next)
|
|
pphase->next = phas;
|
|
}
|
|
|
|
static void piniProcess(int pini)
|
|
{
|
|
phaseData_t phase;
|
|
phase.next = MIN_PHASE;
|
|
phase.pini = pini;
|
|
|
|
/* This scans through the whole database as many times as needed.
|
|
* During the first pass it is unlikely to find any records with
|
|
* PHAS = MIN_PHASE, but during each iteration it looks for the
|
|
* phase value of the next pass to run. Note that PHAS fields can
|
|
* be changed at runtime, so we have to look for the lowest value
|
|
* of PHAS each time.
|
|
*/
|
|
do {
|
|
phase.this = phase.next;
|
|
phase.next = MAX_PHASE + 1;
|
|
iterateRecords(doRecordPini, &phase);
|
|
} while (phase.next != MAX_PHASE + 1);
|
|
}
|
|
|
|
static void piniProcessHook(initHookState state)
|
|
{
|
|
switch (state) {
|
|
case initHookAtIocRun:
|
|
piniProcess(menuPiniRUN);
|
|
break;
|
|
|
|
case initHookAfterIocRunning:
|
|
piniProcess(menuPiniRUNNING);
|
|
break;
|
|
|
|
case initHookAtIocPause:
|
|
piniProcess(menuPiniPAUSE);
|
|
break;
|
|
|
|
case initHookAfterIocPaused:
|
|
piniProcess(menuPiniPAUSED);
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void initialProcess(void)
|
|
{
|
|
initHookRegister(piniProcessHook);
|
|
piniProcess(menuPiniYES);
|
|
}
|
|
|
|
|
|
/*
|
|
* Shutdown processing.
|
|
*/
|
|
static void doCloseLinks(dbRecordType *pdbRecordType, dbCommon *precord,
|
|
void *user)
|
|
{
|
|
devSup *pdevSup;
|
|
struct dsxt *pdsxt;
|
|
int j;
|
|
|
|
for (j = 0; j < pdbRecordType->no_links; j++) {
|
|
dbFldDes *pdbFldDes =
|
|
pdbRecordType->papFldDes[pdbRecordType->link_ind[j]];
|
|
DBLINK *plink = (DBLINK *)((char *)precord + pdbFldDes->offset);
|
|
|
|
if (plink->type == CA_LINK) {
|
|
dbCaRemoveLink(plink);
|
|
}
|
|
}
|
|
|
|
if (precord->dset &&
|
|
(pdevSup = dbDSETtoDevSup(pdbRecordType, precord->dset)) &&
|
|
(pdsxt = pdevSup->pdsxt) &&
|
|
pdsxt->del_record) {
|
|
pdsxt->del_record(precord);
|
|
}
|
|
}
|
|
|
|
static void exitDatabase(void *dummy)
|
|
{
|
|
iterateRecords(doCloseLinks, NULL);
|
|
iocState = iocStopped;
|
|
}
|