| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476 | using Aitex.Core.Common;using Aitex.Core.RT.Log;using Aitex.Core.Util;using MECF.Framework.Common.Equipment;using MECF.Framework.Common.Jobs;using MECF.Framework.Common.RecipeCenter;using MECF.Framework.Common.SubstrateTrackings;using PunkHPX8_RT.Modules;using PunkHPX8_RT.Schedulers;using System;using System.Collections.Concurrent;using System.Collections.Generic;using System.Linq;using System.Text;using System.Threading.Tasks;using System.Windows.Markup;using Aitex.Core.RT.SCCore;using MECF.Framework.Common.ToolLayout;using PunkHPX8_RT.Devices.EFEM;using Aitex.Core.RT.Device;using MECF.Framework.RT.Core.Equipments;using PunkHPX8_RT.Modules.VpwMain;namespace PunkHPX8_RT.Dispatch{    public class WaferTaskManager : Singleton<WaferTaskManager>    {        #region 常量        private const string SIDE_A = "SideA";        private const string SIDE_B = "SideB";        #endregion        #region 内部变量        /// <summary>        /// FA回复        /// </summary>        private SchedulerFACallback _schedulerFACallback = new SchedulerFACallback();        /// <summary>        /// 任务锁        /// </summary>        private object _taskLocker = new object();        /// <summary>        /// 任务集合        /// </summary>        private List<WaferTask> _waferTaskList = new List<WaferTask>();        /// <summary>        /// Wafer task字典        /// </summary>        private ConcurrentDictionary<string, WaferTask> _waferTasksDic = new ConcurrentDictionary<string, WaferTask>();        /// <summary>        /// Wafer配对字典(key-自身WaferId,value-配对WaferId)        /// </summary>        private ConcurrentDictionary<string, string> _waferTaskMatchDic = new ConcurrentDictionary<string, string>();                #endregion        /// <summary>        /// 初始化Task        /// </summary>        /// <param name="task"></param>        public void InitializeTask(WaferTask task)        {            _waferTasksDic[task.WaferId] = task;            lock(_taskLocker)            {                _waferTaskList.Add(task);            }        }        /// <summary>        /// 获取WaferTask        /// </summary>        /// <param name="waferId"></param>        /// <returns></returns>        public WaferTask GetWaferTask(string waferId)        {            return _waferTasksDic.ContainsKey(waferId) ? _waferTasksDic[waferId] : null;        }        /// <summary>        /// 是否包含        /// </summary>        /// <param name="waferId"></param>        /// <returns></returns>        public bool Contains(string waferId)        {            return _waferTasksDic.ContainsKey(waferId);        }        /// <summary>        /// 移除任务        /// </summary>        /// <param name="waferId"></param>        public void RemoveWaferTask(string waferId)        {            if(_waferTasksDic.ContainsKey(waferId))            {                _waferTasksDic.TryRemove(waferId, out var task);                task.Dispose();                task.OnTaskComplete -= WaferTask_OnTaskComplete;                LOG.WriteBackgroundLog(eEvent.EV_SCHEDULER, "Scheduler", $"remove wafer {waferId} task");            }            if(_waferTaskMatchDic.ContainsKey(waferId))            {                _waferTaskMatchDic.TryRemove(waferId,out var taskMath);            }            lock(_taskLocker)            {                int index = _waferTaskList.FindIndex(O => O.WaferId == waferId);                if (index >= 0)                {                    _waferTaskList.RemoveAt(index);                }            }        }        /// <summary>        /// 加载Wafer Task集合        /// </summary>        /// <returns></returns>        public List<WaferTask> LoadWaferTaskList()        {            return _waferTaskList.ToList();         }        /// <summary>        /// 检验是否可以创建Wafer Task        /// </summary>        /// <returns></returns>        public bool CheckEnableCreateWaferTask()        {            lock (_taskLocker)            {                if (_waferTaskList.Count == 0)                {                    return true;                }                if (WaferManager.Instance.CheckHasWafer(ModuleName.Aligner1, 0))                {                    return false;                }                foreach (WaferTask item in _waferTaskList)                {                    //Wafer还未离开Aligner                    if (item.CurrentSequenceIndex <= 1)                    {                        return false;                    }                }                return true;            }        }        /// <summary>        /// 获取下一个执行的ProcessJob        /// </summary>        /// <param name="jobs"></param>        /// <returns></returns>        private ProcessJobInfo GetNextProcessJob(List<ProcessJobInfo> jobs, List<string> executedPPids)        {            try            {                bool isRunInParallelMode = SC.GetValue<bool>("Scheduler.IsRunInParallelMode");                if (!isRunInParallelMode)                {                    return jobs[0];                }                else                {                    if (_waferTaskList.Count == 0)                    {                        return jobs[0];                    }                    WaferTask lastWaferTask = _waferTaskList[_waferTaskList.Count - 1];                    string lastSequence = lastWaferTask.ProcessJobInfo.SequenceRecipe?.Ppid;                    List<ProcessJobInfo> lastSamePjs = new List<ProcessJobInfo>();                    List<ProcessJobInfo> otherPjs = new List<ProcessJobInfo>();                    int lastIndex = jobs.FindIndex(O => O.SequenceRecipe?.Ppid == lastSequence);                    int nextIndex = (lastIndex == -1 || lastIndex >= jobs.Count - 1) ? 0 : lastIndex + 1;                    for (int i = nextIndex; i < jobs.Count; i++)                    {                        ProcessJobInfo processJobInfo = jobs[i];                        if (processJobInfo.SequenceRecipe?.Ppid == lastSequence)                        {                            lastSamePjs.Add(processJobInfo);                            continue;                        }                        if (executedPPids.Contains(processJobInfo.SequenceRecipe?.Ppid))                        {                            continue;                        }                        return processJobInfo;                    }                    for (int i = 0; i < lastIndex; i++)                    {                        ProcessJobInfo processJobInfo = jobs[i];                        if (processJobInfo.SequenceRecipe?.Ppid == lastSequence)                        {                            lastSamePjs.Add(processJobInfo);                            continue;                        }                        if (executedPPids.Contains(processJobInfo.SequenceRecipe?.Ppid))                        {                            continue;                        }                        return processJobInfo;                    }                    if (lastSamePjs.Count != 0)                    {                        return lastSamePjs[0];                    }                    return jobs[0];                }            }            catch            {                return jobs[0];            }        }        /// <summary>        /// 创建Wafer任务        /// </summary>        public void CreateWaferTask(List<ProcessJobInfo> processJobInfos)        {            int count = 0;            string mateWaferTask = "";            List<WaferInfo> wafers = new List<WaferInfo>();            if (processJobInfos.Count == 0)            {                return;            }            List<string> excludes = new List<string>();            ProcessJobInfo processJob = GetNextProcessJob(processJobInfos, excludes);            if (processJob == null)            {                return;            }            for(int i=0; i<processJobInfos.Count;i++)            {                ProcessJobInfo processJobInfo = processJobInfos[i];                if (processJobInfo.SequenceRecipe?.Ppid != processJob.SequenceRecipe?.Ppid)                {                    continue;                }                for(int j=0;j< processJobInfo.SlotWafers.Count; j++)                {                    var item = processJobInfo.SlotWafers[j];                    if (WaferManager.Instance.CheckHasWafer(item.Item1, item.Item2))                    {                        WaferInfo waferInfo = WaferManager.Instance.GetWafer(item.Item1, item.Item2);                        if (waferInfo != null && waferInfo.ProcessState == EnumWaferProcessStatus.Idle)                        {                            if (!Contains(waferInfo.WaferID))                            {                                if (count == 0)                                {                                    wafers.Add(waferInfo);                                    count++;                                }                                else                                {                                    wafers.Add(waferInfo);                                    break;                                }                            }                        }                    }                }            }            List<VpwCellEntity> vpwCellEntities = SchedulerSequenceManager.Instance.GetAvaibleVpwList(processJob.SequenceRecipe.SubstrateSize, false);            if (vpwCellEntities.Count == 0)            {                return;            }            SequenceRecipe sequenceRecipe = processJob.SequenceRecipe;            if (wafers.Count == 1)            {                WaferInfo dummyWaferInfo = GetDummyWafer(sequenceRecipe.SubstrateSize);                if (dummyWaferInfo == null)                {                    return;                }                CreateWaferTaskSchedulerSequence(wafers[0], sequenceRecipe, "", vpwCellEntities[0].Module);                mateWaferTask = wafers[0].WaferID;                CreateDummyWaferTaskSchedulerSequence(sequenceRecipe, dummyWaferInfo, mateWaferTask, vpwCellEntities.Count == 1 ? vpwCellEntities[0].Module : vpwCellEntities[1].Module);                _waferTaskMatchDic[wafers[0].WaferID] = dummyWaferInfo.WaferID;                 LOG.WriteLog(eEvent.EV_SEQUENCE, "Scheduler", $"wafer {wafers[0].WaferID} match {dummyWaferInfo.WaferID}");                _waferTaskMatchDic[dummyWaferInfo.WaferID] = wafers[0].WaferID;                LOG.WriteLog(eEvent.EV_SEQUENCE, "Scheduler", $"wafer {dummyWaferInfo.WaferID} match {wafers[0].WaferID}");            }            else if (wafers.Count >= 2)            {                CreateWaferTaskSchedulerSequence(wafers[0], sequenceRecipe, "",vpwCellEntities[0].Module);                mateWaferTask = wafers[0].WaferID;                CreateWaferTaskSchedulerSequence(wafers[1], sequenceRecipe, mateWaferTask, vpwCellEntities.Count == 1 ? vpwCellEntities[0].Module : vpwCellEntities[1].Module);                _waferTaskMatchDic[wafers[0].WaferID] = wafers[1].WaferID;                _waferTaskMatchDic[wafers[1].WaferID] = wafers[0].WaferID;            }        }        /// <summary>        /// 获取Dummy Wafer        /// </summary>        /// <param name="waferSize"></param>        /// <returns></returns>        private WaferInfo GetDummyWafer(int waferSize)        {            List<string> dummmys= DummyCasseteItemManager.Instance.InstalledModules;            foreach(string item in dummmys)            {                if (Enum.TryParse(item, out ModuleName dummyModule))                {                    DummyDevice dummyDevice= Singleton<RouteManager>.Instance.EFEM.GetDummyDevice(dummyModule - ModuleName.Dummy1);                    if ((int)dummyDevice.WaferSize != waferSize)                    {                        continue;                    }                    int maxSlot = dummyDevice.MaxSlotNumber;                    for (int i = 0; i < maxSlot; i++)                    {                        if (WaferManager.Instance.CheckHasWafer(item, i))                        {                            return WaferManager.Instance.GetWafer(dummyModule, i);                        }                    }                }            }            return null;        }        /// <summary>        /// 创建WaferTask SchedulerSequence        /// </summary>        /// <param name="waferInfo"></param>        /// <param name="pufModuleName"></param>        /// <param name="waferHolderInfo"></param>        /// <param name="sequenceRecipe"></param>        private void CreateWaferTaskSchedulerSequence(WaferInfo waferInfo, SequenceRecipe sequenceRecipe, string mateWafeTask,ModuleName vpwModule)        {            List<SchedulerSequence> sequences = SchedulerSequenceManager.Instance.AnalyWaferAllSchedulerSequence(waferInfo,vpwModule,sequenceRecipe);            WaferTask waferTask = new WaferTask(waferInfo, sequences);            if (!string.IsNullOrEmpty(mateWafeTask))            {                waferTask.MateWaferTask = mateWafeTask;            }            waferTask.OnTaskComplete += WaferTask_OnTaskComplete;            LOG.WriteLog(eEvent.EV_SEQUENCE, "Scheduler", $"Create wafer {waferInfo.WaferID} task");            InitializeTask(waferTask);        }        /// <summary>        /// 创建Dummy Wafer Task 调度工序        /// </summary>        /// <param name="waferInfo"></param>        /// <param name="pufModuleName"></param>        /// <param name="mateWaferTask"></param>        private void CreateDummyWaferTaskSchedulerSequence(SequenceRecipe sequenceRecipe,WaferInfo waferInfo,string mateWaferTask,ModuleName vpwModuleName)        {            List<SchedulerSequence> sequences = SchedulerSequenceManager.Instance.AnalyDummyWaferAllSchedulerSequence(sequenceRecipe,vpwModuleName,waferInfo);                        WaferTask waferTask = new WaferTask(waferInfo,sequences);            if (!string.IsNullOrEmpty(mateWaferTask))            {                waferTask.MateWaferTask = mateWaferTask;            }            LOG.WriteLog(eEvent.EV_SEQUENCE, "Scheduler", $"Create Dummy wafer {waferInfo.WaferID} task");            waferTask.OnTaskStart += WaferTask_OnTaskStart;            waferTask.OnTaskComplete += WaferTask_OnTaskComplete;            InitializeTask(waferTask);        }        /// <summary>        /// 任务开始        /// </summary>        /// <param name="id"></param>        /// <exception cref="NotImplementedException"></exception>        private void WaferTask_OnTaskStart(string id)        {            WaferInfo waferInfo = WaferManager.Instance.GetWaferByWaferId(id);            if (waferInfo != null && waferInfo.WaferType == WaferType.Production)            {                ModuleName moduleName = (ModuleName)waferInfo.OriginStation;                if (ModuleHelper.IsLoadPort(moduleName))                {                    CycleManager.Instance.UpdateCycleWaferCount(moduleName.ToString());                }                _schedulerFACallback.JobWaferStart(JobProcesser.Instance.GetControlJobInfoByProcessJob(waferInfo.ProcessJob),                    waferInfo.SequenceName, waferInfo.OriginSlot);            }        }        /// <summary>        ///  Wafer task完成事件        /// </summary>        /// <param name="id"></param>        private void WaferTask_OnTaskComplete(string id)        {            WaferInfo waferInfo=WaferManager.Instance.GetWaferByWaferId(id);            if (waferInfo != null&&waferInfo.WaferType==WaferType.Production)            {                ModuleName moduleName = (ModuleName)waferInfo.OriginStation;                if (ModuleHelper.IsLoadPort(moduleName))                {                    CycleManager.Instance.UpdateCycleWaferCount(moduleName.ToString());                }                _schedulerFACallback.JobWaferEnd(JobProcesser.Instance.GetControlJobInfoByProcessJob(waferInfo.ProcessJob),                     waferInfo.SequenceName, waferInfo.OriginSlot);            }            RemoveWaferTask(id);        }        /// <summary>        /// 检验所有task是否完成        /// </summary>        /// <returns></returns>        public bool CheckAllTaskComplete()        {            return _waferTaskList.Count == 0;        }        /// <summary>        /// 暂停        /// </summary>        public void PauseAllTask()        {            List<WaferTask> waferTasks = _waferTaskList.ToList();            foreach(WaferTask item in waferTasks)            {                item.Pause();            }        }        /// <summary>        /// 恢复所有任务        /// </summary>        public void ResumeAllTask()        {            List<WaferTask> waferTasks = _waferTaskList.ToList();            foreach (WaferTask item in waferTasks)            {                item.Resume();            }        }        /// <summary>        /// 恢复所有任务        /// </summary>        public void RemoveAllTask()        {            List<WaferTask> waferTasks = _waferTaskList.ToList();            foreach (WaferTask item in waferTasks)            {                RemoveWaferTask(item.WaferId);            }        }        /// <summary>        /// 根据WaferId获取配对另一个WaferId        /// </summary>        /// <param name="waferId"></param>        /// <returns></returns>        public string GetMatchWaferIdByWaferId(string waferId)        {            return _waferTaskMatchDic.ContainsKey(waferId) ? _waferTaskMatchDic[waferId] : "";        }        /// <summary>        /// 检验Wafer存在匹配的wafer任务(wafer是单片任务还是双片任务)        /// </summary>        /// <param name="waferId"></param>        /// <returns></returns>        public string CheckWaferHasMatch(string waferId)        {            if (_waferTaskMatchDic.ContainsKey(waferId))            {                return _waferTaskMatchDic[waferId];            }            else            {                return "";            }        }    }}
 |