Files
open-insight/LSL2/STPROC/SCHEDULE_SERVICES.txt
2025-08-28 09:37:39 -07:00

3608 lines
151 KiB
Plaintext

Function Schedule_Services(@Service, @Params)
/***********************************************************************************************************************
This program is proprietary and is not to be used by or disclosed to others, nor is it to be copied without written
permission from SRP Computer Solutions, Inc.
Name : Schedule_Services
Description : Handler program for all module related services.
Notes : The generic parameters should contain all the necessary information to process the services. Often
this will be information like the data Record and Key ID.
Parameters :
Service [in] -- Name of the service being requested
Param1-10 [in/out] -- Additional request parameter holders
Response [out] -- Response to be sent back to the Controller (MCP) or requesting procedure
Metadata :
History : (Date, Initials, Notes)
06/01/17 dmb Original programmer. - [IREPIOI-8]
01/07/18 dmb Update the AdjustScheduleEvents service to treat 1-day events as previous events so they
will get skipped over when future events are adjusted. - [IREPIOI-14]
05/09/19 dmb Update AdjustOtherWorkOrderEvents gosub to sort AllWorkOrderEvents using the existing
sort order of AllWorkOrderKeys instead of sorting the array directly because the data
doesn't allow a predictible sort. - [IREPIOI-70]
***********************************************************************************************************************/
#pragma precomp SRP_PreCompiler
$insert APP_INSERTS
$insert SERVICE_SETUP
$insert RLIST_EQUATES
$insert REACTOR_EQUATES
$insert SCHED_DET_NG_EQUATES
$insert WO_LOG_EQUATES
$Insert WO_MAT_EQUATES
$insert WO_SCHEDULE_NG_EQUATES
$insert COMPANY_EQUATES
$insert PROD_VER_EQUATES
$insert PRS_LAYER_EQU
$insert SCHEDULE_EVENT_SUMMMARY_EQUATES
$insert EPI_PART_EQUATES
$insert PROD_SPEC_EQUATES
$insert SCHED_HIST_EQUATES
$Insert RDS_EQUATES
Equ new_exist$ To 0 ; * Reduce Mode 0
Equ next_cur$ To 1
Equ add_exist$ To 2
EQU NOTIFICATION_PERIOD$ TO 12
Declare subroutine Error_Services, Schedule_Services, Memory_Services, RList, Database_Services, Logging_Services
Declare subroutine Btree.Extract, Set_Status, Reduce, FSMsg, Messaging_Services, obj_Notes, Make.List
Declare subroutine Mona_Services
Declare function SRP_Array, Schedule_Services, Memory_Services, Database_Services, SRP_Sort_Array, Datetime
Declare function Epi_Part_Services, SRP_Math, SRP_Hash, obj_Prod_Spec, Logging_Services, Environment_Services
Declare function Work_Order_Services, RTI_CreateGUID, Reactor_Services, Schedule_Services, NextKey, SRP_Datetime
Declare function SRP_Time, Lsl_Users_Services, GetTickCount
Date = Oconv(Date(), 'D4/')
LogFileName = Date[7, 4] : '-' : Date[1, 2] : '-' : Date[4, 2] : ' Scheduler Log.csv'
Headers = 'Logging DTM' : @FM : 'Service Step' : @FM : 'Service Notes'
ColumnWidths = 20 : @FM : 30 : @FM : 150
IsProd = Environment_Services('IsProd')
If IsProd EQ True$ then
MonaResource = 'GRP_OPENINSIGHT_MES_OP_FE_SCHEDULESERVICES'
end else
MonaResource = 'GRP_OPENINSIGHT_MES_OP_FE_DEV_SCHEDULESERVICES'
end
GoToService else
Error_Services('Set', Service : ' is not a valid service request within the ' : ServiceModule : ' services module.')
end
Return Response else ''
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
// Service Parameter Options
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
Options BLOCK_GROUP = 'HARD_BLOCK', 'SOFT_BLOCK'
Options ACTIONS = 'ADD', 'STOP', 'CANCEL', 'MODIFY'
Options BOOLEAN = True$, False$
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
// Services
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
//----------------------------------------------------------------------------------------------------------------------
// GetReactors
//
// Returns an array of reactors that will be used to populate the SRP Schedule control. All reactors will be returned
// other than 0.
//----------------------------------------------------------------------------------------------------------------------
Service GetReactors(ReactorType, SusceptorSize)
ServiceKeyID := '*' : Reactortype : '*' : SusceptorSize
Reactors = Memory_Services('GetValue', ServiceKeyID)
If Reactors EQ '' then
TableName = 'REACTOR'
hReactors = Database_Services('GetTableHandle', TableName)
If Error_Services('NoError') then
Select hReactors
Done = False$
ReactorArray = ''
MetaArray = ''
Loop
Readnext ReactorNo ELSE Done = True$
Until Done
If ReactorNo GT 0 then
* ReactorRec = Database_Services('ReadDataRow', TableName, ReactorNo)
ReactorRec = Schedule_Services('GetReactorDetails', ReactorNo)
If Error_Services('NoError') then
AcceptReactor = True$ ; // Assume accepted for now.
Type = ReactorRec<REACTOR_REACT_TYPE$>
Size = ReactorRec<REACTOR_SUSC_POCKET_SIZE$>
SecondChamber = ReactorRec<REACTOR_SECOND_CHAMBER$>
If (ReactorType NE '') AND (Type NE ReactorType) then
If (ReactorType[1, 3] EQ 'ASM' OR ReactorType EQ 'EPS') AND (Type[1, 3] EQ 'ASM' OR Type EQ 'EPS') else
// The reactor type for this work order does not match the required reactor type.
AcceptReactor = False$
end
end
If AcceptReactor AND SusceptorSize NE '' then
If Size NE SusceptorSize then
// The susceptor size for this work order does not match the required susceptor size.
AcceptReactor = False$
end
end
If (Type _EQC 'EPP' OR Type _EQC 'EpiPro') AND SecondChamber EQ '' then
AcceptReactor = False$
end
If AcceptReactor then
Begin Case
Case Type _eqc 'ASM+' ; Type = 'ASM+'
Case Type _eqc 'EPP' ; Type = 'EpiPro'
Case Type _eqc 'EPS' ; Type = 'ASM'
Case Type _eqc 'HTR' ; Type = 'HTR'
Case Type _eqc 'GAN' ; Type = 'GaN'
Case Type _eqc '' ; Type = '***'
End Case
Assignment = ReactorRec<REACTOR_REACT_ASSIGNMENT$>
Location = ReactorRec<REACTOR_LOCATIONX$>
Begin Case
Case Assignment EQ 'M'
// Dedicated
Background = 'Pink'
Case Assignment EQ 'C'
// Non-Dedicated
Background = 'LightGreen'
Case Assignment EQ 'G'
// GaN
Background = 'SkyBlue'
Case Assignment EQ 'O'
// Out-of-Service
Background = 'Silver'
Case Otherwise$
// Default to Dedicated
Background = 'Pink'
End Case
EntityName = ReactorNo
If SecondChamber NE '' then EntityName := '/' : SecondChamber
ReactorArray := 'ENT' : @VM : 1 : @VM : ReactorNo : @VM : EntityName : @VM : Type : CRLF$ : Size : @VM : 'Black' : @VM : Background : @FM
MetaArray := ReactorNo : @VM : Oconv(Assignment, '[REACT_ASSIGN_CONV]') : @VM : Location : @FM
end
end
end
Repeat
ReactorArray[-1, 1] = ''
ReactorArray = SRP_Array('SortRows', ReactorArray, 'AR3', 'LIST', @FM, @VM)
MetaArray[-1, 1] = ''
MetaArray = SRP_Array('SortRows', MetaArray, 'AR1', 'LIST', @FM, @VM)
Reactors = ReactorArray : @RM : MetaArray
Memory_Services('SetValue', ServiceKeyID, Reactors)
end
end
Response = Reactors
end service
//----------------------------------------------------------------------------------------------------------------------
// GetReactorDetails
//
//----------------------------------------------------------------------------------------------------------------------
Service GetReactorDetails(ReactorNo)
ServiceKeyID := '*' : ReactorNo
ReactorDetails = Memory_Services('GetValue', ServiceKeyID)
If ReactorDetails EQ '' then
If (ReactorNo NE '') AND (ReactorNo NE 0) then
TableName = 'REACTOR'
ReactorDetails = Database_Services('ReadDataRow', TableName, ReactorNo)
If Error_Services('NoError') then
Memory_Services('SetValue', ServiceKeyID, ReactorDetails)
end
end else
Error_Services('Add', 'ReactorNo argument was missing from the ' : Service : ' service.')
end
end
Response = ReactorDetails
end service
//----------------------------------------------------------------------------------------------------------------------
// GetReactorAssignments
//
// Returns an array of key pair "Reactor:Assignment" that will be used to populate the SRP Schedule control. All reactors will be returned
// other than 0.
//----------------------------------------------------------------------------------------------------------------------
Service GetReactorAssignments()
ReactorAssignments = Memory_Services('GetValue', ServiceKeyID)
ReactorAssignments = ''
If ReactorAssignments EQ '' then
TableName = 'REACTOR'
hReactors = Database_Services('GetTableHandle', TableName)
If Error_Services('NoError') then
Select hReactors
Done = False$
LOOP
READNEXT ReactorNo ELSE Done = True$
UNTIL Done
If ReactorNo GT 0 then
ReactorRec = Database_Services('ReadDataRow', TableName, ReactorNo)
If Error_Services('NoError') then
Assignment = ReactorRec<REACTOR_REACT_ASSIGNMENT$>
ReactorAssignments := ReactorNo : @VM : Assignment : @FM
end
end
REPEAT
ReactorAssignments[-1, 1] = ''
ReactorAssignments = SRP_Array('SortRows', ReactorAssignments, 'AR1', 'LIST', @FM, @VM)
Memory_Services('SetValue', ServiceKeyID, ReactorAssignments)
end
end
Response = ReactorAssignments
end service
//----------------------------------------------------------------------------------------------------------------------
// GetTimeBlockList
//
// Returns an array of reactors assignments that will be used to populate the SRP Schedule control. All reactors will be
// returned other than 0.
//----------------------------------------------------------------------------------------------------------------------
Service GetTimeBlockList(StartDate, EndDate)
TimeBlockList = ''
ReactorAssignments = Schedule_Services('GetReactorAssignments')
For Each ReactorAssignment in ReactorAssignments using @FM
ReactorNo = ReactorAssignment<0, 1>
Assignment = ReactorAssignment<0, 2>
For Date = StartDate to EndDate
StartDTM = Date : '.0'
EndDTM = (Date + 1) : '.0'
Begin Case
Case Assignment EQ 'M'
// Dedicated
TimeBlockList := ReactorNo : @VM : ReactorNo : '*' : Date : @VM : StartDTM : @VM : EndDTM : @VM : 'Pink' : @FM
Case Assignment EQ 'C'
// Non-Dedicated
TimeBlockList := ReactorNo : @VM : ReactorNo : '*' : Date : @VM : StartDTM : @VM : EndDTM : @VM : 'LightGreen' : @FM
Case Assignment EQ 'G'
// GaN
TimeBlockList := ReactorNo : @VM : ReactorNo : '*' : Date : @VM : StartDTM : @VM : EndDTM : @VM : 'SkyBlue' : @FM
Case Assignment EQ 'O'
// Out-of-Service
TimeBlockList := ReactorNo : @VM : ReactorNo : '*' : Date : @VM : StartDTM : @VM : EndDTM : @VM : 'Silver' : @FM
Case Otherwise$
// Default to Dedicated
TimeBlockList := ReactorNo : @VM : ReactorNo : '*' : Date : @VM : StartDTM : @VM : EndDTM : @VM : 'Pink' : @FM
End Case
Next Date
Next ReactorAssignment
TimeBlockList[-1, 1] = ''
Response = TimeBlockList
end service
//----------------------------------------------------------------------------------------------------------------------
// GetScheduleEvents
//
// Returns an array of schedule details that will be used to populate the SRP Schedule control.
//----------------------------------------------------------------------------------------------------------------------
Service GetScheduleEvents(StartDate, EndDate, ReactorNo, WorkOrderNo, BlockOut)
SRP_Stopwatch('Reset')
SRP_Stopwatch('Start', Service)
ScheduleEvents = ''
If Not(Num(StartDate)) then StartDate = Iconv(StartDate, 'D')
If Not(Num(EndDate)) then EndDate = Iconv(EndDate, 'D')
If BlockOut NE True$ then BlockOut = False$
If (StartDate NE '') then
SchedDetKeyIDs = Schedule_Services('GetScheduleDetailKeys', StartDate, EndDate, ReactorNo, WorkOrderNo, BlockOut)
If SchedDetKeyIDs NE '' then
SchedDetKeyIDs = SRP_Array('SortRows', SchedDetKeyIDs, 'AR1' : @FM : 'AR2' : @FM : 'AR3', 'LIST', @FM, '*')
If Error_Services('NoError') then
PrevWorkOrderNo = ''
Lum = 80
For Each SchedDetKeyID in SchedDetKeyIDs using @FM setting fPos
SchedDetRec = Database_Services('ReadDataRow', 'SCHED_DET_NG', SchedDetKeyID)
ReactorNo = SchedDetRec<SCHED_DET_NG.REACT_NO$>
WorkOrderNo = SchedDetRec<SCHED_DET_NG.WO_NO$>
StartDTM = SchedDetRec<SCHED_DET_NG.START_DTM$>
StopDTM = SchedDetRec<SCHED_DET_NG.STOP_DTM$>
ScheduleEventSummary = Schedule_Services('GetScheduleEventSummary', SchedDetKeyID, False$)
ScheduleEvents := ReactorNo : @VM
ScheduleEvents := SchedDetKeyID : @VM ; //WorkOrderNo : '*' : ScheduleEventSummary<SES_EPIPARTNO$> : '*' : ReactorNo : '*' : ScheduleDate : @VM
ScheduleEvents := ScheduleEventSummary<SES_STARTDTM$> : @VM
ScheduleEvents := ScheduleEventSummary<SES_ENDDTM$> : @VM
ScheduleEvents := ScheduleEventSummary<SES_BACKCOLOR$> : ' L=' : Lum : @VM
ScheduleEvents := ScheduleEventSummary<SES_FORECOLOR$> : @VM
ScheduleEvents := ScheduleEventSummary<SES_TITLE$> : @VM
ScheduleEvents := ScheduleEventSummary<SES_DESCRIPTION$> : @VM
ScheduleEvents := @VM : @VM : @VM : @VM : @VM : @VM : @VM : @VM : @VM : @VM
ScheduleEvents := ScheduleEventSummary<SES_HOTLOTFLAG$> : @SVM : ScheduleEventSummary<SES_NOTEFLAG$> : @SVM : ScheduleEventSummary<SES_WOCLOSEDFLAG$> : @SVM : ScheduleEventSummary<SES_CURRENTFLAG$>
ScheduleEvents := @FM
Next SchedDetKeyID
ScheduleEvents[-1, 1] = ''
ScheduleEvents = SRP_Array('SortRows', ScheduleEvents, 'AR1' : @FM : 'AR3' : @FM : 'AR4' : @FM : 'AL7', 'LIST', @FM, @VM)
end
end
end else
Error_Services('Add', 'StartDate argument was missing from the ' : Service : ' service.')
end
Response = ScheduleEvents
SRP_Stopwatch('Stop', Service)
end service
Service GetScheduleEvent(SchedDetKeyID)
SchedEvent = ''
If SchedDetKeyID NE '' then
Lum = 80
SchedDetRec = Database_Services('ReadDataRow', 'SCHED_DET_NG', SchedDetKeyID)
ReactorNo = SchedDetRec<SCHED_DET_NG.REACT_NO$>
WorkOrderNo = SchedDetRec<SCHED_DET_NG.WO_NO$>
StartDTM = SchedDetRec<SCHED_DET_NG.START_DTM$>
StopDTM = SchedDetRec<SCHED_DET_NG.STOP_DTM$>
ScheduleEventSummary = Schedule_Services('GetScheduleEventSummary', SchedDetKeyID, False$)
SchedEvent = ReactorNo : @VM
SchedEvent := SchedDetKeyID : @VM
SchedEvent := ScheduleEventSummary<SES_STARTDTM$> : @VM
SchedEvent := ScheduleEventSummary<SES_ENDDTM$> : @VM
SchedEvent := ScheduleEventSummary<SES_BACKCOLOR$> : ' L=' : Lum : @VM
SchedEvent := ScheduleEventSummary<SES_FORECOLOR$> : @VM
SchedEvent := ScheduleEventSummary<SES_TITLE$> : @VM
SchedEvent := ScheduleEventSummary<SES_DESCRIPTION$> : @VM
SchedEvent := @VM : @VM : @VM : @VM : @VM : @VM : @VM : @VM : @VM : @VM
SchedEvent := ScheduleEventSummary<SES_HOTLOTFLAG$> : @SVM : ScheduleEventSummary<SES_NOTEFLAG$> : @SVM : ScheduleEventSummary<SES_WOCLOSEDFLAG$> : @SVM : ScheduleEventSummary<SES_CURRENTFLAG$>
end
Response = SchedEvent
end service
//----------------------------------------------------------------------------------------------------------------------
// GetScheduleEventSummary
//
// Returns an @FM array of event summary data related to the indicated reactor and work order.
//----------------------------------------------------------------------------------------------------------------------
Service GetScheduleEventSummary(SchedDetKeyID, IncludeWaferDetails)
If IncludeWaferDetails NE True$ then IncludeWaferDetails = False$
ScheduleEventSummary = ''
If SchedDetKeyID NE '' then
SchedDetRec = Database_Services('ReadDataRow', 'SCHED_DET_NG', SchedDetKeyID)
ReactorNo = SchedDetRec<SCHED_DET_NG.REACT_NO$>
WorkOrderNo = SchedDetRec<SCHED_DET_NG.WO_NO$>
StartDTM = SchedDetRec<SCHED_DET_NG.START_DTM$>
StopDTM = SchedDetRec<SCHED_DET_NG.STOP_DTM$>
Desc = SchedDetRec<SCHED_DET_NG.DESC$>
Modified = SchedDetRec<SCHED_DET_NG.MODIFIED$>
BlockOut = SchedDetRec<SCHED_DET_NG.BLOCK_OUT$>
BlockOutType = SchedDetRec<SCHED_DET_NG.BLOCK_OUT_TYPE$>
BackColor = SchedDetRec<SCHED_DET_NG.BACKCOLOR$>
ForeColor = SchedDetRec<SCHED_DET_NG.FORECOLOR$>
EventWfrQty = SchedDetRec<SCHED_DET_NG.EVENT_TOTAL_WFRS$>
ReactorDetails = Schedule_Services('GetReactorDetails', ReactorNo)
Type = ReactorDetails<REACTOR_REACT_TYPE$>
Size = ReactorDetails<REACTOR_SUSC_POCKET_SIZE$>
WOLogRow = Memory_Services('GetValue', 'WOLogRow' : '*' : ReactorNo : '*' : WorkOrderNo, True$, 5)
If WOLogRow EQ '' then
WOLogRow = Work_Order_Services('GetWorkOrderLogDetail', WorkOrderNo)
Memory_Services('SetValue', 'WOLogRow' : '*' : ReactorNo : '*' : WorkOrderNo, WOLogRow)
end
CustNo = WOLogRow<WO_LOG_CUST_NO$>
EpiPartNo = WOLogRow<WO_LOG_EPI_PART_NO$>
ProdVerNo = WOLogRow<WO_LOG_PROD_VER_NO$>
TotalWafers = SchedDetRec<SCHED_DET_NG.EVENT_TOTAL_WFRS$>
WOClosedFlag = SchedDetRec<SCHED_DET_NG.EVENT_COMP$>
HotLotFlag = WOLogRow<WO_LOG_HOT_FLAG$>
If Memory_Services('IsValueExpired', 'CompanyRow' : '*' : ReactorNo : '*' : WorkOrderNo, 60, True$) then
CompanyRow = Database_Services('ReadDataRow', 'COMPANY', CustNo)
Memory_Services('SetValue', 'CompanyRow' : '*' : ReactorNo : '*' : WorkOrderNo, CompanyRow)
end else
CompanyRow = Memory_Services('GetValue', 'CompanyRow' : '*' : ReactorNo : '*' : WorkOrderNo)
end
CustName = CompanyRow<COMPANY_ABBREV$>
Begin Case
Case CustName _EQC 'International Rectifier' ; CustNameShort = 'IR'
Case CustName _EQC 'IRNewport' ; CustNameShort = 'Newport'
Case CustName _EQC 'IFX (Kulim)' ; CustNameShort = 'Kulim'
Case CustName _EQC 'IFX Austria AG' ; CustNameShort = 'Austria'
Case CustName _EQC 'Tower Semiconductor' ; CustNameShort = 'Tower'
Case Otherwise$ ; CustNameShort = CustName
End Case
TableName = 'SCHED_DET_NG'
NoteFlag = ''
If BlockOut then
Title = 'Block Out' : ' (' : BlockOutType : ')'
end else
Title = WorkOrderNo : ' (' : CustNameShort : ' ' : EpiPartNo : ')'
end
HardBlock = ( (BlockOutType EQ 'Out of Service (OOS)') or (BlockOutType EQ 'Down No Material (DNM)') or (BlockOutType EQ 'Factory Down Time') )
Begin Case
Case BackColor NE ''
// Set in the schedule detail record. Use current color.
Case HotLotFlag
BackColor = 'LightCoral'
Case WOClosedFlag
BackColor = 'LightGray'
Case HardBlock
BackColor = 'LightGoldenRodYellow'
Case BlockOut
BackColor = 'Plum'
Case Otherwise$
BackColor = 'LightSteelBlue'
End Case
Begin Case
Case ForeColor NE ''
// Set in the schedule detail record. Use current color.
Case NoteFlag AND HotLotFlag
ForeColor = 'Yellow'
Case NoteFlag AND BlockOut
ForeColor = 'Indigo'
Case NoteFlag
ForeColor = 'Red'
Case Otherwise$
ForeColor = 'Black'
End Case
If IncludeWaferDetails EQ True$ then
Remaining = Schedule_Services('GetEventRemainingWfrs', SchedDetKeyID)
If TotalWafers GT 0 then
Complete = (TotalWafers - Remaining) / TotalWafers * 100
end else
Complete = 0
end
TotalDays = SRP_Math('ROUND', StopDTM - StartDTM, 1)
ProdVerRow = Memory_Services('GetValue', 'ProdVerRow' : '*' : ReactorNo : '*' : WorkOrderNo, True$, 5)
If ProdVerRow EQ '' then
ProdVerRow = Database_Services('ReadDataRow', 'PROD_VER', ProdVerNo)
Memory_Services('SetValue', 'ProdVerRow' : '*' : ReactorNo : '*' : WorkOrderNo, ProdVerRow)
end
PSN = ProdVerRow<PROD_VER_PROC_STEP_PSN$>
LayerSpecs = obj_Prod_Spec('GetLayerProp',PSN:@RM:@RM:1) ;* Returns specs for all layers in internal format
LayerSpec = FIELD(LayerSpecs,@RM,1) ;* Take the first Layer
LayerSet = FIELD(LayerSpec,@FM,1) ;* Not used here but shown for clarity
LayerSpec = FIELD(LayerSpec,@FM,2,99) ;* LayerSpec without the LayerSet
Recipe = LayerSpec<PRS_LAYER_RECIPE$>
end else
Remaining = ''
TotalWafers = ''
Complete = ''
TotalDays = ''
PSN = ''
Recipe = ''
end
StartDate = Field(StartDTM, '.', 1)
EndDate = Field(StopDTM, '.', 1)
ScheduleEventSummary := Type : @FM
ScheduleEventSummary := Size : @FM
ScheduleEventSummary := CustName : @FM
ScheduleEventSummary := CustNameShort : @FM
ScheduleEventSummary := EpiPartNo : @FM
ScheduleEventSummary := Oconv(StartDate, 'D4/') : @FM
ScheduleEventSummary := Oconv(EndDate, 'D4/') : @FM
ScheduleEventSummary := TotalDays : @FM
ScheduleEventSummary := EventWfrQty : @FM
ScheduleEventSummary := Remaining : @FM
ScheduleEventSummary := Oconv(Complete, 'MD0[ %]S') : @FM
ScheduleEventSummary := WOClosedFlag : @FM
ScheduleEventSummary := HotLotFlag : @FM
ScheduleEventSummary := '' : @FM ; //CurrentFlag : @FM
ScheduleEventSummary := NoteFlag : @FM
ScheduleEventSummary := StartDTM : @FM
ScheduleEventSummary := StopDTM : @FM
ScheduleEventSummary := Title : @FM
ScheduleEventSummary := Desc : @FM
ScheduleEventSummary := BackColor : @FM
ScheduleEventSummary := ForeColor : @FM
ScheduleEventSummary := PSN : @FM
ScheduleEventSummary := Recipe : @FM
ScheduleEventSummary := '' : @FM ; // Day Length Code ?
ScheduleEventSummary := BlockOut : @FM
ScheduleEventSummary := BlockOutType
end else
Error_Services('Add', 'ReactorNo or WorkOrderNo argument was missing from the ' : Service : ' service.')
end
Response = ScheduleEventSummary
end service
//----------------------------------------------------------------------------------------------------------------------
// GetScheduleDetailKeys
//
// Returns an array of schedule detail KeyIDs based on the arguments provided.
//----------------------------------------------------------------------------------------------------------------------
Service GetScheduleDetailKeys(StartDate, EndDate, ReactorNo, WorkOrderNo, BlockOut, InProcess=BOOLEAN)
ScheduleDetailKeys = ''
If Not(Num(StartDate)) then StartDate = Iconv(StartDate, 'DT')
If Not(Num(EndDate)) then EndDate = Iconv(EndDate, 'DT')
If BlockOut NE True$ then BlockOut = False$
If ( (ReactorNo NE '') or (StartDate NE '') ) then
CursorVar = ''
SortList = 'REACT_NO' : @FM : 'START_DTM'
Flag = ''
Done = False$
ReduceScript = ''
If ReactorNo NE '' then ReduceScript := 'WITH {REACT_NO} EQ "':ReactorNo:'" AND '
If StartDate NE '' then ReduceScript := "WITH {START_DTM} GT ":Quote(OConv(StartDate, 'DT/^3H')):' AND '
// Additional filters can be added here
ReduceScript[-1, -5] = ''
GoSub ClearCursors
Reduce(ReduceScript, SortList, 1, 'SCHED_DET_NG', CursorVar, Flag)
If Flag then
Select 'SCHED_DET_NG' By SortList using CursorVar then
Open 'SCHED_DET_NG' to hTable then
Loop
ReadNext Key using CursorVar by AT Else Done = True$
Until Done
Locate Key in ScheduleDetailKeys using @FM setting fPos else
// This ensures the list is unique. Duplicates can be found by the Select statement
// if the index on REACT_NO or START_DTM is compromised.
ScheduleDetailKeys<-1> = Key
end
Repeat
end
end
end
GoSub ClearCursors
If InProcess then
// Perform a second select to ensure currently running events get selected.
Flag = ''
Done = False$
ReduceScript = 'WITH {UNPROCESSED_CASS} NE ""'
// Additional filters can be added here
GoSub ClearCursors
Reduce(ReduceScript, SortList, 1, 'SCHED_DET_NG', CursorVar, Flag)
If Flag then
Select 'SCHED_DET_NG' By SortList using CursorVar then
Open 'SCHED_DET_NG' to hTable then
Loop
ReadNext Key using CursorVar by AT Else Done = True$
Until Done
Locate Key in ScheduleDetailKeys using @FM setting fPos else
// This ensures the list is unique. Duplicates can be found by the Select statement
// if the index on REACT_NO or START_DTM is compromised.
ScheduleDetailKeys<-1> = Key
end
Repeat
end
end
end
end
end else
Error_Services('Add', 'StartDate argument was missing from the ' : Service : ' service.')
end
Response = ScheduleDetailKeys
end service
Service CancelScheduleEvent(SchedDetKeyID, ShiftEvents)
RemSchedDetKeys = ''
AddSchedDetKeys = ''
If SchedDetKeyID NE '' then
SchedDetRec = Database_Services('ReadDataRow', 'SCHED_DET_NG', SchedDetKeyID)
ReactNo = SchedDetRec<SCHED_DET_NG.REACT_NO$>
ReactRec = Database_Services('ReadDataRow', 'REACTOR', ReactNo)
ReactSchedEvents = ReactRec<REACTOR_SCHED_EVENTS$>
Locate SchedDetKeyID in ReactSchedEvents using @VM setting vPos then
PrevSchedKeyID = ReactSchedEvents<0, vPos - 1>
If PrevSchedKeyID NE '' then
// Mark this event as not complete.
PrevSchedDetRec = Database_Services('ReadDataRow', 'SCHED_DET_NG', PrevSchedKeyID)
PrevSchedDetRec<SCHED_DET_NG.EVENT_COMP$> = 0
Database_Services('WriteDataRow', 'SCHED_DET_NG', PrevSchedKeyID, PrevSchedDetRec, True$, False$, True$)
end
end
// Delete canceled event record
Database_Services('DeleteDataRow', 'SCHED_DET_NG', SchedDetKeyID)
end
// Record schedule change in SCHED_HIST table
Schedule_Services('AddSchedHist', SchedDetKeyID, ReactNo, 'CANCEL', @User4)
Response = RemSchedDetKeys
end service
Service GetStartCassNo(SchedDetKey)
RDSNo = ''
CassNo = ''
If SchedDetKey NE '' then
SchedDetRec = Schedule_Services('GetScheduleDetail', SchedDetKey)
WONo = SchedDetRec<SCHED_DET_NG.WO_NO$>
ReactorNo = SchedDetRec<SCHED_DET_NG.REACT_NO$>
EventStartDTM = SchedDetRec<SCHED_DET_NG.START_DTM$>
StartDate = EventStartDTM[1, 'F.']
StartTime = (EventStartDTM[-1, 'B.'] * 86400)
StartTime = StartTime[1, 5]
// Find first cassette with START_DT and START_TIME after event START_DTM
Query = 'SELECT RDS WITH WO EQ ':WONo:' AND WITH REACTOR EQ ':ReactorNo:' BY DATE_IN BY TIME_IN'
Set_Status(0)
GoSub ClearCursors
Rlist(Query, Target_ActiveList$, '', '', '')
If @RecCount then
Done = False$
EOF = False$
Loop
ReadNext RDSNo else EOF = True$
Until EOF EQ True$
DTMIn = Xlate('RDS', RDSNo, 'DATETIME_IN', 'X')
If DTMIN GE EventStartDTM then
Done = True$
end
Until Done EQ True$
Repeat
end
GoSub ClearCursors
errCode = ''
If Get_Status(errCode) then
ErrorMsg = 'Error in service ':Service:' module. Error code':errCode
Error_Services('Add', ErrorMsg)
end
end
If RDSNo NE '' then CassNo = Xlate('RDS', RDSNo, 'CASS_NO', 'X')
Response = CassNo
end service
Service GetEventLastCassNo(SchedDetKey)
RDSNo = ''
CassNo = ''
If SchedDetKey NE '' then
SchedDetRec = Schedule_Services('GetScheduleDetail', SchedDetKey)
ReactorNo = SchedDetRec<SCHED_DET_NG.REACT_NO$>
EventStopDTM = SchedDetRec<SCHED_DET_NG.STOP_DTM$>
WONo = SchedDetRec<SCHED_DET_NG.WO_NO$>
EventStartDTM = SchedDetRec<SCHED_DET_NG.START_DTM$>
StartDate = EventStartDTM[1, 'F.']
StartTime = (EventStartDTM[-1, 'B.'] * 86400)
StartTime = StartTime[1, 5]
StopDate = EventStopDTM[1, 'F.']
StopTime = (EventStopDTM[-1, 'B.'] * 86400)
StopTime = StopTime[1, 5]
// Find first cassette with START_DT and START_TIME after event START_DTM
Query = 'SELECT RDS WITH WO EQ ':WONo:' AND WITH REACTOR EQ ':ReactorNo:' BY-DSND DATE_OUT BY-DSND TIME_OUT'
Set_Status(0)
GoSub ClearCursors
Rlist(Query, Target_ActiveList$, '', '', '')
If @RecCount then
Done = False$
EOF = False$
Loop
ReadNext RDSNo else EOF = True$
Until EOF EQ True$
DTMOut = Xlate('RDS', RDSNo, 'DATETIME_OUT', 'X')
If DTMOut LE EventStopDTM then
Done = True$
end
Until Done EQ True$
Repeat
end
GoSub ClearCursors
errCode = ''
If Get_Status(errCode) then
ErrorMsg = 'Error in service ':Service:' module. Error code':errCode
Error_Services('Add', ErrorMsg)
end
end
If RDSNo NE '' then CassNo = Xlate('RDS', RDSNo, 'CASS_NO', 'X')
Response = CassNo
end service
Service GetEventRemainingWfrs(SchedDetKey)
WfrsRemaining = 0
If SchedDetKey NE '' then
SchedDetRec = Schedule_Services('GetScheduleDetail', SchedDetKey)
WONo = SchedDetRec<SCHED_DET_NG.WO_NO$>
ReactType = Xlate('WO_LOG', WONo, 'REACT_TYPE', 'X')
SchedQty = SchedDetRec<SCHED_DET_NG.EVENT_TOTAL_WFRS$>
UnprocessedCassettes = SchedDetRec<SCHED_DET_NG.UNPROCESSED_CASS$>
NumUnprocessedCassettes = DCount(UnprocessedCassettes, @VM)
If ReactType NE 'EPP' then
WfrsRemaining = 25 * NumUnprocessedCassettes
end else
If UnprocessedCassettes NE '' then
QueryEndDtm = SchedDetRec<SCHED_DET_NG.START_DTM$>
PrevSchedWfrQty = Schedule_Services('GetScheduledWfrQty', WONo, QueryEndDtm)
StartSlot = Mod(PrevSchedWfrQty, 25) + 1
StartCass = SRP_Math('CEILING', ( (PrevSchedWfrQty + 1) / 25))
If StartCass EQ 0 then StartCass = 1
LastUnprocCass = UnprocessedCassettes<0, NumUnprocessedCassettes>
TotalQty = PrevSchedWfrQty + SchedQty
StopSlot = Mod(TotalQty, 25)
If StopSlot EQ 0 then StopSlot = 25
For each UnprocCass in UnprocessedCassettes using @VM setting vPos
WMIKey = WONo:'*1*':UnprocCass
If RowExists('WM_IN', WMIKey) then
RDSKeys = Xlate('WM_IN', WMIKey, 'RDS_NO', 'X')
UnloadDtms = Xlate('RDS', RDSKeys, 'DATETIME_OUT', 'X')
If UnprocCass EQ StartCass then
StartIndex = StartSlot
end else
StartIndex = 1
end
For SlotIndex = StartIndex to 25
If UnloadDtms<0, SlotIndex> EQ '' then WfrsRemaining += 1
Until ( (UnprocCass EQ LastUnprocCass) and (SlotIndex EQ StopSlot) )
Next SlotIndex
end else
WfrsRemaining += 25
end
Next UnprocCass
end
end
Response = WfrsRemaining
end
If WfrsRemaining GT 0 then
Response = WfrsRemaining
end else
Response = 0
Error_Services('Add', 'Error in ':Service:' service. Negative wafers remaining value returned!')
end
end service
Service GetEventCompStatus(SchedDetKey)
If SchedDetKey NE '' then
SchedDetRec = Schedule_Services('GetScheduleDetail', SchedDetKey)
SchedQty = SchedDetRec<SCHED_DET_NG.EVENT_TOTAL_WFRS$>
NumUnprocWfrs = Schedule_Services('GetEventRemainingWfrs', SchedDetKey)
NumWfrsProcessed = SchedQty - NumUnprocWfrs
If ( (SchedQty NE '') and (SchedQty GT 0) ) then
PercentComplete = (NumWfrsProcessed / SchedQty)
Response = PercentComplete
end
end
end service
Service GetCurrentEvent(ReactNo)
StartTick = GetTickCount()
MetricName = 'GetCurrentEvent'
SchedDetKey = ''
ErrorMsg = ''
If ReactNo NE '' then
CurrDTM = Datetime()
Query = 'SELECT SCHED_DET_NG WITH REACT_NO EQ "':ReactNo:'" AND WITH EVENT_COMP NE 1 BY START_DTM'
Set_Status(0)
GoSub ClearCursors
Rlist(Query, TARGET_ACTIVELIST$, '', '', '')
errCode = ''
If Not(Get_Status(errCode)) then
ReadNext SchedDetKey else EOF = True$
Response = SchedDetKey
end else
ErrorMsg = 'Error in service ':Service:' module. Error code ':errCode
end
GoSub ClearCursors
end
EndTick = GetTickCount()
Mona_Services('QueueLatencyAndCountMetrics', MonaResource, MetricName, StartTick, EndTick)
If ErrorMsg NE '' then Error_Services('Add', ErrorMsg)
end service
Service CurrentEventIsBlock(ReactNo)
StartTick = GetTickCount()
MetricName = 'CurrentEventIsBlock'
ErrorMsg = ''
If ReactNo NE '' then
CurrentSchedDetKey = Schedule_Services('GetCurrentEvent', ReactNo)
If Error_Services('NoError') then
CurrentSchedDetRecord = Database_Services('ReadDataRow', 'SCHED_DET_NG', CurrentSchedDetKey)
CurrentEventIsABlock = CurrentSchedDetRecord<SCHED_DET_NG.BLOCK_OUT$>
If CurrentEventIsABlock EQ '' then
CurrentEventIsABlock = False$
end
Response = CurrentEventIsABlock
end else
Error = Error_Services('GetMessage')
ErrorMsg = 'Error in service ':Service:' module. ':Error
end
end else
ErrorMsg = 'Error in service ':Service:' module. Invalid reactor number.'
end
EndTick = GetTickCount()
Mona_Services('QueueLatencyAndCountMetrics', MonaResource, MetricName, StartTick, EndTick)
If ErrorMsg NE '' then Error_Services('Add', ErrorMsg)
end service
Service GetNextEvent(ReactNo)
StartTick = GetTickCount()
MetricName = 'GetNextEvent'
SchedDetKey = ''
ErrorMsg = ''
If ReactNo NE '' then
CurrDTM = Datetime()
Query = 'SELECT SCHED_DET_NG WITH REACT_NO EQ "':ReactNo:'" AND WITH EVENT_COMP NE 1 BY START_DTM'
Set_Status(0)
GoSub ClearCursors
Rlist(Query, TARGET_ACTIVELIST$, '', '', '')
errCode = ''
If Not(Get_Status(errCode)) then
EOF = False$
ReadNext SchedDetKey else EOF = True$
If EOF EQ True$ then
ErrorMsg = 'Error in service ':Service:' module. No incomplete events in schedule'
end else
ReadNext SchedDetKey else EOF = True$
If EOF EQ True$ then
ErrorMsg = 'Error in service ':Service:' module. No incomplete events in schedule after current event'
end else
Response = SchedDetKey
end
end
end else
ErrorMsg = 'Error in service ':Service:' module. Error code ':errCode
end
GoSub ClearCursors
end
EndTick = GetTickCount()
Mona_Services('QueueLatencyAndCountMetrics', MonaResource, MetricName, StartTick, EndTick)
If ErrorMsg NE '' then Error_Services('Add', ErrorMsg)
end service
Service GetEngagedEvent(ReactNo)
SchedDetKey = ''
If ReactNo NE '' then
CurrDTM = Datetime()
Query = 'SELECT SCHED_DET_NG WITH REACT_NO EQ "':ReactNo:'" AND WITH UNPROCESSED_CASS NE "" AND WITH EVENT_COMP NE 1 BY START_DTM'
Set_Status(0)
GoSub ClearCursors
Rlist(Query, TARGET_ACTIVELIST$, '', '', '')
errCode = ''
If Not(Get_Status(errCode)) then
ReadNext SchedDetKey else EOF = True$
Response = SchedDetKey
end else
ErrorMsg = 'Error in service ':Service:' module. Error code ':errCode
Error_Services('Add', ErrorMsg)
end
GoSub ClearCursors
end
end service
Service GetLastEngagedEvent(ReactNo)
SchedDetKey = ''
If ReactNo NE '' then
CurrDTM = Datetime()
Query = 'SELECT SCHED_DET_NG WITH REACT_NO EQ "':ReactNo:'" AND WITH START_DTM LT "':CurrDTM:'" '
Query := 'BY-DSND START_DTM'
Set_Status(0)
GoSub ClearCursors
Rlist(Query, Target_ActiveList$, '', '', '')
ReadNext SchedDetKey else EOF = True$
GoSub ClearCursors
errCode = ''
If Get_Status(errCode) then
ErrorMsg = 'Error in service ':Service:' module. Error code':errCode
Error_Services('Add', ErrorMsg)
end else
Response = SchedDetKey
end
end
end service
Service NextEventIsSamePsn(ReactNo)
StartTick = GetTickCount()
MetricName = 'NextEventIsSamePsn'
Response = False$
ErrorMsg = ''
If ReactNo NE '' then
CurrentSchedDetKey = Schedule_Services('GetCurrentEvent', ReactNo)
If Error_Services('NoError') then
CurrentEventIsABlock = Schedule_Services('CurrentEventIsBlock', ReactNo)
If CurrentEventIsABlock EQ False$ then
CurrentSchedDetRecord = Database_Services('ReadDataRow', 'SCHED_DET_NG', CurrentSchedDetKey)
CurrentEventWoNo = CurrentSchedDetRecord<SCHED_DET_NG.WO_NO$>
CurrentEventWoMatRecord = Database_Services('ReadDataRow', 'WO_MAT', CurrentEventWoNo:'*1')
CurrentEventRdsNo = CurrentEventWoMatRecord<WO_MAT_RDS_NO$>
CurrentEventRdsRecord = Database_Services('ReadDataRow', 'RDS', CurrentEventRdsNo)
CurrentEventPsn = CurrentEventRdsRecord<RDS_PROD_SPEC_ID$>
NextSchedDetKey = Schedule_Services('GetNextEvent', ReactNo)
If Error_Services('NoError') then
NextEventIsABlock = Schedule_Services('NextEventIsBlock', ReactNo)
If NextEventIsABlock EQ False$ then
NextSchedDetRecord = Database_Services('ReadDataRow', 'SCHED_DET_NG', NextSchedDetKey)
NextEventWoNo = NextSchedDetRecord<SCHED_DET_NG.WO_NO$>
NextEventWoMatRecord = Database_Services('ReadDataRow', 'WO_MAT', NextEventWoNo:'*1')
NextEventRdsNo = NextEventWoMatRecord<WO_MAT_RDS_NO$>
NextEventRdsRecord = Database_Services('ReadDataRow', 'RDS', NextEventRdsNo)
NextEventPsn = NextEventRdsRecord<RDS_PROD_SPEC_ID$>
PsnsMatch = CurrentEventPsn EQ NextEventPsn
Response = PsnsMatch
end
end
end
end
end else
ErrorMsg = 'Error in service ':Service:' module. Invalid reactor number.'
end
EndTick = GetTickCount()
Mona_Services('QueueLatencyAndCountMetrics', MonaResource, MetricName, StartTick, EndTick)
If ErrorMsg NE '' then Error_Services('Add', ErrorMsg)
end service
Service NextEventIsBlock(ReactNo)
StartTick = GetTickCount()
MetricName = 'NextEventIsBlock'
ErrorMsg = ''
If ReactNo NE '' then
NextSchedDetKey = Schedule_Services('GetNextEvent', ReactNo)
If Error_Services('NoError') then
NextSchedDetRecord = Database_Services('ReadDataRow', 'SCHED_DET_NG', NextSchedDetKey)
NextEventIsABlock = NextSchedDetRecord<SCHED_DET_NG.BLOCK_OUT$>
If NextEventIsABlock EQ '' then
NextEventIsABlock = False$
end
Response = NextEventIsABlock
end else
Error = Error_Services('GetMessage')
ErrorMsg = 'Error in service ':Service:' module. ':Error
end
end else
ErrorMsg = 'Error in service ':Service:' module. Invalid reactor number.'
end
EndTick = GetTickCount()
Mona_Services('QueueLatencyAndCountMetrics', MonaResource, MetricName, StartTick, EndTick)
If ErrorMsg NE '' then Error_Services('Add', ErrorMsg)
end service
//----------------------------------------------------------------------------------------------------------------------
// AdjustScheduleEvents
//
// Adjusts the schedule detail rows for a given reactor.
//----------------------------------------------------------------------------------------------------------------------
Service AdjustScheduleEvents(ReactNo)
HaveLock = False$
hSysLists = Database_Services('GetTableHandle', 'SYSLISTS')
Lock hSysLists, ServiceKeyID:'*':ReactNo then
HaveLock = True$
CurrDTM = Datetime()
// Log a line break to separate service calls
LogData = ''
LogData<1> = ''
LogData<2> = ''
LogData<3> = ''
Schedule_Services('LogActivity', ReactNo, LogData)
LogData = ''
LogData<1> = OConv(Datetime(), 'DT2/^H')
LogData<2> = '1'
LogData<3> = 'Begin Adjusting Reactor ':ReactNo
Schedule_Services('LogActivity', ReactNo, LogData)
OrigReactNo = ReactNo
Done = False$
// Mark all hard blocks with end datetimes before the current date time as complete.
LogData = ''
LogData<1> = OConv(Datetime(), 'DT2/^H')
LogData<2> = '2'
LogData<3> = 'Checking for hard block events to close.'
Schedule_Services('LogActivity', ReactNo, LogData)
EOF = False$
Query = 'SELECT SCHED_DET_NG WITH REACT_NO EQ ':Quote(ReactNo):' AND WITH BLOCK_OUT EQ ':Quote(True$)
Query := ' AND WITH STOP_DTM LT ':Quote(CurrDtm):' AND WITH EVENT_COMP NE 1 AND WITH HARD_BLOCK EQ 1'
GoSub ClearCursors
Set_Status(0)
RList(Query, TARGET_ACTIVELIST$, '', '', '')
If Not(Get_Status(ErrCode)) then
LoopIndex = 0
Loop
ReadNext OldBlockOutKey else EOF = True$
Until EOF
LoopIndex += 1
OldBlockOutRec = Database_Services('ReadDataRow', 'SCHED_DET_NG', OldBlockOutKey)
If Error_Services('NoError') then
LogData = ''
LogData<1> = OConv(Datetime(), 'DT2/^H')
LogData<2> = '2.':LoopIndex
LogData<3> = 'Marking previous hard block event complete: ':OldBlockOutKey
Schedule_Services('LogActivity', ReactNo, LogData)
OldBlockOutRec<SCHED_DET_NG.EVENT_COMP$> = True$
Database_Services('WriteDataRow', 'SCHED_DET_NG', OldBlockOutKey, OldBlockOutRec, True$, False$, True$)
end
Repeat
end
// Find the current event and start adjusting from there.
CurrEventKey = Schedule_Services('GetCurrentEvent', ReactNo)
If CurrEventKey EQ '' then
CurrEventKey = Schedule_Services('GetLastEngagedEvent', ReactNo)
end
LogData = ''
LogData<1> = OConv(Datetime(), 'DT2/^H')
LogData<2> = '3.1'
LogData<3> = 'Located starting event to adjust from: ':CurrEventKey
Schedule_Services('LogActivity', ReactNo, LogData)
ReactEventList = Xlate('REACTOR', ReactNo, 'SCHED_EVENTS', 'X')
OrigReactEventList = ReactEventList
NewReactEventList = ''
If ReactEventList NE '' then
NumEvents = DCount(ReactEventList, @VM)
NonHBList = ''
HBList = ''
TruncatePos = ''
// Build list of non-hard block events and a list of hard block events.
Locate CurrEventKey in ReactEventList using @VM setting vPos then
TruncatePos = vPos
For EventKeyIndex = vPos to NumEvents
EventKey = ReactEventList<0, EventKeyIndex>
HardBlock = Xlate('SCHED_DET_NG', EventKey, 'HARD_BLOCK', 'X')
If HardBlock EQ True$ then
HBList<0, -1> = EventKey
end else
NonHBList<0, -1> = EventKey
end
Next EventKeyIndex
// Create a new list up to, but not including, the current event.
For EventKeyIndex = 1 to (vPos - 1)
NewReactEventList<0, EventKeyIndex> = ReactEventList<0, EventKeyIndex>
Next EventKeyIndex
end
If NonHBList NE '' then
LogData = ''
LogData<1> = OConv(Datetime(), 'DT2/^H')
LogData<2> = '3.2'
LogData<3> = 'Attempting to merge non-hard block consecutive events with matching work orders'
Schedule_Services('LogActivity', ReactNo, LogData)
// Merge all non-hard block events (merge consecutive events with matching work orders)
NumNonHBEvents = DCount(NonHBList, @VM)
// Stop at the second to last event so that we don't overstep the array
For EventIndex = 1 to (NumNonHBEvents - 1)
// Look ahead at the next event to see if the work order numbers match. If so, then "merge" the events.
// Merging two events
// - Add scheduled quantities together into first event
// - Delete second event record
// - Delete event from NonHBList
// - Decrement NumNonHBEvents
// - Update next event pointer
ThisEventKey = NonHBList<0, EventIndex>
ThisEventWONo = Xlate('SCHED_DET_NG', ThisEventKey, 'WO_NO', 'X')
ThisEventBlockOut = Xlate('SCHED_DET_NG', ThisEventKey, 'BLOCK_OUT', 'X')
NextEventKey = NonHBList<0, EventIndex + 1>
NextEventWONo = Xlate('SCHED_DET_NG', NextEventKey, 'WO_NO', 'X')
NextEventBlockOut = Xlate('SCHED_DET_NG', NextEventKey, 'BLOCK_OUT', 'X')
// Do not merge blockout events!
If ( (ThisEventBlockOut NE True$) and (NextEventBlockOut NE True$) and (ThisEventWONo EQ NextEventWONo) ) then
// Merge the events
NextEventRec = Database_Services('ReadDataRow', 'SCHED_DET_NG', NextEventKey)
NextEventWfrQty = NextEventRec<SCHED_DET_NG.EVENT_TOTAL_WFRS$>
NextEventUnprocCass = NextEventRec<SCHED_DET_NG.UNPROCESSED_CASS$>
NextEventProcCass = NextEventRec<SCHED_DET_NG.PROCESSED_CASS$>
NextNextEventKey = NonHBList<EventIndex + 2>
ThisEventRec = Database_Services('ReadDataRow', 'SCHED_DET_NG', ThisEventKey)
ThisEventWfrQty = ThisEventRec<SCHED_DET_NG.EVENT_TOTAL_WFRS$>
ThisEventUnprocCass = ThisEventRec<SCHED_DET_NG.UNPROCESSED_CASS$>
ThisEventProcCass = ThisEventRec<SCHED_DET_NG.PROCESSED_CASS$>
// Merge wafer quantities and cassette lists
NewEventWfrQty = ThisEventWfrQty + NextEventWfrQty
ThisEventUnprocCass = SRP_Array('Join', ThisEventUnprocCass, NextEventUnprocCass, 'OR', @VM)
ThisEventProcCass = SRP_Array('Join', ThisEventProcCass, NextEventProcCass, 'OR', @VM)
ThisEventUnprocCass = SRP_Array('SortSimpleList', ThisEventUnprocCass, 'AscendingNumbers', @VM)
ThisEventProcCass = SRP_Array('SortSimpleList', ThisEventProcCass, 'AscendingNumbers', @VM)
// Merging split EpiPro events will result in the cassette between the two events
// being in both the processed and unprocessed cassette lists.
CutoverCassList = SRP_Array('Join', ThisEventUnprocCass, ThisEventProcCass, 'AND', @VM)
If CutoverCassList NE '' then
For each CutoverCassNo in CutoverCassList using @VM
Locate CutoverCassNo in ThisEventProcCass using @VM setting vPos then
ThisEventProcCass = Delete(ThisEventProcCass, 0, vPos, 0)
end
Next CutoverCassNo
end
LogData = ''
LogData<1> = OConv(Datetime(), 'DT2/^H')
LogData<2> = '3.3'
LogData<3> = 'Merging ':ThisEventKey:' with ':ThisEventWfrQty:' wafers and ':NextEventKey:' with ':NextEventWfrQty:' wafers'
Schedule_Services('LogActivity', ReactNo, LogData)
ThisEventRec<SCHED_DET_NG.EVENT_TOTAL_WFRS$> = NewEventWfrQty
ThisEventRec<SCHED_DET_NG.UNPROCESSED_CASS$> = ThisEventUnprocCass
ThisEventRec<SCHED_DET_NG.PROCESSED_CASS$> = ThisEventProcCass
ThisEventRec<SCHED_DET_NG.EVENT_COMP$> = False$
Database_Services('WriteDataRow', 'SCHED_DET_NG', ThisEventKey, ThisEventRec, True$, False$, True$)
LogData = ''
LogData<1> = OConv(Datetime(), 'DT2/^H')
LogData<2> = '3.4'
LogData<3> = 'Merged events into ':ThisEventKey:' with a new quantity of ':NewEventWfrQty
Schedule_Services('LogActivity', ReactNo, LogData)
Database_Services('DeleteDataRow', 'SCHED_DET_NG', NextEventKey, True$, False$)
NonHBList = Delete(NonHBList, 0, (EventIndex + 1), 0)
NumNonHBEvents -= 1
end
Next EventIndex
LogData = ''
LogData<1> = OConv(Datetime(), 'DT2/^H')
LogData<2> = '4'
LogData<3> = 'Begin rescheduling non-hard block events'
Schedule_Services('LogActivity', ReactNo, LogData)
// Reschedule all non-hard block events
For each EventKey in NonHBList using @VM setting vPos
LogData = ''
LogData<1> = OConv(Datetime(), 'DT2/^H')
LogData<2> = '4.':vPos
LogData<3> = 'Begin rescheduling event ':EventKey
Schedule_Services('LogActivity', ReactNo, LogData)
EventRec = Schedule_Services('GetScheduleDetail', EventKey)
BlockOut = EventRec<SCHED_DET_NG.BLOCK_OUT$>
EpiPro = (Xlate('REACTOR', ReactNo, 'REACT_TYPE', 'X') EQ 'EPP')
If BlockOut NE True$ then
LogData = ''
LogData<1> = OConv(Datetime(), 'DT2/^H')
LogData<2> = '4.':vPos:'.1'
LogData<3> = 'Original event details:'
Schedule_Services('LogActivity', ReactNo, LogData)
LogData<3> = 'Event work order: ':EventRec<SCHED_DET_NG.WO_NO$>
Schedule_Services('LogActivity', ReactNo, LogData)
Logdata<3> = 'Event start dtm: ':OConv(EventRec<SCHED_DET_NG.START_DTM$>, 'DT2/^H')
Schedule_Services('LogActivity', ReactNo, LogData)
LogData<3> = 'Event stop dtm: ':OConv(EventRec<SCHED_DET_NG.STOP_DTM$>, 'DT2/^H')
Schedule_Services('LogActivity', ReactNo, LogData)
LogData<3> = 'Event quantity: ':EventRec<SCHED_DET_NG.EVENT_TOTAL_WFRS$>
Schedule_Services('LogActivity', ReactNo, LogData)
LogData<3> = 'Event complete: ':OConv(EventRec<SCHED_DET_NG.EVENT_COMP$>, "Btrue,false")
Schedule_Services('LogActivity', ReactNo, LogData)
EventAdjustment = Schedule_Services('GetEventAdjustment', EventKey)
UnprocessedCass = EventRec<SCHED_DET_NG.UNPROCESSED_CASS$>
ProcessedCass = EventRec<SCHED_DET_NG.PROCESSED_CASS$>
AllCassList = SRP_Array('Join', UnprocessedCass, ProcessedCass, 'OR', @VM)
AllCassList = SRP_Array('SortSimpleList', AllCassList, 'AscendingNumbers', @VM)
EventComp = (UnprocessedCass EQ '')
If Not(EventRec<SCHED_DET_NG.EVENT_COMP$>) and EventComp then
LogData = ''
LogData<1> = OConv(Datetime(), 'DT2/^H')
LogData<2> = '4.':vPos:'.1.1'
LogData<3> = 'Closing event ':EventKey
Schedule_Services('LogActivity', ReactNo, LogData)
end
EventRec<SCHED_DET_NG.EVENT_COMP$> = EventComp
WONo = EventRec<SCHED_DET_NG.WO_NO$>
ReactNo = EventRec<SCHED_DET_NG.REACT_NO$>
EventWfrs = EventRec<SCHED_DET_NG.EVENT_TOTAL_WFRS$>
WOQty = Xlate('WO_LOG', WONo, 'QTY', 'X')
// If work order event is engaged then set start DTM to first cassette DTM in.
// If previous event end time overlaps (e.g. soft block finished sooner than scheduled)
// then truncate it. If work order event is not engaged yet, then set start DTM to previous
// event's stop DTM.
RDSNo = ''
DTMIn = ''
EventEngaged = False$
WOMatKeys = ''
For each SchedCassNo in AllCassList using @VM setting SchedIndex
WOMatKeys<0, SchedIndex> = WONo:'*':SchedCassNo
Next SchedCassNo
If EpiPro then
FirstCass = AllCassList<0, 1>
If FirstCass NE '' then
WMIKey = WONo:'*1*':FirstCass
SchedRDSNos = Xlate('WM_IN', WMIKey, 'RDS_NO', 'X')
// Look at the next slot to be scheduled!
QueryEndDtm = EventRec<SCHED_DET_NG.STOP_DTM$>
// Get the number of wafers scheduled up to and including this event.
// This should give us the correct starting wafer slot.
ScheduledWfrQty = Schedule_Services('GetScheduledWfrQty', WONo, QueryEndDtm)
ScheduledWfrQty -= EventWfrs
SchedSlot = Mod(ScheduledWfrQty, 25) + 1
SchedRDSNos = Field(SchedRDSNos, @VM, SchedSlot, (25 - SchedSlot))
end
end else
SchedRDSNos = Xlate('WO_MAT', WOMatKeys, 'RDS_NO', 'X')
end
SchedDateIns = Xlate('RDS', SchedRDSNos, 'DATE_IN', 'X')
If SchedDateIns NE '' then
FirstLoad = ''
For each SchedDateIn in SchedDateIns using @VM setting DateInPos
If SchedDateIn NE '' then
RDSNo = SchedRDSNos<0, DateInPos>
DTMIn = Xlate('RDS', RDSNo, 'DATETIME_IN', 'X')
EventEngaged = True$
end
Until EventEngaged EQ True$
Next SchedDateIn
end
If EventEngaged then
LogData = ''
LogData<1> = OConv(Datetime(), 'DT2/^H')
LogData<2> = '4.':vPos:'.2'
LogData<3> = 'Event ':EventKey:' engaged.'
Schedule_Services('LogActivity', ReactNo, LogData)
LogData<3> = 'Setting start dtm to first RDS load dtm: ':OConv(DTMIn, 'DT2/^H')
Schedule_Services('LogActivity', ReactNo, LogData)
// Work order event is engaged so set start DTM to first cassette DTM in.
EventStartDTM = DTMIn
// Ensure previous event doesn't overlap.
CurrNumEvents = DCount(NewReactEventList, @VM)
If CurrNumEvents GT 0 then
PrevEventKey = NewReactEventList<0, CurrNumEvents>
If PrevEventKey NE '' then
LogData = ''
LogData<1> = OConv(Datetime(), 'DT2/^H')
LogData<2> = '4.':vPos:'.2.1'
LogData<3> = 'Ensuring previous event ':PrevEventKey:' does not overlap.'
Schedule_Services('LogActivity', ReactNo, LogData)
PrevEventRec = Database_Services('ReadDataRow', 'SCHED_DET_NG', PrevEventKey)
PrevEventRecOrig = PrevEventRec
PrevEventOrigStartDTM = PrevEventRec<SCHED_DET_NG.START_DTM$>
PrevEventOrigStopDTM = PrevEventRec<SCHED_DET_NG.STOP_DTM$>
PrevEventOrigQty = PrevEventRec<SCHED_DET_NG.EVENT_TOTAL_WFRS$>
PrevEventWONo = PrevEventRec<SCHED_DET_NG.WO_NO$>
PrevEventBlockOut = PrevEventRec<SCHED_DET_NG.BLOCK_OUT$>
PrevEventDuration = PrevEventOrigStopDTM - PrevEventOrigStartDTM
// Ensure previous event is marked as complete
If PrevEventRec<SCHED_DET_NG.EVENT_COMP$> NE True$ then
LogData = ''
LogData<1> = OConv(Datetime(), 'DT2/^H')
LogData<2> = '4.':vPos:'.2.1.1'
LogData<3> = 'Closing previous event ':PrevEventKey:'.'
Schedule_Services('LogActivity', ReactNo, LogData)
end
PrevEventRec<SCHED_DET_NG.EVENT_COMP$> = True$
If Not(PrevEventBlockOut) then
// Set previous event's stop datetime to the last processed cassette/run's unload datetime
PrevEventProcCass = PrevEventRec<SCHED_DET_NG.PROCESSED_CASS$>
PrevEventUnprocCass = PrevEventRec<SCHED_DET_NG.UNPROCESSED_CASS$>
AllPrevEventSchedCass = SRP_Array('Join', PrevEventProcCass, PrevEventUnprocCass, 'OR', @VM)
AllPrevEventSchedCass = SRP_Array('SortSimpleList', AllPrevEventSchedCass, 'AscendingNumbers', @VM)
NumPrevEventSchedCass = DCount(AllPrevEventSchedCass, @VM)
PrevEventEndCassNo = AllPrevEventSchedCass<0, NumPrevEventSchedCass>
If EpiPro then
PrevSchedWfrQty = Schedule_Services('GetScheduledWfrQty', WONo, PrevEventOrigStopDTM)
PrevEventEndSlotNo = Mod(PrevSchedWfrQty, 25)
If PrevEventEndSlotNo EQ 0 then PrevEventEndSlotNo = 25
WMIKey = PrevEventWONo:'*1*':PrevEventEndCassNo
WMIRdsList = Xlate('WM_IN', WMIKey, 'RDS_NO', 'X')
RDSNo = WMIRdsList<0, PrevEventEndSlotNo>
end else
WOMatKey = PrevEventWONo:'*':PrevEventEndCassNo
RDSNo = Xlate('WO_MAT', WOMatKey, 'RDS_NO', 'X')
end
If RDSNo NE '' then
DtmOut = Xlate('RDS', RDSNo, 'DATETIME_OUT', 'X')
If ( (DtmOut NE '') and (DtmOut LT PrevEventOrigStopDtm) ) then
PrevEventStopDTM = DtmOut
LogData = ''
LogData<1> = OConv(Datetime(), 'DT2/^H')
LogData<2> = '4.':vPos:'.2.2'
LogData<3> = 'Previous event stop dtm set to last RDS unload dtm: ':OConv(DtmOut, 'DT2/^H')
Schedule_Services('LogActivity', ReactNo, LogData)
end else
If PrevEventOrigStopDtm LT EventStartDtm then
PrevEventStopDtm = PrevEventOrigStopDtm
LogData = ''
LogData<1> = OConv(Datetime(), 'DT2/^H')
LogData<2> = '4.':vPos:'.2.2'
LogData<3> = 'Previous event stop dtm less than current event start dtm.'
Schedule_Services('LogActivity', ReactNo, LogData)
LogData<3> = 'Preserving previous event stop dtm.'
Schedule_Services('LogActivity', ReactNo, LogData)
end else
PrevEventStopDtm = EventStartDtm
LogData = ''
LogData<1> = OConv(Datetime(), 'DT2/^H')
LogData<2> = '4.':vPos:'.2.2'
LogData<3> = 'Previous event stop dtm greater than current event start dtm.'
Schedule_Services('LogActivity', ReactNo, LogData)
LogData<3> = 'Setting previous event stop dtm to current event start dtm.'
Schedule_Services('LogActivity', ReactNo, LogData)
end
end
end else
If PrevEventOrigStopDtm LT EventStartDtm then
PrevEventStopDtm = PrevEventOrigStopDtm
end else
PrevEventStopDtm = EventStartDtm
end
end
end else
LogData = ''
LogData<1> = OConv(Datetime(), 'DT2/^H')
LogData<2> = '4.':vPos:'.2.2'
LogData<3> = 'Previous event is a block event. '
Schedule_Services('LogActivity', ReactNo, LogData)
If PrevEventOrigStopDtm GT EventStartDtm then
PrevEventStopDtm = EventStartDTM
LogData<3> = 'Previous event stop dtm greater than current event start dtm.'
Schedule_Services('LogActivity', ReactNo, LogData)
LogData<3> = 'Setting previous event stop dtm to current event start dtm.'
Schedule_Services('LogActivity', ReactNo, LogData)
end else
PrevEventStopDtm = PrevEventOrigStopDtm
LogData<3> = 'Previous event stop dtm less than current event start dtm.'
Schedule_Services('LogActivity', ReactNo, LogData)
LogData<3> = 'Preserving previous event stop dtm.'
Schedule_Services('LogActivity', ReactNo, LogData)
end
Schedule_Services('LogActivity', ReactNo, LogData)
end
PrevEventRec<SCHED_DET_NG.STOP_DTM$> = PrevEventStopDtm
// Check if events will overlap
If PrevEventStopDtm GT EventStartDtm then
LogData = ''
LogData<1> = OConv(Datetime(), 'DT2/^H')
LogData<2> = '4.':vPos:'.2.3'
LogData<3> = 'Calculated previous event stop dtm greater than current event start dtm.'
Schedule_Services('LogActivity', ReactNo, LogData)
LogData<3> = 'Setting current event start dtm to previous event stop dtm.'
Schedule_Services('LogActivity', ReactNo, LogData)
EventStartDtm = PrevEventStopDtm
end
If PrevEventRec NE PrevEventRecOrig then
Database_Services('WriteDataRow', 'SCHED_DET_NG', PrevEventKey, PrevEventRec, True$, False$, True$)
end
end
end
EventStopDTM = CurrDTM + EventAdjustment
end else
LogData = ''
LogData<1> = OConv(Datetime(), 'DT2/^H')
LogData<2> = '4.':vPos:'.2'
LogData<3> = 'Event ':EventKey:' not engaged. '
Schedule_Services('LogActivity', ReactNo, LogData)
// Work order event not engaged, so set start DTM to previous event's stop DTM.
PrevEventKey = ''
EventIsHardBlock = False$
Locate EventKey in OrigReactEventList using @VM setting OrigEventIndex then
If OrigEventIndex GT 1 then
SearchEventIndex = OrigEventIndex - 1
Loop
PrevEventKey = OrigReactEventList<0, SearchEventIndex>
EventExists = RowExists('SCHED_DET_NG', PrevEventKey)
If EventExists then EventIsHardBlock = Xlate('SCHED_DET_NG', PrevEventKey, 'HARD_BLOCK', 'X')
SearchEventIndex -= 1
Until ( ( (EventExists EQ True$) and (EventIsHardBlock NE True$) ) or (SearchEventIndex EQ 1) )
Repeat
end else
PrevEventKey = ''
end
end
If PrevEventKey NE '' then
PrevEventRec = Database_Services('ReadDataRow', 'SCHED_DET_NG', PrevEventKey)
PrevEventStopDTM = PrevEventRec<SCHED_DET_NG.STOP_DTM$>
PrevEventComp = PrevEventRec<SCHED_DET_NG.EVENT_COMP$>
If ( (PrevEventComp NE True$) and (PrevEventStopDTM GT CurrDTM) ) then
EventStartDTM = PrevEventStopDTM
LogData<3> = 'Previous event not complete.'
Schedule_Services('LogActivity', ReactNo, LogData)
LogData<3> = 'Setting start dtm to to previous event stop dtm: ':OConv(PrevEventStopDTM, 'DT2/^H')
Schedule_Services('LogActivity', ReactNo, LogData)
end else
EventStartDTM = CurrDTM
LogData<3> = 'Previous event complete.'
Schedule_Services('LogActivity', ReactNo, LogData)
LogData<3> = 'Setting start dtm to to current dtm: ':OConv(CurrDTM, 'DT2/^H')
Schedule_Services('LogActivity', ReactNo, LogData)
end
end else
EventStartDTM = CurrDTM
LogData<3> = 'No previous event found.'
Schedule_Services('LogActivity', ReactNo, LogData)
LogData<3> = 'Setting start dtm to to current dtm: ':OConv(CurrDTM, 'DT2/^H')
Schedule_Services('LogActivity', ReactNo, LogData)
end
// Set stop DTM to start DTM + event adjustment
EventStopDTM = EventStartDTM + EventAdjustment
LogData<3> = 'New event stop dtm: ':OConv(EventStopDTM, 'DT2/^H')
Schedule_Services('LogActivity', ReactNo, LogData)
end
// Check if event is complete (i.e. no more cassettes to process).
// If event is complete, then set stop DTM to last processed cassette's datetime
// out. Otherwise, adjust stop DTM to current DTM + event adjustment.
If EventComp then
LogData = ''
LogData<1> = OConv(Datetime(), 'DT2/^H')
LogData<2> = '4.':vPos:'.3'
LogData<3> = 'Event ':EventKey:' complete. '
Schedule_Services('LogActivity', ReactNo, LogData)
// Unprocessed cassette list must be empty to get here.
EventRec<SCHED_DET_NG.EVENT_COMP$> = True$
EventWONo = EventRec<SCHED_DET_NG.WO_NO$>
EventProcCass = EventRec<SCHED_DET_NG.PROCESSED_CASS$>
NumEventProcCass = DCount(EventProcCass, @VM)
EventEndCassNo = EventProcCass<0, NumEventProcCass>
If EpiPro then
SchedWfrQty = Schedule_Services('GetScheduledWfrQty', WONo, EventStopDtm)
EventEndSlotNo = Mod(SchedWfrQty, 25)
If EventEndSlotNo EQ 0 then EventEndSlotNo = 25
WMIKey = EventWONo:'*1*':EventEndCassNo
WMIRdsList = Xlate('WM_IN', WMIKey, 'RDS_NO', 'X')
RDSNo = WMIRdsList<0, EventEndSlotNo>
end else
WOMatKey = EventWONo:'*':EventEndCassNo
RDSNo = Xlate('WO_MAT', WOMatKey, 'RDS_NO', 'X')
end
If RDSNo NE '' then
DtmOut = Xlate('RDS', RDSNo, 'DATETIME_OUT', 'X')
If DtmOut NE '' then
EventStopDTM = DtmOut
LogData<3> = 'Setting event stop dtm to last unloaded RDS ':OConv(DtmOut, 'DT2/^H')
Schedule_Services('LogActivity', ReactNo, LogData)
end else
// Failsafe
EventStopDtm = CurrDtm
LogData<3> = 'Last RDS does not unloaded.'
Schedule_Services('LogActivity', ReactNo, LogData)
LogData<3> = 'Setting stop dtm to current dtm ':OConv(CurrDtm, 'DT2/^H')
Schedule_Services('LogActivity', ReactNo, LogData)
end
end else
// Failsafe
EventStopDtm = CurrDtm
LogData<3> = 'Error! Null last RDS.'
Schedule_Services('LogActivity', ReactNo, LogData)
LogData<3> = 'Setting stop dtm to current dtm ':OConv(CurrDtm, 'DT2/^H')
Schedule_Services('LogActivity', ReactNo, LogData)
end
end
EventRec<SCHED_DET_NG.START_DTM$> = EventStartDTM
EventRec<SCHED_DET_NG.STOP_DTM$> = EventStopDTM
Database_Services('WriteDataRow', 'SCHED_DET_NG', EventKey, EventRec, True$, False$, True$)
end else
// Soft block
// Adjust start and stop DTMs so that the soft block immediately follows the previous event
LogData = ''
LogData<1> = OConv(Datetime(), 'DT2/^H')
LogData<2> = '4.':vPos:'.1'
LogData<3> = 'Original event details:'
Schedule_Services('LogActivity', ReactNo, LogData)
LogData<3> = 'Event work order: ':EventRec<SCHED_DET_NG.WO_NO$>
Schedule_Services('LogActivity', ReactNo, LogData)
Logdata<3> = 'Event start dtm: ':OConv(EventRec<SCHED_DET_NG.START_DTM$>, 'DT2/^H')
Schedule_Services('LogActivity', ReactNo, LogData)
LogData<3> = 'Event stop dtm: ':OConv(EventRec<SCHED_DET_NG.STOP_DTM$>, 'DT2/^H')
Schedule_Services('LogActivity', ReactNo, LogData)
LogData<3> = 'Event quantity: ':EventRec<SCHED_DET_NG.EVENT_TOTAL_WFRS$>
Schedule_Services('LogActivity', ReactNo, LogData)
LogData<3> = 'Event complete: ':OConv(EventRec<SCHED_DET_NG.EVENT_COMP$>, "Btrue,false")
Schedule_Services('LogActivity', ReactNo, LogData)
LogData = ''
LogData<1> = OConv(Datetime(), 'DT2/^H')
LogData<2> = '4.':vPos:'.2'
LogData<3> = 'Adjusting soft block event.'
Schedule_Services('LogActivity', ReactNo, LogData)
CurrNumEvents = DCount(NewReactEventList, @VM)
PrevEventKey = NewReactEventList<0, CurrNumEvents>
EventStartDTM = EventRec<SCHED_DET_NG.START_DTM$>
EventStopDTM = EventRec<SCHED_DET_NG.STOP_DTM$>
EventDuration = EventStopDTM - EventStartDTM
If PrevEventKey NE '' then
LogData = ''
LogData<1> = OConv(Datetime(), 'DT2/^H')
LogData<2> = '4.':vPos:'.3'
LogData<3> = 'Previous event found: ':PrevEventKey:' '
Schedule_Services('LogActivity', ReactNo, LogData)
PrevEventRec = Database_Services('ReadDataRow', 'SCHED_DET_NG', PrevEventKey)
PrevEventStopDTM = PrevEventRec<SCHED_DET_NG.STOP_DTM$>
EventStartDTM = PrevEventStopDTM
EventStopDTM = EventStartDTM + EventDuration
EventRec<SCHED_DET_NG.START_DTM$> = EventStartDTM
EventRec<SCHED_DET_NG.STOP_DTM$> = EventStopDTM
Database_Services('WriteDataRow', 'SCHED_DET_NG', EventKey, EventRec, True$, False$, True$)
LogData<3> = 'Setting soft block start dtm to previous event stop dtm.'
Schedule_Services('LogActivity', ReactNo, LogData)
LogData<3> = 'Adjusting soft block stop dtm by original event duration.'
Schedule_Services('LogActivity', ReactNo, LogData)
end
end
// Add event into new reactor event list
NewReactEventList<0, -1> = EventKey
LogData = ''
LogData<1> = OConv(Datetime(), 'DT2/^H')
LogData<2> = '4.':vPos:'.4'
LogData<3> = 'Adjusted event details:'
Schedule_Services('LogActivity', ReactNo, LogData)
LogData<3> = 'Event work order: ':EventRec<SCHED_DET_NG.WO_NO$>
Schedule_Services('LogActivity', ReactNo, LogData)
Logdata<3> = 'Event start dtm: ':OConv(EventRec<SCHED_DET_NG.START_DTM$>, 'DT2/^H')
Schedule_Services('LogActivity', ReactNo, LogData)
LogData<3> = 'Event stop dtm: ':OConv(EventRec<SCHED_DET_NG.STOP_DTM$>, 'DT2/^H')
Schedule_Services('LogActivity', ReactNo, LogData)
LogData<3> = 'Event quantity: ':EventRec<SCHED_DET_NG.EVENT_TOTAL_WFRS$>
Schedule_Services('LogActivity', ReactNo, LogData)
LogData<3> = 'Event complete: ':OConv(EventRec<SCHED_DET_NG.EVENT_COMP$>, "Btrue,false")
Schedule_Services('LogActivity', ReactNo, LogData)
Next EventKey
end
If HBList NE '' then
LogData = ''
LogData<1> = OConv(Datetime(), 'DT2/^H')
LogData<2> = '5'
LogData<3> = 'Hard block event(s) found.'
Schedule_Services('LogActivity', ReactNo, LogData)
LogData<3> = 'Splitting events and inserting as needed.'
Schedule_Services('LogActivity', ReactNo, LogData)
// Insert all hard block events.
// - Find insert position in list.
// - Scan the reactor event list from end to beginning
// - If not a block event then
// - Split work order event
// - Calculate the number of wafers (round down for whole cassettes) that can be run in
// the time window (between event start time and hard block start).
// - Calculate event stop time
// - Insert hard block into list
// - Calculate remaining number of wafers (cassettes)
// - Create a new event with remaining number of wafers (start time will be hard block stop time)
// - Insert new event after hard block
// - Adjust downstream events
For each HBEventKey in HBList using @VM setting vPos
LogData = ''
LogData<1> = OConv(Datetime(), 'DT2/^H')
LogData<2> = '5.':vPos:'.1'
LogData<3> = 'Rescheduling hard block event ':HBEventKey:'.'
Schedule_Services('LogActivity', ReactNo, LogData)
HBInserted = False$
HBEventRec = Schedule_Services('GetScheduleDetail', HBEventKey)
HBEventStartDTM = HBEventRec<SCHED_DET_NG.START_DTM$>
HBEventStopDTM = HBEventRec<SCHED_DET_NG.STOP_DTM$>
// Find position in event list to insert hard block and determine if we need to split an event
// (e.g. if there is an event occuring in the hard block time slot).
NumSchedEvents = DCount(NewReactEventList, @VM)
If CurrEventKey NE '' then
Locate CurrEventKey in NewReactEventList using @VM setting StartEventIndex then
// Traverse forwards
StepInc = 1
end else
// Traverse backwards
StartEventIndex = NumSchedEvents
StepInc = -1
end
end else
// Traverse backwards
StartEventIndex = NumSchedEvents
StepInc = -1
end
EventIndex = StartEventIndex
Loop
ThisEventKey = NewReactEventList<0, EventIndex>
NextEventKey = NewReactEventList<0, EventIndex + StepInc>
If ThisEventKey NE '' then
ThisEventRec = Database_Services('ReadDataRow', 'SCHED_DET_NG', ThisEventKey)
ThisEventWONo = ThisEventRec<SCHED_DET_NG.WO_NO$>
ReactType = Xlate('WO_LOG', ThisEventWONo, 'REACT_TYPE', 'X')
ThisEventStartDTM = ThisEventRec<SCHED_DET_NG.START_DTM$>
ThisEventStopDTM = ThisEventRec<SCHED_DET_NG.STOP_DTM$>
ThisEventTotalWfrQty = ThisEventRec<SCHED_DET_NG.EVENT_TOTAL_WFRS$>
If NextEventKey NE '' then
NextEventStartDtm = Xlate('SCHED_DET_NG', NextEventKey, 'START_DTM', 'X')
end else
NextEventStartDtm = ''
end
LogData = ''
LogData<1> = OConv(Datetime(), 'DT2/^H')
LogData<2> = '5.':vPos:'.2'
LogData<3> = 'Checking for insertion point with event ':ThisEventKey:'.'
Schedule_Services('LogActivity', ReactNo, LogData)
If ( (ThisEventStartDTM LE HBEventStartDTM) and ( (NextEventStartDtm GT HBEventStartDtm) or (NextEventKey EQ '') ) ) then
LogData = ''
LogData<1> = OConv(Datetime(), 'DT2/^H')
LogData<2> = '5.':vPos:'.2.1'
LogData<3> = 'Found insertion point on event ':ThisEventKey:'.'
Schedule_Services('LogActivity', ReactNo, LogData)
HBInserted = True$
// Found insertion position
// Need to determine if we need to split
// Check if event is a block event
BlockEvent = ThisEventRec<SCHED_DET_NG.BLOCK_OUT$>
TimeConflict = (ThisEventStopDTM GT HBEventStartDTM)
If ( (BlockEvent NE True$) and (TimeConflict EQ True$) ) then
LogData = ''
LogData<1> = OConv(Datetime(), 'DT2/^H')
LogData<2> = '5.':vPos:'.2.2'
LogData<3> = 'Confirmed event ':ThisEventKey:' is not a block event and there is time conflict.'
Schedule_Services('LogActivity', ReactNo, LogData)
FirstEventProcCassList = ''
FirstEventUnprocCassList = ''
FirstLoadedRDS = ''
LastLoadedRDS = ''
LastUnloadedRDS = ''
Query = "SELECT RDS WITH WO EQ ":ThisEventWONo:" AND WITH DATETIME_IN GE '"
Query := OConv(ThisEventStartDTM, 'DT/^S'):"' AND WITH DATETIME_IN LE '"
Query := OConv(HBEventStartDTM, 'DT/^S'):"' BY DATETIME_IN"
LogData = ''
LogData<1> = OConv(Datetime(), 'DT2/^H')
LogData<2> = '5.':vPos:'.2.3'
LogData<3> = 'Selecting RDS records to associate with first split event.'
Schedule_Services('LogActivity', ReactNo, LogData)
RList(Query, TARGET_ACTIVELIST$, '', '', '')
If Not(Get_Status(ErrCode)) then
RDSKeys = ''
EOF = False$
Loop
ReadNext RDSKey else EOF = True$
Until EOF
RDSKeys<0, -1> = RDSKey
Repeat
If RDSKeys NE '' then
LogData = ''
LogData<1> = OConv(Datetime(), 'DT2/^H')
LogData<2> = '5.':vPos:'.2.4'
LogData<3> = 'RDS keys found. Building unprocessed and processed cassette lists.'
Schedule_Services('LogActivity', ReactNo, LogData)
For each RDSKey in RDSKeys using @VM
If FirstLoadedRDS EQ '' then FirstLoadedRDS = RDSKey
DtmOut = Xlate('RDS', RDSKey, 'DATETIME_OUT', 'X')
If ReactType EQ 'EPP' then
WMIKeys = Xlate('RDS', RDSKey, 'WM_IN_KEY', 'X')
For each WMIKey in WMIKeys using @VM
CassNo = Field(WMIKey, '*', 3)
If DtmOut NE '' then
LastUnloadedRDS = RDSKey
Locate CassNo in FirstEventProcCassList using @VM setting Dummy else
FirstEventProcCassList<0, -1> = CassNo
end
end else
LastLoadedRDS = RDSKey
Locate CassNo in FirstEventUnprocCassList using @VM setting Dummy else
FirstEventUnprocCassList<0, -1> = CassNo
end
end
Next WMIKey
end else
CassNo = Xlate('RDS', RDSKey, 'CASS_NO', 'X')
If DtmOut NE '' then
LastUnloadedRDS = RDSKey
FirstEventProcCassList<0, -1> = CassNo
end else
LastLoadedRDS = RDSKey
FirstEventUnprocCassList<0, -1> = CassNo
end
end
Next RDSKey
// Cleanup EpiPro cassette lists.
FirstEventUnprocCassList = SRP_Array('Join', FirstEventUnprocCassList, FirstEventProcCassList, 'NOT', @VM)
end else
LogData = ''
LogData<1> = OConv(Datetime(), 'DT2/^H')
LogData<2> = '5.':vPos:'.2.4'
LogData<3> = 'No RDS keys found.'
Schedule_Services('LogActivity', ReactNo, LogData)
end
end
FirstEventProcCassCount = DCount(FirstEventProcCassList, @VM)
FirstEventUnprocCassCount = DCount(FirstEventUnprocCassList, @VM)
FirstEventRemCassCount = 0
If HBEventStartDTM GT CurrDtm then
LogData = ''
LogData<1> = OConv(Datetime(), 'DT2/^H')
LogData<2> = '5.':vPos:'.2.4.1'
LogData<3> = 'Hard block event not engaged yet.'
Schedule_Services('LogActivity', ReactNo, LogData)
LogData<3> = 'Estimating the number of cassettes that can be run before the hard block.'
Schedule_Services('LogActivity', ReactNo, LogData)
TimeWindow = HBEventStartDTM - CurrDTM
// Calculate number of whole cassettes we can run in this window
FirstEventRemWfrEstimate = Schedule_Services('GetWaferEstimate', ThisEventWONo, TimeWindow)
FirstEventRemCassCount += SRP_Math('FLOOR', (FirstEventRemWfrEstimate / 25) )
FirstEventUnprocCassCount += FirstEventRemCassCount
end
// If it is estimated that additional cassettes can run before the hard block event,
// then we'll work with cassette quantities of 25. If it is estimated that no addtional
// whole cassettes can run before the hard block, then determine the exact event wafer
// quantity to ensure the event begins and ends on an RDS and not on a whole cassette.
If FirstEventRemCassCount GT 0 then
LogData = ''
LogData<1> = OConv(Datetime(), 'DT2/^H')
LogData<2> = '5.':vPos:'.2.5'
LogData<3> = 'At least one full cassette can run, so estimating stop dtm of event.'
Schedule_Services('LogActivity', ReactNo, LogData)
FirstEventUnprocWfrQty = FirstEventUnprocCassCount * 25
// Determine stop DTM of first split event
EventRuntime = Schedule_Services('GetEventDuration', ThisEventWONo, FirstEventUnprocWfrQty)
ThisEventStopDTM = CurrDTM + EventRuntime
If ThisEventStopDTM GT HBEventStartDtm then
ThisEventStopDtm = HBEventStartDtm
end
// Set new wafer qty for first split
FirstEventProcWfrQty = FirstEventProcCassCount * 25
FirstEventWfrQty = FirstEventUnprocWfrQty + FirstEventProcWfrQty
FirstEventCassCount = FirstEventWfrQty / 25
LogData = ''
LogData<1> = OConv(Datetime(), 'DT2/^H')
LogData<2> = '5.':vPos:'.2.5.1'
LogData<3> = 'First split event wafer quantity ':FirstEventWfrQty
Schedule_Services('LogActivity', ReactNo, LogData)
// Determine unprocessed cass list
If FirstEventProcCassCount GT 0 then
LastProcCassNo = FirstEventProcCassList<0, FirstEventProcCassCount>
end else
LastProcCassNo = 0
end
FirstEventUnprocCassList = ''
For CassIndex = 1 to FirstEventUnprocCassCount
FirstEventUnprocCassList<0, -1> = (LastProcCassNo + CassIndex)
Next CassIndex
end else
LogData = ''
LogData<1> = OConv(Datetime(), 'DT2/^H')
LogData<2> = '5.':vPos:'.2.5'
LogData<3> = 'It is estimated that no additional cassettes can be processed in the time left before the hard block event.'
Schedule_Services('LogActivity', ReactNo, LogData)
// It is estimated that no additional cassettes can be processed in the time left before the hard block event.
If FirstEventUnprocCassList NE '' then
ThisEventStopDTM = HBEventStartDTM
If ReactType EQ 'EPP' then
// Find slot event starts on
FirstLoadedWMIs = Xlate('RDS', FirstLoadedRDS, 'WM_IN_KEY', 'X')
FirstLoadedWMI = FirstLoadedWMIs[1, 'F':@VM]
FirstLoadedWMIRdsList = Xlate('WM_IN', FirstLoadedWMI, 'RDS_NO', 'X')
Done = False$
For SlotIndex = 1 to 25 Step 1
If FirstLoadedWMIRdsList<0, SlotIndex> EQ FirstLoadedRDS then
StartSlot = SlotIndex
Done = True$
end
Until Done
Next StartSlot
// Find slot event stops on
LastLoadedWMIs = Xlate('RDS', LastLoadedRDS, 'WM_IN_KEY', 'X')
LastLoadedWMI = LastLoadedWMIs[-1, 'B':@VM]
LastLoadedWMIRdsList = Xlate('WM_IN', LastLoadedWMI, 'RDS_NO', 'X')
Done = False$
For SlotIndex = 25 to 1 Step -1
If LastLoadedWMIRdsList<0, SlotIndex> EQ LastLoadedRDS then
StopSlot = SlotIndex
Done = True$
end
Until Done
Next SlotIndex
FirstEventWfrQty = ( (FirstEventProcCassCount - 1) * 25 ) + (25 - StartSlot + 1) + ( (FirstEventUnprocCassCount - 1) * 25 ) + StopSlot
end else
FirstEventWfrQty = ( FirstEventProcCassCount * 25 ) + ( FirstEventUnprocCassCount * 25)
end
LogData = ''
LogData<1> = OConv(Datetime(), 'DT2/^H')
LogData<2> = '5.':vPos:'.2.6'
LogData<3> = 'First split event has unprocessed cassettes left.'
Schedule_Services('LogActivity', ReactNo, LogData)
end else
// Get the last RDS that ran and use that as the end of this event.
LastUnloadDtm = Xlate('RDS', LastUnloadedRDS, 'DATETIME_OUT', 'X')
If LastUnloadDtm LT HBEventStartDtm then
ThisEventStopDtm = LastUnloadDtm
end else
ThisEventStopDTM = HBEventStartDTM
end
If ReactType EQ 'EPP' then
// Find slot event starts on
FirstLoadedWMIs = Xlate('RDS', FirstLoadedRDS, 'WM_IN_KEY', 'X')
FirstLoadedWMI = FirstLoadedWMIs[1, 'F':@VM]
FirstLoadedWMIRdsList = Xlate('WM_IN', FirstLoadedWMI, 'RDS_NO', 'X')
Done = False$
For SlotIndex = 1 to 25 Step 1
If FirstLoadedWMIRdsList<0, SlotIndex> EQ FirstLoadedRDS then
StartSlot = SlotIndex
Done = True$
end
Until Done
Next StartSlot
// Find slot event stops on
LastUnloadedWMIs = Xlate('RDS', LastUnloadedRDS, 'WM_IN_KEY', 'X')
LastUnloadedWMI = LastUnloadedWMIs[-1, 'B':@VM]
LastUnloadedWMIRdsList = Xlate('WM_IN', LastUnloadedWMI, 'RDS_NO', 'X')
Done = False$
For SlotIndex = 25 to 1 Step -1
If LastUnloadedWMIRdsList<0, SlotIndex> EQ LastUnloadedRDS then
StopSlot = SlotIndex
Done = True$
end
Until Done
Next SlotIndex
FirstEventWfrQty = ( (FirstEventProcCassCount - 2) * 25 ) + (25 - StartSlot + 1) + StopSlot
end else
FirstEventWfrQty = ( FirstEventProcCassCount * 25 )
end
ThisEventRec<SCHED_DET_NG.EVENT_COMP$> = True$
LogData = ''
LogData<1> = OConv(Datetime(), 'DT2/^H')
LogData<2> = '5.':vPos:'.2.6'
LogData<3> = 'First split event no unprocessed cassettes left.'
Schedule_Services('LogActivity', ReactNo, LogData)
LogData<3> = 'Marking event complete.'
Schedule_Services('LogActivity', ReactNo, LogData)
end
end
// Update first split event
If FirstEventWfrQty GT 100 then
ThisEventRec<SCHED_DET_NG.PROCESSED_CASS$> = FirstEventProcCassList
ThisEventRec<SCHED_DET_NG.UNPROCESSED_CASS$> = FirstEventUnprocCassList
ThisEventRec<SCHED_DET_NG.STOP_DTM$> = ThisEventStopDTM
ThisEventRec<SCHED_DET_NG.EVENT_TOTAL_WFRS$> = FirstEventWfrQty
Database_Services('WriteDataRow', 'SCHED_DET_NG', ThisEventKey, ThisEventRec, True$, False$, True$)
LogData = ''
LogData<1> = OConv(Datetime(), 'DT2/^H')
LogData<2> = '5.':vPos:'.2.7'
LogData<3> = 'First split event updated. Wafer quantity: ':FirstEventWfrQty:'. Stop dtm ':ThisEventStopDtm
Schedule_Services('LogActivity', ReactNo, LogData)
end else
// Event is too small to schedule. Move the wafers to the second split event.
FirstEventWfrQty = 0
Database_Services('DeleteDataRow', 'SCHED_DET_NG', ThisEventKey, True$, False$)
NewReactEventList = Delete(NewReactEventList, 0, EventIndex, 0)
EventIndex -= 1
LogData = ''
LogData<1> = OConv(Datetime(), 'DT2/^H')
LogData<2> = '5.':vPos:'.2.7'
LogData<3> = 'First split event wafer quantity too small.'
Schedule_Services('LogActivity', ReactNo, LogData)
LogData<3> = 'Deleting event and moving all wafers to second split event.'
Schedule_Services('LogActivity', ReactNo, LogData)
end
// Insert hard block into reactor event list after first split
HBEventIndex = EventIndex + 1
NewReactEventList = Insert(NewReactEventList, 0, HBEventIndex, 0, HBEventKey)
// Calculate second split event quantity
SecondEventWfrQty = ThisEventTotalWfrQty - FirstEventWfrQty
LogData = ''
LogData<1> = OConv(Datetime(), 'DT2/^H')
LogData<2> = '5.':vPos:'.2.8'
LogData<3> = 'Determining if second split event is necessary.'
Schedule_Services('LogActivity', ReactNo, LogData)
LogData<3> = 'Second split event wafer quantity ':SecondEventWfrQty:'.'
Schedule_Services('LogActivity', ReactNo, LogData)
If SecondEventWfrQty GT 0 then
SecondEventUnprocCassList = ''
SecondEventProcCassList = ''
// Look at the next slot to be scheduled!
// Get the number of wafers scheduled up to this event.
// This should give us the correct starting wafer slot.
PrevSchedWfrQty = Schedule_Services('GetScheduledWfrQty', ThisEventWONo, HBEventStopDTM)
StartCassNo = SRP_Math('CEILING', ( (PrevSchedWfrQty + 1) / 25))
If StartCassNo EQ 0 then StartCassNo = 1
SecondEventNumCass = SRP_Math('CEILING', (SecondEventWfrQty / 25))
FirstRDS = ''
If ReactType EQ 'EPP' then
// Example: Event starts on WM_IN cassette 6 (i.e. 172125*1*6), slot 13.
StartSlotNo = Mod(PrevSchedWfrQty, 25) + 1
StopCassNo = StartCassNo + SecondEventNumCass - 1
StopSlotNo = Mod((PrevSchedWfrQty + SecondEventWfrQty), 25)
If StopSlotNo EQ 0 then StopSlotNo = 25
For CassIndex = 1 to SecondEventNumCass
CassNo = (StartCassNo + CassIndex - 1)
WMIKey = ThisEventWONo:'*1*':CassNo
WMIRdsKeys = Xlate('WM_IN', WMIKey, 'RDS_NO', 'X')
EventCassRDSKeys = ''
Begin Case
Case CassNo EQ StartCassNo
StartIndex = StartSlotNo
StopIndex = 25
Case CassNo EQ StopCassNo
StartIndex = 1
StopIndex = StopSlotNo
Case ( (CassNo EQ StartCassNo) and (CassNo EQ StopCassNo) )
StartIndex = StartSlotNo
StopIndex = StopSlotNo
Case Otherwise$
StartIndex = 1
StopIndex = 25
End Case
For SlotIndex = StartIndex to StopIndex
If FirstRDS EQ '' then FirstRDS = WMIRdsKeys<0, SlotIndex>
EventCassRDSKeys<0, -1> = WMIRdsKeys<0, SlotIndex>
Next SlotIndex
EventCassRDSKeys = SRP_Array('Clean', EventCassRDSKeys, 'TrimAndMakeUnique', @VM)
// Now we have a list of RDS keys that only pertain to this event.
If EventCassRDSKeys NE '' then
// Are all of the RDS runs unloaded?
NumUnloadedRDS = 0
UnloadDtms = Xlate('RDS', EventCassRDSKeys, 'DATETIME_OUT', 'X')
If UnloadDtms NE '' then
For each UnloadDtm in UnloadDtms using @VM
If UnloadDtm NE '' then NumUnloadedRds += 1
Next UnloadDtm
end
NumRDS = DCount(EventCassRDSKeys, @VM)
CassProcessed = (NumRds EQ NumUnloadedRds)
end else
CassProcessed = False$
end
If CassProcessed then
SecondEventProcCassList<0, -1> = CassNo
end else
SecondEventUnprocCassList<0, -1> = CassNo
end
Next CassIndex
SecondEventProcCassList = SRP_Array('Join', SecondEventProcCassList, SecondEventUnprocCassList, 'NOT', @VM)
end else
// Example: Event starts on WO_MAT cassette 7. We don't need to schedule down to the
// wafer levels with non-epipro work order events.
For CassIndex = 1 to SecondEventNumCass
CassNo = (StartCassNo + CassIndex - 1)
WOMatKey = ThisEventWONo:'*':CassNo
RDSNo = Xlate('WO_MAT', WOMatKey, 'RDS_NO', 'X')
If FirstRDS EQ '' then FirstRDS = RdsNo
UnloadDtm = Xlate('RDS', RDSNo, 'DATETIME_OUT', 'X')
CassProcessed = (UnloadDtm NE '')
If CassProcessed then
SecondEventProcCassList<0, -1> = CassNo
end else
SecondEventUnprocCassList<0, -1> = CassNo
end
Next CassIndex
end
SecondEventStartDtm = HBEventStopDtm
If FirstRDS NE '' then
FirstLoadDtm = Xlate('RDS', FirstRDS, 'DATETIME_IN', 'X')
If FirstLoadDtm GT HBEventStopDTM then
SecondEventStartDtm = FirstLoadDtm
end
end
SecondEventKey = RTI_CreateGUID()
SecondEventRec = ''
NewEventDuration = Schedule_Services('GetEventDuration', ThisEventWONo, SecondEventWfrQty)
SecondEventRec<SCHED_DET_NG.REACT_NO$> = ThisEventRec<SCHED_DET_NG.REACT_NO$>
SecondEventRec<SCHED_DET_NG.WO_NO$> = ThisEventRec<SCHED_DET_NG.WO_NO$>
SecondEventRec<SCHED_DET_NG.START_DTM$> = SecondEventStartDTM
SecondEventRec<SCHED_DET_NG.STOP_DTM$> = SecondEventStartDTM + NewEventDuration
SecondEventRec<SCHED_DET_NG.DESC$> = ThisEventRec<SCHED_DET_NG.DESC$>
SecondEventRec<SCHED_DET_NG.EVENT_TOTAL_WFRS$> = SecondEventWfrQty
SecondEventRec<SCHED_DET_NG.PROCESSED_CASS$> = SecondEventProcCassList
SecondEventRec<SCHED_DET_NG.UNPROCESSED_CASS$> = SecondEventUnprocCassList
Database_Services('WriteDataRow', 'SCHED_DET_NG', SecondEventKey, SecondEventRec)
// Add second split event to event list
NewEventIndex = EventIndex + 2
NewReactEventList = Insert(NewReactEventList, 0, NewEventIndex, 0, SecondEventKey)
LogData = ''
LogData<1> = OConv(Datetime(), 'DT2/^H')
LogData<2> = '5.':vPos:'.2.9'
LogData<3> = 'Second split event details:'
Schedule_Services('LogActivity', ReactNo, LogData)
LogData<3> = 'Event work order: ':SecondEventRec<SCHED_DET_NG.WO_NO$>
Schedule_Services('LogActivity', ReactNo, LogData)
Logdata<3> = 'Event start dtm: ':OConv(SecondEventRec<SCHED_DET_NG.START_DTM$>, 'DT2/^H')
Schedule_Services('LogActivity', ReactNo, LogData)
LogData<3> = 'Event stop dtm: ':OConv(SecondEventRec<SCHED_DET_NG.STOP_DTM$>, 'DT2/^H')
Schedule_Services('LogActivity', ReactNo, LogData)
LogData<3> = 'Event quantity: ':SecondEventRec<SCHED_DET_NG.EVENT_TOTAL_WFRS$>
Schedule_Services('LogActivity', ReactNo, LogData)
LogData<3> = 'Event complete: ':OConv(SecondEventRec<SCHED_DET_NG.EVENT_COMP$>, "Btrue,false")
Schedule_Services('LogActivity', ReactNo, LogData)
end
// Adjust downstream events
LogData = ''
LogData<1> = OConv(Datetime(), 'DT2/^H')
LogData<2> = '6.':vPos
LogData<3> = 'Begin adjusting downstream events.'
Schedule_Services('LogActivity', ReactNo, LogData)
NumSchedEvents = DCount(NewReactEventList, @VM)
For AdjustmentIndex = (EventIndex + 3) to NumSchedEvents
AdjustEventKey = NewReactEventList<0, AdjustmentIndex>
AdjustEventRec = Database_Services('ReadDataRow', 'SCHED_DET_NG', AdjustEventKey)
AdjustEventWONo = AdjustEventRec<SCHED_DET_NG.WO_NO$>
AdjustEventWfrQty = AdjustEventRec<SCHED_DET_NG.EVENT_TOTAL_WFRS$>
PrevEventKey = NewReactEventList<0, AdjustmentIndex - 1>
PrevEventStopDTM = Xlate('SCHED_DET_NG', PrevEventKey, 'STOP_DTM', 'X')
// Set new start and stop DTMs
AdjustEventStartDTM = PrevEventStopDTM
AdjustEventDuration = Schedule_Services('GetEventDuration', AdjustEventWONo, AdjustEventWfrQty)
AdjustEventStopDTM = AdjustEventStartDTM + AdjustEventDuration
AdjustEventRec<SCHED_DET_NG.START_DTM$> = AdjustEventStartDTM
AdjustEventRec<SCHED_DET_NG.STOP_DTM$> = AdjustEventStopDTM
Database_Services('WriteDataRow', 'SCHED_DET_NG', AdjustEventKey, AdjustEventRec, True$, False$, True$)
LogData<1> = OConv(Datetime(), 'DT2/^H')
LogData<2> = '6.':vPos:'.':AdjustmentIndex
LogData<3> = 'Event ':AdjustEventKey:' adjusted.'
Schedule_Services('LogActivity', ReactNo, LogData)
LogData<3> = 'New start dtm ':OConv(AdjustEventStartDtm, 'DT2/^H'):'.'
Schedule_Services('LogActivity', ReactNo, LogData)
LogData<3> = 'New stop dtm ':OConv(AdjustEventStopDtm, 'DT2/^H'):'.'
Schedule_Services('LogActivity', ReactNo, LogData)
Next AdjustmentIndex
end else
// No time conflict, so just insert hard block event
HBEventIndex = EventIndex + 1
NewReactEventList = Insert(NewReactEventList, 0, HBEventIndex, 0, HBEventKey)
LogData = ''
LogData<1> = OConv(Datetime(), 'DT2/^H')
LogData<2> = '5.':vPos:'.2.2'
LogData<3> = 'No time conflict found between event ':ThisEventKey:' and ':HBEventKey:'.'
Schedule_Services('LogActivity', ReactNo, LogData)
LogData<3> = 'Inserting hard block without splitting.'
Schedule_Services('LogActivity', ReactNo, LogData)
end
end
end
EventIndex += StepInc
Until ( HBInserted or (EventIndex LT 0) or (EventIndex GT NumSchedEvents) )
Repeat
Next HBEventKey
end else
LogData = ''
LogData<1> = OConv(Datetime(), 'DT2/^H')
LogData<2> = '5'
LogData<3> = 'No hard block events to reschedule.'
Schedule_Services('LogActivity', ReactNo, LogData)
end
end
LogData = ''
LogData<1> = OConv(Datetime(), 'DT2/^H')
LogData<2> = '6'
LogData<3> = 'Begin cleaning out "zombie events".'
Schedule_Services('LogActivity', ReactNo, LogData)
NumEvents = DCount(NewReactEventList, @VM)
For EventIndex = 1 to NumEvents
EventID = NewReactEventList<0, EventIndex>
EventRec = Database_Services('ReadDataRow', 'SCHED_DET_NG', EventID)
EventWONo = EventRec<SCHED_DET_NG.WO_NO$>
EventBlockOut = EventRec<SCHED_DET_NG.BLOCK_OUT$>
If ( (EventWONo EQ '') and (EventBlockOut NE True$) ) then
// This is a zombie, so delete it (there's a bug up above somewhere creating or not
// deleting events as necessary.
NewReactEventList = Delete(NewReactEventList, 0, EventIndex, 0)
Database_Services('DeleteDataRow', 'SCHED_DET_NG', EventID, True$, False$)
LogData = ''
LogData<1> = OConv(Datetime(), 'DT2/^H')
LogData<2> = '6.':EventIndex
LogData<3> = 'Zombie event ':EventID:' found and deleted.'
Schedule_Services('LogActivity', ReactNo, LogData)
end
Next EventIndex
ReactNo = OrigReactNo
// Update reactor event list
If ( (NewReactEventList NE '') and (NewReactEventList NE OrigReactEventList) ) then
ReactRec = Database_Services('ReadDataRow', 'REACTOR', ReactNo)
// Clean out duplicate event keys (there's a bug up above somewhere creating them)
NewReactEventList = SRP_Array('Clean', NewReactEventList, 'TrimAndMakeUnique', @VM)
ReactRec<REACTOR_SCHED_EVENTS$> = NewReactEventList
Database_Services('WriteDataRow', 'REACTOR', ReactNo, ReactRec, True$, False$, True$)
LogData = ''
LogData<1> = OConv(Datetime(), 'DT2/^H')
LogData<2> = '7'
LogData<3> = 'Writing new event list for reactor ':ReactNo:'.'
Schedule_Services('LogActivity', ReactNo, LogData)
end
Schedule_Services('SendRefreshMessage')
Unlock hSysLists, ServiceKeyID:'*':ReactNo else Null
end
end service
//----------------------------------------------------------------------------------------------------------------------
// AddSchedEvent
//
// Saves the event to the database.
//----------------------------------------------------------------------------------------------------------------------
Service AddSchedEvent(ReactorNo, WorkOrderNo, StartDTM, StopDTM, Description, WaferQty)
// Create a unique key ID for the new event
NewEventKeyID = RTI_CreateGUID()
NewEventRec = ''
NewEventRec<SCHED_DET_NG.REACT_NO$> = ReactorNo
NewEventRec<SCHED_DET_NG.WO_NO$> = WorkOrderNo
NewEventRec<SCHED_DET_NG.START_DTM$> = StartDTM
NewEventRec<SCHED_DET_NG.DESC$> = Description
NewEventRec<SCHED_DET_NG.EVENT_TOTAL_WFRS$> = WaferQty
EpiPro = (Xlate('WO_LOG', WorkOrderNo, 'REACT_TYPE', 'X') EQ 'EPP')
CurrDTM = Datetime()
EventProcCassList = ''
EventUnprocCassList = ''
LastLoadedRDS = ''
LastUnloadedRDS = ''
// Determine start cassette and slot number
PrevSchedQty = Schedule_Services('GetScheduledWfrQty', WorkOrderNo)
StartCass = SRP_Math('CEILING', ( (PrevSchedQty + 1) / 25))
If StartCass EQ 0 then StartCass = 1
StartSlot = Mod(PrevSchedQty, 25) + 1
TotalSchedQty = PrevSchedQty + WaferQty
StopCass = SRP_Math('CEILING', ( TotalSchedQty / 25 ) )
StopSlot = Mod(TotalSchedQty, 25)
If StopSlot EQ 0 then StopSlot = 25
// Get list of RDS keys associated with this event if they exist
RDSKeys = ''
If EpiPro then
For CassNo = StartCass to StopCass
WMIKey = WorkOrderNo:'*1*':CassNo
WMIRdsList = Xlate('WM_IN', WMIKey, 'RDS_NO', 'X')
If WMIRdsList NE '' then
Begin Case
Case ( (CassNo EQ StartCass) and (CassNo EQ StopCass) )
StartIndex = StartSlot
StopIndex = StopSlot
Case CassNo EQ StartCass
StartIndex = StartSlot
StopIndex = 25
Case CassNo EQ StopCass
StartIndex = 1
StopIndex = StopSlot
Case Otherwise$
StartIndex = 1
StopIndex = 25
End Case
For SlotIndex = StartIndex to StopIndex
RDSKeys<0, -1> = WMIRdsList<0, SlotIndex>
Next SlotIndex
end else
// No RDS keys exist for this cassette yet, so it's unprocessed
EventUnprocCassList<0, -1> = CassNo
end
Next CassNo
If RDSKeys NE '' then
RDSKeys = SRP_Array('Clean', RDSKeys, 'TrimAndMakeUnique', @VM)
For each RDSKey in RDSKeys using @VM setting vPos
DtmOut = Xlate('RDS', RDSKey, 'DATETIME_OUT', 'X')
WMIKeys = Xlate('RDS', RDSKey, 'WM_IN_KEY', 'X')
For each WMIKey in WMIKeys using @VM
CassNo = Field(WMIKey, '*', 3)
If DtmOut NE '' then
LastUnloadedRDS = RDSKey
Locate CassNo in EventProcCassList using @VM setting Dummy else
EventProcCassList<0, -1> = CassNo
end
end else
Locate CassNo in EventUnprocCassList using @VM setting Dummy else
EventUnprocCassList<0, -1> = CassNo
end
end
Next WMIKey
Next RDSKey
end
EventUnprocCassList = SRP_Array('SortSimpleList', EventUnprocCassList, 'AscendingNumbers', @VM)
// This line ensures EpiPro cassettes are not recorded as both processed and unprocessed.
EventProcCassList = SRP_Array('Join', EventProcCassList, EventUnprocCassList, 'NOT', @VM)
end else
For CassNo = StartCass to StopCass
WOMatKey = WorkOrderNo:'*':CassNo
RDSKey = Xlate('WO_MAT', WOMatKey, 'RDS_NO', 'X')
DtmOut = Xlate('RDS', RDSKey, 'DATETIME_OUT', 'X')
If DtmOut NE '' then
LastUnloadedRDS = RDSKey
Locate CassNo in EventProcCassList using @VM setting Dummy else
EventProcCassList<0, -1> = CassNo
end
end else
Locate CassNo in EventUnprocCassList using @VM setting Dummy else
EventUnprocCassList<0, -1> = CassNo
end
end
Next CassNo
end
EventComp = (EventUnprocCassList EQ '')
If EventComp then StopDtm = Xlate('RDS', LastUnloadedRDS, 'DATETIME_OUT', 'X')
// Update event
NewEventRec<SCHED_DET_NG.EVENT_COMP$> = EventComp
NewEventRec<SCHED_DET_NG.PROCESSED_CASS$> = EventProcCassList
NewEventRec<SCHED_DET_NG.UNPROCESSED_CASS$> = EventUnprocCassList
NewEventRec<SCHED_DET_NG.STOP_DTM$> = StopDtm
Database_Services('WriteDataRow', 'SCHED_DET_NG', NewEventKeyID, NewEventRec)
If Error_Services('NoError') then
// Determine where to insert the event in the REACTOR record linked list
// Add the event to the REACTOR record event list
ReactRec = Database_Services('ReadDataRow', 'REACTOR', ReactorNo)
ReactSchedEvents = ReactRec<REACTOR_SCHED_EVENTS$>
CurrNumEvents = DCount(ReactSchedEvents, @VM)
// Add our new event key ID to the reactor event list
NewNumEvents = CurrNumEvents + 1
InsertPos = ''
For EventIndex = 1 to CurrNumEvents Step 1
ThisEventKey = ReactSchedEvents<0, EventIndex>
ThisStartDTM = Xlate('SCHED_DET_NG', ThisEventKey, 'START_DTM', 'X')
IF ThisStartDTM GT StartDTM then
// We found our location
InsertPos = EventIndex
end
Until InsertPos NE ''
Next EventIndex
If InsertPos EQ '' then InsertPos = NewNumEvents
ReactSchedEvents = Insert(ReactSchedEvents, 0, InsertPos, 0, NewEventKeyID)
ReactRec<REACTOR_SCHED_EVENTS$> = ReactSchedEvents
Database_Services('WriteDataRow', 'REACTOR', ReactorNo, ReactRec)
// Check if there is an overlap with the previous event. If so, then set previous event's stop DTM to this event's
// start DTM.
If NewNumEvents GT 1 then
PrevEventKey = ReactSchedEvents<0, (NewNumEvents - 1)>
If PrevEventKey NE '' then
PrevEventRec = Database_Services('ReadDataRow', 'SCHED_DET_NG', PrevEventKey)
PrevEventStopDTM = PrevEventRec<SCHED_DET_NG.STOP_DTM$>
If PrevEventStopDTM GT StartDTM then
PrevEventStopDTM = StartDTM
PrevEventRec<SCHED_DET_NG.STOP_DTM$> = PrevEventStopDTM
Database_Services('WriteDataRow', 'SCHED_DET_NG', PrevEventKey, PrevEventRec, True$, False$, True$)
end
end
end
// Record schedule change in SCHED_HIST table
Schedule_Services('AddSchedHist', NewEventKeyID, ReactorNo, 'ADD', @User4)
Response = NewEventKeyID
end
end service
Service AddSchedHist(SchedDetNGKey, ReactorNo, Action=ACTIONS, User)
Error = ''
If ( (SchedDetNGKey NE '') and (ReactorNo NE '') and (Action NE '') and (User NE '') ) then
SchedHistRec = ''
SchedHistRec<SCHED_HIST.SCHED_DET_NG_KEY$> = SchedDetNGKey
SchedHistRec<SCHED_HIST.REACTOR$> = ReactorNo
SchedHistRec<SCHED_HIST.ACTION$> = Action
SchedHistRec<SCHED_HIST.USER$> = User
SchedHistKey = NextKey('SCHED_HIST')
If SchedHistKey NE '' then
Database_Services('WriteDataRow', 'SCHED_HIST', SchedHistKey, SchedHistRec)
If Error_Services('HasError') then Error = Error_Services('GetMessage')
end else
Error = 'Error in ':Service:' service. Error retrieving new SCHED_HIST key.'
end
end else
Error = 'Error in ':Service:' service. Null parameter passed in.'
end
If Error NE '' then Error_Services('Add', Error)
end service
//----------------------------------------------------------------------------------------------------------------------
// ModifySchedEvent
//
// Modifies an event's details.
//----------------------------------------------------------------------------------------------------------------------
Service ModifySchedEvent(SchedDetNGKey, ReactorNo, WorkOrderNo, StartDTM, StopDTM, Description, WaferQty, AdjustNextEvent=BOOLEAN)
Response = False$
If SchedDetNGKey NE '' then
EventRec = Database_Services('ReadDataRow', 'SCHED_DET_NG', SchedDetNGKey)
If Error_Services('NoError') then
If AdjustNextEvent EQ '' then AdjustNextEvent = True$
CurrDTM = Datetime()
If ReactorNo NE '' then EventRec<SCHED_DET_NG.REACT_NO$> = ReactorNo
If WorkOrderNo NE '' then EventRec<SCHED_DET_NG.WO_NO$> = WorkOrderNo
If StartDTM NE '' then EventRec<SCHED_DET_NG.START_DTM$> = StartDTM
If StopDTM NE '' then EventRec<SCHED_DET_NG.STOP_DTM$> = StopDTM
If Description NE '' then EventRec<SCHED_DET_NG.DESC$> = Description
OrigWfrQty = EventRec<SCHED_DET_NG.EVENT_TOTAL_WFRS$>
// If the user is just changing the description the wafer quantity may not be passed in.
If WaferQty EQ '' then WaferQty = OrigWfrQty
If OrigWfrQty NE WaferQty then
ReactorNo = EventRec<SCHED_DET_NG.REACT_NO$>
WorkOrderNo = EventRec<SCHED_DET_NG.WO_NO$>
StartDTM = EventRec<SCHED_DET_NG.START_DTM$>
StopDTM = EventRec<SCHED_DET_NG.STOP_DTM$>
EpiPro = ( Xlate('REACTOR', ReactorNo, 'REACT_TYPE', 'X') EQ 'EPP' )
EventProcCassList = ''
EventUnprocCassList = ''
LastLoadedRDS = ''
LastUnloadedRDS = ''
// Determine start cassette and slot number
PrevSchedQty = Schedule_Services('GetScheduledWfrQty', WorkOrderNo, StartDtm)
StartCass = SRP_Math('CEILING', ( (PrevSchedQty + 1) / 25))
If StartCass EQ 0 then StartCass = 1
StartSlot = Mod(PrevSchedQty, 25) + 1
TotalSchedQty = PrevSchedQty + WaferQty
StopCass = SRP_Math('CEILING', ( TotalSchedQty / 25 ) )
StopSlot = Mod(TotalSchedQty, 25)
If StopSlot EQ 0 then StopSlot = 25
// Get list of RDS keys associated with this event if they exist
RDSKeys = ''
If EpiPro then
For CassNo = StartCass to StopCass
WMIKey = WorkOrderNo:'*1*':CassNo
WMIRdsList = Xlate('WM_IN', WMIKey, 'RDS_NO', 'X')
If WMIRdsList NE '' then
Begin Case
Case ( (CassNo EQ StartCass) and (CassNo EQ StopCass) )
StartIndex = StartSlot
StopIndex = StopSlot
Case CassNo EQ StartCass
StartIndex = StartSlot
StopIndex = 25
Case CassNo EQ StopCass
StartIndex = 1
StopIndex = StopSlot
Case Otherwise$
StartIndex = 1
StopIndex = 25
End Case
For SlotIndex = StartIndex to StopIndex
RDSKeys<0, -1> = WMIRdsList<0, SlotIndex>
Next SlotIndex
end else
// No RDS keys exist for this cassette yet, so it's unprocessed
EventUnprocCassList<0, -1> = CassNo
end
Next CassNo
If RDSKeys NE '' then
RDSKeys = SRP_Array('Clean', RDSKeys, 'TrimAndMakeUnique', @VM)
For each RDSKey in RDSKeys using @VM setting vPos
DtmOut = Xlate('RDS', RDSKey, 'DATETIME_OUT', 'X')
WMIKeys = Xlate('RDS', RDSKey, 'WM_IN_KEY', 'X')
For each WMIKey in WMIKeys using @VM
CassNo = Field(WMIKey, '*', 3)
If DtmOut NE '' then
LastUnloadedRDS = RDSKey
Locate CassNo in EventProcCassList using @VM setting Dummy else
EventProcCassList<0, -1> = CassNo
end
end else
Locate CassNo in EventUnprocCassList using @VM setting Dummy else
EventUnprocCassList<0, -1> = CassNo
end
end
Next WMIKey
Next RDSKey
end
EventUnprocCassList = SRP_Array('SortSimpleList', EventUnprocCassList, 'AscendingNumbers', @VM)
// This line ensures EpiPro cassettes are not recorded as both processed and unprocessed.
EventProcCassList = SRP_Array('Join', EventProcCassList, EventUnprocCassList, 'NOT', @VM)
end else
For CassNo = StartCass to StopCass
WOMatKey = WorkOrderNo:'*':CassNo
RDSKey = Xlate('WO_MAT', WOMatKey, 'RDS_NO', 'X')
DtmOut = Xlate('RDS', RDSKey, 'DATETIME_OUT', 'X')
If DtmOut NE '' then
LastUnloadedRDS = RDSKey
Locate CassNo in EventProcCassList using @VM setting Dummy else
EventProcCassList<0, -1> = CassNo
end
end else
Locate CassNo in EventUnprocCassList using @VM setting Dummy else
EventUnprocCassList<0, -1> = CassNo
end
end
Next CassNo
end
EventComp = (EventUnprocCassList EQ '')
If EventComp then StopDtm = Xlate('RDS', LastUnloadedRDS, 'DATETIME_OUT', 'X')
// Update event
EventRec<SCHED_DET_NG.EVENT_COMP$> = EventComp
EventRec<SCHED_DET_NG.PROCESSED_CASS$> = EventProcCassList
EventRec<SCHED_DET_NG.UNPROCESSED_CASS$> = EventUnprocCassList
EventRec<SCHED_DET_NG.STOP_DTM$> = StopDtm
EventRec<SCHED_DET_NG.EVENT_TOTAL_WFRS$> = WaferQty
Database_Services('WriteDataRow', 'SCHED_DET_NG', SchedDetNGKey, EventRec, True$, False$, True$)
If Error_Services('NoError') then
Schedule_Services('AddSchedHist', SchedDetNGKey, ReactorNo, 'MODIFY', @User4)
Response = True$
end
If AdjustNextEvent then
// Adjust downstream events if they share the same work order. If the above event increased its
// wafer quantity then the next event will be reduced by the delta and vice versa.
ModifyWfrQty = WaferQty - OrigWfrQty
Query = 'SELECT SCHED_DET_NG WITH WO_NO EQ ':WorkOrderNo:' AND WITH START_DTM GT '
Query := Quote(OConv(StartDtm, 'DT/^3H')):' BY START_DTM'
GoSub ClearCursors
Set_Status(0)
RList(Query, TARGET_ACTIVELIST$, '', '', '')
errCode = ''
Done = False$
DownStreamKey = ''
If Get_Status(errCode) then
ErrorMsg = 'Error retrieving event list from the ':Service:' module. Error code':errCode
Error_Services('Add', ErrorMsg)
end else
EOF = False$
ReadNext DownStreamKey else EOF = True$
If ( Not(EOF) and (DownStreamKey NE SchedDetNGKey) ) then
DownStreamRec = Database_Services('ReadDataRow', 'SCHED_DET_NG', DownStreamKey)
OrigDownStreamQty = DownStreamRec<SCHED_DET_NG.EVENT_TOTAL_WFRS$>
NewDownStreamQty = OrigDownStreamQty - ModifyWfrQty
Schedule_Services('ModifySchedEvent', DownStreamKey, '', '', '', '', '', NewDownStreamQty, False$)
end
GoSub ClearCursors
end
end
end else
// User is likely just adding a comment
Database_Services('WriteDataRow', 'SCHED_DET_NG', SchedDetNGKey, EventRec, True$, False$, True$)
If Error_Services('NoError') then
Schedule_Services('AddSchedHist', SchedDetNGKey, ReactorNo, 'MODIFY', @User4)
Response = True$
end
end
end
end
end service
Service GetBlockEvents(ReactorNo, StartDTM, StopDTM, BlockGroup=BLOCK_GROUP)
ScheduleDetailKeys = ''
If Not(Num(StartDTM)) then StartDTM = Iconv(StartDTM, 'DT')
If Not(Num(StopDTM)) then StopDTM = Iconv(StopDTM, 'DT')
GoSub ClearCursors
Set_Status(0)
Query = ''
If ReactorNo NE '' then
Query = 'SELECT SCHED_DET_NG WITH REACT_NO EQ ':ReactorNo
RList(Query, TARGET_ACTIVELIST$, '', '', '')
end
Begin Case
Case ((StartDTM NE '') and (StopDTM NE ''))
Query = 'SELECT SCHED_DET_NG WITH START_DTM BETWEEN ':Quote(OConv(StartDTM, 'DT/^3H')):' AND ':Quote(OConv(StopDTM, 'DT/^3H')):' OR '
Query := 'WITH STOP_DTM BETWEEN ':Quote(OConv(StartDTM, 'DT/^3H')):' AND ':Quote(OConv(StopDTM, 'DT/^3H'))
RList(Query, TARGET_ACTIVELIST$, '', '', '')
Case ((StartDTM NE '') and (StopDTM EQ ''))
Query = 'SELECT SCHED_DET_NG WITH START_DTM GE ':Quote(OConv(StartDTM, 'DT/^3H'))
RList(Query, TARGET_ACTIVELIST$, '', '', '')
Case ((StartDTM EQ '') and (StopDTM NE ''))
Query = 'SELECT SCHED_DET_NG WITH STOP_DTM LE ':Quote(OConv(StopDTM, 'DT/^3H'))
RList(Query, TARGET_ACTIVELIST$, '', '', '')
Case ((StartDTM EQ '') and (StopDTM EQ ''))
Null
End Case
Begin Case
Case BlockGroup _EQC 'SOFT_BLOCK'
Query = 'SELECT SCHED_DET_NG WITH SOFT_BLOCK EQ ':True$
RList(Query, TARGET_ACTIVELIST$, '', '', '')
Case BlockGroup _EQC 'HARD_BLOCK'
Query = 'SELECT SCHED_DET_NG WITH HARD_BLOCK EQ ':True$
RList(Query, TARGET_ACTIVELIST$, '', '', '')
Case Otherwise$
// Get all block events
Query = 'SELECT SCHED_DET_NG WITH BLOCK_OUT EQ ':True$
RList(Query, TARGET_ACTIVELIST$, '', '', '')
End Case
errCode = ''
If Get_Status(errCode) then
ErrorMsg = 'Error retrieving event list from the ':Service:' module. Error code':errCode
Error_Services('Add', ErrorMsg)
end else
EOF = False$
Loop
ReadNext SchedDetNGKey Else EOF = True$
Until EOF EQ True$
ScheduleDetailKeys<-1> = SchedDetNGKey
Repeat
end
GoSub ClearCursors
Response = ScheduleDetailKeys
end service
Service AddBlockOutEvent(ReactorNo, StartDTM, StopDTM, BlockType, Description, Options)
// Create a unique key ID for the new event
NewEventKeyID = RTI_CreateGUID()
NewEventRec = ''
NewEventRec<SCHED_DET_NG.REACT_NO$> = ReactorNo
NewEventRec<SCHED_DET_NG.START_DTM$> = StartDTM
NewEventRec<SCHED_DET_NG.STOP_DTM$> = StopDTM
NewEventRec<SCHED_DET_NG.DESC$> = Description
NewEventRec<SCHED_DET_NG.BLOCK_OUT$> = True$
NewEventRec<SCHED_DET_NG.BLOCK_OUT_TYPE$> = BlockType
// Determine where to insert the event in the REACTOR record linked list
// Add the event to the REACTOR record linked list
ReactRec = Database_Services('ReadDataRow', 'REACTOR', ReactorNo)
ReactSchedEvents = ReactRec<REACTOR_SCHED_EVENTS$>
CurrNumEvents = DCount(ReactSchedEvents, @VM)
ReactRec = Database_Services('ReadDataRow', 'REACTOR', ReactorNo)
ReactSchedEvents = ReactRec<REACTOR_SCHED_EVENTS$>
CurrNumEvents = DCount(ReactSchedEvents, @VM)
// Add our new event key ID to the reactor event list
NewNumEvents = CurrNumEvents + 1
InsertPos = ''
For EventIndex = 1 to CurrNumEvents Step 1
ThisEventKey = ReactSchedEvents<0, EventIndex>
ThisStartDTM = Xlate('SCHED_DET_NG', ThisEventKey, 'START_DTM', 'X')
IF ThisStartDTM GT StartDTM then
// We found our location
InsertPos = EventIndex
end
Until InsertPos NE ''
Next EventIndex
If InsertPos EQ '' then InsertPos = NewNumEvents
ReactSchedEvents = Insert(ReactSchedEvents, 0, InsertPos, 0, NewEventKeyID)
ReactRec<REACTOR_SCHED_EVENTS$> = ReactSchedEvents
Database_Services('WriteDataRow', 'REACTOR', ReactorNo, ReactRec)
Database_Services('WriteDataRow', 'SCHED_DET_NG', NewEventKeyID, NewEventRec)
// Record schedule change in SCHED_HIST table
Schedule_Services('AddSchedHist', NewEventKeyID, ReactorNo, 'ADD', @User4)
Response = NewEventKeyID
end service
//----------------------------------------------------------------------------------------------------------------------
// GetScheduleDetail
//
// Returns the SCHED_DET_NG row for the indicated Key ID.
//----------------------------------------------------------------------------------------------------------------------
Service GetScheduleDetail(ScheduleKeyID)
ScheduleDetail = ''
TableName = 'SCHED_DET_NG'
If ScheduleKeyID NE '' then
ScheduleDetail = Database_Services('ReadDataRow', Tablename, ScheduleKeyID)
end else
Error_Services('Add', 'ScheduleKeyID argument was missing from the ' : Service : ' service.')
end
Response = ScheduleDetail
end service
//----------------------------------------------------------------------------------------------------------------------
// SetScheduleDetail
//
// Creates or updates the SCHED_DET_NG row for the indicated Key ID.
//----------------------------------------------------------------------------------------------------------------------
Service SetScheduleDetail(ScheduleKeyID, ScheduleDetail)
If (ScheduleKeyID NE '') AND (ScheduleDetail NE '') then
TableName = 'SCHED_DET_NG'
Database_Services('WriteDataRow', Tablename, ScheduleKeyID, ScheduleDetail, True$)
end else
Error_Services('Add', 'ScheduleKeyID or ScheduleDetail argument was missing from the ' : Service : ' service.')
end
end service
//----------------------------------------------------------------------------------------------------------------------
// DeleteScheduleDetail
//
// Deletes the SCHED_DET_NG row for the indicated Key IDs.
//----------------------------------------------------------------------------------------------------------------------
Service DeleteScheduleDetail(ScheduleKeyIDs)
If ScheduleKeyIDs NE '' then
TableName = 'SCHED_DET_NG'
For Each ScheduleKeyID in ScheduleKeyIDs using @FM
ReactNo = Xlate(TableName, ScheduleKeyID, 'REACT_NO', 'X')
ReactRec = Database_Services('ReadDataRow', 'REACTOR', ReactNo)
ReactEvents = ReactRec<REACTOR_SCHED_EVENTS$>
Locate ScheduleKeyID in ReactEvents using @VM setting vPos then
ReactEvents = Delete(ReactEvents, 0, vPos, 0)
ReactRec<REACTOR_SCHED_EVENTS$> = ReactEvents
Database_Services('WriteDataRow', 'REACTOR', ReactNo, ReactRec, True$, False$, True$)
end
Database_Services('DeleteDataRow', Tablename, ScheduleKeyID, True$)
Next ScheduleKeyID
end else
Error_Services('Add', 'ScheduleKeyID argument was missing from the ' : Service : ' service.')
end
end service
//----------------------------------------------------------------------------------------------------------------------
// GetAdjustedDays
//
// Returns the calculated adjusted number of days needed to process the remaining wafers for the indicated work order.
// Note: this can return a negative value if there are more scheduled events needed to complete the remaining wafers.
//----------------------------------------------------------------------------------------------------------------------
Service GetAdjustedDays(WorkOrderNo, ReactorNo, EventWafers)
DaysNeeded = ''
If WorkOrderNo NE '' then
WorkOrderDetail = Work_Order_Services('GetWorkOrderLogDetail', WorkOrderNo)
If Error_Services('NoError') then
If EventWafers EQ '' then
// Get estimate for all remaining wafers in the work order.
EventWafers = Work_Order_Services('GetRemainingWafers', WorkOrderNo)
end
EpiPartNo = WorkOrderDetail<WO_LOG_EPI_PART_NO$>
TotalWafers = WorkOrderDetail<WO_LOG_WO_QTY$>
WOReactorType = WorkOrderDetail<WO_LOG_REACT_TYPE$>
WOClosedFlag = WorkOrderDetail<WO_LOG_CLOSE_DATE$> NE ''
Begin Case
Case WOReactorType _eqc 'ASM+' ; WOReactorType = 'ASM+'
Case WOReactorType _eqc 'EPP' ; WOReactorType = 'EpiPro'
Case WOReactorType _eqc 'EPS' ; WOReactorType = 'ASM'
Case WOReactorType _eqc 'HTR' ; WOReactorType = 'HTR'
Case WOReactorType _eqc 'GAN' ; WOReactorType = 'GaN'
Case WOReactorType _eqc '' ; WOReactorType = '***'
End Case
WafersPerDay = Epi_Part_Services('GetAdjustedWafersPerDayScheduler', EpiPartNo, WOReactorType)
If ( (WafersPerDay NE 0) and (WafersPerDay NE '') ) then
DaysNeeded = (EventWafers / WafersPerDay)
end
end else
GoSub LogError
end
end else
Error_Services('Add', 'WorkOrderNo argument was missing from the ' : Service : ' service.')
GoSub LogError
end
Response = DaysNeeded
end service
Service GetEventAdjustment(SchedDetKey)
DaysNeeded = ''
If SchedDetKey NE '' then
SchedDetRec = Schedule_Services('GetScheduleDetail', SchedDetKey)
BlockOut = SchedDetRec<SCHED_DET_NG.BLOCK_OUT$>
If BlockOut EQ True$ then
// Simply return the original duration of the event (i.e. there is no adjustment)
EventStartDTM = SchedDetRec<SCHED_DET_NG.START_DTM$>
EventStopDTM = SchedDetRec<SCHED_DET_NG.STOP_DTM$>
DaysNeeded = EventStopDTM - EventStartDTM
end else
// Calculate new duration necessary to complete event.
WorkOrderNo = SchedDetRec<SCHED_DET_NG.WO_NO$>
WorkOrderDetail = Work_Order_Services('GetWorkOrderLogDetail', WorkOrderNo)
If Error_Services('NoError') then
CurrDTM = Datetime()
EventRemWfrs = Schedule_Services('GetEventRemainingWfrs', SchedDetKey)
If Error_Services('NoError') then
EpiPartNo = WorkOrderDetail<WO_LOG_EPI_PART_NO$>
TotalWafers = WorkOrderDetail<WO_LOG_WO_QTY$>
WOReactorType = WorkOrderDetail<WO_LOG_REACT_TYPE$>
WOClosedFlag = WorkOrderDetail<WO_LOG_CLOSE_DATE$> NE ''
Begin Case
Case WOReactorType _eqc 'ASM+' ; WOReactorType = 'ASM+'
Case WOReactorType _eqc 'EPP' ; WOReactorType = 'EpiPro'
Case WOReactorType _eqc 'EPS' ; WOReactorType = 'ASM'
Case WOReactorType _eqc 'HTR' ; WOReactorType = 'HTR'
Case WOReactorType _eqc 'GAN' ; WOReactorType = 'GaN'
Case WOReactorType _eqc '' ; WOReactorType = '***'
End Case
WafersPerDay = Epi_Part_Services('GetAdjustedWafersPerDayScheduler', EpiPartNo, WOReactorType)
If Error_Services('NoError') then
DaysNeeded = (EventRemWfrs / WafersPerDay)
end else
GoSub LogError
end
end else
GoSub LogError
end
end else
GoSub LogError
end
end
end else
Error_Services('Add', 'SchedDetKey argument was missing from the ' : Service : ' service.')
GoSub LogError
end
Response = DaysNeeded
end service
Service GetWafersPerDayAdjusted(WONo)
If WONo NE '' then
WorkOrderDetail = Work_Order_Services('GetWorkOrderLogDetail', WONo)
If Error_Services('NoError') then
If Error_Services('NoError') then
EpiPartNo = WorkOrderDetail<WO_LOG_EPI_PART_NO$>
WOReactorType = WorkOrderDetail<WO_LOG_REACT_TYPE$>
Begin Case
Case WOReactorType _eqc 'ASM+' ; WOReactorType = 'ASM+'
Case WOReactorType _eqc 'EPP' ; WOReactorType = 'EpiPro'
Case WOReactorType _eqc 'EPS' ; WOReactorType = 'ASM'
Case WOReactorType _eqc 'HTR' ; WOReactorType = 'HTR'
Case WOReactorType _eqc 'GAN' ; WOReactorType = 'GaN'
Case WOReactorType _eqc '' ; WOReactorType = '***'
End Case
WafersPerDay = Epi_Part_Services('GetAdjustedWafersPerDayScheduler', EpiPartNo, WOReactorType)
If Error_Services('NoError') then
Response = WafersPerDay
end else
GoSub LogError
end
end else
GoSub LogError
end
end else
GoSub LogError
end
end else
Error_Services('Add', 'WONo argument was missing from the ' : Service : ' service.')
GoSub LogError
end
end service
Service GetWaferEstimate(WONo, TimeWindow)
If ( (WONo NE '') and (TimeWindow NE '') ) then
WafersPerDayAdjusted = Schedule_Services('GetWafersPerDayAdjusted', WONo)
WaferEstimate = WafersPerDayAdjusted * TimeWindow
Response = WaferEstimate
end
end service
Service GetEventDuration(WONo, WaferQty)
If ( (WONo NE '') and (WaferQty NE '') ) then
WafersPerDayAdjusted = Schedule_Services('GetWafersPerDayAdjusted', WONo)
If ( (WafersPerDayAdjusted NE '') and (WafersPerDayAdjusted GT 0) ) then
TimeEstimate = WaferQty / WafersPerDayAdjusted
Response = TimeEstimate
end else
WorkOrderDetail = Work_Order_Services('GetWorkOrderLogDetail', WONo)
If WorkOrderDetail NE '' then
EpiPartNo = WorkOrderDetail<WO_LOG_EPI_PART_NO$>
ReactType = WorkOrderDetail<WO_LOG_REACT_TYPE$>
ErrorMsg = 'Error in ':Service:' service. Wafers per day value not set for EpiPart ':EpiPartNo:' for reactor type 'ReactType:'.'
end else
ErrorMsg = 'Error in ':Service:' service. Wafers per day value not set for work order ':WONo:'.'
end
end
end
end service
//----------------------------------------------------------------------------------------------------------------------
// AutoScheduler
//
// Analyzes all work orders events for the current date for all reactors. Determines the number of days required for
// each work order to complete the remaining number of wafers. Each work order is adjusted automatically based on the
// adjuted number of days calculated.
//----------------------------------------------------------------------------------------------------------------------
Service AutoScheduler()
HaveLock = Database_Services('GetKeyIDLock', 'APP_INFO', 'AUTO_SCHEDULER_REACTOR')
If HaveLock then
ReactNos = Reactor_Services('GetReactorNumbers')
If Error_Services('NoError') then
LastReactNo = ''
If RowExists('APP_INFO', 'AUTO_SCHEDULER_REACTOR') then
LastReactNo = Database_Services('ReadDataRow', 'APP_INFO', 'AUTO_SCHEDULER_REACTOR')
end
If LastReactNo NE '' then
Locate LastReactNo in ReactNos using @FM setting fPos then
fPos += 1
end
end else
fPos = 1
end
NextReactNo = ReactNos<fPos>
// WriteDataRow service will release the lock.
Database_Services('WriteDataRow', 'APP_INFO', 'AUTO_SCHEDULER_REACTOR', NextReactNo, True$, False$, False$)
Schedule_Services('AdjustScheduleEvents', NextReactNo)
end
end
end service
//----------------------------------------------------------------------------------------------------------------------
// LogActivity
//
// Logs the current activity indicated in the LogData argument. This should be reactor based.
//----------------------------------------------------------------------------------------------------------------------
Service LogActivity(ReactorNo, LogData, EmailNotification=BOOLEAN)
If ReactorNo EQ '' then ReactorNo = 0
LogPath = Environment_Services('GetApplicationRootPath') : '\LogFiles\Scheduler\' : ReactorNo
LogFileName = Date[7, 4] : '-' : Date[1, 2] : '-' : Date[4, 2] : ' Scheduler Log.csv'
objLog = Logging_Services('NewLog', LogPath, LogFilename, CRLF$, ' ', Headers, ColumnWidths, False$, False$)
If EmailNotification then
Recipients = 'DANIEL_ST'
SentFrom = @USER4
Subject = 'Scheduler Notification'
Message = LogData
Swap @FM with ' ' in Message
AttachWindow = ''
AttachKey = ''
SendToGroup = ''
Parms = Recipients:@RM:SentFrom:@RM:Subject:@RM:Message:@RM:AttachWindow:@RM:AttachKey:@RM:SendToGroup
obj_Notes('Create',Parms)
end
Logging_Services('AppendLog', objLog, LogData, @RM, @FM, False$)
end service
Service GetScheduledWfrQty(WorkOrderNo, EndDtm)
ScheduledQty = ''
TableName = "SCHED_DET_NG"
Open "DICT.":TableName To hTableDict then
ScheduledQty = 0
Column = "WO_NO"
QueryParam = WorkOrderNo
Query = Column:@VM:QueryParam:@FM
If EndDtm NE '' then
If Not(Num(EndDtm)) then EndDtm = IConv(EndDtm, 'DT')
Query := "STOP_DTM":@VM:'<=':EndDtm:@FM
end
Keylist = ""
Option = "S"
Flag = ""
Btree.Extract(Query, TableName, hTableDict, Keylist, Option, Flag)
If Flag EQ 0 then
For each KeyID in Keylist using @VM
ScheduledQty += Xlate('SCHED_DET_NG', KeyID, 'EVENT_TOTAL_WFRS', 'X')
Next KeyID
end
end
Response = ScheduledQty
end service
Service GetScheduledEvents(WorkOrderNo)
Keylist = ""
TableName = "SCHED_DET_NG"
Open "DICT.":TableName To hTableDict then
ScheduledQty = 0
Column = "WO_NO"
QueryParam = WorkOrderNo
Query = Column:@VM:QueryParam:@FM
Option = "S"
Flag = ""
Btree.Extract(Query, TableName, hTableDict, Keylist, Option, Flag)
If Flag EQ 0 then
Error_Services('Add', 'Error in ':Service:' service. Error calling Btree.Extract.')
end
end
Response = KeyList
end service
Service GetScheduledCassettes(WorkOrderNo)
ScheduledCassettes = ''
TableName = "SCHED_DET_NG"
Open "DICT.":TableName To hTableDict then
ScheduledQty = 0
Column = "WO_NO"
QueryParam = WorkOrderNo
Query = Column:@VM:QueryParam:@FM
Keylist = ""
Option = "S"
Flag = ""
Btree.Extract(Query, TableName, hTableDict, Keylist, Option, Flag)
If Flag EQ 0 then
For each KeyID in Keylist using @VM
UnprocessedCass = Xlate('SCHED_DET_NG', KeyID, 'UNPROCESSED_CASS', 'X')
ProcessedCass = Xlate('SCHED_DET_NG', KeyID, 'PROCESSED_CASS', 'X')
ScheduledCassettes = SRP_Array('Join', ScheduledCassettes, UnprocessedCass, 'OR', @VM)
ScheduledCassettes = SRP_Array('Join', ScheduledCassettes, ProcessedCass, 'OR', @VM)
Next KeyID
end
end
ScheduledCassettes = SRP_Array('SortSimpleList', ScheduledCassettes, 'AscendingNumbers', @VM)
Response = ScheduledCassettes
end service
Service GetUnscheduledWfrQty(WorkOrderNo)
If WorkOrderNo NE '' then
ScheduledWfrQty = Schedule_Services('GetScheduledWfrQty', WorkOrderNo)
WorkOrderWfrQty = Xlate('WO_LOG', WorkOrderNo, 'QTY', 'X')
UnscheduledWfrQty = WorkOrderWfrQty - ScheduledWfrQty
Response = UnscheduledWfrQty
end
end service
//----------------------------------------------------------------------------------------------------------------------
// GetAvailableWorkOrders
//
// Returns an array of work orders with wafers available to schedule.
//----------------------------------------------------------------------------------------------------------------------
Service GetAvailableWorkOrders(ReactorType, SusceptorSize, ReactorNo)
WorkOrders = ''
If ( (ReactorType NE '') or (ReactorNo NE '') ) then
If ReactorType EQ '' then ReactorType = Xlate('REACTOR', ReactorNo, 'REACT_TYPE', 'X')
end
If ReactorType _EQC 'ASM+' then ReactorType = 'ASM'
WorkOrderList = Database_Services('ReadDataRow', 'APP_INFO', 'SCHED_WO_LIST')
NumRows = DCount(WorkOrderList, @FM)
For RowIndex = 1 to NumRows
WOLogKeyID = WorkOrderList<RowIndex, 1>
WOLogRow = Database_Services('ReadDataRow', 'WO_LOG', WOLogKeyID)
EpiPartNo = WOLogRow<WO_LOG_EPI_PART_NO$>
ProdVerNo = WOLogRow<WO_LOG_PROD_VER_NO$>
ProdVerRow = Database_Services('ReadDataRow', 'PROD_VER', ProdVerNo)
PSN = ProdVerRow<PROD_VER_PROC_STEP_PSN$>
ProdSpecRow = Database_Services('ReadDataRow', 'PROD_SPEC', PSN)
BlockedReactors = ProdSpecRow<PROD_SPEC_BLOCKED_REACTS$>
AcceptWorkOrder = True$ ; // Assume this work order is acceptable for now.
PSNBlock = False$
If EpiPartNo NE '' then
If SusceptorSize NE '' then
EpiPart = Database_Services('ReadDataRow', 'EPI_PART', EpiPartNo)
SuscSize = EpiPart<EPI_PART_SUB_WAFER_SIZE$>
If SuscSize NE SusceptorSize then
// The susceptor size for this work order does not match the required susceptor size.
AcceptWorkOrder = False$
end
end
If (BlockedReactors NE '') AND (ReactorNo NE '') then
// This work order uses a part spec that is blocked on the current reactor.
Locate ReactorNo in BlockedReactors using @VM setting vPos then PSNBlock = True$
end
RowReactorType = WorkOrderList<RowIndex, 4>
If ( (RowReactorType _NEC ReactorType) and (RowReactorType _NEC ReactorType:'+') ) then
AcceptWorkOrder = False$
end
If AcceptWorkOrder then
Swap False$ with '' in PSNBlock
Swap True$ with 'Yes' in PSNBlock
WorkOrderList<RowIndex, 5> = PSNBlock
WorkOrders = Insert(WorkOrders, -1, 0, 0, WorkOrderList<RowIndex>)
end
end
Next RowIndex
Response = WorkOrders
end service
Service MarkCassProcessed(WONo, CassNo, ProcessedDTM)
If ((WONo NE '') and (CassNo NE '') and (ProcessedDTM NE '') ) then
If Num(ProcessedDTM) then ProcessedDTM = OConv(ProcessedDTM, 'DT2/^H')
// Find the SCHED_DET_NG event record that contains the cassette number
Query = 'SELECT SCHED_DET_NG WITH WO_NO EQ ':WONo:' AND WITH START_DTM LE "':ProcessedDTM:'" '
Query := 'AND WITH UNPROCESSED_CASS EQ ':CassNo:' BY START_DTM'
GoSub ClearCursors
RList(Query, TARGET_ACTIVELIST$, '', '', '')
EOF = False$
Done = False$
Loop
Readnext SchedDetNGKey else EOF = True$
Until EOF = True$
SchedDetNGRec = Database_Services('ReadDataRow', 'SCHED_DET_NG', SchedDetNGKey)
CassComp = False$
ReactType = Xlate('WO_LOG', WONo, 'REACT_TYPE', 'X')
If ReactType EQ 'EPP' then
// Since EpiPro splits WM_IN cassettes out into RDS runs, we need to determine if the
// cassette has been emptied out or if enough wafers have been consumed such that
// the wafer count equals the split work order event quantity.
StopDtm = SchedDetNGRec<SCHED_DET_NG.STOP_DTM$>
SchedWfrQty = Schedule_Services('GetScheduledWfrQty', WONo, StopDtm)
// Determine the cassette and slot this event should end on. This will be used to identify
// cassettes that should be marked as complete when the event ends in the middle of the cassette.
// That is, the EpiPro work order events should end on an RDS when they are split around block
// out events.
UnprocCassList = SchedDetNGRec<SCHED_DET_NG.UNPROCESSED_CASS$>
ProcCassList = SchedDetNGRec<SCHED_DET_NG.PROCESSED_CASS$>
AllSchedCass = SRP_Array('Join', UnprocCassList, ProcCassList, 'OR', @VM)
AllSchedCass = SRP_Array('SortSimpleList', AllSchedCass, 'AscendingNumbers', @VM)
NumSchedCass = DCount(AllSchedCass, @VM)
EndCassNo = AllSchedCass<0, NumSchedCass>
EndSlotNo = Mod(SchedWfrQty, 25)
If EndSlotNo EQ 0 then EndSlotNo = 25
WMIKey = WONo:'*1*':CassNo
WMIRdsList = Xlate('WM_IN', WMIKey, 'RDS_NO', 'X')
UnloadDtms = Xlate('RDS', WMIRdsList, 'DATETIME_OUT', 'X')
NumUnloadedWfrs = 0
If UnloadDtms NE '' then
For each UnloadDtm in UnloadDtms using @VM
NumUnloadedWfrs += (UnloadDtm NE '')
Next UnloadDtm
end
CassRemWfrs = Xlate('WM_IN', WMIKey, 'REM_WFRS', 'X')
If ( (NumUnloadedWfrs EQ 25) or ( (CassNo EQ EndCassNo) and (NumUnloadedWfrs GE EndSlotNo) ) or (CassRemWfrs EQ 0) ) then
CassComp = True$
end
end else
CassComp = True$
end
UnprocessedCassettes = SchedDetNGRec<SCHED_DET_NG.UNPROCESSED_CASS$>
If CassComp then
Locate CassNo in UnprocessedCassettes using @VM setting vPos then
UnprocessedCassettes = Delete(UnprocessedCassettes, 0, vPos, 0)
ProcessedCassettes = SchedDetNGRec<SCHED_DET_NG.PROCESSED_CASS$>
ProcessedCassettes<0, -1> = CassNo
SchedDetNGRec<SCHED_DET_NG.UNPROCESSED_CASS$> = UnprocessedCassettes
SchedDetNGRec<SCHED_DET_NG.PROCESSED_CASS$> = ProcessedCassettes
Database_Services('WriteDataRow', 'SCHED_DET_NG', SchedDetNGKey, SchedDetNGRec, True$, False$, True$)
Done = True$
end
end
Until Done EQ True$
Repeat
end
end service
Service GetEventCurrCass(SchedDetNGKey)
CurrCassNo = ''
If SchedDetNGKey NE '' then
SchedDetNGRec = Database_Services('ReadDataRow', 'SCHED_DET_NG', SchedDetNGKey)
CurrRunNo = Schedule_Services('GetEventCurrRun', SchedDetNGKey)
If CurrRunNo NE '' then
ReactNo = SchedDetNGRec<SCHED_DET_NG.REACT_NO$>
WONo = SchedDetNGRec<SCHED_DET_NG.WO_NO$>
ReactType = Xlate('REACTOR', ReactNo, 'REACT_TYPE', 'X')
AllRuns = Xlate('WO_STEP', WONo:'*1', 'RDS_KEY', 'X')
CurrRDSNo = AllRuns<0, CurrRunNo>
If ReactType EQ 'EPP' then
WMIKeys = Xlate('RDS', CurrRDSNo, 'WM_IN_KEY', 'X')
CurrCassNo = WMIKeys[-1, 'B*']
end else
CurrCassNo = Xlate('RDS', CurrRDSNo, 'CASS_NO', 'X')
end
end else
// No runs have been loaded yet, so use the first cassette assigned to the event
UnprocCassList = SchedDetNGRec<SCHED_DET_NG.UNPROCESSED_CASS$>
ProcCassList = SchedDetNGRec<SCHED_DET_NG.PROCESSED_CASS$>
EventCassList = SRP_Array('Join', ProcCassList, UnprocCassList, 'OR', @VM)
EventCassList = SRP_Array('SortSimpleList', EventCassList, 'AscendingNumbers', @VM)
CurrCassNo = EventCassList<0, 1>
end
end else
Error_Services('Add', 'Error in ':Service:' service. Null SchedDetNGKey passed in.')
end
Response = CurrCassNo
end service
Service GetEventCurrRun(SchedDetNGKey)
CurrRunNo = ''
If SchedDetNGKey NE '' then
EventRDSKeys = Schedule_Services('GetEventRDSKeys', SchedDetNGKey)
Swap @VM with @FM in EventRDSKeys
If EventRDSKeys NE '' then
Query = "WITH {DATE_IN} NE ''"
Tablename = 'RDS'
Open Tablename to hTable then
Open 'DICT.RDS' to hDict then
GoSub ClearCursors
Cursor = 0
Make.List(Cursor, EventRDSKeys, hTable, hDict)
Mode = 2 ; // Use existing cursor initialized above
SortList = '#DATETIME_IN'
Select Tablename by SortList using Cursor then
Reduce(Query, SortList, Mode, Tablename, Cursor, Flag)
If Flag then
// Success
ReadNext RDSKey then CurrRunNo = Xlate('RDS', RDSKey, 'RUN_ORDER_NUM', 'X')
GoSub ClearCursors
end
end
end
end
end
end else
Error_Services('Add', 'Error in ':Service:' service. Null SchedDetNGKey passed in.')
end
Response = CurrRunNo
end service
Service GetEventRDSKeys(SchedDetNGKey)
RDSKeys = ''
If SchedDetNGKey NE '' then
SchedEventRec = Database_Services('ReadDataRow', 'SCHED_DET_NG', SchedDetNGKey)
If Error_Services('NoError') then
WorkOrderNo = SchedEventRec<SCHED_DET_NG.WO_NO$>
ReactNo = SchedEventRec<SCHED_DET_NG.REACT_NO$>
ReactType = Xlate('REACTOR', ReactNo, 'REACT_TYPE', 'X')
UnprocCass = SchedEventRec<SCHED_DET_NG.UNPROCESSED_CASS$>
ProcCass = SchedEventRec<SCHED_DET_NG.PROCESSED_CASS$>
EventComp = SchedEventRec<SCHED_DET_NG.EVENT_COMP$>
AllCass = SRP_Array('Join', UnprocCass, ProcCass, 'OR', @VM)
AllCass = SRP_Array('SortSimpleList', AllCass, 'AscendingNumbers', @VM)
If ReactType NE 'EPP' then
For each CassNo in AllCass using @VM setting vPos
WOMatKey = WorkOrderNo:'*':CassNo
RDSKeys<0, -1> = Xlate('WO_MAT', WOMatKey, 'RDS_NO', 'X')
Next CassNo
end else
// Determine starting slot to exlude runs that are part of previous schedule events
QueryDtm = SchedEventRec<SCHED_DET_NG.START_DTM$>
SchedQty = SchedEventRec<SCHED_DET_NG.EVENT_TOTAL_WFRS$>
EventComp = SchedEventRec<SCHED_DET_NG.EVENT_COMP$>
PrevSchedQty = Schedule_Services('GetScheduledWfrQty', WorkOrderNo, QueryDtm)
StartSlot = Mod(PrevSchedQty, 25) + 1
TotalQty = PrevSchedQty + SchedQty
StopSlot = Mod(TotalQty, 25)
If StopSlot EQ 0 then StopSlot = 25
StartCass = AllCass<0, 1>
StopCass = AllCass[-1, 'B':@VM]
For each CassNo in AllCass using @VM setting vPos
WMInKey = WorkOrderNo:'*1*':CassNo
WMInRDSList = Xlate('WM_IN', WMInKey, 'RDS_NO', 'X')
Begin Case
Case CassNo EQ StartCass
For SlotIndex = StartSlot to 25
RDSKeys<0, -1> = WMInRDSList<0, SlotIndex>
Next SlotIndex
Case CassNo EQ StopCass
For SlotIndex = 1 to StopSlot
RDSKeys<0, -1> = WMInRDSList<0, SlotIndex>
Next SlotIndex
Case ( (CassNo EQ StartCass) and (CassNo EQ StopCass) )
For SlotIndex = StartSlot to StopSlot
RDSKeys<0, -1> = WMInRDSList<0, SlotIndex>
Next SlotIndex
Case Otherwise$
RDSKeys<0, -1> = WMInRDSList
End Case
Next CassNo
end
If Not(EventComp) then
// Add in RDS keys not yet loaded (i.e. not associated with any WM_IN record yet).
Query = "SELECT RDS WITH WO EQ ":WorkOrderNo:" AND WITH DATE_IN EQ '' BY SEQ"
RList(Query, TARGET_ACTIVELIST$, '', '', '')
ErrCode = ''
If Not(Get_Status(ErrCode)) then
EOF = False$
Loop
ReadNext RDSKey else EOF = True$
Until EOF
RDSKeys<0, -1> = RDSKey
Repeat
end
end
RDSKeys = SRP_Array('Clean', RDSKeys, 'TrimAndMakeUnique', @VM)
end
end else
Error_Services('Add', 'Error in ':Service:' service. Null SchedDetNGKey passed in.')
end
Response = RDSKeys
end service
Service GetEventLoadedRDSKeys(SchedDetNGKey)
RDSKeys = ''
If SchedDetNGKey NE '' then
EventRDSKeys = Schedule_Services('GetEventRDSKeys', SchedDetNGKey)
Swap @VM with @FM in EventRDSKeys
If EventRDSKeys NE '' then
Query = "WITH {DATE_IN} NE '' AND WITH {DATE_OUT} EQ ''"
Tablename = 'RDS'
Open Tablename to hTable then
Open 'DICT.RDS' to hDict then
GoSub ClearCursors
Cursor = 0
Make.List(Cursor, EventRDSKeys, hTable, hDict)
Mode = 2 ; // Use existing cursor initialized above
SortList = 'SEQ'
Reduce(Query, SortList, Mode, Tablename, Cursor, Flag)
If Flag then
// Success
EOF = False$
Loop
ReadNext RDSKey else EOF = True$
Until EOF
RDSKeys<0, -1> = RDSKey
Repeat
end
end
end
end
end else
Error_Services('Add', 'Error in ':Service:' service. Null SchedDetNGKey passed in.')
end
Response = RDSKeys
end service
Service GetEventLoadedWfrQty(SchedDetNGKey)
LoadedQty = ''
If SchedDetNGKey NE '' then
ReactNo = Xlate('SCHED_DET_NG', SchedDetNGKey, 'REACT_NO', 'X')
ReactType = Xlate('REACTOR', ReactNo, 'REACT_TYPE', 'X')
LoadedRDS = Schedule_Services('GetEventLoadedRDSKeys', SchedDetNGKey)
If LoadedRDS NE '' then
If ReactType EQ 'EPP' then
LoadedQty = Sum(Xlate('REACT_RUN', LoadedRDS, 'WFRS_EPI_LOAD', 'X'))
end else
LoadedQty = Sum(Xlate('RDS', LoadedRDS, 'WAFERS_IN', 'X'))
end
end
end else
Error_Services('Add', 'Error in ':Service:' service. Null SchedDetNGKey passed in.')
end
Response = LoadedQty
end service
Service NotifySupervisorsIfSameDayChange(NewRec, OrigRec)
If NewRec NE '' then
StartDTM = NewRec<SCHED_DET_NG.START_DTM$>
StopDTM = NewRec<SCHED_DET_NG.STOP_DTM$>
If (StartDTM NE '') and (StopDTM NE '') then
CurrentDTM = SRP_DateTime("Now")
CurrentPlus24DTM = SRP_Datetime("AddHours", CurrentDTM, 24)
IrrelevantStopChange = False$
IrrelevantStartChange = False$
If OrigRec NE '' then
OrigStartDTM = OrigRec<SCHED_DET_NG.START_DTM$>
OrigStopDTM = OrigRec<SCHED_DET_NG.STOP_DTM$>
If (OrigStartDTM NE '') and (OrigStopDTM NE '') then
// Check to make sure not just an extension where original stop was already after 24 hours from now
IrrelevantStopChange = ((OrigStopDTM GE CurrentPlus24DTM) and (StopDTM GE CurrentPlus24DTM)) or (OrigStopDTM EQ StopDTM)
// Check to make sure original start was not already before today and new start still not after now
IrrelevantStartChange = ((OrigStartDTM LT CurrentDTM) and (StartDTM LT CurrentDTM)) or (OrigStartDTM EQ StartDTM)
end
end
If ((IrrelevantStopChange NE True$) or (IrrelevantStartChange NE True$)) and (StartDTM LT StopDTM) then
SchedEventInNext24Hours = False$
GoSub VerifySameDayChange
If SchedEventInNext24Hours EQ True$ then
WorkOrder = NewRec<SCHED_DET_NG.WO_NO$>
ReactorNo = NewRec<SCHED_DET_NG.REACT_NO$>
HotLot = Work_Order_Services("GetHotFlag", WorkOrder)
Description = NewRec<SCHED_DET_NG.DESC$>
Message = ''
If HotLot NE False$ then Message := '[HOT LOT] '
Message := 'A same-day schedule change was made'
If ReactorNo NE '' then
Message := ' on reactor ':ReactorNo:'.'
end else
Message := '.'
end
EmailSubject = ''
If HotLot NE False$ then
EmailSubject = '[HOT LOT] '
end else
EmailSubject = ''
end
EmailSubject := 'Same-day schedule change!'
GoSub CreateScheduleChangeNotification
end
end
end
end else
If OrigRec NE '' then
StartDTM = OrigRec<SCHED_DET_NG.START_DTM$>
StopDTM = OrigRec<SCHED_DET_NG.STOP_DTM$>
If (StartDTM NE '') and (StopDTM NE '') then
CurrentDTM = SRP_DateTime("Now")
CurrentPlus24DTM = SRP_Datetime("AddHours", CurrentDTM, 24)
SchedEventInNext24Hours = False$
GoSub VerifySameDayChange
If SchedEventInNext24Hours EQ True$ then
WorkOrder = OrigRec<SCHED_DET_NG.WO_NO$>
ReactorNo = OrigRec<SCHED_DET_NG.REACT_NO$>
HotLot = Work_Order_Services("GetHotFlag", WorkOrder)
Description = OrigRec<SCHED_DET_NG.DESC$>
Message = 'A scheduled event was cancelled or removed on the same day'
If ReactorNo NE '' then
Message := ' on reactor ':ReactorNo:'.'
end else
Message := '.'
end
EmailSubject = 'Same-day schedule change!'
GoSub CreateScheduleChangeNotification
end
end
end
end
end service
Service SendRefreshMessage()
SendUpdate = True$
LastRefreshDtm = Xlate('APP_INFO', 'AUTO_SCHEDULER_LAST_REFRESH', '', 'X')
If LastRefreshDtm NE '' then
ElapMinutes = SRP_Datetime('MinuteSpan', LastRefreshDtm, Datetime(), True$)
SendUpdate = (ElapMinutes GT 5)
end
If SendUpdate then
Messaging_Services('SendMessage', 'RefreshSchedule', 'Response', @User4, 'All', @User4, 'EventHandler', 'NDW_SCHEDULER' : ',OMNIEVENT,@MESSAGE,@ARGUMENTS')
Database_Services('WriteDataRow', 'APP_INFO', 'AUTO_SCHEDULER_LAST_REFRESH', Datetime(), True$, False$, True$)
end
end service
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
// Internal GoSubs
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
ClearCursors:
For counter = 0 to 8
ClearSelect counter
Next counter
return
LogError:
Message = Error_Services('GetMessage')
LogData = ''
LogData<1> = Oconv(Date(), 'D4/') : ' ' : Oconv(Time(), 'MTS') ; // Logging DTM
LogData<2> = Service : ' Error' ; // Service Step
LogData<3> = Str(' ', 12) : Message
Schedule_Services('LogActivity', ReactorNo, LogData)
return
VerifySameDayChange:
If (StartDTM LE CurrentDTM) and (StopDTM GE CurrentPlus24DTM) then
SchedEventInNext24Hours = True$
end
If (StartDTM GE CurrentDTM) and (StopDTM LE CurrentPlus24DTM) then
SchedEventInNext24Hours = True$
end
If (StartDTM GE CurrentDTM) and (StartDTM LT CurrentPlus24DTM) and (StopDTM GE CurrentPlus24DTM) then
SchedEventInNext24Hours = True$
end
If (StartDTM LE CurrentDTM) and (StopDTM LE CurrentPlus24DTM) and (StopDTM GT CurrentDTM) then
SchedEventInNext24Hours = True$
end
return
CreateScheduleChangeNotification:
If WorkOrder NE '' then Message := CRLF$:'Work Order: ':WorkOrder:'.'
ConvertedStart = OConv(StartDTM, 'DT2/^H')
If StartDTM NE '' then Message := CRLF$:'Start Time: ':ConvertedStart:'.'
ConvertedStop = OConv(StopDTM, 'DT2/^H')
If StopDTM NE '' then Message := CRLF$:'End Time: ':ConvertedStop:'.'
If Description NE '' then Message := CRLF$:'Description: ':Description:'.'
IncludeNextShift = False$
//If we are close to shift change we want to add in the next shift.
CurrTime = Time()
if SRP_Time('Hour', CurrTime) EQ 17 OR SRP_Time('Hour', CurrTime) EQ 5 then
IncludeNextShift = True$
end
OnShiftSupervisors = Lsl_Users_Services('GetOnShiftUsersByClass', 'Si Shift Supervisors', IncludeNextShift)
OnShiftLeads = Lsl_Users_Services('GetOnShiftUsersByClass', 'Lead Operator', IncludeNextShift)
Recipients = OnShiftSupervisors:@VM:OnShiftLeads
SentFrom = @User4
AttachWindow = ''
AttachKey = ''
SendToGroup = ''
Parms = Recipients:@RM:SentFrom:@RM:EmailSubject:@RM:Message:@RM:AttachWindow:@RM:AttachKey:@RM:SendToGroup
obj_Notes('Create',Parms)
return