unit MCAD_MI1201_Thread0Ex; interface uses Windows, Classes, SysUtils, SyncObjs, Registry, xSystem, MMTimer, TMultiThreadList, MiscFunc, MCAD_MI1201_FileOutput, MCAD_MI1201_Range, c_ISSB, MassClbr, c_Mi1201, MITypes, MCAD_MI1201_Registry, MCAD_Event, McadUserTypes, MCAD_MI1201_TChartSeries, MCAD_MI1201_Thread_Types; const cMainTreadName='MI1201_Thread'; cDefaultTimeOut=500; {мс} // Идентификатор сообщения масс-спектрометра CM_MI1201Notify = $9FFE; const cChannels:array[tSeries] of tChannel=(PNC1,PNC2,PNC3,PNC4,PNC5,PNC_SEM,IonCounter,PNC1); type tDataSeries=array[tSeries] of tMSSeries; TUpDownDirection=(updNone, updUp, updDown); tMNIParameters=(mni_X, mni_dM); tCounter=integer; tMsSp=c_Mi1201.tCtrl; tMsSpPtr=^tMsSp; tFlag=(fDataAllocated, fNoMemory, fStopMeasure, fDoMeasureCicleContinues, fDoChangeMass, fDoChangeCounter, fDoPNCCalibration, fDoPNCCalibrationEx, fDoFileWrite, fDoDataFileWrite, fDoDataFileRead, fDoSetON, fDoSetOff, fDoResetDevice, fDoMassZeroSetByHardware, fDoSetIntegrationTime, fError, fDifferentPointsNumberInSeries, fPointProcessed, fDataValid, fCumulativeMeasure, fSourceTimerFails, fVoltmeterTimerFails, fMassLimit, fMassLimitLo, fMassLimitUp, fDataChanged ); const cDoFlags=[fDoChangeMass..fDoSetIntegrationTime]; type tFlags=set of tFlag; tError=(ecOK, ecHardwareError, ecNoMemory, ecUnexpectedError); tSetRangeResult=(rOK, rMin, rMax, rStep, rCantSetup); tMeasureState=(msUnknown, msErrorStopped, msNotAvailableStopped, msSuspended, msStopped, msPaused, msScroll { msSinglePass, msRepeatedlyWithDataReplace, msRepeatedlyWithDataAccumulation}); tScrollDirection=(sdUnknown, sdNoScroll, sdUp, sdDown); const cDoNotMeasureFlags=[fNoMemory, fStopMeasure, fError]; cDoMeasureFlags=[fDataAllocated{, fRangeValid}]; cFullIntegrationTime=Succ(Ord(High(tChannel))); type tNotifyList=class(tMultiThreadList) public procedure Add(var NotifyHanler:tMI1201_Thread_NotifyData); function Remove(var NotifyHanler:tMI1201_Thread_NotifyData):Integer; procedure NotifyByList(Sender:TObject; Event:tMI1201_Thread_Event); end; tNotifyWindowsList=class(tMultiThreadList) public procedure Add(WindowsHanler:HWND); function Remove(WindowsHanler:HWND): Integer; procedure NotifyByList(Sender:TObject; Event:tMI1201_Thread_Event); end; tMassData=record // Pass:uint; StepX:tMass; StepMultiplicator:integer; LastStep:tMass; end; tPointTimeData=record // Время одного замера (запуск-получение данных- запись в хранилище) TimeInterval:cardinal; LastTime,MinTime,MaxTime:cardinal; FullTime:int64; Number:cardinal; end; tCicleTimeData=record // Время цикла замера (запуск-запуск следующего) Start:cardinal; LastTime,MinTime,MaxTime:cardinal; FullTime:int64; Number:cardinal; end; tSpecterDateTime=record StartDateTime,EndDateTime:TDateTime; end; type tVoltmeterData=record Timer:tMMTimer; Channels:tVoltageChannels; Time:cardinal; Data:array[tVoltageChannel] of tMkVolts; end; tSourcesData=record AutoCloseTimes:array[tSource] of cardinal; Timer:tMMTimer; OneSecondTickTimer:tMMTimer; StartDateTime:TDateTime; StartTime:tTimeInterval; end; tPNCCalibrationParameter=(pncAll, pncK, pncU0, pncZeroLevel); tMsThread = class(TThread) private // Ожидание завершения, по истечении предпринимаются решительные действия; prTerminateTimeOut:cardinal; // Доступ к реестру prLocalRegistry:tRegistry; // Данные по шкале масс prMass:tMassData; // Код ошибки prError:tError; // Данные для блокировки prDataLock,prMsSpLock:TMultiReadExclusiveWriteSynchronizer; // Контроль завершения потока prEventNotExecuted:TEvent; prFinished:boolean; // Внутренние флаги prFlags:tFlags; // Внутренние переключатели prSwitches:tMSSwitches; // Начальная емкость данных для спектра prDefaultCapacity:integer; // Приблизительный начальный размер данных prInitialDataSize:cardinal; // Последний считанный сигнал prSignalsV:tSignalsV; // Данные по времени на считывание точки (статистические данные) prPointTimeData:tPointTimeData; // Данные по времени на цикл считывание точки (статистические данные) prCicleTimeData:tCicleTimeData; // Данные по времени на снятие спектра prDateTime:tSpecterDateTime; // Список оповещени prNotifyList:tNotifyList; prNotifyWindowsList:tNotifyWindowsList; // Имя файла конфигурации prConfigFileName:string; // Данные автообновления вольтметра prVoltmeter:tVoltmeterData; // Данные автообновления вольтметра prSourcesData:tSourcesData; // Данные спектра prDataSeries:tDataSeries; // Размещения спектра prAssignedDataSeries:tSeriesSet; // Данные по массе для фоновой установки массы prBackgroundMass:tMass; // Данные по счетчику для фоновой установки счетчика prBackgroundCounter:tCounter; // Главный канал измерени prMainChannel:tSeries; // Формат текстового файла prFileFormat:tMsDataFileFormat; // Набор каналов ПНЧ для фоновой калибровки prPNCCalibrateChannelSet:tChannels; // Набор активных (размещенных в памяти) серий (данных измерения для каналов) prActiveSeries:tSeriesSet; // Время интегрировани prIntegrationTime:cardinal; // для быстрого доступа prIntegrationTimeX:cardinal; // для фоновой установки // список диапазонов prRanges:tRangeList; // активный диапазон prActiveRange:TRange; // номер активного диапазона prActiveRangeIndex:integer; // таймер проверки флагов опасности для оборудовани prEmergencyCheckTimer:tMMTimer; // последнее значение флагов опасности для оборудования считанное таймером prCtrlFlags:tCtrlFlags; // список устройств для установки значений prDevicesToReset:tDevices; // указатель на процедуру фонового измерени prDoMeasure:procedure of object; // Значение различия показаний по счетчику и ИМЧ, // при превышении которого шкала масс синхронизуется автоманически prAutoMassScaleSyncWithIMChValue:tMass; // Число сделанных автосинхронизаций prAutoMassScaleSyncWithIMChCount:cardinal; // Имя файла данных в который записываются-читаются серии функцией // DataSeriesWriteDataToFileX, DataSeriesReadDataFromFileX prDataFileName:string; // размещение и инициализация внутренних данных procedure Init; // установка-очистка внутренних флагов procedure FlagSet(f:tFlag); procedure FlagClear(f:tFlag); // установка начального объема хранилища данных procedure InitialCapacitySet(ic:integer); // управление клапанами procedure ValveSetOpen(AValve:tSource); function ValveGetOpen:tSource; function ValveAutoCloseTimeGet:cardinal; procedure ValveAutoCloseTimeSet(dT_ms:cardinal); function ValveAutoCloseTimesGet(Valve:tSource):cardinal; procedure ValveAutoCloseTimesSet(Valve:tSource; dT_ms:cardinal); function ValveAutoCloseEnabledGet:boolean; procedure ValveAutoCloseActivate(Valve:tSource); procedure ValveAutoCloseActivateTiks; procedure ValveTerminateTimers; procedure ValveDoAutoClose(Sender:tObject); procedure ValveOneSecondTimerTick(Sender:tObject); function ValveAutoCloseTimeLeftGet:cardinal; function ValveOpenedTimeGet:cardinal; // Операция проверки флагов опасности для оборудования считанное таймером procedure DoEmergencyCheck(Sender:tObject); // управление временем интегрировани function IntegrationTimeGet:cardinal; procedure IntegrationTimeSet(Time:cardinal); procedure IntegrationTimeSetInternal(Time:word); procedure IntegrationTimeSetX(Time:cardinal); // готовность к фоновому измерению function MeasureAvailable:boolean; function MeasureAvailableEx:boolean; // управление переключателями function SwitchesGet:tMSSwitches; procedure SwitchesSet(x:tMSSwitches); // автоизмерение напряжений function VoltmeterGet(Channel:tVoltageChannel):tMkVolts; function VoltmeterCurrentGet:tMkVolts; function VoltmeterCurrentChannelGet:tVoltageChannel; procedure VoltmeterCurrentChannelSet(Channel:tVoltageChannel); function VoltmeterUpDateGet(Channel:tVoltageChannel):tMkVolts; procedure VoltmeterUpdateTimeSet(dT:cardinal); procedure VoltmeterUpdateChannelsSet(Channels:tVoltageChannels); procedure VoltmeterInitTimer; procedure VoltmeterDoAutoUpdate(Sender:TObject); // отработка изменений диапазона procedure HandleRangesChanged(Sender:TObject; ARangeIndex:integer; AItem:tRangeItem); // отработка изменения списка диапазонов procedure HandleRangeListChanged(Sender:TObject); // запуск-останов развертки диапазонов procedure RangesProcessStartStop; // развертка диапазона ARange function RangeScroll(ARange:tRange):boolean; // изменение массы при развертке диапазона ARange function RangeChangeMass(ARange:tRange):BOOLEAN; // получение сигнала при развертке диапазона ARange procedure RangeDataGet(ARange:tRange); // Сохранение данных по сигнала при развертке диапазона ARange procedure RangeDataStore(ARange:tRange); // Активный диапазон function RangeActiveGet:tRange; // Следующий диапазон function RangeNext:tRange; // Установка номера активного диапазона procedure ActiveRangeIndexSet(AIndex:integer); // Установка величины границы автосинхронизации с ИМЧ procedure AutoMassScaleSyncWithIMChValueSet(AValue:tMass); // Выполнение процедуры автосинхронизации с ИМЧ procedure DoAutoMassScaleSyncWithIMCh; protected // Драйвер масс-спектрометра prMsSp:tMsSp; // Установка кода ошибки procedure ErrorSet(e:tError); // Текстовое сообщение об ошибке function ErrorMsgGet:string; // Код ошибки драйвера function HardwareErrorGet:word; // Запуск измерения с начала procedure Start; // Время ожидания завершения операции function TimeOutGet:cardinal; // Очистка хранилища данных procedure ClearAll; // Частичная очистка хранилища данных, указанной ASeries procedure ClearSer(ASeries:tSeries); // Процедура фонового измерени procedure DoMeasure; // Процедура ПРОСТОГО измерени procedure DoMeasureSimple; // Процедура измерения ДИАПАЗОНОВ procedure DoMeasureRange; // Процедура изменения массы для ПРОСТОГО измерени procedure ChangeMass; // Процедура получения сигнала function DataGet0Ex(var ASignals:tSignalsV):boolean; procedure DataGet0; // Процедура получения сигнала для ПРОСТОГО измерени procedure DataGet; // Процедура получения и запоминания данных сигнала для ПРОСТОГО измерени function DataGetAndStore0:boolean; // Процедура записи сигнала в хранилище procedure DataStoreEx(const ASignals:tSignalsV; Accumulate:boolean); // Процедура записи сигнала в хранилище для ПРОСТОГО измерени procedure DataStore; // Полный интервал времени развертки function DeltaDateTimeGet:tDateTime; // Установка умножителя шага procedure StepMultiplicatorSet(SMult:integer); // Включение автоподстройки procedure AutoTuningSet(x:boolean); function AutoTuningGet:boolean; // Получение-установка данных по калибровке шкалы масс function CalibrationMGet:tMass; procedure CalibrationMSet(Mass:tMass); function CalibrationKGet:tMass; procedure CalibrationKSet(Koef:tMass); // Получение-установка данных по калибровке ИМЧ function IMChCalibrationMGet:tMass; procedure IMChCalibrationMSet(M0:tMass); function IMChCalibrationKGet:tMass; procedure IMChCalibrationKSet(K:tMass); function IMCh_MassGet:tMass; // Получение данных по флагам оборудовани function EqupmentFlagsGet:tEmergencyFlags; // Основная процедура MsThread procedure Execute; override; // Процедура выполнения фоновых команд MsThread procedure ExecuteCommand; // Код ошибки MsThread property ErrorCode:tError read prError write ErrorSet; // Процедуры оповещения о событиях MsThread procedure Notify(Event:tMI1201_Thread_EventID); procedure NotifyEx(Event:tMI1201_Thread_EventID; parByte:byte; parWord:word); procedure NotifyA(Event:tMI1201_Thread_Event); // Доступ к реестру function Registry:tRegistry; // Установка-чтение имени файла конфигурации procedure ConfigFileNameSet(NewName:string); function ConfigFileNameGet:string; // Запись/чтение конфигурации MsThread в реестр procedure ConfigSave; procedure ConfigRead; // Восстановление (актуализация) конфигурации после включени procedure ConfigRestore; // Запись/чтение конфигурации драйвера в файл function DriverConfigSave:boolean; function DriverConfigRead:boolean; // Установка-чтение счетчика procedure CounterSet(c:tCounter); procedure CounterXSet(c:tCounter); function CounterGet:tCounter; // Установка-чтение массы procedure MassSet0(m:tMass); // без ожидани procedure MassSet(m:tMass); procedure MassXSet(m:tMass); function MassGet:tMass; // Установка-чтение шага по шкале масс procedure MassStepSet(m:tMass); function MassMinStepGet:tMass; function MassStepAbsoluteMinGet:tMass; // Чтение признака аппаратного предела function MassLimitGet:boolean; function MassLimitLowGet:boolean; function MassLimitUpGet:boolean; // Сброс признака аппаратного предела procedure MassLimitClear; // Установка-чтение калибровки ПНЧ function PNCCalibrationGet(cl:tChannel; par:tPNCCalibrationParameter):double; procedure PNCCalibrationSet(cl:tChannel; par:tPNCCalibrationParameter; Value:double); // Чтение состояния фонового измерени function MeasureStateGet:tMeasureState; // Чтение направления развертки function ScrollDirectionGet:tScrollDirection; function DataSeriesGet(s:tSeries):tMSSeries; function AssignedDataSeriesGet:tSeriesSet; function DataPointsCountGet:cardinal; function DataSeriesFirstIndex(Mass:tMass):integer; function DataSeriesLastIndex(Mass:tMass):integer; // Очистка хранилища - одной серии ASeries function ClearSeries(ASeries:tSeries):boolean; procedure DataMassShift(dM:tMass); procedure DataMassCalibrationChange(const OldCalibration:tMassCalibration); procedure MainChannelSet(AChannel:tSeries); procedure DataSeriesDrawLock(Sender:tObject); procedure DataSeriesDrawUnLock(Sender:tObject); procedure DataDeleteAllXRangeByValues(AMin,AMax:tMAss); procedure DataDeleteXRangeByValues(ASeriesSet:TSeriesSet; AMin,AMax:tMAss); procedure DataSeriesWrite(AStream:TStream); procedure DataSeriesRead(AStream:TStream); // Блокировка доступа к данным измерени procedure DataBeginWrite; procedure DataEndWrite; procedure DataBeginRead; procedure DataEndRead; // Блокировка доступа к данным драйвера procedure MsSpBeginWrite; procedure MsSpEndWrite; procedure MsSpBeginRead; procedure MsSpEndRead; procedure FileWriteAutoComment(AFileFormat:tMsDataFileFormat); procedure FileFormatSet(AFileFormat:tMsDataFileFormat); function RangeGet(i:integer):tRange; function RangeCountGet:integer; function RegistryOpenRootKey(ForWrite:boolean):boolean; function RegistryOpenKey(const SubKey:string; ForWrite:boolean):boolean; public // Объект блокировки доступа к MsThread Ready:tMCADThreadEvent; constructor Create(const Name:string); destructor Destroy; override; // Остановка (завершение) витка MsThread procedure Terminate; function WaitForTerminate(TimeOut:cardinal):tWaitResult; property Finished:boolean read prFinished; property InitialCapacity:integer read prDefaultCapacity write InitialCapacitySet; property InitialDataSize:cardinal read prInitialDataSize; // Включение-выключение оборудования (без остановки витка) function SetON:boolean; function SetOff:boolean; procedure SetONX; procedure SetOffX; function IsON:boolean; // Состояние витка MsThread property MeasureState:tMeasureState read MeasureStateGet; // Направление развертки property ScrollDirection:tScrollDirection read ScrollDirectionGet; // Включение-выключение фонового измерени procedure Stop; // ВЫКЛючение развертки и измерени procedure Resume; // ВКЛЮчение развертки и измерени procedure Pause; // выключение развертки (StepMultiplicator:=0); // Прерывание операции драйвера procedure Abort; // Доступ к MsThread с ожиданием готовности function WaitForLimitedTime(dT:cardinal):TWaitResult; function Wait:TWaitResult; function WaitReady:boolean; function SetReady:boolean; // Текущее время ожидания готовности property TimeOut:cardinal read TimeOutGet; // Флаги состояния витка MsThread property Flags:tFlags read prFlags; // Флаги состояния оборудовани property EmergencyFlags:tCtrlFlags read prCtrlFlags; // Хранилище данных property DataSeries[series:tSeries]:tMSSeries read DataSeriesGet; function DataSeriesCount:integer; // число серий function DataValidSeriesCount(ASeriesSet:tSeriesSet):integer; // число реально существующих в хранилище серий из набора ASeriesSet property DataPointsCount:cardinal read DataPointsCountGet; // максимальное число точек во всех сериях function DataPointCount(ASeriesSet:TSeriesSet):integer; // максимальное число точек в сериях из набора ASeriesSe property AssignedDataSeries:tSeriesSet read AssignedDataSeriesGet; // число реально размещенных серий // Запись (чтение) данных хранилища в (из) файл (файла) procedure DataSeriesWriteDataToFile(const AFileName:string); procedure DataSeriesReadDataFromFile(const AFileName:string); procedure DataSeriesWriteData; procedure DataSeriesReadData; procedure DataSeriesWriteDataToFileX(const AFileName:string); procedure DataSeriesReadDataFromFileX(const AFileName:string); // Очистка хранилища function Clear:boolean; // Основной канал - на нем проводятся одноканальные операции property MainChannel:tSeries read prMainChannel write MainChannelSet; // Число точек (максимальное) в диапазоне [AMin,AMax] для набора ASeriesSet function DataRangePointsCount(ASeriesSet:tSeriesSet; AMin,AMax:tMass):integer; // Заполнение массива MathCAD точеками из диапазона [AMin,AMax] для набора ASeriesSet function DataArrayGet(const AMcadArray:ComplexArray; ASeriesSet:tSeriesSet; AMin,AMax:tMass):boolean; function DataPointGet(AMass:tMass; const ASignals:ComplexArray):boolean; // Управление ошибками property Error:tError read prError; // код ошибки property ErrorMsg:string read ErrorMsgGet; // текст-сообщение об ошибке property HardwareError:word read HardwareErrorGet; // код ошибки драйвера procedure ResetError; // сброс ошибки function IsError:boolean; // TRUE - состояние ошибки function IsNoError:boolean; // TRUE - состояние БЕЗ ошибки function DriverErrorGetFirstCtrlWith:tController; function DriverErrorCodeFromSubCtrl(c:tController):word; // Текущая установленная масса для масс-спектрометра (присваивание устанавливает новую массу) property Mass:tMass read MassGet write MassSet; // Текущая установленная фоновая масса для масс-спектрометра // параметры и функции "имя_X" делают то же самое, что и функции "имя", // но в витке MsThread, т.е. в 'фоновом' режиме. Возврат из // функций "имя_X" происходит немедленно, а выполнение операции, // заданной функцией "имя_X" или присваиванием параметру "имя_X" происходит // после завершения текущей операции. // Возврат из функции "имя" или присваиванием параметру "имя_X" // происходит по завершению операции. property MassX:tMass read prBackgroundMass write MassXSet; // Текущая установленное значение счетчика для масс-спектрометра (присваивание устанавливает новое значение счетчика) property Counter:tCounter read CounterGet write CounterSet; // Фоновое значение счетчика для масс-спектрометра (присваивание устанавливает новое значение счетчика) property CounterX:tCounter read prBackgroundCounter write CounterXSet; // Округление массы до ближайшего доступного значени function MassAjust(AMass:tMass):tMass; // Текущий установленный шаг развертки по шкале масс property MassStep:tMass read prMass.StepX write MassStepSet; // Текущий минимально возможный шаг развертки по шкале масс property MassStepAbsoluteMin:tMass read MassStepAbsoluteMinGet; // Минимально возможный шаг развертки по шкале масс для массы AMass function MassStepAbsoluteMinCalculate(AMass:tMass):tMass; // Реальный шаг развертки по шкале масс с которым выполняется развертка, // это значение не меньше MassStepAbsoluteMin и не больше MassStep property MassStepMin:tMass read MassMinStepGet; // Реальный шаг развертки по шкале масс с которым выполнялась быя развертка, // для массы AMass и шага AStep function MassStepMinCalculate(AMass,AStep:tMass):tMass; // Значение последнего выполненого шага авторазвертки property MassStepLast:tMass read prMass.LastStep; // Достигнут аппаратный предел на шкале масс property MassLimit:boolean read MassLimitGet; property MassLimitLow:boolean read MassLimitLowGet; property MassLimitUp:boolean read MassLimitUpGet; // Установить ноль по аппартному пределу procedure MassZeroSetByHardware; procedure MassZeroSetByHardwareX; // Установить текущую массу БЕЗ изменения магнитного пол // (изменение калибровки шкалы масс) procedure RefineMass(m:tMass); procedure RefineMassEx(m:tMass; DoShiftData:boolean); // Установить текущий счетчик БЕЗ изменения магнитного пол // (изменение калибровки шкалы масс) procedure RefineCounter(c:integer); // Установить текущую калибровку шкалы масс function CalibrationSet(Mass0,Coefficient:tMass):boolean; // Прочитать текущую калибровку шкалы масс procedure CalibrationGet(var Mass0,Coefficient:tMass); property K:tMass read CalibrationKGet write CalibrationKSet; property M0:tMass read CalibrationMGet write CalibrationMSet; // Преобразовать счетчик -> массу function Counter2Mass(c:integer):tMass; // Преобразовать массу -> счетчик function Mass2Counter(m:tMass):integer; // Значение сигналов (последнее полученное) property Signals:tSignalsV read prSignalsV; // Статистические данные property PointTimeData:tPointTimeData read prPointTimeData; property CicleTimeData:tCicleTimeData read prCicleTimeData; // очистка статистических данных по времени на измерение точки procedure PointTimeDataClear; procedure CicleTimeDataClear; // Калибровка ПНЧ - преобразование числа импульсов в секунду к вольтам property PNCCalibration[cl:tChannel; par:tPNCCalibrationParameter]:double read PNCCalibrationGet write PNCCalibrationSet; // Выполнить калибровку ПНЧ procedure PNCCalibrate; procedure PNCCalibrateX; // Выполнить калибровку ПНЧ для каналов ChannelSet procedure PNCCalibrateEx(ChannelSet:tChannels); procedure PNCCalibrateExX(ChannelSet:tChannels); // Калибровка ИМЧ (индикатора массовых чисел) function IMChCalibrationSet(Mass0,Coefficient:tMass):boolean; procedure IMChCalibrationGet(var Mass0,Coefficient:tMass); property IMCh_M0:tMass read IMChCalibrationMGet write IMChCalibrationMSet; property IMCh_K:tMass read IMChCalibrationKGet write IMChCalibrationKSet; // Вычислить калибровку ИМЧ на основании текущей шкалы масс procedure IMChCalibrationCalculate; procedure IMChCalibrationRefineOrigin; // Масса по ИМЧ на основании текущей калибровки property IMCh_Mass:tMass read IMCh_MassGet; // Проверка ошибки оборудования - // если ошибка есть - возвращается TRUE и устанавливается состояние ошибки // ecHardwareError для MsThread function CheckHardwareError:boolean; // Установка параметров устройства Device в значение value procedure DeviceSet(Device:tDevice; value:integer); function DeviceSetX(Device:tDevice; value:integer):boolean; // Чтение значения параметра устройства Device function DeviceGet(Device:tDevice):integer; property Device[d:tDevice]:integer read DeviceGet write DeviceSet; // Чтение значения шага параметра устройства Device function DeviceStepGet(Device:tDevice):cardinal; property DeviceStep[d:tDevice]:cardinal read DeviceStepGet; // Чтение максимального числа шагов параметра устройства Device function DeviceMaxStepCount(Device:tDevice):cardinal; // Переустановка текущего значения параметра устройства Device function DeviceReset(d:tDevice):boolean; procedure DeviceResetX(d:tDevice); // Переустановка текущего значения параметров набора устройств ADevices function DevicesReset(ADevices:tDevices):tDevices; // Время, потребное на выполнение DeviceReset function DeviceOperationTime_ms(Device:tDevice; Step:integer):integer; // Выполнить один шаг изменения значения параметра устройства Device // в напрвлении Direction function DeviceDoStep(d:tDevice; Direction:TUpDownDirection):boolean; // Набор устройств для которых значения параметров установлены достоверно // - это связано с отсутствием в приборе обратной связи по некоторым устройствам function DevicesValidated:tDevices; // Аппаратные флаги оборудовани property EqupmentFlags:tEmergencyFlags read EqupmentFlagsGet; // Умножитель шага по шкале масс для авторазвертки. Равный 0 - останавливает развертку. property StepMultiplicator:integer read prMass.StepMultiplicator write StepMultiplicatorSet; // TRUE, если cl правильный и доступный номер канала измерений function ChannelValid(cl:byte):boolean; // Время запуска развертки property StartDateTime:tDateTime read prDateTime.StartDateTime; // Время остановки развертки property EndDateTime:tDateTime read prDateTime.EndDateTime; // Полный интервал времени развертки property DeltaDateTime:tDateTime read DeltaDateTimeGEt; // Открытый клапан property Valve:tSource read ValveGetOpen write ValveSetOpen; // Время открытия клапана property ValveOpenDateTime:tDateTime read prSourcesData.StartDateTime; // Интервал времени, в течение которого клапан открыт property ValveOpenedTime:cardinal read ValveOpenedTimeGet; // Интервалы времени, после которого клапан Valve будет автоматически закрыт property ValveAutoCloseTimes[Valve:tSource]:cardinal read ValveAutoCloseTimesGet write ValveAutoCloseTimesSet; // Интервалы времени, после которого открытый клапан будет автоматически закрыт property ValveAutoCloseTime:cardinal read ValveAutoCloseTimeGet write ValveAutoCloseTimeSet; // Авозакрытие клапана включено - TRUE property ValveAutoCloseEnabled:boolean read ValveAutoCloseEnabledGet; // Интервалы времени, оставшийся до автозакрытия открытого клапана property ValveAutoCloseTimeLeft:cardinal read ValveAutoCloseTimeLeftGet; // Текущие показания вольтметра property VoltmeterCurrent:tMkVolts read VoltmeterCurrentGet; // Текущий канал вольтметра property VoltmeterCurrentChannel: tVoltageChannel read VoltmeterCurrentChannelGet write VoltmeterCurrentChannelSet; // Показания вольтметра для канала MeasurePoint property Voltmeter[MeasurePoint:tVoltageChannel]:tMkVolts read VoltmeterGet; // Период автообновления данных по вольтметру для канала MeasurePoint property VoltmeterUpdateTime:cardinal read prVoltmeter.Time write VoltmeterUpdateTimeSet; // Каналы автообновлени property VoltmeterUpdateChannels:tVoltageChannels read prVoltmeter.Channels write VoltmeterUpdateChannelsSet; // Данные автообновления для канала MeasurePoint property VoltmeterUpdateData[MeasurePoint:tVoltageChannel]:tMkVolts read VoltmeterUpDateGet; // Время интегрировани property IntegrationTime:cardinal read IntegrationTimeGet write IntegrationTimeSet; property IntegrationTimeX:cardinal read IntegrationTimeGet write IntegrationTimeSetX; // Автоподстройка property AutoTuning:boolean read AutoTuningGet write AutoTuningSet; // Состояние переключателей property Switches:tMSSwitches read SwitchesGet write SwitchesSet; // Включить function SwitchON(s:tMSSwitch):boolean; // Выключить function SwitchOFF(s:tMSSwitch):boolean; // Добавить процедуру оповещения об изменениях состояния MsThread procedure NotifyAdd(var NotifyHanler:tMI1201_Thread_NotifyData); // Удалить процедуру оповещения об изменениях состояния MsThread procedure NotifyDel(var NotifyHanler:tMI1201_Thread_NotifyData); // Имя файла сохранения-восстановления конфигурации драйвера. property ConfigFileName:string read prConfigFileName write ConfigFileNameSet; // Имя файла сохранения-восстановления данных по масс-спектру. // Генерируется так DataFileName:=ConfigFileName+'.dat'. function DataFileName:string; // Описание формата текстового файла данных, используемого по умолчанию property FileFormat:tMsDataFileFormat read prFileFormat write FileFormatSet; // Запись текстового файла данных с форматом AFileFormat procedure FileWriteFormatted(AFileFormat:tMsDataFileFormat); // Запись текстового файла данных с форматом по умолчанию procedure FileWrite; // Фоновая запись текстового файла данных с форматом по умолчанию procedure FileWriteX; property Ranges:tRangeList read prRanges; property Range[i:integer]:tRange read RangeGet; property RangeActive:TRange read prActiveRange; property RangeActiveIndex:integer read prActiveRangeIndex write ActiveRangeIndexSet; function RangeActiveSet(AIndex:integer; AState:boolean):boolean; function RangeIsActive(AIndex:integer):boolean; property AutoMassScaleSyncWithIMChValue:tMass read prAutoMassScaleSyncWithIMChValue write AutoMassScaleSyncWithIMChValueSet; property AutoMassScaleSyncWithIMChCount:cardinal read prAutoMassScaleSyncWithIMChCount; property TerminateTimeOut:cardinal read prTerminateTimeOut write prTerminateTimeOut; end; function SwitchesToStr(ASwitches:tMSSwitches; Sep:string):string; implementation Uses MCAD_MiscFuncs; function SwitchesToStr(ASwitches:tMSSwitches; Sep:string):string; var s:tMSSwitch; begin if Sep='' then Sep:='; '; for s:=Low(s) to High(s) do begin if not (s in cInActiveSwitches) then begin Result:=Result+cMSSwitchesName[s]+': '; if s in ASwitches then begin Result:=Result+'1'; end else begin Result:=Result+'0'; end; if s<>High(s) then Result:=Result+Sep; end; end; end; function tMsThread.RangeActiveSet(AIndex:integer; AState:boolean):boolean; begin Result:=(0<=AIndex) and (AIndex=0) and (RangeActiveIndexwrSignaled then Exit; prFinished:=FALSE; Init; Suspend; try while not Terminated do begin if (Flags*cDoFlags<>[]) then begin FlagClear(fDoMeasureCicleContinues); ExecuteCommand; end else if (fError in Flags) then begin FlagClear(fDoMeasureCicleContinues); if CheckHardwareError then begin Suspend; end; end else if (fStopMeasure in Flags) then begin FlagClear(fDoMeasureCicleContinues); prActiveRange.Enabled:=FALSE; prDateTime.EndDateTime:=Now(); Notify(evStop); Suspend; end else if MeasureAvailableEx then begin DoMeasure; end else begin FlagClear(fDoMeasureCicleContinues); Suspend; end; Priority:=tpNormal; end; Priority:=tpHighest; finally Notify(evTerminate); prFinished:=TRUE; prEventNotExecuted.SetEvent; end; end; procedure tMsThread.ExecuteCommand; var f:tFlags; begin f:=Flags*cDoFlags; Priority:=tpLower; if (fDoFileWrite in f) then begin FileWrite; end; if (fDoSetOn in f) then begin SetOn; end; if (fDoSetOff in f) then begin SetOff; end; if (fDoChangeMass in f) then begin Mass:=MassX; end else if (fDoChangeCounter in f) then begin Counter:=CounterX; end; if (fDoPNCCalibration in f) then begin if (fDoPNCCalibrationEx in f) then begin PNCCalibrateEx(prPNCCalibrateChannelSet); end else begin PNCCalibrate; end; end; if (fDoResetDevice in f) then begin DevicesReset(prDevicesToReset); end; if (fDoSetIntegrationTime in f) then begin IntegrationTime:=prIntegrationTimeX; end; if (fDoDataFileRead in f) then begin DataSeriesReadDataFromFile(prDataFileName); end; if (fDoDataFileWrite in f) then begin DataSeriesWriteDataToFile(prDataFileName); end; if (fDoMassZeroSetByHardware in f) then begin MassZeroSetByHardware; end; prFlags:=prFlags-f; end; procedure tMsThread.DoMeasureSimple; begin (* MsSpBeginWrite; try ChangeMass; DataGet; finally MsSpEndWrite; end; if IsNoError then begin DataStore; end;*) MsSpBeginWrite; try ChangeMass; DataGetAndStore0; finally MsSpEndWrite; end; end; function tMsThread.RangeChangeMass(ARange:tRange):boolean; var m,dm:tMass; begin Result:=TRUE; if ARange.PointProcessed then begin m:=MassAjust(ARange.Mass+MassStepMinCalculate(ARange.Mass,ARange.Step)); dm:=m-ARange.Mass; ARange.Mass:=m; if ARange.InRange(MassStepAbsoluteMinCalculate(ARange.Mass)) then begin prMass.LastStep:=dm; ARange.PointProcessed:=FALSE; end else begin ARange.Enabled:=FALSE; Result:=FALSE; end; end; end; function tMsThread.RangeScroll(ARange:tRange):boolean; var DoDataGet:boolean; begin Result:=true; if not ARange.Enabled then begin Result:=FALSE; end else begin if RangeChangeMass(ARange) then begin DoDataGet:= not ARange.UseExistingPoints or (prDataSeries[sIntegrationTime].xValues.Locate(ARange.Mass)=-1); if DoDataGet then begin MsSpBeginWrite; try MassSet0(ARange.Mass); if MassLimit then begin ARange.Enabled:=False; DoDataGet:=FALSE; end else begin RangeDataGet(ARange); end; finally MsSpEndWrite; end; end else ARange.PointProcessed:=TRUE; { if DoDataGet and IsNoError then begin RangeDataStore(ARange); end;} end else begin Result:=true; end; end; end; procedure tMsThread.DoMeasureRange; var DoDataGet:boolean; begin if (StepMultiplicator<>0) then begin if not prActiveRange.Enabled then RangeNext; RangeScroll(prActiveRange); end else begin DoMeasureSimple; end; end; procedure tMsThread.DoMeasure; // var Time:cardinal; begin if WaitReady then begin // Time:=GetTickCount; try prDoMeasure; except ErrorSet(ecUnexpectedError); end; SetReady; (* if mstCalculateStatisticalData in prSwitches then begin // Вычисление статистических данных try Time:=TimeSpentTicks(Time); if (Time>=Signals.Time) then begin Dec(Time,Signals.Time); prPointTimeData.LastTime:=Time; if Time>prPointTimeData.MaxTime then prPointTimeData.MaxTime:=Time; if Time0 then begin m:=Mass; MassSet0(m+dm); prMass.LastStep:=Mass-m; end; FlagClear(fPointProcessed); end; end; function tMsThread.DataGet0Ex(var ASignals:tSignalsV):boolean; begin try MsSpBeginRead; try prMsSp.exSignalsVGet(ASignals); finally MsSpEndRead; end; CheckHardWareError; except ErrorSet(ecUnexpectedError); end; Result:=IsNoError; end; procedure tMsThread.DataGet0; begin DataGet0Ex(prSignalsV); end; procedure tMsThread.DataGet; begin if not (fPointProcessed in Flags) and not (fStopMeasure in Flags) then begin FlagSet(fPointProcessed); DataGet0; end; end; function tMsThread.DataGetAndStore0:boolean; var StartTime,Time:cardinal; begin Result:=FALSE; MsSpBeginRead; try StartTime:=GetTickCount; prMsSp.exSignalsStart; if (fDataValid in Flags) then begin DataStore; FlagClear(fDataValid); if mstCalculateStatisticalData in prSwitches then try if fDoMeasureCicleContinues in Flags then begin Time:=TimeSpentTicks(prCicleTimeData.Start); if Time>=Signals.Time then begin Dec(Time,Signals.Time); prCicleTimeData.LastTime:=Time; if Time>prCicleTimeData.MaxTime then prCicleTimeData.MaxTime:=Time; if Time=Signals.Time) then begin Dec(Time,Signals.Time); prPointTimeData.LastTime:=Time; if Time>prPointTimeData.MaxTime then prPointTimeData.MaxTime:=Time; if Time'') then begin MsSpBeginWrite; try prMsSp.RestoreFromFile(PChar(ConfigFileName)); MsSpEndWrite; except MsSpEndWrite; end; end; prIntegrationTime:=IntegrationTime; ConfigRead; for ds:=Low(ds) to High(ds) do begin if ds in prActiveSeries then begin s:=tMSSeries.CreateX(NIL,InitialCapacity); prDataSeries[ds]:=s; if Assigned(s) then begin s.Title:=cMSSeriesTitles[ds]; if s.Name='' then s.Name:=cMSSeriesNames[ds]; s.BeforeDrawValues:=DataSeriesDrawLock; s.AfterDrawValues:=DataSeriesDrawUnLock; end; end; end; // Поскольку требуется сначала прочитать InitialCapacity if RegistryOpenKey('DataSeries',FALSE) then begin for ds:=Low(ds) to High(ds) do begin s:=prDataSeries[ds]; If Assigned(s) then try s.RestoreParameters(Registry); except end; end; end; if mstAutoSaveData in prSwitches then try DataSeriesReadData; except end; // prMeasureType:=mRepeatedlyWithDataReplace; // prBackgroundMeasureState:=bmsNotStarted; prSourcesData.StartTime.Start(1); try prEmergencyCheckTimer:=tMMTimer.Create; prEmergencyCheckTimer.Enabled:=FALSE; prEmergencyCheckTimer.TimerType:=ttPeriodic; prEmergencyCheckTimer.Resolution:=1000; prEmergencyCheckTimer.Interval:=5000; prEmergencyCheckTimer.OnTimer:=DoEmergencyCheck; except end; GlobalMemoryStatus(ms); Inc(prInitialDataSize,Mem-ms.dwAvailVirtual); finally Ready.Exit; DataEndWrite; end; end; destructor tMsThread.Destroy; var ds:tSeries; begin Notify(evDestroy); FreeOnTerminate:=FALSE; if not Finished and not Suspended and (GetCurrentThreadID<>ThreadID) then begin WaitForTerminate(30000); if not Finished then Suspend; end; SetOff; // Блокировка данных перед уничтожением DataBeginWrite; MsSpBeginWrite; prEmergencyCheckTimer.Enabled:=False; prSourcesData.OneSecondTickTimer.Enabled:=False; prSourcesData.Timer.Enabled:=False; prVoltmeter.Timer.Enabled:=False; for ds:=Low(ds) to High(ds) do begin if Assigned(prDataSeries[ds]) then prDataSeries[ds].ParentChart:=NIL; end; ConfigSave; if (mstAutoSaveData in prSwitches) and (fDataChanged in Flags) then try DataSeriesWriteData; except end; for ds:=Low(ds) to High(ds) do begin FreeAndNil(prDataSeries[ds]); end; FreeAndNil(prSourcesData.OneSecondTickTimer); FreeAndNil(prSourcesData.Timer); FreeAndNil(prVoltmeter.Timer); FreeAndNil(prEmergencyCheckTimer); FreeAndNil(prNotifyList); FreeAndNil(prNotifyWindowsList); FreeAndNil(prLocalRegistry); FreeAndNil(prFileFormat); FreeAndNil(prRanges); FreeAndNil(prActiveRange); prMsSp.Done; FreeAndNil(prMsSpLock); FreeAndNil(prDataLock); FreeAndNil(Ready); FreeAndNil(prEventNotExecuted); Inherited Destroy; end; function tMsThread.Wait; begin Result:=Ready.WaitFor(TimeOut); end; function tMsThread.WaitReady; begin Result:=Ready.WaitFor(TimeOut)=wrSignaled; end; function tMsThread.SetReady; begin Result:=Ready.Exit; end; function tMsThread.TimeOutGet; begin Result:=cDefaultTimeOut+prMsSp.OperationTime; end; function tMsThread.CheckHardwareError:boolean; begin Result:=prMsSp.Error; if Result then begin Ready.WaitForever:=False; Ready.SetEventAnyway; ErrorSet(ecHardwareError); end else if ErrorCode=ecHardwareError then begin ErrorSet(ecOK); end; end; function tMsThread.HardwareErrorGet:word; begin Result:=prMsSp.ErrorCode; end; procedure tMsThread.ErrorSet(e:tError); begin Notify(evError); if e=ecOK then begin prError:=ecOK; FlagClear(fError); prMsSp.SetNoError; Notify(evClearError); Resume; end else if prError=ecOK then begin prError:=e; FlagSet(fError); Notify(evSetError); Stop; end; end; function tMsThread.ErrorMsgGet:string; begin if prMsSp.Error then Result:=prMsSp.ErrorMessageCurrent else case ErrorCode of ecOK: Result:='Все в порядке'; ecHardwareError: Result:='Ошибка оборудования'; ecNoMemory: Result:='Недостаточно памяти'; ecUnexpectedError: Result:='Неожиданная и неизвестная ошибка'; else Result:='Описание ошибки недоступно'; end; end; procedure tMsThread.ClearAll; var s:tSeries; begin DataBeginWrite; try for s:=Low(s) to High(s) do begin ClearSeries(s); end; Notify(evPointsNumberChanged); finally DataEndWrite; end; end; procedure tMsThread.ClearSer(ASeries:tSeries); var sr:tMSSeries; begin DataBeginWrite; try sr:=DataSeries[ASeries]; if Assigned(sr) then begin sr.Clear; FlagSet(fDataChanged); end; finally DataEndWrite; end; end; function tMsThread.Clear; begin Result:=FALSE; if WaitReady then begin Result:=TRUE; ClearAll; SetReady; end; end; function tMsThread.ClearSeries(ASeries:tSeries):boolean; begin Result:=FALSE; if WaitReady then begin Result:=TRUE; ClearSer(ASeries); SetReady; end; end; function tMsThread.CalibrationSet; var OldMC:tMassCalibration; OldM0,OldK:tMass; begin Result:=false; if WaitReady then begin try MsSpBeginRead; try prMsSp.MassCalibrationGet(OldM0,OldK); finally MsSpEndRead; end; if (OldM0<>Mass0) or (OldK<>Coefficient) then begin MsSpBeginWrite; try prMsSp.MassCalibration(OldMC); prMsSp.MassCalibrationSet(Mass0,Coefficient); finally MsSpEndWrite; end; DataMassCalibrationChange(OldMC); Notify(evMassCalibrationChanged); end; Result:=TRUE; finally SetReady; end; end; end; procedure tMsThread.DataMassCalibrationChange(const OldCalibration:tMassCalibration); var i:tSeries; s:tMSSeries; NewMC:tMassCalibration; begin DataBeginWrite; try MsSpBeginRead; try prMsSp.MassCalibration(NewMC); finally MsSpEndRead; end; prSignalsV.Mass:=NewMC.Counter2Mass(OldCalibration.Mass2Counter(prSignalsV.Mass)); for i:=Low(i) to High(i) do begin s:=prDataSeries[i]; if Assigned(s) then begin s.RecalculateXValues(OldCalibration,NewMC); end; end; finally DataEndWrite; end; end; procedure tMsThread.CalibrationGet; begin try prMsSp.MassCalibrationGet(Mass0,Coefficient); except Mass0:=-1; Coefficient:=-1; end; end; function tMsThread.CalibrationMGet:tMass; var x:tMass; begin CalibrationGet(Result,x); end; procedure tMsThread.CalibrationMSet; begin CalibrationSet(Mass,K); end; function tMsThread.CalibrationKGet:tMass; var x:tMass; begin CalibrationGet(x,Result); end; procedure tMsThread.CalibrationKSet; begin CalibrationSet(M0,Koef); end; function tMsThread.IMChCalibrationSet; begin Result:=false; if WaitReady then begin try prMsSp.IMChCalibrationSet(Mass0,Coefficient); Notify(exMNICalibrationChanged); Result:=TRUE; except end; SetReady; end; end; procedure tMsThread.IMChCalibrationCalculate; begin if WaitReady then begin try prMsSp.exIMChCalibrationCalculateDefault; except end; SetReady; end; end; procedure tMsThread.IMChCalibrationRefineOrigin; begin if WaitReady then begin try prMsSp.exIMChCalibrationRefineOrigin; except end; SetReady; end; end; procedure tMsThread.IMChCalibrationGet; begin try prMsSp.IMChCalibrationGet(Mass0,Coefficient); except Mass0:=-1; Coefficient:=-1; end; end; function tMsThread.IMChCalibrationMGet:tMass; var x:tMass; begin IMChCalibrationGet(Result,x); end; procedure tMsThread.IMChCalibrationMSet; begin IMChCalibrationSet(M0,IMCh_K); end; function tMsThread.IMChCalibrationKGet:tMass; var x:tMass; begin IMChCalibrationGet(x,Result); end; procedure tMsThread.IMChCalibrationKSet; begin IMChCalibrationSet(IMCh_M0,K); end; function CheckDataPoint(var dp:ComplexArray):boolean; begin Result:=(dp.hReal<>NIL) and (dp.Rows>=(Ord(High(tChannel))+2)) and (dp.Cols>=1); end; procedure tMsThread.StepMultiplicatorSet; const cMaxMul=1000; begin try if SMult>cMaxMul then SMult:=cMaxMul else if SMult<-cMaxMul then SMult:=-cMaxMul; prMass.StepMultiplicator:=SMult; Notify(evStepMultiplicatorChanged); except end; end; procedure tMsThread.Start; begin prActiveRange.Enabled:=FALSE; FlagClear(fPointProcessed); // prMass.Pass:=1; prDateTime.StartDateTime:=Now(); prDateTime.EndDateTime:=prDateTime.StartDateTime; Notify(evStart); Resume; end; procedure tMsThread.Stop; begin FlagSet(fStopMeasure); end; procedure tMsThread.Resume; var Resumed:boolean; begin FlagClear(fStopMeasure); Resumed:=Suspended; Inherited Resume; if Resumed then Notify(evResume); end; procedure tMsThread.Pause; begin StepMultiplicator:=0; Resume; end; procedure tMsThread.ValveSetOpen; var curValve:tSource; begin curValve:=Valve; try ValveTerminateTimers; if not IsON then Exit; if not prMsSp.exSwitchIsON(fValvesControl) then prMsSp.exSwitchTurnON(fValvesControl); if prMsSp.NoError then begin prMsSp.exSourceSet(AValve); end else begin prMsSp.ctrlPanel.SetNoError; prMsSp.ctrlPanel.exSourceSet(AValve); end; prSourcesData.StartTime.Start(ValveAutoCloseTimes[AValve]); ValveAutoCloseActivate(AValve); if Valve<>curValve then prSourcesData.StartDateTime:=Now(); except ErrorSet(ecUnexpectedError); end; CheckHardwareError; if Valve<>curValve then Notify(evSourceChanged); end; procedure tMsThread.ValveTerminateTimers; begin if Assigned(prSourcesData.Timer) then prSourcesData.Timer.Enabled:=False; if Assigned(prSourcesData.OneSecondTickTimer) then prSourcesData.OneSecondTickTimer.Enabled:=False; end; procedure tMsThread.ValveAutoCloseActivate(Valve:tSource); begin ValveTerminateTimers; if (Valve<>sCloseAll) and (ValveAutoCloseTimes[Valve]>0) then begin if not Assigned(prSourcesData.Timer) then begin try prSourcesData.Timer:=tMMTimer.Create; prSourcesData.Timer.Enabled:=FALSE; prSourcesData.Timer.Resolution:=1; prSourcesData.Timer.TimerType:=ttOnce; prSourcesData.Timer.OnTimer:=ValveDoAutoCLose; except FreeAndNil(prSourcesData.Timer); end; end; if Assigned(prSourcesData.Timer) then begin Exclude(prFlags,fSourceTimerFails); prSourcesData.Timer.Interval:=ValveAutoCloseTimes[Valve]; prSourcesData.Timer.Enabled:=True; if not prSourcesData.Timer.Enabled then begin Include(prFlags,fSourceTimerFails); end else begin ValveAutoCloseActivateTiks; end; end else begin Include(prFlags,fSourceTimerFails); end; end; end; procedure tMsThread.ValveAutoCloseActivateTiks; begin if Assigned(prSourcesData.OneSecondTickTimer) then prSourcesData.OneSecondTickTimer.Enabled:=False; if Assigned(prSourcesData.Timer) and (prSourcesData.Timer.Interval>1000) then begin if not Assigned(prSourcesData.OneSecondTickTimer) then begin try prSourcesData.OneSecondTickTimer:=tMMTimer.Create; prSourcesData.OneSecondTickTimer.Enabled:=FALSE; prSourcesData.OneSecondTickTimer.Resolution:=100; prSourcesData.OneSecondTickTimer.TimerType:=ttPeriodic; prSourcesData.OneSecondTickTimer.OnTimer:=ValveOneSecondTimerTick; except FreeAndNil(prSourcesData.OneSecondTickTimer); end; end; if Assigned(prSourcesData.OneSecondTickTimer) then prSourcesData.OneSecondTickTimer.Enabled:=TRUE; end; end; procedure tMsThread.ValveOneSecondTimerTick; begin Notify(evValveOneSecondTimerTick); end; function tMsThread.ValveGetOpen; begin if not IsOn then begin Result:=sBad; Exit; end; try Result:=prMsSp.exSource; except Result:=sBad; CheckHardwareError; end; end; function tMsThread.DeviceStepGet(Device:tDevice):cardinal; begin if Device in [IonizationVoltage..SEM_Voltage] then begin try Result:=prMsSp.DeviceUStep(Device); except Result:=0; end; end else begin Result:=0; end; end; function tMsThread.DeviceMaxStepCount(Device:tDevice):cardinal; begin if Device in [IonizationVoltage..SEM_Voltage] then begin try Result:=prMsSp.DeviceCounterMax(Device); except Result:=0; end; end else begin Result:=0; end; end; function tMsThread.DeviceOperationTime_ms(Device:tDevice; Step:integer):integer; begin if Device in [IonizationVoltage..SEM_Voltage] then begin try Result:=prMsSp.DeviceCounterMax(Device); if Result>Abs(Step) then Result:=Abs(Step); Result:=Result*prMsSp.ctrlISSB.CurStepDelay*4; except Result:=0; end; end else begin Result:=0; end; end; function tMsThread.DeviceDoStep(d:tDevice; Direction:TUpDownDirection):boolean; var c:word; begin Result:=FALSE; if d in [IonizationVoltage..SEM_Voltage] then begin try c:=prMsSp.DeviceCounter(d); case Direction of updUp: prMsSp.ctrlISSB.exStepUp(c_ISSB.tUnit(d)); updDown: prMsSp.ctrlISSB.exStepDown(c_ISSB.tUnit(d)); else end; CheckHardwareError; Result:=prMsSp.DeviceCounter(d)<>c ; if Result then begin NotifyEx(evDevice,Ord(d),0); end; except end; end; end; function tMsThread.DeviceSetX(Device:tDevice; value:integer):boolean; var DoSet:boolean; begin Result:=False; if not IsON then Exit; try MsSpBeginRead; try DoSet:=(prMsSp.DeviceUMin(Device)<=Value) and (Value<=prMsSp.DeviceUMax(Device)) and (prMsSp.DeviceU(Device)<>Value); finally MsSpEndRead; end; if DoSet then begin MsSpBeginWrite; try prMsSp.exDeviceUSet(Device,Value); finally MsSpEndWrite; end; CheckHardwareError; Result:=(ErrorCode=ecOK); NotifyEx(evDevice,Ord(Device),0); end; except CheckHardwareError; end; end; procedure tMsThread.DeviceResetX(d:tDevice); begin if d in [IonizationVoltage..CorrectionZ] then begin Include(prDevicesToReset,d); FlagSet(fDoResetDevice); end; end; function tMsThread.DevicesReset(ADevices:tDevices):tDevices; var d:tDevice; begin for d:=Low(d) to High(d) do if (d in ADevices) then DeviceReset(d); end; function tMsThread.DeviceReset(d:tDevice):boolean; begin if d in [IonizationVoltage..CorrectionZ] then begin try MsSpBeginWrite; try prMsSp.exDeviceReset(d); finally MsSpEndWrite; end; CheckHardwareError; NotifyEx(evDevice,Ord(d),0); except CheckHardwareError; end; Result:=(ErrorCode=ecOK); end else begin Result:=TRUE; end; end; procedure tMsThread.DeviceSet(Device:tDevice; value:integer); begin DeviceSetX(Device,value); end; function tMsThread.DeviceGet; begin Result:=Low(Result); if Device in [IonizationVoltage..SEM_Voltage] then begin try if IsON then Result:=prMsSp.DeviceU(Device) else Result:=0; except CheckHardwareError; end; end; end; function tMsThread.DevicesValidated:tDevices; begin prMsSp.ctrlISSB.CheckedUnitsGet(c_ISSB.TUnits(Result)); end; function tMsThread.Mass2Counter(m:tMass):integer; begin Result:=prMsSp.Mass2Counter(m); end; function tMsThread.Counter2Mass(c:integer):tMass; begin Result:=prMsSp.Counter2Mass(c); end; function tMsThread.IntegrationTimeGet; begin Result:=prMsSp.IntegrationTime; prIntegrationTime:=Result; end; procedure tMsThread.IntegrationTimeSet; begin if Time=0 then Time:=1 else if Time>prMsSp.IntegrationTimeMax then Time:=prMsSp.IntegrationTimeMax; if Time=IntegrationTime then Exit; if WaitReady then begin try IntegrationTimeSetInternal(Time); NotifyEx(evIntegrationTimeChanged,0,Time); except end; end; SetReady; end; procedure tMsThread.IntegrationTimeSetInternal(Time:word); begin MsSpBeginWrite; try prMsSp.IntegrationTimeSet(Time); prIntegrationTime:=Time; finally MsSpEndWrite; end; end; procedure tMsThread.IntegrationTimeSetX; begin if prIntegrationTime=Time then Exit; prIntegrationTimeX:=Time; FlagSet(fDoSetIntegrationTime); inherited Resume; end; procedure PointTimeDataClearEx(var APointTimeData:tPointTimeData); begin ZeroMemory(@APointTimeData,SizeOf(APointTimeData)); end; procedure tMsThread.PointTimeDataClear; begin ZeroMemory(@prPointTimeData,SizeOf(prPointTimeData)); prPointTimeData.MinTime:=High(prPointTimeData.MinTime); end; procedure tMsThread.CicleTimeDataClear; begin FlagClear(fDoMeasureCicleContinues); ZeroMemory(@prCicleTimeData,SizeOf(prCicleTimeData)); prCicleTimeData.MinTime:=High(prCicleTimeData.MinTime); end; procedure tMsThread.Abort; begin prMsSp.SetErrorCode(word(ecAbort)); Ready.WaitForever:=False; CheckHardwareError; Notify(evAbort); end; function tMsThread.EqupmentFlagsGet; begin prMsSp.exEmergencyFlagsGet(Result); end; function tMsThread.MeasureAvailable:boolean; begin Result:=IsON and IsNoError and (prFlags*cDoNotMeasureFlags=[]) and (prFlags*cDoMeasureFlags=cDoMeasureFlags); end; function tMsThread.MeasureAvailableEx:boolean; begin Result:=IsON and IsNoError and not (fStopMeasure in Flags); end; procedure tMsThread.AutoTuningSet(x:boolean); begin if AutoTuningGet<>x then begin if WaitReady then begin prMsSp.ctrlRoll.exAutoTuning(x); SetReady; end; end; end; function tMsThread.AutoTuningGet:boolean; begin Result:=prMsSp.ctrlRoll.AutoTuning; end; function tMsThread.DeltaDateTimeGet:tDateTime; begin Result:=EndDateTime-StartDateTime; end; function tMsThread.ChannelValid; begin Result:={(Ord(Low(tChannel))<=cl) and} (cl<=Ord(High(tChannel))); end; procedure tMsThread.ResetError; begin ErrorCode:=ecOK; end; (*function tMsThread.SpecialFeaturesGet:tFeatures; begin prMsSp.SpecialFeaturesGet(Result); end; procedure tMsThread.SpecialFeaturesSet(x:tFeatures); begin prMsSp.SpecialFeaturesSet(x); Notify(evSwitchesChanged); end; *) procedure tMsThread.NotifyAdd(var NotifyHanler:tMI1201_Thread_NotifyData); begin if not Assigned(prNotifyList) then Exit; prNotifyList.Add(NotifyHanler); end; procedure tMsThread.NotifyDel(var NotifyHanler:tMI1201_Thread_NotifyData); begin if not Assigned(prNotifyList) then Exit; prNotifyList.Remove(NotifyHanler); end; procedure tMsThread.Notify(Event:tMI1201_Thread_EventID); begin NotifyEx(Event, 0,0); end; procedure tMsThread.NotifyEx(Event:tMI1201_Thread_EventID; parByte:byte; parWord:word); var Ev:tMI1201_Thread_Event; begin Ev.EventID:=Event; Ev.ParameterB:=ParByte; Ev.ParameterW:=ParWord; NotifyA(Ev); end; procedure tMsThread.NotifyA(Event:tMI1201_Thread_Event); begin // Сообщение через Windows-message окну prNotifyWindowsList.NotifyByList(Self,Event); // Прямой вызов-уведомление prNotifyList.NotifyByList(Self,Event); end; procedure tNotifyList.Add(var NotifyHanler:tMI1201_Thread_NotifyData); resourcestring AlreadyExist='Указатель %h уже присутствует в списке.'; Unassigned='Указатель %h = NIL.'; begin // if not (Assigned(Self)) then Exit; if not Assigned(Addr(NotifyHanler)) then begin Error(Unassigned, Integer(NotifyHanler)); end else if (IndexOf(Addr(NotifyHanler))<>-1) then begin Error(AlreadyExist, cardinal(Addr(NotifyHanler))); end else begin Inherited Add(Addr(NotifyHanler)); end; end; function tNotifyList.Remove(var NotifyHanler:tMI1201_Thread_NotifyData):Integer; begin // if not (Assigned(Self)) then Exit; Result:=Inherited Remove(Addr(NotifyHanler)); end; procedure tNotifyList.NotifyByList(Sender:TObject; Event:tMI1201_Thread_Event); var i:integer; p:^tMI1201_Thread_NotifyData; begin // if not (Assigned(Self)) then Exit; MultiReadExclusiveWriteSynchronizer.BeginRead; for i:=0 to Pred(Count) do begin p:=Items[i]; If Assigned(p^.NotifyHandler) then begin if (p^.NotifyMask=[]) or (Event.EventID in p^.NotifyMask) or (Event.EventID = evDestroy) then try p^.NotifyHandler(Sender,Event); except Delete(i); end; end else begin Delete(i); end; end; MultiReadExclusiveWriteSynchronizer.EndRead; end; procedure tNotifyWindowsList.Add(WindowsHanler:HWND); resourcestring AlreadyExist='Windows handle %h уже присутствует в списке.'; Unassigned='Windows handle %h = 0.'; begin // if not (Assigned(Self)) then Exit; if WindowsHanler=0 then begin Error(Unassigned, cardinal(WindowsHanler)); end else if (IndexOf(Pointer(WindowsHanler))<>-1) then begin Error(AlreadyExist, cardinal(WindowsHanler)); end else begin Inherited Add(Pointer(WindowsHanler)); end; end; function tNotifyWindowsList.Remove(WindowsHanler:HWND): Integer; resourcestring Unassigned='Windows handle %h = 0.'; begin // if not (Assigned(Self)) then Exit; if WindowsHanler=0 then begin Result:=-1; Error(Unassigned, cardinal(WindowsHanler)); end else begin Result:=Inherited Remove(Pointer(WindowsHanler)); end; end; procedure tNotifyWindowsList.NotifyByList(Sender:TObject; Event:tMI1201_Thread_Event); var i:integer; h:HWND; begin if not (Assigned(Self)) then Exit; MultiReadExclusiveWriteSynchronizer.BeginRead; // EnterCriticalSection(CriticalSection); for i:=0 to Pred(Count) do begin try h:=HWND(Items[i]); If h<>0 then begin PostMessage(h, CM_MI1201Notify, Longint(Event), Longint(Sender) ); end else begin Delete(i); end; except end; end; // LeaveCriticalSection(CriticalSection); MultiReadExclusiveWriteSynchronizer.EndRead; end; function tMsThread.DriverErrorGetFirstCtrlWith:tController; begin Result:=prMsSp.ErrorGetFirstCtrlWith; end; function tMsThread.DriverErrorCodeFromSubCtrl(c:tController):word; begin Result:=prMsSp.ErrorCodeFromSubCtrl(c); end; procedure AddFeatures(var s:tMSSwitches; f:tFeatures); begin if ffRollFastExInit in f then Include(s,sffRollFastExInit); if ffSkipValvesExInit in f then Include(s,sffSkipValvesExInit); if ffSkipValvesExDone in f then Include(s,sffSkipValvesExDone); if ffSkipExDoneAtAll in f then Include(s,sffSkipExDoneAtAll); if ffFastVoltsInit in f then Include(s,sffFastVoltsInit); if ffUseEmulator in f then Include(s,sffUseEmulator); end; function GetFeatures(s:tMSSwitches):tFeatures; begin Result:=[]; if sffRollFastExInit in s then Include(Result,ffRollFastExInit); if sffSkipValvesExInit in s then Include(Result,ffSkipValvesExInit); if sffSkipValvesExDone in s then Include(Result,ffSkipValvesExDone); if sffSkipExDoneAtAll in s then Include(Result,ffSkipExDoneAtAll); if sffFastVoltsInit in s then Include(Result,ffFastVoltsInit); if sffUseEmulator in s then Include(Result,ffUseEmulator); end; procedure AddControllersOff(var s:tMSSwitches; f:tControllers); begin ; if Bus in f then Include(s,sBus); if CVF in f then Include(s,sCVF); ; if Roll in f then Include(s,sMagnet); if Volts in f then Include(s,sVoltmeter); ; if Panel in f then Include(s,sPanel); ; if ISSB in f then Include(s,sBPGI); if Count in f then Include(s,sCounter); end; function GetControllersOff(s:tMSSwitches):tControllers; begin Result:=[]; if sBus in s then Include(Result,Bus); if sCVF in s then Include(Result,CVF); if sMagnet in s then Include(Result,Roll); if sVoltmeter in s then Include(Result,Volts); if sPanel in s then Include(Result,Panel); if sBPGI in s then Include(Result,ISSB); if sCounter in s then Include(Result,Count); end; procedure AddSwitches(var s:tMSSwitches; f:tSwitches); begin if fCVF_Beam in f then Include(s,sfCVF_Beam); if fCVF_Base0 in f then Include(s,sfCVF_Base0); if fCVF_InputEMU in f then Include(s,sfCVF_InputEMU); if fCVF_DoNotInvertInput in f then Include(s,sfCVF_DoNotInvertInput); if fBPGI in f then Include(s,sfBPGI); if f10kV in f then Include(s,sf10kV); if fSEM in f then Include(s,sfSEM); if fValvesControl in f then Include(s,sfValvesControl); // if fAllowHighVoltageAndSEM in f then Include(s,sfAllowHighVoltageAndSEM); end; function GetSwitches(s:tMSSwitches):tSwitches; begin Result:=[]; if sfCVF_Beam in s then Include(Result,fCVF_Beam); if sfCVF_Base0 in s then Include(Result,fCVF_Base0); if sfCVF_InputEMU in s then Include(Result,fCVF_InputEMU); if sfCVF_DoNotInvertInput in s then Include(Result,fCVF_DoNotInvertInput); if sfBPGI in s then Include(Result,fBPGI); if sf10kV in s then Include(Result,f10kV); if sfSEM in s then Include(Result,fSEM); if sfValvesControl in s then Include(Result,fValvesControl); // if sfAllowHighVoltageAndSEM in s then Include(Result,fAllowHighVoltageAndSEM); end; function tMsThread.SwitchesGet:tMSSwitches; var f:tFeatures; c:tControllers; s:tSwitches; begin Result:=[]; if not Assigned(Self) then Exit; prMsSp.SkipMaskGetForExInit(c); AddControllersOff(Result,c); prMsSp.SpecialFeaturesGet(f); AddFeatures(Result,f); if IsOn then begin prMsSp.exSwitchesGet(s); AddSwitches(Result,s); end; Result:=Result+prSwitches; end; procedure tMsThread.SwitchesSet(x:tMSSwitches); var xx:tMSSwitches; begin if not Assigned(Self) then Exit; xx:=Switches; if x=xx then Exit; if cMSSwitchesRequiredSynchonization*x=cMSSwitchesRequiredSynchonization*xx then begin prMsSp.SpecialFeaturesSet(GetFeatures(x)); prMsSp.SkipMaskSetForExInit(GetControllersOff(x)); if prMsSp.CompletelyInitiated then begin prMsSp.exSwitchesSet(GetSwitches(x)); end; end else if WaitReady then begin try prMsSp.SpecialFeaturesSet(GetFeatures(x)); prMsSp.SkipMaskSetForExInit(GetControllersOff(x)); if prMsSp.CompletelyInitiated then begin prMsSp.exSwitchesSet(GetSwitches(x)); end; finally SetReady; end; end; prSwitches:=x*cMSTSwitches; if (mstCalculateStatisticalData in xx)<>(mstCalculateStatisticalData in x) then FlagClear(fDoMeasureCicleContinues); if (mstUseRanges in xx)<>(mstUseRanges in x) then RangesProcessStartStop; Notify(evSwitchesChanged); end; function tMsThread.SwitchON(s:tMSSwitch):boolean; begin SwitchesSet(SwitchesGet+[s]); Result:=s in SwitchesGet; end; function tMsThread.SwitchOFF(s:tMSSwitch):boolean; begin SwitchesSet(SwitchesGet-[s]); Result:=not (s in SwitchesGet); end; procedure tMsThread.Terminate; begin Inherited Terminate; Priority:=tpHighest; if (GetCurrentThreadID<>ThreadID) and not Suspended then begin WaitForTerminate(TerminateTimeOut+TimeOut); end; if Suspended and FreeOnTerminate then Free; end; procedure tMsThread.ConfigFileNameSet(NewName:string); var ok:boolean; begin if not Assigned(Self) then Exit; NewName:=ExpandFileName(NewName); if UpperCase(ConfigFileName)=UpperCase(NewName) then Exit; ok:=TRUE; if FileExists(NewName) then begin end; if Ok then begin try Registry.CloseKey; Registry.Access:=KEY_WRITE; if Registry.OpenKey(cRegistryKeyRoot,TRUE) then begin try Registry.WriteString('ConfigFileName',NewName); prConfigFileName:=NewName; Notify(evConfigFileNameChanged); finally Registry.CloseKey; end; end; except end; end; end; function tMsThread.ConfigFileNameGet:string; begin if not Assigned(Self) then Exit; Result:=''; try Registry.CloseKey; Registry.Access:=KEY_READ; if Registry.OpenKey(cRegistryKeyRoot,FALSE) then begin try Result:=Registry.ReadString('ConfigFileName'); finally Registry.CloseKey; end; end; except Result:=''; end; if Result='' then begin Result:=ExtractFilePath(CurrentModuleFileName)+'MI-1201 AGM.cfg'; end; if FileExists(Result) then begin end; end; function tMsThread.Registry:tRegistry; begin if not Assigned(prLocalRegistry) then begin prLocalRegistry:=tRegistry.Create(KEY_READ); end; Result:=prLocalRegistry; end; function tMsThread.DriverConfigSave:boolean; begin Result:=FALSE; if (ConfigFileName<>'') then begin if WaitReady then begin try prMsSp.SaveToFile(PChar(ConfigFileName)); Result:=TRUE; except end; SetReady; end; end; end; (*function tMsThread.DriverConfigRestore:boolean; begin Result:=FALSE; if (ConfigFileName<>'') then begin if WaitReady then begin try prMsSp.exRestoreFromFile(PChar(ConfigFileName)); Result:=prMsSp.NoError; finally SetReady; end; end; end; end; *) function tMsThread.DriverConfigRead:boolean; begin Result:=(ConfigFileName<>''); if Result then begin Result:=WaitReady; if Result then begin MsSpBeginWrite; try prMsSp.RestoreFromFile(PChar(ConfigFileName)); Result:=prMsSp.NoError; finally MsSpEndWrite; SetReady; end; end; end; end; (*function tMsThread.DriverConfigDefault:boolean; var sw:tSwitches; begin try prMsSp.exSwitchesGet(sw); sw:=sw+[fCVF_Beam,fCVF_Base0, fCVF_InputEMU, fCVF_DoNotInvertInput, fBPGI,f10kV,fSEM]; prMsSp.exSwitchesSet(sw); Result:=TRUE; except Result:=TRUE; end; end; *) function tMsThread.ValveAutoCloseTimesGet(Valve:tSource):cardinal; begin Result:=prSourcesData.AutoCloseTimes[Valve]; end; function tMsThread.ValveAutoCloseTimeGet:cardinal; begin Result:=ValveAutoCloseTimes[Valve]; end; procedure tMsThread.ValveAutoCloseTimesSet(Valve:tSource; dT_ms:cardinal); begin if prSourcesData.AutoCloseTimes[Valve]<>dT_ms then begin prSourcesData.AutoCloseTimes[Valve]:=dT_ms; Notify(evSourceAutoCloseTimeChanged); end; ValveAutoCloseActivate(Valve); end; procedure tMsThread.ValveAutoCloseTimeSet(dT_ms:cardinal); begin if ValveAutoCloseTime<>dT_ms then begin ValveAutoCloseTimes[Valve]:=dT_ms; end; end; procedure tMsThread.ValveDoAutoCLose; begin if ValveAutoCloseEnabled then begin Valve:=sCloseAll; end; end; function tMsThread.ValveAutoCloseEnabledGet:boolean; begin Result:=(ValveAutoCloseTime>0) and Assigned(prSourcesData.Timer) and prSourcesData.Timer.Enabled; end; function tMsThread.VoltmeterCurrentGet:tMkVolts; begin If IsON then begin prMsSp.exVoltageSelectChannel; Result:=prMsSp.exVoltage; prVoltmeter.Data[prMsSp.VoltageChannel]:=Result; Notify(evVoltmeterNewMeasure); end else begin Result:=0; end; end; function tMsThread.VoltmeterGet(Channel:tVoltageChannel):tMkVolts; begin If IsON then begin prMsSp.VoltageChannelSet(Channel); Result:=VoltmeterCurrentGet; end else begin Result:=0; end; end; procedure tMsThread.VoltmeterUpdateTimeSet(dT:cardinal); begin if prVoltmeter.Time<>dT then begin prVoltmeter.Time:=dT; Notify(evVoltmeterAutoUpdateTimeChanged); end; VoltmeterInitTimer; end; procedure tMsThread.VoltmeterUpdateChannelsSet(Channels:tVoltageChannels); begin if prVoltmeter.Channels<>Channels then begin prVoltmeter.Channels:=Channels; Notify(evVoltmeterAutoUpdateChannelsChanged); end; VoltmeterInitTimer; end; procedure tMsThread.VoltmeterInitTimer; begin If not IsON then Exit; if (VoltmeterUpdateChannels<>[]) and (VoltmeterUpdateTime>0) then begin try if not Assigned(prVoltmeter.Timer) then begin prVoltmeter.Timer:=tMMTimer.Create; prVoltmeter.Timer.Resolution:=1; prVoltmeter.Timer.TimerType:=ttPeriodic; prVoltmeter.Timer.OnTimer:=VoltmeterDoAutoUpdate; end; except end; Exclude(prFlags,fVoltmeterTimerFails); if Assigned(prVoltmeter.Timer) then begin prVoltmeter.Timer.Enabled:=True; prVoltmeter.Timer.Interval:=VoltmeterUpdateTime; if not prVoltmeter.Timer.Enabled then Include(prFlags,fVoltmeterTimerFails); end else begin Include(prFlags,fVoltmeterTimerFails); end; end else begin if Assigned(prVoltmeter.Timer) then prVoltmeter.Timer.Enabled:=False; end; end; procedure tMsThread.VoltmeterDoAutoUpdate; var vuc:tVoltageChannels; c0,c:tVoltageChannel; begin If not IsON then Exit; vuc:=VoltmeterUpdateChannels; SetThreadPriority(GetCurrentThreadID,THREAD_PRIORITY_BELOW_NORMAL); if (vuc=[]) then Exit; try c0:=prMsSp.VoltageChannel; for c:=Low(c) to High(c) do if c in vuc then begin prMsSp.VoltageChannelSet(C); prMsSp.exVoltageSelectChannel; prVoltmeter.Data[c]:=prMsSp.exVoltage; end; prMsSp.VoltageChannelSet(c0); prMsSp.exVoltageSelectChannel; Notify(evVoltmeterAutoUpdate); except end; end; function tMsThread.VoltmeterUpDateGet(Channel:tVoltageChannel):tMkVolts; begin if Assigned(Self) then Result:=prVoltmeter.Data[Channel] else Result:=0; end; function tMsThread.VoltmeterCurrentChannelGet:tVoltageChannel; begin Result:=prMsSp.VoltageChannel end; procedure tMsThread.VoltmeterCurrentChannelSet(Channel:tVoltageChannel); begin If not IsON then Exit; prMsSp.VoltageChannelSet(Channel); Notify(evVoltmeterAutoUpdateChannelsChanged); end; function tMsThread.MassAjust(AMass:tMass):tMass; begin Result:=Counter2Mass(Mass2Counter(AMass)); end; procedure tMsThread.MassSet0(m:tMass); var OldMass:tMass; LargeChange:boolean; begin if m<>Mass then try Notify(evMassChange); LargeChange:=ABS(prMsSp.Mass2Counter(m)-prMsSp.Counter)>10000; if LargeChange then Notify(evMassAboutChangeForLargeValue); OldMass:=Mass; prMsSp.exJumpToMass(m); CheckHardwareError; if LargeChange then Notify(evMassAboutChangeForLargeValueEnd); if (Mass<>OldMass) then begin Notify(evMassChanged); if (fMassLimit in Flags) then MassLimit; end else if not (fMassLimit in Flags) then begin MassLimit; end; except CheckHardwareError; end; end; procedure tMsThread.MassSet(m:tMass); var OldMass:tMass; LargeChange:boolean; begin if m=Mass then Exit; if WaitReady then begin MassSet0(m); SetReady; end; end; procedure tMsThread.MassXSet(m:tMass); begin prBackgroundMass:=m; FlagSet(fDoChangeMass); Inherited Resume; end; procedure tMsThread.CounterXSet(c:tCounter); begin prBackgroundCounter:=c; FlagSet(fDoChangeCounter); Inherited Resume; end; procedure tMsThread.MassStepSet(m:tMass); begin if MassStep=m then Exit; if WaitReady then begin try prMass.StepX:=m; Notify(evMassStepChanged); except end; SetReady; end; end; function tMsThread.MassStepAbsoluteMinCalculate(AMass:tMass):tMass; var c:tCounter; begin c:=Mass2Counter(AMass); Result:=(prMsSp.Counter2Mass(Succ(c))-prMsSp.Counter2Mass(c))+(1E-14); end; function tMsThread.MassStepAbsoluteMinGet:tMass; begin Result:=(prMsSp.Counter2Mass(Succ(prMsSp.Counter))-prMsSp.Mass)+(1E-14); end; function tMsThread.MassStepMinCalculate(AMass,AStep:tMass):tMass; var dm:tMass; begin Result:=AStep*StepMultiplicator; if Result=0 then Exit; dm:=MassStepAbsoluteMinCalculate(AMass); if Result<0 then begin dm:=-dm; if Result>dm then Result:=dm; end else begin if Resultdm then Result:=dm; end else begin dm:=(prMsSp.Counter2Mass(Succ(c))-dm); if ResultMass0 then begin prMsSp.MassCalibration(OldMC); OldM0:=M0; prMsSp.MassCalibrationSetCurrentMass(m); if OldM0<>M0 then Notify(evMassCalibrationChanged); Notify(evMassChanged); if DoShiftData then begin DataMassShift(Mass-Mass0); DataMassCalibrationChange(OldMC); end; end; except end; MsSpEndWrite; DataEndWrite; SetReady; end; end; procedure tMsThread.RefineCounter(c:integer); var Mass0:double; OldMC:tMassCalibration; begin if WaitReady then begin DataBeginWrite; MsSpBeginWrite; try if c<>Counter then begin prMsSp.MassCalibration(OldMC); Mass0:=Mass; prMsSp.MassCalibrationSetCurrentCounter(c); Notify(evMassCalibrationChanged); Notify(evMassChanged); if not (mstDoNotShiftDataOnRefineMass in Switches) then begin DataMassShift(Mass-Mass0); DataMassCalibrationChange(OldMC); end; end; except end; MsSpEndWrite; DataEndWrite; SetReady; end; end; procedure tMsThread.DataMassShift(dM:tMass); var i:tSeries; s:tMSSeries; begin DataBeginWrite; try prSignalsV.Mass:=prSignalsV.Mass+dM; for i:=Low(i) to High(i) do begin s:=prDataSeries[i]; if Assigned(s) then s.ShiftXValues(dM); end; finally DataEndWrite; end; end; function tMsThread.IsError:boolean; begin Result:=Error<>ecOK; end; function tMsThread.IsNoError:boolean; begin Result:=Error=ecOK; end; function tMsThread.WaitForLimitedTime(dT:cardinal):TWaitResult; begin // FlagSet(fPauseMeasure); Result:=Ready.WaitForLimitedTime(dT); end; procedure tMsThread.FlagSet(f:tFlag); begin Include(prFlags,f); end; procedure tMsThread.FlagClear(f:tFlag); begin Exclude(prFlags,f); end; function tMsThread.PNCCalibrationGet(cl:tChannel; par:tPNCCalibrationParameter):double; var K,U0:double; begin if par=pncZeroLevel then begin Result:=prMsSp.SignalZeroLevelGet(tChannel(cl)) end else begin if cl in [PNC1..PNC5] then begin prMsSp.ctrlCVF.CalibrationDataGetV(Succ(Ord(cl)),K,U0); end else if cl=PNC_SEM then begin prMsSp.ctrlCVF.CalibrationDataGetV(9,K,U0); end else begin K:=0; U0:=0; end; case par of pncK: Result:=K; pncU0: Result:=U0; else Result:=0; end; end; end; procedure tMsThread.PNCCalibrationSet(cl:tChannel; par:tPNCCalibrationParameter; Value:double); var K,U0:double; begin if par=pncZeroLevel then begin if prMsSp.SignalZeroLevelGet(cl)=Value then Exit; prMsSp.SignalZeroLevelSet(cl,Value); end else begin if cl in [PNC1..PNC5] then begin prMsSp.ctrlCVF.CalibrationDataGetV(Succ(Ord(cl)),K,U0); end else if cl=PNC_SEM then begin prMsSp.ctrlCVF.CalibrationDataGetV(9,K,U0); end else begin K:=0; U0:=0; end; case par of pncK: begin if K=Value then Exit; K:=Value; end; pncU0: begin if U0=Value then Exit; U0:=Value; end; else ; end; if cl in [PNC1..PNC5] then begin prMsSp.ctrlCVF.CalibrationDataSetV(Succ(Ord(cl)),K,U0); end else if cl=PNC_SEM then begin prMsSp.ctrlCVF.CalibrationDataSetV(9,K,U0); end; NotifyEx(evPNCCalibrationChanged,Ord(cl),Ord(par)); end; end; procedure tMsThread.PNCCalibrate; begin if WaitReady then begin try prMsSp.exCalibrate; CheckHardwareError; NotifyEx(evPNCCalibrationChanged,$FF,Ord(pncAll)); except end; SetReady; end; end; procedure tMsThread.PNCCalibrateX; begin FlagSet(fDoPNCCalibration); Inherited Resume; end; procedure tMsThread.PNCCalibrateEx(ChannelSet:tChannels); begin if WaitReady then begin try FlagClear(fDoPNCCalibrationEx); prMsSp.exCalibrateEx(ChannelSet); CheckHardwareError; NotifyEx(evPNCCalibrationChanged,Byte(ChannelSet),Ord(pncAll)); except end; SetReady; end; end; procedure tMsThread.PNCCalibrateExX(ChannelSet:tChannels); begin prPNCCalibrateChannelSet:=ChannelSet; if ChannelSet<>[] then begin FlagSet(fDoPNCCalibrationEx); FlagSet(fDoPNCCalibration); Inherited Resume; end else begin FlagClear(fDoPNCCalibration); FlagClear(fDoPNCCalibrationEx); end; end; function tMsThread.MeasureStateGet:tMeasureState; begin if not IsNoError then begin Result:=msErrorStopped; end else if (fStopMeasure in Flags) then begin Result:=msStopped; end else if (not MeasureAvailableEx) then begin Result:=msNotAvailableStopped; end else if Suspended then begin Result:=msSuspended; end else if MassStepMin=0 then begin Result:=msPaused; end else begin Result:=msScroll; end; (* end else if (fSinglePass in Flags) then begin Result:=msSinglePass; end else if (fCumulativeMeasure in Flags) then begin Result:=msRepeatedlyWithDataAccumulation; end else begin Result:=msRepeatedlyWithDataReplace; { end else begin Result:=msUnknown;} end;*) end; function tMsThread.ScrollDirectionGet:tScrollDirection; var m:tMass; begin m:=MassStepMin; if m=0 then begin Result:=sdNoScroll; end else if m>0 then begin Result:=sdUp; end else begin Result:=sdDown; end; end; function tMsThread.RegistryOpenRootKey(ForWrite:boolean):boolean; begin if ForWrite then begin Registry.CloseKey; Registry.Access:=KEY_WRITE; Result:=Registry.OpenKey(cRegistryKeyRoot,TRUE) end else Result:=Registry.OpenKeyReadOnly(cRegistryKeyRoot); end; function tMsThread.RegistryOpenKey(const SubKey:string; ForWrite:boolean):boolean; begin if ForWrite then begin Registry.CloseKey; Registry.Access:=KEY_WRITE; Result:=Registry.OpenKey(cRegistryKeyRoot+'\'+SubKey,TRUE) end else Result:=Registry.OpenKeyReadOnly(cRegistryKeyRoot+'\'+SubKey); end; procedure tMsThread.ConfigSave; var ds:TSeries; begin if not Assigned(Self) then Exit; if RegistryOpenRootKey(TRUE) then begin try Registry.WriteInteger('InitialCapacity',InitialCapacity); except end; try Registry.WriteBinaryData('Switches',prSwitches,SizeOf(prSwitches)); except end; try Registry.WriteBinaryData('ActiveSeries',prActiveSeries,SizeOf(prActiveSeries)); except end; try Registry.WriteFloat('MassStep',prMass.StepX); except end; try Registry.WriteFloat('AutoMassScaleSyncWithIMChValue',prAutoMassScaleSyncWithIMChValue); except end; try Registry.WriteBinaryData('SourceAutoCloseTime', prSourcesData.AutoCloseTimes, SizeOF(prSourcesData.AutoCloseTimes)); except end; try Registry.WriteBinaryData('Signals', prSignalsV, SizeOF(prSignalsV)); except end; if Registry.OpenKey('DataSeries',TRUE) then for ds:=Low(ds) to High(ds) do begin If Assigned(prDataSeries[ds]) then try prDataSeries[ds].SaveParameters(Registry); except end; end; if RegistryOpenKey('FileFormat',TRUE) then try FileFormat.WriteToRegistry(Registry); except end; if prRanges.Changed then begin if RegistryOpenKey('Ranges',TRUE) then try prRanges.WriteToRegistry(Registry); except end; end; end; Registry.CloseKey; end; procedure tMsThread.ConfigRead; begin if not Assigned(Self) then Exit; try if RegistryOpenRootKey(FALSE) then begin try InitialCapacity:=Registry.ReadInteger('InitialCapacity'); except end; try prAutoMassScaleSyncWithIMChValue:=Registry.ReadFloat('AutoMassScaleSyncWithIMChValue'); except end; try Registry.ReadBinaryData('Switches',prSwitches,SizeOf(prSwitches)); prSwitches:=prSwitches*cMSTSwitches-[mstCalculateStatisticalData]; except end; try Registry.ReadBinaryData('ActiveSeries',prActiveSeries,SizeOf(prActiveSeries)); prActiveSeries:=prActiveSeries*[Low(tSeries)..High(tSeries)]; except end; try prMass.StepX:=Registry.ReadFloat('MassStep'); except end; try Registry.ReadBinaryData('SourceAutoCloseTime', prSourcesData.AutoCloseTimes, SizeOF(prSourcesData.AutoCloseTimes)); except end; try Registry.ReadBinaryData('Signals', prSignalsV, SizeOF(prSignalsV)); except end; if RegistryOpenKey('FileFormat',FALSE) then try FileFormat.ReadFromRegistry(Registry); except end; if RegistryOpenKey('Ranges',FALSE) then try prRanges.ReadFromRegistry(Registry); prRanges.Changed:=FALSE; except end; end; except end; Registry.CloseKey; end; procedure tMsThread.ConfigRestore; var v:tSource; begin if not Assigned(Self) then Exit; for v:=Low(v) to High(v) do begin ValveAutoCloseTimes[v]:=prSourcesData.AutoCloseTimes[v]; end; RangesProcessStartStop; end; function tMsThread.IMCh_MassGet:tMass; begin try MsSpBeginRead; try Result:=prMsSp.exIMCh_Mass; CheckHardwareError; finally MsSpEndRead; end; except Result:=0; ErrorSet(ecUnexpectedError); end; end; function tMsThread.DataSeriesGet(s:tSeries):tMSSeries; begin DataBeginRead; Result:=prDataSeries[s]; DataEndRead; end; function tMsThread.DataSeriesCount:integer; begin Result:=Ord(High(tSeries)); end; function tMsThread.AssignedDataSeriesGet:tSeriesSet; var s:tSeries; begin Result:=prAssignedDataSeries; if Result<>[] then Exit; DataBeginRead; try for s:=Low(s) to High(s) do if Assigned(prDataSeries[s]) then Include(Result,s); finally DataEndRead; end; end; function tMsThread.DataPointsCountGet:cardinal; begin Result:=DataPointCount([Low(TSeries)..High(TSeries)]); end; function tMsThread.DataPointCount(ASeriesSet:TSeriesSet):integer; var s:tSeries; i:integer; begin if (fDifferentPointsNumberInSeries in Flags) then begin Result:=0; DataBeginRead; try for s:=Low(s) to High(s) do if s in ASeriesSet then begin i:=prDataSeries[s].Count; if i>Result then Result:=i; end; finally DataEndRead; end; end else begin DataBeginRead; try Result:=prDataSeries[sIntegrationTime].Count; finally DataEndRead; end; end; end; procedure tMsThread.DataDeleteXRangeByValues(ASeriesSet:TSeriesSet; AMin,AMax:tMAss); var s:tSeries; pc:pointer; ms:tMSSeries; acv:boolean; begin ASeriesSet:=ASeriesSet*AssignedDataSeries; DataBeginWrite; try for s:=Low(s) to High(s) do if s in ASeriesSet then begin ms:=prDataSeries[s]; acv:=ms.Active; ms.Active:=FALSE; // pc:=prDataSeries[s].ParentChart; // prDataSeries[s].ParentChart:=NIL; prDataSeries[s].DeleteXRAngeByValues(AMin,AMax); // prDataSeries[s].ParentChart:=pc; ms.Active:=acv; end; finally DataEndWrite; end; end; procedure tMsThread.DataDeleteAllXRangeByValues(AMin,AMax:tMAss); begin DataDeleteXRangeByValues([Low(TSeries)..High(TSeries)],AMin,AMax); end; procedure tMsThread.MassZeroSetByHardware; begin if WaitReady then begin try MsSpBeginWrite; try Notify(evMassAboutChangeForLargeValue); prMsSp.ctrlRoll.exScrollToAbsoluteLowLimit; CheckHardwareError; Notify(evMassAboutChangeForLargeValueEnd); finally MsSpEndWrite; end; if IsNoError then RefineMass(0); except ErrorSet(ecUnexpectedError); end; SetReady; end; end; procedure tMsThread.MassZeroSetByHardwareX; begin FlagSet(fDoMassZeroSetByHardware); end; procedure tMsThread.InitialCapacitySet(ic:integer); begin if ic<0 then ic:=0 else if ic>1000000 then ic:=1000000; prDefaultCapacity:=ic; Notify(evInitialCapacityChanged); end; function tMsThread.WaitForTerminate(TimeOut:cardinal):tWaitResult; begin if (TimeOut=INFINITE) and not Terminated then Terminate; Result:=prEventNotExecuted.WaitFor(TimeOut); if Result=wrSignaled then prEventNotExecuted.SetEvent; end; procedure tMsThread.DataBeginWrite; begin prDataLock.BeginWrite; end; procedure tMsThread.DataEndWrite; begin prDataLock.EndWrite; end; procedure tMsThread.DataBeginRead; begin prDataLock.BeginRead; end; procedure tMsThread.DataEndRead; begin prDataLock.EndRead; end; procedure tMsThread.MsSpBeginWrite; begin prMsSpLock.BeginWrite; end; procedure tMsThread.MsSpEndWrite; begin prMsSpLock.EndWrite; end; procedure tMsThread.MsSpBeginRead; begin prMsSpLock.BeginRead; end; procedure tMsThread.MsSpEndRead; begin prMsSpLock.EndRead; end; function tMsThread.ValveAutoCloseTimeLeftGet:cardinal; begin if ValveAutoCloseEnabled then Result:=prSourcesData.StartTime.TimeLeft else Result:=High(Result); end; function tMsThread.ValveOpenedTimeGet:cardinal; begin Result:=prSourcesData.StartTime.TimeSpendSinceStarted; end; procedure tMsThread.MainChannelSet(AChannel:tSeries); begin prMainChannel:=AChannel; Notify(evMainChannelChanged); end; procedure tMsThread.MassLimitClear; begin if Flags*[fMassLimitLo,fMassLimitUp,fMassLimit]=[] then Exit; prFlags:=prFlags-[fMassLimitLo,fMassLimitUp,fMassLimit]; Notify(evMassLimit); end; function tMsThread.MassLimitGet:boolean; begin Result:=IsON and prMsSp.ctrlRoll.LimitReached; if Result then begin if not (fMassLimit in Flags) then begin FlagSet(fMassLimit); Notify(evMassLimit); end; end else if (fMassLimit in Flags) then begin MassLimitClear; end; end; function tMsThread.MassLimitLowGet:boolean; begin Result:=MassLimit; if Result then begin Result:=prMsSp.ctrlRoll.LowLimit; if Result and not (fMassLimitLo in Flags) then begin FlagSet(fMassLimitLo); end; end; end; function tMsThread.MassLimitUpGet:boolean; begin Result:=MassLimit; if Result then begin Result:=prMsSp.ctrlRoll.UpLimit; if Result and not (fMassLimitUp in Flags) then begin FlagSet(fMassLimitUp); end; end; end; procedure WriteFormattedValue(ss:TStringStream; x:double; ATextFormat:tMsDataTextFileFormat); var xs:string; i,di:integer; begin xs:=FloatToStrF(X,ATextFormat.Format,ATextFormat.Precision,ATextFormat.Digits); if ATextFormat.NumberSeparator<>'' then xs:=xs+ATextFormat.NumberSeparator else xs:=xs+' '; i:=Length(xs); if i[] then begin s:=Low(s); while not (s in AFileFormat.Series) and (s<>High(s)) do begin s:=Succ(s); end; Include(AFileFormat.WriteMassSeries,s); end; end else begin AFileFormat.WriteMassSeries:=AFileFormat.Series; end; if (msdffCommentAtBegin in AFileFormat.Flags) then begin AFileFormat.WriteComment; if (msdffAddAutoComment in AFileFormat.Flags) then FileWriteAutoComment(AFileFormat); end; ss:=TStringStream.Create(''); if AFileFormat.TextFile.NumberSeparator<>'' then dPos:=Length(AFileFormat.TextFile.NumberSeparator) else dPos:=1; if AFileFormat.TextFile.Precision>15 then AFileFormat.TextFile.Precision:=15; DataBeginWrite; try { for s:=Low(s) to High(s) do begin if not Assigned(prDataSeries[s]) then begin Exclude(AFileFormat.Series,s); end; end;} if AFileFormat.FullMassRange then begin i0:=0; i1:=DataPointCount(AFileFormat.Series); end else begin i0:=DataSeriesFirstIndex(AFileFormat.MassMin); i1:=DataSeriesLastIndex(AFileFormat.MassMax); if (i0<0) or (i1<0) then begin i0:=0; i1:=0; end; end; for i:=i0 to Pred(i1) do begin ss.Position:=0; for s:=Low(s) to High(s) do begin if (s in AFileFormat.Series) then begin ms:=prDataSeries[s]; if (ms.Count>i) then begin if s in AFileFormat.WriteMassSeries then XY.X:=ms.XValues[i]; XY.Y:=ms.YValues[i]; end else begin if s in AFileFormat.WriteMassSeries then XY.X:=ms.XValues[Pred(ms.Count)]; XY.Y:=0; end; if s in AFileFormat.WriteMassSeries then WriteFormattedValue(ss, XY.x, AFileFormat.TextFile); WriteFormattedValue(ss, XY.Y, AFileFormat.TextFile); end; end; ss.Position:=ss.Position-dPos; ss.WriteString(cEOL); AFileFormat.WriteString(ss.DataString); end; if not (msdffCommentAtBegin in AFileFormat.Flags) then begin if (msdffAddAutoComment in AFileFormat.Flags) then FileWriteAutoComment(AFileFormat); AFileFormat.WriteComment; end; finally DataEndWrite; ss.Free; AFileFormat.Close; end; end; procedure tMsThread.FileWriteAutoComment(AFileFormat:tMsDataFileFormat); var sw:tMSSwitch; sws:tMSSwitches; d:tDevice; begin AFileFormat.WriteCommentBegin; if not (msdffCommentAtBegin in AFileFormat.Flags) then begin AFileFormat.WriteSeriesTitleLine; end else begin AFileFormat.Writeln; end; AFileFormat.WritelnCommentString('АВТОКОММЕНТАРИЙ'); AFileFormat.Writeln; AFileFormat.WritelnCommentString('Дата записи: '+DateTimeToStr(Now())); AFileFormat.WriteCommentString('Состояние масс-спектрометра: '); AFileFormat.WriteON_OFF(IsOn); AFileFormat.Writeln; AFileFormat.Writeln; AFileFormat.WritelnCommentString('ВНИМАНИЕ! Параметры могут не соответствовать состоянию прибора в момент записи спектра.'); AFileFormat.Writeln; AFileFormat.WritelnCommentString(' Время интегрирования, мс: '+IntToStr(IntegrationTime)); sws:=Switches; AFileFormat.Writeln; if IsOn and (not (sVoltmeter in sws)) then begin AFileFormat.WritelnCommentString('Показания вольтметра:'); AFileFormat.WritelnCommentString(' Ускоряющее напряжение, мкВ: '+IntToStr(Voltmeter[Acceleration])); AFileFormat.WritelnCommentString(' ВЭУ, мкВ: '+IntToStr(Voltmeter[Mutiplicator])); AFileFormat.WritelnCommentString(' Антидинатрон, мкВ: '+IntToStr(Voltmeter[AntiDinatron])); // AFileFormat.WritelnCommentString(' Линза, мкВ: '+IntToStr(Voltmeter[Lens])); end; AFileFormat.Writeln; AFileFormat.WritelnCommentString('Параметры устройств:'); for d:=Low(d) to Pred(High(d)) do begin AFileFormat.WritelnCommentString(' '+cDeviceName[d]+': '+IntToStr(Device[d])); end; AFileFormat.Writeln; AFileFormat.WritelnCommentString('Состояние переключателей: '); for sw:=sffUseEmulator to sfValvesControl do begin AFileFormat.WriteString(' '+cMSSwitchesName[sw]+': '); AFileFormat.WriteON_OFF(sw in sws); AFileFormat.Writeln; end; AFileFormat.Writeln; AFileFormat.WritelnCommentString('Состояние клапанов: '); AFileFormat.WritelnCommentString(' Открыт: '+cSourceNames[Valve]); AFileFormat.WritelnCommentString(' Время открытия: '+DateTimeToStr(ValveOpenDateTime)); AFileFormat.WritelnCommentString(' Период открытия, мс: '+IntToStr(ValveOpenedTime)); AFileFormat.Writeln; if (msdffCommentAtBegin in AFileFormat.Flags) then begin AFileFormat.WriteSeriesTitleLine; end; AFileFormat.WriteCommentEnd; end; procedure tMsThread.FileWrite; begin FlagClear(fDoFileWrite); try FileWriteFormatted(FileFormat); except Notify(evFileWriteFail); end; end; procedure tMsThread.FileWriteX; begin FlagSet(fDoFileWrite); inherited Resume; end; procedure tMsThread.DataSeriesDrawLock(Sender:tObject); begin DataBeginRead; end; procedure tMsThread.DataSeriesDrawUnLock(Sender:tObject); begin DataEndRead; end; procedure tMsThread.DataSeriesWriteDataToFile(const AFileName:string); var fs:tFileStream; begin fs:=tFileStream.Create(AFileName,fmCreate or fmShareDenyWrite); try DataSeriesWrite(fs); finally fs.Free; end; end; procedure tMsThread.DataSeriesWrite(AStream:TStream); var ds:tSeries; s:tMsSeries; begin DataBeginRead; try for ds:=Low(ds) to High(ds) do begin s:=prDataSeries[ds]; if Assigned(s) then begin s.WriteData(AStream); end; end; finally DataEndRead; end; end; procedure tMsThread.DataSeriesReadDataFromFile(const AFileName:string); var fs:tFileStream; begin if not FileExists(AFileName) then Exit; fs:=tFileStream.Create(AFileName,fmOpenRead or fmShareDenyWrite); try DataSeriesRead(fs); finally fs.Free; end; end; procedure tMsThread.DataSeriesRead(AStream:TStream); var ds:tSeries; s:tMsSeries; begin DataBeginWrite; try for ds:=Low(ds) to High(ds) do begin s:=prDataSeries[ds]; if Assigned(s) then begin s.ReadData(AStream); end; end; finally DataEndWrite; end; end; procedure tMsThread.DataSeriesWriteData; begin if DataFileName<>'' then DataSeriesWriteDataToFile(DataFileName); end; procedure tMsThread.DataSeriesReadData; begin if DataFileName<>'' then DataSeriesReadDataFromFile(DataFileName); end; function tMsThread.DataFileName:string; begin if ConfigFileName<>'' then Result:=ConfigFileName+'.dat' else Result:=''; end; procedure tMsThread.FileFormatSet(AFileFormat:tMsDataFileFormat); begin prFileFormat.Assign(AFileFormat); end; function tMsThread.RangeGet(i:integer):tRange; begin if (i>=0) and (i<=prRanges.Count) then Result:=prRanges[i] else Result:=NIL; end; function tMsThread.RangeCountGet:integer; begin Result:=prRanges.Count; end; function tMsThread.DataSeriesFirstIndex(Mass:tMass):integer; begin Result:=prDataSeries[sPNC1].FindFirstXAboveEqual(Mass); end; function tMsThread.DataSeriesLastIndex(Mass:tMass):integer; begin Result:=prDataSeries[sPNC1].FindLastXBelowEqual(Mass); end; procedure tMsThread.DoEmergencyCheck(Sender:tObject); begin if IsON then begin if prMsSp.ctrlISSB.NoError then begin prMsSp.ctrlISSB.exCurFlags(prCtrlFlags); if (fBPGI_ON in prCtrlFlags) and (prCtrlFlags*[fCatodOK, fOverload]<>[fCatodOK]) then Notify(evEmergency); end; end; end; procedure tMsThread.HandleRangesChanged(Sender:TObject; ARangeIndex:integer; AItem:tRangeItem); begin case AItem of riEnabled: NotifyEx(evRangeChanged,Ord(AItem),ARangeIndex); end; end; procedure tMsThread.HandleRangeListChanged(Sender:TObject); begin Notify(evRangeListChanged); end; function tMsThread.DataValidSeriesCount(ASeriesSet:tSeriesSet):integer; var s:tSeries; i:integer; begin Result:=0; if ASeriesSet=[] then Exit; for s:=Low(s) to High(s) do begin if s in ASeriesSet then if Assigned(prDataSeries[s]) then Inc(Result); end; end; function tMsThread.DataRangePointsCount(ASeriesSet:tSeriesSet; AMin,AMax:tMass):integer; var s:tSeries; i:integer; ms:tMsSeries; begin Result:=0; if ASeriesSet=[] then Exit; DataBeginRead; try for s:=Low(s) to High(s) do begin if s in ASeriesSet then begin ms:=prDataSeries[s]; if Assigned(ms) then try i:=ms.RangePointsCount(AMin,AMax); if i>Result then Result:=i; except end; end; end; finally DataEndRead; end; end; function tMsThread.DataArrayGet(const AMcadArray:ComplexArray; ASeriesSet:tSeriesSet; AMin,AMax:tMass):boolean; var s:tSeries; i,n:integer; ms:tMsSeries; begin Result:=ASeriesSet<>[]; if not Result then Exit; s:=Low(s); i:=0; Result:=AMcadArray.cols>=2; if not Result then Exit; DataBeginRead; try n:=AMcadArray.cols; if n>Succ(Ord(High(s))) then n:=Succ(Ord(High(s))); repeat if s in ASeriesSet then begin ms:=prDataSeries[s]; if Assigned(ms) then begin try if i=0 then begin Inc(i); ms.RangeXYPointsGet(AMcadArray.hReal[0]^,AMcadArray.hReal[1]^,AMcadArray.rows,AMin,AMax); end else begin ms.RangeYPointsGet(AMcadArray.hReal[i]^,AMcadArray.rows,AMin,AMax); end; except // ColumnClear(AMcadArray.hReal[i]^, AMcadArray.rows); end; end else begin // ColumnClear(AMcadArray.hReal[i]^, AMcadArray.rows); end; Inc(i); end; if (s<>High(s)) then s:=Succ(s) else n:=i; until (i>=n); finally DataEndRead; end; end; function tMsThread.DataPointGet(AMass:tMass; const ASignals:ComplexArray):boolean; var i:integer; rows:integer; begin if WaitReady then begin try Mass:=AMass; DataGet0; if IsNoError then begin DataStore; if ASignals.Cols>0 then begin rows:=ASignals.Rows; if rows>0 then begin ASignals.hReal[0,0]:=Signals.Mass; if rows>1 then begin if Rows>Signals.Count then i:=Signals.Count else i:=Rows; for i:=0 to Pred(Signals.Count) do begin ASignals.hReal[0,Succ(i)]:=Signals.Signals[tChannel(i)]; end; if Rows>Succ(Signals.Count) then begin ASignals.hReal[0,Succ(Signals.Count)]:=Signals.Time; end; end; end; end; end; except end; SetReady; end; end; procedure tMsThread.AutoMassScaleSyncWithIMChValueSet(AValue:tMass); begin AValue:=Abs(AValue); if AValue=AutoMassScaleSyncWithIMChValue then Exit; prAutoMassScaleSyncWithIMChValue:=AValue; Notify(evAutoMassScaleSyncWithIMChValueChanged); end; procedure tMsThread.DoAutoMassScaleSyncWithIMCh; var mIMCh:tMass; begin if (mstAutoMassScaleSyncWithIMCh in prSwitches) and (AutoMassScaleSyncWithIMChValue<>0) then begin if WaitReady then begin try mIMCh:=IMCh_Mass; if (mIMCh>0) and (Abs(mIMCh-Mass)>=AutoMassScaleSyncWithIMChValue) then begin RefineMassEx(mIMCh,FALSE); Inc(prAutoMassScaleSyncWithIMChCount); Notify(evDoAutoMassScaleSyncWithIMCh); end; except end; SetReady; end; end; end; procedure tMsThread.DataSeriesWriteDataToFileX(const AFileName:string); begin FlagClear(fDoDataFileRead); prDataFileName:=AFileName; FlagSet(fDoDataFileWrite); inherited Resume; end; procedure tMsThread.DataSeriesReadDataFromFileX(const AFileName:string); begin FlagClear(fDoDataFileWrite); prDataFileName:=AFileName; FlagSet(fDoDataFileRead); inherited Resume; end; END.