| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593 | 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 MECF.Framework.Common.WaferHolder;using CyberX8_RT.Modules.PUF;using CyberX8_RT.Modules;using CyberX8_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;namespace CyberX8_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>();        /// <summary>        /// Wafer--WaferHolder任务字典(key-自身WaferId,value-WaferHolder任务)        /// </summary>        private ConcurrentDictionary<string, WaferHolderTask> _waferWaferHolderTaskDic = new ConcurrentDictionary<string, WaferHolderTask>();        #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任务        /// </summary>        public (int productWaferCount,int dummyWaferCount) CreateWaferTask(List<ProcessJobInfo> processJobInfos, WaferHolderInfo waferHolderInfo,WaferHolderTask waferHolderTask)        {            int count = 0;            string mateWaferTask = "";            List<WaferInfo> wafers = new List<WaferInfo>();            if (processJobInfos.Count == 0)            {                return (0,0);            }            for(int i=0; i<processJobInfos.Count;i++)            {                ProcessJobInfo processJobInfo = processJobInfos[i];                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;                                }                            }                        }                    }                }            }            SequenceRecipe sequenceRecipe = processJobInfos[0].SequenceRecipe;            if (wafers.Count == 1)            {                WaferInfo waferInfo = wafers[0];                ModuleName pufModuleName = ModuleName.PUF1;                string waferSide = "";                string dummySide = "";                bool lastSingleWaferToSideB = sequenceRecipe.LastSingleWaferToSideB;                if (lastSingleWaferToSideB)                {                    if (CheckLoaderHasSameWaferSize())                    {                        waferSide = SIDE_B;                        dummySide = SIDE_A;                    }                    else                    {                        string side = AnalyseLoadSide(sequenceRecipe);                        waferSide = side;                        dummySide = side;                    }                }                else                {                    if (CheckLoaderHasSameWaferSize())                    {                        waferSide = SIDE_A;                        dummySide = SIDE_B;                    }                    else                    {                        string side = AnalyseLoadSide(sequenceRecipe);                        waferSide = side;                        dummySide = side;                    }                }                if (CheckWaferHolderHasSameTypeWafers(waferHolderInfo))                {                    List<WaferInfo> dummyWafers = DummyWaferManager.Instance.LoadDummyWafersByWaferHolder(waferHolderInfo);                    if (dummyWafers.Count != 0)                    {                        CreateWaferTaskSchedulerSequence(waferInfo, pufModuleName, waferHolderInfo, waferInfo.ProcessJob.SequenceRecipe, "",waferSide);                        WaferInfo dummyInfo = null;                        if (dummyWafers.Count > 1)                        {                            foreach (WaferInfo info in dummyWafers)                            {                                //最后一片Wafer使用B面,那么dummy片使用0 slot                                if (lastSingleWaferToSideB && info.OriginSlot == 0)                                {                                    dummyInfo = info;                                    break;                                }                                //最后一片Wafer使用A面,那么dummy片使用1 slot                                if (!lastSingleWaferToSideB && info.OriginSlot == 1)                                {                                    dummyInfo = info;                                    break;                                }                            }                            if (dummyInfo == null)                            {                                dummyInfo = dummyWafers[0];                            }                        }                        else                        {                            dummyInfo = dummyWafers[0];                        }                        CreateDummyWaferTaskSchedulerSequence(waferHolderInfo.SequenceRecipe, dummyInfo, pufModuleName, waferInfo.WaferID,dummySide);                        _waferTaskMatchDic[waferInfo.WaferID] = dummyWafers[0].WaferID;                        _waferTaskMatchDic[dummyWafers[0].WaferID] = waferInfo.WaferID;                        _waferWaferHolderTaskDic[waferInfo.WaferID] = waferHolderTask;                        _waferWaferHolderTaskDic[dummyWafers[0].WaferID] = waferHolderTask;                        return (1,1);                    }                    else                    {                        CreateWaferTaskSchedulerSequence(waferInfo, pufModuleName, waferHolderInfo, waferInfo.ProcessJob.SequenceRecipe, "",waferSide);                        _waferWaferHolderTaskDic[waferInfo.WaferID] = waferHolderTask;                        return (1,0);                    }                }                else                {                    CreateWaferTaskSchedulerSequence(waferInfo, pufModuleName, waferHolderInfo, waferInfo.ProcessJob.SequenceRecipe, "",waferSide);                    _waferWaferHolderTaskDic[waferInfo.WaferID] = waferHolderTask;                    return (1, 0);                }            }            else if (wafers.Count >= 2)            {                for (int i = 0; i < 2; i++)                {                    WaferInfo waferInfo = wafers[i];                    ModuleName pufModuleName = ModuleName.PUF1;                    if (i == 0)                    {                        mateWaferTask = wafers[0].WaferID;                        if (CheckLoaderHasSameWaferSize())                        {                            CreateWaferTaskSchedulerSequence(waferInfo, pufModuleName, waferHolderInfo, waferInfo.ProcessJob.SequenceRecipe, "", SIDE_A);                        }                        else                        {                            string side = AnalyseLoadSide(sequenceRecipe);                            CreateWaferTaskSchedulerSequence(waferInfo, pufModuleName, waferHolderInfo, waferInfo.ProcessJob.SequenceRecipe, "", side);                        }                    }                    else                    {                        if (CheckLoaderHasSameWaferSize())                        {                            CreateWaferTaskSchedulerSequence(waferInfo, pufModuleName, waferHolderInfo, waferInfo.ProcessJob.SequenceRecipe, mateWaferTask, SIDE_B);                        }                        else                        {                            string side = AnalyseLoadSide(sequenceRecipe);                             CreateWaferTaskSchedulerSequence(waferInfo, pufModuleName, waferHolderInfo, waferInfo.ProcessJob.SequenceRecipe, mateWaferTask, side);                        }                    }                }                _waferTaskMatchDic[wafers[0].WaferID] = wafers[1].WaferID;                _waferTaskMatchDic[wafers[1].WaferID] = wafers[0].WaferID;                _waferWaferHolderTaskDic[wafers[0].WaferID] = waferHolderTask;                _waferWaferHolderTaskDic[wafers[1].WaferID] = waferHolderTask;                return (2, 0);            }            return (0,0);        }        /// <summary>        /// 检验Loader两边是不是尺寸一致        /// </summary>        /// <returns></returns>        private bool CheckLoaderHasSameWaferSize()        {            int sideAWaferSize = SC.GetValue<int>($"Loader1.SideAWaferSize");            int sideBWaferSize = SC.GetValue<int>($"Loader1.SideBWaferSize");            return sideAWaferSize == sideBWaferSize;        }        /// <summary>        /// Loader两边尺寸不一致,解析Loader使用side        /// </summary>        /// <param name="recipe"></param>        /// <returns></returns>        private string AnalyseLoadSide(SequenceRecipe recipe)        {            int sideAWaferSize = SC.GetValue<int>($"Loader1.SideAWaferSize");            int sideBWaferSize = SC.GetValue<int>($"Loader1.SideBWaferSize");            if (sideAWaferSize != sideBWaferSize)            {                if(recipe.SubstrateSize==sideAWaferSize)                {                    return SIDE_A;                }                else                {                    return SIDE_B;                }            }            return "";        }        /// <summary>        /// 检验WaferHolder是否存在两片生产片        /// </summary>        /// <param name="waferHolderInfo"></param>        /// <returns></returns>        private bool CheckWaferHolderHasSameTypeWafers(WaferHolderInfo waferHolderInfo)        {            int count = 0;            WaferInfo waferAInfo = WaferManager.Instance.GetWaferByWaferId(waferHolderInfo.WaferAId);            WaferType waferType = WaferType.Production;            if (waferAInfo != null)            {                waferType = waferAInfo.WaferType;                count++;            }            WaferInfo waferBInfo = WaferManager.Instance.GetWaferByWaferId(waferHolderInfo.WaferBId);            if (waferBInfo != null)            {                if (waferBInfo.WaferType == waferType)                {                    count++;                }            }            return count >= 2;        }        /// <summary>        /// 检验WaferHolder是否存在两片生产片        /// </summary>        /// <param name="waferHolderInfo"></param>        /// <returns></returns>        private bool CheckWaferHolderHasTwoProductionWafers(WaferHolderInfo waferHolderInfo)        {            int count = 0;            WaferInfo waferAInfo = WaferManager.Instance.GetWaferByWaferId(waferHolderInfo.WaferAId);            if (waferAInfo != null && waferAInfo.WaferType==WaferType.Production)            {                count++;            }            WaferInfo waferBInfo = WaferManager.Instance.GetWaferByWaferId(waferHolderInfo.WaferBId);            if (waferBInfo != null && waferBInfo.WaferType == WaferType.Production)            {                count++;            }            return count >= 2;        }        /// <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, ModuleName pufModuleName, WaferHolderInfo waferHolderInfo, SequenceRecipe sequenceRecipe, string mateWafeTask,string side)        {            List<SchedulerSequence> sequences = SchedulerSequenceManager.Instance.AnalyWaferAllSchedulerSequence(waferInfo, pufModuleName,side, waferHolderInfo, sequenceRecipe);            PUFEntity pufEntity = Singleton<RouteManager>.Instance.GetModule<PUFEntity>(pufModuleName.ToString());            WaferTask waferTask = new WaferTask(waferInfo, pufEntity, 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>        /// 创建DummyWaferTask        /// </summary>        /// <param name="waferInfos"></param>        public int CreateDummyWaferTask(List<WaferInfo> waferInfos,SequenceRecipe sequenceRecipe,WaferHolderTask waferHolderTask)        {            string mateWaferTask = "";            int waferCount = 0;            for (int i = 0; i < waferInfos.Count; i++)            {                WaferInfo waferInfo = waferInfos[i];                int orginalSlot = waferInfo.OriginSlot;                ModuleName pufModuleName = ModuleName.PUF1;                string side = orginalSlot == 0 ? SIDE_A : SIDE_B;                if(!CheckLoaderHasSameWaferSize())                {                    side = AnalyseLoadSide(sequenceRecipe);                }                waferCount = i;                if (!Contains(waferInfo.WaferID))                {                    if (i == 0)                    {                        mateWaferTask = waferInfo.WaferID;                        CreateDummyWaferTaskSchedulerSequence(sequenceRecipe,waferInfo, pufModuleName,"", side);                        _waferWaferHolderTaskDic[waferInfo.WaferID] = waferHolderTask;                    }                    else                    {                        CreateDummyWaferTaskSchedulerSequence(sequenceRecipe,waferInfo, pufModuleName, mateWaferTask,side);                        _waferTaskMatchDic[mateWaferTask] = waferInfo.WaferID;                        _waferTaskMatchDic[waferInfo.WaferID] = mateWaferTask;                        _waferWaferHolderTaskDic[waferInfo.WaferID]= waferHolderTask;                        break;                    }                }            }            return waferCount + 1;        }        /// <summary>        /// 创建Dummy Wafer Task 调度工序        /// </summary>        /// <param name="waferInfo"></param>        /// <param name="pufModuleName"></param>        /// <param name="mateWaferTask"></param>        private void CreateDummyWaferTaskSchedulerSequence(SequenceRecipe sequenceRecipe,WaferInfo waferInfo,ModuleName pufModuleName,string mateWaferTask,string side)        {            List<SchedulerSequence> sequences = SchedulerSequenceManager.Instance.AnalyDummyWaferAllSchedulerSequence(sequenceRecipe,waferInfo,pufModuleName, side);            PUFEntity pufEntity = Singleton<RouteManager>.Instance.GetModule<PUFEntity>(pufModuleName.ToString());            WaferTask waferTask = new WaferTask(waferInfo, pufEntity,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>        /// 移除WaferId匹配WaferHolderTask字典        /// </summary>        /// <param name="waferId"></param>        public void RemoveWaferIdMatchWaferHolderTaskDic(string waferId)        {            if(_waferWaferHolderTaskDic.ContainsKey(waferId))            {                _waferWaferHolderTaskDic.TryRemove(waferId, out var waferHolderTask);            }        }        /// <summary>        /// 获取WaferId匹配WaferHolderTask字典        /// </summary>        /// <param name="waferId"></param>        /// <returns></returns>        public WaferHolderTask GetWaferHolderTaskByWaferId(string waferId)        {            return _waferWaferHolderTaskDic.ContainsKey(waferId) ? _waferWaferHolderTaskDic[waferId] : null;        }        /// <summary>        /// 获取WaferHolder Task对应的Wafer Task WaferId集合        /// </summary>        /// <param name="waferHolderId"></param>        /// <returns></returns>        public List<string> GetWaferHolderMathWaferId(string waferHolderId)        {            List<string> keys = _waferWaferHolderTaskDic.Keys.ToList();            List<string> result = new List<string>();            foreach(string item in keys)            {                if(_waferWaferHolderTaskDic[item].WaferHolderInfo.Id==waferHolderId)                {                    result.Add(item);                }            }            return result;        }        /// <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 "";            }        }    }}
 |