|
@@ -27,6 +27,7 @@ using Venus_Unity;
|
|
|
using MECF.Framework.RT.EquipmentLibrary.HardwareUnits.PMs;
|
|
|
using System.Linq.Expressions;
|
|
|
using Aitex.Core.UI.Control;
|
|
|
+using SecsGem.Core.ItemModel;
|
|
|
|
|
|
namespace Venus_RT.Modules
|
|
|
{
|
|
@@ -91,11 +92,12 @@ namespace Venus_RT.Modules
|
|
|
|
|
|
private DateTime _scheduledTime;
|
|
|
|
|
|
+ public ModuleName DestinationPM { get; set; }
|
|
|
public WaferTask(ModuleName source, int srcSlot, ModuleName dest, int dstSlot, float temp, Guid waferID, Guid lotID, string recipeName, string wtwClean, SequenceLLInOutPath inOutPath, int LLDelay, bool needAlign)
|
|
|
{
|
|
|
sourceMod = currentMod = nextMod = routedMod = source;
|
|
|
sourceSlot = currentSlot = nextSlot = routedSlot = srcSlot;
|
|
|
-
|
|
|
+ DestinationPM = dest;
|
|
|
destMod = dest;
|
|
|
destSlot = dstSlot;
|
|
|
hand = Hand.None;
|
|
@@ -593,6 +595,7 @@ namespace Venus_RT.Modules
|
|
|
|
|
|
public void InvokePreJobClean(string preJobClean)
|
|
|
{
|
|
|
+
|
|
|
_preJobCleanRecipe = preJobClean;
|
|
|
_pendingCleanTask.Enqueue(CleanType.PreJobClean);
|
|
|
//LOG.Write(eEvent.INFO_TROUGHPUT_ENHANCE_TRACE, Module, $"Invoke preClean {preJobClean}");
|
|
@@ -947,6 +950,9 @@ namespace Venus_RT.Modules
|
|
|
public List<string> LP1InUseRecipes = new List<string>();
|
|
|
public List<string> LP2InUseRecipes = new List<string>();
|
|
|
public List<string> LP3InUseRecipes = new List<string>();
|
|
|
+
|
|
|
+ private Dictionary<ModuleName, DateTime> _pmLastWaferEndTime = new Dictionary<ModuleName, DateTime>();
|
|
|
+ private int _preCleanMaxIdleTime = 60;
|
|
|
#region public interface
|
|
|
public SystemDispatcher()
|
|
|
{
|
|
@@ -984,6 +990,15 @@ namespace Venus_RT.Modules
|
|
|
{
|
|
|
LOG.ModuleErrorInterrupt += (modulename, id) => ModuleErrorInterrupt(modulename, id);
|
|
|
}
|
|
|
+
|
|
|
+ ModuleHelper.InstalledModules.ForEach(x =>
|
|
|
+ {
|
|
|
+ if (ModuleHelper.IsPm(x))
|
|
|
+ {
|
|
|
+ _pmLastWaferEndTime.Add(x, DateTime.Now.AddDays(-1));
|
|
|
+ }
|
|
|
+ });
|
|
|
+
|
|
|
}
|
|
|
private void ModuleErrorInterrupt(ModuleName pmname, eEvent eEventid)
|
|
|
{
|
|
@@ -993,7 +1008,7 @@ namespace Venus_RT.Modules
|
|
|
x.Sequence.PMs.ForEach(m => modules.Add(m));
|
|
|
});
|
|
|
if (modules.Contains(pmname) && (eEventid == eEvent.ERR_PM_SCRUBBER_NOOK_CONTINUEPROCESS || eEventid == eEvent.ERR_PM_SCRUBBER_NOOK_STOPPROCESS))
|
|
|
- {
|
|
|
+ {
|
|
|
Singleton<RouteManager>.Instance.InvokeAbort(new object[] { ModuleName.System });
|
|
|
LOG.Write(eEvent.ERR_PM_SCRUBBER_NOOK_STOPTRANSFER, ModuleName.System, $"{pmname} Scrubber Alarm ,Tranfer Stop");
|
|
|
}
|
|
@@ -1027,7 +1042,7 @@ namespace Venus_RT.Modules
|
|
|
_tmRobotSingleArmOption = SC.GetValue<int>("TM.SingleArmOption");
|
|
|
_LLSlotInOutOption = (LLSlotInOutOpt)SC.GetValue<int>("System.LoadlockSlotInOutOption");
|
|
|
_specialRoutingPattern = (SpecialRoutingPattern)SC.GetValue<int>("System.SpecialRoutingPattern");
|
|
|
-
|
|
|
+ _preCleanMaxIdleTime = SC.GetValue<int>("System.Job.PreCleanIdleMaxTime");
|
|
|
// rounding TM robot single arm option
|
|
|
if (_tmRobotSingleArmOption > 2 && _LLSlotInOutOption == LLSlotInOutOpt.AllInAllOut)
|
|
|
{
|
|
@@ -1062,7 +1077,7 @@ namespace Venus_RT.Modules
|
|
|
RoutingVacWafers();
|
|
|
|
|
|
epilogue();
|
|
|
-
|
|
|
+
|
|
|
return _cycleState;
|
|
|
}
|
|
|
|
|
@@ -1075,6 +1090,26 @@ namespace Venus_RT.Modules
|
|
|
reason = "";
|
|
|
|
|
|
string[] slotSequence = (string[])param["SlotSequence"];
|
|
|
+ string[] slotWaferId = new string[slotSequence.Length];
|
|
|
+ if (param.Keys.Contains("WaferId"))
|
|
|
+ {
|
|
|
+ slotWaferId = (string[])param["WaferId"];
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ for (int i = 0; i < slotWaferId.Length; i++)
|
|
|
+ {
|
|
|
+ slotWaferId[i] = "";
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ string lotType = "";
|
|
|
+ if (param.Keys.Contains("LotType"))
|
|
|
+ {
|
|
|
+ lotType = (string)param["LotType"];
|
|
|
+ }
|
|
|
+
|
|
|
+
|
|
|
string module = (string)param["Module"];
|
|
|
|
|
|
string jobId = (string)param["JobId"];
|
|
@@ -1188,6 +1223,8 @@ namespace Venus_RT.Modules
|
|
|
WaferInfo waferInfo = WaferManager.Instance.GetWafer(ModuleHelper.Converter(module), i);
|
|
|
cj.LotWafers.Add(waferInfo);
|
|
|
waferInfo.SequenceName = slotSequence[i];
|
|
|
+ waferInfo.EAPWaferID = slotWaferId[i];
|
|
|
+ waferInfo.LotType = lotType;
|
|
|
waferInfo.LotId = lotId;
|
|
|
//--- 2024-03-21 增加了waferinfo 相应的lotId信息 end---
|
|
|
|
|
@@ -1580,26 +1617,26 @@ namespace Venus_RT.Modules
|
|
|
ModuleName SourceModule = (ModuleName)objs[0];
|
|
|
int SourceSlot = (int)objs[1];
|
|
|
|
|
|
- if (!_dictModuleTask.Keys.Contains(SourceModule))
|
|
|
- {
|
|
|
- LOG.Write(eEvent.WARN_ROUTER, ModuleName.System, $"Invalid source module {SourceModule} for manual return wafer");
|
|
|
- return false;
|
|
|
- }
|
|
|
+ //if (!_dictModuleTask.Keys.Contains(SourceModule))
|
|
|
+ //{
|
|
|
+ // LOG.Write(eEvent.WARN_ROUTER, ModuleName.System, $"Invalid source module {SourceModule} for manual return wafer");
|
|
|
+ // return false;
|
|
|
+ //}
|
|
|
|
|
|
- if (WaferManager.Instance.CheckNoWafer(SourceModule, SourceSlot))
|
|
|
- {
|
|
|
- LOG.Write(eEvent.WARN_ROUTER, ModuleName.System, $"Can not return wafer as {SourceModule} {SourceSlot} has no wafer");
|
|
|
- return false;
|
|
|
- }
|
|
|
+ //if (WaferManager.Instance.CheckNoWafer(SourceModule, SourceSlot))
|
|
|
+ //{
|
|
|
+ // LOG.Write(eEvent.WARN_ROUTER, ModuleName.System, $"Can not return wafer as {SourceModule} {SourceSlot} has no wafer");
|
|
|
+ // return false;
|
|
|
+ //}
|
|
|
|
|
|
- if (!_dictModuleTask[SourceModule].Scheduler.IsIdle)
|
|
|
- {
|
|
|
- LOG.Write(eEvent.WARN_ROUTER, ModuleName.System, $"The module: {SourceModule} is not ready for return wafer");
|
|
|
- return false;
|
|
|
- }
|
|
|
+ //if (!_dictModuleTask[SourceModule].Scheduler.IsIdle)
|
|
|
+ //{
|
|
|
+ // LOG.Write(eEvent.WARN_ROUTER, ModuleName.System, $"The module: {SourceModule} is not ready for return wafer");
|
|
|
+ // return false;
|
|
|
+ //}
|
|
|
|
|
|
var wafer = WaferManager.Instance.GetWafer(SourceModule, SourceSlot);
|
|
|
- LOG.Write(eEvent.EV_ROUTER, ModuleName.System, $"Manual return wafer: {wafer.WaferOrigin} at {SourceModule} {SourceSlot} while system is auto running");
|
|
|
+ //LOG.Write(eEvent.EV_ROUTER, ModuleName.System, $"Manual return wafer: {wafer.WaferOrigin} at {SourceModule} {SourceSlot} while system is auto running");
|
|
|
|
|
|
_qeReturnWafers.Enqueue(wafer);
|
|
|
return true;
|
|
@@ -1609,17 +1646,17 @@ namespace Venus_RT.Modules
|
|
|
{
|
|
|
var pmWaferCount = _dictModuleTask.Where(mod => ModuleHelper.IsPm(mod.Key) && WaferManager.Instance.CheckHasWafer(mod.Key, 0)).Count();
|
|
|
var tmRobotWaferCount = (WaferManager.Instance.CheckHasWafer(ModuleName.TMRobot, 0) ? 1 : 0) + (WaferManager.Instance.CheckHasWafer(ModuleName.TMRobot, 1) ? 1 : 0);
|
|
|
- if (!_dictModuleTask[ModuleName.TMRobot].Scheduler.IsIdle && (pmWaferCount > 0 || tmRobotWaferCount > 0))
|
|
|
- {
|
|
|
- LOG.Write(eEvent.ERR_ROUTER, ModuleName.TMRobot, $"The TM Robot is not ready for return wafer.");
|
|
|
- return RState.Failed;
|
|
|
- }
|
|
|
+ //if (!_dictModuleTask[ModuleName.TMRobot].Scheduler.IsIdle && (pmWaferCount > 0 || tmRobotWaferCount > 0))
|
|
|
+ //{
|
|
|
+ // LOG.Write(eEvent.ERR_ROUTER, ModuleName.TMRobot, $"The TM Robot is not ready for return wafer.");
|
|
|
+ // return RState.Failed;
|
|
|
+ //}
|
|
|
|
|
|
- if (!_dictModuleTask[ModuleName.EfemRobot].Scheduler.IsIdle)
|
|
|
- {
|
|
|
- LOG.Write(eEvent.ERR_ROUTER, ModuleName.EfemRobot, $"The EFEM Robot is not ready for return wafer.");
|
|
|
- return RState.Failed;
|
|
|
- }
|
|
|
+ //if (!_dictModuleTask[ModuleName.EfemRobot].Scheduler.IsIdle)
|
|
|
+ //{
|
|
|
+ // LOG.Write(eEvent.ERR_ROUTER, ModuleName.EfemRobot, $"The EFEM Robot is not ready for return wafer.");
|
|
|
+ // return RState.Failed;
|
|
|
+ //}
|
|
|
|
|
|
foreach (var mod in _dictModuleTask)
|
|
|
{
|
|
@@ -1629,11 +1666,11 @@ namespace Venus_RT.Modules
|
|
|
if (!mod.Value.Scheduler.IsInclude)
|
|
|
continue;
|
|
|
|
|
|
- if (!mod.Value.Scheduler.IsIdle)
|
|
|
- {
|
|
|
- LOG.Write(eEvent.ERR_ROUTER, mod.Key, $"{mod.Key} is not ready for return wafer.");
|
|
|
- return RState.Failed;
|
|
|
- }
|
|
|
+ //if (!mod.Value.Scheduler.IsIdle)
|
|
|
+ //{
|
|
|
+ // LOG.Write(eEvent.ERR_ROUTER, mod.Key, $"{mod.Key} is not ready for return wafer.");
|
|
|
+ // return RState.Failed;
|
|
|
+ //}
|
|
|
|
|
|
int nSlotNumber = ModuleHelper.IsLoadLock(mod.Key) ? (mod.Key == ModuleName.LLA ? _LLASlotNumber : _LLBSlotNumber) : (ModuleHelper.IsTMRobot(mod.Key) || ModuleHelper.IsEFEMRobot(mod.Key) ? 2 : 1);
|
|
|
for (int slot = 0; slot < nSlotNumber; slot++)
|
|
@@ -1643,18 +1680,19 @@ namespace Venus_RT.Modules
|
|
|
continue;
|
|
|
|
|
|
var destLP = (ModuleName)wafer.OriginStation;
|
|
|
- if (!_dictModuleTask[destLP].Scheduler.IsAvailable)
|
|
|
- {
|
|
|
- LOG.Write(eEvent.ERR_ROUTER, destLP, $"The destination Loadport {destLP} is not ready for return wafer.");
|
|
|
- return RState.Failed;
|
|
|
- }
|
|
|
+ //if (!_dictModuleTask[destLP].Scheduler.IsAvailable)
|
|
|
+ //{
|
|
|
+ // LOG.Write(eEvent.ERR_ROUTER, destLP, $"The destination Loadport {destLP} is not ready for return wafer.");
|
|
|
+ // return RState.Failed;
|
|
|
+ //}
|
|
|
|
|
|
- if (!ModuleHelper.IsLoadPort(destLP))
|
|
|
- {
|
|
|
- LOG.Write(eEvent.ERR_ROUTER, ModuleName.System, $"The wafer {wafer.WaferOrigin} cannot be return, please manually drag it.");
|
|
|
- return RState.Failed;
|
|
|
- }
|
|
|
+ //if (!ModuleHelper.IsLoadPort(destLP))
|
|
|
+ //{
|
|
|
+ // LOG.Write(eEvent.ERR_ROUTER, ModuleName.System, $"The wafer {wafer.WaferOrigin} cannot be return, please manually drag it.");
|
|
|
+ // return RState.Failed;
|
|
|
+ //}
|
|
|
}
|
|
|
+
|
|
|
}
|
|
|
|
|
|
Clear();
|
|
@@ -1696,6 +1734,7 @@ namespace Venus_RT.Modules
|
|
|
if (systemWaferCount == 0)
|
|
|
return RState.End;
|
|
|
|
|
|
+
|
|
|
if (_qeWaitCoolingLLs.Count > 0 && _dictModuleTask[_qeWaitCoolingLLs.First()].Scheduler.IsIdle && !IsLLReservedByEFEM(_qeWaitCoolingLLs.First()) && !IsLLReservedByTM(_qeWaitCoolingLLs.First()))
|
|
|
{
|
|
|
if ((_dictModuleTask[_qeWaitCoolingLLs.First()] as LoadlockTask).Cooling())
|
|
@@ -1770,6 +1809,17 @@ namespace Venus_RT.Modules
|
|
|
RunLotCleanTasks();
|
|
|
|
|
|
CreateNewWaferTask();
|
|
|
+
|
|
|
+ //var endWaferTask = _lstWaferTasks.Where(item => ModuleHelper.IsLoadPort(item.destMod) && ModuleHelper.IsLoadPort(item.currentMod) && item.pressureStatus == RState.End);
|
|
|
+ //if (endWaferTask.Count() > 0)
|
|
|
+ //{
|
|
|
+ // foreach (var waferTask in endWaferTask)
|
|
|
+ // {
|
|
|
+ // var pj = _lstProcessJobs.Find(process => process.InnerId == waferTask.lotId);
|
|
|
+ // //_pmLastWaferEndTime[waferTask.DestinationPM] = DateTime.Now;
|
|
|
+ // }
|
|
|
+ //}
|
|
|
+
|
|
|
_lstWaferTasks.RemoveAll(item => ModuleHelper.IsLoadPort(item.destMod) && ModuleHelper.IsLoadPort(item.currentMod) && item.pressureStatus == RState.End);
|
|
|
|
|
|
ReDispatchBlockingWafers();
|
|
@@ -2242,6 +2292,10 @@ namespace Venus_RT.Modules
|
|
|
private void WaferLeaved(WaferTask wafer, MoveItem item)
|
|
|
{
|
|
|
_dictModuleTask[item.SourceModule].WaferLeaved(wafer, item.DestinationSlot);
|
|
|
+ if (ModuleHelper.IsPm(item.SourceModule))
|
|
|
+ {
|
|
|
+ _pmLastWaferEndTime[item.SourceModule] = DateTime.Now;
|
|
|
+ }
|
|
|
|
|
|
//--2024-03-21 增加了PortJobWaferStart 上报事件 start--
|
|
|
if (ModuleHelper.IsLoadPort(item.SourceModule))
|
|
@@ -3199,7 +3253,6 @@ namespace Venus_RT.Modules
|
|
|
hands.Add((Hand)i);
|
|
|
}
|
|
|
}
|
|
|
-
|
|
|
return hands;
|
|
|
}
|
|
|
|
|
@@ -3527,7 +3580,7 @@ namespace Venus_RT.Modules
|
|
|
var inLLWafers = _lstWaferTasks.FindAll(wt => wt.currentMod == inLL).ToList();
|
|
|
|
|
|
int pickCount = 0;
|
|
|
- foreach(var inWafer in inLLWafers)
|
|
|
+ foreach (var inWafer in inLLWafers)
|
|
|
{
|
|
|
if (!_dictModuleTask[inWafer.destMod].HasWafer && _dictModuleTask[inWafer.destMod].TimeToReady < 20 && pickCount < freeHands.Count)
|
|
|
{
|
|
@@ -3537,21 +3590,21 @@ namespace Venus_RT.Modules
|
|
|
}
|
|
|
}
|
|
|
|
|
|
- if(pickCount == 2)
|
|
|
+ if (pickCount == 2)
|
|
|
{
|
|
|
_tmSchdActions.Enqueue(pickActions);
|
|
|
- foreach(var ac in placeActions)
|
|
|
+ foreach (var ac in placeActions)
|
|
|
{
|
|
|
_tmSchdActions.Enqueue(new List<MoveItem> { ac });
|
|
|
}
|
|
|
}
|
|
|
- else if(pickCount == 1)
|
|
|
+ else if (pickCount == 1)
|
|
|
{
|
|
|
- if(inLLWafers.Count == 2 && freeHands.Count == 2)
|
|
|
+ if (inLLWafers.Count == 2 && freeHands.Count == 2)
|
|
|
{
|
|
|
var peerSlot = 1 - pickActions.First().SourceSlot;
|
|
|
var peerWafer = inLLWafers.Find(wt => wt.currentSlot == peerSlot);
|
|
|
- if(peerWafer != null && _dictModuleTask[peerWafer.destMod].TimeToReady < 60)
|
|
|
+ if (peerWafer != null && _dictModuleTask[peerWafer.destMod].TimeToReady < 60)
|
|
|
{
|
|
|
pickActions.Add(new MoveItem(peerWafer.currentMod, peerWafer.currentSlot, ModuleName.TMRobot, (int)freeHands[pickCount], freeHands[pickCount]));
|
|
|
}
|
|
@@ -4146,7 +4199,7 @@ namespace Venus_RT.Modules
|
|
|
destLL = readyPushInLLs.Last().Key;
|
|
|
}
|
|
|
|
|
|
- var pickHand = SelectTMPickArm(destLL);
|
|
|
+ var pickHand = SelectTMPickArm(destLL);
|
|
|
var llWafers = _lstWaferTasks.Where(x => x.currentMod == destLL).ToList();
|
|
|
if (llWafers.Count() == 0)
|
|
|
{
|
|
@@ -4154,11 +4207,11 @@ namespace Venus_RT.Modules
|
|
|
}
|
|
|
for (int i = 0; i < llWafers.Count(); i++)
|
|
|
{
|
|
|
- if (_lstWaferTasks.Count(x => ModuleHelper.IsPm(x.currentMod) && llWafers[i].destMod==x.currentMod)==0)
|
|
|
+ if (_lstWaferTasks.Count(x => ModuleHelper.IsPm(x.currentMod) && llWafers[i].destMod == x.currentMod) == 0)
|
|
|
{
|
|
|
_tmSchdActions.Enqueue(new List<MoveItem> { new MoveItem(destLL, GetLLFixedReadyInOutSlots(destLL).tOutSlot.First(), ModuleName.TMRobot, (int)pickHand, pickHand) });
|
|
|
}
|
|
|
- }
|
|
|
+ }
|
|
|
//if (_lstWaferTasks.Count(x => ModuleHelper.IsPm(x.currentMod)) == 0)
|
|
|
//{
|
|
|
//_tmSchdActions.Enqueue(new List<MoveItem> { new MoveItem(destLL, GetLLFixedReadyInOutSlots(destLL).tOutSlot.First(), ModuleName.TMRobot, (int)pickHand, pickHand) });
|
|
@@ -4705,7 +4758,7 @@ namespace Venus_RT.Modules
|
|
|
schdPumpingLLs.Add(ModuleName.LLA);
|
|
|
}
|
|
|
else if ((llaWaferStatus.outSlot.Count == _LLASlotNumber) ||
|
|
|
- (llaWaferStatus.outSlot.Count > 0 && llaWaferStatus.outSlot.Count < _LLASlotNumber && !NeedPairingForNexWafer()))
|
|
|
+ (llaWaferStatus.outSlot.Count > 0 && llaWaferStatus.outSlot.Count < _LLASlotNumber))
|
|
|
{
|
|
|
if (!schdVentingLLs.Contains(ModuleName.LLA))
|
|
|
schdCoolingLLs.Add(ModuleName.LLA);
|
|
@@ -4718,7 +4771,7 @@ namespace Venus_RT.Modules
|
|
|
schdPumpingLLs.Add(ModuleName.LLB);
|
|
|
}
|
|
|
else if ((llbWaferStatus.outSlot.Count == _LLBSlotNumber) ||
|
|
|
- (llbWaferStatus.outSlot.Count > 0 && llbWaferStatus.outSlot.Count < _LLBSlotNumber && !NeedPairingForNexWafer()))
|
|
|
+ (llbWaferStatus.outSlot.Count > 0 && llbWaferStatus.outSlot.Count < _LLBSlotNumber))
|
|
|
{
|
|
|
if (!schdVentingLLs.Contains(ModuleName.LLB))
|
|
|
schdCoolingLLs.Add(ModuleName.LLB);
|
|
@@ -4781,7 +4834,10 @@ namespace Venus_RT.Modules
|
|
|
var preClean = pj.Sequence.GetPreCleanRecipe(pm.Key);
|
|
|
if (!string.IsNullOrWhiteSpace(preClean))
|
|
|
{
|
|
|
- pmTask.InvokePreJobClean(preClean);
|
|
|
+ if (DateTime.Now.Subtract(_pmLastWaferEndTime[pm.Key]).TotalSeconds > _preCleanMaxIdleTime)
|
|
|
+ {
|
|
|
+ pmTask.InvokePreJobClean(preClean);
|
|
|
+ }
|
|
|
_preLotCleanMarks[pm.Key].Add(waitInWafer.lotId);
|
|
|
}
|
|
|
}
|
|
@@ -5011,7 +5067,7 @@ namespace Venus_RT.Modules
|
|
|
return;
|
|
|
|
|
|
var llActions = _curTmAction.Find(ac => ModuleHelper.IsLoadLock(ac.DestinationModule));
|
|
|
- if(llActions != null && !_qeWaitCoolingLLs.Contains(llActions.DestinationModule) && !RouteManager.IsATMMode)
|
|
|
+ if (llActions != null && !_qeWaitCoolingLLs.Contains(llActions.DestinationModule) && !RouteManager.IsATMMode)
|
|
|
{
|
|
|
_qeWaitCoolingLLs.Enqueue(llActions.DestinationModule);
|
|
|
}
|