Sic03-8inch/SicRT/Equipments/Schedulers/SchedulerModule.cs

517 lines
16 KiB
C#
Raw Permalink Normal View History

2023-03-03 15:42:13 +08:00
using System;
using System.Collections.Generic;
using Aitex.Core.Common;
using Aitex.Core.RT.Event;
using Aitex.Core.RT.SCCore;
using Aitex.Core.Util;
using MECF.Framework.Common.Equipment;
using MECF.Framework.Common.Schedulers;
using MECF.Framework.Common.SubstrateTrackings;
namespace SicRT.Modules.Schedulers
{
public class SchedulerModule
{
protected enum TaskType
{
None,
PrepareTransfer,
Pick,
Place,
PickAndPlace,
Preprocess,
Process,
PostProcess,
OpenCover,
CloseCover,
Load,
Unload,
Align,
TransferTarget,
Cooling,
WarmUp,
Vent,
Pump,
Purge,
Group,
Separate,
Map
}
/// <summary>
/// 访问变量时先锁定,避免多线程竞争问题。
/// </summary>
protected readonly object SyncRoot = new object();
public ModuleName Module
{
get { return ModuleHelper.Converter(_module); }
}
public virtual bool IsAvailable { get; }
public virtual bool IsOnline { get; }
public virtual bool IsError { get; }
public virtual bool IsReady { get; }
public virtual bool IsService { get; }
private R_TRIG[] firstDetectWaferArriveTrigs = new R_TRIG[25];
private R_TRIG[] firstDetectWaferLeaveTrigs = new R_TRIG[25];
private R_TRIG[] firstDetectTrayArriveTrigs = new R_TRIG[25];
private R_TRIG[] firstDetectTrayLeaveTrigs = new R_TRIG[25];
protected string _module;
protected TaskType _task = TaskType.None;
protected ModuleName _inProcessRobot;
public SchedulerModule(string module)
{
_module = module;
for (int i = 0; i < firstDetectWaferArriveTrigs.Length; i++)
{
firstDetectWaferArriveTrigs[i] = new R_TRIG();
}
for (int i = 0; i < firstDetectWaferLeaveTrigs.Length; i++)
{
firstDetectWaferLeaveTrigs[i] = new R_TRIG();
}
for (int i = 0; i < firstDetectTrayArriveTrigs.Length; i++)
{
firstDetectTrayArriveTrigs[i] = new R_TRIG();
}
for (int i = 0; i < firstDetectTrayLeaveTrigs.Length; i++)
{
firstDetectTrayLeaveTrigs[i] = new R_TRIG();
}
}
public string GetTask()
{
return _task.ToString();
}
/// <summary>
/// 检查Module的任务是否完成。
/// <para>完成的条件除关键动作完成例如Wafer到位以外还需要判断Module的状态是否为Idle或Error。</para>
/// <para>1. 如果关键动作完成但Module还未Idle则说明上个Routine还未执行完毕则认为Task还未完成。</para>
/// <para>2. 如果关键动作未完成但Module已经Idle或Error则说明Routine已执行完毕但可能出错则认为Task完成。</para>
/// </summary>
/// <param name="keyActionDone"></param>
/// <param name="fsmStopped"></param>
/// <returns></returns>
protected bool SuperCheckTaskDone(bool keyActionDone, bool fsmStopped)
{
//TODO 需进一步抽象将Module.IsIdle和Module.IsError的判断写在基类。
keyActionDone &= fsmStopped;
if (keyActionDone)
{
if (_task != TaskType.None)
{
LogTaskDone(_task, "");
_task = TaskType.None;
}
return true;
}
if (fsmStopped)
{
LogTaskDone(_task, "Task Failed");
_task = TaskType.None;
return true;
}
return false;
}
protected void LogTaskStart(TaskType cmd, string message)
{
EV.PostInfoLog("Scheduler", $"Task start:{_module},{cmd} {message}");
}
protected void LogTaskDone(TaskType cmd, string message)
{
EV.PostInfoLog("Scheduler", $"Task done:{_module},{cmd} {message}");
}
public void ResetTask()
{
_task = TaskType.None;
}
public bool WaitTransfer(ModuleName robot)
{
_task = TaskType.TransferTarget;
_inProcessRobot = robot;
LogTaskStart(_task, $"Note {robot} in transfer");
return true;
}
public bool IsWaitTransfer(ModuleName robot)
{
return _task == TaskType.TransferTarget && _inProcessRobot == robot;
}
public bool StopWaitTransfer(ModuleName robot)
{
LogTaskDone(_task, $"Note {robot} transfer complete");
_inProcessRobot = ModuleName.System;
_task = TaskType.None;
return true;
}
2023-03-13 17:37:55 +08:00
public WaferInfoRt GetWaferInfo(int slot)
2023-03-03 15:42:13 +08:00
{
return WaferManager.Instance.GetWafer(ModuleHelper.Converter(_module), slot);
}
public bool CheckWaferStatus(int slot, WaferStatus waferStatus)
{
return WaferManager.Instance.CheckWafer(ModuleHelper.Converter(_module), slot, waferStatus);
}
public bool HasWafer(int slot)
{
return WaferManager.Instance.CheckHasWafer(ModuleHelper.Converter(_module), slot);
}
public bool NoWafer(int slot)
{
return WaferManager.Instance.CheckNoWafer(ModuleHelper.Converter(_module), slot);
}
public bool HasTray(int slot)
{
return WaferManager.Instance.CheckHasTray(ModuleHelper.Converter(_module), slot);
}
public bool NoTray(int slot)
{
return WaferManager.Instance.CheckNoTray(ModuleHelper.Converter(_module), slot);
}
public bool FirstDetectWaferArrive(int slot)
{
firstDetectWaferArriveTrigs[slot].CLK = HasWafer(slot);
return firstDetectWaferArriveTrigs[slot].Q;
}
public bool FirstDetectWaferLeave(int slot)
{
firstDetectWaferLeaveTrigs[slot].CLK = NoWafer(slot);
return firstDetectWaferLeaveTrigs[slot].Q;
}
public bool FirstDetectTrayArrive(int slot)
{
firstDetectTrayArriveTrigs[slot].CLK = HasTray(slot);
return firstDetectTrayArriveTrigs[slot].Q;
}
public bool FirstDetectTrayLeave(int slot)
{
firstDetectTrayLeaveTrigs[slot].CLK = NoTray(slot);
return firstDetectTrayLeaveTrigs[slot].Q;
}
public bool HasTrayAndExceedProcessCount(int slot)
{
2023-03-13 17:37:55 +08:00
var wi = WaferManager.Instance.GetWafer(ModuleHelper.Converter(_module), slot);
2023-03-03 15:42:13 +08:00
return wi.TrayState == WaferTrayStatus.Normal && wi.TrayProcessCount <=0;
}
public bool HasTrayAndNotExceedProcessCount(int slot)
{
2023-03-13 17:37:55 +08:00
var wi = WaferManager.Instance.GetWafer(ModuleHelper.Converter(_module), slot);
2023-03-03 15:42:13 +08:00
return wi.TrayState == WaferTrayStatus.Normal && wi.TrayProcessCount >0;
}
public virtual bool CheckWaferNextStepIsThis(ModuleName module, int slot)
{
if (!WaferManager.Instance.CheckHasWafer(module, slot))
return false;
2023-03-13 17:37:55 +08:00
var wafer = WaferManager.Instance.GetWafer(module, slot);
2023-03-03 15:42:13 +08:00
if (wafer.ProcessJob == null || wafer.ProcessJob.Sequence == null)
return false;
if (wafer.NextSequenceStep >= wafer.ProcessJob.Sequence.Steps.Count)
return false;
if (!wafer.ProcessJob.Sequence.Steps[wafer.NextSequenceStep].StepModules.Contains(Module))
return false;
return true;
}
/// <summary>
/// 判断下一步的模块是否有Wafer
/// </summary>
/// <param name="slot"></param>
/// <returns></returns>
public virtual bool CheckWaferNextStepModuleNoWafer(int slot)
{
if (!WaferManager.Instance.CheckHasWafer(Module, slot))
return false;
2023-03-13 17:37:55 +08:00
var wafer = WaferManager.Instance.GetWafer(Module, slot);
2023-03-03 15:42:13 +08:00
if (wafer.ProcessJob == null || wafer.ProcessJob.Sequence == null)
return false;
if (wafer.NextSequenceStep >= wafer.ProcessJob.Sequence.Steps.Count)
return false;
List<ModuleName> lstModuleName = wafer.ProcessJob.Sequence.Steps[wafer.NextSequenceStep].StepModules;
if (lstModuleName.Count > 0)
{
for (int i = 0; i < lstModuleName.Count; i++)
{
if (WaferManager.Instance.CheckNoWafer(lstModuleName[i], 0))
{
return true;
}
}
}
return false;
}
public virtual bool CheckWaferNextStepModuleNoTray(int slot)
{
if (!WaferManager.Instance.CheckHasWafer(Module, slot))
return false;
2023-03-13 17:37:55 +08:00
var wafer = WaferManager.Instance.GetWafer(Module, slot);
2023-03-03 15:42:13 +08:00
if (wafer.ProcessJob == null || wafer.ProcessJob.Sequence == null)
return false;
if (wafer.NextSequenceStep >= wafer.ProcessJob.Sequence.Steps.Count)
return false;
List<ModuleName> lstModuleName = wafer.ProcessJob.Sequence.Steps[wafer.NextSequenceStep].StepModules;
if (lstModuleName.Count > 0)
{
for (int j = 0; j < lstModuleName.Count; j++)
{
if (lstModuleName[j] == ModuleName.Buffer)
{
if (CheckWaferNeedProcess(slot, ModuleName.PM1))
{
if (WaferManager.Instance.CheckNoTray(ModuleName.Buffer, 0))
{
return true;
}
}
else if (CheckWaferNeedProcess(slot, ModuleName.PM2))
{
if (WaferManager.Instance.CheckNoTray(ModuleName.Buffer, 1))
{
return true;
}
}
}
else if (WaferManager.Instance.CheckNoTray(lstModuleName[j], 0))
{
return true;
}
}
}
return false;
}
/// <summary>
/// 判断Wafer的步骤是否都完成了
/// </summary>
/// <param name="waferModule"></param>
/// <param name="waferSlot"></param>
/// <returns></returns>
public virtual bool CheckWaferSequenceStepDone(int slot)
{
2023-03-13 17:37:55 +08:00
var wafer = WaferManager.Instance.GetWafer(Module, slot);
2023-03-03 15:42:13 +08:00
if (wafer.IsEmpty)
{
return false;
}
if (wafer.ProcessJob == null || wafer.ProcessJob.Sequence == null)
{
return true;
}
if (wafer.NextSequenceStep >= wafer.ProcessJob.Sequence.Steps.Count && wafer.ProcessJob.Sequence.Steps.Count > 0)
{
return true;
}
return false;
}
public virtual bool CheckWaferNeedProcess(int waferSlot, ModuleName processIn = ModuleName.System)
{
2023-03-13 17:37:55 +08:00
var wafer = WaferManager.Instance.GetWafer(Module, waferSlot);
2023-03-03 15:42:13 +08:00
if (wafer.IsEmpty)
return false;
//if (wafer.Status == WaferStatus.Dummy && wafer.ProcessState == EnumWaferProcessStatus.Wait)
//{
// if (ModuleHelper.IsPm(processIn))
// {
// return (CheckNeedRunClean(out bool withWafer, out _) && withWafer);
// }
// return true;
//}
if (wafer.ProcessJob == null || wafer.ProcessJob.Sequence == null || wafer.ProcessJob.Sequence.Steps == null)
return false;
if (wafer.NextSequenceStep >= wafer.ProcessJob.Sequence.Steps.Count || wafer.ProcessJob.Sequence.Steps[wafer.NextSequenceStep].StepModules == null)
return false;
for (int i = wafer.NextSequenceStep; i < wafer.ProcessJob.Sequence.Steps.Count; i++)
{
foreach (var stepModule in wafer.ProcessJob.Sequence.Steps[i].StepModules)
{
if (ModuleHelper.IsPm(stepModule))
{
if (processIn != ModuleName.System)
{
return stepModule == processIn ? true : false;
}
else
{
return true;
}
}
}
}
return false;
}
public virtual bool CheckNeedRunClean(out bool withWafer, out string recipeName)
{
recipeName = string.Empty;
withWafer = false;
//SequenceInfo seq = GetCurrentSequenceInfo();
//int waferProcessed = StatsDataManager.Instance.GetValue($"{Module}.WaferProcessedSincePreviousClean");
//if (seq == null)
// return false;
//foreach (var stepInfo in seq.Steps)
//{
// if (!stepInfo.StepModules.Contains(Module))
// continue;
// if (stepInfo.StepParameter.ContainsKey("CleanIntervalWaferless")
// && int.TryParse((string)stepInfo.StepParameter["CleanIntervalWaferless"], out int interval)
// && stepInfo.StepParameter.ContainsKey("CleanRecipeWaferless")
// && !string.IsNullOrEmpty((string)stepInfo.StepParameter["CleanRecipeWaferless"]))
// {
// if (interval > 0 && waferProcessed >= interval)
// {
// recipeName = (string)stepInfo.StepParameter["CleanRecipeWaferless"];
// withWafer = false;
// return true;
// }
// }
// if (stepInfo.StepParameter.ContainsKey("CleanIntervalWafer")
// && int.TryParse((string)stepInfo.StepParameter["CleanIntervalWafer"], out interval)
// && stepInfo.StepParameter.ContainsKey("CleanRecipeWafer")
// && !string.IsNullOrEmpty((string)stepInfo.StepParameter["CleanRecipeWafer"]))
// {
// if (interval > 0 && waferProcessed >= interval)
// {
// recipeName = (string)stepInfo.StepParameter["CleanRecipeWafer"];
// withWafer = true;
// return true;
// }
// }
//}
return false;
}
public virtual bool IsReadyForPick(ModuleName robot, int slot)
{
return true;
}
public virtual bool IsReadyForPlace(ModuleName robot, int slot)
{
return true;
}
public virtual bool PrepareTransfer(ModuleName robot, EnumTransferType type, int slot)
{
return true;
}
public virtual bool PostTransfer(ModuleName robot, EnumTransferType type, int slot)
{
return true;
}
public virtual bool Process(string recipeName, bool isCleanRecipe, bool withDummyWafer)
{
return true;
}
public virtual bool Cooling(bool coolingType, int coolingTime)
{
return true;
}
public virtual bool Aligning()
{
return true;
}
public virtual bool WarmUp(int warmUpTime)
{
return true;
}
public virtual bool CheckWaferTraySeparated()
{
return true;
}
public virtual bool CheckWaferTrayGrouped()
{
return true;
}
public virtual bool CheckSlitValveClosed() //检查闸板阀是否关闭
{
return true;
}
}
}