SystemDispatcher.cs 236 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463246424652466246724682469247024712472247324742475247624772478247924802481248224832484248524862487248824892490249124922493249424952496249724982499250025012502250325042505250625072508250925102511251225132514251525162517251825192520252125222523252425252526252725282529253025312532253325342535253625372538253925402541254225432544254525462547254825492550255125522553255425552556255725582559256025612562256325642565256625672568256925702571257225732574257525762577257825792580258125822583258425852586258725882589259025912592259325942595259625972598259926002601260226032604260526062607260826092610261126122613261426152616261726182619262026212622262326242625262626272628262926302631263226332634263526362637263826392640264126422643264426452646264726482649265026512652265326542655265626572658265926602661266226632664266526662667266826692670267126722673267426752676267726782679268026812682268326842685268626872688268926902691269226932694269526962697269826992700270127022703270427052706270727082709271027112712271327142715271627172718271927202721272227232724272527262727272827292730273127322733273427352736273727382739274027412742274327442745274627472748274927502751275227532754275527562757275827592760276127622763276427652766276727682769277027712772277327742775277627772778277927802781278227832784278527862787278827892790279127922793279427952796279727982799280028012802280328042805280628072808280928102811281228132814281528162817281828192820282128222823282428252826282728282829283028312832283328342835283628372838283928402841284228432844284528462847284828492850285128522853285428552856285728582859286028612862286328642865286628672868286928702871287228732874287528762877287828792880288128822883288428852886288728882889289028912892289328942895289628972898289929002901290229032904290529062907290829092910291129122913291429152916291729182919292029212922292329242925292629272928292929302931293229332934293529362937293829392940294129422943294429452946294729482949295029512952295329542955295629572958295929602961296229632964296529662967296829692970297129722973297429752976297729782979298029812982298329842985298629872988298929902991299229932994299529962997299829993000300130023003300430053006300730083009301030113012301330143015301630173018301930203021302230233024302530263027302830293030303130323033303430353036303730383039304030413042304330443045304630473048304930503051305230533054305530563057305830593060306130623063306430653066306730683069307030713072307330743075307630773078307930803081308230833084308530863087308830893090309130923093309430953096309730983099310031013102310331043105310631073108310931103111311231133114311531163117311831193120312131223123312431253126312731283129313031313132313331343135313631373138313931403141314231433144314531463147314831493150315131523153315431553156315731583159316031613162316331643165316631673168316931703171317231733174317531763177317831793180318131823183318431853186318731883189319031913192319331943195319631973198319932003201320232033204320532063207320832093210321132123213321432153216321732183219322032213222322332243225322632273228322932303231323232333234323532363237323832393240324132423243324432453246324732483249325032513252325332543255325632573258325932603261326232633264326532663267326832693270327132723273327432753276327732783279328032813282328332843285328632873288328932903291329232933294329532963297329832993300330133023303330433053306330733083309331033113312331333143315331633173318331933203321332233233324332533263327332833293330333133323333333433353336333733383339334033413342334333443345334633473348334933503351335233533354335533563357335833593360336133623363336433653366336733683369337033713372337333743375337633773378337933803381338233833384338533863387338833893390339133923393339433953396339733983399340034013402340334043405340634073408340934103411341234133414341534163417341834193420342134223423342434253426342734283429343034313432343334343435343634373438343934403441344234433444344534463447344834493450345134523453345434553456345734583459346034613462346334643465346634673468346934703471347234733474347534763477347834793480348134823483348434853486348734883489349034913492349334943495349634973498349935003501350235033504350535063507350835093510351135123513351435153516351735183519352035213522352335243525352635273528352935303531353235333534353535363537353835393540354135423543354435453546354735483549355035513552355335543555355635573558355935603561356235633564356535663567356835693570357135723573357435753576357735783579358035813582358335843585358635873588358935903591359235933594359535963597359835993600360136023603360436053606360736083609361036113612361336143615361636173618361936203621362236233624362536263627362836293630363136323633363436353636363736383639364036413642364336443645364636473648364936503651365236533654365536563657365836593660366136623663366436653666366736683669367036713672367336743675367636773678367936803681368236833684368536863687368836893690369136923693369436953696369736983699370037013702370337043705370637073708370937103711371237133714371537163717371837193720372137223723372437253726372737283729373037313732373337343735373637373738373937403741374237433744374537463747374837493750375137523753375437553756375737583759376037613762376337643765376637673768376937703771377237733774377537763777377837793780378137823783378437853786378737883789379037913792379337943795379637973798379938003801380238033804380538063807380838093810381138123813381438153816381738183819382038213822382338243825382638273828382938303831383238333834383538363837383838393840384138423843384438453846384738483849385038513852385338543855385638573858385938603861386238633864386538663867386838693870387138723873387438753876387738783879388038813882388338843885388638873888388938903891389238933894389538963897389838993900390139023903390439053906390739083909391039113912391339143915391639173918391939203921392239233924392539263927392839293930393139323933393439353936393739383939394039413942394339443945394639473948394939503951395239533954395539563957395839593960396139623963396439653966396739683969397039713972397339743975397639773978397939803981398239833984398539863987398839893990399139923993399439953996399739983999400040014002400340044005400640074008400940104011401240134014401540164017401840194020402140224023402440254026402740284029403040314032403340344035403640374038403940404041404240434044404540464047404840494050405140524053405440554056405740584059406040614062406340644065406640674068406940704071407240734074407540764077407840794080408140824083408440854086408740884089409040914092409340944095409640974098409941004101410241034104410541064107410841094110411141124113411441154116411741184119412041214122412341244125412641274128412941304131413241334134413541364137413841394140414141424143414441454146414741484149415041514152415341544155415641574158415941604161416241634164416541664167416841694170417141724173417441754176417741784179418041814182418341844185418641874188418941904191419241934194419541964197419841994200420142024203420442054206420742084209421042114212421342144215421642174218421942204221422242234224422542264227422842294230423142324233423442354236423742384239424042414242424342444245424642474248424942504251425242534254425542564257425842594260426142624263426442654266426742684269427042714272427342744275427642774278427942804281428242834284428542864287428842894290429142924293429442954296429742984299430043014302430343044305430643074308430943104311431243134314431543164317431843194320432143224323432443254326432743284329433043314332433343344335433643374338433943404341434243434344434543464347434843494350435143524353435443554356435743584359436043614362436343644365436643674368436943704371437243734374437543764377437843794380438143824383438443854386438743884389439043914392439343944395439643974398439944004401440244034404440544064407440844094410441144124413441444154416441744184419442044214422442344244425442644274428442944304431443244334434443544364437443844394440444144424443444444454446444744484449445044514452445344544455445644574458445944604461446244634464446544664467446844694470447144724473447444754476447744784479448044814482448344844485448644874488448944904491449244934494449544964497449844994500450145024503450445054506450745084509451045114512451345144515451645174518451945204521452245234524452545264527452845294530453145324533453445354536453745384539454045414542454345444545454645474548454945504551455245534554455545564557455845594560456145624563456445654566456745684569457045714572457345744575457645774578457945804581458245834584458545864587458845894590459145924593459445954596459745984599460046014602460346044605460646074608460946104611461246134614461546164617461846194620462146224623462446254626462746284629463046314632463346344635463646374638463946404641464246434644464546464647464846494650465146524653465446554656465746584659466046614662466346644665466646674668466946704671467246734674467546764677467846794680468146824683468446854686468746884689469046914692469346944695469646974698469947004701470247034704470547064707470847094710471147124713471447154716471747184719472047214722472347244725472647274728472947304731473247334734473547364737473847394740474147424743474447454746474747484749475047514752475347544755475647574758475947604761476247634764476547664767476847694770477147724773477447754776477747784779478047814782478347844785478647874788478947904791479247934794479547964797479847994800480148024803480448054806480748084809481048114812481348144815481648174818481948204821482248234824482548264827482848294830483148324833483448354836483748384839484048414842484348444845484648474848484948504851485248534854485548564857485848594860486148624863486448654866486748684869487048714872487348744875487648774878487948804881488248834884488548864887488848894890489148924893489448954896489748984899490049014902490349044905490649074908490949104911491249134914491549164917491849194920492149224923492449254926492749284929493049314932493349344935493649374938493949404941494249434944494549464947494849494950495149524953495449554956495749584959496049614962496349644965496649674968496949704971497249734974497549764977497849794980498149824983498449854986498749884989499049914992499349944995499649974998499950005001500250035004500550065007500850095010501150125013501450155016501750185019502050215022502350245025502650275028502950305031503250335034503550365037503850395040504150425043504450455046504750485049505050515052505350545055505650575058505950605061506250635064506550665067506850695070507150725073507450755076507750785079508050815082508350845085508650875088508950905091509250935094509550965097509850995100510151025103510451055106510751085109511051115112511351145115511651175118511951205121512251235124512551265127512851295130513151325133513451355136513751385139514051415142514351445145514651475148514951505151515251535154515551565157515851595160516151625163516451655166516751685169517051715172517351745175517651775178517951805181518251835184518551865187518851895190519151925193519451955196519751985199520052015202520352045205520652075208520952105211521252135214521552165217521852195220522152225223522452255226522752285229523052315232523352345235523652375238523952405241524252435244524552465247524852495250525152525253525452555256525752585259526052615262526352645265526652675268526952705271527252735274527552765277527852795280528152825283528452855286528752885289529052915292529352945295
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Linq;
  4. using System.Diagnostics;
  5. using Aitex.Core.Common;
  6. using Aitex.Core.RT.Routine;
  7. using Aitex.Core.RT.SCCore;
  8. using Aitex.Sorter.Common;
  9. using Aitex.Core.RT.Log;
  10. using Aitex.Core.Util;
  11. using Aitex.Core.RT.DataCenter;
  12. using Aitex.Core.RT.RecipeCenter;
  13. using Aitex.Core.RT.Fsm;
  14. using MECF.Framework.Common.Jobs;
  15. using MECF.Framework.Common.Routine;
  16. using MECF.Framework.Common.Equipment;
  17. using MECF.Framework.Common.SubstrateTrackings;
  18. using MECF.Framework.Common.Schedulers;
  19. using MECF.Framework.Common.DBCore;
  20. using Venus_Core;
  21. using Venus_RT.Modules.Schedulers;
  22. using Venus_RT.Scheduler;
  23. using Venus_Unity;
  24. using MECF.Framework.RT.EquipmentLibrary.HardwareUnits.PMs;
  25. using System.Linq.Expressions;
  26. using Aitex.Core.UI.Control;
  27. using SecsGem.Core.ItemModel;
  28. namespace Venus_RT.Modules
  29. {
  30. public class WaferTask
  31. {
  32. public ModuleName sourceMod { get; }
  33. public int sourceSlot { get; }
  34. public ModuleName destMod { get; private set; }
  35. public int destSlot { get; private set; }
  36. public ModuleName currentMod { get; private set; }
  37. public int currentSlot { get; private set; }
  38. public ModuleName nextMod { get; private set; }
  39. public int nextSlot { get; private set; }
  40. public ModuleName routedMod { get; private set; }
  41. public int routedSlot { get; private set; }
  42. public Hand hand { get; private set; }
  43. public SequenceLLInOutPath llInOutPath { get; }
  44. public string processRecipe { get; }
  45. public string wtwCleanRecipe { get; }
  46. public float temperature { get; }
  47. public Guid waferId { get; }
  48. public Guid lotId { get; }
  49. public int elapseTime { get { return (int)(DateTime.Now - _scheduledTime).TotalSeconds; } }
  50. public int llDelayTime { get; }
  51. public RState pressureStatus { get; private set; }
  52. public RState temperatureStatus { get; private set; }
  53. public RState movingStatus { get; private set; }
  54. public bool IsCompleted { get { return movingStatus == RState.End && currentMod == destMod && currentSlot == destSlot; } }
  55. public bool IsTimeout { get { return elapseTime > 600; } }
  56. public bool IsAligned { get; set; }
  57. public bool IsWaitingAtmMoving
  58. {
  59. get
  60. {
  61. if (movingStatus != RState.Init)
  62. return false;
  63. return !IsWaitingVacMoving;
  64. }
  65. }
  66. public bool IsWaitingVacMoving
  67. {
  68. get
  69. {
  70. if (movingStatus == RState.Init &&
  71. (ModuleHelper.IsPm(nextMod) ||
  72. ModuleHelper.IsTMRobot(nextMod) ||
  73. ModuleHelper.IsPm(currentMod) ||
  74. ModuleHelper.IsTMRobot(currentMod)))
  75. return true;
  76. return false;
  77. }
  78. }
  79. public delegate void WaferMoveHandler(WaferTask wafer, MoveItem item);
  80. public event WaferMoveHandler OnWaferArrived;
  81. public event WaferMoveHandler OnWaferLeaved;
  82. private DateTime _scheduledTime;
  83. public ModuleName DestinationPM { get; set; }
  84. 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)
  85. {
  86. sourceMod = currentMod = nextMod = routedMod = source;
  87. sourceSlot = currentSlot = nextSlot = routedSlot = srcSlot;
  88. DestinationPM = dest;
  89. destMod = dest;
  90. destSlot = dstSlot;
  91. hand = Hand.None;
  92. temperature = temp;
  93. waferId = waferID;
  94. lotId = lotID;
  95. processRecipe = recipeName;
  96. wtwCleanRecipe = wtwClean;
  97. llInOutPath = inOutPath;
  98. llDelayTime = LLDelay;
  99. IsAligned = !needAlign;
  100. pressureStatus = RState.Init;
  101. temperatureStatus = RState.End;
  102. movingStatus = RState.Init;
  103. _scheduledTime = DateTime.Now;
  104. }
  105. public RState Run()
  106. {
  107. if (movingStatus == RState.Running)
  108. {
  109. var wafer = WaferManager.Instance.GetWafer(nextMod, nextSlot);
  110. if (!wafer.IsEmpty && wafer.InnerId == waferId)
  111. {
  112. LOG.Write(eEvent.EV_ROUTER, ModuleName.System, $"Wafer {wafer.WaferOrigin} from {currentMod}.{currentSlot + 1} move to {nextMod}.{nextSlot + 1}");
  113. if (!ModuleHelper.IsTMRobot(currentMod) && !ModuleHelper.IsEFEMRobot(currentMod))
  114. {
  115. OnWaferLeaved(this, new MoveItem(currentMod, currentSlot, nextMod, nextSlot, hand));
  116. }
  117. if (!ModuleHelper.IsTMRobot(nextMod) && !ModuleHelper.IsEFEMRobot(nextMod))
  118. {
  119. _scheduledTime = DateTime.Now;
  120. wafer.NextSequenceStep++;
  121. OnWaferArrived(this, new MoveItem(currentMod, currentSlot, nextMod, nextSlot, hand));
  122. }
  123. currentMod = nextMod;
  124. currentSlot = nextSlot;
  125. movingStatus = RState.End;
  126. }
  127. }
  128. return RState.Running;
  129. }
  130. public void RouteTo(ModuleName mod, int slot)
  131. {
  132. routedMod = mod;
  133. routedSlot = slot;
  134. movingStatus = RState.Init;
  135. }
  136. public void MoveTo(ModuleName mod, int slot)
  137. {
  138. nextMod = mod;
  139. nextSlot = slot;
  140. movingStatus = RState.Running;
  141. if (ModuleHelper.IsLoadLock(currentMod) && ModuleHelper.IsTMRobot(nextMod) && ModuleHelper.IsLoadLock(routedMod))
  142. {
  143. routedMod = ModuleName.TMRobot; // fix the bug which missing call RouteTo() on Vac side
  144. }
  145. }
  146. public void ReAssignPM(ModuleName pm, int slot)
  147. {
  148. destMod = pm;
  149. destSlot = slot;
  150. }
  151. public void Return()
  152. {
  153. destMod = sourceMod;
  154. destSlot = sourceSlot;
  155. movingStatus = RState.End;
  156. pressureStatus = RState.End;
  157. temperatureStatus = RState.End;
  158. }
  159. }
  160. public enum ModuleStatus
  161. {
  162. Idle,
  163. WaitMove,
  164. Moving,
  165. NotReady,
  166. // PM Status
  167. WaitProcess,
  168. StartProcess,
  169. Processing,
  170. StartIdleClean,
  171. IdleClean,
  172. WaitPreJobClean,
  173. StartPreJobClean,
  174. PreJobClean,
  175. WaitPostJobClean,
  176. StartPostJobClean,
  177. PostJobClean,
  178. WaitWTWClean,
  179. StartWTWClean,
  180. WTWClean,
  181. WaitLongIdleClean,
  182. StartLongIdleClean,
  183. LongIdleClean,
  184. // Align Status
  185. WaitAlign,
  186. StartAlign,
  187. Aligning,
  188. // Loadlock Status
  189. StartPump,
  190. Pumping,
  191. VacReady,
  192. StartVent,
  193. Venting,
  194. AtmReady,
  195. WaitCooling,
  196. StartCooling,
  197. Cooling,
  198. CoolingDone,
  199. // Unknown
  200. Unknown,
  201. }
  202. public class ModuleTask
  203. {
  204. public ModuleName Module { get; }
  205. public ModuleStatus Status { get; set; }
  206. public SchedulerModule Scheduler { get; }
  207. public virtual int SlotNum { get; protected set; }
  208. public virtual bool IsIdle { get { return Status == ModuleStatus.Idle; } }
  209. public virtual bool HasWafer { get { return WaferManager.Instance.CheckHasWafer(Module, 0); } }
  210. private int _currentActionTime = 0;
  211. public virtual int IdleTime
  212. {
  213. get
  214. {
  215. if (Status != ModuleStatus.Idle)
  216. return 0;
  217. return (int)(DateTime.Now - _moduleTimer).TotalSeconds;
  218. }
  219. }
  220. public virtual int BusyTime
  221. {
  222. get
  223. {
  224. if (Status == ModuleStatus.Idle)
  225. return 0;
  226. return (int)(DateTime.Now - _moduleTimer).TotalSeconds;
  227. }
  228. }
  229. public virtual int BusyLastTime
  230. {
  231. get
  232. {
  233. if (Status == ModuleStatus.Idle)
  234. return 0;
  235. return _currentActionTime - BusyTime;
  236. }
  237. }
  238. public virtual int TimeToReady
  239. {
  240. get
  241. {
  242. if (Scheduler.IsAvailable)
  243. return 0;
  244. return int.MaxValue / 2;
  245. }
  246. }
  247. protected DateTime _moduleTimer;
  248. public ModuleTask(ModuleName mod)
  249. {
  250. Module = mod;
  251. Scheduler = Singleton<TransferModule>.Instance.GetScheduler(mod);
  252. SlotNum = 1;
  253. _moduleTimer = DateTime.Now;
  254. }
  255. public virtual RState Run()
  256. {
  257. Status = Scheduler.IsAvailable ? ModuleStatus.Idle : ModuleStatus.NotReady;
  258. return RState.Running;
  259. }
  260. public virtual void WaferArrived(WaferTask wafer, int slot)
  261. {
  262. }
  263. public virtual void WaferLeaved(WaferTask wafer, int slot)
  264. {
  265. }
  266. public virtual void ResetTask()
  267. {
  268. }
  269. }
  270. public class PMTask : ModuleTask
  271. {
  272. enum CleanType
  273. {
  274. IdleClean,
  275. PreJobClean,
  276. PostJobClean,
  277. WTWClean,
  278. }
  279. public override int TimeToReady
  280. {
  281. get
  282. {
  283. if (!Scheduler.IsOnline)
  284. return int.MaxValue / 2;
  285. switch (Status)
  286. {
  287. case ModuleStatus.Idle:
  288. case ModuleStatus.Processing:
  289. case ModuleStatus.IdleClean:
  290. case ModuleStatus.PreJobClean:
  291. {
  292. return Scheduler.IsAvailable ? 0 : (Scheduler.TimeToReady + 500) / 1000;
  293. }
  294. case ModuleStatus.PostJobClean:
  295. case ModuleStatus.WTWClean:
  296. {
  297. if (_pendingCleanTask.Count == 0)
  298. {
  299. return Scheduler.IsAvailable ? 0 : (Scheduler.TimeToReady + 500) / 1000;
  300. }
  301. }
  302. break;
  303. }
  304. return int.MaxValue / 2;
  305. }
  306. }
  307. public bool HasPendingCleanTask => _pendingCleanTask.Count > 0;
  308. private WaferTask _wafer;
  309. private SchedulerPM _pmScheduler => Scheduler as SchedulerPM;
  310. private string _preJobCleanRecipe;
  311. private string _postJobCleanRecipe;
  312. private string _wtwCleanRecipe;
  313. private Queue<CleanType> _pendingCleanTask = new Queue<CleanType>();
  314. public PMTask(ModuleName pm) : base(pm)
  315. {
  316. }
  317. public override RState Run()
  318. {
  319. if (Scheduler.IsIdle)
  320. {
  321. switch (Status)
  322. {
  323. case ModuleStatus.Idle:
  324. {
  325. if (WaferManager.Instance.CheckNoWafer(Module, 0) && Scheduler.IsOnline)
  326. {
  327. if (_pendingCleanTask.Count > 0)
  328. {
  329. var cleanTask = _pendingCleanTask.Dequeue();
  330. if (cleanTask == CleanType.PostJobClean && !string.IsNullOrWhiteSpace(_postJobCleanRecipe))
  331. {
  332. Status = ModuleStatus.WaitPostJobClean;
  333. }
  334. else if (cleanTask == CleanType.PreJobClean && !string.IsNullOrWhiteSpace(_preJobCleanRecipe))
  335. {
  336. Status = ModuleStatus.WaitPreJobClean;
  337. }
  338. }
  339. else if (_pmScheduler.RunIdleCleanTask()) // Check Idle Clean
  340. {
  341. Status = ModuleStatus.StartIdleClean;
  342. }
  343. }
  344. }
  345. break;
  346. case ModuleStatus.WaitProcess:
  347. {
  348. Scheduler.EventWaferArrived?.Invoke(this, new WaferMoveArgs(ModuleName.TMRobot, 0, Module, 0));
  349. Status = ModuleStatus.StartProcess;
  350. }
  351. break;
  352. case ModuleStatus.WaitPreJobClean:
  353. {
  354. if (WaferManager.Instance.CheckNoWafer(Module, 0))
  355. {
  356. if (_pmScheduler.RunJobCleanTask(_preJobCleanRecipe))
  357. {
  358. Status = ModuleStatus.StartPreJobClean;
  359. }
  360. else
  361. {
  362. LOG.Write(eEvent.WARN_ROUTER, Module, $"Run Prejob clean recipe{_preJobCleanRecipe} failed");
  363. Status = ModuleStatus.Idle;
  364. _preJobCleanRecipe = string.Empty;
  365. }
  366. }
  367. }
  368. break;
  369. case ModuleStatus.WaitPostJobClean:
  370. {
  371. if (WaferManager.Instance.CheckNoWafer(Module, 0))
  372. {
  373. if (_pmScheduler.RunJobCleanTask(_postJobCleanRecipe))
  374. {
  375. Status = ModuleStatus.StartPostJobClean;
  376. }
  377. else
  378. {
  379. LOG.Write(eEvent.WARN_ROUTER, Module, $"Run Postjob clean recipe{_postJobCleanRecipe} failed");
  380. Status = ModuleStatus.Idle;
  381. _postJobCleanRecipe = string.Empty;
  382. }
  383. }
  384. }
  385. break;
  386. case ModuleStatus.WaitWTWClean:
  387. {
  388. if (WaferManager.Instance.CheckNoWafer(Module, 0))
  389. {
  390. if (_pmScheduler.RunJobCleanTask(_wtwCleanRecipe))
  391. {
  392. Status = ModuleStatus.StartWTWClean;
  393. }
  394. else
  395. {
  396. LOG.Write(eEvent.WARN_ROUTER, Module, $"Run WTW Wafer clean recipe{_wtwCleanRecipe} failed");
  397. Status = ModuleStatus.Idle;
  398. _wtwCleanRecipe = string.Empty;
  399. }
  400. }
  401. }
  402. break;
  403. case ModuleStatus.Processing:
  404. {
  405. var wafer = WaferManager.Instance.GetWafer(Module, 0);
  406. if (Scheduler.IsOnline)
  407. {
  408. if (!wafer.IsEmpty && wafer.ProcessState == EnumWaferProcessStatus.Completed)
  409. {
  410. _wafer.Return();
  411. Status = ModuleStatus.Idle;
  412. }
  413. }
  414. else // handle offline exception situation
  415. {
  416. Status = ModuleStatus.Idle;
  417. }
  418. }
  419. break;
  420. case ModuleStatus.PreJobClean:
  421. {
  422. _preJobCleanRecipe = string.Empty;
  423. Status = ModuleStatus.Idle;
  424. }
  425. break;
  426. case ModuleStatus.PostJobClean:
  427. {
  428. _postJobCleanRecipe = string.Empty;
  429. if (_pendingCleanTask.Count > 0)
  430. {
  431. var cleanTask = _pendingCleanTask.Dequeue();
  432. if (cleanTask == CleanType.PreJobClean && !string.IsNullOrWhiteSpace(_preJobCleanRecipe))
  433. {
  434. Status = ModuleStatus.WaitPreJobClean;
  435. break;
  436. }
  437. }
  438. Status = ModuleStatus.Idle;
  439. }
  440. break;
  441. case ModuleStatus.WTWClean:
  442. {
  443. if (_pendingCleanTask.Count > 0)
  444. {
  445. var cleanTask = _pendingCleanTask.Dequeue();
  446. if (cleanTask == CleanType.PostJobClean && !string.IsNullOrWhiteSpace(_postJobCleanRecipe))
  447. {
  448. Status = ModuleStatus.WaitPostJobClean;
  449. }
  450. else if (cleanTask == CleanType.PreJobClean && !string.IsNullOrWhiteSpace(_preJobCleanRecipe))
  451. {
  452. Status = ModuleStatus.WaitPreJobClean;
  453. }
  454. }
  455. else
  456. {
  457. Status = ModuleStatus.Idle;
  458. }
  459. _wtwCleanRecipe = string.Empty;
  460. }
  461. break;
  462. case ModuleStatus.IdleClean:
  463. {
  464. Status = ModuleStatus.Idle;
  465. }
  466. break;
  467. case ModuleStatus.StartProcess:
  468. {
  469. if (RouteManager.IsATMMode)
  470. {
  471. Status = ModuleStatus.Processing;
  472. }
  473. }
  474. break;
  475. }
  476. }
  477. else
  478. {
  479. switch (Status)
  480. {
  481. case ModuleStatus.StartProcess:
  482. Status = ModuleStatus.Processing;
  483. break;
  484. case ModuleStatus.StartIdleClean:
  485. Status = ModuleStatus.IdleClean;
  486. break;
  487. case ModuleStatus.StartPreJobClean:
  488. Status = ModuleStatus.PreJobClean;
  489. break;
  490. case ModuleStatus.StartPostJobClean:
  491. Status = ModuleStatus.PostJobClean;
  492. break;
  493. case ModuleStatus.StartWTWClean:
  494. Status = ModuleStatus.WTWClean;
  495. break;
  496. }
  497. }
  498. return RState.Running;
  499. }
  500. public override void WaferArrived(WaferTask wafer, int slot)
  501. {
  502. _wafer = wafer;
  503. Status = ModuleStatus.WaitProcess;
  504. }
  505. public override void WaferLeaved(WaferTask wafer, int slot)
  506. {
  507. if (!string.IsNullOrWhiteSpace(_wafer.wtwCleanRecipe.Trim()) && (_pendingCleanTask.Count == 0 || _pendingCleanTask.First() != CleanType.PostJobClean))
  508. {
  509. Status = ModuleStatus.WaitWTWClean;
  510. _wtwCleanRecipe = _wafer.wtwCleanRecipe;
  511. }
  512. else if (_pendingCleanTask.Count > 0)
  513. {
  514. var cleanTask = _pendingCleanTask.Dequeue();
  515. if (cleanTask == CleanType.PostJobClean && !string.IsNullOrWhiteSpace(_postJobCleanRecipe))
  516. {
  517. Status = ModuleStatus.WaitPostJobClean;
  518. }
  519. else if (cleanTask == CleanType.PreJobClean && !string.IsNullOrWhiteSpace(_preJobCleanRecipe))
  520. {
  521. Status = ModuleStatus.WaitPreJobClean;
  522. }
  523. }
  524. _wafer = null;
  525. }
  526. public void InvokePreJobClean(string preJobClean)
  527. {
  528. _preJobCleanRecipe = preJobClean;
  529. _pendingCleanTask.Enqueue(CleanType.PreJobClean);
  530. //LOG.Write(eEvent.INFO_TROUGHPUT_ENHANCE_TRACE, Module, $"Invoke preClean {preJobClean}");
  531. }
  532. public bool IsPreJobCleanDone()
  533. {
  534. return !_pendingCleanTask.Contains(CleanType.PreJobClean) && string.IsNullOrWhiteSpace(_preJobCleanRecipe);
  535. }
  536. public void InvokePostJobClean(string postJobClean)
  537. {
  538. _postJobCleanRecipe = postJobClean;
  539. _pendingCleanTask.Enqueue(CleanType.PostJobClean);
  540. //LOG.Write(eEvent.INFO_TROUGHPUT_ENHANCE_TRACE, Module, $"Invoke postClean {postJobClean}");
  541. }
  542. public bool IsPostJobCleanDone()
  543. {
  544. return !_pendingCleanTask.Contains(CleanType.PostJobClean) && string.IsNullOrWhiteSpace(_postJobCleanRecipe);
  545. }
  546. }
  547. public class AlignerTask : ModuleTask
  548. {
  549. private WaferTask _wafer;
  550. public AlignerTask(ModuleName aligner) : base(aligner)
  551. {
  552. }
  553. public override RState Run()
  554. {
  555. var efemRobot = Singleton<TransferModule>.Instance.GetScheduler(ModuleName.EfemRobot) as SchedulerEfemRobot;
  556. if (efemRobot == null || !efemRobot.IsAvailable)
  557. return RState.Running;
  558. switch (Status)
  559. {
  560. case ModuleStatus.WaitAlign:
  561. {
  562. efemRobot.Align(0F);
  563. Status = ModuleStatus.Aligning;
  564. }
  565. break;
  566. case ModuleStatus.Aligning:
  567. {
  568. _wafer.IsAligned = true;
  569. Status = ModuleStatus.Idle;
  570. }
  571. break;
  572. }
  573. return RState.Running;
  574. }
  575. public override void WaferArrived(WaferTask wafer, int slot)
  576. {
  577. _wafer = wafer;
  578. Status = ModuleStatus.WaitAlign;
  579. }
  580. public override void WaferLeaved(WaferTask wafer, int slot)
  581. {
  582. _wafer = null;
  583. }
  584. }
  585. public class LoadPortTask : ModuleTask
  586. {
  587. public LoadPortTask(ModuleName lp) : base(lp)
  588. {
  589. SlotNum = SC.GetValue<int>($"EFEM.LoadPort.SlotNumber");
  590. }
  591. public override RState Run()
  592. {
  593. return base.Run();
  594. }
  595. }
  596. public class TMRobotTask : ModuleTask
  597. {
  598. public TMRobotTask(ModuleName tmRobot) : base(tmRobot)
  599. {
  600. SlotNum = 2;
  601. }
  602. public override RState Run()
  603. {
  604. return base.Run();
  605. }
  606. }
  607. public class EFEMRobotTask : ModuleTask
  608. {
  609. public EFEMRobotTask(ModuleName efemRobot) : base(efemRobot)
  610. {
  611. SlotNum = 2;
  612. }
  613. public override RState Run()
  614. {
  615. return base.Run();
  616. }
  617. public bool PostMoveItems(MoveItem[] items)
  618. {
  619. Status = ModuleStatus.Moving;
  620. return (Scheduler as SchedulerEfemRobot).PostMoveItems(items);
  621. }
  622. }
  623. public enum LLStatus
  624. {
  625. StartPump,
  626. Pumping,
  627. VacReady,
  628. StartVent,
  629. Venting,
  630. AtmReady,
  631. Unknow,
  632. }
  633. public class LoadlockTask : ModuleTask
  634. {
  635. private Dictionary<int, int> _wafersDelayTime = new Dictionary<int, int>();
  636. public LoadlockTask(ModuleName ll) : base(ll)
  637. {
  638. SlotNum = SC.GetValue<int>($"{ll}.SlotNumber");
  639. for (int slot = 0; slot < SlotNum; slot++)
  640. _wafersDelayTime[slot] = -1;
  641. }
  642. public override int TimeToReady
  643. {
  644. get { return Scheduler.IsAvailable ? 0 : (Scheduler.TimeToReady + 500) / 1000; }
  645. }
  646. public bool ReayForTMInTime(int seconds)
  647. {
  648. return ((RouteManager.IsATMMode ? Status == ModuleStatus.AtmReady : Status == ModuleStatus.Pumping) && TimeToReady < seconds) || Status == ModuleStatus.VacReady;
  649. }
  650. public bool ReayForEfemInTime(int seconds)
  651. {
  652. return (Status == ModuleStatus.Venting && TimeToReady < seconds) || Status == ModuleStatus.AtmReady;
  653. }
  654. public override void WaferArrived(WaferTask wafer, int slot)
  655. {
  656. Scheduler.WaferArrived(slot);
  657. if (ModuleHelper.IsLoadPort(wafer.destMod))
  658. {
  659. _wafersDelayTime[slot] = wafer.llDelayTime;
  660. }
  661. else
  662. {
  663. _wafersDelayTime[slot] = -1;
  664. }
  665. }
  666. public override void WaferLeaved(WaferTask wafer, int slot)
  667. {
  668. _wafersDelayTime[slot] = -1;
  669. }
  670. public bool PreVent()
  671. {
  672. if (Scheduler.IsAvailable && !Scheduler.IsAtm)
  673. {
  674. if ((Scheduler as SchedulerLoadLock).PreVent())
  675. {
  676. Status = ModuleStatus.StartVent;
  677. return true;
  678. }
  679. }
  680. return false;
  681. }
  682. public bool PrePump()
  683. {
  684. if (Scheduler.IsAvailable && !Scheduler.IsVac)
  685. {
  686. if ((Scheduler as SchedulerLoadLock).PrePump())
  687. {
  688. Status = ModuleStatus.StartPump;
  689. return true;
  690. }
  691. }
  692. return false;
  693. }
  694. public bool Cooling()
  695. {
  696. if (Scheduler.IsAvailable && !Scheduler.IsAtm)
  697. {
  698. if ((Scheduler as SchedulerLoadLock).Cooling())
  699. {
  700. Status = ModuleStatus.StartCooling;
  701. return true;
  702. }
  703. }
  704. return false;
  705. }
  706. public override RState Run()
  707. {
  708. if (Scheduler.IsIdle)
  709. {
  710. if (Scheduler.IsAvailable)
  711. {
  712. switch (Status)
  713. {
  714. case ModuleStatus.Pumping:
  715. if (Scheduler.IsVac)
  716. {
  717. Status = ModuleStatus.VacReady;
  718. }
  719. break;
  720. case ModuleStatus.Venting:
  721. if (Scheduler.IsAtm)
  722. {
  723. Status = ModuleStatus.AtmReady;
  724. }
  725. break;
  726. case ModuleStatus.Cooling:
  727. if (Scheduler.IsAtm)
  728. {
  729. Status = ModuleStatus.CoolingDone;
  730. }
  731. break;
  732. default:
  733. {
  734. Status = Scheduler.IsAtm ? ModuleStatus.AtmReady : Scheduler.IsVac ? ModuleStatus.VacReady : Status;
  735. }
  736. break;
  737. }
  738. }
  739. }
  740. else
  741. {
  742. switch (Status)
  743. {
  744. case ModuleStatus.StartPump:
  745. Status = ModuleStatus.Pumping;
  746. break;
  747. case ModuleStatus.StartVent:
  748. Status = ModuleStatus.Venting;
  749. break;
  750. case ModuleStatus.StartCooling:
  751. Status = ModuleStatus.Cooling;
  752. break;
  753. }
  754. }
  755. return RState.Running;
  756. }
  757. public int Distance
  758. {
  759. get
  760. {
  761. switch (Status)
  762. {
  763. case ModuleStatus.Idle:
  764. return Scheduler.IsVac ? 1 : Scheduler.IsAtm ? 7 : 5;
  765. case ModuleStatus.StartPump:
  766. case ModuleStatus.Pumping:
  767. return 3;
  768. case ModuleStatus.StartCooling:
  769. case ModuleStatus.Cooling:
  770. case ModuleStatus.StartVent:
  771. case ModuleStatus.Venting:
  772. return 9;
  773. default:
  774. return 5;
  775. }
  776. }
  777. }
  778. }
  779. enum LLSlotInOutOpt
  780. {
  781. AllInAllOut,
  782. UpperInLowerOut,
  783. LowerInUpperOut,
  784. }
  785. enum SpecialRoutingPattern
  786. {
  787. Disable,
  788. LongTimeRecipe,
  789. }
  790. class SystemDispatcher : ICycle
  791. {
  792. private List<ControlJobInfo> _lstControlJobs = new List<ControlJobInfo>();
  793. private List<ProcessJobInfo> _lstProcessJobs = new List<ProcessJobInfo>();
  794. private Queue<ControlJobInfo> _qePollingJobs = new Queue<ControlJobInfo>();
  795. private List<WaferTask> _lstWaferTasks = new List<WaferTask>();
  796. private Dictionary<ModuleName, ModuleTask> _dictModuleTask = new Dictionary<ModuleName, ModuleTask>();
  797. private Queue<WaferInfo> _qeReturnWafers = new Queue<WaferInfo>();
  798. private Queue<Guid> _qeWaitInWafers = new Queue<Guid>();
  799. private Queue<ModuleName> _qeWaitCoolingLLs = new Queue<ModuleName>();
  800. private Queue<List<MoveItem>> _efemSchdActions = new Queue<List<MoveItem>>();
  801. private Queue<List<MoveItem>> _tmSchdActions = new Queue<List<MoveItem>>();
  802. private List<MoveItem> _curEfemAction = new List<MoveItem>();
  803. private List<MoveItem> _curTmAction = new List<MoveItem>();
  804. private RState _efemRobotStatus { get { return (_dictModuleTask[ModuleName.EfemRobot].Scheduler as SchedulerEfemRobot).RobotStatus; } }
  805. private RState _tmRobotStatus { get { return (_dictModuleTask[ModuleName.TMRobot].Scheduler as SchedulerTMRobot).RobotStatus; } }
  806. private SchedulerFACallback _faCallback;
  807. private SchedulerDBCallback _dbCallback;
  808. private readonly int _LLASlotNumber = 4;
  809. private readonly int _LLBSlotNumber = 4;
  810. private int _efemRobotSingleArmOption = 0;
  811. private int _tmRobotSingleArmOption = 0;
  812. private LLSlotInOutOpt _LLSlotInOutOption = 0;
  813. private Dictionary<ModuleName, int> _lpCycleWafer = new Dictionary<ModuleName, int> { { ModuleName.LP1, 0 }, { ModuleName.LP2, 0 }, { ModuleName.LP3, 0 } };
  814. private Dictionary<ModuleName, int> _lpCycleCount = new Dictionary<ModuleName, int> { { ModuleName.LP1, 0 }, { ModuleName.LP2, 0 }, { ModuleName.LP3, 0 } };
  815. private Dictionary<ModuleName, int> _lpCycleSP = new Dictionary<ModuleName, int> { { ModuleName.LP1, 1 }, { ModuleName.LP2, 1 }, { ModuleName.LP3, 1 } };
  816. private Dictionary<ModuleName, float> _lpThroughput = new Dictionary<ModuleName, float> { { ModuleName.LP1, 0 }, { ModuleName.LP2, 0 }, { ModuleName.LP3, 0 } };
  817. private Dictionary<ModuleName, Stopwatch> _lpCycleWatch = new Dictionary<ModuleName, Stopwatch> { { ModuleName.LP1, new Stopwatch() }, { ModuleName.LP2, new Stopwatch() }, { ModuleName.LP3, new Stopwatch() } };
  818. private Dictionary<ModuleName, List<Guid>> _preLotCleanMarks = new Dictionary<ModuleName, List<Guid>>();
  819. private Dictionary<ModuleName, List<Guid>> _postLotCleanMarks = new Dictionary<ModuleName, List<Guid>>();
  820. private Stopwatch _cycleWatch = new Stopwatch();
  821. private SequenceLLInOutPath _LLInOutPath = SequenceLLInOutPath.DInDOut;
  822. private SpecialRoutingPattern _specialRoutingPattern = SpecialRoutingPattern.Disable;
  823. public SequenceLLInOutPath LLInOutPath => _LLInOutPath;
  824. public bool HasJobRunning => _lstControlJobs.Count > 0;
  825. private RState _cycleState = RState.Init;
  826. public RState CycleState => _cycleState;
  827. private Dictionary<string, ControlJobInfo> _loadportControlJobDic = new Dictionary<string, ControlJobInfo>();
  828. public List<string> InUseRecipes = new List<string>();
  829. public List<string> LP1InUseRecipes = new List<string>();
  830. public List<string> LP2InUseRecipes = new List<string>();
  831. public List<string> LP3InUseRecipes = new List<string>();
  832. private Dictionary<ModuleName, DateTime> _pmLastWaferEndTime = new Dictionary<ModuleName, DateTime>();
  833. private int _preCleanMaxIdleTime = 60;
  834. #region public interface
  835. public SystemDispatcher()
  836. {
  837. _faCallback = new SchedulerFACallback();
  838. _dbCallback = new SchedulerDBCallback();
  839. _efemRobotSingleArmOption = SC.GetValue<int>("EFEM.SingleArmOption");
  840. _tmRobotSingleArmOption = SC.GetValue<int>("TM.SingleArmOption");
  841. _LLASlotNumber = SC.GetValue<int>("LLA.SlotNumber");
  842. _LLBSlotNumber = SC.GetValue<int>("LLB.SlotNumber");
  843. InitModules();
  844. DATA.Subscribe("Scheduler.PjIdList", () => Array.ConvertAll(_lstProcessJobs.ToArray(), x => x.InnerId.ToString()).ToList());
  845. DATA.Subscribe("Scheduler.PjNameList", () => Array.ConvertAll(_lstProcessJobs.ToArray(), x => x.LotName.ToString()).ToList());
  846. DATA.Subscribe("Scheduler.InUsingRecipe", () => InUseRecipes);
  847. foreach (var lp in new List<ModuleName> { ModuleName.LP1, ModuleName.LP2, ModuleName.LP3 })
  848. {
  849. _loadportControlJobDic[lp.ToString()] = null;
  850. DATA.Subscribe($"{lp}.CurrentControlJob", () => _loadportControlJobDic[lp.ToString()], SubscriptionAttribute.FLAG.IgnoreSaveDB);
  851. DATA.Subscribe($"{lp}.CycledCount", () => _lpCycleCount[lp], SubscriptionAttribute.FLAG.IgnoreSaveDB);
  852. DATA.Subscribe($"{lp}.CycledWafer", () => _lpCycleWafer[lp], SubscriptionAttribute.FLAG.IgnoreSaveDB);
  853. DATA.Subscribe($"{lp}.CycleSetPoint", () => _lpCycleSP[lp], SubscriptionAttribute.FLAG.IgnoreSaveDB);
  854. DATA.Subscribe($"{lp}.Throughput", () => _lpThroughput[lp], SubscriptionAttribute.FLAG.IgnoreSaveDB);
  855. }
  856. DATA.Subscribe("Scheduler.LP1.InUsingRecipe", () => LP1InUseRecipes);
  857. DATA.Subscribe("Scheduler.LP2.InUsingRecipe", () => LP2InUseRecipes);
  858. DATA.Subscribe("Scheduler.LP3.InUsingRecipe", () => LP3InUseRecipes);
  859. if (RtInstance.ConfigType == ConfigType.Kepler2200)
  860. {
  861. LOG.ModuleErrorInterrupt += (modulename, id) => ModuleErrorInterrupt(modulename, id);
  862. }
  863. ModuleHelper.InstalledModules.ForEach(x =>
  864. {
  865. if (ModuleHelper.IsPm(x))
  866. {
  867. _pmLastWaferEndTime.Add(x, DateTime.Now.AddDays(-1));
  868. }
  869. });
  870. }
  871. private void ModuleErrorInterrupt(ModuleName pmname, eEvent eEventid)
  872. {
  873. List<ModuleName> modules = new List<ModuleName>();
  874. _lstProcessJobs.ForEach(x =>
  875. {
  876. x.Sequence.PMs.ForEach(m => modules.Add(m));
  877. });
  878. if (modules.Contains(pmname) && (eEventid == eEvent.ERR_PM_SCRUBBER_NOOK_CONTINUEPROCESS || eEventid == eEvent.ERR_PM_SCRUBBER_NOOK_STOPPROCESS))
  879. {
  880. Singleton<RouteManager>.Instance.InvokeAbort(new object[] { ModuleName.System });
  881. LOG.Write(eEvent.ERR_PM_SCRUBBER_NOOK_STOPTRANSFER, ModuleName.System, $"{pmname} Scrubber Alarm ,Tranfer Stop");
  882. }
  883. }
  884. /// <summary>
  885. /// 获取lp当前的ControlJob
  886. /// </summary>
  887. /// <param name="lp"></param>
  888. /// <returns></returns>
  889. public ControlJobInfo GetLoadPortCurrentControlJob(ModuleName lp)
  890. {
  891. if (ModuleHelper.IsLoadPort(lp))
  892. {
  893. return _loadportControlJobDic.ContainsKey(lp.ToString()) ? _loadportControlJobDic[lp.ToString()] : null;
  894. }
  895. else
  896. {
  897. return null;
  898. }
  899. }
  900. public RState Start(params object[] objs)
  901. {
  902. if (WaferManager.Instance.HasDuplicatedWafer)
  903. {
  904. LOG.Write(eEvent.ERR_ROUTER, ModuleName.System, "System has dummy wafers, please verify all the wafer position, and delete the invalid wafer");
  905. return RState.Failed;
  906. }
  907. _efemRobotSingleArmOption = SC.GetValue<int>("EFEM.SingleArmOption");
  908. _tmRobotSingleArmOption = SC.GetValue<int>("TM.SingleArmOption");
  909. _LLSlotInOutOption = (LLSlotInOutOpt)SC.GetValue<int>("System.LoadlockSlotInOutOption");
  910. _specialRoutingPattern = (SpecialRoutingPattern)SC.GetValue<int>("System.SpecialRoutingPattern");
  911. _preCleanMaxIdleTime = SC.GetValue<int>("System.Job.PreCleanIdleMaxTime");
  912. // rounding TM robot single arm option
  913. if (_tmRobotSingleArmOption > 2 && _LLSlotInOutOption == LLSlotInOutOpt.AllInAllOut)
  914. {
  915. _tmRobotSingleArmOption = 0;
  916. }
  917. _cycleWatch.Stop();
  918. Clear();
  919. // reset cycle/throughput couter
  920. _lpCycleWafer = new Dictionary<ModuleName, int> { { ModuleName.LP1, 0 }, { ModuleName.LP2, 0 }, { ModuleName.LP3, 0 } };
  921. _lpCycleCount = new Dictionary<ModuleName, int> { { ModuleName.LP1, 0 }, { ModuleName.LP2, 0 }, { ModuleName.LP3, 0 } };
  922. _lpCycleSP = new Dictionary<ModuleName, int> { { ModuleName.LP1, 1 }, { ModuleName.LP2, 1 }, { ModuleName.LP3, 1 } };
  923. _lpThroughput = new Dictionary<ModuleName, float> { { ModuleName.LP1, 0 }, { ModuleName.LP2, 0 }, { ModuleName.LP3, 0 } };
  924. _lpCycleWatch = new Dictionary<ModuleName, Stopwatch> { { ModuleName.LP1, new Stopwatch() }, { ModuleName.LP2, new Stopwatch() }, { ModuleName.LP3, new Stopwatch() } };
  925. return RState.Running;
  926. }
  927. public RState Monitor()
  928. {
  929. if (_lstControlJobs.Any(x => x.State == EnumControlJobState.Paused))
  930. {
  931. return RState.Running;
  932. }
  933. prelude();
  934. RunWaferTask();
  935. RunModuleTasks();
  936. RoutingATMWafers();
  937. RoutingVacWafers();
  938. epilogue();
  939. return _cycleState;
  940. }
  941. public void Abort()
  942. {
  943. }
  944. public bool CreateJob(Dictionary<string, object> param, out string reason)
  945. {
  946. reason = "";
  947. string[] slotSequence = (string[])param["SlotSequence"];
  948. string[] slotWaferId = new string[slotSequence.Length];
  949. if (param.Keys.Contains("WaferId"))
  950. {
  951. slotWaferId = (string[])param["WaferId"];
  952. }
  953. else
  954. {
  955. for (int i = 0; i < slotWaferId.Length; i++)
  956. {
  957. slotWaferId[i] = "";
  958. }
  959. }
  960. string lotType = "";
  961. if (param.Keys.Contains("LotType"))
  962. {
  963. lotType = (string)param["LotType"];
  964. }
  965. string module = (string)param["Module"];
  966. string jobId = (string)param["JobId"];
  967. if (string.IsNullOrEmpty(jobId))
  968. {
  969. jobId = "CJ_Local_" + module;
  970. }
  971. foreach (var item in _loadportControlJobDic.Values)
  972. {
  973. if (item?.Name == jobId)
  974. {
  975. reason = $"The same jobid {jobId} already exists, please re-enter the jobid";
  976. LOG.Write(eEvent.WARN_ROUTER, ModuleName.System, reason);
  977. return false;
  978. }
  979. }
  980. string lotId = (string)param["LotId"];
  981. if (string.IsNullOrEmpty(lotId))
  982. {
  983. lotId = "CJ_Local_" + module;
  984. }
  985. string preCleanRecipe = param.ContainsKey("PreCleanRecipeName") ? (string)param["PreCleanRecipeName"] : string.Empty;
  986. string postCleanRecipe = param.ContainsKey("PostCleanRecipeName") ? (string)param["PostCleanRecipeName"] : string.Empty;
  987. int cycleNumber = param.ContainsKey("CycleNumber") ? (int)param["CycleNumber"] : 1;
  988. if (slotSequence.Length != SC.GetValue<int>("EFEM.LoadPort.SlotNumber"))
  989. {
  990. reason = $"slot sequence parameter not valid, length is {slotSequence.Length}, should be {SC.GetValue<int>("EFEM.LoadPort.SlotNumber")}";
  991. LOG.Write(eEvent.WARN_ROUTER, ModuleName.System, reason);
  992. _faCallback.JobCreateFailed(module, lotId, jobId, "");
  993. return false;
  994. }
  995. if (!ModuleHelper.IsLoadPort(ModuleHelper.Converter(module)))
  996. {
  997. reason = $"{module} should be LoadPort";
  998. LOG.Write(eEvent.WARN_ROUTER, ModuleName.System, reason);
  999. _faCallback.JobCreateFailed(module, lotId, jobId, "");
  1000. return false;
  1001. }
  1002. if (_lstControlJobs.Exists(x => x.Name == jobId))
  1003. {
  1004. reason = $"{jobId} already created";
  1005. LOG.Write(eEvent.WARN_ROUTER, ModuleName.System, reason);
  1006. _faCallback.JobCreateFailed(module, lotId, jobId, "");
  1007. return false;
  1008. }
  1009. ControlJobInfo cj = new ControlJobInfo();
  1010. cj.Name = jobId;
  1011. cj.Module = module;
  1012. cj.LotName = lotId;
  1013. cj.LotInnerId = Guid.NewGuid();
  1014. cj.LotWafers = new List<WaferInfo>();
  1015. cj.SetState(EnumControlJobState.WaitingForStart);
  1016. cj.JetState = EnumJetCtrlJobState.Created;
  1017. cj.PreJobClean = preCleanRecipe;
  1018. cj.PostJobClean = postCleanRecipe;
  1019. cj.SequenceNameList = slotSequence;
  1020. cj.CycleNumber = cycleNumber;
  1021. cj.CarrierID = (Singleton<TransferModule>.Instance.GetScheduler(ModuleHelper.Converter(module)) as SchedulerLoadPort).CarrierId;
  1022. _lpCycleSP[ModuleHelper.Converter(module)] = cj.CycleNumber;
  1023. Dictionary<string, bool[]> seqSlot = new Dictionary<string, bool[]>();
  1024. Dictionary<string, List<Tuple<ModuleName, int>>> seqSlotWafers = new Dictionary<string, List<Tuple<ModuleName, int>>>();
  1025. Dictionary<string, string> indexSequence = new Dictionary<string, string>();
  1026. bool enableGroupBySequence = SC.GetValue<bool>("Scheduler.GroupWaferBySequence");
  1027. for (int i = 0; i < SC.GetValue<int>("EFEM.LoadPort.SlotNumber"); i++)
  1028. {
  1029. if (string.IsNullOrEmpty(slotSequence[i]) || string.IsNullOrEmpty(slotSequence[i].Trim()))
  1030. continue;
  1031. string groupName = enableGroupBySequence ? slotSequence[i].Trim() : i.ToString();
  1032. indexSequence[groupName] = slotSequence[i];
  1033. if (!seqSlot.ContainsKey(groupName))
  1034. {
  1035. seqSlot[groupName] = new bool[SC.GetValue<int>("EFEM.LoadPort.SlotNumber")];
  1036. }
  1037. if (!seqSlotWafers.ContainsKey(groupName))
  1038. {
  1039. seqSlotWafers[groupName] = new List<Tuple<ModuleName, int>>();
  1040. }
  1041. seqSlot[groupName][i] = true;
  1042. if (!WaferManager.Instance.CheckHasWafer(module, i))
  1043. {
  1044. LOG.Write(eEvent.WARN_ROUTER, ModuleName.System, $"job wafer: {module} slot {i + 1} not in the carrier");
  1045. return false;
  1046. }
  1047. if (!WaferManager.Instance.CheckWafer(ModuleHelper.Converter(module), i, WaferStatus.Normal))
  1048. {
  1049. reason = $"job wafer: {module} slot {i + 1} status is {WaferManager.Instance.GetWafer(ModuleHelper.Converter(module), i).Status}";
  1050. LOG.Write(eEvent.WARN_ROUTER, ModuleName.System, reason);
  1051. _faCallback.JobCreateFailed(module, lotId, jobId, "");
  1052. return false;
  1053. }
  1054. if (WaferManager.Instance.GetWafer(ModuleHelper.Converter(module), i).ProcessState != EnumWaferProcessStatus.Idle)
  1055. {
  1056. reason = $"job wafer: {module} slot {i + 1} process status is {WaferManager.Instance.GetWafer(ModuleHelper.Converter(module), i).ProcessState}";
  1057. LOG.Write(eEvent.WARN_ROUTER, ModuleName.System, reason);
  1058. _faCallback.JobCreateFailed(module, lotId, jobId, "");
  1059. return false;
  1060. }
  1061. //--- 2024-03-21 增加了waferinfo 相应的lotId信息 start---
  1062. WaferInfo waferInfo = WaferManager.Instance.GetWafer(ModuleHelper.Converter(module), i);
  1063. cj.LotWafers.Add(waferInfo);
  1064. waferInfo.SequenceName = slotSequence[i];
  1065. waferInfo.EAPWaferID = slotWaferId[i];
  1066. waferInfo.LotType = lotType;
  1067. waferInfo.LotId = lotId;
  1068. //--- 2024-03-21 增加了waferinfo 相应的lotId信息 end---
  1069. seqSlotWafers[groupName].Add(Tuple.Create(ModuleHelper.Converter(module), i));
  1070. cj.JobWaferSize = WaferManager.Instance.GetWafer(ModuleHelper.Converter(module), i).Size;
  1071. LOG.Write(eEvent.EV_ROUTER, ModuleName.System, $"Assigned wafer job, wafer {module}.{i + 1}, sequence: {slotSequence[i]}");
  1072. }
  1073. if (seqSlotWafers.Count == 0)
  1074. {
  1075. reason = $"job has not assign wafer";
  1076. LOG.Write(eEvent.WARN_ROUTER, ModuleName.System, reason);
  1077. _faCallback.JobCreateFailed(module, lotId, jobId, "");
  1078. return false;
  1079. }
  1080. List<ProcessJobInfo> pjs = new List<ProcessJobInfo>();
  1081. string[] seqs = seqSlot.Keys.ToArray();
  1082. for (int i = 0; i < seqs.Length; i++)
  1083. {
  1084. ProcessJobInfo pj = new ProcessJobInfo();
  1085. pj.Name = jobId + "_" + (i + 1);
  1086. pj.Sequence = SequenceInfoHelper.KeplerGetInfo(indexSequence[seqs[i]]);
  1087. if (pj.Sequence == null)
  1088. {
  1089. reason = $"invalid sequence[{indexSequence[seqs[i]]}]";
  1090. return false;
  1091. }
  1092. pj.ControlJobName = cj.Name;
  1093. pj.LotName = lotId;
  1094. pj.SlotWafers = seqSlotWafers[seqs[i]];
  1095. pj.SetState(EnumProcessJobState.Queued);
  1096. if (!CheckSequencePmReady(pj.Sequence, out reason))
  1097. {
  1098. LOG.Write(eEvent.WARN_ROUTER, ModuleName.System, $"no valid chamber for the {reason}");
  1099. _faCallback.JobCreateFailed(module, lotId, jobId, "");
  1100. return false;
  1101. }
  1102. if (!RouteManager.IsATMMode && !CheckSequenceRecipeFileValid(pj.Sequence, out reason))
  1103. {
  1104. LOG.Write(eEvent.WARN_ROUTER, ModuleName.System, $"recipe file not valid in the sequence, {reason}");
  1105. _faCallback.JobCreateFailed(module, lotId, jobId, "");
  1106. return false;
  1107. }
  1108. pjs.Add(pj);
  1109. }
  1110. _dbCallback.LotUpdate(cj);
  1111. foreach (var pj in pjs)
  1112. {
  1113. cj.ProcessJobNameList.Add(pj.Name);
  1114. _lstProcessJobs.Add(pj);
  1115. }
  1116. _lstControlJobs.Add(cj);
  1117. //AssociatedPMWithLP(cj);
  1118. _loadportControlJobDic[cj.Module] = cj;
  1119. _faCallback.JobCreated(cj, GetFirstProcessJob(cj));
  1120. return true;
  1121. }
  1122. public bool CheckAllJobDone()
  1123. {
  1124. foreach (var cj in _lstControlJobs)
  1125. {
  1126. if (cj.State == EnumControlJobState.Executing || cj.State == EnumControlJobState.Paused)
  1127. return false;
  1128. }
  1129. InUseRecipes.Clear();
  1130. LP1InUseRecipes.Clear();
  1131. LP2InUseRecipes.Clear();
  1132. LP3InUseRecipes.Clear();
  1133. return true;
  1134. }
  1135. public bool CheckJobJustDone(out string sJobName)
  1136. {
  1137. foreach (var cj in _lstControlJobs)
  1138. {
  1139. if (cj.State == EnumControlJobState.Completed && !cj.BeenPosted)
  1140. {
  1141. if (cj.Module == "LP1")
  1142. {
  1143. LP1InUseRecipes.Clear();
  1144. }
  1145. else if (cj.Module == "LP2")
  1146. {
  1147. LP2InUseRecipes.Clear();
  1148. }
  1149. else if (cj.Module == "LP3")
  1150. {
  1151. LP3InUseRecipes.Clear();
  1152. }
  1153. //_lstControlJobs.Remove(doneJob);
  1154. _loadportControlJobDic[cj.Module] = null;
  1155. //LP;WaferSize;Lot;Number;Start;End;
  1156. sJobName = $"{cj.Module};{cj.JobWaferSize};{cj.LotName};{cj.LotWafers.Count};{cj.StartTime:T};{cj.EndTime:T}";
  1157. cj.BeenPosted = true;
  1158. return true;
  1159. }
  1160. }
  1161. //var doneJob = _lstControlJobs.Where(cj => cj.State == EnumControlJobState.Completed && !cj.BeenPosted).FirstOrDefault();
  1162. //if (doneJob != null)
  1163. //{
  1164. // return true;
  1165. //}
  1166. sJobName = "NULL";
  1167. return false;
  1168. }
  1169. public ProcessJobInfo GetFirstProcessJob(ControlJobInfo cj)
  1170. {
  1171. foreach (var pj in _lstProcessJobs)
  1172. {
  1173. if (pj.ControlJobName == cj.Name)
  1174. return pj;
  1175. }
  1176. return null;
  1177. }
  1178. public bool AbortJob(string jobName, out string reason)
  1179. {
  1180. reason = "";
  1181. ControlJobInfo cj = _lstControlJobs.Find(x => x.Name == jobName);
  1182. if (cj == null)
  1183. {
  1184. reason = $"abort job rejected, not found job with id {jobName}";
  1185. LOG.Write(eEvent.WARN_ROUTER, ModuleName.System, reason);
  1186. return false;
  1187. }
  1188. List<ProcessJobInfo> pjAbortList = new List<ProcessJobInfo>();
  1189. foreach (var pj in _lstProcessJobs)
  1190. {
  1191. if (pj.ControlJobName == cj.Name)
  1192. {
  1193. pj.SetState(EnumProcessJobState.Aborting);
  1194. pjAbortList.Add(pj);
  1195. int unprocessed = 0;
  1196. int aborted = 0;
  1197. WaferInfo[] wafers = WaferManager.Instance.GetWaferByProcessJob(pj.Name);
  1198. foreach (var waferInfo in wafers)
  1199. {
  1200. waferInfo.ProcessJob = null;
  1201. waferInfo.NextSequenceStep = 0;
  1202. if (waferInfo.ProcessState != EnumWaferProcessStatus.Completed)
  1203. unprocessed++;
  1204. }
  1205. JobDataRecorder.EndPJ(pj.InnerId.ToString(), aborted, unprocessed);
  1206. }
  1207. }
  1208. _faCallback.JobAborted(cj, GetFirstProcessJob(cj));
  1209. _dbCallback.LotFinished(cj);
  1210. // remove control job
  1211. var deletedPollingJobs = new Queue<ControlJobInfo>();
  1212. foreach (var item in _qePollingJobs)
  1213. {
  1214. if (cj.InnerId != item.InnerId)
  1215. deletedPollingJobs.Enqueue(item);
  1216. }
  1217. _qePollingJobs = deletedPollingJobs;
  1218. foreach (var pj in pjAbortList)
  1219. {
  1220. _lstProcessJobs.Remove(pj);
  1221. }
  1222. _lstControlJobs.Remove(cj);
  1223. if (_loadportControlJobDic.ContainsKey(cj.Module))
  1224. {
  1225. _loadportControlJobDic[cj.Module] = null;
  1226. }
  1227. return true;
  1228. }
  1229. public bool StopJob(string jobName, out string reason)
  1230. {
  1231. reason = "";
  1232. ControlJobInfo cj = _lstControlJobs.Find(x => x.Name == jobName);
  1233. if (cj == null)
  1234. {
  1235. reason = $"stop job rejected, not found job with id {jobName}";
  1236. LOG.Write(eEvent.WARN_ROUTER, ModuleName.System, reason);
  1237. return false;
  1238. }
  1239. foreach (var pj in _lstProcessJobs)
  1240. {
  1241. if (pj.ControlJobName == cj.Name)
  1242. {
  1243. pj.SetState(EnumProcessJobState.Stopping);
  1244. }
  1245. }
  1246. cj.LotWafers.Clear();
  1247. LOG.Write(eEvent.WARN_ROUTER, ModuleName.System, $"{cj.Module} {jobName} job stop, 剩余wafer不再抓取");
  1248. //_faCallback.JobStopped(cj, GetFirstProcessJob(cj));
  1249. //_dbCallback.LotFinished(cj);
  1250. //if (_loadportControlJobDic.ContainsKey(cj.Module))
  1251. //{
  1252. // _loadportControlJobDic[cj.Module] = null;
  1253. //}
  1254. //var deletedPollingJobs = new Queue<ControlJobInfo>();
  1255. //foreach (var item in _qePollingJobs)
  1256. //{
  1257. // if (cj.InnerId != item.InnerId)
  1258. // deletedPollingJobs.Enqueue(item);
  1259. //}
  1260. //_qePollingJobs = deletedPollingJobs;
  1261. return true;
  1262. }
  1263. public bool ResumeJob(string jobName, out string reason)
  1264. {
  1265. reason = "";
  1266. ControlJobInfo cj = _lstControlJobs.Find(x => x.Name == jobName);
  1267. if (cj == null)
  1268. {
  1269. reason = $"resume job rejected, not found job with id {jobName}";
  1270. //LOG.Write(eEvent.WARN_ROUTER, ModuleName.System, $"resume job rejected, not found job with id {jobName}");
  1271. return false;
  1272. }
  1273. if (cj.State == EnumControlJobState.Paused)
  1274. {
  1275. cj.SetState(EnumControlJobState.Executing);
  1276. }
  1277. _faCallback.JobResumed(cj, GetFirstProcessJob(cj));
  1278. _cycleState = RState.Running;
  1279. return true;
  1280. }
  1281. public bool PauseJob(string jobName, out string reason)
  1282. {
  1283. reason = "";
  1284. ControlJobInfo cj = _lstControlJobs.Find(x => x.Name == jobName);
  1285. if (cj == null)
  1286. {
  1287. reason = $"pause job rejected, not found job with id {jobName}";
  1288. //LOG.Write(eEvent.WARN_ROUTER, ModuleName.System, $"pause job rejected, not found job with id {jobName}");
  1289. return false;
  1290. }
  1291. if (cj.State == EnumControlJobState.Executing)
  1292. {
  1293. cj.SetState(EnumControlJobState.Paused);
  1294. }
  1295. _faCallback.JobPaused(cj, GetFirstProcessJob(cj));
  1296. var deletedPollingJobs = new Queue<ControlJobInfo>();
  1297. foreach (var item in _qePollingJobs)
  1298. {
  1299. if (cj.InnerId != item.InnerId)
  1300. deletedPollingJobs.Enqueue(item);
  1301. }
  1302. _qePollingJobs = deletedPollingJobs;
  1303. if (!_lstControlJobs.Exists(job => job.State == EnumControlJobState.Executing))
  1304. _cycleState = RState.Paused;
  1305. return true;
  1306. }
  1307. public bool StartJob(string jobName, out string reason)
  1308. {
  1309. reason = "";
  1310. ControlJobInfo cj = _lstControlJobs.Find(x => x.Name == jobName);
  1311. if (cj == null)
  1312. {
  1313. reason = $"start job rejected, not found job with id {jobName}";
  1314. LOG.Write(eEvent.WARN_ROUTER, ModuleName.System, reason);
  1315. return false;
  1316. }
  1317. if (RtInstance.ConfigType == ConfigType.Kepler2200)
  1318. {
  1319. var item = _lstProcessJobs.Where(x => x.LotName == cj.LotName).FirstOrDefault();
  1320. if (item != null && item.ControlJobName == cj.Name && !CheckSequenceKepler2200TemperatureReady(item.Sequence))
  1321. {
  1322. return false;
  1323. }
  1324. if (cj.Module == "LP1")
  1325. {
  1326. GetAllJobRecipe(LP1InUseRecipes, cj, item.Sequence);
  1327. }
  1328. else if (cj.Module == "LP2")
  1329. {
  1330. GetAllJobRecipe(LP2InUseRecipes, cj, item.Sequence);
  1331. }
  1332. else if (cj.Module == "LP3")
  1333. {
  1334. GetAllJobRecipe(LP3InUseRecipes, cj, item.Sequence);
  1335. }
  1336. }
  1337. if (cj.State == EnumControlJobState.WaitingForStart)
  1338. {
  1339. cj.SetState(EnumControlJobState.Executing);
  1340. //PreJobClean(cj);
  1341. cj.JetState = EnumJetCtrlJobState.Quequed;
  1342. cj.StartTime = DateTime.Now;
  1343. _dbCallback.LotCreated(cj);
  1344. _faCallback.JobStarted(cj, GetFirstProcessJob(cj));
  1345. //if (!_blockingWatcher.IsRunning)
  1346. // _blockingWatcher.Restart();
  1347. //ResetTraceFlag();
  1348. }
  1349. if (!_cycleWatch.IsRunning)
  1350. {
  1351. _cycleWatch.Restart();
  1352. }
  1353. var lp = ModuleHelper.Converter(cj.Module);
  1354. _lpCycleCount[lp] = 0;
  1355. _lpCycleWafer[lp] = 0;
  1356. _lpThroughput[lp] = 0;
  1357. _cycleState = RState.Running;
  1358. return true;
  1359. }
  1360. public void Clear()
  1361. {
  1362. foreach (var module in _dictModuleTask)
  1363. {
  1364. module.Value.Status = ModuleStatus.Idle;
  1365. module.Value.Scheduler.ResetTask();
  1366. }
  1367. _lstWaferTasks.ForEach(x =>
  1368. {
  1369. if (ModuleHelper.IsLoadLock(x.nextMod) && ModuleHelper.IsLoadPort(x.destMod) && !_qeWaitCoolingLLs.Contains(x.nextMod))
  1370. {
  1371. _qeWaitCoolingLLs.Enqueue(x.nextMod);
  1372. }
  1373. });
  1374. List<string> keys = _loadportControlJobDic.Keys.ToList();
  1375. foreach (var key in keys)
  1376. {
  1377. _loadportControlJobDic[key] = null;
  1378. }
  1379. foreach (var preLot in _preLotCleanMarks)
  1380. {
  1381. preLot.Value.Clear();
  1382. }
  1383. foreach (var postLot in _postLotCleanMarks)
  1384. {
  1385. postLot.Value.Clear();
  1386. }
  1387. _lstWaferTasks.Clear();
  1388. _qeWaitInWafers.Clear();
  1389. _tmSchdActions.Clear();
  1390. _curTmAction.Clear();
  1391. _efemSchdActions.Clear();
  1392. _curEfemAction.Clear();
  1393. _lstControlJobs.Clear();
  1394. _lstProcessJobs.Clear();
  1395. InUseRecipes.Clear();
  1396. LP1InUseRecipes.Clear();
  1397. LP2InUseRecipes.Clear();
  1398. LP3InUseRecipes.Clear();
  1399. _qePollingJobs.Clear();
  1400. //_qeWaitCoolingLLs.Clear();
  1401. _cycleState = RState.End;
  1402. }
  1403. // manual return one wafer while system is auto running
  1404. public bool ManualReturnWafer(object[] objs)
  1405. {
  1406. ModuleName SourceModule = (ModuleName)objs[0];
  1407. int SourceSlot = (int)objs[1];
  1408. //if (!_dictModuleTask.Keys.Contains(SourceModule))
  1409. //{
  1410. // LOG.Write(eEvent.WARN_ROUTER, ModuleName.System, $"Invalid source module {SourceModule} for manual return wafer");
  1411. // return false;
  1412. //}
  1413. //if (WaferManager.Instance.CheckNoWafer(SourceModule, SourceSlot))
  1414. //{
  1415. // LOG.Write(eEvent.WARN_ROUTER, ModuleName.System, $"Can not return wafer as {SourceModule} {SourceSlot} has no wafer");
  1416. // return false;
  1417. //}
  1418. //if (!_dictModuleTask[SourceModule].Scheduler.IsIdle)
  1419. //{
  1420. // LOG.Write(eEvent.WARN_ROUTER, ModuleName.System, $"The module: {SourceModule} is not ready for return wafer");
  1421. // return false;
  1422. //}
  1423. var wafer = WaferManager.Instance.GetWafer(SourceModule, SourceSlot);
  1424. //LOG.Write(eEvent.EV_ROUTER, ModuleName.System, $"Manual return wafer: {wafer.WaferOrigin} at {SourceModule} {SourceSlot} while system is auto running");
  1425. _qeReturnWafers.Enqueue(wafer);
  1426. return true;
  1427. }
  1428. public RState CheckManualReturnWafer()
  1429. {
  1430. var pmWaferCount = _dictModuleTask.Where(mod => ModuleHelper.IsPm(mod.Key) && WaferManager.Instance.CheckHasWafer(mod.Key, 0)).Count();
  1431. var tmRobotWaferCount = (WaferManager.Instance.CheckHasWafer(ModuleName.TMRobot, 0) ? 1 : 0) + (WaferManager.Instance.CheckHasWafer(ModuleName.TMRobot, 1) ? 1 : 0);
  1432. //if (!_dictModuleTask[ModuleName.TMRobot].Scheduler.IsIdle && (pmWaferCount > 0 || tmRobotWaferCount > 0))
  1433. //{
  1434. // LOG.Write(eEvent.ERR_ROUTER, ModuleName.TMRobot, $"The TM Robot is not ready for return wafer.");
  1435. // return RState.Failed;
  1436. //}
  1437. //if (!_dictModuleTask[ModuleName.EfemRobot].Scheduler.IsIdle)
  1438. //{
  1439. // LOG.Write(eEvent.ERR_ROUTER, ModuleName.EfemRobot, $"The EFEM Robot is not ready for return wafer.");
  1440. // return RState.Failed;
  1441. //}
  1442. foreach (var mod in _dictModuleTask)
  1443. {
  1444. if (ModuleHelper.IsLoadPort(mod.Key))
  1445. continue;
  1446. if (!mod.Value.Scheduler.IsInclude)
  1447. continue;
  1448. //if (!mod.Value.Scheduler.IsIdle)
  1449. //{
  1450. // LOG.Write(eEvent.ERR_ROUTER, mod.Key, $"{mod.Key} is not ready for return wafer.");
  1451. // return RState.Failed;
  1452. //}
  1453. int nSlotNumber = ModuleHelper.IsLoadLock(mod.Key) ? (mod.Key == ModuleName.LLA ? _LLASlotNumber : _LLBSlotNumber) : (ModuleHelper.IsTMRobot(mod.Key) || ModuleHelper.IsEFEMRobot(mod.Key) ? 2 : 1);
  1454. for (int slot = 0; slot < nSlotNumber; slot++)
  1455. {
  1456. var wafer = WaferManager.Instance.GetWafer(mod.Key, slot);
  1457. if (wafer.IsEmpty)
  1458. continue;
  1459. //var destLP = (ModuleName)wafer.OriginStation;
  1460. //if (!_dictModuleTask[destLP].Scheduler.IsAvailable)
  1461. //{
  1462. // LOG.Write(eEvent.ERR_ROUTER, destLP, $"The destination Loadport {destLP} is not ready for return wafer.");
  1463. // return RState.Failed;
  1464. //}
  1465. //if (!ModuleHelper.IsLoadPort(destLP))
  1466. //{
  1467. // LOG.Write(eEvent.ERR_ROUTER, ModuleName.System, $"The wafer {wafer.WaferOrigin} cannot be return, please manually drag it.");
  1468. // return RState.Failed;
  1469. //}
  1470. }
  1471. }
  1472. Clear();
  1473. return RState.Running;
  1474. }
  1475. public RState ReturnAllWafers()
  1476. {
  1477. int systemWaferCount = 0;
  1478. foreach (var mod in _dictModuleTask)
  1479. {
  1480. if (ModuleHelper.IsLoadPort(mod.Key) || !mod.Value.Scheduler.IsInclude)
  1481. continue;
  1482. int nSlotNumber = ModuleHelper.IsLoadLock(mod.Key) ? (mod.Key == ModuleName.LLA ? _LLASlotNumber : _LLBSlotNumber) : (ModuleHelper.IsTMRobot(mod.Key) || ModuleHelper.IsEFEMRobot(mod.Key) ? 2 : 1);
  1483. for (int slot = 0; slot < nSlotNumber; slot++)
  1484. {
  1485. var wafer = WaferManager.Instance.GetWafer(mod.Key, slot);
  1486. if (!wafer.IsEmpty)
  1487. systemWaferCount++;
  1488. }
  1489. //var schdCoolingLLs = new List<ModuleName>();
  1490. //if (ModuleHelper.IsLoadLock(mod.Key))
  1491. //{
  1492. // //WaferManager.Instance.GetWafers(mod.Key).Where(x => x.ProcessState != EnumWaferProcessStatus.Idle).Count() > 0
  1493. // if (mod.Value.HasWafer)
  1494. // {
  1495. // schdCoolingLLs.Add(mod.Key);
  1496. // }
  1497. // foreach (var coolLL in schdCoolingLLs)
  1498. // {
  1499. // if (!_dictModuleTask[coolLL].Scheduler.IsAtm)
  1500. // (_dictModuleTask[coolLL] as LoadlockTask).Cooling();
  1501. // }
  1502. //}
  1503. }
  1504. if (systemWaferCount == 0)
  1505. return RState.End;
  1506. if (_qeWaitCoolingLLs.Count > 0 && _dictModuleTask[_qeWaitCoolingLLs.First()].Scheduler.IsIdle && !IsLLReservedByEFEM(_qeWaitCoolingLLs.First()) && !IsLLReservedByTM(_qeWaitCoolingLLs.First()))
  1507. {
  1508. (_dictModuleTask[_qeWaitCoolingLLs.First()] as LoadlockTask).Cooling();
  1509. {
  1510. _qeWaitCoolingLLs.Dequeue();
  1511. return RState.Running;
  1512. }
  1513. }
  1514. ReturnVacWafers();
  1515. ReturnAtmWafers();
  1516. return RState.Running;
  1517. }
  1518. #endregion
  1519. #region internal implementation
  1520. private void InitModules()
  1521. {
  1522. foreach (var module in ModuleHelper.InstalledModules)
  1523. {
  1524. if (ModuleHelper.IsInstalled(module))
  1525. {
  1526. if (ModuleHelper.IsPm(module))
  1527. {
  1528. _dictModuleTask[module] = new PMTask(module);
  1529. _preLotCleanMarks[module] = new List<Guid>();
  1530. _postLotCleanMarks[module] = new List<Guid>();
  1531. }
  1532. else if (ModuleHelper.IsLoadLock(module))
  1533. {
  1534. _dictModuleTask[module] = new LoadlockTask(module);
  1535. }
  1536. else if (ModuleHelper.IsLoadPort(module))
  1537. {
  1538. _dictModuleTask[module] = new LoadPortTask(module);
  1539. }
  1540. else if (ModuleHelper.IsTMRobot(module))
  1541. {
  1542. _dictModuleTask[module] = new TMRobotTask(module);
  1543. }
  1544. else if (ModuleHelper.IsEFEMRobot(module))
  1545. {
  1546. _dictModuleTask[module] = new EFEMRobotTask(module);
  1547. }
  1548. else if (ModuleHelper.IsAligner(module))
  1549. {
  1550. _dictModuleTask[module] = new AlignerTask(module);
  1551. }
  1552. }
  1553. }
  1554. }
  1555. private void prelude()
  1556. {
  1557. bool available = true;
  1558. foreach (var mod in _dictModuleTask)
  1559. {
  1560. available = mod.Value.Scheduler.IsAvailable; // force scheduler update
  1561. }
  1562. }
  1563. private void epilogue()
  1564. {
  1565. UpdateProcessJobStatus();
  1566. UpdateControlJobStatus();
  1567. UpdateLLInOutPathProperty();
  1568. PrepareLLPressure();
  1569. RunLotCleanTasks();
  1570. CreateNewWaferTask();
  1571. _lstWaferTasks.RemoveAll(item => ModuleHelper.IsLoadPort(item.destMod) && ModuleHelper.IsLoadPort(item.currentMod) && item.pressureStatus == RState.End);
  1572. ReDispatchBlockingWafers();
  1573. }
  1574. private void RunWaferTask()
  1575. {
  1576. foreach (var task in _lstWaferTasks)
  1577. {
  1578. task.Run();
  1579. }
  1580. }
  1581. private void RunModuleTasks()
  1582. {
  1583. foreach (var task in _dictModuleTask)
  1584. {
  1585. task.Value.Run();
  1586. }
  1587. }
  1588. private bool CheckAllWaferReturned(ProcessJobInfo pj, bool checkAllProcessed)
  1589. {
  1590. bool allWaferReturn = true;
  1591. foreach (var slot in pj.SlotWafers)
  1592. {
  1593. var wafer = WaferManager.Instance.GetWafer(slot.Item1, slot.Item2);
  1594. if (wafer.IsEmpty || (wafer.ProcessState != EnumWaferProcessStatus.Completed && checkAllProcessed))
  1595. {
  1596. allWaferReturn = false;
  1597. break;
  1598. }
  1599. }
  1600. return allWaferReturn;
  1601. }
  1602. private void UpdateProcessJobStatus()
  1603. {
  1604. if (_efemRobotStatus == RState.Running || _tmRobotStatus == RState.Running)
  1605. return;
  1606. foreach (var pj in _lstProcessJobs)
  1607. {
  1608. if (CheckAllWaferReturned(pj, pj.State != EnumProcessJobState.Stopping))
  1609. {
  1610. pj.SetState(EnumProcessJobState.ProcessingComplete);
  1611. JobDataRecorder.EndPJ(pj.InnerId.ToString(), 0, 0);
  1612. }
  1613. }
  1614. }
  1615. private bool IsControlJobIntersect(ControlJobInfo cj1, ControlJobInfo cj2)
  1616. {
  1617. var cj1Pms = GetWaitPreCleanPMsByCtrlJob(cj1);
  1618. var cj2Pms = GetWaitPreCleanPMsByCtrlJob(cj2);
  1619. return cj1Pms.Intersect(cj2Pms).Count() > 0;
  1620. }
  1621. private void UpdateControlJobStatus()
  1622. {
  1623. if (_lstControlJobs.Count == 0 || _efemRobotStatus == RState.Running || _tmRobotStatus == RState.Running)
  1624. return;
  1625. var runningJobs = _lstControlJobs.FindAll(cj => IsCtrlJobRuning(cj));
  1626. var queueJobs = _lstControlJobs.FindAll(cj => cj.JetState == EnumJetCtrlJobState.Quequed).OrderBy(cj => cj.StartTime);
  1627. if (runningJobs.Count == 0)
  1628. {
  1629. if (queueJobs.Count() > 0)
  1630. {
  1631. // start the first waiting job
  1632. RunControlJob(queueJobs.First());
  1633. }
  1634. }
  1635. else
  1636. {
  1637. foreach (var runningjob in runningJobs)
  1638. {
  1639. // update the running job
  1640. RunControlJob(runningjob);
  1641. }
  1642. foreach (var queueJob in queueJobs)
  1643. {
  1644. if (!runningJobs.Exists(cj => IsControlJobIntersect(cj, queueJob)))
  1645. {
  1646. // parallelly start the not intersect job
  1647. RunControlJob(queueJob);
  1648. }
  1649. else
  1650. {
  1651. foreach (var cj in runningJobs)
  1652. {
  1653. if (!_qePollingJobs.Contains(cj))
  1654. _qePollingJobs.Enqueue(cj);
  1655. }
  1656. RunControlJob(queueJob);
  1657. if (!_qePollingJobs.Contains(queueJob))
  1658. _qePollingJobs.Enqueue(queueJob);
  1659. }
  1660. }
  1661. }
  1662. // update system cycle status
  1663. if (_lstControlJobs.All(cj => cj.State == EnumControlJobState.Completed))
  1664. _cycleState = RState.End;
  1665. List<ControlJobInfo> cjRemoveList = new List<ControlJobInfo>();
  1666. foreach (var cj in _lstControlJobs)
  1667. {
  1668. LoadportCassetteState state = (LoadportCassetteState)DATA.Poll($"{cj.Module}.CassetteState");
  1669. if (cj.State == EnumControlJobState.Completed && state != LoadportCassetteState.Normal)
  1670. {
  1671. cjRemoveList.Add(cj);
  1672. }
  1673. }
  1674. foreach (var cj in cjRemoveList)
  1675. {
  1676. List<ProcessJobInfo> pjRemoveList = new List<ProcessJobInfo>();
  1677. foreach (var pj in _lstProcessJobs)
  1678. {
  1679. if (pj.ControlJobName == cj.Name)
  1680. pjRemoveList.Add(pj);
  1681. }
  1682. foreach (var pj in pjRemoveList)
  1683. {
  1684. _lstProcessJobs.Remove(pj);
  1685. }
  1686. _lstControlJobs.Remove(cj);
  1687. }
  1688. }
  1689. private bool ActiveProcessJob(ProcessJobInfo pj)
  1690. {
  1691. foreach (var pjSlotWafer in pj.SlotWafers)
  1692. {
  1693. WaferInfo wafer = WaferManager.Instance.GetWafer(pjSlotWafer.Item1, pjSlotWafer.Item2);
  1694. wafer.ProcessJob = pj;
  1695. wafer.NextSequenceStep = 0;
  1696. WaferDataRecorder.SetPjInfo(wafer.InnerId.ToString(), pj.InnerId.ToString());
  1697. }
  1698. ControlJobInfo cj = _lstControlJobs.Find(x => x.Name == pj.ControlJobName);
  1699. CarrierInfo carrier = CarrierManager.Instance.GetCarrier(cj.Module);
  1700. JobDataRecorder.StartPJ(pj.InnerId.ToString(), carrier.InnerId.ToString(), cj.InnerId.ToString(), pj.Name, cj.Module, cj.Module, pj.SlotWafers.Count);
  1701. pj.SetState(EnumProcessJobState.Processing);
  1702. return true;
  1703. }
  1704. private bool ActiveControlJob(ControlJobInfo cj)
  1705. {
  1706. cj.JetState = EnumJetCtrlJobState.Processing;
  1707. foreach (var pjName in cj.ProcessJobNameList)
  1708. {
  1709. var pj = _lstProcessJobs.Find(x => x.Name == pjName);
  1710. if (pj == null)
  1711. {
  1712. LOG.Write(eEvent.ERR_ROUTER, ModuleName.System, $"Not find pj named {pjName} in {cj.Name}");
  1713. continue;
  1714. }
  1715. if (pj.State == EnumProcessJobState.Queued)
  1716. {
  1717. ActiveProcessJob(pj);
  1718. }
  1719. }
  1720. return true;
  1721. }
  1722. private bool RunControlJob(ControlJobInfo cj)
  1723. {
  1724. // polling the sequencial jobs
  1725. if (_qePollingJobs.Contains(cj))
  1726. {
  1727. if (_qePollingJobs.First() == cj)
  1728. {
  1729. if (!cj.LotWafers.Exists(wafer => !wafer.IsEmpty && wafer.NextSequenceStep == 0) && (IsAllJobWaferProcessedOrProcessing(cj) || IsAllProcessJobComplete(cj)))
  1730. {
  1731. _qePollingJobs.Dequeue();
  1732. _qePollingJobs.Enqueue(cj);
  1733. }
  1734. }
  1735. }
  1736. switch (cj.JetState)
  1737. {
  1738. case EnumJetCtrlJobState.Quequed:
  1739. {
  1740. ActiveControlJob(cj);
  1741. }
  1742. break;
  1743. case EnumJetCtrlJobState.Processing:
  1744. {
  1745. if (IsAllProcessJobComplete(cj))
  1746. {
  1747. cj.JetState = EnumJetCtrlJobState.Completed;
  1748. }
  1749. }
  1750. break;
  1751. }
  1752. // caculate processed wafer by control job
  1753. var lp = ModuleHelper.Converter(cj.Module);
  1754. int countProcessed = 0;
  1755. foreach (var pjName in cj.ProcessJobNameList)
  1756. {
  1757. var pj = _lstProcessJobs.Find(x => x.Name == pjName);
  1758. if (pj == null)
  1759. {
  1760. LOG.Write(eEvent.ERR_ROUTER, ModuleName.System, $"Not find pj named {pjName} in {cj.Name}");
  1761. continue;
  1762. }
  1763. // caculate process wafer by process
  1764. if (_lpCycleCount[lp] < cj.CycleNumber)
  1765. {
  1766. foreach (var pjSlotWafer in pj.SlotWafers)
  1767. {
  1768. WaferInfo wafer = WaferManager.Instance.GetWafer(pjSlotWafer.Item1, pjSlotWafer.Item2);
  1769. if (!wafer.IsEmpty && wafer.ProcessState == EnumWaferProcessStatus.Completed)
  1770. countProcessed++;
  1771. }
  1772. }
  1773. }
  1774. int lpCycleWafer = 0;
  1775. if (cj.JetState == EnumJetCtrlJobState.Completed)
  1776. {
  1777. cj.EndTime = DateTime.Now;
  1778. _faCallback.JobFinished(cj, GetFirstProcessJob(cj));
  1779. _dbCallback.LotFinished(cj);
  1780. if (_lpCycleCount[lp] >= cj.CycleNumber)
  1781. (Singleton<TransferModule>.Instance.GetScheduler(lp) as SchedulerLoadPort).NoteJobComplete();
  1782. _lpCycleCount[lp]++;
  1783. lpCycleWafer = _lpCycleCount[lp] * cj.LotWafers.Count;
  1784. _lpThroughput[lp] = (float)(lpCycleWafer / (DateTime.Now - cj.StartTime).TotalHours);
  1785. LOG.Write(eEvent.EV_ROUTER, lp, $"Cycle Count: {_lpCycleCount[lp]}, Total Processed Wafer: {lpCycleWafer}, Throughput: {_lpThroughput[lp]}");
  1786. if (_lpCycleCount[lp] < cj.CycleNumber)
  1787. {
  1788. cj.SetState(EnumControlJobState.Executing);
  1789. cj.JetState = EnumJetCtrlJobState.Quequed;
  1790. cj.LotInnerId = Guid.NewGuid();
  1791. _dbCallback.LotCreated(cj);
  1792. foreach (var pj in _lstProcessJobs)
  1793. {
  1794. if (pj.ControlJobName == cj.Name)
  1795. {
  1796. foreach (var marks in _preLotCleanMarks)
  1797. {
  1798. if (marks.Value.Contains(pj.InnerId))
  1799. marks.Value.Remove(pj.InnerId);
  1800. }
  1801. foreach (var marks in _postLotCleanMarks)
  1802. {
  1803. if (marks.Value.Contains(pj.InnerId))
  1804. marks.Value.Remove(pj.InnerId);
  1805. }
  1806. pj.SetState(EnumProcessJobState.Queued);
  1807. pj.InnerId = Guid.NewGuid();
  1808. foreach (var pjSlotWafer in pj.SlotWafers)
  1809. {
  1810. WaferInfo wafer = WaferManager.Instance.GetWafer(pjSlotWafer.Item1, pjSlotWafer.Item2);
  1811. wafer.ProcessJob = null;
  1812. wafer.NextSequenceStep = 0;
  1813. wafer.ProcessState = EnumWaferProcessStatus.Idle;
  1814. }
  1815. }
  1816. }
  1817. _lstWaferTasks.RemoveAll(wt => wt.currentMod == lp);
  1818. }
  1819. else
  1820. {
  1821. cj.SetState(EnumControlJobState.Completed);
  1822. if (_qePollingJobs.Contains(cj))
  1823. {
  1824. Queue<ControlJobInfo> removedPolingJobs = new Queue<ControlJobInfo>();
  1825. foreach (var pj in _qePollingJobs)
  1826. {
  1827. if (pj != cj)
  1828. removedPolingJobs.Enqueue(pj);
  1829. }
  1830. if (removedPolingJobs.Count > 1)
  1831. _qePollingJobs = removedPolingJobs;
  1832. else
  1833. _qePollingJobs.Clear();
  1834. }
  1835. }
  1836. }
  1837. else
  1838. {
  1839. lpCycleWafer = _lpCycleCount[lp] * cj.LotWafers.Count + countProcessed;
  1840. }
  1841. if (lpCycleWafer != _lpCycleWafer[lp])
  1842. {
  1843. _lpCycleWafer[lp] = lpCycleWafer;
  1844. RefreshThroughput();
  1845. }
  1846. return cj.State == EnumControlJobState.Completed;
  1847. }
  1848. void RefreshThroughput()
  1849. {
  1850. foreach (var cj in _lstControlJobs)
  1851. {
  1852. if (cj.State == EnumControlJobState.Executing)
  1853. {
  1854. var lp = ModuleHelper.Converter(cj.Module);
  1855. if (_lpCycleCount[lp] > 0)
  1856. {
  1857. _lpThroughput[lp] = (float)((_lpCycleWafer[lp] - 2) / _lpCycleWatch[lp].Elapsed.TotalHours);
  1858. }
  1859. }
  1860. }
  1861. }
  1862. private bool IsAllJobWaferProcessedOrProcessing(ControlJobInfo cj)
  1863. {
  1864. List<ModuleName> allModules = _dictModuleTask.Keys.ToList();
  1865. int original = (int)ModuleHelper.Converter(cj.Module);
  1866. foreach (var mod in allModules)
  1867. {
  1868. if (ModuleHelper.IsLoadPort(mod) && (int)mod != original)
  1869. continue;
  1870. var wafers = WaferManager.Instance.GetWafers(mod);
  1871. foreach (var wafer in wafers)
  1872. {
  1873. if (wafer.IsEmpty || wafer.ProcessJob == null || wafer.ProcessJob.Sequence == null || wafer.ProcessJob.ControlJobName != cj.Name)
  1874. continue;
  1875. if (wafer.ProcessState != EnumWaferProcessStatus.Completed && wafer.ProcessState != EnumWaferProcessStatus.InProcess)
  1876. return false; ;
  1877. }
  1878. }
  1879. return true;
  1880. }
  1881. private bool IsAllProcessJobComplete(ControlJobInfo cj)
  1882. {
  1883. foreach (var pj in _lstProcessJobs)
  1884. {
  1885. if (pj.ControlJobName == cj.Name && pj.State != EnumProcessJobState.ProcessingComplete)
  1886. {
  1887. return false;
  1888. }
  1889. }
  1890. return true;
  1891. }
  1892. private bool IsCtrlJobExecuting(ModuleName lp)
  1893. {
  1894. var cJob = _lstControlJobs.Find(cj => ModuleHelper.Converter(cj.Module) == lp);
  1895. if (cJob != null && cJob.State == EnumControlJobState.Executing)
  1896. return true;
  1897. return false;
  1898. }
  1899. private bool IsCtrlJobRuning(ControlJobInfo cj)
  1900. {
  1901. return cj.JetState == EnumJetCtrlJobState.PreJobClean ||
  1902. cj.JetState == EnumJetCtrlJobState.Processing ||
  1903. cj.JetState == EnumJetCtrlJobState.PostJobClean;
  1904. }
  1905. private bool IsCtrlJobNeedPreClean(ControlJobInfo cj)
  1906. {
  1907. return !string.IsNullOrWhiteSpace(cj.PreJobClean);
  1908. }
  1909. private bool IsCtrlJobNeedPostClean(ControlJobInfo cj)
  1910. {
  1911. return !string.IsNullOrWhiteSpace(cj.PostJobClean.Trim());
  1912. }
  1913. private List<ModuleName> GetWaitPreCleanPMsByCtrlJob(ControlJobInfo cj)
  1914. {
  1915. var pmlist = new List<ModuleName>();
  1916. foreach (var pj in _lstProcessJobs)
  1917. {
  1918. if (pj.ControlJobName == cj.Name)
  1919. {
  1920. foreach (var pm in pj.Sequence.PMs)
  1921. {
  1922. if (!pmlist.Contains(pm))
  1923. pmlist.Add(pm);
  1924. }
  1925. }
  1926. }
  1927. return pmlist;
  1928. }
  1929. private bool IsPreJobCleanDone(ControlJobInfo cj)
  1930. {
  1931. var pms = GetWaitPreCleanPMsByCtrlJob(cj);
  1932. foreach (var pm in pms)
  1933. {
  1934. var pmTask = _dictModuleTask[pm] as PMTask;
  1935. if (pmTask.Scheduler.IsOnline && !pmTask.IsPreJobCleanDone())
  1936. return false;
  1937. }
  1938. return true;
  1939. }
  1940. private bool IsPostJobCleanDone(ControlJobInfo cj)
  1941. {
  1942. var pms = GetWaitPreCleanPMsByCtrlJob(cj);
  1943. foreach (var pm in pms)
  1944. {
  1945. var pmTask = _dictModuleTask[pm] as PMTask;
  1946. if (pmTask.Scheduler.IsOnline && !pmTask.IsPostJobCleanDone())
  1947. return false;
  1948. }
  1949. return true;
  1950. }
  1951. private void WaferArrived(WaferTask wafer, MoveItem item)
  1952. {
  1953. _dictModuleTask[item.DestinationModule].WaferArrived(wafer, item.DestinationSlot);
  1954. //--2024-03-21 增加了PortJobWaferEnd 上报事件 start--
  1955. if (ModuleHelper.IsLoadPort(item.DestinationModule))
  1956. {
  1957. ControlJobInfo currentControlJob = GetLoadPortCurrentControlJob(item.DestinationModule);
  1958. if (currentControlJob != null)
  1959. {
  1960. _faCallback.JobWaferEnd(currentControlJob, item.DestinationSlot);
  1961. }
  1962. }
  1963. if (ModuleHelper.IsLoadPort(item.DestinationModule))
  1964. {
  1965. if (!_lpCycleWatch[wafer.sourceMod].IsRunning)
  1966. _lpCycleWatch[wafer.sourceMod].Start();
  1967. }
  1968. //--2024-03-21 增加了PortJobWaferEnd 上报事件 end--
  1969. }
  1970. private void WaferLeaved(WaferTask wafer, MoveItem item)
  1971. {
  1972. _dictModuleTask[item.SourceModule].WaferLeaved(wafer, item.DestinationSlot);
  1973. if (ModuleHelper.IsPm(item.SourceModule))
  1974. {
  1975. _pmLastWaferEndTime[item.SourceModule] = DateTime.Now;
  1976. }
  1977. //--2024-03-21 增加了PortJobWaferStart 上报事件 start--
  1978. if (ModuleHelper.IsLoadPort(item.SourceModule))
  1979. {
  1980. ControlJobInfo currentControlJob = GetLoadPortCurrentControlJob(item.SourceModule);
  1981. if (currentControlJob != null)
  1982. {
  1983. _faCallback.JobWaferStart(currentControlJob, item.SourceSlot);
  1984. }
  1985. }
  1986. //--2024-03-21 增加了PortJobWaferStart 上报事件 end--
  1987. }
  1988. private ModuleName GetComingAvailablePM(ControlJobInfo cj)
  1989. {
  1990. var lstInProcessPMs = new List<ModuleName>();
  1991. foreach (var wafer in cj.LotWafers)
  1992. {
  1993. if (!wafer.IsEmpty && wafer.NextSequenceStep == 0 && wafer.ProcessJob != null)
  1994. {
  1995. foreach (var pm in wafer.ProcessJob.Sequence.PMs)
  1996. {
  1997. if (!lstInProcessPMs.Contains(pm) && _dictModuleTask[pm].Scheduler.IsOnline)
  1998. lstInProcessPMs.Add(pm);
  1999. }
  2000. }
  2001. }
  2002. // blocking check while only one loadlock in out
  2003. if (LLInOutPath == SequenceLLInOutPath.AInAOut ||
  2004. LLInOutPath == SequenceLLInOutPath.BInBOut ||
  2005. !_dictModuleTask[ModuleName.LLA].Scheduler.IsOnline ||
  2006. !_dictModuleTask[ModuleName.LLB].Scheduler.IsOnline)
  2007. {
  2008. if (_lstWaferTasks.Count >= lstInProcessPMs.Count + _LLASlotNumber / 2)
  2009. return ModuleName.System;
  2010. }
  2011. Dictionary<ModuleName, int> pmAssociatedWaferCount = new Dictionary<ModuleName, int>();
  2012. foreach (var mod in _dictModuleTask)
  2013. {
  2014. if (ModuleHelper.IsPm(mod.Key) && mod.Value.Scheduler.IsOnline && lstInProcessPMs.Contains(mod.Key))
  2015. {
  2016. pmAssociatedWaferCount[mod.Key] = 0;
  2017. }
  2018. }
  2019. var unprocessedWafer = _lstWaferTasks.Where(task => ModuleHelper.IsPm(task.destMod));
  2020. foreach (var wafer in unprocessedWafer)
  2021. {
  2022. if (ModuleHelper.IsPm(wafer.destMod) && _dictModuleTask[wafer.destMod].Scheduler.IsOnline && lstInProcessPMs.Contains(wafer.destMod))
  2023. {
  2024. pmAssociatedWaferCount[wafer.destMod]++;
  2025. }
  2026. }
  2027. var oderedPMCount = pmAssociatedWaferCount.OrderBy(p => p.Value).ToDictionary(p => p.Key, o => o.Value);
  2028. if (oderedPMCount.Count > 0 && oderedPMCount.First().Value < 2)
  2029. return oderedPMCount.First().Key;
  2030. return ModuleName.System;
  2031. }
  2032. private void CreateNewWaferTask()
  2033. {
  2034. var runningJobs = _lstControlJobs.FindAll(cj => cj.JetState == EnumJetCtrlJobState.PreJobClean || cj.JetState == EnumJetCtrlJobState.Processing);
  2035. var onDutyPms = new Dictionary<ModuleName, int>();
  2036. foreach (var job in runningJobs)
  2037. {
  2038. var pms = GetWaitPreCleanPMsByCtrlJob(job);
  2039. foreach (var pm in pms)
  2040. {
  2041. if (_dictModuleTask[pm].Scheduler.IsOnline && !onDutyPms.ContainsKey(pm))
  2042. {
  2043. onDutyPms[pm] = _lstWaferTasks.Where(wt => wt.destMod == pm || wt.currentMod == pm).Count();
  2044. }
  2045. }
  2046. }
  2047. if (runningJobs.Count == 0 || onDutyPms.Count == 0)
  2048. return;
  2049. var oderedPMs = onDutyPms.OrderBy(kv => kv.Value).OrderBy(kv => _dictModuleTask[kv.Key].TimeToReady);
  2050. if (oderedPMs.First().Value < 1)
  2051. {
  2052. foreach (var runningJob in runningJobs)
  2053. {
  2054. //if (_qePollingJobs.Contains(runningJob) && _qePollingJobs.First() != runningJob)
  2055. // continue;
  2056. foreach (var wafer in runningJob.LotWafers)
  2057. {
  2058. if (wafer.IsEmpty || wafer.ProcessJob == null)
  2059. continue;
  2060. if (wafer.ProcessJob.Sequence.PMs.Contains(oderedPMs.First().Key) && wafer.NextSequenceStep == 0 && !_lstWaferTasks.Exists(task => task.sourceMod == (ModuleName)wafer.OriginStation && task.sourceSlot == wafer.OriginSlot))
  2061. {
  2062. CreateWaferTasks(wafer, oderedPMs.First().Key);
  2063. return;
  2064. }
  2065. }
  2066. }
  2067. }
  2068. if (_tmRobotStatus == RState.Running)
  2069. return;
  2070. // keep two raw atm wafer, try match a double place to LL
  2071. var atmWaferCount = _lstWaferTasks.Count(wt => ModuleHelper.IsPm(wt.destMod) && (ModuleHelper.IsLoadPort(wt.currentMod) || ModuleHelper.IsAligner(wt.currentMod) || ModuleHelper.IsEFEMRobot(wt.currentMod) || ModuleHelper.IsLoadLock(wt.currentMod)));
  2072. if (atmWaferCount < 2)
  2073. {
  2074. var busyPMs = oderedPMs.Where(pv => pv.Value == 1 && _dictModuleTask[pv.Key].HasWafer).OrderBy(pv => _dictModuleTask[pv.Key].TimeToReady);
  2075. if (busyPMs.Count() == 0)
  2076. return;
  2077. ModuleName nextPM = busyPMs.First().Key;
  2078. if (_lpCycleWafer.Sum(kv => kv.Value) == 0)
  2079. {
  2080. if (busyPMs.ToList().Exists(kv => _dictModuleTask[kv.Key].TimeToReady > 36000) && busyPMs.ToList().Exists(pv => _dictModuleTask[pv.Key].TimeToReady < 3600))
  2081. {
  2082. foreach (var pm in busyPMs)
  2083. {
  2084. if (_dictModuleTask[pm.Key].TimeToReady > 36000)
  2085. {
  2086. nextPM = pm.Key;
  2087. break;
  2088. }
  2089. }
  2090. }
  2091. }
  2092. foreach (var runningJob in runningJobs)
  2093. {
  2094. //if (_qePollingJobs.Contains(runningJob) && _qePollingJobs.First() != runningJob)
  2095. // continue;
  2096. foreach (var wafer in runningJob.LotWafers)
  2097. {
  2098. if (wafer.IsEmpty || wafer.ProcessJob == null)
  2099. continue;
  2100. if (wafer.ProcessJob.Sequence.PMs.Contains(nextPM) && wafer.NextSequenceStep == 0 && !_lstWaferTasks.Exists(task => task.sourceMod == (ModuleName)wafer.OriginStation && task.sourceSlot == wafer.OriginSlot))
  2101. {
  2102. CreateWaferTasks(wafer, nextPM);
  2103. return;
  2104. }
  2105. }
  2106. }
  2107. }
  2108. }
  2109. private void CreateWaferTasks(WaferInfo wafer, ModuleName pm)
  2110. {
  2111. var recipeName = wafer.ProcessJob.Sequence.GetRecipe(pm);
  2112. var recipeContent = RecipeFileManager.Instance.LoadRecipe(pm.ToString(), recipeName, false, RecipeType.Process.ToString());
  2113. Recipe recipe = Recipe.Load(recipeContent);
  2114. int temperature = 0;
  2115. if (int.TryParse(recipe.Header.Temperature, out int temp))
  2116. {
  2117. temperature = temp;
  2118. }
  2119. var waferTask = new WaferTask((ModuleName)wafer.OriginStation,
  2120. wafer.OriginSlot,
  2121. pm,
  2122. 0,
  2123. temperature,
  2124. wafer.InnerId,
  2125. wafer.ProcessJob.InnerId,
  2126. recipeName,
  2127. wafer.ProcessJob.Sequence.GetWTWCleanRecipe(pm),
  2128. wafer.ProcessJob.Sequence.LLInOutPath,
  2129. wafer.ProcessJob.Sequence.LLDelayTime,
  2130. IsSequenceNeedAlign(wafer.ProcessJob.Sequence));
  2131. waferTask.OnWaferArrived += WaferArrived;
  2132. waferTask.OnWaferLeaved += WaferLeaved;
  2133. _lstWaferTasks.Add(waferTask);
  2134. _qeWaitInWafers.Enqueue(wafer.InnerId);
  2135. //LOG.Write(eEvent.INFO_TROUGHPUT_ENHANCE_TRACE, waferTask.sourceMod, $"Routing wafer: {waferTask.sourceMod}.{waferTask.sourceSlot + 1} to {pm}");
  2136. }
  2137. private void ReDispatchBlockingWafers()
  2138. {
  2139. foreach (var wt in _lstWaferTasks)
  2140. {
  2141. if (ModuleHelper.IsPm(wt.destMod) && !ModuleHelper.IsPm(wt.currentMod) && !ModuleHelper.IsLoadPort(wt.currentMod))
  2142. {
  2143. if (!_dictModuleTask[wt.destMod].Scheduler.IsOnline && wt.elapseTime > 120)
  2144. {
  2145. var wafer = WaferManager.Instance.GetWafer(wt.currentMod, wt.currentSlot);
  2146. var otherPMs = _dictModuleTask.Where(mod => wafer.ProcessJob.Sequence.PMs.Contains(mod.Key) && mod.Value.Scheduler.IsOnline).OrderBy(mod => _lstWaferTasks.Where(task => task.destMod == mod.Key).Count()).ToList();
  2147. if (otherPMs.Count > 0)
  2148. {
  2149. LOG.Write(eEvent.WARN_ROUTER, wt.currentMod, $"Re-assign process chamber {otherPMs.First().Key} for {wafer.WaferOrigin} as {wt.destMod} unexpected offline");
  2150. wt.ReAssignPM(otherPMs.First().Key, 0);
  2151. }
  2152. }
  2153. }
  2154. }
  2155. }
  2156. private bool ForwardATMWafers(ModuleName ll)
  2157. {
  2158. if (_qeWaitInWafers.Count == 0)
  2159. return false;
  2160. var emptySlots = GetLLReadyInOutSlots(ll).emptySlot;
  2161. var atmWafers = _lstWaferTasks.Where(wafer => (wafer.movingStatus == RState.End || wafer.movingStatus == RState.Init) && (ModuleHelper.IsEFEMRobot(wafer.currentMod) || ModuleHelper.IsAligner(wafer.currentMod))).ToList();
  2162. var notAlignedWafer = atmWafers.Where(wafer => !wafer.IsAligned).ToList();
  2163. var freeHands = GetEFEMFreeHand();
  2164. var waitInWafer = _lstWaferTasks.Find(wt => wt.waferId == _qeWaitInWafers.First());
  2165. if (atmWafers.Count >= 2 && freeHands.Count == 0 || waitInWafer == null)
  2166. return false;
  2167. // do not need pass by aligner
  2168. if (waitInWafer.IsAligned)
  2169. {
  2170. _efemSchdActions.Enqueue(new List<MoveItem> { new MoveItem(waitInWafer.currentMod, waitInWafer.currentSlot, ModuleName.EfemRobot, (int)freeHands[0], freeHands[0]) });
  2171. _qeWaitInWafers.Dequeue();
  2172. }
  2173. else
  2174. {
  2175. if (atmWafers.Count == 0)
  2176. {
  2177. // push one wafer to aligner
  2178. _efemSchdActions.Enqueue(new List<MoveItem> { new MoveItem(waitInWafer.currentMod, waitInWafer.currentSlot, ModuleName.EfemRobot, (int)freeHands[0], freeHands[0]) });
  2179. _efemSchdActions.Enqueue(new List<MoveItem> { new MoveItem(ModuleName.EfemRobot, (int)freeHands[0], ModuleName.Aligner1, 0, freeHands[0]) });
  2180. _qeWaitInWafers.Dequeue();
  2181. }
  2182. else if (_lstWaferTasks.Exists(wt => ModuleHelper.IsAligner(wt.currentMod)) && freeHands.Count == 2)
  2183. {
  2184. // forward another wafer to aligner
  2185. _efemSchdActions.Enqueue(new List<MoveItem> { new MoveItem(waitInWafer.currentMod, waitInWafer.currentSlot, ModuleName.EfemRobot, (int)freeHands[1], freeHands[1]) });
  2186. _efemSchdActions.Enqueue(new List<MoveItem> { new MoveItem(ModuleName.Aligner1, 0, ModuleName.EfemRobot, (int)freeHands[0], freeHands[0]) });
  2187. _efemSchdActions.Enqueue(new List<MoveItem> { new MoveItem(ModuleName.EfemRobot, (int)freeHands[1], ModuleName.Aligner1, 0, freeHands[1]) });
  2188. _qeWaitInWafers.Dequeue();
  2189. }
  2190. }
  2191. return _efemSchdActions.Count > 0;
  2192. }
  2193. private bool ExchangeWaferWithLL(ModuleName ll)
  2194. {
  2195. if (IsLLReservedByTM(ll))
  2196. return false;
  2197. var waferStaus = GetLLReadyInOutSlots(ll);
  2198. var atmWafers = _lstWaferTasks.Where(wafer => (wafer.movingStatus == RState.End || wafer.movingStatus == RState.Init) && wafer.IsAligned && (ModuleHelper.IsEFEMRobot(wafer.currentMod) || ModuleHelper.IsAligner(wafer.currentMod))).ToList();
  2199. var freeHands = GetEFEMFreeHand();
  2200. var swapActions = new List<MoveItem>();
  2201. var validHands = new List<Hand>();
  2202. int placeCount = 0;
  2203. if (waferStaus.emptySlot.Count > 0 &&
  2204. freeHands.Count >= 1 &&
  2205. atmWafers.Count(wafer => ModuleHelper.IsAligner(wafer.currentMod)) == 1 &&
  2206. (atmWafers.Count(wafer => ModuleHelper.IsEFEMRobot(wafer.currentMod)) == 0 || waferStaus.emptySlot.Count >= 2))
  2207. {
  2208. var alignerWafer = atmWafers.Where(wafer => ModuleHelper.IsAligner(wafer.currentMod)).First();
  2209. if (CanWaferGotoLL(alignerWafer, ll))
  2210. {
  2211. alignerWafer.RouteTo(ll, waferStaus.emptySlot[placeCount]);
  2212. _efemSchdActions.Enqueue(new List<MoveItem> { new MoveItem(alignerWafer.currentMod, alignerWafer.currentSlot, ModuleName.EfemRobot, (int)freeHands[0], freeHands[0]) });
  2213. swapActions.Add(new MoveItem(ModuleName.EfemRobot, (int)freeHands[0], ll, waferStaus.emptySlot[placeCount], freeHands[0]));
  2214. validHands.Add(freeHands[0]);
  2215. placeCount++;
  2216. }
  2217. }
  2218. var armWafers = atmWafers.Where(wafer => ModuleHelper.IsEFEMRobot(wafer.currentMod)).ToArray();
  2219. foreach (var wafer in armWafers)
  2220. {
  2221. if (CanWaferGotoLL(wafer, ll) && placeCount < waferStaus.emptySlot.Count)
  2222. {
  2223. wafer.RouteTo(ll, waferStaus.emptySlot[placeCount]);
  2224. swapActions.Add(new MoveItem(wafer.currentMod, wafer.currentSlot, ll, waferStaus.emptySlot[placeCount], (Hand)wafer.currentSlot));
  2225. if (!validHands.Contains((Hand)wafer.currentSlot))
  2226. validHands.Add((Hand)wafer.currentSlot);
  2227. placeCount++;
  2228. }
  2229. }
  2230. if (!validHands.Contains(Hand.Blade1) && WaferManager.Instance.CheckNoWafer(ModuleName.EfemRobot, (int)Hand.Blade1) && _efemRobotSingleArmOption != 2)
  2231. validHands.Add(Hand.Blade1);
  2232. if (!validHands.Contains(Hand.Blade2) && WaferManager.Instance.CheckNoWafer(ModuleName.EfemRobot, (int)Hand.Blade2) && _efemRobotSingleArmOption != 1)
  2233. validHands.Add(Hand.Blade2);
  2234. int pickCount = 0;
  2235. var returnActions = new List<MoveItem>();
  2236. foreach (var slot in waferStaus.outSlot)
  2237. {
  2238. if (pickCount < validHands.Count)
  2239. {
  2240. var outWafer = _lstWaferTasks.Find(wafer => (wafer.movingStatus == RState.End || wafer.movingStatus == RState.Init) && wafer.currentMod == ll && wafer.currentSlot == slot);
  2241. if (outWafer != null)
  2242. {
  2243. outWafer.RouteTo(outWafer.destMod, outWafer.destSlot);
  2244. swapActions.Add(new MoveItem(ll, waferStaus.outSlot[pickCount], ModuleName.EfemRobot, (int)validHands[pickCount], validHands[pickCount]));
  2245. returnActions.Add(new MoveItem(ModuleName.EfemRobot, (int)validHands[pickCount], outWafer.destMod, outWafer.destSlot, validHands[pickCount]));
  2246. pickCount++;
  2247. }
  2248. }
  2249. }
  2250. if (swapActions.Count > 0)
  2251. {
  2252. _efemSchdActions.Enqueue(swapActions);
  2253. }
  2254. foreach (var action in returnActions)
  2255. {
  2256. _efemSchdActions.Enqueue(new List<MoveItem> { action });
  2257. }
  2258. return _efemSchdActions.Count > 0;
  2259. }
  2260. private bool ExchangeWaferWithFixedSlotLL(ModuleName ll)
  2261. {
  2262. if (IsLLReservedByTM(ll))
  2263. return false;
  2264. var llWaferStatus = GetLLFixedReadyInOutSlots(ll);
  2265. var atmWafers = _lstWaferTasks.Where(wafer => (wafer.movingStatus == RState.End || wafer.movingStatus == RState.Init) && wafer.IsAligned && (ModuleHelper.IsEFEMRobot(wafer.currentMod) || ModuleHelper.IsAligner(wafer.currentMod))).ToList();
  2266. var freeHands = GetEFEMFreeHand();
  2267. var swapActions = new List<MoveItem>();
  2268. var validHands = new List<Hand>();
  2269. int placeCount = 0;
  2270. if (llWaferStatus.eInSlot.Count > 0 &&
  2271. freeHands.Count >= 1 &&
  2272. atmWafers.Count(wafer => ModuleHelper.IsAligner(wafer.currentMod)) == 1 &&
  2273. (atmWafers.Count(wafer => ModuleHelper.IsEFEMRobot(wafer.currentMod)) == 0 || llWaferStatus.eInSlot.Count >= 2))
  2274. {
  2275. var alignerWafer = atmWafers.Where(wafer => ModuleHelper.IsAligner(wafer.currentMod)).First();
  2276. if (CanWaferGotoLL(alignerWafer, ll))
  2277. {
  2278. alignerWafer.RouteTo(ll, llWaferStatus.eInSlot[placeCount]);
  2279. _efemSchdActions.Enqueue(new List<MoveItem> { new MoveItem(alignerWafer.currentMod, alignerWafer.currentSlot, ModuleName.EfemRobot, (int)freeHands[0], freeHands[0]) });
  2280. swapActions.Add(new MoveItem(ModuleName.EfemRobot, (int)freeHands[0], ll, llWaferStatus.eInSlot[placeCount], freeHands[0]));
  2281. validHands.Add(freeHands[0]);
  2282. placeCount++;
  2283. }
  2284. }
  2285. var armWafers = atmWafers.Where(wafer => ModuleHelper.IsEFEMRobot(wafer.currentMod)).ToArray();
  2286. foreach (var wafer in armWafers)
  2287. {
  2288. if (CanWaferGotoLL(wafer, ll) && placeCount < llWaferStatus.eInSlot.Count)
  2289. {
  2290. wafer.RouteTo(ll, llWaferStatus.eInSlot[placeCount]);
  2291. swapActions.Add(new MoveItem(wafer.currentMod, wafer.currentSlot, ll, llWaferStatus.eInSlot[placeCount], (Hand)wafer.currentSlot));
  2292. if (!validHands.Contains((Hand)wafer.currentSlot))
  2293. validHands.Add((Hand)wafer.currentSlot);
  2294. placeCount++;
  2295. }
  2296. }
  2297. if (!validHands.Contains(Hand.Blade1) && WaferManager.Instance.CheckNoWafer(ModuleName.EfemRobot, (int)Hand.Blade1) && _efemRobotSingleArmOption != 2)
  2298. validHands.Add(Hand.Blade1);
  2299. if (!validHands.Contains(Hand.Blade2) && WaferManager.Instance.CheckNoWafer(ModuleName.EfemRobot, (int)Hand.Blade2) && _efemRobotSingleArmOption != 1)
  2300. validHands.Add(Hand.Blade2);
  2301. int pickCount = 0;
  2302. var returnActions = new List<MoveItem>();
  2303. foreach (var slot in llWaferStatus.eOutSlot)
  2304. {
  2305. if (pickCount < validHands.Count)
  2306. {
  2307. var outWafer = _lstWaferTasks.Find(wafer => (wafer.movingStatus == RState.End || wafer.movingStatus == RState.Init) && wafer.currentMod == ll && wafer.currentSlot == slot);
  2308. if (outWafer != null)
  2309. {
  2310. outWafer.RouteTo(outWafer.destMod, outWafer.destSlot);
  2311. swapActions.Add(new MoveItem(ll, llWaferStatus.eOutSlot[pickCount], ModuleName.EfemRobot, (int)validHands[pickCount], validHands[pickCount]));
  2312. returnActions.Add(new MoveItem(ModuleName.EfemRobot, (int)validHands[pickCount], outWafer.destMod, outWafer.destSlot, validHands[pickCount]));
  2313. pickCount++;
  2314. }
  2315. }
  2316. }
  2317. if (swapActions.Count > 0)
  2318. {
  2319. _efemSchdActions.Enqueue(swapActions);
  2320. }
  2321. foreach (var action in returnActions)
  2322. {
  2323. _efemSchdActions.Enqueue(new List<MoveItem> { action });
  2324. }
  2325. return _efemSchdActions.Count > 0;
  2326. }
  2327. private void RoutingATMWafers()
  2328. {
  2329. if (_efemSchdActions.Count > 0 || _curEfemAction.Count > 0)
  2330. {
  2331. RunSchdEFEMActions();
  2332. return;
  2333. }
  2334. if (_efemRobotStatus != RState.End)
  2335. return;
  2336. if (_LLSlotInOutOption == LLSlotInOutOpt.AllInAllOut)
  2337. {
  2338. if (_LLASlotNumber == 2)
  2339. {
  2340. Routing2SlotATMSystem();
  2341. }
  2342. else if (_LLASlotNumber == 4)
  2343. {
  2344. Routing4SlotATMSystem();
  2345. }
  2346. }
  2347. else
  2348. {
  2349. RoutingFixedSlotATMSystem();
  2350. }
  2351. }
  2352. private void Routing4SlotATMSystem()
  2353. {
  2354. var atmWaferCount = _lstWaferTasks.Where(wafer => ModuleHelper.IsEFEMRobot(wafer.currentMod) || ModuleHelper.IsAligner(wafer.currentMod)).Count();
  2355. var efemRobotWafers = _lstWaferTasks.Where(wafer => ModuleHelper.IsEFEMRobot(wafer.currentMod)).ToList();
  2356. if (LLInOutPath == SequenceLLInOutPath.AInBOut || LLInOutPath == SequenceLLInOutPath.BInAOut) // don need check system wafer existence as no blocking risk
  2357. {
  2358. var inLL = LLInOutPath == SequenceLLInOutPath.AInBOut ? ModuleName.LLA : ModuleName.LLB;
  2359. var outLL = LLInOutPath == SequenceLLInOutPath.AInBOut ? ModuleName.LLB : ModuleName.LLA;
  2360. var inLLWaferStatus = GetLLReadyInOutSlots(inLL);
  2361. var outLLWaferStatus = GetLLReadyInOutSlots(outLL);
  2362. if (efemRobotWafers.Count() >= 1)
  2363. {
  2364. if (inLLWaferStatus.emptySlot.Count >= 2 && !IsLLReservedByTM(inLL)) // wait for in loadlock ready
  2365. {
  2366. if (_dictModuleTask[inLL].TimeToReady < 2)
  2367. {
  2368. ExchangeWaferWithLL(inLL);
  2369. }
  2370. }
  2371. if ((inLLWaferStatus.emptySlot.Count == 0 || (outLLWaferStatus.outSlot.Count >= 1 && _dictModuleTask[outLL].TimeToReady < 2)) && !IsLLReservedByTM(outLL)) // ready return wafer
  2372. {
  2373. ExchangeWaferWithLL(outLL);
  2374. }
  2375. }
  2376. else
  2377. {
  2378. if (((outLLWaferStatus.outSlot.Count >= 2 && _dictModuleTask[outLL].TimeToReady < 10) ||
  2379. (outLLWaferStatus.outSlot.Count == 1 && _dictModuleTask[outLL].TimeToReady < 2)) && !IsLLReservedByTM(outLL))
  2380. {
  2381. ExchangeWaferWithLL(outLL);
  2382. }
  2383. else if (inLLWaferStatus.emptySlot.Count >= 2)
  2384. {
  2385. if (((_dictModuleTask[inLL].TimeToReady > 20 || _dictModuleTask[inLL].Scheduler.IsVac) && atmWaferCount < 2) || atmWaferCount == 0)
  2386. {
  2387. ForwardATMWafers(inLL);
  2388. }
  2389. else if (atmWaferCount >= 2 && !IsLLReservedByTM(inLL))
  2390. {
  2391. ExchangeWaferWithLL(inLL);
  2392. }
  2393. }
  2394. }
  2395. }
  2396. else
  2397. {
  2398. RoutingSameLLInOutATMSystem();
  2399. }
  2400. }
  2401. private void Routing2SlotATMSystem()
  2402. {
  2403. var atmWaferCount = _lstWaferTasks.Where(wafer => ModuleHelper.IsEFEMRobot(wafer.currentMod) || ModuleHelper.IsAligner(wafer.currentMod)).Count();
  2404. var efemRobotWafers = _lstWaferTasks.Where(wafer => ModuleHelper.IsEFEMRobot(wafer.currentMod)).ToList();
  2405. var validHands = (_efemRobotSingleArmOption == 0 ? 2 : 1);
  2406. if (LLInOutPath == SequenceLLInOutPath.AInBOut || LLInOutPath == SequenceLLInOutPath.BInAOut) // don need check system wafer existence as no blocking risk
  2407. {
  2408. var inLL = LLInOutPath == SequenceLLInOutPath.AInBOut ? ModuleName.LLA : ModuleName.LLB;
  2409. var outLL = LLInOutPath == SequenceLLInOutPath.AInBOut ? ModuleName.LLB : ModuleName.LLA;
  2410. var inLLWaferStatus = GetLLReadyInOutSlots(inLL);
  2411. var outLLWaferStatus = GetLLReadyInOutSlots(outLL);
  2412. var inLLModule = _dictModuleTask[inLL] as LoadlockTask;
  2413. var outLLModule = _dictModuleTask[outLL] as LoadlockTask;
  2414. var needAlign = _lstProcessJobs.Exists(pj => pj.State == EnumProcessJobState.Processing && IsSequenceNeedAlign(pj.Sequence));
  2415. // return wafer while out LL ready or cooling state
  2416. //if ((outLLWaferStatus.outSlot.Count > 0 && outLLWaferStatus.outSlot.Count <= (validHands - efemRobotWafers.Count)) && outLLModule.ReayForEfemInTime(10))
  2417. if (outLLWaferStatus.outSlot.Count > 0 && outLLModule.ReayForEfemInTime(10))
  2418. {
  2419. if (ExchangeWaferWithLL(outLL))
  2420. {
  2421. //LOG.Write(eEvent.INFO_TROUGHPUT_ENHANCE_TRACE, outLL, $"{outLL} will be ready in {outLLModule.TimeToReady}");
  2422. return;
  2423. }
  2424. }
  2425. if (needAlign)
  2426. {
  2427. if (atmWaferCount == 0 ||
  2428. (inLLModule.ReayForEfemInTime(20) && Math.Min(validHands, inLLWaferStatus.emptySlot.Count) > atmWaferCount))
  2429. {
  2430. if (ForwardATMWafers(inLL))
  2431. {
  2432. ////LOG.Write(eEvent.INFO_TROUGHPUT_ENHANCE_TRACE, inLL, $"{inLL} will be ready in {inLLModule.TimeToReady}");
  2433. return;
  2434. }
  2435. }
  2436. }
  2437. else
  2438. {
  2439. if (inLLModule.ReayForEfemInTime(20) && inLLWaferStatus.emptySlot.Count > efemRobotWafers.Count && ((outLLModule.TimeToReady - inLLModule.TimeToReady > 10) || !outLLModule.ReayForEfemInTime(30) || outLLWaferStatus.emptySlot.Count < efemRobotWafers.Count))
  2440. {
  2441. if (ForwardATMWafers(inLL))
  2442. {
  2443. //LOG.Write(eEvent.INFO_TROUGHPUT_ENHANCE_TRACE, inLL, $"{inLL} will be ready in {inLLModule.TimeToReady}");
  2444. return;
  2445. }
  2446. }
  2447. }
  2448. if (inLLModule.ReayForEfemInTime(5) &&
  2449. (Math.Min(validHands, inLLWaferStatus.emptySlot.Count) == atmWaferCount || _qeWaitInWafers.Count == 0))
  2450. {
  2451. if (ExchangeWaferWithLL(inLL))
  2452. {
  2453. //LOG.Write(eEvent.INFO_TROUGHPUT_ENHANCE_TRACE, inLL, $"{inLL} will be ready in {inLLModule.TimeToReady}");
  2454. return;
  2455. }
  2456. }
  2457. }
  2458. else
  2459. {
  2460. RoutingSameLLInOutATMSystem();
  2461. }
  2462. }
  2463. private void RoutingFixedSlotATMSystem()
  2464. {
  2465. var atmWaferCount = _lstWaferTasks.Where(wafer => ModuleHelper.IsEFEMRobot(wafer.currentMod) || ModuleHelper.IsAligner(wafer.currentMod)).Count();
  2466. var lls = _dictModuleTask.Where(mod => ModuleHelper.IsLoadLock(mod.Key) && mod.Value.Scheduler.IsOnline);
  2467. if (atmWaferCount == 0 || (_efemRobotSingleArmOption == 0 && atmWaferCount == 1))
  2468. {
  2469. foreach (var ll in lls)
  2470. {
  2471. if (ForwardATMWafers(ll.Key))
  2472. return;
  2473. }
  2474. }
  2475. // try to match a ll swap action
  2476. var readyReturnLL = lls.Where(ll => (GetLLFixedReadyInOutSlots(ll.Key).eOutSlot.Count + GetLLFixedReadyInOutSlots(ll.Key).eInSlot.Count > 0) && ll.Value.TimeToReady <= 10)
  2477. .OrderByDescending(ll => GetLLFixedReadyInOutSlots(ll.Key).eInSlot.Count + GetLLFixedReadyInOutSlots(ll.Key).eOutSlot.Count)
  2478. .OrderBy(ll => ll.Value.TimeToReady);
  2479. foreach (var ll in readyReturnLL)
  2480. {
  2481. if (ExchangeWaferWithFixedSlotLL(ll.Key))
  2482. return;
  2483. }
  2484. }
  2485. private void RoutingSameLLInOutATMSystem()
  2486. {
  2487. var atmWaferCount = _lstWaferTasks.Where(wafer => ModuleHelper.IsEFEMRobot(wafer.currentMod) || ModuleHelper.IsAligner(wafer.currentMod)).Count();
  2488. var vacWaferCount = _lstWaferTasks.Where(task => ModuleHelper.IsPm(task.currentMod) || ModuleHelper.IsTMRobot(task.currentMod) || ModuleHelper.IsLoadLock(task.currentMod)).Count();
  2489. var preferWaferVacCount = _dictModuleTask.Where(mod => ModuleHelper.IsPm(mod.Key) && mod.Value.Scheduler.IsOnline).Count() +
  2490. (LLInOutPath == SequenceLLInOutPath.AInAOut ? _LLASlotNumber / 2 : (LLInOutPath == SequenceLLInOutPath.BInBOut ? _LLBSlotNumber / 2 : _LLASlotNumber));
  2491. var inOutIndicator = vacWaferCount - preferWaferVacCount;
  2492. if (inOutIndicator < 0)
  2493. {
  2494. // ready to push in
  2495. var readyLLs = _dictModuleTask.Where(mod => ModuleHelper.IsLoadLock(mod.Key) &&
  2496. ((mod.Key == ModuleName.LLA && LLInOutPath != SequenceLLInOutPath.BInBOut) || (mod.Key == ModuleName.LLB && LLInOutPath != SequenceLLInOutPath.AInAOut)) &&
  2497. mod.Value.TimeToReady <= 10 &&
  2498. !IsLLReservedByTM(mod.Key) &&
  2499. GetLLReadyInOutSlots(mod.Key).emptySlot.Count >= 1).
  2500. OrderBy(mod => mod.Value.TimeToReady).ToList();
  2501. if (atmWaferCount >= 1)
  2502. {
  2503. foreach (var ll in readyLLs)
  2504. {
  2505. if (ExchangeWaferWithLL(ll.Key))
  2506. return;
  2507. }
  2508. }
  2509. // forward wafer to system
  2510. if (atmWaferCount < 2)
  2511. {
  2512. var targetLL = readyLLs.Count > 0 ? readyLLs.First().Key : (LLInOutPath == SequenceLLInOutPath.BInBOut ? ModuleName.LLB : ModuleName.LLA);
  2513. ForwardATMWafers(targetLL);
  2514. return;
  2515. }
  2516. if (_efemSchdActions.Count == 0)
  2517. {
  2518. var readyReturnLL = _dictModuleTask.Where(mod => ModuleHelper.IsLoadLock(mod.Key) &&
  2519. ((mod.Key == ModuleName.LLA && LLInOutPath != SequenceLLInOutPath.BInBOut) || (mod.Key == ModuleName.LLB && LLInOutPath != SequenceLLInOutPath.AInAOut)) &&
  2520. mod.Value.TimeToReady <= 5 &&
  2521. !IsLLReservedByTM(mod.Key) &&
  2522. GetLLReadyInOutSlots(mod.Key).outSlot.Count >= 1).
  2523. OrderByDescending(mod => GetLLReadyInOutSlots(mod.Key).outSlot.Count).ToList();
  2524. foreach (var ll in readyReturnLL)
  2525. {
  2526. if (ExchangeWaferWithLL(readyReturnLL.First().Key))
  2527. return;
  2528. }
  2529. }
  2530. }
  2531. else
  2532. {
  2533. // ready double return LL
  2534. var readyDoubleReturnLL = _dictModuleTask.Where(mod => ModuleHelper.IsLoadLock(mod.Key) &&
  2535. mod.Value.TimeToReady < 10 &&
  2536. !IsLLReservedByTM(mod.Key) &&
  2537. GetLLReadyInOutSlots(mod.Key).outSlot.Count > 1).
  2538. OrderBy(mod => mod.Value.TimeToReady).ToList();
  2539. foreach (var ll in readyDoubleReturnLL)
  2540. {
  2541. if (ExchangeWaferWithLL(ll.Key))
  2542. return;
  2543. }
  2544. // ready single return LL
  2545. var readySingleReturnLL = _dictModuleTask.Where(mod => ModuleHelper.IsLoadLock(mod.Key) &&
  2546. mod.Value.TimeToReady < 2 &&
  2547. !IsLLReservedByTM(mod.Key) &&
  2548. GetLLReadyInOutSlots(mod.Key).outSlot.Count == 1).
  2549. OrderBy(mod => mod.Value.TimeToReady).ToList();
  2550. foreach (var ll in readySingleReturnLL)
  2551. {
  2552. if (ExchangeWaferWithLL(ll.Key))
  2553. return;
  2554. }
  2555. if (atmWaferCount == 0)
  2556. {
  2557. // ready to push in
  2558. var readyLLs = _dictModuleTask.Where(mod => ModuleHelper.IsLoadLock(mod.Key) &&
  2559. ((mod.Key != ModuleName.LLA && LLInOutPath == SequenceLLInOutPath.BInBOut) || (mod.Key != ModuleName.LLB && LLInOutPath == SequenceLLInOutPath.AInAOut)) &&
  2560. mod.Value.TimeToReady <= 10 &&
  2561. !IsLLReservedByTM(mod.Key) &&
  2562. GetLLReadyInOutSlots(mod.Key).emptySlot.Count > 1).
  2563. OrderBy(mod => mod.Value.TimeToReady).ToList();
  2564. foreach (var ll in readyLLs)
  2565. {
  2566. if (ForwardATMWafers(ll.Key))
  2567. return;
  2568. }
  2569. }
  2570. }
  2571. }
  2572. private bool isReturnActionsDone(List<MoveItem> items)
  2573. {
  2574. foreach (var item in items)
  2575. {
  2576. if (WaferManager.Instance.CheckHasWafer(item.SourceModule, item.SourceSlot) || WaferManager.Instance.CheckNoWafer(item.DestinationModule, item.DestinationSlot))
  2577. return false;
  2578. }
  2579. return true;
  2580. }
  2581. private bool IsMovingActionsDone(List<MoveItem> actions)
  2582. {
  2583. bool CheckWaferExistence(ModuleName mod, int slot)
  2584. {
  2585. return ModuleHelper.IsLoadPort(mod) ? WaferManager.Instance.CheckHasWafer(mod, slot) : _lstWaferTasks.Exists(wt => wt.currentMod == mod && wt.currentSlot == slot);
  2586. }
  2587. if (actions.Count == 1)
  2588. {
  2589. if (CheckWaferExistence(actions.First().SourceModule, actions.First().SourceSlot) ||
  2590. !CheckWaferExistence(actions.First().DestinationModule, actions.First().DestinationSlot))
  2591. return false;
  2592. }
  2593. else
  2594. {
  2595. // initialize all the wafer existance before move
  2596. var slotWafers = new Dictionary<KeyValuePair<ModuleName, int>, bool>();
  2597. foreach (var ac in actions)
  2598. {
  2599. var scrSlot = new KeyValuePair<ModuleName, int>(ac.SourceModule, ac.SourceSlot);
  2600. var destSlot = new KeyValuePair<ModuleName, int>(ac.DestinationModule, ac.DestinationSlot);
  2601. if (!slotWafers.ContainsKey(scrSlot))
  2602. {
  2603. slotWafers[scrSlot] = true;
  2604. }
  2605. if (!slotWafers.ContainsKey(destSlot))
  2606. {
  2607. slotWafers[destSlot] = false;
  2608. }
  2609. }
  2610. // simulate moved result
  2611. foreach (var ac in actions)
  2612. {
  2613. var scrSlot = new KeyValuePair<ModuleName, int>(ac.SourceModule, ac.SourceSlot);
  2614. var destSlot = new KeyValuePair<ModuleName, int>(ac.DestinationModule, ac.DestinationSlot);
  2615. if (!slotWafers[scrSlot])
  2616. {
  2617. LOG.Write(eEvent.WARN_ROUTER, ModuleName.System, $"{slotWafers[scrSlot]} do not has a wafer");
  2618. }
  2619. else
  2620. {
  2621. slotWafers[scrSlot] = false;
  2622. }
  2623. if (slotWafers[destSlot])
  2624. {
  2625. LOG.Write(eEvent.WARN_ROUTER, ModuleName.System, $"{slotWafers[destSlot]} has a wafer");
  2626. }
  2627. else
  2628. {
  2629. slotWafers[destSlot] = true;
  2630. }
  2631. }
  2632. foreach (var slot in slotWafers)
  2633. {
  2634. if (slot.Value != CheckWaferExistence(slot.Key.Key, slot.Key.Value))
  2635. return false;
  2636. }
  2637. }
  2638. return true;
  2639. }
  2640. private bool IsLLReservedByTM(ModuleName ll)
  2641. {
  2642. foreach (var item in _curTmAction)
  2643. {
  2644. if (item.DestinationModule == ll || item.SourceModule == ll)
  2645. return true;
  2646. }
  2647. foreach (var action in _tmSchdActions)
  2648. {
  2649. foreach (var item in action)
  2650. {
  2651. if (item.DestinationModule == ll || item.SourceModule == ll)
  2652. return true;
  2653. }
  2654. }
  2655. return false;
  2656. }
  2657. private bool IsLLReservedByEFEM(ModuleName ll)
  2658. {
  2659. foreach (var item in _curEfemAction)
  2660. {
  2661. if (item.DestinationModule == ll || item.SourceModule == ll)
  2662. return true;
  2663. }
  2664. foreach (var action in _efemSchdActions)
  2665. {
  2666. foreach (var item in action)
  2667. {
  2668. if (item.DestinationModule == ll || item.SourceModule == ll)
  2669. return true;
  2670. }
  2671. }
  2672. return false;
  2673. }
  2674. private void RunSchdEFEMActions()
  2675. {
  2676. var efemRobot = Singleton<TransferModule>.Instance.GetScheduler(ModuleName.EfemRobot) as SchedulerEfemRobot;
  2677. if (efemRobot == null || !efemRobot.IsAvailable)
  2678. return;
  2679. if (_dictModuleTask[ModuleName.EfemRobot].IsIdle)
  2680. {
  2681. if (_efemSchdActions.Count > 0)
  2682. {
  2683. if (_curEfemAction.Count == 0 || IsMovingActionsDone(_curEfemAction))
  2684. {
  2685. var nextActions = _efemSchdActions.First();
  2686. if (nextActions.Exists(action => !_lstWaferTasks.Exists(wafer => (wafer.movingStatus == RState.End || wafer.movingStatus == RState.Init) && wafer.currentMod == action.SourceModule && wafer.currentSlot == action.SourceSlot)))
  2687. return;
  2688. var nextModule = nextActions.First().Module;
  2689. if (ModuleHelper.IsLoadLock(nextModule) && !_dictModuleTask[nextModule].Scheduler.IsAvailable)
  2690. return;
  2691. _curEfemAction = _efemSchdActions.Dequeue();
  2692. foreach (var action in _curEfemAction)
  2693. {
  2694. var waferTask = _lstWaferTasks.Find(wafer => (wafer.movingStatus == RState.End || wafer.movingStatus == RState.Init) && wafer.currentMod == action.SourceModule && wafer.currentSlot == action.SourceSlot);
  2695. waferTask.MoveTo(action.DestinationModule, action.DestinationSlot);
  2696. //--2024-03-21 增加了PortJobWaferStart 上报事件 start--
  2697. if (ModuleHelper.IsLoadPort(action.SourceModule))
  2698. {
  2699. ControlJobInfo currentControlJob = GetLoadPortCurrentControlJob(action.SourceModule);
  2700. if (currentControlJob != null)
  2701. {
  2702. _faCallback.JobWaferStart(currentControlJob, action.SourceSlot);
  2703. }
  2704. }
  2705. if (ModuleHelper.IsLoadPort(action.DestinationModule))
  2706. {
  2707. ControlJobInfo currentControlJob = GetLoadPortCurrentControlJob(action.DestinationModule);
  2708. if (currentControlJob != null)
  2709. {
  2710. _faCallback.JobWaferEnd(currentControlJob, action.DestinationSlot);
  2711. }
  2712. }
  2713. //--2024-03-21 增加了PortJobWaferStart 上报事件 end--
  2714. }
  2715. (_dictModuleTask[ModuleName.EfemRobot].Scheduler as SchedulerEfemRobot).PostMoveItems(_curEfemAction.ToArray());
  2716. }
  2717. }
  2718. else if (_curEfemAction.Count >= 0 && IsMovingActionsDone(_curEfemAction)) // all scheduled actions done
  2719. {
  2720. _curEfemAction.Clear();
  2721. }
  2722. }
  2723. }
  2724. private void RunSchdTMActions()
  2725. {
  2726. if (_dictModuleTask[ModuleName.TMRobot].IsIdle)
  2727. {
  2728. if (_tmSchdActions.Count > 0)
  2729. {
  2730. if (_curTmAction.Count == 0 || IsMovingActionsDone(_curTmAction))
  2731. {
  2732. var nextActions = _tmSchdActions.First();
  2733. if (nextActions.Exists(action => !_lstWaferTasks.Exists(wafer => (wafer.movingStatus == RState.End || wafer.movingStatus == RState.Init) && wafer.currentMod == action.SourceModule && wafer.currentSlot == action.SourceSlot)))
  2734. return;
  2735. if (ModuleHelper.IsPm(nextActions.First().Module) && !_dictModuleTask[nextActions.First().Module].IsIdle) /// wait PMTask status update to idle
  2736. return;
  2737. _curTmAction = _tmSchdActions.Dequeue();
  2738. foreach (var action in _curTmAction)
  2739. {
  2740. var waferTask = _lstWaferTasks.Find(wafer => (wafer.movingStatus == RState.End || wafer.movingStatus == RState.Init) && wafer.currentMod == action.SourceModule && wafer.currentSlot == action.SourceSlot);
  2741. waferTask.MoveTo(action.DestinationModule, action.DestinationSlot);
  2742. }
  2743. (_dictModuleTask[ModuleName.TMRobot].Scheduler as SchedulerTMRobot).SendMoveItems(_curTmAction.ToArray());
  2744. }
  2745. }
  2746. else if (_curTmAction.Count >= 0 && IsMovingActionsDone(_curTmAction)) // all scheduled actions done
  2747. {
  2748. _curTmAction.Clear();
  2749. }
  2750. }
  2751. }
  2752. private int TimeForNextModuleReady(WaferTask task)
  2753. {
  2754. if (ModuleHelper.IsPm(task.destMod) && (ModuleHelper.IsLoadLock(task.currentMod) || ModuleHelper.IsTMRobot(task.currentMod)))
  2755. return _dictModuleTask[task.destMod].TimeToReady;
  2756. var llbWaferStatus = GetLLReadyInOutSlots(ModuleName.LLB);
  2757. var llaWaferStatus = GetLLReadyInOutSlots(ModuleName.LLA);
  2758. if (ModuleHelper.IsLoadPort(task.destMod) && (ModuleHelper.IsPm(task.currentMod) || ModuleHelper.IsTMRobot(task.currentMod)))
  2759. {
  2760. if (task.llInOutPath == SequenceLLInOutPath.AInBOut || task.llInOutPath == SequenceLLInOutPath.BInBOut)
  2761. {
  2762. if (llbWaferStatus.emptySlot.Count > 0)
  2763. {
  2764. return _dictModuleTask[ModuleName.LLB].TimeToReady;
  2765. }
  2766. }
  2767. else if (task.llInOutPath == SequenceLLInOutPath.BInAOut || task.llInOutPath == SequenceLLInOutPath.AInAOut)
  2768. {
  2769. if (llaWaferStatus.emptySlot.Count > 0)
  2770. {
  2771. return _dictModuleTask[ModuleName.LLA].TimeToReady;
  2772. }
  2773. }
  2774. else if (task.llInOutPath == SequenceLLInOutPath.DInDOut)
  2775. {
  2776. if (llaWaferStatus.emptySlot.Count > 0 && llbWaferStatus.emptySlot.Count > 0)
  2777. {
  2778. if (_dictModuleTask[ModuleName.LLA].TimeToReady == _dictModuleTask[ModuleName.LLB].TimeToReady)
  2779. {
  2780. return _dictModuleTask[ModuleName.LLA].TimeToReady;
  2781. }
  2782. else
  2783. {
  2784. return Math.Min(_dictModuleTask[ModuleName.LLA].TimeToReady, _dictModuleTask[ModuleName.LLB].TimeToReady);
  2785. }
  2786. }
  2787. else if (llaWaferStatus.emptySlot.Count > 0)
  2788. {
  2789. return _dictModuleTask[ModuleName.LLA].TimeToReady;
  2790. }
  2791. else if (llbWaferStatus.emptySlot.Count > 0)
  2792. {
  2793. return _dictModuleTask[ModuleName.LLB].TimeToReady;
  2794. }
  2795. }
  2796. }
  2797. else if (ModuleHelper.IsPm(task.destMod) && (ModuleHelper.IsLoadLock(task.currentMod) || ModuleHelper.IsTMRobot(task.currentMod)))
  2798. {
  2799. if (_lstWaferTasks.Find(wafer => wafer.currentMod == task.destMod || wafer.routedMod == task.destMod) == null)
  2800. {
  2801. return _dictModuleTask[task.destMod].TimeToReady;
  2802. }
  2803. }
  2804. return int.MaxValue / 2;
  2805. }
  2806. List<Hand> GetTMValidFreeHands(List<WaferTask> robotWafers, List<MoveItem> swapActions)
  2807. {
  2808. var hands = new List<Hand>();
  2809. for (int i = 0; i < 2; i++)
  2810. {
  2811. if (_tmRobotSingleArmOption == 0 || _tmRobotSingleArmOption == i + 1)
  2812. {
  2813. if ((!robotWafers.Exists(wafer => wafer.currentSlot == i) ||
  2814. (robotWafers.Exists(wafer => wafer.currentSlot == i) &&
  2815. swapActions.Exists(item => item.SourceModule == ModuleName.TMRobot && item.SourceSlot == i))) &&
  2816. !swapActions.Exists(item => item.DestinationModule == ModuleName.TMRobot && item.DestinationSlot == i))
  2817. hands.Add((Hand)i);
  2818. }
  2819. }
  2820. return hands;
  2821. }
  2822. (ModuleName llPath, List<MoveItem> swapActions) FindBestLLSwapPlan(ModuleName ll, List<WaferTask> robotWafers)
  2823. {
  2824. ModuleName path = ll == ModuleName.System ? ModuleName.System : ll;
  2825. List<ModuleName> lls = ll == ModuleName.System ? new List<ModuleName> { ModuleName.LLA, ModuleName.LLB } : new List<ModuleName> { ll };
  2826. List<MoveItem> actions = new List<MoveItem>();
  2827. Dictionary<ModuleName, List<MoveItem>> llSwapActions = new Dictionary<ModuleName, List<MoveItem>>();
  2828. foreach (var loadlock in lls)
  2829. {
  2830. llSwapActions[loadlock] = new List<MoveItem>();
  2831. if (IsLLReservedByEFEM(loadlock))
  2832. continue;
  2833. var llWaferStatus = GetLLReadyInOutSlots(loadlock);
  2834. foreach (var wafer in robotWafers)
  2835. {
  2836. if (_dictModuleTask[loadlock].TimeToReady == 0 && !IsLLReservedByEFEM(loadlock) && CanWaferGotoLL(wafer, loadlock) && llWaferStatus.emptySlot.Count > llSwapActions[loadlock].Count)
  2837. {
  2838. llSwapActions[loadlock].Add(new MoveItem(ModuleName.TMRobot, wafer.currentSlot, loadlock, llWaferStatus.emptySlot[llSwapActions[loadlock].Count], (Hand)wafer.currentSlot));
  2839. }
  2840. }
  2841. int prePickTime = 20;
  2842. int prePickCount = 0;
  2843. var hands = GetTMValidFreeHands(robotWafers, llSwapActions[loadlock]);
  2844. foreach (var slot in llWaferStatus.inSlot)
  2845. {
  2846. var wafer = _lstWaferTasks.Find(task => task.currentMod == loadlock && task.currentSlot == slot);
  2847. if (wafer != null && _dictModuleTask[wafer.destMod].TimeToReady <= prePickTime && _dictModuleTask[wafer.destMod].Scheduler.IsOnline && hands.Count > prePickCount)
  2848. {
  2849. if (_lstWaferTasks.Exists(task => task.currentMod == wafer.destMod))
  2850. {
  2851. if (hands.Count == 2 && prePickCount == 0 && (_dictModuleTask.Count(mod => ModuleHelper.IsPm(mod.Key) && mod.Value.TimeToReady < prePickTime && WaferManager.Instance.CheckNoWafer(mod.Key, 0)) == 0 || llWaferStatus.inSlot.Count == 1))
  2852. {
  2853. llSwapActions[loadlock].Add(new MoveItem(loadlock, slot, ModuleName.TMRobot, (int)hands[prePickCount], hands[prePickCount]));
  2854. prePickCount++;
  2855. break;
  2856. }
  2857. }
  2858. else
  2859. {
  2860. llSwapActions[loadlock].Add(new MoveItem(loadlock, slot, ModuleName.TMRobot, (int)hands[prePickCount], hands[prePickCount]));
  2861. prePickCount++;
  2862. }
  2863. }
  2864. }
  2865. }
  2866. var validateActions = llSwapActions.Where(item => item.Value.Count > 0).ToDictionary(k => k.Key, v => v.Value);
  2867. if (validateActions.Count == 1)
  2868. {
  2869. (path, actions) = (validateActions.First().Key, validateActions.First().Value);
  2870. }
  2871. else if (validateActions.Count == 2)
  2872. {
  2873. var sortedActions = validateActions.OrderByDescending(item => item.Value.Count + (_dictModuleTask[item.Key].Scheduler.IsVac ? 2 : 0)).ToDictionary(k => k.Key, v => v.Value);
  2874. (path, actions) = (sortedActions.First().Key, sortedActions.First().Value);
  2875. }
  2876. return (path, actions);
  2877. }
  2878. (List<WaferTask> pmWafers, ModuleName destLL, List<MoveItem> swapAction) FindBestReturnWafersPlan(Hand[] freeHands)
  2879. {
  2880. Dictionary<ModuleName, List<WaferTask>> returnWafers = new Dictionary<ModuleName, List<WaferTask>>();
  2881. Dictionary<ModuleName, List<MoveItem>> llSwapActions = new Dictionary<ModuleName, List<MoveItem>>();
  2882. Dictionary<ModuleName, int> llValues = new Dictionary<ModuleName, int>();
  2883. foreach (var ll in new List<ModuleName> { ModuleName.LLA, ModuleName.LLB })
  2884. {
  2885. var llWaferStatus = GetLLReadyInOutSlots(ll);
  2886. returnWafers[ll] = new List<WaferTask>();
  2887. llSwapActions[ll] = new List<MoveItem>();
  2888. llValues[ll] = 0;
  2889. if (IsLLReservedByEFEM(ll))
  2890. continue;
  2891. var readyReturnWafers = _lstWaferTasks.Where(wafer => ModuleHelper.IsPm(wafer.currentMod) &&
  2892. ModuleHelper.IsLoadPort(wafer.destMod) &&
  2893. wafer.movingStatus == RState.End &&
  2894. _dictModuleTask[wafer.currentMod].TimeToReady <= 2 &&
  2895. CanWaferGotoLL(wafer, ll)).Take(Math.Min(freeHands.Length, llWaferStatus.emptySlot.Count)).ToList();
  2896. int prePickTime = 10 * llSwapActions[ll].Count;
  2897. int handIndex = 0;
  2898. if (_dictModuleTask[ll].TimeToReady <= prePickTime)
  2899. {
  2900. foreach (var wafer in readyReturnWafers)
  2901. {
  2902. if (handIndex < freeHands.Length)
  2903. {
  2904. returnWafers[ll].Add(wafer);
  2905. llSwapActions[ll].Add(new MoveItem(ModuleName.TMRobot, (int)freeHands[handIndex], ll, llWaferStatus.emptySlot[handIndex], freeHands[handIndex]));
  2906. handIndex++;
  2907. }
  2908. }
  2909. int prePickCount = 0;
  2910. handIndex = 0;
  2911. foreach (var slot in llWaferStatus.inSlot)
  2912. {
  2913. var wafer = _lstWaferTasks.Find(task => task.currentMod == ll && task.currentSlot == slot);
  2914. if (wafer != null && _dictModuleTask[wafer.destMod].TimeToReady <= prePickTime && _dictModuleTask[wafer.destMod].Scheduler.IsOnline && freeHands.Length > prePickCount)
  2915. {
  2916. if (_lstWaferTasks.Exists(task => task.currentMod == wafer.destMod))
  2917. {
  2918. if (freeHands.Length == 2 && prePickCount == 0 && (_dictModuleTask.Count(mod => ModuleHelper.IsPm(mod.Key) && mod.Value.TimeToReady <= prePickTime && WaferManager.Instance.CheckNoWafer(mod.Key, 0)) == 0 || llWaferStatus.inSlot.Count == 1))
  2919. {
  2920. llSwapActions[ll].Add(new MoveItem(ll, slot, ModuleName.TMRobot, (int)freeHands[prePickCount], freeHands[prePickCount]));
  2921. prePickCount++;
  2922. break;
  2923. }
  2924. }
  2925. else
  2926. {
  2927. llSwapActions[ll].Add(new MoveItem(ll, slot, ModuleName.TMRobot, (int)freeHands[prePickCount], freeHands[prePickCount]));
  2928. prePickCount++;
  2929. }
  2930. }
  2931. }
  2932. llValues[ll] = returnWafers[ll].Count * 2 + llSwapActions[ll].Count + (_dictModuleTask[ll].Scheduler.IsVac && returnWafers[ll].Count > 0 ? 2 : 0);
  2933. }
  2934. }
  2935. var preferLL = llValues.OrderByDescending(v => v.Value).First().Key;
  2936. return (returnWafers[preferLL], preferLL, llSwapActions[preferLL]);
  2937. }
  2938. ModuleName FindTheBestMovePathWithFixSlot(IEnumerable<KeyValuePair<ModuleName, ModuleTask>> lls, List<WaferTask> pmWafers)
  2939. {
  2940. Dictionary<ModuleName, int> moveValues = new Dictionary<ModuleName, int>();
  2941. foreach (var ll in lls)
  2942. {
  2943. var llWaferStatus = GetLLFixedReadyInOutSlots(ll.Key);
  2944. moveValues[ll.Key] = Math.Min(pmWafers.Where(wt => CanWaferGotoLL(wt, ll.Key)).Count(), GetLLFixedReadyInOutSlots(ll.Key).tInSlot.Count) + llWaferStatus.tOutSlot.Count;
  2945. }
  2946. var preferLL = moveValues.OrderByDescending(v => v.Value).First().Key;
  2947. return moveValues[preferLL] > 0 ? preferLL : ModuleName.System;
  2948. }
  2949. void RoutingTMSwapActions(List<MoveItem> actions)
  2950. {
  2951. foreach (var action in actions)
  2952. {
  2953. if (ModuleHelper.IsTMRobot(action.SourceModule))
  2954. {
  2955. var waferTask = _lstWaferTasks.Find(wafer => wafer.currentMod == ModuleName.TMRobot && wafer.currentSlot == action.SourceSlot);
  2956. waferTask.RouteTo(action.DestinationModule, action.DestinationSlot);
  2957. }
  2958. else
  2959. {
  2960. var waferTask = _lstWaferTasks.Find(wafer => wafer.currentMod == action.SourceModule && wafer.currentSlot == action.SourceSlot);
  2961. waferTask.RouteTo(action.DestinationModule, action.DestinationSlot);
  2962. }
  2963. }
  2964. }
  2965. private bool ManaulReturnPMWafer()
  2966. {
  2967. var mReturnWafer = _qeReturnWafers.Peek();
  2968. if (ModuleHelper.IsPm((ModuleName)mReturnWafer.Station) && _lstWaferTasks.Count(wt => ModuleHelper.IsTMRobot(wt.currentMod)) == 0)
  2969. {
  2970. var hands = GetTMValidFreeHands(_lstWaferTasks.Where(wt => ModuleHelper.IsTMRobot(wt.currentMod)).ToList(), new List<MoveItem>());
  2971. var waferTask = _lstWaferTasks.Find(wt => wt.currentMod == (ModuleName)mReturnWafer.Station && wt.waferId == mReturnWafer.InnerId);
  2972. if (waferTask != null && hands.Count > 0)
  2973. {
  2974. waferTask.Return();
  2975. var outLL = _dictModuleTask.Where(mod => ModuleHelper.IsLoadLock(mod.Key) && CanWaferGotoLL(waferTask, mod.Key) && GetLLReadyInOutSlots(mod.Key).emptySlot.Count > 0 && !IsLLReservedByEFEM(mod.Key));
  2976. if (outLL.Count() > 0)
  2977. {
  2978. var slot = GetLLReadyInOutSlots(outLL.First().Key).emptySlot.First();
  2979. _tmSchdActions.Enqueue(new List<MoveItem> { new MoveItem(waferTask.currentMod, 0, ModuleName.TMRobot, (int)hands.First(), hands.First()) });
  2980. _tmSchdActions.Enqueue(new List<MoveItem> { new MoveItem(ModuleName.TMRobot, (int)hands.First(), outLL.First().Key, slot, hands.First()) });
  2981. _qeReturnWafers.Dequeue();
  2982. return true;
  2983. }
  2984. }
  2985. }
  2986. return false;
  2987. }
  2988. private void RoutingVacWafers()
  2989. {
  2990. if (_tmSchdActions.Count > 0 || _curTmAction.Count > 0)
  2991. {
  2992. RunSchdTMActions();
  2993. return;
  2994. }
  2995. if (_tmRobotStatus == RState.End)
  2996. {
  2997. // handle manual return wafer if exist
  2998. if (_qeReturnWafers.Count > 0)
  2999. {
  3000. if (ManaulReturnPMWafer())
  3001. return;
  3002. }
  3003. if (_LLSlotInOutOption == LLSlotInOutOpt.AllInAllOut)
  3004. {
  3005. if (_LLASlotNumber == 2)
  3006. {
  3007. Routing2SlotVacSystem();
  3008. }
  3009. else if (_LLASlotNumber == 4)
  3010. {
  3011. Routing4SlotVacSystem();
  3012. }
  3013. }
  3014. else
  3015. {
  3016. RoutingFixedSlotVacSystem();
  3017. }
  3018. }
  3019. }
  3020. private bool HasPendingCleanTask()
  3021. {
  3022. foreach (var mod in _dictModuleTask)
  3023. {
  3024. if (ModuleHelper.IsPm(mod.Key) && mod.Value.Scheduler.IsOnline)
  3025. {
  3026. if ((mod.Value as PMTask).HasPendingCleanTask)
  3027. return true;
  3028. }
  3029. }
  3030. return false;
  3031. }
  3032. private void Routing2SlotVacSystem()
  3033. {
  3034. var robotWafers = _lstWaferTasks.Where(wafer => ModuleHelper.IsTMRobot(wafer.currentMod)).OrderBy(wafer => TimeForNextModuleReady(wafer)).ToList();
  3035. var freeHands = GetTMFreeHand();
  3036. if (LLInOutPath == SequenceLLInOutPath.AInBOut || LLInOutPath == SequenceLLInOutPath.BInAOut)
  3037. {
  3038. var inLL = LLInOutPath == SequenceLLInOutPath.AInBOut ? ModuleName.LLA : ModuleName.LLB;
  3039. var outLL = LLInOutPath == SequenceLLInOutPath.AInBOut ? ModuleName.LLB : ModuleName.LLA;
  3040. var inLLWaferStatus = GetLLReadyInOutSlots(inLL);
  3041. var outLLWaferStatus = GetLLReadyInOutSlots(outLL);
  3042. var inLLModule = _dictModuleTask[inLL] as LoadlockTask;
  3043. var outLLModule = _dictModuleTask[outLL] as LoadlockTask;
  3044. if (robotWafers.Count == 0)
  3045. {
  3046. #region special routing pattern
  3047. if (_specialRoutingPattern != SpecialRoutingPattern.Disable)
  3048. {
  3049. if (_specialRoutingPattern == SpecialRoutingPattern.LongTimeRecipe && freeHands.Count == 2 && !HasPendingCleanTask())
  3050. {
  3051. if (_dictModuleTask.Where(mod => ModuleHelper.IsPm(mod.Key) && !mod.Value.HasWafer && mod.Value.Scheduler.IsOnline).Count() == 0 && inLLWaferStatus.inSlot.Count > 0)
  3052. {
  3053. var readyOutPMs = _dictModuleTask.Where(mod => ModuleHelper.IsPm(mod.Key) && mod.Value.HasWafer && mod.Value.TimeToReady < 25).OrderBy(mod => mod.Value.TimeToReady).ToList();
  3054. if (readyOutPMs.Count > 0 && inLLModule.ReayForTMInTime(5) && _lstWaferTasks.Exists(wt => wt.currentMod == inLL && readyOutPMs.Exists(kv => kv.Key == wt.destMod)))
  3055. {
  3056. var llWafer = _lstWaferTasks.Find(wt => wt.currentMod == inLL && wt.destMod == readyOutPMs.First().Key);
  3057. if (llWafer == null)
  3058. {
  3059. llWafer = _lstWaferTasks.Find(wt => wt.currentMod == inLL && readyOutPMs.Exists(pm => pm.Key == wt.destMod && pm.Value.TimeToReady < 5));
  3060. }
  3061. if (llWafer != null && readyOutPMs.Count(x=>x.Key==llWafer.destMod)>0)
  3062. {
  3063. _tmSchdActions.Enqueue(new List<MoveItem> { new MoveItem(inLL, llWafer.currentSlot, ModuleName.TMRobot, (int)freeHands[0], freeHands[0]) });
  3064. var pmSwap = new List<MoveItem>
  3065. { new MoveItem(llWafer.destMod, 0, ModuleName.TMRobot, (int)freeHands[1], freeHands[1]),
  3066. new MoveItem(ModuleName.TMRobot, (int)freeHands[0], llWafer.destMod, 0, freeHands[0])
  3067. };
  3068. _tmSchdActions.Enqueue(pmSwap);
  3069. if (outLLModule.ReayForTMInTime(25) && outLLWaferStatus.emptySlot.Count > 0)
  3070. {
  3071. _tmSchdActions.Enqueue(new List<MoveItem> { new MoveItem(ModuleName.TMRobot, (int)freeHands[1], outLL, outLLWaferStatus.emptySlot[0], freeHands[1]) });
  3072. }
  3073. }
  3074. }
  3075. return;
  3076. }
  3077. }
  3078. }
  3079. #endregion
  3080. if (_dictModuleTask.Count(mod => ModuleHelper.IsPm(mod.Key) && mod.Value.HasWafer && mod.Value.TimeToReady < 5) > 0 && outLLModule.ReayForTMInTime(30) && outLLWaferStatus.emptySlot.Count > 0)
  3081. {
  3082. var readyOutPMs = _dictModuleTask.Where(mod => ModuleHelper.IsPm(mod.Key) && mod.Value.HasWafer && mod.Value.TimeToReady < 25).OrderBy(mod => mod.Value.TimeToReady).ToList();
  3083. var placeActions = new List<MoveItem>();
  3084. int pickCount = 0;
  3085. foreach (var pm in readyOutPMs)
  3086. {
  3087. var pmWafer = _lstWaferTasks.Find(wt => wt.currentMod == pm.Key);
  3088. if (pmWafer != null && pickCount < freeHands.Count && pickCount < outLLWaferStatus.emptySlot.Count)
  3089. {
  3090. //_tmSchdActions.Enqueue(new List<MoveItem> { new MoveItem(0, 0, pm.Key, 0, Hand.Both) });
  3091. _tmSchdActions.Enqueue(new List<MoveItem> { new MoveItem(pm.Key, 0, ModuleName.TMRobot, (int)freeHands[pickCount], freeHands[pickCount]) });
  3092. placeActions.Add(new MoveItem(ModuleName.TMRobot, (int)freeHands[pickCount], outLL, outLLWaferStatus.emptySlot[pickCount], freeHands[pickCount]));
  3093. pickCount++;
  3094. //LOG.Write(eEvent.INFO_TROUGHPUT_ENHANCE_TRACE, pm.Key, $"{pm.Key} will be ready in {pm.Value.TimeToReady} seconds");
  3095. }
  3096. }
  3097. if (pickCount > 0)
  3098. {
  3099. //LOG.Write(eEvent.INFO_TROUGHPUT_ENHANCE_TRACE, outLL, $"{outLL} will be ready in {outLLModule.TimeToReady} seconds");
  3100. _tmSchdActions.Enqueue(placeActions);
  3101. return;
  3102. }
  3103. }
  3104. // push wafer from LL to PM and avoid blocking two robot arms
  3105. if (inLLModule.ReayForTMInTime(10) && inLLWaferStatus.inSlot.Count > 0)
  3106. {
  3107. var pickActions = new List<MoveItem>();
  3108. var placeActions = new List<MoveItem>();
  3109. var inLLWafers = _lstWaferTasks.FindAll(wt => wt.currentMod == inLL).ToList();
  3110. int pickCount = 0;
  3111. foreach (var inWafer in inLLWafers)
  3112. {
  3113. if (!_dictModuleTask[inWafer.destMod].HasWafer && _dictModuleTask[inWafer.destMod].TimeToReady < 20 && pickCount < freeHands.Count)
  3114. {
  3115. pickActions.Add(new MoveItem(inWafer.currentMod, inWafer.currentSlot, ModuleName.TMRobot, (int)freeHands[pickCount], freeHands[pickCount]));
  3116. placeActions.Add(new MoveItem(ModuleName.TMRobot, (int)freeHands[pickCount], inWafer.destMod, 0, freeHands[pickCount]));
  3117. pickCount++;
  3118. }
  3119. }
  3120. if (pickCount == 2)
  3121. {
  3122. _tmSchdActions.Enqueue(pickActions);
  3123. foreach (var ac in placeActions)
  3124. {
  3125. _tmSchdActions.Enqueue(new List<MoveItem> { ac });
  3126. }
  3127. }
  3128. else if (pickCount == 1)
  3129. {
  3130. if (inLLWafers.Count == 2 && freeHands.Count == 2)
  3131. {
  3132. var peerSlot = 1 - pickActions.First().SourceSlot;
  3133. var peerWafer = inLLWafers.Find(wt => wt.currentSlot == peerSlot);
  3134. if (peerWafer != null && _dictModuleTask[peerWafer.destMod].TimeToReady < 60)
  3135. {
  3136. pickActions.Add(new MoveItem(peerWafer.currentMod, peerWafer.currentSlot, ModuleName.TMRobot, (int)freeHands[pickCount], freeHands[pickCount]));
  3137. }
  3138. }
  3139. _tmSchdActions.Enqueue(pickActions);
  3140. _tmSchdActions.Enqueue(placeActions);
  3141. }
  3142. else
  3143. {
  3144. if (inLLWafers.Count == 1 && _dictModuleTask[inLLWafers.First().destMod].TimeToReady < 60 && freeHands.Count == 2)
  3145. {
  3146. _tmSchdActions.Enqueue(new List<MoveItem> { new MoveItem(inLL, inLLWafers.First().currentSlot, ModuleName.TMRobot, (int)freeHands[0], freeHands[0]) });
  3147. }
  3148. }
  3149. }
  3150. }
  3151. else
  3152. {
  3153. #region special routing pattern
  3154. if (_specialRoutingPattern != SpecialRoutingPattern.Disable)
  3155. {
  3156. if (_specialRoutingPattern == SpecialRoutingPattern.LongTimeRecipe && _tmRobotSingleArmOption == 0 && !HasPendingCleanTask())
  3157. {
  3158. if (_dictModuleTask.Where(mod => ModuleHelper.IsPm(mod.Key) && !mod.Value.HasWafer && mod.Value.Scheduler.IsOnline).Count() == 0 && ModuleHelper.IsLoadPort(robotWafers[0].destMod))
  3159. {
  3160. if (outLLModule.ReayForTMInTime(10) && outLLWaferStatus.emptySlot.Count > 0)
  3161. {
  3162. _tmSchdActions.Enqueue(new List<MoveItem> { new MoveItem(ModuleName.TMRobot, robotWafers[0].currentSlot, outLL, outLLWaferStatus.emptySlot[0], (Hand)(robotWafers[0].currentSlot)) });
  3163. }
  3164. return;
  3165. }
  3166. }
  3167. }
  3168. #endregion
  3169. foreach (var robotWafer in robotWafers)
  3170. {
  3171. if (ModuleHelper.IsPm(robotWafer.destMod))
  3172. {
  3173. if (_dictModuleTask[robotWafer.destMod].TimeToReady < 5)
  3174. {
  3175. var pmWafer = _lstWaferTasks.Find(wt => wt.currentMod == robotWafer.destMod);
  3176. if (_dictModuleTask[robotWafer.destMod].HasWafer && pmWafer != null)
  3177. {
  3178. // PM Swap
  3179. if (freeHands.Count > 0)
  3180. {
  3181. if (string.IsNullOrWhiteSpace(pmWafer.wtwCleanRecipe) && !(_dictModuleTask[pmWafer.currentMod] as PMTask).HasPendingCleanTask)
  3182. {
  3183. var pmActions = new List<MoveItem>
  3184. {
  3185. new MoveItem(robotWafer.destMod, 0, ModuleName.TMRobot, (int)freeHands[0], freeHands[0]),
  3186. new MoveItem(ModuleName.TMRobot, robotWafer.currentSlot, robotWafer.destMod, 0, (Hand)robotWafer.currentSlot)
  3187. };
  3188. _tmSchdActions.Enqueue(pmActions);
  3189. //LOG.Write(eEvent.INFO_TROUGHPUT_ENHANCE_TRACE, robotWafer.destMod, $"{robotWafer.destMod} will be ready in {_dictModuleTask[robotWafer.destMod].TimeToReady} seconds");
  3190. return;
  3191. }
  3192. else
  3193. {
  3194. _tmSchdActions.Enqueue(new List<MoveItem> { new MoveItem(robotWafer.destMod, 0, ModuleName.TMRobot, (int)freeHands[0], freeHands[0]) });
  3195. return;
  3196. }
  3197. }
  3198. }
  3199. else
  3200. {
  3201. //LOG.Write(eEvent.INFO_TROUGHPUT_ENHANCE_TRACE, robotWafer.destMod, $"{robotWafer.destMod} will be ready in {_dictModuleTask[robotWafer.destMod].TimeToReady} seconds");
  3202. _tmSchdActions.Enqueue(new List<MoveItem> { new MoveItem(ModuleName.TMRobot, robotWafer.currentSlot, robotWafer.destMod, 0, (Hand)robotWafer.currentSlot) });
  3203. return;
  3204. }
  3205. }
  3206. else if (_dictModuleTask[robotWafer.destMod].TimeToReady > 20 && freeHands.Count > 0)
  3207. {
  3208. if (outLLWaferStatus.emptySlot.Count > 0 && outLLModule.ReayForTMInTime(10) && _lstWaferTasks.Count(wt => ModuleHelper.IsPm(wt.currentMod) && ModuleHelper.IsLoadPort(wt.destMod)) > 0)
  3209. {
  3210. // try to return one wafer from PM to LL
  3211. var readyOutWafer = _lstWaferTasks.Find(wt => ModuleHelper.IsPm(wt.currentMod) && ModuleHelper.IsLoadPort(wt.destMod));
  3212. if (readyOutWafer != null && _dictModuleTask[readyOutWafer.currentMod].TimeToReady < 5)
  3213. {
  3214. //LOG.Write(eEvent.INFO_TROUGHPUT_ENHANCE_TRACE, readyOutWafer.currentMod, $"{readyOutWafer.currentMod} will be ready in {_dictModuleTask[readyOutWafer.currentMod].TimeToReady} seconds");
  3215. //LOG.Write(eEvent.INFO_TROUGHPUT_ENHANCE_TRACE, outLL, $"{outLL} will be ready in {outLLModule.TimeToReady} seconds");
  3216. _tmSchdActions.Enqueue(new List<MoveItem> { new MoveItem(readyOutWafer.currentMod, 0, ModuleName.TMRobot, (int)freeHands[0], freeHands[0]) });
  3217. _tmSchdActions.Enqueue(new List<MoveItem> { new MoveItem(ModuleName.TMRobot, (int)freeHands[0], outLL, outLLWaferStatus.emptySlot[0], freeHands[0]) });
  3218. return;
  3219. }
  3220. }
  3221. if (inLLModule.ReayForTMInTime(5) && inLLWaferStatus.inSlot.Count > 0)
  3222. {
  3223. var inLLWafers = _lstWaferTasks.FindAll(wt => wt.currentMod == inLL).OrderBy(wt => _dictModuleTask[wt.destMod].TimeToReady);
  3224. foreach (var inWafer in inLLWafers)
  3225. {
  3226. if (!_dictModuleTask[inWafer.destMod].HasWafer && _dictModuleTask[inWafer.destMod].TimeToReady < 10)
  3227. {
  3228. //LOG.Write(eEvent.INFO_TROUGHPUT_ENHANCE_TRACE, inWafer.destMod, $"{inWafer.destMod} will be ready in {_dictModuleTask[inWafer.destMod].TimeToReady} seconds");
  3229. //LOG.Write(eEvent.INFO_TROUGHPUT_ENHANCE_TRACE, inLL, $"{inLL} will be ready in {inLLModule.TimeToReady} seconds");
  3230. _tmSchdActions.Enqueue(new List<MoveItem> { new MoveItem(inLL, inWafer.currentSlot, ModuleName.TMRobot, (int)freeHands[0], freeHands[0]) });
  3231. _tmSchdActions.Enqueue(new List<MoveItem> { new MoveItem(ModuleName.TMRobot, (int)freeHands[0], inWafer.destMod, 0, freeHands[0]) });
  3232. return;
  3233. }
  3234. }
  3235. }
  3236. }
  3237. }
  3238. else
  3239. {
  3240. if (freeHands.Count > 0 &&
  3241. outLLWaferStatus.emptySlot.Count == 2 &&
  3242. outLLModule.ReayForTMInTime(10) &&
  3243. _lstWaferTasks.Count(wt => ModuleHelper.IsPm(wt.currentMod) && ModuleHelper.IsLoadPort(wt.destMod)) > 0)
  3244. {
  3245. // try to get one wafer from PM, and match a double place next time
  3246. var readyOutWafer = _lstWaferTasks.Find(wt => ModuleHelper.IsPm(wt.currentMod) && ModuleHelper.IsLoadPort(wt.destMod));
  3247. if (readyOutWafer != null && _dictModuleTask[readyOutWafer.currentMod].TimeToReady < 5)
  3248. {
  3249. _tmSchdActions.Enqueue(new List<MoveItem> { new MoveItem(readyOutWafer.currentMod, 0, ModuleName.TMRobot, (int)freeHands[0], freeHands[0]) });
  3250. return;
  3251. }
  3252. }
  3253. if (outLLModule.ReayForTMInTime(5))
  3254. {
  3255. // try match a double place
  3256. if (robotWafers.Count == 2 && robotWafers.All(wt => ModuleHelper.IsLoadPort(wt.destMod)) && outLLWaferStatus.emptySlot.Count == 2)
  3257. {
  3258. var doublePlace = new List<MoveItem>
  3259. {
  3260. new MoveItem(ModuleName.TMRobot, 0, outLL, 0, Hand.Blade1),
  3261. new MoveItem(ModuleName.TMRobot, 1, outLL, 1, Hand.Blade2)
  3262. };
  3263. //LOG.Write(eEvent.INFO_TROUGHPUT_ENHANCE_TRACE, outLL, $"{outLL} will be ready in {outLLModule.TimeToReady} seconds");
  3264. _tmSchdActions.Enqueue(doublePlace);
  3265. return;
  3266. }
  3267. if (outLLWaferStatus.emptySlot.Count > 0)
  3268. {
  3269. //LOG.Write(eEvent.INFO_TROUGHPUT_ENHANCE_TRACE, outLL, $"{outLL} will be ready in {outLLModule.TimeToReady} seconds");
  3270. _tmSchdActions.Enqueue(new List<MoveItem> { new MoveItem(ModuleName.TMRobot, robotWafer.currentSlot, outLL, outLLWaferStatus.emptySlot[0], (Hand)robotWafer.currentSlot) });
  3271. return;
  3272. }
  3273. }
  3274. }
  3275. }
  3276. if (freeHands.Count == 0)
  3277. return;
  3278. // try to push one wafer in pm
  3279. if (inLLWaferStatus.inSlot.Count == 1 && inLLModule.ReayForTMInTime(5))
  3280. {
  3281. var inWafer = _lstWaferTasks.Find(wt => wt.currentMod == inLL && wt.currentSlot == inLLWaferStatus.inSlot.First());
  3282. if (inWafer != null && !_dictModuleTask[inWafer.destMod].HasWafer && _dictModuleTask[inWafer.destMod].TimeToReady < 20)
  3283. {
  3284. //LOG.Write(eEvent.INFO_TROUGHPUT_ENHANCE_TRACE, inWafer.destMod, $"{inWafer.destMod} will be ready in {_dictModuleTask[inWafer.destMod].TimeToReady} seconds");
  3285. //LOG.Write(eEvent.INFO_TROUGHPUT_ENHANCE_TRACE, inLL, $"{inLL} will be ready in {inLLModule.TimeToReady} seconds");
  3286. _tmSchdActions.Enqueue(new List<MoveItem> { new MoveItem(inLL, inLLWaferStatus.inSlot.First(), ModuleName.TMRobot, (int)freeHands[0], freeHands[0]) });
  3287. _tmSchdActions.Enqueue(new List<MoveItem> { new MoveItem(ModuleName.TMRobot, (int)freeHands[0], inWafer.destMod, 0, freeHands[0]) });
  3288. return;
  3289. }
  3290. }
  3291. // try to return one wafer
  3292. if (outLLModule.ReayForTMInTime(15) && outLLWaferStatus.emptySlot.Count > 0)
  3293. {
  3294. var readyReturnPMs = _dictModuleTask.Where(mod => ModuleHelper.IsPm(mod.Key) && mod.Value.HasWafer && mod.Value.TimeToReady < 5).OrderBy(mod => mod.Value.TimeToReady).ToList();
  3295. if (readyReturnPMs.Count > 0)
  3296. {
  3297. //LOG.Write(eEvent.INFO_TROUGHPUT_ENHANCE_TRACE, readyReturnPMs.First().Key, $"{readyReturnPMs.First().Key} will be ready in {readyReturnPMs.First().Value.TimeToReady} seconds");
  3298. //LOG.Write(eEvent.INFO_TROUGHPUT_ENHANCE_TRACE, outLL, $"{outLL} will be ready in {outLLModule.TimeToReady} seconds");
  3299. _tmSchdActions.Enqueue(new List<MoveItem> { new MoveItem(readyReturnPMs.First().Key, 0, ModuleName.TMRobot, (int)freeHands[0], freeHands[0]) });
  3300. _tmSchdActions.Enqueue(new List<MoveItem> { new MoveItem(ModuleName.TMRobot, (int)freeHands[0], outLL, outLLWaferStatus.emptySlot.First(), freeHands[0]) });
  3301. return;
  3302. }
  3303. }
  3304. }
  3305. }
  3306. else
  3307. {
  3308. if (robotWafers.Count == 2)
  3309. {
  3310. foreach (var wafer in robotWafers)
  3311. {
  3312. if (wafer.movingStatus == RState.End &&
  3313. ModuleHelper.IsPm(wafer.destMod) &&
  3314. _dictModuleTask[wafer.destMod].TimeToReady == 0 &&
  3315. !_lstWaferTasks.Exists(waferT => waferT.currentMod == wafer.destMod) &&
  3316. !_lstWaferTasks.Exists(waferT => waferT.routedMod == wafer.destMod))
  3317. {
  3318. wafer.RouteTo(wafer.destMod, 0);
  3319. _tmSchdActions.Enqueue(new List<MoveItem> { new MoveItem(ModuleName.TMRobot, wafer.currentSlot, wafer.destMod, 0, (Hand)wafer.currentSlot) });
  3320. }
  3321. }
  3322. if (_tmSchdActions.Count == 0)
  3323. {
  3324. var swapActions = FindBestLLSwapPlan(ModuleName.System, robotWafers);
  3325. if (swapActions.llPath != ModuleName.System && swapActions.swapActions.Count > 0)
  3326. {
  3327. RoutingTMSwapActions(swapActions.swapActions);
  3328. _tmSchdActions.Enqueue(swapActions.swapActions);
  3329. }
  3330. }
  3331. }
  3332. else if (robotWafers.Count == 1)
  3333. {
  3334. if (ModuleHelper.IsPm(robotWafers[0].destMod) && _dictModuleTask[robotWafers[0].destMod].TimeToReady == 0)
  3335. {
  3336. var pmActions = new List<MoveItem>();
  3337. var pmWafer = _lstWaferTasks.Find(wafer => (wafer.movingStatus == RState.End || wafer.movingStatus == RState.Init) && wafer.currentMod == robotWafers[0].destMod && ModuleHelper.IsLoadPort(wafer.destMod));
  3338. if (pmWafer != null)
  3339. {
  3340. int pickSlot = 1 - robotWafers[0].currentSlot;
  3341. pmWafer.RouteTo(ModuleName.TMRobot, pickSlot);
  3342. pmActions.Add(new MoveItem(pmWafer.currentMod, 0, ModuleName.TMRobot, pickSlot, (Hand)pickSlot));
  3343. if (string.IsNullOrWhiteSpace(pmWafer.wtwCleanRecipe) && !(_dictModuleTask[robotWafers[0].destMod] as PMTask).HasPendingCleanTask)
  3344. {
  3345. pmActions.Add(new MoveItem(ModuleName.TMRobot, robotWafers[0].currentSlot, pmWafer.currentMod, 0, (Hand)robotWafers[0].currentSlot));
  3346. }
  3347. _tmSchdActions.Enqueue(pmActions);
  3348. return;
  3349. }
  3350. if (!_lstWaferTasks.Exists(wafer => wafer.currentMod == robotWafers[0].destMod))
  3351. {
  3352. robotWafers[0].RouteTo(robotWafers[0].destMod, 0);
  3353. pmActions.Add(new MoveItem(ModuleName.TMRobot, robotWafers[0].currentSlot, robotWafers[0].destMod, 0, (Hand)robotWafers[0].currentSlot));
  3354. _tmSchdActions.Enqueue(pmActions);
  3355. return;
  3356. }
  3357. }
  3358. // try to pick a wafer from LL
  3359. if (_tmSchdActions.Count == 0)
  3360. {
  3361. var swapActions = FindBestLLSwapPlan(ModuleName.System, robotWafers);
  3362. if (swapActions.llPath != ModuleName.System && swapActions.swapActions.Count > 0)
  3363. {
  3364. RoutingTMSwapActions(swapActions.swapActions);
  3365. _tmSchdActions.Enqueue(swapActions.swapActions);
  3366. return;
  3367. }
  3368. }
  3369. // try to return a wafer from PM
  3370. var freeHand = GetTMValidFreeHands(robotWafers, new List<MoveItem>());
  3371. var moveActions = FindBestReturnWafersPlan(freeHand.ToArray());
  3372. if (moveActions.pmWafers.Count > 0)
  3373. {
  3374. MoveItem placeAction = moveActions.swapAction.Find(ac => ModuleHelper.IsLoadLock(ac.DestinationModule));
  3375. moveActions.pmWafers.First().RouteTo(placeAction.DestinationModule, placeAction.DestinationSlot);
  3376. _tmSchdActions.Enqueue(new List<MoveItem> { new MoveItem(moveActions.pmWafers.First().currentMod, 0, ModuleName.TMRobot, (int)freeHand[0], freeHand[0]) });
  3377. MoveItem pickAction = moveActions.swapAction.Find(ac => ModuleHelper.IsLoadLock(ac.SourceModule));
  3378. if (pickAction != null)
  3379. {
  3380. var pickWafer = _lstWaferTasks.Find(wafer => wafer.currentMod == pickAction.SourceModule && wafer.currentSlot == pickAction.SourceSlot);
  3381. pickWafer.RouteTo(pickAction.DestinationModule, pickAction.DestinationSlot);
  3382. }
  3383. _tmSchdActions.Enqueue(moveActions.swapAction);
  3384. }
  3385. }
  3386. else // robot arm is empty
  3387. {
  3388. // try return wafers from PM
  3389. var freeHand = GetTMValidFreeHands(robotWafers, new List<MoveItem>());
  3390. var moveActions = FindBestReturnWafersPlan(freeHand.ToArray());
  3391. if (moveActions.pmWafers.Count > 0)
  3392. {
  3393. int handIndex = 0;
  3394. var placeActions = moveActions.swapAction.Where(ac => ModuleHelper.IsLoadLock(ac.DestinationModule));
  3395. foreach (var action in placeActions)
  3396. {
  3397. moveActions.pmWafers[handIndex].RouteTo(action.DestinationModule, action.DestinationSlot);
  3398. _tmSchdActions.Enqueue(new List<MoveItem> { new MoveItem(moveActions.pmWafers[handIndex].currentMod, 0, ModuleName.TMRobot, (int)freeHand[handIndex], freeHand[handIndex]) });
  3399. handIndex++;
  3400. }
  3401. var pickActions = moveActions.swapAction.Where(ac => ModuleHelper.IsLoadLock(ac.SourceModule)).ToList();
  3402. foreach (var ac in pickActions)
  3403. {
  3404. var llWafer = _lstWaferTasks.Find(wafer => wafer.currentMod == ac.SourceModule && wafer.currentSlot == ac.SourceSlot);
  3405. llWafer.RouteTo(ac.DestinationModule, ac.DestinationSlot);
  3406. }
  3407. _tmSchdActions.Enqueue(moveActions.swapAction);
  3408. }
  3409. if (_tmSchdActions.Count == 0)
  3410. {
  3411. var swapActions = FindBestLLSwapPlan(ModuleName.System, robotWafers);
  3412. if (swapActions.llPath != ModuleName.System && swapActions.swapActions.Count > 0)
  3413. {
  3414. RoutingTMSwapActions(swapActions.swapActions);
  3415. _tmSchdActions.Enqueue(swapActions.swapActions);
  3416. }
  3417. }
  3418. }
  3419. }
  3420. }
  3421. private void Routing4SlotVacSystem()
  3422. {
  3423. var robotWafers = _lstWaferTasks.Where(wafer => (wafer.movingStatus == RState.End || wafer.movingStatus == RState.Init) && ModuleHelper.IsTMRobot(wafer.currentMod)).OrderBy(wafer => TimeForNextModuleReady(wafer)).ToList();
  3424. if (robotWafers.Count == 2)
  3425. {
  3426. foreach (var wafer in robotWafers)
  3427. {
  3428. if (wafer.movingStatus == RState.End &&
  3429. ModuleHelper.IsPm(wafer.destMod) &&
  3430. _dictModuleTask[wafer.destMod].TimeToReady == 0 &&
  3431. !_lstWaferTasks.Exists(waferT => waferT.currentMod == wafer.destMod) &&
  3432. !_lstWaferTasks.Exists(waferT => waferT.routedMod == wafer.destMod))
  3433. {
  3434. wafer.RouteTo(wafer.destMod, 0);
  3435. _tmSchdActions.Enqueue(new List<MoveItem> { new MoveItem(ModuleName.TMRobot, wafer.currentSlot, wafer.destMod, 0, (Hand)wafer.currentSlot) });
  3436. }
  3437. }
  3438. if (_tmSchdActions.Count == 0)
  3439. {
  3440. var swapActions = FindBestLLSwapPlan(ModuleName.System, robotWafers);
  3441. if (swapActions.llPath != ModuleName.System && swapActions.swapActions.Count > 0)
  3442. {
  3443. RoutingTMSwapActions(swapActions.swapActions);
  3444. _tmSchdActions.Enqueue(swapActions.swapActions);
  3445. }
  3446. }
  3447. }
  3448. else if (robotWafers.Count == 1)
  3449. {
  3450. if (ModuleHelper.IsPm(robotWafers[0].destMod) && _dictModuleTask[robotWafers[0].destMod].TimeToReady == 0)
  3451. {
  3452. var pmActions = new List<MoveItem>();
  3453. var pmWafer = _lstWaferTasks.Find(wafer => (wafer.movingStatus == RState.End || wafer.movingStatus == RState.Init) && wafer.currentMod == robotWafers[0].destMod && ModuleHelper.IsLoadPort(wafer.destMod));
  3454. if (pmWafer != null && string.IsNullOrWhiteSpace(pmWafer.wtwCleanRecipe))
  3455. {
  3456. int pickSlot = 1 - robotWafers[0].currentSlot;
  3457. pmWafer.RouteTo(ModuleName.TMRobot, pickSlot);
  3458. pmActions.Add(new MoveItem(pmWafer.currentMod, 0, ModuleName.TMRobot, pickSlot, (Hand)pickSlot));
  3459. pmActions.Add(new MoveItem(ModuleName.TMRobot, robotWafers[0].currentSlot, pmWafer.currentMod, 0, (Hand)robotWafers[0].currentSlot));
  3460. _tmSchdActions.Enqueue(pmActions);
  3461. return;
  3462. }
  3463. if (!_lstWaferTasks.Exists(wafer => wafer.currentMod == robotWafers[0].destMod))
  3464. {
  3465. robotWafers[0].RouteTo(robotWafers[0].destMod, 0);
  3466. pmActions.Add(new MoveItem(ModuleName.TMRobot, robotWafers[0].currentSlot, robotWafers[0].destMod, 0, (Hand)robotWafers[0].currentSlot));
  3467. _tmSchdActions.Enqueue(pmActions);
  3468. return;
  3469. }
  3470. }
  3471. // try to pick a wafer from LL
  3472. if (_tmSchdActions.Count == 0)
  3473. {
  3474. var swapActions = FindBestLLSwapPlan(ModuleName.System, robotWafers);
  3475. if (swapActions.llPath != ModuleName.System && swapActions.swapActions.Count > 0)
  3476. {
  3477. RoutingTMSwapActions(swapActions.swapActions);
  3478. _tmSchdActions.Enqueue(swapActions.swapActions);
  3479. return;
  3480. }
  3481. }
  3482. // try to return a wafer from PM
  3483. var freeHand = GetTMValidFreeHands(robotWafers, new List<MoveItem>());
  3484. var moveActions = FindBestReturnWafersPlan(freeHand.ToArray());
  3485. if (moveActions.pmWafers.Count > 0)
  3486. {
  3487. MoveItem placeAction = moveActions.swapAction.Find(ac => ModuleHelper.IsLoadLock(ac.DestinationModule));
  3488. moveActions.pmWafers.First().RouteTo(placeAction.DestinationModule, placeAction.DestinationSlot);
  3489. _tmSchdActions.Enqueue(new List<MoveItem> { new MoveItem(moveActions.pmWafers.First().currentMod, 0, ModuleName.TMRobot, (int)freeHand[0], freeHand[0]) });
  3490. MoveItem pickAction = moveActions.swapAction.Find(ac => ModuleHelper.IsLoadLock(ac.SourceModule));
  3491. if (pickAction != null)
  3492. {
  3493. var pickWafer = _lstWaferTasks.Find(wafer => wafer.currentMod == pickAction.SourceModule && wafer.currentSlot == pickAction.SourceSlot);
  3494. pickWafer.RouteTo(pickAction.DestinationModule, pickAction.DestinationSlot);
  3495. }
  3496. _tmSchdActions.Enqueue(moveActions.swapAction);
  3497. }
  3498. }
  3499. else // robot arm is empty
  3500. {
  3501. // try return wafers from PM
  3502. var freeHand = GetTMValidFreeHands(robotWafers, new List<MoveItem>());
  3503. var moveActions = FindBestReturnWafersPlan(freeHand.ToArray());
  3504. if (moveActions.pmWafers.Count > 0)
  3505. {
  3506. int handIndex = 0;
  3507. var placeActions = moveActions.swapAction.Where(ac => ModuleHelper.IsLoadLock(ac.DestinationModule));
  3508. foreach (var action in placeActions)
  3509. {
  3510. moveActions.pmWafers[handIndex].RouteTo(action.DestinationModule, action.DestinationSlot);
  3511. _tmSchdActions.Enqueue(new List<MoveItem> { new MoveItem(moveActions.pmWafers[handIndex].currentMod, 0, ModuleName.TMRobot, (int)freeHand[handIndex], freeHand[handIndex]) });
  3512. handIndex++;
  3513. }
  3514. var pickActions = moveActions.swapAction.Where(ac => ModuleHelper.IsLoadLock(ac.SourceModule)).ToList();
  3515. foreach (var ac in pickActions)
  3516. {
  3517. var llWafer = _lstWaferTasks.Find(wafer => wafer.currentMod == ac.SourceModule && wafer.currentSlot == ac.SourceSlot);
  3518. llWafer.RouteTo(ac.DestinationModule, ac.DestinationSlot);
  3519. }
  3520. _tmSchdActions.Enqueue(moveActions.swapAction);
  3521. }
  3522. if (_tmSchdActions.Count == 0)
  3523. {
  3524. var swapActions = FindBestLLSwapPlan(ModuleName.System, robotWafers);
  3525. if (swapActions.llPath != ModuleName.System && swapActions.swapActions.Count > 0)
  3526. {
  3527. RoutingTMSwapActions(swapActions.swapActions);
  3528. _tmSchdActions.Enqueue(swapActions.swapActions);
  3529. }
  3530. }
  3531. }
  3532. }
  3533. private void RoutingFixedSlotVacSystem()
  3534. {
  3535. var robotWafers = _lstWaferTasks.Where(wt => ModuleHelper.IsTMRobot(wt.currentMod));
  3536. var lls = _dictModuleTask.Where(mod => ModuleHelper.IsLoadLock(mod.Key) && mod.Value.Scheduler.IsOnline);
  3537. if (_LLASlotNumber == 2)
  3538. {
  3539. if (robotWafers.Count() == 1)
  3540. {
  3541. if (ModuleHelper.IsPm(robotWafers.First().destMod))
  3542. {
  3543. var destPM = robotWafers.First().destMod;
  3544. if (_dictModuleTask[destPM].TimeToReady <= 5)
  3545. {
  3546. if (_lstWaferTasks.Exists(wt => wt.currentMod == destPM))
  3547. {
  3548. var pickHand = SelectTMPickArm(destPM);
  3549. if (pickHand != Hand.None)
  3550. {
  3551. var pmSwap = new List<MoveItem>
  3552. {
  3553. new MoveItem(destPM, 0, ModuleName.TMRobot, (int)pickHand, pickHand),
  3554. new MoveItem(ModuleName.TMRobot, robotWafers.First().currentSlot, destPM, 0, (Hand)robotWafers.First().currentSlot)
  3555. };
  3556. _tmSchdActions.Enqueue(pmSwap);
  3557. return;
  3558. }
  3559. }
  3560. else
  3561. {
  3562. _tmSchdActions.Enqueue(new List<MoveItem> { new MoveItem(ModuleName.TMRobot, robotWafers.First().currentSlot, destPM, 0, (Hand)robotWafers.First().currentSlot) });
  3563. return;
  3564. }
  3565. }
  3566. }
  3567. else
  3568. {
  3569. var readySwapLL = lls.Where(ll => CanWaferGotoLL(robotWafers.First(), ll.Key) &&
  3570. GetLLFixedReadyInOutSlots(ll.Key).tOutSlot.Count == 1 &&
  3571. GetLLFixedReadyInOutSlots(ll.Key).tInSlot.Count == 1 &&
  3572. ll.Value.TimeToReady <= 10);
  3573. if (readySwapLL.Count() > 0)
  3574. {
  3575. var destLL = readySwapLL.First().Key;
  3576. Hand pickHand = SelectTMPickArm(destLL);
  3577. if (pickHand != Hand.None)
  3578. {
  3579. Hand placeHand = (Hand)robotWafers.First().currentSlot;
  3580. var llSwap = new List<MoveItem>();
  3581. var llSlots = GetLLFixedReadyInOutSlots(destLL);
  3582. llSwap.Add(new MoveItem(destLL, llSlots.tOutSlot.First(), ModuleName.TMRobot, (int)pickHand, pickHand));
  3583. llSwap.Add(new MoveItem(ModuleName.TMRobot, (int)placeHand, destLL, llSlots.tInSlot.First(), placeHand));
  3584. _tmSchdActions.Enqueue(llSwap);
  3585. return;
  3586. }
  3587. }
  3588. var emptyLL = lls.Where(ll => CanWaferGotoLL(robotWafers.First(), ll.Key) &&
  3589. GetLLFixedReadyInOutSlots(ll.Key).tInSlot.Count == 1 &&
  3590. ll.Value.TimeToReady <= 2);
  3591. if (emptyLL.Count() > 0)
  3592. {
  3593. var destLL = emptyLL.First().Key;
  3594. _tmSchdActions.Enqueue(new List<MoveItem> { new MoveItem( ModuleName.TMRobot,
  3595. robotWafers.First().currentSlot,
  3596. destLL,
  3597. GetLLFixedReadyInOutSlots(destLL).tInSlot.First(),
  3598. (Hand)robotWafers.First().currentSlot) });
  3599. return;
  3600. }
  3601. }
  3602. // try to return a wafer
  3603. var readyReturnWafers = _lstWaferTasks.Where(wt => ModuleHelper.IsPm(wt.currentMod) && ModuleHelper.IsLoadPort(wt.destMod) && _dictModuleTask[wt.currentMod].TimeToReady <= 2);
  3604. foreach (var pmWafer in readyReturnWafers)
  3605. {
  3606. var readyLL = lls.Where(ll => CanWaferGotoLL(pmWafer, ll.Key) && GetLLFixedReadyInOutSlots(ll.Key).tInSlot.Count > 0 && ll.Value.TimeToReady <= 2).OrderBy(ll => ll.Value.Scheduler.IsVac ? 0 : 1);
  3607. if (readyLL.Count() > 0)
  3608. {
  3609. var validHand = (Hand)(1 - robotWafers.First().currentSlot);
  3610. _tmSchdActions.Enqueue(new List<MoveItem> { new MoveItem(pmWafer.currentMod, 0, ModuleName.TMRobot, (int)validHand, validHand) });
  3611. _tmSchdActions.Enqueue(new List<MoveItem> { new MoveItem(ModuleName.TMRobot, (int)validHand, readyLL.First().Key, GetLLFixedReadyInOutSlots(readyLL.First().Key).tInSlot.First(), validHand) });
  3612. return;
  3613. }
  3614. }
  3615. // try to push in a wafer
  3616. var readyInLL = lls.Where(ll => GetLLFixedReadyInOutSlots(ll.Key).tOutSlot.Count > 0 && ll.Value.TimeToReady <= 2).OrderBy(ll => ll.Value.Scheduler.IsVac ? 0 : 1);
  3617. foreach (var ll in readyInLL)
  3618. {
  3619. var waferStatus = GetLLFixedReadyInOutSlots(ll.Key);
  3620. var wafer = _lstWaferTasks.Find(wt => wt.currentMod == ll.Key && wt.currentSlot == waferStatus.tOutSlot.First());
  3621. if (wafer != null &&
  3622. ModuleHelper.IsPm(wafer.destMod) &&
  3623. !_lstWaferTasks.Exists(wt => wt.currentMod == wafer.destMod) &&
  3624. _dictModuleTask[wafer.destMod].TimeToReady <= 2)
  3625. {
  3626. var validHand = (Hand)(1 - robotWafers.First().currentSlot);
  3627. _tmSchdActions.Enqueue(new List<MoveItem> { new MoveItem(wafer.currentMod, wafer.currentSlot, ModuleName.TMRobot, (int)validHand, validHand) });
  3628. _tmSchdActions.Enqueue(new List<MoveItem> { new MoveItem(ModuleName.TMRobot, (int)validHand, wafer.destMod, 0, validHand) });
  3629. return;
  3630. }
  3631. }
  3632. }
  3633. else if (robotWafers.Count() == 0)
  3634. {
  3635. var readyReturnWafers = _lstWaferTasks.Where(wt => ModuleHelper.IsPm(wt.currentMod) && ModuleHelper.IsLoadPort(wt.destMod)).ToList();
  3636. if (readyReturnWafers.Count > 0)
  3637. {
  3638. // try to do a LL swap action
  3639. var readySwapLL = lls.Where(ll => GetLLFixedReadyInOutSlots(ll.Key).tOutSlot.Count == 1 && GetLLFixedReadyInOutSlots(ll.Key).tInSlot.Count == 1 && ll.Value.TimeToReady <= 10);
  3640. foreach (var pmWafer in readyReturnWafers)
  3641. {
  3642. foreach (var ll in readySwapLL)
  3643. {
  3644. if (CanWaferGotoLL(pmWafer, ll.Key))
  3645. {
  3646. var returnHand = SelectTMPickArm(pmWafer.currentMod);
  3647. var pickHand = SelectTMPickArm(ll.Key);
  3648. var llWaferStatus = GetLLFixedReadyInOutSlots(ll.Key);
  3649. _tmSchdActions.Enqueue(new List<MoveItem> { new MoveItem(pmWafer.currentMod, pmWafer.currentSlot, ModuleName.TMRobot, (int)returnHand, returnHand) });
  3650. var llSwapActions = new List<MoveItem>();
  3651. llSwapActions.Add(new MoveItem(ModuleName.TMRobot, (int)returnHand, ll.Key, llWaferStatus.tInSlot.First(), returnHand));
  3652. llSwapActions.Add(new MoveItem(ll.Key, llWaferStatus.tOutSlot.First(), ModuleName.TMRobot, (int)pickHand, pickHand));
  3653. _tmSchdActions.Enqueue(llSwapActions);
  3654. return;
  3655. }
  3656. }
  3657. }
  3658. // just return to LL
  3659. var readyInLL = lls.Where(ll => GetLLFixedReadyInOutSlots(ll.Key).tInSlot.Count == 1 && ll.Value.TimeToReady <= 5);
  3660. foreach (var pmWafer in readyReturnWafers)
  3661. {
  3662. foreach (var ll in readyInLL)
  3663. {
  3664. if (CanWaferGotoLL(pmWafer, ll.Key))
  3665. {
  3666. var returnHand = SelectTMPickArm(pmWafer.currentMod);
  3667. _tmSchdActions.Enqueue(new List<MoveItem> { new MoveItem(pmWafer.currentMod, pmWafer.currentSlot, ModuleName.TMRobot, (int)returnHand, returnHand) });
  3668. _tmSchdActions.Enqueue(new List<MoveItem> { new MoveItem(ModuleName.TMRobot, (int)returnHand, ll.Key, GetLLFixedReadyInOutSlots(ll.Key).tInSlot.First(), returnHand) });
  3669. return;
  3670. }
  3671. }
  3672. }
  3673. }
  3674. var readyPushInLLs = lls.Where(ll => GetLLFixedReadyInOutSlots(ll.Key).tOutSlot.Count == 1 && ll.Value.TimeToReady <= 2);
  3675. if (readyPushInLLs.Count() > 0)
  3676. {
  3677. var destLL = readyPushInLLs.First().Key;
  3678. if (readyPushInLLs.Count() == 2 && readyPushInLLs.First().Value.Scheduler.IsAtm && readyPushInLLs.Last().Value.Scheduler.IsVac)
  3679. {
  3680. destLL = readyPushInLLs.Last().Key;
  3681. }
  3682. var pickHand = SelectTMPickArm(destLL);
  3683. var llWafers = _lstWaferTasks.Where(x => x.currentMod == destLL).ToList();
  3684. if (llWafers.Count() == 0)
  3685. {
  3686. return;
  3687. }
  3688. for (int i = 0; i < llWafers.Count(); i++)
  3689. {
  3690. if (_lstWaferTasks.Count(x => ModuleHelper.IsPm(x.currentMod) && llWafers[i].destMod == x.currentMod) == 0)
  3691. {
  3692. _tmSchdActions.Enqueue(new List<MoveItem> { new MoveItem(destLL, GetLLFixedReadyInOutSlots(destLL).tOutSlot.First(), ModuleName.TMRobot, (int)pickHand, pickHand) });
  3693. }
  3694. }
  3695. //if (_lstWaferTasks.Count(x => ModuleHelper.IsPm(x.currentMod)) == 0)
  3696. //{
  3697. //_tmSchdActions.Enqueue(new List<MoveItem> { new MoveItem(destLL, GetLLFixedReadyInOutSlots(destLL).tOutSlot.First(), ModuleName.TMRobot, (int)pickHand, pickHand) });
  3698. //}
  3699. return;
  3700. }
  3701. }
  3702. else
  3703. {
  3704. // should not go here
  3705. LOG.Write(eEvent.WARN_ROUTER, ModuleName.System, "Had better avoid picking two wafers on TM robot arms under Loadlock fixed slot pattern");
  3706. foreach (var wafer in robotWafers)
  3707. {
  3708. if (ModuleHelper.IsPm(wafer.destMod))
  3709. {
  3710. if (!_lstWaferTasks.Exists(wt => wt.currentMod == wafer.destMod) && _dictModuleTask[wafer.destMod].TimeToReady <= 2)
  3711. {
  3712. _tmSchdActions.Enqueue(new List<MoveItem> { new MoveItem(ModuleName.TMRobot, wafer.currentSlot, wafer.destMod, 0, (Hand)wafer.currentSlot) });
  3713. return;
  3714. }
  3715. }
  3716. else
  3717. {
  3718. var readyInLL = lls.Where(ll => GetLLFixedReadyInOutSlots(ll.Key).tInSlot.Count == 1 && CanWaferGotoLL(wafer, ll.Key) && ll.Value.TimeToReady <= 2);
  3719. if (readyInLL.Count() > 0)
  3720. {
  3721. _tmSchdActions.Enqueue(new List<MoveItem> { new MoveItem(ModuleName.TMRobot, wafer.currentSlot, readyInLL.First().Key, GetLLFixedReadyInOutSlots(readyInLL.First().Key).tInSlot.First(), (Hand)wafer.currentSlot) });
  3722. return;
  3723. }
  3724. }
  3725. }
  3726. }
  3727. }
  3728. else // 4 slot Loadlock
  3729. {
  3730. if (robotWafers.Count() > 0)
  3731. {
  3732. if (robotWafers.Count() == 2)
  3733. {
  3734. if (robotWafers.All(wt => ModuleHelper.IsLoadPort(wt.destMod)))
  3735. {
  3736. var destLL = lls.Where(ll => robotWafers.All(wt => CanWaferGotoLL(wt, ll.Key)) &&
  3737. ll.Value.TimeToReady <= 10 &&
  3738. GetLLFixedReadyInOutSlots(ll.Key).tInSlot.Count == 2).
  3739. OrderByDescending(ll => GetLLFixedReadyInOutSlots(ll.Key).tOutSlot.Count).
  3740. OrderBy(ll => ll.Value.TimeToReady);
  3741. if (destLL.Count() > 0)
  3742. {
  3743. var swapLL = destLL.First().Key;
  3744. var llWaferStatus = GetLLFixedReadyInOutSlots(swapLL);
  3745. var swapActions = new List<MoveItem>();
  3746. swapActions.Add(new MoveItem(ModuleName.TMRobot, robotWafers.First().currentSlot, swapLL, llWaferStatus.tInSlot.First(), (Hand)robotWafers.First().currentSlot));
  3747. swapActions.Add(new MoveItem(ModuleName.TMRobot, robotWafers.Last().currentSlot, swapLL, llWaferStatus.tInSlot.Last(), (Hand)robotWafers.Last().currentSlot));
  3748. int pickCount = 0;
  3749. foreach (var slot in llWaferStatus.eOutSlot)
  3750. {
  3751. swapActions.Add(new MoveItem(swapLL, slot, ModuleName.TMRobot, pickCount, (Hand)pickCount));
  3752. }
  3753. _tmSchdActions.Enqueue(swapActions);
  3754. return;
  3755. }
  3756. }
  3757. foreach (var wafer in robotWafers)
  3758. {
  3759. if (ModuleHelper.IsPm(wafer.destMod))
  3760. {
  3761. if (!_lstWaferTasks.Exists(wt => wt.currentMod == wafer.destMod) && _dictModuleTask[wafer.destMod].TimeToReady <= 5)
  3762. {
  3763. _tmSchdActions.Enqueue(new List<MoveItem> { new MoveItem(ModuleName.TMRobot, wafer.currentSlot, wafer.destMod, 0, (Hand)wafer.currentSlot) });
  3764. return;
  3765. }
  3766. }
  3767. else
  3768. {
  3769. var placeLL = lls.Where(ll => CanWaferGotoLL(wafer, ll.Key) && ll.Value.TimeToReady <= 5 && GetLLFixedReadyInOutSlots(ll.Key).tInSlot.Count > 0).OrderBy(ll => ll.Value.TimeToReady);
  3770. if (placeLL.Count() > 0)
  3771. {
  3772. _tmSchdActions.Enqueue(new List<MoveItem> { new MoveItem(ModuleName.TMRobot, wafer.currentSlot, placeLL.First().Key, GetLLFixedReadyInOutSlots(placeLL.First().Key).tInSlot.First(), (Hand)wafer.currentSlot) });
  3773. return;
  3774. }
  3775. }
  3776. }
  3777. }
  3778. else if (robotWafers.Count() == 1)
  3779. {
  3780. var robotWafer = robotWafers.First();
  3781. if (ModuleHelper.IsPm(robotWafer.destMod))
  3782. {
  3783. if (_dictModuleTask[robotWafer.destMod].TimeToReady <= 5)
  3784. {
  3785. if (_lstWaferTasks.Exists(wt => wt.currentMod == robotWafer.destMod))
  3786. {
  3787. var pmSwap = new List<MoveItem>();
  3788. pmSwap.Add(new MoveItem(robotWafer.destMod, 0, ModuleName.TMRobot, 1 - robotWafer.currentSlot, (Hand)(1 - robotWafer.currentSlot)));
  3789. pmSwap.Add(new MoveItem(ModuleName.TMRobot, robotWafer.currentSlot, robotWafer.destMod, 0, (Hand)robotWafer.currentSlot));
  3790. _tmSchdActions.Enqueue(pmSwap);
  3791. return;
  3792. }
  3793. else
  3794. {
  3795. _tmSchdActions.Enqueue(new List<MoveItem> { new MoveItem(ModuleName.TMRobot, robotWafer.currentSlot, robotWafer.destMod, 0, (Hand)robotWafer.currentSlot) });
  3796. return;
  3797. }
  3798. }
  3799. }
  3800. else
  3801. {
  3802. var destLL = lls.Where(ll => CanWaferGotoLL(robotWafer, ll.Key) &&
  3803. ll.Value.TimeToReady <= 10 &&
  3804. GetLLFixedReadyInOutSlots(ll.Key).tInSlot.Count > 0).
  3805. OrderByDescending(ll => GetLLFixedReadyInOutSlots(ll.Key).tOutSlot.Count).
  3806. OrderBy(ll => ll.Value.TimeToReady);
  3807. if (destLL.Count() > 0)
  3808. {
  3809. var swapLL = destLL.First().Key;
  3810. var llWaferStatus = GetLLFixedReadyInOutSlots(swapLL);
  3811. var llSwap = new List<MoveItem>();
  3812. llSwap.Add(new MoveItem(ModuleName.TMRobot, robotWafer.currentSlot, swapLL, llWaferStatus.tInSlot.First(), (Hand)robotWafer.currentSlot));
  3813. int pickCount = 0;
  3814. foreach (var slot in llWaferStatus.tOutSlot)
  3815. {
  3816. llSwap.Add(new MoveItem(swapLL, slot, ModuleName.TMRobot, pickCount, (Hand)pickCount));
  3817. pickCount++;
  3818. }
  3819. _tmSchdActions.Enqueue(llSwap);
  3820. return;
  3821. }
  3822. }
  3823. // try to return a wafer
  3824. var readyReturnWafers = _lstWaferTasks.Where(wt => ModuleHelper.IsPm(wt.currentMod) && ModuleHelper.IsLoadPort(wt.destMod) && _dictModuleTask[wt.currentMod].TimeToReady <= 2);
  3825. foreach (var pmWafer in readyReturnWafers)
  3826. {
  3827. var readyLL = lls.Where(ll => CanWaferGotoLL(pmWafer, ll.Key) && GetLLFixedReadyInOutSlots(ll.Key).tInSlot.Count > 0 && ll.Value.TimeToReady <= 2).OrderBy(ll => ll.Value.Scheduler.IsVac ? 0 : 1);
  3828. if (readyLL.Count() > 0)
  3829. {
  3830. var validHand = (Hand)(1 - robotWafers.First().currentSlot);
  3831. _tmSchdActions.Enqueue(new List<MoveItem> { new MoveItem(pmWafer.currentMod, 0, ModuleName.TMRobot, (int)validHand, validHand) });
  3832. _tmSchdActions.Enqueue(new List<MoveItem> { new MoveItem(ModuleName.TMRobot, (int)validHand, readyLL.First().Key, GetLLFixedReadyInOutSlots(readyLL.First().Key).tInSlot.First(), validHand) });
  3833. return;
  3834. }
  3835. }
  3836. // try to push in a wafer
  3837. var readyInLL = lls.Where(ll => GetLLFixedReadyInOutSlots(ll.Key).tOutSlot.Count > 0 && ll.Value.TimeToReady <= 2).OrderBy(ll => ll.Value.Scheduler.IsVac ? 0 : 1);
  3838. foreach (var ll in readyInLL)
  3839. {
  3840. var waferStatus = GetLLFixedReadyInOutSlots(ll.Key);
  3841. var wafer = _lstWaferTasks.Find(wt => wt.currentMod == ll.Key && wt.currentSlot == waferStatus.tOutSlot.First());
  3842. if (wafer != null &&
  3843. ModuleHelper.IsPm(wafer.destMod) &&
  3844. !_lstWaferTasks.Exists(wt => wt.currentMod == wafer.destMod) &&
  3845. _dictModuleTask[wafer.destMod].TimeToReady <= 2)
  3846. {
  3847. var validHand = (Hand)(1 - robotWafers.First().currentSlot);
  3848. _tmSchdActions.Enqueue(new List<MoveItem> { new MoveItem(wafer.currentMod, wafer.currentSlot, ModuleName.TMRobot, (int)validHand, validHand) });
  3849. _tmSchdActions.Enqueue(new List<MoveItem> { new MoveItem(ModuleName.TMRobot, (int)validHand, wafer.destMod, 0, validHand) });
  3850. return;
  3851. }
  3852. }
  3853. }
  3854. }
  3855. else // robot empty
  3856. {
  3857. var nearlyReadyReturnWafers = _lstWaferTasks.Where(wt => ModuleHelper.IsPm(wt.currentMod) && (ModuleHelper.IsLoadPort(wt.destMod) || _dictModuleTask[wt.currentMod].TimeToReady <= 10)).OrderBy(wt => _dictModuleTask[wt.currentMod].TimeToReady).ToList();
  3858. var readyLL = lls.Where(ll => ll.Value.TimeToReady <= 10);
  3859. if (readyLL.Count() > 0)
  3860. {
  3861. var llPath = FindTheBestMovePathWithFixSlot(readyLL, nearlyReadyReturnWafers);
  3862. if (ModuleHelper.IsLoadLock(llPath))
  3863. {
  3864. int returnCount = 0;
  3865. var llWaferStatus = GetLLFixedReadyInOutSlots(llPath);
  3866. var llSwapActions = new List<MoveItem>();
  3867. foreach (var wafer in nearlyReadyReturnWafers)
  3868. {
  3869. if (CanWaferGotoLL(wafer, llPath) && returnCount < Math.Min(llWaferStatus.tInSlot.Count, 2))
  3870. {
  3871. _tmSchdActions.Enqueue(new List<MoveItem> { new MoveItem(wafer.currentMod, 0, ModuleName.TMRobot, returnCount, (Hand)returnCount) });
  3872. llSwapActions.Add(new MoveItem(ModuleName.TMRobot, returnCount, llPath, llWaferStatus.tInSlot[returnCount], (Hand)returnCount));
  3873. returnCount++;
  3874. }
  3875. }
  3876. int pickCount = 0;
  3877. foreach (var slot in llWaferStatus.tOutSlot)
  3878. {
  3879. if (pickCount < 2)
  3880. {
  3881. llSwapActions.Add(new MoveItem(llPath, slot, ModuleName.TMRobot, pickCount, (Hand)pickCount));
  3882. pickCount++;
  3883. }
  3884. }
  3885. if (llSwapActions.Count > 0)
  3886. {
  3887. _tmSchdActions.Enqueue(llSwapActions);
  3888. }
  3889. return;
  3890. }
  3891. }
  3892. }
  3893. }
  3894. }
  3895. private bool IsTMRobotArmNotReserved(Hand hand)
  3896. {
  3897. if (WaferManager.Instance.CheckHasWafer(ModuleName.TMRobot, (int)hand) || _lstWaferTasks.Exists(wt => wt.currentMod == ModuleName.TMRobot && wt.currentSlot == (int)hand))
  3898. return false;
  3899. foreach (var acs in _tmSchdActions)
  3900. {
  3901. foreach (var ac in acs)
  3902. {
  3903. if ((ac.DestinationModule == ModuleName.TMRobot || ac.SourceModule == ModuleName.TMRobot) && ac.DestinationSlot == (int)hand)
  3904. return false;
  3905. }
  3906. }
  3907. return true;
  3908. }
  3909. private Hand SelectTMPickArm(ModuleName mod)
  3910. {
  3911. switch (_tmRobotSingleArmOption)
  3912. {
  3913. case 0:
  3914. {
  3915. if (IsTMRobotArmNotReserved(Hand.Blade1))
  3916. return Hand.Blade1;
  3917. if (IsTMRobotArmNotReserved(Hand.Blade2))
  3918. return Hand.Blade2;
  3919. }
  3920. break;
  3921. case 1:
  3922. {
  3923. if (IsTMRobotArmNotReserved(Hand.Blade1))
  3924. return Hand.Blade1;
  3925. }
  3926. break;
  3927. case 2:
  3928. {
  3929. if (IsTMRobotArmNotReserved(Hand.Blade2))
  3930. return Hand.Blade2;
  3931. }
  3932. break;
  3933. // Blade1 In Blade2 Out
  3934. case 3:
  3935. {
  3936. if (ModuleHelper.IsLoadLock(mod) && IsTMRobotArmNotReserved(Hand.Blade1))
  3937. return Hand.Blade1;
  3938. if (ModuleHelper.IsPm(mod) && IsTMRobotArmNotReserved(Hand.Blade2))
  3939. return Hand.Blade2;
  3940. }
  3941. break;
  3942. // Blade2 In Blade1 Out
  3943. case 4:
  3944. {
  3945. if (ModuleHelper.IsLoadLock(mod) && IsTMRobotArmNotReserved(Hand.Blade2))
  3946. return Hand.Blade2;
  3947. if (ModuleHelper.IsPm(mod) && IsTMRobotArmNotReserved(Hand.Blade1))
  3948. return Hand.Blade1;
  3949. }
  3950. break;
  3951. }
  3952. return Hand.None;
  3953. }
  3954. private List<Hand> GetTMFreeHand()
  3955. {
  3956. var lstHands = new List<Hand>();
  3957. if (WaferManager.Instance.CheckNoWafer(ModuleName.TMRobot, 0) && _tmRobotSingleArmOption != 2)
  3958. lstHands.Add(Hand.Blade1);
  3959. if (WaferManager.Instance.CheckNoWafer(ModuleName.TMRobot, 1) && _tmRobotSingleArmOption != 1)
  3960. lstHands.Add(Hand.Blade2);
  3961. return lstHands;
  3962. }
  3963. private List<Hand> GetEFEMFreeHand()
  3964. {
  3965. var lstHands = new List<Hand>();
  3966. if (WaferManager.Instance.CheckNoWafer(ModuleName.EfemRobot, 0) && _efemRobotSingleArmOption != 2)
  3967. lstHands.Add(Hand.Blade1);
  3968. if (WaferManager.Instance.CheckNoWafer(ModuleName.EfemRobot, 1) && _efemRobotSingleArmOption != 1)
  3969. lstHands.Add(Hand.Blade2);
  3970. return lstHands;
  3971. }
  3972. private bool CanWaferGotoLL(WaferTask task, ModuleName ll)
  3973. {
  3974. bool bVacWafer = ModuleHelper.IsPm(task.currentMod) || ModuleHelper.IsTMRobot(task.currentMod);
  3975. if (bVacWafer)
  3976. {
  3977. if (ModuleHelper.IsPm(task.destMod))
  3978. return false;
  3979. switch (task.llInOutPath)
  3980. {
  3981. case SequenceLLInOutPath.AInAOut:
  3982. case SequenceLLInOutPath.BInAOut:
  3983. return ll == ModuleName.LLA;
  3984. case SequenceLLInOutPath.AInBOut:
  3985. case SequenceLLInOutPath.BInBOut:
  3986. return ll == ModuleName.LLB;
  3987. case SequenceLLInOutPath.DInDOut:
  3988. return true;
  3989. }
  3990. }
  3991. else
  3992. {
  3993. if (ModuleHelper.IsLoadPort(task.destMod))
  3994. return false;
  3995. switch (task.llInOutPath)
  3996. {
  3997. case SequenceLLInOutPath.AInAOut:
  3998. case SequenceLLInOutPath.AInBOut:
  3999. return ll == ModuleName.LLA;
  4000. case SequenceLLInOutPath.BInAOut:
  4001. case SequenceLLInOutPath.BInBOut:
  4002. return ll == ModuleName.LLB;
  4003. case SequenceLLInOutPath.DInDOut:
  4004. return true;
  4005. }
  4006. }
  4007. return false;
  4008. }
  4009. private (List<int> inSlot, List<int> outSlot, List<int> emptySlot) GetLLReadyInOutSlots(ModuleName ll)
  4010. {
  4011. var readInSlots = new List<int>();
  4012. var readyOutSlots = new List<int>();
  4013. var emptySlots = new List<int>();
  4014. for (int slot = 0; slot < (ll == ModuleName.LLA ? _LLASlotNumber : _LLBSlotNumber); slot++)
  4015. {
  4016. if (WaferManager.Instance.CheckNoWafer(ll, slot) && !_lstWaferTasks.Exists(wafer => (wafer.routedMod == ll && wafer.routedSlot == slot) || (wafer.currentMod == ll && wafer.currentSlot == slot) || (wafer.nextMod == ll && wafer.nextSlot == slot)))
  4017. {
  4018. emptySlots.Add(slot);
  4019. }
  4020. if (WaferManager.Instance.CheckHasWafer(ll, slot))
  4021. {
  4022. Guid waferID = WaferManager.Instance.GetWafer(ll, slot).InnerId;
  4023. var waferTask = _lstWaferTasks.Find(wafer => wafer.waferId == waferID);
  4024. if (waferTask != null && waferTask.movingStatus == RState.End)
  4025. {
  4026. if (ModuleHelper.IsPm(waferTask.destMod))
  4027. readInSlots.Add(slot);
  4028. else if (_dictModuleTask[ll].Scheduler.WaferArrivedTicks(slot) > waferTask.llDelayTime * 1000)
  4029. readyOutSlots.Add(slot);
  4030. }
  4031. }
  4032. }
  4033. return (readInSlots, readyOutSlots, emptySlots);
  4034. }
  4035. private (List<int> eInSlot, List<int> eOutSlot, List<int> tInSlot, List<int> tOutSlot) GetLLFixedReadyInOutSlots(ModuleName ll)
  4036. {
  4037. var eInSlot = new List<int>();
  4038. var eOutSlot = new List<int>();
  4039. var tInSlot = new List<int>();
  4040. var tOutSlot = new List<int>();
  4041. int SlotNumber = ll == ModuleName.LLA ? _LLASlotNumber : _LLBSlotNumber;
  4042. for (int slot = 0; slot < SlotNumber; slot++)
  4043. {
  4044. if (WaferManager.Instance.CheckNoWafer(ll, slot) && !_lstWaferTasks.Exists(wafer => (wafer.routedMod == ll && wafer.routedSlot == slot) || (wafer.currentMod == ll && wafer.currentSlot == slot) || (wafer.nextMod == ll && wafer.nextSlot == slot)))
  4045. {
  4046. if ((slot >= SlotNumber / 2 && _LLSlotInOutOption == LLSlotInOutOpt.UpperInLowerOut) ||
  4047. (slot < SlotNumber / 2 && _LLSlotInOutOption == LLSlotInOutOpt.LowerInUpperOut))
  4048. {
  4049. eInSlot.Add(slot);
  4050. }
  4051. else
  4052. {
  4053. tInSlot.Add(slot);
  4054. }
  4055. }
  4056. if (WaferManager.Instance.CheckHasWafer(ll, slot))
  4057. {
  4058. var wafer = WaferManager.Instance.GetWafer(ll, slot);
  4059. var waferTask = _lstWaferTasks.Find(wt => wt.waferId == wafer.InnerId && wt.currentMod == ll && wt.currentSlot == slot);
  4060. if (waferTask == null)
  4061. {
  4062. LOG.Write(eEvent.EV_ROUTER, ll, $"Routing trace: did not find inner task associated with {wafer.WaferOrigin} at {ll}.{slot + 1}");
  4063. }
  4064. else if (waferTask.movingStatus == RState.End)
  4065. {
  4066. if (ModuleHelper.IsPm(waferTask.destMod))
  4067. tOutSlot.Add(slot);
  4068. else if (_dictModuleTask[ll].Scheduler.WaferArrivedTicks(slot) > waferTask.llDelayTime * 1000)
  4069. eOutSlot.Add(slot);
  4070. }
  4071. }
  4072. }
  4073. return (eInSlot, eOutSlot, tInSlot, tOutSlot);
  4074. }
  4075. private (List<WaferInfo> wafers, List<int> emptySlots) GetLLWaferExistance(ModuleName ll)
  4076. {
  4077. var lstWafers = new List<WaferInfo>();
  4078. var lstEmptySlots = new List<int>();
  4079. for (int slot = 0; slot < (ll == ModuleName.LLA ? _LLASlotNumber : _LLBSlotNumber); slot++)
  4080. {
  4081. var wafer = WaferManager.Instance.GetWafer(ll, slot);
  4082. if (wafer != null && !wafer.IsEmpty)
  4083. {
  4084. lstWafers.Add(wafer);
  4085. }
  4086. else
  4087. {
  4088. lstEmptySlots.Add(slot);
  4089. }
  4090. }
  4091. return (lstWafers, lstEmptySlots);
  4092. }
  4093. private bool NeedPairingForNexWafer()
  4094. {
  4095. if (_lstWaferTasks.Exists(wt => ModuleHelper.IsLoadPort(wt.destMod) && (ModuleHelper.IsPm(wt.currentMod) || ModuleHelper.IsTMRobot(wt.currentMod))))
  4096. return true;
  4097. var processingPMs = _dictModuleTask.Where(mod => ModuleHelper.IsPm(mod.Key) && mod.Value.HasWafer && mod.Value.Scheduler.IsOnline).OrderBy(wt => wt.Value.TimeToReady);
  4098. if (processingPMs.Count() > 0)
  4099. {
  4100. int maxWaitingTime = SC.GetValue<int>("TM.MaxWaitTimePairingOutWafer");
  4101. return processingPMs.First().Value.TimeToReady < maxWaitingTime;
  4102. }
  4103. return false;
  4104. }
  4105. private void PrepareLLPressure()
  4106. {
  4107. if (RouteManager.IsATMMode)
  4108. return;
  4109. var schdPumpingLLs = new List<ModuleName>();
  4110. var schdVentingLLs = new List<ModuleName>();
  4111. var schdCoolingLLs = new List<ModuleName>();
  4112. var llaWaferStatus = GetLLReadyInOutSlots(ModuleName.LLA);
  4113. var llbWaferStatus = GetLLReadyInOutSlots(ModuleName.LLB);
  4114. // pre pumping/venting for schedule actions
  4115. if (_efemSchdActions.Count > 0)
  4116. {
  4117. var ventLLs = _efemSchdActions.Where(acs => ModuleHelper.IsLoadLock(acs.First().Module)).Select(item => item.First().Module).ToList();
  4118. foreach (var ll in ventLLs)
  4119. {
  4120. if (!schdVentingLLs.Contains(ll))
  4121. schdVentingLLs.Add(ll);
  4122. }
  4123. }
  4124. if (_tmSchdActions.Count > 0)
  4125. {
  4126. var pumpLLs = _tmSchdActions.Where(acs => ModuleHelper.IsLoadLock(acs.First().Module)).Select(item => item.First().Module).ToList();
  4127. foreach (var ll in pumpLLs)
  4128. {
  4129. if (!schdPumpingLLs.Contains(ll))
  4130. schdPumpingLLs.Add(ll);
  4131. }
  4132. }
  4133. // loadlock in out path advance check
  4134. if (_LLInOutPath == SequenceLLInOutPath.AInBOut)
  4135. {
  4136. if (llaWaferStatus.emptySlot.Count == _LLASlotNumber)
  4137. {
  4138. if (!schdVentingLLs.Contains(ModuleName.LLA))
  4139. schdVentingLLs.Add(ModuleName.LLA);
  4140. }
  4141. else if ((llaWaferStatus.inSlot.Count == _LLASlotNumber) ||
  4142. (llaWaferStatus.inSlot.Count > 0 && llaWaferStatus.inSlot.Count < _LLASlotNumber && _lstWaferTasks.Count(wt => ModuleHelper.IsLoadPort(wt.currentMod) && ModuleHelper.IsPm(wt.destMod)) == 0))
  4143. {
  4144. if (!schdPumpingLLs.Contains(ModuleName.LLA))
  4145. schdPumpingLLs.Add(ModuleName.LLA);
  4146. }
  4147. if (llbWaferStatus.emptySlot.Count == _LLBSlotNumber)
  4148. {
  4149. if (!schdPumpingLLs.Contains(ModuleName.LLB))
  4150. schdPumpingLLs.Add(ModuleName.LLB);
  4151. }
  4152. else if ((llbWaferStatus.outSlot.Count == _LLBSlotNumber) ||
  4153. (llbWaferStatus.outSlot.Count > 0 && llbWaferStatus.outSlot.Count < _LLBSlotNumber && !NeedPairingForNexWafer()))
  4154. {
  4155. schdCoolingLLs.Add(ModuleName.LLB);
  4156. }
  4157. }
  4158. else if (_LLInOutPath == SequenceLLInOutPath.BInAOut)
  4159. {
  4160. if (llbWaferStatus.emptySlot.Count == _LLBSlotNumber)
  4161. {
  4162. if (!schdVentingLLs.Contains(ModuleName.LLA))
  4163. schdVentingLLs.Add(ModuleName.LLB);
  4164. }
  4165. else if ((llbWaferStatus.inSlot.Count == _LLBSlotNumber) ||
  4166. (llbWaferStatus.inSlot.Count > 0 && llbWaferStatus.inSlot.Count < _LLBSlotNumber && _lstWaferTasks.Count(wt => ModuleHelper.IsLoadPort(wt.currentMod) && ModuleHelper.IsPm(wt.destMod)) == 0))
  4167. {
  4168. if (!schdPumpingLLs.Contains(ModuleName.LLB))
  4169. schdPumpingLLs.Add(ModuleName.LLB);
  4170. }
  4171. if (llaWaferStatus.emptySlot.Count == _LLASlotNumber)
  4172. {
  4173. if (!schdPumpingLLs.Contains(ModuleName.LLA))
  4174. schdPumpingLLs.Add(ModuleName.LLA);
  4175. }
  4176. else if ((llaWaferStatus.outSlot.Count == _LLASlotNumber) ||
  4177. (llaWaferStatus.outSlot.Count > 0 && llaWaferStatus.outSlot.Count < _LLASlotNumber && !NeedPairingForNexWafer()))
  4178. {
  4179. schdCoolingLLs.Add(ModuleName.LLA);
  4180. }
  4181. }
  4182. else
  4183. {
  4184. if ((llaWaferStatus.inSlot.Count == _LLASlotNumber) ||
  4185. (llaWaferStatus.inSlot.Count > 0 && llaWaferStatus.inSlot.Count < _LLASlotNumber && _lstWaferTasks.Count(wt => ModuleHelper.IsLoadPort(wt.currentMod) && ModuleHelper.IsPm(wt.destMod)) == 0))
  4186. {
  4187. if (!schdPumpingLLs.Contains(ModuleName.LLA))
  4188. schdPumpingLLs.Add(ModuleName.LLA);
  4189. }
  4190. else if ((llaWaferStatus.outSlot.Count == _LLASlotNumber) ||
  4191. (llaWaferStatus.outSlot.Count > 0 && llaWaferStatus.outSlot.Count < _LLASlotNumber))
  4192. {
  4193. if (!schdVentingLLs.Contains(ModuleName.LLA))
  4194. schdCoolingLLs.Add(ModuleName.LLA);
  4195. }
  4196. if ((llbWaferStatus.inSlot.Count == _LLBSlotNumber) ||
  4197. (llbWaferStatus.inSlot.Count > 0 && llbWaferStatus.inSlot.Count < _LLBSlotNumber && _lstWaferTasks.Count(wt => ModuleHelper.IsLoadPort(wt.currentMod) && ModuleHelper.IsPm(wt.destMod)) == 0))
  4198. {
  4199. if (!schdPumpingLLs.Contains(ModuleName.LLB))
  4200. schdPumpingLLs.Add(ModuleName.LLB);
  4201. }
  4202. else if ((llbWaferStatus.outSlot.Count == _LLBSlotNumber) ||
  4203. (llbWaferStatus.outSlot.Count > 0 && llbWaferStatus.outSlot.Count < _LLBSlotNumber))
  4204. {
  4205. if (!schdVentingLLs.Contains(ModuleName.LLB))
  4206. schdCoolingLLs.Add(ModuleName.LLB);
  4207. }
  4208. }
  4209. foreach (var ventLL in schdVentingLLs)
  4210. {
  4211. if (_dictModuleTask[ventLL].Scheduler.IsAvailable && !_dictModuleTask[ventLL].Scheduler.IsAtm && !schdCoolingLLs.Contains(ventLL))
  4212. (_dictModuleTask[ventLL] as LoadlockTask).PreVent();
  4213. }
  4214. foreach (var pumpLL in schdPumpingLLs)
  4215. {
  4216. if (_dictModuleTask[pumpLL].Scheduler.IsAvailable && !_dictModuleTask[pumpLL].Scheduler.IsVac && !schdVentingLLs.Contains(pumpLL))
  4217. (_dictModuleTask[pumpLL] as LoadlockTask).PrePump();
  4218. }
  4219. foreach (var coolLL in schdCoolingLLs)
  4220. {
  4221. if (_dictModuleTask[coolLL].Scheduler.IsOnline && !_dictModuleTask[coolLL].Scheduler.IsAtm)
  4222. (_dictModuleTask[coolLL] as LoadlockTask).Cooling();
  4223. }
  4224. }
  4225. private void RunLotCleanTasks()
  4226. {
  4227. foreach (var pm in _dictModuleTask)
  4228. {
  4229. if (!ModuleHelper.IsPm(pm.Key))
  4230. continue;
  4231. var pmTask = pm.Value as PMTask;
  4232. if (pmTask.HasWafer)
  4233. {
  4234. var pmWafer = _lstWaferTasks.Find(wt => wt.currentMod == pm.Key);
  4235. if (pmWafer != null && IsProcessJobEnding(pmWafer.lotId, pm.Key) && !_postLotCleanMarks[pm.Key].Contains(pmWafer.lotId))
  4236. {
  4237. var pj = _lstProcessJobs.Find(process => process.InnerId == pmWafer.lotId);
  4238. if (pj != null)
  4239. {
  4240. var postClean = pj.Sequence.GetPostCleanRecipe(pm.Key);
  4241. if (!string.IsNullOrWhiteSpace(postClean))
  4242. {
  4243. pmTask.InvokePostJobClean(postClean);
  4244. _postLotCleanMarks[pm.Key].Add(pmWafer.lotId);
  4245. }
  4246. }
  4247. }
  4248. }
  4249. // pre clean
  4250. var waitInWafer = GetFirstWaitInWafer(pm.Key);
  4251. if (waitInWafer != null && !_preLotCleanMarks[pm.Key].Contains(waitInWafer.lotId))
  4252. {
  4253. var pj = _lstProcessJobs.Find(process => process.InnerId == waitInWafer.lotId);
  4254. if (pj != null)
  4255. {
  4256. var preClean = pj.Sequence.GetPreCleanRecipe(pm.Key);
  4257. if (!string.IsNullOrWhiteSpace(preClean))
  4258. {
  4259. if (DateTime.Now.Subtract(_pmLastWaferEndTime[pm.Key]).TotalSeconds > _preCleanMaxIdleTime)
  4260. {
  4261. pmTask.InvokePreJobClean(preClean);
  4262. }
  4263. _preLotCleanMarks[pm.Key].Add(waitInWafer.lotId);
  4264. }
  4265. }
  4266. }
  4267. }
  4268. }
  4269. bool IsProcessJobEnding(Guid pjId, ModuleName pm)
  4270. {
  4271. if (!_lstWaferTasks.Exists(wt => wt.lotId == pjId && wt.destMod == pm && !ModuleHelper.IsPm(wt.currentMod)))
  4272. {
  4273. var processJob = _lstProcessJobs.Find(pj => pj.InnerId == pjId);
  4274. if (processJob == null)
  4275. return false;
  4276. foreach (var slot in processJob.SlotWafers)
  4277. {
  4278. if (WaferManager.Instance.CheckHasWafer(slot.Item1, slot.Item2))
  4279. {
  4280. if (WaferManager.Instance.GetWafer(slot.Item1, slot.Item2).NextSequenceStep == 0)
  4281. return false;
  4282. }
  4283. }
  4284. return true;
  4285. }
  4286. return false;
  4287. }
  4288. WaferTask GetFirstWaitInWafer(ModuleName pm)
  4289. {
  4290. int distance(WaferTask wafer)
  4291. {
  4292. if (ModuleHelper.IsTMRobot(wafer.currentMod))
  4293. return 0;
  4294. else if (ModuleHelper.IsLoadLock(wafer.currentMod))
  4295. {
  4296. return (_dictModuleTask[wafer.currentMod] as LoadlockTask).Distance;
  4297. }
  4298. else if (ModuleHelper.IsEFEMRobot(wafer.currentMod))
  4299. {
  4300. if (wafer.IsAligned)
  4301. return 15;
  4302. else
  4303. return 19;
  4304. }
  4305. return 17;
  4306. }
  4307. var waitInWafers = _lstWaferTasks.Where(wt => (wt.destMod == pm) && !ModuleHelper.IsPm(wt.currentMod) && !ModuleHelper.IsLoadPort(wt.currentMod)).OrderBy(wt => distance(wt));
  4308. if (waitInWafers.Count() > 0)
  4309. return waitInWafers.First();
  4310. return null;
  4311. }
  4312. private void ReturnVacWafers()
  4313. {
  4314. if (_tmSchdActions.Count > 0 || _curTmAction.Count > 0)
  4315. {
  4316. RunSchdTMReturnActions();
  4317. return;
  4318. }
  4319. if (_tmRobotStatus != RState.End)
  4320. return;
  4321. var hands = GetTMFreeHand();
  4322. var lls = _dictModuleTask.Where(mod => ModuleHelper.IsLoadLock(mod.Key) && !IsLLReservedByEFEM(mod.Key) && !_qeWaitCoolingLLs.Contains(mod.Key) && mod.Value.Scheduler.IsIdle && mod.Value.Scheduler.IsInclude && GetLLWaferExistance(mod.Key).emptySlots.Count > 0).OrderByDescending(mod => GetLLReadyInOutSlots(mod.Key).emptySlot.Count);
  4323. if (lls.Count() > 0)
  4324. {
  4325. // return robot wafers
  4326. int moveCount = 0;
  4327. var llActions = new List<MoveItem>();
  4328. var emptySlots = GetLLWaferExistance(lls.First().Key).emptySlots;
  4329. for (int arm = 0; arm < 2; arm++)
  4330. {
  4331. var wafer = WaferManager.Instance.GetWafer(ModuleName.TMRobot, arm);
  4332. if (wafer != null && !wafer.IsEmpty && moveCount < emptySlots.Count)
  4333. {
  4334. llActions.Add(new MoveItem(ModuleName.TMRobot, arm, lls.First().Key, emptySlots[moveCount], (Hand)arm));
  4335. moveCount++;
  4336. }
  4337. }
  4338. if (moveCount > 0)
  4339. {
  4340. _tmSchdActions.Enqueue(llActions);
  4341. return;
  4342. }
  4343. // return PM wafers
  4344. var PickActions = new List<MoveItem>();
  4345. var hasWaferPMs = _dictModuleTask.Where(mod => ModuleHelper.IsPm(mod.Key) && mod.Value.Scheduler.IsIdle && mod.Value.Scheduler.IsInclude && WaferManager.Instance.CheckHasWafer(mod.Key, 0));
  4346. foreach (var pm in hasWaferPMs)
  4347. {
  4348. if (moveCount < hands.Count && moveCount < emptySlots.Count)
  4349. {
  4350. PickActions.Add(new MoveItem(pm.Key, 0, ModuleName.TMRobot, (int)hands[moveCount], hands[moveCount]));
  4351. llActions.Add(new MoveItem(ModuleName.TMRobot, (int)hands[moveCount], lls.First().Key, emptySlots[moveCount], hands[moveCount]));
  4352. moveCount++;
  4353. }
  4354. }
  4355. if (moveCount > 0)
  4356. {
  4357. foreach (var ac in PickActions)
  4358. {
  4359. _tmSchdActions.Enqueue(new List<MoveItem> { ac });
  4360. }
  4361. _tmSchdActions.Enqueue(llActions);
  4362. }
  4363. }
  4364. }
  4365. private void ReturnAtmWafers()
  4366. {
  4367. if (_efemSchdActions.Count > 0 || _curEfemAction.Count > 0)
  4368. {
  4369. RunSchdEFEMReturnActions();
  4370. return;
  4371. }
  4372. if (_efemRobotStatus != RState.End)
  4373. return;
  4374. // return Robot Wafer
  4375. for (int i = 0; i < 2; i++)
  4376. {
  4377. var wafer = WaferManager.Instance.GetWafer(ModuleName.EfemRobot, i);
  4378. if (wafer != null && !wafer.IsEmpty)
  4379. {
  4380. _efemSchdActions.Enqueue(new List<MoveItem> { new MoveItem(ModuleName.EfemRobot, i, (ModuleName)wafer.OriginStation, wafer.OriginSlot, (Hand)i) });
  4381. }
  4382. }
  4383. if (_efemSchdActions.Count > 0)
  4384. return;
  4385. var hands = GetEFEMFreeHand();
  4386. // return Aligner wafer
  4387. var aligner = ModuleHelper.InstalledModules.Where(mod => ModuleHelper.IsAligner(mod)).ToList();
  4388. if (aligner.Count > 0 && WaferManager.Instance.CheckHasWafer(aligner.First(), 0) && hands.Count > 0)
  4389. {
  4390. var wafer = WaferManager.Instance.GetWafer(aligner.First(), 0);
  4391. _efemSchdActions.Enqueue(new List<MoveItem> { new MoveItem(aligner.First(), 0, ModuleName.EfemRobot, (int)hands.First(), hands.First()) });
  4392. _efemSchdActions.Enqueue(new List<MoveItem> { new MoveItem(ModuleName.EfemRobot, (int)hands.First(), (ModuleName)wafer.OriginStation, wafer.OriginSlot, hands.First()) });
  4393. return;
  4394. }
  4395. // return Loadlock wafer
  4396. var lls = _dictModuleTask.Where(mod => ModuleHelper.IsLoadLock(mod.Key) && !IsLLReservedByTM(mod.Key) && !_qeWaitCoolingLLs.Contains(mod.Key) && mod.Value.Scheduler.IsIdle && GetLLWaferExistance(mod.Key).wafers.Count > 0).OrderByDescending(mod => GetLLWaferExistance(mod.Key).wafers.Count);
  4397. if (lls.Count() > 0)
  4398. {
  4399. int returnCount = 0;
  4400. var llActions = new List<MoveItem>();
  4401. var placActions = new List<MoveItem>();
  4402. foreach (var wafer in GetLLWaferExistance(lls.First().Key).wafers)
  4403. {
  4404. if (returnCount < hands.Count)
  4405. {
  4406. llActions.Add(new MoveItem(lls.First().Key, wafer.Slot, ModuleName.EfemRobot, (int)hands[returnCount], hands[returnCount]));
  4407. placActions.Add(new MoveItem(ModuleName.EfemRobot, (int)hands[returnCount], (ModuleName)wafer.OriginStation, wafer.OriginSlot, hands[returnCount]));
  4408. returnCount++;
  4409. }
  4410. }
  4411. if (returnCount > 0)
  4412. {
  4413. _efemSchdActions.Enqueue(llActions);
  4414. foreach (var ac in placActions)
  4415. {
  4416. _efemSchdActions.Enqueue(new List<MoveItem> { ac });
  4417. }
  4418. }
  4419. }
  4420. }
  4421. private void RunSchdEFEMReturnActions()
  4422. {
  4423. var efemRobot = Singleton<TransferModule>.Instance.GetScheduler(ModuleName.EfemRobot) as SchedulerEfemRobot;
  4424. if (efemRobot == null || !efemRobot.IsAvailable)
  4425. return;
  4426. if (_efemSchdActions.Count > 0)
  4427. {
  4428. if (_curEfemAction.Count == 0 || isReturnActionsDone(_curEfemAction))
  4429. {
  4430. var nextActions = _efemSchdActions.First();
  4431. var nextModule = nextActions.First().Module;
  4432. if (ModuleHelper.IsLoadLock(nextModule) && !_dictModuleTask[nextModule].Scheduler.IsIdle)
  4433. return;
  4434. _curEfemAction = _efemSchdActions.Dequeue();
  4435. efemRobot.PostMoveItems(_curEfemAction.ToArray());
  4436. }
  4437. }
  4438. else if (_curEfemAction.Count >= 0 && isReturnActionsDone(_curEfemAction)) // all scheduled actions done
  4439. {
  4440. _curEfemAction.Clear();
  4441. }
  4442. }
  4443. private void RunSchdTMReturnActions()
  4444. {
  4445. var tmRobot = Singleton<TransferModule>.Instance.GetScheduler(ModuleName.TMRobot) as SchedulerTMRobot;
  4446. if (tmRobot == null || !tmRobot.IsAvailable)
  4447. return;
  4448. if (_tmSchdActions.Count > 0)
  4449. {
  4450. if (_curTmAction.Count == 0 || isReturnActionsDone(_curTmAction))
  4451. {
  4452. var nextActions = _tmSchdActions.First();
  4453. var nextModule = nextActions.First().Module;
  4454. if (ModuleHelper.IsLoadLock(nextModule) && !_dictModuleTask[nextModule].Scheduler.IsIdle)
  4455. return;
  4456. var llActions = _curTmAction.Find(ac => ModuleHelper.IsLoadLock(ac.DestinationModule));
  4457. if (llActions != null && !_qeWaitCoolingLLs.Contains(llActions.DestinationModule) && !RouteManager.IsATMMode)
  4458. {
  4459. _qeWaitCoolingLLs.Enqueue(llActions.DestinationModule);
  4460. }
  4461. _curTmAction = _tmSchdActions.Dequeue();
  4462. tmRobot.SendMoveItems(_curTmAction.ToArray());
  4463. }
  4464. }
  4465. else if (_curTmAction.Count >= 0 && isReturnActionsDone(_curTmAction)) // all scheduled actions done
  4466. {
  4467. var llActions = _curTmAction.Find(ac => ModuleHelper.IsLoadLock(ac.DestinationModule));
  4468. if (llActions != null && !_qeWaitCoolingLLs.Contains(llActions.DestinationModule) && !RouteManager.IsATMMode)
  4469. {
  4470. _qeWaitCoolingLLs.Enqueue(llActions.DestinationModule);
  4471. }
  4472. _curTmAction.Clear();
  4473. }
  4474. }
  4475. #endregion
  4476. #region sequence/recipe operation
  4477. private bool CheckSequencePmReady(SequenceInfo seq, out string reason)
  4478. {
  4479. reason = "";
  4480. foreach (var pm in seq.PMs)
  4481. {
  4482. if (ModuleHelper.IsInstalled(pm) && (_dictModuleTask[pm].Scheduler.IsOnline || !Singleton<RouteManager>.Instance.IsAutoMode))
  4483. return true;
  4484. }
  4485. reason = $"Sequence {seq.Name} no valid PM, " + string.Join("|", seq.PMs);
  4486. return false;
  4487. }
  4488. private bool CheckSequenceKepler2200TemperatureReady(SequenceInfo seq)
  4489. {
  4490. for (int i = 0; i < seq.Steps.Count; i++)
  4491. {
  4492. SequenceStepInfo stepInfo = seq.Steps[i];
  4493. foreach (var module in stepInfo.StepModules)
  4494. {
  4495. if (ModuleHelper.IsPm(module))
  4496. {
  4497. string attr = $"{module}Recipe";
  4498. if (stepInfo.StepParameter.ContainsKey(attr)
  4499. && !string.IsNullOrWhiteSpace((string)stepInfo.StepParameter[attr]))
  4500. {
  4501. var recipeName = seq.GetRecipe(module);
  4502. var recipeContent =
  4503. RecipeFileManager.Instance.LoadRecipe($"{module}", recipeName, false, "Process");
  4504. var recipe = Recipe.Load(recipeContent);
  4505. float currentChamberTemperature;
  4506. switch (module)
  4507. {
  4508. case ModuleName.PMA:
  4509. currentChamberTemperature = Singleton<RouteManager>.Instance.PMA.ChamberTemperature;
  4510. break;
  4511. case ModuleName.PMB:
  4512. currentChamberTemperature = Singleton<RouteManager>.Instance.PMB.ChamberTemperature;
  4513. break;
  4514. case ModuleName.PMC:
  4515. currentChamberTemperature = Singleton<RouteManager>.Instance.PMC.ChamberTemperature;
  4516. break;
  4517. case ModuleName.PMD:
  4518. currentChamberTemperature = Singleton<RouteManager>.Instance.PMD.ChamberTemperature;
  4519. break;
  4520. default:
  4521. currentChamberTemperature = 0;
  4522. break;
  4523. }
  4524. if (recipe.Header.Temperature != null && recipe.Header.Temperature != "" && (currentChamberTemperature > Convert.ToSingle(recipe.Header.Temperature) + 10 || currentChamberTemperature < Convert.ToSingle(recipe.Header.Temperature) - 10))
  4525. {
  4526. LOG.Write(eEvent.ERR_ROUTER, ModuleName.System, $"Start job失败,由于{module}腔体温度{currentChamberTemperature}与{recipeName} Recipe温度{recipe.Header.Temperature}不匹配");
  4527. return false;
  4528. }
  4529. }
  4530. }
  4531. }
  4532. }
  4533. return true;
  4534. }
  4535. private bool CheckSequenceRecipeFileValid(SequenceInfo seq, out string reason)
  4536. {
  4537. for (int i = 0; i < seq.Steps.Count; i++)
  4538. {
  4539. SequenceStepInfo stepInfo = seq.Steps[i];
  4540. foreach (var module in stepInfo.StepModules)
  4541. {
  4542. if (ModuleHelper.IsPm(module))
  4543. {
  4544. string attr = $"{module}Recipe";
  4545. if (stepInfo.StepParameter.ContainsKey(attr)
  4546. && !string.IsNullOrWhiteSpace((string)stepInfo.StepParameter[attr]))
  4547. {
  4548. var recipeName = (string)stepInfo.StepParameter[attr];
  4549. if (!string.IsNullOrWhiteSpace(recipeName))
  4550. {
  4551. var recipeContent =
  4552. RecipeFileManager.Instance.LoadRecipe($"{module}", recipeName, false, "Process");
  4553. if (string.IsNullOrWhiteSpace(recipeContent))
  4554. {
  4555. reason = $"Can not find recipe file{recipeName}";
  4556. return false;
  4557. }
  4558. }
  4559. }
  4560. }
  4561. }
  4562. }
  4563. reason = "";
  4564. return true;
  4565. }
  4566. private void UpdateLLInOutPathProperty()
  4567. {
  4568. if (_lstControlJobs.Count == 0)
  4569. return;
  4570. var inOutPaths = new List<SequenceLLInOutPath>();
  4571. foreach (var wt in _lstWaferTasks)
  4572. {
  4573. if (!inOutPaths.Contains(wt.llInOutPath))
  4574. inOutPaths.Add(wt.llInOutPath);
  4575. }
  4576. if (inOutPaths.Count == 1)
  4577. {
  4578. _LLInOutPath = inOutPaths[0];
  4579. }
  4580. else
  4581. _LLInOutPath = SequenceLLInOutPath.DInDOut;
  4582. }
  4583. private bool GetAllJobRecipe(List<string> InUseRecipes, ControlJobInfo cj, SequenceInfo seq)
  4584. {
  4585. for (int i = 0; i < seq.Steps.Count; i++)
  4586. {
  4587. SequenceStepInfo stepInfo = seq.Steps[i];
  4588. foreach (var module in stepInfo.StepModules)
  4589. {
  4590. if (ModuleHelper.IsPm(module))
  4591. {
  4592. string attr = $"{module}Recipe";
  4593. if (stepInfo.StepParameter.ContainsKey(attr)
  4594. && !string.IsNullOrWhiteSpace((string)stepInfo.StepParameter[attr]))
  4595. {
  4596. var recipeName = seq.GetRecipe(module);
  4597. var recipeContent =
  4598. RecipeFileManager.Instance.LoadRecipe($"{module}", recipeName, false, "Process");
  4599. var recipe = Recipe.Load(recipeContent);
  4600. if (recipe.Header.ChuckRecipe != null && recipe.Header.ChuckRecipe != "")
  4601. {
  4602. InUseRecipes.Add($"{module}.ChuckRecipe.{recipe.Header.ChuckRecipe}");
  4603. }
  4604. if (recipe.Header.DechuckRecipe != null && recipe.Header.DechuckRecipe != "")
  4605. {
  4606. InUseRecipes.Add($"{module}.DechuckRecipe.{recipe.Header.DechuckRecipe}");
  4607. }
  4608. InUseRecipes.Add($"{module}.Process.{recipeName}");
  4609. if (cj.PreJobClean != "")
  4610. {
  4611. InUseRecipes.Add($"{module}.Clean.{cj.PreJobClean}");
  4612. }
  4613. if (cj.PostJobClean != "")
  4614. {
  4615. InUseRecipes.Add($"{module}.Clean.{cj.PostJobClean}");
  4616. }
  4617. //if (stepInfo.StepParameter["WTWClean"].ToString() != "")
  4618. //{
  4619. // InUseRecipes.Add($"{module}.Clean.{stepInfo.StepParameter["WTWClean"]}");
  4620. //}
  4621. }
  4622. }
  4623. }
  4624. }
  4625. return true;
  4626. }
  4627. private bool IsSequenceNeedAlign(SequenceInfo sequence)
  4628. {
  4629. // check wether need align
  4630. foreach (var step in sequence.Steps)
  4631. {
  4632. foreach (var mod in step.StepModules)
  4633. {
  4634. if (ModuleHelper.IsAligner(mod))
  4635. return true;
  4636. }
  4637. }
  4638. return false;
  4639. }
  4640. //private bool CheckKepler2200Scrubber()
  4641. //{
  4642. // if (RtInstance.ConfigType != ConfigType.Kepler2200)
  4643. // {
  4644. // return true;
  4645. // }
  4646. // List<ModuleName> modules = new List<ModuleName>();
  4647. // _lstProcessJobs.ForEach(x =>
  4648. // {
  4649. // x.Sequence.PMs.ForEach(m => modules.Add(m));
  4650. // });
  4651. // modules.Distinct().ToList().ForEach(x => _dictModuleTask[x].Scheduler.IsAvailable);
  4652. // //var item = _dictModuleTask.Where(x => ModuleHelper.IsPm(x.Key));
  4653. // //var item1 = _lstProcessJobs[0].Sequence.PMs;
  4654. //}
  4655. #endregion
  4656. }
  4657. }